Подтвердить что ты не робот

Реализация виджета в приложении Actionbar в приложении ListFragment

В настоящее время у меня установлено мое приложение с ListFragment слева и DetailsFragment справа (похоже на макет на планшете ниже).

layout

На панели действий у меня есть виджет поиска, который при отправке поискового запроса должен обновлять содержимое списка (в ListFragment). Я слежу за поиском по android dev (http://developer.android.com/guide/topics/search/search-dialog.html), но у меня возникают проблемы с поиском, чтобы обновить список в том же деятельность.

Мне удалось получить некоторую форму реализации, вызвав новое действие при отправке поискового запроса, но это заставляет кнопку "Назад" вести себя странно (т.е. вы возвращаетесь назад, и вы переходите к предыдущему действию, пока виджет поиска все еще отображается поисковый запрос).

Я добавил строки намерения в манифест, как показано ниже.

<intent-filter>
        <action android:name="android.intent.action.SEARCH" />
    </intent-filter>
    <meta-data android:name="android.app.searchable"
               android:resource="@xml/searchable"/>

Возможно ли, чтобы виджет поиска обновил список текущей активности?

4b9b3361

Ответ 1

Да, по сути, это "нормальный" вариант использования. Вы должны помнить, что фрагмент, который использует виджет панели поиска, должен быть частью, которая регистрирует его в целом, поскольку виджет будет уничтожен вместе с этим фрагментом.

Вот как правильно связать его:

в вашем ListFragment пусть будет известно, что у вас есть пункты меню опций...

@Override
public void onActivityCreated(Bundle savedInstanceState) {
    ..
    setHasOptionsMenu(true);
    ..
}

затем внутри того же ListFragment создайте меню параметров, переопределив обратный вызов, после того, как у вас есть ссылка на виджете SearchView, зарегистрируйте обратный вызов QueryTextListener:

@Override 
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
    inflater.inflate(R.menu.grid_default, menu); 
    SearchView searchView = (SearchView)menu.findItem(R.id.grid_default_search).getActionView();
    searchView.setOnQueryTextListener(queryListener);
}

создайте свой виджет поиска через программную (Java) или декларативно (XML) здесь XML (с именем grid_default.xml, сверху):

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android">

    <item     
        android:id="@+id/grid_default_search"
        android:icon="@android:drawable/ic_menu_search"
        android:title="search"
        android:showAsAction="always"
        android:actionViewClass="android.widget.SearchView" 
    />

    <!-- other items or whatever  -->

</menu>

Теперь в ListFragment вам нужно создать queryListener, который мы зарегистрировали выше:

private String grid_currentQuery = null; // holds the current query...

final private OnQueryTextListener queryListener = new OnQueryTextListener() {       

    @Override
    public boolean onQueryTextChange(String newText) {
        if (TextUtils.isEmpty(newText)) {
            getActivity().getActionBar().setSubtitle("List");               
            grid_currentQuery = null;
        } else {
            getActivity().getActionBar().setSubtitle("List - Searching for: " + newText);
            grid_currentQuery = newText;

        }   
        getLoaderManager().restartLoader(0, null, MyListFragment.this); 
        return false;
    }

    @Override
    public boolean onQueryTextSubmit(String query) {            
        Toast.makeText(getActivity(), "Searching for: " + query + "...", Toast.LENGTH_SHORT).show();
        return false;
    }
};

Теперь вы знаете, когда запрос изменен или отправлен, в моем примере я повторно запрашиваю в любое время нажатия клавиши, потому что моя база данных была небольшой и быстрой с результатами, вам может потребоваться изменить это. Но вы заметите как я просто использую виджет, чтобы установить частное поле класса внутри ListFragment в текущий текст виджета поиска (grid_currentQuery), а затем я просто вызываю getLoaderManager().restartLoader(0, null, MyListFragment.this);, где внутри my onCreateLoader я просто обновляю запрос, который я использовал следующим образом:

@Override
public Loader<Cursor> onCreateLoader(int id, Bundle bargs) {

    String sort = "SortColumn ASC";
    String[] grid_columns = new String[] { "ColumnA", "ColumnB", "Etc..." };
    String grid_whereClause = "ColumnToSearchBy LIKE ?"

    if (!TextUtils.isEmpty(grid_currentQuery)) {            
        return new CursorLoader(getActivity(), DataProvider.CONTENT_URI, grid_columns, grid_whereClause, new String[] { grid_currentQuery + "%" }, sort);
    }       

    return new CursorLoader(getActivity(), DataProvider.CONTENT_URI, grid_columns, null, null, sort);
}

и что все, что вам действительно нужно делать, я понимаю, что это отчасти нарезается, но это так, потому что есть больше вещей, которые должны произойти, установки адаптера, начиная загрузчика, используя getLoaderManager().initLoader(0, null, this); и все остальные обратные вызовы загрузчика, которые нужно обрабатывать... но если вы следуете лучшим практикам, установленным людьми андроида, (используя LoaderManager в ваших ListActivity и ListFragments, это должно быть довольно легко для вы...

Я надеюсь, что это поможет не просить разъяснений в комментариях, если вам нужно -ck