2024-04-06 09:58:42 +08:00
|
|
|
import { useState } from "react";
|
2024-05-16 11:44:39 +08:00
|
|
|
import { Button, Collapse, TextArea, Popover, Input } from "@douyinfe/semi-ui";
|
2024-04-06 09:58:42 +08:00
|
|
|
import { IconDeleteStroked, IconCheckboxTick } from "@douyinfe/semi-icons";
|
|
|
|
import { noteThemes, Action, ObjectType } from "../../../data/constants";
|
|
|
|
import { useNotes, useUndoRedo } from "../../../hooks";
|
2024-05-16 11:44:39 +08:00
|
|
|
import { useTranslation } from "react-i18next";
|
2024-04-06 09:58:42 +08:00
|
|
|
|
|
|
|
export default function NoteInfo({ data, nid }) {
|
|
|
|
const { updateNote, deleteNote } = useNotes();
|
|
|
|
const { setUndoStack, setRedoStack } = useUndoRedo();
|
|
|
|
const [editField, setEditField] = useState({});
|
2024-05-16 11:44:39 +08:00
|
|
|
const { t } = useTranslation();
|
2024-04-06 09:58:42 +08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Collapse.Panel
|
|
|
|
header={
|
|
|
|
<div className="overflow-hidden text-ellipsis whitespace-nowrap">
|
|
|
|
{data.title}
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
itemKey={`${data.id}`}
|
|
|
|
id={`scroll_note_${data.id}`}
|
|
|
|
>
|
|
|
|
<div className="flex items-center mb-2">
|
2024-05-16 11:44:39 +08:00
|
|
|
<div className="font-semibold me-2 break-keep">{t("title")}:</div>
|
2024-04-06 09:58:42 +08:00
|
|
|
<Input
|
|
|
|
value={data.title}
|
2024-05-16 11:44:39 +08:00
|
|
|
placeholder={t("title")}
|
2024-04-06 09:58:42 +08:00
|
|
|
onChange={(value) => updateNote(data.id, { title: value })}
|
|
|
|
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,
|
|
|
|
nid: data.id,
|
|
|
|
undo: editField,
|
|
|
|
redo: { title: e.target.value },
|
2024-05-16 11:44:39 +08:00
|
|
|
message: t("edit_note", {
|
|
|
|
noteTitle: e.target.value,
|
|
|
|
extra: "[title]",
|
|
|
|
}),
|
2024-04-06 09:58:42 +08:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div className="flex justify-between align-top">
|
|
|
|
<TextArea
|
2024-05-16 11:44:39 +08:00
|
|
|
placeholder={t("content")}
|
2024-04-06 09:58:42 +08:00
|
|
|
value={data.content}
|
|
|
|
autosize
|
|
|
|
onChange={(value) => {
|
|
|
|
const textarea = document.getElementById(`note_${data.id}`);
|
|
|
|
textarea.style.height = "0";
|
|
|
|
textarea.style.height = textarea.scrollHeight + "px";
|
|
|
|
const newHeight = textarea.scrollHeight + 16 + 20 + 4;
|
|
|
|
updateNote(data.id, { height: newHeight, content: value });
|
|
|
|
}}
|
|
|
|
onFocus={(e) =>
|
|
|
|
setEditField({ content: e.target.value, height: data.height })
|
|
|
|
}
|
|
|
|
onBlur={(e) => {
|
2024-05-15 17:40:51 +08:00
|
|
|
if (e.target.value === editField.content) return;
|
2024-04-06 09:58:42 +08:00
|
|
|
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: nid,
|
|
|
|
undo: editField,
|
|
|
|
redo: { content: e.target.value, height: newHeight },
|
2024-05-16 11:44:39 +08:00
|
|
|
message: t("edit_note", {
|
|
|
|
noteTitle: e.target.value,
|
|
|
|
extra: "[content]",
|
|
|
|
}),
|
2024-04-06 09:58:42 +08:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
|
|
|
}}
|
|
|
|
rows={3}
|
|
|
|
/>
|
|
|
|
<div className="ms-2">
|
|
|
|
<Popover
|
|
|
|
content={
|
|
|
|
<div className="popover-theme">
|
2024-05-16 11:44:39 +08:00
|
|
|
<div className="font-medium mb-1">{t("theme")}</div>
|
2024-04-06 09:58:42 +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,
|
|
|
|
nid: nid,
|
|
|
|
undo: { color: data.color },
|
|
|
|
redo: { color: c },
|
2024-05-16 11:44:39 +08:00
|
|
|
message: t("edit_note", {
|
|
|
|
noteTitle: data.title,
|
|
|
|
extra: "[color]",
|
|
|
|
}),
|
2024-04-06 09:58:42 +08:00
|
|
|
},
|
|
|
|
]);
|
|
|
|
setRedoStack([]);
|
|
|
|
updateNote(nid, { color: c });
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
{data.color === c ? (
|
|
|
|
<IconCheckboxTick style={{ color: "white" }} />
|
|
|
|
) : (
|
|
|
|
<IconCheckboxTick style={{ color: c }} />
|
|
|
|
)}
|
|
|
|
</button>
|
|
|
|
))}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
trigger="click"
|
|
|
|
position="rightTop"
|
|
|
|
showArrow
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className="h-[32px] w-[32px] rounded mb-2"
|
|
|
|
style={{ backgroundColor: data.color }}
|
|
|
|
/>
|
|
|
|
</Popover>
|
|
|
|
<Button
|
|
|
|
icon={<IconDeleteStroked />}
|
|
|
|
type="danger"
|
2024-05-16 11:44:39 +08:00
|
|
|
onClick={() => deleteNote(nid, true)}
|
2024-04-06 09:58:42 +08:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Collapse.Panel>
|
|
|
|
);
|
|
|
|
}
|