您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 
 
 

391 行
15 KiB

using System;
using System.Collections.Generic;
using System.Linq;
using RSG;
using UIWidgets.animation;
using UIWidgets.foundation;
using UIWidgets.gestures;
using UIWidgets.painting;
using UIWidgets.rendering;
using UIWidgets.scheduler;
using UIWidgets.ui;
namespace UIWidgets.widgets {
public delegate Widget ViewportBuilder(BuildContext context, ViewportOffset position);
public class Scrollable : StatefulWidget {
public Scrollable(
Key key = null,
AxisDirection axisDirection = AxisDirection.down,
ScrollController controller = null,
ScrollPhysics physics = null,
ViewportBuilder viewportBuilder = null
) : base(key: key) {
D.assert(viewportBuilder != null);
this.axisDirection = axisDirection;
this.controller = controller;
this.physics = physics;
this.viewportBuilder = viewportBuilder;
}
public readonly AxisDirection axisDirection;
public readonly ScrollController controller;
public readonly ScrollPhysics physics;
public readonly ViewportBuilder viewportBuilder;
public Axis axis {
get { return AxisUtils.axisDirectionToAxis(this.axisDirection); }
}
public override State createState() {
return new ScrollableState();
}
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
base.debugFillProperties(properties);
properties.add(new EnumProperty<AxisDirection>("axisDirection", this.axisDirection));
properties.add(new DiagnosticsProperty<ScrollPhysics>("physics", this.physics));
}
public static ScrollableState of(BuildContext context) {
_ScrollableScope widget = (_ScrollableScope) context.inheritFromWidgetOfExactType(typeof(_ScrollableScope));
return widget == null ? null : widget.scrollable;
}
public static IPromise ensureVisible(BuildContext context,
double alignment = 0.0,
TimeSpan? duration = null,
Curve curve = null
) {
duration = duration ?? TimeSpan.Zero;
curve = curve ?? Curves.ease;
List<IPromise> futures = new List<IPromise>();
ScrollableState scrollable = Scrollable.of(context);
while (scrollable != null) {
futures.Add(scrollable.position.ensureVisible(
context.findRenderObject(),
alignment: alignment,
duration: duration,
curve: curve
));
context = scrollable.context;
scrollable = Scrollable.of(context);
}
if (futures.isEmpty() || duration == TimeSpan.Zero) {
return Promise.Resolved();
}
if (futures.Count == 1) {
return futures.Single();
}
return Promise.All(futures);
}
}
class _ScrollableScope : InheritedWidget {
internal _ScrollableScope(
Key key = null,
ScrollableState scrollable = null,
ScrollPosition position = null,
Widget child = null
) : base(key: key, child: child) {
D.assert(scrollable != null);
D.assert(child != null);
this.scrollable = scrollable;
this.position = position;
}
public readonly ScrollableState scrollable;
public readonly ScrollPosition position;
public override bool updateShouldNotify(InheritedWidget old) {
return this.position != ((_ScrollableScope) old).position;
}
}
public class ScrollableState : TickerProviderStateMixin<Scrollable>, ScrollContext {
public ScrollPosition position {
get { return this._position; }
}
ScrollPosition _position;
public AxisDirection axisDirection {
get { return this.widget.axisDirection; }
}
ScrollBehavior _configuration;
ScrollPhysics _physics;
void _updatePosition() {
this._configuration = ScrollConfiguration.of(this.context);
this._physics = this._configuration.getScrollPhysics(this.context);
if (this.widget.physics != null) {
this._physics = this.widget.physics.applyTo(this._physics);
}
ScrollController controller = this.widget.controller;
ScrollPosition oldPosition = this.position;
if (oldPosition != null) {
if (controller != null) {
controller.detach(oldPosition);
}
Window.instance.scheduleMicrotask(oldPosition.dispose);
}
this._position = controller == null
? null
: controller.createScrollPosition(this._physics, this, oldPosition);
this._position = this._position
?? new ScrollPositionWithSingleContext(physics: this._physics, context: this,
oldPosition: oldPosition);
D.assert(this.position != null);
if (controller != null) {
controller.attach(this.position);
}
}
public override void didChangeDependencies() {
base.didChangeDependencies();
this._updatePosition();
}
bool _shouldUpdatePosition(Scrollable oldWidget) {
ScrollPhysics newPhysics = this.widget.physics;
ScrollPhysics oldPhysics = oldWidget.physics;
do {
Type newPhysicsType = newPhysics != null ? newPhysics.GetType() : null;
Type oldPhysicsType = oldPhysics != null ? oldPhysics.GetType() : null;
if (newPhysicsType != oldPhysicsType) {
return true;
}
if (newPhysics != null) {
newPhysics = newPhysics.parent;
}
if (oldPhysics != null) {
oldPhysics = oldPhysics.parent;
}
} while (newPhysics != null || oldPhysics != null);
Type controllerType = this.widget.controller == null ? null : this.widget.controller.GetType();
Type oldControllerType = oldWidget.controller == null ? null : oldWidget.controller.GetType();
return controllerType != oldControllerType;
}
public override void didUpdateWidget(StatefulWidget oldWidgetRaw) {
Scrollable oldWidget = (Scrollable) oldWidgetRaw;
base.didUpdateWidget(oldWidget);
if (this.widget.controller != oldWidget.controller) {
if (oldWidget.controller != null) {
oldWidget.controller.detach(this.position);
}
if (this.widget.controller != null) {
this.widget.controller.attach(this.position);
}
}
if (this._shouldUpdatePosition(oldWidget)) {
this._updatePosition();
}
}
public override void dispose() {
if (this.widget.controller != null) {
this.widget.controller.detach(this.position);
}
this.position.dispose();
base.dispose();
}
readonly GlobalKey<RawGestureDetectorState> _gestureDetectorKey = GlobalKey<RawGestureDetectorState>.key();
readonly GlobalKey _ignorePointerKey = GlobalKey.key();
Dictionary<Type, GestureRecognizerFactory> _gestureRecognizers =
new Dictionary<Type, GestureRecognizerFactory>();
bool _shouldIgnorePointer = false;
bool _lastCanDrag;
Axis _lastAxisDirection;
public void setCanDrag(bool canDrag) {
if (canDrag == this._lastCanDrag && (!canDrag || this.widget.axis == this._lastAxisDirection)) {
return;
}
if (!canDrag) {
this._gestureRecognizers = new Dictionary<Type, GestureRecognizerFactory>();
} else {
switch (this.widget.axis) {
case Axis.vertical:
this._gestureRecognizers = new Dictionary<Type, GestureRecognizerFactory>();
this._gestureRecognizers.Add(typeof(VerticalDragGestureRecognizer),
new GestureRecognizerFactoryWithHandlers<VerticalDragGestureRecognizer>(
() => new VerticalDragGestureRecognizer(),
instance => {
instance.onDown = this._handleDragDown;
instance.onStart = this._handleDragStart;
instance.onUpdate = this._handleDragUpdate;
instance.onEnd = this._handleDragEnd;
instance.onCancel = this._handleDragCancel;
instance.minFlingDistance =
this._physics == null ? (double?) null : this._physics.minFlingDistance;
instance.minFlingVelocity =
this._physics == null ? (double?) null : this._physics.minFlingVelocity;
instance.maxFlingVelocity =
this._physics == null ? (double?) null : this._physics.maxFlingVelocity;
}
));
break;
case Axis.horizontal:
this._gestureRecognizers = new Dictionary<Type, GestureRecognizerFactory>();
this._gestureRecognizers.Add(typeof(HorizontalDragGestureRecognizer),
new GestureRecognizerFactoryWithHandlers<HorizontalDragGestureRecognizer>(
() => new HorizontalDragGestureRecognizer(),
instance => {
instance.onDown = this._handleDragDown;
instance.onStart = this._handleDragStart;
instance.onUpdate = this._handleDragUpdate;
instance.onEnd = this._handleDragEnd;
instance.onCancel = this._handleDragCancel;
instance.minFlingDistance =
this._physics == null ? (double?) null : this._physics.minFlingDistance;
instance.minFlingVelocity =
this._physics == null ? (double?) null : this._physics.minFlingVelocity;
instance.maxFlingVelocity =
this._physics == null ? (double?) null : this._physics.maxFlingVelocity;
}
));
break;
}
}
this._lastCanDrag = canDrag;
this._lastAxisDirection = this.widget.axis;
if (this._gestureDetectorKey.currentState != null) {
this._gestureDetectorKey.currentState.replaceGestureRecognizers(this._gestureRecognizers);
}
}
public TickerProvider vsync {
get { return this; }
}
public void setIgnorePointer(bool value) {
if (this._shouldIgnorePointer == value) {
return;
}
this._shouldIgnorePointer = value;
if (this._ignorePointerKey.currentContext != null) {
var renderBox = (RenderIgnorePointer) this._ignorePointerKey.currentContext.findRenderObject();
renderBox.ignoring = this._shouldIgnorePointer;
}
}
public BuildContext notificationContext {
get { return this._gestureDetectorKey.currentContext; }
}
public BuildContext storageContext {
get { return this.context; }
}
Drag _drag;
ScrollHoldController _hold;
void _handleDragDown(DragDownDetails details) {
D.assert(this._drag == null);
D.assert(this._hold == null);
this._hold = this.position.hold(this._disposeHold);
}
void _handleDragStart(DragStartDetails details) {
D.assert(this._drag == null);
this._drag = this.position.drag(details, this._disposeDrag);
D.assert(this._drag != null);
D.assert(this._hold == null);
}
void _handleDragUpdate(DragUpdateDetails details) {
D.assert(this._hold == null || this._drag == null);
if (this._drag != null) {
this._drag.update(details);
}
}
void _handleDragEnd(DragEndDetails details) {
D.assert(this._hold == null || this._drag == null);
if (this._drag != null) {
this._drag.end(details);
}
D.assert(this._drag == null);
}
void _handleDragCancel() {
D.assert(this._hold == null || this._drag == null);
if (this._hold != null) {
this._hold.cancel();
}
if (this._drag != null) {
this._drag.cancel();
}
D.assert(this._hold == null);
D.assert(this._drag == null);
}
void _disposeHold() {
this._hold = null;
}
void _disposeDrag() {
this._drag = null;
}
public override Widget build(BuildContext context) {
D.assert(this.position != null);
Widget result = new RawGestureDetector(
key: this._gestureDetectorKey,
gestures: this._gestureRecognizers,
behavior: HitTestBehavior.opaque,
child: new IgnorePointer(
key: this._ignorePointerKey,
ignoring: this._shouldIgnorePointer,
child: new _ScrollableScope(
scrollable: this,
position: this.position,
child: this.widget.viewportBuilder(context, this.position)
)
)
);
return this._configuration.buildViewportChrome(context, result, this.widget.axisDirection);
}
public override void debugFillProperties(DiagnosticPropertiesBuilder properties) {
base.debugFillProperties(properties);
properties.add(new DiagnosticsProperty<ScrollPosition>("position", this.position));
}
}
}