using System.Collections.Generic; using BansheeEngine; namespace BansheeEditor { internal sealed class DefaultHandleManager : Handle { private struct HandledObject { public HandledObject(SceneObject so) { this.so = so; initialPosition = so.Position; initialRotation = so.Rotation; initialScale = so.LocalScale; } public SceneObject so; public Vector3 initialPosition; public Quaternion initialRotation; public Vector3 initialScale; } private SceneViewTool activeHandleType = SceneViewTool.View; private DefaultHandle activeHandle; private HandledObject[] activeSelection; private bool isDragged; protected override void PreInput() { SceneObject[] selectedSceneObjects = Selection.sceneObjects; if (selectedSceneObjects.Length == 0) { if (activeHandle != null) { activeHandle.Destroy(); activeHandle = null; } } else { if (activeHandleType != EditorApplication.ActiveSceneTool || activeHandle == null) { if (activeHandle != null) { activeHandle.Destroy(); activeHandle = null; } switch (EditorApplication.ActiveSceneTool) { case SceneViewTool.Move: activeHandle = new MoveHandle(); break; case SceneViewTool.Rotate: activeHandle = new RotateHandle(); break; case SceneViewTool.Scale: activeHandle = new ScaleHandle(); break; } activeHandleType = EditorApplication.ActiveSceneTool; } } if (activeHandle != null) { Quaternion rotation; if (EditorApplication.ActiveCoordinateMode == HandleCoordinateMode.World) rotation = Quaternion.identity; else rotation = selectedSceneObjects[0].Rotation; // We don't average rotation in case of multi-selection Vector3 position; if (EditorApplication.ActivePivotMode == HandlePivotMode.Pivot) position = selectedSceneObjects[0].Position; // Just take pivot from the first one, no averaging else { List flatenedHierarchy = new List(); foreach (var so in selectedSceneObjects) { flatenedHierarchy.AddRange(EditorUtility.FlattenHierarchy(so)); } AABox selectionBounds = EditorUtility.CalculateBounds(flatenedHierarchy.ToArray()); position = selectionBounds.center; } activeHandle.Position = position; activeHandle.Rotation = rotation; activeHandle.DoPreInput(); } } protected override void PostInput() { if (activeHandle != null) { if (activeHandle.IsDragged()) { if (!isDragged) { isDragged = true; SceneObject[] selectedSceneObjects = Selection.sceneObjects; activeSelection = new HandledObject[selectedSceneObjects.Length]; for (int i = 0; i < selectedSceneObjects.Length; i++) activeSelection[i] = new HandledObject(selectedSceneObjects[0]); } } else { isDragged = false; activeSelection = null; } activeHandle.DoPostInput(); if (activeHandle.IsDragged()) { switch (activeHandleType) { case SceneViewTool.Move: { MoveHandle moveHandle = (MoveHandle) activeHandle; foreach (var selectedObj in activeSelection) selectedObj.so.Position = selectedObj.initialPosition + moveHandle.Delta; } break; case SceneViewTool.Rotate: { RotateHandle rotateHandle = (RotateHandle) activeHandle; foreach (var selectedObj in activeSelection) selectedObj.so.Rotation = selectedObj.initialRotation * rotateHandle.Delta; } break; case SceneViewTool.Scale: { ScaleHandle scaleHandle = (ScaleHandle) activeHandle; if (EditorApplication.ActivePivotMode == HandlePivotMode.Pivot) { foreach (var selectedObj in activeSelection) selectedObj.so.LocalScale = selectedObj.initialScale + scaleHandle.Delta; } else { foreach (var selectedObj in activeSelection) { selectedObj.so.LocalScale = selectedObj.initialScale + scaleHandle.Delta; // TODO } } } break; } } } else { isDragged = false; activeSelection = null; } } protected override void Draw() { if (activeHandle != null) activeHandle.DoDraw(); } } }