您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
1480 行
54 KiB
1480 行
54 KiB
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Unity.UIWidgets.animation;
|
|
using Unity.UIWidgets.async;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.gestures;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.physics;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.scheduler;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
using Color = Unity.UIWidgets.ui.Color;
|
|
|
|
namespace Unity.UIWidgets.widgets {
|
|
public delegate List<Widget> NestedScrollViewHeaderSliversBuilder(BuildContext context, bool innerBoxIsScrolled);
|
|
|
|
public class NestedScrollView : StatefulWidget {
|
|
public NestedScrollView(
|
|
Key key = null,
|
|
ScrollController controller = null,
|
|
Axis scrollDirection = Axis.vertical,
|
|
bool reverse = false,
|
|
ScrollPhysics physics = null,
|
|
NestedScrollViewHeaderSliversBuilder headerSliverBuilder = null,
|
|
Widget body = null,
|
|
DragStartBehavior dragStartBehavior = DragStartBehavior.start
|
|
) : base(key: key) {
|
|
D.assert(headerSliverBuilder != null);
|
|
D.assert(body != null);
|
|
this.controller = controller;
|
|
this.scrollDirection = scrollDirection;
|
|
this.reverse = reverse;
|
|
this.physics = physics;
|
|
this.headerSliverBuilder = headerSliverBuilder;
|
|
this.body = body;
|
|
this.dragStartBehavior = dragStartBehavior;
|
|
}
|
|
|
|
public readonly ScrollController controller;
|
|
|
|
public readonly Axis scrollDirection;
|
|
|
|
public readonly bool reverse;
|
|
|
|
public readonly ScrollPhysics physics;
|
|
|
|
public readonly NestedScrollViewHeaderSliversBuilder headerSliverBuilder;
|
|
|
|
public readonly Widget body;
|
|
|
|
public readonly DragStartBehavior dragStartBehavior;
|
|
|
|
public static SliverOverlapAbsorberHandle sliverOverlapAbsorberHandleFor(BuildContext context) {
|
|
_InheritedNestedScrollView target = context.dependOnInheritedWidgetOfExactType<_InheritedNestedScrollView>();
|
|
D.assert(target != null,
|
|
() => "NestedScrollView.sliverOverlapAbsorberHandleFor must be called with a context that contains a NestedScrollView."
|
|
);
|
|
return target.state._absorberHandle;
|
|
}
|
|
|
|
internal List<Widget> _buildSlivers(BuildContext context, ScrollController innerController,
|
|
bool bodyIsScrolled) {
|
|
List<Widget> slivers = new List<Widget>();
|
|
slivers.AddRange(headerSliverBuilder(context, bodyIsScrolled));
|
|
slivers.Add(new SliverFillRemaining(
|
|
child: new PrimaryScrollController(
|
|
controller: innerController,
|
|
child: body
|
|
)
|
|
));
|
|
return slivers;
|
|
}
|
|
|
|
public override State createState() {
|
|
return new NestedScrollViewState();
|
|
}
|
|
}
|
|
|
|
class NestedScrollViewState : State<NestedScrollView> {
|
|
internal readonly SliverOverlapAbsorberHandle _absorberHandle = new SliverOverlapAbsorberHandle();
|
|
|
|
ScrollController innerController => _coordinator._innerController;
|
|
|
|
ScrollController outerController => _coordinator._outerController;
|
|
|
|
_NestedScrollCoordinator _coordinator;
|
|
|
|
public override void initState() {
|
|
base.initState();
|
|
_coordinator =
|
|
new _NestedScrollCoordinator(this, widget.controller, _handleHasScrolledBodyChanged);
|
|
}
|
|
|
|
public override void didChangeDependencies() {
|
|
base.didChangeDependencies();
|
|
_coordinator.setParent(widget.controller);
|
|
}
|
|
|
|
public override void didUpdateWidget(StatefulWidget _oldWidget) {
|
|
NestedScrollView oldWidget = _oldWidget as NestedScrollView;
|
|
base.didUpdateWidget(oldWidget);
|
|
if (oldWidget.controller != widget.controller) {
|
|
_coordinator.setParent(widget.controller);
|
|
}
|
|
}
|
|
|
|
public override void dispose() {
|
|
_coordinator.dispose();
|
|
_coordinator = null;
|
|
base.dispose();
|
|
}
|
|
|
|
bool _lastHasScrolledBody;
|
|
|
|
void _handleHasScrolledBodyChanged() {
|
|
if (!mounted) {
|
|
return;
|
|
}
|
|
|
|
bool newHasScrolledBody = _coordinator.hasScrolledBody;
|
|
if (_lastHasScrolledBody != newHasScrolledBody) {
|
|
setState(() => { });
|
|
}
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
return new _InheritedNestedScrollView(
|
|
state: this,
|
|
child: new Builder(
|
|
builder: (BuildContext _context) => {
|
|
_lastHasScrolledBody = _coordinator.hasScrolledBody;
|
|
return new _NestedScrollViewCustomScrollView(
|
|
dragStartBehavior: widget.dragStartBehavior,
|
|
scrollDirection: widget.scrollDirection,
|
|
reverse: widget.reverse,
|
|
physics: widget.physics != null
|
|
? widget.physics.applyTo(new ClampingScrollPhysics())
|
|
: new ClampingScrollPhysics(),
|
|
controller: _coordinator._outerController,
|
|
slivers: widget._buildSlivers(
|
|
_context,
|
|
_coordinator._innerController,
|
|
_lastHasScrolledBody
|
|
),
|
|
handle: _absorberHandle
|
|
);
|
|
}
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
class _NestedScrollViewCustomScrollView : CustomScrollView {
|
|
public _NestedScrollViewCustomScrollView(
|
|
Axis scrollDirection,
|
|
bool reverse,
|
|
ScrollPhysics physics,
|
|
ScrollController controller,
|
|
List<Widget> slivers,
|
|
SliverOverlapAbsorberHandle handle,
|
|
DragStartBehavior dragStartBehavior = DragStartBehavior.start
|
|
) : base(
|
|
scrollDirection: scrollDirection,
|
|
reverse: reverse,
|
|
physics: physics,
|
|
controller: controller,
|
|
slivers: slivers,
|
|
dragStartBehavior: dragStartBehavior
|
|
) {
|
|
this.handle = handle;
|
|
}
|
|
|
|
public readonly SliverOverlapAbsorberHandle handle;
|
|
|
|
protected override Widget buildViewport(
|
|
BuildContext context,
|
|
ViewportOffset offset,
|
|
AxisDirection? axisDirection,
|
|
List<Widget> slivers
|
|
) {
|
|
D.assert(!shrinkWrap);
|
|
return new NestedScrollViewViewport(
|
|
axisDirection: axisDirection,
|
|
offset: offset,
|
|
slivers: slivers,
|
|
handle: handle
|
|
);
|
|
}
|
|
}
|
|
|
|
class _InheritedNestedScrollView : InheritedWidget {
|
|
public _InheritedNestedScrollView(
|
|
Key key = null,
|
|
NestedScrollViewState state = null,
|
|
Widget child = null
|
|
) : base(key: key, child: child) {
|
|
D.assert(state != null);
|
|
D.assert(child != null);
|
|
this.state = state;
|
|
}
|
|
|
|
public readonly NestedScrollViewState state;
|
|
|
|
public override bool updateShouldNotify(InheritedWidget _old) {
|
|
_InheritedNestedScrollView old = _old as _InheritedNestedScrollView;
|
|
return state != old.state;
|
|
}
|
|
}
|
|
|
|
class _NestedScrollMetrics : FixedScrollMetrics {
|
|
public _NestedScrollMetrics(
|
|
float minScrollExtent,
|
|
float maxScrollExtent,
|
|
float pixels,
|
|
float viewportDimension,
|
|
AxisDirection axisDirection,
|
|
float minRange,
|
|
float maxRange,
|
|
float correctionOffset
|
|
) : base(
|
|
minScrollExtent: minScrollExtent,
|
|
maxScrollExtent: maxScrollExtent,
|
|
pixels: pixels,
|
|
viewportDimension: viewportDimension,
|
|
axisDirection: axisDirection
|
|
) {
|
|
this.minRange = minRange;
|
|
this.maxRange = maxRange;
|
|
this.correctionOffset = correctionOffset;
|
|
}
|
|
|
|
public _NestedScrollMetrics copyWith(
|
|
float? minScrollExtent = null,
|
|
float? maxScrollExtent = null,
|
|
float? pixels = null,
|
|
float? viewportDimension = null,
|
|
AxisDirection? axisDirection = null,
|
|
float? minRange = null,
|
|
float? maxRange = null,
|
|
float? correctionOffset = null
|
|
) {
|
|
return new _NestedScrollMetrics(
|
|
minScrollExtent: minScrollExtent ?? this.minScrollExtent,
|
|
maxScrollExtent: maxScrollExtent ?? this.maxScrollExtent,
|
|
pixels: pixels ?? this.pixels,
|
|
viewportDimension: viewportDimension ?? this.viewportDimension,
|
|
axisDirection: axisDirection ?? this.axisDirection,
|
|
minRange: minRange ?? this.minRange,
|
|
maxRange: maxRange ?? this.maxRange,
|
|
correctionOffset: correctionOffset ?? this.correctionOffset
|
|
);
|
|
}
|
|
|
|
public readonly float minRange;
|
|
|
|
public readonly float maxRange;
|
|
|
|
public readonly float correctionOffset;
|
|
}
|
|
|
|
delegate ScrollActivity _NestedScrollActivityGetter(_NestedScrollPosition position);
|
|
|
|
class _NestedScrollCoordinator : ScrollActivityDelegate, ScrollHoldController {
|
|
public _NestedScrollCoordinator(
|
|
NestedScrollViewState _state,
|
|
ScrollController _parent,
|
|
VoidCallback _onHasScrolledBodyChanged) {
|
|
float initialScrollOffset = _parent?.initialScrollOffset ?? 0.0f;
|
|
_outerController =
|
|
new _NestedScrollController(this, initialScrollOffset: initialScrollOffset, debugLabel: "outer");
|
|
_innerController = new _NestedScrollController(this, initialScrollOffset: 0.0f, debugLabel: "inner");
|
|
this._state = _state;
|
|
this._parent = _parent;
|
|
this._onHasScrolledBodyChanged = _onHasScrolledBodyChanged;
|
|
}
|
|
|
|
public readonly NestedScrollViewState _state;
|
|
ScrollController _parent;
|
|
public readonly VoidCallback _onHasScrolledBodyChanged;
|
|
|
|
internal _NestedScrollController _outerController;
|
|
internal _NestedScrollController _innerController;
|
|
|
|
_NestedScrollPosition _outerPosition {
|
|
get {
|
|
if (!_outerController.hasClients) {
|
|
return null;
|
|
}
|
|
return _outerController.nestedPositions.Single();
|
|
}
|
|
}
|
|
|
|
IEnumerable<_NestedScrollPosition> _innerPositions {
|
|
get { return _innerController.nestedPositions; }
|
|
}
|
|
|
|
public bool canScrollBody {
|
|
get {
|
|
_NestedScrollPosition outer = _outerPosition;
|
|
if (outer == null) {
|
|
return true;
|
|
}
|
|
|
|
return outer.haveDimensions && outer.extentAfter() == 0.0f;
|
|
}
|
|
}
|
|
|
|
public bool hasScrolledBody {
|
|
get {
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
if (position.pixels > position.minScrollExtent) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public void updateShadow() {
|
|
if (_onHasScrolledBodyChanged != null) {
|
|
_onHasScrolledBodyChanged();
|
|
}
|
|
}
|
|
|
|
public ScrollDirection userScrollDirection {
|
|
get { return _userScrollDirection; }
|
|
}
|
|
|
|
ScrollDirection _userScrollDirection = ScrollDirection.idle;
|
|
|
|
public void updateUserScrollDirection(ScrollDirection value) {
|
|
if (userScrollDirection == value) {
|
|
return;
|
|
}
|
|
_userScrollDirection = value;
|
|
_outerPosition.didUpdateScrollDirection(value);
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
position.didUpdateScrollDirection(value);
|
|
}
|
|
}
|
|
|
|
ScrollDragController _currentDrag;
|
|
|
|
public void beginActivity(ScrollActivity newOuterActivity, _NestedScrollActivityGetter innerActivityGetter) {
|
|
_outerPosition.beginActivity(newOuterActivity);
|
|
bool scrolling = newOuterActivity.isScrolling;
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
ScrollActivity newInnerActivity = innerActivityGetter(position);
|
|
position.beginActivity(newInnerActivity);
|
|
scrolling = scrolling && newInnerActivity.isScrolling;
|
|
}
|
|
|
|
_currentDrag?.dispose();
|
|
_currentDrag = null;
|
|
if (!scrolling) {
|
|
updateUserScrollDirection(ScrollDirection.idle);
|
|
}
|
|
}
|
|
|
|
public AxisDirection axisDirection {
|
|
get { return _outerPosition.axisDirection; }
|
|
}
|
|
|
|
static IdleScrollActivity _createIdleScrollActivity(_NestedScrollPosition position) {
|
|
return new IdleScrollActivity(position);
|
|
}
|
|
|
|
public void goIdle() {
|
|
beginActivity(_createIdleScrollActivity(_outerPosition), _createIdleScrollActivity);
|
|
}
|
|
|
|
public void goBallistic(float velocity) {
|
|
beginActivity(
|
|
createOuterBallisticScrollActivity(velocity),
|
|
(_NestedScrollPosition position) => createInnerBallisticScrollActivity(position, velocity)
|
|
);
|
|
}
|
|
|
|
public ScrollActivity createOuterBallisticScrollActivity(float velocity) {
|
|
_NestedScrollPosition innerPosition = null;
|
|
if (velocity != 0.0f) {
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
if (innerPosition != null) {
|
|
if (velocity > 0.0f) {
|
|
if (innerPosition.pixels < position.pixels) {
|
|
continue;
|
|
}
|
|
}
|
|
else {
|
|
D.assert(velocity < 0.0f);
|
|
if (innerPosition.pixels > position.pixels) {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
innerPosition = position;
|
|
}
|
|
}
|
|
|
|
if (innerPosition == null) {
|
|
return _outerPosition.createBallisticScrollActivity(
|
|
_outerPosition.physics.createBallisticSimulation(_outerPosition, velocity),
|
|
mode: _NestedBallisticScrollActivityMode.independent
|
|
);
|
|
}
|
|
|
|
_NestedScrollMetrics metrics = _getMetrics(innerPosition, velocity);
|
|
|
|
return _outerPosition.createBallisticScrollActivity(
|
|
_outerPosition.physics.createBallisticSimulation(metrics, velocity),
|
|
mode: _NestedBallisticScrollActivityMode.outer,
|
|
metrics: metrics
|
|
);
|
|
}
|
|
|
|
protected internal ScrollActivity createInnerBallisticScrollActivity(_NestedScrollPosition position,
|
|
float velocity) {
|
|
return position.createBallisticScrollActivity(
|
|
position.physics.createBallisticSimulation(
|
|
velocity == 0f ? (ScrollMetrics) position : _getMetrics(position, velocity),
|
|
velocity
|
|
),
|
|
mode: _NestedBallisticScrollActivityMode.inner
|
|
);
|
|
}
|
|
|
|
_NestedScrollMetrics _getMetrics(_NestedScrollPosition innerPosition, float velocity) {
|
|
D.assert(innerPosition != null);
|
|
float pixels, minRange, maxRange, correctionOffset, extra;
|
|
if (innerPosition.pixels == innerPosition.minScrollExtent) {
|
|
pixels = _outerPosition.pixels.clamp(_outerPosition.minScrollExtent,
|
|
_outerPosition.maxScrollExtent); // TODO(ianh): gracefully handle out-of-range outer positions
|
|
minRange = _outerPosition.minScrollExtent;
|
|
maxRange = _outerPosition.maxScrollExtent;
|
|
D.assert(minRange <= maxRange);
|
|
correctionOffset = 0.0f;
|
|
extra = 0.0f;
|
|
}
|
|
else {
|
|
D.assert(innerPosition.pixels != innerPosition.minScrollExtent);
|
|
if (innerPosition.pixels < innerPosition.minScrollExtent) {
|
|
pixels = innerPosition.pixels - innerPosition.minScrollExtent + _outerPosition.minScrollExtent;
|
|
}
|
|
else {
|
|
D.assert(innerPosition.pixels > innerPosition.minScrollExtent);
|
|
pixels = innerPosition.pixels - innerPosition.minScrollExtent + _outerPosition.maxScrollExtent;
|
|
}
|
|
|
|
if ((velocity > 0.0f) && (innerPosition.pixels > innerPosition.minScrollExtent)) {
|
|
extra = _outerPosition.maxScrollExtent - _outerPosition.pixels;
|
|
D.assert(extra >= 0.0f);
|
|
minRange = pixels;
|
|
maxRange = pixels + extra;
|
|
D.assert(minRange <= maxRange);
|
|
correctionOffset = _outerPosition.pixels - pixels;
|
|
}
|
|
else if ((velocity < 0.0f) && (innerPosition.pixels < innerPosition.minScrollExtent)) {
|
|
extra = _outerPosition.pixels - _outerPosition.minScrollExtent;
|
|
D.assert(extra >= 0.0f);
|
|
minRange = pixels - extra;
|
|
maxRange = pixels;
|
|
D.assert(minRange <= maxRange);
|
|
correctionOffset = _outerPosition.pixels - pixels;
|
|
}
|
|
else {
|
|
if (velocity > 0.0f) {
|
|
extra = _outerPosition.minScrollExtent - _outerPosition.pixels;
|
|
}
|
|
else {
|
|
D.assert(velocity < 0.0f);
|
|
extra = _outerPosition.pixels -
|
|
(_outerPosition.maxScrollExtent - _outerPosition.minScrollExtent);
|
|
}
|
|
|
|
D.assert(extra <= 0.0f);
|
|
minRange = _outerPosition.minScrollExtent;
|
|
maxRange = _outerPosition.maxScrollExtent + extra;
|
|
D.assert(minRange <= maxRange);
|
|
correctionOffset = 0.0f;
|
|
}
|
|
}
|
|
|
|
return new _NestedScrollMetrics(
|
|
minScrollExtent: _outerPosition.minScrollExtent,
|
|
maxScrollExtent: _outerPosition.maxScrollExtent + innerPosition.maxScrollExtent -
|
|
innerPosition.minScrollExtent + extra,
|
|
pixels: pixels,
|
|
viewportDimension: _outerPosition.viewportDimension,
|
|
axisDirection: _outerPosition.axisDirection,
|
|
minRange: minRange,
|
|
maxRange: maxRange,
|
|
correctionOffset: correctionOffset
|
|
);
|
|
}
|
|
|
|
public float unnestOffset(float value, _NestedScrollPosition source) {
|
|
if (source == _outerPosition) {
|
|
return value.clamp(_outerPosition.minScrollExtent, _outerPosition.maxScrollExtent);
|
|
}
|
|
|
|
if (value < source.minScrollExtent) {
|
|
return value - source.minScrollExtent + _outerPosition.minScrollExtent;
|
|
}
|
|
|
|
return value - source.minScrollExtent + _outerPosition.maxScrollExtent;
|
|
}
|
|
|
|
public float nestOffset(float value, _NestedScrollPosition target) {
|
|
if (target == _outerPosition) {
|
|
return value.clamp(_outerPosition.minScrollExtent, _outerPosition.maxScrollExtent);
|
|
}
|
|
|
|
if (value < _outerPosition.minScrollExtent) {
|
|
return value - _outerPosition.minScrollExtent + target.minScrollExtent;
|
|
}
|
|
|
|
if (value > _outerPosition.maxScrollExtent) {
|
|
return value - _outerPosition.maxScrollExtent + target.minScrollExtent;
|
|
}
|
|
|
|
return target.minScrollExtent;
|
|
}
|
|
|
|
public void updateCanDrag() {
|
|
if (!_outerPosition.haveDimensions) {
|
|
return;
|
|
}
|
|
|
|
float maxInnerExtent = 0.0f;
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
if (!position.haveDimensions) {
|
|
return;
|
|
}
|
|
|
|
maxInnerExtent = Mathf.Max(maxInnerExtent, position.maxScrollExtent - position.minScrollExtent);
|
|
}
|
|
|
|
_outerPosition.updateCanDrag(maxInnerExtent);
|
|
}
|
|
|
|
public Future animateTo(
|
|
float to,
|
|
TimeSpan duration,
|
|
Curve curve
|
|
) {
|
|
DrivenScrollActivity outerActivity = _outerPosition.createDrivenScrollActivity(
|
|
nestOffset(to, _outerPosition),
|
|
duration,
|
|
curve
|
|
);
|
|
List<Future> resultFutures = new List<Future> {outerActivity.done};
|
|
beginActivity(
|
|
outerActivity,
|
|
(_NestedScrollPosition position) => {
|
|
DrivenScrollActivity innerActivity = position.createDrivenScrollActivity(
|
|
nestOffset(to, position),
|
|
duration,
|
|
curve
|
|
);
|
|
resultFutures.Add(innerActivity.done);
|
|
return innerActivity;
|
|
}
|
|
);
|
|
return Future.wait<object>(resultFutures);
|
|
}
|
|
|
|
public void jumpTo(float to) {
|
|
goIdle();
|
|
_outerPosition.localJumpTo(nestOffset(to, _outerPosition));
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
position.localJumpTo(nestOffset(to, position));
|
|
}
|
|
|
|
goBallistic(0.0f);
|
|
}
|
|
|
|
public float setPixels(float newPixels) {
|
|
D.assert(false);
|
|
return 0.0f;
|
|
}
|
|
|
|
public ScrollHoldController hold(VoidCallback holdCancelCallback) {
|
|
beginActivity(
|
|
new HoldScrollActivity(del: _outerPosition, onHoldCanceled: holdCancelCallback),
|
|
(_NestedScrollPosition position) => new HoldScrollActivity(del: position)
|
|
);
|
|
return this;
|
|
}
|
|
|
|
public void cancel() {
|
|
goBallistic(0.0f);
|
|
}
|
|
|
|
public Drag drag(DragStartDetails details, VoidCallback dragCancelCallback) {
|
|
ScrollDragController drag = new ScrollDragController(
|
|
del: this,
|
|
details: details,
|
|
onDragCanceled: dragCancelCallback
|
|
);
|
|
beginActivity(
|
|
new DragScrollActivity(_outerPosition, drag),
|
|
(_NestedScrollPosition position) => new DragScrollActivity(position, drag)
|
|
);
|
|
D.assert(_currentDrag == null);
|
|
_currentDrag = drag;
|
|
return drag;
|
|
}
|
|
|
|
public void applyUserOffset(float delta) {
|
|
updateUserScrollDirection(delta > 0.0f ? ScrollDirection.forward : ScrollDirection.reverse);
|
|
D.assert(delta != 0.0f);
|
|
if (!_innerPositions.Any()) {
|
|
_outerPosition.applyFullDragUpdate(delta);
|
|
}
|
|
else if (delta < 0.0f) {
|
|
float innerDelta = _outerPosition.applyClampedDragUpdate(delta);
|
|
if (innerDelta != 0.0f) {
|
|
foreach (_NestedScrollPosition position in _innerPositions) {
|
|
position.applyFullDragUpdate(innerDelta);
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
float outerDelta = 0.0f; // it will go positive if it changes
|
|
List<float> overscrolls = new List<float> { };
|
|
List<_NestedScrollPosition> innerPositions = _innerPositions.ToList();
|
|
foreach (_NestedScrollPosition position in innerPositions) {
|
|
float overscroll = position.applyClampedDragUpdate(delta);
|
|
outerDelta = Mathf.Max(outerDelta, overscroll);
|
|
overscrolls.Add(overscroll);
|
|
}
|
|
|
|
if (outerDelta != 0.0f) {
|
|
outerDelta -= _outerPosition.applyClampedDragUpdate(outerDelta);
|
|
}
|
|
|
|
for (int i = 0; i < innerPositions.Count; ++i) {
|
|
float remainingDelta = overscrolls[i] - outerDelta;
|
|
if (remainingDelta > 0.0f) {
|
|
innerPositions[i].applyFullDragUpdate(remainingDelta);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public void applyUserScrollOffset(float delta) {
|
|
// TODO: replace with real implementation
|
|
applyUserOffset(delta);
|
|
}
|
|
|
|
public void setParent(ScrollController value) {
|
|
_parent = value;
|
|
updateParent();
|
|
}
|
|
|
|
public void updateParent() {
|
|
_outerPosition?.setParent(_parent ?? PrimaryScrollController.of(_state.context));
|
|
}
|
|
|
|
public void dispose() {
|
|
_currentDrag?.dispose();
|
|
_currentDrag = null;
|
|
_outerController.dispose();
|
|
_innerController.dispose();
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"{GetType()}(outer={_outerController}; inner={_innerController})";
|
|
}
|
|
}
|
|
|
|
class _NestedScrollController : ScrollController {
|
|
public _NestedScrollController(
|
|
_NestedScrollCoordinator coordinator,
|
|
float initialScrollOffset = 0.0f,
|
|
string debugLabel = null
|
|
) : base(initialScrollOffset: initialScrollOffset, debugLabel: debugLabel) {
|
|
this.coordinator = coordinator;
|
|
}
|
|
|
|
public readonly _NestedScrollCoordinator coordinator;
|
|
|
|
public override ScrollPosition createScrollPosition(
|
|
ScrollPhysics physics,
|
|
ScrollContext context,
|
|
ScrollPosition oldPosition
|
|
) {
|
|
return new _NestedScrollPosition(
|
|
coordinator: coordinator,
|
|
physics: physics,
|
|
context: context,
|
|
initialPixels: initialScrollOffset,
|
|
oldPosition: oldPosition,
|
|
debugLabel: debugLabel
|
|
);
|
|
}
|
|
|
|
public override void attach(ScrollPosition position) {
|
|
D.assert(position is _NestedScrollPosition);
|
|
base.attach(position);
|
|
coordinator.updateParent();
|
|
coordinator.updateCanDrag();
|
|
position.addListener(_scheduleUpdateShadow);
|
|
_scheduleUpdateShadow();
|
|
}
|
|
|
|
public override void detach(ScrollPosition position) {
|
|
D.assert(position is _NestedScrollPosition);
|
|
position.removeListener(_scheduleUpdateShadow);
|
|
base.detach(position);
|
|
_scheduleUpdateShadow();
|
|
}
|
|
|
|
void _scheduleUpdateShadow() {
|
|
SchedulerBinding.instance.addPostFrameCallback(
|
|
(TimeSpan timeStamp) => { coordinator.updateShadow(); }
|
|
);
|
|
}
|
|
|
|
public IEnumerable<_NestedScrollPosition> nestedPositions {
|
|
get {
|
|
foreach (var scrollPosition in positions) {
|
|
yield return (_NestedScrollPosition) scrollPosition;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
class _NestedScrollPosition : ScrollPosition, ScrollActivityDelegate {
|
|
public _NestedScrollPosition(
|
|
ScrollPhysics physics,
|
|
ScrollContext context,
|
|
float initialPixels = 0.0f,
|
|
ScrollPosition oldPosition = null,
|
|
string debugLabel = null,
|
|
_NestedScrollCoordinator coordinator = null
|
|
) : base(
|
|
physics: physics,
|
|
context: context,
|
|
oldPosition: oldPosition,
|
|
debugLabel: debugLabel,
|
|
coordinator: coordinator
|
|
) {
|
|
D.assert(coordinator != null);
|
|
if (!havePixels) {
|
|
correctPixels(initialPixels);
|
|
}
|
|
|
|
if (activity == null) {
|
|
goIdle();
|
|
}
|
|
|
|
D.assert(activity != null);
|
|
saveScrollOffset(); // in case we didn't restore but could, so that we don't restore it later
|
|
}
|
|
|
|
public _NestedScrollCoordinator coordinator {
|
|
get { return (_NestedScrollCoordinator) _coordinator; }
|
|
}
|
|
|
|
public TickerProvider vsync {
|
|
get { return context.vsync; }
|
|
}
|
|
|
|
ScrollController _parent;
|
|
|
|
public void setParent(ScrollController value) {
|
|
_parent?.detach(this);
|
|
_parent = value;
|
|
_parent?.attach(this);
|
|
}
|
|
|
|
public override AxisDirection axisDirection {
|
|
get { return context.axisDirection; }
|
|
}
|
|
|
|
protected override void absorb(ScrollPosition other) {
|
|
base.absorb(other);
|
|
activity.updateDelegate(this);
|
|
}
|
|
|
|
protected override void restoreScrollOffset() {
|
|
if (coordinator.canScrollBody) {
|
|
base.restoreScrollOffset();
|
|
}
|
|
}
|
|
|
|
public float applyClampedDragUpdate(float delta) {
|
|
D.assert(delta != 0.0f);
|
|
float min = delta < 0.0f ? -float.PositiveInfinity : Mathf.Min(minScrollExtent, pixels);
|
|
float max = delta > 0.0f ? float.PositiveInfinity : Mathf.Max(maxScrollExtent, pixels);
|
|
float oldPixels = pixels;
|
|
float newPixels = (pixels - delta).clamp(min, max);
|
|
float clampedDelta = newPixels - pixels;
|
|
if (clampedDelta == 0.0f) {
|
|
return delta;
|
|
}
|
|
|
|
float overscroll = physics.applyBoundaryConditions(this, newPixels);
|
|
float actualNewPixels = newPixels - overscroll;
|
|
float offset = actualNewPixels - oldPixels;
|
|
if (offset != 0.0f) {
|
|
forcePixels(actualNewPixels);
|
|
didUpdateScrollPositionBy(offset);
|
|
}
|
|
|
|
return delta + offset;
|
|
}
|
|
|
|
public float applyFullDragUpdate(float delta) {
|
|
D.assert(delta != 0.0f);
|
|
float oldPixels = pixels;
|
|
float newPixels = pixels - physics.applyPhysicsToUserOffset(this, delta);
|
|
if (oldPixels == newPixels) {
|
|
return 0.0f; // delta must have been so small we dropped it during floating point addition
|
|
}
|
|
|
|
float overscroll = physics.applyBoundaryConditions(this, newPixels);
|
|
float actualNewPixels = newPixels - overscroll;
|
|
if (actualNewPixels != oldPixels) {
|
|
forcePixels(actualNewPixels);
|
|
didUpdateScrollPositionBy(actualNewPixels - oldPixels);
|
|
}
|
|
|
|
if (overscroll != 0.0f) {
|
|
didOverscrollBy(overscroll);
|
|
return overscroll;
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
public float applyClampedPointerSignalUpdate(float delta) {
|
|
D.assert(delta != 0.0f);
|
|
|
|
float min = delta > 0.0f
|
|
? float.NegativeInfinity
|
|
: Mathf.Min(minScrollExtent, pixels);
|
|
// The logic for max is equivalent but on the other side.
|
|
float max = delta < 0.0f
|
|
? float.PositiveInfinity
|
|
: Mathf.Max(maxScrollExtent, pixels);
|
|
float newPixels = (pixels + delta).clamp(min, max);
|
|
float clampedDelta = newPixels - pixels;
|
|
if (clampedDelta == 0.0f)
|
|
return delta;
|
|
forcePixels(newPixels);
|
|
didUpdateScrollPositionBy(clampedDelta);
|
|
return delta - clampedDelta;
|
|
}
|
|
|
|
public override ScrollDirection userScrollDirection {
|
|
get { return coordinator.userScrollDirection; }
|
|
}
|
|
|
|
public DrivenScrollActivity createDrivenScrollActivity(float to, TimeSpan duration, Curve curve) {
|
|
return new DrivenScrollActivity(
|
|
this,
|
|
from: pixels,
|
|
to: to,
|
|
duration: duration,
|
|
curve: curve,
|
|
vsync: vsync
|
|
);
|
|
}
|
|
|
|
public void applyUserOffset(float delta) {
|
|
D.assert(false);
|
|
}
|
|
|
|
public void applyUserScrollOffset(float delta) {
|
|
// TODO: replace with real implementation
|
|
applyUserOffset(delta);
|
|
}
|
|
|
|
public void goIdle() {
|
|
beginActivity(new IdleScrollActivity(this));
|
|
}
|
|
|
|
public void goBallistic(float velocity) {
|
|
Simulation simulation = null;
|
|
if (velocity != 0.0f || this.outOfRange()) {
|
|
simulation = physics.createBallisticSimulation(this, velocity);
|
|
}
|
|
|
|
beginActivity(createBallisticScrollActivity(
|
|
simulation,
|
|
mode: _NestedBallisticScrollActivityMode.independent
|
|
));
|
|
}
|
|
|
|
public ScrollActivity createBallisticScrollActivity(
|
|
Simulation simulation = null,
|
|
_NestedBallisticScrollActivityMode? mode = null,
|
|
_NestedScrollMetrics metrics = null
|
|
) {
|
|
if (simulation == null) {
|
|
return new IdleScrollActivity(this);
|
|
}
|
|
|
|
D.assert(mode != null);
|
|
switch (mode) {
|
|
case _NestedBallisticScrollActivityMode.outer:
|
|
D.assert(metrics != null);
|
|
if (metrics.minRange == metrics.maxRange) {
|
|
return new IdleScrollActivity(this);
|
|
}
|
|
|
|
return new _NestedOuterBallisticScrollActivity(coordinator, this, metrics, simulation,
|
|
context.vsync);
|
|
case _NestedBallisticScrollActivityMode.inner:
|
|
return new _NestedInnerBallisticScrollActivity(coordinator, this, simulation,
|
|
context.vsync);
|
|
case _NestedBallisticScrollActivityMode.independent:
|
|
return new BallisticScrollActivity(this, simulation, context.vsync);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public override Future animateTo(float to,
|
|
TimeSpan duration,
|
|
Curve curve
|
|
) {
|
|
return coordinator.animateTo(coordinator.unnestOffset(to, this), duration: duration,
|
|
curve: curve);
|
|
}
|
|
|
|
public override void jumpTo(float value) {
|
|
coordinator.jumpTo(coordinator.unnestOffset(value, this));
|
|
}
|
|
|
|
public void localJumpTo(float value) {
|
|
if (pixels != value) {
|
|
float oldPixels = pixels;
|
|
forcePixels(value);
|
|
didStartScroll();
|
|
didUpdateScrollPositionBy(pixels - oldPixels);
|
|
didEndScroll();
|
|
}
|
|
}
|
|
|
|
protected override void applyNewDimensions() {
|
|
base.applyNewDimensions();
|
|
coordinator.updateCanDrag();
|
|
}
|
|
|
|
public void updateCanDrag(float totalExtent) {
|
|
context.setCanDrag(totalExtent > (viewportDimension - maxScrollExtent) ||
|
|
minScrollExtent != maxScrollExtent);
|
|
}
|
|
|
|
public override ScrollHoldController hold(VoidCallback holdCancelCallback) {
|
|
return coordinator.hold(holdCancelCallback);
|
|
}
|
|
|
|
public override Drag drag(DragStartDetails details, VoidCallback dragCancelCallback) {
|
|
return coordinator.drag(details, dragCancelCallback);
|
|
}
|
|
|
|
public override void dispose() {
|
|
_parent?.detach(this);
|
|
base.dispose();
|
|
}
|
|
}
|
|
|
|
enum _NestedBallisticScrollActivityMode {
|
|
outer,
|
|
inner,
|
|
independent
|
|
}
|
|
|
|
class _NestedInnerBallisticScrollActivity : BallisticScrollActivity {
|
|
public _NestedInnerBallisticScrollActivity(
|
|
_NestedScrollCoordinator coordinator,
|
|
_NestedScrollPosition position,
|
|
Simulation simulation,
|
|
TickerProvider vsync
|
|
) : base(position, simulation, vsync) {
|
|
this.coordinator = coordinator;
|
|
}
|
|
|
|
public readonly _NestedScrollCoordinator coordinator;
|
|
|
|
public new _NestedScrollPosition del {
|
|
get { return (_NestedScrollPosition) base.del; }
|
|
}
|
|
|
|
public override void resetActivity() {
|
|
del.beginActivity(coordinator.createInnerBallisticScrollActivity(del, velocity));
|
|
}
|
|
|
|
public override void applyNewDimensions() {
|
|
del.beginActivity(coordinator.createInnerBallisticScrollActivity(del, velocity));
|
|
}
|
|
|
|
protected override bool applyMoveTo(float value) {
|
|
return base.applyMoveTo(coordinator.nestOffset(value, del));
|
|
}
|
|
}
|
|
|
|
class _NestedOuterBallisticScrollActivity : BallisticScrollActivity {
|
|
public _NestedOuterBallisticScrollActivity(
|
|
_NestedScrollCoordinator coordinator,
|
|
_NestedScrollPosition position,
|
|
_NestedScrollMetrics metrics,
|
|
Simulation simulation,
|
|
TickerProvider vsync
|
|
) : base(position, simulation, vsync) {
|
|
D.assert(metrics.minRange != metrics.maxRange);
|
|
D.assert(metrics.maxRange > metrics.minRange);
|
|
this.coordinator = coordinator;
|
|
this.metrics = metrics;
|
|
}
|
|
|
|
public readonly _NestedScrollCoordinator coordinator;
|
|
public readonly _NestedScrollMetrics metrics;
|
|
|
|
public new _NestedScrollPosition del {
|
|
get { return (_NestedScrollPosition) base.del; }
|
|
}
|
|
|
|
public override void resetActivity() {
|
|
del.beginActivity(coordinator.createOuterBallisticScrollActivity(velocity));
|
|
}
|
|
|
|
public override void applyNewDimensions() {
|
|
del.beginActivity(coordinator.createOuterBallisticScrollActivity(velocity));
|
|
}
|
|
|
|
protected override bool applyMoveTo(float value) {
|
|
bool done = false;
|
|
if (velocity > 0.0f) {
|
|
if (value < metrics.minRange) {
|
|
return true;
|
|
}
|
|
|
|
if (value > metrics.maxRange) {
|
|
value = metrics.maxRange;
|
|
done = true;
|
|
}
|
|
}
|
|
else if (velocity < 0.0f) {
|
|
if (value > metrics.maxRange) {
|
|
return true;
|
|
}
|
|
|
|
if (value < metrics.minRange) {
|
|
value = metrics.minRange;
|
|
done = true;
|
|
}
|
|
}
|
|
else {
|
|
value = value.clamp(metrics.minRange, metrics.maxRange);
|
|
done = true;
|
|
}
|
|
|
|
bool result = base.applyMoveTo(value + metrics.correctionOffset);
|
|
D.assert(result); // since we tried to pass an in-range value, it shouldn"t ever overflow
|
|
return !done;
|
|
}
|
|
|
|
public override string ToString() {
|
|
return
|
|
$"{GetType()}({metrics.minRange} .. {metrics.maxRange}; correcting by {metrics.correctionOffset})";
|
|
}
|
|
}
|
|
|
|
public class SliverOverlapAbsorberHandle : ChangeNotifier {
|
|
internal int _writers = 0;
|
|
|
|
public float layoutExtent {
|
|
get { return _layoutExtent; }
|
|
}
|
|
|
|
float _layoutExtent;
|
|
|
|
public float scrollExtent {
|
|
get { return _scrollExtent; }
|
|
}
|
|
|
|
float _scrollExtent;
|
|
|
|
internal void _setExtents(float layoutValue, float scrollValue) {
|
|
D.assert(_writers == 1,
|
|
() => "Multiple RenderSliverOverlapAbsorbers have been provided the same SliverOverlapAbsorberHandle.");
|
|
_layoutExtent = layoutValue;
|
|
_scrollExtent = scrollValue;
|
|
}
|
|
|
|
internal void _markNeedsLayout() {
|
|
notifyListeners();
|
|
}
|
|
|
|
public override string ToString() {
|
|
string extra = "";
|
|
switch (_writers) {
|
|
case 0:
|
|
extra = ", orphan";
|
|
break;
|
|
case 1:
|
|
break;
|
|
default:
|
|
extra = ", $_writers WRITERS ASSIGNED";
|
|
break;
|
|
}
|
|
|
|
return $"{GetType()}({layoutExtent}{extra})";
|
|
}
|
|
}
|
|
|
|
public class SliverOverlapAbsorber : SingleChildRenderObjectWidget {
|
|
public SliverOverlapAbsorber(
|
|
Key key = null,
|
|
SliverOverlapAbsorberHandle handle = null,
|
|
Widget child = null,
|
|
Widget sliver = null
|
|
) : base(key: key, child: sliver ?? child) {
|
|
D.assert(handle != null);
|
|
D.assert(child == null || sliver == null);
|
|
this.handle = handle;
|
|
}
|
|
|
|
public readonly SliverOverlapAbsorberHandle handle;
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return new RenderSliverOverlapAbsorber(
|
|
handle: handle
|
|
);
|
|
}
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject _renderObject) {
|
|
RenderSliverOverlapAbsorber renderObject = _renderObject as RenderSliverOverlapAbsorber;
|
|
renderObject.handle = handle;
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
|
|
public class RenderSliverOverlapAbsorber : RenderObjectWithChildMixinRenderSliver<RenderSliver> {
|
|
public RenderSliverOverlapAbsorber(
|
|
SliverOverlapAbsorberHandle handle,
|
|
RenderSliver child = null,
|
|
RenderSliver sliver = null
|
|
) {
|
|
D.assert(handle != null);
|
|
D.assert(child == null || sliver == null);
|
|
_handle = handle;
|
|
this.child = sliver ?? child;
|
|
}
|
|
|
|
public SliverOverlapAbsorberHandle handle {
|
|
get { return _handle; }
|
|
set {
|
|
D.assert(value != null);
|
|
if (handle == value) {
|
|
return;
|
|
}
|
|
|
|
if (attached) {
|
|
handle._writers -= 1;
|
|
value._writers += 1;
|
|
value._setExtents(handle.layoutExtent, handle.scrollExtent);
|
|
}
|
|
|
|
_handle = value;
|
|
}
|
|
}
|
|
|
|
SliverOverlapAbsorberHandle _handle;
|
|
|
|
public override void attach(object owner) {
|
|
base.attach(owner);
|
|
handle._writers += 1;
|
|
}
|
|
|
|
public override void detach() {
|
|
handle._writers -= 1;
|
|
base.detach();
|
|
}
|
|
|
|
protected override void performLayout() {
|
|
D.assert(handle._writers == 1,
|
|
() =>
|
|
"A SliverOverlapAbsorberHandle cannot be passed to multiple RenderSliverOverlapAbsorber objects at the same time.");
|
|
if (child == null) {
|
|
geometry = new SliverGeometry();
|
|
return;
|
|
}
|
|
|
|
child.layout(constraints, parentUsesSize: true);
|
|
SliverGeometry childLayoutGeometry = child.geometry;
|
|
geometry = new SliverGeometry(
|
|
scrollExtent: childLayoutGeometry.scrollExtent - childLayoutGeometry.maxScrollObstructionExtent,
|
|
paintExtent: childLayoutGeometry.paintExtent,
|
|
paintOrigin: childLayoutGeometry.paintOrigin,
|
|
layoutExtent: Mathf.Max(0, childLayoutGeometry.paintExtent - childLayoutGeometry.maxScrollObstructionExtent),
|
|
maxPaintExtent: childLayoutGeometry.maxPaintExtent,
|
|
maxScrollObstructionExtent: childLayoutGeometry.maxScrollObstructionExtent,
|
|
hitTestExtent: childLayoutGeometry.hitTestExtent,
|
|
visible: childLayoutGeometry.visible,
|
|
hasVisualOverflow: childLayoutGeometry.hasVisualOverflow,
|
|
scrollOffsetCorrection: childLayoutGeometry.scrollOffsetCorrection
|
|
);
|
|
handle._setExtents(childLayoutGeometry.maxScrollObstructionExtent,
|
|
childLayoutGeometry.maxScrollObstructionExtent);
|
|
}
|
|
|
|
public override void applyPaintTransform(RenderObject child, Matrix4 transform) {
|
|
}
|
|
|
|
protected override bool hitTestChildren(
|
|
SliverHitTestResult result,
|
|
float mainAxisPosition = 0,
|
|
float crossAxisPosition = 0
|
|
) {
|
|
if (child != null) {
|
|
return child.hitTest(result, mainAxisPosition: mainAxisPosition,
|
|
crossAxisPosition: crossAxisPosition);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public override void paint(PaintingContext context, Offset offset) {
|
|
if (child != null) {
|
|
context.paintChild(child, offset);
|
|
}
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
|
|
public class SliverOverlapInjector : SingleChildRenderObjectWidget {
|
|
public SliverOverlapInjector(
|
|
Key key = null,
|
|
SliverOverlapAbsorberHandle handle = null,
|
|
Widget child = null,
|
|
Widget sliver = null
|
|
) : base(key: key, child: sliver ?? child) {
|
|
D.assert(handle != null);
|
|
D.assert(child == null || sliver == null);
|
|
this.handle = handle;
|
|
}
|
|
|
|
public readonly SliverOverlapAbsorberHandle handle;
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return new RenderSliverOverlapInjector(
|
|
handle: handle
|
|
);
|
|
}
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject _renderObject) {
|
|
RenderSliverOverlapInjector renderObject = _renderObject as RenderSliverOverlapInjector;
|
|
renderObject.handle = handle;
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
|
|
public class RenderSliverOverlapInjector : RenderSliver {
|
|
public RenderSliverOverlapInjector(
|
|
SliverOverlapAbsorberHandle handle
|
|
) {
|
|
D.assert(handle != null);
|
|
_handle = handle;
|
|
}
|
|
|
|
float _currentLayoutExtent;
|
|
float _currentMaxExtent;
|
|
|
|
public SliverOverlapAbsorberHandle handle {
|
|
get { return _handle; }
|
|
set {
|
|
D.assert(value != null);
|
|
if (handle == value) {
|
|
return;
|
|
}
|
|
|
|
if (attached) {
|
|
handle.removeListener(markNeedsLayout);
|
|
}
|
|
|
|
_handle = value;
|
|
if (attached) {
|
|
handle.addListener(markNeedsLayout);
|
|
if (handle.layoutExtent != _currentLayoutExtent ||
|
|
handle.scrollExtent != _currentMaxExtent) {
|
|
markNeedsLayout();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
SliverOverlapAbsorberHandle _handle;
|
|
|
|
public override void attach(object owner) {
|
|
base.attach(owner);
|
|
handle.addListener(markNeedsLayout);
|
|
if (handle.layoutExtent != _currentLayoutExtent ||
|
|
handle.scrollExtent != _currentMaxExtent) {
|
|
markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
public override void detach() {
|
|
handle.removeListener(markNeedsLayout);
|
|
base.detach();
|
|
}
|
|
|
|
protected override void performLayout() {
|
|
_currentLayoutExtent = handle.layoutExtent;
|
|
_currentMaxExtent = handle.layoutExtent;
|
|
float clampedLayoutExtent = Mathf.Min(_currentLayoutExtent - constraints.scrollOffset,
|
|
constraints.remainingPaintExtent);
|
|
geometry = new SliverGeometry(
|
|
scrollExtent: _currentLayoutExtent,
|
|
paintExtent: Mathf.Max(0.0f, clampedLayoutExtent),
|
|
maxPaintExtent: _currentMaxExtent
|
|
);
|
|
}
|
|
|
|
public override void debugPaint(PaintingContext context, Offset offset) {
|
|
D.assert(() => {
|
|
if (D.debugPaintSizeEnabled) {
|
|
Paint paint = new Paint();
|
|
paint.color = new Color(0xFFCC9933);
|
|
paint.strokeWidth = 3.0f;
|
|
paint.style = PaintingStyle.stroke;
|
|
Offset start, end, delta;
|
|
switch (constraints.axis) {
|
|
case Axis.vertical:
|
|
float x = offset.dx + constraints.crossAxisExtent / 2.0f;
|
|
start = new Offset(x, offset.dy);
|
|
end = new Offset(x, offset.dy + geometry.paintExtent);
|
|
delta = new Offset(constraints.crossAxisExtent / 5.0f, 0.0f);
|
|
break;
|
|
case Axis.horizontal:
|
|
float y = offset.dy + constraints.crossAxisExtent / 2.0f;
|
|
start = new Offset(offset.dx, y);
|
|
end = new Offset(offset.dy + geometry.paintExtent, y);
|
|
delta = new Offset(0.0f, constraints.crossAxisExtent / 5.0f);
|
|
break;
|
|
default:
|
|
throw new Exception("");
|
|
}
|
|
|
|
for (int index = -2; index <= 2; index += 1) {
|
|
PaintingUtilities.paintZigZag(context.canvas, paint, start - delta * (float) index,
|
|
end - delta * (float) index, 10, 10.0f);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
});
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
|
|
public class NestedScrollViewViewport : Viewport {
|
|
public NestedScrollViewViewport(
|
|
Key key = null,
|
|
AxisDirection? axisDirection = AxisDirection.down,
|
|
AxisDirection crossAxisDirection = AxisDirection.right,
|
|
float anchor = 0.0f,
|
|
ViewportOffset offset = null,
|
|
Key center = null,
|
|
List<Widget> slivers = null,
|
|
SliverOverlapAbsorberHandle handle = null
|
|
) : base(
|
|
key: key,
|
|
axisDirection: axisDirection,
|
|
crossAxisDirection: crossAxisDirection,
|
|
anchor: anchor,
|
|
offset: offset,
|
|
center: center,
|
|
slivers: slivers ?? new List<Widget>()
|
|
) {
|
|
D.assert(handle != null);
|
|
D.assert(this.offset != null);
|
|
this.handle = handle;
|
|
}
|
|
|
|
public readonly SliverOverlapAbsorberHandle handle;
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return new RenderNestedScrollViewViewport(
|
|
axisDirection: axisDirection,
|
|
crossAxisDirection: crossAxisDirection ??
|
|
getDefaultCrossAxisDirection(context, axisDirection),
|
|
anchor: anchor,
|
|
offset: offset,
|
|
handle: handle
|
|
);
|
|
}
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject _renderObject) {
|
|
RenderNestedScrollViewViewport renderObject = _renderObject as RenderNestedScrollViewViewport;
|
|
renderObject.axisDirection = axisDirection.Value;
|
|
renderObject.crossAxisDirection = (crossAxisDirection ?? getDefaultCrossAxisDirection(context, axisDirection)).Value;
|
|
if (crossAxisDirection == null) {
|
|
renderObject.anchor = anchor;
|
|
}
|
|
|
|
renderObject.offset = offset;
|
|
renderObject.handle = handle;
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
|
|
public class RenderNestedScrollViewViewport : RenderViewport {
|
|
public RenderNestedScrollViewViewport(
|
|
AxisDirection? axisDirection = AxisDirection.down,
|
|
AxisDirection? crossAxisDirection = AxisDirection.right,
|
|
ViewportOffset offset = null,
|
|
float anchor = 0.0f,
|
|
List<RenderSliver> children = null,
|
|
RenderSliver center = null,
|
|
SliverOverlapAbsorberHandle handle = null
|
|
) : base(
|
|
axisDirection: axisDirection,
|
|
crossAxisDirection: crossAxisDirection,
|
|
offset: offset,
|
|
anchor: anchor,
|
|
children: children,
|
|
center: center
|
|
) {
|
|
D.assert(handle != null);
|
|
D.assert(offset != null);
|
|
_handle = handle;
|
|
}
|
|
|
|
public SliverOverlapAbsorberHandle handle {
|
|
get { return _handle; }
|
|
set {
|
|
D.assert(value != null);
|
|
if (handle == value) {
|
|
return;
|
|
}
|
|
|
|
_handle = value;
|
|
handle._markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
SliverOverlapAbsorberHandle _handle;
|
|
|
|
public override void markNeedsLayout() {
|
|
handle._markNeedsLayout();
|
|
base.markNeedsLayout();
|
|
}
|
|
|
|
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
base.debugFillProperties(properties);
|
|
properties.add(new DiagnosticsProperty<SliverOverlapAbsorberHandle>("handle", handle));
|
|
}
|
|
}
|
|
}
|