android viewpager 滑動菜單欄Tab導(dǎo)航

在許多項目中逮壁,我們都可以看到可以滑動的tab導(dǎo)航欄颓遏,最常見的比如新聞客戶端徐矩,

剛好最近項目中又有需要用到的地方,之前寫過叁幢,但是不久之后就忘記了滤灯,所以記錄下來方便下次查看。

實現(xiàn)滾動的tab導(dǎo)航欄,主要考慮的就是這幾點

1可以滑動頂部tab
2當(dāng)切換viewpager的時候鳞骤,tab會隨之變化
3一般的tab下面會有橫線跟隨變化

簡單的來想就這么幾點需要實現(xiàn)的窒百,所以我們誰用horizontalscrollview來實現(xiàn)這個tab.

首先自定義一個horizontalscrollview,來完成我們的內(nèi)部邏輯豫尽,由于其內(nèi)部是橫向的textview,都會有選中的高亮效果篙梢,所以我們可以在horizontalscrollview里面包含一個RadioGroup,然后設(shè)置選中效果美旧,這樣就可以比較方便的實現(xiàn)切換效果了渤滞,先添加一個布局文件:

<?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"
android:orientation="vertical">


<RadioGroup
    android:id="@+id/rg_all_title"
    android:layout_width="wrap_content"
    android:layout_height="48dp"
    android:orientation="horizontal">

</RadioGroup>

<ImageView
    android:id="@+id/iv_leng"
    android:layout_width="45dp"
    android:layout_height="1dp"
    android:layout_gravity="bottom"
    android:background="@color/colorPrimary" />

</LinearLayout>

這里我們只聲明了一個RadioGroup,和一個imageview,在代碼中,我們會根據(jù)導(dǎo)航欄的長度來手動new RadioButton,imageview代表的是下面的指示線陈症。

接下來我們繼承一個horizontalscrollview蔼水,來寫我們具體的方法,首頁加載上面寫好的布局

        mView = inflate(getContext(), R.layout.item_title_scrollview, this);
        rg_all_title = mView.findViewById(R.id.rg_all_title);
        iv_leng = mView.findViewById(R.id.iv_leng);

這樣一來录肯,基本的布局就完成了趴腋,只需要聲明一個導(dǎo)航的集合來創(chuàng)建就可以了。
當(dāng)我們手動添加RadioButton的時候论咏,需要給每一個RadioButton設(shè)置一個Tag或者Id,這樣方便查找到我們添加的RadioButton,我這里給添加一個默認(rèn)的Id,然后往后面疊加


    private int mDefaultId = 56845;//默認(rèn)Id,一直往后疊加
    private List<String> mDatas;
    public void setDatas(List<String> datas) {
        this.mDatas = datas;
        if(mDatas!=null){
             addTitle();
        }
    }

    public void addTitle() {
        //添加之前先清理掉之前的所有title
        rg_all_title.removeAllViews();
        for (int i = 0; i < mDatas.size(); i++) {
            RadioButton radioButton = new RadioButton(getContext());
            //這一塊可以根據(jù)項目的具體需求优炬,來添加圖片或者字體邊距等等
            radioButton.setBackground(null);
            radioButton.setButtonDrawable(null);
            radioButton.setText(mDatas.get(i));
            //默認(rèn)顏色,和選中時候的顏色
            radioButton.setTextColor(getResources().getColorStateList(R.color.select_home_text));
            radioButton.setTextSize(13);
            radioButton.setPadding(40, 0, 40, 0);
            //設(shè)置每個id,這里加i代表減去默認(rèn)的之后對應(yīng)的就是viewpager的頁面
            radioButton.setId(mDefaultId + i);
            rg_all_title.addView(radioButton);
            if (i == 0) {
                //第一次添加的時候厅贪,默認(rèn)第一個為選中的
                radioButton.setChecked(true);
            }
        }
    }

到了這里蠢护,我們的第一步已經(jīng)完成,接下來看看第二步养涮,當(dāng)viewpager切換的時候葵硕,Tab也隨之變化。
大家都知道贯吓,viewpager可以監(jiān)聽pager的切換變化懈凹,然后當(dāng)監(jiān)聽到pager切換之后,找到對應(yīng)的RadioButton設(shè)置為Cheack狀態(tài)就可以了

   private ViewPager viewPager;
//在我們的具體實現(xiàn)頁面中悄谐,把viewpager傳遞過來介评,更好的操作
   public void setViewPager(ViewPager viewPager) {
        this.viewPager = viewPager;
        initViewPager();
    }

    private void initViewPager() {
        //手動添加一個pager改變的監(jiān)聽
        viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                
            }

            @Override
            public void onPageSelected(int position) {
                onMoveRadio(position);
            }

            @Override
            public void onPageScrollStateChanged(int state) {

            }
        });
    }

    public void onMoveRadio(int id) {
        RadioButton button = findViewById(id + mDefaultId);
        button.setChecked(true);
    }

   @Override
    public void onCheckedChanged(RadioGroup group, int checkedId) {
        //當(dāng)點擊RadioButton的時候,也切換到對應(yīng)的viewpager頁面
        if (viewPager != null) {
            viewPager.setCurrentItem(checkedId - mDefaultId);
        }
    }

這樣一來爬舰,第二條也可以實現(xiàn)了们陆,但是當(dāng)viewpager滑動的后面之后,tab不會隨之往后面移動情屹,這樣肯定不能滿足使用需求坪仇,結(jié)合第三條,tab下面的線條跟隨移動垃你,總結(jié)下來就是需要監(jiān)聽onPageScrolled,當(dāng)viewpager滑動時椅文,我們跟隨滑動即可颈墅。

  viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                onMoveLeng(position, positionOffset);
            }

            @Override
            public void onPageSelected(int position) {
                onMoveRadio(position);
            }

            @Override
            public void onPageScrollStateChanged(int state) {

            }
        });

    public void onMoveLeng(int id, float positionOffset) {
        RadioButton button = findViewById(id + mDefaultId);
        //選中tab距離左邊的距離,來計算移動的
        int left = button.getLeft();
        int width = button.getWidth() / 2;
        //橫線的移動距離
        int move = (int) (left + (width - iv_leng.getWidth() / 2) + (positionOffset * width * 2));
        //ScreenUtils.getScreenWidth(getContext())獲取屏幕的寬度雾袱,當(dāng)移動到屏幕右邊的tab時候,scrollview向左邊移動
        int moveX = (int) (left - ScreenUtils.getScreenWidth(getContext()) / 2 + width + (positionOffset * width * 2));
        smoothScrollTo(moveX, 0);//scrollview移動
        iv_leng.setTranslationX(move);// 橫線的移動
    }

    public void onMoveRadio(int id) {
        RadioButton button = findViewById(id + mDefaultId);
        button.setChecked(true);
    }

這樣一來我們的代碼就基本全部實現(xiàn)了官还,在具體需要實現(xiàn)的地方芹橡,需要實現(xiàn)其setDatas()方法,以及setViewPager()就可以實現(xiàn)想要的效果了望伦。

完整代碼如下:

/**
 * Created by hy on 2018/9/11.
 */
public class MenuScrollTitle extends HorizontalScrollView implements RadioGroup.OnCheckedChangeListener {
    private View mView;
    private RadioGroup rg_all_title;
    private ImageView iv_leng;
    private List<String> mDatas;
    private int mDefaultId = 56845;
    private ViewPager viewPager;

    public MenuScrollTitle(Context context) {
        this(context, null);
    }

    public MenuScrollTitle(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public MenuScrollTitle(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initView();
    }

    private void initView() {
        mView = inflate(getContext(), R.layout.item_title_scrollview, this);
        rg_all_title = mView.findViewById(R.id.rg_all_title);
        iv_leng = mView.findViewById(R.id.iv_leng);
        rg_all_title.setOnCheckedChangeListener(this);
    }

    public void setViewPager(ViewPager viewPager) {
        this.viewPager = viewPager;
        initViewPager();
    }

    private void initViewPager() {
        viewPager.addOnPageChangeListener(new ViewPager.OnPageChangeListener() {
            @Override
            public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
                onMoveLeng(position, positionOffset);
            }

            @Override
            public void onPageSelected(int position) {
                onMoveRadio(position);
            }

            @Override
            public void onPageScrollStateChanged(int state) {

            }
        });
    }

    public void onMoveLeng(int id, float positionOffset) {
        RadioButton button = findViewById(id + mDefaultId);
        int left = button.getLeft();
        int width = button.getWidth() / 2;
        int lengmove = (int) (left + (width - iv_leng.getWidth() / 2) + (positionOffset * width * 2));
        final int moveX = (int) (left - ScreenUtils.getScreenWidth(getContext()) / 2 + width + (positionOffset * width * 2));
        smoothScrollTo(moveX, 0);
        iv_leng.setTranslationX(lengmove);
    }

    public void onMoveRadio(int id) {
        RadioButton button = findViewById(id + mDefaultId);
        button.setChecked(true);
    }

    public void setDatas(List<String> datas) {
        this.mDatas = datas;
        addTitle();
    }

    public void addTitle() {
        rg_all_title.removeAllViews();
        for (int i = 0; i < mDatas.size(); i++) {
            RadioButton radioButton = new RadioButton(getContext());
            radioButton.setBackground(null);
            radioButton.setButtonDrawable(null);
            radioButton.setText(mDatas.get(i));
            radioButton.setTextColor(getResources().getColorStateList(R.color.select_home_text));
            radioButton.setTextSize(13);
            radioButton.setPadding(40, 0, 40, 0);
            radioButton.setId(mDefaultId + i);
            rg_all_title.addView(radioButton);
            if (i == 0) {
                radioButton.setChecked(true);
            }
        }

    }

    @Override
    public void onCheckedChanged(RadioGroup group, int checkedId) {
        if (viewPager != null) {
            viewPager.setCurrentItem(checkedId - mDefaultId);
        }
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末林说,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子屯伞,更是在濱河造成了極大的恐慌腿箩,老刑警劉巖,帶你破解...
    沈念sama閱讀 210,914評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件劣摇,死亡現(xiàn)場離奇詭異珠移,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)末融,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評論 2 383
  • 文/潘曉璐 我一進(jìn)店門钧惧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人勾习,你說我怎么就攤上這事浓瞪。” “怎么了巧婶?”我有些...
    開封第一講書人閱讀 156,531評論 0 345
  • 文/不壞的土叔 我叫張陵乾颁,是天一觀的道長。 經(jīng)常有香客問我艺栈,道長英岭,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,309評論 1 282
  • 正文 為了忘掉前任眼滤,我火速辦了婚禮巴席,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘诅需。我一直安慰自己漾唉,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 65,381評論 5 384
  • 文/花漫 我一把揭開白布堰塌。 她就那樣靜靜地躺著赵刑,像睡著了一般。 火紅的嫁衣襯著肌膚如雪场刑。 梳的紋絲不亂的頭發(fā)上般此,一...
    開封第一講書人閱讀 49,730評論 1 289
  • 那天蚪战,我揣著相機(jī)與錄音,去河邊找鬼铐懊。 笑死邀桑,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的科乎。 我是一名探鬼主播壁畸,決...
    沈念sama閱讀 38,882評論 3 404
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼茅茂!你這毒婦竟也來了捏萍?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,643評論 0 266
  • 序言:老撾萬榮一對情侶失蹤空闲,失蹤者是張志新(化名)和其女友劉穎令杈,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體碴倾,經(jīng)...
    沈念sama閱讀 44,095評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡逗噩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,448評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了影斑。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片给赞。...
    茶點故事閱讀 38,566評論 1 339
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖矫户,靈堂內(nèi)的尸體忽然破棺而出片迅,到底是詐尸還是另有隱情,我是刑警寧澤皆辽,帶...
    沈念sama閱讀 34,253評論 4 328
  • 正文 年R本政府宣布柑蛇,位于F島的核電站,受9級特大地震影響驱闷,放射性物質(zhì)發(fā)生泄漏耻台。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,829評論 3 312
  • 文/蒙蒙 一空另、第九天 我趴在偏房一處隱蔽的房頂上張望盆耽。 院中可真熱鬧,春花似錦扼菠、人聲如沸摄杂。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,715評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽析恢。三九已至,卻和暖如春秧饮,著一層夾襖步出監(jiān)牢的瞬間映挂,已是汗流浹背泽篮。 一陣腳步聲響...
    開封第一講書人閱讀 31,945評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留柑船,地道東北人帽撑。 一個月前我還...
    沈念sama閱讀 46,248評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像鞍时,于是被迫代替她去往敵國和親油狂。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,440評論 2 348

推薦閱讀更多精彩內(nèi)容