From 85c61e042f80f5f727dca6c609d34c07c7a9ee47 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Wed, 29 Jun 2022 14:54:39 +0300 Subject: [PATCH 01/15] feat: clone atoms inside parent instead of top level --- editor/js/Menubar.Edit.js | 3 ++- editor/js/Sidebar.Scene.js | 11 ++++++----- 2 files changed, 8 insertions(+), 6 deletions(-) diff --git a/editor/js/Menubar.Edit.js b/editor/js/Menubar.Edit.js index cea91b86fc6d0..e7de551e050c5 100644 --- a/editor/js/Menubar.Edit.js +++ b/editor/js/Menubar.Edit.js @@ -120,12 +120,13 @@ function MenubarEdit( editor ) { option.onClick( function () { let object = editor.selected; + const originalParent = editor.selected.parent; if ( object === null || object.parent === null ) return; // avoid cloning the camera or scene object = object.clone(); - editor.execute( new AddObjectCommand( editor, object ) ); + editor.execute( new AddObjectCommand( editor, object, originalParent ) ); } ); options.add( option ); diff --git a/editor/js/Sidebar.Scene.js b/editor/js/Sidebar.Scene.js index f224ddbff3817..c9fd114e20466 100644 --- a/editor/js/Sidebar.Scene.js +++ b/editor/js/Sidebar.Scene.js @@ -2,6 +2,7 @@ import * as THREE from 'three'; import { UIPanel, UIBreak, UIRow, UIColor, UISelect, UIText, UINumber } from './libs/ui.js'; import { UIOutliner, UITexture } from './libs/ui.three.js'; +import { AddObjectCommand, RemoveObjectCommand } from './commands/Commands.js'; function SidebarScene( editor ) { @@ -349,7 +350,7 @@ function SidebarScene( editor ) { } - // Hide utility scene objects and labels from materials editor + // Hide utility scene objects and labels from materials editor if ( ! [ 'OrthographicCamera', @@ -391,15 +392,15 @@ function SidebarScene( editor ) { } else if ( value === 'Cut' ) { copiedObject = editor.selected; - editor.execute( new RemoveObjectCommand( editor.selected ) ); + editor.execute( new RemoveObjectCommand( editor, editor.selected ) ); } else if ( value === 'Delete' ) { - editor.execute( new RemoveObjectCommand( editor.selected ) ); + editor.execute( new RemoveObjectCommand( editor, editor.selected ) ); } else if ( value === 'Clone' ) { - editor.execute( new AddObjectCommand( editor.selected.clone() ) ); + editor.execute( new AddObjectCommand( editor, editor.selected.clone(), editor.selected.parent ) ); } else if ( value === 'Paste' && copiedObject !== undefined ) { @@ -409,7 +410,7 @@ function SidebarScene( editor ) { } - editor.execute( new AddObjectCommand( copiedObject, editor.selected ) ); + editor.execute( new AddObjectCommand( editor, copiedObject, editor.selected ) ); } From 71d611831bf11ac0a29980aa7c9b258757bb2e7f Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Tue, 5 Jul 2022 21:31:41 +0300 Subject: [PATCH 02/15] feat: add "multiple selection" option to toolbar --- editor/images/multiple-selection.svg | 49 +++++++++++++++++++++++++ editor/js/Config.js | 3 +- editor/js/Editor.js | 3 ++ editor/js/Sidebar.Settings.Shortcuts.js | 8 +++- editor/js/Strings.js | 3 ++ editor/js/Toolbar.js | 26 +++++++++++++ editor/js/Viewport.js | 10 ++++- 7 files changed, 99 insertions(+), 3 deletions(-) create mode 100644 editor/images/multiple-selection.svg diff --git a/editor/images/multiple-selection.svg b/editor/images/multiple-selection.svg new file mode 100644 index 0000000000000..74df8d3e9cd7e --- /dev/null +++ b/editor/images/multiple-selection.svg @@ -0,0 +1,49 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/editor/js/Config.js b/editor/js/Config.js index c586fe86fb74d..c883496f85953 100644 --- a/editor/js/Config.js +++ b/editor/js/Config.js @@ -24,7 +24,8 @@ function Config() { 'settings/shortcuts/rotate': 'e', 'settings/shortcuts/scale': 'r', 'settings/shortcuts/undo': 'z', - 'settings/shortcuts/focus': 'f' + 'settings/shortcuts/focus': 'f', + 'settings/shortcuts/multiple-selection': 'm' }; if ( window.localStorage[ name ] === undefined ) { diff --git a/editor/js/Editor.js b/editor/js/Editor.js index 3282c13dfd1d2..7931c861ce808 100644 --- a/editor/js/Editor.js +++ b/editor/js/Editor.js @@ -19,6 +19,9 @@ function Editor( providedDefaultCamera ) { var signal = new Signal(); this.signals = { + // multipleSelection + + toggleMultipleSelection: new Signal(), // script diff --git a/editor/js/Sidebar.Settings.Shortcuts.js b/editor/js/Sidebar.Settings.Shortcuts.js index 12dccbd78ff1b..a8847f471f101 100644 --- a/editor/js/Sidebar.Settings.Shortcuts.js +++ b/editor/js/Sidebar.Settings.Shortcuts.js @@ -23,7 +23,7 @@ function SidebarSettingsShortcuts( editor ) { headerRow.add( new UIText( strings.getKey( 'sidebar/settings/shortcuts' ).toUpperCase() ) ); container.add( headerRow ); - const shortcuts = [ 'translate', 'rotate', 'scale', 'undo', 'focus' ]; + const shortcuts = [ 'translate', 'rotate', 'scale', 'multiple-selection', 'undo', 'focus' ]; function createShortcutInput( name ) { @@ -134,6 +134,12 @@ function SidebarSettingsShortcuts( editor ) { break; + case config.getKey( 'settings/shortcuts/multiple-selection' ) : + + signals.toggleMultipleSelection.dispatch(); + + break; + case config.getKey( 'settings/shortcuts/undo' ): if ( IS_MAC ? event.metaKey : event.ctrlKey ) { diff --git a/editor/js/Strings.js b/editor/js/Strings.js index af67edd890933..18fd294af05f0 100644 --- a/editor/js/Strings.js +++ b/editor/js/Strings.js @@ -331,6 +331,7 @@ function Strings( config ) { 'toolbar/rotate': 'Rotate', 'toolbar/scale': 'Scale', 'toolbar/local': 'Local', + 'toolbar/multiple-selection': 'Toggle multiple selection', 'viewport/info/objects': 'Objects', 'viewport/info/vertices': 'Vertices', @@ -666,6 +667,7 @@ function Strings( config ) { 'toolbar/rotate': 'Rotation', 'toolbar/scale': 'Échelle', 'toolbar/local': 'Local', + 'toolbar/multiple-selection': 'Basculer la sélection multiple', 'viewport/info/objects': 'Objets', 'viewport/info/vertices': 'Sommets', @@ -1001,6 +1003,7 @@ function Strings( config ) { 'toolbar/rotate': '旋转', 'toolbar/scale': '缩放', 'toolbar/local': '本地', + 'toolbar/multiple-selection': '切换多项选择', 'viewport/info/objects': '物体', 'viewport/info/vertices': '顶点', diff --git a/editor/js/Toolbar.js b/editor/js/Toolbar.js index 164034510363f..f328ed22a5994 100644 --- a/editor/js/Toolbar.js +++ b/editor/js/Toolbar.js @@ -6,6 +6,8 @@ const translateImg = ' const rotateImg = ''; /* '../images/scale.svg' */ const scaleImg = ''; +/* '../images/multiple-selection.svg' */ +const multipleSelectionImg = ''; function Toolbar( editor ) { @@ -57,6 +59,19 @@ function Toolbar( editor ) { } ); container.add( scale ); + const multipleSelectionIcon = document.createElement( 'img' ); + multipleSelectionIcon.title = strings.getKey( 'toolbar/multiple-selection' ); + multipleSelectionIcon.src = multipleSelectionImg; + + const multipleSelection = new UIButton(); + multipleSelection.dom.appendChild( multipleSelectionIcon ); + multipleSelection.onClick( function () { + + signals.toggleMultipleSelection.dispatch( 'multiple-selection' ); + + } ); + container.add( multipleSelection ); + const local = new UICheckbox( false ); local.dom.title = strings.getKey( 'toolbar/local' ); local.onChange( function () { @@ -73,6 +88,7 @@ function Toolbar( editor ) { translate.dom.classList.remove( 'selected' ); rotate.dom.classList.remove( 'selected' ); scale.dom.classList.remove( 'selected' ); + multipleSelection.dom.classList.remove( 'selected' ); switch ( mode ) { @@ -84,6 +100,16 @@ function Toolbar( editor ) { } ); + signals.toggleMultipleSelection.add( function () { + + translate.dom.classList.remove( 'selected' ); + rotate.dom.classList.remove( 'selected' ); + scale.dom.classList.remove( 'selected' ); + + multipleSelection.dom.classList.toggle( 'selected' ); + + } ); + return container; } diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index ccbaa00141205..91f5eaea69070 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -55,7 +55,7 @@ function Viewport( editor ) { grid.rotateX( Math.PI / 2 ); - // additioanl axis lines for usability considerations + // additioanl axis lines for usability considerations const length = 30; const lineMaterial = new THREE.LineDashedMaterial( { dashSize: 1, @@ -333,6 +333,8 @@ function Viewport( editor ) { const controls = new EditorControls( camera, container.dom ); controls.addEventListener( 'change', function () { + console.log( { camera } ); + signals.cameraChanged.dispatch( camera ); signals.refreshSidebarObject3D.dispatch( camera ); @@ -348,6 +350,12 @@ function Viewport( editor ) { } ); + signals.toggleMultipleSelection.add( function () { + + console.log( 'toggle multiple selection' ); + + } ); + signals.transformModeChanged.add( function ( mode ) { transformControls.setMode( mode ); From 2eead04f6a72cad39122fa6141509a96ae46e2fb Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Thu, 7 Jul 2022 21:57:17 +0300 Subject: [PATCH 03/15] feat: basic implementation of capturing selected data --- editor/js/Viewport.js | 9 +- .../multiple-selection/multiple-selection.js | 98 +++++++++++++++++++ .../multiple-selection/selection-drawer.js | 81 +++++++++++++++ 3 files changed, 185 insertions(+), 3 deletions(-) create mode 100644 editor/js/libs/multiple-selection/multiple-selection.js create mode 100644 editor/js/libs/multiple-selection/selection-drawer.js diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index 91f5eaea69070..69eb1414621c3 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -16,6 +16,7 @@ import { SetRotationCommand } from './commands/SetRotationCommand.js'; import { SetScaleCommand } from './commands/SetScaleCommand.js'; import { RoomEnvironment } from '../../examples/jsm/environments/RoomEnvironment.js'; +import { MultipleSelection } from './libs/multiple-selection/multiple-selection.js'; function Viewport( editor ) { @@ -90,6 +91,8 @@ function Viewport( editor ) { selectionBox.visible = false; sceneHelpers.add( selectionBox ); + const multipleSelection = new MultipleSelection( editor.viewportCamera, scene ); + let objectPositionOnDown = null; let objectRotationOnDown = null; let objectScaleOnDown = null; @@ -333,8 +336,6 @@ function Viewport( editor ) { const controls = new EditorControls( camera, container.dom ); controls.addEventListener( 'change', function () { - console.log( { camera } ); - signals.cameraChanged.dispatch( camera ); signals.refreshSidebarObject3D.dispatch( camera ); @@ -352,7 +353,8 @@ function Viewport( editor ) { signals.toggleMultipleSelection.add( function () { - console.log( 'toggle multiple selection' ); + multipleSelection.toggle(); + controls.enabled = ! multipleSelection.enabled; } ); @@ -406,6 +408,7 @@ function Viewport( editor ) { renderer.setAnimationLoop( animate ); renderer.setClearColor( 0xaaaaaa ); + multipleSelection.renderer = renderer; if ( window.matchMedia ) { diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js new file mode 100644 index 0000000000000..31f17c73d8518 --- /dev/null +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -0,0 +1,98 @@ +import { SelectionBox } from '../../../../examples/jsm/interactive/SelectionBox.js'; +import { SelectionDrawer } from './selection-drawer.js'; + +class MultipleSelection { + + constructor( camera, scene, renderer ) { + + this.enabled = false; + this._camera = camera; + this._scene = scene; + this._renderer = renderer; + this._selectionDrawer = null; + this._selectionBox = null; + + console.log( this.enabled, this._camera, this._scene, this._renderer ); + + } + + set renderer( renderer ) { + + console.log( { renderer } ); + + this._renderer = renderer; + + } + + toggle() { + + this.enabled = ! this.enabled; + + if ( this.enabled ) { + + this._selectionBox = new SelectionBox( this._camera, this._scene ); + this._selectionDrawer = new SelectionDrawer( this._renderer, 'selectBox' ); + + document.addEventListener( 'pointerdown', this._pointerDown ); + document.addEventListener( 'pointerup', this._pointerUp ); + + } else { + + this._selectionBox = null; + this._selectionDrawer = null; + + document.removeEventListener( 'pointerdown', this._pointerDown ); + document.removeEventListener( 'pointerup', this._pointerUp ); + + } + + } + + _pointerDown = ( event ) => { + + this._selectionDrawer.onPointerDown( event ); + + this._selectionBox.startPoint.set( + ( event.clientX / window.innerWidth ) * 2 - 1, + - ( event.clientY / window.innerHeight ) * 2 + 1, + 0.5 ); + + console.log( 'pointerdown', event ); + + document.addEventListener( 'pointermove', this._pointerMove ); + + }; + + _pointerMove = ( event ) => { + + this._selectionDrawer.onPointerMove( event ); + + this._selectionBox.endPoint.set( + ( event.clientX / window.innerWidth ) * 2 - 1, + - ( event.clientY / window.innerHeight ) * 2 + 1, + 0.5 ); + + console.log( 'pointermove', event ); + + }; + + _pointerUp = ( event ) => { + + this._selectionDrawer.onPointerUp(); + + this._selectionBox.endPoint.set( + ( event.clientX / window.innerWidth ) * 2 - 1, + - ( event.clientY / window.innerHeight ) * 2 + 1, + 0.5 ); + + const intersectedMeshes = this._selectionBox.select(); + + alert( 'Intersected meshes: ' + ( intersectedMeshes.map( m => m.name ).join( ',' ) || 'Not Found' ) ); + + document.removeEventListener( 'pointermove', this._pointerMove ); + + }; + +} + +export { MultipleSelection }; diff --git a/editor/js/libs/multiple-selection/selection-drawer.js b/editor/js/libs/multiple-selection/selection-drawer.js new file mode 100644 index 0000000000000..7736d9195af90 --- /dev/null +++ b/editor/js/libs/multiple-selection/selection-drawer.js @@ -0,0 +1,81 @@ +import { Vector2 } from 'three'; + +class SelectionDrawer { + + constructor( renderer, cssClassName ) { + + this.element = document.createElement( 'div' ); + this.element.classList.add( cssClassName ); + this.element.style.pointerEvents = 'none'; + + this.renderer = renderer; + + this.startPoint = new Vector2(); + this.pointTopLeft = new Vector2(); + this.pointBottomRight = new Vector2(); + + this.isDown = false; + + this.onPointerDown = function ( event ) { + + this.isDown = true; + this.onSelectStart( event ); + + }.bind( this ); + + this.onPointerMove = function ( event ) { + + if ( this.isDown ) { + + this.onSelectMove( event ); + + } + + }.bind( this ); + + this.onPointerUp = function ( ) { + + this.isDown = false; + this.onSelectOver(); + + }.bind( this ); + + } + + onSelectStart( event ) { + + this.renderer.domElement.parentElement.appendChild( this.element ); + + this.element.style.left = event.clientX + 'px'; + this.element.style.top = event.clientY + 'px'; + this.element.style.width = '0px'; + this.element.style.height = '0px'; + + this.startPoint.x = event.clientX; + this.startPoint.y = event.clientY; + + } + + onSelectMove( event ) { + + this.pointBottomRight.x = Math.max( this.startPoint.x, event.clientX ); + this.pointBottomRight.y = Math.max( this.startPoint.y, event.clientY ); + this.pointTopLeft.x = Math.min( this.startPoint.x, event.clientX ); + this.pointTopLeft.y = Math.min( this.startPoint.y, event.clientY ); + + this.element.style.left = this.pointTopLeft.x + 'px'; + this.element.style.top = this.pointTopLeft.y + 'px'; + this.element.style.width = ( this.pointBottomRight.x - this.pointTopLeft.x ) + 'px'; + this.element.style.height = ( this.pointBottomRight.y - this.pointTopLeft.y ) + 'px'; + + } + + onSelectOver() { + + this.element.parentElement.removeChild( this.element ); + + } + +} + +export { SelectionDrawer }; From 3e8e431abbef333366f6740261ecffd72154a56b Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Tue, 12 Jul 2022 19:33:35 +0300 Subject: [PATCH 04/15] feat: create a group from multiple selection captured items --- editor/js/Viewport.js | 14 ++++- .../multiple-selection/multiple-selection.js | 57 ++++++++++++++++--- 2 files changed, 62 insertions(+), 9 deletions(-) diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index 69eb1414621c3..2b1935cd45bc6 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -17,6 +17,7 @@ import { SetScaleCommand } from './commands/SetScaleCommand.js'; import { RoomEnvironment } from '../../examples/jsm/environments/RoomEnvironment.js'; import { MultipleSelection } from './libs/multiple-selection/multiple-selection.js'; +import { AddObjectCommand } from './commands/AddObjectCommand.js'; function Viewport( editor ) { @@ -92,6 +93,17 @@ function Viewport( editor ) { sceneHelpers.add( selectionBox ); const multipleSelection = new MultipleSelection( editor.viewportCamera, scene ); + multipleSelection.addEventListener( 'pointerup', ( selectedMeshes ) => { + + if ( ! selectedMeshes.length ) return; + + const group = new THREE.Group(); + group.name = 'Multiple Selection Group'; + selectedMeshes.forEach( mesh => group.add( mesh ) ); + + editor.execute( new AddObjectCommand( editor, group ) ); + + } ); let objectPositionOnDown = null; let objectRotationOnDown = null; @@ -114,7 +126,7 @@ function Viewport( editor ) { } - signals.refreshSidebarObject3D.dispatch( object ); + signals.sceneGraphChanged.dispatch( ); } diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js index 31f17c73d8518..97efe4c8954d3 100644 --- a/editor/js/libs/multiple-selection/multiple-selection.js +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -1,3 +1,4 @@ +import { Object3D } from 'three'; import { SelectionBox } from '../../../../examples/jsm/interactive/SelectionBox.js'; import { SelectionDrawer } from './selection-drawer.js'; @@ -11,15 +12,15 @@ class MultipleSelection { this._renderer = renderer; this._selectionDrawer = null; this._selectionBox = null; - - console.log( this.enabled, this._camera, this._scene, this._renderer ); + this._selectedMeshes = []; + this._pointerUpCustomCallback = null; + this._pointerDownCustomCallback = null; + this._pointerMoveCustomCallback = null; } set renderer( renderer ) { - console.log( { renderer } ); - this._renderer = renderer; } @@ -48,6 +49,32 @@ class MultipleSelection { } + addEventListener( event, callback ) { + + switch ( event ) { + + case 'pointerup': + this._pointerUpCustomCallback = callback; + break; + case 'pointerdown': + this._pointerDownCustomCallback = callback; + break; + case 'pointermove': + this._pointerMoveCustomCallback = callback; + break; + default: + break; + + } + + } + + _keepOnlyMeshes( sceneObjects ) { + + return sceneObjects.filter( object => object instanceof Object3D && object.isMesh ); + + } + _pointerDown = ( event ) => { this._selectionDrawer.onPointerDown( event ); @@ -57,7 +84,11 @@ class MultipleSelection { - ( event.clientY / window.innerHeight ) * 2 + 1, 0.5 ); - console.log( 'pointerdown', event ); + if ( this._pointerDownCustomCallback ) { + + this._pointerDownCustomCallback( event ); + + } document.addEventListener( 'pointermove', this._pointerMove ); @@ -72,7 +103,13 @@ class MultipleSelection { - ( event.clientY / window.innerHeight ) * 2 + 1, 0.5 ); - console.log( 'pointermove', event ); + this._selectedMeshes = this._keepOnlyMeshes( this._selectionBox.select() ); + + if ( this._pointerMoveCustomCallback ) { + + this._pointerMoveCustomCallback( this._selectedMeshes ); + + } }; @@ -85,9 +122,13 @@ class MultipleSelection { - ( event.clientY / window.innerHeight ) * 2 + 1, 0.5 ); - const intersectedMeshes = this._selectionBox.select(); + this._selectedMeshes = this._keepOnlyMeshes( this._selectionBox.select() ); + + if ( this._pointerUpCustomCallback ) { - alert( 'Intersected meshes: ' + ( intersectedMeshes.map( m => m.name ).join( ',' ) || 'Not Found' ) ); + this._pointerUpCustomCallback( this._selectedMeshes ); + + } document.removeEventListener( 'pointermove', this._pointerMove ); From b4bef633c8b5c294feb79e2ccba4b4f175d0da26 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Tue, 12 Jul 2022 22:11:08 +0300 Subject: [PATCH 05/15] feat: keep only one instance of active multiple selection group --- editor/js/Viewport.js | 34 ++++++++++- .../multiple-selection/multiple-selection.js | 57 ++++++++++++++++--- 2 files changed, 81 insertions(+), 10 deletions(-) diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index 2b1935cd45bc6..4a10da8a443d7 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -18,6 +18,7 @@ import { SetScaleCommand } from './commands/SetScaleCommand.js'; import { RoomEnvironment } from '../../examples/jsm/environments/RoomEnvironment.js'; import { MultipleSelection } from './libs/multiple-selection/multiple-selection.js'; import { AddObjectCommand } from './commands/AddObjectCommand.js'; +import { RemoveObjectCommand } from './commands/Commands.js'; function Viewport( editor ) { @@ -93,9 +94,34 @@ function Viewport( editor ) { sceneHelpers.add( selectionBox ); const multipleSelection = new MultipleSelection( editor.viewportCamera, scene ); + + multipleSelection.addEventListener( 'pointerdown', () => { + + const group = scene.getObjectByName( 'Multiple Selection Group' ); + + if ( group ) { + + editor.execute( new RemoveObjectCommand( editor, group ) ); + + } + + } ); + multipleSelection.addEventListener( 'pointerup', ( selectedMeshes ) => { - if ( ! selectedMeshes.length ) return; + if ( ! selectedMeshes.length ) { + + const previousMultipleSelectionGroup = scene.getObjectByName( 'Multiple Selection Group' ); + + if ( previousMultipleSelectionGroup ) { + + editor.execute( new RemoveObjectCommand( editor, previousMultipleSelectionGroup ) ); + + } + + return; + + } const group = new THREE.Group(); group.name = 'Multiple Selection Group'; @@ -105,6 +131,12 @@ function Viewport( editor ) { } ); + multipleSelection.addEventListener( 'pointermove', ( ) => { + + console.log( 'pointermove' ); + + } ); + let objectPositionOnDown = null; let objectRotationOnDown = null; let objectScaleOnDown = null; diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js index 97efe4c8954d3..4ebfaed8ce89a 100644 --- a/editor/js/libs/multiple-selection/multiple-selection.js +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -13,6 +13,7 @@ class MultipleSelection { this._selectionDrawer = null; this._selectionBox = null; this._selectedMeshes = []; + this._hiddenOriginalMeshes = []; this._pointerUpCustomCallback = null; this._pointerDownCustomCallback = null; this._pointerMoveCustomCallback = null; @@ -34,16 +35,16 @@ class MultipleSelection { this._selectionBox = new SelectionBox( this._camera, this._scene ); this._selectionDrawer = new SelectionDrawer( this._renderer, 'selectBox' ); - document.addEventListener( 'pointerdown', this._pointerDown ); - document.addEventListener( 'pointerup', this._pointerUp ); + this._renderer.domElement.parentElement.addEventListener( 'pointerdown', this._pointerDown ); + this._renderer.domElement.parentElement.addEventListener( 'pointerup', this._pointerUp ); } else { this._selectionBox = null; this._selectionDrawer = null; - document.removeEventListener( 'pointerdown', this._pointerDown ); - document.removeEventListener( 'pointerup', this._pointerUp ); + this._renderer.domElement.parentElement.removeEventListener( 'pointerdown', this._pointerDown ); + this._renderer.domElement.parentElement.removeEventListener( 'pointerup', this._pointerUp ); } @@ -69,6 +70,42 @@ class MultipleSelection { } + _displayOriginalMeshes() { + + this._hiddenOriginalMeshes.forEach( mesh => { + + mesh.visible = true; + + } ); + + this._hiddenOriginalMeshes = []; + + } + + _hideOriginalMeshes( clonedMeshes ) { + + clonedMeshes.forEach( mesh => { + + mesh._original_mesh.visible = false; + this._hiddenOriginalMeshes.push( mesh._original_mesh ); + + } ); + + } + + _cloneMeshes( meshes ) { + + return meshes.map( mesh => { + + const clone = mesh.clone(); + clone._original_mesh = mesh; + + return clone; + + } ); + + } + _keepOnlyMeshes( sceneObjects ) { return sceneObjects.filter( object => object instanceof Object3D && object.isMesh ); @@ -77,6 +114,8 @@ class MultipleSelection { _pointerDown = ( event ) => { + this._displayOriginalMeshes(); + this._selectionDrawer.onPointerDown( event ); this._selectionBox.startPoint.set( @@ -90,7 +129,7 @@ class MultipleSelection { } - document.addEventListener( 'pointermove', this._pointerMove ); + this._renderer.domElement.parentElement.addEventListener( 'pointermove', this._pointerMove ); }; @@ -103,8 +142,6 @@ class MultipleSelection { - ( event.clientY / window.innerHeight ) * 2 + 1, 0.5 ); - this._selectedMeshes = this._keepOnlyMeshes( this._selectionBox.select() ); - if ( this._pointerMoveCustomCallback ) { this._pointerMoveCustomCallback( this._selectedMeshes ); @@ -122,7 +159,9 @@ class MultipleSelection { - ( event.clientY / window.innerHeight ) * 2 + 1, 0.5 ); - this._selectedMeshes = this._keepOnlyMeshes( this._selectionBox.select() ); + this._selectedMeshes = this._cloneMeshes( this._keepOnlyMeshes( this._selectionBox.select() ) ); + + this._hideOriginalMeshes( this._selectedMeshes ); if ( this._pointerUpCustomCallback ) { @@ -130,7 +169,7 @@ class MultipleSelection { } - document.removeEventListener( 'pointermove', this._pointerMove ); + this._renderer.domElement.parentElement.removeEventListener( 'pointermove', this._pointerMove ); }; From db4d7306011fabfa04d86e88400cc08f5707e504 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Thu, 14 Jul 2022 20:45:38 +0300 Subject: [PATCH 06/15] fix: selection accuracy issue --- .../multiple-selection/multiple-selection.js | 32 ++++++++++++------- 1 file changed, 20 insertions(+), 12 deletions(-) diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js index 4ebfaed8ce89a..f6bffbe19b9ee 100644 --- a/editor/js/libs/multiple-selection/multiple-selection.js +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -112,16 +112,26 @@ class MultipleSelection { } + _getCursorPosition( event ) { + + const rect = event.target.getBoundingClientRect(); + + const x = ( ( event.clientX - rect.x ) / rect.width ) * 2 - 1; + const y = - ( ( event.clientY - rect.y ) / rect.height ) * 2 + 1; + + return { x, y }; + + } + _pointerDown = ( event ) => { this._displayOriginalMeshes(); this._selectionDrawer.onPointerDown( event ); - this._selectionBox.startPoint.set( - ( event.clientX / window.innerWidth ) * 2 - 1, - - ( event.clientY / window.innerHeight ) * 2 + 1, - 0.5 ); + const cursorPosition = this._getCursorPosition( event ); + + this._selectionBox.startPoint.set( cursorPosition.x, cursorPosition.y, 0.5 ); if ( this._pointerDownCustomCallback ) { @@ -137,10 +147,9 @@ class MultipleSelection { this._selectionDrawer.onPointerMove( event ); - this._selectionBox.endPoint.set( - ( event.clientX / window.innerWidth ) * 2 - 1, - - ( event.clientY / window.innerHeight ) * 2 + 1, - 0.5 ); + const cursorPosition = this._getCursorPosition( event ); + + this._selectionBox.endPoint.set( cursorPosition.x, cursorPosition.y, 0.5 ); if ( this._pointerMoveCustomCallback ) { @@ -154,10 +163,9 @@ class MultipleSelection { this._selectionDrawer.onPointerUp(); - this._selectionBox.endPoint.set( - ( event.clientX / window.innerWidth ) * 2 - 1, - - ( event.clientY / window.innerHeight ) * 2 + 1, - 0.5 ); + const cursorPosition = this._getCursorPosition( event ); + + this._selectionBox.endPoint.set( cursorPosition.x, cursorPosition.y, 0.5 ); this._selectedMeshes = this._cloneMeshes( this._keepOnlyMeshes( this._selectionBox.select() ) ); From b1af054f3e3421e3c1cb53d15a8d9fbd5ac9e012 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Thu, 14 Jul 2022 20:55:05 +0300 Subject: [PATCH 07/15] feat: add "submit" and "cancel" buttons for selection process --- editor/js/Editor.js | 2 + editor/js/Toolbar.js | 4 +- editor/js/Viewport.js | 48 +++++----- .../js/commands/AddMultipleSelectionGroup.js | 28 ++++++ editor/js/commands/Commands.js | 2 + .../commands/RemoveMultipleSelectionGroup.js | 28 ++++++ .../multiple-selection/multiple-selection.js | 89 ++++++++++++++++--- 7 files changed, 167 insertions(+), 34 deletions(-) create mode 100644 editor/js/commands/AddMultipleSelectionGroup.js create mode 100644 editor/js/commands/RemoveMultipleSelectionGroup.js diff --git a/editor/js/Editor.js b/editor/js/Editor.js index 7931c861ce808..c296c0e187cfa 100644 --- a/editor/js/Editor.js +++ b/editor/js/Editor.js @@ -22,6 +22,8 @@ function Editor( providedDefaultCamera ) { // multipleSelection toggleMultipleSelection: new Signal(), + enableMultipleSelection: new Signal(), + disableMultipleSelection: new Signal(), // script diff --git a/editor/js/Toolbar.js b/editor/js/Toolbar.js index f328ed22a5994..d8ca7991ffd4b 100644 --- a/editor/js/Toolbar.js +++ b/editor/js/Toolbar.js @@ -67,7 +67,7 @@ function Toolbar( editor ) { multipleSelection.dom.appendChild( multipleSelectionIcon ); multipleSelection.onClick( function () { - signals.toggleMultipleSelection.dispatch( 'multiple-selection' ); + signals.toggleMultipleSelection.dispatch(); } ); container.add( multipleSelection ); @@ -90,6 +90,8 @@ function Toolbar( editor ) { scale.dom.classList.remove( 'selected' ); multipleSelection.dom.classList.remove( 'selected' ); + signals.disableMultipleSelection.dispatch(); + switch ( mode ) { case 'translate': translate.dom.classList.add( 'selected' ); break; diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index 4a10da8a443d7..b2d170bf577e1 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -17,8 +17,7 @@ import { SetScaleCommand } from './commands/SetScaleCommand.js'; import { RoomEnvironment } from '../../examples/jsm/environments/RoomEnvironment.js'; import { MultipleSelection } from './libs/multiple-selection/multiple-selection.js'; -import { AddObjectCommand } from './commands/AddObjectCommand.js'; -import { RemoveObjectCommand } from './commands/Commands.js'; +import { AddMultipleSelectionGroup, RemoveMultipleSelectionGroup } from './commands/Commands.js'; function Viewport( editor ) { @@ -97,13 +96,9 @@ function Viewport( editor ) { multipleSelection.addEventListener( 'pointerdown', () => { - const group = scene.getObjectByName( 'Multiple Selection Group' ); + editor.execute( new RemoveMultipleSelectionGroup( editor ) ); - if ( group ) { - - editor.execute( new RemoveObjectCommand( editor, group ) ); - - } + multipleSelection.hideControlButtons(); } ); @@ -111,29 +106,23 @@ function Viewport( editor ) { if ( ! selectedMeshes.length ) { - const previousMultipleSelectionGroup = scene.getObjectByName( 'Multiple Selection Group' ); - - if ( previousMultipleSelectionGroup ) { - - editor.execute( new RemoveObjectCommand( editor, previousMultipleSelectionGroup ) ); - - } + editor.execute( new RemoveMultipleSelectionGroup( editor ) ); return; } - const group = new THREE.Group(); - group.name = 'Multiple Selection Group'; - selectedMeshes.forEach( mesh => group.add( mesh ) ); + editor.execute( new AddMultipleSelectionGroup( editor, selectedMeshes ) ); - editor.execute( new AddObjectCommand( editor, group ) ); + multipleSelection.showControlButtons(); } ); - multipleSelection.addEventListener( 'pointermove', ( ) => { + multipleSelection.addEventListener( 'cancel-selection', () => { + + editor.execute( new RemoveMultipleSelectionGroup( editor ) ); - console.log( 'pointermove' ); + multipleSelection.hideControlButtons(); } ); @@ -397,11 +386,26 @@ function Viewport( editor ) { signals.toggleMultipleSelection.add( function () { + multipleSelection.toggle(); controls.enabled = ! multipleSelection.enabled; } ); + signals.enableMultipleSelection.add( function () { + + multipleSelection.enable(); + controls.enabled = false; + + } ); + + signals.disableMultipleSelection.add( function () { + + multipleSelection.disable(); + controls.enabled = true; + + } ); + signals.transformModeChanged.add( function ( mode ) { transformControls.setMode( mode ); @@ -562,6 +566,8 @@ function Viewport( editor ) { signals.objectRemoved.add( function ( object ) { + if ( multipleSelection.enabled ) return; + controls.enabled = true; // see #14180 if ( object === transformControls.object ) { diff --git a/editor/js/commands/AddMultipleSelectionGroup.js b/editor/js/commands/AddMultipleSelectionGroup.js new file mode 100644 index 0000000000000..c951a3d23916b --- /dev/null +++ b/editor/js/commands/AddMultipleSelectionGroup.js @@ -0,0 +1,28 @@ +import { Group } from 'three'; +import { Command } from '../Command.js'; +import { AddObjectCommand } from './AddObjectCommand.js'; + +class AddMultipleSelectionGroup extends Command { + + constructor( editor, meshes ) { + + super( editor ); + + this.type = 'AddMultipleSelectionGroup'; + this.meshes = meshes; + + } + + execute() { + + const group = new Group(); + group.name = 'Multiple Selection Group'; + this.meshes.forEach( mesh => group.add( mesh ) ); + + this.editor.execute( new AddObjectCommand( this.editor, group ) ); + + } + +} + +export { AddMultipleSelectionGroup }; diff --git a/editor/js/commands/Commands.js b/editor/js/commands/Commands.js index 40dd6ec045f27..5b6158c8cf3f9 100644 --- a/editor/js/commands/Commands.js +++ b/editor/js/commands/Commands.js @@ -19,3 +19,5 @@ export { SetSceneCommand } from './SetSceneCommand.js'; export { SetScriptValueCommand } from './SetScriptValueCommand.js'; export { SetUuidCommand } from './SetUuidCommand.js'; export { SetValueCommand } from './SetValueCommand.js'; +export { RemoveMultipleSelectionGroup } from './RemoveMultipleSelectionGroup.js'; +export { AddMultipleSelectionGroup } from './AddMultipleSelectionGroup.js'; diff --git a/editor/js/commands/RemoveMultipleSelectionGroup.js b/editor/js/commands/RemoveMultipleSelectionGroup.js new file mode 100644 index 0000000000000..0131fb40e8e1a --- /dev/null +++ b/editor/js/commands/RemoveMultipleSelectionGroup.js @@ -0,0 +1,28 @@ +import { Command } from '../Command.js'; +import { RemoveObjectCommand } from './RemoveObjectCommand.js'; + +class RemoveMultipleSelectionGroup extends Command { + + constructor( editor ) { + + super( editor ); + + this.type = 'RemoveMultipleSelectionGroup'; + + } + + execute() { + + const group = this.editor.scene.getObjectByName( 'Multiple Selection Group' ); + + if ( group ) { + + this.editor.execute( new RemoveObjectCommand( this.editor, group ) ); + + } + + } + +} + +export { RemoveMultipleSelectionGroup }; diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js index f6bffbe19b9ee..9249f242ec566 100644 --- a/editor/js/libs/multiple-selection/multiple-selection.js +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -17,6 +17,9 @@ class MultipleSelection { this._pointerUpCustomCallback = null; this._pointerDownCustomCallback = null; this._pointerMoveCustomCallback = null; + this._cancelBtnClickCustomCallback = null; + this._submitBtn = null; + this._cancelBtn = null; } @@ -28,28 +31,42 @@ class MultipleSelection { toggle() { - this.enabled = ! this.enabled; - if ( this.enabled ) { - this._selectionBox = new SelectionBox( this._camera, this._scene ); - this._selectionDrawer = new SelectionDrawer( this._renderer, 'selectBox' ); - - this._renderer.domElement.parentElement.addEventListener( 'pointerdown', this._pointerDown ); - this._renderer.domElement.parentElement.addEventListener( 'pointerup', this._pointerUp ); + this.disable(); } else { - this._selectionBox = null; - this._selectionDrawer = null; - - this._renderer.domElement.parentElement.removeEventListener( 'pointerdown', this._pointerDown ); - this._renderer.domElement.parentElement.removeEventListener( 'pointerup', this._pointerUp ); + this.enable(); } } + enable() { + + this.enabled = true; + + this._selectionBox = new SelectionBox( this._camera, this._scene ); + this._selectionDrawer = new SelectionDrawer( this._renderer, 'selectBox' ); + + this._renderer.domElement.parentElement.addEventListener( 'pointerdown', this._pointerDown ); + this._renderer.domElement.parentElement.addEventListener( 'pointerup', this._pointerUp ); + + } + + disable() { + + this.enabled = false; + + this._selectionBox = null; + this._selectionDrawer = null; + + this._renderer.domElement.parentElement.removeEventListener( 'pointerdown', this._pointerDown ); + this._renderer.domElement.parentElement.removeEventListener( 'pointerup', this._pointerUp ); + + } + addEventListener( event, callback ) { switch ( event ) { @@ -63,6 +80,9 @@ class MultipleSelection { case 'pointermove': this._pointerMoveCustomCallback = callback; break; + case 'cancel-selection': + this._cancelBtnClickCustomCallback = callback; + break; default: break; @@ -70,6 +90,51 @@ class MultipleSelection { } + showControlButtons() { + + this._submitBtn = document.createElement( 'button' ); + this._submitBtn.innerHTML = 'Submit'; + this._renderer.domElement.parentElement.appendChild( this._submitBtn ); + + this._submitBtn.style = 'position: absolute; top: 50px; right: 10px;'; + + this._cancelBtn = document.createElement( 'button' ); + this._cancelBtn.innerHTML = 'Cancel'; + this._renderer.domElement.parentElement.appendChild( this._cancelBtn ); + + this._cancelBtn.style = 'position: absolute; top: 50px; right: 75px;'; + + this._cancelBtn.addEventListener( 'click', this._cancelBtnClick ); + + } + + hideControlButtons() { + + if ( this._submitBtn ) { + + this._submitBtn.remove(); + + } + + if ( this._cancelBtn ) { + + this._cancelBtn.removeEventListener( 'click', this._cancelBtnClick ); + this._cancelBtn.remove(); + + } + + } + + _cancelBtnClick() { + + if ( this._cancelBtnClickCustomCallback ) { + + this._cancelBtnClickCustomCallback(); + + } + + } + _displayOriginalMeshes() { this._hiddenOriginalMeshes.forEach( mesh => { From c74438a9b3e45570a675befa6e9c6dc7364356bc Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Mon, 18 Jul 2022 20:34:55 +0300 Subject: [PATCH 08/15] feat: implement "cancel" and "submit" handlers functionality --- editor/images/multiple-selection.svg | 49 ----------- editor/js/Editor.js | 4 + editor/js/MultipleSelectionControls.js | 85 +++++++++++++++++++ editor/js/Strings.js | 12 ++- editor/js/Toolbar.js | 25 +++--- editor/js/Viewport.js | 41 +++++---- ...js => AddMultipleSelectionGroupCommand.js} | 6 +- .../CancelMultipleSelectionCommand.js | 27 ++++++ editor/js/commands/Commands.js | 6 +- ...=> RemoveMultipleSelectionGroupCommand.js} | 6 +- .../SubmitMultipleSelectionCommand.js | 34 ++++++++ .../multiple-selection/multiple-selection.js | 67 ++------------- 12 files changed, 213 insertions(+), 149 deletions(-) delete mode 100644 editor/images/multiple-selection.svg create mode 100644 editor/js/MultipleSelectionControls.js rename editor/js/commands/{AddMultipleSelectionGroup.js => AddMultipleSelectionGroupCommand.js} (73%) create mode 100644 editor/js/commands/CancelMultipleSelectionCommand.js rename editor/js/commands/{RemoveMultipleSelectionGroup.js => RemoveMultipleSelectionGroupCommand.js} (68%) create mode 100644 editor/js/commands/SubmitMultipleSelectionCommand.js diff --git a/editor/images/multiple-selection.svg b/editor/images/multiple-selection.svg deleted file mode 100644 index 74df8d3e9cd7e..0000000000000 --- a/editor/images/multiple-selection.svg +++ /dev/null @@ -1,49 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/editor/js/Editor.js b/editor/js/Editor.js index c296c0e187cfa..7fea3ae5b3d9c 100644 --- a/editor/js/Editor.js +++ b/editor/js/Editor.js @@ -24,6 +24,10 @@ function Editor( providedDefaultCamera ) { toggleMultipleSelection: new Signal(), enableMultipleSelection: new Signal(), disableMultipleSelection: new Signal(), + showMultipleSelectionControls: new Signal(), + hideMultipleSelectionControls: new Signal(), + cancelMultipleSelection: new Signal(), + submitMultipleSelection: new Signal(), // script diff --git a/editor/js/MultipleSelectionControls.js b/editor/js/MultipleSelectionControls.js new file mode 100644 index 0000000000000..cfdc98c6af00e --- /dev/null +++ b/editor/js/MultipleSelectionControls.js @@ -0,0 +1,85 @@ +import { UIButton, UIPanel } from './libs/ui.js'; + +const toggleMultipleSelectionImg = ''; +const submitMultipleSelectionImg = ''; +const cancelMultipleSelectionImg = ''; + +function MultipleSelectionControls( editor ) { + + const signals = editor.signals; + const strings = editor.strings; + + const container = new UIPanel(); + container.setId( 'multiple-selection-buttons' ); + + // Toggle Button + const toggleIcon = document.createElement( 'img' ); + toggleIcon.title = strings.getKey( 'toolbar/toggle-multiple-selection' ); + toggleIcon.src = toggleMultipleSelectionImg; + + const toggleBtn = new UIButton(); + toggleBtn.setClass( 'ms-button ms-button__toggle' ); + toggleBtn.dom.appendChild( toggleIcon ); + toggleBtn.onClick( function () { + + signals.toggleMultipleSelection.dispatch(); + + } ); + + // Cancel Button + const cancelIcon = document.createElement( 'img' ); + cancelIcon.title = strings.getKey( 'toolbar/cancel-multiple-selection' ); + cancelIcon.src = cancelMultipleSelectionImg; + + const cancelBtn = new UIButton(); + cancelBtn.setClass( 'ms-button ms-button__cancel' ); + cancelBtn.dom.appendChild( cancelIcon ); + cancelBtn.dom.style.display = 'none'; + cancelBtn.onClick( function () { + + signals.cancelMultipleSelection.dispatch(); + + } ); + + // Submit Button + const submitIcon = document.createElement( 'img' ); + submitIcon.title = strings.getKey( 'toolbar/submit-multiple-selection' ); + submitIcon.src = submitMultipleSelectionImg; + + const submitBtn = new UIButton(); + submitBtn.setClass( 'ms-button ms-button__submit' ); + submitBtn.dom.appendChild( submitIcon ); + submitBtn.dom.style.display = 'none'; + submitBtn.onClick( function () { + + signals.submitMultipleSelection.dispatch(); + + } ); + + container.add( toggleBtn ); + container.add( cancelBtn ); + container.add( submitBtn ); + + signals.showMultipleSelectionControls.add( function () { + + submitBtn.dom.style.display = 'block'; + cancelBtn.dom.style.display = 'block'; + + } ); + + signals.hideMultipleSelectionControls.add( function () { + + submitBtn.dom.style.display = 'none'; + cancelBtn.dom.style.display = 'none'; + + } ); + + container.toggleBtn = toggleBtn; + container.cancelBtn = cancelBtn; + container.submitBtn = submitBtn; + + return container; + +} + +export { MultipleSelectionControls }; diff --git a/editor/js/Strings.js b/editor/js/Strings.js index 18fd294af05f0..bfb15fbccd8e3 100644 --- a/editor/js/Strings.js +++ b/editor/js/Strings.js @@ -331,7 +331,9 @@ function Strings( config ) { 'toolbar/rotate': 'Rotate', 'toolbar/scale': 'Scale', 'toolbar/local': 'Local', - 'toolbar/multiple-selection': 'Toggle multiple selection', + 'toolbar/toggle-multiple-selection': 'Toggle multiple selection', + 'toolbar/submit-multiple-selection': 'Submit multiple selection', + 'toolbar/cancel-multiple-selection': 'Cancel multiple selection', 'viewport/info/objects': 'Objects', 'viewport/info/vertices': 'Vertices', @@ -667,7 +669,9 @@ function Strings( config ) { 'toolbar/rotate': 'Rotation', 'toolbar/scale': 'Échelle', 'toolbar/local': 'Local', - 'toolbar/multiple-selection': 'Basculer la sélection multiple', + 'toolbar/toggle-multiple-selection': 'Basculer la sélection multiple', + 'toolbar/submit-multiple-selection': 'Soumettre une sélection multiple', + 'toolbar/cancel-multiple-selection': 'Annuler la sélection multiple', 'viewport/info/objects': 'Objets', 'viewport/info/vertices': 'Sommets', @@ -1003,7 +1007,9 @@ function Strings( config ) { 'toolbar/rotate': '旋转', 'toolbar/scale': '缩放', 'toolbar/local': '本地', - 'toolbar/multiple-selection': '切换多项选择', + 'toolbar/toggle-multiple-selection': '切换多项选择', + 'toolbar/submit-multiple-selection': '提交多项选择', + 'toolbar/cancel-multiple-selection': '取消多项选择', 'viewport/info/objects': '物体', 'viewport/info/vertices': '顶点', diff --git a/editor/js/Toolbar.js b/editor/js/Toolbar.js index d8ca7991ffd4b..1ba7f90fd9a2f 100644 --- a/editor/js/Toolbar.js +++ b/editor/js/Toolbar.js @@ -1,4 +1,5 @@ import { UIPanel, UIButton, UICheckbox } from './libs/ui.js'; +import { MultipleSelectionControls } from './MultipleSelectionControls.js'; /* '../images/translate.svg' */ const translateImg = ''; @@ -6,8 +7,6 @@ const translateImg = ' const rotateImg = ''; /* '../images/scale.svg' */ const scaleImg = ''; -/* '../images/multiple-selection.svg' */ -const multipleSelectionImg = ''; function Toolbar( editor ) { @@ -59,17 +58,7 @@ function Toolbar( editor ) { } ); container.add( scale ); - const multipleSelectionIcon = document.createElement( 'img' ); - multipleSelectionIcon.title = strings.getKey( 'toolbar/multiple-selection' ); - multipleSelectionIcon.src = multipleSelectionImg; - - const multipleSelection = new UIButton(); - multipleSelection.dom.appendChild( multipleSelectionIcon ); - multipleSelection.onClick( function () { - - signals.toggleMultipleSelection.dispatch(); - - } ); + const multipleSelection = new MultipleSelectionControls( editor ); container.add( multipleSelection ); const local = new UICheckbox( false ); @@ -88,7 +77,7 @@ function Toolbar( editor ) { translate.dom.classList.remove( 'selected' ); rotate.dom.classList.remove( 'selected' ); scale.dom.classList.remove( 'selected' ); - multipleSelection.dom.classList.remove( 'selected' ); + multipleSelection.toggleBtn.dom.classList.remove( 'selected' ); signals.disableMultipleSelection.dispatch(); @@ -108,7 +97,13 @@ function Toolbar( editor ) { rotate.dom.classList.remove( 'selected' ); scale.dom.classList.remove( 'selected' ); - multipleSelection.dom.classList.toggle( 'selected' ); + multipleSelection.toggleBtn.dom.classList.toggle( 'selected' ); + + } ); + + signals.disableMultipleSelection.add( function () { + + multipleSelection.toggleBtn.dom.classList.remove( 'selected' ); } ); diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index b2d170bf577e1..dd10c6fe3166d 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -17,7 +17,12 @@ import { SetScaleCommand } from './commands/SetScaleCommand.js'; import { RoomEnvironment } from '../../examples/jsm/environments/RoomEnvironment.js'; import { MultipleSelection } from './libs/multiple-selection/multiple-selection.js'; -import { AddMultipleSelectionGroup, RemoveMultipleSelectionGroup } from './commands/Commands.js'; +import { + AddMultipleSelectionGroupCommand, + RemoveMultipleSelectionGroupCommand, + SubmitMultipleSelectionCommand, + CancelMultipleSelectionCommand +} from './commands/Commands.js'; function Viewport( editor ) { @@ -92,13 +97,13 @@ function Viewport( editor ) { selectionBox.visible = false; sceneHelpers.add( selectionBox ); - const multipleSelection = new MultipleSelection( editor.viewportCamera, scene ); + const multipleSelection = new MultipleSelection( editor ); multipleSelection.addEventListener( 'pointerdown', () => { - editor.execute( new RemoveMultipleSelectionGroup( editor ) ); + editor.execute( new RemoveMultipleSelectionGroupCommand( editor ) ); - multipleSelection.hideControlButtons(); + signals.hideMultipleSelectionControls.dispatch(); } ); @@ -106,23 +111,15 @@ function Viewport( editor ) { if ( ! selectedMeshes.length ) { - editor.execute( new RemoveMultipleSelectionGroup( editor ) ); + editor.execute( new RemoveMultipleSelectionGroupCommand( editor ) ); return; } - editor.execute( new AddMultipleSelectionGroup( editor, selectedMeshes ) ); + editor.execute( new AddMultipleSelectionGroupCommand( editor, selectedMeshes ) ); - multipleSelection.showControlButtons(); - - } ); - - multipleSelection.addEventListener( 'cancel-selection', () => { - - editor.execute( new RemoveMultipleSelectionGroup( editor ) ); - - multipleSelection.hideControlButtons(); + signals.showMultipleSelectionControls.dispatch(); } ); @@ -406,6 +403,20 @@ function Viewport( editor ) { } ); + signals.submitMultipleSelection.add( function () { + + editor.execute( new SubmitMultipleSelectionCommand( editor, multipleSelection ) ); + render(); + + } ); + + signals.cancelMultipleSelection.add( function () { + + editor.execute( new CancelMultipleSelectionCommand( editor, multipleSelection ) ); + render(); + + } ); + signals.transformModeChanged.add( function ( mode ) { transformControls.setMode( mode ); diff --git a/editor/js/commands/AddMultipleSelectionGroup.js b/editor/js/commands/AddMultipleSelectionGroupCommand.js similarity index 73% rename from editor/js/commands/AddMultipleSelectionGroup.js rename to editor/js/commands/AddMultipleSelectionGroupCommand.js index c951a3d23916b..5aacea1c7d21f 100644 --- a/editor/js/commands/AddMultipleSelectionGroup.js +++ b/editor/js/commands/AddMultipleSelectionGroupCommand.js @@ -2,13 +2,13 @@ import { Group } from 'three'; import { Command } from '../Command.js'; import { AddObjectCommand } from './AddObjectCommand.js'; -class AddMultipleSelectionGroup extends Command { +class AddMultipleSelectionGroupCommand extends Command { constructor( editor, meshes ) { super( editor ); - this.type = 'AddMultipleSelectionGroup'; + this.type = 'AddMultipleSelectionGroupCommand'; this.meshes = meshes; } @@ -25,4 +25,4 @@ class AddMultipleSelectionGroup extends Command { } -export { AddMultipleSelectionGroup }; +export { AddMultipleSelectionGroupCommand }; diff --git a/editor/js/commands/CancelMultipleSelectionCommand.js b/editor/js/commands/CancelMultipleSelectionCommand.js new file mode 100644 index 0000000000000..4355a02af7b68 --- /dev/null +++ b/editor/js/commands/CancelMultipleSelectionCommand.js @@ -0,0 +1,27 @@ +import { Command } from '../Command.js'; +import { RemoveMultipleSelectionGroupCommand } from './RemoveMultipleSelectionGroupCommand.js'; + +class CancelMultipleSelectionCommand extends Command { + + constructor( editor, multipleSelection ) { + + super( editor ); + + this.type = 'CancelMultipleSelectionCommand'; + this.multipleSelection = multipleSelection; + + } + + execute() { + + this.editor.execute( new RemoveMultipleSelectionGroupCommand( this.editor ) ); + this.multipleSelection.disable(); + this.multipleSelection._displayOriginalMeshes(); + this.editor.signals.hideMultipleSelectionControls.dispatch(); + this.editor.signals.disableMultipleSelection.dispatch(); + + } + +} + +export { CancelMultipleSelectionCommand }; diff --git a/editor/js/commands/Commands.js b/editor/js/commands/Commands.js index 5b6158c8cf3f9..81b65acb3940d 100644 --- a/editor/js/commands/Commands.js +++ b/editor/js/commands/Commands.js @@ -19,5 +19,7 @@ export { SetSceneCommand } from './SetSceneCommand.js'; export { SetScriptValueCommand } from './SetScriptValueCommand.js'; export { SetUuidCommand } from './SetUuidCommand.js'; export { SetValueCommand } from './SetValueCommand.js'; -export { RemoveMultipleSelectionGroup } from './RemoveMultipleSelectionGroup.js'; -export { AddMultipleSelectionGroup } from './AddMultipleSelectionGroup.js'; +export { RemoveMultipleSelectionGroupCommand } from './RemoveMultipleSelectionGroupCommand.js'; +export { AddMultipleSelectionGroupCommand } from './AddMultipleSelectionGroupCommand.js'; +export { CancelMultipleSelectionCommand } from './CancelMultipleSelectionCommand.js'; +export { SubmitMultipleSelectionCommand } from './SubmitMultipleSelectionCommand.js'; diff --git a/editor/js/commands/RemoveMultipleSelectionGroup.js b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js similarity index 68% rename from editor/js/commands/RemoveMultipleSelectionGroup.js rename to editor/js/commands/RemoveMultipleSelectionGroupCommand.js index 0131fb40e8e1a..81fc152abe50a 100644 --- a/editor/js/commands/RemoveMultipleSelectionGroup.js +++ b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js @@ -1,13 +1,13 @@ import { Command } from '../Command.js'; import { RemoveObjectCommand } from './RemoveObjectCommand.js'; -class RemoveMultipleSelectionGroup extends Command { +class RemoveMultipleSelectionGroupCommand extends Command { constructor( editor ) { super( editor ); - this.type = 'RemoveMultipleSelectionGroup'; + this.type = 'RemoveMultipleSelectionGroupCommand'; } @@ -25,4 +25,4 @@ class RemoveMultipleSelectionGroup extends Command { } -export { RemoveMultipleSelectionGroup }; +export { RemoveMultipleSelectionGroupCommand }; diff --git a/editor/js/commands/SubmitMultipleSelectionCommand.js b/editor/js/commands/SubmitMultipleSelectionCommand.js new file mode 100644 index 0000000000000..e7274da7d3b41 --- /dev/null +++ b/editor/js/commands/SubmitMultipleSelectionCommand.js @@ -0,0 +1,34 @@ +import { Command } from '../Command.js'; +import { SetPositionCommand } from './SetPositionCommand.js'; +import { RemoveMultipleSelectionGroupCommand } from './RemoveMultipleSelectionGroupCommand.js'; + +class SubmitMultipleSelectionCommand extends Command { + + constructor( editor, multipleSelection ) { + + super( editor ); + + this.type = 'SubmitMultipleSelectionCommand'; + this.multipleSelection = multipleSelection; + + } + + execute() { + + for ( const selectedMesh of this.multipleSelection.selectedMeshes ) { + + this.editor.execute( new SetPositionCommand( this.editor, selectedMesh._original_mesh, selectedMesh.matrixWorld.getPosition() ) ); + + } + + this.editor.execute( new RemoveMultipleSelectionGroupCommand( this.editor ) ); + this.multipleSelection.disable(); + this.multipleSelection._displayOriginalMeshes(); + this.editor.signals.hideMultipleSelectionControls.dispatch(); + this.editor.signals.disableMultipleSelection.dispatch(); + + } + +} + +export { SubmitMultipleSelectionCommand }; diff --git a/editor/js/libs/multiple-selection/multiple-selection.js b/editor/js/libs/multiple-selection/multiple-selection.js index 9249f242ec566..6e7efe06ee9c1 100644 --- a/editor/js/libs/multiple-selection/multiple-selection.js +++ b/editor/js/libs/multiple-selection/multiple-selection.js @@ -4,22 +4,19 @@ import { SelectionDrawer } from './selection-drawer.js'; class MultipleSelection { - constructor( camera, scene, renderer ) { + constructor( editor, renderer ) { this.enabled = false; - this._camera = camera; - this._scene = scene; + this._camera = editor.camera; + this._scene = editor.scene; this._renderer = renderer; this._selectionDrawer = null; this._selectionBox = null; - this._selectedMeshes = []; + this.selectedMeshes = []; this._hiddenOriginalMeshes = []; this._pointerUpCustomCallback = null; this._pointerDownCustomCallback = null; this._pointerMoveCustomCallback = null; - this._cancelBtnClickCustomCallback = null; - this._submitBtn = null; - this._cancelBtn = null; } @@ -80,9 +77,6 @@ class MultipleSelection { case 'pointermove': this._pointerMoveCustomCallback = callback; break; - case 'cancel-selection': - this._cancelBtnClickCustomCallback = callback; - break; default: break; @@ -90,51 +84,6 @@ class MultipleSelection { } - showControlButtons() { - - this._submitBtn = document.createElement( 'button' ); - this._submitBtn.innerHTML = 'Submit'; - this._renderer.domElement.parentElement.appendChild( this._submitBtn ); - - this._submitBtn.style = 'position: absolute; top: 50px; right: 10px;'; - - this._cancelBtn = document.createElement( 'button' ); - this._cancelBtn.innerHTML = 'Cancel'; - this._renderer.domElement.parentElement.appendChild( this._cancelBtn ); - - this._cancelBtn.style = 'position: absolute; top: 50px; right: 75px;'; - - this._cancelBtn.addEventListener( 'click', this._cancelBtnClick ); - - } - - hideControlButtons() { - - if ( this._submitBtn ) { - - this._submitBtn.remove(); - - } - - if ( this._cancelBtn ) { - - this._cancelBtn.removeEventListener( 'click', this._cancelBtnClick ); - this._cancelBtn.remove(); - - } - - } - - _cancelBtnClick() { - - if ( this._cancelBtnClickCustomCallback ) { - - this._cancelBtnClickCustomCallback(); - - } - - } - _displayOriginalMeshes() { this._hiddenOriginalMeshes.forEach( mesh => { @@ -218,7 +167,7 @@ class MultipleSelection { if ( this._pointerMoveCustomCallback ) { - this._pointerMoveCustomCallback( this._selectedMeshes ); + this._pointerMoveCustomCallback( this.selectedMeshes ); } @@ -232,13 +181,13 @@ class MultipleSelection { this._selectionBox.endPoint.set( cursorPosition.x, cursorPosition.y, 0.5 ); - this._selectedMeshes = this._cloneMeshes( this._keepOnlyMeshes( this._selectionBox.select() ) ); + this.selectedMeshes = this._cloneMeshes( this._keepOnlyMeshes( this._selectionBox.select() ) ); - this._hideOriginalMeshes( this._selectedMeshes ); + this._hideOriginalMeshes( this.selectedMeshes ); if ( this._pointerUpCustomCallback ) { - this._pointerUpCustomCallback( this._selectedMeshes ); + this._pointerUpCustomCallback( this.selectedMeshes ); } From 253af73ec31080507e13a0d0eb95050780b52bca Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Tue, 19 Jul 2022 20:29:45 +0300 Subject: [PATCH 09/15] feat: add more specific MultipleSelectionGroup, preserve scale and rotation changes on submit too --- .../AddMultipleSelectionGroupCommand.js | 5 +-- .../RemoveMultipleSelectionGroupCommand.js | 2 +- .../SubmitMultipleSelectionCommand.js | 36 ++++++++++++++++++- editor/js/objects/MultipleSelectionGroup.js | 17 +++++++++ 4 files changed, 56 insertions(+), 4 deletions(-) create mode 100644 editor/js/objects/MultipleSelectionGroup.js diff --git a/editor/js/commands/AddMultipleSelectionGroupCommand.js b/editor/js/commands/AddMultipleSelectionGroupCommand.js index 5aacea1c7d21f..2a5a7d198a196 100644 --- a/editor/js/commands/AddMultipleSelectionGroupCommand.js +++ b/editor/js/commands/AddMultipleSelectionGroupCommand.js @@ -1,6 +1,6 @@ -import { Group } from 'three'; import { Command } from '../Command.js'; import { AddObjectCommand } from './AddObjectCommand.js'; +import { MultipleSelectionGroup } from '../objects/MultipleSelectionGroup.js'; class AddMultipleSelectionGroupCommand extends Command { @@ -15,8 +15,9 @@ class AddMultipleSelectionGroupCommand extends Command { execute() { - const group = new Group(); + const group = new MultipleSelectionGroup(); group.name = 'Multiple Selection Group'; + this.meshes.forEach( mesh => group.add( mesh ) ); this.editor.execute( new AddObjectCommand( this.editor, group ) ); diff --git a/editor/js/commands/RemoveMultipleSelectionGroupCommand.js b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js index 81fc152abe50a..d09616526cc9f 100644 --- a/editor/js/commands/RemoveMultipleSelectionGroupCommand.js +++ b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js @@ -13,7 +13,7 @@ class RemoveMultipleSelectionGroupCommand extends Command { execute() { - const group = this.editor.scene.getObjectByName( 'Multiple Selection Group' ); + const group = this.editor.scene.getObjectByProperty( 'type', 'MultipleSelectionGroup' ); if ( group ) { diff --git a/editor/js/commands/SubmitMultipleSelectionCommand.js b/editor/js/commands/SubmitMultipleSelectionCommand.js index e7274da7d3b41..7360ad7d49342 100644 --- a/editor/js/commands/SubmitMultipleSelectionCommand.js +++ b/editor/js/commands/SubmitMultipleSelectionCommand.js @@ -1,5 +1,8 @@ +import { Vector3, Quaternion, Euler } from 'three'; import { Command } from '../Command.js'; import { SetPositionCommand } from './SetPositionCommand.js'; +import { SetScaleCommand } from './SetScaleCommand.js'; +import { SetRotationCommand } from './SetRotationCommand.js'; import { RemoveMultipleSelectionGroupCommand } from './RemoveMultipleSelectionGroupCommand.js'; class SubmitMultipleSelectionCommand extends Command { @@ -13,11 +16,42 @@ class SubmitMultipleSelectionCommand extends Command { } + _getPositionValue( mesh ) { + + const vector = new Vector3(); + mesh.getWorldPosition( vector ); + + return vector; + + } + + _getRotationValue( mesh ) { + + const quaternion = new Quaternion(); + mesh.getWorldQuaternion( quaternion ); + const rotation = new Euler(); + rotation.setFromQuaternion( quaternion ); + + return rotation; + + } + + _getScaleValue( mesh ) { + + const vector = new Vector3(); + mesh.getWorldScale( vector ); + + return vector; + + } + execute() { for ( const selectedMesh of this.multipleSelection.selectedMeshes ) { - this.editor.execute( new SetPositionCommand( this.editor, selectedMesh._original_mesh, selectedMesh.matrixWorld.getPosition() ) ); + this.editor.execute( new SetPositionCommand( this.editor, selectedMesh._original_mesh, this._getPositionValue( selectedMesh ) ) ); + this.editor.execute( new SetScaleCommand( this.editor, selectedMesh._original_mesh, this._getScaleValue( selectedMesh ) ) ); + this.editor.execute( new SetRotationCommand( this.editor, selectedMesh._original_mesh, this._getRotationValue( selectedMesh ) ) ); } diff --git a/editor/js/objects/MultipleSelectionGroup.js b/editor/js/objects/MultipleSelectionGroup.js new file mode 100644 index 0000000000000..e40492ab421d5 --- /dev/null +++ b/editor/js/objects/MultipleSelectionGroup.js @@ -0,0 +1,17 @@ +import { Group } from 'three'; + +class MultipleSelectionGroup extends Group { + + constructor() { + + super(); + + this.type = 'MultipleSelectionGroup'; + + } + +} + +MultipleSelectionGroup.prototype.isGroup = true; + +export { MultipleSelectionGroup }; From eb18708a96ceab5bca08411d122affdadce61cb2 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Wed, 20 Jul 2022 21:28:04 +0300 Subject: [PATCH 10/15] feat: enhance context menu options for multiple selection group --- editor/js/Sidebar.Scene.js | 18 +++++++++++++++++- 1 file changed, 17 insertions(+), 1 deletion(-) diff --git a/editor/js/Sidebar.Scene.js b/editor/js/Sidebar.Scene.js index c9fd114e20466..de12f7ade309a 100644 --- a/editor/js/Sidebar.Scene.js +++ b/editor/js/Sidebar.Scene.js @@ -379,7 +379,15 @@ function SidebarScene( editor ) { outliner.setOptions( options ); - outliner.setContextMenuOptions( [ 'Copy', 'Cut', 'Paste', 'Clone', 'Delete' ] ); + if ( editor.selected && editor.selected.type === 'MultipleSelectionGroup' ) { + + outliner.setContextMenuOptions( [ 'Cancel', 'Submit' ] ); + + } else { + + outliner.setContextMenuOptions( [ 'Copy', 'Cut', 'Paste', 'Clone', 'Delete' ] ); + + } outliner.onContextMenuChange( function ( value ) { @@ -412,6 +420,14 @@ function SidebarScene( editor ) { editor.execute( new AddObjectCommand( editor, copiedObject, editor.selected ) ); + } else if ( value === 'Submit' ) { + + signals.submitMultipleSelection.dispatch(); + + } else if ( value === 'Cancel' ) { + + signals.cancelMultipleSelection.dispatch(); + } } From c33ea5b4705882f472da1032ac83d09dbdd9f267 Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Wed, 27 Jul 2022 13:45:29 +0300 Subject: [PATCH 11/15] chore: add brief explanation of camera blocking functionality --- editor/js/Viewport.js | 1 + 1 file changed, 1 insertion(+) diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index dd10c6fe3166d..6d6758bd1c61d 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -577,6 +577,7 @@ function Viewport( editor ) { signals.objectRemoved.add( function ( object ) { + // we explicitly prevent manipulations of the camera when multiple selection is enabled if ( multipleSelection.enabled ) return; controls.enabled = true; // see #14180 From 3878f1ab9c71b0f328bf06ceeb81444dc2d72d7f Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Wed, 27 Jul 2022 14:14:45 +0300 Subject: [PATCH 12/15] feat: add "File/Exit" option to the menubar --- editor/js/Editor.js | 1 + editor/js/Menubar.File.js | 14 ++++++++++++++ editor/js/Strings.js | 3 +++ 3 files changed, 18 insertions(+) diff --git a/editor/js/Editor.js b/editor/js/Editor.js index 7fea3ae5b3d9c..83306cfc9a7d1 100644 --- a/editor/js/Editor.js +++ b/editor/js/Editor.js @@ -46,6 +46,7 @@ function Editor( providedDefaultCamera ) { // notifications editorCleared: new Signal(), + editorClosed: new Signal(), savingStarted: new Signal(), savingFinished: new Signal(), diff --git a/editor/js/Menubar.File.js b/editor/js/Menubar.File.js index 59eb58236d139..4118f4f8c042f 100644 --- a/editor/js/Menubar.File.js +++ b/editor/js/Menubar.File.js @@ -8,6 +8,7 @@ function MenubarFile( editor ) { const config = editor.config; const strings = editor.strings; + const signals = editor.signals; const container = new UIPanel(); container.setClass( 'menu' ); @@ -481,6 +482,19 @@ function MenubarFile( editor ) { } ); options.add( option ); + // Exit + option = new UIRow(); + option.setClass( 'option' ); + option.setTextContent( strings.getKey( 'menubar/file/exit' ) ); + option.onClick( function () { + + signals.editorClosed.dispatch(); + + } ); + + options.add( option ); + + // const link = document.createElement( 'a' ); diff --git a/editor/js/Strings.js b/editor/js/Strings.js index bfb15fbccd8e3..46ea3d9a37bf3 100644 --- a/editor/js/Strings.js +++ b/editor/js/Strings.js @@ -23,6 +23,7 @@ function Strings( config ) { 'menubar/file/export/stl_binary': 'Export STL (Binary)', 'menubar/file/export/usdz': 'Export USDZ', 'menubar/file/publish': 'Publish', + 'menubar/file/exit': 'Exit', 'menubar/edit': 'Edit', 'menubar/edit/undo': 'Undo (Ctrl+Z)', @@ -361,6 +362,7 @@ function Strings( config ) { 'menubar/file/export/stl_binary': 'Exporter STL (Binaire)', 'menubar/file/export/usdz': 'Exporter USDZ', 'menubar/file/publish': 'Publier', + 'menubar/file/exit': 'Quitter', 'menubar/edit': 'Edition', 'menubar/edit/undo': 'Annuler (Ctrl+Z)', @@ -699,6 +701,7 @@ function Strings( config ) { 'menubar/file/export/stl_binary': '导出STL(二进制)', 'menubar/file/export/usdz': '导出USDZ', 'menubar/file/publish': '发布', + 'menubar/file/exit': '退出', 'menubar/edit': '编辑', 'menubar/edit/undo': '撤销 (Ctrl+Z)', From 2a48f6781b311034db53c0a3a9071e317a0446ff Mon Sep 17 00:00:00 2001 From: fluoxe7ine Date: Wed, 27 Jul 2022 14:43:56 +0300 Subject: [PATCH 13/15] feat: add shortcut info in toolbar button hint text --- editor/js/MultipleSelectionControls.js | 8 +++++--- editor/js/Strings.js | 6 +++--- editor/js/Toolbar.js | 10 ++++++---- editor/js/libs/getToolbarBtnHintText.js | 15 +++++++++++++++ 4 files changed, 29 insertions(+), 10 deletions(-) create mode 100644 editor/js/libs/getToolbarBtnHintText.js diff --git a/editor/js/MultipleSelectionControls.js b/editor/js/MultipleSelectionControls.js index cfdc98c6af00e..3a674ba202b27 100644 --- a/editor/js/MultipleSelectionControls.js +++ b/editor/js/MultipleSelectionControls.js @@ -1,4 +1,5 @@ import { UIButton, UIPanel } from './libs/ui.js'; +import { getToolbarBtnHintText } from './libs/getToolbarBtnHintText.js'; const toggleMultipleSelectionImg = ''; const submitMultipleSelectionImg = ''; @@ -8,13 +9,14 @@ function MultipleSelectionControls( editor ) { const signals = editor.signals; const strings = editor.strings; + const config = editor.config; const container = new UIPanel(); container.setId( 'multiple-selection-buttons' ); // Toggle Button const toggleIcon = document.createElement( 'img' ); - toggleIcon.title = strings.getKey( 'toolbar/toggle-multiple-selection' ); + toggleIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/toggle-multiple-selection' ), config.getKey( 'settings/shortcuts/multiple-selection' ) ); toggleIcon.src = toggleMultipleSelectionImg; const toggleBtn = new UIButton(); @@ -28,7 +30,7 @@ function MultipleSelectionControls( editor ) { // Cancel Button const cancelIcon = document.createElement( 'img' ); - cancelIcon.title = strings.getKey( 'toolbar/cancel-multiple-selection' ); + cancelIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/cancel-multiple-selection' ) ); cancelIcon.src = cancelMultipleSelectionImg; const cancelBtn = new UIButton(); @@ -43,7 +45,7 @@ function MultipleSelectionControls( editor ) { // Submit Button const submitIcon = document.createElement( 'img' ); - submitIcon.title = strings.getKey( 'toolbar/submit-multiple-selection' ); + submitIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/submit-multiple-selection' ) ); submitIcon.src = submitMultipleSelectionImg; const submitBtn = new UIButton(); diff --git a/editor/js/Strings.js b/editor/js/Strings.js index 46ea3d9a37bf3..50fbc9af1d8d5 100644 --- a/editor/js/Strings.js +++ b/editor/js/Strings.js @@ -332,9 +332,9 @@ function Strings( config ) { 'toolbar/rotate': 'Rotate', 'toolbar/scale': 'Scale', 'toolbar/local': 'Local', - 'toolbar/toggle-multiple-selection': 'Toggle multiple selection', - 'toolbar/submit-multiple-selection': 'Submit multiple selection', - 'toolbar/cancel-multiple-selection': 'Cancel multiple selection', + 'toolbar/toggle-multiple-selection': 'Toggle Multiple Selection', + 'toolbar/submit-multiple-selection': 'Submit Multiple Selection', + 'toolbar/cancel-multiple-selection': 'Cancel Multiple Selection', 'viewport/info/objects': 'Objects', 'viewport/info/vertices': 'Vertices', diff --git a/editor/js/Toolbar.js b/editor/js/Toolbar.js index 1ba7f90fd9a2f..e04034ed19f48 100644 --- a/editor/js/Toolbar.js +++ b/editor/js/Toolbar.js @@ -1,5 +1,6 @@ -import { UIPanel, UIButton, UICheckbox } from './libs/ui.js'; +import { UIButton, UICheckbox, UIPanel } from './libs/ui.js'; import { MultipleSelectionControls } from './MultipleSelectionControls.js'; +import { getToolbarBtnHintText } from './libs/getToolbarBtnHintText.js'; /* '../images/translate.svg' */ const translateImg = ''; @@ -12,6 +13,7 @@ function Toolbar( editor ) { const signals = editor.signals; const strings = editor.strings; + const config = editor.config; const container = new UIPanel(); container.setId( 'toolbar' ); @@ -19,7 +21,7 @@ function Toolbar( editor ) { // translate / rotate / scale const translateIcon = document.createElement( 'img' ); - translateIcon.title = strings.getKey( 'toolbar/translate' ); + translateIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/translate' ), config.getKey( 'settings/shortcuts/translate' ) ); translateIcon.src = translateImg; const translate = new UIButton(); @@ -33,7 +35,7 @@ function Toolbar( editor ) { container.add( translate ); const rotateIcon = document.createElement( 'img' ); - rotateIcon.title = strings.getKey( 'toolbar/rotate' ); + rotateIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/rotate' ), config.getKey( 'settings/shortcuts/rotate' ) ); rotateIcon.src = rotateImg; const rotate = new UIButton(); @@ -46,7 +48,7 @@ function Toolbar( editor ) { container.add( rotate ); const scaleIcon = document.createElement( 'img' ); - scaleIcon.title = strings.getKey( 'toolbar/scale' ); + scaleIcon.title = getToolbarBtnHintText( strings.getKey( 'toolbar/scale' ), config.getKey( 'settings/shortcuts/scale' ) ); scaleIcon.src = scaleImg; const scale = new UIButton(); diff --git a/editor/js/libs/getToolbarBtnHintText.js b/editor/js/libs/getToolbarBtnHintText.js new file mode 100644 index 0000000000000..d9bee375c30a4 --- /dev/null +++ b/editor/js/libs/getToolbarBtnHintText.js @@ -0,0 +1,15 @@ +function getToolbarBtnHintText( optionText, shortcutText ) { + + let result = optionText; + + if ( shortcutText ) { + + result += ' ' + '(' + shortcutText.toUpperCase() + ')'; + + } + + return result; + +} + +export { getToolbarBtnHintText }; From e939741ebfb92b7a3cfba3eddc11dcf5254cbd9f Mon Sep 17 00:00:00 2001 From: Anton Slipchenko Date: Tue, 9 Aug 2022 01:10:28 +0300 Subject: [PATCH 14/15] added rotation functonallity --- editor/js/Viewport.js | 2 ++ .../AddMultipleSelectionGroupCommand.js | 14 +++++++++- .../RemoveMultipleSelectionGroupCommand.js | 7 +++++ examples/jsm/controls/ArcballControls.js | 2 +- examples/jsm/controls/TransformControls.js | 28 +++++++++++++------ examples/jsm/interactive/HTMLMesh.js | 4 +-- examples/jsm/nodes/core/InputNode.js | 2 +- examples/jsm/nodes/core/NodeUtils.js | 2 +- examples/jsm/nodes/lights/LightsNode.js | 6 ++-- examples/jsm/nodes/materials/Materials.js | 2 +- examples/jsm/nodes/shadernode/ShaderNode.js | 10 +++---- .../jsm/renderers/webgl/nodes/WebGLNodes.js | 2 +- examples/jsm/utils/BufferGeometryUtils.js | 2 +- 13 files changed, 58 insertions(+), 25 deletions(-) diff --git a/editor/js/Viewport.js b/editor/js/Viewport.js index 6d6758bd1c61d..9c4e21cf4beed 100644 --- a/editor/js/Viewport.js +++ b/editor/js/Viewport.js @@ -98,6 +98,7 @@ function Viewport( editor ) { sceneHelpers.add( selectionBox ); const multipleSelection = new MultipleSelection( editor ); + editor.multipleSelection = multipleSelection; multipleSelection.addEventListener( 'pointerdown', () => { @@ -420,6 +421,7 @@ function Viewport( editor ) { signals.transformModeChanged.add( function ( mode ) { transformControls.setMode( mode ); + render(); } ); diff --git a/editor/js/commands/AddMultipleSelectionGroupCommand.js b/editor/js/commands/AddMultipleSelectionGroupCommand.js index 2a5a7d198a196..20cf11cfa5f4b 100644 --- a/editor/js/commands/AddMultipleSelectionGroupCommand.js +++ b/editor/js/commands/AddMultipleSelectionGroupCommand.js @@ -1,6 +1,7 @@ import { Command } from '../Command.js'; import { AddObjectCommand } from './AddObjectCommand.js'; import { MultipleSelectionGroup } from '../objects/MultipleSelectionGroup.js'; +import { Group, Sphere, Vector3 } from 'three'; class AddMultipleSelectionGroupCommand extends Command { @@ -20,7 +21,18 @@ class AddMultipleSelectionGroupCommand extends Command { this.meshes.forEach( mesh => group.add( mesh ) ); - this.editor.execute( new AddObjectCommand( this.editor, group ) ); + const positions = this.meshes.map( ( { matrixWorld } ) => { + + return new Vector3().setFromMatrixPosition( matrixWorld ); + + } ); + const sphere = new Sphere().setFromPoints( positions ); + const pivotGroup = new Group(); + pivotGroup.name = 'Pivot Group'; + this.editor.execute( new AddObjectCommand( this.editor, pivotGroup ) ); + this.editor.execute( new AddObjectCommand( this.editor, group, pivotGroup ) ); + pivotGroup.position.set( sphere.center.x, sphere.center.y, sphere.center.z ); + group.position.set( - sphere.center.x, - sphere.center.y, - sphere.center.z ); } diff --git a/editor/js/commands/RemoveMultipleSelectionGroupCommand.js b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js index d09616526cc9f..250f7678915f6 100644 --- a/editor/js/commands/RemoveMultipleSelectionGroupCommand.js +++ b/editor/js/commands/RemoveMultipleSelectionGroupCommand.js @@ -14,6 +14,13 @@ class RemoveMultipleSelectionGroupCommand extends Command { execute() { const group = this.editor.scene.getObjectByProperty( 'type', 'MultipleSelectionGroup' ); + const pivotGroup = this.editor.scene.getObjectByProperty( 'name', 'Pivot Group' ); + + if ( pivotGroup ) { + + this.editor.execute( new RemoveObjectCommand( this.editor, pivotGroup ) ); + + } if ( group ) { diff --git a/examples/jsm/controls/ArcballControls.js b/examples/jsm/controls/ArcballControls.js index e7e57c7b017ab..dd08c5f5f6700 100644 --- a/examples/jsm/controls/ArcballControls.js +++ b/examples/jsm/controls/ArcballControls.js @@ -1040,7 +1040,7 @@ class ArcballControls extends EventDispatcher { } - this._v3_1.setFromMatrixPosition(this._gizmoMatrixState); + this._v3_1.setFromMatrixPosition( this._gizmoMatrixState ); this.applyTransformMatrix( this.scale( size, this._v3_1 ) ); diff --git a/examples/jsm/controls/TransformControls.js b/examples/jsm/controls/TransformControls.js index 16fe1c105aef6..45f8caa61b0a4 100644 --- a/examples/jsm/controls/TransformControls.js +++ b/examples/jsm/controls/TransformControls.js @@ -17,8 +17,9 @@ import { Raycaster, SphereGeometry, TorusGeometry, - Vector3 + Vector3, } from 'three'; +import { MultipleSelectionGroup } from '../../../editor/js/objects/MultipleSelectionGroup.js'; const _raycaster = new Raycaster(); @@ -546,8 +547,19 @@ class TransformControls extends Object3D { // Set current object attach( object ) { - this.object = object; - this.visible = true; + if ( object instanceof MultipleSelectionGroup ) { + + this.object = object.parent; + console.log(this.object, this.mode); + console.log( 'this mode', this.mode ); + + } else { + + this.object = object; + this.visible = true; + + } + return this; @@ -598,7 +610,9 @@ class TransformControls extends Object3D { } setMode( mode ) { - + if (!this.visible) { + this.visible = true; + } this.mode = mode; } @@ -1130,7 +1144,6 @@ class TransformControlsGizmo extends Object3D { this.add( this.helper[ 'scale' ] = setupGizmo( helperScale ) ); // Pickers should be hidden always - this.picker[ 'translate' ].visible = false; this.picker[ 'rotate' ].visible = false; this.picker[ 'scale' ].visible = false; @@ -1146,7 +1159,7 @@ class TransformControlsGizmo extends Object3D { const quaternion = ( space === 'local' ) ? this.worldQuaternion : _identityQuaternion; // Show only gizmos for current transform mode - + //TODO: why do we need this logic if below in cycle we do the same??? this.gizmo[ 'translate' ].visible = this.mode === 'translate'; this.gizmo[ 'rotate' ].visible = this.mode === 'rotate'; this.gizmo[ 'scale' ].visible = this.mode === 'scale'; @@ -1378,7 +1391,6 @@ class TransformControlsGizmo extends Object3D { } else if ( this.mode === 'rotate' ) { // Align handles to current local or world rotation - _tempQuaternion2.copy( quaternion ); _alignVector.copy( this.eye ).applyQuaternion( _tempQuaternion.copy( quaternion ).invert() ); @@ -1528,7 +1540,7 @@ class TransformControlsPlane extends Mesh { case 'rotate': default: // special case for rotate - _dirVector.set( 0, 0, 0 ); + // _dirVector.set( 0, 0, 0 ); } diff --git a/examples/jsm/interactive/HTMLMesh.js b/examples/jsm/interactive/HTMLMesh.js index d9212e8f15ce1..84ed9b384f832 100644 --- a/examples/jsm/interactive/HTMLMesh.js +++ b/examples/jsm/interactive/HTMLMesh.js @@ -249,8 +249,8 @@ function html2canvas( element ) { context.save(); const dpr = window.devicePixelRatio; - context.scale(1/dpr, 1/dpr); - context.drawImage(element, 0, 0 ); + context.scale( 1 / dpr, 1 / dpr ); + context.drawImage( element, 0, 0 ); context.restore(); } else { diff --git a/examples/jsm/nodes/core/InputNode.js b/examples/jsm/nodes/core/InputNode.js index 01268edf9d6a7..5df1e845e26c1 100644 --- a/examples/jsm/nodes/core/InputNode.js +++ b/examples/jsm/nodes/core/InputNode.js @@ -51,7 +51,7 @@ class InputNode extends Node { generate( /*builder, output*/ ) { - console.warn('Abstract function.'); + console.warn( 'Abstract function.' ); } diff --git a/examples/jsm/nodes/core/NodeUtils.js b/examples/jsm/nodes/core/NodeUtils.js index efafa7d776346..42610b7487ce7 100644 --- a/examples/jsm/nodes/core/NodeUtils.js +++ b/examples/jsm/nodes/core/NodeUtils.js @@ -62,7 +62,7 @@ export const getValueType = ( value ) => { export const getValueFromType = ( type, ...params ) => { - const last4 = type?.slice( -4 ); + const last4 = type?.slice( - 4 ); if ( type === 'color' ) { diff --git a/examples/jsm/nodes/lights/LightsNode.js b/examples/jsm/nodes/lights/LightsNode.js index 9c59c5d453ba0..52ac6ebe4da1b 100644 --- a/examples/jsm/nodes/lights/LightsNode.js +++ b/examples/jsm/nodes/lights/LightsNode.js @@ -44,7 +44,7 @@ class LightsNode extends Node { if ( this._hash === null ) { let hash = ''; - + const lightNodes = this.lightNodes; for ( const lightNode of lightNodes ) { @@ -52,9 +52,9 @@ class LightsNode extends Node { hash += lightNode.light.uuid + ' '; } - + this._hash = hash; - + } return this._hash; diff --git a/examples/jsm/nodes/materials/Materials.js b/examples/jsm/nodes/materials/Materials.js index e787b52c793e5..f0ad731894014 100644 --- a/examples/jsm/nodes/materials/Materials.js +++ b/examples/jsm/nodes/materials/Materials.js @@ -47,7 +47,7 @@ NodeMaterial.fromMaterial = function ( material ) { const nodeMaterial = new materialLib[ type ]( material ); - for ( let key in material ) { + for ( const key in material ) { if ( nodeMaterial[ key ] === undefined ) { diff --git a/examples/jsm/nodes/shadernode/ShaderNode.js b/examples/jsm/nodes/shadernode/ShaderNode.js index 9256e7cdc8ac9..3636d64ff038b 100644 --- a/examples/jsm/nodes/shadernode/ShaderNode.js +++ b/examples/jsm/nodes/shadernode/ShaderNode.js @@ -187,17 +187,17 @@ const ints = [ - 1, - 2 ]; const floats = [ 0.5, 1.5, 1 / 3, 1e-6, 1e6, Math.PI, Math.PI * 2, 1 / Math.PI, 2 / Math.PI, 1 / ( Math.PI * 2 ), Math.PI / 2 ]; const boolsCacheMap = new Map(); -for ( let bool of bools ) boolsCacheMap.set( bool, new ConstNode( bool ) ); +for ( const bool of bools ) boolsCacheMap.set( bool, new ConstNode( bool ) ); const uintsCacheMap = new Map(); -for ( let uint of uints ) uintsCacheMap.set( uint, new ConstNode( uint, 'uint' ) ); +for ( const uint of uints ) uintsCacheMap.set( uint, new ConstNode( uint, 'uint' ) ); const intsCacheMap = new Map( [ ...uintsCacheMap ].map( el => new ConstNode( el.value, 'int' ) ) ); -for ( let int of ints ) intsCacheMap.set( int, new ConstNode( int, 'int' ) ); +for ( const int of ints ) intsCacheMap.set( int, new ConstNode( int, 'int' ) ); const floatsCacheMap = new Map( [ ...intsCacheMap ].map( el => new ConstNode( el.value ) ) ); -for ( let float of floats ) floatsCacheMap.set( float, new ConstNode( float ) ); -for ( let float of floats ) floatsCacheMap.set( - float, new ConstNode( - float ) ); +for ( const float of floats ) floatsCacheMap.set( float, new ConstNode( float ) ); +for ( const float of floats ) floatsCacheMap.set( - float, new ConstNode( - float ) ); export const cacheMaps = { bool: boolsCacheMap, uint: uintsCacheMap, ints: intsCacheMap, float: floatsCacheMap }; diff --git a/examples/jsm/renderers/webgl/nodes/WebGLNodes.js b/examples/jsm/renderers/webgl/nodes/WebGLNodes.js index 9648f17e23f0d..830173b24a4e0 100644 --- a/examples/jsm/renderers/webgl/nodes/WebGLNodes.js +++ b/examples/jsm/renderers/webgl/nodes/WebGLNodes.js @@ -37,7 +37,7 @@ Material.prototype.onBeforeRender = function ( renderer, scene, camera, geometry nodeFrame.updateNode( node ); } - + } } diff --git a/examples/jsm/utils/BufferGeometryUtils.js b/examples/jsm/utils/BufferGeometryUtils.js index 16024e1d241ef..ce4da6b92f162 100644 --- a/examples/jsm/utils/BufferGeometryUtils.js +++ b/examples/jsm/utils/BufferGeometryUtils.js @@ -98,7 +98,7 @@ function computeMikkTSpaceTangents( geometry, MikkTSpace, negateSign = true ) { if ( geometry !== _geometry ) { - geometry.copy( _geometry ) + geometry.copy( _geometry ); } From ce2be578ef7c05dd0b8b458e64393ceca8eab940 Mon Sep 17 00:00:00 2001 From: Anton Slipchenko Date: Tue, 9 Aug 2022 01:12:24 +0300 Subject: [PATCH 15/15] resolve lint issues --- examples/jsm/controls/TransformControls.js | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/examples/jsm/controls/TransformControls.js b/examples/jsm/controls/TransformControls.js index 45f8caa61b0a4..f3f5a5ee285ef 100644 --- a/examples/jsm/controls/TransformControls.js +++ b/examples/jsm/controls/TransformControls.js @@ -550,8 +550,6 @@ class TransformControls extends Object3D { if ( object instanceof MultipleSelectionGroup ) { this.object = object.parent; - console.log(this.object, this.mode); - console.log( 'this mode', this.mode ); } else { @@ -610,9 +608,13 @@ class TransformControls extends Object3D { } setMode( mode ) { - if (!this.visible) { + + if ( ! this.visible ) { + this.visible = true; + } + this.mode = mode; }