2024-03-15 22:00:23 +08:00
|
|
|
import { useState } from "react";
|
2024-04-05 10:12:50 +08:00
|
|
|
import {
|
|
|
|
Action,
|
|
|
|
ObjectType,
|
|
|
|
Tab,
|
|
|
|
State,
|
|
|
|
noteThemes,
|
|
|
|
} from "../../data/constants";
|
2023-09-19 20:50:20 +08:00
|
|
|
import { Input, Button, Popover, Toast } from "@douyinfe/semi-ui";
|
|
|
|
import {
|
|
|
|
IconEdit,
|
|
|
|
IconDeleteStroked,
|
|
|
|
IconCheckboxTick,
|
|
|
|
} from "@douyinfe/semi-icons";
|
2024-04-05 10:12:50 +08:00
|
|
|
import {
|
|
|
|
useLayout,
|
|
|
|
useUndoRedo,
|
|
|
|
useSelect,
|
|
|
|
useNotes,
|
|
|
|
useSaveState,
|
|
|
|
} from "../../hooks";
|
2023-09-19 20:49:09 +08:00
|
|
|
|
2024-03-14 02:39:16 +08:00
|
|
|
export default function Note({ data, onMouseDown }) {
|
2023-09-19 20:49:11 +08:00
|
|
|
const w = 180;
|
2023-09-19 20:49:09 +08:00
|
|
|
const r = 3;
|
|
|
|
const fold = 24;
|
2024-03-14 02:39:16 +08:00
|
|
|
const [editField, setEditField] = useState({});
|
|
|
|
const [hovered, setHovered] = useState(false);
|
|
|
|
const { layout } = useLayout();
|
2024-03-15 22:00:23 +08:00
|
|
|
const { setSaveState } = useSaveState();
|
2024-03-13 05:36:49 +08:00
|
|
|
const { updateNote, deleteNote } = useNotes();
|
2024-03-11 05:55:23 +08:00
|
|
|
const { setUndoStack, setRedoStack } = useUndoRedo();
|
|
|
|
const { selectedElement, setSelectedElement } = useSelect();
|
2023-09-19 20:50:20 +08:00
|
|
|
|
2023-09-19 20:49:09 +08:00
|
|
|
const handleChange = (e) => {
|
2024-03-14 02:39:16 +08:00
|
|
|
const textarea = document.getElementById(`note_${data.id}`);
|
2023-09-19 20:49:09 +08:00
|
|
|
textarea.style.height = "0";
|
|
|
|
textarea.style.height = textarea.scrollHeight + "px";
|
2024-03-16 08:23:10 +08:00
|
|
|
const newHeight = textarea.scrollHeight + 42;
|
2024-03-14 02:39:16 +08:00
|
|
|
updateNote(data.id, { content: e.target.value, height: newHeight });
|
2023-09-19 20:49:09 +08:00
|
|
|
};
|
|
|
|
|
2024-03-16 08:23:10 +08:00
|
|
|
const handleBlur = (e) => {
|
|
|
|
if (e.target.value === editField.content) return;
|
|
|
|
const textarea = document.getElementById(`note_${data.id}`);
|
|
|
|
textarea.style.height = "0";
|
|
|
|
textarea.style.height = textarea.scrollHeight + "px";
|
|
|
|
const newHeight = textarea.scrollHeight + 16 + 20 + 4;
|
|
|
|
setUndoStack((prev) => [
|
|
|
|
...prev,
|
|
|
|
{
|
|
|
|
action: Action.EDIT,
|
|
|
|
element: ObjectType.NOTE,
|
|
|
|
nid: data.id,
|
|
|
|
undo: editField,
|
|
|
|
redo: { content: e.target.value, height: newHeight },
|
|
|
|
message: `Edit note content to "${e.target.value}"`,
|
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
|
|
|
};
|
|
|
|
|
|
|
|
const edit = () => {
|
|
|
|
if (layout.sidebar) {
|
|
|
|
setSelectedElement((prev) => ({
|
|
|
|
...prev,
|
|
|
|
currentTab: Tab.NOTES,
|
|
|
|
}));
|
|
|
|
if (selectedElement.currentTab !== Tab.NOTES) return;
|
|
|
|
document
|
|
|
|
.getElementById(`scroll_note_${data.id}`)
|
|
|
|
.scrollIntoView({ behavior: "smooth" });
|
|
|
|
} else {
|
|
|
|
setSelectedElement((prev) => ({
|
|
|
|
...prev,
|
|
|
|
element: ObjectType.NOTE,
|
|
|
|
id: data.id,
|
|
|
|
open: true,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2023-09-19 20:49:09 +08:00
|
|
|
return (
|
2023-09-19 20:50:20 +08:00
|
|
|
<g
|
|
|
|
onMouseEnter={() => setHovered(true)}
|
|
|
|
onMouseLeave={() => setHovered(false)}
|
|
|
|
>
|
2023-09-19 20:49:09 +08:00
|
|
|
<path
|
2024-03-14 02:39:16 +08:00
|
|
|
d={`M${data.x + fold} ${data.y} L${data.x + w - r} ${
|
|
|
|
data.y
|
|
|
|
} A${r} ${r} 0 0 1 ${data.x + w} ${data.y + r} L${data.x + w} ${
|
|
|
|
data.y + data.height - r
|
|
|
|
} A${r} ${r} 0 0 1 ${data.x + w - r} ${data.y + data.height} L${
|
|
|
|
data.x + r
|
|
|
|
} ${data.y + data.height} A${r} ${r} 0 0 1 ${data.x} ${
|
|
|
|
data.y + data.height - r
|
|
|
|
} L${data.x} ${data.y + fold}`}
|
|
|
|
fill={data.color}
|
2023-09-19 20:51:11 +08:00
|
|
|
stroke={
|
|
|
|
hovered
|
|
|
|
? "rgb(59 130 246)"
|
|
|
|
: selectedElement.element === ObjectType.NOTE &&
|
2024-03-14 02:39:16 +08:00
|
|
|
selectedElement.id === data.id
|
2023-09-19 20:51:11 +08:00
|
|
|
? "rgb(59 130 246)"
|
|
|
|
: "rgb(168 162 158)"
|
|
|
|
}
|
|
|
|
strokeDasharray={hovered ? 4 : 0}
|
2023-09-19 20:49:09 +08:00
|
|
|
strokeLinejoin="round"
|
2023-09-19 20:51:11 +08:00
|
|
|
strokeWidth="1.2"
|
2023-09-19 20:49:09 +08:00
|
|
|
/>
|
|
|
|
<path
|
2024-03-14 02:39:16 +08:00
|
|
|
d={`M${data.x} ${data.y + fold} L${data.x + fold - r} ${
|
|
|
|
data.y + fold
|
|
|
|
} A${r} ${r} 0 0 0 ${data.x + fold} ${data.y + fold - r} L${
|
|
|
|
data.x + fold
|
|
|
|
} ${data.y} L${data.x} ${data.y + fold} Z`}
|
|
|
|
fill={data.color}
|
2023-09-19 20:51:11 +08:00
|
|
|
stroke={
|
|
|
|
hovered
|
|
|
|
? "rgb(59 130 246)"
|
|
|
|
: selectedElement.element === ObjectType.NOTE &&
|
2024-03-14 02:39:16 +08:00
|
|
|
selectedElement.id === data.id
|
2023-09-19 20:51:11 +08:00
|
|
|
? "rgb(59 130 246)"
|
|
|
|
: "rgb(168 162 158)"
|
|
|
|
}
|
|
|
|
strokeDasharray={hovered ? 4 : 0}
|
2023-09-19 20:49:09 +08:00
|
|
|
strokeLinejoin="round"
|
2023-09-19 20:51:11 +08:00
|
|
|
strokeWidth="1.2"
|
2023-09-19 20:49:09 +08:00
|
|
|
/>
|
|
|
|
<foreignObject
|
2024-03-14 02:39:16 +08:00
|
|
|
x={data.x}
|
|
|
|
y={data.y}
|
2023-09-19 20:49:11 +08:00
|
|
|
width={w}
|
2024-03-14 02:39:16 +08:00
|
|
|
height={data.height}
|
|
|
|
onMouseDown={onMouseDown}
|
2023-09-19 20:49:09 +08:00
|
|
|
>
|
|
|
|
<div className="text-gray-900 select-none w-full h-full cursor-move px-3 py-2">
|
2024-03-14 02:39:16 +08:00
|
|
|
<label htmlFor={`note_${data.id}`} className="ms-5">
|
|
|
|
{data.title}
|
2023-09-19 20:50:00 +08:00
|
|
|
</label>
|
2023-09-19 20:49:09 +08:00
|
|
|
<textarea
|
2024-03-14 02:39:16 +08:00
|
|
|
id={`note_${data.id}`}
|
|
|
|
value={data.content}
|
2023-09-19 20:50:00 +08:00
|
|
|
onChange={handleChange}
|
2023-09-19 20:50:09 +08:00
|
|
|
onFocus={(e) =>
|
|
|
|
setEditField({
|
|
|
|
content: e.target.value,
|
2024-03-14 02:39:16 +08:00
|
|
|
height: data.height,
|
2023-09-19 20:50:09 +08:00
|
|
|
})
|
|
|
|
}
|
2024-03-16 08:23:10 +08:00
|
|
|
onBlur={handleBlur}
|
2023-09-19 20:50:00 +08:00
|
|
|
className="w-full resize-none outline-none overflow-y-hidden border-none select-none"
|
2024-03-14 02:39:16 +08:00
|
|
|
style={{ backgroundColor: data.color }}
|
2024-03-16 08:23:10 +08:00
|
|
|
/>
|
2023-09-19 20:50:28 +08:00
|
|
|
{(hovered ||
|
|
|
|
(selectedElement.element === ObjectType.NOTE &&
|
2024-03-14 02:39:16 +08:00
|
|
|
selectedElement.id === data.id &&
|
|
|
|
selectedElement.open &&
|
2023-09-19 20:50:28 +08:00
|
|
|
!layout.sidebar)) && (
|
2023-09-19 20:50:20 +08:00
|
|
|
<div className="absolute top-2 right-3">
|
|
|
|
<Popover
|
2023-09-19 20:50:28 +08:00
|
|
|
visible={
|
|
|
|
selectedElement.element === ObjectType.NOTE &&
|
2024-03-14 02:39:16 +08:00
|
|
|
selectedElement.id === data.id &&
|
|
|
|
selectedElement.open &&
|
2023-09-19 20:50:28 +08:00
|
|
|
!layout.sidebar
|
|
|
|
}
|
|
|
|
onClickOutSide={() => {
|
2024-03-14 02:39:16 +08:00
|
|
|
if (selectedElement.editFromToolbar) {
|
|
|
|
setSelectedElement((prev) => ({
|
|
|
|
...prev,
|
|
|
|
editFromToolbar: false,
|
|
|
|
}));
|
|
|
|
return;
|
|
|
|
}
|
2023-09-19 20:50:28 +08:00
|
|
|
setSelectedElement((prev) => ({
|
|
|
|
...prev,
|
2024-03-14 02:39:16 +08:00
|
|
|
open: false,
|
2023-09-19 20:50:28 +08:00
|
|
|
}));
|
2024-03-15 22:00:23 +08:00
|
|
|
setSaveState(State.SAVING);
|
2023-09-19 20:50:28 +08:00
|
|
|
}}
|
|
|
|
stopPropagation
|
2023-09-19 20:50:20 +08:00
|
|
|
content={
|
2023-09-19 20:51:08 +08:00
|
|
|
<div className="popover-theme">
|
2023-09-19 20:50:20 +08:00
|
|
|
<div className="font-semibold mb-2 ms-1">Edit note</div>
|
|
|
|
<div className="w-[280px] flex items-center mb-2">
|
|
|
|
<Input
|
2024-03-14 02:39:16 +08:00
|
|
|
value={data.title}
|
2023-09-19 20:50:20 +08:00
|
|
|
placeholder="Title"
|
|
|
|
className="me-2"
|
|
|
|
onChange={(value) =>
|
2024-03-14 02:39:16 +08:00
|
|
|
updateNote(data.id, { title: value })
|
2023-09-19 20:50:20 +08:00
|
|
|
}
|
|
|
|
onFocus={(e) => setEditField({ title: e.target.value })}
|
|
|
|
onBlur={(e) => {
|
|
|
|
if (e.target.value === editField.title) return;
|
|
|
|
setUndoStack((prev) => [
|
|
|
|
...prev,
|
|
|
|
{
|
|
|
|
action: Action.EDIT,
|
|
|
|
element: ObjectType.NOTE,
|
2024-03-14 02:39:16 +08:00
|
|
|
nid: data.id,
|
2023-09-19 20:50:20 +08:00
|
|
|
undo: editField,
|
|
|
|
redo: { title: e.target.value },
|
2024-01-22 23:56:57 +08:00
|
|
|
message: `Edit note title to "${e.target.value}"`,
|
2023-09-19 20:50:20 +08:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<Popover
|
|
|
|
content={
|
2023-09-19 20:51:08 +08:00
|
|
|
<div className="popover-theme">
|
|
|
|
<div className="font-medium mb-1">Theme</div>
|
2023-09-19 20:50:20 +08:00
|
|
|
<hr />
|
|
|
|
<div className="py-3">
|
|
|
|
{noteThemes.map((c) => (
|
|
|
|
<button
|
|
|
|
key={c}
|
|
|
|
style={{ backgroundColor: c }}
|
|
|
|
className="p-3 rounded-full mx-1"
|
|
|
|
onClick={() => {
|
|
|
|
setUndoStack((prev) => [
|
|
|
|
...prev,
|
|
|
|
{
|
|
|
|
action: Action.EDIT,
|
|
|
|
element: ObjectType.NOTE,
|
2024-03-14 02:39:16 +08:00
|
|
|
nid: data.id,
|
|
|
|
undo: { color: data.color },
|
2023-09-19 20:50:20 +08:00
|
|
|
redo: { color: c },
|
2023-09-19 20:50:52 +08:00
|
|
|
message: `Edit note color to ${c}`,
|
2023-09-19 20:50:20 +08:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
2024-03-14 02:39:16 +08:00
|
|
|
updateNote(data.id, { color: c });
|
2023-09-19 20:50:20 +08:00
|
|
|
}}
|
|
|
|
>
|
2024-03-14 02:39:16 +08:00
|
|
|
{data.color === c ? (
|
2023-09-19 20:50:20 +08:00
|
|
|
<IconCheckboxTick
|
|
|
|
style={{ color: "white" }}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<IconCheckboxTick style={{ color: c }} />
|
|
|
|
)}
|
|
|
|
</button>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
position="rightTop"
|
|
|
|
showArrow
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className="h-[32px] w-[32px] rounded"
|
2024-03-14 02:39:16 +08:00
|
|
|
style={{ backgroundColor: data.color }}
|
2023-09-19 20:50:20 +08:00
|
|
|
/>
|
|
|
|
</Popover>
|
|
|
|
</div>
|
|
|
|
<div className="flex">
|
|
|
|
<Button
|
|
|
|
icon={<IconDeleteStroked />}
|
|
|
|
type="danger"
|
|
|
|
block
|
|
|
|
onClick={() => {
|
|
|
|
Toast.success(`Note deleted!`);
|
2024-03-14 02:39:16 +08:00
|
|
|
deleteNote(data.id, true);
|
2023-09-19 20:50:20 +08:00
|
|
|
}}
|
|
|
|
>
|
|
|
|
Delete
|
|
|
|
</Button>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
trigger="custom"
|
|
|
|
position="rightTop"
|
|
|
|
showArrow
|
|
|
|
>
|
|
|
|
<Button
|
|
|
|
icon={<IconEdit />}
|
|
|
|
size="small"
|
|
|
|
theme="solid"
|
|
|
|
style={{
|
|
|
|
backgroundColor: "#2f68ad",
|
|
|
|
opacity: "0.7",
|
|
|
|
}}
|
2024-03-16 08:23:10 +08:00
|
|
|
onClick={edit}
|
|
|
|
/>
|
2023-09-19 20:50:20 +08:00
|
|
|
</Popover>
|
|
|
|
</div>
|
|
|
|
)}
|
2023-09-19 20:49:09 +08:00
|
|
|
</div>
|
|
|
|
</foreignObject>
|
|
|
|
</g>
|
|
|
|
);
|
|
|
|
}
|