Trying to understand view hierarchy
I'm writing an app with the sole purpose of trying to understand how the view hierarchy in Android works. I am having some really harsh problems in it right now. I'll try to be concise in my explanation here.
Setup:
Currently I have three Views. 2 are ViewGroups
and 1 is just a View
. Let's say they're in this order:
TestA extends ViewGroup
TestB extends ViewGroup
TestC extends View
TestA->TestB->TestC
Where TestC is in TestB and TestB is in TestA.
In my Activity
I simply display the views like so:
TestA myView = new TestA(context);
myView.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.MATCH_PARENT));
setContentView(myView);
Problems:
The
onDraw(Canvas canvas)
method of TestA is never called. I've seen a couple solutions to this saying that my view doesn't have any dimensions (height/width = 0), however, this is not the case. When I overrideonLayout()
, I get the dimensions of my layout and they are correct. Also, getHeight()/Width() are exactly as they should be. I can also overridedispatchDraw()
and get my base views to draw themselves.I want to animate an object in
TestB
. Traditionally, I would override theonDraw()
method on callinvalidate()
on itself开发者_Go百科 until the object finish the animation it was supposed to do. However, inTestB
, when I callinvalidate()
the view never gets redrawn. I'm under the impression that it's the job of my parent view to call theonDraw()
method again, but my parent view does not call thedispatchDraw()
again.
I guess my questions are, why would my onDraw()
method of my parent view never get called to begin with? What methods in my parent view are supposed to be called when one of it's children invalidate itself? Is the parent the one responsible for ensure it's children get drawn or does Android take care of that? If Android responds to invalidate()
, why does my TestB
never get drawn again?
Ok, after some research and a lot of trying, I've found out I was doing three things wrong in regards to problem #2. A lot of it was simple answers but not very obvious.
You need to override
onMeasure()
for every view. WithinonMeasure()
, you need to callmeasure()
for every child contained in theViewGroup
passing in the MeasureSpec that the child needs.You need to call
addView()
for every child you want to include. Originally, I was simply created a view object and using it directly. This allowed me to draw it once, but the view was not include in the view tree, thus it when I calledinvalidate()
it wasn't invalidating the view tree and not redrawing. For example:
In testA:
TestB childView;
TestA(Context context){
****** setup code *******
LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
childView = new TestB(context);
childView.setLayoutParams(params);
}
@Override
protected void dispatchDraw(Canvas canvas){
childView.draw(canvas);
}
This will draw the child view once. However, if that view needs updating for animations or whatever, that was it. I put addView(childView)
in the constructor of TestA to add it to the view tree. Final code is as such:
TestB childView;
TestA(Context context){
****** setup code *******
LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
childView = new TestB(context);
childView.setLayoutParams(params);
addView(childView);
}
@Override
protected void dispatchDraw(Canvas canvas){
childView.draw(canvas);
}
Alternatively, I could override dispatchDraw(Canvas canvas)
like so if I had many more children to draw, but I need some custom element between each drawing like grid lines or something.
@Override
protectd void dispatchDraw(Canvas canvas){
int childCount = getChildCount();
for(int i = 0; i < size; i++)
{
drawCustomElement();
getChildAt(i).draw(canvas);
}
}
- You must override
onLayout()
(it's abstract inViewGroup
anyway, so it's required anyway). Within the this method, you must calllayout
for every child. Even after doing the first two things, my views wouldn't invalidate. As soon as I did this, everything worked just perfectly.
UPDATE: Problem #1 has been solved. Another extremely simply but not-so-obvious solution.
When I create an instance of TestA
, I have to call setWillNotDraw(false)
or else Android will not draw it for optimization reasons. So the full setup is:
TestA myView = new TestA(context);
myView.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,ViewGroup.LayoutParams.MATCH_PARENT));
myView.setWillNotDraw(false);
setContentView(myView);
This isn't a direct answer, but here is a fantastic tutorial on how to draw custom views and gives a great crash cours in how to animate a view. The code is very simple and clean too.
Hope this helps!
精彩评论