使用 Fragment 内的方法更改 TextView 值(NullPointerException)

Change TextView values with method inside Fragment (NullPointerException)

我正在尝试制作一个应用程序(用于学习目的),其中有几个包含可滚动内容的选项卡。 我有一个折叠的应用程序栏,主要布局有一个 ViewPager 和三个相同的布局,在 NestedScrollView 中有一个 TextView,唯一的区别是文本顶部的图像。我还有三个片段 classes,里面有相同的代码。一切正常。

但由于我在每个选项卡中使用相同的布局,我认为我可以只为每个选项卡使用 xml 文件和 java class,而不是每个选项卡中的一个,然后以某种方式更改我的适配器中每个选项卡的内容。

所以我试图在我的 tabFragment class 中创建一个方法来执行此操作。但我似乎找不到修改 TextView 的方法。我每次都收到 NullPointerException。

我错过了什么? 我的方法错了吗?如果是这样,什么会更好?我该如何解决?

提前谢谢你:)


这是我现在的片段 class(仅剩一个):

import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import androidx.fragment.app.Fragment;
import com.example.tablayoutwithscrollview.R;

public class TabFragment extends Fragment {

    private TextView textView;
    private int text, image;

    public TabFragment(){}

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
    {
        View view = inflater.inflate(R.layout.tab1, container, false);
        //textView = view.findViewById(R.id.textView);
        return view;
    }

    @Override
    public void onViewCreated(View view, Bundle savedInstanceState)
    {
        super.onViewCreated(view, savedInstanceState);
        textView = getView().findViewById(R.id.textView);
    }

    public void setTabContent(int text, int image)
    {
        textView.setText(text);
        textView.setCompoundDrawablesWithIntrinsicBounds(0,image,0,0);
    }
}

我的适配器class:

import android.content.Context;
import androidx.fragment.app.Fragment;
import androidx.fragment.app.FragmentManager;
import androidx.fragment.app.FragmentPagerAdapter;

import com.example.tablayoutwithscrollview.Tabs.TabFragment;

public class MyAdapter extends FragmentPagerAdapter {

    private static int tabCount;
    Context context;

    public MyAdapter(FragmentManager fm, int tabCount, Context context){
        super(fm);
        this.tabCount = tabCount;
        this.context = context;
    }

    @Override
    public Fragment getItem(int position) {

        TabFragment fragment;
        switch (position)
        {
            case 0:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.jackfruit);
                return fragment;
            case 1:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.guava);
                return fragment;
            case 2:
                fragment = new TabFragment();
                fragment.setTabContent(R.string.lorem_ipsum, R.drawable.pomegranate);
                return fragment;
            default:
                return null;
        }
    }

    @Override
    public CharSequence getPageTitle(int position)
    {
        switch (position)
        {
            case 0:
                return context.getString(R.string.tab1_title);
            case 1:
                return context.getString(R.string.tab2_title);
            case 2:
                return context.getString(R.string.tab3_title);
            default:
                return null;
        }
    }

    @Override
    public int getCount() {
        return tabCount;
    }
}

主要活动:

package com.example.tablayoutwithscrollview;

import androidx.appcompat.app.AppCompatActivity;
import androidx.core.widget.NestedScrollView;
import androidx.viewpager.widget.ViewPager;

import android.os.Bundle;

import com.google.android.material.tabs.TabLayout;

public class MainActivity extends AppCompatActivity {

    private TabLayout tabLayout;
    private ViewPager viewPager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tabLayout = findViewById(R.id.tabLayout);
        viewPager = findViewById(R.id.viewPager);
        populatesTabLayout(tabLayout);
        MyAdapter myAdapter = new MyAdapter(getSupportFragmentManager(), tabLayout.getTabCount(), this);
        viewPager.setAdapter(myAdapter);

        tabLayout.setupWithViewPager(viewPager);
    }

    private void populatesTabLayout(TabLayout tabLayout)
    {
        for (int i = 0; i<3; i++)
        {
            tabLayout.addTab(tabLayout.newTab());
        }
    }
}

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout 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:id="@+id/coordLayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <com.google.android.material.appbar.AppBarLayout
        android:id="@+id/appBarLayout"

        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <com.google.android.material.appbar.CollapsingToolbarLayout
            android:id="@+id/collapseToolBar"
            android:layout_width="match_parent"
            android:layout_height="match_parent"

            android:fitsSystemWindows="true"
            app:contentScrim="?attr/colorPrimary"
            app:layout_scrollFlags="scroll|exitUntilCollapsed"
            app:toolbarId="@+id/toolbar">


            <ImageView
                android:id="@+id/imageView7"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:srcCompat="@drawable/strawberry" />

            <androidx.appcompat.widget.Toolbar
                android:id="@+id/toolbar"
                android:layout_width="match_parent"
                android:layout_height="?attr/actionBarSize"></androidx.appcompat.widget.Toolbar>

        </com.google.android.material.appbar.CollapsingToolbarLayout>

        <com.google.android.material.tabs.TabLayout
            android:id="@+id/tabLayout"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:visibility="visible" />

    </com.google.android.material.appbar.AppBarLayout>


    <androidx.viewpager.widget.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />


</androidx.coordinatorlayout.widget.CoordinatorLayout>

我的选项卡布局:

<?xml version="1.0" encoding="utf-8"?>
<androidx.core.widget.NestedScrollView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/tabNestedScrollView"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/textView"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:drawableTop="@drawable/jackfruit"
        android:text="@string/lorem_ipsum" />
</androidx.core.widget.NestedScrollView>

这是它应该如何工作的(从之前每个选项卡都有一个片段 class 和一个布局文件):

问题是创建的 TabFragment 实例还没有附加到 activity,因此 textView 在那个时候没有膨胀。

您应该将文本作为 MyAdapter#getItem(..) 方法中的参数传递给 TabFragment 实例。

...    
case 0:
   fragment = new TabFragment();

   Bundle args = new Bundle();
   args.putString("key", textString);
   fragment.setArguments(args);
   return fragment
...

之后在 onViewCreated(...) 方法中设置 TextView 实例的文本:

@Override
public void onViewCreated(View view, Bundle savedInstanceState) {
    super.onViewCreated(view, savedInstanceState);
    textView = getView().findViewById(R.id.textView);
    textView.setText(getArguments().getString("key"));
}

或者您可以在 Android Reference 看到一个很好的例子。