自定义控件、事件的处理

时间:2022-08-31 08:10:32

一、控件相关API
view.getchildcount()    获取该控件下的子控件个数
view.getchildat()    根据位置获取相应子控件
二、控件相关类
AnimationListener    动画监听的接口
三、自定义属性
               参考网站:http://blog.csdn.net/lmj623565791/article/details/45022631
自定义一个CustomView(extends View )类
 在两个参数的构造方法中,获取属性ID,并将其转换为BitMap对象,通过onDraw(Canvas canvas)方法中的canvas
 将其画在面板中,这就涉及到自定义属性:
 有两种方式:TypedArray 和AttributeSet
      第一种:通过  TypedArray 获取int类型的ID
public MyTextView(Context context, AttributeSet attrs) {
super(context, attrs);

TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.test);

String text = ta.getString(R.styleable.test_testAttr);
int textAttr = ta.getInteger(R.styleable.test_text, -1);

Log.e(TAG, "text = " + text + " , textAttr = " + textAttr);

ta.recycle();
}
        第二种:通过命名空间AttributeSet的方法

public ToggleButton(Context context, attrs) {
super(context, attrs);
//自定义的命名空间
String namespace="http://schemas.android.com/apk/res-auto";
//自定义背景图片属性
int switchBackgroundID = attrs.getAttributeResourceValue(namespace, "switchBackgroundResource", -1);
//自定义滑块图片属性
int slideButtonBackgroundID = attrs.getAttributeResourceValue(namespace, "slideButtonBackgroundResource", -1);
//自定义开关状态属性
currentState = attrs.getAttributeBooleanValue(namespace, "currentState", false);

setSwitchBackgroundResource(switchBackgroundID);
setSlideButtonBackgroundResource(slideButtonBackgroundID);

}
编写values/attrs.xml,在其中编写styleable和item等标签元素
<resources>
<declare-styleable name="ToggleButton">

<!-- 开关背景图片的属性, 引用的是R.drawable.xx. -->
<attr name="switchBackgroundResource" format="reference" />

<!-- 开关滑动块图片的属性, 引用的是R.drawable.xx. -->
<attr name="slideButtonBackgroundResource" format="reference" />

<!-- 开关当前的状态的属性, true 打开, false 关闭 -->
<attr name="currentState" format="boolean" />
</declare-styleable>
</resources>
在布局文件中CustomView使用自定义的属性(注意namespace)
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
xmlns:zhy="http://schemas.android.com/apk/res/com.example.test"
android:layout_width="match_parent"
android:layout_height="match_parent" >

<com.example.test.MyTextView
android:layout_width="100dp"
android:layout_height="200dp"
zhy:testAttr="520"
zhy:text="helloworld" />

</RelativeLayout>

onMeasure()方法
测量并设置控件的宽和高    setMeasuredDimension()
onDraw()方法
绘制控件     canvas.drawBitmap()

四、下拉刷新,上拉加载
在获取view的高宽之前,需要先调用 measure()方法进行测量

五、事件分发机制
//分发事件
dispatchTouchEvent
//拦截事件
onInterceptTouchEvent
//处理事件
onTouchEvent

 
Android滑动事件冲突

首先,我们假设这样一个场景:一个ViewPager里面嵌套一个ViewPager,内部滑动方向和外部滑动方向一样时,该怎么解决这一冲突呢? 
针对滑动冲突这里给出两种解决方案:外部拦截法,内部拦截法。

外部拦截法
情景:一个ViewPager嵌套了一个Listview,一个是左右滑动,一个上下滑动。这个时候我们可以用外部拦截法,来处理冲突。在父容器ViewPager中,重写onInterceptTouchEvent()方法,判断当左右滑动时就拦截事件,上下滑动就不拦截,将事件交由子元素Listview来处理。首先我们需要重写一个ViewPager,叫MyViewPager,然后重写onInterceptTouchEvent()方法。具体代码如下:


1 public class MyViewPager extends ViewPager {
2 private int startX;
3 private int startY;
4 public MyViewPager(Context context) {
5 super(context);
6 }
7
8
9 @Override
10 public boolean onInterceptTouchEvent(MotionEvent ev) {
11 switch (ev.getAction())
12 {
13 case MotionEvent.ACTION_DOWN:
14 startX= (int) ev.getX();
15 startY= (int) ev.getY();
16 break;
17 case MotionEvent.ACTION_MOVE:
18
19 int dX= (int) (ev.getX()-startX);
20 int dY= (int) (ev.getY()-startX);
21 if(Math.abs(dX)>Math.abs(dY)){//左右滑动
22 return true;
23 }else {//上下滑动
24 return false;
25 }
26 case MotionEvent.ACTION_UP:
27 break;
28 }
29 return super.onInterceptTouchEvent(ev);
30 }
31 }

这样就解决这种情况下的滑动冲突, 程序演示入下图: 

 

上述代码是外部拦截的典型逻辑,只需要重写onInterceptTouchEvent()方法,修改父容器当前需要的事件即可。

 

内部拦截法
情景:一个ViewPager嵌套了一个ViewPager,两个都是左右滑动。这个时候我们可以用内部拦截法,来处理冲突。即重写子元素的dispatchTouchEvent()方法,并调用getParent().requestDisallowInterceptTouchEvent(true)是父容器不能拦截子元素需要的事件。下面来看具体代码:


1 public boolean dispatchTouchEvent(MotionEvent event) {
2 ...
3
4 switch (action) {
5 case MotionEvent.ACTION_MOVE:
6 getParent().requestDisallowInterceptTouchEvent(true);
7
8 break;
9 case MotionEvent.ACTION_MOVE:
10 if(子元素需要处理此事件)
11 getParent().requestDisallowInterceptTouchEvent(true);
12
13 break;
14 case MotionEvent.ACTION_UP: {
15 break;
16 }
17 ...
18 return super.dispatchTouchEvent(event);
19 ;
20 }

当然,还需要修改父容器的onInterceptTouchEvent()方法,代码如下:


1 @Override
2 public boolean onInterceptTouchEvent(MotionEvent ev) {
3
4 int action=ev.getAction();
5 if(action==MotionEvent.ACTION_DOWN){
6 return false;
7 }else {
8 return true;
9 }
10 }

运行结果如图:

 

以上就是两种解决滑动冲突的解决方案。

来源: http://www.cnblogs.com/yxx123/p/5250101.html
六、ViewDragHelper
处理控件在界面中的拖拽问题、滑动帮助类
public DragLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
// 1. 创建 ViewDragHelper 辅助类
mHelper = ViewDragHelper.create(this, 1.0f, callback);
}

ViewDragHelper.Callback callback=new ViewDragHelper.Callback() {

/**
*1. 返回值, 决定了child是否可以被拖拽
*/
@Override
public boolean tryCaptureView(View child, int pointerId) {
// child 被用户拖拽的子View
// pointerId 多点触摸的手指id
System.out.println("tryCaptureView: ");
return true;
}

/**
*2. 返回拖拽的范围. 返回一个 >0 的值, 决定了动画的执行时长, 水平方向是否可以被滑开
*/
@Override
public int getViewHorizontalDragRange(View child) {
return mRange;
}

/**
* 3. 修正子View水平方向的位置. 此时还没有发生真正的移动.
返回值决定了View将会移动到的位置
*/
@Override
public int clampViewPositionHorizontal(View child, int left, int dx) {
if(child == mMainContent){
left = fixLeft(left);
}
return left;
}

/**
* 4. 当控件位置变化时 调用, 可以做 : 伴随动画, 状态的更新, 事件的回调.
*/
@Override
public void onViewPositionChanged(View changedView, int left, int top, int dx, int dy) {
if(changedView == mLeftContent){
// 如果移动的是左面板
// 1. 放回原来的位置
mLeftContent.layout(0, 0, 0 + mWidth, 0 + mHeight);
// 2. 把左面板发生的变化量dx转递给主面板
int newLeft = mMainContent.getLeft() + dx;

// 修正左边值.
newLeft = fixLeft(newLeft);
mMainContent.layout(newLeft, 0, newLeft + mWidth, 0 + mHeight);
}

dispatchDragEvent();

// 为了兼容低版本, 手动重绘界面所有内容.
invalidate();
}

/**
* 5. 决定了松手之后要做的事情, 结束的动画
*/
@Override
public void onViewReleased(View releasedChild, float xvel, float yvel) {
super.onViewReleased(releasedChild, xvel, yvel);
System.out.println("onViewReleased: xvel: " + xvel);
// releasedChild 被释放的孩子
// xvel 水平方向的速度 向右为+, 向左为-
if(xvel == 0 && mMainContent.getLeft() > mRange * 0.5f){
open();
} else if (xvel > 0) {
open();
} else {
close();
}
}

};

七、自定义控件
(一)自定义控件的步骤
自定义的控件类需要继承哪个类?
View、ViewGroup、FrameLayout。。。
实现三个构造方法
根据情况重写方法
onDraw()、onMeasure()、onLayout()
处理触摸事件
onTouchEvent():点击事件
OnScrollListener:滑动监听
onInterceptTouchEvent():事件的拦截
设置回调监听
当处于某个动作时需要回调相应方法作处理

(二)其他情况:
继承某个控件(比如ListView),重写该控件特有的方法,在其基础上达到效果的叠加
还有一种:也不叫自定义控件,就是完全纯代码写控件LayoutParams、addView(titleLayout)等。

(三)自定义控件的种类:
组合控件,已有的控件效果叠加
在某个已有控件上进行功能的叠加
完全自定义

(四)获取控件的宽高

我们以高度为例子: 
android提供了两个API来动态获取View的高:

view.getMeasuredHeight()
view.getHeight()
那么这两个API的区别是什么呢?

简单来说就是:

view.getMeasuredHeight() 是由view中的测量方法赋值的,这个值包含了隐藏的高度(比如一个view部分超出屏幕,他也会计算出来)。
view.getHeight() 由view的底部位置减去顶部位置,即实际显示的View的高度,不包含隐藏了的高度。
1.主动测量
代码:

view.measure(0, 0);
view.getMeasuredWidth();
view.getMeasuredHeight();
说明: 
onMeasure传入的两个参数是由父控件的大小, 
也可以使用 View.MeasureSpec.makeMeasureSpec(0,mode);设置值 
其中mode可以选择

MeasureSpec.UNSPECIFIED 未指定尺寸,比如listview中尺寸由父控件决定
MeasureSpec.EXACTLY 适合match_parent或者具体值(确定值)
MeasureSpec.AT_MOST 适合wrap_content不确定值(最大值)
注意: 
这种方法不一定能测出正确的值,因为onMesure会多次调用(由于onMesure自上而下,父控件如果对于子控件的宽高不满意,即如子控件没有限制宽高,父控件会重新调用onMesure重新测量),所以测量结果不一定正确。

关于测量的时机

2.使用OnPreDrawListener

3.使用OnGlobalLayoutListener

4.使用OnLayoutChangeListener