این درس نحوه ردیابی حرکت را در رویدادهای لمسی توضیح می دهد.
هر زمان که موقعیت، فشار یا اندازه تماس لمسی فعلی تغییر کند، یک onTouchEvent()
جدید با یک رویداد ACTION_MOVE
فعال میشود. همانطور که در Detect common gestures توضیح داده شد، همه این رویدادها در پارامتر MotionEvent
onTouchEvent()
ثبت میشوند.
از آنجایی که لمس مبتنی بر انگشت همیشه دقیق ترین شکل تعامل نیست، تشخیص رویدادهای لمسی اغلب بیشتر بر اساس حرکت است تا تماس ساده. برای کمک به برنامهها برای تمایز بین حرکات مبتنی بر حرکت (مانند تند کشیدن) و ژستهای غیرحرکتی (مانند یک ضربه)، اندروید مفهوم شیب لمسی را در بر میگیرد. شیب لمسی به فاصلهای بر حسب پیکسلی اشاره دارد که لمس کاربر میتواند قبل از اینکه ژست به عنوان یک حرکت حرکتی تفسیر شود، سرگردان شود. برای اطلاعات بیشتر در مورد این موضوع، به مدیریت رویدادهای لمسی در یک ViewGroup مراجعه کنید.
بسته به نیاز برنامه شما، راه های مختلفی برای ردیابی حرکت در ژست وجود دارد. موارد زیر نمونه هایی هستند:
- موقعیت شروع و پایان یک اشاره گر، مانند حرکت یک شی روی صفحه از نقطه A به نقطه B.
- جهتی که اشاره گر در آن حرکت می کند، همانطور که با مختصات X و Y تعیین می شود.
- تاریخچه. شما می توانید اندازه تاریخچه یک ژست را با فراخوانی متد
MotionEvent
getHistorySize()
بیابید. سپس میتوانید موقعیت، اندازه، زمان و فشار هر یک از رویدادهای تاریخی را با استفاده از روشهایgetHistorical <Value>
رویداد حرکتی بدست آورید. تاریخچه هنگام رندر کردن دنباله انگشت کاربر، مانند طراحی لمسی، مفید است. برای جزئیات به مرجعMotionEvent
مراجعه کنید. - سرعت نشانگر هنگام حرکت در صفحه لمسی.
به منابع مرتبط زیر مراجعه کنید:
سرعت ردیابی
شما می توانید یک ژست مبتنی بر حرکت داشته باشید که بر اساس مسافت یا جهتی است که نشانگر طی می کند. با این حال، سرعت اغلب یک عامل تعیین کننده در ردیابی ویژگی های یک ژست یا تصمیم گیری در مورد اینکه آیا ژست رخ داده است یا خیر است. برای آسانتر کردن محاسبه سرعت، اندروید کلاس VelocityTracker
را ارائه میکند. VelocityTracker
به شما کمک می کند تا سرعت رویدادهای لمسی را ردیابی کنید. این برای حرکاتی که در آنها سرعت بخشی از معیارهای حرکت است، مانند پرت کردن، مفید است.
در اینجا مثالی آورده شده است که هدف روش ها در VelocityTracker
API را نشان می دهد:
private const val DEBUG_TAG = "Velocity" class MainActivity : Activity() { private var mVelocityTracker: VelocityTracker? = null override fun onTouchEvent(event: MotionEvent): Boolean { when (event.actionMasked) { MotionEvent.ACTION_DOWN -> { // Reset the velocity tracker back to its initial state. mVelocityTracker?.clear() // If necessary, retrieve a new VelocityTracker object to watch // the velocity of a motion. mVelocityTracker = mVelocityTracker ?: VelocityTracker.obtain() // Add a user's movement to the tracker. mVelocityTracker?.addMovement(event) } MotionEvent.ACTION_MOVE -> { mVelocityTracker?.apply { val pointerId: Int = event.getPointerId(event.actionIndex) addMovement(event) // When you want to determine the velocity, call // computeCurrentVelocity(). Then, call getXVelocity() and // getYVelocity() to retrieve the velocity for each pointer // ID. computeCurrentVelocity(1000) // Log velocity of pixels per second. It's best practice to // use VelocityTrackerCompat where possible. Log.d("", "X velocity: ${getXVelocity(pointerId)}") Log.d("", "Y velocity: ${getYVelocity(pointerId)}") } } MotionEvent.ACTION_UP, MotionEvent.ACTION_CANCEL -> { // Return a VelocityTracker object back to be re-used by others. mVelocityTracker?.recycle() mVelocityTracker = null } } return true } }
public class MainActivity extends Activity { private static final String DEBUG_TAG = "Velocity"; ... private VelocityTracker mVelocityTracker = null; @Override public boolean onTouchEvent(MotionEvent event) { int index = event.getActionIndex(); int action = event.getActionMasked(); int pointerId = event.getPointerId(index); switch(action) { case MotionEvent.ACTION_DOWN: if(mVelocityTracker == null) { // Retrieve a new VelocityTracker object to watch the // velocity of a motion. mVelocityTracker = VelocityTracker.obtain(); } else { // Reset the velocity tracker back to its initial state. mVelocityTracker.clear(); } // Add a user's movement to the tracker. mVelocityTracker.addMovement(event); break; case MotionEvent.ACTION_MOVE: mVelocityTracker.addMovement(event); // When you want to determine the velocity, call // computeCurrentVelocity(). Then call getXVelocity() and // getYVelocity() to retrieve the velocity for each pointer ID. mVelocityTracker.computeCurrentVelocity(1000); // Log velocity of pixels per second. It's best practice to use // VelocityTrackerCompat where possible. Log.d("", "X velocity: " + mVelocityTracker.getXVelocity(pointerId)); Log.d("", "Y velocity: " + mVelocityTracker.getYVelocity(pointerId)); break; case MotionEvent.ACTION_UP: case MotionEvent.ACTION_CANCEL: // Return a VelocityTracker object back to be re-used by others. mVelocityTracker.recycle(); break; } return true; } }
از گرفتن اشاره گر استفاده کنید
برخی از برنامهها، مانند بازیها، دسکتاپ از راه دور و کلاینتهای مجازیسازی، از کنترل نشانگر ماوس سود میبرند. ضبط اشاره گر ویژگی موجود در اندروید 8.0 (سطح API 26) و بالاتر است که این کنترل را با ارائه همه رویدادهای ماوس به نمای متمرکز در برنامه شما فراهم می کند.
درخواست گرفتن اشاره گر
یک نمای در برنامه شما فقط زمانی میتواند درخواست ثبت اشاره گر را داشته باشد که سلسله مراتب نمای حاوی آن فوکوس داشته باشد. به همین دلیل، زمانی که یک کنش کاربر خاص روی نما وجود دارد، درخواست ضبط اشاره گر را بدهید، مثلاً در طول یک رویداد onClick()
یا در کنترل کننده رویداد onWindowFocusChanged()
فعالیت شما.
برای درخواست گرفتن اشاره گر، متد requestPointerCapture()
را در view فراخوانی کنید. مثال کد زیر نحوه درخواست گرفتن اشاره گر را هنگامی که کاربر روی نما کلیک می کند نشان می دهد:
fun onClick(view: View) { view.requestPointerCapture() }
@Override public void onClick(View view) { view.requestPointerCapture(); }
پس از موفقیت آمیز بودن درخواست برای گرفتن اشاره گر، Android onPointerCaptureChange(true)
فراخوانی می کند. این سیستم رویدادهای ماوس را به نمای متمرکز در برنامه شما تحویل می دهد تا زمانی که در همان سلسله مراتب نمای نمایی باشد که درخواست تصویربرداری کرده است. سایر برنامهها تا زمانی که عکس منتشر نشود، رویدادهای ماوس را دریافت نمیکنند، از جمله رویدادهای ACTION_OUTSIDE
. Android رویدادهای اشاره گر را از منابعی غیر از ماوس به طور معمول ارائه می دهد، اما نشانگر ماوس دیگر قابل مشاهده نیست.
رویدادهای نشانگر گرفته شده را مدیریت کنید
هنگامی که یک view با موفقیت ثبت اشاره گر را به دست آورد، Android رویدادهای ماوس را ارائه می دهد. نمای متمرکز شما میتواند با انجام یکی از کارهای زیر، رویدادها را مدیریت کند:
- اگر از نمای سفارشی استفاده میکنید،
onCapturedPointerEvent(MotionEvent)
را لغو کنید. - در غیر این صورت، یک
OnCapturedPointerListener
ثبت کنید.
مثال کد زیر نحوه پیاده سازی onCapturedPointerEvent(MotionEvent)
را نشان می دهد:
override fun onCapturedPointerEvent(motionEvent: MotionEvent): Boolean { // Get the coordinates required by your app. val verticalOffset: Float = motionEvent.y // Use the coordinates to update your view and return true if the event is // successfully processed. return true }
@Override public boolean onCapturedPointerEvent(MotionEvent motionEvent) { // Get the coordinates required by your app. float verticalOffset = motionEvent.getY(); // Use the coordinates to update your view and return true if the event is // successfully processed. return true; }
مثال کد زیر نحوه ثبت OnCapturedPointerListener
را نشان می دهد:
myView.setOnCapturedPointerListener { view, motionEvent -> // Get the coordinates required by your app. val horizontalOffset: Float = motionEvent.x // Use the coordinates to update your view and return true if the event is // successfully processed. true }
myView.setOnCapturedPointerListener(new View.OnCapturedPointerListener() { @Override public boolean onCapturedPointer (View view, MotionEvent motionEvent) { // Get the coordinates required by your app. float horizontalOffset = motionEvent.getX(); // Use the coordinates to update your view and return true if the event is // successfully processed. return true; } });
چه از یک نمای سفارشی استفاده کنید یا یک شنونده را ثبت کنید، نمای شما یک MotionEvent
با مختصات اشاره گر دریافت می کند که حرکات نسبی مانند دلتاهای X یا Y را مشخص می کند، مشابه مختصات ارائه شده توسط یک دستگاه ترک بال. می توانید مختصات را با استفاده از getX()
و getY()
بازیابی کنید.
ضبط اشاره گر را رها کنید
نمای برنامه شما میتواند با فراخوانی releasePointerCapture()
ضبط اشاره گر را آزاد کند، همانطور که در مثال کد زیر نشان داده شده است:
override fun onClick(view: View) { view.releasePointerCapture() }
@Override public void onClick(View view) { view.releasePointerCapture(); }
این سیستم میتواند بدون فراخوانی صریح releasePointerCapture()
عکسبرداری را از view دور کند، معمولاً به این دلیل که سلسلهمراتب view حاوی نمایی که درخواست ضبط را میدهد، فوکوس را از دست میدهد.