最近很蛋疼,项目要模仿网易新闻的样式去做。上次把仿网易新闻客户端的下拉刷新写出来了,这次是ViewPager的滑动,同时ViewPager的上面标题下划线跟随者移动,本来通过ViewPager的OnPagerChangeListener的监听事件就可以完成,但是做出来之后,因为需要一直的刷新,所以很卡,一气之下,呵呵,自己完全的画了。整个点击事件,滑动事件都自己处理了。
效果图如下:
 
 
下标的长宽是随之改变的。
使用方式:
我的布局文件:
<LinearLayout 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"
    android:orientation="vertical"
    >
 
 
    <com.flyme.columnhorizontalscrollview.ColumnHorizontalScrollView 
        android:layout_width="match_parent"
        android:paddingLeft="20dip"
        android:paddingRight="20dip"
        android:layout_height="100dip"
        android:id="@+id/title"
        />
    
    <android.support.v4.view.ViewPager
        android:background="@android:color/holo_orange_light"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:id="@+id/viewpager"
        />
    

</LinearLayout>

很简单,把自定义的布局放在ViewPager的上面,很随意了,你想放哪就放哪。
 
在Activity中的使用方式为:
 
ViewPager mViewPager = (ViewPager) findViewById(R.id.viewpager);
        ColumnHorizontalScrollView title = (ColumnHorizontalScrollView) findViewById(R.id.title);
        mViewPager.setAdapter(new FragmentStatePagerAdapter(getSupportFragmentManager()) {
            @Override
            public int getCount() {
                return 10;
            }
            
            @Override
            public Fragment getItem(int arg0) {
                return new MyFragment(arg0);
            }
        });
        title.setTitle("上市的公司","历史","游戏啊","房产证","精选的搞笑片段","段子","电脑硬件","热点","轻松一刻","时尚");  //这个是设置标题的  
 
        title.setspace(40);

title.setViewPager(mViewPager);  //这个是将ViewPager对象给自定义的View

 
我就把自定义的类代码贴出来吧:比较懒了,时间比较紧,代码封装的不好,也没怎么封装,见谅。
public class ColumnHorizontalScrollView extends View implements OnPageChangeListener {
 
    private Context context;
    private int width ;
    private int padingLeft;
    private int padingRight ; 
    private float lineStartX = 0  ;
    private float lineEndX = 0  ;
    private Paint textPaint;
    private int textWidth ;
    private int lastPosition = 0 ;
    private float lineScale = 0 ; 
    
    private ViewPager mViewPager ;
    
    
    /**
     * 这些事默认设置的
     */
    private static final int DEFULT_TEXT_COLOR = Color.BLACK;
    private static final int DEFULT_TEXT_SIZE = 40;
    private  static final int DEFULT_LINE_COLOR = Color.RED;
    private static final int DEFULT_LINE_STRO = 5;
    
    
    private String [] str ; // 存储 题要显示的字符
    private float [] strX ; // 存储 每个字符  开始的X位置
    private float [] strWidth ;  // 存储每个字符的长度
    private int startX; // 与手势相关    记录按下的X的坐标
    
    private int recordDownX = 0 ;
    
    private OnClickOneListener mOnClickOneListener ;
    private OverScroller mScroller; 
 
    public ColumnHorizontalScrollView(Context context, AttributeSet attrs,
            int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(context);
    }
    
    public ColumnHorizontalScrollView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init(context);
    }
    public ColumnHorizontalScrollView(Context context) {
        super(context);
        init(context);
    }
    
    
    /**
     * 初始化的时候需要new出来的一些类,和必要的设置
     * @param context
     */
    private void init(Context context){
        this.context = context ;
        mScroller = new OverScroller(context,new LinearInterpolator());
        textPaint = new Paint();
        linePaint = new Paint();
        
        textPaint.setColor(DEFULT_TEXT_COLOR);
        textPaint.setTextSize(DEFULT_TEXT_SIZE);
        textPaint.setTypeface(Typeface.SANS_SERIF);
        
        linePaint.setColor(DEFULT_LINE_COLOR);
        linePaint.setStrokeWidth(DEFULT_LINE_STRO);
    }
    
    
    
    
    /**
     * 画图
     */
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        if (str == null) {
            return ;
        }
        measureSize();
        drawText(canvas); //画字
        measureLineSize();//测量下划线的位置
        drawLine(canvas);//画下划线
    }
    
    
    
    /**
     * 这个是用来通过Scroller来控制缓冲变化的   后来不需要了
     */
    @Override
    public void computeScroll() {
        if (mScroller.computeScrollOffset()) {
            scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
        }
        super.computeScroll();
        postInvalidate();
    }
    
    
    /**
     * 画字的具体代码
     * @param canvas
     */
    private  void drawText(Canvas canvas){
        int totel = 0 ;
        for (int i = 0; i < str.length; i++) {
            if (i == 0 ) {
                totel += padingLeft;
            }
            strX[i] = totel ;
            canvas.drawText(str[i], strX[i], textHeight, textPaint);
            float oneTextWidth = textPaint.measureText(str[i]);
            totel =(int) (totel +  oneTextWidth +space);
            strWidth[i] = oneTextWidth ;
        }
        textWidth = totel - space  +padingLeft ;
    }
    
    
    private void drawLine(Canvas canvas){
        canvas.drawLine(lineStartX, lineHeight ,lineEndX , lineHeight, linePaint);
    }
    
    
    
    private void measureSize(){
        width = getWidth();
        height = getHeight();
        padingLeft = getPaddingLeft();
        padingRight = getPaddingRight();
        textHeight = (int) (height / 2 - (textPaint.descent() + textPaint.ascent())/2); 
        lineHeight = (int) (textHeight - (textPaint.descent() + textPaint.ascent()));
    }
    
    
    
    private void measureLineSize(){
        lineStartX = strX[lastPosition] - space / 2 + (strWidth[lastPosition] + space )* lineScale;
        if ((lastPosition + 1) == mViewPager.getAdapter().getCount()) {
            lineEndX = strX[lastPosition] + strWidth[lastPosition] + space / 2 ;          
        }else {
            lineEndX = strX[lastPosition] + strWidth[lastPosition] + space / 2 + ( strWidth[lastPosition + 1] + space ) * lineScale;          
        }
    }
    
    
    
    public void setTitle(String ...str){
        this.str = str ;
        strX = new float[str.length];
        strWidth = new float[str.length];
    }
    
    
    private int space = 0 ;
    private Paint linePaint;
    private int lineHeight;
    private int height;
    private int textHeight;
    
    public void setspace(int space){
        this.space = space ;
    }
    
    public void setTextColorResourceId(int colorid){
        int color = context.getResources().getColor(colorid);
        textPaint.setColor(color);
    }
    
    public void setTextColor(int color){
        textPaint.setColor(color);
    }
    
    
    
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            startX = recordDownX = (int) event.getX();
            break;
        case MotionEvent.ACTION_MOVE:
            int moveX = (int) event.getX();
            int scrollX = getScrollX();
            if (Math.abs(moveX - recordDownX ) > 32 ) {
                smootMove(scrollX , startX - moveX);
            }
            startX = (int) event.getX();
            break;
        case MotionEvent.ACTION_UP:
            int upX = (int) event.getX();
            if (Math.abs(upX - recordDownX ) < 32 ) {
                int position = decidePosition(upX);
                setPosition(position);
                if (mOnClickOneListener != null) {
                    mOnClickOneListener.onClick(position) ;
                }
            }
            break;
        }
        return true;
    }
    
    
    private int  decidePosition(int upX){
        for (int i = 0; i < strX.length; i++) {
            if ((upX + getScrollX()) < (strX[i] + strWidth[i]) + space / 2) {
                return i ;
            }
        }
        return 0 ;
    }
    
    
    public void setPosition(int position){
//        lastPosition = position ;
//        int scrollX = getScrollX();
//        int location = (int) (strX[lastPosition] - space - 20);
//        int dy = (int)(location  - getScrollX());
//        if ((location + width) > (textWidth + padingRight ) ) {
//            mScroller.startScroll(scrollX, 0,textWidth + padingRight - width - getScrollX()  , 0, 200);
//        }else {
//            mScroller.startScroll(scrollX, 0, dy , 0, 200);
//        }
//        if ((scrollX + dy ) > (textWidth+padingRight - width)) {
//            mScroller.startScroll(scrollX, 0, textWidth+padingRight - width - scrollX , 0, 200);
//        }else {
//            mScroller.startScroll(scrollX, 0, dy , 0, 200);
//        }
        mViewPager.setCurrentItem(position);
    }
    
    
    private void smootMove(int scrollX , int dy){
        if ((scrollX <= 0 && dy > 0 && textWidth > width)  || (scrollX > 0)) {
            if ((scrollX + dy) < 0 ) {
                scrollTo(0 , 0);
            }else {
                if ((scrollX + dy + getWidth()) > (textWidth+padingRight )) {
                    scrollTo(textWidth +padingRight - getWidth() , 0);
                }else {
                    scrollTo(scrollX + dy , 0);
                }
            }
        }
    }
    
    public void setViewPager(ViewPager mViewPager){
        this.mViewPager = mViewPager;
        if (this.mViewPager != null) {
            this.mViewPager.setOnPageChangeListener(this);
        }
    }
    
    /**
     * dip转为 px
     */
    public int dip2px(float dipValue) {
        final float scale = context.getResources().getDisplayMetrics().density;
        return (int) (dipValue * scale + 0.5f);
    }
 
    @Override
    public void onPageScrollStateChanged(int arg0) {
        
    }
 
    @Override
    public void onPageScrolled(int lastPosition, float scale, int location) {
        this.lastPosition = lastPosition ;
        lineScale = scale ;
        invalidate();
        smootMove((int)(strX[lastPosition] - space - 20 ) ,(int)((strWidth[lastPosition] + space )*  lineScale));
    }
 
    @Override
    public void onPageSelected(int arg0) {
    }
    
    
    public void setOnClickOneListener (OnClickOneListener mOnClickOneListener){
        this.mOnClickOneListener = mOnClickOneListener ;
    }
    
    
    /**
     * 内部接口    是点击事件的接口
     * @author lenovo
     *
     */
    interface OnClickOneListener{
        public void onClick(int position);
    }
    

}

 
over。
关于bug:
这个做的还不是很完善,还有一些东西没有做适配。如果有人用,在使用过程中出现问题,你可以自己做一下简单的适配。
 
 
源码下载地址: 

http://pan.baidu.com/s/1sj0lIxR

 
我的github地址:https://github.com/flyme2012
我的博客地址:http://www.cnblogs.com/flyme2012/

Android之自定义(上方标题随ViewPager手势慢慢滑动)的更多相关文章

  1. Android之自己定义(上方标题随ViewPager手势慢慢滑动)

    近期非常蛋疼,项目要模仿网易新闻的样式去做.上次把仿网易新闻client的下拉刷新写出来了.这次是ViewPager的滑动,同一时候ViewPager的上面标题下划线尾随者移动.本来通过ViewPag ...

  2. Android Studio 去除上方标题

    选中代码再Ctrl+shift+'/' 可加/***/注释 https://blog.csdn.net/wuqingsen1/article/details/78554117 styles.xml & ...

  3. 【Android Developers Training】 70. 使用ViewPager实现屏幕滑动

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  4. Android典型界面设计(6)——ActionBar Tab+ViewPager+Fagment实现滑动导航

    一.问题描述 在Android典型界面设计一文中,实现典型滑动导航界面,其实使用ActionBar 也可以轻松实现这一效果,甚至也可实现类似Android典型界面设计(3)的双导航效果.可见Actio ...

  5. 自定义actionbar标题

    这是我自己封装的一个activity,主要作为所有Activity的基类,可以一键关掉所有的activity, 并共享一个自定义actionbar.直接切入主题吧. 第一步就是创建自定义标题的布局文件 ...

  6. 114、Android禁止ViewPager的左右滑动

    有时候在开发中会遇到一些“诡异”的要求,比如在ViewPager中嵌入ListView,或者再嵌入一个ViewPager,那么在滑动的时候就会造成被嵌入的XXView不能滑动了,那么现在就把最外层的V ...

  7. 自定义ViewPager控制是否滑动

    package com.lvshandian.menshen.view;/** * Created by zhang on 2016/11/8. * 创建自定义滑动,禁止滑动的ViewPager */ ...

  8. Android 自定义View修炼-【2014年最后的分享啦】Android实现自定义刮刮卡效果View

    一.简介: 今天是2014年最后一天啦,首先在这里,我祝福大家在新的2015年都一个个的新健康,新收入,新顺利,新如意!!! 上一偏,我介绍了用Xfermode实现自定义圆角和椭圆图片view的博文& ...

  9. Android RatingBar 自定义样式

    Android RatingBar 自定义样式 1.先定义Style: <style name="RadingStyle" parent="@android:sty ...

随机推荐

  1. SLAM技术在国内的发展现状

    近年来,由于扫地机的出现使得SLAM技术名声大噪,如今,已在机器人.无人机.AVG等领域相继出现它的身影,今天就来跟大家聊一聊国内SLAM的发展现状. SLAM的多领域应用 SLAM应用领域广泛,按其 ...

  2. PAT天梯赛L3-015 球队食物链

    读题可以知道是DFS,注意一点,题目说的是赢过,所以str[i][j]=‘W',那么g[i][j]=1,str[i][j]='L',g[j][i]=1 然后就常规搜索即可,还有一点就是剪枝,如果没有可 ...

  3. linux系统下的日志,此日志对于系统安全来说是非常重要的一 个机制!!

    var/log/messages /etc/logrotate.conf 日志切割配置文件 (参考https://my.oschina.net/u/2000675/blog/908189) dmesg ...

  4. day13--------python 内置函数(一)

    一:内置函数 内置函数就是python直接提供可以用的 01:作用域相关: locals() 返回当前作用域中的名字 globals() 返回全局作用域中的名字 02:迭代器相关: range() 生 ...

  5. HDU - 3033 滚动数组有坑

    每层至少一个,滚动时要判上一层非法与否,所以每次都要memset #include<bits/stdc++.h> #define rep(i,j,k) for(int i=j;i<= ...

  6. Oracle SET UNUSED的用法

    SET UNUSED的用法 原理:清楚掉字典信息(撤消存储空间),不可恢复.    可以使用 SET UNUSED选项标记一列或者多列不可用.    使用DROP SET UNUSED选项删除被被标记 ...

  7. 金融量化分析-python量化分析系列之---使用python获取股票历史数据和实时分笔数据

    财经数据接口包tushare的使用(一) Tushare是一款开源免费的金融数据接口包,可以用于获取股票的历史数据.年度季度报表数据.实时分笔数据.历史分笔数据,本文对tushare的用法,已经存在的 ...

  8. strus2配置strus.xml问题-The content of element type "package" must match "(result-types?,interceptors?

    搭建strus2项目,在配置strus.xml时候碰到了这个问题: The content of element type "package" must match "( ...

  9. 切换myEclipse工作空间后设置,myEclipse添加注释/设置豆沙背景颜色/调节字体大小

    一.添加注释 操作位置: 注释规范 Files/** * @文件名称: ${file_name} * @文件路径: ${package_name} * @功能描述: ${todo} * @作者: ${ ...

  10. cxf 框架 webservice

    cxf 内置了一个web服务器 cxf简单入门实例 package test; import org.apache.cxf.jaxws.JaxWsServerFactoryBean; import c ...