#java #android #android-fragments #android-recyclerview #android-viewpager
#java #Android #android-фрагменты #android-recyclerview #android-viewpager
Вопрос:
У меня есть tablayout viewpager в моем макете.
Фрагменты и их вкладки создаются динамически из категорий в моей базе данных, так что они обновляются, когда пользователь создает или удаляет новые категории.
Я хочу иметь возможность отображать элементы для соответствующих категорий в качестве элементов recyclerview на каждой из этих вкладок. У меня уже есть эти элементы, отображаемые в других местах обычных просмотров вторсырья, где они создаются / редактируются, поэтому остается только заставить их отображаться таким же образом в этом viewpager, отсортированном по вкладкам. Элементы также будут отфильтрованы по дате с помощью calendarview.
Пока у меня есть DynamicFragmentAdapter:
package com.example.navgraphtest;
import android.os.Bundle;
import androidx.annotation.NonNull;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentStatePagerAdapter;
import org.jetbrains.annotations.NotNull;
public class DynamicFragmentAdapter extends FragmentStatePagerAdapter {
private int mNumOfTabs;
DynamicFragmentAdapter(FragmentManager fm, int NumOfTabs) {
super(fm);
this.mNumOfTabs = NumOfTabs;
}
@NonNull
@Override
public Fragment getItem(int position) {
Bundle b = new Bundle();
b.putInt("position", position);
Fragment frag = DynamicFragment.newInstance();
frag.setArguments(b);
return frag;
}
@Override
public int getCount() {
return mNumOfTabs;
}
}
Сам DynamicFragment, который на данный момент просто отображает индекс, относящийся к его категории
package com.example.navgraphtest;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentActivity;
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;
public class DynamicFragment extends Fragment {
ArrayList<CategoryModel> arrayList;
public static DynamicFragment newInstance() {
return new DynamicFragment();
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.dynamic_fragment_layout, container, false);
initViews(view);
return view;
}
private void initViews(View view) {
DatabaseHelper databaseHelper = new DatabaseHelper(getActivity());
arrayList = new ArrayList<CategoryModel>(databaseHelper.getCategories());
TextView textView=view.findViewById(R.id.commonTextView);
{
textView.setText(String.valueOf("Category : " getArguments().getInt("position")));
}
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
}
@Override
public void onAttach(@NotNull Context context) {
super.onAttach(context);
}
@Override
public void onPause() {
super.onPause();
}
@Override
public void onResume() { super.onResume(); }
@Override
public void onStop() {
super.onStop();
}
@Override
public void onDestroyView() {
super.onDestroyView();
}
@Override
public void onDestroy() {
super.onDestroy();
}
}
Внутри основного фрагмента
package com.example.navgraphtest;
import android.os.Bundle;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.fragment.app.Fragment;
import androidx.navigation.NavController;
import androidx.navigation.Navigation;
import androidx.viewpager.widget.ViewPager;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import com.google.android.material.tabs.TabLayout;
import com.prolificinteractive.materialcalendarview.CalendarDay;
import com.prolificinteractive.materialcalendarview.MaterialCalendarView;
import com.prolificinteractive.materialcalendarview.OnDateSelectedListener;
import org.threeten.bp.LocalDate;
import java.util.ArrayList;
import static android.content.ContentValues.TAG;
public class CalendarViewFragment extends Fragment {
private MaterialCalendarView mCalendarView;
private CalendarDay currentDate;
private TextView mTextViewCurrentDate;
private ViewPager mViewPager;
private TabLayout mTabLayout;
ArrayList<CategoryModel> arrayList;
@Override
public void onResume() {
Log.d(TAG, "onResume: ");
super.onResume();
// Update ActionBar titles in MainActivity to reflect current fragment
// Actionbar is in main layout not the fragments.
((MainActivity) getActivity()).setActionBarTitle("Calendar View");
((MainActivity) getActivity()).getSupportActionBar().setSubtitle("View your consumption history");
// Open fragment with today's date selected on calendarview.
mCalendarView.setSelectedDate(CalendarDay.today());
// initViews for category tabs
initViews();
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
Log.d(TAG, "onCreateView: ");
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_calendar_view, container, false);
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
Log.d(TAG, "onViewCreated: ");
mTextViewCurrentDate = getActivity().findViewById(R.id.currentDateTextView);
mCalendarView = getActivity().findViewById(R.id.calendarView);
mCalendarView.setOnDateChangedListener(new OnDateSelectedListener(){
@Override
public void onDateSelected(@NonNull MaterialCalendarView widget, @NonNull CalendarDay date, boolean selected) {
mTextViewCurrentDate.setText(String.valueOf(date));
}
});
final NavController navController = Navigation.findNavController(view);
}
private void initViews(){
Log.d(TAG, "initViews: ");
mViewPager = getActivity().findViewById(R.id.viewPagerCalendarView);
mTabLayout = getActivity().findViewById(R.id.tabLayoutCalendarView);
mViewPager.setOffscreenPageLimit(5);
mViewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));
mTabLayout.addOnTabSelectedListener(new TabLayout.OnTabSelectedListener() {
@Override
public void onTabSelected(TabLayout.Tab tab) {
mViewPager.setCurrentItem(tab.getPosition());
}
@Override
public void onTabUnselected(TabLayout.Tab tab) {
}
@Override
public void onTabReselected(TabLayout.Tab tab) {
}
});
setDynamicFragmentToTabLayout();
}
private void setDynamicFragmentToTabLayout() {
DatabaseHelper databaseHelper = new DatabaseHelper(getActivity());
arrayList = new ArrayList<CategoryModel>(databaseHelper.getCategories());
Log.d(TAG, "setDynamicFragmentToTabLayout: ");
for (int i = 0; i < arrayList.size() ; i ) {
mTabLayout.addTab(mTabLayout.newTab().setText(arrayList.get(i).getCategoryTitle()));
}
DynamicFragmentAdapter mDynamicFragmentAdapter = new DynamicFragmentAdapter(getActivity().getSupportFragmentManager(), mTabLayout.getTabCount());
mViewPager.setAdapter(mDynamicFragmentAdapter);
mViewPager.setCurrentItem(0);
}
}
Я действительно не понимаю, как я мог бы это реализовать, за исключением отказа от возможности для пользователя создавать категории и просто иметь фиксированный номер, но тогда я не знаю, как я получу элементы для отображения в tablayout / viewpager.
Ответ №1:
Ну, ваш dynamicFragmentAdapter должен быть таким
public class DynamicFragmentAdapter extends FragmentStatePagerAdapter {
private int mNumOfTabs;
Fragment fragment = null;
ViewPager viewPager;
ArrayList<Menu> fragDataList=new ArrayList<>();
private List<String> tabNameList;
Context context;
public DynamicFragmentAdapter (Context context,FragmentManager fm, int mNumOfTabs, List<String> tabNameList, ArrayList<Menu> fragDataList,ViewPager viewPager) {
super(fm);
this.mNumOfTabs = mNumOfTabs;
this.tabNameList = tabNameList;
this.fragDataList=fragDataList;
this.context=context;
this.viewPager=viewPager;
}
@Override
public Fragment getItem(final int position) {
for (int i = 0; i < mNumOfTabs ; i ) {
fragment = GenericFragment.newInstance( fragDataList,position);
Log.i("info",String.valueOf(position));
break;
}
return fragment;
}
@Override
public int getCount() {
return mNumOfTabs;
}
@Override
public CharSequence getPageTitle(int position) {
return tabNameList.get(position);
// return super.getPageTitle(position);
}
}
и ваш основной фрагмент, через который вы можете получать данные из базы данных, viewpager и tablayout будут выглядеть следующим образом
//this method is used to set number of tabs dynamically
private void sendData(int /*sizeof tabNames list*/ size, final List<String> tabName, final ArrayList<Menu> fragArrayList, List<Menu> positioncheck) {
Log.d("My Msg", "Tab countert check = " size);
viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
int pos = 0;
@Override
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
Bundle bundle = new Bundle();
bundle.putInt("position", position);
}
@Override
public void onPageSelected(int position) {
pos = position;
}
@Override
public void onPageScrollStateChanged(int state) {
}
});
final DynamicFragmentAdapter dynamicFragmentAdapter = new DynamicFragmentAdapter (getApplicationContext(), getSupportFragmentManager(), size, tabName, fragArrayList, viewPager);
viewPager.setAdapter(dynamicFragmentAdapter );
//viewPager.setOffscreenPageLimit(5);
// viewPager.setCurrentItem(tabLayout.getSelectedTabPosition());
viewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(tabLayout));
tabLayout.addOnTabSelectedListener(new TabLayout.BaseOnTabSelectedListener() {
@Override
public void onTabSelected(TabLayout.Tab tab) {
viewPager.setCurrentItem(tab.getPosition());
}
@Override
public void onTabUnselected(TabLayout.Tab tab) {
}
@Override
public void onTabReselected(TabLayout.Tab tab) {
}
});
tabLayout.setupWithViewPager(viewPager);
}
и ваш dynamicFragment получит данные
`public class DynamicFragment extends Fragment {
static List<fragDAta> fragDataList;
static int loopsize = 0;
int mnum = 0;
public static DynamicFragment newInstance(ArrayList<fragData> fragDataList, int position) {
DynamicFragment.fragDataList = fragDataList;
DynamicFragment f = new DynamicFragment ();
// Supply num input as an argument.
Bundle args = new Bundle();
args.putInt("position", position);
f.setArguments(args);
DynamicFragment.loopsize = loopsize;
return f;
}
public DynamicFragment () {
// Required empty public constructor
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mnum = getArguments() != null ? getArguments().getInt("position") : 1;
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_dynamic, container, false);
}
@Override
public void setUserVisibleHint(boolean isVisibleToUser) {
super.setUserVisibleHint(isVisibleToUser);
}
@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
// here is your data
Log.d("MYMSG", "DATA CHECK = " fragDataList.size());
}
@Override
public void onResume() {
Log.d("MYMSG", "DATA CHECK = " fragDataList.size());
super.onResume(); }
}`
Комментарии:
1. Привет, очень ценю ваш ответ. Я пытался реализовать то, что вы предложили, но это не работает. Фрагменты не создаются, и никакие данные не отображаются, только табличное описание с заголовками.
2. можете ли вы поделиться со мной кодом, как вы его реализовали, чтобы я мог видеть и более точно помогать вам
3. извините за поздний ответ @Ifoste17 вы получаете результат. если нет, я помогу вам