در اندروید، اسکرول معمولاً با استفاده از کلاس ScrollView
انجام می شود. هر طرحبندی استانداردی را که ممکن است فراتر از محدوده محفظه آن گسترش یابد، در یک ScrollView
قرار دهید تا نمای قابل پیمایشی را ارائه دهد که توسط چارچوب مدیریت میشود. پیاده سازی یک اسکرول سفارشی فقط برای سناریوهای خاص ضروری است. این سند نحوه نمایش افکت اسکرول را در پاسخ به حرکات لمسی با استفاده از اسکرول توضیح می دهد.
برنامه شما میتواند از پیمایشها Scroller
یا OverScroller
- برای جمعآوری دادههای مورد نیاز برای تولید یک انیمیشن پیمایشی در پاسخ به یک رویداد لمسی استفاده کند. آنها مشابه هستند، اما OverScroller
همچنین شامل روش هایی برای نشان دادن زمانی است که کاربران پس از حرکت حرکت یا حرکت به لبه های محتوا می رسند.
- با شروع در Android 12 (سطح API 31)، عناصر بصری در یک رویداد درگ کشیده میشوند و به عقب باز میگردند و در یک رویداد پرتاب به عقب باز میگردند.
- در اندروید 11 (سطح API 30) و قبل از آن، مرزها یک جلوه "درخشش" را پس از حرکت کشیدن یا پرت کردن به لبه نشان می دهند.
نمونه InteractiveChart
در این سند از کلاس EdgeEffect
برای نمایش این افکت های overscroll استفاده می کند.
میتوانید با استفاده از فیزیک اسکرول استاندارد پلتفرم مانند اصطکاک، سرعت و کیفیتهای دیگر، از پیمایش برای متحرک کردن پیمایش در طول زمان استفاده کنید. خود اسکرول چیزی نمی کشد. اسکرولها در طول زمان انحرافهای پیمایش را برای شما دنبال میکنند، اما بهطور خودکار این موقعیتها را در نمای شما اعمال نمیکنند. شما باید مختصات جدید را با سرعتی دریافت و اعمال کنید که انیمیشن اسکرول را صاف به نظر برساند.
اصطلاحات اسکرول را درک کنید
اسکرول کلمه ای است که بسته به زمینه آن در اندروید می تواند معانی مختلفی داشته باشد.
اسکرول فرآیند کلی جابجایی درگاه نمایش است - یعنی "پنجره" محتوایی که به آن نگاه می کنید. هنگامی که پیمایش در هر دو محور x و y باشد، به آن پیمایش می گویند. برنامه نمونه InteractiveChart
در این سند دو نوع مختلف اسکرول، کشیدن و پرت کردن را نشان می دهد:
- کشیدن: این نوع پیمایشی است که زمانی اتفاق میافتد که کاربر انگشت خود را روی صفحه لمسی بکشد. میتوانید کشیدن را با نادیده گرفتن
onScroll()
درGestureDetector.OnGestureListener
پیادهسازی کنید. برای اطلاعات بیشتر در مورد کشیدن، به کشیدن و مقیاس نگاه کنید. - Flinging: این نوعی پیمایش است که زمانی رخ می دهد که کاربر انگشت خود را به سرعت بکشد و بلند کند. پس از اینکه کاربر انگشت خود را بلند کرد، عموماً می خواهید به حرکت درگاه نمایش ادامه دهید، اما سرعت را تا زمانی که درگاه نمایش متوقف شود، کاهش دهید. میتوانید flinging را با نادیده گرفتن
onFling()
درGestureDetector.OnGestureListener
و با استفاده از یک شی اسکرول پیادهسازی کنید. - Panning: پیمایش همزمان در امتداد هر دو محور x و y را پیمایش میگویند.
استفاده از اشیاء اسکرول همراه با ژست پرتاب معمول است، اما میتوانید از آنها در هر زمینهای که میخواهید رابط کاربری اسکرول را در پاسخ به یک رویداد لمسی نمایش دهد، استفاده کنید. برای مثال، میتوانید onTouchEvent()
را لغو کنید تا رویدادهای لمسی را مستقیماً پردازش کنید و در پاسخ به آن رویدادهای لمسی، یک افکت اسکرول یا یک انیمیشن «snap-to-page» تولید کنید.
مولفه هایی که شامل پیاده سازی های اسکرول داخلی هستند
اجزای Android زیر شامل پشتیبانی داخلی برای رفتار پیمایش و اسکرول است:
-
GridView
-
HorizontalScrollView
-
ListView
-
NestedScrollView
-
RecyclerView
-
ScrollView
-
ViewPager
-
ViewPager2
اگر برنامه شما نیاز به پشتیبانی از پیمایش و اسکرول در یک مؤلفه دیگر دارد، مراحل زیر را کامل کنید:
- یک پیادهسازی پیمایش مبتنی بر لمسی سفارشی ایجاد کنید .
- برای پشتیبانی از دستگاههایی که Android 12 و جدیدتر را اجرا میکنند، جلوه کششی overscroll را اجرا کنید .
یک پیاده سازی اسکرول مبتنی بر لمس سفارشی ایجاد کنید
اگر برنامه شما از مؤلفهای استفاده میکند که شامل پشتیبانی داخلی برای اسکرول و اسکرول نیست، این بخش نحوه ایجاد پیمایش خود را توضیح میدهد.
قطعه زیر از نمونه InteractiveChart
می آید. از یک GestureDetector
استفاده می کند و متد onFling()
GestureDetector.SimpleOnGestureListener
را لغو می کند. از OverScroller
برای ردیابی ژست پرت کردن استفاده می کند. اگر کاربر پس از انجام ژست پرت کردن به لبه های محتوا برسد، ظرف زمانی را نشان می دهد که کاربر به پایان محتوا می رسد. نشانه بستگی به نسخه اندرویدی دارد که دستگاه اجرا می کند:
- در اندروید 12 و نسخههای جدیدتر، عناصر بصری کشیده شده و به عقب باز میگردند.
- در اندروید 11 و نسخههای قبلی، عناصر بصری جلوهای درخشش نشان میدهند.
قسمت اول قطعه زیر اجرای onFling()
را نشان می دهد:
کاتلین
// Viewport extremes. See currentViewport for a discussion of the viewport. private val AXIS_X_MIN = -1f private val AXIS_X_MAX = 1f private val AXIS_Y_MIN = -1f private val AXIS_Y_MAX = 1f // The current viewport. This rectangle represents the visible chart // domain and range. The viewport is the part of the app that the // user manipulates via touch gestures. private val currentViewport = RectF(AXIS_X_MIN, AXIS_Y_MIN, AXIS_X_MAX, AXIS_Y_MAX) // The current destination rectangle—in pixel coordinates—into which // the chart data must be drawn. private lateinit var contentRect: Rect private lateinit var scroller: OverScroller private lateinit var scrollerStartViewport: RectF ... private val gestureListener = object : GestureDetector.SimpleOnGestureListener() { override fun onDown(e: MotionEvent): Boolean { // Initiates the decay phase of any active edge effects. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects() } scrollerStartViewport.set(currentViewport) // Aborts any active scroll animations and invalidates. scroller.forceFinished(true) ViewCompat.postInvalidateOnAnimation(this@InteractiveLineGraphView) return true } ... override fun onFling( e1: MotionEvent, e2: MotionEvent, velocityX: Float, velocityY: Float ): Boolean { fling((-velocityX).toInt(), (-velocityY).toInt()) return true } } private fun fling(velocityX: Int, velocityY: Int) { // Initiates the decay phase of any active edge effects. // On Android 12 and later, the edge effect (stretch) must // continue. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects() } // Flings use math in pixels, as opposed to math based on the viewport. val surfaceSize: Point = computeScrollSurfaceSize() val (startX: Int, startY: Int) = scrollerStartViewport.run { set(currentViewport) (surfaceSize.x * (left - AXIS_X_MIN) / (AXIS_X_MAX - AXIS_X_MIN)).toInt() to (surfaceSize.y * (AXIS_Y_MAX - bottom) / (AXIS_Y_MAX - AXIS_Y_MIN)).toInt() } // Before flinging, stops the current animation. scroller.forceFinished(true) // Begins the animation. scroller.fling( // Current scroll position. startX, startY, velocityX, velocityY, /* * Minimum and maximum scroll positions. The minimum scroll * position is generally 0 and the maximum scroll position * is generally the content size less the screen size. So if the * content width is 1000 pixels and the screen width is 200 * pixels, the maximum scroll offset is 800 pixels. */ 0, surfaceSize.x - contentRect.width(), 0, surfaceSize.y - contentRect.height(), // The edges of the content. This comes into play when using // the EdgeEffect class to draw "glow" overlays. contentRect.width() / 2, contentRect.height() / 2 ) // Invalidates to trigger computeScroll(). ViewCompat.postInvalidateOnAnimation(this) }
جاوا
// Viewport extremes. See currentViewport for a discussion of the viewport. private static final float AXIS_X_MIN = -1f; private static final float AXIS_X_MAX = 1f; private static final float AXIS_Y_MIN = -1f; private static final float AXIS_Y_MAX = 1f; // The current viewport. This rectangle represents the visible chart // domain and range. The viewport is the part of the app that the // user manipulates via touch gestures. private RectF currentViewport = new RectF(AXIS_X_MIN, AXIS_Y_MIN, AXIS_X_MAX, AXIS_Y_MAX); // The current destination rectangle—in pixel coordinates—into which // the chart data must be drawn. private final Rect contentRect = new Rect(); private final OverScroller scroller; private final RectF scrollerStartViewport = new RectF(); // Used only for zooms and flings. ... private final GestureDetector.SimpleOnGestureListener gestureListener = new GestureDetector.SimpleOnGestureListener() { @Override public boolean onDown(MotionEvent e) { if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects(); } scrollerStartViewport.set(currentViewport); scroller.forceFinished(true); ViewCompat.postInvalidateOnAnimation(InteractiveLineGraphView.this); return true; } ... @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { fling((int) -velocityX, (int) -velocityY); return true; } }; private void fling(int velocityX, int velocityY) { // Initiates the decay phase of any active edge effects. // On Android 12 and later, the edge effect (stretch) must // continue. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects(); } // Flings use math in pixels, as opposed to math based on the viewport. Point surfaceSize = computeScrollSurfaceSize(); scrollerStartViewport.set(currentViewport); int startX = (int) (surfaceSize.x * (scrollerStartViewport.left - AXIS_X_MIN) / ( AXIS_X_MAX - AXIS_X_MIN)); int startY = (int) (surfaceSize.y * (AXIS_Y_MAX - scrollerStartViewport.bottom) / ( AXIS_Y_MAX - AXIS_Y_MIN)); // Before flinging, stops the current animation. scroller.forceFinished(true); // Begins the animation. scroller.fling( // Current scroll position. startX, startY, velocityX, velocityY, /* * Minimum and maximum scroll positions. The minimum scroll * position is generally 0 and the maximum scroll position * is generally the content size less the screen size. So if the * content width is 1000 pixels and the screen width is 200 * pixels, the maximum scroll offset is 800 pixels. */ 0, surfaceSize.x - contentRect.width(), 0, surfaceSize.y - contentRect.height(), // The edges of the content. This comes into play when using // the EdgeEffect class to draw "glow" overlays. contentRect.width() / 2, contentRect.height() / 2); // Invalidates to trigger computeScroll(). ViewCompat.postInvalidateOnAnimation(this); }
هنگامی که onFling()
postInvalidateOnAnimation()
را فراخوانی می کند، computeScroll()
برای به روز رسانی مقادیر x و y فعال می کند. همانطور که در مثال قبل نشان داده شده است، این معمولاً زمانی انجام میشود که یک view child با استفاده از یک شیء پیمایشی، یک اسکرول را متحرک میکند.
اکثر نماها موقعیت x و y شی اسکرولر را مستقیماً به scrollTo()
منتقل می کنند. پیادهسازی computeScroll()
رویکرد متفاوتی دارد: computeScrollOffset()
را برای بدست آوردن مکان فعلی x و y فراخوانی میکند. وقتی معیارهای نمایش افکت لبه "درخشش" overscroll برآورده می شود - یعنی نمایشگر بزرگنمایی شده است، x یا y خارج از محدوده است و برنامه قبلاً یک overscroll را نشان نمی دهد - کد درخشش overscroll را تنظیم می کند. افکت کرده و postInvalidateOnAnimation()
را فراخوانی می کند تا یک invalidate در view ایجاد کند.
کاتلین
// Edge effect/overscroll tracking objects. private lateinit var edgeEffectTop: EdgeEffect private lateinit var edgeEffectBottom: EdgeEffect private lateinit var edgeEffectLeft: EdgeEffect private lateinit var edgeEffectRight: EdgeEffect private var edgeEffectTopActive: Boolean = false private var edgeEffectBottomActive: Boolean = false private var edgeEffectLeftActive: Boolean = false private var edgeEffectRightActive: Boolean = false override fun computeScroll() { super.computeScroll() var needsInvalidate = false // The scroller isn't finished, meaning a fling or // programmatic pan operation is active. if (scroller.computeScrollOffset()) { val surfaceSize: Point = computeScrollSurfaceSize() val currX: Int = scroller.currX val currY: Int = scroller.currY val (canScrollX: Boolean, canScrollY: Boolean) = currentViewport.run { (left > AXIS_X_MIN || right < AXIS_X_MAX) to (top > AXIS_Y_MIN || bottom < AXIS_Y_MAX) } /* * If you are zoomed in, currX or currY is * outside of bounds, and you aren't already * showing overscroll, then render the overscroll * glow edge effect. */ if (canScrollX && currX < 0 && edgeEffectLeft.isFinished && !edgeEffectLeftActive) { edgeEffectLeft.onAbsorb(scroller.currVelocity.toInt()) edgeEffectLeftActive = true needsInvalidate = true } else if (canScrollX && currX > surfaceSize.x - contentRect.width() && edgeEffectRight.isFinished && !edgeEffectRightActive) { edgeEffectRight.onAbsorb(scroller.currVelocity.toInt()) edgeEffectRightActive = true needsInvalidate = true } if (canScrollY && currY < 0 && edgeEffectTop.isFinished && !edgeEffectTopActive) { edgeEffectTop.onAbsorb(scroller.currVelocity.toInt()) edgeEffectTopActive = true needsInvalidate = true } else if (canScrollY && currY > surfaceSize.y - contentRect.height() && edgeEffectBottom.isFinished && !edgeEffectBottomActive) { edgeEffectBottom.onAbsorb(scroller.currVelocity.toInt()) edgeEffectBottomActive = true needsInvalidate = true } ... } }
جاوا
// Edge effect/overscroll tracking objects. private EdgeEffectCompat edgeEffectTop; private EdgeEffectCompat edgeEffectBottom; private EdgeEffectCompat edgeEffectLeft; private EdgeEffectCompat edgeEffectRight; private boolean edgeEffectTopActive; private boolean edgeEffectBottomActive; private boolean edgeEffectLeftActive; private boolean edgeEffectRightActive; @Override public void computeScroll() { super.computeScroll(); boolean needsInvalidate = false; // The scroller isn't finished, meaning a fling or // programmatic pan operation is active. if (scroller.computeScrollOffset()) { Point surfaceSize = computeScrollSurfaceSize(); int currX = scroller.getCurrX(); int currY = scroller.getCurrY(); boolean canScrollX = (currentViewport.left > AXIS_X_MIN || currentViewport.right < AXIS_X_MAX); boolean canScrollY = (currentViewport.top > AXIS_Y_MIN || currentViewport.bottom < AXIS_Y_MAX); /* * If you are zoomed in, currX or currY is * outside of bounds, and you aren't already * showing overscroll, then render the overscroll * glow edge effect. */ if (canScrollX && currX < 0 && edgeEffectLeft.isFinished() && !edgeEffectLeftActive) { edgeEffectLeft.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectLeftActive = true; needsInvalidate = true; } else if (canScrollX && currX > (surfaceSize.x - contentRect.width()) && edgeEffectRight.isFinished() && !edgeEffectRightActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectRightActive = true; needsInvalidate = true; } if (canScrollY && currY < 0 && edgeEffectTop.isFinished() && !edgeEffectTopActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectTopActive = true; needsInvalidate = true; } else if (canScrollY && currY > (surfaceSize.y - contentRect.height()) && edgeEffectBottom.isFinished() && !edgeEffectBottomActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectBottomActive = true; needsInvalidate = true; } ... }
در اینجا بخشی از کد است که بزرگنمایی واقعی را انجام می دهد:
کاتلین
lateinit var zoomer: Zoomer val zoomFocalPoint = PointF() ... // If a zoom is in progress—either programmatically // or through double touch—this performs the zoom. if (zoomer.computeZoom()) { val newWidth: Float = (1f - zoomer.currZoom) * scrollerStartViewport.width() val newHeight: Float = (1f - zoomer.currZoom) * scrollerStartViewport.height() val pointWithinViewportX: Float = (zoomFocalPoint.x - scrollerStartViewport.left) / scrollerStartViewport.width() val pointWithinViewportY: Float = (zoomFocalPoint.y - scrollerStartViewport.top) / scrollerStartViewport.height() currentViewport.set( zoomFocalPoint.x - newWidth * pointWithinViewportX, zoomFocalPoint.y - newHeight * pointWithinViewportY, zoomFocalPoint.x + newWidth * (1 - pointWithinViewportX), zoomFocalPoint.y + newHeight * (1 - pointWithinViewportY) ) constrainViewport() needsInvalidate = true } if (needsInvalidate) { ViewCompat.postInvalidateOnAnimation(this) }
جاوا
// Custom object that is functionally similar to Scroller. Zoomer zoomer; private PointF zoomFocalPoint = new PointF(); ... // If a zoom is in progress—either programmatically // or through double touch—this performs the zoom. if (zoomer.computeZoom()) { float newWidth = (1f - zoomer.getCurrZoom()) * scrollerStartViewport.width(); float newHeight = (1f - zoomer.getCurrZoom()) * scrollerStartViewport.height(); float pointWithinViewportX = (zoomFocalPoint.x - scrollerStartViewport.left) / scrollerStartViewport.width(); float pointWithinViewportY = (zoomFocalPoint.y - scrollerStartViewport.top) / scrollerStartViewport.height(); currentViewport.set( zoomFocalPoint.x - newWidth * pointWithinViewportX, zoomFocalPoint.y - newHeight * pointWithinViewportY, zoomFocalPoint.x + newWidth * (1 - pointWithinViewportX), zoomFocalPoint.y + newHeight * (1 - pointWithinViewportY)); constrainViewport(); needsInvalidate = true; } if (needsInvalidate) { ViewCompat.postInvalidateOnAnimation(this); }
این متد computeScrollSurfaceSize()
است که در قطعه قبلی فراخوانی شده است. اندازه سطح قابل پیمایش فعلی را بر حسب پیکسل محاسبه می کند. به عنوان مثال، اگر کل منطقه نمودار قابل مشاهده است، این اندازه فعلی mContentRect
است. اگر نمودار در هر دو جهت 200 درصد بزرگنمایی شود، اندازه برگشتی دو برابر بزرگتر بهصورت افقی و عمودی است.
کاتلین
private fun computeScrollSurfaceSize(): Point { return Point( (contentRect.width() * (AXIS_X_MAX - AXIS_X_MIN) / currentViewport.width()).toInt(), (contentRect.height() * (AXIS_Y_MAX - AXIS_Y_MIN) / currentViewport.height()).toInt() ) }
جاوا
private Point computeScrollSurfaceSize() { return new Point( (int) (contentRect.width() * (AXIS_X_MAX - AXIS_X_MIN) / currentViewport.width()), (int) (contentRect.height() * (AXIS_Y_MAX - AXIS_Y_MIN) / currentViewport.height())); }
برای مثال دیگری از استفاده از اسکرول، کد منبع کلاس ViewPager
را ببینید. در پاسخ به flings پیمایش می کند و از اسکرول برای اجرای انیمیشن "snap-to-page" استفاده می کند.
افکت کشش overscroll را اجرا کنید
با شروع اندروید 12، EdgeEffect
API های زیر را برای اجرای افکت Overscroll کششی اضافه می کند:
-
getDistance()
-
onPullDistance()
برای ارائه بهترین تجربه کاربری با کشش overscroll، موارد زیر را انجام دهید:
- هنگامی که وقتی کاربر محتویات را لمس می کند انیمیشن کششی اعمال می شود، لمس را به عنوان "گیر" ثبت کنید. کاربر انیمیشن را متوقف می کند و دوباره شروع به دستکاری کشش می کند.
- وقتی کاربر انگشت خود را در جهت مخالف کشش حرکت میدهد، کشش را رها کنید تا کاملاً از بین برود و سپس پیمایش را شروع کنید.
- هنگامی که کاربر در حین کشش پرت می کند،
EdgeEffect
برای افزایش افکت کشش پرتاب کنید.
انیمیشن را بگیرید
وقتی کاربر یک انیمیشن کششی فعال را میگیرد، EdgeEffect.getDistance()
0
برمیگرداند. این حالت نشان می دهد که کشش باید با حرکت لمسی دستکاری شود. در اکثر کانتینرها، catch در onInterceptTouchEvent()
شناسایی می شود، همانطور که در قطعه کد زیر نشان داده شده است:
کاتلین
override fun onInterceptTouchEvent(ev: MotionEvent): Boolean { ... when (action and MotionEvent.ACTION_MASK) { MotionEvent.ACTION_DOWN -> ... isBeingDragged = EdgeEffectCompat.getDistance(edgeEffectBottom) > 0f || EdgeEffectCompat.getDistance(edgeEffectTop) > 0f ... } return isBeingDragged }
جاوا
@Override public boolean onInterceptTouchEvent(MotionEvent ev) { ... switch (action & MotionEvent.ACTION_MASK) { case MotionEvent.ACTION_DOWN: ... isBeingDragged = EdgeEffectCompat.getDistance(edgeEffectBottom) > 0 || EdgeEffectCompat.getDistance(edgeEffectTop) > 0; ... } }
در مثال قبل، زمانی که mIsBeingDragged
true
باشد onInterceptTouchEvent()
true
برمی گرداند، بنابراین کافی است رویداد را قبل از اینکه کودک فرصتی برای مصرف داشته باشد مصرف کنید.
افکت overscroll را رها کنید
مهم است که جلوه کشش را قبل از پیمایش آزاد کنید تا از اعمال کشش به محتوای اسکرول جلوگیری کنید. نمونه کد زیر این بهترین روش را اعمال می کند:
کاتلین
override fun onTouchEvent(ev: MotionEvent): Boolean { val activePointerIndex = ev.actionIndex when (ev.getActionMasked()) { MotionEvent.ACTION_MOVE -> val x = ev.getX(activePointerIndex) val y = ev.getY(activePointerIndex) var deltaY = y - lastMotionY val pullDistance = deltaY / height val displacement = x / width if (deltaY < 0f && EdgeEffectCompat.getDistance(edgeEffectTop) > 0f) { deltaY -= height * EdgeEffectCompat.onPullDistance(edgeEffectTop, pullDistance, displacement); } if (deltaY > 0f && EdgeEffectCompat.getDistance(edgeEffectBottom) > 0f) { deltaY += height * EdgeEffectCompat.onPullDistance(edgeEffectBottom, -pullDistance, 1 - displacement); } ... }
جاوا
@Override public boolean onTouchEvent(MotionEvent ev) { final int actionMasked = ev.getActionMasked(); switch (actionMasked) { case MotionEvent.ACTION_MOVE: final float x = ev.getX(activePointerIndex); final float y = ev.getY(activePointerIndex); float deltaY = y - lastMotionY; float pullDistance = deltaY / getHeight(); float displacement = x / getWidth(); if (deltaY < 0 && EdgeEffectCompat.getDistance(edgeEffectTop) > 0) { deltaY -= getHeight() * EdgeEffectCompat.onPullDistance(edgeEffectTop, pullDistance, displacement); } if (deltaY > 0 && EdgeEffectCompat.getDistance(edgeEffectBottom) > 0) { deltaY += getHeight() * EdgeEffectCompat.onPullDistance(edgeEffectBottom, -pullDistance, 1 - displacement); } ...
هنگامی که کاربر در حال کشیدن است، قبل از اینکه رویداد لمسی را به محفظه پیمایشی تو در تو منتقل کنید یا اسکرول را بکشید، فاصله کشش EdgeEffect
را مصرف کنید. در نمونه کد قبلی، getDistance()
یک مقدار مثبت را هنگامی که یک افکت لبه نمایش داده می شود برمی گرداند و می تواند با حرکت آزاد شود. هنگامی که رویداد لمسی کشش را آزاد می کند، ابتدا توسط EdgeEffect
مصرف می شود تا قبل از نمایش جلوه های دیگر، مانند پیمایش تو در تو، به طور کامل آزاد شود. میتوانید از getDistance()
استفاده کنید تا بفهمید چقدر فاصله کششی برای آزاد کردن اثر فعلی لازم است.
برخلاف onPull()
، onPullDistance()
مقدار مصرف شده دلتای عبور شده را برمی گرداند. در اندروید 12، اگر onPull()
یا onPullDistance()
به مقادیر deltaDistance
منفی منتقل شوند زمانی که getDistance()
0
باشد، افکت کشش تغییر نمی کند. در اندروید 11 و نسخههای قبلی، onPull()
به مقادیر منفی برای کل مسافت اجازه میدهد جلوههای درخشندگی را نشان دهد.
از overscroll انصراف دهید
میتوانید از overscroll در فایل طرحبندی خود یا به صورت برنامهنویسی انصراف دهید.
برای انصراف در فایل طرح بندی خود، android:overScrollMode
را مانند مثال زیر تنظیم کنید:
<MyCustomView android:overScrollMode="never"> ... </MyCustomView>
برای انصراف از طریق برنامه، از کدهایی مانند زیر استفاده کنید:
کاتلین
customView.overScrollMode = View.OVER_SCROLL_NEVER
جاوا
customView.setOverScrollMode(View.OVER_SCROLL_NEVER);
منابع اضافی
به منابع مرتبط زیر مراجعه کنید:
، در اندروید، اسکرول معمولاً با استفاده از کلاس ScrollView
انجام می شود. هر طرحبندی استانداردی را که ممکن است فراتر از محدوده محفظه آن گسترش یابد، در یک ScrollView
قرار دهید تا نمای قابل پیمایشی را ارائه دهد که توسط چارچوب مدیریت میشود. پیاده سازی یک اسکرول سفارشی فقط برای سناریوهای خاص ضروری است. این سند نحوه نمایش افکت اسکرول را در پاسخ به حرکات لمسی با استفاده از اسکرول توضیح می دهد.
برنامه شما میتواند از پیمایشها Scroller
یا OverScroller
- برای جمعآوری دادههای مورد نیاز برای تولید یک انیمیشن پیمایشی در پاسخ به یک رویداد لمسی استفاده کند. آنها مشابه هستند، اما OverScroller
همچنین شامل روش هایی برای نشان دادن زمانی است که کاربران پس از حرکت حرکت یا حرکت به لبه های محتوا می رسند.
- با شروع در Android 12 (سطح API 31)، عناصر بصری در یک رویداد درگ کشیده میشوند و به عقب باز میگردند و در یک رویداد پرتاب به عقب باز میگردند.
- در اندروید 11 (سطح API 30) و قبل از آن، مرزها یک جلوه "درخشش" را پس از حرکت کشیدن یا پرت کردن به لبه نشان می دهند.
نمونه InteractiveChart
در این سند از کلاس EdgeEffect
برای نمایش این افکت های overscroll استفاده می کند.
میتوانید با استفاده از فیزیک اسکرول استاندارد پلتفرم مانند اصطکاک، سرعت و کیفیتهای دیگر، از پیمایش برای متحرک کردن پیمایش در طول زمان استفاده کنید. خود اسکرول چیزی نمی کشد. اسکرولها در طول زمان انحرافهای پیمایش را برای شما دنبال میکنند، اما بهطور خودکار این موقعیتها را در نمای شما اعمال نمیکنند. شما باید مختصات جدید را با سرعتی دریافت و اعمال کنید که انیمیشن اسکرول را صاف به نظر برساند.
اصطلاحات اسکرول را درک کنید
اسکرول کلمه ای است که بسته به زمینه آن در اندروید می تواند معانی مختلفی داشته باشد.
اسکرول فرآیند کلی جابجایی درگاه نمایش است - یعنی "پنجره" محتوایی که به آن نگاه می کنید. هنگامی که پیمایش در هر دو محور x و y باشد، به آن پیمایش می گویند. برنامه نمونه InteractiveChart
در این سند دو نوع مختلف اسکرول، کشیدن و پرت کردن را نشان می دهد:
- کشیدن: این نوع پیمایشی است که زمانی اتفاق میافتد که کاربر انگشت خود را روی صفحه لمسی بکشد. میتوانید کشیدن را با نادیده گرفتن
onScroll()
درGestureDetector.OnGestureListener
پیادهسازی کنید. برای اطلاعات بیشتر در مورد کشیدن، به کشیدن و مقیاس نگاه کنید. - Flinging: این نوعی پیمایش است که زمانی رخ می دهد که کاربر انگشت خود را به سرعت بکشد و بلند کند. پس از اینکه کاربر انگشت خود را بلند کرد، عموماً می خواهید به حرکت درگاه نمایش ادامه دهید، اما سرعت را تا زمانی که درگاه نمایش متوقف شود، کاهش دهید. میتوانید flinging را با نادیده گرفتن
onFling()
درGestureDetector.OnGestureListener
و با استفاده از یک شی اسکرول پیادهسازی کنید. - Panning: پیمایش همزمان در امتداد هر دو محور x و y را پیمایش میگویند.
استفاده از اشیاء اسکرول همراه با ژست پرتاب معمول است، اما میتوانید از آنها در هر زمینهای که میخواهید رابط کاربری اسکرول را در پاسخ به یک رویداد لمسی نمایش دهد، استفاده کنید. برای مثال، میتوانید onTouchEvent()
را لغو کنید تا رویدادهای لمسی را مستقیماً پردازش کنید و در پاسخ به آن رویدادهای لمسی، یک افکت اسکرول یا یک انیمیشن «snap-to-page» تولید کنید.
مولفه هایی که شامل پیاده سازی های اسکرول داخلی هستند
اجزای Android زیر شامل پشتیبانی داخلی برای رفتار پیمایش و اسکرول است:
-
GridView
-
HorizontalScrollView
-
ListView
-
NestedScrollView
-
RecyclerView
-
ScrollView
-
ViewPager
-
ViewPager2
اگر برنامه شما نیاز به پشتیبانی از پیمایش و اسکرول در یک مؤلفه دیگر دارد، مراحل زیر را کامل کنید:
- یک پیادهسازی پیمایش مبتنی بر لمسی سفارشی ایجاد کنید .
- برای پشتیبانی از دستگاههایی که Android 12 و جدیدتر را اجرا میکنند، جلوه کششی overscroll را اجرا کنید .
یک پیاده سازی اسکرول مبتنی بر لمس سفارشی ایجاد کنید
اگر برنامه شما از مؤلفهای استفاده میکند که شامل پشتیبانی داخلی برای اسکرول و اسکرول نیست، این بخش نحوه ایجاد پیمایش خود را توضیح میدهد.
قطعه زیر از نمونه InteractiveChart
می آید. از یک GestureDetector
استفاده می کند و متد onFling()
GestureDetector.SimpleOnGestureListener
را لغو می کند. از OverScroller
برای ردیابی ژست پرت کردن استفاده می کند. اگر کاربر پس از انجام ژست پرت کردن به لبه های محتوا برسد، ظرف زمانی را نشان می دهد که کاربر به پایان محتوا می رسد. نشانه بستگی به نسخه اندرویدی دارد که دستگاه اجرا می کند:
- در اندروید 12 و نسخههای جدیدتر، عناصر بصری کشیده شده و به عقب باز میگردند.
- در اندروید 11 و نسخههای قبلی، عناصر بصری جلوهای درخشش نشان میدهند.
قسمت اول قطعه زیر اجرای onFling()
را نشان می دهد:
کاتلین
// Viewport extremes. See currentViewport for a discussion of the viewport. private val AXIS_X_MIN = -1f private val AXIS_X_MAX = 1f private val AXIS_Y_MIN = -1f private val AXIS_Y_MAX = 1f // The current viewport. This rectangle represents the visible chart // domain and range. The viewport is the part of the app that the // user manipulates via touch gestures. private val currentViewport = RectF(AXIS_X_MIN, AXIS_Y_MIN, AXIS_X_MAX, AXIS_Y_MAX) // The current destination rectangle—in pixel coordinates—into which // the chart data must be drawn. private lateinit var contentRect: Rect private lateinit var scroller: OverScroller private lateinit var scrollerStartViewport: RectF ... private val gestureListener = object : GestureDetector.SimpleOnGestureListener() { override fun onDown(e: MotionEvent): Boolean { // Initiates the decay phase of any active edge effects. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects() } scrollerStartViewport.set(currentViewport) // Aborts any active scroll animations and invalidates. scroller.forceFinished(true) ViewCompat.postInvalidateOnAnimation(this@InteractiveLineGraphView) return true } ... override fun onFling( e1: MotionEvent, e2: MotionEvent, velocityX: Float, velocityY: Float ): Boolean { fling((-velocityX).toInt(), (-velocityY).toInt()) return true } } private fun fling(velocityX: Int, velocityY: Int) { // Initiates the decay phase of any active edge effects. // On Android 12 and later, the edge effect (stretch) must // continue. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects() } // Flings use math in pixels, as opposed to math based on the viewport. val surfaceSize: Point = computeScrollSurfaceSize() val (startX: Int, startY: Int) = scrollerStartViewport.run { set(currentViewport) (surfaceSize.x * (left - AXIS_X_MIN) / (AXIS_X_MAX - AXIS_X_MIN)).toInt() to (surfaceSize.y * (AXIS_Y_MAX - bottom) / (AXIS_Y_MAX - AXIS_Y_MIN)).toInt() } // Before flinging, stops the current animation. scroller.forceFinished(true) // Begins the animation. scroller.fling( // Current scroll position. startX, startY, velocityX, velocityY, /* * Minimum and maximum scroll positions. The minimum scroll * position is generally 0 and the maximum scroll position * is generally the content size less the screen size. So if the * content width is 1000 pixels and the screen width is 200 * pixels, the maximum scroll offset is 800 pixels. */ 0, surfaceSize.x - contentRect.width(), 0, surfaceSize.y - contentRect.height(), // The edges of the content. This comes into play when using // the EdgeEffect class to draw "glow" overlays. contentRect.width() / 2, contentRect.height() / 2 ) // Invalidates to trigger computeScroll(). ViewCompat.postInvalidateOnAnimation(this) }
جاوا
// Viewport extremes. See currentViewport for a discussion of the viewport. private static final float AXIS_X_MIN = -1f; private static final float AXIS_X_MAX = 1f; private static final float AXIS_Y_MIN = -1f; private static final float AXIS_Y_MAX = 1f; // The current viewport. This rectangle represents the visible chart // domain and range. The viewport is the part of the app that the // user manipulates via touch gestures. private RectF currentViewport = new RectF(AXIS_X_MIN, AXIS_Y_MIN, AXIS_X_MAX, AXIS_Y_MAX); // The current destination rectangle—in pixel coordinates—into which // the chart data must be drawn. private final Rect contentRect = new Rect(); private final OverScroller scroller; private final RectF scrollerStartViewport = new RectF(); // Used only for zooms and flings. ... private final GestureDetector.SimpleOnGestureListener gestureListener = new GestureDetector.SimpleOnGestureListener() { @Override public boolean onDown(MotionEvent e) { if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects(); } scrollerStartViewport.set(currentViewport); scroller.forceFinished(true); ViewCompat.postInvalidateOnAnimation(InteractiveLineGraphView.this); return true; } ... @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { fling((int) -velocityX, (int) -velocityY); return true; } }; private void fling(int velocityX, int velocityY) { // Initiates the decay phase of any active edge effects. // On Android 12 and later, the edge effect (stretch) must // continue. if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) { releaseEdgeEffects(); } // Flings use math in pixels, as opposed to math based on the viewport. Point surfaceSize = computeScrollSurfaceSize(); scrollerStartViewport.set(currentViewport); int startX = (int) (surfaceSize.x * (scrollerStartViewport.left - AXIS_X_MIN) / ( AXIS_X_MAX - AXIS_X_MIN)); int startY = (int) (surfaceSize.y * (AXIS_Y_MAX - scrollerStartViewport.bottom) / ( AXIS_Y_MAX - AXIS_Y_MIN)); // Before flinging, stops the current animation. scroller.forceFinished(true); // Begins the animation. scroller.fling( // Current scroll position. startX, startY, velocityX, velocityY, /* * Minimum and maximum scroll positions. The minimum scroll * position is generally 0 and the maximum scroll position * is generally the content size less the screen size. So if the * content width is 1000 pixels and the screen width is 200 * pixels, the maximum scroll offset is 800 pixels. */ 0, surfaceSize.x - contentRect.width(), 0, surfaceSize.y - contentRect.height(), // The edges of the content. This comes into play when using // the EdgeEffect class to draw "glow" overlays. contentRect.width() / 2, contentRect.height() / 2); // Invalidates to trigger computeScroll(). ViewCompat.postInvalidateOnAnimation(this); }
هنگامی که onFling()
postInvalidateOnAnimation()
را فراخوانی می کند، computeScroll()
برای به روز رسانی مقادیر x و y فعال می کند. همانطور که در مثال قبل نشان داده شده است، این معمولاً زمانی انجام میشود که یک view child با استفاده از یک شیء پیمایشی، یک اسکرول را متحرک میکند.
اکثر نماها موقعیت x و y شی اسکرولر را مستقیماً به scrollTo()
منتقل می کنند. پیادهسازی computeScroll()
رویکرد متفاوتی دارد: computeScrollOffset()
را برای بدست آوردن مکان فعلی x و y فراخوانی میکند. وقتی معیارهای نمایش افکت لبه "درخشش" overscroll برآورده می شود - یعنی نمایشگر بزرگنمایی شده است، x یا y خارج از محدوده است و برنامه قبلاً یک overscroll را نشان نمی دهد - کد درخشش overscroll را تنظیم می کند. افکت کرده و postInvalidateOnAnimation()
را فراخوانی می کند تا یک invalidate در view ایجاد کند.
کاتلین
// Edge effect/overscroll tracking objects. private lateinit var edgeEffectTop: EdgeEffect private lateinit var edgeEffectBottom: EdgeEffect private lateinit var edgeEffectLeft: EdgeEffect private lateinit var edgeEffectRight: EdgeEffect private var edgeEffectTopActive: Boolean = false private var edgeEffectBottomActive: Boolean = false private var edgeEffectLeftActive: Boolean = false private var edgeEffectRightActive: Boolean = false override fun computeScroll() { super.computeScroll() var needsInvalidate = false // The scroller isn't finished, meaning a fling or // programmatic pan operation is active. if (scroller.computeScrollOffset()) { val surfaceSize: Point = computeScrollSurfaceSize() val currX: Int = scroller.currX val currY: Int = scroller.currY val (canScrollX: Boolean, canScrollY: Boolean) = currentViewport.run { (left > AXIS_X_MIN || right < AXIS_X_MAX) to (top > AXIS_Y_MIN || bottom < AXIS_Y_MAX) } /* * If you are zoomed in, currX or currY is * outside of bounds, and you aren't already * showing overscroll, then render the overscroll * glow edge effect. */ if (canScrollX && currX < 0 && edgeEffectLeft.isFinished && !edgeEffectLeftActive) { edgeEffectLeft.onAbsorb(scroller.currVelocity.toInt()) edgeEffectLeftActive = true needsInvalidate = true } else if (canScrollX && currX > surfaceSize.x - contentRect.width() && edgeEffectRight.isFinished && !edgeEffectRightActive) { edgeEffectRight.onAbsorb(scroller.currVelocity.toInt()) edgeEffectRightActive = true needsInvalidate = true } if (canScrollY && currY < 0 && edgeEffectTop.isFinished && !edgeEffectTopActive) { edgeEffectTop.onAbsorb(scroller.currVelocity.toInt()) edgeEffectTopActive = true needsInvalidate = true } else if (canScrollY && currY > surfaceSize.y - contentRect.height() && edgeEffectBottom.isFinished && !edgeEffectBottomActive) { edgeEffectBottom.onAbsorb(scroller.currVelocity.toInt()) edgeEffectBottomActive = true needsInvalidate = true } ... } }
جاوا
// Edge effect/overscroll tracking objects. private EdgeEffectCompat edgeEffectTop; private EdgeEffectCompat edgeEffectBottom; private EdgeEffectCompat edgeEffectLeft; private EdgeEffectCompat edgeEffectRight; private boolean edgeEffectTopActive; private boolean edgeEffectBottomActive; private boolean edgeEffectLeftActive; private boolean edgeEffectRightActive; @Override public void computeScroll() { super.computeScroll(); boolean needsInvalidate = false; // The scroller isn't finished, meaning a fling or // programmatic pan operation is active. if (scroller.computeScrollOffset()) { Point surfaceSize = computeScrollSurfaceSize(); int currX = scroller.getCurrX(); int currY = scroller.getCurrY(); boolean canScrollX = (currentViewport.left > AXIS_X_MIN || currentViewport.right < AXIS_X_MAX); boolean canScrollY = (currentViewport.top > AXIS_Y_MIN || currentViewport.bottom < AXIS_Y_MAX); /* * If you are zoomed in, currX or currY is * outside of bounds, and you aren't already * showing overscroll, then render the overscroll * glow edge effect. */ if (canScrollX && currX < 0 && edgeEffectLeft.isFinished() && !edgeEffectLeftActive) { edgeEffectLeft.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectLeftActive = true; needsInvalidate = true; } else if (canScrollX && currX > (surfaceSize.x - contentRect.width()) && edgeEffectRight.isFinished() && !edgeEffectRightActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectRightActive = true; needsInvalidate = true; } if (canScrollY && currY < 0 && edgeEffectTop.isFinished() && !edgeEffectTopActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectTopActive = true; needsInvalidate = true; } else if (canScrollY && currY > (surfaceSize.y - contentRect.height()) && edgeEffectBottom.isFinished() && !edgeEffectBottomActive) { edgeEffectRight.onAbsorb((int)mScroller.getCurrVelocity()); edgeEffectBottomActive = true; needsInvalidate = true; } ... }
در اینجا بخشی از کد است که بزرگنمایی واقعی را انجام می دهد:
کاتلین
lateinit var zoomer: Zoomer val zoomFocalPoint = PointF() ... // If a zoom is in progress—either programmatically // or through double touch—this performs the zoom. if (zoomer.computeZoom()) { val newWidth: Float = (1f - zoomer.currZoom) * scrollerStartViewport.width() val newHeight: Float = (1f - zoomer.currZoom) * scrollerStartViewport.height() val pointWithinViewportX: Float = (zoomFocalPoint.x - scrollerStartViewport.left) / scrollerStartViewport.width() val pointWithinViewportY: Float = (zoomFocalPoint.y - scrollerStartViewport.top) / scrollerStartViewport.height() currentViewport.set( zoomFocalPoint.x - newWidth * pointWithinViewportX, zoomFocalPoint.y - newHeight * pointWithinViewportY, zoomFocalPoint.x + newWidth * (1 - pointWithinViewportX), zoomFocalPoint.y + newHeight * (1 - pointWithinViewportY) ) constrainViewport() needsInvalidate = true } if (needsInvalidate) { ViewCompat.postInvalidateOnAnimation(this) }
جاوا
// Custom object that is functionally similar to Scroller. Zoomer zoomer; private PointF zoomFocalPoint = new PointF(); ... // If a zoom is in progress—either programmatically // or through double touch—this performs the zoom. if (zoomer.computeZoom()) { float newWidth = (1f - zoomer.getCurrZoom()) * scrollerStartViewport.width(); float newHeight = (1f - zoomer.getCurrZoom()) * scrollerStartViewport.height(); float pointWithinViewportX = (zoomFocalPoint.x - scrollerStartViewport.left) / scrollerStartViewport.width(); float pointWithinViewportY = (zoomFocalPoint.y - scrollerStartViewport.top) / scrollerStartViewport.height(); currentViewport.set( zoomFocalPoint.x - newWidth * pointWithinViewportX, zoomFocalPoint.y - newHeight * pointWithinViewportY, zoomFocalPoint.x + newWidth * (1 - pointWithinViewportX), zoomFocalPoint.y + newHeight * (1 - pointWithinViewportY)); constrainViewport(); needsInvalidate = true; } if (needsInvalidate) { ViewCompat.postInvalidateOnAnimation(this); }
این متد computeScrollSurfaceSize()
است که در قطعه قبلی فراخوانی شده است. اندازه سطح قابل پیمایش فعلی را بر حسب پیکسل محاسبه می کند. به عنوان مثال، اگر کل منطقه نمودار قابل مشاهده است، این اندازه فعلی mContentRect
است. اگر نمودار در هر دو جهت 200 درصد بزرگنمایی شود، اندازه برگشتی دو برابر بزرگتر بهصورت افقی و عمودی است.
کاتلین
private fun computeScrollSurfaceSize(): Point { return Point( (contentRect.width() * (AXIS_X_MAX - AXIS_X_MIN) / currentViewport.width()).toInt(), (contentRect.height() * (AXIS_Y_MAX - AXIS_Y_MIN) / currentViewport.height()).toInt() ) }
جاوا
private Point computeScrollSurfaceSize() { return new Point( (int) (contentRect.width() * (AXIS_X_MAX - AXIS_X_MIN) / currentViewport.width()), (int) (contentRect.height() * (AXIS_Y_MAX - AXIS_Y_MIN) / currentViewport.height())); }
برای مثال دیگری از استفاده از اسکرول، کد منبع کلاس ViewPager
را ببینید. در پاسخ به flings پیمایش می کند و از اسکرول برای اجرای انیمیشن "snap-to-page" استفاده می کند.
افکت کشش overscroll را اجرا کنید
با شروع اندروید 12، EdgeEffect
API های زیر را برای اجرای افکت Overscroll کششی اضافه می کند:
-
getDistance()
-
onPullDistance()
برای ارائه بهترین تجربه کاربری با کشش overscroll، موارد زیر را انجام دهید:
- هنگامی که وقتی کاربر محتویات را لمس می کند انیمیشن کششی اعمال می شود، لمس را به عنوان "گیر" ثبت کنید. کاربر انیمیشن را متوقف می کند و دوباره شروع به دستکاری کشش می کند.
- وقتی کاربر انگشت خود را در جهت مخالف کشش حرکت میدهد، کشش را رها کنید تا کاملاً از بین برود و سپس پیمایش را شروع کنید.
- هنگامی که کاربر در حین کشش پرت می کند،
EdgeEffect
برای افزایش افکت کشش پرتاب کنید.
انیمیشن را بگیرید
وقتی کاربر یک انیمیشن کششی فعال را میگیرد، EdgeEffect.getDistance()
0
برمیگرداند. این حالت نشان می دهد که کشش باید با حرکت لمسی دستکاری شود. در اکثر کانتینرها، catch در onInterceptTouchEvent()
شناسایی می شود، همانطور که در قطعه کد زیر نشان داده شده است:
کاتلین
override fun onInterceptTouchEvent(ev: MotionEvent): Boolean { ... when (action and MotionEvent.ACTION_MASK) { MotionEvent.ACTION_DOWN -> ... isBeingDragged = EdgeEffectCompat.getDistance(edgeEffectBottom) > 0f || EdgeEffectCompat.getDistance(edgeEffectTop) > 0f ... } return isBeingDragged }
جاوا
@Override public boolean onInterceptTouchEvent(MotionEvent ev) { ... switch (action & MotionEvent.ACTION_MASK) { case MotionEvent.ACTION_DOWN: ... isBeingDragged = EdgeEffectCompat.getDistance(edgeEffectBottom) > 0 || EdgeEffectCompat.getDistance(edgeEffectTop) > 0; ... } }
در مثال قبل، زمانی که mIsBeingDragged
true
باشد onInterceptTouchEvent()
true
برمی گرداند، بنابراین کافی است رویداد را قبل از اینکه کودک فرصتی برای مصرف داشته باشد مصرف کنید.
افکت overscroll را رها کنید
مهم است که جلوه کشش را قبل از پیمایش آزاد کنید تا از اعمال کشش به محتوای اسکرول جلوگیری کنید. نمونه کد زیر این بهترین روش را اعمال می کند:
کاتلین
override fun onTouchEvent(ev: MotionEvent): Boolean { val activePointerIndex = ev.actionIndex when (ev.getActionMasked()) { MotionEvent.ACTION_MOVE -> val x = ev.getX(activePointerIndex) val y = ev.getY(activePointerIndex) var deltaY = y - lastMotionY val pullDistance = deltaY / height val displacement = x / width if (deltaY < 0f && EdgeEffectCompat.getDistance(edgeEffectTop) > 0f) { deltaY -= height * EdgeEffectCompat.onPullDistance(edgeEffectTop, pullDistance, displacement); } if (deltaY > 0f && EdgeEffectCompat.getDistance(edgeEffectBottom) > 0f) { deltaY += height * EdgeEffectCompat.onPullDistance(edgeEffectBottom, -pullDistance, 1 - displacement); } ... }
جاوا
@Override public boolean onTouchEvent(MotionEvent ev) { final int actionMasked = ev.getActionMasked(); switch (actionMasked) { case MotionEvent.ACTION_MOVE: final float x = ev.getX(activePointerIndex); final float y = ev.getY(activePointerIndex); float deltaY = y - lastMotionY; float pullDistance = deltaY / getHeight(); float displacement = x / getWidth(); if (deltaY < 0 && EdgeEffectCompat.getDistance(edgeEffectTop) > 0) { deltaY -= getHeight() * EdgeEffectCompat.onPullDistance(edgeEffectTop, pullDistance, displacement); } if (deltaY > 0 && EdgeEffectCompat.getDistance(edgeEffectBottom) > 0) { deltaY += getHeight() * EdgeEffectCompat.onPullDistance(edgeEffectBottom, -pullDistance, 1 - displacement); } ...
هنگامی که کاربر در حال کشیدن است، قبل از اینکه رویداد لمسی را به محفظه پیمایشی تو در تو منتقل کنید یا اسکرول را بکشید، فاصله کشش EdgeEffect
را مصرف کنید. در نمونه کد قبلی، getDistance()
یک مقدار مثبت را هنگامی که یک افکت لبه نمایش داده می شود برمی گرداند و می تواند با حرکت آزاد شود. هنگامی که رویداد لمسی کشش را آزاد می کند، ابتدا توسط EdgeEffect
مصرف می شود تا قبل از نمایش جلوه های دیگر، مانند پیمایش تو در تو، به طور کامل آزاد شود. میتوانید از getDistance()
استفاده کنید تا بفهمید چقدر فاصله کششی برای آزاد کردن اثر فعلی لازم است.
برخلاف onPull()
، onPullDistance()
مقدار مصرف شده دلتای عبور شده را برمی گرداند. در اندروید 12، اگر onPull()
یا onPullDistance()
به مقادیر deltaDistance
منفی منتقل شوند زمانی که getDistance()
0
باشد، افکت کشش تغییر نمی کند. در اندروید 11 و نسخههای قبلی، onPull()
به مقادیر منفی برای کل مسافت اجازه میدهد جلوههای درخشندگی را نشان دهد.
از overscroll انصراف دهید
میتوانید از overscroll در فایل طرحبندی خود یا به صورت برنامهنویسی انصراف دهید.
برای انصراف در فایل طرح بندی خود، android:overScrollMode
را مانند مثال زیر تنظیم کنید:
<MyCustomView android:overScrollMode="never"> ... </MyCustomView>
برای انصراف از طریق برنامه، از کدهایی مانند زیر استفاده کنید:
کاتلین
customView.overScrollMode = View.OVER_SCROLL_NEVER
جاوا
customView.setOverScrollMode(View.OVER_SCROLL_NEVER);
منابع اضافی
به منابع مرتبط زیر مراجعه کنید: