【问题标题】:Icons in TabLayoutTabLayout 中的图标
【发布时间】:2023-03-07 08:58:01
【问题描述】:

我有一个简单的AppCompatActivity,一个简单的Toolbar (android.support.v7.widget.Toolbar),一个设计库TabLayout 和一个ViewPager

现在,在这种情况下,我有 4 个片段,它们位于 viewpager 内,可以从侧面滑动。这很好用。 但是一旦我使用mTabLayout.setupWithViewPager(mViewPager);,图标就会消失,但我可以在空白处使用标签,效果很好。如果我删除该行,图标再次可见,“滑动”有效,但如果我选择一个图标,什么也不会发生。

我的代码

MainActivity

public class MainActivity extends AppCompatActivity {

    private Toolbar mToolbar;
    private TabLayout mTabLayout;
    private ViewPager mViewPager;
    private ViewPagerAdapter mAdapter;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initToolbar();
        initTabLayout();
        initViewPager();

        mTabLayout.setupWithViewPager(mViewPager);//Here is the code from above
        mViewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(mTabLayout));

    }

    private void initToolbar() {
        mToolbar = (Toolbar) findViewById(R.id.app_bar);
        setSupportActionBar(mToolbar);
        getSupportActionBar().setDisplayHomeAsUpEnabled(false);

    }

    private void initTabLayout() {
        mTabLayout = (TabLayout) findViewById(R.id.tabLayout);
        mTabLayout.addTab(mTabLayout.newTab().setIcon(R.drawable.ic_fire_white));
        mTabLayout.addTab(mTabLayout.newTab().setIcon(R.drawable.ic_apps_white));
        mTabLayout.addTab(mTabLayout.newTab().setIcon(R.drawable.ic_account_plus_white));
        mTabLayout.addTab(mTabLayout.newTab().setIcon(R.drawable.ic_help_white));

    }

    private void initViewPager() {
        mAdapter = new ViewPagerAdapter(getSupportFragmentManager(), MainActivity.this);
        mViewPager = (ViewPager) findViewById(R.id.viewPager);
        mViewPager.setAdapter(mAdapter);

    }

    //MENUE STUFF

}

//ViewPagerAdapter
class ViewPagerAdapter extends FragmentStatePagerAdapter {

    public static final int TABS_COUNT = 4;
    private Context context;

    public ViewPagerAdapter(FragmentManager fm, Context context) {
        super(fm);
        this.context = context;
    }

    @Override
    public Fragment getItem(int position) {
        switch (position) {
            case 0: return new HomeFragment1();
            case 1: return new HomeFragment2();
            case 2: return new HomeFragment3();
            case 3: return new HomeFragment4();
        }
        return null;
    }

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

}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:orientation="vertical">

    <include
        android:id="@+id/app_bar"
        layout="@layout/app_bar" />

    <android.support.design.widget.TabLayout
        android:id="@+id/tabLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@color/primary"
        app:tabGravity="fill"
        app:tabMode="fixed"
        app:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar" />

    <android.support.v4.view.ViewPager
        android:id="@+id/viewPager"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

</LinearLayout>

【问题讨论】:

    标签: android android-tabs android-design-library


    【解决方案1】:

    setupWithViewPager() 实例化 Tab 仅具有从 ViewPager.Adapter 提供的标题。当然你可以把这个方法重写成这样:

    public class IconicTabLayout extends TabLayout {
    
        private TabViewProvider mTabViewProvider;
    
        public interface TabViewProvider {
            Tab newTabInstance(int tabPosition);
        }
    
        public IconicTabLayout(Context context) {
            super(context);
        }
    
        public IconicTabLayout(Context context, AttributeSet attrs) {
            super(context, attrs);
        }
    
        public IconicTabLayout(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
        }
    
        public void setTabProvider(TabViewProvider provider){
            mTabViewProvider = provider;
        }
    
        @Override
        public void setupWithViewPager(ViewPager viewPager) {
            PagerAdapter adapter = viewPager.getAdapter();
            if (adapter == null) {
                throw new IllegalArgumentException("ViewPager does not have a PagerAdapter set");
            } else {
                this.setTabsFromPagerAdapter(adapter);
                viewPager.addOnPageChangeListener(new TabLayout.TabLayoutOnPageChangeListener(this));
                this.setOnTabSelectedListener(new TabLayout.ViewPagerOnTabSelectedListener(viewPager));
            }
    
        }
    
        @Override
        public void setTabsFromPagerAdapter(PagerAdapter adapter) {
            removeAllTabs();
            mTabViewProvider = mTabViewProvider == null && adapter instanceof TabViewProvider? (TabViewProvider)adapter : null;
      if(mTabViewProvider == null){   
            super.setTabsFromPagerAdapter(adapter);
        }else{
            for (int i = 0, count = adapter.getCount(); i < count; ++i) {
                this.addTab(mTabViewProvider.newTabInstance(i));
            }
    
        }
       }
    }
    

    用法:

     mIconicTabLayout.setTabProvider(new TabViewProvider() {
            @Override
            public Tab newTabInstance(int tabPosition) {
                return mIconicTabLayout.newTab().setIcon(icons[position]);
            }
        });
        mIconicTabLayout.setupWithViewPager(mViewPager);
    

    或者

    class ViewPagerAdapter extends FragmentStatePagerAdapter implements TabViewProvider{
    
       public void setTabLayout(TabLayout t){
          mTabLayout = t;
       }
      //...
    
        @Override
        public Tab newTabInstance(int position) {
                    return mTabLayout.newTab().setIcon(icons[position]);
                }
    
    }
    

    注意: 确保您首先调用setTabProvider 方法或您的PagerAdapter 实现TabViewProvider,否则它将调用默认实现。

    【讨论】:

    • 使用第一种方法,它不会在标签上为我显示图标。我已经尝试过 PNG 文件和 XML(使用选择器)。两者都没有工作。使用第二种方法,图标会不时闪烁,尤其是当我在选择器中使用 exitFadeDuration 或快速滑动时。怎么会?
    【解决方案2】:

    您应该在 PagerAdapter 中启动并添加图标。

    private int[] imageResId = {
        R.drawable.ic_one,
        R.drawable.ic_two,
        R.drawable.ic_three
    };
    
    // ...
    
    @Override
    public CharSequence getPageTitle(int position) {
       // Generate title based on item position
     // return tabTitles[position];
    Drawable image = context.getResources().getDrawable(imageResId[position]);
    image.setBounds(0, 0, image.getIntrinsicWidth(), image.getIntrinsicHeight());
    SpannableString sb = new SpannableString(" ");
    ImageSpan imageSpan = new ImageSpan(image, ImageSpan.ALIGN_BOTTOM);
    sb.setSpan(imageSpan, 0, 1, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
    return sb;
    }
    

    Source

    【讨论】:

    • 虽然我不一定将其称为正确答案,但我对它的独创性给予了心理 +1。更好的方法是标签上的 setIcon() 。
    • 我不知道。有一些您可以展示的示例代码吗? @androiddeveloper
    • 看不到 Chris Banes 添加了文本和图标。 ;)
    • @androiddeveloper 我遇到了和你一样的问题,并且drawables没有出现在标题中。 lagos 的答案中只有一个缺失的部分,问题是标签标签的样式将属性 textAllCaps 设置为 true这将阻止 ImageSpan 被渲染!因此,您只需要为您的 TabLayout 设置一个自定义样式,其中 textAllCaps 设置为 false。您可以在此处找到更多信息和所需样式:link
    • 这显示了图标,但它们失去了选择器功能(点击时它们没有改变颜色)
    【解决方案3】:

    另一种方法是不调用setupWithViewPager()。相反,您可以按住Tabs 中的List,并使用PagerListener 选择正确的Tab

    类似这样的:

    private List<TabLayout.Tab> tabList;
    
    private void initTabLayout() {
        tabList = new ArrayList<>();
        tabList.add(mTabLayout.newTab().setIcon(R.drawable.ic_fire_white));
        tabList.add(mTabLayout.newTab().setIcon(R.drawable.ic_apps_white));
        tabList.add(mTabLayout.newTab().setIcon(R.drawable.ic_account_plus_white));
        tabList.add(mTabLayout.newTab().setIcon(R.drawable.ic_help_white));
        for (TabLayout.Tab tab : tabList){
            mTabLayout.addTab(tab);
        }
        mViewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
    
            }
    
            @Override
            public void onPageSelected(int position) {
                tabList.get(position).select();
            }
    
            @Override
            public void onPageScrollStateChanged(int state) {
    
            }
        });
    
    }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2023-03-16
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-12-24
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多