您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
402 行
12 KiB
402 行
12 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using Unity.UIWidgets.async;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.service;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
using UnityEngine;
|
|
|
|
namespace Unity.UIWidgets.editor {
|
|
|
|
#if UNITY_EDITOR
|
|
|
|
using UnityEditor;
|
|
public abstract class UIWidgetsEditorWindow : EditorWindow {
|
|
WindowAdapter _windowAdapter;
|
|
public UIWidgetsEditorWindow()
|
|
{
|
|
wantsMouseMove = true;
|
|
wantsMouseEnterLeaveWindow = true;
|
|
}
|
|
|
|
void OnEnable() {
|
|
if (this._windowAdapter == null) {
|
|
this._windowAdapter = new EditorWindowAdapter(this);
|
|
}
|
|
|
|
this._windowAdapter.OnEnable();
|
|
|
|
var rootRenderBox = this.rootRenderBox();
|
|
if (rootRenderBox != null) {
|
|
this._windowAdapter.attachRootRenderBox(rootRenderBox);
|
|
return;
|
|
}
|
|
|
|
this._windowAdapter.attachRootWidget(this.rootWidget());
|
|
}
|
|
|
|
void OnDisable() {
|
|
this._windowAdapter.OnDisable();
|
|
}
|
|
|
|
void OnGUI() {
|
|
this._windowAdapter.OnGUI();
|
|
}
|
|
|
|
void Update() {
|
|
this._windowAdapter.Update();
|
|
}
|
|
|
|
protected virtual RenderBox rootRenderBox() {
|
|
return null;
|
|
}
|
|
|
|
protected abstract Widget rootWidget();
|
|
}
|
|
|
|
public class EditorWindowAdapter : WindowAdapter
|
|
{
|
|
|
|
public readonly EditorWindow editorWindow;
|
|
|
|
public EditorWindowAdapter(EditorWindow editorWindow)
|
|
{
|
|
this.editorWindow = editorWindow;
|
|
}
|
|
|
|
public override void scheduleFrame(bool regenerateLayerTree = true) {
|
|
base.scheduleFrame(regenerateLayerTree);
|
|
editorWindow.Repaint();
|
|
}
|
|
|
|
public override GUIContent titleContent
|
|
{
|
|
get { return editorWindow.titleContent; }
|
|
}
|
|
|
|
protected override double queryDevicePixelRatio()
|
|
{
|
|
return EditorGUIUtility.pixelsPerPoint;
|
|
}
|
|
|
|
protected override Vector2 queryWindowSize()
|
|
{
|
|
return editorWindow.position.size;
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
public abstract class WindowAdapter : Window {
|
|
|
|
static readonly List<WindowAdapter> _windowAdapters = new List<WindowAdapter>();
|
|
public static IEnumerable<WindowAdapter> windowAdapters {
|
|
get { return _windowAdapters; }
|
|
}
|
|
|
|
public WidgetInspectorService widgetInspectorService
|
|
{
|
|
get {
|
|
D.assert(this._binding != null);
|
|
return this._binding.widgetInspectorService;
|
|
}
|
|
}
|
|
|
|
WidgetsBinding _binding;
|
|
float _lastWindowWidth;
|
|
float _lastWindowHeight;
|
|
|
|
readonly TimeSpan _epoch = new TimeSpan(Stopwatch.GetTimestamp());
|
|
readonly MicrotaskQueue _microtaskQueue = new MicrotaskQueue();
|
|
readonly TimerProvider _timerProvider = new TimerProvider();
|
|
readonly WindowCallbacks _windowCallbacks = new WindowCallbacks();
|
|
readonly TextInput _textInput = new TextInput();
|
|
readonly Rasterizer _rasterizer = new Rasterizer();
|
|
|
|
bool _regenerateLayerTree;
|
|
Surface _surface;
|
|
|
|
bool _alive;
|
|
|
|
public bool alive
|
|
{
|
|
get { return this._alive; }
|
|
}
|
|
|
|
public void OnEnable() {
|
|
this._devicePixelRatio = queryDevicePixelRatio();
|
|
var size = queryWindowSize();
|
|
this._lastWindowWidth = size.x;
|
|
this._lastWindowHeight =size.y;
|
|
this._physicalSize = new Size(
|
|
this._lastWindowWidth * this._devicePixelRatio,
|
|
this._lastWindowHeight * this._devicePixelRatio);
|
|
|
|
D.assert(this._surface == null);
|
|
this._surface = createSurface();
|
|
|
|
this._rasterizer.setup(this._surface);
|
|
_windowAdapters.Add(this);
|
|
this._alive = true;
|
|
}
|
|
|
|
public void OnDisable() {
|
|
_windowAdapters.Remove(this);
|
|
this._alive = false;
|
|
|
|
this._rasterizer.teardown();
|
|
|
|
D.assert(this._surface != null);
|
|
this._surface.Dispose();
|
|
this._surface = null;
|
|
}
|
|
|
|
public override IDisposable getScope() {
|
|
instance = this;
|
|
if (this._binding == null) {
|
|
this._binding = new WidgetsBinding();
|
|
}
|
|
|
|
WidgetsBinding.instance = this._binding;
|
|
|
|
return new WindowDisposable(this);
|
|
}
|
|
|
|
class WindowDisposable : IDisposable {
|
|
readonly WindowAdapter _window;
|
|
|
|
public WindowDisposable(WindowAdapter window) {
|
|
this._window = window;
|
|
}
|
|
|
|
public void Dispose() {
|
|
D.assert(instance == this._window);
|
|
instance = null;
|
|
|
|
D.assert(WidgetsBinding.instance == this._window._binding);
|
|
WidgetsBinding.instance = null;
|
|
}
|
|
}
|
|
|
|
public void PostPointerEvent(List<PointerData> data)
|
|
{
|
|
WithBinding(() =>
|
|
{
|
|
this.onPointerEvent(new PointerDataPacket(data));
|
|
});
|
|
}
|
|
|
|
public void PostPointerEvent(PointerData data)
|
|
{
|
|
PostPointerEvent(new List<PointerData>(){data});
|
|
}
|
|
|
|
public void WithBinding(Action fn) {
|
|
using (this.getScope()) {
|
|
fn();
|
|
}
|
|
}
|
|
|
|
public T WithBindingFunc<T>(Func<T> fn) {
|
|
using (this.getScope()) {
|
|
return fn();
|
|
}
|
|
}
|
|
|
|
public virtual void OnGUI() {
|
|
using (this.getScope()) {
|
|
bool dirty = false;
|
|
|
|
if (this._devicePixelRatio != queryDevicePixelRatio()) {
|
|
dirty = true;
|
|
}
|
|
|
|
var size = queryWindowSize();
|
|
if (this._lastWindowWidth != size.x
|
|
|| this._lastWindowHeight != size.y) {
|
|
dirty = true;
|
|
}
|
|
|
|
if (dirty) {
|
|
this._devicePixelRatio = queryDevicePixelRatio();
|
|
this._lastWindowWidth = size.x;
|
|
this._lastWindowHeight = size.y;
|
|
this._physicalSize = new Size(
|
|
this._lastWindowWidth * this._devicePixelRatio,
|
|
this._lastWindowHeight * this._devicePixelRatio);
|
|
|
|
if (this.onMetricsChanged != null) {
|
|
this.onMetricsChanged();
|
|
}
|
|
}
|
|
|
|
this._doOnGUI();
|
|
}
|
|
}
|
|
|
|
public virtual GUIContent titleContent
|
|
{
|
|
get { return null; }
|
|
}
|
|
|
|
protected abstract double queryDevicePixelRatio();
|
|
protected abstract Vector2 queryWindowSize();
|
|
|
|
protected virtual Surface createSurface()
|
|
{
|
|
return new EditorWindowSurface();
|
|
}
|
|
|
|
|
|
void _beginFrame() {
|
|
if (this.onBeginFrame != null) {
|
|
this.onBeginFrame(new TimeSpan(Stopwatch.GetTimestamp()) - this._epoch);
|
|
}
|
|
|
|
this.flushMicrotasks();
|
|
|
|
if (this.onDrawFrame != null) {
|
|
this.onDrawFrame();
|
|
}
|
|
}
|
|
|
|
void _doOnGUI() {
|
|
var evt = Event.current;
|
|
|
|
if (evt.type == EventType.Repaint) {
|
|
if (this._regenerateLayerTree) {
|
|
this._regenerateLayerTree = false;
|
|
this._beginFrame();
|
|
} else {
|
|
this._rasterizer.drawLastLayerTree();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (this.onPointerEvent != null) {
|
|
PointerData pointerData = null;
|
|
|
|
if (evt.type == EventType.MouseDown) {
|
|
pointerData = new PointerData(
|
|
timeStamp: Timer.timespanSinceStartup,
|
|
change: PointerChange.down,
|
|
kind: PointerDeviceKind.mouse,
|
|
device: evt.button,
|
|
physicalX: evt.mousePosition.x * this._devicePixelRatio,
|
|
physicalY: evt.mousePosition.y * this._devicePixelRatio
|
|
);
|
|
} else if (evt.type == EventType.MouseUp || evt.rawType == EventType.MouseUp) {
|
|
pointerData = new PointerData(
|
|
timeStamp: Timer.timespanSinceStartup,
|
|
change: PointerChange.up,
|
|
kind: PointerDeviceKind.mouse,
|
|
device: evt.button,
|
|
physicalX: evt.mousePosition.x * this._devicePixelRatio,
|
|
physicalY: evt.mousePosition.y * this._devicePixelRatio
|
|
);
|
|
} else if (evt.type == EventType.MouseDrag) {
|
|
pointerData = new PointerData(
|
|
timeStamp: Timer.timespanSinceStartup,
|
|
change: PointerChange.move,
|
|
kind: PointerDeviceKind.mouse,
|
|
device: evt.button,
|
|
physicalX: evt.mousePosition.x * this._devicePixelRatio,
|
|
physicalY: evt.mousePosition.y * this._devicePixelRatio
|
|
);
|
|
} else if (evt.type == EventType.MouseMove) {
|
|
pointerData = new PointerData(
|
|
timeStamp: Timer.timespanSinceStartup,
|
|
change: PointerChange.hover,
|
|
kind: PointerDeviceKind.mouse,
|
|
device: evt.button,
|
|
physicalX: evt.mousePosition.x * this._devicePixelRatio,
|
|
physicalY: evt.mousePosition.y * this._devicePixelRatio
|
|
);
|
|
}
|
|
|
|
if (pointerData != null) {
|
|
this.onPointerEvent(new PointerDataPacket(new List<PointerData> {
|
|
pointerData
|
|
}));
|
|
}
|
|
}
|
|
|
|
if (_textInput != null) {
|
|
_textInput.OnGUI();
|
|
}
|
|
}
|
|
|
|
public void Update() {
|
|
Timer.update();
|
|
|
|
using (this.getScope()) {
|
|
this._doUpdate();
|
|
}
|
|
}
|
|
|
|
void _doUpdate() {
|
|
this.flushMicrotasks();
|
|
this._windowCallbacks.update();
|
|
this._timerProvider.update();
|
|
}
|
|
|
|
public override void scheduleFrame(bool regenerateLayerTree = true) {
|
|
if (regenerateLayerTree) {
|
|
this._regenerateLayerTree = true;
|
|
}
|
|
}
|
|
|
|
public override void render(Scene scene) {
|
|
var layerTree = scene.takeLayerTree();
|
|
if (layerTree == null) {
|
|
return;
|
|
}
|
|
|
|
if (this._physicalSize.isEmpty) {
|
|
return;
|
|
}
|
|
|
|
layerTree.frameSize = this._physicalSize;
|
|
layerTree.devicePixelRatio = this._devicePixelRatio;
|
|
this._rasterizer.draw(layerTree);
|
|
}
|
|
|
|
public override void scheduleMicrotask(Action callback) {
|
|
this._microtaskQueue.scheduleMicrotask(callback);
|
|
}
|
|
|
|
public override void flushMicrotasks() {
|
|
this._microtaskQueue.flushMicrotasks();
|
|
}
|
|
|
|
public override Timer run(TimeSpan duration, Action callback, bool periodic = false) {
|
|
return periodic
|
|
? this._timerProvider.periodic(duration, callback)
|
|
: this._timerProvider.run(duration, callback);
|
|
}
|
|
|
|
public override IDisposable onUpdate(VoidCallback callback) {
|
|
return this._windowCallbacks.onUpdate(callback);
|
|
}
|
|
|
|
public void attachRootRenderBox(RenderBox root) {
|
|
using (this.getScope()) {
|
|
this._binding.renderView.child = root;
|
|
}
|
|
}
|
|
|
|
public void attachRootWidget(Widget root) {
|
|
using (this.getScope()) {
|
|
this._binding.attachRootWidget(root);
|
|
}
|
|
}
|
|
|
|
public override TextInput textInput {
|
|
get { return this._textInput; }
|
|
}
|
|
}
|
|
}
|