mirror of
https://github.com/kremalicious/umami.git
synced 2025-02-14 21:10:34 +01:00
127 lines
3.8 KiB
TypeScript
127 lines
3.8 KiB
TypeScript
import { useState, useMemo, useEffect } from 'react';
|
|
import { DragDropContext, Draggable, Droppable } from 'react-beautiful-dnd';
|
|
import classNames from 'classnames';
|
|
import { Button, Loading } from 'react-basics';
|
|
import { firstBy } from 'thenby';
|
|
import useDashboard, { saveDashboard } from 'store/dashboard';
|
|
import { useMessages, useWebsites } from 'components/hooks';
|
|
import styles from './DashboardEdit.module.css';
|
|
|
|
const DRAG_ID = 'dashboard-website-ordering';
|
|
|
|
export function DashboardEdit({ teamId }: { teamId: string }) {
|
|
const settings = useDashboard();
|
|
const { websiteOrder } = settings;
|
|
const { formatMessage, labels } = useMessages();
|
|
const [order, setOrder] = useState(websiteOrder || []);
|
|
const [websites, setWebsites] = useState([]);
|
|
|
|
const {
|
|
result,
|
|
query: { isLoading },
|
|
setParams,
|
|
} = useWebsites({ teamId });
|
|
|
|
useEffect(() => {
|
|
if (result?.data) {
|
|
setWebsites(prevWebsites => {
|
|
const newWebsites = [...prevWebsites, ...result.data];
|
|
if (newWebsites.length < result.count) {
|
|
setParams(prevParams => ({ ...prevParams, page: prevParams.page + 1 }));
|
|
}
|
|
return newWebsites;
|
|
});
|
|
}
|
|
}, [result]);
|
|
|
|
const ordered = useMemo(() => {
|
|
if (websites) {
|
|
return websites
|
|
.map((website: { id: any }) => ({ ...website, order: order.indexOf(website.id) }))
|
|
.sort(firstBy('order'));
|
|
}
|
|
return [];
|
|
}, [websites, order]);
|
|
|
|
function handleWebsiteDrag({ destination, source }) {
|
|
if (!destination || destination.index === source.index) return;
|
|
|
|
const orderedWebsites = [...ordered];
|
|
const [removed] = orderedWebsites.splice(source.index, 1);
|
|
orderedWebsites.splice(destination.index, 0, removed);
|
|
|
|
setOrder(orderedWebsites.map(website => website?.id || 0));
|
|
}
|
|
|
|
function handleSave() {
|
|
saveDashboard({
|
|
editing: false,
|
|
websiteOrder: order,
|
|
});
|
|
}
|
|
|
|
function handleCancel() {
|
|
saveDashboard({ editing: false, websiteOrder });
|
|
}
|
|
|
|
function handleReset() {
|
|
setOrder([]);
|
|
}
|
|
|
|
if (isLoading) {
|
|
return <Loading />;
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className={styles.buttons}>
|
|
<Button onClick={handleSave} variant="primary" size="sm">
|
|
{formatMessage(labels.save)}
|
|
</Button>
|
|
<Button onClick={handleCancel} size="sm">
|
|
{formatMessage(labels.cancel)}
|
|
</Button>
|
|
<Button onClick={handleReset} size="sm">
|
|
{formatMessage(labels.reset)}
|
|
</Button>
|
|
</div>
|
|
<div className={styles.dragActive}>
|
|
<DragDropContext onDragEnd={handleWebsiteDrag}>
|
|
<Droppable droppableId={DRAG_ID}>
|
|
{(provided, snapshot) => (
|
|
<div
|
|
{...provided.droppableProps}
|
|
ref={provided.innerRef}
|
|
style={{ marginBottom: snapshot.isDraggingOver ? 260 : null }}
|
|
>
|
|
{ordered.map(({ id, name, domain }, index) => (
|
|
<Draggable key={id} draggableId={`${DRAG_ID}-${id}`} index={index}>
|
|
{(provided, snapshot) => (
|
|
<div
|
|
ref={provided.innerRef}
|
|
className={classNames(styles.item, {
|
|
[styles.active]: snapshot.isDragging,
|
|
})}
|
|
{...provided.draggableProps}
|
|
{...provided.dragHandleProps}
|
|
>
|
|
<div className={styles.text}>
|
|
<h1>{name}</h1>
|
|
<h2>{domain}</h2>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</Draggable>
|
|
))}
|
|
{provided.placeholder}
|
|
</div>
|
|
)}
|
|
</Droppable>
|
|
</DragDropContext>
|
|
</div>
|
|
</>
|
|
);
|
|
}
|
|
|
|
export default DashboardEdit;
|