mirror of
https://github.com/cheeaun/phanpy.git
synced 2024-12-30 14:18:15 +03:00
307 lines
8.3 KiB
JavaScript
307 lines
8.3 KiB
JavaScript
import './lists.css';
|
|
|
|
import { Menu, MenuItem } from '@szhsin/react-menu';
|
|
import { useEffect, useRef, useState } from 'preact/hooks';
|
|
import { InView } from 'react-intersection-observer';
|
|
import { useNavigate, useParams } from 'react-router-dom';
|
|
import { useSnapshot } from 'valtio';
|
|
|
|
import AccountBlock from '../components/account-block';
|
|
import Icon from '../components/icon';
|
|
import Link from '../components/link';
|
|
import ListAddEdit from '../components/list-add-edit';
|
|
import Menu2 from '../components/menu2';
|
|
import Modal from '../components/modal';
|
|
import Timeline from '../components/timeline';
|
|
import { api } from '../utils/api';
|
|
import { filteredItems } from '../utils/filters';
|
|
import states, { saveStatus } from '../utils/states';
|
|
import useTitle from '../utils/useTitle';
|
|
|
|
const LIMIT = 20;
|
|
|
|
function List(props) {
|
|
const snapStates = useSnapshot(states);
|
|
const { masto, instance } = api();
|
|
const id = props?.id || useParams()?.id;
|
|
const navigate = useNavigate();
|
|
const latestItem = useRef();
|
|
// const [reloadCount, reload] = useReducer((c) => c + 1, 0);
|
|
|
|
const listIterator = useRef();
|
|
async function fetchList(firstLoad) {
|
|
if (firstLoad || !listIterator.current) {
|
|
listIterator.current = masto.v1.timelines.listList(id, {
|
|
limit: LIMIT,
|
|
});
|
|
}
|
|
const results = await listIterator.current.next();
|
|
let { value } = results;
|
|
if (value?.length) {
|
|
if (firstLoad) {
|
|
latestItem.current = value[0].id;
|
|
}
|
|
|
|
value = filteredItems(value, 'home');
|
|
value.forEach((item) => {
|
|
saveStatus(item, instance);
|
|
});
|
|
}
|
|
return {
|
|
...results,
|
|
value,
|
|
};
|
|
}
|
|
|
|
async function checkForUpdates() {
|
|
try {
|
|
const results = await masto.v1.timelines.listList(id, {
|
|
limit: 1,
|
|
since_id: latestItem.current,
|
|
});
|
|
let { value } = results;
|
|
value = filteredItems(value, 'home');
|
|
if (value?.length) {
|
|
return true;
|
|
}
|
|
return false;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
const [list, setList] = useState({ title: 'List' });
|
|
// const [title, setTitle] = useState(`List`);
|
|
useTitle(list.title, `/l/:id`);
|
|
useEffect(() => {
|
|
(async () => {
|
|
try {
|
|
const list = await masto.v1.lists.fetch(id);
|
|
setList(list);
|
|
// setTitle(list.title);
|
|
} catch (e) {
|
|
console.error(e);
|
|
}
|
|
})();
|
|
}, [id]);
|
|
|
|
const [showListAddEditModal, setShowListAddEditModal] = useState(false);
|
|
const [showManageMembersModal, setShowManageMembersModal] = useState(false);
|
|
|
|
return (
|
|
<>
|
|
<Timeline
|
|
key={id}
|
|
title={list.title}
|
|
id="list"
|
|
emptyText="Nothing yet."
|
|
errorText="Unable to load posts."
|
|
instance={instance}
|
|
fetchItems={fetchList}
|
|
checkForUpdates={checkForUpdates}
|
|
useItemID
|
|
boostsCarousel={snapStates.settings.boostsCarousel}
|
|
allowFilters
|
|
// refresh={reloadCount}
|
|
headerStart={
|
|
<Link to="/l" class="button plain">
|
|
<Icon icon="list" size="l" />
|
|
</Link>
|
|
}
|
|
headerEnd={
|
|
<Menu2
|
|
portal
|
|
setDownOverflow
|
|
overflow="auto"
|
|
viewScroll="close"
|
|
position="anchor"
|
|
menuButton={
|
|
<button type="button" class="plain">
|
|
<Icon icon="more" size="l" />
|
|
</button>
|
|
}
|
|
>
|
|
<MenuItem
|
|
onClick={() =>
|
|
setShowListAddEditModal({
|
|
list,
|
|
})
|
|
}
|
|
>
|
|
<Icon icon="pencil" size="l" />
|
|
<span>Edit</span>
|
|
</MenuItem>
|
|
<MenuItem onClick={() => setShowManageMembersModal(true)}>
|
|
<Icon icon="group" size="l" />
|
|
<span>Manage members</span>
|
|
</MenuItem>
|
|
</Menu2>
|
|
}
|
|
/>
|
|
{showListAddEditModal && (
|
|
<Modal
|
|
class="light"
|
|
onClick={(e) => {
|
|
if (e.target === e.currentTarget) {
|
|
setShowListAddEditModal(false);
|
|
}
|
|
}}
|
|
>
|
|
<ListAddEdit
|
|
list={showListAddEditModal?.list}
|
|
onClose={(result) => {
|
|
if (result.state === 'success' && result.list) {
|
|
setList(result.list);
|
|
// reload();
|
|
} else if (result.state === 'deleted') {
|
|
navigate('/l');
|
|
}
|
|
setShowListAddEditModal(false);
|
|
}}
|
|
/>
|
|
</Modal>
|
|
)}
|
|
{showManageMembersModal && (
|
|
<Modal
|
|
class="light"
|
|
onClick={(e) => {
|
|
if (e.target === e.currentTarget) {
|
|
setShowManageMembersModal(false);
|
|
}
|
|
}}
|
|
>
|
|
<ListManageMembers
|
|
listID={id}
|
|
onClose={() => setShowManageMembersModal(false)}
|
|
/>
|
|
</Modal>
|
|
)}
|
|
</>
|
|
);
|
|
}
|
|
|
|
const MEMBERS_LIMIT = 40;
|
|
function ListManageMembers({ listID, onClose }) {
|
|
// Show list of members with [Remove] button
|
|
// API only returns 40 members at a time, so this need to be paginated with infinite scroll
|
|
// Show [Add] button after removing a member
|
|
const { masto, instance } = api();
|
|
const [members, setMembers] = useState([]);
|
|
const [uiState, setUIState] = useState('default');
|
|
const [showMore, setShowMore] = useState(false);
|
|
|
|
const membersIterator = useRef();
|
|
|
|
async function fetchMembers(firstLoad) {
|
|
setShowMore(false);
|
|
setUIState('loading');
|
|
(async () => {
|
|
try {
|
|
if (firstLoad || !membersIterator.current) {
|
|
membersIterator.current = masto.v1.lists.listAccounts(listID, {
|
|
limit: MEMBERS_LIMIT,
|
|
});
|
|
}
|
|
const results = await membersIterator.current.next();
|
|
let { done, value } = results;
|
|
if (value?.length) {
|
|
if (firstLoad) {
|
|
setMembers(value);
|
|
} else {
|
|
setMembers(members.concat(value));
|
|
}
|
|
setShowMore(!done);
|
|
} else {
|
|
setShowMore(false);
|
|
}
|
|
setUIState('default');
|
|
} catch (e) {
|
|
setUIState('error');
|
|
}
|
|
})();
|
|
}
|
|
|
|
useEffect(() => {
|
|
fetchMembers(true);
|
|
}, []);
|
|
|
|
return (
|
|
<div class="sheet" id="list-manage-members-container">
|
|
{!!onClose && (
|
|
<button type="button" class="sheet-close" onClick={onClose}>
|
|
<Icon icon="x" />
|
|
</button>
|
|
)}
|
|
<header>
|
|
<h2>Manage members</h2>
|
|
</header>
|
|
<main>
|
|
<ul>
|
|
{members.map((member) => (
|
|
<li key={member.id}>
|
|
<AccountBlock account={member} instance={instance} />
|
|
<RemoveAddButton account={member} listID={listID} />
|
|
</li>
|
|
))}
|
|
{showMore && uiState === 'default' && (
|
|
<InView as="li" onChange={(inView) => inView && fetchMembers()}>
|
|
<button type="button" class="light block" onClick={fetchMembers}>
|
|
Show more…
|
|
</button>
|
|
</InView>
|
|
)}
|
|
</ul>
|
|
</main>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function RemoveAddButton({ account, listID }) {
|
|
const { masto } = api();
|
|
const [uiState, setUIState] = useState('default');
|
|
const [removed, setRemoved] = useState(false);
|
|
|
|
return (
|
|
<button
|
|
type="button"
|
|
class={`light ${removed ? '' : 'danger'}`}
|
|
disabled={uiState === 'loading'}
|
|
onClick={() => {
|
|
if (removed) {
|
|
setUIState('loading');
|
|
(async () => {
|
|
try {
|
|
await masto.v1.lists.addAccount(listID, {
|
|
accountIds: [account.id],
|
|
});
|
|
setUIState('default');
|
|
setRemoved(false);
|
|
} catch (e) {
|
|
setUIState('error');
|
|
}
|
|
})();
|
|
} else {
|
|
const yes = confirm(`Remove ${account.username} from this list?`);
|
|
if (!yes) return;
|
|
setUIState('loading');
|
|
|
|
(async () => {
|
|
try {
|
|
await masto.v1.lists.removeAccount(listID, {
|
|
accountIds: [account.id],
|
|
});
|
|
setUIState('default');
|
|
setRemoved(true);
|
|
} catch (e) {
|
|
setUIState('error');
|
|
}
|
|
})();
|
|
}
|
|
}}
|
|
>
|
|
{removed ? 'Add' : 'Remove…'}
|
|
</button>
|
|
);
|
|
}
|
|
|
|
export default List;
|