1 /* Copyright (c) 2019 Samsung Electronics Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
7 * http://www.apache.org/licenses/LICENSE-2.0
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
17 using Tizen.NUI.BaseComponents;
18 using System.ComponentModel;
19 using System.Diagnostics;
21 namespace Tizen.NUI.Components
24 /// [Draft] This class provides a View that can scroll a single View with a layout. This View can be a nest of Views.
26 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API.
27 [EditorBrowsable(EditorBrowsableState.Never)]
28 public class ScrollableBase : Control
30 static bool LayoutDebugScrollableBase = false; // Debug flag
31 private Direction mScrollingDirection = Direction.Vertical;
32 private bool mScrollEnabled = true;
33 private int mPageWidth = 0;
35 private class ScrollableBaseCustomLayout : LayoutGroup
37 protected override void OnMeasure(MeasureSpecification widthMeasureSpec, MeasureSpecification heightMeasureSpec)
39 Extents padding = Padding;
40 float totalHeight = padding.Top + padding.Bottom;
41 float totalWidth = padding.Start + padding.End;
43 MeasuredSize.StateType childWidthState = MeasuredSize.StateType.MeasuredSizeOK;
44 MeasuredSize.StateType childHeightState = MeasuredSize.StateType.MeasuredSizeOK;
46 Direction scrollingDirection = Direction.Vertical;
47 ScrollableBase scrollableBase = this.Owner as ScrollableBase;
50 scrollingDirection = scrollableBase.ScrollingDirection;
53 // measure child, should be a single scrolling child
54 foreach( LayoutItem childLayout in LayoutChildren )
56 if (childLayout != null)
59 // Use an Unspecified MeasureSpecification mode so scrolling child is not restricted to it's parents size in Height (for vertical scrolling)
60 // or Width for horizontal scrolling
61 MeasureSpecification unrestrictedMeasureSpec = new MeasureSpecification( heightMeasureSpec.Size, MeasureSpecification.ModeType.Unspecified);
63 if (scrollingDirection == Direction.Vertical)
65 MeasureChild( childLayout, widthMeasureSpec, unrestrictedMeasureSpec ); // Height unrestricted by parent
69 MeasureChild( childLayout, unrestrictedMeasureSpec, heightMeasureSpec ); // Width unrestricted by parent
72 float childWidth = childLayout.MeasuredWidth.Size.AsDecimal();
73 float childHeight = childLayout.MeasuredHeight.Size.AsDecimal();
75 // Determine the width and height needed by the children using their given position and size.
76 // Children could overlap so find the left most and right most child.
77 Position2D childPosition = childLayout.Owner.Position2D;
78 float childLeft = childPosition.X;
79 float childTop = childPosition.Y;
81 // Store current width and height needed to contain all children.
82 Extents childMargin = childLayout.Margin;
83 totalWidth = childWidth + childMargin.Start + childMargin.End;
84 totalHeight = childHeight + childMargin.Top + childMargin.Bottom;
86 if (childLayout.MeasuredWidth.State == MeasuredSize.StateType.MeasuredSizeTooSmall)
88 childWidthState = MeasuredSize.StateType.MeasuredSizeTooSmall;
90 if (childLayout.MeasuredWidth.State == MeasuredSize.StateType.MeasuredSizeTooSmall)
92 childHeightState = MeasuredSize.StateType.MeasuredSizeTooSmall;
98 MeasuredSize widthSizeAndState = ResolveSizeAndState(new LayoutLength(totalWidth), widthMeasureSpec, MeasuredSize.StateType.MeasuredSizeOK);
99 MeasuredSize heightSizeAndState = ResolveSizeAndState(new LayoutLength(totalHeight), heightMeasureSpec, MeasuredSize.StateType.MeasuredSizeOK);
100 totalWidth = widthSizeAndState.Size.AsDecimal();
101 totalHeight = heightSizeAndState.Size.AsDecimal();
103 // Ensure layout respects it's given minimum size
104 totalWidth = Math.Max( totalWidth, SuggestedMinimumWidth.AsDecimal() );
105 totalHeight = Math.Max( totalHeight, SuggestedMinimumHeight.AsDecimal() );
107 widthSizeAndState.State = childWidthState;
108 heightSizeAndState.State = childHeightState;
110 SetMeasuredDimensions( ResolveSizeAndState( new LayoutLength(totalWidth), widthMeasureSpec, childWidthState ),
111 ResolveSizeAndState( new LayoutLength(totalHeight), heightMeasureSpec, childHeightState ) );
113 // Size of ScrollableBase is changed. Change Page width too.
114 scrollableBase.mPageWidth = (int)MeasuredWidth.Size.AsRoundedValue();
117 protected override void OnLayout(bool changed, LayoutLength left, LayoutLength top, LayoutLength right, LayoutLength bottom)
119 foreach( LayoutItem childLayout in LayoutChildren )
121 if( childLayout != null )
123 LayoutLength childWidth = childLayout.MeasuredWidth.Size;
124 LayoutLength childHeight = childLayout.MeasuredHeight.Size;
126 Position2D childPosition = childLayout.Owner.Position2D;
127 Extents padding = Padding;
128 Extents childMargin = childLayout.Margin;
130 LayoutLength childLeft = new LayoutLength(childPosition.X + childMargin.Start + padding.Start);
131 LayoutLength childTop = new LayoutLength(childPosition.Y + childMargin.Top + padding.Top);
133 childLayout.Layout( childLeft, childTop, childLeft + childWidth, childTop + childHeight );
137 } // ScrollableBaseCustomLayout
140 /// The direction axis to scroll.
142 /// <since_tizen> 6 </since_tizen>
143 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API.
144 [EditorBrowsable(EditorBrowsableState.Never)]
145 public enum Direction
150 /// <since_tizen> 6 </since_tizen>
156 /// <since_tizen> 6 </since_tizen>
161 /// [Draft] Configurable speed threshold that register the gestures as a flick.
162 /// If the flick speed less than the threshold then will not be considered a flick.
164 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API.
165 [EditorBrowsable(EditorBrowsableState.Never)]
166 public float FlickThreshold { get; set; } = 0.2f;
169 /// [Draft] Configurable duration modifer for the flick animation.
170 /// Determines the speed of the scroll, large value results in a longer flick animation. Range (0.1 - 1.0)
172 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
173 [EditorBrowsable(EditorBrowsableState.Never)]
174 public float FlickAnimationSpeed { get; set; } = 0.4f;
177 /// [Draft] Configurable modifer for the distance to be scrolled when flicked detected.
178 /// It a ratio of the ScrollableBase's length. (not child's length).
179 /// First value is the ratio of the distance to scroll with the weakest flick.
180 /// Second value is the ratio of the distance to scroll with the strongest flick.
183 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
184 [EditorBrowsable(EditorBrowsableState.Never)]
185 public Vector2 FlickDistanceMultiplierRange { get; set; } = new Vector2(0.6f, 1.8f);
188 /// [Draft] Scrolling direction mode.
189 /// Default is Vertical scrolling.
191 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
192 [EditorBrowsable(EditorBrowsableState.Never)]
193 public Direction ScrollingDirection
197 return mScrollingDirection;
201 if(value != mScrollingDirection)
203 mScrollingDirection = value;
204 mPanGestureDetector.RemoveDirection(value == Direction.Horizontal ? PanGestureDetector.DirectionVertical : PanGestureDetector.DirectionHorizontal);
205 mPanGestureDetector.AddDirection(value == Direction.Horizontal ? PanGestureDetector.DirectionHorizontal : PanGestureDetector.DirectionVertical);
211 /// [Draft] Enable or disable scrolling.
213 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
214 [EditorBrowsable(EditorBrowsableState.Never)]
215 public bool ScrollEnabled
219 return mScrollEnabled;
223 if (value != mScrollEnabled)
225 mScrollEnabled = value;
228 mPanGestureDetector.Detected += OnPanGestureDetected;
229 mTapGestureDetector.Detected += OnTapGestureDetected;
233 mPanGestureDetector.Detected -= OnPanGestureDetected;
234 mTapGestureDetector.Detected -= OnTapGestureDetected;
241 /// [Draft] Pages mode, enables moving to the next or return to current page depending on pan displacement.
242 /// Default is false.
244 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
245 [EditorBrowsable(EditorBrowsableState.Never)]
246 public bool SnapToPage { set; get; } = false;
249 /// [Draft] Get current page.
250 /// Working propery with SnapToPage property.
252 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
253 [EditorBrowsable(EditorBrowsableState.Never)]
254 public int CurrentPage { get; private set; } = 0;
257 /// [Draft] Duration of scroll animation.
259 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
260 [EditorBrowsable(EditorBrowsableState.Never)]
261 public int ScrollDuration { set; get; } = 125;
264 /// ScrollEventArgs is a class to record scroll event arguments which will sent to user.
266 /// <since_tizen> 6 </since_tizen>
267 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
268 [EditorBrowsable(EditorBrowsableState.Never)]
269 public class ScrollEventArgs : EventArgs
274 /// Default constructor.
276 /// <param name="position">Current scroll position</param>
277 /// <since_tizen> 6 </since_tizen>
278 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
279 public ScrollEventArgs(Position position)
281 this.position = position;
285 /// [Draft] Current scroll position.
287 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
288 [EditorBrowsable(EditorBrowsableState.Never)]
289 public Position Position
299 /// An event emitted when user starts dragging ScrollableBase, user can subscribe or unsubscribe to this event handler.<br />
301 /// <since_tizen> 6 </since_tizen>
302 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
303 [EditorBrowsable(EditorBrowsableState.Never)]
304 public event EventHandler<ScrollEventArgs> ScrollDragStartEvent;
307 /// An event emitted when user stops dragging ScrollableBase, user can subscribe or unsubscribe to this event handler.<br />
309 /// <since_tizen> 6 </since_tizen>
310 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
311 [EditorBrowsable(EditorBrowsableState.Never)]
312 public event EventHandler<ScrollEventArgs> ScrollDragEndEvent;
316 /// An event emitted when the scrolling slide animation starts, user can subscribe or unsubscribe to this event handler.<br />
318 /// <since_tizen> 6 </since_tizen>
319 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
320 [EditorBrowsable(EditorBrowsableState.Never)]
321 public event EventHandler<ScrollEventArgs> ScrollAnimationStartEvent;
324 /// An event emitted when the scrolling slide animation ends, user can subscribe or unsubscribe to this event handler.<br />
326 /// <since_tizen> 6 </since_tizen>
327 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
328 [EditorBrowsable(EditorBrowsableState.Never)]
329 public event EventHandler<ScrollEventArgs> ScrollAnimationEndEvent;
333 /// An event emitted when scrolling, user can subscribe or unsubscribe to this event handler.<br />
335 /// <since_tizen> 6 </since_tizen>
336 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
337 [EditorBrowsable(EditorBrowsableState.Never)]
338 public event EventHandler<ScrollEventArgs> ScrollEvent;
340 private Animation scrollAnimation;
341 private float maxScrollDistance;
342 private float childTargetPosition = 0.0f;
343 private PanGestureDetector mPanGestureDetector;
344 private TapGestureDetector mTapGestureDetector;
345 private View mScrollingChild;
346 private float multiplier =1.0f;
347 private bool scrolling = false;
348 private float ratioOfScreenWidthToCompleteScroll = 0.5f;
349 private float totalDisplacementForPan = 0.0f;
351 // If false then can only flick pages when the current animation/scroll as ended.
352 private bool flickWhenAnimating = false;
353 private PropertyNotification propertyNotification;
356 /// [Draft] Constructor
358 /// <since_tizen> 6 </since_tizen>
359 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
360 [EditorBrowsable(EditorBrowsableState.Never)]
361 public ScrollableBase() : base()
363 mPanGestureDetector = new PanGestureDetector();
364 mPanGestureDetector.Attach(this);
365 mPanGestureDetector.AddDirection(PanGestureDetector.DirectionVertical);
366 mPanGestureDetector.Detected += OnPanGestureDetected;
368 mTapGestureDetector = new TapGestureDetector();
369 mTapGestureDetector.Attach(this);
370 mTapGestureDetector.Detected += OnTapGestureDetected;
373 ClippingMode = ClippingModeType.ClipToBoundingBox;
375 mScrollingChild = new View();
376 mScrollingChild.Name = "DefaultScrollingChild";
378 Layout = new ScrollableBaseCustomLayout();
381 private void OnPropertyChanged(object source, PropertyNotification.NotifyEventArgs args)
387 /// Called after a child has been added to the owning view.
389 /// <param name="view">The child which has been added.</param>
390 /// <since_tizen> 6 </since_tizen>
391 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
392 [EditorBrowsable(EditorBrowsableState.Never)]
393 public override void OnChildAdd(View view)
395 if(mScrollingChild.Name != "DefaultScrollingChild")
397 propertyNotification.Notified -= OnPropertyChanged;
398 mScrollingChild.RemovePropertyNotification(propertyNotification);
401 mScrollingChild = view;
402 propertyNotification = mScrollingChild?.AddPropertyNotification("position", PropertyCondition.Step(1.0f));
403 propertyNotification.Notified += OnPropertyChanged;
406 if (Children.Count > 1)
407 Log.Error("ScrollableBase", $"Only 1 child should be added to ScrollableBase.");
412 /// Called after a child has been removed from the owning view.
414 /// <param name="view">The child which has been removed.</param>
415 /// <since_tizen> 6 </since_tizen>
416 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
417 [EditorBrowsable(EditorBrowsableState.Never)]
418 public override void OnChildRemove(View view)
420 propertyNotification.Notified -= OnPropertyChanged;
421 mScrollingChild.RemovePropertyNotification(propertyNotification);
423 mScrollingChild = new View();
428 /// Scrolls to the item at the specified index.
430 /// <param name="index">Index of item.</param>
431 /// <since_tizen> 6 </since_tizen>
432 /// This may be public opened in tizen_6.0 after ACR done. Before ACR, need to be hidden as inhouse API
433 [EditorBrowsable(EditorBrowsableState.Never)]
434 public void ScrollToIndex(int index)
436 if(mScrollingChild.ChildCount-1 < index || index < 0)
446 maxScrollDistance = CalculateMaximumScrollDistance();
448 float targetPosition = Math.Min(ScrollingDirection == Direction.Vertical ? mScrollingChild.Children[index].Position.Y : mScrollingChild.Children[index].Position.X, maxScrollDistance);
449 AnimateChildTo(ScrollDuration, -targetPosition);
452 private void OnScrollDragStart()
454 ScrollEventArgs eventArgs = new ScrollEventArgs(mScrollingChild.CurrentPosition);
455 ScrollDragStartEvent?.Invoke(this, eventArgs);
458 private void OnScrollDragEnd()
460 ScrollEventArgs eventArgs = new ScrollEventArgs(mScrollingChild.CurrentPosition);
461 ScrollDragEndEvent?.Invoke(this, eventArgs);
464 private void OnScrollAnimationStart()
466 ScrollEventArgs eventArgs = new ScrollEventArgs(mScrollingChild.CurrentPosition);
467 ScrollAnimationStartEvent?.Invoke(this, eventArgs);
470 private void OnScrollAnimationEnd()
472 ScrollEventArgs eventArgs = new ScrollEventArgs(mScrollingChild.CurrentPosition);
473 ScrollAnimationEndEvent?.Invoke(this, eventArgs);
476 private void OnScroll()
478 ScrollEventArgs eventArgs = new ScrollEventArgs(mScrollingChild.CurrentPosition);
479 ScrollEvent?.Invoke(this, eventArgs);
482 private void StopScroll()
484 if (scrollAnimation != null)
486 if (scrollAnimation.State == Animation.States.Playing)
488 Debug.WriteLineIf(LayoutDebugScrollableBase, "StopScroll Animation Playing");
489 scrollAnimation.Stop(Animation.EndActions.Cancel);
490 OnScrollAnimationEnd();
492 scrollAnimation.Clear();
496 // static constructor registers the control type
497 static ScrollableBase()
499 // ViewRegistry registers control type with DALi type registry
500 // also uses introspection to find any properties that need to be registered with type registry
501 CustomViewRegistry.Instance.Register(CreateInstance, typeof(ScrollableBase));
504 internal static CustomView CreateInstance()
506 return new ScrollableBase();
509 private void AnimateChildTo(int duration, float axisPosition)
511 Debug.WriteLineIf(LayoutDebugScrollableBase, "AnimationTo Animation Duration:" + duration + " Destination:" + axisPosition);
513 StopScroll(); // Will replace previous animation so will stop existing one.
515 if (scrollAnimation == null)
517 scrollAnimation = new Animation();
518 scrollAnimation.Finished += ScrollAnimationFinished;
521 scrollAnimation.Duration = duration;
522 scrollAnimation.DefaultAlphaFunction = new AlphaFunction(AlphaFunction.BuiltinFunctions.EaseOutSine);
523 scrollAnimation.AnimateTo(mScrollingChild, (ScrollingDirection == Direction.Horizontal) ? "PositionX" : "PositionY", axisPosition);
525 OnScrollAnimationStart();
526 scrollAnimation.Play();
529 private void ScrollBy(float displacement, bool animate)
531 if (GetChildCount() == 0 || displacement == 0 || maxScrollDistance < 0)
536 float childCurrentPosition = (ScrollingDirection == Direction.Horizontal) ? mScrollingChild.PositionX: mScrollingChild.PositionY;
538 Debug.WriteLineIf(LayoutDebugScrollableBase, "ScrollBy childCurrentPosition:" + childCurrentPosition +
539 " displacement:" + displacement,
540 " maxScrollDistance:" + maxScrollDistance );
542 childTargetPosition = childCurrentPosition + displacement; // child current position + gesture displacement
543 childTargetPosition = Math.Min(0,childTargetPosition);
544 childTargetPosition = Math.Max(-maxScrollDistance,childTargetPosition);
546 Debug.WriteLineIf( LayoutDebugScrollableBase, "ScrollBy currentAxisPosition:" + childCurrentPosition + "childTargetPosition:" + childTargetPosition);
550 // Calculate scroll animaton duration
551 float scrollDistance = 0.0f;
552 if (childCurrentPosition < childTargetPosition)
554 scrollDistance = Math.Abs(childCurrentPosition + childTargetPosition);
558 scrollDistance = Math.Abs(childCurrentPosition - childTargetPosition);
561 int duration = (int)((320*FlickAnimationSpeed) + (scrollDistance * FlickAnimationSpeed));
562 Debug.WriteLineIf(LayoutDebugScrollableBase, "Scroll Animation Duration:" + duration + " Distance:" + scrollDistance);
564 AnimateChildTo(duration, childTargetPosition);
568 // Set position of scrolling child without an animation
569 if (ScrollingDirection == Direction.Horizontal)
571 mScrollingChild.PositionX = childTargetPosition;
575 mScrollingChild.PositionY = childTargetPosition;
581 /// you can override it to clean-up your own resources.
583 /// <param name="type">DisposeTypes</param>
584 /// <since_tizen> 6 </since_tizen>
585 /// This will be public opened in tizen_5.5 after ACR done. Before ACR, need to be hidden as inhouse API.
586 [EditorBrowsable(EditorBrowsableState.Never)]
587 protected override void Dispose(DisposeTypes type)
594 if (type == DisposeTypes.Explicit)
598 if (mPanGestureDetector != null)
600 mPanGestureDetector.Detected -= OnPanGestureDetected;
601 mPanGestureDetector.Dispose();
602 mPanGestureDetector = null;
605 if (mTapGestureDetector != null)
607 mTapGestureDetector.Detected -= OnTapGestureDetected;
608 mTapGestureDetector.Dispose();
609 mTapGestureDetector = null;
615 private float CalculateDisplacementFromVelocity(float axisVelocity)
617 // Map: flick speed of range (2.0 - 6.0) to flick multiplier of range (0.7 - 1.6)
618 float speedMinimum = FlickThreshold;
619 float speedMaximum = FlickThreshold + 6.0f;
620 float multiplierMinimum = FlickDistanceMultiplierRange.X;
621 float multiplierMaximum = FlickDistanceMultiplierRange.Y;
623 float flickDisplacement = 0.0f;
625 float speed = Math.Min(4.0f,Math.Abs(axisVelocity));
627 Debug.WriteLineIf(LayoutDebugScrollableBase, "ScrollableBase Candidate Flick speed:" + speed);
629 if (speed > FlickThreshold)
631 // Flick length is the length of the ScrollableBase.
632 float flickLength = (ScrollingDirection == Direction.Horizontal) ?CurrentSize.Width:CurrentSize.Height;
634 // Calculate multiplier by mapping speed between the multiplier minimum and maximum.
635 multiplier =( (speed - speedMinimum) / ( (speedMaximum - speedMinimum) * (multiplierMaximum - multiplierMinimum) ) )+ multiplierMinimum;
637 // flick displacement is the product of the flick length and multiplier
638 flickDisplacement = ((flickLength * multiplier) * speed) / axisVelocity; // *speed and /velocity to perserve sign.
640 Debug.WriteLineIf(LayoutDebugScrollableBase, "Calculated FlickDisplacement[" + flickDisplacement +"] from speed[" + speed + "] multiplier:"
643 return flickDisplacement;
646 private float CalculateMaximumScrollDistance()
648 int scrollingChildLength = 0;
649 int scrollerLength = 0;
650 if (ScrollingDirection == Direction.Horizontal)
652 Debug.WriteLineIf(LayoutDebugScrollableBase, "Horizontal");
654 scrollingChildLength = (int)mScrollingChild.Layout.MeasuredWidth.Size.AsRoundedValue();
655 scrollerLength = CurrentSize.Width;
659 Debug.WriteLineIf(LayoutDebugScrollableBase, "Vertical");
660 scrollingChildLength = (int)mScrollingChild.Layout.MeasuredHeight.Size.AsRoundedValue();
661 scrollerLength = CurrentSize.Height;
664 Debug.WriteLineIf(LayoutDebugScrollableBase, "ScrollBy maxScrollDistance:" + (scrollingChildLength - scrollerLength) +
665 " parent length:" + scrollerLength +
666 " scrolling child length:" + scrollingChildLength);
668 return Math.Max(scrollingChildLength - scrollerLength,0);
671 private void PageSnap()
673 Debug.WriteLineIf(LayoutDebugScrollableBase, "PageSnap with pan candidate totalDisplacement:" + totalDisplacementForPan +
674 " currentPage[" + CurrentPage + "]" );
676 //Increment current page if total displacement enough to warrant a page change.
677 if (Math.Abs(totalDisplacementForPan) > (mPageWidth * ratioOfScreenWidthToCompleteScroll))
679 if (totalDisplacementForPan < 0)
681 CurrentPage = Math.Min(Math.Max(mScrollingChild.Children.Count - 1,0), ++CurrentPage);
685 CurrentPage = Math.Max(0, --CurrentPage);
689 // Animate to new page or reposition to current page
690 int destinationX = -(CurrentPage * mPageWidth);
691 Debug.WriteLineIf(LayoutDebugScrollableBase, "Snapping to page[" + CurrentPage + "] to:"+ destinationX + " from:" + mScrollingChild.PositionX);
692 AnimateChildTo(ScrollDuration, destinationX);
695 private void Flick(float flickDisplacement)
699 if ( ( flickWhenAnimating && scrolling == true) || ( scrolling == false) )
701 if(flickDisplacement < 0)
703 CurrentPage = Math.Min(Math.Max(mScrollingChild.Children.Count - 1,0), CurrentPage + 1);
704 Debug.WriteLineIf(LayoutDebugScrollableBase, "Snap - to page:" + CurrentPage);
708 CurrentPage = Math.Max(0, CurrentPage - 1);
709 Debug.WriteLineIf(LayoutDebugScrollableBase, "Snap + to page:" + CurrentPage);
711 float targetPosition = -(CurrentPage* mPageWidth); // page size
712 Debug.WriteLineIf(LayoutDebugScrollableBase, "Snapping to :" + targetPosition);
713 AnimateChildTo(ScrollDuration,targetPosition);
718 ScrollBy(flickDisplacement, true); // Animate flickDisplacement.
722 private void OnPanGestureDetected(object source, PanGestureDetector.DetectedEventArgs e)
724 if (e.PanGesture.State == Gesture.StateType.Started)
726 Debug.WriteLineIf(LayoutDebugScrollableBase, "Gesture Start");
727 if (scrolling && !SnapToPage)
731 maxScrollDistance = CalculateMaximumScrollDistance();
732 totalDisplacementForPan = 0.0f;
735 else if (e.PanGesture.State == Gesture.StateType.Continuing)
737 if (ScrollingDirection == Direction.Horizontal)
739 ScrollBy(e.PanGesture.Displacement.X, false);
740 totalDisplacementForPan += e.PanGesture.Displacement.X;
744 ScrollBy(e.PanGesture.Displacement.Y, false);
745 totalDisplacementForPan += e.PanGesture.Displacement.Y;
747 Debug.WriteLineIf(LayoutDebugScrollableBase, "OnPanGestureDetected Continue totalDisplacementForPan:" + totalDisplacementForPan);
749 else if (e.PanGesture.State == Gesture.StateType.Finished)
751 float axisVelocity = (ScrollingDirection == Direction.Horizontal) ? e.PanGesture.Velocity.X : e.PanGesture.Velocity.Y;
752 float flickDisplacement = CalculateDisplacementFromVelocity(axisVelocity);
754 Debug.WriteLineIf(LayoutDebugScrollableBase, "FlickDisplacement:" + flickDisplacement + "TotalDisplacementForPan:" + totalDisplacementForPan);
757 if (flickDisplacement > 0 | flickDisplacement < 0)// Flick detected
759 Flick(flickDisplacement);
763 // End of panning gesture but was not a flick
769 totalDisplacementForPan = 0;
773 private new void OnTapGestureDetected(object source, TapGestureDetector.DetectedEventArgs e)
775 if (e.TapGesture.Type == Gesture.GestureType.Tap)
777 // Stop scrolling if tap detected (press then relase).
778 // Unless in Pages mode, do not want a page change to stop part way.
779 if(scrolling && !SnapToPage)
786 private void ScrollAnimationFinished(object sender, EventArgs e)
789 OnScrollAnimationEnd();