浏览代码
Merge branch 'shiyun/skia' of github.com:Unity-Technologies/com.unity.uiwidgets into fix_runtime_errors
/siyaoH-1.17-PlatformMessage
Merge branch 'shiyun/skia' of github.com:Unity-Technologies/com.unity.uiwidgets into fix_runtime_errors
/siyaoH-1.17-PlatformMessage
xingweizhu
4 年前
当前提交
f8a96fc0
共有 6 个文件被更改,包括 902 次插入 和 41 次删除
-
40com.unity.uiwidgets/Runtime/cupertino/text_selection.cs
-
2com.unity.uiwidgets/Runtime/widgets/basic.cs
-
480com.unity.uiwidgets/Runtime/widgets/focus_manager.cs
-
29com.unity.uiwidgets/Runtime/widgets/focus_scope.cs
-
13com.unity.uiwidgets/Runtime/widgets/text_selection.cs
-
379com.unity.uiwidgets/Runtime/widgets/focus_traversal.cs
|
|||
using System; |
|||
using System.Collections.Generic; |
|||
using System.Linq; |
|||
using Unity.UIWidgets.foundation; |
|||
using Unity.UIWidgets.ui; |
|||
using Unity.UIWidgets.widgets; |
|||
|
|||
|
|||
namespace Unity.UIWidgets.widgets { |
|||
/*public class FocusTraversalGroup : StatefulWidget { |
|||
public FocusTraversalGroup( |
|||
Key key = null, |
|||
FocusTraversalPolicy policy = null, |
|||
Widget child = null |
|||
) : base(key: key) { |
|||
policy = policy ?? new ReadingOrderTraversalPolicy(); |
|||
this.child = child; |
|||
} |
|||
|
|||
public readonly Widget child; |
|||
public readonly FocusTraversalPolicy policy; |
|||
public static FocusTraversalPolicy of(BuildContext context, bool nullOk = false) { |
|||
D.assert(context != null); |
|||
_FocusTraversalGroupMarker inherited = context?.dependOnInheritedWidgetOfExactType<_FocusTraversalGroupMarker>(); |
|||
D.assert(() =>{ |
|||
if (nullOk) { |
|||
return true; |
|||
} |
|||
if (inherited == null) { |
|||
throw new UIWidgetsError( |
|||
"Unable to find a FocusTraversalGroup widget in the context.\n" + |
|||
"FocusTraversalGroup.of() was called with a context that does not contain a " + |
|||
"FocusTraversalGroup.\n" + |
|||
"No FocusTraversalGroup ancestor could be found starting from the context that was " + |
|||
"passed to FocusTraversalGroup.of(). This can happen because there is not a " + |
|||
"WidgetsApp or MaterialApp widget (those widgets introduce a FocusTraversalGroup), " + |
|||
"or it can happen if the context comes from a widget above those widgets.\n" + |
|||
"The context used was:\n" + |
|||
$" {context}" |
|||
); |
|||
} |
|||
return true; |
|||
}); |
|||
return inherited?.policy; |
|||
} |
|||
|
|||
public override State createState() { |
|||
return new _FocusTraversalGroupState(); |
|||
} |
|||
|
|||
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) { |
|||
base.debugFillProperties(properties); |
|||
properties.add(new TextTreeRenderer.DiagnosticsProperty<FocusTraversalPolicy>("policy", policy)); |
|||
} |
|||
} |
|||
public class _FocusTraversalGroupState : State<FocusTraversalGroup> { |
|||
FocusNode focusNode; |
|||
public override void initState() { |
|||
base.initState(); |
|||
focusNode = new FocusNode( |
|||
canRequestFocus: false, |
|||
skipTraversal: true, |
|||
debugLabel: "FocusTraversalGroup" |
|||
); |
|||
} |
|||
public override void dispose() { |
|||
focusNode?.dispose(); |
|||
base.dispose(); |
|||
} |
|||
public override Widget build(BuildContext context) { |
|||
return new _FocusTraversalGroupMarker( |
|||
policy: widget.policy, |
|||
focusNode: focusNode, |
|||
child: new Focus( |
|||
focusNode: focusNode, |
|||
canRequestFocus: false, |
|||
skipTraversal: true, |
|||
includeSemantics: false, |
|||
child: widget.child |
|||
) |
|||
); |
|||
} |
|||
} |
|||
public class _FocusTraversalGroupMarker : InheritedWidget { |
|||
public _FocusTraversalGroupMarker( |
|||
FocusTraversalPolicy policy = null, |
|||
FocusNode focusNode = null, |
|||
Widget child = null |
|||
) : base(child: child) { |
|||
D.assert(policy != null); |
|||
D.assert(focusNode != null); |
|||
this.policy = policy; |
|||
this.focusNode = focusNode; |
|||
} |
|||
public readonly FocusTraversalPolicy policy; |
|||
public readonly FocusNode focusNode; |
|||
public override bool updateShouldNotify(InheritedWidget oldWidget) => false; |
|||
} |
|||
|
|||
public abstract class FocusTraversalPolicy : TextTreeRenderer.Diagnosticable { |
|||
public FocusTraversalPolicy() { |
|||
} |
|||
|
|||
public virtual FocusNode findFirstFocus(FocusNode currentNode) { |
|||
D.assert(currentNode != null); |
|||
FocusScopeNode scope = currentNode.nearestScope; |
|||
FocusNode candidate = scope.focusedChild; |
|||
if (candidate == null && scope.descendants.Count() != 0) { |
|||
List<FocusNode> sorted = _sortAllDescendants(scope); |
|||
candidate = sorted.isNotEmpty() ? sorted.First() : null; |
|||
} |
|||
|
|||
candidate ??= currentNode; |
|||
return candidate; |
|||
} |
|||
|
|||
public abstract FocusNode findFirstFocusInDirection(FocusNode currentNode, TraversalDirection direction); |
|||
|
|||
public abstract void invalidateScopeData(FocusScopeNode node); |
|||
|
|||
public abstract void changedScope(FocusNode node = null, FocusScopeNode oldScope = null); |
|||
bool next(FocusNode currentNode) => _moveFocus(currentNode, forward: true); |
|||
|
|||
bool previous(FocusNode currentNode) => _moveFocus(currentNode, forward: false); |
|||
|
|||
public abstract bool inDirection(FocusNode currentNode, TraversalDirection direction); |
|||
|
|||
public abstract IEnumerable<FocusNode> sortDescendants(IEnumerable<FocusNode> descendants); |
|||
public _FocusTraversalGroupMarker _getMarker(BuildContext context) { |
|||
return context?.getElementForInheritedWidgetOfExactType<_FocusTraversalGroupMarker>()?.widget as _FocusTraversalGroupMarker; |
|||
} |
|||
public List<FocusNode> _sortAllDescendants(FocusScopeNode scope) { |
|||
D.assert(scope != null); |
|||
_FocusTraversalGroupMarker scopeGroupMarker = _getMarker(scope.context); |
|||
FocusTraversalPolicy defaultPolicy = scopeGroupMarker?.policy ?? ReadingOrderTraversalPolicy(); |
|||
Dictionary<FocusNode, _FocusTraversalGroupInfo> groups = new Dictionary<FocusNode, _FocusTraversalGroupInfo>(); |
|||
foreach(FocusNode node in scope.descendants) { |
|||
_FocusTraversalGroupMarker groupMarker = _getMarker(node.context); |
|||
FocusNode groupNode = groupMarker?.focusNode; |
|||
if (node == groupNode) { |
|||
BuildContext parentContext = _getAncestor(groupNode.context, count: 2); |
|||
_FocusTraversalGroupMarker parentMarker = _getMarker(parentContext); |
|||
FocusNode parentNode = parentMarker?.focusNode; |
|||
groups[parentNode] ??= new _FocusTraversalGroupInfo(parentMarker, members: new List<FocusNode>(), defaultPolicy: defaultPolicy); |
|||
D.assert( !groups[parentNode].members.Contains(node) ); |
|||
groups[parentNode].members.add(groupNode); |
|||
continue; |
|||
} |
|||
if (node.canRequestFocus && !node.skipTraversal) { |
|||
groups[groupNode] ??= _FocusTraversalGroupInfo(groupMarker, members: new List<FocusNode>(), defaultPolicy: defaultPolicy); |
|||
D.assert(!groups[groupNode].members.contains(node)); |
|||
groups[groupNode].members.add(node); |
|||
} |
|||
} |
|||
HashSet<FocusNode> groupKeys = groups.Keys.ToSet(); |
|||
foreach ( FocusNode key in groups.keys) { |
|||
List<FocusNode> sortedMembers = groups[key].policy.sortDescendants(groups[key].members).toList(); |
|||
groups[key].members.clear(); |
|||
groups[key].members.addAll(sortedMembers); |
|||
} |
|||
|
|||
List<FocusNode> sortedDescendants = new List<FocusNode>(); |
|||
void visitGroups(_FocusTraversalGroupInfo info) { |
|||
foreach ( FocusNode node in info.members) { |
|||
if (groupKeys.Contains(node)) { |
|||
visitGroups(groups[node]); |
|||
} else { |
|||
sortedDescendants.Add(node); |
|||
} |
|||
} |
|||
} |
|||
visitGroups(groups[scopeGroupMarker?.focusNode]); |
|||
D.assert( |
|||
sortedDescendants.toSet().difference(scope.traversalDescendants.toSet()).isEmpty, |
|||
$"sorted descendants contains more nodes than it should: ({sortedDescendants.toSet().difference(scope.traversalDescendants.toSet())})" |
|||
); |
|||
D.assert( |
|||
scope.traversalDescendants.toSet().difference(sortedDescendants.toSet()).isEmpty, |
|||
"sorted descendants are missing some nodes: (${scope.traversalDescendants.toSet().difference(sortedDescendants.toSet())})" |
|||
); |
|||
return sortedDescendants; |
|||
} |
|||
protected bool _moveFocus(FocusNode currentNode, bool forward = false) { |
|||
D.assert(forward != null); |
|||
if (currentNode == null) { |
|||
return false; |
|||
} |
|||
FocusScopeNode nearestScope = currentNode.nearestScope; |
|||
invalidateScopeData(nearestScope); |
|||
FocusNode focusedChild = nearestScope.focusedChild; |
|||
if (focusedChild == null) { |
|||
FocusNode firstFocus = findFirstFocus(currentNode); |
|||
if (firstFocus != null) { |
|||
_focusAndEnsureVisible( |
|||
firstFocus, |
|||
alignmentPolicy: forward ? ScrollPositionAlignmentPolicy.keepVisibleAtEnd : ScrollPositionAlignmentPolicy.keepVisibleAtStart, |
|||
); |
|||
return true; |
|||
} |
|||
} |
|||
List<FocusNode> sortedNodes = _sortAllDescendants(nearestScope); |
|||
if (forward && focusedChild == sortedNodes.Last()) { |
|||
_focusAndEnsureVisible(sortedNodes.First(), alignmentPolicy: ScrollPositionAlignmentPolicy.keepVisibleAtEnd); |
|||
return true; |
|||
} |
|||
if (!forward && focusedChild == sortedNodes.First()) { |
|||
_focusAndEnsureVisible(sortedNodes.Last(), alignmentPolicy: ScrollPositionAlignmentPolicy.keepVisibleAtStart); |
|||
return true; |
|||
} |
|||
|
|||
IEnumerable<FocusNode> maybeFlipped = new List<FocusNode>(); |
|||
if (forward) { |
|||
maybeFlipped = sortedNodes; |
|||
} |
|||
else { |
|||
sortedNodes.Reverse(); |
|||
maybeFlipped = sortedNodes; |
|||
} |
|||
FocusNode previousNode = null; |
|||
foreach ( FocusNode node in maybeFlipped) { |
|||
if (previousNode == focusedChild) { |
|||
_focusAndEnsureVisible( |
|||
node, |
|||
alignmentPolicy: forward ? ScrollPositionAlignmentPolicy.keepVisibleAtEnd : ScrollPositionAlignmentPolicy.keepVisibleAtStart, |
|||
); |
|||
return true; |
|||
} |
|||
previousNode = node; |
|||
} |
|||
return false; |
|||
} |
|||
} |
|||
public class _DirectionalPolicyDataEntry { |
|||
public _DirectionalPolicyDataEntry( |
|||
TraversalDirection direction , |
|||
FocusNode node ) { |
|||
this.direction = direction; |
|||
this.node = node; |
|||
|
|||
} |
|||
public readonly TraversalDirection direction; |
|||
public readonly FocusNode node; |
|||
} |
|||
|
|||
class _ReadingOrderDirectionalGroupData : Diagnosticable { |
|||
public _ReadingOrderDirectionalGroupData(List<_ReadingOrderSortData> members) { |
|||
this.members = members; |
|||
} |
|||
|
|||
public readonly List<_ReadingOrderSortData> members; |
|||
|
|||
TextDirection directionality { |
|||
get { |
|||
return members.First().directionality; |
|||
} |
|||
} |
|||
|
|||
Rect _rect; |
|||
Rect rect { |
|||
get {if (_rect == null) { |
|||
foreach(Rect rect in members.Select<Rect>( |
|||
(_ReadingOrderSortData data) => data.rect)){ |
|||
_rect ??= rect; |
|||
_rect = _rect.expandToInclude(rect); |
|||
} |
|||
} |
|||
return _rect; |
|||
} |
|||
} |
|||
List<Directionality> memberAncestors { |
|||
get { if (_memberAncestors == null) { |
|||
_memberAncestors = new List<Directionality>(); |
|||
foreach (_ReadingOrderSortData member in members) { |
|||
_memberAncestors.AddRange(member.directionalAncestors); |
|||
} |
|||
} |
|||
return _memberAncestors; } |
|||
|
|||
} |
|||
List<Directionality> _memberAncestors; |
|||
public static void sortWithDirectionality(List<_ReadingOrderDirectionalGroupData> list, TextDirection directionality) { |
|||
mergeSort<_ReadingOrderDirectionalGroupData>(list, compare: (_ReadingOrderDirectionalGroupData a, _ReadingOrderDirectionalGroupData b) =>{ |
|||
switch (directionality) { |
|||
case TextDirection.ltr: |
|||
return a.rect.left.compareTo(b.rect.left); |
|||
case TextDirection.rtl: |
|||
return b.rect.right.compareTo(a.rect.right); |
|||
} |
|||
D.assert(false, ()=>"Unhandled directionality $directionality"); |
|||
return 0; |
|||
}); |
|||
} |
|||
public override void debugFillProperties(TextTreeRenderer.DiagnosticPropertiesBuilder properties) { |
|||
base.debugFillProperties(properties); |
|||
properties.add(DiagnosticsProperty<TextDirection>("directionality", directionality)); |
|||
properties.add(DiagnosticsProperty<Rect>("rect", rect)); |
|||
properties.add(IterableProperty<String>("members", members.map<String>((_ReadingOrderSortData member) { |
|||
return ""${member.node.debugLabel}"(${member.rect})"; |
|||
}))); |
|||
} |
|||
} |
|||
public class ReadingOrderTraversalPolicy : FocusTraversalPolicy , DirectionalFocusTraversalPolicyMixin { |
|||
public List<_ReadingOrderDirectionalGroupData> _collectDirectionalityGroups(IEnumerable<_ReadingOrderSortData> candidates) { |
|||
TextDirection currentDirection = candidates.First().directionality; |
|||
List<_ReadingOrderSortData> currentGroup = new List<_ReadingOrderSortData>(); |
|||
List<_ReadingOrderDirectionalGroupData> result = new List<_ReadingOrderDirectionalGroupData>(); |
|||
foreach ( _ReadingOrderSortData candidate in candidates) { |
|||
if (candidate.directionality == currentDirection) { |
|||
currentGroup.Add(candidate); |
|||
continue; |
|||
} |
|||
currentDirection = candidate.directionality; |
|||
result.Add(_ReadingOrderDirectionalGroupData(currentGroup)); |
|||
currentGroup = new List<_ReadingOrderSortData>(){candidate}; |
|||
} |
|||
if (currentGroup.isNotEmpty()) { |
|||
result.Add(_ReadingOrderDirectionalGroupData(currentGroup)); |
|||
} |
|||
|
|||
foreach ( _ReadingOrderDirectionalGroupData bandGroup in result) { |
|||
if (bandGroup.members.length == 1) { |
|||
continue; |
|||
} |
|||
_ReadingOrderSortData.sortWithDirectionality(bandGroup.members, bandGroup.directionality); |
|||
} |
|||
return result; |
|||
} |
|||
public _ReadingOrderSortData _pickNext(List<_ReadingOrderSortData> candidates) { |
|||
|
|||
mergeSort<_ReadingOrderSortData>(candidates, compare: (_ReadingOrderSortData a, _ReadingOrderSortData b) => a.rect.top.compareTo(b.rect.top)); |
|||
_ReadingOrderSortData topmost = candidates.First(); |
|||
|
|||
List<_ReadingOrderSortData> inBand(_ReadingOrderSortData current, IEnumerable<_ReadingOrderSortData> candidates) { |
|||
Rect band = Rect.fromLTRB(double.negativeInfinity, current.rect.top, double.infinity, current.rect.bottom); |
|||
return candidates.Where((_ReadingOrderSortData item)=> { |
|||
return !item.rect.intersect(band).isEmpty; |
|||
}).ToList(); |
|||
} |
|||
List<_ReadingOrderSortData> inBandOfTop = inBand(topmost, candidates); |
|||
D.assert(topmost.rect.isEmpty || inBandOfTop.isNotEmpty); |
|||
if (inBandOfTop.Count <= 1) { |
|||
return topmost; |
|||
} |
|||
TextDirection nearestCommonDirectionality = _ReadingOrderSortData.commonDirectionalityOf(inBandOfTop); |
|||
_ReadingOrderSortData.sortWithDirectionality(inBandOfTop, nearestCommonDirectionality); |
|||
List<_ReadingOrderDirectionalGroupData> bandGroups = _collectDirectionalityGroups(inBandOfTop); |
|||
if (bandGroups.Count == 1) { |
|||
return bandGroups.First().members.first; |
|||
} |
|||
_ReadingOrderDirectionalGroupData.sortWithDirectionality(bandGroups, nearestCommonDirectionality); |
|||
return bandGroups.First().members.first; |
|||
} |
|||
|
|||
public override IEnumerable<FocusNode> sortDescendants(IEnumerable<FocusNode> descendants) { |
|||
D.assert(descendants != null); |
|||
if (descendants.Count() <= 1) { |
|||
return descendants; |
|||
} |
|||
List<_ReadingOrderSortData> data = new List<_ReadingOrderSortData>(){ |
|||
foreach ( FocusNode node in descendants) |
|||
_ReadingOrderSortData(node), |
|||
}; |
|||
List<FocusNode> sortedList = new List<FocusNode>(); |
|||
List<_ReadingOrderSortData> unplaced = data; |
|||
_ReadingOrderSortData current = _pickNext(unplaced); |
|||
sortedList.Add(current.node); |
|||
unplaced.Remove(current); |
|||
while (unplaced.isNotEmpty()) { |
|||
_ReadingOrderSortData next = _pickNext(unplaced); |
|||
current = next; |
|||
sortedList.Add(current.node); |
|||
unplaced.Remove(current); |
|||
} |
|||
return sortedList; |
|||
} |
|||
}*/ |
|||
|
|||
|
|||
} |
撰写
预览
正在加载...
取消
保存
Reference in new issue