2017-03-21 27 views
20

Tôi có một số RecyclerView trong một đoạn lặp lại trong TabLayout. Tôi gặp vấn đề về chế độ xem không thay đổi trong RecyclerView. Tôi có một spinner trên mỗi tab. Tôi muốn thay đổi dữ liệu khi các mục spinner được chọn.RecyclerView Các mục không thay đổi

trường hợp của tôi:

  1. khi chuyển đổi giữa các tab - mục thay đổi
  2. khi lựa chọn một giá trị trong spinner trong tab đầu tiên - mục không thay đổi. (nhưng dữ liệu đang thay đổi trong lớp adapter.Ie. Đầu tiên, nó không phải là null sau đó null trong khi lựa chọn. Nhưng dữ liệu đầu tiên không nulled không xuất hiện, nó thay thế bằng null. Tìm thấy nó bằng cách sử dụng breakpoint).

    Lưu ý: Trong trường hợp này, khi chuyển tab, các mục sẽ được đổi thành các mục được chọn trong tab trước đó. Và sau đó nó biến mất và hiển thị các mục hiện tại trong tab.

  3. khi chọn giá trị khác trong spinner trong tab cuối cùng - mục thay đổi.

My xem pager bộ chuyển đổi lớp


public class StudentViewPagerAdapter extends FragmentStatePagerAdapter { 
    private final List<StudentList> mFragmentList = new ArrayList<>(); 
    private final List<Clazz> mFragmentTitleList = new ArrayList<>(); 
    public StudentViewPagerAdapter(FragmentManager fm) { 
     super(fm); 
    } 
    public void addFragment(StudentList fragment,Clazz clazz){ 
     mFragmentList.add(fragment); 
     mFragmentTitleList.add(clazz); 
    } 
    @Override 
    public Fragment getItem(int position) { 
     return StudentList.newInstance(mFragmentTitleList.get(position)); 
    } 

    @Override 
    public int getCount() { 
     return mFragmentList.size(); 
    } 

    @Override 
    public CharSequence getPageTitle(int position) { 
     return mFragmentTitleList.get(position).getName(); 
    } 
} 

lớp RecyclerView bộ chuyển đổi của tôi

public class PeopleAdapter extends RecyclerView.Adapter<PeopleAdapter.MyViewHolder> implements View.OnClickListener { 
    private List<Student> dataList; 
    private Context context; 
    private Clicker clicker; 
    public PeopleAdapter(List<Student> data, Context context, Clicker clicker) { 
     this.dataList = data; 
     this.context = context; 
     this.clicker = clicker; 
    } 

    @Override 
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { 
     View itemView = LayoutInflater.from(parent.getContext()) 
       .inflate(R.layout.people_list_item, parent, false); 

     return new MyViewHolder(itemView); 
    } 

    @Override 
    public void onBindViewHolder(MyViewHolder holder, int position) { 
     Student data=dataList.get(position); 
     holder.email.setText(data.getEmail()); 
     holder.name.setText(data.getName()); 
     holder.phone.setText(data.getPhone()); 
     Glide.with(context).load(Method.getImageUrl(MyConfiguration.STUDENT_IMAGE_URL, 
       data.getStudentId())).asBitmap().into(holder.profilePic); 
     holder.edit.setOnClickListener(this); 
     holder.edit.setTag(position); 
    } 

    @Override 
    public int getItemCount() { 
     return dataList.size(); 
    } 

    @Override 
    public void onClick(View v) { 
     clicker.OnItemClicked((int) v.getTag(),null); 
    } 


    static class MyViewHolder extends RecyclerView.ViewHolder { 
    @BindView(R.id.name) 
     TextView name; 
     @BindView(R.id.email) 
     TextView email; 
     @BindView(R.id.phone) 
     TextView phone; 
     @BindView(R.id.image) 
     ImageView profilePic; 
     @BindView(R.id.imageedit) 
     ImageView edit; 
    MyViewHolder(View view) { 
     super(view); 
     ButterKnife.bind(this,view); 
    } 
} 

} 

mảnh tab

public class StudentList extends Fragment implements SectionChanger { 
     @BindView(R.id.studentlist) 
     RecyclerView mRecyclerview; 
     CompositeDisposable disposable; 
     private Unbinder unbinder; 
     private Clazz clazz; 
     private Requester requester; 

     public StudentList() { 
      StudentInformation.bindSectionChangeListener(this); 
     } 
     public static StudentList newInstance(Clazz clazz) { 
      StudentList fragment=new StudentList(); 
      Bundle args = new Bundle(); 
      args.putSerializable(MyConfiguration.SECTIONS, clazz); 
      fragment.setArguments(args); 
      return fragment; 
     } 
     @Override 
     public void onDestroyView() { 
      super.onDestroyView(); 
      unbinder.unbind(); 
      disposable.clear(); 
     } 

     @Override 
     public View onCreateView(LayoutInflater inflater, ViewGroup container, 
           Bundle savedInstanceState) { 
      // Inflate the layout for this fragment 
      View view = inflater.inflate(R.layout.fragment_student_list, container, false); 
      unbinder = ButterKnife.bind(this, view); 
      LinearLayoutManager mLayoutManager = new LinearLayoutManager(getActivity()); 
      mLayoutManager.setOrientation(LinearLayoutManager.VERTICAL); 
      mRecyclerview.setLayoutManager(mLayoutManager); 
      initializeRetrofit(); 
      if (getArguments() != null) { 
       clazz = (Clazz) getArguments().getSerializable(MyConfiguration.SECTIONS); 
       loadStudentJson(clazz != null ? clazz.getClassId() : null, 
         clazz != null ? clazz.getSections().get(0).getSectionId() : null); 
      } 
      return view; 
     } 

     /** 
     * Load students list 
     */ 
     public void loadStudentJson(String class_id,String section_id) { 


      disposable = new CompositeDisposable(requester.getStudentsInSection(class_id,section_id) 
        .observeOn(AndroidSchedulers.mainThread()) 
        .subscribeOn(Schedulers.io()) 
        .subscribe(
          this::handleResponse, 
          this::handleError 
        ) 
      ); 
     } 

     private void handleResponse(List<Student> list) { 
      PeopleAdapter adapter=new PeopleAdapter(list, getActivity(), 
        (position, name) -> Toast.makeText(getActivity(), position, Toast.LENGTH_LONG).show()); 
      mRecyclerview.setAdapter(adapter); 
      adapter.notifyDataSetChanged(); 
     } 

     private void handleError(Throwable error) { 
      Toast.makeText(getActivity(), "Error " + error.getLocalizedMessage(), Toast.LENGTH_SHORT).show(); 
     } 

     @Override 
     public void ChangeData(Section section) { 
      loadStudentJson(section.getClassId(),section.getSectionId()); 
     } 
     public void initializeRetrofit(){ 
      HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor(); 
      interceptor.setLevel(HttpLoggingInterceptor.Level.BODY); 
      CookieHandler handler=new Cookies(getActivity()); 
      //ClearableCookieJar cookieJar = new PersistentCookieJar(new SetCookieCache(), new SharedPrefsCookiePersistor(getActivity())); 
      OkHttpClient client = new OkHttpClient.Builder() 
        .addInterceptor(interceptor) 
        .cookieJar(new JavaNetCookieJar(handler)) 
        .build(); 

      requester = new Retrofit.Builder() 
        .baseUrl(MyConfiguration.BASE_URL) 
        .addCallAdapterFactory(RxJava2CallAdapterFactory.create()) 
        .addConverterFactory(JacksonConverterFactory.create()) 
        .client(client) 
        .build().create(Requester.class); 
     } 
    } 

ViewPager thiết lập đoạn

public class StudentInformation extends Fragment implements TabLayout.OnTabSelectedListener, AdapterView.OnItemSelectedListener { 
    // 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"; 
    private Unbinder unbinder; 
    // TODO: Rename and change types of parameters 
    private String mParam1; 
    private String mParam2; 
    CompositeDisposable disposable; 
    List<Section> sectionsList=new ArrayList<>(); 
    private OnConnectingFragments mListener; 
    @BindView(R.id.tabs) 
    TabLayout mTabLayout; 
    @BindView(R.id.viewpager) 
    ViewPager viewPager; 
    @BindView(R.id.secSelector) 
    Spinner spinner; 
    @BindView(R.id.className) 
    TextView className; 
    private static SectionChanger sectionChanger; 
    public StudentInformation() { 
     // Required empty public constructor 
    } 

    /** 
    * Use this factory method to create a new instance of 
    * this fragment using the provided parameters. 
    * 
    * @param param1 Parameter 1. 
    * @param param2 Parameter 2. 
    * @return A new instance of fragment StudentInformation. 
    */ 
    // TODO: Rename and change types and number of parameters 
    public static StudentInformation newInstance(String param1, String param2) { 
     StudentInformation fragment = new StudentInformation(); 
     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 void onDestroyView() { 
     super.onDestroyView(); 
     unbinder.unbind(); 
     disposable.clear(); 
    } 
    @Override 
    public View onCreateView(LayoutInflater inflater, ViewGroup container, 
          Bundle savedInstanceState) { 
     View view=inflater.inflate(R.layout.fragment_student_information, container, false); 
     unbinder = ButterKnife.bind(this, view); 
     LoadDataAndSetupViewPager(); 
     //setupViewPager(viewPager); 
     mTabLayout.setupWithViewPager(viewPager); 
     mTabLayout.addOnTabSelectedListener(this); 
     spinner.setOnItemSelectedListener(this); 
//  ArrayAdapter<String> arrayAdapter = new ArrayAdapter<>(getContext(), android.R.layout.simple_spinner_dropdown_item, MyConfiguration.CLASS_SECTIONS); 
//  spinner.setAdapter(arrayAdapter); 

     return view; 
    } 

    // TODO: Rename method, update argument and hook method into UI event 
    public void onButtonPressed(Fragment fragment,String tag) { 
     if (mListener != null) { 
      mListener.onClickedMenu(fragment,tag); 
     } 
    } 

    @Override 
    public void onAttach(Context context) { 
     super.onAttach(context); 
     if (context instanceof OnConnectingFragments) { 
      mListener = (OnConnectingFragments) context; 
     } else { 
      throw new RuntimeException(context.toString() 
        + " must implement OnFragmentInteractionListener"); 
     } 
    } 

    @Override 
    public void onDetach() { 
     super.onDetach(); 
     mListener = null; 
    } 
    /** 
    * This interface must be implemented by activities that contain this 
    * fragment to allow an interaction in this fragment to be communicated 
    * to the activity and potentially other fragments contained in that 
    * activity. 
    * <p> 
    * See the Android Training lesson <a href= 
    * "http://developer.android.com/training/basics/fragments/communicating.html" 
    * >Communicating with Other Fragments</a> for more information. 
    */ 
    /*private void setupViewPager(ViewPager viewPager) { 
     StudentViewPagerAdapter adapter = new StudentViewPagerAdapter(getFragmentManager()); 
     //adapter.addFragment(new StudentList(),"exam"); 

     for (String claz: MyConfiguration.CLASS) 
      adapter.addFragment(new StudentList(), claz); 
     viewPager.setAdapter(adapter); 
    }*/ 
    @OnClick(R.id.add) 
    public void OnClicked(LinearLayout view){ 
     onButtonPressed(new AddStudent(),"addStudent"); 
    } 

    @Override 
    public void onTabSelected(TabLayout.Tab tab) { 
     className.setText(String.format(getString(R.string.class_name), tab.getPosition()+1)); 
    } 

    @Override 
    public void onTabUnselected(TabLayout.Tab tab) { 

    } 

    @Override 
    public void onTabReselected(TabLayout.Tab tab) { 

    } 
    public void LoadDataAndSetupViewPager() { 
     HttpLoggingInterceptor interceptor = new HttpLoggingInterceptor(); 
     interceptor.setLevel(HttpLoggingInterceptor.Level.BODY); 
     ClearableCookieJar cookieJar = 
       new PersistentCookieJar(new SetCookieCache(), new SharedPrefsCookiePersistor(getActivity())); 
     OkHttpClient client = new OkHttpClient.Builder().addInterceptor(interceptor).build(); 

     Requester requester=new Retrofit.Builder() 
       .baseUrl(MyConfiguration.BASE_URL) 
       .addCallAdapterFactory(RxJava2CallAdapterFactory.create()) 
       .addConverterFactory(JacksonConverterFactory.create()) 
       .client(client) 
       .build().create(Requester.class); 
     disposable=new CompositeDisposable(requester.getClasses() ////GETTING CLASSES//// 
       .observeOn(AndroidSchedulers.mainThread()) 
       .subscribeOn(Schedulers.io()) 
       .flatMapIterable(clazzs -> clazzs) 
       .flatMap(clazz -> requester.getDivision(clazz.getClassId()) ////GETTING SECTIONS//// 
         .observeOn(AndroidSchedulers.mainThread()) 
         .subscribeOn(Schedulers.io()) 
         .flatMapIterable(sections -> sections) 
         .doOnNext(section -> {sectionsList.add(section); 
          Log.v("section_id",section.getSectionId());}) 
         .takeLast(1) 
         .map(section -> clazz) 
       ) 
       .doOnNext(clazz -> {clazz.setSections(sectionsList); 
        Log.v("List Size",sectionsList.size()+""); 
        sectionsList=new ArrayList<>(); 
       }) 
       .toList() 
       .subscribe(this::SetupViewPager, throwable -> Log.e("retroerror",throwable.toString()))); 

    } 
    public void SetupViewPager(List<Clazz> classList){ 
     StudentViewPagerAdapter adapter = new StudentViewPagerAdapter(getFragmentManager()); 
     //adapter.addFragment(new StudentList(),"exam"); 

     for (Clazz claz: classList){ 
      adapter.addFragment(new StudentList(), claz); 
     } 

     viewPager.setAdapter(adapter); 
     viewPager.setOffscreenPageLimit(3); 
     viewPager.addOnPageChangeListener(new ViewPager.SimpleOnPageChangeListener() { 
      @Override 
      public void onPageSelected(int position) { 
       super.onPageSelected(position); 
       List<Section>sections=classList.get(position).getSections(); 
       ArrayAdapter<Section> arrayAdapter = new ArrayAdapter<>(getContext(), android.R.layout.simple_spinner_dropdown_item, sections); 
       spinner.setAdapter(arrayAdapter); 
      } 
     }); 
    } 

    @Override 
    public void onItemSelected(AdapterView<?> parent, View view, int position, long id) { 
       Section section= (Section) parent.getItemAtPosition(position); 
       sectionChanger.ChangeData(section); 
    } 

    @Override 
    public void onNothingSelected(AdapterView<?> parent) { 

    } 
    public static void bindSectionChangeListener(SectionChanger changer){ 
     sectionChanger=changer; 
    } 
} 

enter image description here

Câu hỏi đặt ra là: Tại sao các dữ liệu có được không thay đổi khi chọn tùy chọn trên spinner đôi khi? (xem trường hợp của tôi)

+2

thậm chí tôi đang làm một số loại thiết lập tương tự như bạn đã làm và tôi phải đối mặt với cùng một vấn đề. Vấn đề là người xem đã trả lại cho tôi một ví dụ khác về đoạn của tôi. Vì vậy, hãy kiểm tra nó một lần nếu bạn đang nhận được instace đúng của mảnh. Tôi đã sử dụng phương thức dưới đây để lấy phân đoạn đúng, ngoài ra không có phương thức nào đã cho tôi một cá thể đúng - ((MyClass) viewPager.getAdapter(). InstantiateItem (viewPager, 0)). RefreshData(); – Gautam

+0

Vì vậy, trong 'handleResponse()' bạn đặt chính xác dữ liệu thành 'PeopleAdapter', nhưng dữ liệu không được hiển thị? – azizbekian

+0

Có chính xác. Nhưng khi tôi chuyển sang tab tiếp theo, dữ liệu đã chọn trước đó sẽ xuất hiện, sau đó đột nhiên thay thế bằng dữ liệu mới trong tab hiện tại. Ngoài ra, tab cuối cùng không có vấn đề gì. Nó cho thấy sự thay đổi dữ liệu về thời gian. @azizbekian –

Trả lời

1

Vấn đề chủ yếu là ở đây:

public StudentList() { 
      StudentInformation.bindSectionChangeListener(this); 
     } 

này không đảm bảo rằng các mảnh vỡ có thể nhìn thấy cho người sử dụng là các ràng buộc cuối cùng. Bạn thiết lập viewPager.setOffscreenPageLimit(3); có nghĩa là ba mảnh mỗi thời gian có thể được khởi tạo, vì vậy khi bạn đang ở trên một mảnh cũng là bên phải và bên trái được tạo ra.

Điều này giải thích lý do giải pháp cuối cùng hoạt động tốt, bởi vì không có phân đoạn phù hợp và có thể là đoạn cuối cùng được khởi tạo và ràng buộc.

Cá nhân tôi sẽ thay đổi việc triển khai xử lý spinner bên trong đoạn trang, vì nó chỉ hoạt động trên đoạn trang.Giải pháp

với một setUserVisibleHint

Bind đoạn trang cho chính một khi phân đoạn trở thành visble cho người dùng. Hãy chú ý đến bộ nhớ bị rò rỉ và phát hành các tài liệu tham khảo tĩnh

@Override 
    public void setUserVisibleHint(boolean isVisibleToUser) { 
    super.setUserVisibleHint(isVisibleToUser); 
    if(isVisibleToUser){ 
     StudentInformation.bindSectionChangeListener(this); 
    } 
    } 

Giải pháp với một chiếc xe buýt

Một giải pháp nhanh chóng có thể được sử dụng một chiếc xe buýt như thế này: http://square.github.io/otto/ Bằng cách này mỗi mảnh sẽ đăng ký vào một sự kiện SelectedItemChanged và làm mới chúng tự. Đoạn chính sẽ đăng các cập nhật mỗi lần lựa chọn spinner được thay đổi.

Tuy nhiên, ví dụ này khá lớn, vì vậy tôi không chắc chắn rằng không có vấn đề gì khác. Cố gắng chia sẻ một dự án hoàn chỉnh để nhận được trợ giúp cụ thể hơn.

+0

Không thay đổi. Nhưng tôi tìm thấy danh sách dữ liệu không phải là null và sau đó null khi chọn spinner –

7

thử sử dụng getChildFragmentManager() thay vì getFragmentManager() sẽ giải quyết vấn đề của bạn

+0

Thật không may, nó đã không làm việc .. :( –

1

Khi bạn làm việc với recycleview chỉ dựa vào mô hình lớp giá trị nộp của bạn. Thực hiện tất cả các thay đổi đối với lớp mô hình và tạo các khung nhìn hàng dựa trên các trường lớp mô hình này.

tức là bạn phải tạo hoặc thay đổi giá trị hoặc xem trạng thái dựa trên giá trị của các trường lớp mô hình.

Đối với ví dụ: nếu bạn muốn thực hiện một hàng (vị trí 30) vô hình, sau đó tôi sẽ thiết lập một lá cờ trong lớp mô hình gọi là tầm nhìn và đặt nó vào sai.

trong trường hợp này bất cứ khi nào xem vẽ lại trong giao diện người dùng, chúng tôi sẽ phải kiểm tra cờ và đặt chế độ hiển thị dựa trên giá trị trường lớp mô hình đó.

Như thế này bạn có thể tạo cờ, giá trị nội dung, tham số tuần tự hóa từ API vv trong lớp mô hình. Phương pháp này sẽ làm cho reclycleview của bạn nhất quán hơn thông qua các tương tác của người dùng và thay đổi API.

1

Kiểm tra của bạn StudentViewPagerAdapter

@Override 
    public Fragment getItem(int position) { 
     return mFragmentList.get(position).newInstance(mFragmentTitleList.get(position)); 
    } 

tại sao bạn gọi newInstance một lần nữa?

Thay thế nó với những điều sau đây, return mFragmentList.get(position);

+0

Không, khi tôi làm điều đó ném NullPointerException –

+0

tại sao? Nó không nên ném NullPointerException phải không? có nghĩa là khởi tạo StudentViewPagerAdapter của bạn có một số lỗi – Darish

+0

Tôi biết, nhưng tôi không thể tìm thấy .. hãy giúp đỡ .. –

-1

Thứ nhất, kiểm tra xem bạn thêm dòng này sau khi

setadapter : 
adapter.notifyDataSetChanged(); 

Một cách khác là làm cho cuối cùng MyViewHolder giữ và int vị trí

ví dụ.

public void onBindViewHolder(final MyViewHolder holder, final int position) { 
3

Tôi nghĩ rằng bạn nên thực hiện onNothingSelected() như sau

Section mCurrentSection; 

@Override 
public void onNothingSelected(AdapterView<?> parent) { 
      int position = parent.getSelectedItemPosition(); 
      Section section= (Section) parent.getItemAtPosition(position); 
      if (mCurrentSection == null || !mCurrentSection.equals(section)) { 
       mCurrentSection = section; 
       sectionChanger.ChangeData(section); 
      } 
} 

Bởi vì, onNothingSelected được gọi thay vì onItemSelected khi bạn lại chọn một mục của spinner.

Các vấn đề liên quan