bewcloud/public/components/notes/MainNotes.js
Bruno Bernardino 917649b97a
Some checks are pending
Build Docker Image / build-and-push (push) Waiting to run
Deploy / deploy (push) Waiting to run
Run Tests / test (push) Waiting to run
Fix ring opacity in dropdowns
2026-02-25 15:14:18 +00:00

233 lines
No EOL
7.6 KiB
JavaScript

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
}));
}