[英]Fragments not being displayed correctly on android
我正在 Android 中创建预算应用程序。 目前,我正在尝试使用fragments
来显示所有用户当前的钱包(每个钱包一个片段)。 每个fragment
将占据整个屏幕并使用屏幕顶部的TabLayout
进行导航。 由于我的fragments
显示的数据会经常更改,因此它们需要是动态的。 现在的问题是它们正在被创建(我可以使用断点等在FragmentManager
看到它们),但是,它们没有显示在ViewPager
或TabLayout
。
我的主要活动的 XML 代码:
<androidx.constraintlayout.widget.ConstraintLayout 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"
tools:context=".MainActivity">
<androidx.viewpager.widget.ViewPager
android:id="@+id/pager"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<com.google.android.material.tabs.TabLayout
android:id="@+id/tabLayout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:elevation="4dp"
android:minHeight="?attr/actionBarSize"
android:background="@color/colorTest"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent"
app:tabGravity="fill"
app:tabMode="fixed" />
<Button
android:id="@+id/createWallet"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Create wallet"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
<Button
android:id="@+id/currentFragment"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="Current Fragment"
app:layout_constraintBottom_toBottomOf="@+id/createWallet"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>
我在MainActivity
创建fragments
并将ViewPager
和TabLayout
在一起的代码:
ViewPager viewPager = findViewById(R.id.pager);
ViewPagerAdapter viewPagerAdapter = new ViewPagerAdapter(getSupportFragmentManager(), MainActivity.this);
viewPager.setAdapter(viewPagerAdapter);
TabLayout tabLayout = findViewById(R.id.tabLayout);
tabLayout.setupWithViewPager(viewPager);
public void loadFragments()
{
ViewPagerAdapter viewPagerAdapter = new ViewPagerAdapter(getSupportFragmentManager(), MainActivity.this);
for (int i = 0; i < walletList.size(); i++)
{
WalletClass object = walletList.get(i);
FragmentTransaction fragmentTransaction = getSupportFragmentManager().beginTransaction();
fragment frag = new fragment().newInstance(object);
fragmentTransaction.add(R.id.pager, frag, "fragment" + i);
viewPagerAdapter.addFragment(frag); // this line can cause crashes
fragmentTransaction.commit();
}
}
我的片段类代码:
public fragment newInstance(WalletClass wallet)
{
fragment frag = new fragment();
Bundle bundle = new Bundle();
bundle.putString("walletName", wallet.getWalletName());
bundle.putInt("walletBalance", wallet.getBalance());
frag.setArguments(bundle);
return frag;
}
@Override
public void onCreate(@Nullable Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
Bundle bundle = getArguments();
Toast.makeText(getContext(), "" + bundle.getString("walletName"), Toast.LENGTH_SHORT).show();
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
{
// Inflate the layout for this fragment
View view = inflater.inflate(R.layout.fragment_layout, container, false);
Bundle bundle = getArguments();
TextView walletName = view.findViewById(R.id.walletName);
walletName.setText(bundle.getString("walletName"));
return view;
}
还有我的ViewPager
适配器代码。 这段代码主要来自其他一些在线教程,我认为这基本上是需要重新编写的:
List<fragment> fragmentList = new ArrayList<>(); // this line can cause crashes
List<String> fragmentNameList = new ArrayList<>();
private Context context;
public ViewPagerAdapter(FragmentManager fm, Context context)
{
super(fm);
this.context = context;
}
@Override
public fragment getItem(int position)
{
return fragmentList.get(position);
}
public void addFragment (Fragment fragment, WalletClass wallet) // this line can cause crashes
{
Bundle bundle = new Bundle();
bundle.putString("walletName", wallet.getWalletName());
bundle.putInt("walletBalance", wallet.getBalance());
fragment.setArguments(bundle);
fragmentList.add(fragment);
fragmentNameList.add(wallet.getWalletName());
}
public void removeFragment (fragment fragment)
{
fragmentList.remove(fragment);
}
@Override
public int getCount()
{
return fragmentList.size();
}
@Nullable
public CharSequence getPageTitle(int position)
{
return fragmentNameList.get(position);
}
那么有人对如何使用ViewPager
适配器动态创建fragments
有任何想法吗? 或者我这样做是完全错误的方式吗?
FragmentPagerAdapter 已经管理 Fragment 的生命周期,您不需要手动添加它们。 但是, FragmentPagerAdapter 将 Fragment 的位置编码到其片段标记中,因此不能安全地用于处理动态寻呼适配器。
您需要根据 FragmentPagerAdapter 的代码使用以下代码或等效代码(最终可能会成为以下代码)。
public class DynamicFragmentPagerAdapter extends PagerAdapter {
private static final String TAG = "DynamicFragmentPagerAdapter";
private final FragmentManager fragmentManager;
public static abstract class FragmentIdentifier implements Parcelable {
private final String fragmentTag;
private final Bundle args;
public FragmentIdentifier(@NonNull String fragmentTag, @Nullable Bundle args) {
this.fragmentTag = fragmentTag;
this.args = args;
}
protected FragmentIdentifier(Parcel in) {
fragmentTag = in.readString();
args = in.readBundle(getClass().getClassLoader());
}
@Override
public void writeToParcel(Parcel dest, int flags) {
dest.writeString(fragmentTag);
dest.writeBundle(args);
}
protected final Fragment newFragment() {
Fragment fragment = createFragment();
Bundle oldArgs = fragment.getArguments();
Bundle newArgs = new Bundle();
if(oldArgs != null) {
newArgs.putAll(oldArgs);
}
if(args != null) {
newArgs.putAll(args);
}
fragment.setArguments(newArgs);
return fragment;
}
protected abstract Fragment createFragment();
}
private ArrayList<FragmentIdentifier> fragmentIdentifiers = new ArrayList<>();
private FragmentTransaction currentTransaction = null;
private Fragment currentPrimaryItem = null;
public DynamicFragmentPagerAdapter(FragmentManager fragmentManager) {
this.fragmentManager = fragmentManager;
}
private int findIndexIfAdded(FragmentIdentifier fragmentIdentifier) {
for (int i = 0, size = fragmentIdentifiers.size(); i < size; i++) {
FragmentIdentifier identifier = fragmentIdentifiers.get(i);
if (identifier.fragmentTag.equals(fragmentIdentifier.fragmentTag)) {
return i;
}
}
return -1;
}
public void addFragment(FragmentIdentifier fragmentIdentifier) {
if (findIndexIfAdded(fragmentIdentifier) < 0) {
fragmentIdentifiers.add(fragmentIdentifier);
notifyDataSetChanged();
}
}
public void removeFragment(FragmentIdentifier fragmentIdentifier) {
int index = findIndexIfAdded(fragmentIdentifier);
if (index >= 0) {
fragmentIdentifiers.remove(index);
notifyDataSetChanged();
}
}
@Override
public int getCount() {
return fragmentIdentifiers.size();
}
@Override
public void startUpdate(@NonNull ViewGroup container) {
if (container.getId() == View.NO_ID) {
throw new IllegalStateException("ViewPager with adapter " + this
+ " requires a view id");
}
}
@SuppressWarnings("ReferenceEquality")
@NonNull
@Override
public Object instantiateItem(@NonNull ViewGroup container, int position) {
if (currentTransaction == null) {
currentTransaction = fragmentManager.beginTransaction();
}
final FragmentIdentifier fragmentIdentifier = fragmentIdentifiers.get(position);
// Do we already have this fragment?
final String name = fragmentIdentifier.fragmentTag;
Fragment fragment = fragmentManager.findFragmentByTag(name);
if (fragment != null) {
currentTransaction.attach(fragment);
} else {
fragment = fragmentIdentifier.newFragment();
currentTransaction.add(container.getId(), fragment, fragmentIdentifier.fragmentTag);
}
if (fragment != currentPrimaryItem) {
fragment.setMenuVisibility(false);
fragment.setUserVisibleHint(false);
}
return fragment;
}
@Override
public void destroyItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
if (currentTransaction == null) {
currentTransaction = fragmentManager.beginTransaction();
}
currentTransaction.detach((Fragment) object);
}
@SuppressWarnings("ReferenceEquality")
@Override
public void setPrimaryItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
Fragment fragment = (Fragment) object;
if (fragment != currentPrimaryItem) {
if (currentPrimaryItem != null) {
currentPrimaryItem.setMenuVisibility(false);
currentPrimaryItem.setUserVisibleHint(false);
}
fragment.setMenuVisibility(true);
fragment.setUserVisibleHint(true);
currentPrimaryItem = fragment;
}
}
@Override
public void finishUpdate(@NonNull ViewGroup container) {
if (currentTransaction != null) {
currentTransaction.commitNowAllowingStateLoss();
currentTransaction = null;
}
}
@Override
public boolean isViewFromObject(@NonNull View view, @NonNull Object object) {
return ((Fragment) object).getView() == view;
}
@Override
public Parcelable saveState() {
Bundle bundle = new Bundle();
bundle.putParcelableArrayList("fragmentIdentifiers", fragmentIdentifiers);
return bundle;
}
@Override
public void restoreState(Parcelable state, ClassLoader loader) {
Bundle bundle = ((Bundle)state);
bundle.setClassLoader(loader);
fragmentIdentifiers = bundle.getParcelableArrayList("fragmentIdentifiers");
}
}
然后你可以将FragmentIdentifier
s 传入这个寻呼机,它会根据你的片段标识符正确管理你的片段。
List<fragment> fragmentList = new ArrayList<>();
这些教程试图让您的应用程序崩溃,请检查https://proandroiddev.com/the-seven-actually-10-cardinal-sins-of-android-development-491d2f64c8e0 中的#3
我认为您的fragment
没有正确附加到您的ViewPager
。
首先,在您的loadFragment()
您的新添加的 Fragment 尚未在您的viewPager
上设置。 在将新片段添加到适配器后,您必须再次添加setAdapter()
。 正如官方文档中提到的(它来自 ListAdapter 类,但基本相同):
为了显示列表中的项目,调用 setAdapter(android.widget.ListAdapter) 将适配器与列表关联。
其次,同样在您的loadFragment()
,您不必在方法内创建新的ViewPagerAdapter
实例。 使用您早期实例化的 viewPagerAdapter。 只要您在onCreate()
方法之外的开头声明 ViewPager。
第三,您可能希望在loadFragment()
删除您的FragmentTransaction
。 我不知道是不是意味着。 但是,只要你有适配器,你的新片段就可以自动附加而无需调用FragmentTransaction
这是MainActivity
样子:
// declare ViewPager viewPager in the beginning of class.
viewPager = findViewById(R.id.pager);
viewPagerAdapter = new ViewPagerAdapter(getSupportFragmentManager(), MainActivity.this);
viewPager.setAdapter(viewPagerAdapter);
TabLayout tabLayout = findViewById(R.id.tabLayout);
tabLayout.setupWithViewPager(viewPager);
public void loadFragments()
{
for (int i = 0; i < walletList.size(); i++)
{
WalletClass object = walletList.get(i);
fragment frag = new fragment().newInstance(object);
viewPagerAdapter.addFragment(frag, object);
viewPager.setAdapter(viewPagerAdapter)
}
}
注意:当我运行您的代码时,我认为问题不在于您的适配器,而在于您调用适配器的方式
希望能帮助到你。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.