Как отфильтровать RecyclerView внутри фрагмента, который находится внутри ViewPager

#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);
}