2025-09-11 22:22:50 +03:00
|
|
|
import { Dispatch, StateUpdater, useCallback, useEffect, useMemo, useRef, useState } from "preact/hooks";
|
2025-09-10 20:18:17 +03:00
|
|
|
import { ViewModeProps } from "../interface";
|
|
|
|
|
import "./index.css";
|
|
|
|
|
import { ColumnMap, getBoardData } from "./data";
|
2025-09-11 19:14:54 +03:00
|
|
|
import { useNoteLabelWithDefault, useTriliumEvent } from "../../react/hooks";
|
2025-09-10 21:10:31 +03:00
|
|
|
import Icon from "../../react/Icon";
|
|
|
|
|
import { t } from "../../../services/i18n";
|
2025-09-11 19:14:54 +03:00
|
|
|
import Api from "./api";
|
2025-09-10 22:20:17 +03:00
|
|
|
import FormTextBox from "../../react/FormTextBox";
|
2025-09-11 20:02:58 +03:00
|
|
|
import { createContext } from "preact";
|
2025-09-11 21:20:25 +03:00
|
|
|
import { onWheelHorizontalScroll } from "../../widget_utils";
|
2025-09-11 21:42:59 +03:00
|
|
|
import Column from "./column";
|
2025-09-12 14:48:05 +03:00
|
|
|
import BoardApi from "./api";
|
2025-09-12 15:58:38 +03:00
|
|
|
import FormTextArea from "../../react/FormTextArea";
|
2025-09-10 20:18:17 +03:00
|
|
|
|
|
|
|
|
export interface BoardViewData {
|
|
|
|
|
columns?: BoardColumnData[];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export interface BoardColumnData {
|
|
|
|
|
value: string;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-11 20:02:58 +03:00
|
|
|
interface BoardViewContextData {
|
2025-09-12 14:48:05 +03:00
|
|
|
api?: BoardApi;
|
2025-09-11 20:02:58 +03:00
|
|
|
branchIdToEdit?: string;
|
2025-09-11 21:34:10 +03:00
|
|
|
columnNameToEdit?: string;
|
2025-09-11 22:22:50 +03:00
|
|
|
setColumnNameToEdit?: Dispatch<StateUpdater<string | undefined>>;
|
|
|
|
|
setBranchIdToEdit?: Dispatch<StateUpdater<string | undefined>>;
|
2025-09-12 14:48:05 +03:00
|
|
|
draggedColumn: { column: string, index: number } | null;
|
|
|
|
|
setDraggedColumn: (column: { column: string, index: number } | null) => void;
|
|
|
|
|
dropPosition: { column: string, index: number } | null;
|
|
|
|
|
setDropPosition: (position: { column: string, index: number } | null) => void;
|
|
|
|
|
draggedCard: { noteId: string, branchId: string, fromColumn: string, index: number } | null;
|
|
|
|
|
setDraggedCard: (card: { noteId: string, branchId: string, fromColumn: string, index: number } | null) => void;
|
|
|
|
|
setDropTarget: (target: string | null) => void,
|
|
|
|
|
dropTarget: string | null
|
2025-09-11 20:02:58 +03:00
|
|
|
}
|
|
|
|
|
|
2025-09-11 21:42:59 +03:00
|
|
|
export const BoardViewContext = createContext<BoardViewContextData>({});
|
2025-09-11 20:02:58 +03:00
|
|
|
|
2025-09-10 20:18:17 +03:00
|
|
|
export default function BoardView({ note: parentNote, noteIds, viewConfig, saveConfig }: ViewModeProps<BoardViewData>) {
|
2025-09-11 19:14:54 +03:00
|
|
|
const [ statusAttribute ] = useNoteLabelWithDefault(parentNote, "board:groupBy", "status");
|
2025-09-10 20:18:17 +03:00
|
|
|
const [ byColumn, setByColumn ] = useState<ColumnMap>();
|
|
|
|
|
const [ columns, setColumns ] = useState<string[]>();
|
2025-09-11 18:27:42 +03:00
|
|
|
const [ draggedCard, setDraggedCard ] = useState<{ noteId: string, branchId: string, fromColumn: string, index: number } | null>(null);
|
2025-09-11 18:05:09 +03:00
|
|
|
const [ dropTarget, setDropTarget ] = useState<string | null>(null);
|
2025-09-11 18:11:12 +03:00
|
|
|
const [ dropPosition, setDropPosition ] = useState<{ column: string, index: number } | null>(null);
|
2025-09-11 18:42:32 +03:00
|
|
|
const [ draggedColumn, setDraggedColumn ] = useState<{ column: string, index: number } | null>(null);
|
|
|
|
|
const [ columnDropPosition, setColumnDropPosition ] = useState<number | null>(null);
|
2025-09-11 20:02:58 +03:00
|
|
|
const [ branchIdToEdit, setBranchIdToEdit ] = useState<string>();
|
2025-09-11 21:34:10 +03:00
|
|
|
const [ columnNameToEdit, setColumnNameToEdit ] = useState<string>();
|
2025-09-11 19:14:54 +03:00
|
|
|
const api = useMemo(() => {
|
2025-09-11 20:02:58 +03:00
|
|
|
return new Api(byColumn, columns ?? [], parentNote, statusAttribute, viewConfig ?? {}, saveConfig, setBranchIdToEdit );
|
|
|
|
|
}, [ byColumn, columns, parentNote, statusAttribute, viewConfig, saveConfig, setBranchIdToEdit ]);
|
|
|
|
|
const boardViewContext = useMemo<BoardViewContextData>(() => ({
|
2025-09-12 15:10:20 +03:00
|
|
|
api,
|
2025-09-12 14:48:05 +03:00
|
|
|
branchIdToEdit, setBranchIdToEdit,
|
|
|
|
|
columnNameToEdit, setColumnNameToEdit,
|
|
|
|
|
draggedColumn, setDraggedColumn,
|
|
|
|
|
dropPosition, setDropPosition,
|
|
|
|
|
draggedCard, setDraggedCard,
|
|
|
|
|
dropTarget, setDropTarget
|
2025-09-12 15:10:20 +03:00
|
|
|
}), [
|
|
|
|
|
api,
|
|
|
|
|
branchIdToEdit, setBranchIdToEdit,
|
2025-09-12 15:08:00 +03:00
|
|
|
columnNameToEdit, setColumnNameToEdit,
|
|
|
|
|
draggedColumn, setDraggedColumn,
|
|
|
|
|
dropPosition, setDropPosition,
|
|
|
|
|
draggedCard, setDraggedCard,
|
|
|
|
|
dropTarget, setDropTarget
|
|
|
|
|
]);
|
2025-09-10 20:18:17 +03:00
|
|
|
|
2025-09-10 21:41:15 +03:00
|
|
|
function refresh() {
|
2025-09-11 19:14:54 +03:00
|
|
|
getBoardData(parentNote, statusAttribute, viewConfig ?? {}).then(({ byColumn, newPersistedData }) => {
|
2025-09-10 20:18:17 +03:00
|
|
|
setByColumn(byColumn);
|
|
|
|
|
|
|
|
|
|
if (newPersistedData) {
|
|
|
|
|
viewConfig = { ...newPersistedData };
|
|
|
|
|
saveConfig(newPersistedData);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Use the order from persistedData.columns, then add any new columns found
|
|
|
|
|
const orderedColumns = viewConfig?.columns?.map(col => col.value) || [];
|
|
|
|
|
const allColumns = Array.from(byColumn.keys());
|
|
|
|
|
const newColumns = allColumns.filter(col => !orderedColumns.includes(col));
|
|
|
|
|
setColumns([...orderedColumns, ...newColumns]);
|
|
|
|
|
});
|
2025-09-10 21:41:15 +03:00
|
|
|
}
|
|
|
|
|
|
2025-09-11 20:37:09 +03:00
|
|
|
useEffect(refresh, [ parentNote, noteIds, viewConfig ]);
|
2025-09-10 21:41:15 +03:00
|
|
|
|
2025-09-11 18:42:32 +03:00
|
|
|
const handleColumnDrop = useCallback((fromIndex: number, toIndex: number) => {
|
|
|
|
|
if (!columns || fromIndex === toIndex) return;
|
|
|
|
|
|
|
|
|
|
const newColumns = [...columns];
|
|
|
|
|
const [movedColumn] = newColumns.splice(fromIndex, 1);
|
|
|
|
|
newColumns.splice(toIndex, 0, movedColumn);
|
|
|
|
|
|
|
|
|
|
// Update view config with new column order
|
|
|
|
|
const newViewConfig = {
|
|
|
|
|
...viewConfig,
|
|
|
|
|
columns: newColumns.map(col => ({ value: col }))
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
saveConfig(newViewConfig);
|
|
|
|
|
setColumns(newColumns);
|
|
|
|
|
setDraggedColumn(null);
|
|
|
|
|
setColumnDropPosition(null);
|
|
|
|
|
}, [columns, viewConfig, saveConfig]);
|
|
|
|
|
|
2025-09-10 21:41:15 +03:00
|
|
|
useTriliumEvent("entitiesReloaded", ({ loadResults }) => {
|
|
|
|
|
// Check if any changes affect our board
|
|
|
|
|
const hasRelevantChanges =
|
|
|
|
|
// React to changes in status attribute for notes in this board
|
|
|
|
|
loadResults.getAttributeRows().some(attr => attr.name === statusAttribute && noteIds.includes(attr.noteId!)) ||
|
|
|
|
|
// React to changes in note title
|
|
|
|
|
loadResults.getNoteIds().some(noteId => noteIds.includes(noteId)) ||
|
|
|
|
|
// React to changes in branches for subchildren (e.g., moved, added, or removed notes)
|
|
|
|
|
loadResults.getBranchRows().some(branch => noteIds.includes(branch.noteId!)) ||
|
|
|
|
|
// React to changes in note icon or color.
|
|
|
|
|
loadResults.getAttributeRows().some(attr => [ "iconClass", "color" ].includes(attr.name ?? "") && noteIds.includes(attr.noteId ?? "")) ||
|
|
|
|
|
// React to attachment change
|
|
|
|
|
loadResults.getAttachmentRows().some(att => att.ownerId === parentNote.noteId && att.title === "board.json") ||
|
|
|
|
|
// React to changes in "groupBy"
|
|
|
|
|
loadResults.getAttributeRows().some(attr => attr.name === "board:groupBy" && attr.noteId === parentNote.noteId);
|
|
|
|
|
|
|
|
|
|
if (hasRelevantChanges) {
|
|
|
|
|
refresh();
|
|
|
|
|
}
|
|
|
|
|
});
|
2025-09-10 20:18:17 +03:00
|
|
|
|
2025-09-11 18:42:32 +03:00
|
|
|
const handleColumnDragOver = useCallback((e: DragEvent) => {
|
|
|
|
|
if (!draggedColumn) return;
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
|
|
|
|
const container = e.currentTarget as HTMLElement;
|
|
|
|
|
const columns = Array.from(container.querySelectorAll('.board-column'));
|
|
|
|
|
const mouseX = e.clientX;
|
|
|
|
|
|
|
|
|
|
let newIndex = columns.length;
|
|
|
|
|
for (let i = 0; i < columns.length; i++) {
|
|
|
|
|
const col = columns[i] as HTMLElement;
|
|
|
|
|
const rect = col.getBoundingClientRect();
|
|
|
|
|
const colMiddle = rect.left + rect.width / 2;
|
|
|
|
|
|
|
|
|
|
if (mouseX < colMiddle) {
|
|
|
|
|
newIndex = i;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setColumnDropPosition(newIndex);
|
|
|
|
|
}, [draggedColumn]);
|
|
|
|
|
|
|
|
|
|
const handleContainerDrop = useCallback((e: DragEvent) => {
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
if (draggedColumn && columnDropPosition !== null) {
|
|
|
|
|
handleColumnDrop(draggedColumn.index, columnDropPosition);
|
|
|
|
|
}
|
|
|
|
|
}, [draggedColumn, columnDropPosition, handleColumnDrop]);
|
|
|
|
|
|
2025-09-10 20:18:17 +03:00
|
|
|
return (
|
2025-09-11 21:20:25 +03:00
|
|
|
<div
|
|
|
|
|
className="board-view"
|
|
|
|
|
onWheel={onWheelHorizontalScroll}
|
|
|
|
|
>
|
2025-09-11 20:02:58 +03:00
|
|
|
<BoardViewContext.Provider value={boardViewContext}>
|
|
|
|
|
<div
|
|
|
|
|
className="board-view-container"
|
|
|
|
|
onDragOver={handleColumnDragOver}
|
|
|
|
|
onDrop={handleContainerDrop}
|
|
|
|
|
>
|
|
|
|
|
{byColumn && columns?.map((column, index) => (
|
|
|
|
|
<>
|
|
|
|
|
{columnDropPosition === index && draggedColumn?.column !== column && (
|
|
|
|
|
<div className="column-drop-placeholder show" />
|
|
|
|
|
)}
|
|
|
|
|
<Column
|
|
|
|
|
api={api}
|
|
|
|
|
column={column}
|
|
|
|
|
columnIndex={index}
|
|
|
|
|
columnItems={byColumn.get(column)}
|
|
|
|
|
isDraggingColumn={draggedColumn?.column === column}
|
|
|
|
|
/>
|
|
|
|
|
</>
|
|
|
|
|
))}
|
|
|
|
|
{columnDropPosition === columns?.length && draggedColumn && (
|
|
|
|
|
<div className="column-drop-placeholder show" />
|
|
|
|
|
)}
|
|
|
|
|
|
2025-09-12 17:05:17 +03:00
|
|
|
<AddNewColumn api={api} />
|
2025-09-11 20:02:58 +03:00
|
|
|
</div>
|
|
|
|
|
</BoardViewContext.Provider>
|
2025-09-10 20:18:17 +03:00
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-12 17:05:17 +03:00
|
|
|
function AddNewColumn({ api }: { api: BoardApi }) {
|
2025-09-10 22:20:17 +03:00
|
|
|
const [ isCreatingNewColumn, setIsCreatingNewColumn ] = useState(false);
|
|
|
|
|
|
|
|
|
|
const addColumnCallback = useCallback(() => {
|
|
|
|
|
setIsCreatingNewColumn(true);
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<div className={`board-add-column ${isCreatingNewColumn ? "editing" : ""}`} onClick={addColumnCallback}>
|
|
|
|
|
{!isCreatingNewColumn
|
|
|
|
|
? <>
|
|
|
|
|
<Icon icon="bx bx-plus" />{" "}
|
|
|
|
|
{t("board_view.add-column")}
|
|
|
|
|
</>
|
2025-09-12 17:05:17 +03:00
|
|
|
: (
|
|
|
|
|
<TitleEditor
|
2025-09-11 22:35:31 +03:00
|
|
|
placeholder={t("board_view.add-column-placeholder")}
|
2025-09-12 17:05:17 +03:00
|
|
|
save={(columnName) => api.addNewColumn(columnName)}
|
|
|
|
|
dismiss={() => setIsCreatingNewColumn(false)}
|
|
|
|
|
isNewItem
|
2025-09-10 22:20:17 +03:00
|
|
|
/>
|
2025-09-12 17:05:17 +03:00
|
|
|
)}
|
2025-09-10 22:20:17 +03:00
|
|
|
</div>
|
|
|
|
|
)
|
|
|
|
|
}
|
2025-09-11 21:51:02 +03:00
|
|
|
|
2025-09-12 17:05:17 +03:00
|
|
|
export function TitleEditor({ currentValue, placeholder, save, dismiss, multiline, isNewItem }: {
|
|
|
|
|
currentValue?: string;
|
|
|
|
|
placeholder?: string;
|
|
|
|
|
save: (newValue: string) => void;
|
|
|
|
|
dismiss: () => void;
|
|
|
|
|
multiline?: boolean;
|
|
|
|
|
isNewItem?: boolean;
|
2025-09-11 21:51:02 +03:00
|
|
|
}) {
|
2025-09-12 15:58:38 +03:00
|
|
|
const inputRef = useRef<any>(null);
|
2025-09-11 21:51:02 +03:00
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
inputRef.current?.focus();
|
|
|
|
|
inputRef.current?.select();
|
|
|
|
|
}, [ inputRef ]);
|
|
|
|
|
|
2025-09-12 15:58:38 +03:00
|
|
|
const Element = multiline ? FormTextArea : FormTextBox;
|
|
|
|
|
|
2025-09-11 21:51:02 +03:00
|
|
|
return (
|
2025-09-12 15:58:38 +03:00
|
|
|
<Element
|
2025-09-11 21:51:02 +03:00
|
|
|
inputRef={inputRef}
|
2025-09-12 17:05:17 +03:00
|
|
|
currentValue={currentValue ?? ""}
|
|
|
|
|
placeholder={placeholder}
|
2025-09-12 15:58:38 +03:00
|
|
|
rows={multiline ? 4 : undefined}
|
2025-09-11 21:51:02 +03:00
|
|
|
onKeyDown={(e) => {
|
|
|
|
|
if (e.key === "Enter") {
|
|
|
|
|
const newValue = e.currentTarget.value;
|
2025-09-12 16:57:23 +03:00
|
|
|
if (newValue !== currentValue || isNewItem) {
|
2025-09-11 21:51:02 +03:00
|
|
|
save(newValue);
|
|
|
|
|
}
|
|
|
|
|
dismiss();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (e.key === "Escape") {
|
|
|
|
|
dismiss();
|
|
|
|
|
}
|
|
|
|
|
}}
|
|
|
|
|
onBlur={(newValue) => {
|
|
|
|
|
if (newValue !== currentValue) {
|
|
|
|
|
save(newValue);
|
|
|
|
|
}
|
|
|
|
|
dismiss();
|
|
|
|
|
}}
|
|
|
|
|
/>
|
2025-09-12 17:05:17 +03:00
|
|
|
);
|
2025-09-11 21:51:02 +03:00
|
|
|
}
|