ViewPager + HorizontalScrollView 实现可滚动的标签栏

这是一个可滑动的标签栏的自定义控件,参考此文章http://blog.csdn.net/fx_sky/article/details/8990573,我将主要的功能整合成一个类,配上2个特定的布局即可使用。 
效果图:

主要布局文件:

<?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" >
        <RelativeLayout
                android:id="@+id/rl_nav"
                android:layout_width="fill_parent"
                android:layout_height="wrap_content"
                android:layout_gravity="top" >

                <RadioGroup
                    android:id="@+id/rg_nav_content"
                    android:layout_width="fill_parent"
                    android:layout_height="38dip"
                    android:layout_alignParentTop="true"
                    android:background="#F2F2F2"
                    android:orientation="horizontal" >
                </RadioGroup>

                <ImageView
                    android:id="@+id/iv_nav_indicator"
                    android:layout_width="1dip"
                    android:layout_height="5dip"
                    android:layout_alignParentBottom="true"
                    android:background="#FF0000"
                    android:contentDescription="@string/mygo_share"
                    android:scaleType="matrix" />
            </RelativeLayout>
        </LinearLayout>

标签的布局: 
sync_nav_radiogroup_item.xml

<?xml version="1.0" encoding="utf-8"?>
<RadioButton xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="0dip"
    android:layout_height="fill_parent"
    android:background="#F2F2F2"
    android:button="@null"
    android:checked="true"
    android:gravity="center"
    android:textSize="15sp" />

以下是工具类代码:

import android.app.Activity;
import android.content.Context;
import android.support.v4.view.ViewPager;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.animation.LinearInterpolator;
import android.view.animation.TranslateAnimation;
import android.widget.HorizontalScrollView;
import android.widget.ImageView;
import android.widget.RadioButton;
import android.widget.RadioGroup;
import android.widget.RadioGroup.OnCheckedChangeListener;

public class SyncHorizontalScrollView extends HorizontalScrollView {

    private View view;
    private ImageView leftImage;
    private ImageView rightImage;
    private int windowWitdh = 0;
    private Activity mContext;
    private RadioGroup rg_nav_content;
    private ImageView iv_nav_indicator;
    private LayoutInflater mInflater;
    private int count;// 屏幕显示的标签个数
    private int indicatorWidth;// 每个标签所占的宽度
    private int currentIndicatorLeft = 0;// 当前所在标签页面的位移
    private ViewPager mViewPager;
    private int scrollX;

    public SyncHorizontalScrollView(Context context) {
        super(context);
        // TODO Auto-generated constructor stub
    }

    public SyncHorizontalScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
        // TODO Auto-generated constructor stub
    }

    /**
     *
     * 方法描述:
     * @param mViewPager
     * @param leftImage 左箭頭
     * @param rightImage 右箭頭
     * @param tabTitle 標籤欄的名稱
     * @param count 一頁顯示的標籤個數
     * @param context
     * <pre>
     * 修改日期      修改人       修改说明
     * 2014-2-17   chen     新建
     * </pre>
     */
    public void setSomeParam(ViewPager mViewPager, ImageView leftImage,
            ImageView rightImage, String[] tabTitle, int count, Activity context) {
        this.mContext = context;
        this.mViewPager = mViewPager;
        // this.view = view;
        mInflater = LayoutInflater.from(context);
        this.view = mInflater.inflate(R.layout.sync_hsv_item, null);
        this.addView(view);
        this.leftImage = leftImage;
        this.rightImage = rightImage;
        DisplayMetrics dm = new DisplayMetrics();
        context.getWindowManager().getDefaultDisplay().getMetrics(dm);
        windowWitdh = dm.widthPixels;
        this.count = count;
        indicatorWidth = windowWitdh / count;
        init(tabTitle);
    }

    private void init(String[] tabTitle) {
        rg_nav_content = (RadioGroup) view.findViewById(R.id.rg_nav_content);
        iv_nav_indicator = (ImageView) view.findViewById(R.id.iv_nav_indicator);
        initIndicatorWidth();
        initNavigationHSV(tabTitle);
        setListener();
    }

    // 初始化滑动下标的宽
    private void initIndicatorWidth() {
        ViewGroup.LayoutParams cursor_Params = iv_nav_indicator
                .getLayoutParams();
        cursor_Params.width = indicatorWidth;
        iv_nav_indicator.setLayoutParams(cursor_Params);
    }

    // 添加顶部标签
    private void initNavigationHSV(String[] tabTitle) {
        rg_nav_content.removeAllViews();
        for (int i = 0; i < tabTitle.length; i++) {
            RadioButton rb = (RadioButton) mInflater.inflate(
                    R.layout.sync_nav_radiogroup_item, null);
            rb.setId(i);
            rb.setText(tabTitle[i]);
            rb.setLayoutParams(new LayoutParams(indicatorWidth,
                    LayoutParams.MATCH_PARENT));
            rg_nav_content.addView(rb);
        }
        RadioButton rb = (RadioButton) mInflater.inflate(
                R.layout.sync_nav_radiogroup_item, null);
        rg_nav_content.addView(rb);

    }

    private void setListener() {
        rg_nav_content
                .setOnCheckedChangeListener(new OnCheckedChangeListener() {
                    @Override
                    public void onCheckedChanged(RadioGroup group, int checkedId) {
                        if (rg_nav_content.getChildAt(checkedId) != null) {
                            moveAnimation(checkedId);
                            mViewPager.setCurrentItem(checkedId); // ViewPager
                            // 跟随一起 切换
                        }
                    }
                });
    }
    //动画移动效果
    private void moveAnimation(int checkedId){
        TranslateAnimation animation = new TranslateAnimation(currentIndicatorLeft,
                indicatorWidth * checkedId,0f, 0f);
        animation.setInterpolator(new LinearInterpolator());
        animation.setDuration(100);
        animation.setFillAfter(true);

        // 执行位移动画
        iv_nav_indicator.startAnimation(animation);

        // 记录当前 下标的距最左侧的 距离
        currentIndicatorLeft = indicatorWidth * checkedId;
        scrollX = (checkedId > 1 ? currentIndicatorLeft: 0)- indicatorWidth * 2;
        this.post(runnable);
    }

    // 模拟点击事件
    public void performLabelClick(int position) {
        if (rg_nav_content != null && rg_nav_content.getChildCount() > position) {
            ((RadioButton) rg_nav_content.getChildAt(position)).performClick();
        }
    }

    private Runnable runnable = new Runnable() {

        @Override
        public void run() {
            // TODO Auto-generated method stub
            smoothScrollTo(scrollX, 0);
        }
    };

    // 显示和隐藏左右两边的箭头
    protected void onScrollChanged(int l, int t, int oldl, int oldt) {
        super.onScrollChanged(l, t, oldl, oldt);
        if (!mContext.isFinishing() && view != null && rightImage != null
                && leftImage != null) {
            if (view.getWidth() <= windowWitdh) {
                leftImage.setVisibility(View.GONE);
                rightImage.setVisibility(View.GONE);
            } else {
                if (l == 0) {
                    leftImage.setVisibility(View.GONE);
                    rightImage.setVisibility(View.VISIBLE);
                } else if (view.getWidth() - l == windowWitdh) {
                    leftImage.setVisibility(View.VISIBLE);
                    rightImage.setVisibility(View.GONE);
                } else {
                    leftImage.setVisibility(View.VISIBLE);
                    rightImage.setVisibility(View.VISIBLE);
                }
            }
        }
    }
    public int getIndicatorWidth(){
        return indicatorWidth;
    }
}

在调用时,先调用setSomeParam方法将需要用的的控件与数据传入,然后控件内部开始初始化。 
由于项目有需求,要在进入此控件使用到的页面时,自动定位到某一个标签,在此需使用View.post方法执行HorizontalScrollView 控件的smoothScrollTo方法,才能确保进入页面后标签已定位。原因是scrollTo方法要等到界面显示完毕才能有效,而view.post方法也是在界面刷新完毕之后才执行。 
定义一个performLabelClick方法,让外部调用此类来实现相应的页面跳转即可。

不过此处使用时有一个问题,就是从上一级页面跳转至此页面时,无法固定在最后一个标签位,调试时发现虽然执行了performLabelClick方法模拟点击事件,但是监听事件并没有被响应。暂时没找出问题所在。不过用了一个方法来解决此问题,即在调用initNavigationHSV方法添加标签时,为其多加一个空标签,宽度为0即可,这样即可解决标签栏上无法定位到最后一位的问题,因为真正的最后一位实际上是宽度为0的空标签。 
如果大家有遇到此问题或是有解决方法,欢迎告知,谢谢。

附上代码下载地址: 
http://www.oschina.net/code/snippet_1409622_33243

				
时间: 2024-11-08 10:20:32

ViewPager + HorizontalScrollView 实现可滚动的标签栏的相关文章

Android ViewPager+HorizontalScrollView实现标题栏滑动(腾讯新闻)

1) ViewPager提供了左右滑动切换页面的方法,但是它所提供的标题只是无语,估计没有真正的项目会照搬拿过来;并且它只能一页一页滑,我想直接查看最后一页要滑半天; 2) 看了腾讯新闻客户端感觉体验很好,所以就仿着写了,因为只是做个demo供大家交流也是给自己做个笔记,所以功能实现就行demo比较简单; 3) 有兴趣的可以在demo的基础拓展,如果哪里写得不好还望大家多多赐教.一起交流 4) 直接上主要代码,所以注释都写在代码里,最后会给工程包.(PS是在AS环境下生成的) 先放个效果图: M

【Android实战】ScrollView+GridView+ViewPager实现导航页

按照常规思路,实现导航页有专门的TabHost或ViewPager,但自定义空间不大,再者,自己想熟悉一下多重布局和动画效果的使用,因此采用这种ScrollView+GridView+ViewPager的效果. 其中比较麻烦的是GridView实现横向的加载,并且下面的滚动条随着滑动也得滚动和动态发生位置变化. public class MainAct extends FragmentActivity { ViewPager viewPager; HorizontalScrollView scr

使用TabLayout快速实现一个导航栏

在没有Material Design的年代,要实现一个类似微信主页面的效果,我们有以下几种解决方案: 1.Fragment + ViewPager  +  RadioGroup自定义固定导航条 2.Fragment + ViewPager  带滑动导航条 3.Fragment + ViewPager +  HorizontalScrollView自定义滑动导航条 当然,除了这些之外,还有许多已经被Google丢弃的方案,我们就不说了.当有了Material Design之后,一切都变得那么漂亮,

android下拉刷新控件之第三方开源控件的使用实现

本次使用的第三方下拉刷新控件是:Android-Pull-Refresh,下载地址:https://github.com/chrisbanes/Android-PullToRefresh 该控件适用于: ViewPager HorizontalScrollView ScrollView WebView GridView ListView ExpandableListView ListFragment 从github上下载解压后,将library,PullToRefreshListFragment

Android开发之自定义HorizontalScrollView视图实现仿ViewPager效果

开发过程中,需要达到 HorizontalScrollView和ViewPager的效果,于是直接重写了HorizontalScrollView来达到实现ViewPager的效果. 实际效果图如下: (1)自定义HorizontalScrollView类:AppHorizontalScrollView实现: package com.czm.ui.view; import java.util.ArrayList; import android.content.Context; import and

HorizontalScrollView滑动 ViewPager切换

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent"    tools

Android重写HorizontalScrollView模仿ViewPager效果

Android提供的ViewPager类太复杂,有时候没有必要使用,所以重写一个HorizontalScrollView来实现类似的效果,也可以当做Gallery来用 思路很简单,就是重写onTouchEvent事件,在手指抬起或者取消的时候,进行smoothScroll的操作,具体请看代码: 布局文件:activity_test.xml ? 1 2 3 4 5 6 1 <!--?xml version=1.0 encoding=utf-8?-->  2 <com.example.tes

ViewPager中嵌套HorizontalScrollView导致无法滑动换页冲突解决

在之前的开发过程中,遇到过ViewPager中嵌套HorizontalScrollView导致无法横向滑动换页,最终也是通过对onTouchEvent方法判断滑动的状态来进行控制左右的滑动,注意的是onTouchEvent的事件分发的方向正好与onInterceptTouchEvent相反,是从下往上分发的,所以也会先执行子View的onTouchEvent方法(如果想进一步了解View的分发机制可以去进一步查阅资料,比如http://blog.csdn.net/a553181867/artic

viewpager标签栏之PagerTab

import android.content.Context; import android.graphics.Canvas; import android.graphics.Paint; import android.graphics.Typeface; import android.support.v4.view.ViewCompat; import android.support.v4.view.ViewPager; import android.support.v4.widget.Edg