TabSwitcher自定义控件(带拖动&滑动效果)

[url="http://gundumw100.iteye.com/blog/1139708"]上一篇[/url]实现了滑动效果,还是不爽,这次还添加了拖动效果,请大家帮忙测试。

上传于8.5 1:30
8.5 4:15一次更新

[img]http://dl.iteye.com/upload/attachment/531394/834f1935-9fa6-395c-bd69-fa8044c9e213.png[/img]


package com.ql.view;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Color;
import android.util.AttributeSet;
import android.util.Log;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.animation.LinearInterpolator;
import android.view.animation.TranslateAnimation;
import android.widget.FrameLayout;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.ql.app.R;

public class TabSwitcher extends FrameLayout{

private static final String tag="TabSwitcher";
private Context context;
private String[] texts;
private int arrayId;
private int selectedPosition=0;
private int oldPosition=selectedPosition;
private ImageView iv;
private LinearLayout.LayoutParams params;
private LinearLayout layout;
private int iv_width;
private TextView[] tvs;
public TabSwitcher(Context context) {
super(context);
// TODO Auto-generated constructor stub
init();
}
public TabSwitcher(Context context, AttributeSet attrs) {
super(context, attrs);
// TODO Auto-generated constructor stub
Log.i(tag, "--------------TabSwitcher2---------------------");
init();
TypedArray a=context.obtainStyledAttributes(attrs,R.styleable.custom);
arrayId=a.getResourceId(R.styleable.custom_arrayId, 0);
// selectedPosition=a.getInt(R.styleable.custom_selectedPosition, 0);
a.recycle();
}
private void init(){
context=getContext();
FrameLayout.LayoutParams params=new FrameLayout.LayoutParams(FrameLayout.LayoutParams.FILL_PARENT,FrameLayout.LayoutParams.WRAP_CONTENT);
setLayoutParams(params);
setBackgroundResource(R.drawable.tabswitcher_long);

}

@Override
protected void onFinishInflate() {
// TODO Auto-generated method stub
super.onFinishInflate();
Log.i(tag, "--------------onFinishInflate---------------------");
if(arrayId!=0){
texts=getResources().getStringArray(arrayId);
}else{
texts=new String[]{};
}
tvs=new TextView[texts.length];
}

OnClickListener listener = new OnClickListener(){

@Override
public void onClick(View v) {
// TODO Auto-generated method stub
selectedPosition=(Integer)v.getTag();
if(tvs[selectedPosition].isClickable()){
tvs[oldPosition].setClickable(true);
tvs[selectedPosition].setClickable(false);
doAnimation();

oldPosition=selectedPosition;
if(onItemClickLisener!=null){
onItemClickLisener.onItemClickLisener(v, selectedPosition);
}
}

}

};

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
// TODO Auto-generated method stub
super.onSizeChanged(w, h, oldw, oldh);
Log.i("tag", "---------------onSizeChanged--------------------");
if(selectedPosition>texts.length-1){
throw new IllegalArgumentException("The selectedPosition can't be > texts.length.");
}
layout=new LinearLayout(context);
params=new LinearLayout.LayoutParams(LinearLayout.LayoutParams.FILL_PARENT,this.getMeasuredHeight());//为了居中显示文字
params.weight=1;
params.gravity=Gravity.CENTER_VERTICAL;
for(int i=0;i<texts.length;i++){
TextView child=new TextView(context);
child.setTag(i);
child.setText(texts[i]);
child.setTextSize(16);
child.setTextColor(Color.BLACK);
child.setGravity(Gravity.CENTER);
child.setOnClickListener(listener);
if(i==selectedPosition){
child.setClickable(false);
}else{
child.setClickable(true);
}
tvs[i]=child;
layout.addView(child,params);
}

oldPosition=selectedPosition;
//
iv_width=this.getMeasuredWidth()/texts.length;//计算ImageView的宽
// LinearLayout.LayoutParams p=new LinearLayout.LayoutParams(iv_width,LinearLayout.LayoutParams.FILL_PARENT);
LinearLayout.LayoutParams p=new LinearLayout.LayoutParams(iv_width,this.getMeasuredHeight());
// p.leftMargin=selectedPosition*iv_width;//无效,因为FrameLayout必须对齐左上角。
iv=new ImageView(context);
// iv.setImageResource(R.drawable.tabswitcher_short);
// iv.setScaleType(ScaleType.FIT_XY);
iv.setBackgroundResource(R.drawable.tabswitcher_short);
iv.setOnTouchListener(touchListener);
iv.setClickable(true);
this.addView(iv,p);
this.addView(layout,params);
}

private void doAnimation(){
// TranslateAnimation animation = new TranslateAnimation(oldPosition*iv_width, selectedPosition*iv_width, 0, 0);
iv.layout(selectedPosition*iv_width, iv.getTop(),
selectedPosition*iv_width+iv.getWidth(), iv.getBottom());//很关键!
TranslateAnimation animation = new TranslateAnimation((oldPosition-selectedPosition)*iv_width, 0, 0, 0);
animation.setInterpolator(new LinearInterpolator());
animation.setDuration(400);
animation.setFillAfter(true);
iv.startAnimation(animation);
}

private OnItemClickLisener onItemClickLisener;
public void setOnItemClickLisener(OnItemClickLisener onItemClickLisener) {
this.onItemClickLisener = onItemClickLisener;
}
public interface OnItemClickLisener{
void onItemClickLisener(View view,int position);
}

public void setTexts(String[] texts) {
this.texts = texts;
}
// public void setSelectedPosition(int selectedPosition) {
// this.selectedPosition = selectedPosition;
// }

OnTouchListener touchListener=new OnTouchListener(){
int temp[] = new int[]{0, 0};
@Override
public boolean onTouch(View v, MotionEvent event) {
// TODO Auto-generated method stub
int action = event.getAction();
int x = (int)event.getRawX()-getLeft();//获得拖动点在屏幕的坐标
// int x = (int)event.getX();//很奇怪,为什么这样x就会“跳”呢?
// Log.i(tag, "x======="+x);
// int y = (int)event.getRawY();//只能水平拖动,所以y方向不需要
switch (action) {
case MotionEvent.ACTION_DOWN:
temp[0] = (int)event.getX();
// temp[1] = (int)(y-v.getTop());
v.postInvalidate();
break;
case MotionEvent.ACTION_MOVE:
int left = x - temp[0];
int right = left + v.getWidth();
// int top = y - temp[1];
// int bottom = top + v.getHeight();
int top=0;//y
int bottom=0+v.getHeight();
if (left < 0){//边界判断
left = 0;
right = left + v.getWidth();
}

if (right > getMeasuredWidth()){
right = getMeasuredWidth();
left = right - v.getWidth();
}

/*if (top < 0) {
top = 0;
bottom = top + v.getHeight();
}

if (bottom > getMeasuredHeight()) {
bottom = getMeasuredHeight();
top = bottom - v.getHeight();
}*/
v.layout(left, top, right, bottom);
v.postInvalidate();
break;
case MotionEvent.ACTION_UP:
tvs[oldPosition].setClickable(true);
setBestPosition();
if(oldPosition!=selectedPosition){//回调
if(onItemClickLisener!=null){
onItemClickLisener.onItemClickLisener(tvs[selectedPosition], selectedPosition);
}
}
oldPosition=selectedPosition;//
tvs[selectedPosition].setClickable(false);
break;
}
return false;
}

};
/**
* 获得最佳停留位置
*/
private void setBestPosition() {
int left = iv.getLeft();
selectedPosition = Math.round(1.0F*left/iv_width);//四舍五入
int toPosition = selectedPosition*iv_width;
iv.layout(selectedPosition*iv_width, iv.getTop(),
selectedPosition*iv_width+iv.getWidth(), iv.getBottom());
TranslateAnimation animation = new TranslateAnimation(left-toPosition,0,0,0);
animation.setInterpolator(new LinearInterpolator());
animation.setDuration(400);
animation.setFillAfter(true);
iv.startAnimation(animation);
// iv.invalidate();

}
}


package com.ql.app;

import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

import com.ql.view.TabSwitcher;
import com.ql.view.TabSwitcher.OnItemClickLisener;

public class App extends Activity{
private Context context;
private TabSwitcher tabSwitcher;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
context=this;
tabSwitcher=(TabSwitcher)findViewById(R.id.tabSwitcher);

tabSwitcher.setOnItemClickLisener(onItemClickLisener);
// tabSwitcher.setTexts(new String[]{"1","2","3"});
}


OnItemClickLisener onItemClickLisener=new OnItemClickLisener(){

@Override
public void onItemClickLisener(View view, int position) {
// TODO Auto-generated method stub
//
switch (position) {
case 0:
case 1:
case 2:
case 3:
case 4:
Log.i("App", "position:"+position);
// Toast.makeText(context, "position clicked:"+position, Toast.LENGTH_SHORT).show();
break;

default:
break;
}
}

};
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值