Android瀏覽圖片,點擊放大至全屏效果

最近做一個項目類似於QQ空間,做到照片瀏覽的功能,對於QQ空間中點擊圖片放大至全屏,感覺效果很贊,於是也做了個類似的效果。如下。



我不知道QQ那個是怎麼做的,我的思路如下:

首先,從圖片縮略界面跳轉到圖片詳情頁面,應該是從一個Activity跳轉到另外一個Activity,應該圖片詳情頁面也有很多操作,用View或者Dialog不是很好。所以現在難點就是,如何使得前一個界面的ImageView在另外一個界面做縮放切割動畫。

一般縮略界面的ImageView的是如上圖所示的正方形的,並且是CENTER_CROP縮放屬性的。CENTER_CROP屬性會導致ImageView中顯示的Bitmap有被切割達到填充的效果。

而詳情頁面的ImageView一般都是FIT_CENTER的縮放屬性。所以要保證這個跳轉動畫的流暢,要做如下的變化:

1、Bitmap的縮放,因爲縮略圖和詳情圖的縮放比例肯定不一樣

2、Bitmap位置的平移,因爲縮略圖的位置是不確定的,我們要使他平移到中間

3、Bitmap的切割,因爲CENTER_CROP是切割過得,而FIT_CENTER是沒有切割的,那麼兩幅圖顯示的內容區域是不同的,所以也要顯示區域的平滑變換。


要完成上面的效果,如果單單是指對ImageView做一個動畫變換,我覺得是完成不了這個要求的。所以自己重寫了ImageView來完成上述的變換。

直接貼上主要的ImageView

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
  1. package com.roamer.ui.view;  
  2.   
  3. import android.animation.Animator;  
  4. import android.animation.PropertyValuesHolder;  
  5. import android.animation.ValueAnimator;  
  6. import android.app.Activity;  
  7. import android.content.Context;  
  8. import android.graphics.Bitmap;  
  9. import android.graphics.Canvas;  
  10. import android.graphics.Matrix;  
  11. import android.graphics.Paint;  
  12. import android.graphics.Paint.Style;  
  13. import android.graphics.drawable.BitmapDrawable;  
  14. import android.util.AttributeSet;  
  15. import android.util.Log;  
  16. import android.view.animation.AccelerateDecelerateInterpolator;  
  17. import android.widget.ImageView;  
  18.   
  19. /** 
  20.  * 2d平滑變化的顯示圖片的ImageView 
  21.  * 僅限於用於:從一個ScaleType==CENTER_CROP的ImageView,切換到另一個ScaleType= 
  22.  * FIT_CENTER的ImageView,或者反之 (當然,得使用同樣的圖片最好) 
  23.  *  
  24.  * @author Dean Tao 
  25.  *  
  26.  */  
  27. public class SmoothImageView extends ImageView {  
  28.   
  29.     private static final int STATE_NORMAL = 0;  
  30.     private static final int STATE_TRANSFORM_IN = 1;  
  31.     private static final int STATE_TRANSFORM_OUT = 2;  
  32.     private int mOriginalWidth;  
  33.     private int mOriginalHeight;  
  34.     private int mOriginalLocationX;  
  35.     private int mOriginalLocationY;  
  36.     private int mState = STATE_NORMAL;  
  37.     private Matrix mSmoothMatrix;  
  38.     private Bitmap mBitmap;  
  39.     private boolean mTransformStart = false;  
  40.     private Transfrom mTransfrom;  
  41.     private final int mBgColor = 0xFF000000;  
  42.     private int mBgAlpha = 0;  
  43.     private Paint mPaint;  
  44.       
  45.     public SmoothImageView(Context context) {  
  46.         super(context);  
  47.         init();  
  48.     }  
  49.   
  50.     public SmoothImageView(Context context, AttributeSet attrs) {  
  51.         super(context, attrs);  
  52.         init();  
  53.     }  
  54.   
  55.     public SmoothImageView(Context context, AttributeSet attrs, int defStyle) {  
  56.         super(context, attrs, defStyle);  
  57.         init();  
  58.     }  
  59.   
  60.     private void init() {  
  61.         mSmoothMatrix = new Matrix();  
  62.         mPaint=new Paint();  
  63.         mPaint.setColor(mBgColor);  
  64.         mPaint.setStyle(Style.FILL);  
  65. //      setBackgroundColor(mBgColor);  
  66.     }  
  67.   
  68.     public void setOriginalInfo(int width, int height, int locationX, int locationY) {  
  69.         mOriginalWidth = width;  
  70.         mOriginalHeight = height;  
  71.         mOriginalLocationX = locationX;  
  72.         mOriginalLocationY = locationY;  
  73.         // 因爲是屏幕座標,所以要轉換爲該視圖內的座標,因爲我所用的該視圖是MATCH_PARENT,所以不用定位該視圖的位置,如果不是的話,還需要定位視圖的位置,然後計算mOriginalLocationX和mOriginalLocationY  
  74.         mOriginalLocationY = mOriginalLocationY - getStatusBarHeight(getContext());  
  75.     }  
  76.   
  77.     /** 
  78.      * 獲取狀態欄高度 
  79.      *  
  80.      * @return 
  81.      */  
  82.     public static int getStatusBarHeight(Context context) {  
  83.         Class<?> c = null;  
  84.         Object obj = null;  
  85.         java.lang.reflect.Field field = null;  
  86.         int x = 0;  
  87.         int statusBarHeight = 0;  
  88.         try {  
  89.             c = Class.forName("com.android.internal.R$dimen");  
  90.             obj = c.newInstance();  
  91.             field = c.getField("status_bar_height");  
  92.             x = Integer.parseInt(field.get(obj).toString());  
  93.             statusBarHeight = context.getResources().getDimensionPixelSize(x);  
  94.             return statusBarHeight;  
  95.         } catch (Exception e) {  
  96.             e.printStackTrace();  
  97.         }  
  98.         return statusBarHeight;  
  99.     }  
  100.   
  101.     /** 
  102.      * 用於開始進入的方法。 調用此方前,需已經調用過setOriginalInfo 
  103.      */  
  104.     public void transformIn() {  
  105.         mState = STATE_TRANSFORM_IN;  
  106.         mTransformStart = true;  
  107.         invalidate();  
  108.     }  
  109.   
  110.     /** 
  111.      * 用於開始退出的方法。 調用此方前,需已經調用過setOriginalInfo 
  112.      */  
  113.     public void transformOut() {  
  114.         mState = STATE_TRANSFORM_OUT;  
  115.         mTransformStart = true;  
  116.         invalidate();  
  117.     }  
  118.   
  119.     private class Transfrom {  
  120.         float startScale;// 圖片開始的縮放值  
  121.         float endScale;// 圖片結束的縮放值  
  122.         float scale;// 屬性ValueAnimator計算出來的值  
  123.         LocationSizeF startRect;// 開始的區域  
  124.         LocationSizeF endRect;// 結束的區域  
  125.         LocationSizeF rect;// 屬性ValueAnimator計算出來的值  
  126.   
  127.         void initStartIn() {  
  128.             scale = startScale;  
  129.             try {  
  130.                 rect = (LocationSizeF) startRect.clone();  
  131.             } catch (CloneNotSupportedException e) {  
  132.                 e.printStackTrace();  
  133.             }  
  134.         }  
  135.   
  136.         void initStartOut() {  
  137.             scale = endScale;  
  138.             try {  
  139.                 rect = (LocationSizeF) endRect.clone();  
  140.             } catch (CloneNotSupportedException e) {  
  141.                 e.printStackTrace();  
  142.             }  
  143.         }  
  144.           
  145.     }  
  146.   
  147.     /** 
  148.      * 初始化進入的變量信息 
  149.      */  
  150.     private void initTransform() {  
  151.         if (getDrawable() == null) {  
  152.             return;  
  153.         }  
  154.         if (mBitmap == null || mBitmap.isRecycled()) {  
  155.             mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();  
  156.         }  
  157.         //防止mTransfrom重複的做同樣的初始化  
  158.         if (mTransfrom != null) {  
  159.             return;  
  160.         }  
  161.         if (getWidth() == 0 || getHeight() == 0) {  
  162.             return;  
  163.         }  
  164.         mTransfrom = new Transfrom();  
  165.   
  166.         /** 下面爲縮放的計算 */  
  167.         /* 計算初始的縮放值,初始值因爲是CENTR_CROP效果,所以要保證圖片的寬和高至少1個能匹配原始的寬和高,另1個大於 */  
  168.         float xSScale = mOriginalWidth / ((float) mBitmap.getWidth());  
  169.         float ySScale = mOriginalHeight / ((float) mBitmap.getHeight());  
  170.         float startScale = xSScale > ySScale ? xSScale : ySScale;  
  171.         mTransfrom.startScale = startScale;  
  172.         /* 計算結束時候的縮放值,結束值因爲要達到FIT_CENTER效果,所以要保證圖片的寬和高至少1個能匹配原始的寬和高,另1個小於 */  
  173.         float xEScale = getWidth() / ((float) mBitmap.getWidth());  
  174.         float yEScale = getHeight() / ((float) mBitmap.getHeight());  
  175.         float endScale = xEScale < yEScale ? xEScale : yEScale;  
  176.         mTransfrom.endScale = endScale;  
  177.   
  178.         /** 
  179.          * 下面計算Canvas Clip的範圍,也就是圖片的顯示的範圍,因爲圖片是慢慢變大,並且是等比例的,所以這個效果還需要裁減圖片顯示的區域 
  180.          * ,而顯示區域的變化範圍是在原始CENTER_CROP效果的範圍區域 
  181.          * ,到最終的FIT_CENTER的範圍之間的,區域我用LocationSizeF更好計算 
  182.          * ,他就包括左上頂點座標,和寬高,最後轉爲Canvas裁減的Rect. 
  183.          */  
  184.         /* 開始區域 */  
  185.         mTransfrom.startRect = new LocationSizeF();  
  186.         mTransfrom.startRect.left = mOriginalLocationX;  
  187.         mTransfrom.startRect.top = mOriginalLocationY;  
  188.         mTransfrom.startRect.width = mOriginalWidth;  
  189.         mTransfrom.startRect.height = mOriginalHeight;  
  190.         /* 結束區域 */  
  191.         mTransfrom.endRect = new LocationSizeF();  
  192.         float bitmapEndWidth = mBitmap.getWidth() * mTransfrom.endScale;// 圖片最終的寬度  
  193.         float bitmapEndHeight = mBitmap.getHeight() * mTransfrom.endScale;// 圖片最終的寬度  
  194.         mTransfrom.endRect.left = (getWidth() - bitmapEndWidth) / 2;  
  195.         mTransfrom.endRect.top = (getHeight() - bitmapEndHeight) / 2;  
  196.         mTransfrom.endRect.width = bitmapEndWidth;  
  197.         mTransfrom.endRect.height = bitmapEndHeight;  
  198.   
  199.         mTransfrom.rect = new LocationSizeF();  
  200.     }  
  201.   
  202.     private class LocationSizeF implements Cloneable{  
  203.         float left;  
  204.         float top;  
  205.         float width;  
  206.         float height;  
  207.         @Override  
  208.         public String toString() {  
  209.             return "[left:"+left+" top:"+top+" width:"+width+" height:"+height+"]";  
  210.         }  
  211.           
  212.         @Override  
  213.         public Object clone() throws CloneNotSupportedException {  
  214.             // TODO Auto-generated method stub  
  215.             return super.clone();  
  216.         }  
  217.           
  218.     }  
  219.   
  220.     /* 下面實現了CENTER_CROP的功能 的Matrix,在優化的過程中,已經不用了 */  
  221.     private void getCenterCropMatrix() {  
  222.         if (getDrawable() == null) {  
  223.             return;  
  224.         }  
  225.         if (mBitmap == null || mBitmap.isRecycled()) {  
  226.             mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();  
  227.         }  
  228.         /* 下面實現了CENTER_CROP的功能 */  
  229.         float xScale = mOriginalWidth / ((float) mBitmap.getWidth());  
  230.         float yScale = mOriginalHeight / ((float) mBitmap.getHeight());  
  231.         float scale = xScale > yScale ? xScale : yScale;  
  232.         mSmoothMatrix.reset();  
  233.         mSmoothMatrix.setScale(scale, scale);  
  234.         mSmoothMatrix.postTranslate(-(scale * mBitmap.getWidth() / 2 - mOriginalWidth / 2), -(scale * mBitmap.getHeight() / 2 - mOriginalHeight / 2));  
  235.     }  
  236.   
  237.     private void getBmpMatrix() {  
  238.         if (getDrawable() == null) {  
  239.             return;  
  240.         }  
  241.         if (mTransfrom == null) {  
  242.             return;  
  243.         }  
  244.         if (mBitmap == null || mBitmap.isRecycled()) {  
  245.             mBitmap = ((BitmapDrawable) getDrawable()).getBitmap();  
  246.         }  
  247.         /* 下面實現了CENTER_CROP的功能 */  
  248.         mSmoothMatrix.setScale(mTransfrom.scale, mTransfrom.scale);  
  249.         mSmoothMatrix.postTranslate(-(mTransfrom.scale * mBitmap.getWidth() / 2 - mTransfrom.rect.width / 2),  
  250.                 -(mTransfrom.scale * mBitmap.getHeight() / 2 - mTransfrom.rect.height / 2));  
  251.     }  
  252.   
  253.     @Override  
  254.     protected void onDraw(Canvas canvas) {  
  255.         if (getDrawable() == null) {  
  256.             return// couldn't resolve the URI  
  257.         }  
  258.   
  259.         if (mState == STATE_TRANSFORM_IN || mState == STATE_TRANSFORM_OUT) {  
  260.             if (mTransformStart) {  
  261.                 initTransform();  
  262.             }  
  263.             if (mTransfrom == null) {  
  264.                 super.onDraw(canvas);  
  265.                 return;  
  266.             }  
  267.   
  268.             if (mTransformStart) {  
  269.                 if (mState == STATE_TRANSFORM_IN) {  
  270.                     mTransfrom.initStartIn();  
  271.                 } else {  
  272.                     mTransfrom.initStartOut();  
  273.                 }  
  274.             }  
  275.   
  276.             if(mTransformStart){  
  277.                 Log.d("Dean""mTransfrom.startScale:"+mTransfrom.startScale);  
  278.                 Log.d("Dean""mTransfrom.startScale:"+mTransfrom.endScale);  
  279.                 Log.d("Dean""mTransfrom.scale:"+mTransfrom.scale);  
  280.                 Log.d("Dean""mTransfrom.startRect:"+mTransfrom.startRect.toString());  
  281.                 Log.d("Dean""mTransfrom.endRect:"+mTransfrom.endRect.toString());  
  282.                 Log.d("Dean""mTransfrom.rect:"+mTransfrom.rect.toString());  
  283.             }  
  284.               
  285.             mPaint.setAlpha(mBgAlpha);  
  286.             canvas.drawPaint(mPaint);  
  287.               
  288.             int saveCount = canvas.getSaveCount();  
  289.             canvas.save();  
  290.             // 先得到圖片在此刻的圖像Matrix矩陣  
  291.             getBmpMatrix();  
  292.             canvas.translate(mTransfrom.rect.left, mTransfrom.rect.top);  
  293.             canvas.clipRect(00, mTransfrom.rect.width, mTransfrom.rect.height);  
  294.             canvas.concat(mSmoothMatrix);  
  295.             getDrawable().draw(canvas);  
  296.             canvas.restoreToCount(saveCount);  
  297.             if (mTransformStart) {  
  298.                 mTransformStart=false;  
  299.                 startTransform(mState);  
  300.             }   
  301.         } else {  
  302.             //當Transform In變化完成後,把背景改爲黑色,使得Activity不透明  
  303.             mPaint.setAlpha(255);  
  304.             canvas.drawPaint(mPaint);  
  305.             super.onDraw(canvas);  
  306.         }  
  307.     }  
  308.   
  309.     private void startTransform(final int state) {  
  310.         if (mTransfrom == null) {  
  311.             return;  
  312.         }  
  313.         ValueAnimator valueAnimator = new ValueAnimator();  
  314.         valueAnimator.setDuration(300);  
  315.         valueAnimator.setInterpolator(new AccelerateDecelerateInterpolator());  
  316.         if (state == STATE_TRANSFORM_IN) {  
  317.             PropertyValuesHolder scaleHolder = PropertyValuesHolder.ofFloat("scale", mTransfrom.startScale, mTransfrom.endScale);  
  318.             PropertyValuesHolder leftHolder = PropertyValuesHolder.ofFloat("left", mTransfrom.startRect.left, mTransfrom.endRect.left);  
  319.             PropertyValuesHolder topHolder = PropertyValuesHolder.ofFloat("top", mTransfrom.startRect.top, mTransfrom.endRect.top);  
  320.             PropertyValuesHolder widthHolder = PropertyValuesHolder.ofFloat("width", mTransfrom.startRect.width, mTransfrom.endRect.width);  
  321.             PropertyValuesHolder heightHolder = PropertyValuesHolder.ofFloat("height", mTransfrom.startRect.height, mTransfrom.endRect.height);  
  322.             PropertyValuesHolder alphaHolder = PropertyValuesHolder.ofInt("alpha"0255);  
  323.             valueAnimator.setValues(scaleHolder, leftHolder, topHolder, widthHolder, heightHolder, alphaHolder);  
  324.         } else {  
  325.             PropertyValuesHolder scaleHolder = PropertyValuesHolder.ofFloat("scale", mTransfrom.endScale, mTransfrom.startScale);  
  326.             PropertyValuesHolder leftHolder = PropertyValuesHolder.ofFloat("left", mTransfrom.endRect.left, mTransfrom.startRect.left);  
  327.             PropertyValuesHolder topHolder = PropertyValuesHolder.ofFloat("top", mTransfrom.endRect.top, mTransfrom.startRect.top);  
  328.             PropertyValuesHolder widthHolder = PropertyValuesHolder.ofFloat("width", mTransfrom.endRect.width, mTransfrom.startRect.width);  
  329.             PropertyValuesHolder heightHolder = PropertyValuesHolder.ofFloat("height", mTransfrom.endRect.height, mTransfrom.startRect.height);  
  330.             PropertyValuesHolder alphaHolder = PropertyValuesHolder.ofInt("alpha"2550);  
  331.             valueAnimator.setValues(scaleHolder, leftHolder, topHolder, widthHolder, heightHolder, alphaHolder);  
  332.         }  
  333.   
  334.         valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {  
  335.             @Override  
  336.             public synchronized void onAnimationUpdate(ValueAnimator animation) {  
  337.                 mTransfrom.scale = (Float) animation.getAnimatedValue("scale");  
  338.                 mTransfrom.rect.left = (Float) animation.getAnimatedValue("left");  
  339.                 mTransfrom.rect.top = (Float) animation.getAnimatedValue("top");  
  340.                 mTransfrom.rect.width = (Float) animation.getAnimatedValue("width");  
  341.                 mTransfrom.rect.height = (Float) animation.getAnimatedValue("height");  
  342.                 mBgAlpha = (Integer) animation.getAnimatedValue("alpha");  
  343.                 invalidate();  
  344.                 ((Activity)getContext()).getWindow().getDecorView().invalidate();  
  345.             }  
  346.         });  
  347.         valueAnimator.addListener(new ValueAnimator.AnimatorListener() {  
  348.             @Override  
  349.             public void onAnimationStart(Animator animation) {  
  350.   
  351.             }  
  352.   
  353.             @Override  
  354.             public void onAnimationRepeat(Animator animation) {  
  355.   
  356.             }  
  357.   
  358.             @Override  
  359.             public void onAnimationEnd(Animator animation) {  
  360.                 /* 
  361.                  * 如果是進入的話,當然是希望最後停留在center_crop的區域。但是如果是out的話,就不應該是center_crop的位置了 
  362.                  * , 而應該是最後變化的位置,因爲當out的時候結束時,不回覆視圖是Normal,要不然會有一個突然閃動回去的bug 
  363.                  */  
  364.                 // TODO 這個可以根據實際需求來修改  
  365.                 if (state == STATE_TRANSFORM_IN) {  
  366.                     mState = STATE_NORMAL;  
  367.                 }  
  368.                 if (mTransformListener != null) {  
  369.                     mTransformListener.onTransformComplete(state);  
  370.                 }  
  371.             }  
  372.   
  373.             @Override  
  374.             public void onAnimationCancel(Animator animation) {  
  375.   
  376.             }  
  377.         });  
  378.         valueAnimator.start();  
  379.     }  
  380.   
  381.     public void setOnTransformListener(TransformListener listener) {  
  382.         mTransformListener = listener;  
  383.     }  
  384.   
  385.     private TransformListener mTransformListener;  
  386.   
  387.     public static interface TransformListener {  
  388.         /** 
  389.          *  
  390.          * @param mode 
  391.          *            STATE_TRANSFORM_IN 1 ,STATE_TRANSFORM_OUT 2 
  392.          */  
  393.         void onTransformComplete(int mode);// mode 1  
  394.     }  
  395.   
  396. }  

使用的時候,從前一個Activity傳遞到詳情Activity下面幾個主要的信息:

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
  1. Intent intent = new Intent(MainActivity.this, SpaceImageDetailActivity.class);  
  2.                     intent.putExtra("images", (ArrayList<String>) datas);//非必須  
  3.                     intent.putExtra("position", position);  
  4.                     int[] location = new int[2];  
  5.                     imageView.getLocationOnScreen(location);  
  6.                     intent.putExtra("locationX", location[0]);//必須  
  7.                     intent.putExtra("locationY", location[1]);//必須  
  8.   
  9.                     intent.putExtra("width", imageView.getWidth());//必須  
  10.                     intent.putExtra("height", imageView.getHeight());//必須  
  11.                     startActivity(intent);  
  12.                     overridePendingTransition(00);  

在詳情Activity接受到這些參數,並對SmoothImageView初始化位置信息,然後就可以進行變化了。

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
  1. mDatas = (ArrayList<String>) getIntent().getSerializableExtra("images");  
  2. mPosition = getIntent().getIntExtra("position"0);  
  3. mLocationX = getIntent().getIntExtra("locationX"0);  
  4. mLocationY = getIntent().getIntExtra("locationY"0);  
  5. mWidth = getIntent().getIntExtra("width"0);  
  6. mHeight = getIntent().getIntExtra("height"0);  
  7.   
  8. imageView = new SmoothImageView(this);  
  9. imageView.setOriginalInfo(mWidth, mHeight, mLocationX, mLocationY);  
  10. imageView.transformIn();  
  11. imageView.setLayoutParams(new ViewGroup.LayoutParams(-1, -1));  
  12. imageView.setScaleType(ScaleType.FIT_CENTER);  
  13. setContentView(imageView);  
  14. ImageLoader.getInstance().displayImage(mDatas.get(mPosition), imageView);  



上面的就已經完成了圖片的縮放效果,但是還需要設置下Activity透明的風格,才能使得alpha效果體驗出來,用戶體驗更好。

對Activity設置如下風格,另外說明,在SmoothImageView中沒有定位視圖的位置,只是做了對狀態欄的處理,所以要設置Activity 爲NotitleBar,具體style如下:


在CODE上查看代碼片派生到我的代碼片
  1. <style name="IMTheme.Transparent" >  
  2.        <item name="android:windowBackground">@android:color/transparent</item>  
  3.        <item name="android:windowIsTranslucent">true</item>  
  4.        <item name="android:windowNoTitle">true</item>  
  5.        <item name="android:windowContentOverlay">@null</item>  
  6. lt;/style>  
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章