it-swarm.com.ru

OnCloseListener в SearchView не работает

Я пытаюсь добавить поддержку SearchView в Android 3.0+ ActionBar, но не могу заставить работать OnCloseListener.

Вот мой код:

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.menu, menu);
    searchView = (SearchView) menu.findItem(R.id.search_textbox).getActionView();
    searchView.setOnQueryTextListener(new OnQueryTextListener() {
        @Override
        public boolean onQueryTextChange(String newText) {
            searchLibrary(newText);
            return false;
        }
        @Override
        public boolean onQueryTextSubmit(String query) { return false; }
    });
    searchView.setOnCloseListener(new OnCloseListener() {
        @Override
        public boolean onClose() {
            System.out.println("Testing. 1, 2, 3...");
            return false;
        }
    });
    return true;
}

Поиск работает отлично, и все работает, кроме OnCloseListener. Ничего не печатается в Logcat. Вот Logcat, когда я нажимаю кнопку «Закрыть»:

02-17 13:01:52.914: I/TextType(446): TextType = 0x0
02-17 13:01:57.344: I/TextType(446): TextType = 0x0
02-17 13:02:02.944: I/TextType(446): TextType = 0x0

Я просмотрел документацию и образцы, но, похоже, ничего не изменило. Я запускаю его на Asus Transformer Prime и Galaxy Nexus, оба на бутерброде с мороженым. Есть идеи?

Обновление:

Да - System.out.println()делает работает. Вот доказательство:

   @Override
 public boolean onQueryTextChange(String newText) {
    System.out.println(newText + "hello");
    searchLibrary(newText);
    return false;
 }

Результаты в этом Logcat:

02-17 13:04:20.094: I/System.out(21152): hello
02-17 13:04:24.914: I/System.out(21152): thello
02-17 13:04:25.394: I/System.out(21152): tehello
02-17 13:04:25.784: I/System.out(21152): teshello
02-17 13:04:26.064: I/System.out(21152): testhello
91
Michell Bak

Я также сталкиваюсь с этой проблемой, и у меня нет выбора, кроме как отказаться от "oncloselistener". Вместо этого вы можете получить ваш menuItem, затем setOnActionExpandListener. Затем переопределите unimplents методы. 

@Override
public boolean onMenuItemActionExpand(MenuItem item) {
    // TODO Auto-generated method stub
    Log.d("*******","onMenuItemActionExpand");
    return true;
}

@Override
public boolean onMenuItemActionCollapse(MenuItem item) {
    //do what you want to when close the sesarchview
    //remember to return true;
    Log.d("*******","onMenuItemActionCollapse");
    return true;
}
135
niki huang

Для Android API 14+ (ICS и выше) используйте этот код:

// When using the support library, the setOnActionExpandListener() method is
// static and accepts the MenuItem object as an argument
MenuItemCompat.setOnActionExpandListener(menuItem, new OnActionExpandListener() {
    @Override
    public boolean onMenuItemActionCollapse(MenuItem item) {
        // Do something when collapsed
        return true;  // Return true to collapse action view
    }

    @Override
    public boolean onMenuItemActionExpand(MenuItem item) {
        // Do something when expanded
        return true;  // Return true to expand action view
    }
});

Для получения дополнительной информации: http://developer.Android.com/guide/topics/ui/actionbar.html#ActionView

Ссылка: onActionCollapse/onActionExpand

59
lomza

Для этой проблемы я придумал что-то вроде этого,

private SearchView mSearchView;

@TargetApi(14)
@Override
public boolean onCreateOptionsMenu(Menu menu)
{

    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.conversation_index_activity_menu, menu);

    mSearchView = (SearchView) menu.findItem(R.id.itemSearch).getActionView();

    MenuItem menuItem = menu.findItem(R.id.itemSearch);

    int currentapiVersion = Android.os.Build.VERSION.SDK_INT;
    if (currentapiVersion >= Android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH)
    {
        menuItem.setOnActionExpandListener(new OnActionExpandListener()
        {

            @Override
            public boolean onMenuItemActionCollapse(MenuItem item)
            {
                // Do something when collapsed
                Log.i(TAG, "onMenuItemActionCollapse " + item.getItemId());
                return true; // Return true to collapse action view
            }

            @Override
            public boolean onMenuItemActionExpand(MenuItem item)
            {
                // TODO Auto-generated method stub
                Log.i(TAG, "onMenuItemActionExpand " + item.getItemId());
                return true;
            }
        });
    } else
    {
        // do something for phones running an SDK before froyo
        mSearchView.setOnCloseListener(new OnCloseListener()
        {

            @Override
            public boolean onClose()
            {
                Log.i(TAG, "mSearchView on close ");
                // TODO Auto-generated method stub
                return false;
            }
        });
    }


    return super.onCreateOptionsMenu(menu);

}
27
Ben Benson

Я столкнулся с той же проблемой на Android 4.1.1. Похоже, это известная ошибка: https://code.google.com/p/Android/issues/detail?id=25758

В любом случае, в качестве обходного пути я использовал слушателя изменения состояния (когда SearchView отсоединен от панели действий, он также, очевидно, закрыт).

view.addOnAttachStateChangeListener(new OnAttachStateChangeListener() {

    @Override
    public void onViewDetachedFromWindow(View arg0) {
        // search was detached/closed
    }

    @Override
    public void onViewAttachedToWindow(View arg0) {
        // search was opened
    }
});

Выше код работал хорошо в моем случае.


Я отправляю тот же ответ здесь: https://stackoverflow.com/a/24573266/2162924

17
Dario

В итоге я использовал хак, который хорошо работает для моих целей - не уверен, что он будет работать во всех целях. В любом случае, я проверяю, пустой ли поисковый запрос. Это на самом деле не связано с SearchView's OnCloseListener - это все равно не работает!

searchView.setOnQueryTextListener(new OnQueryTextListener() {
            @Override
            public boolean onQueryTextChange(String newText) {
                if (newText.length() > 0) {
                    // Search
                } else {
                    // Do something when there's no input
                }
                return false;
            }
            @Override
            public boolean onQueryTextSubmit(String query) { return false; }
        });
7
Michell Bak

Ну, это решило мою проблему:

Пункт меню с showAsAction="always"

<item
    Android:id="@+id/action_search"
    Android:icon="@drawable/ic_action_search"
    Android:title="Search"
    app:actionViewClass="Android.support.v7.widget.SearchView"
    app:showAsAction="always"/>

и в деятельности

searchView.setOnCloseListener(new OnCloseListener() {

        @Override
        public boolean onClose() {

            Log.i("SearchView:", "onClose");
            searchView.onActionViewCollapsed();
            return false;
        }
    });
6
Sushant

Чтобы заставить OnCloseListener работать, убедитесь, что для showAsAction установлено значение always в пункте меню поиска.

<menu xmlns:Android="http://schemas.Android.com/apk/res/Android"
      xmlns:app="http://schemas.Android.com/apk/res-auto"
      xmlns:tools="http://schemas.Android.com/tools"
      tools:context=".SearchActivity">

    <item
        Android:id="@+id/search"
        Android:title="@string/search"
        Android:icon="@drawable/ic_search_toolbar"
        app:showAsAction="always"
        app:actionViewClass="Android.support.v7.widget.SearchView"/>
</menu>
2
Julio Betta

Я столкнулся с той же проблемой, когда onCloseListener не вызывает SearchView . Поймите из ошибки, поднятой в 25758 , и некоторые сообщения, которые я прочитал, чтобы вызвать onCloseListener, вам нужно установить:

searchView.setIconifiedByDefault(true);

Но для моего случая я хотел, чтобы окно поиска было открыто и не отображалось постоянно. Мне удается решить эту проблему, добавив еще одну строку ниже:

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.search_bar, menu);
    SearchManager searchManager = (SearchManager) getSystemService(Context.SEARCH_SERVICE);
    searchView = (SearchView) menu.findItem(R.id.search).getActionView();
    searchView.setSearchableInfo(searchManager.getSearchableInfo(getComponentName()));
    searchView.setOnQueryTextListener(queryTextListener);
    searchView.setIconifiedByDefault(true);
    searchView.setIconified(false);
    return true;
}

SearchView.setIconified (false) приведет к открытию searchView, несмотря на то, что в предыдущей строке для значка по умолчанию было указано значение true. Таким образом, мне удалось получить SearchView , Который все время открывается, и заставить его вызывать onCloseListener.

2
gkl

Для проблемы MenuItemCompat я добавил ViewTreeObserver для отслеживания состояния видимости. Вы можете проверить мой ответ здесь: https://stackoverflow.com/a/28762632/1633609

1
Sir NIkolay Cesar The First

Создайте пункт меню с app:showAsAction, установленным в всегда. 

<item   
 Android:id="@+id/action_search"  
 Android:title="..."  
 Android:icon="..."  
 app:actionViewClass="Android.support.v7.widget.SearchView"  
 app:showAsAction="always"/>

При создании SearchView в методе onCreateOptionsMenu сделайте что-то похожее

inflater.inflate(R.menu.menu_search, menu);
final MenuItem item = menu.findItem(R.id.action_search);
final SearchView search = (SearchView) item.getActionView();
search.setQueryHint(getString(R.string.search_brand_item));
search.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
  @Override
  public boolean onQueryTextSubmit(String query) {
    // add your code
    return false;
  }

  @Override
  public boolean onQueryTextChange(String newText) {
    // add your code 
    return false;
  }
});
search.setOnCloseListener(new SearchView.OnCloseListener() {
  @Override
  public boolean onClose() {
    // add your code here
    return false;
  }
});
search.setIconifiedByDefault(true); // make sure to set this to true

Для search.setIconifiedByDefault(true) должно быть установлено значение true, чтобы вызывать метод onClose() для функции SearchView.OnCloseListener(), созданной выше.

1
Ray Hunter

Это обходной путь, но сработал для меня

  searchView.setOnQueryTextListener(new Android.widget.SearchView.OnQueryTextListener() {

                String lastText;

                @Override
                public boolean onQueryTextChange(final String newText) {
                    if (lastText != null && lastText.length() > 1 && newText.isEmpty()) {
                        // close ctn clicked

                        return true;
                    }
}
0
Nativ

кажется, старая ветка уже, но я думал, что я получил ту же проблему API 18 в первом начале. После того, как погуглил, нашел эту ветку, еще час прочитал javadoc и попытался ошибиться из-за чего-то, что я не притворяюсь, что полностью понимаю в javadoc, следующая работа для меня сейчас:

searchView.setIconifiedByDefault(true);

   // OnQueryTextListener
   @Override
   public boolean onQueryTextSubmit(String query) {
      Log.d(tag, "onQueryTextSubmit: " + query);
      return true;
   }

   @Override
   public boolean onQueryTextChange(String query) {
      Log.d(tag, "onQueryTextChange: " + query);
      return true;
   }

   // OnCloseListener
   @Override
   public boolean onClose() {
      Log.w(tag, "onClose: ");
      return false;
   }

Я немного поиграл с true/false, это как-то имеет значение, и теперь это работает для меня. Надеюсь, это может сэкономить кому-то время.

0
Sean

Причина, по которой OnCloseListener не вызывается, заключается в том, что в коде Android есть ошибка - слушатель вызывается, только если вы также вызываете setIconifiedByDefault(true).

0
Joseph Earl
    searchView.setOnCloseListener {
        d("click", "close clicked")
        [email protected] false
    }

если вы нажмете на закрыть searchView ->

D/click: закрыть нажал

0
Oleksandr Yahnenko

Я столкнулся с этой проблемой, пытаясь обнаружить показ/отклонение SearchView. Я закончил тем, что использовал другого слушателя, и он работал для того, что мне нужно:

        setOnQueryTextFocusChangeListener { _, hasFocus ->
            if (hasFocus) {
                // SearchView is being shown
            } else {
                // SearchView was dismissed
            }
        }
0
Psylocke