ObjectAnimator animate LinearLayout width

Android

Android Problem Overview


Ok, so i checked out http://android-developers.blogspot.com/2011/02/animation-in-honeycomb.html

He says you can animate the property of an object in a given time. And i tried moving around objects and it looks fine. I encountered a problem when i went changing the width of a LinearLayout. I got this:

10-26 14:51:27.190: E/PropertyValuesHolder(12681): Couldn't find setter/getter for       property width with value type float

Then i tried extending LinearLayout, with "myWidth"

public void setMyWidth(int myWidth) {
    LinearLayout.LayoutParams params = (LayoutParams) getLayoutParams();
    params.weight = myWidth;
    setLayoutParams(params);
    requestLayout();
    this.myWidth = myWidth;
}

No luck. Then i tried changing LayoutParams.width, turns out width and height are the only public properties in java history, and ObjectAnimator needs getter and setter. No luck. I'm embarassed to say i tried extending LayoutParams too... with no luck ofc.

Anybody succeded doing such a thing? I used old android.view.animation and i got what i wanted, but i'm curious for the future.

Android Solutions


Solution 1 - Android

In situations where there isn't a simple property getter/setter you should use ValueAnimator and perform the animation manually.

Assuming:

  • v is the view you're animating
  • END_WIDTH is the target width of the view in pixels.
  • DURATION is the desired length of the animation in milliseconds.

Your code should look something like this:

    ValueAnimator anim = ValueAnimator.ofInt(v.getMeasuredWidth(), END_WIDTH);
    anim.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
        @Override
        public void onAnimationUpdate(ValueAnimator valueAnimator) {
            int val = (Integer) valueAnimator.getAnimatedValue();
            ViewGroup.LayoutParams layoutParams = v.getLayoutParams();
            layoutParams.width = val;
            v.setLayoutParams(layoutParams);
        }
    });
    anim.setDuration(DURATION);
    anim.start(); 

Solution 2 - Android

For what it's worth this works. After you change the width on the layout params you have to reset the layout params object.

private class WidthEvaluator extends IntEvaluator {
	
	private View v;
	public WidthEvaluator(View v) {
		this.v = v;
	}
	
	@Override
	public Object evaluate(float fraction, Object startValue, Object endValue) {
		int num = (Integer)super.evaluate(fraction, startValue, endValue);
		ViewGroup.LayoutParams params = v.getLayoutParams();
		params.width = num;
		v.setLayoutParams(params);
		return num;
	}
}

// your client code
ValueAnimator.ofObject(new WidthEvaluator(box), box.getWidth(), v.getWidth()).start();

Solution 3 - Android

I've created a small library ViewPropertyObjectAnimator that can do that in a very simple way (and uses a similar approach to this proposed by Tomer Weller).

You could achieve this animation with (assuming the mLinearLayout is the animated View and mEndWidth is the end value of the View's width):

ViewPropertyObjectAnimator.animate(mLinearLayout).width(mEndWidth).start();

Solution 4 - Android

You have made a mistake

 params.weight = myWidth;

I think it is params.width not params.weight

Solution 5 - Android

I think a better way to accomplish this would be to use View's scaleX and scaleY property which are defined all the way down in View class, so it would be valid with practically any view or layout.

For example, consider this

ObjectAnimator objectAnimator=ObjectAnimator.ofFloat(MyLinearLayout,"scaleX",0f,1f);
objectAnimator.setDuration(300);
objectAnimator.start();

It works.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestioneyahnView Question on Stackoverflow
Solution 1 - AndroidTomer WellerView Answer on Stackoverflow
Solution 2 - Androiduser123321View Answer on Stackoverflow
Solution 3 - AndroidBartek LipinskiView Answer on Stackoverflow
Solution 4 - AndroidRead MarkView Answer on Stackoverflow
Solution 5 - AndroidAmit GuptaView Answer on Stackoverflow