drawDB/src/components/EditorCanvas/Note.jsx

296 lines
10 KiB
React
Raw Normal View History

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