您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
155 行
4.3 KiB
155 行
4.3 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Runtime.CompilerServices;
|
|
|
|
namespace UIWidgets.foundation {
|
|
{% macro AbstractNodeMixin(with) %}
|
|
{% set className = 'AbstractNode' if with == '' else 'AbstractNodeMixin' + with %}
|
|
|
|
{% set extends = '' if with == '' else ' : ' + with %}
|
|
|
|
public class {{className}} {{extends}} {
|
|
public int depth {
|
|
get { return this._depth; }
|
|
}
|
|
|
|
int _depth = 0;
|
|
|
|
protected void redepthChild({{className}} child) {
|
|
D.assert(child.owner == this.owner);
|
|
if (child._depth <= this._depth) {
|
|
child._depth = this._depth + 1;
|
|
child.redepthChildren();
|
|
}
|
|
}
|
|
|
|
public virtual void redepthChildren() {
|
|
}
|
|
|
|
public object owner {
|
|
get { return this._owner; }
|
|
}
|
|
|
|
object _owner;
|
|
|
|
public bool attached {
|
|
get { return this._owner != null; }
|
|
}
|
|
|
|
public virtual void attach(object owner) {
|
|
D.assert(owner != null);
|
|
D.assert(this._owner == null);
|
|
this._owner = owner;
|
|
}
|
|
|
|
public virtual void detach() {
|
|
D.assert(this._owner != null);
|
|
this._owner = null;
|
|
}
|
|
|
|
public {{className}} parent {
|
|
get { return this._parent; }
|
|
}
|
|
|
|
{{className}} _parent;
|
|
|
|
protected virtual void adoptChild({{className}} child) {
|
|
D.assert(child != null);
|
|
D.assert(child._parent == null);
|
|
D.assert(() => {
|
|
var node = this;
|
|
while (node.parent != null)
|
|
node = node.parent;
|
|
D.assert(node != child); // indicates we are about to create a cycle
|
|
return true;
|
|
});
|
|
|
|
child._parent = this;
|
|
if (this.attached) {
|
|
child.attach(this._owner);
|
|
}
|
|
|
|
this.redepthChild(child);
|
|
}
|
|
|
|
protected virtual void dropChild({{className}} child) {
|
|
D.assert(child != null);
|
|
D.assert(child._parent == this);
|
|
D.assert(child.attached == this.attached);
|
|
|
|
child._parent = null;
|
|
if (this.attached) {
|
|
child.detach();
|
|
}
|
|
}
|
|
}
|
|
{% endmacro %}
|
|
|
|
{{ AbstractNodeMixin('') }}
|
|
|
|
{{ AbstractNodeMixin('DiagnosticableTree') }}
|
|
|
|
|
|
{% macro CanonicalMixin(with) %}
|
|
public abstract class CanonicalMixin{{with}} : {{with}} {
|
|
_DependencyList _dependencyList;
|
|
|
|
_DependencyList _getDependencyList() {
|
|
if (this._dependencyList == null) {
|
|
this._dependencyList = new _DependencyList(this.GetType(), this);
|
|
}
|
|
|
|
return this._dependencyList;
|
|
}
|
|
|
|
CanonicalMixin{{with}} _canonical;
|
|
|
|
CanonicalMixin{{with}} _getCanonical() {
|
|
if (this._canonical != null) {
|
|
return this._canonical;
|
|
}
|
|
|
|
var weakReference = _canonicalObjects.putIfAbsent(this._getDependencyList(), () => {
|
|
return new WeakReference(this);
|
|
});
|
|
if (weakReference.Target == null) {
|
|
weakReference.Target = this;
|
|
}
|
|
|
|
return this._canonical = (CanonicalMixin{{with}}) weakReference.Target;
|
|
}
|
|
|
|
~CanonicalMixin{{with}}() {
|
|
if (this == this._canonical) {
|
|
_canonicalObjects.Remove(this._dependencyList);
|
|
}
|
|
}
|
|
|
|
static readonly Dictionary<_DependencyList, WeakReference> _canonicalObjects =
|
|
new Dictionary<_DependencyList, WeakReference>();
|
|
|
|
public override bool Equals(object obj) {
|
|
if (object.ReferenceEquals(null, obj)) {
|
|
return false;
|
|
}
|
|
|
|
if (object.ReferenceEquals(this, obj)) {
|
|
return true;
|
|
}
|
|
|
|
if (obj.GetType() != this.GetType()) {
|
|
return false;
|
|
}
|
|
|
|
return object.ReferenceEquals(this._getCanonical(), ((CanonicalMixin{{with}}) obj)._getCanonical());
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
return RuntimeHelpers.GetHashCode(this._getCanonical());
|
|
}
|
|
}
|
|
{% endmacro %}
|
|
|
|
{{ CanonicalMixin('DiagnosticableTree') }}
|
|
|
|
}
|