您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
405 行
14 KiB
405 行
14 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using Unity.UIWidgets.async;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.scheduler;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
|
|
namespace Unity.UIWidgets.widgets {
|
|
public interface WidgetsBindingObserver {
|
|
void didChangeMetrics();
|
|
|
|
void didChangeTextScaleFactor();
|
|
|
|
void didChangePlatformBrightness();
|
|
|
|
void didChangeLocales(List<Locale> locale);
|
|
|
|
Future<bool> didPopRoute();
|
|
|
|
Future<bool> didPushRoute(string route);
|
|
|
|
void didChangeAccessibilityFeatures();
|
|
}
|
|
|
|
public static partial class ui_ {
|
|
public static void Each<T>(this IEnumerable<T> source, Action<T> fn) {
|
|
foreach (var item in source) {
|
|
fn.Invoke(item);
|
|
}
|
|
}
|
|
|
|
public static void Each<T>(this IEnumerable<T> source, Action<T, int> fn) {
|
|
int index = 0;
|
|
|
|
foreach (T item in source) {
|
|
fn.Invoke(item, index);
|
|
index++;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Convert a variable length argument list of items to an enumerable.
|
|
/// </summary>
|
|
public static IEnumerable<T> FromItems<T>(params T[] items) {
|
|
foreach (var item in items) {
|
|
yield return item;
|
|
}
|
|
}
|
|
|
|
public static void runApp(Widget app) {
|
|
var instance = UiWidgetsBinding.ensureInitialized();
|
|
instance.scheduleAttachRootWidget(app);
|
|
instance.scheduleWarmUpFrame();
|
|
}
|
|
}
|
|
|
|
public class WidgetsBinding : RendererBinding {
|
|
public new static WidgetsBinding instance {
|
|
get { return (WidgetsBinding) RendererBinding.instance; }
|
|
set { RendererBinding.instance = value; }
|
|
}
|
|
|
|
protected override void initInstances() {
|
|
base.initInstances();
|
|
instance = this;
|
|
|
|
D.assert(() => {
|
|
// _debugAddStackFilters();
|
|
return true;
|
|
});
|
|
|
|
_buildOwner = new BuildOwner();
|
|
buildOwner.onBuildScheduled = _handleBuildScheduled;
|
|
window.onLocaleChanged += handleLocaleChanged;
|
|
widgetInspectorService = new WidgetInspectorService();
|
|
|
|
// window.onAccessibilityFeaturesChanged = handleAccessibilityFeaturesChanged;
|
|
// SystemChannels.navigation.setMethodCallHandler(_handleNavigationInvocation);
|
|
// FlutterErrorDetails.propertiesTransformers.add(transformDebugCreator);
|
|
}
|
|
|
|
public BuildOwner buildOwner {
|
|
get { return _buildOwner; }
|
|
}
|
|
|
|
BuildOwner _buildOwner;
|
|
|
|
public FocusManager focusManager {
|
|
get { return _buildOwner.focusManager; }
|
|
}
|
|
|
|
readonly List<WidgetsBindingObserver> _observers = new List<WidgetsBindingObserver>();
|
|
|
|
public void addObserver(WidgetsBindingObserver observer) {
|
|
_observers.Add(observer);
|
|
}
|
|
|
|
public bool removeObserver(WidgetsBindingObserver observer) {
|
|
return _observers.Remove(observer);
|
|
}
|
|
|
|
public void handlePopRoute() {
|
|
var idx = -1;
|
|
|
|
void _handlePopRouteSub(bool result) {
|
|
if (!result) {
|
|
idx++;
|
|
if (idx >= _observers.Count) {
|
|
Application.Quit();
|
|
return;
|
|
}
|
|
|
|
_observers[idx].didPopRoute().then_(_handlePopRouteSub);
|
|
}
|
|
}
|
|
|
|
_handlePopRouteSub(false);
|
|
}
|
|
|
|
public WidgetInspectorService widgetInspectorService;
|
|
|
|
protected override void handleMetricsChanged() {
|
|
base.handleMetricsChanged();
|
|
foreach (WidgetsBindingObserver observer in _observers) {
|
|
observer.didChangeMetrics();
|
|
}
|
|
}
|
|
|
|
protected override void handleTextScaleFactorChanged() {
|
|
base.handleTextScaleFactorChanged();
|
|
foreach (WidgetsBindingObserver observer in _observers) {
|
|
observer.didChangeTextScaleFactor();
|
|
}
|
|
}
|
|
|
|
protected override void handlePlatformBrightnessChanged() {
|
|
base.handlePlatformBrightnessChanged();
|
|
foreach (WidgetsBindingObserver observer in _observers) {
|
|
observer.didChangePlatformBrightness();
|
|
}
|
|
}
|
|
|
|
protected virtual void handleLocaleChanged() {
|
|
dispatchLocalesChanged(Window.instance.locales);
|
|
}
|
|
|
|
protected virtual void dispatchLocalesChanged(List<Locale> locales) {
|
|
foreach (WidgetsBindingObserver observer in _observers) {
|
|
observer.didChangeLocales(locales);
|
|
}
|
|
}
|
|
|
|
void _handleBuildScheduled() {
|
|
D.assert(() => {
|
|
if (debugBuildingDirtyElements) {
|
|
throw new UIWidgetsError(new List<DiagnosticsNode>{
|
|
new ErrorSummary("Build scheduled during frame."),
|
|
new ErrorDescription(
|
|
"While the widget tree was being built, laid out, and painted, " +
|
|
"a new frame was scheduled to rebuild the widget tree."
|
|
),
|
|
new ErrorHint(
|
|
"This might be because setState() was called from a layout or " +
|
|
"paint callback. " +
|
|
"If a change is needed to the widget tree, it should be applied " +
|
|
"as the tree is being built. Scheduling a change for the subsequent " +
|
|
"frame instead results in an interface that lags behind by one frame. " +
|
|
"If this was done to make your build dependent on a size measured at " +
|
|
"layout time, consider using a LayoutBuilder, CustomSingleChildLayout, " +
|
|
"or CustomMultiChildLayout. If, on the other hand, the one frame delay " +
|
|
"is the desired effect, for example because this is an " +
|
|
"animation, consider scheduling the frame in a post-frame callback " +
|
|
"using SchedulerBinding.addPostFrameCallback or " +
|
|
"using an AnimationController to trigger the animation."
|
|
)
|
|
});
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
ensureVisualUpdate();
|
|
}
|
|
|
|
protected bool debugBuildingDirtyElements = false;
|
|
|
|
protected override void drawFrame() {
|
|
D.assert(!debugBuildingDirtyElements);
|
|
D.assert(() => {
|
|
debugBuildingDirtyElements = true;
|
|
return true;
|
|
});
|
|
try {
|
|
if (renderViewElement != null) {
|
|
buildOwner.buildScope(renderViewElement);
|
|
}
|
|
|
|
base.drawFrame();
|
|
buildOwner.finalizeTree();
|
|
}
|
|
finally {
|
|
D.assert(() => {
|
|
debugBuildingDirtyElements = false;
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
|
|
public RenderObjectToWidgetElement<RenderBox> renderViewElement {
|
|
get { return _renderViewElement; }
|
|
}
|
|
|
|
RenderObjectToWidgetElement<RenderBox> _renderViewElement;
|
|
|
|
public void detachRootWidget() {
|
|
if (_renderViewElement == null) {
|
|
return;
|
|
}
|
|
|
|
//The former widget tree must be layout first before its destruction
|
|
drawFrame();
|
|
attachRootWidget(null);
|
|
buildOwner.buildScope(_renderViewElement);
|
|
buildOwner.finalizeTree();
|
|
|
|
pipelineOwner.rootNode = null;
|
|
_renderViewElement.deactivate();
|
|
_renderViewElement.unmount();
|
|
_renderViewElement = null;
|
|
}
|
|
|
|
public void scheduleAttachRootWidget(Widget rootWidget) {
|
|
Timer.run(() => {
|
|
attachRootWidget(rootWidget);
|
|
return null;
|
|
});
|
|
}
|
|
|
|
public void attachRootWidget(Widget rootWidget) {
|
|
_renderViewElement = new RenderObjectToWidgetAdapter<RenderBox>(
|
|
container: renderView,
|
|
debugShortDescription: "[root]",
|
|
child: rootWidget
|
|
).attachToRenderTree(buildOwner, _renderViewElement);
|
|
}
|
|
|
|
bool isRootWidgetAttached {
|
|
get { return _renderViewElement != null; }
|
|
}
|
|
}
|
|
|
|
public class RenderObjectToWidgetAdapter<T> : RenderObjectWidget where T : RenderObject {
|
|
public RenderObjectToWidgetAdapter(
|
|
Widget child = null,
|
|
RenderObjectWithChildMixin<T> container = null,
|
|
string debugShortDescription = null
|
|
) : base(
|
|
new GlobalObjectKey<State>(container)) {
|
|
this.child = child;
|
|
this.container = container;
|
|
this.debugShortDescription = debugShortDescription;
|
|
}
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly RenderObjectWithChildMixin<T> container;
|
|
|
|
public readonly string debugShortDescription;
|
|
|
|
public override Element createElement() {
|
|
return new RenderObjectToWidgetElement<T>(this);
|
|
}
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return (RenderObject) container;
|
|
}
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject renderObject) {
|
|
}
|
|
|
|
public RenderObjectToWidgetElement<T> attachToRenderTree(BuildOwner owner,
|
|
RenderObjectToWidgetElement<T> element) {
|
|
if (element == null) {
|
|
owner.lockState(() => {
|
|
element = (RenderObjectToWidgetElement<T>) createElement();
|
|
D.assert(element != null);
|
|
element.assignOwner(owner);
|
|
});
|
|
owner.buildScope(element, () => { element.mount(null, null); });
|
|
SchedulerBinding.instance.ensureVisualUpdate();
|
|
}
|
|
else {
|
|
element._newWidget = this;
|
|
element.markNeedsBuild();
|
|
}
|
|
|
|
return element;
|
|
}
|
|
|
|
public override string toStringShort() {
|
|
return debugShortDescription ?? base.toStringShort();
|
|
}
|
|
}
|
|
|
|
public class RenderObjectToWidgetElement<T> : RootRenderObjectElement where T : RenderObject {
|
|
public RenderObjectToWidgetElement(RenderObjectToWidgetAdapter<T> widget) : base(widget) {
|
|
}
|
|
|
|
public new RenderObjectToWidgetAdapter<T> widget {
|
|
get { return (RenderObjectToWidgetAdapter<T>) base.widget; }
|
|
}
|
|
|
|
Element _child;
|
|
|
|
static readonly object _rootChildSlot = new object();
|
|
|
|
public override void visitChildren(ElementVisitor visitor) {
|
|
if (_child != null) {
|
|
visitor(_child);
|
|
}
|
|
}
|
|
|
|
public override void forgetChild(Element child) {
|
|
D.assert(child == _child);
|
|
_child = null;
|
|
base.forgetChild(child);
|
|
}
|
|
|
|
public override void mount(Element parent, object newSlot) {
|
|
D.assert(parent == null);
|
|
base.mount(parent, newSlot);
|
|
_rebuild();
|
|
}
|
|
|
|
public override void update(Widget newWidget) {
|
|
base.update(newWidget);
|
|
D.assert(widget == newWidget);
|
|
_rebuild();
|
|
}
|
|
|
|
internal Widget _newWidget;
|
|
|
|
protected override void performRebuild() {
|
|
if (_newWidget != null) {
|
|
Widget newWidget = _newWidget;
|
|
_newWidget = null;
|
|
update((RenderObjectToWidgetAdapter<T>)newWidget);
|
|
}
|
|
|
|
base.performRebuild();
|
|
D.assert(_newWidget == null);
|
|
}
|
|
|
|
void _rebuild() {
|
|
try {
|
|
_child = updateChild(_child, widget.child,
|
|
_rootChildSlot);
|
|
// allow
|
|
}
|
|
catch (Exception ex) {
|
|
var details = new UIWidgetsErrorDetails(
|
|
exception: ex,
|
|
library: "widgets library",
|
|
context: new ErrorDescription("attaching to the render tree")
|
|
);
|
|
UIWidgetsError.reportError(details);
|
|
|
|
Widget error = ErrorWidget.builder(details);
|
|
_child = updateChild(null, error, _rootChildSlot);
|
|
}
|
|
}
|
|
|
|
public new RenderObjectWithChildMixin<T> renderObject {
|
|
get { return (RenderObjectWithChildMixin<T>) base.renderObject; }
|
|
}
|
|
|
|
protected override void insertChildRenderObject(RenderObject child, object slot) {
|
|
D.assert(slot == _rootChildSlot);
|
|
D.assert(renderObject.debugValidateChild(child));
|
|
renderObject.child = (T) child;
|
|
}
|
|
|
|
protected override void moveChildRenderObject(RenderObject child, object slot) {
|
|
D.assert(false);
|
|
}
|
|
|
|
protected override void removeChildRenderObject(RenderObject child) {
|
|
D.assert(renderObject.child == child);
|
|
renderObject.child = null;
|
|
}
|
|
}
|
|
|
|
public class UiWidgetsBinding : WidgetsBinding {
|
|
// todo
|
|
public static WidgetsBinding ensureInitialized() {
|
|
if (WidgetsBinding.instance == null) {
|
|
return new UiWidgetsBinding();
|
|
}
|
|
|
|
return WidgetsBinding.instance;
|
|
}
|
|
}
|
|
}
|