ユーザーに提案を表示するためにAutoCompleteTextView
を提供しました。ユーザーが選択したアイテムに基づいて、アイテムのIDを取得し、データベース側で使用しています。今私の問題は、ユーザーにAutoCompleteTextView
からのみ選択を強制することです(つまり、ユーザーは自分のテキストを入力しないでください)。これはクライアントの要件です。これを行う方法?
これはかなり簡単な解決策です:
setOnItemClickListener
にAutoCompleteTextView
を設定することで、選択した値を格納する変数を作成できます。その後、ユーザーがフィールドにnull
を追加することで、フィールドに入力するたびにその値をTextWatcher
できます。最後に、続行する前に変数がnullでないことを検証できます。
String my_var; //keep track!
AutoCompleteTextView tv = (AutoCompleteTextView) layout.findViewById(R.id.tv);
tv.setAdapter(my_adapter);
tv.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
my_var = my_adapter.getItem(position).toString();
}
});
/**
* Unset the var whenever the user types. Validation will
* then fail. This is how we enforce selecting from the list.
*/
tv.addTextChangedListener(new TextWatcher() {
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
my_var = null;
}
@Override
public void afterTextChanged(Editable s) {}
});
私が取り組んでいるプロジェクトにはたまたまこのような要件が必要でしたが、私は要件を実装した方法を皆さんと共有します。
オートコンプリートテキストビューのフォーカス変更リスナーを追加し、ユーザーがオートコンプリートからフォーカスを変更したときにチェックし、状況を簡単に処理しました。
autoTextViewCountry.setOnFocusChangeListener(new View.OnFocusChangeListener() {
@Override
public void onFocusChange(View view, boolean b) {
if(!b) {
// on focus off
String str = autoTextViewCountry.getText().toString();
ListAdapter listAdapter = autoTextViewCountry.getAdapter();
for(int i = 0; i < listAdapter.getCount(); i++) {
String temp = listAdapter.getItem(i).toString();
if(str.compareTo(temp) == 0) {
return;
}
}
autoTextViewCountry.setText("");
}
}
});
したがって、私の実装は次のとおりです。入力されたテキストが配列アダプターに存在しない場合、フォーカスでテキストビューが空に変更され、後で登録の次のステージに進むときに、このテキストビューが空かどうかを確認します。
このアプローチが誰かを助けることを願っています。
ハッピーコーディング。
NiceAutoCompleteTextView は、isSelectionFromPopup()
を呼び出して、ドロップダウンポップアップから選択が行われたかどうかを確認する機能を提供します
OKユーザーの入力を、提案ボックスにリストされている項目のリストに含まれているテキストに制限したいと思います。
たとえば、次の場合:
この場合、ユーザーは最初の文字「O」と「T」のみを入力できます。前に入力したテキストに応じて同様に続きます。
これを実現するには、TextViewのsetFiltersメソッドを利用できます。
editBox = (TextView) findViewById(R.id.editBox);
editBox.setFilters(getFilters());
editBox.addTextChangedListener(this);
editBox.setOnFocusChangeListener(this);
さらに、おそらくテキスト変更リスナーとフォーカスリスナーが反応する必要がありますそして、新しい文字が入力されたときにフィルターされたリストを更新します...さらに、フィルターを更新します。
以下は、プロジェクトで使用した10進数フィルターの例です。
protected InputFilter[] getFilters()
{
InputFilter[] filters = new InputFilter[1];
filters[0] = new InputFilter()
{
public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend)
{
// limit input to digits and decimal / thousand separator only
// in case thousand separator is pressed, change it to decimal
// separator instead
String output = "";
if (number.isFocused())
{
for (int i = start; i < end; i++)
{
char c = source.charAt(i);
if (isDecimalOrThousandSeparator(c))
{
output = output + Character.toString(decimalSeparator);
}
else if (Character.isDigit(c))
{
output = output + Character.toString(c);
}
}
return output == "" ? null : output;
}
return null;
}
};
return filters;
}
単純な解決策は、現在の入力がアダプターの項目の1つであるかどうかを確認することです。あなたはこのようにそれを行うことができます:
val AutoCompleteTextView.isValid: Boolean
get() {
for (i in 0 until adapter.count) {
if (adapter.getItem(i) == text.toString()) {
return true
}
}
return false
}