Android中用Matrix实现ImageView里的图片平移和缩放动画

  注: 这里说的图片的平移和缩放不是对ImageView整个view进行的,而是对ImageView里面的图片进行的(view本身没有什么变化),所以Android自带的动画效果不能满足需求。

功能点
1、一开始可以像centerCrop一样显示图片(觉得scaleType为centerCrop时显示效果比较好,图片会铺满整个View,而且图片本身的分辨率不变)
2、对ImageView里的图片平移或放大
3、将这个平移或放大做成一个动画效果

一、在ScaleType为matrix的情况下实现centerCrop的显示效果

要充分利用Android的源码,看ImageView的源代码就可以简单很多,在configureBounds()方法中:
Android中用Matrix实现ImageView里的图片平移和缩放动画 - Johnny - Lucky Johnnys blog
 不过为了之后的平移,要保证图片的宽度比View的宽度大

public void setImage ( Drawable drawable ) {

if ( null == drawable ) return;

int dwidth = drawable.getIntrinsicWidth ();

int dheight = drawable.getIntrinsicHeight ();

int vwidth = mWidth;

int vheight = mHeight;

float scale = 1.0f;

float dx = 0, dy = 0;

if ( dwidth * vheight > vwidth * dheight ) {

scale = ( float ) vheight / ( float ) dheight;

dx = ( vwidth - dwidth * scale ) * 0.5f + 0.5f;

} else {

scale = ( float ) vwidth / ( float ) dwidth;

dy = ( vheight - dheight * scale ) * 0.5f + 0.5f;

}

if ( dwidth * scale < 2 * vwidth ) {

scale = ( float ) vwidth / ( 2.0f * dwidth );

dx = - dwidth / 2;

dy = ( vheight - dheight * scale ) * 0.5f + 0.5f;

}

Matrix matrix = new Matrix ();

matrix.setScale ( scale, scale );

matrix.postTranslate ( dx, dy );

mImage.setImageMatrix ( matrix );

mImage.setImageDrawable ( drawable );

}

// vwidth和vheight使用确定的值,因为如果在应用初始化时view.getWidth()可能为0


二、利用Matrix实现平移和放大

平移和放大都是在当前图片的基础上,先用mImage.getImageMatrix()得到当前的matrix,再用Matrix的postTranslate或postScale方法就可以了。
需要注意的是:
如果用Matrix matrix  = mImage.getImageMatrix ();  //matrix只是得到一个对象的引用
应该用Matrix matrix = new Matrix ( mImage.getImageMatrix () );//这样才是得到一个克隆对象

三、用ValueAnimator实现动画效果

1、平移

dx为负是向左平移,为正是向右平移
因为图片的宽度设置为至少是view的2倍,所以向左和向右平移的最大距离都是vwidth / 2。

private class MyTransXAnimatorListener implements AnimatorUpdateListener {

private Matrix mPrimaryMatrix;

public MyTransXAnimatorListener ( Matrix matrix ) {

mPrimaryMatrix = new Matrix ( matrix );

}

@Override

public void onAnimationUpdate ( ValueAnimator animation ) {

int dx = ( Integer ) animation.getAnimatedValue ();

Matrix matrix = new Matrix ( mPrimaryMatrix );

matrix.postTranslate ( dx, 0 );

mImage.setImageMatrix ( matrix );

}

}

public void setTranslateAnimation () {

ValueAnimator animator = ValueAnimator.ofInt ( 0, - 60 );

animator.addUpdateListener ( new MyTransXAnimatorListener ( mImage.getImageMatrix() ) );

animator.setDuration ( 1000 );

animator.setInterpolator ( new DecelerateInterpolator () );

animator.setStartDelay ( 500 );

animator.start ();

}


2、放大

放大要设置中心点为ImageView的中心

private class MyScaleAnimatorListener implements AnimatorUpdateListener {

private Matrix mPrimaryMatrix;

public MyScaleAnimatorListener ( Matrix matrix ) {

mPrimaryMatrix = matrix;

}

@Override

public void onAnimationUpdate ( ValueAnimator animation ) {

float scale = ( Float ) animation.getAnimatedValue ();

Matrix matrix = new Matrix ( mPrimaryMatrix );

matrix.postScale ( scale, scale, mWidth / 2, mHeight / 2 );

mImage.setImageMatrix ( matrix );

}

}

public void setScaleAnimation () {

ValueAnimator animator = ValueAnimator.ofFloat ( 1.0f, 1.2f );

animator.addUpdateListener ( new MyScaleAnimatorListener ( mImage.getImageMatrix () ) );

animator.setDuration ( 1000 );

animator.setInterpolator ( new DecelerateInterpolator () );

animator.setStartDelay ( 500 );

animator.start ();

}


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值