Android自定义布局 - onDraw()永远不会被调用

时间:2012-10-24 19:28:36

标签: android layout view viewgroup

public class MainActivity extends Activity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        TableView tv = new TableView(this);
        tv.setLayoutParams(new LayoutParams(LayoutParams.MATCH_PARENT,LayoutParams.MATCH_PARENT));
        setContentView(tv);      
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }
}

public class TableView extends ViewGroup {
    private Paint oval;
    private RectF rect;

    public TableView(Context context) {
        super(context);
        oval= new Paint(Paint.ANTI_ALIAS_FLAG);
        oval.setColor(Color.GREEN);
    }


    public void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        canvas.drawOval(rect , oval);
    }
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int wspec = MeasureSpec.makeMeasureSpec(
                getMeasuredWidth(), MeasureSpec.EXACTLY);
        int hspec = MeasureSpec.makeMeasureSpec(
                getMeasuredHeight(), MeasureSpec.EXACTLY);
        for(int i=0; i<getChildCount(); i++){
            View v = getChildAt(i);
            v.measure(wspec, hspec);
        }
    }


    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        float w=r-l;
        float h=b-t;
        rect=new RectF(w/8,h/8,7*w/8,7*h/8);
        float theta = (float) (2 * Math.PI / getChildCount());
        for(int i=0; i< getChildCount(); i++) {
            View v = getChildAt(i);
            w = (rect.right-rect.left)/2;
            h = (rect.bottom-rect.top)/2;
            float half = Math.max(w, h)/2;
            float centerX = rect.centerX()+w*FloatMath.cos(theta);
            float centerY = rect.centerY()+h*FloatMath.sin(theta);
            v.layout((int)(centerX-half),(int)(centerY-half),(int)(centerX+half),(int)(centerY+half));
        }
    }
}

那么几乎没有好的和深入的教程,几乎没有关于如何正确进行自定义布局的任何数据,所以我试图弄清楚它是如何完成的,我想要实现的是一个绘制绿色椭圆的布局在屏幕的中心,我希望这个布局的每个孩子都围绕着椭圆形。

你可以把椭圆形想象成一张扑克桌,我希望这种布局的孩子可以坐在它周围。

这段代码目前发生的是我得到一个没有椭圆形的白色app scren,所以我调试了它,看到onDraw永远不会被调用...

3个问题:

  1. 为什么onDraw没有被调用?
  2. sdk警告我,我不应该在onLayout方法中分配新对象,那么我应该在哪里计算RectF,以便为onDraw调用做好准备呢?
  3. 调用super.onDraw()会让所有孩子自己画画吗?或者我应该明确调用他们的draw()?
  4. 如果我弄错了,你们可以引导我朝着正确的方向前进,或者有任何与这个主题相关的示例或教程的链接也会有所帮助!

2 个答案:

答案 0 :(得分:137)

默认情况下,onDraw()对象不会调用ViewGroup。相反,您可以覆盖dispatchDraw()

或者,您可以通过调用ViewGroup构造函数中的setWillNotDraw(false)来启用TableView绘图。

编辑:

对于#2: - 在构造函数中初始化它,然后只需在rect.set()方法中调用onLayout()

对于#3: - 是的,据我所知,超级电话会处理它,你不应该处理它,除非你需要自定义孩子的绘制方式。

答案 1 :(得分:-3)

如果你想要画布重新调用invalidate(),那么onDraw()方法将被重新执行