#android
#Android
Вопрос:
Я хочу использовать SearchView для фильтрации моего RecyclerView (RecyclerView), который находится внутри фрагмента, который находится внутри ViewPager, ViewPager находится внутри пользовательского макета, и этот пользовательский макет включен в макет основного действия.
Я реализовал логику фильтрации в адаптере RecyclerView.
Что я хочу знать, так это где вызвать метод filter и как получить ссылку на RecyclerView.
Это класс адаптера для RecyclerView:
public class cardAdapter extends RecyclerView.Adapter<cardAdapter.noteHolder> implements Filterable {
private List<singleNoteDetail> myNotesList;
private List<singleNoteDetail> myNotesListFull;
private OnItemClickListenerMyNotes listener;
public static class noteHolder extends RecyclerView.ViewHolder {
protected static TextView titleField;
protected static TextView dateField;
public noteHolder(View v) {
super(v);
titleField = v.findViewById(R.id.singleNoteTitle);
dateField = v.findViewById(R.id.singleNoteDate);
}
public void bind(final singleNoteDetail item, final OnItemClickListenerMyNotes listener) {
itemView.setOnClickListener(new View.OnClickListener() {
@Override public void onClick(View v) {
listener.onItemClick(item);
}
});
}
}
public cardAdapter(List<singleNoteDetail> myNotesList,OnItemClickListenerMyNotes listener) {
this.myNotesList = myNotesList;
this.myNotesListFull = new ArrayList<>(myNotesList);
this.listener = listener;
}
@NonNull
@Override
public noteHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
View itemView = LayoutInflater.from(viewGroup.getContext())
.inflate(R.layout.single_note_card, viewGroup, false);
return new noteHolder(itemView);
}
@Override
public void onBindViewHolder(@NonNull noteHolder viewHolder, int i) {
singleNoteDetail singleNoteDetail = myNotesList.get(i);
noteHolder.titleField.setText(singleNoteDetail.title);
noteHolder.dateField.setText(singleNoteDetail.date);
viewHolder.bind(myNotesList.get(i), listener);
}
@Override
public int getItemCount() {
return myNotesList.size();
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public int getItemViewType(int position) {
return position;
}
@Override
public Filter getFilter() {
return filter;
}
private Filter filter = new Filter() {
@Override
protected FilterResults performFiltering(CharSequence constraint) {
List<singleNoteDetail> filteredList = new ArrayList<>();
if(constraint == null || constraint.length() == 0){
filteredList.addAll(myNotesListFull);
}else{
String pattern = constraint.toString().toLowerCase().trim();
for(singleNoteDetail item : myNotesListFull){
if(item.getTitle().toLowerCase().contains(pattern)){
filteredList.add(item);
}
}
}
FilterResults results = new FilterResults();
results.values = filteredList;
return results;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
myNotesList.clear();
myNotesList.addAll((List)results.values);
notifyDataSetChanged();
}
};
}
Это класс fragment:
public class MyNotes extends Fragment {
SharedPreferences prefs;
RecyclerView recyclerView;
SearchView searchView;
public RecyclerView.LayoutManager layoutManager;
public List<singleNoteDetail> cardList;
public RecyclerView.Adapter adapter;
ProgressBar progressBar;
// TODO: Rename parameter arguments, choose names that match
// the fragment initialization parameters, e.g. ARG_ITEM_NUMBER
private static final String ARG_PARAM1 = "param1";
private static final String ARG_PARAM2 = "param2";
// TODO: Rename and change types of parameters
private String mParam1;
private String mParam2;
private OnFragmentInteractionListener mListener;
public MyNotes() {
// Required empty public constructor
}
public void syncMyNotes(){
//
}
public static MyNotes newInstance(String param1, String param2) {
MyNotes fragment = new MyNotes();
Bundle args = new Bundle();
args.putString(ARG_PARAM1, param1);
args.putString(ARG_PARAM2, param2);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mParam1 = getArguments().getString(ARG_PARAM1);
mParam2 = getArguments().getString(ARG_PARAM2);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
prefs = getActivity().getSharedPreferences("Username",Context.MODE_PRIVATE);
View view = inflater.inflate(R.layout.fragment_my_notes, container, false);
recyclerView = view.findViewById(R.id.myNotesList);
progressBar = view.findViewById(R.id.progress);
layoutManager = new LinearLayoutManager( getActivity());
recyclerView.setLayoutManager(layoutManager);
recyclerView.setItemAnimator(new DefaultItemAnimator());
syncMyNotes();
adapter = new cardAdapter(cardList, new OnItemClickListenerMyNotes() {
@Override
public void onItemClick(singleNoteDetail item) {
Intent i = new Intent(getActivity(),NoteEditingActivity.class);
i.putExtra("noteTitle",item.title);
startActivity(i);
}
});
recyclerView.setAdapter(adapter);
return view;
}
public void onButtonPressed(Uri uri) {
if (mListener != null) {
mListener.onFragmentInteraction(uri);
}
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
if (context instanceof OnFragmentInteractionListener) {
mListener = (OnFragmentInteractionListener) context;
} else {
throw new RuntimeException(context.toString()
" must implement OnFragmentInteractionListener");
}
}
@Override
public void onAttachFragment(Fragment fragment) {
// TODO Auto-generated method stub
super.onAttachFragment(fragment);
Toast.makeText(getActivity().getApplicationContext(), String.valueOf(fragment.getId()), Toast.LENGTH_SHORT).show();
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
public interface OnFragmentInteractionListener {
// TODO: Update argument type and name
void onFragmentInteraction(Uri uri);
}
}
И вот макет, который содержит SearchView и ViewPager
existing_notes_layout.xml :
<LinearLayout
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"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@drawable/customborder"
android:orientation="vertical"
android:gravity="center_horizontal">
<android.support.design.widget.TabLayout
android:id="@ id/tablayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/colorPrimary"
android:minHeight="?attr/actionBarSize"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar">
</android.support.design.widget.TabLayout>
<SearchView
android:id="@ id/searchNotes"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
>
</SearchView>
<android.support.v4.view.ViewPager
android:id="@ id/pager"
android:layout_width="match_parent"
android:layout_height="match_parent">
</android.support.v4.view.ViewPager>
Ответ №1:
ну, у меня была точно такая же проблема. что я сделал, так это: сначала дайте идентификатор вашему пользовательскому описанию, то есть existing_notes_layout. и затем в вашем фрагменте внутри onCreateView напишите следующий код:
View view1 = getActivity().findViewById(R.id.activity); where "activity" is the id given to the customlayout i.e. existing_notes_layout.Then, find the id for the searchview you defined in your customlayout. like,
SearchView searchview = view1.findViewById(R.id.searchNotes);
затем, наконец, вы можете реализовать следующие методы searchview.
searchview.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextSubmit(String query) {
return false;
}
@Override
public boolean onQueryTextChange(String newText) {
cardAdapter.getFilter().filter(newText);
return false;
}
});
Ответ №2:
Поскольку вы уже включили представление поиска в свой фрагмент. затем вы можете получить свой фильтр при изменении текста запроса и отправить. Примерный способ выполнения этого заключается в следующем:
public class MyNotes extends Fragment implements SearchView.OnQueryTextListener {
.........
..........
........
@Override
public boolean onQueryTextSubmit(String query) {
return false;
}
@Override
public boolean onQueryTextChange(String newText) {
// If you have not instantiated your adapter then you must instantiate it
CardAdapter adapter = new cardAdapter(cardList, new OnItemClickListenerMyNotes() {
@Override
public void onItemClick(singleNoteDetail item) {
Intent i = new Intent(getActivity(),NoteEditingActivity.class);
i.putExtra("noteTitle",item.title);
startActivity(i);
}
});
// If you have already instantiated it you can start with this
if (adapter != null amp;amp; newText != null) {
adapter.getFilter().filter(newText);
return true;
}
return false;
}
}
Комментарии:
1. SearchView отсутствует во фрагменте, SearchView и ViewPager находятся в одном пользовательском макете, а фрагмент находится в ViewPager
Ответ №3:
Кто бы ни пришел сюда в будущем, вот решение для такого сценария.
Лучшим подходом может быть использование EventBus. Это действительно полезно в подобном сценарии, в котором мы должны передать определенные данные из Activity во фрагмент.
Для использования EventBus добавьте реализацию этой библиотеки ‘org.greenrobot:eventbus: 3.0.0’, затем вам нужно определить класс POJO, примерно такой:
class SearchQueryEvent {
String query;
public SearchQueryEvent(String query) {
this.query=query;
}
public String getQuery() {
return query;
}
}
Теперь ваш код должен быть изменен следующим образом:
MainActivity.java
SearchView searchView;
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate( R.menu.main, menu);
MenuItem myActionMenuItem = menu.findItem( R.id.action_search);
searchView = (SearchView) myActionMenuItem.getActionView();
searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextSubmit(String query) {
return false;
}
@Override
public boolean onQueryTextChange(String s) {
EventBus.getDefault().post(new SearchQueryEvent(s));
return false;
}
});
return true;
}
YourFragment.java
@Override
public void onResume() {
super.onResume();
EventBus.getDefault().register(this);
}
@Override
public void onPause() {
EventBus.getDefault().unregister(this);
super.onPause();
}
@Subscribe(threadMode = ThreadMode.MAIN)
public void onSearchQuery(SearchQueryEvent event) {
String query=event.getQuery();
adapter.getFilter().filter(query);
}