查看大图之超长图处理

关于Android的超长图处理,可以很容易的找到解决方案,即用BitmapRegionDecoder来分区域生成bitmap来实现,但是在实践过程中发现,各中细节并不是那么容易,下面分享一下其中的技术难点。

实现目标

类似于微博和微信,对于超长图的处理。

  1. 双击进入超长图模式,超长图自动占满全屏方便阅读
  2. 滑动到哪里,哪个区域变得清晰
  3. 带惯性的流畅滑动

实现思路

  1. 捕获双击手势,利用matrix放大原始小图得到模糊的大图
  2. 捕获手势,利用scrollByOverScroller 来实现滑动和惯性滑动
  3. 监听滑动事件,在滑动事件中判断是否需要获取新的bitmap。如需获取则开始异步获取bitmap
  4. 将异步获取到的bitmapondraw中绘制到屏幕的对应区域

手势处理

手势处理可以利用GestureDetector这个类捕获

双击事件

用来放大缩小图片,进入和退出长图模式

  @Override
            public boolean onDoubleTap(MotionEvent e) {
                if (isAnim || isLoading||!canMove)
                    return true;
                if (!isScale) {
                    BigImgImageView.this.setScaleType(ScaleType.MATRIX);
                    scrollTo(0, 0);
                    RectF rect = bigImgViewUtils.getMatrixMapRect(currentMaritx);
                    float downXRatio = calcScaleScrollRatio(true, e, rect);
                    float downYRatio = calcScaleScrollRatio(false, e, rect);
                    animToScale(downXRatio, downYRatio);
                } else {
                    scrollTo(0, 0);
                    bigImgViewRealImgHelper.cancelDrawBigImg();
                    animToMatrix(currentMaritx, originMatrix);
                    destroyBigImg();
                }

                return true;
            }

计算放大倍率

 private float calcScaleScrollRatio(boolean isX, MotionEvent event, RectF rect) {
        float ratio = 0;
        if (isX) {
            if (event.getX() < (getWidth() - rect.width()) / 2)
                ratio = 0;
            else if (event.getX() > (getWidth() + rect.height()) / 2) {
                ratio = 1;
            } else {
                ratio = (event.getX() - (getWidth() - rect.width()) / 2) / rect.width();
            }
        } else {
            if (event.getY() < (getHeight() - rect.height()) / 2)
                ratio = 0;
            else if (event.getY() > (getHeight() + rect.height()) / 2) {
                ratio = 1;
            } else {
                ratio = (event.getY() - (getHeight() - rect.height()) / 2) / rect.height();
            }
        }
        return ratio;
    }

滑动事件

 @Override
            public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
                if (isAnim || isLoading||!canMove)
                    return true;
                if (isScale) {
                    RectF rectf = bigImgViewUtils.getMatrixMapRect(currentMaritx);
                    int maxX = (int) (rectf.width() / 2 - getWidth() / 2);
                    int maxY = (int) (rectf.height() / 2 - getHeight() / 2);
                    int minX = -maxX;
                    int minY = -maxY;
                    boolean cross = false;
//避免超出滑动范围
                    if (getScrollX() + distanceX > maxX) {
                        distanceX = maxX - getScrollX();
                        cross = true;
                    }


                    if (getScrollX() + distanceX < minX) {
                        cross = true;
                        distanceX = minX - getScrollX();
                    }


                    if (getScrollY() + distanceY > maxY)
                        distanceY = maxY - getScrollY();

                    if (getScrollY() + distanceY < minY)
                        distanceY = minY - getScrollY();

                    requestIntercept(true);
        
                    BigImgImageView.this.scrollBy((int) distanceX, (int) distanceY);

                }
                return true;
            }
        });

惯性滑动

@Override
            public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
                if (isAnim || isLoading||!canMove)
                    return true;
                if (isScale) {
                    requestIntercept(true);
                    RectF rectf = bigImgViewUtils.getMatrixMapRect(currentMaritx);
               
                    scroller.fling(getScrollX(), getScrollY(), -(int) velocityX, (int) -velocityY,
                            -(int) (rectf.width() / 2 - getWidth() / 2), (int) (rectf.width() / 2 - getWidth() / 2),
                            -(int) (rectf.height() / 2 - getHeight() / 2), (int) (rectf.height() / 2) - getHeight() / 2);
               
                    scrollStart = true;
                    invalidate();
                }
                return true;
            }

大图变换

这里各个地方需要注意,利用matrix放大的倍率精度是有限的,我们不要用开始计算好的倍率来处理后续业务,等matrix放大完毕后,测量matrix真正的放大倍率,再利用这个放大倍率进行后续计算

   //计算放大倍率 
    private void animToScale(final float downXRatio, final float downYRatio) {
        RectF rectF = bigImgViewUtils.getMatrixMapRect(originMatrix);
        float widthRatio = getWidth() / rectF.width();
        float heightRatio = getHeight() / rectF.height();
        float scaleRatio;
        boolean isWidthMore = widthRatio > heightRatio;
        if (widthRatio <= 1f && heightRatio <= 1f) {
            scaleRatio = maxScale;
        } else {
            scaleRatio = isWidthMore ? widthRatio : heightRatio;
        }
        if (scaleRatio < maxScale)
            scaleRatio = maxScale;

        bigImgViewRealImgHelper.needLoadRealBySize = scaleRatio > scrollMinRatio;
        if (!bigImgViewRealImgHelper.needLoadRealBySize) {
            int dx = 0;
            int dy = 0;
            dx = -(int) ((scaleRatio * rectF.width() - getWidth()) / 2 - downXRatio * scaleRatio * rectF.width() + getWidth() * downXRatio);
            dy = -(int) ((scaleRatio * rectF.height() - getHeight()) / 2 - downYRatio * scaleRatio * rectF.height() + getHeight() * downYRatio);
            scroller.startScroll(0, 0, dx, dy, 150);
        }
        playScaleAnim(downXRatio, downYRatio, scaleRatio);
    }

播放放大动画 ,并在动画结束后根据双击坐标,改变当前位置scrollX 与scrollY

 private void playScaleAnim(final float downXRatio, final float downYRatio, float scaleRatio) {
        ValueAnimator valueAnimator = ValueAnimator.ofFloat(1, scaleRatio);
        valueAnimator.setDuration(150);
        valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                currentMaritx = new Matrix(originMatrix);
                currentMaritx.postScale((Float) animation.getAnimatedValue(), (Float) animation.getAnimatedValue(), getWidth() / 2, getHeight() / 2);
                BigImgImageView.this.setImageMatrix(currentMaritx);
            }
        });
        valueAnimator.addListener(new Animator.AnimatorListener() {
            @Override
            public void onAnimationStart(Animator animation) {
                setAnim(true);

            }

            @Override
            public void onAnimationEnd(Animator animation) {
                setAnim(false);
                isScale = true;
                changeMode(true);
                RectF realRect = bigImgViewUtils.getMatrixMapRect(currentMaritx);

                if (bigImgViewRealImgHelper.needLoadRealBySize) {
                    int dx = 0;
                    int dy = 0;
                    dx = realRect.width() > realRect.height() ? (int) (downXRatio * (realRect.width() - getWidth())) : 0;
                    dy = realRect.height() > realRect.width() ? (int) (downYRatio * realRect.height() - getHeight()) : 0;
                    scrollTo((int) (-realRect.width() / 2 + dx + getWidth() / 2), (int) (-realRect.height() / 2 + dy + getHeight() / 2));
                }


                if (bigImgViewRealImgHelper.needToLoadRealBigImg) {
                    bigImgViewRealImgHelper.initBitmapRegion(uri);
                    bigImgViewRealImgHelper.onBigImgFlingStop(currentMaritx);
                }
            }

            @Override
            public void onAnimationCancel(Animator animation) {

            }

            @Override
            public void onAnimationRepeat(Animator animation) {

            }
        });
        valueAnimator.start();
    }

加载区域图片

这里的处理要注注意,不能每次生成的区域太小。避免 BitmapRegionDecoder 频繁创建bitmap ,这里很容易导致 oom 或者过于频繁的GC造成卡顿。因为我们是在滑动的回调中处理这些业务,调用次数很频繁,所以要尽可能的避免在过程中创建对象。
同时这里我整合了几个对象

RealBitmap

包含需要绘制的bitmap 和相关区域信息

 //超长图加载区域信息
    public static class RealBitmap {
//需要绘制的图片
        public Bitmap bitmap;
//图片需要绘制的区域
        public Rect rect1;
//图片绘制的目标区域
        public RectF targetRect;
//该图片在原始图片中的区域
        private Rect calcRect;

        public RealBitmap(RealBitmap realBitmap) {
            this.bitmap = realBitmap.bitmap;
            this.rect1 = realBitmap.rect1;
            this.targetRect = realBitmap.targetRect;
            this.calcRect = realBitmap.calcRect;
        }

        private RealBitmap() {
        }

        public void recycle() {
            if (bitmap != null)
                bitmap.recycle();
        }

        public boolean isRecycled() {
            return bitmap == null || bitmap.isRecycled();
        }

        @Override
        public String toString() {
            return bitmap.getWidth() + "---" + bitmap.getHeight() + "----" + rect1.toString() + "----" + targetRect.toString();
        }
    }

RealBitmapWrapper

我们加载过程要根据滑动方向进行预加载 ,所以包装了一个之前和当前的 RealBitmap 。
预计加载的方向如下,每次多加载一屏的bitmap可以有效地的避免bitmap创建过于频繁。

  //超大图预加载
    protected enum Orientation {
        toLeft, toRight, toTop, toBottom, none
    }
    //超长图加载信息
    public class RealBitmapWrapper {
        public RealBitmap last;
        public RealBitmap current;

        private synchronized void add(RealBitmap bitmap) {
            if (current == null)
                current = bitmap;
            else {
                if (last != null)
                    last.recycle();
                last = current;
                current = bitmap;
            }
        }

        public void recycle() {
            if (last != null)
                last.recycle();
            if (current != null)
                current.recycle();
            last = null;
            current = null;
        }

        //是否包含
        public boolean contains(Rect rect) {
            if (last == null && current == null)
                return false;
            if (last != null && current != null) {
                tempRectF.set(Math.min(current.targetRect.left, last.targetRect.left),
                        Math.min(current.targetRect.top, last.targetRect.top),
                        Math.max(current.targetRect.right, last.targetRect.right),
                        Math.max(current.targetRect.bottom, last.targetRect.bottom));
                return tempRectF.contains(RectToRectF(rect));
            }
            if (current != null)
                return current.targetRect.contains(RectToRectF(rect));
            else
                return last.targetRect.contains(RectToRectF(rect));
        }

        //获取下一次加载方向
        private Orientation containsGetNext(Rect rect) {
            RectF finial;
            if (last == null && current == null)
                return Orientation.none;
            if (last != null && current != null) {
                tempRectF.set(Math.min(current.targetRect.left, last.targetRect.left),
                        Math.min(current.targetRect.top, last.targetRect.top),
                        Math.max(current.targetRect.right, last.targetRect.right),
                        Math.max(current.targetRect.bottom, last.targetRect.bottom));
                finial = tempRectF;
            } else if (current != null)
                finial = current.targetRect;
            else
                finial = last.targetRect;
            if (finial.left > rect.left)
                return toLeft;
            else if (finial.right < rect.right)
                return toRight;
            else if (finial.top < rect.top)
                return toBottom;
            else
                return toTop;
        }
    }

判断是否需要加载

 //是否需要去加载
    private boolean needToLoad() {
        if (realBitmapWrapper == null)
            return true;
        currentScrollRect.set(imageView.getScrollX(), imageView.getScrollY(), imageView.getScrollX() + imageView.getWidth(),
                imageView.getScrollY() + imageView.getHeight());
        return !realBitmapWrapper.contains(currentScrollRect);
    }

获取图片

计算当前参数,确定需要获取的图片在原图片的坐标目标绘制坐标

  //获取清晰的真实图片
    private void getOriginBitmapRect(Orientation preloadFlag, Matrix currentMaritx) {
        tempMatrixRect.setEmpty();
        tempMatrixRect.right = imageView.getDrawable().getIntrinsicWidth();
        tempMatrixRect.bottom = imageView.getDrawable().getIntrinsicHeight();
        currentMaritx.mapRect(tempMatrixRect);

        RectF current = tempMatrixRect;
        float ratio = (float) bigImgRealWidth / current.width();
        float ratioHeight = (float) bigImgRealHeight / current.height();
        bigAsyncData.rect = calcBitmapRect(ratio, ratioHeight, bigImgRealWidth, bigImgRealHeight, current
                , preloadFlag, imageView.getScrollX(), imageView.getScrollY());
        bigAsyncData.target = calcDrawRect(ratio, ratioHeight, bigAsyncData.rect, imageView.getScrollX(), imageView.getScrollY(), preloadFlag);

        if (bigAsyncData.target.equals(currentRequestRect))
            return;
        currentRequestRect = bigAsyncData.target;

        if (asyncBigImg != null) {
            asyncBigImg.cancel(true);
        }
        asyncBigImg = new AsyncBigImg();
        asyncBigImg.execute(bigAsyncData);
    }

交由异步任务执行获取过程

 //获取大图异步放大
    private class AsyncBigImg extends AsyncTask<BigAsyncData, Object, RealBitmap> {
        private boolean isCancel = false;

        @Override
        protected RealBitmap doInBackground(BigAsyncData... params) {
            BigAsyncData bigAsyncData = params[0];
            if (bitmapRegionDecoder == null)
                return null;

            RealBitmap realBitmapT = null;
            BitmapFactory.Options options = new BitmapFactory.Options();
            options.inPreferredConfig = Bitmap.Config.RGB_565;
            Bitmap bitmap = null;
            try {
                bitmap = bitmapRegionDecoder.decodeRegion(changRotateRect(imgRotate, bigAsyncData.rect), options);
                if (imgRotate != 0) {
                    Bitmap old = bitmap;
                    bitmap = FileUntil.rotateBitmap(bitmap, imgRotate);
                    old.recycle();
                }
            } catch (Exception e) {
                e.printStackTrace();
                return null;
            }
            if (bitmap != null)
                realBitmapT = new RealBitmap();
            else
                return null;
            realBitmapT.bitmap = bitmap;
            realBitmapT.rect1 = new Rect(0, 0, bitmap.getWidth(), bitmap.getHeight());
            realBitmapT.targetRect = bigAsyncData.target;
            realBitmapT.calcRect = bigAsyncData.rect;
            if (isCancel) {
                realBitmapT.recycle();
                realBitmapT = null;
            }
            return realBitmapT;
        }

 @Override
        protected void onCancelled() {
            super.onCancelled();
            isCancel = true;
        }

        @Override
        protected void onPreExecute() {
            super.onPreExecute();
        }

        @Override
        protected void onPostExecute(RealBitmap realBitmap) {
            if (realBitmap == null)
                return;
            realBitmapWrapper.add(realBitmap);
            drawRealBig = true;
            imageView.invalidate();
        }
}

图片绘制

首先我们需要一个标志位来确定是否需要绘制。另外需要一个对象来保存异步获取的绘制图片信息,方便在ondraw中调用

 //是否可以绘制大图
    public boolean drawRealBig = false;
 //原始图片信息
    public RealBitmapWrapper realBitmapWrapper = new RealBitmapWrapper();

最后再ondraw中绘制bitmap即可

canvas.drawBitmap(realBitmapWrapper.current.bitmap, realBitmapWrapper.current.rect1,
                        bigImgViewRealImgHelper.realBitmapWrapper.current.targetRect, null);

总结

这里的核心难点在于对内存的把控,这里可能要频繁的生成bitmap 注意要及时释放无用的。另外,为了避免bitmap过于频繁生成,我们加入了预加载机制,根据滑动的方向,预加载部分图片。按这套机制处理出来的超大图与微博,微信效果无异。我们来看一下最终效果图

效果图
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容