using UnityEngine; using UnityEditor; namespace UnityEditor.U2D.Common.Path.GUIFramework { /// /// An implementation of an IGUIState that represents a generic GUI state. /// internal class GUIState : IGUIState { private Handles.CapFunction nullCap = (int c, Vector3 p , Quaternion r, float s, EventType ev) => {}; /// /// The current mouse position. /// public Vector2 mousePosition { get { return Event.current.mousePosition; } } /// /// The currently pressed button. /// public int mouseButton { get { return Event.current.button; } } /// /// The current number of mouse clicks. /// public int clickCount { get { return Event.current.clickCount; } set { Event.current.clickCount = Mathf.Max(0, value); } } /// /// Indicates whether the shift key is pressed. /// public bool isShiftDown { get { return Event.current.shift; } } /// /// Indicates whether the alt key is pressed. /// public bool isAltDown { get { return Event.current.alt; } } /// /// Indicates whether the action key is pressed. /// public bool isActionKeyDown { get { return EditorGUI.actionKey; } } /// /// The KeyCode of the currently pressed key. /// public KeyCode keyCode { get { return Event.current.keyCode; } } /// /// The type of the current event. /// public EventType eventType { get { return Event.current.type; } } /// /// The name of the current event's command. /// public string commandName { get { return Event.current.commandName; } } /// /// The closest control to the event. /// public int nearestControl { get { return HandleUtility.nearestControl; } set { HandleUtility.nearestControl = value; } } /// /// Hot Control /// public int hotControl { get { return GUIUtility.hotControl; } set { GUIUtility.hotControl = value; } } /// /// Indicates whether the GUI has changed. /// public bool changed { get { return GUI.changed; } set { GUI.changed = value; } } /// /// Gets the ID of a nested control by a hint and focus type. /// /// The hint this function uses to identify the control ID. /// The focus Type /// Returns the ID of the control that matches the hint and focus type. public int GetControlID(int hint, FocusType focusType) { return GUIUtility.GetControlID(hint, focusType); } /// /// Adds a control to the GUIState. /// /// The ID of the control to add. /// The distance from the camera to the control. public void AddControl(int controlID, float distance) { HandleUtility.AddControl(controlID, distance); } /// /// Checks whether a slider value has changed. /// /// The ID of the slider to check. /// The slider's data. /// The new position of the slider. /// Returns `true` if the slider has changed. Otherwise, returns `false`. public bool Slider(int id, SliderData sliderData, out Vector3 newPosition) { if (mouseButton == 0 && eventType == EventType.MouseDown) { hotControl = 0; nearestControl = id; } EditorGUI.BeginChangeCheck(); newPosition = Handles.Slider2D(id, sliderData.position, sliderData.forward, sliderData.right, sliderData.up, 1f, nullCap, Vector2.zero); return EditorGUI.EndChangeCheck(); } /// /// Uses the current event. /// public void UseEvent() { Event.current.Use(); } /// /// Repaints the GUI. /// public void Repaint() { HandleUtility.Repaint(); } /// /// Checks if the current camera is valid. /// /// Returns `true` if the current camera is not null. Otherwise, returns `false`. public bool HasCurrentCamera() { return Camera.current != null; } /// /// Gets the size of the handle. /// /// The position of the handle. /// Returns the size of the handle. public float GetHandleSize(Vector3 position) { var scale = HasCurrentCamera() ? 0.01f : 0.05f; return HandleUtility.GetHandleSize(position) * scale; } /// /// Measures the GUI-space distance between two points of a segment. /// /// The first point. /// The seconde point. /// Returns the GUI-space distance between p1 and p2. public float DistanceToSegment(Vector3 p1, Vector3 p2) { p1 = HandleUtility.WorldToGUIPoint(p1); p2 = HandleUtility.WorldToGUIPoint(p2); return HandleUtility.DistancePointToLineSegment(Event.current.mousePosition, p1, p2); } /// /// Measures the distance to a circle. /// /// The center of the circle. /// The radius of the circle. /// Returns the distance to a circle with the specified center and radius. public float DistanceToCircle(Vector3 center, float radius) { return HandleUtility.DistanceToCircle(center, radius); } /// /// Transforms a GUI-space position into world space. /// /// The GUI position /// The plane normal. /// The plane position. /// Returns the world-space position of `guiPosition`. public Vector3 GUIToWorld(Vector2 guiPosition, Vector3 planeNormal, Vector3 planePos) { Vector3 worldPos = Handles.inverseMatrix.MultiplyPoint(guiPosition); if (Camera.current) { Ray ray = HandleUtility.GUIPointToWorldRay(guiPosition); planeNormal = Handles.matrix.MultiplyVector(planeNormal); planePos = Handles.matrix.MultiplyPoint(planePos); Plane plane = new Plane(planeNormal, planePos); float distance = 0f; if (plane.Raycast(ray, out distance)) { worldPos = Handles.inverseMatrix.MultiplyPoint(ray.GetPoint(distance)); } } return worldPos; } } }