54
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));
        }
    }
}

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

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

这段代码目前发生的事情是我得到一个没有椭圆形的白色应用程序屏幕,所以我调试它并看到 onDraw 永远不会被调用......

3个问题:

  1. 为什么 onDraw 没有被调用?
  2. sdk 警告我不应该在 onLayout 方法中分配新对象,那么我应该在哪里计算 RectF 以便为 onDraw 调用做好准备?
  3. 调用 super.onDraw() 会让所有孩子自己画画吗?还是我应该明确调用他们的draw()?

如果我全都错了,你们可以指导我朝着正确的方向前进,或者提供与该主题相关的示例或教程的任何链接,这也会有所帮助!

4

2 回答 2

163

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

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

编辑:

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

对于#3: - 是的,据我所知,超级调用将处理它,除非您需要自定义绘制孩子的方式,否则您不必处理它。

于 2012-10-24T19:32:57.223 回答
-3

如果要重新绘制画布调用invalidate(),方法onDraw() 将被重新执行

于 2014-02-13T15:44:33.543 回答