android自定义控件 onMeasure() 测量尺寸

时间:2021-11-21 02:37:18

上次讲的自定义控件刷新点屏幕的任意地方都会刷新,而且在xml里自定义控件下面放一个textview的话,这个TextView是显示不出来的,不只这个,以前的几个自定义控件都是

为什么呢?今天来讲下onMeasure()

在自定义刷新控件的基础上重写onMeasure方法

根据上一篇自定义组件修改

注释在代码里

  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. android:orientation="vertical"
  3. android:layout_width="fill_parent"
  4. android:layout_height="wrap_content"
  5. >
  6. <xue.test.CusView3
  7. android:id="@+id/cusview3"
  8. android:layout_width="wrap_content"
  9. android:layout_height="wrap_content"
  10. >
  11. </xue.test.CusView3>
  12. <TextView
  13. android:layout_width="wrap_content"
  14. android:layout_height="wrap_content"
  15. android:text="我终于出现了" />
  16. </LinearLayout>

这里的TextView无法显示,想要显示的话,要测量控件的大小

  1. public class CusView3 extends View {
  2. private int color = 0;
  3. private String text = "点击我刷新";
  4. private Paint mPaint;
  5. private int mAscent;
  6. public CusView3(Context context, AttributeSet attrs) {
  7. super(context, attrs);
  8. mPaint = new Paint();
  9. mPaint.setStyle(Style.FILL);
  10. mPaint.setTextSize(35.0f);
  11. setPadding(20, 60, 0, 0); //设置padding
  12. }
  13. @Override
  14. protected void onDraw(Canvas canvas) {
  15. super.onDraw(canvas);
  16. if (color > 2) {
  17. color = 0;
  18. }
  19. switch (color) {
  20. case 0:
  21. mPaint.setColor(Color.GREEN);
  22. break;
  23. case 1:
  24. mPaint.setColor(Color.RED);
  25. break;
  26. case 2:
  27. mPaint.setColor(Color.BLUE);
  28. break;
  29. default:
  30. break;
  31. }
  32. canvas.drawText(text, getPaddingLeft(), getPaddingTop(), mPaint);
  33. }
  34. public void changeColor() {
  35. color++;
  36. }
  37. /**
  38. * 比onDraw先执行
  39. *
  40. * 一个MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureSpec代表了一组宽度和高度的要求。
  41. * 一个MeasureSpec由大小和模式组成
  42. * 它有三种模式:UNSPECIFIED(未指定),父元素部队自元素施加任何束缚,子元素可以得到任意想要的大小;
  43. *              EXACTLY(完全),父元素决定自元素的确切大小,子元素将被限定在给定的边界里而忽略它本身大小;
  44. *              AT_MOST(至多),子元素至多达到指定大小的值。
  45. *
  46. *   它常用的三个函数:   
  47. * 1.static int getMode(int measureSpec):根据提供的测量值(格式)提取模式(上述三个模式之一)
  48. * 2.static int getSize(int measureSpec):根据提供的测量值(格式)提取大小值(这个大小也就是我们通常所说的大小)
  49. * 3.static int makeMeasureSpec(int size,int mode):根据提供的大小值和模式创建一个测量值(格式)
  50. */
  51. @Override
  52. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  53. setMeasuredDimension(measureWidth(widthMeasureSpec), measureHeight(heightMeasureSpec));
  54. }
  55. private int measureWidth(int measureSpec) {
  56. int result = 0;
  57. int specMode = MeasureSpec.getMode(measureSpec);
  58. int specSize = MeasureSpec.getSize(measureSpec);
  59. if (specMode == MeasureSpec.EXACTLY) {
  60. // We were told how big to be
  61. result = specSize;
  62. } else {
  63. // Measure the text
  64. result = (int) mPaint.measureText(text) + getPaddingLeft() + getPaddingRight();
  65. if (specMode == MeasureSpec.AT_MOST) {
  66. // Respect AT_MOST value if that was what is called for by
  67. // measureSpec
  68. result = Math.min(result, specSize);// 60,480
  69. }
  70. }
  71. return result;
  72. }
  73. private int measureHeight(int measureSpec) {
  74. int result = 0;
  75. int specMode = MeasureSpec.getMode(measureSpec);
  76. int specSize = MeasureSpec.getSize(measureSpec);
  77. mAscent = (int) mPaint.ascent();
  78. if (specMode == MeasureSpec.EXACTLY) {
  79. // We were told how big to be
  80. result = specSize;
  81. } else {
  82. // Measure the text (beware: ascent is a negative number)
  83. result = (int) (-mAscent + mPaint.descent()) + getPaddingTop() + getPaddingBottom();
  84. if (specMode == MeasureSpec.AT_MOST) {
  85. // Respect AT_MOST value if that was what is called for by
  86. // measureSpec
  87. result = Math.min(result, specSize);
  88. }
  89. }
  90. return result;
  91. }
  92. }

效果图

android自定义控件 onMeasure() 测量尺寸

代码 http://download.csdn.net/detail/ethan_xue/4178423