Android半视图展开/折叠动画

3
我想折叠/展开ImageView,但是从50%的图片开始展开到100%,折叠到50%而不是以下。
我已经查看了一些在SO上流行的问题和答案,但是我没有找到如何仅管理一半的方法。我还想修改视图的高度,而不是宽度。
我尝试过的方法:
public static void expand(final View v) {
    v.measure(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
    final int targtetHeight = v.getMeasuredHeight();

    v.getLayoutParams().height = 0;
    v.setVisibility(View.VISIBLE);
    Animation a = new Animation()
    {
        @Override
        protected void applyTransformation(float interpolatedTime, Transformation t) {
            v.getLayoutParams().height = interpolatedTime == 1
                    ? ViewGroup.LayoutParams.WRAP_CONTENT
                    : (int)(targtetHeight * interpolatedTime);
            v.requestLayout();
        }

        @Override
        public boolean willChangeBounds() {
            return true;
        }
    };

    a.setDuration((int)(targtetHeight / v.getContext().getResources().getDisplayMetrics().density));
    v.startAnimation(a);
}

public static void collapse(final View v) {
    final int initialHeight = v.getMeasuredHeight();

    Animation a = new Animation() {
        @Override
        protected void applyTransformation(float interpolatedTime, Transformation t) {
            if (interpolatedTime == 1) {
                v.setVisibility(View.VISIBLE);
            } else {
                v.getLayoutParams().height = initialHeight - (int) (initialHeight * interpolatedTime);
                v.requestLayout();
            }
        }

        @Override
        public boolean willChangeBounds() {
            return true;
        }
    };

    a.setDuration((int) (initialHeight / v.getContext().getResources().getDisplayMetrics().density));
    v.startAnimation(a);
}

作为我之前所说的,这不是我想要的,因为它会完全消失并且改变宽度。
我也尝试了这个代码片段,但没有动画效果:
mImageDrawable = (ClipDrawable) pic.getDrawable();
mImageDrawable.setLevel(5000);//use set level to expand or collapse manually but no animation.

clip:

    <?xml version="1.0" encoding="utf-8"?>
<clip xmlns:android="http://schemas.android.com/apk/res/android"
    android:clipOrientation="vertical"
    android:drawable="@drawable/test_pic"
    android:gravity="top" />
1个回答

6
使用支持包(androidx)中提供的转换API。只需调用TransitionManager.beginDelayedTransition,然后更改视图的高度即可。 TransitionManager将处理这些更改,并提供转换,该转换将通过动画更改imageView。

enter image description here

这里的ImageViewscaleTypecenterCrop,所以在折叠和展开时图像会缩放。不幸的是,没有“填充宽度并裁剪底部”的scaleType,所以如果您需要它,我认为可以通过scaleType = matrix实现。
import androidx.appcompat.app.AppCompatActivity;
import androidx.transition.TransitionManager;

public class MainActivity extends AppCompatActivity {

  private ImageView image;
  private ViewGroup parent;
  boolean collapse = true;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    image = findViewById(R.id.image);
    parent = findViewById(R.id.parent);

    findViewById(R.id.btn).setOnClickListener(view -> {
        collapse = !collapse;
        collapse();
    });
  }

  private void collapse() {
    TransitionManager.beginDelayedTransition(parent);
    //change layout params
    int height = image.getHeight();
    LayoutParams layoutParams = image.getLayoutParams();
    layoutParams.height = !collapse ? height / 2 : height * 2;
    image.requestLayout();
  }
}

布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/parent"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">

<Button
    android:id="@+id/btn"
    android:text="start"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content" />

<ImageView
    android:id="@+id/image"
    android:layout_width="match_parent"
    android:layout_height="300dp"
    android:scaleType="centerCrop"
    android:src="@drawable/qwe" />

<TextView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:text="random text"
    android:layout_margin="8dp"/>
</LinearLayout>

更新:

现在有beginDelayedTransition(ViewGroup, Transition)方法。默认情况下,beginDelayedTransition(ViewGroup)使用AutoTransition作为过渡效果。 因此,如果您需要处理过渡的开始/结束,可以按照以下方式操作:

    AutoTransition transition = new AutoTransition();
    transition.addListener(new TransitionListenerAdapter(){
        @Override
        public void onTransitionStart(@NonNull Transition transition) {
            //TODO
        }
        @Override
        public void onTransitionEnd(@NonNull Transition transition) {
            //TODO 
        }
    });
    TransitionManager.beginDelayedTransition(parent, transition);

我也想要这种缩放效果!所以你的代码也可以,而且正是我想要的! :) - undefined
我有一个与这个转换相关的问题。我该如何知道动画/过渡何时完成?我需要在过渡/动画完成后立即执行一些操作。 - undefined
工作得非常好!感谢您如此快速地更新您的答案! - undefined

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接