import { fireEvent, render, screen, waitFor } from "@testing-library/react"; import { act } from "react"; import { afterEach, beforeEach, describe, expect, it, vi } from "vitest"; import { App } from "../../src/app/App"; import { createEditorStore } from "../../src/app/editor-store"; import { createBoxBrush } from "../../src/document/brushes"; import { createEmptySceneDocument } from "../../src/document/scene-document"; const { MockViewportHost, viewportHostInstances } = vi.hoisted(() => { const viewportHostInstances: Array<{ setTransformCommitHandler: ReturnType; setPanelId: ReturnType; mount: ReturnType; dispose: ReturnType; updateWorld: ReturnType; updateSimulation: ReturnType; updateAssets: ReturnType; updateDocument: ReturnType; updateSelection: ReturnType; setRenderEnabled: ReturnType; setGridVisible: ReturnType; setViewMode: ReturnType; setDisplayMode: ReturnType; setCameraState: ReturnType; setBrushSelectionChangeHandler: ReturnType; setCameraStateChangeHandler: ReturnType; setCreationPreviewChangeHandler: ReturnType; setCreationCommitHandler: ReturnType; setTransformSessionChangeHandler: ReturnType; setTransformPreviewChangeHandler: ReturnType; setTransformCancelHandler: ReturnType; setWhiteboxHoverLabelChangeHandler: ReturnType; setWhiteboxSelectionMode: ReturnType; setWhiteboxSnapSettings: ReturnType; setToolMode: ReturnType; setCreationPreview: ReturnType; setTransformSession: ReturnType; focusSelection: ReturnType; }> = []; class MockViewportHost { setPanelId = vi.fn(); mount = vi.fn(); dispose = vi.fn(); updateWorld = vi.fn(); updateSimulation = vi.fn(); updateAssets = vi.fn(); updateDocument = vi.fn(); updateSelection = vi.fn(); setRenderEnabled = vi.fn(); setGridVisible = vi.fn(); setViewMode = vi.fn(); setDisplayMode = vi.fn(); setCameraState = vi.fn(); setBrushSelectionChangeHandler = vi.fn(); setCameraStateChangeHandler = vi.fn(); setCreationPreviewChangeHandler = vi.fn(); setCreationCommitHandler = vi.fn(); setTransformSessionChangeHandler = vi.fn(); setTransformPreviewChangeHandler = vi.fn(); setTransformCommitHandler = vi.fn(); setTransformCancelHandler = vi.fn(); setWhiteboxHoverLabelChangeHandler = vi.fn(); setWhiteboxSelectionMode = vi.fn(); setWhiteboxSnapSettings = vi.fn(); setToolMode = vi.fn(); setCreationPreview = vi.fn(); setTransformSession = vi.fn(); focusSelection = vi.fn(); constructor() { viewportHostInstances.push(this); } } return { MockViewportHost, viewportHostInstances }; }); vi.mock("../../src/viewport-three/viewport-host", () => ({ ViewportHost: MockViewportHost })); vi.mock("../../src/assets/project-asset-storage", () => ({ getBrowserProjectAssetStorageAccess: vi.fn(async () => ({ storage: null, diagnostic: null })) })); describe("water volume integration", () => { beforeEach(() => { viewportHostInstances.length = 0; vi.spyOn(HTMLCanvasElement.prototype, "getContext").mockImplementation(() => ({}) as never); }); afterEach(() => { vi.restoreAllMocks(); }); it("keeps the selected water color after committing the picker change", async () => { const brush = createBoxBrush({ id: "brush-water-sidebar", volume: { mode: "water", water: { colorHex: "#4da6d9", surfaceOpacity: 0.55, waveStrength: 0.35, foamContactLimit: 6, surfaceDisplacementEnabled: false } } }); const store = createEditorStore({ initialDocument: { ...createEmptySceneDocument({ name: "Water Sidebar Test" }), brushes: { [brush.id]: brush } } }); render(); await waitFor(() => { expect(viewportHostInstances.length).toBeGreaterThan(0); expect(viewportHostInstances[0]?.setTransformCommitHandler).toHaveBeenCalled(); }); act(() => { store.setSelection({ kind: "brushes", ids: [brush.id] }); }); const input = (await screen.findByTestId("brush-water-color")) as HTMLInputElement; act(() => { fireEvent.change(input, { target: { value: "#12a4ff" } }); }); await waitFor(() => { expect(store.getState().document.brushes[brush.id]?.volume).toEqual({ mode: "water", water: { colorHex: "#12a4ff", surfaceOpacity: 0.55, waveStrength: 0.35, foamContactLimit: 6, surfaceDisplacementEnabled: false } }); }); expect(input.value).toBe("#12a4ff"); }); it("keeps vertical surface motion enabled after toggling the checkbox", async () => { const brush = createBoxBrush({ id: "brush-water-displacement-toggle", volume: { mode: "water", water: { colorHex: "#4da6d9", surfaceOpacity: 0.55, waveStrength: 0.35, foamContactLimit: 6, surfaceDisplacementEnabled: false } } }); const store = createEditorStore({ initialDocument: { ...createEmptySceneDocument({ name: "Water Displacement Toggle Test" }), brushes: { [brush.id]: brush } } }); render(); await waitFor(() => { expect(viewportHostInstances.length).toBeGreaterThan(0); }); act(() => { store.setSelection({ kind: "brushes", ids: [brush.id] }); }); const checkbox = (await screen.findByTestId("brush-water-surface-displacement-enabled")) as HTMLInputElement; expect(checkbox.checked).toBe(false); act(() => { fireEvent.click(checkbox); }); await waitFor(() => { expect(store.getState().document.brushes[brush.id]?.volume).toEqual({ mode: "water", water: { colorHex: "#4da6d9", surfaceOpacity: 0.55, waveStrength: 0.35, foamContactLimit: 6, surfaceDisplacementEnabled: true } }); }); expect(checkbox.checked).toBe(true); }); });