drawDB/src/components/area.jsx

352 lines
13 KiB
React
Raw Normal View History

2023-09-19 20:48:59 +08:00
import { React, useContext, useState } from "react";
2023-09-19 20:50:18 +08:00
import { Button, Popover, Input, Toast } from "@douyinfe/semi-ui";
import {
IconEdit,
IconCheckboxTick,
IconDeleteStroked,
} from "@douyinfe/semi-icons";
import {
Tab,
Action,
ObjectType,
tableThemes,
defaultTableTheme,
} from "../data/data";
import {
AreaContext,
LayoutContext,
2023-09-19 20:50:28 +08:00
SelectContext,
2023-09-19 20:51:08 +08:00
SettingsContext,
2023-09-19 20:50:18 +08:00
TabContext,
UndoRedoContext,
} from "../pages/editor";
2023-09-19 20:48:04 +08:00
export default function Area(props) {
2023-09-19 20:48:08 +08:00
const [hovered, setHovered] = useState(false);
2023-09-19 20:50:18 +08:00
const [saved, setSaved] = useState(false);
2023-09-19 20:50:20 +08:00
const [editField, setEditField] = useState({});
2023-09-19 20:48:59 +08:00
const { layout } = useContext(LayoutContext);
2023-09-19 20:51:08 +08:00
const { settings } = useContext(SettingsContext);
2023-09-19 20:48:59 +08:00
const { tab, setTab } = useContext(TabContext);
2023-09-19 20:50:18 +08:00
const { updateArea, deleteArea } = useContext(AreaContext);
const { setUndoStack, setRedoStack } = useContext(UndoRedoContext);
2023-09-19 20:50:28 +08:00
const { selectedElement, setSelectedElement } = useContext(SelectContext);
2023-09-19 20:48:06 +08:00
const handleMouseDown = (e, dir) => {
2023-09-19 20:48:55 +08:00
props.setResize({ id: props.areaData.id, dir: dir });
2023-09-19 20:48:42 +08:00
props.setInitCoords({
2023-09-19 20:48:07 +08:00
x: props.areaData.x,
y: props.areaData.y,
width: props.areaData.width,
height: props.areaData.height,
2023-09-19 20:49:57 +08:00
mouseX: e.clientX / props.zoom,
mouseY: e.clientY / props.zoom,
2023-09-19 20:48:07 +08:00
});
2023-09-19 20:48:06 +08:00
};
2023-09-19 20:48:04 +08:00
return (
2023-09-19 20:48:08 +08:00
<g
onMouseEnter={() => setHovered(true)}
2023-09-19 20:50:18 +08:00
onMouseLeave={() => {
setHovered(false);
setSaved(false);
}}
2023-09-19 20:48:08 +08:00
>
2023-09-19 20:48:04 +08:00
<foreignObject
key={props.areaData.id}
x={props.areaData.x}
y={props.areaData.y}
2023-09-19 20:48:08 +08:00
width={props.areaData.width > 0 ? props.areaData.width : 0}
height={props.areaData.height > 0 ? props.areaData.height : 0}
2023-09-19 20:48:04 +08:00
onMouseDown={props.onMouseDown}
>
2023-09-19 20:48:55 +08:00
<div
className={`${
hovered
? "border-4 border-dashed border-[#5891db]"
: "border-2 border-slate-400"
} w-full h-full cursor-move rounded relative`}
>
<div
className="opacity-40 w-fill p-2 h-full"
style={{ backgroundColor: props.areaData.color }}
/>
</div>
2023-09-19 20:51:08 +08:00
<div className="text-color absolute top-2 left-3 select-none">
2023-09-19 20:48:04 +08:00
{props.areaData.name}
</div>
2023-09-19 20:50:28 +08:00
{(hovered ||
(selectedElement.element === ObjectType.AREA &&
selectedElement.id === props.areaData.id &&
selectedElement.openDialogue &&
!layout.sidebar)) && (
2023-09-19 20:48:59 +08:00
<div className="absolute top-2 right-3">
2023-09-19 20:50:18 +08:00
<Popover
2023-09-19 20:50:28 +08:00
visible={
selectedElement.element === ObjectType.AREA &&
selectedElement.id === props.areaData.id &&
selectedElement.openDialogue &&
!layout.sidebar
}
onClickOutSide={() => {
setSelectedElement((prev) => ({
...prev,
openDialogue: false,
}));
}}
stopPropagation
2023-09-19 20:50:18 +08:00
content={
2023-09-19 20:51:08 +08:00
<div className="popover-theme">
2023-09-19 20:50:18 +08:00
<div className="font-semibold mb-2 ms-1">
Edit subject area
</div>
<div className="w-[280px] flex items-center mb-2">
<Input
value={props.areaData.name}
placeholder="Name"
className="me-2"
onChange={(value) =>
updateArea(props.areaData.id, { name: value })
}
onFocus={(e) => setEditField({ name: e.target.value })}
onBlur={(e) => {
setSaved(true);
if (e.target.value === editField.name) return;
setUndoStack((prev) => [
...prev,
{
action: Action.EDIT,
element: ObjectType.AREA,
aid: props.areaData.id,
undo: editField,
redo: { name: e.target.value },
2023-09-19 20:50:52 +08:00
message: `Edit area name to ${e.target.value}`,
2023-09-19 20:50:18 +08:00
},
]);
setRedoStack([]);
}}
/>
<Popover
content={
2023-09-19 20:51:08 +08:00
<div className="popover-theme">
2023-09-19 20:50:18 +08:00
<div className="flex justify-between items-center p-2">
<div className="font-medium">Theme</div>
<Button
type="tertiary"
size="small"
onClick={() =>
updateArea(props.areaData.id, {
color: defaultTableTheme,
})
}
>
Clear
</Button>
</div>
<hr />
<div className="py-3">
<div>
{tableThemes
.slice(0, Math.ceil(tableThemes.length / 2))
.map((c) => (
<button
key={c}
style={{ backgroundColor: c }}
className="p-3 rounded-full mx-1"
onClick={() => {
setUndoStack((prev) => [
...prev,
{
action: Action.EDIT,
element: ObjectType.AREA,
aid: props.areaData.id,
undo: { color: props.areaData.color },
redo: { color: c },
2023-09-19 20:50:52 +08:00
message: `Edit area color to ${c}`,
2023-09-19 20:50:18 +08:00
},
]);
setRedoStack([]);
updateArea(props.areaData.id, {
color: c,
});
}}
>
{props.areaData.color === c ? (
<IconCheckboxTick
style={{ color: "white" }}
/>
) : (
<IconCheckboxTick style={{ color: c }} />
)}
</button>
))}
</div>
<div className="mt-3">
{tableThemes
.slice(Math.ceil(tableThemes.length / 2))
.map((c) => (
<button
key={c}
style={{ backgroundColor: c }}
className="p-3 rounded-full mx-1"
onClick={() => {
setUndoStack((prev) => [
...prev,
{
action: Action.EDIT,
element: ObjectType.AREA,
aid: props.areaData.id,
undo: { color: props.areaData.color },
redo: { color: c },
2023-09-19 20:50:52 +08:00
message: `Edit area color to ${c}`,
2023-09-19 20:50:18 +08:00
},
]);
setRedoStack([]);
updateArea(props.areaData.id, {
color: c,
});
}}
>
<IconCheckboxTick
style={{
color:
props.areaData.color === c
? "white"
: c,
}}
/>
</button>
))}
</div>
</div>
</div>
}
2023-09-19 20:50:20 +08:00
position="rightTop"
2023-09-19 20:50:18 +08:00
showArrow
>
<div
className="h-[32px] w-[32px] rounded"
style={{ backgroundColor: props.areaData.color }}
/>
</Popover>
</div>
<div className="flex">
<Button
icon={<IconDeleteStroked />}
type="danger"
block
onClick={() => {
Toast.success(`Area deleted!`);
deleteArea(props.areaData.id, true);
}}
>
Delete
</Button>
<Button
block
style={{ marginLeft: "8px" }}
onClick={() => {
if (!saved) {
if (props.areaData.name === editField.name) return;
setUndoStack((prev) => [
...prev,
{
action: Action.EDIT,
element: ObjectType.AREA,
aid: props.areaData.id,
undo: editField,
redo: { name: props.areaData.name },
2023-09-19 20:50:52 +08:00
message: `Edit area name to ${props.areaData.name}`,
2023-09-19 20:50:18 +08:00
},
]);
setRedoStack([]);
setSaved(false);
}
}}
>
Save
</Button>
</div>
</div>
}
trigger="custom"
position="rightTop"
showArrow
>
<Button
icon={<IconEdit />}
size="small"
theme="solid"
style={{
backgroundColor: "#2f68ad",
opacity: "0.7",
}}
onClick={() => {
if (layout.sidebar) {
setTab(Tab.subject_areas);
if (tab !== Tab.subject_areas) return;
document
.getElementById(`scroll_area_${props.areaData.id}`)
.scrollIntoView({ behavior: "smooth" });
} else {
2023-09-19 20:50:28 +08:00
setSelectedElement({
element: ObjectType.AREA,
id: props.areaData.id,
openDialogue: true,
openCollapse: false,
});
2023-09-19 20:50:18 +08:00
}
}}
></Button>
</Popover>
2023-09-19 20:48:59 +08:00
</div>
)}
2023-09-19 20:48:04 +08:00
</foreignObject>
2023-09-19 20:48:08 +08:00
{hovered && (
<>
2023-09-19 20:48:55 +08:00
<circle
cx={props.areaData.x}
cy={props.areaData.y}
r={6}
2023-09-19 20:51:08 +08:00
fill={settings.mode === "light" ? "white" : "rgb(28, 31, 35)"}
2023-09-19 20:48:55 +08:00
stroke="#5891db"
strokeWidth={3}
2023-09-19 20:48:08 +08:00
cursor="nwse-resize"
onMouseDown={(e) => handleMouseDown(e, "tl")}
/>
2023-09-19 20:48:55 +08:00
<circle
cx={props.areaData.x + props.areaData.width}
cy={props.areaData.y}
r={6}
2023-09-19 20:51:08 +08:00
fill={settings.mode === "light" ? "white" : "rgb(28, 31, 35)"}
2023-09-19 20:48:55 +08:00
stroke="#5891db"
strokeWidth={3}
2023-09-19 20:48:08 +08:00
cursor="nesw-resize"
onMouseDown={(e) => handleMouseDown(e, "tr")}
/>
2023-09-19 20:48:55 +08:00
<circle
cx={props.areaData.x}
cy={props.areaData.y + props.areaData.height}
r={6}
2023-09-19 20:51:08 +08:00
fill={settings.mode === "light" ? "white" : "rgb(28, 31, 35)"}
2023-09-19 20:48:55 +08:00
stroke="#5891db"
strokeWidth={3}
2023-09-19 20:48:08 +08:00
cursor="nesw-resize"
onMouseDown={(e) => handleMouseDown(e, "bl")}
/>
2023-09-19 20:48:55 +08:00
<circle
cx={props.areaData.x + props.areaData.width}
cy={props.areaData.y + props.areaData.height}
r={6}
2023-09-19 20:51:08 +08:00
fill={settings.mode === "light" ? "white" : "rgb(28, 31, 35)"}
2023-09-19 20:48:55 +08:00
stroke="#5891db"
strokeWidth={3}
2023-09-19 20:48:08 +08:00
cursor="nwse-resize"
onMouseDown={(e) => handleMouseDown(e, "br")}
/>
</>
)}
2023-09-19 20:48:04 +08:00
</g>
);
}