在处理轮班闹钟显示异常的bug时,自己在代码里手动测量了View的宽高并进行显示,结果bug并没有改好,只改对了一半。
我的处理流程:在“不提醒”时,测量了当前View的宽高并进行赋值。然后在6:30时,直接把测量的宽高给了6:30导致6:30的0被挤到第二行。
言哥后来在代码review时传授了View的三种测量模式,今天又看书看到了,决定做个笔记。
三种模式:
EXACTLY
精确值模式,当控件的layout_width和layout_height属性指定为具体数值或match_parent时。
AT_MOST
最大值模式,当空间的宽高设置为wrap_content时。
UNSPECIFIED
未指定模式,View想多大就多大,通常在绘制自定义View时才会用。
而我在代码里改的方案就是EXACTILY模式,View类默认的onMeasure方法只支持EXACTLY模式,所以如果要搞一些骚操作,就得重写onMeasure方法。
举个例子:
- /**
- * Created by qiaopc on 2017/12/15 0015.
- */
- public class MyView extends View {
- // 只实现单参数的构造方法会报错,具体原因未知,猜测可能和第二个参数有关
- public MyView(Context context, @Nullable AttributeSet attrs) {
- super(context, attrs);
- }
- @Override
- protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
- setMeasuredDimension(measureWidth(widthMeasureSpec),measureHeight(heightMeasureSpec));
- }
- private int measureWidth(int measureSpec) {
- int specMode = MeasureSpec.getMode(measureSpec);
- int specSize = MeasureSpec.getSize(measureSpec);
- int result = 0;
- if (specMode == MeasureSpec.EXACTLY) {
- result = specSize;
- } else {
- result = 200;
- if (specMode == MeasureSpec.AT_MOST) {
- result = result < specSize ? result : specSize;
- }
- }
- return result;
- }
- private int measureHeight(int measureSpec) {
- int specMode = MeasureSpec.getMode(measureSpec);
- int specSize = MeasureSpec.getSize(measureSpec);
- int result = 0;
- if (specMode == MeasureSpec.EXACTLY) {
- return specSize;
- } else {
- result = 200;
- if (specMode == MeasureSpec.AT_MOST) {
- result = Math.min(result, specSize);
- }
- }
- return result;
- }
- }
其实最后设置成MATCH_PARENT就好了。