import { useSignal } from '@preact/signals'; import ListFiles from "/public/components/files/ListFiles.js"; import FilesBreadcrumb from "/public/components/files/FilesBreadcrumb.js"; import CreateDirectoryModal from "/public/components/files/CreateDirectoryModal.js"; import CreateNoteModal from "./CreateNoteModal.js"; export default function MainNotes({ initialDirectories, initialFiles, initialPath }) { const isAdding = useSignal(false); const isDeleting = useSignal(false); const directories = useSignal(initialDirectories); const files = useSignal(initialFiles); const path = useSignal(initialPath); const areNewOptionsOption = useSignal(false); const isNewNoteModalOpen = useSignal(false); const isNewDirectoryModalOpen = useSignal(false); function onClickCreateNote() { if (isNewNoteModalOpen.value) { isNewNoteModalOpen.value = false; return; } isNewNoteModalOpen.value = true; } async function onClickSaveNote(newNoteName) { if (isAdding.value) { return; } if (!newNoteName) { return; } areNewOptionsOption.value = false; isAdding.value = true; const requestBody = new FormData(); requestBody.set('parent_path', path.value); requestBody.set('path_in_view', path.value); requestBody.set('name', `${newNoteName}.md`); requestBody.set('contents', `# ${newNoteName}\n\nStart your new note!\n`); try { const response = await fetch(`/api/files/upload`, { method: 'POST', body: requestBody }); if (!response.ok) { throw new Error(`Failed to create note. ${response.statusText} ${await response.text()}`); } const result = await response.json(); if (!result.success) { throw new Error('Failed to create note!'); } files.value = [...result.newFiles]; isNewNoteModalOpen.value = false; } catch (error) { console.error(error); } isAdding.value = false; } function onCloseCreateNote() { isNewNoteModalOpen.value = false; } function onClickCreateDirectory() { if (isNewDirectoryModalOpen.value) { isNewDirectoryModalOpen.value = false; return; } isNewDirectoryModalOpen.value = true; } async function onClickSaveDirectory(newDirectoryName) { if (isAdding.value) { return; } if (!newDirectoryName) { return; } areNewOptionsOption.value = false; isAdding.value = true; try { const requestBody = { parentPath: path.value, name: newDirectoryName }; const response = await fetch(`/api/files/create-directory`, { method: 'POST', body: JSON.stringify(requestBody) }); if (!response.ok) { throw new Error(`Failed to create directory. ${response.statusText} ${await response.text()}`); } const result = await response.json(); if (!result.success) { throw new Error('Failed to create directory!'); } directories.value = [...result.newDirectories]; isNewDirectoryModalOpen.value = false; } catch (error) { console.error(error); } isAdding.value = false; } function onCloseCreateDirectory() { isNewDirectoryModalOpen.value = false; } function toggleNewOptionsDropdown() { areNewOptionsOption.value = !areNewOptionsOption.value; } async function onClickDeleteDirectory(parentPath, name) { if (confirm('Are you sure you want to delete this directory?')) { if (isDeleting.value) { return; } isDeleting.value = true; try { const requestBody = { parentPath, name }; const response = await fetch(`/api/files/delete-directory`, { method: 'POST', body: JSON.stringify(requestBody) }); if (!response.ok) { throw new Error(`Failed to delete directory. ${response.statusText} ${await response.text()}`); } const result = await response.json(); if (!result.success) { throw new Error('Failed to delete directory!'); } directories.value = [...result.newDirectories]; } catch (error) { console.error(error); } isDeleting.value = false; } } async function onClickDeleteFile(parentPath, name) { if (confirm('Are you sure you want to delete this note?')) { if (isDeleting.value) { return; } isDeleting.value = true; try { const requestBody = { parentPath, name }; const response = await fetch(`/api/files/delete`, { method: 'POST', body: JSON.stringify(requestBody) }); if (!response.ok) { throw new Error(`Failed to delete note. ${response.statusText} ${await response.text()}`); } const result = await response.json(); if (!result.success) { throw new Error('Failed to delete note!'); } files.value = [...result.newFiles]; } catch (error) { console.error(error); } isDeleting.value = false; } } return h(Fragment, null, h("section", { class: "flex flex-row items-center justify-between mb-4" }, h("section", { class: "flex items-center justify-end w-full" }, h(FilesBreadcrumb, { path: path.value, isShowingNotes: true }), h("section", { class: "relative inline-block text-left ml-2" }, h("div", null, h("button", { class: "inline-block justify-center gap-x-1.5 rounded-md bg-[#51A4FB] px-3 py-2 text-sm font-semibold text-white shadow-sm hover:bg-sky-400 ml-2", type: "button", title: "Add new note or directory", id: "new-button", "aria-expanded": "true", "aria-haspopup": "true", onClick: () => toggleNewOptionsDropdown() }, h("img", { src: "/public/images/add.svg", alt: "Add new note or directory", class: `white ${isAdding.value ? 'animate-spin' : ''}`, width: 20, height: 20 }))), h("div", { class: `absolute right-0 z-10 mt-2 w-44 origin-top-right rounded-md bg-slate-700 shadow-lg ring-1 ring-black/15 focus:outline-none ${!areNewOptionsOption.value ? 'hidden' : ''}`, role: "menu", "aria-orientation": "vertical", "aria-labelledby": "new-button", tabindex: -1 }, h("div", { class: "py-1" }, h("button", { class: `text-white block px-4 py-2 text-sm w-full text-left hover:bg-slate-600`, onClick: () => onClickCreateNote(), type: "button" }, "New Note"), h("button", { class: `text-white block px-4 py-2 text-sm w-full text-left hover:bg-slate-600`, onClick: () => onClickCreateDirectory(), type: "button" }, "New Directory")))))), h("section", { class: "mx-auto max-w-7xl my-8" }, h(ListFiles, { directories: directories.value, files: files.value, onClickDeleteDirectory: onClickDeleteDirectory, onClickDeleteFile: onClickDeleteFile, isShowingNotes: true }), h("span", { class: `flex justify-end items-center text-sm mt-1 mx-2 text-slate-100` }, isDeleting.value ? h(Fragment, null, h("img", { src: "/public/images/loading.svg", class: "white mr-2", width: 18, height: 18 }), "Deleting...") : null, isAdding.value ? h(Fragment, null, h("img", { src: "/public/images/loading.svg", class: "white mr-2", width: 18, height: 18 }), "Creating...") : null, !isDeleting.value && !isAdding.value ? h(Fragment, null, "\xA0") : null)), h(CreateDirectoryModal, { isOpen: isNewDirectoryModalOpen.value, onClickSave: onClickSaveDirectory, onClose: onCloseCreateDirectory }), h(CreateNoteModal, { isOpen: isNewNoteModalOpen.value, onClickSave: onClickSaveNote, onClose: onCloseCreateNote })); }