Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add controls feature to tweak the visualizer easier #1180

Merged
merged 34 commits into from
Feb 27, 2024
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
4621302
feat(visualizer): randomize sinusoidal period times
VmMad Feb 21, 2024
d474eab
chore: remove comments
VmMad Feb 21, 2024
02bb2c3
feat(visualizer): randomize sinusoid amplitudes
VmMad Feb 21, 2024
bf80194
refactor: rename HALF_PERIOD constants to PERIOD
VmMad Feb 22, 2024
8d6c88d
Merge branch 'feat/nova-visualizer/randomize-period' into feat/nova-v…
VmMad Feb 22, 2024
58bd427
Merge branch 'dev' into feat/nova-visualizer/paused-data-output
VmMad Feb 22, 2024
f3013c7
Merge branch 'dev' into feat/nova-visualizer/randomize-amplitude
VmMad Feb 22, 2024
df6d034
feat(visualizer): add tangle tilting factor
VmMad Feb 22, 2024
ca31490
Merge branch 'feat/nova-visualizer/randomize-amplitude' into feat/nov…
VmMad Feb 22, 2024
0fa4dc0
Merge branch 'dev' into feat/nova-visualizer/add-emitter-tilting
VmMad Feb 22, 2024
ac9bd5d
feat: improve spray
VmMad Feb 22, 2024
9cefde3
feat: controls for visualizer (PoC)
panteleymonchuk Feb 22, 2024
9778b0b
feat: add all fields, cover by feature flag.
panteleymonchuk Feb 23, 2024
2bd7073
feat: control values by list.
panteleymonchuk Feb 23, 2024
1d9fb10
feat: add "reset" btn.
panteleymonchuk Feb 23, 2024
eb79ebf
fix: camera zoom
VmMad Feb 23, 2024
fb0d405
Merge branch 'dev' into feat/nova-visualizer/add-emitter-tilting
VmMad Feb 23, 2024
0dd05cc
fix: import `features` bug
panteleymonchuk Feb 23, 2024
d2c64d9
Merge remote-tracking branch 'origin/feat/nova-visualizer/add-emitter…
panteleymonchuk Feb 23, 2024
e5371fe
feat: dynamic zoom.
panteleymonchuk Feb 23, 2024
d58ec08
fix: clean code.
panteleymonchuk Feb 23, 2024
2d7b6a1
fix: spray shape
VmMad Feb 23, 2024
79214b7
feat: commit suggestion for client/src/features/visualizer-threejs/Co…
panteleymonchuk Feb 26, 2024
d5947d4
feat: commit suggestion for client/src/features/visualizer-threejs/Co…
panteleymonchuk Feb 26, 2024
e04a744
fix: review comments
panteleymonchuk Feb 26, 2024
cba1854
Merge remote-tracking branch 'origin/feat/nova-visualizer/add-emitter…
panteleymonchuk Feb 26, 2024
c21b1d4
Merge branch 'dev' into feat/nova-visualizer/add-emitter-tilting
begonaalvarezd Feb 26, 2024
b63059c
Merge remote-tracking branch 'origin/feat/nova-visualizer/add-emitter…
panteleymonchuk Feb 26, 2024
561e632
fix: add label color for dark mode.
panteleymonchuk Feb 26, 2024
a8d945d
Merge branch 'dev' into feat/issues-1171-add-controls-for-visualizer
panteleymonchuk Feb 26, 2024
7b132b2
feat: add emitterSpeedMultiplier
panteleymonchuk Feb 22, 2024
7c32232
fix: change types.
panteleymonchuk Feb 27, 2024
eee1634
Merge remote-tracking branch 'origin/dev' into feat/issues-1171-add-c…
begonaalvarezd Feb 27, 2024
3cbb37c
feat: move visualizer config controls to bottom
begonaalvarezd Feb 27, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
54 changes: 34 additions & 20 deletions client/src/features/visualizer-threejs/CameraControls.tsx
Original file line number Diff line number Diff line change
@@ -1,45 +1,47 @@
import { CameraControls as DreiCameraControls } from "@react-three/drei";
import React, { useEffect } from "react";
import { getCameraAngles } from "./utils";
import React, { useEffect, useState } from "react";
import { useThree } from "@react-three/fiber";
import { CanvasElement } from "./enums";
import { useConfigStore } from "./store";
import { useTangleStore, useConfigStore } from "./store";
import { VISUALIZER_PADDINGS } from "./constants";
import { getCameraAngles } from "./utils";

const CAMERA_ANGLES = getCameraAngles();

const CameraControls = () => {
const { camera } = useThree();
const controls = React.useRef<DreiCameraControls>(null);
const [shouldLockZoom, setShouldLockZoom] = useState<boolean>(false);

const scene = useThree((state) => state.scene);
const zoom = useTangleStore((state) => state.zoom);
const forcedZoom = useTangleStore((state) => state.forcedZoom);
const mesh = scene.getObjectByName(CanvasElement.TangleWrapperMesh) as THREE.Mesh | undefined;
const canvasDimensions = useConfigStore((state) => state.dimensions);

/**
* Locks the camera zoom to the current zoom value.
*/
function lockCameraZoom(controls: DreiCameraControls) {
const zoom = controls.camera.zoom;
controls.maxZoom = zoom;
controls.minZoom = zoom;
}
useEffect(() => {
if (!forcedZoom) return;

/**
* Unlocks the camera zoom for free movement.
*/
function unlockCameraZoom(controls: DreiCameraControls) {
controls.maxZoom = Infinity;
controls.minZoom = 0.01;
}
(async () => {
if (camera && controls.current) {
controls.current.minZoom = forcedZoom;
controls.current.minZoom = forcedZoom;
await controls.current.zoomTo(forcedZoom, true);
}
})();
}, [forcedZoom]);

/**
* Fits the camera to the TangleMesh.
*/
function fitCameraToTangle(controls: DreiCameraControls | null, mesh?: THREE.Mesh) {
if (controls && mesh) {
unlockCameraZoom(controls);
const previousZoom = controls.camera.zoom;
controls.minZoom = 0.01;
controls.maxZoom = Infinity;
controls.fitToBox(mesh, false, { ...VISUALIZER_PADDINGS });
lockCameraZoom(controls);
controls.minZoom = previousZoom;
controls.maxZoom = previousZoom;
}
}

Expand All @@ -53,7 +55,9 @@ const CameraControls = () => {
const camera = controls.current?.camera;
const renderVerticalScene = canvasDimensions.width < canvasDimensions.height;
const cameraUp: [number, number, number] = renderVerticalScene ? [1, 0, 0] : [0, 1, 0];
setShouldLockZoom(false);
camera.up.set(...cameraUp);
setShouldLockZoom(true);
}
}, [canvasDimensions, controls, mesh]);

Expand All @@ -70,6 +74,16 @@ const CameraControls = () => {
};
}, [controls, mesh]);

/**
* Locks the camera zoom to the current zoom value.
*/
useEffect(() => {
if (controls.current) {
controls.current.maxZoom = shouldLockZoom ? zoom : Infinity;
controls.current.minZoom = shouldLockZoom ? zoom : 0.01;
}
}, [controls.current, shouldLockZoom, zoom]);

return <DreiCameraControls ref={controls} makeDefault {...CAMERA_ANGLES} />;
};

Expand Down
236 changes: 236 additions & 0 deletions client/src/features/visualizer-threejs/ConfigControls.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
import React, { useState } from "react";
import {
MIN_SINUSOID_PERIOD,
MAX_SINUSOID_PERIOD,
MIN_SINUSOID_AMPLITUDE,
MAX_SINUSOID_AMPLITUDE,
MIN_TILT_FACTOR_DEGREES,
MAX_TILT_FACTOR_DEGREES,
TILT_DURATION_SECONDS,
features,
} from "./constants";
import "./Controls.scss";
VmMad marked this conversation as resolved.
Show resolved Hide resolved
import { useTangleStore } from "~features/visualizer-threejs/store";

interface IControlsVisualiser {
MIN_SINUSOID_PERIOD: number;
MAX_SINUSOID_PERIOD: number;
MIN_SINUSOID_AMPLITUDE: number;
MAX_SINUSOID_AMPLITUDE: number;
MIN_TILT_FACTOR_DEGREES: number;
MAX_TILT_FACTOR_DEGREES: number;
TILT_DURATION_SECONDS: number;
}
panteleymonchuk marked this conversation as resolved.
Show resolved Hide resolved

const defaultControlsVisualiser: IControlsVisualiser = {
panteleymonchuk marked this conversation as resolved.
Show resolved Hide resolved
MIN_SINUSOID_PERIOD: MIN_SINUSOID_PERIOD,
MAX_SINUSOID_PERIOD: MAX_SINUSOID_PERIOD,
MIN_SINUSOID_AMPLITUDE: MIN_SINUSOID_AMPLITUDE,
MAX_SINUSOID_AMPLITUDE: MAX_SINUSOID_AMPLITUDE,
MIN_TILT_FACTOR_DEGREES: MIN_TILT_FACTOR_DEGREES,
MAX_TILT_FACTOR_DEGREES: MAX_TILT_FACTOR_DEGREES,
TILT_DURATION_SECONDS: TILT_DURATION_SECONDS,
};

type TKey = keyof IControlsVisualiser;
VmMad marked this conversation as resolved.
Show resolved Hide resolved

/**
* Retrieves a value from localStorage and parses it as JSON.
*/
VmMad marked this conversation as resolved.
Show resolved Hide resolved
const LOCAL_STORAGE_KEY = "controlsVisualiser";
VmMad marked this conversation as resolved.
Show resolved Hide resolved

export const getFromLocalStorage = (): IControlsVisualiser => {
if (features.controlsVisualiserEnabled) {
const item = localStorage.getItem(LOCAL_STORAGE_KEY);
return item ? JSON.parse(item) : defaultControlsVisualiser;
} else {
localStorage.removeItem(LOCAL_STORAGE_KEY);
return defaultControlsVisualiser;
}
};
VmMad marked this conversation as resolved.
Show resolved Hide resolved

/**
* Saves a value to localStorage as a JSON string.
*/
function setToLocalStorage(value: IControlsVisualiser) {
localStorage.setItem(LOCAL_STORAGE_KEY, JSON.stringify(value));
}

/**
*/
VmMad marked this conversation as resolved.
Show resolved Hide resolved
function isExistsInLocalStorage(): boolean {
VmMad marked this conversation as resolved.
Show resolved Hide resolved
return !!localStorage.getItem(LOCAL_STORAGE_KEY);
}

export const ConfigControls = () => {
const forcedZoom = useTangleStore((state) => state.forcedZoom);
const setForcedZoom = useTangleStore((state) => state.setForcedZoom);
const [localZoom, setLocalZoom] = useState<number | undefined>(forcedZoom);

const [state, setState] = useState<IControlsVisualiser>(() => {
// Use getFromLocalStorage to retrieve the state
return getFromLocalStorage() || defaultControlsVisualiser;
});
VmMad marked this conversation as resolved.
Show resolved Hide resolved
const [isShowReset, setIsShowReset] = useState(() => {
return isExistsInLocalStorage();
});
VmMad marked this conversation as resolved.
Show resolved Hide resolved

const [errors, setErrors] = useState<{
[k: string]: string;
}>({});

const inputs: {
key: TKey;
label: string;
min: number;
max: number;
}[] = [
{
key: "MIN_SINUSOID_PERIOD",
VmMad marked this conversation as resolved.
Show resolved Hide resolved
label: "Min sinusoid period",
min: 1,
max: 7,
},
{
key: "MAX_SINUSOID_PERIOD",
label: "Max sinusoid period",
min: 8,
max: 15,
},
{
key: "MIN_SINUSOID_AMPLITUDE",
label: "Min sinusoid amplitude",
min: 100,
VmMad marked this conversation as resolved.
Show resolved Hide resolved
max: 199,
},
{
key: "MAX_SINUSOID_AMPLITUDE",
label: "Max sinusoid amplitude",
min: 200,
max: 500,
},
{
key: "MIN_TILT_FACTOR_DEGREES",
label: "Min tilt factor degrees",
min: 1,
max: 15,
VmMad marked this conversation as resolved.
Show resolved Hide resolved
},
{
key: "MAX_TILT_FACTOR_DEGREES",
label: "Max tilt factor degrees",
min: 16,
max: 100,
VmMad marked this conversation as resolved.
Show resolved Hide resolved
},
{
key: "TILT_DURATION_SECONDS",
label: "Tilt_duration_seconds",
VmMad marked this conversation as resolved.
Show resolved Hide resolved
min: 1,
max: 10,
VmMad marked this conversation as resolved.
Show resolved Hide resolved
},
];

const handleApply = () => {
if (Object.keys(errors).some((key) => errors[key])) {
// Handle the error case, e.g., display a message
console.error("There are errors in the form.");
return;
}

setToLocalStorage(state);
location.reload();
};

const handleChange = (key: TKey, val: string) => {
const input = inputs.find((input) => input.key === key);
if (!input) return;

if (!val) {
setErrors((prevErrors) => ({ ...prevErrors, [key]: "Value is required" }));
setState((prevState) => ({ ...prevState, [key]: "" }));
return;
}

const numericValue = Number(val);
if (numericValue < input.min || numericValue > input.max) {
setErrors((prevErrors) => ({ ...prevErrors, [key]: `Value must be between ${input.min} and ${input.max}` }));
} else {
setErrors((prevErrors) => ({ ...prevErrors, [key]: "" }));
}

setState((prevState) => ({ ...prevState, [key]: numericValue }));
};

if (!features.controlsVisualiserEnabled) {
return null;
}

return (
VmMad marked this conversation as resolved.
Show resolved Hide resolved
<div className={"controls-container"}>
<div className="controls__list">
{inputs.map((i) => {
return (
<div key={i.key} className="controls__item">
<div
style={{
display: "flex",
flexDirection: "column",
}}
>
<label>{i.label}</label>
<input type="number" value={state[i.key]} onChange={(e) => handleChange(i.key, e.target.value)} />
{!!errors[i.key] && <div>{errors[i.key]}</div>}
</div>
</div>
);
})}
</div>

<div className="controls__actions">
<button type={"button"} onClick={handleApply}>
Apply
</button>
{isShowReset && (
<button
type={"button"}
VmMad marked this conversation as resolved.
Show resolved Hide resolved
onClick={() => {
localStorage.removeItem(LOCAL_STORAGE_KEY);
setIsShowReset(false);
location.reload();
}}
>
Reset
</button>
)}
</div>

<div style={{ marginTop: "16px" }}>
<label style={{ display: "block" }}>Zoom</label>
<input
type="number"
VmMad marked this conversation as resolved.
Show resolved Hide resolved
value={localZoom || `${localZoom}`}
onChange={(e) => {
if (!e.target.value) {
setLocalZoom(undefined);
return;
}

const value = Number(e.target.value);

if (value > 2) {
setLocalZoom(2);
return;
}
setLocalZoom(Number(e.target.value));
}}
/>
<div className="controls__actions">
<button type={"button"} onClick={() => setForcedZoom(localZoom)}>
Apply
</button>
</div>
</div>
</div>
);
};

export default React.memo(ConfigControls);
23 changes: 23 additions & 0 deletions client/src/features/visualizer-threejs/Controls.scss
VmMad marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
.controls-container {
background: var(--body-background);
border: 1px solid var(--border-color);
padding: 8px 16px;
border-radius: 8px;
.controls__list {
display: flex;
gap: 8px;
}
.controls__item {
flex: 1;

input {
padding: 8px 16px;
width: 100%;
}
}
.controls__actions {
margin-top: 16px;
display: flex;
gap: 8px;
}
}
Loading
Loading