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.start ) : 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, scrollDirection, reverse) ?? AxisDirection.down; } public override Widget build(BuildContext context) { AxisDirection axisDirection = _getDirection(context); Widget contents = child; if (padding != null) { contents = new Padding( padding: padding, child: contents); } ScrollController scrollController = primary ? PrimaryScrollController.of(context) : controller; Scrollable scrollable = new Scrollable( dragStartBehavior: dragStartBehavior, axisDirection: axisDirection, controller: scrollController, physics: physics, viewportBuilder: (BuildContext subContext, ViewportOffset offset) => { return new _SingleChildViewport( axisDirection: axisDirection, offset: offset, child: contents); } ); if (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: axisDirection, offset: offset ); } public override void updateRenderObject(BuildContext context, RenderObject renderObject) { _RenderSingleChildViewport _renderObject = (_RenderSingleChildViewport) renderObject; _renderObject.axisDirection = axisDirection; _renderObject.offset = offset; } } class _RenderSingleChildViewport : RenderObjectWithChildMixinRenderBox, RenderAbstractViewport { public _RenderSingleChildViewport( AxisDirection axisDirection = AxisDirection.down, ViewportOffset offset = null, float cacheExtent = RenderViewportUtils.defaultCacheExtent, RenderBox child = null) { D.assert(offset != null); _axisDirection = axisDirection; _offset = offset; _cacheExtent = cacheExtent; this.child = child; } public new RenderObject parent { get { return (RenderObject) base.parent; } } public AxisDirection axisDirection { get { return _axisDirection; } set { if (value == _axisDirection) { return; } _axisDirection = value; markNeedsLayout(); } } AxisDirection _axisDirection; public Axis axis { get { return AxisUtils.axisDirectionToAxis(axisDirection); } } public ViewportOffset offset { get { return _offset; } set { D.assert(value != null); if (value == _offset) { return; } if (attached) { _offset.removeListener(_hasScrolled); } _offset = value; if (attached) { _offset.addListener(_hasScrolled); } markNeedsLayout(); } } ViewportOffset _offset; public float cacheExtent { get { return _cacheExtent; } set { if (value == _cacheExtent) { return; } _cacheExtent = value; markNeedsLayout(); } } float _cacheExtent; void _hasScrolled() { 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); _offset.addListener(_hasScrolled); } public override void detach() { _offset.removeListener(_hasScrolled); base.detach(); } public override bool isRepaintBoundary { get { return true; } } float _viewportExtent { get { D.assert(hasSize); switch (axis) { case Axis.horizontal: return size.width; case Axis.vertical: return size.height; } D.assert(false); return 0.0f; } } float _minScrollExtent { get { D.assert(hasSize); return 0.0f; } } float _maxScrollExtent { get { D.assert(hasSize); if (child == null) { return 0.0f; } switch (axis) { case Axis.horizontal: return Mathf.Max(0.0f, child.size.width - size.width); case Axis.vertical: return Mathf.Max(0.0f, child.size.height - size.height); } D.assert(false); return 0.0f; } } BoxConstraints _getInnerConstraints(BoxConstraints constraints) { switch (axis) { case Axis.horizontal: return constraints.heightConstraints(); case Axis.vertical: return constraints.widthConstraints(); } return null; } protected override float computeMinIntrinsicWidth(float height) { if (child != null) { return child.getMinIntrinsicWidth(height); } return 0.0f; } protected override float computeMaxIntrinsicWidth(float height) { if (child != null) { return child.getMaxIntrinsicWidth(height); } return 0.0f; } protected override float computeMinIntrinsicHeight(float width) { if (child != null) { return child.getMinIntrinsicHeight(width); } return 0.0f; } protected internal override float computeMaxIntrinsicHeight(float width) { if (child != null) { return child.getMaxIntrinsicHeight(width); } return 0.0f; } protected override void performLayout() { if (child == null) { size = constraints.smallest; } else { child.layout(_getInnerConstraints(constraints), parentUsesSize: true); size = constraints.constrain(child.size); } offset.applyViewportDimension(_viewportExtent); offset.applyContentDimensions(_minScrollExtent, _maxScrollExtent); } Offset _paintOffset { get { return _paintOffsetForPosition(offset.pixels); } } Offset _paintOffsetForPosition(float position) { switch (axisDirection) { case AxisDirection.up: return new Offset(0.0f, position - child.size.height + size.height); case AxisDirection.down: return new Offset(0.0f, -position); case AxisDirection.left: return new Offset(position - child.size.width + size.width, 0.0f); case AxisDirection.right: return new Offset(-position, 0.0f); } return null; } bool _shouldClipAtPaintOffset(Offset paintOffset) { D.assert(child != null); return paintOffset < Offset.zero || !(Offset.zero & size).contains((paintOffset & child.size).bottomRight); } public override void paint(PaintingContext context, Offset offset) { if (child != null) { Offset paintOffset = _paintOffset; void paintContents(PaintingContext subContext, Offset SubOffset) { subContext.paintChild(child, SubOffset + paintOffset); } if (_shouldClipAtPaintOffset(paintOffset)) { context.pushClipRect(needsCompositing, offset, Offset.zero & size, paintContents); } else { paintContents(context, offset); } } } public override void applyPaintTransform(RenderObject child, Matrix3 transform) { Offset paintOffset = _paintOffset; transform.preTranslate(paintOffset.dx, paintOffset.dy); } public override Rect describeApproximatePaintClip(RenderObject child) { if (child != null && _shouldClipAtPaintOffset(_paintOffset)) { return Offset.zero & size; } return null; } protected override bool hitTestChildren(HitTestResult result, Offset position = null) { if (child != null) { Offset transformed = position + (-_paintOffset); return 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: offset.pixels, rect: rect); } RenderBox targetBox = (RenderBox) target; Matrix3 transform = targetBox.getTransformTo(this); Rect bounds = transform.mapRect(rect); Size contentSize = child.size; float leadingScrollOffset = 0.0f; float targetMainAxisExtent = 0.0f; float mainAxisExtent = 0.0f; switch (axisDirection) { case AxisDirection.up: mainAxisExtent = size.height; leadingScrollOffset = contentSize.height - bounds.bottom; targetMainAxisExtent = bounds.height; break; case AxisDirection.right: mainAxisExtent = size.width; leadingScrollOffset = bounds.left; targetMainAxisExtent = bounds.width; break; case AxisDirection.down: mainAxisExtent = size.height; leadingScrollOffset = bounds.top; targetMainAxisExtent = bounds.height; break; case AxisDirection.left: mainAxisExtent = size.width; leadingScrollOffset = contentSize.width - bounds.right; targetMainAxisExtent = bounds.width; break; } float targetOffset = leadingScrollOffset - (mainAxisExtent - targetMainAxisExtent) * alignment; Rect targetRect = bounds.shift(_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 (!offset.allowImplicitScrolling) { base.showOnScreen( descendant: descendant, rect: rect, duration: duration, curve: curve ); } Rect newRect = RenderViewport.showInViewport( descendant: descendant, viewport: this, offset: offset, rect: rect, duration: duration, curve: curve); base.showOnScreen( rect: newRect, duration: duration, curve: curve); } } }