android自定义View之(五)------ViewPagerIndicator

1.写在前面的话:

      曾经,我看到QQ浏览器的界面指示图标随着手指滑动的效果,觉得非常好,自己也想实现,自己做了一个《android viewpager 之 基本使用方法》,基本上实现了界面指示图标的功能,但是对于界面指示图标随着手指滑动的效果,自己只是知道是要在OnPageChangeListener类中的onPageScrolled方法中实现,但是多次尝试都没有成功。直到,我看到这篇博文<<Android 教你打造炫酷的ViewPagerIndicator 不仅仅是高仿MIUI>>,才明白了其中的实现方法。

      这篇博文,我主要是做二个事情。

      一是把指示图标从三角形改换为点,

      第二是,将我以前的界面指示图标随着手指滑动的效果实现。


2.简单的分析一下核心类--ViewPagerIndicator

    类ViewPagerIndicator的关键方法:

     在博文<<Android 教你打造炫酷的ViewPagerIndicator 不仅仅是高仿MIUI>>中,核心类ViewPagerIndicator的关键方法:

     onFinishInflate()--------Called after a view and all of its children has been inflated from XML.  (从XML文件导入布局文件后调用)

     onSizeChanged(int w, int h, int oldw, int oldh)------Called when the size of this view has changed.  (当子view 改变大小时调用,主要用来确定子view的大小)

     dispatchDraw(Canvas canvas)-------这个方法主要是ViewGroup容器组件的绘制,并分发给子组件进行绘制。所以,在ViewGroup上绘制东西的时候往往重写的是此方法,                                                                         而不是onDraw()方法。

    scroll(int position, float offset)-----此方法是OnPageChangeListener类中的onPageScrolled方法中的一个方法,主要是在界面滑动时,实时的更新界面指示图标的位置和实时的滚动控件的位置。


3.指示图标为圆点的ViewPagerIndicator

3.1 效果图:



3.2 代码修改:

(1)res/values/attr.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <attr name="item_count" format="integer"></attr>
    <declare-styleable name="ViewPagerIndicator">
        <attr name="item_count" />
         <attr name="indicaytor_type">  
         <enum name="indicaytor_type_circle" value="1"></enum>  
         <enum name="indicaytor_type_triangle" value="2"></enum>  
     </attr>  
    </declare-styleable>
</resources>

我们提供一个指示图标的样式选择(indicaytor_type),一个是圆点(indicaytor_type_circle),一个是三角形(indicaytor_type_triangle)


(2)在布局文件中添加指示图标的样式的属性定义:

custom:indicaytor_type="indicaytor_type_triangle"
custom:indicaytor_type="indicaytor_type_circle"

<!--      
custom:indicaytor_type="indicaytor_type_triangle" 
custom:indicaytor_type="indicaytor_type_circle"
-->
    <com.example.customviewpagerindicator.ViewPagerIndicator
        android:id="@+id/id_indicator"
        android:layout_width="match_parent"
        android:layout_height="45dp"
        android:background="@drawable/title_bar_bg_one_row"
        android:orientation="horizontal"
        custom:item_count="4"
        custom:indicaytor_type="indicaytor_type_circle"  >

(3)ViewPagerIndicator.java---代码修改:

定义二种指示图标的样式:

	private int indicaytor_type;
	private int indicaytor_type_circle = 1;
	private int indicaytor_type_triangle = 2;


选择指示图标的样式:
indicaytor_type =  a.getInt(R.styleable.ViewPagerIndicator_indicaytor_type,indicaytor_type_circle);

圆点指示图标的实现:

	protected void dispatchDraw(Canvas canvas)
	{
		if(indicaytor_type == indicaytor_type_circle){
			canvas.save();
			int radius = 6;
			canvas.translate(mInitTranslationX + mTranslationX, getHeight() - 10);
			canvas.drawCircle(radius, 0, radius, mPaint);
			canvas.restore();
		}else{
			mPaint.setPathEffect(new CornerPathEffect(3));
			canvas.save();
			canvas.translate(mInitTranslationX + mTranslationX, getHeight() + 1);
			canvas.drawPath(mPath, mPaint);
			canvas.restore();
		}

		super.dispatchDraw(canvas);
	}


4.android viewpager 之 基本使用方法---之界面指示图标随着手指滑动的效果实现

4.1  效果图




4.2  关键实现代码:

   这个主要是参考<<Android 教你打造炫酷的ViewPagerIndicator 不仅仅是高仿MIUI>>中的scroll(int position, float offset)方法来实现的:

定义移动的参数:

	private int mInitTranslationX;
	private float mTranslationX;
	private int tabCount = 3;

定义开始显示的位置:

	private void InitImageView() {
		// TODO Auto-generated method stub
		imageView= (ImageView) findViewById(R.id.cursor);  
        bmpW = BitmapFactory.decodeResource(getResources(), R.drawable.cursor).getWidth();
        Log.i(TAG, "bmpW:"+bmpW);
        screenW = getScreenWidth();
        offset = (screenW / tabCount - bmpW) / 2;    
        mInitTranslationX = screenW / tabCount / 2 - bmpW/ 2;
        
        Matrix matrix = new Matrix();  
        matrix.postTranslate(mInitTranslationX, 0);  
        imageView.setImageMatrix(matrix); 
	} 


实现实现移动的效果:

        public void onPageScrolled(int position, float positionOffset,int positionOffsetPixels) {  
        	mTranslationX = getScreenWidth() / tabCount * (position + positionOffset);     	       	
            Matrix matrix = new Matrix();  
            matrix.postTranslate(mInitTranslationX+mTranslationX, 0);  
            imageView.setImageMatrix(matrix);                  	      	
        }  


5.附核心类ViewPagerIndicator的完整代码:

import java.util.List;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.CornerPathEffect;
import android.graphics.Paint;
import android.graphics.Paint.Style;
import android.graphics.Path;
import android.support.v4.view.ViewPager;
import android.support.v4.view.ViewPager.OnPageChangeListener;
import android.util.AttributeSet;
import android.util.DisplayMetrics;
import android.util.TypedValue;
import android.view.Gravity;
import android.view.View;
import android.view.WindowManager;
import android.widget.LinearLayout;
import android.widget.TextView;

/**
 * http://blog.csdn.net/lmj623565791/article/details/42160391
 * @author zhy
 *
 */
public class ViewPagerIndicator extends LinearLayout
{

	private Paint mPaint;
	private Path mPath;


	private int mTriangleWidth;
	private int mTriangleHeight;

	private static final float RADIO_TRIANGEL = 1.0f / 6;
	private final int DIMENSION_TRIANGEL_WIDTH = (int) (getScreenWidth() / 3 * RADIO_TRIANGEL);

	private int mInitTranslationX;
	private float mTranslationX;

	private static final int COUNT_DEFAULT_TAB = 4;
	private int mTabVisibleCount = COUNT_DEFAULT_TAB;

	private List<String> mTabTitles;
	public ViewPager mViewPager;

	private static final int COLOR_TEXT_NORMAL = 0x77FFFFFF;
	private static final int COLOR_TEXT_HIGHLIGHTCOLOR = 0xFFFFFFFF;
	
	private int indicaytor_type;
	private int indicaytor_type_circle = 1;
	private int indicaytor_type_triangle = 2;

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

	public ViewPagerIndicator(Context context, AttributeSet attrs)
	{
		super(context, attrs);

		TypedArray a = context.obtainStyledAttributes(attrs,R.styleable.ViewPagerIndicator);
		mTabVisibleCount = a.getInt(R.styleable.ViewPagerIndicator_item_count,COUNT_DEFAULT_TAB);
		indicaytor_type =  a.getInt(R.styleable.ViewPagerIndicator_indicaytor_type,indicaytor_type_circle);
		if (mTabVisibleCount < 0)
			mTabVisibleCount = COUNT_DEFAULT_TAB;
		a.recycle();

		mPaint = new Paint();
		mPaint.setAntiAlias(true);
		mPaint.setColor(Color.parseColor("#ffffffff"));
		mPaint.setStyle(Style.FILL);
		//mPaint.setPathEffect(new CornerPathEffect(3));
	}
	
	@Override
	protected void onFinishInflate()
	{
		super.onFinishInflate();
		int cCount = getChildCount();
		if (cCount == 0)
			return;
		for (int i = 0; i < cCount; i++)
		{
			View view = getChildAt(i);
			LinearLayout.LayoutParams lp = (LayoutParams) view.getLayoutParams();
			lp.weight = 0;
			lp.width = getScreenWidth() / mTabVisibleCount;
			view.setLayoutParams(lp);
		}
		setItemClickEvent();
	}

	
	@Override
	protected void onSizeChanged(int w, int h, int oldw, int oldh)
	{
		super.onSizeChanged(w, h, oldw, oldh);
		mTriangleWidth = (int) (w / mTabVisibleCount * RADIO_TRIANGEL);// 1/6 of width
		mTriangleWidth = Math.min(DIMENSION_TRIANGEL_WIDTH, mTriangleWidth);
		initTriangle();
		mInitTranslationX = getWidth() / mTabVisibleCount / 2 - mTriangleWidth/ 2;
	}	
	
	@Override
	protected void dispatchDraw(Canvas canvas)
	{
		if(indicaytor_type == indicaytor_type_circle){
			canvas.save();
			int radius = 6;
			canvas.translate(mInitTranslationX + mTranslationX, getHeight() - 10);
			canvas.drawCircle(radius, 0, radius, mPaint);
			canvas.restore();
		}else{
			mPaint.setPathEffect(new CornerPathEffect(3));
			canvas.save();
			canvas.translate(mInitTranslationX + mTranslationX, getHeight() + 1);
			canvas.drawPath(mPath, mPaint);
			canvas.restore();
		}

		super.dispatchDraw(canvas);
	}
	
	private void initTriangle()
	{
		mPath = new Path();
		mTriangleHeight = (int) (mTriangleWidth / 2 / Math.sqrt(2));
		mPath.moveTo(0, 0);
		mPath.lineTo(mTriangleWidth, 0);
		mPath.lineTo(mTriangleWidth / 2, -mTriangleHeight);
		mPath.close();
	}

	public interface PageChangeListener
	{
		public void onPageScrolled(int position, float positionOffset,int positionOffsetPixels);
		public void onPageSelected(int position);
		public void onPageScrollStateChanged(int state);
	}

	private PageChangeListener onPageChangeListener;

	public void setOnPageChangeListener(PageChangeListener pageChangeListener)
	{
		this.onPageChangeListener = pageChangeListener;
	}

	public void setViewPager(ViewPager mViewPager, int pos)
	{
		this.mViewPager = mViewPager;
		mViewPager.setOnPageChangeListener(new OnPageChangeListener()
		{
			@Override
			public void onPageSelected(int position)
			{
				resetTextViewColor();
				highLightTextView(position);

				if (onPageChangeListener != null)
				{
					onPageChangeListener.onPageSelected(position);
				}
			}

			@Override
			public void onPageScrolled(int position, float positionOffset,int positionOffsetPixels)
			{
				scroll(position, positionOffset);

				if (onPageChangeListener != null)
				{
					onPageChangeListener.onPageScrolled(position,positionOffset, positionOffsetPixels);
				}
			}

			@Override
			public void onPageScrollStateChanged(int state)
			{
				if (onPageChangeListener != null)
				{
					onPageChangeListener.onPageScrollStateChanged(state);
				}
			}
		});

		mViewPager.setCurrentItem(pos);
		highLightTextView(pos);
	}

	public void scroll(int position, float offset)
	{
		mTranslationX = getWidth() / mTabVisibleCount * (position + offset);
		int tabWidth = getScreenWidth() / mTabVisibleCount;
		
		if (offset > 0 && position >= (mTabVisibleCount - 2) && getChildCount() > mTabVisibleCount)
		{
			if (mTabVisibleCount != 1)
			{
				this.scrollTo((position - (mTabVisibleCount - 2)) * tabWidth+ (int) (tabWidth * offset), 0);
			} else
			{
				this.scrollTo(position * tabWidth + (int) (tabWidth * offset), 0);
			}
		}
		invalidate();
	}	
	
	protected void highLightTextView(int position)
	{
		View view = getChildAt(position);
		if (view instanceof TextView)
		{
			((TextView) view).setTextColor(COLOR_TEXT_HIGHLIGHTCOLOR);
		}
	}

	private void resetTextViewColor()
	{
		for (int i = 0; i < getChildCount(); i++)
		{
			View view = getChildAt(i);
			if (view instanceof TextView)
			{
				((TextView) view).setTextColor(COLOR_TEXT_NORMAL);
			}
		}
	}

	public void setItemClickEvent()
	{
		int cCount = getChildCount();
		for (int i = 0; i < cCount; i++)
		{
			final int j = i;
			View view = getChildAt(i);
			view.setOnClickListener(new OnClickListener()
			{
				@Override
				public void onClick(View v)
				{
					mViewPager.setCurrentItem(j);
				}
			});
		}
	}

	private TextView generateTextView(String text)
	{
		TextView tv = new TextView(getContext());
		LinearLayout.LayoutParams lp = new LinearLayout.LayoutParams(
				LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
		lp.width = getScreenWidth() / mTabVisibleCount;
		tv.setGravity(Gravity.CENTER);
		tv.setTextColor(COLOR_TEXT_NORMAL);
		tv.setText(text);
		tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);
		tv.setLayoutParams(lp);
		return tv;
	}

	public void setVisibleTabCount(int count)
	{
		this.mTabVisibleCount = count;
	}

	public void setTabItemTitles(List<String> datas)
	{
		if (datas != null && datas.size() > 0)
		{
			this.removeAllViews();
			this.mTabTitles = datas;
			for (String title : mTabTitles)
			{
				addView(generateTextView(title));
			}
			setItemClickEvent();
		}
	}

	public int getScreenWidth()
	{
		WindowManager wm = (WindowManager) getContext().getSystemService(Context.WINDOW_SERVICE);
		DisplayMetrics outMetrics = new DisplayMetrics();
		wm.getDefaultDisplay().getMetrics(outMetrics);
		return outMetrics.widthPixels;
	}
}

6.项目代码下载地址:

http://download.csdn.net/detail/hfreeman2008/8401103

7.参考资料:

1.Android 教你打造炫酷的ViewPagerIndicator 不仅仅是高仿MIUI

http://blog.csdn.net/lmj623565791/article/details/42160391

2.android viewpager 之 基本使用方法

http://blog.csdn.net/hfreeman2008/article/details/38796449

3.view组件draw,onDraw,dispatchDraw

http://blog.csdn.net/zxxjj/article/details/7478482


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

hfreeman2008

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值