نمای کلی رویدادهای ورودی

روش Compose را امتحان کنید
Jetpack Compose جعبه ابزار UI توصیه شده برای اندروید است. با نحوه استفاده از لمس و ورودی در Compose آشنا شوید.

در Android، بیش از یک راه برای رهگیری رویدادهای تعامل کاربر با برنامه شما وجود دارد. هنگامی که رویدادها را در رابط کاربری خود در نظر می گیرید، رویکرد این است که رویدادها را از شی View خاصی که کاربر با آن تعامل دارد، ضبط کنید. کلاس View ابزاری برای این کار فراهم می کند.

در کلاس‌های View مختلف که برای نوشتن طرح‌بندی خود استفاده می‌کنید، ممکن است چندین روش تماس عمومی را مشاهده کنید که برای رویدادهای رابط کاربری مفید به نظر می‌رسند. این متدها توسط فریمورک اندروید زمانی فراخوانی می شوند که عمل مربوطه روی آن شیء انجام شود. به عنوان مثال، هنگامی که یک View (مانند یک دکمه) لمس می شود، متد onTouchEvent() روی آن شی فراخوانی می شود. با این حال، برای رهگیری این، باید کلاس را گسترش دهید و متد را لغو کنید. با این حال، گسترش هر شی View به منظور رسیدگی به چنین رویدادی عملی نخواهد بود. به همین دلیل است که کلاس View همچنین شامل مجموعه‌ای از رابط‌های تودرتو با callback است که می‌توانید خیلی راحت‌تر آن‌ها را تعریف کنید. این رابط‌ها که شنوندگان رویداد نامیده می‌شوند، بلیط شما برای ثبت تعامل کاربر با رابط کاربری شما هستند.

در حالی که معمولاً از شنوندگان رویداد برای گوش دادن به تعامل با کاربر استفاده می کنید، ممکن است زمانی فرا برسد که بخواهید یک کلاس View را گسترش دهید تا یک جزء سفارشی بسازید. شاید بخواهید کلاس Button را گسترش دهید تا چیزی فانتزی تر بسازید. در این حالت، می‌توانید با استفاده از کنترل‌کننده‌های رویداد کلاس، رفتارهای رویداد پیش‌فرض را برای کلاس خود تعریف کنید.

شنوندگان رویداد

شنونده رویداد یک رابط در کلاس View است که شامل یک متد پاسخ به تماس است. این روش‌ها توسط چارچوب Android زمانی فراخوانی می‌شوند که نمایی که شنونده در آن ثبت شده است توسط تعامل کاربر با آیتم در UI فعال شود.

در رابط های شنونده رویداد، روش های پاسخ به تماس زیر گنجانده شده است:

onClick()
از View.OnClickListener . زمانی که کاربر مورد را لمس می‌کند (در حالت لمسی)، یا با کلیدهای ناوبری یا گوی گوی روی آیتم تمرکز می‌کند و کلید "ورود" مناسب را فشار می‌دهد یا توپ را فشار می‌دهد.
onLongClick()
از View.OnLongClickListener . زمانی که کاربر مورد را لمس کرده و نگه می‌دارد (هنگامی که در حالت لمسی است)، یا با کلیدهای ناوبری یا گوی گوی بر روی مورد تمرکز می‌کند و کلید مناسب «ورود» را فشار داده و نگه می‌دارد یا توپ را فشار داده و نگه می‌دارد، نامیده می‌شود. برای یک ثانیه).
onFocusChange()
از View.OnFocusChangeListener . هنگامی که کاربر با استفاده از کلیدهای ناوبری یا گوی مسیریابی، روی آیتم یا از آن دور می شود، این نام خوانده می شود.
onKey()
از View.OnKeyListener . هنگامی که کاربر بر روی مورد متمرکز شده و یک کلید سخت افزاری روی دستگاه را فشار داده یا رها می کند، این نام خوانده می شود.
onTouch()
از View.OnTouchListener . زمانی که کاربر اقدامی را انجام می‌دهد که به عنوان یک رویداد لمسی، از جمله فشار دادن، رها کردن، یا هر حرکت حرکتی روی صفحه نمایش (در محدوده مورد) را انجام می‌دهد، این نام خوانده می‌شود.
onCreateContextMenu()
از View.OnCreateContextMenuListener . هنگامی که یک منوی زمینه در حال ساخت است (به عنوان نتیجه یک "کلیک طولانی" پایدار) این نام خوانده می شود. بحث در مورد منوهای زمینه را در راهنمای برنامه‌نویس منوها ببینید.

این روش ها تنها ساکنان رابط مربوطه خود هستند. برای تعریف یکی از این روش ها و مدیریت رویدادهای خود، رابط تودرتو را در Activity خود پیاده سازی کنید یا آن را به عنوان یک کلاس ناشناس تعریف کنید. سپس، یک نمونه از پیاده سازی خود را به متد View.set...Listener() مربوطه ارسال کنید. (به عنوان مثال، setOnClickListener() را فراخوانی کنید و پیاده سازی OnClickListener را به آن ارسال کنید.)

مثال زیر نحوه ثبت یک شنونده روی کلیک را برای یک دکمه نشان می دهد.

کاتلین

protected void onCreate(savedValues: Bundle) {
    ...
    val button: Button = findViewById(R.id.corky)
    // Register the onClick listener with the implementation above
    button.setOnClickListener { view ->
        // do something when the button is clicked
    }
    ...
}

جاوا

// Create an anonymous implementation of OnClickListener
private OnClickListener corkyListener = new OnClickListener() {
    public void onClick(View v) {
      // do something when the button is clicked
    }
};

protected void onCreate(Bundle savedValues) {
    ...
    // Capture our button from layout
    Button button = (Button)findViewById(R.id.corky);
    // Register the onClick listener with the implementation above
    button.setOnClickListener(corkyListener);
    ...
}

همچنین ممکن است پیاده سازی OnClickListener به عنوان بخشی از فعالیت خود راحت تر باشد. این از بار کلاس اضافی و تخصیص شی جلوگیری می کند. به عنوان مثال:

کاتلین

class ExampleActivity : Activity(), OnClickListener {
  
    protected fun onCreate(savedValues: Bundle) {
        val button: Button = findViewById(R.id.corky)
        button.setOnClickListener(this)
    }

    // Implement the OnClickListener callback
    fun onClick(v: View) {
        // do something when the button is clicked
    }
}

جاوا

public class ExampleActivity extends Activity implements OnClickListener {
    protected void onCreate(Bundle savedValues) {
        ...
        Button button = (Button)findViewById(R.id.corky);
        button.setOnClickListener(this);
    }

    // Implement the OnClickListener callback
    public void onClick(View v) {
      // do something when the button is clicked
    }
    ...
}

توجه داشته باشید که onClick() در مثال بالا مقدار بازگشتی ندارد، اما برخی از متدهای شنونده رویداد دیگر باید یک Boolean برگردانند. دلیل بستگی به رویداد دارد. برای معدودی که این کار را انجام می دهند، در اینجا دلیل آن است:

  • onLongClick() - این یک Boolean برمی گرداند تا نشان دهد که آیا شما رویداد را مصرف کرده اید و نباید بیشتر انجام شود. یعنی به درستی برگردید تا نشان دهید که شما رویداد را مدیریت کرده اید و باید در اینجا متوقف شود. اگر شما آن را مدیریت نکرده اید، false را بازگردانید و/یا رویداد باید برای سایر شنوندگان کلیکی ادامه یابد.
  • onKey() - این یک Boolean برمی گرداند تا نشان دهد که آیا شما رویداد را مصرف کرده اید و نباید بیشتر انجام شود. یعنی به درستی برگردید تا نشان دهید که شما رویداد را مدیریت کرده اید و باید در اینجا متوقف شود. اگر آن را مدیریت نکرده‌اید، false را بازگردانید و/یا رویداد باید برای سایر شنوندگان کلیدی ادامه یابد.
  • onTouch() - این یک Boolean برمی گرداند تا نشان دهد که آیا شنونده شما این رویداد را مصرف می کند یا خیر. نکته مهم این است که این رویداد می تواند چندین کنش داشته باشد که به دنبال یکدیگر می آیند. بنابراین، اگر هنگام دریافت رویداد down action، false را برگردانید، نشان می‌دهید که رویداد را مصرف نکرده‌اید و همچنین علاقه‌ای به اقدامات بعدی این رویداد ندارید. بنابراین، از شما برای هیچ گونه اقدام دیگری در رویداد، مانند اشاره انگشت، یا رویداد اقدام نهایی فراخوانی نمی شود.

به یاد داشته باشید که رویدادهای کلیدی سخت‌افزاری همیشه به نمای فعلی در فوکوس ارائه می‌شوند. آنها از بالای سلسله مراتب View فرستاده می شوند و سپس به پایین تا رسیدن به مقصد مناسب ارسال می شوند. اگر نمای شما (یا فرزند نمای شما) در حال حاضر فوکوس دارد، می‌توانید سفر رویداد را از طریق متد dispatchKeyEvent() مشاهده کنید. به عنوان جایگزینی برای ثبت رویدادهای کلیدی از طریق View خود، می‌توانید تمام رویدادهای داخل Activity خود را با onKeyDown() و onKeyUp() دریافت کنید.

همچنین، هنگامی که به ورودی متن برای برنامه خود فکر می کنید، به یاد داشته باشید که بسیاری از دستگاه ها فقط روش های ورودی نرم افزاری دارند. لازم نیست چنین روش هایی مبتنی بر کلید باشند. برخی ممکن است از ورودی صوتی، دست خط و غیره استفاده کنند. حتی اگر یک متد ورودی یک رابط صفحه‌کلید مانند ارائه دهد، معمولاً خانواده رویدادهای onKeyDown() راه‌اندازی نمی‌کند . هرگز نباید رابط کاربری ایجاد کنید که نیاز به فشار دادن کلیدهای خاصی برای کنترل داشته باشد، مگر اینکه بخواهید برنامه خود را به دستگاه هایی با صفحه کلید سخت افزاری محدود کنید. به ویژه، هنگامی که کاربر کلید بازگشت را فشار می دهد، برای تأیید اعتبار ورودی به این روش ها اعتماد نکنید. در عوض، از اقداماتی مانند IME_ACTION_DONE استفاده کنید تا به روش ورودی سیگنال دهید که برنامه شما چگونه انتظار دارد واکنش نشان دهد، بنابراین ممکن است رابط کاربری خود را به روشی معنادار تغییر دهد. از فرضیات در مورد نحوه عملکرد یک روش ورودی نرم افزار اجتناب کنید و فقط به آن اعتماد کنید تا متن فرمت شده قبلی را به برنامه شما ارائه دهد.

توجه: اندروید ابتدا کنترل‌کننده‌های رویداد و سپس کنترل‌کننده‌های پیش‌فرض مناسب را از تعریف کلاس فراخوانی می‌کند. به این ترتیب، بازگرداندن true از این شنوندگان رویداد، انتشار رویداد را به سایر شنوندگان رویداد متوقف می‌کند و همچنین پاسخ تماس به کنترل‌کننده رویداد پیش‌فرض در View را مسدود می‌کند. بنابراین مطمئن باشید که می خواهید رویداد را با بازگشت true خاتمه دهید.

گردانندگان رویداد

اگر یک مؤلفه سفارشی از View ایجاد می‌کنید، می‌توانید چندین روش پاسخ به تماس را که به‌عنوان کنترل‌کننده رویداد پیش‌فرض استفاده می‌شوند، تعریف کنید. در سند مربوط به مؤلفه‌های نمای سفارشی ، برخی از تماس‌های متداول مورد استفاده برای مدیریت رویداد را خواهید آموخت، از جمله:

  • onKeyDown(int, KeyEvent) - زمانی که یک رویداد کلیدی جدید رخ می دهد، فراخوانی می شود.
  • onKeyUp(int, KeyEvent) - زمانی که یک رویداد کلیدی رخ می دهد، فراخوانی می شود.
  • onTrackballEvent(MotionEvent) - هنگامی که یک رویداد حرکت گوی رخ می دهد، تماس می گیرد.
  • onTouchEvent(MotionEvent) - زمانی که یک رویداد حرکتی صفحه نمایش لمسی رخ می دهد، فراخوانی می شود.
  • onFocusChanged(boolean, int, Rect) - زمانی فراخوانی می شود که نمای فوکوس را به دست آورد یا از دست بدهد.

روش‌های دیگری نیز وجود دارد که باید از آنها آگاه باشید، که بخشی از کلاس View نیستند، اما می‌توانند مستقیماً بر نحوه مدیریت رویدادها تأثیر بگذارند. بنابراین، هنگام مدیریت رویدادهای پیچیده تر در یک طرح، این روش های دیگر را در نظر بگیرید:

حالت لمسی

هنگامی که کاربر در حال پیمایش یک رابط کاربری با کلیدهای جهت دار یا گوی ردیابی است، لازم است به موارد عملی (مانند دکمه ها) توجه شود تا کاربر بتواند ببیند چه چیزی ورودی را می پذیرد. با این حال، اگر دستگاه دارای قابلیت لمسی باشد و کاربر با لمس آن با رابط ارتباط برقرار کند، دیگر نیازی به برجسته کردن موارد یا تمرکز روی یک نمای خاص نیست. بنابراین، حالتی برای تعامل به نام "حالت لمسی" وجود دارد.

برای یک دستگاه با قابلیت لمس، هنگامی که کاربر صفحه را لمس کند، دستگاه وارد حالت لمسی می شود. از این مرحله به بعد، فقط نماهایی که برای آنها isFocusableInTouchMode() true است، مانند ویجت های ویرایش متن، قابل فوکوس خواهند بود. سایر نماهایی که قابل لمس هستند، مانند دکمه‌ها، با لمس فوکوس نمی‌شوند. آنها به سادگی شنوندگان کلیکی خود را با فشار دادن فعال می کنند.

هر زمان که کاربر کلید جهت را بزند یا با یک گوی پیمایش کند، دستگاه از حالت لمسی خارج می‌شود و نمایی برای تمرکز پیدا می‌کند. اکنون، کاربر ممکن است بدون لمس صفحه، تعامل با رابط کاربری را از سر بگیرد.

حالت حالت لمسی در کل سیستم (همه پنجره ها و فعالیت ها) حفظ می شود. برای پرس و جو از وضعیت فعلی، می توانید با isInTouchMode() تماس بگیرید تا ببینید آیا دستگاه در حال حاضر در حالت لمسی است یا خیر.

رسیدگی به تمرکز

این چارچوب حرکت معمول فوکوس را در پاسخ به ورودی کاربر مدیریت می کند. این شامل تغییر تمرکز با حذف یا پنهان شدن نماها یا در دسترس قرار گرفتن نماهای جدید است. View ها تمایل آنها را برای تمرکز از طریق متد isFocusable() نشان می دهد. برای تغییر اینکه آیا یک View می تواند فوکوس کند یا خیر، setFocusable() را فراخوانی کنید. هنگامی که در حالت لمسی هستید، می‌توانید بپرسید که آیا View اجازه فوکوس را با isFocusableInTouchMode() می‌دهد. می توانید این را با setFocusableInTouchMode() تغییر دهید.

در دستگاه‌های دارای Android 9 (سطح API 28) یا بالاتر، فعالیت‌ها تمرکز اولیه ندارند. در عوض، در صورت تمایل، باید صراحتاً فوکوس اولیه را درخواست کنید.

حرکت فوکوس بر اساس الگوریتمی است که نزدیکترین همسایه را در یک جهت مشخص پیدا می کند. در موارد نادر، الگوریتم پیش فرض ممکن است با رفتار مورد نظر توسعه دهنده مطابقت نداشته باشد. در این شرایط، می‌توانید با ویژگی‌های XML زیر در فایل طرح‌بندی، بازنویسی‌های صریح ارائه کنید: nextFocusDown ، nextFocusLeft ، nextFocusRight و nextFocusUp . یکی از این ویژگی ها را به View که فوکوس از آن خارج می شود اضافه کنید. مقدار مشخصه را به عنوان شناسه نمایه ای که باید به آن فوکوس داده شود، تعیین کنید. به عنوان مثال:

<LinearLayout
    android:orientation="vertical"
    ... >
  <Button android:id="@+id/top"
          android:nextFocusUp="@+id/bottom"
          ... />
  <Button android:id="@+id/bottom"
          android:nextFocusDown="@+id/top"
          ... />
</LinearLayout>

به طور معمول، در این طرح عمودی، پیمایش از دکمه اول به بالا به جایی نمی رسد، و همچنین حرکت از دکمه دوم به پایین نمی تواند انجام شود. اکنون که دکمه بالا، دکمه پایین را به عنوان nextFocusUp (و بالعکس) تعریف کرده است، فوکوس ناوبری از بالا به پایین و از پایین به بالا چرخه خواهد شد.

اگر می‌خواهید یک View را در رابط کاربری خود به‌عنوان قابل فوکوس‌پذیر اعلام کنید (در صورتی که معمولاً اینطور نیست)، ویژگی android:focusable XML را در اعلان طرح‌بندی خود به View اضافه کنید. مقدار true را تنظیم کنید. همچنین می‌توانید در حالت لمسی با android:focusableInTouchMode ، یک View را به‌عنوان قابل فوکوس اعلام کنید.

برای درخواست یک View خاص برای فوکوس کردن، requestFocus() را فراخوانی کنید.

برای گوش دادن به رویدادهای فوکوس (در صورت دریافت یا از دست دادن فوکوس یک View مطلع شوید)، از onFocusChange() استفاده کنید، همانطور که در بخش شنوندگان رویداد بحث شد.