Как скрыть мягкую клавиатуру изнутри фрагмента? - PullRequest
78 голосов
/ 29 октября 2011

У меня есть FragmentActivity, использующий ViewPager для обслуживания нескольких фрагментов. Каждый из них представляет собой ListFragment со следующим макетом:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent">
    <LinearLayout
        android:orientation="vertical"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:padding="8dp">
        <ListView android:id="@id/android:list"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent" />

        <EditText android:id="@+id/entertext"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content" />
    </LinearLayout>
</LinearLayout>

При запуске упражнения отображается программная клавиатура. Чтобы исправить это, я сделал следующее внутри фрагмента:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    //Save the container view so we can access the window token
    viewContainer = container;
    //get the input method manager service
    imm = (InputMethodManager)getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
    . . .
}

@Override
public void onStart() {
    super.onStart();

    //Hide the soft keyboard
    imm.hideSoftInputFromWindow(viewContainer.getWindowToken(), 0);
}

Я сохраняю входящий параметр ViewGroup container из onCreateView как способ доступа к токену окна для основного действия. Это работает без ошибок, но клавиатура не скрывается от вызова на hideSoftInputFromWindow в onStart.

Первоначально я пытался использовать раздутый макет вместо container, т.е.:

imm.hideSoftInputFromWindow(myInflatedLayout.getWindowToken(), 0);

но это бросило NullPointerException, предположительно, потому что сам фрагмент не является активностью и не имеет уникального жетона окна?

Есть ли способ скрыть программную клавиатуру из фрагмента или мне следует создать метод в FragmentActivity и вызвать его из фрагмента?

Ответы [ 11 ]

161 голосов
/ 02 апреля 2012

Пока ваш фрагмент создает представление, вы можете использовать IBinder (оконный токен) из этого представления после того, как было прикреплено.Например, вы можете переопределить onActivityCreated в вашем фрагменте:

@Override
public void onActivityCreated(Bundle savedInstanceState) {
    super.onActivityCreated(savedInstanceState);
    final InputMethodManager imm = (InputMethodManager) getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
    imm.hideSoftInputFromWindow(getView().getWindowToken(), 0);
}
77 голосов
/ 29 мая 2014

Ничего, кроме следующей строки кода, работало для меня:

getActivity().getWindow().setSoftInputMode(
            WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
21 голосов
/ 09 ноября 2011

Если вы добавите следующий атрибут к определению манифеста своей деятельности, он полностью запретит выдвижение клавиатуры при открытии вашей активности. Надеюсь, это поможет:

(Добавить к определению манифеста вашей деятельности):

android:windowSoftInputMode="stateHidden"
12 голосов
/ 12 мая 2015
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.fragment_my, container,
                false);
        someClass.onCreate(rootView);
        return rootView;
    }

Сохранить экземпляр моего корневого представления в моем классе

View view;

public void onCreate(View rootView) {
    view = rootView;

Используйте вид, чтобы скрыть клавиатуру

 public void removePhoneKeypad() {
    InputMethodManager inputManager = (InputMethodManager) view
            .getContext()
            .getSystemService(Context.INPUT_METHOD_SERVICE);

    IBinder binder = view.getWindowToken();
    inputManager.hideSoftInputFromWindow(binder,
            InputMethodManager.HIDE_NOT_ALWAYS);
}
9 голосов
/ 11 декабря 2013

Исключение составляет DialogFragment, однако, фокус встроенного Dialog должен быть скрыт, вместо этого только первый EditText внутри встроенного Dialog

this.getDialog().getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN);
6 голосов
/ 19 июля 2017

во Fragemnt, работающем с этим кодом

    getActivity().getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
3 голосов
/ 17 августа 2018

это будет работать в моем случае, когда во вкладках я переключаюсь с одного фрагмента на другой фрагмент

@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
    super.setUserVisibleHint(isVisibleToUser);
    if (isVisibleToUser) {
        try {
            InputMethodManager mImm = (InputMethodManager) getActivity().getSystemService(Context.INPUT_METHOD_SERVICE);
            mImm.hideSoftInputFromWindow(getView().getWindowToken(), 0);
            mImm.hideSoftInputFromWindow(getActivity().getCurrentFocus().getWindowToken(), 0);
        } catch (Exception e) {
            Log.e(TAG, "setUserVisibleHint: ", e);
        }
    }
}
2 голосов
/ 14 июля 2018

Используйте этот статический метод из любого места (действие / фрагмент), которое вам нравится.

public static void hideKeyboard(Activity activity) {
    try{
        InputMethodManager inputManager = (InputMethodManager) activity
                .getSystemService(Context.INPUT_METHOD_SERVICE);
        View currentFocusedView = activity.getCurrentFocus();
        if (currentFocusedView != null) {
            inputManager.hideSoftInputFromWindow(currentFocusedView.getWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);
        }
    }catch (Exception e){
        e.printStackTrace();
    }
}

Если вы хотите использовать для фрагмента, просто позвоните hideKeyboard(((Activity) getActivity())).

0 голосов
/ 18 мая 2019

В Котлине:

(activity?.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager).hideSoftInputFromWindow(view?.windowToken,0)
0 голосов
/ 19 сентября 2018

Просто добавьте эту строку в ваш код:

getActivity().getWindow().setSoftInputMode(
                WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
...