From de4fd4b93019787a9da0572d24af0e4794e1cf4f Mon Sep 17 00:00:00 2001 From: Philippe Morier Date: Fri, 17 Nov 2023 19:26:34 +0100 Subject: [PATCH] feat(gizmo): custom colors for gizmos (#14525) * feat(gizmo): custom colors for plan-rotation gizmo * docs(contributing): fix typos * feat(gizmo): custom colors for axis-drag gizmo * fix(gizmo): avoid white reflection on hover material * style: run `npm run format` * refactor: ensure rgb values always contain '.' * refactor: getter to avoid breaking change & default hover color * refactor: set uniform in shader instead of dynamic string --- contributing.md | 6 +-- packages/dev/core/src/Gizmos/axisDragGizmo.ts | 36 +++++++++++++++--- .../dev/core/src/Gizmos/planeDragGizmo.ts | 6 ++- .../dev/core/src/Gizmos/planeRotationGizmo.ts | 38 ++++++++++++++----- 4 files changed, 68 insertions(+), 18 deletions(-) diff --git a/contributing.md b/contributing.md index a290d1b8929..c15eb578a11 100644 --- a/contributing.md +++ b/contributing.md @@ -14,7 +14,7 @@ The first golden rule is a really important one because we want our users to tru ### Performance -Babylon.js is a 3D rendering engine. So every piece of code has to be scrutinized to look for potential bottlenecks or slow downs. Ultimately the goal is to render more with less resources. +Babylon.js is a 3D rendering engine. So every piece of code has to be scrutinized to look for potential bottlenecks or slowdowns. Ultimately the goal is to render more with less resources. ### Simplicity @@ -58,7 +58,7 @@ If you intend to only update the doc, this [documentation](https://doc.babylonjs To validate your PR, please follow these steps: -- Run "npm run build:dev" locally and make sure that no error is generated +- Run `npm run build:dev` locally and make sure that no error is generated - Make sure that all public functions and classes are commented using JSDoc/TSDoc syntax - Run `npm run test:unit` for unit tests, and check the buildSystem.md file for information regarding the visualization tests. @@ -71,4 +71,4 @@ In order to not bloat the core engine with unwanted or unnecessary features (tha - Is there a general use case for this feature? - Does this feature already exist in a similar framework? -If your PR is does not fall into the core category you can consider using our [Extensions repo](https://github.com/BabylonJS/Extensions) for more high level features. +If your PR does not fall into the core category you can consider using our [Extensions repo](https://github.com/BabylonJS/Extensions) for more high level features. diff --git a/packages/dev/core/src/Gizmos/axisDragGizmo.ts b/packages/dev/core/src/Gizmos/axisDragGizmo.ts index f010f782929..6143f961cce 100644 --- a/packages/dev/core/src/Gizmos/axisDragGizmo.ts +++ b/packages/dev/core/src/Gizmos/axisDragGizmo.ts @@ -16,6 +16,7 @@ import type { Scene } from "../scene"; import type { PositionGizmo } from "./positionGizmo"; import { Color3 } from "../Maths/math.color"; import { TmpVectors } from "../Maths/math.vector"; + /** * Interface for axis drag gizmo */ @@ -82,13 +83,32 @@ export class AxisDragGizmo extends Gizmo implements IAxisDragGizmo { public get disableMaterial() { return this._disableMaterial; } + /** * @internal */ public static _CreateArrow(scene: Scene, material: StandardMaterial, thickness: number = 1, isCollider = false): TransformNode { const arrow = new TransformNode("arrow", scene); - const cylinder = CreateCylinder("cylinder", { diameterTop: 0, height: 0.075, diameterBottom: 0.0375 * (1 + (thickness - 1) / 4), tessellation: 96 }, scene); - const line = CreateCylinder("cylinder", { diameterTop: 0.005 * thickness, height: 0.275, diameterBottom: 0.005 * thickness, tessellation: 96 }, scene); + const cylinder = CreateCylinder( + "cylinder", + { + diameterTop: 0, + height: 0.075, + diameterBottom: 0.0375 * (1 + (thickness - 1) / 4), + tessellation: 96, + }, + scene + ); + const line = CreateCylinder( + "cylinder", + { + diameterTop: 0.005 * thickness, + height: 0.275, + diameterBottom: 0.005 * thickness, + tessellation: 96, + }, + scene + ); // Position arrow pointing in its drag axis cylinder.parent = arrow; @@ -127,13 +147,17 @@ export class AxisDragGizmo extends Gizmo implements IAxisDragGizmo { * @param gizmoLayer The utility layer the gizmo will be added to * @param parent * @param thickness display gizmo axis thickness + * @param hoverColor The color of the gizmo when hovering over and dragging + * @param disableColor The Color of the gizmo when its disabled */ constructor( dragAxis: Vector3, color: Color3 = Color3.Gray(), gizmoLayer: UtilityLayerRenderer = UtilityLayerRenderer.DefaultUtilityLayer, parent: Nullable = null, - thickness: number = 1 + thickness: number = 1, + hoverColor: Color3 = Color3.Yellow(), + disableColor: Color3 = Color3.Gray() ) { super(gizmoLayer); this._parent = parent; @@ -144,10 +168,10 @@ export class AxisDragGizmo extends Gizmo implements IAxisDragGizmo { this._coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1)); this._hoverMaterial = new StandardMaterial("", gizmoLayer.utilityLayerScene); - this._hoverMaterial.diffuseColor = Color3.Yellow(); + this._hoverMaterial.diffuseColor = hoverColor; this._disableMaterial = new StandardMaterial("", gizmoLayer.utilityLayerScene); - this._disableMaterial.diffuseColor = Color3.Gray(); + this._disableMaterial.diffuseColor = disableColor; this._disableMaterial.alpha = 0.4; // Build Mesh + Collider @@ -253,6 +277,7 @@ export class AxisDragGizmo extends Gizmo implements IAxisDragGizmo { this._setGizmoMeshMaterial(cache.gizmoMeshes, newState ? cache.material : cache.disableMaterial); }); } + protected _attachedNodeChanged(value: Nullable) { if (this.dragBehavior) { this.dragBehavior.enabled = value ? true : false; @@ -274,6 +299,7 @@ export class AxisDragGizmo extends Gizmo implements IAxisDragGizmo { } } } + public get isEnabled(): boolean { return this._isEnabled; } diff --git a/packages/dev/core/src/Gizmos/planeDragGizmo.ts b/packages/dev/core/src/Gizmos/planeDragGizmo.ts index 52c28fe7721..8e4d3648ee5 100644 --- a/packages/dev/core/src/Gizmos/planeDragGizmo.ts +++ b/packages/dev/core/src/Gizmos/planeDragGizmo.ts @@ -26,7 +26,7 @@ export interface IPlaneDragGizmo extends IGizmo { snapDistance: number; /** * Event that fires each time the gizmo snaps to a new location. - * * snapDistance is the the change in distance + * * snapDistance is the change in distance */ onSnapObservable: Observable<{ snapDistance: number }>; /** If the gizmo is enabled */ @@ -82,6 +82,7 @@ export class PlaneDragGizmo extends Gizmo implements IPlaneDragGizmo { public get disableMaterial() { return this._disableMaterial; } + /** * @internal */ @@ -205,6 +206,7 @@ export class PlaneDragGizmo extends Gizmo implements IPlaneDragGizmo { this._setGizmoMeshMaterial(cache.gizmoMeshes, newState ? this._coloredMaterial : this._disableMaterial); }); } + protected _attachedNodeChanged(value: Nullable) { if (this.dragBehavior) { this.dragBehavior.enabled = value ? true : false; @@ -224,9 +226,11 @@ export class PlaneDragGizmo extends Gizmo implements IPlaneDragGizmo { } } } + public get isEnabled(): boolean { return this._isEnabled; } + /** * Disposes of the gizmo */ diff --git a/packages/dev/core/src/Gizmos/planeRotationGizmo.ts b/packages/dev/core/src/Gizmos/planeRotationGizmo.ts index d5ebfeed113..7e30a8db3cb 100644 --- a/packages/dev/core/src/Gizmos/planeRotationGizmo.ts +++ b/packages/dev/core/src/Gizmos/planeRotationGizmo.ts @@ -33,7 +33,7 @@ export interface IPlaneRotationGizmo extends IGizmo { sensitivity: number; /** * Event that fires each time the gizmo snaps to a new location. - * * snapDistance is the the change in distance + * * snapDistance is the change in distance */ onSnapObservable: Observable<{ snapDistance: number }>; /** Accumulated relative angle value for rotation on the axis. */ @@ -65,7 +65,7 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { public snapDistance = 0; /** * Event that fires each time the gizmo snaps to a new location. - * * snapDistance is the the change in distance + * * snapDistance is the change in distance */ public onSnapObservable = new Observable<{ snapDistance: number }>(); @@ -117,6 +117,7 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { uniform mat4 worldViewProjection; varying vec3 vPosition; varying vec2 vUV; + void main(void) { gl_Position = worldViewProjection * vec4(position, 1.0); vUV = uv; @@ -127,7 +128,10 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { varying vec2 vUV; varying vec3 vPosition; uniform vec3 angles; + uniform vec3 hoverColor; + #define twopi 6.283185307 + void main(void) { vec2 uv = vUV - vec2(0.5); float angle = atan(uv.y, uv.x) + 3.141592; @@ -148,8 +152,9 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { intensity += max(step(start, angle) - step(end, angle), 0.); angle += twopi; } - gl_FragColor = vec4(1.,1.,0., min(intensity * 0.25, 0.8)) * opacity; - }`; + gl_FragColor = vec4(hoverColor, min(intensity * 0.25, 0.8)) * opacity; + } + `; protected _rotationShaderMaterial: ShaderMaterial; @@ -162,6 +167,8 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { * @param parent * @param useEulerRotation Use and update Euler angle instead of quaternion * @param thickness display gizmo axis thickness + * @param hoverColor The color of the gizmo when hovering over and dragging + * @param disableColor The Color of the gizmo when its disabled */ constructor( planeNormal: Vector3, @@ -171,7 +178,9 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { parent: Nullable = null, // eslint-disable-next-line @typescript-eslint/no-unused-vars useEulerRotation = false, - thickness: number = 1 + thickness: number = 1, + hoverColor: Color3 = Color3.Yellow(), + disableColor: Color3 = Color3.Gray() ) { super(gizmoLayer); this._parent = parent; @@ -181,10 +190,11 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { this._coloredMaterial.specularColor = color.subtract(new Color3(0.1, 0.1, 0.1)); this._hoverMaterial = new StandardMaterial("", gizmoLayer.utilityLayerScene); - this._hoverMaterial.diffuseColor = Color3.Yellow(); + this._hoverMaterial.diffuseColor = hoverColor; + this._hoverMaterial.specularColor = hoverColor; this._disableMaterial = new StandardMaterial("", gizmoLayer.utilityLayerScene); - this._disableMaterial.diffuseColor = Color3.Gray(); + this._disableMaterial.diffuseColor = disableColor; this._disableMaterial.alpha = 0.4; // Build mesh on root node @@ -192,7 +202,14 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { const { rotationMesh, collider } = this._createGizmoMesh(this._gizmoMesh, thickness, tessellation); // Setup Rotation Circle - this._rotationDisplayPlane = CreatePlane("rotationDisplay", { size: 0.6, updatable: false }, this.gizmoLayer.utilityLayerScene); + this._rotationDisplayPlane = CreatePlane( + "rotationDisplay", + { + size: 0.6, + updatable: false, + }, + this.gizmoLayer.utilityLayerScene + ); this._rotationDisplayPlane.rotation.z = Math.PI * 0.5; this._rotationDisplayPlane.parent = this._gizmoMesh; this._rotationDisplayPlane.setEnabled(false); @@ -208,10 +225,11 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { }, { attributes: ["position", "uv"], - uniforms: ["worldViewProjection", "angles"], + uniforms: ["worldViewProjection", "angles", "hoverColor"], } ); this._rotationShaderMaterial.backFaceCulling = false; + this._rotationShaderMaterial.setColor3("hoverColor", hoverColor); this._rotationDisplayPlane.material = this._rotationShaderMaterial; this._rotationDisplayPlane.visibility = 0.999; @@ -458,9 +476,11 @@ export class PlaneRotationGizmo extends Gizmo implements IPlaneRotationGizmo { } } } + public get isEnabled(): boolean { return this._isEnabled; } + /** * Disposes of the gizmo */