您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
258 行
9.0 KiB
258 行
9.0 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using UIWidgets.foundation;
|
|
using UIWidgets.rendering;
|
|
using UIWidgets.ui;
|
|
|
|
namespace UIWidgets.widgets {
|
|
public interface WidgetsBindingObserver {
|
|
void didChangeMetrics();
|
|
}
|
|
|
|
public class WidgetsBinding : RendererBinding {
|
|
public static new WidgetsBinding instance {
|
|
get { return (WidgetsBinding) RendererBinding.instance; }
|
|
set { RendererBinding.instance = value; }
|
|
}
|
|
|
|
public WidgetsBinding() {
|
|
this.buildOwner.onBuildScheduled = this._handleBuildScheduled;
|
|
Window.instance.onLocaleChanged += this.handleLocaleChanged;
|
|
}
|
|
|
|
public BuildOwner buildOwner {
|
|
get { return this._buildOwner; }
|
|
}
|
|
|
|
readonly BuildOwner _buildOwner = new BuildOwner();
|
|
|
|
public FocusManager focusManager {
|
|
get { return this._buildOwner.focusManager; }
|
|
}
|
|
|
|
readonly List<WidgetsBindingObserver> _observers = new List<WidgetsBindingObserver>();
|
|
|
|
public void addObserver(WidgetsBindingObserver observer) {
|
|
this._observers.Add(observer);
|
|
}
|
|
|
|
public bool removeObserver(WidgetsBindingObserver observer) {
|
|
return this._observers.Remove(observer);
|
|
}
|
|
|
|
protected override void handleMetricsChanged() {
|
|
base.handleMetricsChanged();
|
|
foreach (WidgetsBindingObserver observer in this._observers) {
|
|
observer.didChangeMetrics();
|
|
}
|
|
}
|
|
|
|
void handleLocaleChanged() {
|
|
// todo
|
|
// dispatchLocaleChanged(window.locale);
|
|
}
|
|
|
|
void _handleBuildScheduled() {
|
|
D.assert(() => {
|
|
if (this.debugBuildingDirtyElements) {
|
|
throw new UIWidgetsError(
|
|
"Build scheduled during frame.\n" +
|
|
"While the widget tree was being built, laid out, and painted, " +
|
|
"a new frame was scheduled to rebuild the widget tree. " +
|
|
"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;
|
|
});
|
|
|
|
this.ensureVisualUpdate();
|
|
}
|
|
|
|
protected bool debugBuildingDirtyElements = false;
|
|
|
|
protected override void drawFrame() {
|
|
D.assert(!this.debugBuildingDirtyElements);
|
|
D.assert(() => {
|
|
this.debugBuildingDirtyElements = true;
|
|
return true;
|
|
});
|
|
try {
|
|
if (this.renderViewElement != null) {
|
|
this.buildOwner.buildScope(this.renderViewElement);
|
|
}
|
|
|
|
base.drawFrame();
|
|
this.buildOwner.finalizeTree();
|
|
}
|
|
finally {
|
|
D.assert(() => {
|
|
this.debugBuildingDirtyElements = false;
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
|
|
public RenderObjectToWidgetElement<RenderBox> renderViewElement {
|
|
get { return this._renderViewElement; }
|
|
}
|
|
|
|
RenderObjectToWidgetElement<RenderBox> _renderViewElement;
|
|
|
|
public void attachRootWidget(Widget rootWidget) {
|
|
this._renderViewElement = new RenderObjectToWidgetAdapter<RenderBox>(
|
|
container: this.renderView,
|
|
debugShortDescription: "[root]",
|
|
child: rootWidget
|
|
).attachToRenderTree(this.buildOwner, this._renderViewElement);
|
|
}
|
|
}
|
|
|
|
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) this.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>) this.createElement();
|
|
D.assert(element != null);
|
|
element.assignOwner(owner);
|
|
});
|
|
owner.buildScope(element, () => { element.mount(null, null); });
|
|
} else {
|
|
element._newWidget = this;
|
|
element.markNeedsBuild();
|
|
}
|
|
|
|
return element;
|
|
}
|
|
|
|
public override string toStringShort() {
|
|
return this.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 (this._child != null) {
|
|
visitor(this._child);
|
|
}
|
|
}
|
|
|
|
protected override void forgetChild(Element child) {
|
|
D.assert(child == this._child);
|
|
this._child = null;
|
|
}
|
|
|
|
public override void mount(Element parent, object newSlot) {
|
|
D.assert(parent == null);
|
|
base.mount(parent, newSlot);
|
|
this._rebuild();
|
|
}
|
|
|
|
public override void update(Widget newWidget) {
|
|
base.update(newWidget);
|
|
D.assert(this.widget == newWidget);
|
|
this._rebuild();
|
|
}
|
|
|
|
internal Widget _newWidget;
|
|
|
|
protected override void performRebuild() {
|
|
if (this._newWidget != null) {
|
|
Widget newWidget = this._newWidget;
|
|
this._newWidget = null;
|
|
this.update(newWidget);
|
|
}
|
|
|
|
base.performRebuild();
|
|
D.assert(this._newWidget == null);
|
|
}
|
|
|
|
void _rebuild() {
|
|
try {
|
|
this._child = this.updateChild(this._child, this.widget.child, _rootChildSlot);
|
|
// allow
|
|
}
|
|
catch (Exception ex) {
|
|
var details = new UIWidgetsErrorDetails(
|
|
exception: ex,
|
|
library: "widgets library",
|
|
context: "attaching to the render tree"
|
|
);
|
|
UIWidgetsError.reportError(details);
|
|
|
|
Widget error = ErrorWidget.builder(details);
|
|
this._child = this.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(this.renderObject.debugValidateChild(child));
|
|
this.renderObject.child = (T) child;
|
|
}
|
|
|
|
protected override void moveChildRenderObject(RenderObject child, object slot) {
|
|
D.assert(false);
|
|
}
|
|
|
|
protected override void removeChildRenderObject(RenderObject child) {
|
|
D.assert(this.renderObject.child == child);
|
|
this.renderObject.child = null;
|
|
}
|
|
}
|
|
}
|