mirror of
https://github.com/sinai-dev/UnityExplorer.git
synced 2025-06-16 14:17:51 +08:00
372 lines
12 KiB
C#
372 lines
12 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using UnityEngine;
|
|
using UnityEngine.UI;
|
|
using UnityExplorer.Input;
|
|
using System.IO;
|
|
#if CPP
|
|
using UnityExplorer.Unstrip;
|
|
#endif
|
|
|
|
namespace UnityExplorer.UI
|
|
{
|
|
// Handles dragging and resizing for the main explorer window.
|
|
|
|
public class PanelDragger
|
|
{
|
|
public static PanelDragger Instance { get; private set; }
|
|
|
|
public RectTransform Panel { get; set; }
|
|
|
|
private static bool s_loadedCursorImage;
|
|
|
|
public PanelDragger(RectTransform dragArea, RectTransform panelToDrag)
|
|
{
|
|
Instance = this;
|
|
DragableArea = dragArea;
|
|
Panel = panelToDrag;
|
|
|
|
UpdateResizeCache();
|
|
}
|
|
|
|
public void Update()
|
|
{
|
|
Vector3 rawMousePos = InputManager.MousePosition;
|
|
|
|
ResizeTypes type;
|
|
Vector3 resizePos = Panel.InverseTransformPoint(rawMousePos);
|
|
Vector3 dragPos = DragableArea.InverseTransformPoint(rawMousePos);
|
|
|
|
if (WasHoveringResize && m_resizeCursorImage)
|
|
{
|
|
UpdateHoverImagePos();
|
|
}
|
|
|
|
// If Mouse pressed this frame
|
|
if (InputManager.GetMouseButtonDown(0))
|
|
{
|
|
if (DragableArea.rect.Contains(dragPos))
|
|
{
|
|
OnBeginDrag();
|
|
return;
|
|
}
|
|
else if (MouseInResizeArea(resizePos))
|
|
{
|
|
type = GetResizeType(resizePos);
|
|
if (type != ResizeTypes.NONE)
|
|
{
|
|
OnBeginResize(type);
|
|
}
|
|
}
|
|
}
|
|
// If mouse still pressed from last frame
|
|
else if (InputManager.GetMouseButton(0))
|
|
{
|
|
if (WasDragging)
|
|
{
|
|
OnDrag();
|
|
}
|
|
else if (WasResizing)
|
|
{
|
|
OnResize();
|
|
}
|
|
}
|
|
// If mouse not pressed
|
|
else
|
|
{
|
|
if (WasDragging)
|
|
{
|
|
OnEndDrag();
|
|
}
|
|
else if (WasResizing)
|
|
{
|
|
OnEndResize();
|
|
}
|
|
else if (MouseInResizeArea(resizePos) && (type = GetResizeType(resizePos)) != ResizeTypes.NONE)
|
|
{
|
|
OnHoverResize(type);
|
|
}
|
|
else if (WasHoveringResize)
|
|
{
|
|
OnHoverResizeEnd();
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
#region DRAGGING
|
|
|
|
public RectTransform DragableArea { get; set; }
|
|
public bool WasDragging { get; set; }
|
|
private Vector3 m_lastDragPosition;
|
|
|
|
public void OnBeginDrag()
|
|
{
|
|
WasDragging = true;
|
|
m_lastDragPosition = InputManager.MousePosition;
|
|
}
|
|
|
|
public void OnDrag()
|
|
{
|
|
Vector3 diff = InputManager.MousePosition - m_lastDragPosition;
|
|
m_lastDragPosition = InputManager.MousePosition;
|
|
|
|
Vector3 pos = Panel.localPosition;
|
|
float z = pos.z;
|
|
pos += diff;
|
|
pos.z = z;
|
|
Panel.localPosition = pos;
|
|
}
|
|
|
|
public void OnEndDrag()
|
|
{
|
|
WasDragging = false;
|
|
//UpdateResizeCache();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region RESIZE
|
|
|
|
private const int RESIZE_THICKNESS = 10;
|
|
|
|
private readonly Vector2 minResize = new Vector2(733, 542);
|
|
|
|
private bool WasResizing { get; set; }
|
|
private ResizeTypes m_currentResizeType = ResizeTypes.NONE;
|
|
private Vector2 m_lastResizePos;
|
|
|
|
private bool WasHoveringResize { get; set; }
|
|
private ResizeTypes m_lastResizeHoverType;
|
|
public GameObject m_resizeCursorImage;
|
|
|
|
private Rect m_resizeRect;
|
|
|
|
private readonly Dictionary<ResizeTypes, Rect> m_resizeMask = new Dictionary<ResizeTypes, Rect>
|
|
{
|
|
{ ResizeTypes.Top, Rect.zero },
|
|
{ ResizeTypes.Left, Rect.zero },
|
|
{ ResizeTypes.Right, Rect.zero },
|
|
{ ResizeTypes.Bottom, Rect.zero },
|
|
};
|
|
|
|
[Flags]
|
|
public enum ResizeTypes
|
|
{
|
|
NONE = 0,
|
|
Top = 1,
|
|
Left = 2,
|
|
Right = 4,
|
|
Bottom = 8,
|
|
TopLeft = Top | Left,
|
|
TopRight = Top | Right,
|
|
BottomLeft = Bottom | Left,
|
|
BottomRight = Bottom | Right,
|
|
}
|
|
|
|
private void UpdateResizeCache()
|
|
{
|
|
int halfThick = RESIZE_THICKNESS / 2;
|
|
int dblThick = RESIZE_THICKNESS * 2;
|
|
|
|
// calculate main outer rect
|
|
// the resize area is both outside and inside the panel,
|
|
// to give a bit of buffer and make it easier to use.
|
|
|
|
// outer rect is the outer-most bounds of our resize area
|
|
Rect outer = new Rect(Panel.rect.x - halfThick,
|
|
Panel.rect.y - halfThick,
|
|
Panel.rect.width + dblThick,
|
|
Panel.rect.height + dblThick);
|
|
m_resizeRect = outer;
|
|
|
|
// calculate the four cross sections to use as flags
|
|
|
|
m_resizeMask[ResizeTypes.Bottom] = new Rect(outer.x, outer.y, outer.width, RESIZE_THICKNESS);
|
|
|
|
m_resizeMask[ResizeTypes.Left] = new Rect(outer.x, outer.y, RESIZE_THICKNESS, outer.height);
|
|
|
|
m_resizeMask[ResizeTypes.Top] = new Rect(outer.x, outer.y + Panel.rect.height, outer.width, RESIZE_THICKNESS);
|
|
|
|
m_resizeMask[ResizeTypes.Right] = new Rect(outer.x + Panel.rect.width, outer.y, RESIZE_THICKNESS, outer.height);
|
|
}
|
|
|
|
private bool MouseInResizeArea(Vector2 mousePos)
|
|
{
|
|
return m_resizeRect.Contains(mousePos);
|
|
}
|
|
|
|
private ResizeTypes GetResizeType(Vector2 mousePos)
|
|
{
|
|
// Calculate which part of the resize area we're in, if any.
|
|
// We do this via a bitmask with the ResizeTypes enum.
|
|
// We can return Top/Right/Bottom/Left, or a corner like TopLeft.
|
|
|
|
int mask = 0;
|
|
|
|
if (m_resizeMask[ResizeTypes.Top].Contains(mousePos))
|
|
{
|
|
mask |= (int)ResizeTypes.Top;
|
|
}
|
|
else if (m_resizeMask[ResizeTypes.Bottom].Contains(mousePos))
|
|
{
|
|
mask |= (int)ResizeTypes.Bottom;
|
|
}
|
|
|
|
if (m_resizeMask[ResizeTypes.Left].Contains(mousePos))
|
|
{
|
|
mask |= (int)ResizeTypes.Left;
|
|
}
|
|
else if (m_resizeMask[ResizeTypes.Right].Contains(mousePos))
|
|
{
|
|
mask |= (int)ResizeTypes.Right;
|
|
}
|
|
|
|
return (ResizeTypes)mask;
|
|
}
|
|
|
|
public void OnHoverResize(ResizeTypes resizeType)
|
|
{
|
|
if (WasHoveringResize && m_lastResizeHoverType == resizeType)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (!s_loadedCursorImage)
|
|
LoadCursorImage();
|
|
|
|
// we are entering resize, or the resize type has changed.
|
|
|
|
WasHoveringResize = true;
|
|
m_lastResizeHoverType = resizeType;
|
|
|
|
m_resizeCursorImage.SetActive(true);
|
|
|
|
// set the rotation for the resize icon
|
|
float iconRotation = 0f;
|
|
switch (resizeType)
|
|
{
|
|
case ResizeTypes.TopRight:
|
|
case ResizeTypes.BottomLeft:
|
|
iconRotation = 45f; break;
|
|
case ResizeTypes.Top:
|
|
case ResizeTypes.Bottom:
|
|
iconRotation = 90f; break;
|
|
case ResizeTypes.TopLeft:
|
|
case ResizeTypes.BottomRight:
|
|
iconRotation = 135f; break;
|
|
}
|
|
|
|
Quaternion rot = m_resizeCursorImage.transform.rotation;
|
|
rot.eulerAngles = new Vector3(0, 0, iconRotation);
|
|
m_resizeCursorImage.transform.rotation = rot;
|
|
|
|
UpdateHoverImagePos();
|
|
}
|
|
|
|
// update the resize icon position to be above the mouse
|
|
private void UpdateHoverImagePos()
|
|
{
|
|
RectTransform t = UIManager.CanvasRoot.GetComponent<RectTransform>();
|
|
m_resizeCursorImage.transform.localPosition = t.InverseTransformPoint(InputManager.MousePosition);
|
|
}
|
|
|
|
public void OnHoverResizeEnd()
|
|
{
|
|
WasHoveringResize = false;
|
|
m_resizeCursorImage.SetActive(false);
|
|
}
|
|
|
|
public void OnBeginResize(ResizeTypes resizeType)
|
|
{
|
|
m_currentResizeType = resizeType;
|
|
m_lastResizePos = InputManager.MousePosition;
|
|
WasResizing = true;
|
|
}
|
|
|
|
public void OnResize()
|
|
{
|
|
Vector3 mousePos = InputManager.MousePosition;
|
|
Vector2 diff = m_lastResizePos - (Vector2)mousePos;
|
|
|
|
if ((Vector2)mousePos == m_lastResizePos)
|
|
return;
|
|
|
|
m_lastResizePos = mousePos;
|
|
|
|
float diffX = (float)((decimal)diff.x / Screen.width);
|
|
float diffY = (float)((decimal)diff.y / Screen.height);
|
|
|
|
Vector2 anchorMin = Panel.anchorMin;
|
|
Vector2 anchorMax = Panel.anchorMax;
|
|
|
|
if (m_currentResizeType.HasFlag(ResizeTypes.Left))
|
|
anchorMin.x -= diffX;
|
|
else if (m_currentResizeType.HasFlag(ResizeTypes.Right))
|
|
anchorMax.x -= diffX;
|
|
|
|
if (m_currentResizeType.HasFlag(ResizeTypes.Top))
|
|
anchorMax.y -= diffY;
|
|
else if (m_currentResizeType.HasFlag(ResizeTypes.Bottom))
|
|
anchorMin.y -= diffY;
|
|
|
|
var newWidth = (anchorMax.x - anchorMin.x) * Screen.width;
|
|
var newHeight = (anchorMax.y - anchorMin.y) * Screen.height;
|
|
|
|
if (newWidth >= minResize.x)
|
|
{
|
|
Panel.anchorMin = new Vector2(anchorMin.x, Panel.anchorMin.y);
|
|
Panel.anchorMax = new Vector2(anchorMax.x, Panel.anchorMax.y);
|
|
}
|
|
if (newHeight >= minResize.y)
|
|
{
|
|
Panel.anchorMin = new Vector2(Panel.anchorMin.x, anchorMin.y);
|
|
Panel.anchorMax = new Vector2(Panel.anchorMax.x, anchorMax.y);
|
|
}
|
|
}
|
|
|
|
public void OnEndResize()
|
|
{
|
|
WasResizing = false;
|
|
UpdateResizeCache();
|
|
}
|
|
|
|
private void LoadCursorImage()
|
|
{
|
|
try
|
|
{
|
|
//string path = ExplorerCore.EXPLORER_FOLDER + @"\cursor.png";
|
|
//byte[] data = File.ReadAllBytes(path);
|
|
|
|
//Texture2D tex = new Texture2D(32, 32);
|
|
//tex.LoadImage(data, false);
|
|
//UnityEngine.Object.DontDestroyOnLoad(tex);
|
|
|
|
//Sprite sprite = UIManager.CreateSprite(tex, new Rect(0, 0, 32, 32));
|
|
//UnityEngine.Object.DontDestroyOnLoad(sprite);
|
|
|
|
var sprite = UIManager.ResizeCursor;
|
|
|
|
m_resizeCursorImage = new GameObject("ResizeCursorImage");
|
|
m_resizeCursorImage.transform.SetParent(UIManager.CanvasRoot.transform);
|
|
|
|
Image image = m_resizeCursorImage.AddGraphic<Image>();
|
|
image.sprite = sprite;
|
|
RectTransform rect = image.transform.GetComponent<RectTransform>();
|
|
rect.SetSizeWithCurrentAnchors(RectTransform.Axis.Horizontal, 32);
|
|
rect.SetSizeWithCurrentAnchors(RectTransform.Axis.Vertical, 32);
|
|
|
|
//m_resizeCursorImage.SetActive(false);
|
|
|
|
s_loadedCursorImage = true;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
ExplorerCore.LogWarning("Exception loading cursor image!\r\n" + e.ToString());
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
} |