您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
457 行
15 KiB
457 行
15 KiB
using System;
|
|
using Unity.UIWidgets.animation;
|
|
using Unity.UIWidgets.foundation;
|
|
using Unity.UIWidgets.gestures;
|
|
using Unity.UIWidgets.painting;
|
|
using Unity.UIWidgets.rendering;
|
|
using Unity.UIWidgets.ui;
|
|
using UnityEngine;
|
|
using Rect = Unity.UIWidgets.ui.Rect;
|
|
|
|
namespace Unity.UIWidgets.widgets {
|
|
public class SingleChildScrollView : StatelessWidget {
|
|
public SingleChildScrollView(
|
|
Key key = null,
|
|
Axis scrollDirection = Axis.vertical,
|
|
bool reverse = false,
|
|
EdgeInsets padding = null,
|
|
bool? primary = null,
|
|
ScrollPhysics physics = null,
|
|
ScrollController controller = null,
|
|
Widget child = null,
|
|
DragStartBehavior dragStartBehavior = DragStartBehavior.down
|
|
) : base(key: key) {
|
|
D.assert(!(controller != null && primary == true),
|
|
() => "Primary ScrollViews obtain their ScrollController via inheritance from a PrimaryScrollController widget. " +
|
|
"You cannot both set primary to true and pass an explicit controller.");
|
|
this.scrollDirection = scrollDirection;
|
|
this.reverse = reverse;
|
|
this.padding = padding;
|
|
this.primary = primary ?? controller == null && scrollDirection == Axis.vertical;
|
|
this.physics = physics;
|
|
this.controller = controller;
|
|
this.child = child;
|
|
this.dragStartBehavior = dragStartBehavior;
|
|
}
|
|
|
|
public readonly Axis scrollDirection;
|
|
|
|
public readonly bool reverse;
|
|
|
|
public readonly EdgeInsets padding;
|
|
|
|
public readonly ScrollController controller;
|
|
|
|
public readonly bool primary;
|
|
|
|
public readonly ScrollPhysics physics;
|
|
|
|
public readonly Widget child;
|
|
|
|
public readonly DragStartBehavior dragStartBehavior;
|
|
|
|
AxisDirection _getDirection(BuildContext context) {
|
|
return AxisDirectionUtils.getAxisDirectionFromAxisReverseAndDirectionality(context, this.scrollDirection,
|
|
this.reverse) ?? AxisDirection.down;
|
|
}
|
|
|
|
public override Widget build(BuildContext context) {
|
|
AxisDirection axisDirection = this._getDirection(context);
|
|
Widget contents = this.child;
|
|
if (this.padding != null) {
|
|
contents = new Padding(
|
|
padding: this.padding,
|
|
child: contents);
|
|
}
|
|
|
|
ScrollController scrollController = this.primary
|
|
? PrimaryScrollController.of(context)
|
|
: this.controller;
|
|
|
|
Scrollable scrollable = new Scrollable(
|
|
dragStartBehavior: this.dragStartBehavior,
|
|
axisDirection: axisDirection,
|
|
controller: scrollController,
|
|
physics: this.physics,
|
|
viewportBuilder: (BuildContext subContext, ViewportOffset offset) => {
|
|
return new _SingleChildViewport(
|
|
axisDirection: axisDirection,
|
|
offset: offset,
|
|
child: contents);
|
|
}
|
|
);
|
|
|
|
if (this.primary && scrollController != null) {
|
|
return PrimaryScrollController.none(child: scrollable);
|
|
}
|
|
|
|
return scrollable;
|
|
}
|
|
}
|
|
|
|
|
|
class _SingleChildViewport : SingleChildRenderObjectWidget {
|
|
public _SingleChildViewport(
|
|
Key key = null,
|
|
AxisDirection axisDirection = AxisDirection.down,
|
|
ViewportOffset offset = null,
|
|
Widget child = null
|
|
) : base(key: key, child: child) {
|
|
this.axisDirection = axisDirection;
|
|
this.offset = offset;
|
|
}
|
|
|
|
public readonly AxisDirection axisDirection;
|
|
|
|
public readonly ViewportOffset offset;
|
|
|
|
|
|
public override RenderObject createRenderObject(BuildContext context) {
|
|
return new _RenderSingleChildViewport(
|
|
axisDirection: this.axisDirection,
|
|
offset: this.offset
|
|
);
|
|
}
|
|
|
|
|
|
public override void updateRenderObject(BuildContext context, RenderObject renderObject) {
|
|
_RenderSingleChildViewport _renderObject = (_RenderSingleChildViewport) renderObject;
|
|
_renderObject.axisDirection = this.axisDirection;
|
|
_renderObject.offset = this.offset;
|
|
}
|
|
}
|
|
|
|
|
|
class _RenderSingleChildViewport : RenderObjectWithChildMixinRenderBox<RenderBox>, RenderAbstractViewport {
|
|
public _RenderSingleChildViewport(
|
|
AxisDirection axisDirection = AxisDirection.down,
|
|
ViewportOffset offset = null,
|
|
float cacheExtent = RenderViewportUtils.defaultCacheExtent,
|
|
RenderBox child = null) {
|
|
D.assert(offset != null);
|
|
this._axisDirection = axisDirection;
|
|
this._offset = offset;
|
|
this._cacheExtent = cacheExtent;
|
|
this.child = child;
|
|
}
|
|
|
|
public new RenderObject parent {
|
|
get { return (RenderObject) base.parent; }
|
|
}
|
|
|
|
public AxisDirection axisDirection {
|
|
get { return this._axisDirection; }
|
|
set {
|
|
if (value == this._axisDirection) {
|
|
return;
|
|
}
|
|
|
|
this._axisDirection = value;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
AxisDirection _axisDirection;
|
|
|
|
public Axis axis {
|
|
get { return AxisUtils.axisDirectionToAxis(this.axisDirection); }
|
|
}
|
|
|
|
public ViewportOffset offset {
|
|
get { return this._offset; }
|
|
set {
|
|
D.assert(value != null);
|
|
if (value == this._offset) {
|
|
return;
|
|
}
|
|
|
|
if (this.attached) {
|
|
this._offset.removeListener(this._hasScrolled);
|
|
}
|
|
|
|
this._offset = value;
|
|
if (this.attached) {
|
|
this._offset.addListener(this._hasScrolled);
|
|
}
|
|
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
ViewportOffset _offset;
|
|
|
|
public float cacheExtent {
|
|
get { return this._cacheExtent; }
|
|
set {
|
|
if (value == this._cacheExtent) {
|
|
return;
|
|
}
|
|
|
|
this._cacheExtent = value;
|
|
this.markNeedsLayout();
|
|
}
|
|
}
|
|
|
|
float _cacheExtent;
|
|
|
|
void _hasScrolled() {
|
|
this.markNeedsPaint();
|
|
}
|
|
|
|
public override void setupParentData(RenderObject child) {
|
|
if (!(child.parentData is ParentData)) {
|
|
child.parentData = new ParentData();
|
|
}
|
|
}
|
|
|
|
public override void attach(object owner) {
|
|
base.attach(owner);
|
|
this._offset.addListener(this._hasScrolled);
|
|
}
|
|
|
|
public override void detach() {
|
|
this._offset.removeListener(this._hasScrolled);
|
|
base.detach();
|
|
}
|
|
|
|
public override bool isRepaintBoundary {
|
|
get { return true; }
|
|
}
|
|
|
|
|
|
float _viewportExtent {
|
|
get {
|
|
D.assert(this.hasSize);
|
|
switch (this.axis) {
|
|
case Axis.horizontal:
|
|
return this.size.width;
|
|
case Axis.vertical:
|
|
return this.size.height;
|
|
}
|
|
|
|
D.assert(false);
|
|
return 0.0f;
|
|
}
|
|
}
|
|
|
|
float _minScrollExtent {
|
|
get {
|
|
D.assert(this.hasSize);
|
|
return 0.0f;
|
|
}
|
|
}
|
|
|
|
float _maxScrollExtent {
|
|
get {
|
|
D.assert(this.hasSize);
|
|
if (this.child == null) {
|
|
return 0.0f;
|
|
}
|
|
|
|
switch (this.axis) {
|
|
case Axis.horizontal:
|
|
return Mathf.Max(0.0f, this.child.size.width - this.size.width);
|
|
case Axis.vertical:
|
|
return Mathf.Max(0.0f, this.child.size.height - this.size.height);
|
|
}
|
|
|
|
D.assert(false);
|
|
return 0.0f;
|
|
}
|
|
}
|
|
|
|
BoxConstraints _getInnerConstraints(BoxConstraints constraints) {
|
|
switch (this.axis) {
|
|
case Axis.horizontal:
|
|
return constraints.heightConstraints();
|
|
case Axis.vertical:
|
|
return constraints.widthConstraints();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
|
|
protected override float computeMinIntrinsicWidth(float height) {
|
|
if (this.child != null) {
|
|
return this.child.getMinIntrinsicWidth(height);
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
protected override float computeMaxIntrinsicWidth(float height) {
|
|
if (this.child != null) {
|
|
return this.child.getMaxIntrinsicWidth(height);
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
protected override float computeMinIntrinsicHeight(float width) {
|
|
if (this.child != null) {
|
|
return this.child.getMinIntrinsicHeight(width);
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
protected override float computeMaxIntrinsicHeight(float width) {
|
|
if (this.child != null) {
|
|
return this.child.getMaxIntrinsicHeight(width);
|
|
}
|
|
|
|
return 0.0f;
|
|
}
|
|
|
|
protected override void performLayout() {
|
|
if (this.child == null) {
|
|
this.size = this.constraints.smallest;
|
|
}
|
|
else {
|
|
this.child.layout(this._getInnerConstraints(this.constraints), parentUsesSize: true);
|
|
this.size = this.constraints.constrain(this.child.size);
|
|
}
|
|
|
|
this.offset.applyViewportDimension(this._viewportExtent);
|
|
this.offset.applyContentDimensions(this._minScrollExtent, this._maxScrollExtent);
|
|
}
|
|
|
|
Offset _paintOffset {
|
|
get { return this._paintOffsetForPosition(this.offset.pixels); }
|
|
}
|
|
|
|
Offset _paintOffsetForPosition(float position) {
|
|
switch (this.axisDirection) {
|
|
case AxisDirection.up:
|
|
return new Offset(0.0f, position - this.child.size.height + this.size.height);
|
|
case AxisDirection.down:
|
|
return new Offset(0.0f, -position);
|
|
case AxisDirection.left:
|
|
return new Offset(position - this.child.size.width + this.size.width, 0.0f);
|
|
case AxisDirection.right:
|
|
return new Offset(-position, 0.0f);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
bool _shouldClipAtPaintOffset(Offset paintOffset) {
|
|
D.assert(this.child != null);
|
|
return paintOffset < Offset.zero ||
|
|
!(Offset.zero & this.size).contains((paintOffset & this.child.size).bottomRight);
|
|
}
|
|
|
|
public override void paint(PaintingContext context, Offset offset) {
|
|
if (this.child != null) {
|
|
Offset paintOffset = this._paintOffset;
|
|
|
|
void paintContents(PaintingContext subContext, Offset SubOffset) {
|
|
subContext.paintChild(this.child, SubOffset + paintOffset);
|
|
}
|
|
|
|
if (this._shouldClipAtPaintOffset(paintOffset)) {
|
|
context.pushClipRect(this.needsCompositing, offset, Offset.zero & this.size, paintContents);
|
|
}
|
|
else {
|
|
paintContents(context, offset);
|
|
}
|
|
}
|
|
}
|
|
|
|
public override void applyPaintTransform(RenderObject child, Matrix3 transform) {
|
|
Offset paintOffset = this._paintOffset;
|
|
transform.preTranslate(paintOffset.dx, paintOffset.dy);
|
|
}
|
|
|
|
public override Rect describeApproximatePaintClip(RenderObject child) {
|
|
if (child != null && this._shouldClipAtPaintOffset(this._paintOffset)) {
|
|
return Offset.zero & this.size;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
protected override bool hitTestChildren(HitTestResult result, Offset position = null) {
|
|
if (this.child != null) {
|
|
Offset transformed = position + (-this._paintOffset);
|
|
return this.child.hitTest(result, position: transformed);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
public RevealedOffset getOffsetToReveal(RenderObject target, float alignment, Rect rect = null) {
|
|
rect = rect ?? target.paintBounds;
|
|
if (!(target is RenderBox)) {
|
|
return new RevealedOffset(offset: this.offset.pixels, rect: rect);
|
|
}
|
|
|
|
RenderBox targetBox = (RenderBox) target;
|
|
Matrix3 transform = targetBox.getTransformTo(this);
|
|
Rect bounds = transform.mapRect(rect);
|
|
Size contentSize = this.child.size;
|
|
|
|
float leadingScrollOffset = 0.0f;
|
|
float targetMainAxisExtent = 0.0f;
|
|
float mainAxisExtent = 0.0f;
|
|
|
|
switch (this.axisDirection) {
|
|
case AxisDirection.up:
|
|
mainAxisExtent = this.size.height;
|
|
leadingScrollOffset = contentSize.height - bounds.bottom;
|
|
targetMainAxisExtent = bounds.height;
|
|
break;
|
|
case AxisDirection.right:
|
|
mainAxisExtent = this.size.width;
|
|
leadingScrollOffset = bounds.left;
|
|
targetMainAxisExtent = bounds.width;
|
|
break;
|
|
case AxisDirection.down:
|
|
mainAxisExtent = this.size.height;
|
|
leadingScrollOffset = bounds.top;
|
|
targetMainAxisExtent = bounds.height;
|
|
break;
|
|
case AxisDirection.left:
|
|
mainAxisExtent = this.size.width;
|
|
leadingScrollOffset = contentSize.width - bounds.right;
|
|
targetMainAxisExtent = bounds.width;
|
|
break;
|
|
}
|
|
|
|
float targetOffset = leadingScrollOffset - (mainAxisExtent - targetMainAxisExtent) * alignment;
|
|
Rect targetRect = bounds.shift(this._paintOffsetForPosition(targetOffset));
|
|
return new RevealedOffset(offset: targetOffset, rect: targetRect);
|
|
}
|
|
|
|
public override void showOnScreen(
|
|
RenderObject descendant = null,
|
|
Rect rect = null,
|
|
TimeSpan? duration = null,
|
|
Curve curve = null
|
|
) {
|
|
if (!this.offset.allowImplicitScrolling) {
|
|
base.showOnScreen(
|
|
descendant: descendant,
|
|
rect: rect,
|
|
duration: duration,
|
|
curve: curve
|
|
);
|
|
}
|
|
|
|
Rect newRect = RenderViewport.showInViewport(
|
|
descendant: descendant,
|
|
viewport: this,
|
|
offset: this.offset,
|
|
rect: rect,
|
|
duration: duration,
|
|
curve: curve);
|
|
|
|
base.showOnScreen(
|
|
rect: newRect,
|
|
duration: duration,
|
|
curve: curve);
|
|
}
|
|
}
|
|
}
|