feat(canvas): add persistent node favorites with toolbar star and glow

This commit is contained in:
2026-04-09 14:12:43 +02:00
parent e4d39a21fd
commit b08e448be0
18 changed files with 625 additions and 76 deletions

View File

@@ -81,6 +81,9 @@
--sidebar-accent-foreground: oklch(0.25 0.012 60); --sidebar-accent-foreground: oklch(0.25 0.012 60);
--sidebar-border: oklch(0.91 0.01 75); --sidebar-border: oklch(0.91 0.01 75);
--sidebar-ring: oklch(0.52 0.09 178); --sidebar-ring: oklch(0.52 0.09 178);
--node-favorite-ring: oklch(0.72 0.16 88);
--node-favorite-glow: oklch(0.78 0.18 90 / 0.24);
--node-favorite-fill: oklch(0.93 0.08 92 / 0.45);
} }
.dark { .dark {
@@ -115,6 +118,9 @@
--sidebar-accent-foreground: oklch(0.93 0.008 80); --sidebar-accent-foreground: oklch(0.93 0.008 80);
--sidebar-border: oklch(1 0 0 / 8%); --sidebar-border: oklch(1 0 0 / 8%);
--sidebar-ring: oklch(0.62 0.1 178); --sidebar-ring: oklch(0.62 0.1 178);
--node-favorite-ring: oklch(0.84 0.14 92);
--node-favorite-glow: oklch(0.82 0.16 92 / 0.36);
--node-favorite-fill: oklch(0.48 0.12 90 / 0.28);
} }
@layer base { @layer base {
@@ -306,4 +312,21 @@
transition: none; transition: none;
} }
} }
.node-favorite-chrome {
position: relative;
isolation: isolate;
}
.node-favorite-chrome::after {
content: "";
position: absolute;
inset: -1px;
border-radius: inherit;
pointer-events: none;
box-shadow:
0 0 0 1px var(--node-favorite-ring),
0 0 0 3px var(--node-favorite-fill),
0 0 22px -8px var(--node-favorite-glow);
}
} }

View File

@@ -0,0 +1,134 @@
// @vitest-environment jsdom
import React, { act } from "react";
import { createRoot, type Root } from "react-dom/client";
import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
const mocks = vi.hoisted(() => ({
queueNodeDataUpdate: vi.fn(async () => undefined),
createNodeWithIntersection: vi.fn(async () => undefined),
getNode: vi.fn(),
getNodes: vi.fn(() => []),
getEdges: vi.fn(() => []),
setNodes: vi.fn(),
deleteElements: vi.fn(async () => undefined),
}));
vi.mock("@xyflow/react", () => ({
NodeToolbar: ({ children }: { children: React.ReactNode }) => (
<div data-testid="node-toolbar">{children}</div>
),
NodeResizeControl: () => null,
Position: { Top: "top" },
useNodeId: () => "node-1",
useReactFlow: () => ({
getNode: mocks.getNode,
getNodes: mocks.getNodes,
getEdges: mocks.getEdges,
setNodes: mocks.setNodes,
deleteElements: mocks.deleteElements,
}),
getConnectedEdges: () => [],
}));
vi.mock("@/components/canvas/canvas-sync-context", () => ({
useCanvasSync: () => ({
queueNodeDataUpdate: mocks.queueNodeDataUpdate,
}),
}));
vi.mock("@/components/canvas/canvas-placement-context", () => ({
useCanvasPlacement: () => ({
createNodeWithIntersection: mocks.createNodeWithIntersection,
}),
}));
import BaseNodeWrapper from "@/components/canvas/nodes/base-node-wrapper";
(globalThis as typeof globalThis & { IS_REACT_ACT_ENVIRONMENT?: boolean }).IS_REACT_ACT_ENVIRONMENT = true;
describe("BaseNodeWrapper", () => {
let container: HTMLDivElement | null = null;
let root: Root | null = null;
beforeEach(() => {
mocks.queueNodeDataUpdate.mockClear();
mocks.createNodeWithIntersection.mockClear();
mocks.getNode.mockReset();
mocks.getNodes.mockClear();
mocks.getEdges.mockClear();
mocks.setNodes.mockClear();
mocks.deleteElements.mockClear();
container = document.createElement("div");
document.body.appendChild(container);
root = createRoot(container);
});
afterEach(async () => {
if (root) {
await act(async () => {
root?.unmount();
});
}
container?.remove();
container = null;
root = null;
});
async function renderWrapper(nodeData: Record<string, unknown>, selected = true) {
mocks.getNode.mockReturnValue({
id: "node-1",
type: "text",
data: nodeData,
position: { x: 0, y: 0 },
style: {},
});
await act(async () => {
root?.render(
<BaseNodeWrapper nodeType="text" selected={selected}>
<div>Inner node content</div>
</BaseNodeWrapper>,
);
});
}
it("shows favorite toggle with duplicate and delete controls for selected nodes", async () => {
await renderWrapper({ label: "Frame" }, true);
expect(container?.querySelector('button[title="Favorite"]')).toBeTruthy();
expect(container?.querySelector('button[title="Duplicate"]')).toBeTruthy();
expect(container?.querySelector('button[title="Delete"]')).toBeTruthy();
});
it("toggles favorite and queues merged node data update", async () => {
await renderWrapper({ label: "Frame" }, true);
const favoriteButton = container?.querySelector('button[title="Favorite"]');
if (!(favoriteButton instanceof HTMLButtonElement)) {
throw new Error("Favorite button not found");
}
await act(async () => {
favoriteButton.dispatchEvent(new MouseEvent("click", { bubbles: true }));
});
expect(mocks.queueNodeDataUpdate).toHaveBeenCalledWith({
nodeId: "node-1",
data: {
label: "Frame",
isFavorite: true,
},
});
expect(container?.querySelector('button[title="Duplicate"]')).toBeTruthy();
expect(container?.querySelector('button[title="Delete"]')).toBeTruthy();
});
it("applies favorite chrome marker on favorite nodes", async () => {
await renderWrapper({ label: "Frame", isFavorite: true }, true);
const rootElement = container?.firstElementChild;
expect(rootElement?.className).toContain("node-favorite-chrome");
});
});

View File

@@ -1,4 +1,6 @@
import { describe, expect, it, vi } from "vitest"; import { describe, expect, it, vi } from "vitest";
import { readFileSync } from "node:fs";
import { resolve } from "node:path";
import type { Id } from "@/convex/_generated/dataModel"; import type { Id } from "@/convex/_generated/dataModel";
import { createCanvasSyncEngineController } from "@/components/canvas/use-canvas-sync-engine"; import { createCanvasSyncEngineController } from "@/components/canvas/use-canvas-sync-engine";
@@ -75,6 +77,67 @@ describe("useCanvasSyncEngine", () => {
expect(controller.pendingDataAfterCreateRef.current.has("req-2")).toBe(false); expect(controller.pendingDataAfterCreateRef.current.has("req-2")).toBe(false);
}); });
it("keeps favorite fields in pinned and deferred optimistic data updates", async () => {
const enqueueSyncMutation = vi.fn(async () => undefined);
const controller = createCanvasSyncEngineController({
canvasId: asCanvasId("canvas-1"),
isSyncOnline: true,
getEnqueueSyncMutation: () => enqueueSyncMutation,
getRunBatchRemoveNodes: () => vi.fn(async () => undefined),
getRunSplitEdgeAtExistingNode: () => vi.fn(async () => undefined),
});
const favoritePayload = {
storageId: "storage-next",
filename: "hero.png",
isFavorite: true,
};
await controller.queueNodeDataUpdate({
nodeId: asNodeId("optimistic_req-favorite"),
data: favoritePayload,
});
expect(
controller.pendingLocalNodeDataUntilConvexMatchesRef.current.get(
"optimistic_req-favorite",
),
).toEqual(favoritePayload);
await controller.syncPendingMoveForClientRequest(
"req-favorite",
asNodeId("node-favorite"),
);
expect(enqueueSyncMutation).toHaveBeenCalledWith("updateData", {
nodeId: asNodeId("node-favorite"),
data: favoritePayload,
});
expect(
controller.pendingLocalNodeDataUntilConvexMatchesRef.current.get("node-favorite"),
).toEqual(favoritePayload);
});
it("uses favorite-preserving payloads in media replacement write paths", () => {
const imageNodeSource = readFileSync(
resolve(process.cwd(), "components/canvas/nodes/image-node.tsx"),
"utf8",
);
const assetBrowserSource = readFileSync(
resolve(process.cwd(), "components/canvas/asset-browser-panel.tsx"),
"utf8",
);
const videoBrowserSource = readFileSync(
resolve(process.cwd(), "components/canvas/video-browser-panel.tsx"),
"utf8",
);
expect(imageNodeSource).toContain("preserveNodeFavorite(");
expect(assetBrowserSource).toContain("preserveNodeFavorite(");
expect(videoBrowserSource).toContain("preserveNodeFavorite(");
});
it("pins local node data immediately when queueing an update", async () => { it("pins local node data immediately when queueing an update", async () => {
const enqueueSyncMutation = vi.fn(async () => undefined); const enqueueSyncMutation = vi.fn(async () => undefined);

View File

@@ -2,6 +2,7 @@
import React, { act, useEffect } from "react"; import React, { act, useEffect } from "react";
import { createRoot, type Root } from "react-dom/client"; import { createRoot, type Root } from "react-dom/client";
import { renderToStaticMarkup } from "react-dom/server";
import { afterEach, describe, expect, it, vi } from "vitest"; import { afterEach, describe, expect, it, vi } from "vitest";
import { import {
@@ -10,6 +11,7 @@ import {
useCanvasGraphPreviewOverrides, useCanvasGraphPreviewOverrides,
} from "@/components/canvas/canvas-graph-context"; } from "@/components/canvas/canvas-graph-context";
import { useNodeLocalData } from "@/components/canvas/nodes/use-node-local-data"; import { useNodeLocalData } from "@/components/canvas/nodes/use-node-local-data";
import { readNodeFavorite } from "@/lib/canvas-node-favorite";
type AdjustmentData = { type AdjustmentData = {
exposure: number; exposure: number;
@@ -582,3 +584,159 @@ describe("useNodeLocalData preview overrides", () => {
vi.useRealTimers(); vi.useRealTimers();
}); });
}); });
describe("favorite retention in strict local node flows", () => {
type LocalDataConfig = {
normalize: (value: unknown) => unknown;
onSave: (value: unknown) => Promise<void> | void;
data: unknown;
};
const createNodeProps = (data: Record<string, unknown>) =>
({
id: "node-1",
data,
selected: false,
width: 320,
height: 240,
dragging: false,
zIndex: 0,
isConnectable: true,
type: "curves",
xPos: 0,
yPos: 0,
positionAbsoluteX: 0,
positionAbsoluteY: 0,
}) as const;
const setupNodeHarness = async (modulePath: string) => {
vi.resetModules();
let capturedConfig: LocalDataConfig | null = null;
const queueNodeDataUpdate = vi.fn(async () => undefined);
vi.doMock("@/components/canvas/canvas-sync-context", () => ({
useCanvasSync: () => ({
queueNodeDataUpdate,
queueNodeResize: vi.fn(async () => undefined),
status: { isOffline: false },
}),
}));
vi.doMock("@/components/canvas/canvas-graph-context", () => ({
useCanvasGraph: () => ({ nodes: [], edges: [], previewNodeDataOverrides: new Map() }),
}));
vi.doMock("@/components/canvas/canvas-presets-context", () => ({
useCanvasAdjustmentPresets: () => [],
useSaveCanvasAdjustmentPreset: () => vi.fn(async () => undefined),
}));
vi.doMock("@/components/canvas/nodes/base-node-wrapper", () => ({
default: ({ children }: { children: React.ReactNode }) => <div>{children}</div>,
}));
vi.doMock("@/components/canvas/nodes/adjustment-preview", () => ({
default: () => null,
}));
vi.doMock("@/components/ui/select", () => ({
Select: ({ children }: { children: React.ReactNode }) => <>{children}</>,
SelectContent: ({ children }: { children: React.ReactNode }) => <>{children}</>,
SelectItem: ({ children }: { children: React.ReactNode }) => <>{children}</>,
SelectTrigger: ({ children }: { children: React.ReactNode }) => <>{children}</>,
SelectValue: () => null,
}));
vi.doMock("@/src/components/tool-ui/parameter-slider", () => ({
ParameterSlider: () => null,
}));
vi.doMock("@/hooks/use-pipeline-preview", () => ({
usePipelinePreview: () => ({
canvasRef: { current: null },
hasSource: false,
isRendering: false,
previewAspectRatio: 1,
histogram: null,
error: null,
}),
}));
vi.doMock("@/lib/canvas-render-preview", () => ({
collectPipelineFromGraph: () => [],
getSourceImageFromGraph: () => null,
shouldFastPathPreviewPipeline: () => false,
findSourceNodeFromGraph: () => null,
resolveRenderPreviewInputFromGraph: () => ({ sourceUrl: null, steps: [] }),
}));
vi.doMock("@/components/ui/dialog", () => ({
Dialog: ({ children }: { children: React.ReactNode }) => <>{children}</>,
DialogContent: ({ children }: { children: React.ReactNode }) => <>{children}</>,
DialogTitle: ({ children }: { children: React.ReactNode }) => <>{children}</>,
}));
vi.doMock("@/components/canvas/nodes/use-node-local-data", () => ({
useNodeLocalData: (config: LocalDataConfig) => {
capturedConfig = config;
return {
localData: config.normalize(config.data),
applyLocalData: vi.fn(),
updateLocalData: vi.fn(),
};
},
}));
vi.doMock("next-intl", () => ({
useTranslations: () => () => "",
}));
vi.doMock("@/lib/toast", () => ({
toast: { success: vi.fn() },
}));
vi.doMock("@xyflow/react", () => ({
Handle: () => null,
Position: { Left: "left", Right: "right" },
}));
const importedModule = (await import(modulePath)) as {
default: React.ComponentType<Record<string, unknown>>;
};
renderToStaticMarkup(React.createElement(importedModule.default, createNodeProps({ isFavorite: true })));
if (capturedConfig === null) {
throw new Error("useNodeLocalData config was not captured");
}
const resolvedConfig = capturedConfig as LocalDataConfig;
return { capturedConfig: resolvedConfig, queueNodeDataUpdate };
};
it("preserves isFavorite in normalized local data and saved payloads", async () => {
const targets = [
"@/components/canvas/nodes/crop-node",
"@/components/canvas/nodes/curves-node",
"@/components/canvas/nodes/color-adjust-node",
"@/components/canvas/nodes/light-adjust-node",
"@/components/canvas/nodes/detail-adjust-node",
];
for (const modulePath of targets) {
const { capturedConfig, queueNodeDataUpdate } = await setupNodeHarness(modulePath);
const normalizedWithFavorite = capturedConfig.normalize({ isFavorite: true });
expect(readNodeFavorite(normalizedWithFavorite)).toBe(true);
const strictNextData = capturedConfig.normalize({});
expect(readNodeFavorite(strictNextData)).toBe(false);
await capturedConfig.onSave(strictNextData);
const queueCalls = (queueNodeDataUpdate as unknown as { mock: { calls: Array<Array<unknown>> } })
.mock.calls;
const queuedPayload = queueCalls[0]?.[0] as { data?: unknown } | undefined;
expect(readNodeFavorite(queuedPayload?.data)).toBe(true);
}
});
});

View File

@@ -11,6 +11,7 @@ import {
} from "react"; } from "react";
import { createPortal } from "react-dom"; import { createPortal } from "react-dom";
import { useAction } from "convex/react"; import { useAction } from "convex/react";
import { useReactFlow } from "@xyflow/react";
import { X, Search, Loader2, AlertCircle } from "lucide-react"; import { X, Search, Loader2, AlertCircle } from "lucide-react";
import { api } from "@/convex/_generated/api"; import { api } from "@/convex/_generated/api";
import type { Id } from "@/convex/_generated/dataModel"; import type { Id } from "@/convex/_generated/dataModel";
@@ -19,6 +20,7 @@ import { Button } from "@/components/ui/button";
import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs"; import { Tabs, TabsList, TabsTrigger } from "@/components/ui/tabs";
import { Badge } from "@/components/ui/badge"; import { Badge } from "@/components/ui/badge";
import { computeMediaNodeSize } from "@/lib/canvas-utils"; import { computeMediaNodeSize } from "@/lib/canvas-utils";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { useCanvasSync } from "@/components/canvas/canvas-sync-context"; import { useCanvasSync } from "@/components/canvas/canvas-sync-context";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
@@ -90,6 +92,7 @@ export function AssetBrowserPanel({
const [selectingAssetKey, setSelectingAssetKey] = useState<string | null>(null); const [selectingAssetKey, setSelectingAssetKey] = useState<string | null>(null);
const searchFreepik = useAction(api.freepik.search); const searchFreepik = useAction(api.freepik.search);
const { getNode } = useReactFlow();
const { queueNodeDataUpdate, queueNodeResize, status } = useCanvasSync(); const { queueNodeDataUpdate, queueNodeResize, status } = useCanvasSync();
const shouldSkipInitialSearchRef = useRef(Boolean(initialState?.results?.length)); const shouldSkipInitialSearchRef = useRef(Boolean(initialState?.results?.length));
const requestSequenceRef = useRef(0); const requestSequenceRef = useRef(0);
@@ -198,9 +201,11 @@ export function AssetBrowserPanel({
const assetKey = `${asset.assetType}-${asset.id}`; const assetKey = `${asset.assetType}-${asset.id}`;
setSelectingAssetKey(assetKey); setSelectingAssetKey(assetKey);
try { try {
const currentNode = getNode(nodeId);
await queueNodeDataUpdate({ await queueNodeDataUpdate({
nodeId: nodeId as Id<"nodes">, nodeId: nodeId as Id<"nodes">,
data: { data: preserveNodeFavorite(
{
assetId: asset.id, assetId: asset.id,
assetType: asset.assetType, assetType: asset.assetType,
title: asset.title, title: asset.title,
@@ -214,6 +219,8 @@ export function AssetBrowserPanel({
orientation: asset.orientation, orientation: asset.orientation,
canvasId, canvasId,
}, },
currentNode?.data,
),
}); });
const targetSize = computeMediaNodeSize("asset", { const targetSize = computeMediaNodeSize("asset", {
@@ -234,7 +241,7 @@ export function AssetBrowserPanel({
setSelectingAssetKey(null); setSelectingAssetKey(null);
} }
}, },
[canvasId, isSelecting, nodeId, onClose, queueNodeDataUpdate, queueNodeResize, status.isOffline], [canvasId, getNode, isSelecting, nodeId, onClose, queueNodeDataUpdate, queueNodeResize, status.isOffline],
); );
const handlePreviousPage = useCallback(() => { const handlePreviousPage = useCallback(() => {

View File

@@ -25,6 +25,7 @@ import {
normalizeColorAdjustData, normalizeColorAdjustData,
type ColorAdjustData, type ColorAdjustData,
} from "@/lib/image-pipeline/adjustment-types"; } from "@/lib/image-pipeline/adjustment-types";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { COLOR_PRESETS } from "@/lib/image-pipeline/presets"; import { COLOR_PRESETS } from "@/lib/image-pipeline/presets";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
@@ -53,10 +54,13 @@ export default function ColorAdjustNode({ id, data, selected, width }: NodeProps
const [presetSelection, setPresetSelection] = useState("custom"); const [presetSelection, setPresetSelection] = useState("custom");
const normalizeData = useCallback( const normalizeData = useCallback(
(value: unknown) => (value: unknown) =>
preserveNodeFavorite(
normalizeColorAdjustData({ normalizeColorAdjustData({
...cloneAdjustmentData(DEFAULT_COLOR_ADJUST_DATA), ...cloneAdjustmentData(DEFAULT_COLOR_ADJUST_DATA),
...(value as Record<string, unknown>), ...(value as Record<string, unknown>),
}), }),
value,
) as ColorAdjustData,
[], [],
); );
const { localData, applyLocalData, updateLocalData } = useNodeLocalData<ColorAdjustData>({ const { localData, applyLocalData, updateLocalData } = useNodeLocalData<ColorAdjustData>({
@@ -67,7 +71,7 @@ export default function ColorAdjustNode({ id, data, selected, width }: NodeProps
onSave: (next) => onSave: (next) =>
queueNodeDataUpdate({ queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: next, data: preserveNodeFavorite(next, data),
}), }),
debugLabel: "color-adjust", debugLabel: "color-adjust",
}); });

View File

@@ -21,6 +21,7 @@ import {
type CropNodeData, type CropNodeData,
type CropResizeMode, type CropResizeMode,
} from "@/lib/image-pipeline/crop-node-data"; } from "@/lib/image-pipeline/crop-node-data";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import type { Id } from "@/convex/_generated/dataModel"; import type { Id } from "@/convex/_generated/dataModel";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
@@ -188,7 +189,11 @@ export default function CropNode({ id, data, selected, width }: NodeProps<CropNo
const { queueNodeDataUpdate } = useCanvasSync(); const { queueNodeDataUpdate } = useCanvasSync();
const graph = useCanvasGraph(); const graph = useCanvasGraph();
const normalizeData = useCallback((value: unknown) => normalizeCropNodeData(value), []); const normalizeData = useCallback(
(value: unknown) =>
preserveNodeFavorite(normalizeCropNodeData(value), value) as CropNodeData,
[],
);
const previewAreaRef = useRef<HTMLDivElement | null>(null); const previewAreaRef = useRef<HTMLDivElement | null>(null);
const interactionRef = useRef<CropInteractionState | null>(null); const interactionRef = useRef<CropInteractionState | null>(null);
const { localData, updateLocalData } = useNodeLocalData<CropNodeData>({ const { localData, updateLocalData } = useNodeLocalData<CropNodeData>({
@@ -199,7 +204,7 @@ export default function CropNode({ id, data, selected, width }: NodeProps<CropNo
onSave: (next) => onSave: (next) =>
queueNodeDataUpdate({ queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: next, data: preserveNodeFavorite(next, data),
}), }),
debugLabel: "crop", debugLabel: "crop",
}); });

View File

@@ -25,6 +25,7 @@ import {
normalizeCurvesData, normalizeCurvesData,
type CurvesData, type CurvesData,
} from "@/lib/image-pipeline/adjustment-types"; } from "@/lib/image-pipeline/adjustment-types";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { CURVE_PRESETS } from "@/lib/image-pipeline/presets"; import { CURVE_PRESETS } from "@/lib/image-pipeline/presets";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
@@ -53,10 +54,13 @@ export default function CurvesNode({ id, data, selected, width }: NodeProps<Curv
const [presetSelection, setPresetSelection] = useState("custom"); const [presetSelection, setPresetSelection] = useState("custom");
const normalizeData = useCallback( const normalizeData = useCallback(
(value: unknown) => (value: unknown) =>
preserveNodeFavorite(
normalizeCurvesData({ normalizeCurvesData({
...cloneAdjustmentData(DEFAULT_CURVES_DATA), ...cloneAdjustmentData(DEFAULT_CURVES_DATA),
...(value as Record<string, unknown>), ...(value as Record<string, unknown>),
}), }),
value,
) as CurvesData,
[], [],
); );
const { localData, applyLocalData, updateLocalData } = useNodeLocalData<CurvesData>({ const { localData, applyLocalData, updateLocalData } = useNodeLocalData<CurvesData>({
@@ -67,7 +71,7 @@ export default function CurvesNode({ id, data, selected, width }: NodeProps<Curv
onSave: (next) => onSave: (next) =>
queueNodeDataUpdate({ queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: next, data: preserveNodeFavorite(next, data),
}), }),
debugLabel: "curves", debugLabel: "curves",
}); });

View File

@@ -25,6 +25,7 @@ import {
normalizeDetailAdjustData, normalizeDetailAdjustData,
type DetailAdjustData, type DetailAdjustData,
} from "@/lib/image-pipeline/adjustment-types"; } from "@/lib/image-pipeline/adjustment-types";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { DETAIL_PRESETS } from "@/lib/image-pipeline/presets"; import { DETAIL_PRESETS } from "@/lib/image-pipeline/presets";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
@@ -53,10 +54,13 @@ export default function DetailAdjustNode({ id, data, selected, width }: NodeProp
const [presetSelection, setPresetSelection] = useState("custom"); const [presetSelection, setPresetSelection] = useState("custom");
const normalizeData = useCallback( const normalizeData = useCallback(
(value: unknown) => (value: unknown) =>
preserveNodeFavorite(
normalizeDetailAdjustData({ normalizeDetailAdjustData({
...cloneAdjustmentData(DEFAULT_DETAIL_ADJUST_DATA), ...cloneAdjustmentData(DEFAULT_DETAIL_ADJUST_DATA),
...(value as Record<string, unknown>), ...(value as Record<string, unknown>),
}), }),
value,
) as DetailAdjustData,
[], [],
); );
const { localData, applyLocalData, updateLocalData } = useNodeLocalData<DetailAdjustData>({ const { localData, applyLocalData, updateLocalData } = useNodeLocalData<DetailAdjustData>({
@@ -67,7 +71,7 @@ export default function DetailAdjustNode({ id, data, selected, width }: NodeProp
onSave: (next) => onSave: (next) =>
queueNodeDataUpdate({ queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: next, data: preserveNodeFavorite(next, data),
}), }),
debugLabel: "detail-adjust", debugLabel: "detail-adjust",
}); });

View File

@@ -36,6 +36,7 @@ import {
createCompressedImagePreview, createCompressedImagePreview,
getImageDimensions, getImageDimensions,
} from "@/components/canvas/canvas-media-utils"; } from "@/components/canvas/canvas-media-utils";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
const ALLOWED_IMAGE_TYPES = new Set([ const ALLOWED_IMAGE_TYPES = new Set([
"image/png", "image/png",
@@ -302,13 +303,16 @@ export default function ImageNode({
await queueNodeDataUpdate({ await queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: { data: preserveNodeFavorite(
{
storageId, storageId,
...(previewUpload ?? {}), ...(previewUpload ?? {}),
filename: file.name, filename: file.name,
mimeType: file.type, mimeType: file.type,
...(dimensions ? { width: dimensions.width, height: dimensions.height } : {}), ...(dimensions ? { width: dimensions.width, height: dimensions.height } : {}),
}, },
data,
),
}); });
if (dimensions) { if (dimensions) {
@@ -354,6 +358,7 @@ export default function ImageNode({
} }
}, },
[ [
data,
generateUploadUrl, generateUploadUrl,
id, id,
isUploading, isUploading,
@@ -377,7 +382,8 @@ export default function ImageNode({
try { try {
await queueNodeDataUpdate({ await queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: { data: preserveNodeFavorite(
{
storageId: item.storageId, storageId: item.storageId,
previewStorageId: item.previewStorageId, previewStorageId: item.previewStorageId,
filename: item.filename, filename: item.filename,
@@ -387,6 +393,8 @@ export default function ImageNode({
previewWidth: item.previewWidth, previewWidth: item.previewWidth,
previewHeight: item.previewHeight, previewHeight: item.previewHeight,
}, },
data,
),
}); });
setMediaLibraryPhase("syncing"); setMediaLibraryPhase("syncing");
@@ -414,7 +422,7 @@ export default function ImageNode({
); );
} }
}, },
[id, isNodeLoading, queueNodeDataUpdate, queueNodeResize, t], [data, id, isNodeLoading, queueNodeDataUpdate, queueNodeResize, t],
); );
const handleClick = useCallback(() => { const handleClick = useCallback(() => {

View File

@@ -25,6 +25,7 @@ import {
normalizeLightAdjustData, normalizeLightAdjustData,
type LightAdjustData, type LightAdjustData,
} from "@/lib/image-pipeline/adjustment-types"; } from "@/lib/image-pipeline/adjustment-types";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { LIGHT_PRESETS } from "@/lib/image-pipeline/presets"; import { LIGHT_PRESETS } from "@/lib/image-pipeline/presets";
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
@@ -53,10 +54,13 @@ export default function LightAdjustNode({ id, data, selected, width }: NodeProps
const [presetSelection, setPresetSelection] = useState("custom"); const [presetSelection, setPresetSelection] = useState("custom");
const normalizeData = useCallback( const normalizeData = useCallback(
(value: unknown) => (value: unknown) =>
preserveNodeFavorite(
normalizeLightAdjustData({ normalizeLightAdjustData({
...cloneAdjustmentData(DEFAULT_LIGHT_ADJUST_DATA), ...cloneAdjustmentData(DEFAULT_LIGHT_ADJUST_DATA),
...(value as Record<string, unknown>), ...(value as Record<string, unknown>),
}), }),
value,
) as LightAdjustData,
[], [],
); );
const { localData, applyLocalData, updateLocalData } = useNodeLocalData<LightAdjustData>({ const { localData, applyLocalData, updateLocalData } = useNodeLocalData<LightAdjustData>({
@@ -67,7 +71,7 @@ export default function LightAdjustNode({ id, data, selected, width }: NodeProps
onSave: (next) => onSave: (next) =>
queueNodeDataUpdate({ queueNodeDataUpdate({
nodeId: id as Id<"nodes">, nodeId: id as Id<"nodes">,
data: next, data: preserveNodeFavorite(next, data),
}), }),
debugLabel: "light-adjust", debugLabel: "light-adjust",
}); });

View File

@@ -26,6 +26,7 @@ import {
isPipelineAbortError, isPipelineAbortError,
renderFullWithWorkerFallback, renderFullWithWorkerFallback,
} from "@/lib/image-pipeline/worker-client"; } from "@/lib/image-pipeline/worker-client";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import type { Id } from "@/convex/_generated/dataModel"; import type { Id } from "@/convex/_generated/dataModel";
import { Dialog, DialogContent, DialogTitle } from "@/components/ui/dialog"; import { Dialog, DialogContent, DialogTitle } from "@/components/ui/dialog";
@@ -105,6 +106,7 @@ type PersistedRenderData = {
lastUploadFilename?: string; lastUploadFilename?: string;
lastUploadError?: string; lastUploadError?: string;
lastUploadErrorHash?: string; lastUploadErrorHash?: string;
isFavorite?: true;
}; };
const DEFAULT_OUTPUT_RESOLUTION: RenderResolutionOption = "original"; const DEFAULT_OUTPUT_RESOLUTION: RenderResolutionOption = "original";
@@ -348,7 +350,7 @@ function sanitizeRenderData(data: RenderNodeData): PersistedRenderData {
next.lastUploadErrorHash = data.lastUploadErrorHash; next.lastUploadErrorHash = data.lastUploadErrorHash;
} }
return next; return preserveNodeFavorite(next, data) as PersistedRenderData;
} }
function formatBytes(bytes: number | undefined): string { function formatBytes(bytes: number | undefined): string {
@@ -496,6 +498,7 @@ export default function RenderNode({ id, data, selected, width, height }: NodePr
); );
const steps = renderPreviewInput.steps; const steps = renderPreviewInput.steps;
const hasCropStep = useMemo(() => steps.some((step) => step.type === "crop"), [steps]);
const previewDebounceMs = shouldFastPathPreviewPipeline( const previewDebounceMs = shouldFastPathPreviewPipeline(
steps, steps,
graph.previewNodeDataOverrides, graph.previewNodeDataOverrides,
@@ -592,6 +595,15 @@ export default function RenderNode({ id, data, selected, width, height }: NodePr
}); });
const targetAspectRatio = useMemo(() => { const targetAspectRatio = useMemo(() => {
if (
hasCropStep &&
typeof previewAspectRatio === "number" &&
Number.isFinite(previewAspectRatio) &&
previewAspectRatio > 0
) {
return previewAspectRatio;
}
const sourceAspectRatio = resolveSourceAspectRatio(sourceNode); const sourceAspectRatio = resolveSourceAspectRatio(sourceNode);
if (sourceAspectRatio && Number.isFinite(sourceAspectRatio) && sourceAspectRatio > 0) { if (sourceAspectRatio && Number.isFinite(sourceAspectRatio) && sourceAspectRatio > 0) {
return sourceAspectRatio; return sourceAspectRatio;
@@ -606,7 +618,7 @@ export default function RenderNode({ id, data, selected, width, height }: NodePr
} }
return null; return null;
}, [previewAspectRatio, sourceNode]); }, [hasCropStep, previewAspectRatio, sourceNode]);
useEffect(() => { useEffect(() => {
if (!hasSource || targetAspectRatio === null) { if (!hasSource || targetAspectRatio === null) {

View File

@@ -11,6 +11,7 @@ import {
} from "react"; } from "react";
import { createPortal } from "react-dom"; import { createPortal } from "react-dom";
import { useAction } from "convex/react"; import { useAction } from "convex/react";
import { useReactFlow } from "@xyflow/react";
import { X, Search, Loader2, AlertCircle, Play, Pause } from "lucide-react"; import { X, Search, Loader2, AlertCircle, Play, Pause } from "lucide-react";
import { api } from "@/convex/_generated/api"; import { api } from "@/convex/_generated/api";
import type { Id } from "@/convex/_generated/dataModel"; import type { Id } from "@/convex/_generated/dataModel";
@@ -18,6 +19,7 @@ import { Input } from "@/components/ui/input";
import { Button } from "@/components/ui/button"; import { Button } from "@/components/ui/button";
import type { PexelsVideo, PexelsVideoFile } from "@/lib/pexels-types"; import type { PexelsVideo, PexelsVideoFile } from "@/lib/pexels-types";
import { pickPreviewVideoFile, pickVideoFile } from "@/lib/pexels-types"; import { pickPreviewVideoFile, pickVideoFile } from "@/lib/pexels-types";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
import { toast } from "@/lib/toast"; import { toast } from "@/lib/toast";
import { useCanvasSync } from "@/components/canvas/canvas-sync-context"; import { useCanvasSync } from "@/components/canvas/canvas-sync-context";
@@ -83,6 +85,7 @@ export function VideoBrowserPanel({
const searchVideos = useAction(api.pexels.searchVideos); const searchVideos = useAction(api.pexels.searchVideos);
const popularVideos = useAction(api.pexels.popularVideos); const popularVideos = useAction(api.pexels.popularVideos);
const { getNode } = useReactFlow();
const { queueNodeDataUpdate, queueNodeResize, status } = useCanvasSync(); const { queueNodeDataUpdate, queueNodeResize, status } = useCanvasSync();
const shouldSkipInitialSearchRef = useRef( const shouldSkipInitialSearchRef = useRef(
Boolean(initialState?.results?.length), Boolean(initialState?.results?.length),
@@ -216,9 +219,11 @@ export function VideoBrowserPanel({
return; return;
} }
try { try {
const currentNode = getNode(nodeId);
await queueNodeDataUpdate({ await queueNodeDataUpdate({
nodeId: nodeId as Id<"nodes">, nodeId: nodeId as Id<"nodes">,
data: { data: preserveNodeFavorite(
{
pexelsId: video.id, pexelsId: video.id,
mp4Url: file.link, mp4Url: file.link,
thumbnailUrl: video.image, thumbnailUrl: video.image,
@@ -232,6 +237,8 @@ export function VideoBrowserPanel({
}, },
canvasId, canvasId,
}, },
currentNode?.data,
),
}); });
// Auto-resize to match aspect ratio // Auto-resize to match aspect ratio
@@ -253,7 +260,7 @@ export function VideoBrowserPanel({
setSelectingVideoId(null); setSelectingVideoId(null);
} }
}, },
[canvasId, isSelecting, nodeId, onClose, queueNodeDataUpdate, queueNodeResize, status.isOffline], [canvasId, getNode, isSelecting, nodeId, onClose, queueNodeDataUpdate, queueNodeResize, status.isOffline],
); );
const handlePreviousPage = useCallback(() => { const handlePreviousPage = useCallback(() => {

View File

@@ -10,6 +10,7 @@ import {
} from "../lib/canvas-connection-policy"; } from "../lib/canvas-connection-policy";
import { nodeTypeValidator } from "./node_type_validator"; import { nodeTypeValidator } from "./node_type_validator";
import { normalizeCropNodeData } from "../lib/image-pipeline/crop-node-data"; import { normalizeCropNodeData } from "../lib/image-pipeline/crop-node-data";
import { preserveNodeFavorite } from "../lib/canvas-node-favorite";
// ============================================================================ // ============================================================================
// Interne Helpers // Interne Helpers
@@ -393,9 +394,12 @@ function normalizeNodeDataForWrite(
data: unknown, data: unknown,
): unknown { ): unknown {
if (nodeType === "crop") { if (nodeType === "crop") {
return normalizeCropNodeData(data, { return preserveNodeFavorite(
normalizeCropNodeData(data, {
rejectDisallowedPayloadFields: true, rejectDisallowedPayloadFields: true,
}); }),
data,
);
} }
if (!isAdjustmentNodeType(nodeType)) { if (!isAdjustmentNodeType(nodeType)) {
@@ -407,11 +411,11 @@ function normalizeNodeDataForWrite(
} }
if (nodeType === "render") { if (nodeType === "render") {
return normalizeRenderData(data); return preserveNodeFavorite(normalizeRenderData(data), data);
} }
assertNoAdjustmentImagePayload(nodeType, data); assertNoAdjustmentImagePayload(nodeType, data);
return data; return preserveNodeFavorite(data, data);
} }
async function countIncomingEdges( async function countIncomingEdges(

View File

@@ -0,0 +1,36 @@
function isRecord(value: unknown): value is Record<string, unknown> {
return typeof value === "object" && value !== null && !Array.isArray(value);
}
function toRecord(value: unknown): Record<string, unknown> {
return isRecord(value) ? value : {};
}
export function readNodeFavorite(data: unknown): boolean {
const source = toRecord(data);
return source.isFavorite === true;
}
export function setNodeFavorite(
nextValue: boolean,
currentData: unknown,
): Record<string, unknown> {
const source = toRecord(currentData);
if (nextValue) {
return {
...source,
isFavorite: true,
};
}
const { isFavorite: _isFavorite, ...rest } = source;
return rest;
}
export function preserveNodeFavorite(
nextData: unknown,
previousData: unknown,
): Record<string, unknown> {
return setNodeFavorite(readNodeFavorite(previousData), nextData);
}

View File

@@ -0,0 +1,54 @@
import { describe, expect, it } from "vitest";
import {
preserveNodeFavorite,
readNodeFavorite,
setNodeFavorite,
} from "@/lib/canvas-node-favorite";
describe("canvas node favorite helpers", () => {
it("reads favorite from object data", () => {
expect(readNodeFavorite({ isFavorite: true })).toBe(true);
});
it("returns false when favorite flag is missing", () => {
expect(readNodeFavorite({})).toBe(false);
});
it("persists favorite when enabled", () => {
expect(setNodeFavorite(true, { label: "Frame" })).toEqual({
label: "Frame",
isFavorite: true,
});
});
it("removes favorite key when disabled", () => {
expect(setNodeFavorite(false, { label: "Frame", isFavorite: true })).toEqual({
label: "Frame",
});
});
it("preserves favorite after strict normalization", () => {
expect(
preserveNodeFavorite(
{
crop: {
x: 0,
y: 0,
width: 1,
height: 1,
},
},
{ isFavorite: true },
),
).toEqual({
crop: {
x: 0,
y: 0,
width: 1,
height: 1,
},
isFavorite: true,
});
});
});

View File

@@ -4,6 +4,7 @@ import {
DEFAULT_CROP_NODE_DATA, DEFAULT_CROP_NODE_DATA,
normalizeCropNodeData, normalizeCropNodeData,
} from "@/lib/image-pipeline/crop-node-data"; } from "@/lib/image-pipeline/crop-node-data";
import { preserveNodeFavorite } from "@/lib/canvas-node-favorite";
describe("crop node data validation", () => { describe("crop node data validation", () => {
it("normalizes and clamps crop rectangle data", () => { it("normalizes and clamps crop rectangle data", () => {
@@ -81,4 +82,24 @@ describe("crop node data validation", () => {
), ),
).toThrow("Crop node accepts parameter data only. 'imageData' is not allowed in data."); ).toThrow("Crop node accepts parameter data only. 'imageData' is not allowed in data.");
}); });
it("preserves favorite after strict crop normalization", () => {
const normalized = normalizeCropNodeData(
{
...DEFAULT_CROP_NODE_DATA,
isFavorite: true,
},
{ rejectDisallowedPayloadFields: true },
);
expect(
preserveNodeFavorite(normalized, {
...DEFAULT_CROP_NODE_DATA,
isFavorite: true,
}),
).toEqual({
...DEFAULT_CROP_NODE_DATA,
isFavorite: true,
});
});
}); });

View File

@@ -25,6 +25,7 @@ export default defineConfig({
"components/canvas/__tests__/use-canvas-node-interactions.test.tsx", "components/canvas/__tests__/use-canvas-node-interactions.test.tsx",
"components/canvas/__tests__/canvas-delete-handlers.test.tsx", "components/canvas/__tests__/canvas-delete-handlers.test.tsx",
"components/canvas/__tests__/canvas-media-utils.test.ts", "components/canvas/__tests__/canvas-media-utils.test.ts",
"components/canvas/__tests__/base-node-wrapper.test.tsx",
"components/canvas/__tests__/use-node-local-data.test.tsx", "components/canvas/__tests__/use-node-local-data.test.tsx",
"components/canvas/__tests__/use-canvas-sync-engine.test.ts", "components/canvas/__tests__/use-canvas-sync-engine.test.ts",
"components/canvas/__tests__/use-canvas-sync-engine-hook.test.tsx", "components/canvas/__tests__/use-canvas-sync-engine-hook.test.tsx",