android 大图预览,android 大图预览

导入moudle ,

implementation 'com.ycjiang:imgepreviewlibrary:1.1.3'

然后在图片点击事件添加以下代码

Intent intent = new Intent(WXJiLuDetailActivity.this, EnlargeImageDetailActivity.class);

intent.putExtra(CommonUtils.SpaceImageDetail.IMAGEURL, HttpConfig.IMG_URL + "/" + imglist.get(position));//传入 图片的URL

int[] location = new int[2];

view.getLocationOnScreen(location);//view 就是所点击的ImageView

intent.putExtra(CommonUtils.SpaceImageDetail.LOCATIONX, location[0]);

intent.putExtra(CommonUtils.SpaceImageDetail.LOCATIONY, location[1]);

intent.putExtra(CommonUtils.SpaceImageDetail.WIDTH, view.getWidth());

intent.putExtra(CommonUtils.SpaceImageDetail.HEIGHT, view.getHeight());

startActivity(intent);

overridePendingTransition(0, 0);

EnlargeImageView

public class EnlargeImageView extends ImageView {

private static final int STATE_NORMAL = 0;

private static final int STATE_TRANSFORM_IN = 1;

private static final int STATE_TRANSFORM_OUT = 2;

private int mOriginalWidth;

private int mOriginalHeight;

private int mOriginalLocationX;

private int mOriginalLocationY;

private int mState = STATE_NORMAL;

private Matrix mSmoothMatrix;

private Bitmap mBitmap;

private boolean mTransformStart = false;

private Transfrom mTransfrom;

private final int mBgColor = 0xFF000000;

private int mBgAlpha = 0;

private Paint mPaint;

public EnlargeImageView(Context context) {

super(context);

init();

}

public EnlargeImageView(Context context, AttributeSet attrs) {

super(context, attrs);

init();

}

public EnlargeImageView(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

init();

}

private void init() {

mSmoothMatrix = new Matrix();

mPaint=new Paint();

mPaint.setColor(mBgColor);

mPaint.setStyle(Style.FILL);

// setBackgroundColor(mBgColor);

}

public void setOriginalInfo(int width, int height, int locationX, int locationY) {

mOriginalWidth = width;

mOriginalHeight = height;

mOriginalLocationX = locationX;

mOriginalLocationY = locationY;

// 因为是屏幕坐标,所以要转换为该视图内的坐标,因为我所用的该视图是MATCH_PARENT,所以不用定位该视图的位置,如果不是的话,还需要定位视图的位置,然后计算mOriginalLocationX和mOriginalLocationY

mOriginalLocationY = mOriginalLocationY - getStatusBarHeight(getContext());

}

/**

* 获取状态栏高度

*

* @return

*/

public static int getStatusBarHeight(Context context) {

Class> c = null;

Object obj = null;

java.lang.reflect.Field field = null;

int x = 0;

int statusBarHeight = 0;

try {

c = Class.forName("com.android.internal.R$dimen");

obj = c.newInstance();

field = c.getField("status_bar_height");

x = Integer.parseInt(field.get(obj).toString());

statusBarHeight = context.getResources().getDimensionPixelSize(x);

return statusBarHeight;

} catch (Exception e) {

e.printStackTrace();

}

return statusBarHeight;

}

/**

* 用于开始进入的方法。 调用此方前,需已经调用过setOriginalInfo

*/

public void transformIn() {

mState = STATE_TRANSFORM_IN;

mTransformStart = true;

invalidate();

}

/**

* 用于开始退出的方法。 调用此方前,需已经调用过setOriginalInfo

*/

public void transformOut() {

mState = STATE_TRANSFORM_OUT;

mTransformStart = true;

invalidate();

}

private class Transfrom {

float startScale;// 图片开始的缩放值

float endScale;// 图片结束的缩放值

float scale;// 属性ValueAnimator计算出来的值

LocationSizeF startRect;// 开始的区域

LocationSizeF endRect;// 结束的区域

LocationSizeF rect;// 属性ValueAnimator计算出来的值

void initStartIn() {

scale = startScale;

try {

rect = (LocationSizeF) startRect.clone();

} catch (CloneNotSupportedException e) {

e.printStackTrace();

}

}

void initStartOut() {

scale = endScale;

try {

rect = (LocationSizeF) endRect.clone();

} catch (CloneNotSupportedException e) {

e.printStackTrace();

}

}

}

/**

* 初始化进入的变量信息

*/

private void initTransform() {

if (getDrawable() == null) {

return;

}

if (mBitmap == null || mBitmap.isRecycled()) {

mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();

}

//防止mTransfrom重复的做同样的初始化

if (mTransfrom != null) {

return;

}

if (getWidth() == 0 || getHeight() == 0) {

return;

}

mTransfrom = new Transfrom();

/** 下面为缩放的计算 */

/* 计算初始的缩放值,初始值因为是CENTR_CROP效果,所以要保证图片的宽和高至少1个能匹配原始的宽和高,另1个大于 */

float xSScale = mOriginalWidth / ((float) mBitmap.getWidth());

float ySScale = mOriginalHeight / ((float) mBitmap.getHeight());

float startScale = xSScale > ySScale ? xSScale : ySScale;

mTransfrom.startScale = startScale;

/* 计算结束时候的缩放值,结束值因为要达到FIT_CENTER效果,所以要保证图片的宽和高至少1个能匹配原始的宽和高,另1个小于 */

float xEScale = getWidth() / ((float) mBitmap.getWidth());

float yEScale = getHeight() / ((float) mBitmap.getHeight());

float endScale = xEScale < yEScale ? xEScale : yEScale;

mTransfrom.endScale = endScale;

/**

* 下面计算Canvas Clip的范围,也就是图片的显示的范围,因为图片是慢慢变大,并且是等比例的,所以这个效果还需要裁减图片显示的区域

* ,而显示区域的变化范围是在原始CENTER_CROP效果的范围区域

* ,到最终的FIT_CENTER的范围之间的,区域我用LocationSizeF更好计算

* ,他就包括左上顶点坐标,和宽高,最后转为Canvas裁减的Rect.

*/

/* 开始区域 */

mTransfrom.startRect = new LocationSizeF();

mTransfrom.startRect.left = mOriginalLocationX;

mTransfrom.startRect.top = mOriginalLocationY;

mTransfrom.startRect.width = mOriginalWidth;

mTransfrom.startRect.height = mOriginalHeight;

/* 结束区域 */

mTransfrom.endRect = new LocationSizeF();

float bitmapEndWidth = mBitmap.getWidth() * mTransfrom.endScale;// 图片最终的宽度

float bitmapEndHeight = mBitmap.getHeight() * mTransfrom.endScale;// 图片最终的宽度

mTransfrom.endRect.left = (getWidth() - bitmapEndWidth) / 2;

mTransfrom.endRect.top = (getHeight() - bitmapEndHeight) / 2;

mTransfrom.endRect.width = bitmapEndWidth;

mTransfrom.endRect.height = bitmapEndHeight;

mTransfrom.rect = new LocationSizeF();

}

private class LocationSizeF implements Cloneable {

float left;

float top;

float width;

float height;

@Override

public String toString() {

return "[left:"+left+" top:"+top+" width:"+width+" height:"+height+"]";

}

@Override

public Object clone() throws CloneNotSupportedException {

// TODO Auto-generated method stub

return super.clone();

}

}

/* 下面实现了CENTER_CROP的功能 的Matrix,在优化的过程中,已经不用了 */

private void getCenterCropMatrix() {

if (getDrawable() == null) {

return;

}

if (mBitmap == null || mBitmap.isRecycled()) {

mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();

}

/* 下面实现了CENTER_CROP的功能 */

float xScale = mOriginalWidth / ((float) mBitmap.getWidth());

float yScale = mOriginalHeight / ((float) mBitmap.getHeight());

float scale = xScale > yScale ? xScale : yScale;

mSmoothMatrix.reset();

mSmoothMatrix.setScale(scale, scale);

mSmoothMatrix.postTranslate(-(scale * mBitmap.getWidth() / 2 - mOriginalWidth / 2), -(scale * mBitmap.getHeight() / 2 - mOriginalHeight / 2));

}

private void getBmpMatrix() {

if (getDrawable() == null) {

return;

}

if (mTransfrom == null) {

return;

}

if (mBitmap == null || mBitmap.isRecycled()) {

mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();

}

/* 下面实现了CENTER_CROP的功能 */

mSmoothMatrix.setScale(mTransfrom.scale, mTransfrom.scale);

mSmoothMatrix.postTranslate(-(mTransfrom.scale * mBitmap.getWidth() / 2 - mTransfrom.rect.width / 2),

-(mTransfrom.scale * mBitmap.getHeight() / 2 - mTransfrom.rect.height / 2));

}

@Override

protected void onDraw(Canvas canvas) {

if (getDrawable() == null) {

return; // couldn't resolve the URI

}

if (mState == STATE_TRANSFORM_IN || mState == STATE_TRANSFORM_OUT) {

if (mTransformStart) {

initTransform();

}

if (mTransfrom == null) {

super.onDraw(canvas);

return;

}

if (mTransformStart) {

if (mState == STATE_TRANSFORM_IN) {

mTransfrom.initStartIn();

} else {

mTransfrom.initStartOut();

}

}

if(mTransformStart){

Log.d("Dean", "mTransfrom.startScale:"+mTransfrom.startScale);

Log.d("Dean", "mTransfrom.startScale:"+mTransfrom.endScale);

Log.d("Dean", "mTransfrom.scale:"+mTransfrom.scale);

Log.d("Dean", "mTransfrom.startRect:"+mTransfrom.startRect.toString());

Log.d("Dean", "mTransfrom.endRect:"+mTransfrom.endRect.toString());

Log.d("Dean", "mTransfrom.rect:"+mTransfrom.rect.toString());

}

mPaint.setAlpha(mBgAlpha);

canvas.drawPaint(mPaint);

int saveCount = canvas.getSaveCount();

canvas.save();

// 先得到图片在此刻的图像Matrix矩阵

getBmpMatrix();

canvas.translate(mTransfrom.rect.left, mTransfrom.rect.top);

canvas.clipRect(0, 0, mTransfrom.rect.width, mTransfrom.rect.height);

canvas.concat(mSmoothMatrix);

getDrawable().draw(canvas);

canvas.restoreToCount(saveCount);

if (mTransformStart) {

mTransformStart=false;

startTransform(mState);

}

} else {

//当Transform In变化完成后,把背景改为黑色,使得Activity不透明

mPaint.setAlpha(255);

canvas.drawPaint(mPaint);

super.onDraw(canvas);

}

}

private void startTransform(final int state) {

if (mTransfrom == null) {

return;

}

ValueAnimator valueAnimator = new ValueAnimator();

valueAnimator.setDuration(300);

valueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());

if (state == STATE_TRANSFORM_IN) {

PropertyValuesHolder scaleHolder = PropertyValuesHolder.ofFloat("scale", mTransfrom.startScale, mTransfrom.endScale);

PropertyValuesHolder leftHolder = PropertyValuesHolder.ofFloat("left", mTransfrom.startRect.left, mTransfrom.endRect.left);

PropertyValuesHolder topHolder = PropertyValuesHolder.ofFloat("top", mTransfrom.startRect.top, mTransfrom.endRect.top);

PropertyValuesHolder widthHolder = PropertyValuesHolder.ofFloat("width", mTransfrom.startRect.width, mTransfrom.endRect.width);

PropertyValuesHolder heightHolder = PropertyValuesHolder.ofFloat("height", mTransfrom.startRect.height, mTransfrom.endRect.height);

PropertyValuesHolder alphaHolder = PropertyValuesHolder.ofInt("alpha", 0, 255);

valueAnimator.setValues(scaleHolder, leftHolder, topHolder, widthHolder, heightHolder, alphaHolder);

} else {

PropertyValuesHolder scaleHolder = PropertyValuesHolder.ofFloat("scale", mTransfrom.endScale, mTransfrom.startScale);

PropertyValuesHolder leftHolder = PropertyValuesHolder.ofFloat("left", mTransfrom.endRect.left, mTransfrom.startRect.left);

PropertyValuesHolder topHolder = PropertyValuesHolder.ofFloat("top", mTransfrom.endRect.top, mTransfrom.startRect.top);

PropertyValuesHolder widthHolder = PropertyValuesHolder.ofFloat("width", mTransfrom.endRect.width, mTransfrom.startRect.width);

PropertyValuesHolder heightHolder = PropertyValuesHolder.ofFloat("height", mTransfrom.endRect.height, mTransfrom.startRect.height);

PropertyValuesHolder alphaHolder = PropertyValuesHolder.ofInt("alpha", 255, 0);

valueAnimator.setValues(scaleHolder, leftHolder, topHolder, widthHolder, heightHolder, alphaHolder);

}

valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {

@Override

public synchronized void onAnimationUpdate(ValueAnimator animation) {

mTransfrom.scale = (Float) animation.getAnimatedValue("scale");

mTransfrom.rect.left = (Float) animation.getAnimatedValue("left");

mTransfrom.rect.top = (Float) animation.getAnimatedValue("top");

mTransfrom.rect.width = (Float) animation.getAnimatedValue("width");

mTransfrom.rect.height = (Float) animation.getAnimatedValue("height");

mBgAlpha = (Integer) animation.getAnimatedValue("alpha");

invalidate();

((Activity)getContext()).getWindow().getDecorView().invalidate();

}

});

valueAnimator.addListener(new ValueAnimator.AnimatorListener() {

@Override

public void onAnimationStart(Animator animation) {

}

@Override

public void onAnimationRepeat(Animator animation) {

}

@Override

public void onAnimationEnd(Animator animation) {

/*

* 如果是进入的话,当然是希望最后停留在center_crop的区域。但是如果是out的话,就不应该是center_crop的位置了

* , 而应该是最后变化的位置,因为当out的时候结束时,不回复视图是Normal,要不然会有一个突然闪动回去的bug

*/

// TODO 这个可以根据实际需求来修改

if (state == STATE_TRANSFORM_IN) {

mState = STATE_NORMAL;

}

if (mTransformListener != null) {

mTransformListener.onTransformComplete(state);

}

}

@Override

public void onAnimationCancel(Animator animation) {

}

});

valueAnimator.start();

}

public void setOnTransformListener(TransformListener listener) {

mTransformListener = listener;

}

private TransformListener mTransformListener;

public static interface TransformListener {

/**

*

* @param mode

* STATE_TRANSFORM_IN 1 ,STATE_TRANSFORM_OUT 2

*/

void onTransformComplete(int mode);// mode 1

}

}

package com.luchongbin.enlarge.myenlarge;

import android.app.Activity;

import android.content.Intent;

import android.os.Bundle;

import android.view.View;

import android.view.ViewGroup;

import android.widget.ImageView.ScaleType;

import com.bumptech.glide.Glide;

/**

* Created by luchongbin on 2018/11/5/005.

*/

public class EnlargeImageDetailActivity extends Activity implements View.OnClickListener, EnlargeImageView.TransformListener {

private EnlargeImageView imageView = null;

private String imageUrl;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

Intent mIntent = getIntent();

imageUrl = mIntent.getStringExtra(CommonUtils.SpaceImageDetail.IMAGEURL);

int mLocationX = mIntent.getIntExtra(CommonUtils.SpaceImageDetail.LOCATIONX, 0);

int mLocationY = mIntent.getIntExtra(CommonUtils.SpaceImageDetail.LOCATIONY, 0);

int mWidth = mIntent.getIntExtra(CommonUtils.SpaceImageDetail.WIDTH, 0);

int mHeight = mIntent.getIntExtra(CommonUtils.SpaceImageDetail.HEIGHT, 0);

imageView = new EnlargeImageView(this);

imageView.setOriginalInfo(mWidth, mHeight, mLocationX, mLocationY);

imageView.transformIn();

imageView.setLayoutParams(new ViewGroup.LayoutParams(-1, -1));

imageView.setScaleType(ScaleType.FIT_CENTER);

setContentView(imageView);

initData(imageUrl);

setListener();

}

private void initData(String imageUrl){

Glide.with(this)

.load(imageUrl)

.into(imageView);

}

private void setListener(){

imageView.setOnClickListener(this);

}

@Override

public void onClick(View view) {

if(view.getId()==imageView.getId()){

onBackPressed();

}

}

@Override

public void onTransformComplete(int mode) {

if (mode == 2) {

finish();

}

}

@Override

public void onBackPressed() {

imageView.setOnTransformListener(this);

imageView.transformOut();

}

@Override

protected void onPause() {

super.onPause();

if (isFinishing()) {

overridePendingTransition(0, 0);

}

}

}

public class CommonUtils {

public static class SpaceImageDetail{

public static final String IMAGEURL = "imageUrl";

public static final String LOCATIONX = "locationX";

public static final String LOCATIONY = "locationY";

public static final String WIDTH = "width";

public static final String HEIGHT = "height";

}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值