ویرایشگر روش ورودی (IME) یک کنترل کاربر است که به کاربران اجازه می دهد متن را وارد کنند. Android یک چارچوب روش ورودی توسعهیافته ارائه میکند که به برنامهها اجازه میدهد روشهای ورودی جایگزینی مانند صفحهکلید روی صفحه یا ورودی گفتار را در اختیار کاربران قرار دهند. پس از نصب IME ها، کاربر می تواند یکی از تنظیمات سیستم را انتخاب کرده و در کل سیستم استفاده کند. فقط یک IME را می توان در یک زمان فعال کرد.
برای افزودن یک IME به سیستم Android، یک برنامه Android حاوی کلاسی ایجاد کنید که InputMethodService
گسترش دهد. علاوه بر این، شما معمولاً یک فعالیت "تنظیمات" ایجاد می کنید که گزینه ها را به سرویس IME منتقل می کند. همچنین می توانید رابط کاربری تنظیماتی را که به عنوان بخشی از تنظیمات سیستم نمایش داده می شود، تعریف کنید.
این صفحه موضوعات زیر را پوشش می دهد:
- چرخه حیات IME
- اعلان اجزای IME در مانیفست برنامه
- IME API
- طراحی رابط کاربری IME
- ارسال متن از یک IME به یک برنامه
- کار با زیرگروه های IME
- سایر ملاحظات IME
اگر با IME کار نکردهاید، ابتدا مقاله مقدماتی روشهای ورودی روی صفحه را بخوانید.
چرخه حیات IME
نمودار زیر چرخه حیات یک IME را شرح می دهد:
بخشهای زیر نحوه پیادهسازی UI و کد مرتبط با یک IME را که از این چرخه حیات پیروی میکند، توضیح میدهد.
اجزای IME را در مانیفست اعلام کنید
در سیستم اندروید، IME یک برنامه اندرویدی است که شامل یک سرویس ویژه IME است. فایل مانیفست برنامه باید سرویس را اعلام کند، مجوزهای لازم را درخواست کند، یک فیلتر قصد مطابق با action.view.InputMethod
ارائه کند، و متادیتا را ارائه دهد که ویژگیهای IME را تعریف میکند. علاوه بر این، برای ارائه یک رابط تنظیمات که به کاربر اجازه می دهد رفتار IME را تغییر دهد، می توانید یک فعالیت "تنظیمات" را تعریف کنید که می تواند از تنظیمات سیستم راه اندازی شود.
قطعه زیر یک سرویس IME را اعلام می کند. این مجوز BIND_INPUT_METHOD
را درخواست می کند تا به سرویس اجازه دهد IME را به سیستم متصل کند، یک فیلتر هدف تنظیم می کند که با عملکرد android.view.InputMethod
مطابقت داشته باشد، و ابرداده را برای IME تعریف می کند:
<!-- Declares the input method service. --> <service android:name="FastInputIME" android:label="@string/fast_input_label" android:permission="android.permission.BIND_INPUT_METHOD"> <intent-filter> <action android:name="android.view.InputMethod" /> </intent-filter> <meta-data android:name="android.view.im" android:resource="@xml/method" /> </service>
قطعه بعدی فعالیت تنظیمات را برای IME اعلام می کند. دارای یک فیلتر هدف برای ACTION_MAIN
است که نشان میدهد این فعالیت نقطه ورودی اصلی برنامه IME است:
<!-- Optional: an activity for controlling the IME settings. --> <activity android:name="FastInputIMESettings" android:label="@string/fast_input_settings"> <intent-filter> <action android:name="android.intent.action.MAIN"/> </intent-filter> </activity>
همچنین میتوانید مستقیماً از رابط کاربری آن به تنظیمات IME دسترسی داشته باشید.
API روش ورودی
کلاسهای مخصوص IME در بستههای android.inputmethodservice
و android.view.inputmethod
یافت میشوند. کلاس KeyEvent
برای مدیریت کاراکترهای صفحه کلید مهم است.
بخش مرکزی یک IME یک جزء سرویس است - کلاسی که InputMethodService
را گسترش می دهد. این کلاس علاوه بر پیادهسازی چرخه عمر سرویس عادی، برای ارائه رابط کاربری IME شما، مدیریت ورودی کاربر و تحویل متن به فیلدی که فوکوس دارد، تماسهایی نیز دارد. بهطور پیشفرض، کلاس InputMethodService
بیشتر پیادهسازی را برای مدیریت وضعیت و دید IME و برقراری ارتباط با فیلد ورودی فعلی ارائه میکند.
کلاس های زیر نیز مهم هستند:
-
BaseInputConnection
- کانال ارتباطی را از یک
InputMethod
به برنامهای که ورودی آن را دریافت میکند، تعریف میکند. شما از آن برای خواندن متن در اطراف مکان نما، ارسال متن به جعبه متن و ارسال رویدادهای کلید خام به برنامه استفاده می کنید. برنامه ها باید این کلاس را به جای پیاده سازی رابط پایهInputConnection
گسترش دهند. -
KeyboardView
- یک برنامه افزودنی از
View
که صفحه کلید را رندر می کند و به رویدادهای ورودی کاربر پاسخ می دهد. چیدمان صفحه کلید توسط نمونه ای ازKeyboard
مشخص می شود که می توانید آن را در یک فایل XML تعریف کنید.
UI روش ورودی را طراحی کنید
دو عنصر بصری اصلی برای یک IME وجود دارد: نمای ورودی و نمای کاندیدا . شما فقط باید عناصری را پیاده سازی کنید که با روش ورودی که طراحی می کنید مرتبط هستند.
نمای ورودی
نمای ورودی رابط کاربری است که در آن کاربر متن را به صورت کلیک کلید، دست خط یا اشاره وارد می کند. هنگامی که IME برای اولین بار نمایش داده می شود، سیستم onCreateInputView()
را فراخوانی می کند. در اجرای این روش، طرحی را که می خواهید در پنجره IME نمایش دهید ایجاد کنید و طرح را به سیستم برگردانید. قطعه زیر نمونه ای از پیاده سازی متد onCreateInputView()
را نشان می دهد:
کاتلین
override fun onCreateInputView(): View { return layoutInflater.inflate(R.layout.input, null).apply { if (this is MyKeyboardView) { setOnKeyboardActionListener(this@MyInputMethod) keyboard = latinKeyboard } } }
جاوا
@Override public View onCreateInputView() { MyKeyboardView inputView = (MyKeyboardView) getLayoutInflater().inflate(R.layout.input, null); inputView.setOnKeyboardActionListener(this); inputView.setKeyboard(latinKeyboard); return inputView; }
در این مثال، MyKeyboardView
نمونهای از پیادهسازی سفارشی KeyboardView
است که یک Keyboard
را ارائه میکند.
مشاهده نامزدها
نمای نامزدها رابط کاربری است که در آن IME اصلاحات یا پیشنهادات بالقوه کلمه را برای کاربر نمایش می دهد. در چرخه حیات IME، زمانی که سیستم برای نمایش نمای نامزدها آماده باشد onCreateCandidatesView()
را فرا می خواند. در اجرای این روش، طرحی را برگردانید که پیشنهادات کلمه را نشان می دهد، یا اگر نمی خواهید چیزی نشان داده شود، null را برگردانید. پاسخ تهی رفتار پیشفرض است، بنابراین در صورت عدم ارائه پیشنهاد، نیازی به اجرای آن ندارید.
ملاحظات طراحی رابط کاربری
این بخش برخی از ملاحظات طراحی UI برای IME ها را شرح می دهد.
اندازه های صفحه نمایش متعدد را مدیریت کنید
UI برای IME شما باید بتواند برای اندازه های مختلف صفحه نمایش مقیاس بندی کند و هم جهت افقی و هم عمودی را مدیریت کند. در حالت IME غیر تمام صفحه، فضای کافی برای برنامه برای نمایش فیلد متن و هر زمینه مرتبط با آن بگذارید، به طوری که بیش از نیمی از صفحه توسط IME اشغال نشود. در حالت تمام صفحه IME، این یک مشکل نیست.
انواع مختلف ورودی را مدیریت کنید
فیلدهای متنی Android به شما امکان می دهد نوع ورودی خاصی مانند متن آزاد، اعداد، URL ها، آدرس های ایمیل و رشته های جستجو را تنظیم کنید. هنگامی که یک IME جدید را پیاده سازی می کنید، نوع ورودی هر فیلد را شناسایی کرده و رابط مناسب را برای آن فراهم کنید. با این حال، لازم نیست IME خود را تنظیم کنید تا بررسی کنید آیا کاربر متن معتبری را برای نوع ورودی وارد می کند یا خیر. این مسئولیت بر عهده برنامه ای است که دارای فیلد متنی است.
به عنوان مثال، در اینجا رابطی است که IME لاتین برای ورودی متن پلت فرم اندروید ارائه می کند:
و اینجا رابطی است که IME لاتین برای ورودی عددی پلتفرم اندروید ارائه می کند:
وقتی یک فیلد ورودی فوکوس دریافت میکند و IME شما شروع میشود، سیستم onStartInputView()
را فراخوانی میکند و یک شی EditorInfo
را ارسال میکند که حاوی جزئیات مربوط به نوع ورودی و سایر ویژگیهای فیلد متنی است. در این شی، فیلد inputType
شامل نوع ورودی فیلد متنی است.
فیلد inputType
یک int
است که شامل الگوهای بیت برای تنظیمات مختلف نوع ورودی است. برای آزمایش آن برای نوع ورودی فیلد متن، آن را با ثابت TYPE_MASK_CLASS
ماسک کنید، مانند این:
کاتلین
inputType and InputType.TYPE_MASK_CLASS
جاوا
inputType & InputType.TYPE_MASK_CLASS
الگوی بیت نوع ورودی می تواند یکی از چندین مقدار داشته باشد، از جمله:
-
TYPE_CLASS_NUMBER
- فیلد متنی برای وارد کردن اعداد. همانطور که در شکل 3 نشان داده شده است، IME لاتین یک صفحه شماره برای فیلدهایی از این نوع نمایش می دهد.
-
TYPE_CLASS_DATETIME
- یک فیلد متنی برای وارد کردن تاریخ و زمان.
-
TYPE_CLASS_PHONE
- فیلد متنی برای وارد کردن شماره تلفن.
-
TYPE_CLASS_TEXT
- یک فیلد متنی برای وارد کردن هر کاراکتر پشتیبانی شده.
این ثابت ها با جزئیات بیشتری در مستندات مرجع برای InputType
توضیح داده شده اند.
فیلد inputType
میتواند حاوی بیتهای دیگری باشد که نشاندهنده گونهای از نوع فیلد متنی است، مانند:
-
TYPE_TEXT_VARIATION_PASSWORD
- نوعی از
TYPE_CLASS_TEXT
برای وارد کردن رمزهای عبور. روش ورودی، dingbats را به جای متن واقعی نمایش می دهد. -
TYPE_TEXT_VARIATION_URI
- نوعی از
TYPE_CLASS_TEXT
برای وارد کردن URL های وب و سایر شناسه های منبع یکنواخت (URI). -
TYPE_TEXT_FLAG_AUTO_COMPLETE
- گونه ای از
TYPE_CLASS_TEXT
برای وارد کردن متنی که برنامه به طور خودکار از فرهنگ لغت، جستجو یا امکانات دیگر تکمیل می کند.
هنگام آزمایش این انواع، inputType
با ثابت مناسب پوشش دهید. ثابتهای ماسک موجود در مستندات مرجع برای InputType
فهرست شدهاند.
ارسال متن به برنامه
همانطور که کاربر متن را با IME شما وارد می کند، می توانید با ارسال رویدادهای کلیدی جداگانه یا با ویرایش متن اطراف مکان نما در قسمت متن برنامه، متن را به برنامه ارسال کنید. در هر صورت، از یک نمونه از InputConnection
برای تحویل متن استفاده کنید. برای دریافت این نمونه، InputMethodService.getCurrentInputConnection()
را فراخوانی کنید.
متن اطراف مکان نما را ویرایش کنید
هنگامی که ویرایش متن موجود را انجام می دهید، برخی از روش های مفید در BaseInputConnection
به شرح زیر است:
-
getTextBeforeCursor()
- یک
CharSequence
حاوی تعداد نویسههای درخواستی قبل از موقعیت مکاننمای فعلی را برمیگرداند. -
getTextAfterCursor()
- یک
CharSequence
حاوی تعداد کاراکترهای درخواستی را به دنبال موقعیت مکان نما برمیگرداند. -
deleteSurroundingText()
- تعداد مشخص شده نویسه را قبل و بعد از موقعیت مکان نما فعلی حذف می کند.
-
commitText()
- یک
CharSequence
به فیلد متنی متعهد می کند و یک مکان مکان نما جدید تعیین می کند.
به عنوان مثال، قطعه زیر نشان می دهد که چگونه چهار کاراکتر سمت چپ مکان نما را با متن "Hello!" جایگزین کنید:
کاتلین
currentInputConnection.also { ic: InputConnection -> ic.deleteSurroundingText(4, 0) ic.commitText("Hello", 1) ic.commitText("!", 1) }
جاوا
InputConnection ic = getCurrentInputConnection(); ic.deleteSurroundingText(4, 0); ic.commitText("Hello", 1); ic.commitText("!", 1);
از نوشتن متن قبل از ارتکاب پشتیبانی کنید
اگر IME شما متن را پیشبینی میکند یا برای نوشتن یک علامت یا کلمه به چندین مرحله نیاز دارد، میتوانید پیشرفت را در قسمت متن نشان دهید تا زمانی که کاربر کلمه را متعهد کند و سپس میتوانید ترکیب جزئی را با متن تکمیلشده جایگزین کنید. وقتی متن را به setComposingText()
میفرستید، میتوانید با افزودن یک span به متن، رفتار خاصی را به آن بدهید.
قطعه زیر نحوه نمایش پیشرفت در یک فیلد متنی را نشان می دهد:
کاتلین
currentInputConnection.also { ic: InputConnection -> ic.setComposingText("Composi", 1) ic.setComposingText("Composin", 1) ic.commitText("Composing ", 1) }
جاوا
InputConnection ic = getCurrentInputConnection(); ic.setComposingText("Composi", 1); ic.setComposingText("Composin", 1); ic.commitText("Composing ", 1);
رهگیری رویدادهای کلیدی سخت افزاری
اگرچه پنجره روش ورودی فوکوس واضحی ندارد، ابتدا رویدادهای کلیدی سخت افزاری را دریافت می کند و می تواند آنها را مصرف کند یا به برنامه ارسال کند. به عنوان مثال، ممکن است بخواهید از کلیدهای جهت دار برای پیمایش در UI خود برای انتخاب نامزد در طول ترکیب استفاده کنید. همچنین ممکن است بخواهید کلید بازگشت را به دام بیاندازید تا هر گفتگوی منشأ گرفته از پنجره روش ورودی را رد کنید.
برای رهگیری کلیدهای سخت افزاری، onKeyDown()
و onKeyUp()
را لغو کنید.
متد super()
را برای کلیدهایی که نمی خواهید خودتان کنترل کنید، فراخوانی کنید.
یک زیرنوع IME ایجاد کنید
زیرگروهها به IME اجازه میدهند حالتها و زبانهای ورودی متعددی را که توسط یک IME پشتیبانی میشوند، نمایش دهد. یک زیرگروه می تواند نشان دهنده موارد زیر باشد:
- محلی، مانند en_US یا fr_FR
- یک حالت ورودی، مانند صدا، صفحه کلید، یا دست خط
- سایر سبکهای ورودی، فرمها یا ویژگیهای خاص IME، مانند طرحبندی صفحهکلید 10 کلیدی یا QWERTY
حالت می تواند هر متنی باشد، مانند "صفحه کلید" یا "صدا". یک زیرمجموعه همچنین میتواند ترکیبی از این موارد را نشان دهد.
اطلاعات نوع فرعی برای گفتگوی تعویض کننده IME که از نوار اعلان و تنظیمات IME در دسترس است استفاده می شود. این اطلاعات همچنین به چارچوب اجازه میدهد تا زیرنوع خاصی از یک IME را مستقیماً بیاورد. هنگامی که یک IME می سازید، از تسهیلات subtype استفاده کنید، زیرا به کاربر کمک می کند زبان ها و حالت های مختلف IME را شناسایی کرده و بین آنها سوئیچ کند.
با استفاده از عنصر <subtype>
، انواع فرعی را در یکی از فایل های منبع XML روش ورودی تعریف کنید. قطعه کد زیر یک IME را با دو نوع فرعی تعریف میکند: یک زیرگروه صفحه کلید برای زبان انگلیسی ایالات متحده و یک زیرنوع صفحه کلید دیگر برای زبان فرانسه برای زبان فرانسه:
<input-method xmlns:android="http://schemas.android.com/apk/res/android" android:settingsActivity="com.example.softkeyboard.Settings" android:icon="@drawable/ime_icon"> <subtype android:name="@string/display_name_english_keyboard_ime" android:icon="@drawable/subtype_icon_english_keyboard_ime" android:languageTag="en-US" android:imeSubtypeMode="keyboard" android:imeSubtypeExtraValue="somePrivateOption=true" /> <subtype android:name="@string/display_name_french_keyboard_ime" android:icon="@drawable/subtype_icon_french_keyboard_ime" android:languageTag="fr-FR" android:imeSubtypeMode="keyboard" android:imeSubtypeExtraValue="someVariable=30,someInternalOption=false" /> <subtype android:name="@string/display_name_german_keyboard_ime" ... /> </input-method>
برای اطمینان از اینکه زیرگروههای شما به درستی در UI برچسبگذاری شدهاند، از «%s» استفاده کنید تا یک برچسب فرعی که همان برچسب محلی نوع فرعی است، دریافت کنید. این در دو قطعه کد بعدی نشان داده شده است. اولین قطعه بخشی از فایل XML روش ورودی را نشان می دهد:
<subtype android:label="@string/label_subtype_generic" android:imeSubtypeLocale="en_US" android:icon="@drawable/icon_en_us" android:imeSubtypeMode="keyboard" />
قطعه بعدی بخشی از فایل strings.xml
IME است. منبع رشته label_subtype_generic
، که توسط تعریف UI روش ورودی برای تنظیم برچسب نوع فرعی استفاده میشود، به صورت زیر تعریف میشود:
<string name="label_subtype_generic">%s</string>
این تنظیم باعث می شود که نام نمایشی نوع فرعی با تنظیمات محلی مطابقت داشته باشد. به عنوان مثال، در هر زبان انگلیسی، نام نمایشی "English (ایالات متحده آمریکا)" است.
زیر انواع IME را از نوار اعلان انتخاب کنید
سیستم اندروید همه زیرشاخههای افشا شده توسط همه IMEها را مدیریت میکند. زیرگروه های IME به عنوان حالت های IME که به آن تعلق دارند در نظر گرفته می شوند. همانطور که در شکل زیر نشان داده شده است، کاربر می تواند از نوار اعلان یا برنامه تنظیمات به منوی زیرگروه های IME موجود پیمایش کند:
زیر انواع IME را از تنظیمات سیستم انتخاب کنید
کاربر همچنین میتواند نحوه استفاده از زیرگروهها را در پانل تنظیمات زبان و ورودی در تنظیمات سیستم کنترل کند:
جابهجایی بین زیرگروههای IME
میتوانید با ارائه یک کلید جابجایی، مانند نماد زبان کرهای روی صفحهکلید، به کاربران اجازه دهید به راحتی بین انواع فرعی IME جابجا شوند. این قابلیت استفاده از صفحه کلید را بهبود می بخشد و برای کاربر راحت است. برای فعال کردن این تغییر، مراحل زیر را انجام دهید:
- در فایل های منبع XML روش ورودی
supportsSwitchingToNextInputMethod = "true"
اعلام کنید. اعلان شما باید شبیه قطعه کد زیر باشد:<input-method xmlns:android="http://schemas.android.com/apk/res/android" android:settingsActivity="com.example.softkeyboard.Settings" android:icon="@drawable/ime_icon" android:supportsSwitchingToNextInputMethod="true">
- متد
shouldOfferSwitchingToNextInputMethod()
را فراخوانی کنید. - اگر روش true را برگرداند، یک کلید سوئیچینگ نمایش دهید.
- هنگامی که کاربر روی کلید سوئیچینگ ضربه میزند،
switchToNextInputMethod()
را فراخوانی میکند که false است. مقدار false به سیستم میگوید که همه زیرگروهها را بدون در نظر گرفتن اینکه به کدام IME تعلق دارند، یکسان رفتار کند. تعیین true مستلزم چرخش سیستم در میان انواع فرعی در IME فعلی است.
ملاحظات عمومی IME
در اینجا موارد دیگری وجود دارد که باید هنگام پیاده سازی IME خود در نظر بگیرید:
- راهی برای کاربران فراهم کنید تا گزینهها را مستقیماً از رابط کاربری IME تنظیم کنند.
- راهی را برای کاربران فراهم کنید تا مستقیماً از رابط کاربری روش ورودی به IME دیگری تغییر کنند، زیرا ممکن است چندین IME روی دستگاه نصب شده باشد.
- رابط کاربری IME را به سرعت مطرح کنید. هر منبع بزرگی را از قبل بارگیری یا بارگیری کنید تا کاربران به محض ضربه زدن روی یک فیلد متن، IME را ببینند. منابع و نماهای کش برای فراخوانی های بعدی روش ورودی.
- بلافاصله پس از پنهان شدن پنجره روش ورودی، تخصیص حافظه بزرگ را آزاد کنید تا برنامه ها حافظه کافی برای اجرا داشته باشند. اگر IME برای چند ثانیه مخفی شد، از پیام تاخیری برای انتشار منابع استفاده کنید.
- اطمینان حاصل کنید که کاربران میتوانند تا حد ممکن نویسهها را برای زبان یا محلی مرتبط با IME وارد کنند. کاربران ممکن است از علائم نگارشی در گذرواژهها یا نامهای کاربری استفاده کنند، بنابراین IME شما باید کاراکترهای مختلفی را ارائه کند تا به کاربران اجازه دهد رمز عبور را وارد کرده و به دستگاه دسترسی داشته باشند.