ピン入力用の 1 つのレイアウトに 17 個の API、DPAD コントロール、および 4 個の編集テキストがあります。ユーザーがアプリでロックされているものを開こうとすると、4 桁のピンを入力する必要があるフラグメントに直面しました。ビューが開いたら、プログラムでビュー フォーカスを最初のピン edittext に設定し、キーボードを表示します。ユーザーが最初のピンの数字を入力すると、ユーザーが 2 番目のピンの値を入力する必要がある 2 番目のピン edittext にプログラムでビュー フォーカスを設定します。
問題: ユーザーが最初のピンの値を入力すると、キーボード フォーカス (強調表示)が消え、2 番目のピンの場合、ユーザーは最初からキーボードをナビゲートする必要があります。21 api では、同じコードがうまく機能し、キーボード フォーカス (強調表示)は以前の値のままでした。ビューのフォーカスを別のビューに 設定しているときに、以前に17 APIを使用していたときにキーボードのフォーカス(強調表示)を残す方法
コードは次のとおりです。
public class PasswordFragment extends BaseFragment {
private final ArrayList<Disposable> disposables = new ArrayList<>();
@BindViews({R.id.passFirst, R.id.passSecond, R.id.passThird, R.id.passFour})
EditText[] pinEdits;
@BindDrawable(R.drawable.border_light)
Drawable borderLight;
@BindDrawable(R.drawable.border_white)
Drawable borderWhite;
@Inject
PrefsRepo prefsRepo;
@Inject
ToastUtil toastUtil;
@BindView(R.id.passTitle)
TextView passTitle;
@BindString(R.string.insert_pin)
String defDialogMes;
private InputMethodManager inputMethodManager;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
App.getAppComponent().inject(this);
inputMethodManager = (InputMethodManager) getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
}
@Nullable
@Override
public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View v = super.onCreateView(inflater, container, savedInstanceState);
for (EditText pinEdit : pinEdits) {
pinEdit.setOnFocusChangeListener((view, hasFocus) -> {
if (hasFocus) {
view.setBackground(borderLight);
}
});
}
disposables.add(RxTextView.textChanges(pinEdits[0]).subscribe(charSequence -> {
if (charSequence.length() == 1) {
pinEdits[1].requestFocus();
}
}));
disposables.add(RxTextView.textChanges(pinEdits[1]).subscribe(charSequence -> {
if (charSequence.length() == 1) {
pinEdits[2].requestFocus();
}
}));
disposables.add(RxTextView.textChanges(pinEdits[2]).subscribe(charSequence -> {
if (charSequence.length() == 1) {
pinEdits[3].requestFocus();
}
}));
disposables.add(RxTextView.textChanges(pinEdits[3])
.subscribe(charSequence -> {
if (charSequence.length() == 1) {
String pin = pinEdits[0].getText().toString() + pinEdits[1].getText().toString() + pinEdits[2].getText().toString() + pinEdits[3].getText().toString();
if (pin.length() == 4) {
do someting;
} else {
toastUtil.showToast(getString(R.string.pin_incorrect));
}
}
}));
return v;
}
@Override
public void onResume() {
clearEditTexts();
super.onResume();
}
private void clearEditTexts() {
for (EditText pinEdit : pinEdits) {
pinEdit.getText().clear();
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.JELLY_BEAN_MR1){
pinEdit.setOnClickListener(view -> {
showKeyBoard(pinEdit);
});
}
}
setFocusOnFirst();
}
private void setFocusOnFirst() {
pinEdits[0].setBackground(borderLight);
pinEdits[0].requestFocus();
pinEdits[0].postDelayed(() -> {
showKeyBoard(pinEdits[0]);
}, 50);
}
@Override
public void onDestroy() {
super.onDestroy();
for (Disposable disposable : disposables) {
disposable.dispose();
}
}
@Override
protected int getLayoutId() {
return R.layout.fragment_password;
}
private void showKeyBoard(EditText editText){
inputMethodManager.showSoftInput(editText, InputMethodManager.SHOW_IMPLICIT);
}
}