您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
1429 行
47 KiB
1429 行
47 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Unity.UIWidgets.async2;
|
|
using Unity.UIWidgets.cupertino;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.gestures;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.service;
|
|
using Unity.UIWidgets.ui;
|
|
using Unity.UIWidgets.widgets;
|
|
|
|
namespace Unity.UIWidgets.widgets {
|
|
public enum FocusHighlightMode {
|
|
touch,
|
|
traditional,
|
|
}
|
|
enum FocusHighlightStrategy {
|
|
automatic,
|
|
alwaysTouch,
|
|
alwaysTraditional,
|
|
}
|
|
|
|
public enum UnfocusDisposition {
|
|
|
|
scope,
|
|
|
|
previouslyFocusedChild,
|
|
}
|
|
public delegate bool FocusOnKeyCallback(FocusNode node, RawKeyEvent Event);
|
|
|
|
public class FocusManagerUtils {
|
|
public static readonly bool _kDebugFocus = false;
|
|
|
|
public static bool _focusDebug(string message, List<string> details = null) {
|
|
if (_kDebugFocus) {
|
|
//UnityEngine.Debug.Log();
|
|
UnityEngine.Debug.Log($"FOCUS: {message}");
|
|
if (details != null && details.Count() != 0) {
|
|
foreach (string detail in details) {
|
|
UnityEngine.Debug.Log($" {detail}");
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public static FocusNode primaryFocus {
|
|
get { return WidgetsBinding.instance.focusManager.primaryFocus; }
|
|
}
|
|
|
|
|
|
/*string debugDescribeFocusTree() {
|
|
D.assert(WidgetsBinding.instance != null);
|
|
string result = null;
|
|
D.assert(() => {
|
|
result = FocusManager.instance.toStringDeep();
|
|
return true;
|
|
}());
|
|
return result ?? "";
|
|
}*/
|
|
|
|
|
|
/*void debugDumpFocusTree() {
|
|
D.assert(() => {
|
|
UnityEngine.Debug.Log(debugDescribeFocusTree());
|
|
return true;
|
|
});
|
|
}*/
|
|
}
|
|
|
|
public class FocusAttachment {
|
|
public FocusAttachment(FocusNode _node) {
|
|
D.assert(_node != null);
|
|
this._node = _node;
|
|
}
|
|
|
|
public readonly FocusNode _node;
|
|
|
|
bool isAttached {
|
|
get {
|
|
return _node._attachment == this;
|
|
}
|
|
}
|
|
|
|
public void detach() {
|
|
D.assert(_node != null);
|
|
D.assert(FocusManagerUtils._focusDebug("Detaching node:", new List<string>{_node.ToString(), $"With enclosing scope {_node.enclosingScope}"}));
|
|
if (isAttached) {
|
|
if (_node.hasPrimaryFocus || (_node._manager != null && _node._manager._markedForFocus == _node)) {
|
|
_node.unfocus(disposition: UnfocusDisposition.previouslyFocusedChild);
|
|
}
|
|
|
|
_node._manager?._markDetached(_node);
|
|
_node._parent?._removeChild(_node);
|
|
_node._attachment = null;
|
|
D.assert(!_node.hasPrimaryFocus);
|
|
D.assert(_node._manager?._markedForFocus != _node);
|
|
}
|
|
D.assert(!isAttached);
|
|
}
|
|
|
|
public void reparent(FocusNode parent = null) {
|
|
D.assert(_node != null);
|
|
if (isAttached) {
|
|
D.assert(_node.context != null);
|
|
parent = parent ?? Focus.of(_node.context, nullOk: true, scopeOk: true);
|
|
parent = parent ?? _node.context.owner.focusManager.rootScope;
|
|
D.assert(parent != null);
|
|
parent._reparent(_node);
|
|
}
|
|
}
|
|
}
|
|
|
|
public class FocusNode : ChangeNotifier {
|
|
//DiagnosticableTreeMixin,
|
|
public FocusNode(
|
|
string debugLabel = "",
|
|
FocusOnKeyCallback onKey = null,
|
|
bool skipTraversal = false,
|
|
bool canRequestFocus = true
|
|
) {
|
|
D.assert(skipTraversal != null);
|
|
D.assert(canRequestFocus != null);
|
|
_skipTraversal = skipTraversal;
|
|
_canRequestFocus = canRequestFocus;
|
|
/*_onKey = onKey {
|
|
this.debugLabel = debugLabel; ///????
|
|
}*/
|
|
}
|
|
|
|
public bool skipTraversal {
|
|
get { return _skipTraversal; }
|
|
set {
|
|
if (value != _skipTraversal) {
|
|
_skipTraversal = value;
|
|
_manager?._markPropertiesChanged(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool _skipTraversal;
|
|
|
|
|
|
public bool canRequestFocus {
|
|
get {
|
|
FocusScopeNode scope = enclosingScope;
|
|
return _canRequestFocus && (scope == null || scope.canRequestFocus);
|
|
}
|
|
set {
|
|
if (value != _canRequestFocus) {
|
|
if (!value) {
|
|
unfocus(disposition: UnfocusDisposition.previouslyFocusedChild);
|
|
}
|
|
|
|
_canRequestFocus = value;
|
|
_manager?._markPropertiesChanged(this);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
bool _canRequestFocus;
|
|
|
|
|
|
public BuildContext context {
|
|
get { return _context; }
|
|
|
|
}
|
|
|
|
BuildContext _context;
|
|
|
|
FocusOnKeyCallback onKey {
|
|
get { return _onKey; }
|
|
|
|
}
|
|
|
|
FocusOnKeyCallback _onKey;
|
|
|
|
public FocusManager _manager;
|
|
List<FocusNode> _ancestors;
|
|
List<FocusNode> _descendants;
|
|
bool _hasKeyboardToken = false;
|
|
|
|
FocusNode parent {
|
|
get { return _parent; }
|
|
|
|
}
|
|
|
|
public FocusNode _parent;
|
|
|
|
|
|
IEnumerable<FocusNode> children {
|
|
get { return _children; }
|
|
|
|
}
|
|
|
|
public readonly List<FocusNode> _children = new List<FocusNode>();
|
|
|
|
IEnumerable<FocusNode> traversalChildren {
|
|
get {
|
|
if (!canRequestFocus) {
|
|
return new List<FocusNode>();
|
|
}
|
|
|
|
List<FocusNode> nodes = new List<FocusNode>();
|
|
foreach (FocusNode node in children) {
|
|
if (!node.skipTraversal && node.canRequestFocus)
|
|
nodes.Add(node);
|
|
}
|
|
|
|
return nodes;
|
|
}
|
|
|
|
}
|
|
|
|
public string debugLabel {
|
|
get { return _debugLabel; }
|
|
set {
|
|
D.assert(() => {
|
|
// Only set the value in debug builds.
|
|
_debugLabel = value;
|
|
return true;
|
|
});
|
|
}
|
|
|
|
}
|
|
|
|
string _debugLabel;
|
|
|
|
|
|
public FocusAttachment _attachment;
|
|
|
|
|
|
public IEnumerable<FocusNode> descendants {
|
|
|
|
get {
|
|
if (_descendants == null) {
|
|
List<FocusNode> result = new List<FocusNode>();
|
|
foreach (FocusNode child in _children) {
|
|
foreach (var childDescendant in child.descendants) {
|
|
result.Add(childDescendant);
|
|
}
|
|
|
|
//result.addAll(child.descendants);
|
|
result.Add(child);
|
|
}
|
|
|
|
_descendants = result;
|
|
}
|
|
|
|
return _descendants;
|
|
}
|
|
|
|
}
|
|
|
|
public IEnumerable<FocusNode> traversalDescendants {
|
|
get {
|
|
List<FocusNode> nodes = new List<FocusNode>();
|
|
foreach (FocusNode node in descendants) {
|
|
if (!node.skipTraversal && node.canRequestFocus) {
|
|
nodes.Add(node);
|
|
}
|
|
|
|
}
|
|
|
|
// descendants.where((FocusNode node) => !node.skipTraversal && node.canRequestFocus);
|
|
return nodes;
|
|
}
|
|
|
|
}
|
|
|
|
public IEnumerable<FocusNode> ancestors {
|
|
get {
|
|
if (_ancestors == null) {
|
|
List<FocusNode> result = new List<FocusNode>();
|
|
FocusNode parent = _parent;
|
|
while (parent != null) {
|
|
result.Add(parent);
|
|
parent = parent._parent;
|
|
}
|
|
|
|
_ancestors = result;
|
|
}
|
|
|
|
return _ancestors;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
public bool hasFocus {
|
|
get { return hasPrimaryFocus || (_manager?.primaryFocus?.ancestors?.Contains(this) ?? false); }
|
|
}
|
|
|
|
public bool hasPrimaryFocus {
|
|
get { return _manager?.primaryFocus == this; }
|
|
}
|
|
|
|
/// Returns the [FocusHighlightMode] that is currently in effect for this node.
|
|
FocusHighlightMode highlightMode {
|
|
get { return FocusManager.instance.highlightMode; }
|
|
}
|
|
|
|
public FocusScopeNode nearestScope {
|
|
get { return enclosingScope; }
|
|
}
|
|
|
|
|
|
public FocusScopeNode enclosingScope {
|
|
get {
|
|
foreach (FocusNode node in ancestors) {
|
|
if (node is FocusScopeNode) {
|
|
return (FocusScopeNode) node;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
Size size {
|
|
get {
|
|
D.assert(
|
|
context != null, () =>
|
|
"Tried to get the size of a focus node that didn't have its context set yet.\n" +
|
|
"The context needs to be set before trying to evaluate traversal policies. This " +
|
|
"is typically done with the attach method."
|
|
);
|
|
return context.findRenderObject().semanticBounds.size;
|
|
}
|
|
}
|
|
|
|
|
|
Offset offset {
|
|
get {
|
|
D.assert(
|
|
context != null, () =>
|
|
"Tried to get the offset of a focus node that didn't have its context set yet.\n" +
|
|
"The context needs to be set before trying to evaluate traversal policies. This " +
|
|
"is typically done with the attach method.");
|
|
RenderObject renderObject = context.findRenderObject();
|
|
return MatrixUtils.transformPoint(renderObject.getTransformTo(null),
|
|
renderObject.semanticBounds.topLeft);
|
|
}
|
|
}
|
|
|
|
Rect rect {
|
|
get {
|
|
D.assert(
|
|
context != null, () =>
|
|
"Tried to get the bounds of a focus node that didn't have its context set yet.\n" +
|
|
"The context needs to be set before trying to evaluate traversal policies. This " +
|
|
"is typically done with the attach method.");
|
|
RenderObject renderObject = context.findRenderObject();
|
|
Offset globalOffset = MatrixUtils.transformPoint(renderObject.getTransformTo(null),
|
|
renderObject.semanticBounds.topLeft);
|
|
return globalOffset & renderObject.semanticBounds.size;
|
|
}
|
|
}
|
|
|
|
public void unfocus(
|
|
UnfocusDisposition disposition = UnfocusDisposition.scope
|
|
) {
|
|
D.assert(disposition != null);
|
|
if (!hasFocus && (_manager == null || _manager._markedForFocus != this)) {
|
|
return;
|
|
}
|
|
|
|
FocusScopeNode scope = enclosingScope;
|
|
if (scope == null) {
|
|
|
|
return;
|
|
}
|
|
|
|
switch (disposition) {
|
|
case UnfocusDisposition.scope:
|
|
if (scope.canRequestFocus) {
|
|
scope._focusedChildren.Clear();
|
|
}
|
|
|
|
while (!scope.canRequestFocus) {
|
|
scope = scope.enclosingScope ?? _manager?.rootScope;
|
|
}
|
|
|
|
scope?._doRequestFocus(findFirstFocus: false);
|
|
break;
|
|
case UnfocusDisposition.previouslyFocusedChild:
|
|
|
|
if (scope.canRequestFocus) {
|
|
scope?._focusedChildren?.Remove(this);
|
|
}
|
|
|
|
while (!scope.canRequestFocus) {
|
|
scope.enclosingScope?._focusedChildren?.Remove(scope);
|
|
scope = scope.enclosingScope ?? _manager?.rootScope;
|
|
}
|
|
|
|
scope?._doRequestFocus(findFirstFocus: true);
|
|
break;
|
|
}
|
|
|
|
D.assert(FocusManagerUtils._focusDebug("Unfocused node:",
|
|
new List<string> {$"primary focus was {this}", $"next focus will be {_manager?._markedForFocus}"}));
|
|
}
|
|
|
|
|
|
public bool consumeKeyboardToken() {
|
|
if (!_hasKeyboardToken) {
|
|
return false;
|
|
}
|
|
|
|
_hasKeyboardToken = false;
|
|
return true;
|
|
}
|
|
|
|
public void _markNextFocus(FocusNode newFocus) {
|
|
if (_manager != null) {
|
|
// If we have a manager, then let it handle the focus change.
|
|
_manager._markNextFocus(this);
|
|
return;
|
|
}
|
|
|
|
newFocus?._setAsFocusedChildForScope();
|
|
newFocus?._notify();
|
|
if (newFocus != this) {
|
|
_notify();
|
|
}
|
|
}
|
|
|
|
|
|
public void _removeChild(FocusNode node, bool removeScopeFocus = true) {
|
|
D.assert(node != null);
|
|
D.assert(_children.Contains(node), () => "Tried to remove a node that wasn't a child.");
|
|
D.assert(node._parent == this);
|
|
D.assert(node._manager == _manager);
|
|
|
|
if (removeScopeFocus) {
|
|
node.enclosingScope?._focusedChildren?.Remove(node);
|
|
}
|
|
|
|
node._parent = null;
|
|
_children.Remove(node);
|
|
foreach (FocusNode ancestor in ancestors) {
|
|
ancestor._descendants = null;
|
|
}
|
|
|
|
_descendants = null;
|
|
D.assert(_manager == null || !_manager.rootScope.descendants.Contains(node));
|
|
}
|
|
|
|
void _updateManager(FocusManager manager) {
|
|
_manager = manager;
|
|
foreach (FocusNode descendant in descendants) {
|
|
descendant._manager = manager;
|
|
descendant._ancestors = null;
|
|
}
|
|
}
|
|
|
|
public void _reparent(FocusNode child) {
|
|
D.assert(child != null);
|
|
D.assert(child != this, () => "Tried to make a child into a parent of itself.");
|
|
if (child._parent == this) {
|
|
D.assert(_children.Contains(child),
|
|
() => "Found a node that says it's a child, but doesn't appear in the child list.");
|
|
// The child is already a child of this parent.
|
|
return;
|
|
}
|
|
|
|
D.assert(_manager == null || child != _manager.rootScope, () => "Reparenting the root node isn't allowed.");
|
|
D.assert(!ancestors.Contains(child),
|
|
() => "The supplied child is already an ancestor of this node. Loops are not allowed.");
|
|
FocusScopeNode oldScope = child.enclosingScope;
|
|
bool hadFocus = child.hasFocus;
|
|
child._parent?._removeChild(child, removeScopeFocus: oldScope != nearestScope);
|
|
_children.Add(child);
|
|
child._parent = this;
|
|
child._ancestors = null;
|
|
child._updateManager(_manager);
|
|
foreach (FocusNode ancestor in child.ancestors) {
|
|
ancestor._descendants = null;
|
|
}
|
|
|
|
if (hadFocus) {
|
|
_manager?.primaryFocus?._setAsFocusedChildForScope();
|
|
}
|
|
|
|
if (oldScope != null && child.context != null && child.enclosingScope != oldScope) {
|
|
UnityEngine.Debug.Log(
|
|
"FocusTraversalGroup.of(child.context, nullOk: true)?.changedScope(node: child, oldScope: oldScope);");
|
|
//FocusTraversalGroup.of(child.context, nullOk: true)?.changedScope(node: child, oldScope: oldScope);
|
|
}
|
|
|
|
if (child._requestFocusWhenReparented) {
|
|
child._doRequestFocus(findFirstFocus: true);
|
|
child._requestFocusWhenReparented = false;
|
|
}
|
|
}
|
|
|
|
public FocusAttachment attach(BuildContext context, FocusOnKeyCallback onKey = null) {
|
|
_context = context;
|
|
_onKey = onKey ?? _onKey;
|
|
_attachment = new FocusAttachment(this);
|
|
return _attachment;
|
|
}
|
|
|
|
|
|
public override void dispose() {
|
|
_attachment?.detach();
|
|
base.dispose();
|
|
}
|
|
|
|
|
|
public void _notify() {
|
|
if (_parent == null) {
|
|
|
|
return;
|
|
}
|
|
|
|
if (hasPrimaryFocus) {
|
|
_setAsFocusedChildForScope();
|
|
}
|
|
|
|
notifyListeners();
|
|
}
|
|
|
|
public void requestFocus(FocusNode node = null) {
|
|
if (node != null) {
|
|
if (node._parent == null) {
|
|
_reparent(node);
|
|
}
|
|
|
|
D.assert(node.ancestors.Contains(this),
|
|
() =>
|
|
"Focus was requested for a node that is not a descendant of the scope from which it was requested.");
|
|
node._doRequestFocus(findFirstFocus: true);
|
|
return;
|
|
}
|
|
|
|
_doRequestFocus(findFirstFocus: true);
|
|
}
|
|
|
|
// Note that this is overridden in FocusScopeNode.
|
|
public virtual void _doRequestFocus(bool findFirstFocus = false) {
|
|
D.assert(findFirstFocus != null);
|
|
if (!canRequestFocus) {
|
|
D.assert(FocusManagerUtils._focusDebug(
|
|
"Node NOT requesting focus because canRequestFocus is false: $this"));
|
|
return;
|
|
}
|
|
|
|
if (_parent == null) {
|
|
_requestFocusWhenReparented = true;
|
|
return;
|
|
}
|
|
|
|
_setAsFocusedChildForScope();
|
|
if (hasPrimaryFocus && (_manager._markedForFocus == null || _manager._markedForFocus == this)) {
|
|
return;
|
|
}
|
|
|
|
_hasKeyboardToken = true;
|
|
D.assert(FocusManagerUtils._focusDebug($"Node requesting focus: {this}"));
|
|
_markNextFocus(this);
|
|
}
|
|
|
|
|
|
bool _requestFocusWhenReparented = false;
|
|
|
|
|
|
public void _setAsFocusedChildForScope() {
|
|
FocusNode scopeFocus = this;
|
|
foreach (FocusScopeNode ancestor in ancestors) {
|
|
if (ancestor is FocusScopeNode) {
|
|
D.assert(scopeFocus != ancestor, () => "Somehow made a loop by setting focusedChild to its scope.");
|
|
D.assert(FocusManagerUtils._focusDebug($"Setting {scopeFocus} as focused child for scope:",
|
|
new List<string> {ancestor.ToString()}));
|
|
ancestor._focusedChildren.Remove(scopeFocus);
|
|
|
|
ancestor._focusedChildren.Add(scopeFocus);
|
|
scopeFocus = ancestor;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*bool nextFocus() {
|
|
return FocusTraversalGroup.of(context).next(this);
|
|
}*
|
|
|
|
|
|
bool previousFocus() {
|
|
return FocusTraversalGroup.of(context).previous(this);
|
|
}
|
|
|
|
|
|
bool focusInDirection(TraversalDirection direction) {
|
|
return FocusTraversalGroup.of(context).inDirection(this, direction);
|
|
}*/
|
|
|
|
|
|
/*public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<BuildContext>("context", context, defaultValue: null));
|
|
properties.add(new FlagProperty("canRequestFocus", value: canRequestFocus, ifFalse: "NOT FOCUSABLE", defaultValue: true));
|
|
properties.add(new FlagProperty("hasFocus", value: hasFocus && !hasPrimaryFocus, ifTrue: "IN FOCUS PATH", defaultValue: false));
|
|
properties.add(new FlagProperty("hasPrimaryFocus", value: hasPrimaryFocus, ifTrue: "PRIMARY FOCUS", defaultValue: false));
|
|
}*/
|
|
|
|
|
|
/*public override List<DiagnosticsNode> debugDescribeChildren() {
|
|
int count = 1;
|
|
return _children.map<DiagnosticsNode>((FocusNode child) {
|
|
return child.toDiagnosticsNode(name: "Child ${count++}");
|
|
}).toList();
|
|
}*/
|
|
|
|
|
|
/*public string toStringShort() {//override
|
|
bool hasDebugLabel = debugLabel != null && debugLabel.isNotEmpty();
|
|
string nullStr = "";
|
|
string extraData = $"{(hasDebugLabel ? debugLabel : nullStr)} " +
|
|
$"{(hasFocus && hasDebugLabel ? nullStr : nullStr)}" +
|
|
$"{(hasFocus && !hasPrimaryFocus ? "[IN FOCUS PATH]" : nullStr)}"+
|
|
$"{(hasPrimaryFocus ? "[PRIMARY FOCUS]" : nullStr)}";
|
|
return $"{describeIdentity(this)}" + $"{(extraData.isNotEmpty() ? extraData : nullStr)}";
|
|
}
|
|
}*/
|
|
}
|
|
|
|
public class FocusScopeNode : FocusNode {
|
|
public FocusScopeNode(
|
|
string debugLabel = null,
|
|
FocusOnKeyCallback onKey = null,
|
|
bool skipTraversal = false,
|
|
bool canRequestFocus = true
|
|
) : base(
|
|
debugLabel: debugLabel,
|
|
onKey: onKey,
|
|
canRequestFocus: canRequestFocus,
|
|
skipTraversal: skipTraversal
|
|
) {
|
|
D.assert(skipTraversal != null);
|
|
D.assert(canRequestFocus != null);
|
|
|
|
}
|
|
|
|
public FocusScopeNode nearestScope {
|
|
get { return enclosingScope; }
|
|
}
|
|
|
|
public bool isFirstFocus {
|
|
get {
|
|
return enclosingScope.focusedChild == this;
|
|
}
|
|
}
|
|
|
|
|
|
public FocusNode focusedChild {
|
|
get {
|
|
D.assert(_focusedChildren.isEmpty() || _focusedChildren.Last().enclosingScope == this,()=> "Focused child does not have the same idea of its enclosing scope as the scope does.");
|
|
return _focusedChildren.isNotEmpty() ? _focusedChildren.Last() : null;
|
|
}
|
|
}
|
|
|
|
public readonly List<FocusNode> _focusedChildren = new List<FocusNode>();
|
|
|
|
public void setFirstFocus(FocusScopeNode scope) {
|
|
D.assert(scope != null);
|
|
D.assert(scope != this, ()=>"Unexpected self-reference in setFirstFocus.");
|
|
D.assert(FocusManagerUtils._focusDebug("Setting scope as first focus in $this to node:", new List<string>{scope.ToString()}));
|
|
if (scope._parent == null) {
|
|
_reparent(scope);
|
|
}
|
|
D.assert(scope.ancestors.Contains(this), ()=>$"{typeof(FocusScopeNode)}" + $"{scope} must be a child of"+ $" {this} to set it as first focus.");
|
|
if (hasFocus) {
|
|
scope._doRequestFocus(findFirstFocus: true);
|
|
} else {
|
|
scope._setAsFocusedChildForScope();
|
|
}
|
|
}
|
|
|
|
public void autofocus(FocusNode node) {
|
|
D.assert(FocusManagerUtils._focusDebug("Node autofocusing: $node"));
|
|
if (focusedChild == null) {
|
|
if (node._parent == null) {
|
|
_reparent(node);
|
|
}
|
|
D.assert(node.ancestors.Contains(this), ()=>"Autofocus was requested for a node that is not a descendant of the scope from which it was requested.");
|
|
node._doRequestFocus(findFirstFocus: true);
|
|
}
|
|
}
|
|
|
|
public override void _doRequestFocus( bool findFirstFocus = false) {
|
|
D.assert(findFirstFocus != null);
|
|
|
|
// It is possible that a previously focused child is no longer focusable.
|
|
while (focusedChild != null && !focusedChild.canRequestFocus)
|
|
_focusedChildren.removeLast();
|
|
|
|
|
|
if (!findFirstFocus) {
|
|
if (canRequestFocus) {
|
|
_setAsFocusedChildForScope();
|
|
_markNextFocus(this);
|
|
}
|
|
return;
|
|
}
|
|
|
|
FocusNode primaryFocus = focusedChild ?? this;
|
|
|
|
while (primaryFocus is FocusScopeNode && (( FocusScopeNode)primaryFocus).focusedChild != null) {
|
|
FocusScopeNode scope = primaryFocus as FocusScopeNode;
|
|
primaryFocus = scope.focusedChild;
|
|
}
|
|
if (primaryFocus == this) {
|
|
|
|
if (primaryFocus.canRequestFocus) {
|
|
_setAsFocusedChildForScope();
|
|
_markNextFocus(this);
|
|
}
|
|
} else {
|
|
|
|
primaryFocus._doRequestFocus(findFirstFocus: findFirstFocus);
|
|
}
|
|
}
|
|
|
|
/*public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
if (_focusedChildren.isEmpty()) {
|
|
return;
|
|
}
|
|
List<string> childList = _focusedChildren.reversed.map<string>((FocusNode child) {
|
|
return child.toStringShort();
|
|
}).toList();
|
|
properties.add(new IEnumerableProperty<string>("focusedChildren", childList, defaultValue: new List<string>()));
|
|
}*/
|
|
}
|
|
|
|
public class FocusManager : ChangeNotifier {//DiagnosticableTreeMixin,
|
|
public FocusManager()
|
|
{
|
|
rootScope._manager = this;
|
|
//RawKeyboard.instance.addListener(_handleRawKeyEvent);
|
|
GestureBinding.instance.pointerRouter.addGlobalRoute(_handlePointerEvent);
|
|
}
|
|
public static FocusManager instance {
|
|
get {
|
|
return WidgetsBinding.instance.focusManager;
|
|
}
|
|
}
|
|
|
|
bool _lastInteractionWasTouch = true;
|
|
|
|
|
|
FocusHighlightStrategy highlightStrategy {
|
|
get {
|
|
return _highlightStrategy;
|
|
}
|
|
set {
|
|
_highlightStrategy = highlightStrategy;
|
|
_updateHighlightMode();
|
|
}
|
|
}
|
|
FocusHighlightStrategy _highlightStrategy = FocusHighlightStrategy.automatic;
|
|
|
|
public FocusHighlightMode highlightMode {
|
|
get {
|
|
return _highlightMode;
|
|
}
|
|
}
|
|
FocusHighlightMode _highlightMode = FocusHighlightMode.touch;
|
|
|
|
|
|
void _updateHighlightMode() {
|
|
_lastInteractionWasTouch = false;//(Platform.isAndroid || Platform.isIOS || !WidgetsBinding.instance.mouseTracker.mouseIsConnected);
|
|
FocusHighlightMode newMode = FocusHighlightMode.touch;
|
|
switch (highlightStrategy) {
|
|
case FocusHighlightStrategy.automatic:
|
|
if (_lastInteractionWasTouch) {
|
|
newMode = FocusHighlightMode.touch;
|
|
} else {
|
|
newMode = FocusHighlightMode.traditional;
|
|
}
|
|
break;
|
|
case FocusHighlightStrategy.alwaysTouch:
|
|
newMode = FocusHighlightMode.touch;
|
|
break;
|
|
case FocusHighlightStrategy.alwaysTraditional:
|
|
newMode = FocusHighlightMode.traditional;
|
|
break;
|
|
}
|
|
if (newMode != _highlightMode) {
|
|
_highlightMode = newMode;
|
|
//_notifyHighlightModeListeners();
|
|
}
|
|
}
|
|
|
|
// The list of listeners for [highlightMode] state changes.
|
|
public readonly List<ValueChanged<FocusHighlightMode>> _listeners = new List<ValueChanged<FocusHighlightMode>>();
|
|
|
|
public void addHighlightModeListener(ValueChanged<FocusHighlightMode> listener) => _listeners.Add(listener);
|
|
|
|
public void removeHighlightModeListener(ValueChanged<FocusHighlightMode> listener) => _listeners?.Remove(listener);
|
|
|
|
/*void _notifyHighlightModeListeners() {
|
|
if (_listeners.isEmpty()) {
|
|
return;
|
|
}
|
|
List<ValueChanged<FocusHighlightMode>> localListeners = List<ValueChanged<FocusHighlightMode>>.from(_listeners);
|
|
foreach( ValueChanged<FocusHighlightMode> listener in localListeners) {
|
|
try {
|
|
if (_listeners.Contains(listener)) {
|
|
listener(_highlightMode);
|
|
}
|
|
} catch (exception, stack) {
|
|
InformationCollector collector;
|
|
D.assert(() =>{
|
|
collector = () sync* {
|
|
yield DiagnosticsProperty<FocusManager>(
|
|
"The $runtimeType sending notification was",
|
|
this,
|
|
style: DiagnosticsTreeStyle.errorProperty,
|
|
);
|
|
};
|
|
return true;
|
|
}());
|
|
FlutterError.reportError(FlutterErrorDetails(
|
|
exception: exception,
|
|
stack: stack,
|
|
library: "widgets library",
|
|
context: ErrorDescription("while dispatching notifications for $runtimeType"),
|
|
informationCollector: collector,
|
|
));
|
|
}
|
|
}
|
|
}*/
|
|
|
|
public readonly FocusScopeNode rootScope = new FocusScopeNode(debugLabel: "Root Focus Scope");
|
|
|
|
void _handlePointerEvent(PointerEvent Event) {
|
|
bool currentInteractionIsTouch = false;
|
|
switch (Event.kind) {
|
|
case PointerDeviceKind.touch:
|
|
case PointerDeviceKind.stylus:
|
|
case PointerDeviceKind.invertedStylus:
|
|
currentInteractionIsTouch = true;
|
|
break;
|
|
case PointerDeviceKind.mouse:
|
|
case PointerDeviceKind.unknown:
|
|
currentInteractionIsTouch = false;
|
|
break;
|
|
}
|
|
if (_lastInteractionWasTouch != currentInteractionIsTouch) {
|
|
_lastInteractionWasTouch = currentInteractionIsTouch;
|
|
_updateHighlightMode();
|
|
}
|
|
}
|
|
|
|
/*void _handleRawKeyEvent(RawKeyEvent Event) {
|
|
|
|
if (_lastInteractionWasTouch) {
|
|
_lastInteractionWasTouch = false;
|
|
_updateHighlightMode();
|
|
}
|
|
|
|
D.assert(FocusManagerUtils._focusDebug($"Received key event {Event.logicalKey}"));
|
|
|
|
if (_primaryFocus == null) {
|
|
D.assert(FocusManagerUtils._focusDebug($"No primary focus for key event, ignored: {Event}"));
|
|
return;
|
|
}
|
|
|
|
bool handled = false;
|
|
foreach (FocusNode node in List < FocusNode >{
|
|
_primaryFocus, ..._primaryFocus.ancestors
|
|
}) {
|
|
if (node.onKey != null && node.onKey(node, Event)) {
|
|
D.assert(FocusManagerUtils._focusDebug($"Node {node} handled key event {Event}."));
|
|
handled = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!handled) {
|
|
D.assert(FocusManagerUtils._focusDebug($"Key event not handled by anyone: {Event}."));
|
|
}
|
|
}*/
|
|
|
|
|
|
public FocusNode primaryFocus {
|
|
get{return _primaryFocus;}
|
|
}
|
|
FocusNode _primaryFocus;
|
|
|
|
|
|
public readonly HashSet<FocusNode> _dirtyNodes = new HashSet<FocusNode>();
|
|
|
|
public FocusNode _markedForFocus;
|
|
|
|
public void _markDetached(FocusNode node) {
|
|
|
|
D.assert(FocusManagerUtils._focusDebug($"Node was detached: {node}"));
|
|
if (_primaryFocus == node) {
|
|
_primaryFocus = null;
|
|
}
|
|
_dirtyNodes?.Remove(node);
|
|
}
|
|
|
|
public void _markPropertiesChanged(FocusNode node) {
|
|
_markNeedsUpdate();
|
|
D.assert(FocusManagerUtils._focusDebug($"Properties changed for node {node}."));
|
|
_dirtyNodes?.Add(node);
|
|
}
|
|
|
|
public void _markNextFocus(FocusNode node) {
|
|
if (_primaryFocus == node) {
|
|
|
|
_markedForFocus = null;
|
|
} else {
|
|
_markedForFocus = node;
|
|
_markNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
// True indicates that there is an update pending.
|
|
bool _haveScheduledUpdate = false;
|
|
|
|
void _markNeedsUpdate() {
|
|
D.assert(FocusManagerUtils._focusDebug($"Scheduling update, current focus is {_primaryFocus}, next focus will be {_markedForFocus}"));
|
|
if (_haveScheduledUpdate) {
|
|
return;
|
|
}
|
|
_haveScheduledUpdate = true;
|
|
//scheduleMicrotask(_applyFocusChange);
|
|
}
|
|
/*public void reparentIfNeeded(FocusNode node) {
|
|
D.assert(node != null);
|
|
if (node._parent == null || node._parent == this) {
|
|
return;
|
|
}
|
|
|
|
node.unfocus();
|
|
D.assert(node._parent == null);
|
|
if (_focus == null) {
|
|
_setFocus(node);
|
|
}
|
|
}*/
|
|
|
|
/*void _applyFocusChange() {
|
|
_haveScheduledUpdate = false;
|
|
FocusNode previousFocus = _primaryFocus;
|
|
if (_primaryFocus == null && _markedForFocus == null) {
|
|
|
|
_markedForFocus = rootScope;
|
|
}
|
|
D.assert(FocusManagerUtils._focusDebug($"Refreshing focus state. Next focus will be {_markedForFocus}"));
|
|
|
|
if (_markedForFocus != null && _markedForFocus != _primaryFocus) {
|
|
HashSet<FocusNode> previousPath = previousFocus?.ancestors?.toSet() ?? new HashSet<FocusNode>();
|
|
HashSet<FocusNode> nextPath = _markedForFocus.ancestors.toSet();
|
|
// Notify nodes that are newly focused.
|
|
_dirtyNodes.addAll(nextPath.difference(previousPath));
|
|
// Notify nodes that are no longer focused
|
|
_dirtyNodes.addAll(previousPath.difference(nextPath));
|
|
|
|
_primaryFocus = _markedForFocus;
|
|
_markedForFocus = null;
|
|
}
|
|
if (previousFocus != _primaryFocus) {
|
|
D.assert(FocusManagerUtils._focusDebug("Updating focus from $previousFocus to $_primaryFocus"));
|
|
if (previousFocus != null) {
|
|
_dirtyNodes.Add(previousFocus);
|
|
}
|
|
if (_primaryFocus != null) {
|
|
_dirtyNodes.Add(_primaryFocus);
|
|
}
|
|
}
|
|
D.assert(FocusManagerUtils._focusDebug("Notifying ${_dirtyNodes.length} dirty nodes:", _dirtyNodes.toList().map<String>((FocusNode node) => node.toString())));
|
|
foreach ( FocusNode node in _dirtyNodes) {
|
|
node._notify();
|
|
}
|
|
_dirtyNodes.Clear();
|
|
if (previousFocus != _primaryFocus) {
|
|
notifyListeners();
|
|
}
|
|
D.assert(()=> {
|
|
if (_kDebugFocus) {
|
|
debugDumpFocusTree();
|
|
}
|
|
return true;
|
|
});
|
|
}*/
|
|
|
|
|
|
/*public override List<DiagnosticsNode> debugDescribeChildren() {
|
|
return new List<DiagnosticsNode>{
|
|
rootScope.toDiagnosticsNode(name: "rootScope")
|
|
};
|
|
}*/
|
|
|
|
|
|
/*public void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
properties.add(new FlagProperty("haveScheduledUpdate", value: _haveScheduledUpdate, ifTrue: "UPDATE SCHEDULED"));
|
|
properties.add(new DiagnosticsProperty<FocusNode>("primaryFocus", primaryFocus, defaultValue: null));
|
|
properties.add(new DiagnosticsProperty<FocusNode>("nextFocus", _markedForFocus, defaultValue: null));
|
|
Element element = primaryFocus?.context as Element;
|
|
if (element != null) {
|
|
properties.add(new DiagnosticsProperty<String>("primaryFocusCreator", element.debugGetCreatorChain(20)));
|
|
}
|
|
}*/
|
|
}
|
|
|
|
}
|
|
/*
|
|
public class FocusNode : ChangeNotifier {
|
|
internal FocusScopeNode _parent;
|
|
internal FocusManager _manager;
|
|
internal bool _hasKeyboardToken = false;
|
|
|
|
public bool hasFocus {
|
|
get {
|
|
FocusNode node = null;
|
|
if (_manager != null) {
|
|
node = _manager._currentFocus;
|
|
}
|
|
|
|
return node == this;
|
|
}
|
|
}
|
|
|
|
public bool consumeKeyboardToken() {
|
|
if (!_hasKeyboardToken) {
|
|
return false;
|
|
}
|
|
|
|
_hasKeyboardToken = false;
|
|
return true;
|
|
}
|
|
|
|
public void unfocus() {
|
|
if (_parent != null) {
|
|
_parent._resignFocus(this);
|
|
}
|
|
|
|
D.assert(_parent == null);
|
|
D.assert(_manager == null);
|
|
}
|
|
|
|
public override void dispose() {
|
|
if (_manager != null) {
|
|
_manager._willDisposeFocusNode(this);
|
|
}
|
|
|
|
if (_parent != null) {
|
|
_parent._resignFocus(this);
|
|
}
|
|
|
|
D.assert(_parent == null);
|
|
D.assert(_manager == null);
|
|
base.dispose();
|
|
}
|
|
|
|
internal void _notify() {
|
|
notifyListeners();
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"{foundation_.describeIdentity(this)} hasFocus: {hasFocus}";
|
|
}
|
|
}
|
|
|
|
public class FocusScopeNode : DiagnosticableTree {
|
|
internal FocusManager _manager;
|
|
internal FocusScopeNode _parent;
|
|
|
|
internal FocusScopeNode _nextSibling;
|
|
internal FocusScopeNode _previousSibling;
|
|
|
|
internal FocusScopeNode _firstChild;
|
|
internal FocusScopeNode _lastChild;
|
|
|
|
internal FocusNode _focus;
|
|
internal List<FocusScopeNode> _focusPath;
|
|
|
|
public bool isFirstFocus {
|
|
get { return _parent == null || _parent._firstChild == this; }
|
|
}
|
|
|
|
internal List<FocusScopeNode> _getFocusPath() {
|
|
List<FocusScopeNode> nodes = new List<FocusScopeNode> {this};
|
|
FocusScopeNode node = _parent;
|
|
while (node != null && node != _manager?.rootScope) {
|
|
nodes.Add(node);
|
|
node = node._parent;
|
|
}
|
|
|
|
return nodes;
|
|
}
|
|
|
|
internal void _prepend(FocusScopeNode child) {
|
|
D.assert(child != this);
|
|
D.assert(child != _firstChild);
|
|
D.assert(child != _lastChild);
|
|
D.assert(child._parent == null);
|
|
D.assert(child._manager == null);
|
|
D.assert(child._nextSibling == null);
|
|
D.assert(child._previousSibling == null);
|
|
D.assert(() => {
|
|
var node = this;
|
|
while (node._parent != null) {
|
|
node = node._parent;
|
|
}
|
|
|
|
D.assert(node != child);
|
|
return true;
|
|
});
|
|
child._parent = this;
|
|
child._nextSibling = _firstChild;
|
|
if (_firstChild != null) {
|
|
_firstChild._previousSibling = child;
|
|
}
|
|
|
|
_firstChild = child;
|
|
_lastChild = _lastChild ?? child;
|
|
child._updateManager(_manager);
|
|
}
|
|
|
|
void _updateManager(FocusManager manager) {
|
|
Action<FocusScopeNode> update = null;
|
|
update = (child) => {
|
|
if (child._manager == manager) {
|
|
return;
|
|
}
|
|
|
|
child._manager = manager;
|
|
// We don"t proactively null out the manager for FocusNodes because the
|
|
// manager holds the currently active focus node until the end of the
|
|
// microtask, even if that node is detached from the focus tree.
|
|
if (manager != null && child._focus != null) {
|
|
child._focus._manager = manager;
|
|
}
|
|
|
|
child._visitChildren(update);
|
|
};
|
|
update(this);
|
|
}
|
|
|
|
void _visitChildren(Action<FocusScopeNode> vistor) {
|
|
FocusScopeNode child = _firstChild;
|
|
while (child != null) {
|
|
vistor.Invoke(child);
|
|
child = child._nextSibling;
|
|
}
|
|
}
|
|
|
|
bool _debugUltimatePreviousSiblingOf(FocusScopeNode child, FocusScopeNode equals) {
|
|
while (child._previousSibling != null) {
|
|
D.assert(child._previousSibling != child);
|
|
child = child._previousSibling;
|
|
}
|
|
|
|
return child == equals;
|
|
}
|
|
|
|
bool _debugUltimateNextSiblingOf(FocusScopeNode child, FocusScopeNode equals) {
|
|
while (child._nextSibling != null) {
|
|
D.assert(child._nextSibling != child);
|
|
child = child._nextSibling;
|
|
}
|
|
|
|
return child == equals;
|
|
}
|
|
|
|
internal void _remove(FocusScopeNode child) {
|
|
D.assert(child._parent == this);
|
|
D.assert(child._manager == _manager);
|
|
D.assert(_debugUltimatePreviousSiblingOf(child, equals: _firstChild));
|
|
D.assert(_debugUltimateNextSiblingOf(child, equals: _lastChild));
|
|
if (child._previousSibling == null) {
|
|
D.assert(_firstChild == child);
|
|
_firstChild = child._nextSibling;
|
|
}
|
|
else {
|
|
child._previousSibling._nextSibling = child._nextSibling;
|
|
}
|
|
|
|
if (child._nextSibling == null) {
|
|
D.assert(_lastChild == child);
|
|
_lastChild = child._previousSibling;
|
|
}
|
|
else {
|
|
child._nextSibling._previousSibling = child._previousSibling;
|
|
}
|
|
|
|
child._previousSibling = null;
|
|
child._nextSibling = null;
|
|
child._parent = null;
|
|
child._updateManager(null);
|
|
}
|
|
|
|
internal void _didChangeFocusChain() {
|
|
if (isFirstFocus && _manager != null) {
|
|
_manager._markNeedsUpdate();
|
|
}
|
|
}
|
|
|
|
// TODO: need update
|
|
public void requestFocus(FocusNode node = null) {
|
|
// D.assert(node != null);
|
|
var focusPath = _manager?._getCurrentFocusPath();
|
|
if (_focus == node &&
|
|
(_focusPath == focusPath || (focusPath != null && _focusPath != null &&
|
|
_focusPath.SequenceEqual(focusPath)))) {
|
|
return;
|
|
}
|
|
|
|
if (_focus != null) {
|
|
_focus.unfocus();
|
|
}
|
|
|
|
node._hasKeyboardToken = true;
|
|
_setFocus(node);
|
|
}
|
|
|
|
public void autofocus(FocusNode node) {
|
|
D.assert(node != null);
|
|
if (_focus == null) {
|
|
node._hasKeyboardToken = true;
|
|
_setFocus(node);
|
|
}
|
|
}
|
|
|
|
public void reparentIfNeeded(FocusNode node) {
|
|
D.assert(node != null);
|
|
if (node._parent == null || node._parent == this) {
|
|
return;
|
|
}
|
|
|
|
node.unfocus();
|
|
D.assert(node._parent == null);
|
|
if (_focus == null) {
|
|
_setFocus(node);
|
|
}
|
|
}
|
|
|
|
internal void _setFocus(FocusNode node) {
|
|
D.assert(node != null);
|
|
D.assert(node._parent == null);
|
|
D.assert(_focus == null);
|
|
_focus = node;
|
|
_focus._parent = this;
|
|
_focus._manager = _manager;
|
|
_focus._hasKeyboardToken = true;
|
|
_didChangeFocusChain();
|
|
_focusPath = _getFocusPath();
|
|
}
|
|
|
|
internal void _resignFocus(FocusNode node) {
|
|
D.assert(node != null);
|
|
if (_focus != node) {
|
|
return;
|
|
}
|
|
|
|
_focus._parent = null;
|
|
_focus._manager = null;
|
|
_focus = null;
|
|
_didChangeFocusChain();
|
|
}
|
|
|
|
public void setFirstFocus(FocusScopeNode child) {
|
|
D.assert(child != null);
|
|
D.assert(child._parent == null || child._parent == this);
|
|
if (_firstChild == child) {
|
|
return;
|
|
}
|
|
|
|
child.detach();
|
|
_prepend(child);
|
|
D.assert(child._parent == this);
|
|
_didChangeFocusChain();
|
|
}
|
|
|
|
public void reparentScopeIfNeeded(FocusScopeNode child) {
|
|
D.assert(child != null);
|
|
if (child._parent == null || child._parent == this) {
|
|
return;
|
|
}
|
|
|
|
if (child.isFirstFocus) {
|
|
setFirstFocus(child);
|
|
}
|
|
else {
|
|
child.detach();
|
|
}
|
|
}
|
|
|
|
public void detach() {
|
|
_didChangeFocusChain();
|
|
if (_parent != null) {
|
|
_parent._remove(this);
|
|
}
|
|
|
|
D.assert(_parent == null);
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
if (_focus != null) {
|
|
properties.add(new DiagnosticsProperty<FocusNode>("focus", _focus));
|
|
}
|
|
}
|
|
|
|
public override List<DiagnosticsNode> debugDescribeChildren() {
|
|
var children = new List<DiagnosticsNode>();
|
|
if (_firstChild != null) {
|
|
FocusScopeNode child = _firstChild;
|
|
int count = 1;
|
|
while (true) {
|
|
children.Add(child.toDiagnosticsNode(name: $"child {count}"));
|
|
if (child == _lastChild) {
|
|
break;
|
|
}
|
|
|
|
child = child._nextSibling;
|
|
count += 1;
|
|
}
|
|
}
|
|
|
|
return children;
|
|
}
|
|
}
|
|
|
|
public class FocusManager {
|
|
public FocusManager() {
|
|
rootScope._manager = this;
|
|
D.assert(rootScope._firstChild == null);
|
|
D.assert(rootScope._lastChild == null);
|
|
}
|
|
|
|
public readonly FocusScopeNode rootScope = new FocusScopeNode();
|
|
internal readonly FocusScopeNode _noneScope = new FocusScopeNode();
|
|
|
|
public FocusNode currentFocus {
|
|
get { return _currentFocus; }
|
|
}
|
|
|
|
internal FocusNode _currentFocus;
|
|
|
|
internal void _willDisposeFocusNode(FocusNode node) {
|
|
D.assert(node != null);
|
|
if (_currentFocus == node) {
|
|
_currentFocus = null;
|
|
}
|
|
}
|
|
|
|
bool _haveScheduledUpdate = false;
|
|
|
|
internal void _markNeedsUpdate() {
|
|
if (_haveScheduledUpdate) {
|
|
return;
|
|
}
|
|
|
|
_haveScheduledUpdate = true;
|
|
async_.scheduleMicrotask(() => {
|
|
_update();
|
|
return null;
|
|
});
|
|
}
|
|
|
|
internal FocusNode _findNextFocus() {
|
|
FocusScopeNode scope = rootScope;
|
|
while (scope._firstChild != null) {
|
|
scope = scope._firstChild;
|
|
}
|
|
|
|
return scope._focus;
|
|
}
|
|
|
|
internal void _update() {
|
|
_haveScheduledUpdate = false;
|
|
var nextFocus = _findNextFocus();
|
|
if (_currentFocus == nextFocus) {
|
|
return;
|
|
}
|
|
|
|
var previousFocus = _currentFocus;
|
|
_currentFocus = nextFocus;
|
|
if (previousFocus != null) {
|
|
previousFocus._notify();
|
|
}
|
|
|
|
if (_currentFocus != null) {
|
|
_currentFocus._notify();
|
|
}
|
|
}
|
|
|
|
internal List<FocusScopeNode> _getCurrentFocusPath() {
|
|
return _currentFocus?._parent?._getFocusPath();
|
|
}
|
|
|
|
public void focusNone(bool focus) {
|
|
if (focus) {
|
|
if (_noneScope._parent != null && _noneScope.isFirstFocus) {
|
|
return;
|
|
}
|
|
|
|
rootScope.setFirstFocus(_noneScope);
|
|
}
|
|
else {
|
|
if (_noneScope._parent == null) {
|
|
return;
|
|
}
|
|
|
|
_noneScope.detach();
|
|
}
|
|
}
|
|
|
|
public override string ToString() {
|
|
var status = _haveScheduledUpdate ? " UPDATE SCHEDULED" : "";
|
|
var indent = " ";
|
|
return string.Format("{1}{2}\n{0}currentFocus: {3}\n{4}", indent, foundation_.describeIdentity(this),
|
|
status, _currentFocus,
|
|
rootScope.toStringDeep(prefixLineOne: indent, prefixOtherLines: indent));
|
|
}
|
|
}*/
|