微商怎么做 和淘宝网站一样吗,建手机网站要多少钱,郑州有哪些做网站的公司,站长工具seo综合查询收费吗一、重写onMeasure()来修改已有的View的尺寸 步骤#xff1a;
重写 onMeasure()#xff0c;并调用 super.onMeasure() 触发原先的测量用 getMeasuredWidth() 和 getMeasuredHeight() 取到之前测得的尺寸#xff0c;利用这两个尺寸来计算出最终尺寸使用 setMeasuredDimensio…
一、重写onMeasure()来修改已有的View的尺寸 步骤
重写 onMeasure()并调用 super.onMeasure() 触发原先的测量用 getMeasuredWidth() 和 getMeasuredHeight() 取到之前测得的尺寸利用这两个尺寸来计算出最终尺寸使用 setMeasuredDimension() 保存尺寸 代码
Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {//先执行原测量算法super.onMeasure(widthMeasureSpec, heightMeasureSpec);//获取原先的测量结果int measureWidthgetMeasuredWidth();int measureHeightgetMeasuredHeight();//利用原先的测量结果计算出新的尺寸if(measureWidthmeasureHeight){measureWidthmeasureHeight;}else{measureHeightmeasureWidth;}//保存计算后的结果setMeasuredDimension(measureWidth,measureHeight);}
二、重写onMeasure()来全新计算自定义View的尺寸 步骤
重写 onMeasure0) 把尺寸计算出来把计算的结果用 resolveSize() 过滤一遍后保存 Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {...measuredWidth...;measuredHeight...;measuredWidthresolveSize(measuredWidth,widthMeasureSpec);measuredHeightresolveSize(measuredHeight,heightMeasureSpec);setMeasuredDimension(measuredWidth,measuredHeight);} onMeasure方法的两个参数 widthMeasureSpec和heightMeasureSpec是父View对子View的尺寸限制子View在计算自己尺寸的时候需要遵守这两个参数所包含的限制MeasureSpec。 理解MeasureSpec
在 Android 中View 的大小是由父容器和 View 自身的测量规格MeasureSpec共同决定的。
MeasureSpec 由大小和测量模式组成测量模式有三种取值 UNSPECIFIED未指定父容器对子 View 没有施加任何限制子 View 可以任意大小。 EXACTLY精确父容器已经为子 View 精确指定了大小子 View 应该匹配这个大小。 AT_MOST至多子 View 可以是任何大小但不能超过父容器指定的大小。
MeasureSpec 是通过静态方法 MeasureSpec.makeMeasureSpec() 创建的该方法接受两个参数大小和测量模式。在自定义 View 或者自定义布局中我们通常会使用 MeasureSpec 来测量子 View 的大小并根据测量模式来决定子 View 的大小。
在自定义 View 中我们通常会在 onMeasure() 方法中使用 MeasureSpec 来测量 View 的大小。在这个方法中我们可以通过 MeasureSpec.getMode() 和 MeasureSpec.getSize() 方法来获取测量模式和大小然后根据这些信息来确定 View 的最终大小。 解释resolveSize这个方法
//代码简化不是源码
public static int resolveSize(int size, int measureSpec) {final int specMode MeasureSpec.getMode(measureSpec);final int specSize MeasureSpec.getSize(measureSpec);switch (specMode) {case MeasureSpec.AT_MOST:if (specSize size) {result specSize | MEASURED_STATE_TOO_SMALL;} else {result size;}break;case MeasureSpec.EXACTLY:result specSize;break;case MeasureSpec.UNSPECIFIED:default:result size;}
}
resolveSize这个方法父View传进来的尺寸限制measureSpec是由类型和尺寸值组成的首先要调用MeasureSpec.getModemeasureSpec方法和MeasureSpec.getSize(measureSpec)方法获取限制measureSpec的类型mode和size尺寸值。
限制的类型mode
MeasureSpec.AT_MOST 限制上线
MeasureSpec.EXACTLY 限制固定尺寸
MeasureSpec.UNSPECIFIED 无限制 三、重写onMeasure和onLayout来全新计算自定义ViewGroup的内部布局
onMeasure的重写对于ViewGroup来说包含三部分内容
步骤
调用每个子View的measure让子View自我测量根据子View给出的尺寸得出子View的位置并保存它们的位置和尺寸根据子View的位置和尺寸计算出自己的尺寸并用setMeasuredDimension()保存
理解LayoutParams 在父View里调用子View的getLayoutParams方法可以获得一个LayoutParams对象它包含了xml文件里的layout_打头的参数的对应值其中它的width和height这两个属性就分别对应了layout_width和layout_height的值并且是转换过了的值。 Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
for(int i0;igetChildCount();i){View childViewgetChildAt(i);LayoutParams lpchildView.getLayoutParams();//lp.height lp.width}}
结合自己的可用空间来计算出对子View的宽度和高度的限制
可以根据layout_width和layout_height的值分成三种情况
第一种情况固定值
不需要考虑可用空间的问题直接用EXACTLY把子View尺寸限制为这个固定值就可以了。
第二种情况match_parent
把子View的尺寸限制为固定值可用宽度或者高度 可用空间的判断方法 根据自己的MeasureSpec中mode的不同 1.EXACTLY/AT_MOST 可用空间MeasureSpec中的size 2.UNSPECIFIED 可用空间无限大 第三种情况wrap_content
不能超过父View的边界的情况下子View自己测量
public class SomeView extends ViewGroup {
...Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {...for(int i0;igetChildCount();i){View childViewgetChildAt(i);LayoutParams lpchildView.getLayoutParams();int selfwidthSpecModeMeasureSpec.getMode(widthMeasureSpec);int selfwidthSpecSizeMeasureSpec.getSize(widthMeasureSpec);switch (lp.width){case MATCH_PARENT:if(selfwidthSpecModeEXACTLY||selfwidthSpecModeMeasureSpec.AT_MOST){childWidthSpecMeasureSpec.makeMeasureSpec(selfwidthSpecSize-usedWidth,EXACTLY);}else{childWidthSpecMeasureSpec.makeMeasureSpec(0,MeasureSpec.UNSPECIFIED);}break;case WRAP_CONTENT:if(selfwidthSpecModeEXACTLY||selfwidthSpecModeMeasureSpec.AT_MOST){childWidthSpecMeasureSpec.makeMeasureSpec(selfwidthSpecSize-usedWidth,MeasureSpec.AT_MOST);}else{childWidthSpecMeasureSpec.makeMeasureSpec(0,MeasureSpec.UNSPECIFIED);}break;default:childWidthSpecMeasureSpec.makeMeasureSpec(lp.width, EXACTLY);break;}}}
} 关于保存子View位置的两点说明 1.不是所有的Layout都需要保存子View的位置因为有的Layout可以在布局阶段实时推导出子View的位置例如LinearLayout 2.有时候对某些子View需要重复测量两次或多次才能得到正确的尺寸和位置 重写onLayout来摆放子View Overrideprotected void onLayout(boolean changed, int l, int t, int r, int b) {for(int i0;igetChildCount();i){View childViewgetChildAt(i);childView.layout(childLeft[i],childTop[i],childRight[i],childBottom[i]);}}