17.39.自定义view

/**
* 每天一个知识点day17 TODO 自定义View
* 自定义view的三个流程:
* onMeasure onLayout onDraw
* onMeasure 测量 系统会根据xml布局文件和代码对控件属性的设置,来获取或者计算出每一个view
* 或者viewGroup的尺寸,并将这些尺寸保存下来。
* onMeasure的过程根据view的类型分为两种
* 1.单一的view:通过measure的过程就完成了其测量的过程。
* 过程:measure() -> onMeasure() -> setMeasureDimension() -> getDefaultSize() -> 完成测量
* 2.ViewGroup:除了完成自身的测量以外,还会遍历调用子元素的measure方法,各个子元素再递归执行该流程。
* 过程:view.measure()->view.onMeasure()->viewGroup.measureChild()->
* viewGroup.getChildMeasureSpec->单一view的measure过程->合并到viewGroup测量值->
* view.setMeasureDimension()->完成测量。
*
* onLayout 布局 根据测量出的结果以及对应的参数,来确定每一个控件应该显示的位置。
* 1.单一view: layout->setOpticalFrame -> onLayout(空实现) ->完成布局
* 2.viewGroup: vg.layout()->调用super.layout->view.layout()->onLayout()->遍历子view->
* 计算当前view的位置->确定子view在自身的位置->child.layout->单一view的layout过程->完成
*
* onDraw 绘制 确定好位置后将这些控件绘制到屏幕上
* 单一view:绘制自身
* 过程:draw()->绘制background->绘制自己->绘制子view->绘制装饰(foreground,滚动条等)->结束
* viewGroup: 除了绘制自身view外,还需要绘制所有的子view
* 过程:view.draw()->view.drawBackground()->view.onDraw()->vg.dispatchDraw()->
* 单一view的draw过程->view.onDrawForeground()->结束
*
* onMeasure的MeasureSpec MeasureSpec概括了从父布局传递给子view布局的要求,每一个MeasureSpec代表了
* 宽度或者高度的要求,由size和mode组成。重写onMeasure方法时必须调用setMeasureDimension(w,h)
* 来存储view测量出的宽高
* 一个int32位,前两位用来保存mode,后面30为用来保存size
*
* UNSPECIFIED 父容器没有对当前View有任何限制,当前View可以任意取尺寸
* EXACTLY 当前的尺寸就是当前View应该取的尺寸 match_parent 100dp
* AT_MOST 当前尺寸是当前View能取的最大尺寸 wrap_content
*
*
* invalidate() 和 postInvalidate(),requestLayout() 的区别?
* invalidate和postInvalidate都是进行UI刷新的,invalidate方法运行在UI线程。postInvalidate运行在
* 非UI线程。用于将线程切换到UI线程,postInvalidate最后还是调用了invalidate。
* requestLayout():刷新View,重新执行measure()、layout()、draw()方法。
/
/
*
* 每天一个知识点day39 TODO view的draw() onDraw() dispatchDraw()方法的调用顺序
*
* view中的draw(Canvas canvas){
*
* 1. Draw the background 绘制背景
* 2. If necessary, save the canvas' layers to prepare for fading
* 如有必要,颜色渐变淡之前保存画布层(即锁定原有的画布内容)
* 3. Draw view's content 绘制view的内容
* 4. Draw children 绘制子view
* 5. If necessary, draw the fading edges and restore layers
* 如有必要,绘制颜色渐变淡的边框,并恢复画布(即画布改变的内容附加到原有内容上)
* 6. Draw decorations (scrollbars for instance)
* 绘制装饰,比如滚动条
*
* onDraw(canvas);
* dispatchDraw(canvas);
* }
*
* viewGroup中的dispatchDraw(Canvas canvas){
*
* drawChild(...)
* }
* 画完背景后,draw过程会调用onDraw(Canvas canvas)方法,
* 然后就是dispatchDraw(Canvas canvas)方法,
* dispatchDraw()主要是分发给子组件进行绘制,
* 我们通常定制组件的时候重写的是onDraw()方法。
* 值得注意的是ViewGroup容器组件的绘制,
* 当它没有背景时直接调用的是dispatchDraw()方法,
* 而绕过了draw()方法,当它有背景的时候就调用draw()方法,
* 而draw()方法里包含了dispatchDraw()方法的调用。
* 因此要在ViewGroup上绘制东西的时候往往重写的是dispatchDraw()方法而不是onDraw()方法,
* 或者自定制一个Drawable,重写它的draw(Canvas c)和 getIntrinsicWidth(),
* getIntrinsicHeight()方法,然后设为背景。
*
* draw() -> onDraw() -> dispatchDraw()
*
*/
https://blog.csdn.net/u012732170/article/details/55045472
自定义view的padding margin的处理。

子view的padding是在onDraw()里处理的
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
paddingLeft = getPaddingLeft();
paddingTop = getPaddingTop();
paddingRight = getPaddingRight();
paddingBottom = getPaddingBottom();
Rect rect = new Rect();
rect.left = 0 + paddingLeft;
rect.top = 0 + paddingTop;
rect.right = width - paddingRight;
rect.bottom = height - paddingBottom;
canvas.drawRect(rect,mPaint);
}

viewgroup的padding:

  1. 在onMeasure方法中处理ViewGroup的layout_width和layout_height为wrap_content的情况。本例模拟的是类似LinearLayout垂直布局的情景,因此在宽度方面,选择子View中宽度最大的那一个再加上padding值作为ViewGroup预设的宽度;在高度方面,算上所有子View的高度再加上padding值作为ViewGroup预设的高度,最后将预设值与剩余空间进行比较,选择值最小的作为ViewGroup测量之后的值。
    2.在onLayout中处理padding,只需要在布置子View的时候加上padding的值即可。

viewgroup的margin:
margin的处理方式和自定义ViewGroup中padding的处理方式有点类似,在ViewGroup测量时,算上子View与ViewGroup的margin;在给子View布局时,算上margin即可。

  1. onMeasure方法可能会调用多次,因此在onMeasure中累加得做好初始化准备 。
  2. 直接使用MarginLayoutParams类会报错,因此得在ViewGroup中重载generateLayoutParams方法并且返回一个新的MarginLayoutParams对象,才能获取到margin值。
    总结一下:
    在自定义View中处理padding,只需要在onDraw()中处理,别忘记处理布局为wrap_content的情况。
    在自定义ViewGroup中处理padding,只需要在onLayout()中,给子View布局时算上padding的值即可,也别忘记处理布局为wrap_content的情况。
    自定义View无需处理margin,在自定义ViewGroup中处理margin时,需要在onMeasure()中根据margin计算ViewGroup的宽、高,同时在onLayout中布局子View时也别忘记根据margin来布局。

滑动冲突解决:https://blog.csdn.net/u012732170/article/details/54897422
滑动冲突包括:
(1)不同方向冲突,比如HorizontalScrollView内嵌ListView
(2)同方向冲突,比如纵向ScrollView嵌套ListView,两个滑动方向相同
(3)包含1、2中情况的冲突。
(1)对于第一种情况,一个是横向滑动,一个是纵向滑动,那么我们就让它在横向滑动时将事件交由HorizontalScrollView全权处理,在纵向滑动时将事件交由ListView全权处理,也就是说,当横向滑动时,HorizontalScrollView将拦截事件,此时产生的点击事件就由HorizontalScrollView的onTouchEvent来完成;当纵向滑动时,HorizontalScrollView不拦截事件,此时产生的点击事件就由ListView的onTouchEvent来完成。
(2)对于第二种情况,解决方案也是类似,视具体的业务需求来解决,比如ListView上头还有一个View,当滑动距离在那个View的高度之间时,ScrollView拦截事件,由ScrollView处理上下滑动的事件,当滑动距离到达ListView时,就不拦截事件,让ListView继续处理上下滑动事件,当然,这只是其中的一个例子。
(3)对于第三种情况也是类似的,就是要考虑是否拦截的判定条件变得更为复杂,我下面要讲的例子就涉及了这方面的内容。
该如何解决滑动冲突呢?
有2种方法:
(1)外部拦截法,顾名思义,就是从父元素着手来拦截,具体实现就是在父元素的onInterceptTouchEvent方法中根据解决思路来拦截事件。
(2)内部拦截法,类似的,就是从子元素着手,主动告诉父元素是否要拦截,具体实现就是在子元素的dispatchTouchEvent方法中使用getParent().requestDisallowInterceptTouchEvent(false);
false的意思就是告诉父元素此时需要拦截事件了,true就是不让父元素拦截,至于getParent()方法,视你的布局而定,如果只有一个ViewGroup和一个View(或者是Viewgroup),那么用一个getParent()就可以了,如果有两个ViewGroup,那么对于最底层的那个View(或者ViewGroup)就要使用两次getParent(),即
getParent().getParent().requestDisallowInterceptTouchEvent(false);

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353