Android Glide 3.7.0 源码解析(五) , 如何获得ImageView的宽高

个人博客传送门

前言

通过前面的 Android Glide 3.7.0 源码解析 (二) , 从一次图片加载流程看源码 我们知道
Request(真实) 只有在图片组件的大小准备好了才会开始真正的加载

// GenericRequest
	public void begin() {
        startTime = LogTime.getLogTime();
        if (model == null) {
            onException(null);
            return;
        }

        status = Status.WAITING_FOR_SIZE;
        if (Util.isValidDimensions(overrideWidth, overrideHeight)) {
        	// 指定了特定的 size
            onSizeReady(overrideWidth, overrideHeight);
        } else {
        	// 未指定, 则去获取 size , 得到后调用 onSizeReady()...
            target.getSize(this);
        }

        if (!isComplete() && !isFailed() && canNotifyStatusChanged()) {
            target.onLoadStarted(getPlaceholderDrawable());
        }
        if (Log.isLoggable(TAG, Log.VERBOSE)) {
            logV("finished run method in " + LogTime.getElapsedMillis(startTime));
        }
    }

我们参考 Android Glide 3.7.0 源码解析 (二) , 从一次图片加载流程看源码 一文中介绍的流程, 可得 target 是一个 GlideDrawableImageViewTarget 类型, 下面就来看看是如何获取 size 的

如何获取 size

// GlideDrawableImageViewTarget

	private final SizeDeterminer sizeDeterminer;
	@Override
    public void getSize(SizeReadyCallback cb) {
        sizeDeterminer.getSize(cb);
    }

// SizeDeterminer
	public void getSize(SizeReadyCallback cb) {
			// 计算当前View宽高的代码
            int currentWidth = getViewWidthOrParam();
            int currentHeight = getViewHeightOrParam();
            // 直接开始计算View的宽高, 如果>0或者是WRAP_CONTENT(-2)就代表有效
            if (isSizeValid(currentWidth) && isSizeValid(currentHeight)) {
            	// View 已经加载完, 能算出宽高了
                cb.onSizeReady(currentWidth, currentHeight);
            } else {
                if (!cbs.contains(cb)) {
                    cbs.add(cb);
                }
                if (layoutListener == null) {
                	// 给View的刷新注册监听, 希望View加载完, SizeDeterminerLayoutListener 能收到通知事件
                    final ViewTreeObserver observer = view.getViewTreeObserver();
                    layoutListener = new SizeDeterminerLayoutListener(this);
                    observer.addOnPreDrawListener(layoutListener);
                }
            }
        }

	private boolean isSizeValid(int size) {
            return size > 0 || size == LayoutParams.WRAP_CONTENT;
        }

关于 isSizeValid >0 有效可以理解, 为啥 size == LayoutParams.WRAP_CONTENT 也会被判定有效???
其实在分析完后, 发现这条判断条件好像也用不上

需要弄清楚两个问题:

  1. getViewWidthOrParam()getViewHeightOrParam() 方法是如何计算宽高的
  2. SizeDeterminerLayoutListener 监听 View 的变化之后, 又做了些啥

计算Size的基本函数

我们来看看如何计算 当前状态下(可能没加载完全) 的View的宽高

// SizeDeterminer
	private static final int PENDING_SIZE = 0;
	private int getViewHeightOrParam() {
            final LayoutParams layoutParams = view.getLayoutParams();
            if (isSizeValid(view.getHeight())) {
            	// 直接 getHeight 如果 >0 的话
                return view.getHeight();
            } else if (layoutParams != null) {
            	// 其他情况
                return getSizeForParam(layoutParams.height, true /*isHeight*/);
            } else {
            	// 默认返回计算失败 0 
                return PENDING_SIZE;
            }
        }

        private int getViewWidthOrParam() {
            final LayoutParams layoutParams = view.getLayoutParams();
            if (isSizeValid(view.getWidth())) {
            	// 直接 getWidth 如果 >0 的话
                return view.getWidth();
            } else if (layoutParams != null) {
            	// 其他情况
                return getSizeForParam(layoutParams.width, false /*isHeight*/);
            } else {
            	// 默认返回计算失败 0
                return PENDING_SIZE;
            }
        }
		
		// WRAP_CONTETNT 的情况计算
        private int getSizeForParam(int param, boolean isHeight) {
            if (param == LayoutParams.WRAP_CONTENT) {
            	// 计算 WRAP_CONTETNT 的核心代码
                Point displayDimens = getDisplayDimens();
                return isHeight ? displayDimens.y : displayDimens.x;
            } else {
                return param;
            }
        }

	private Point getDisplayDimens() {
            if (displayDimens != null) {
                return displayDimens;
            }
            WindowManager windowManager = (WindowManager) view.getContext().getSystemService(Context.WINDOW_SERVICE);
            // 很简单, 就是获取 View 的展示区域
            Display display = windowManager.getDefaultDisplay();
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB_MR2) {
                displayDimens = new Point();
                display.getSize(displayDimens);
            } else {
                displayDimens = new Point(display.getWidth(), display.getHeight());
            }
            return displayDimens;
        }

至此, 我们看完了计算 View 大小的函数 ( getViewWidthOrParam() / getViewHeightOrParam() ) , 原理很简单

  1. getHeight / getWidth 能计算出来 >0 的值, 就以它为准
  2. LayoutParams.WRAP_CONTENT 或者 LayoutParams.MATCH_PARENT 时, windowManager.getDefaultDisplay() 来获得 View 的显示区域

简单画了一个流程图
windowManager.getDefaultDisplay()
3. 一种是返回 0 或者 MATCH_PARENT , 表示我没计算出来
4. 一种是返回 getHeight / getWidth 或者 windowManager.getDefaultDisplay() , 表示我计算出来了

分析完基本的计算宽高的函数, 我们再来看看, SizeDeterminerLayoutListener 监听 View 刷新之后都做了啥

监听 View 刷新

// SizeDeterminerLayoutListener
	private static class SizeDeterminerLayoutListener implements ViewTreeObserver.OnPreDrawListener {
            private final WeakReference<SizeDeterminer> sizeDeterminerRef;

            public SizeDeterminerLayoutListener(SizeDeterminer sizeDeterminer) {
                sizeDeterminerRef = new WeakReference<SizeDeterminer>(sizeDeterminer);
            }

            @Override
            public boolean onPreDraw() {
                // View 刷新的回调
                SizeDeterminer sizeDeterminer = sizeDeterminerRef.get();
                if (sizeDeterminer != null) {
                	// SizeDeterminer 去检查当前 View 的宽高是否有效
                    sizeDeterminer.checkCurrentDimens();
                }
                return true;
            }
        }

// SizeDeterminer
	private void checkCurrentDimens() {
            if (cbs.isEmpty()) {
                return;
            }

			// 重新再计算宽高
            int currentWidth = getViewWidthOrParam();
            int currentHeight = getViewHeightOrParam();
            if (!isSizeValid(currentWidth) || !isSizeValid(currentHeight)) {
                return;
            }
			
			// 宽高有效, 通知 GenericRequest
            notifyCbs(currentWidth, currentHeight);
            
            // 移除对 View 刷新的监听
            ViewTreeObserver observer = view.getViewTreeObserver();
            if (observer.isAlive()) {
                observer.removeOnPreDrawListener(layoutListener);
            }
            layoutListener = null;
        }
	
	private void notifyCbs(int width, int height) {
            for (SizeReadyCallback cb : cbs) {
            	// 这里的 cb 就是 GenericRequest 实例了
                cb.onSizeReady(width, height);
            }
            cbs.clear();
        }

逻辑很简单

  1. View 刷新了
  2. 再次计算宽高, 有效 -> GenericRequest.onSizeReady
    () 开始加载图片; 移除刷新监听
  3. 再次计算宽高, 无效 -> 啥也不做, 继续傻等

小结

说了那么多, 总结一下流程
在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android图片框架Glide-3.7.0(最新,很强大),超好用的图片框架(包含jar和源码Glide 是一个效、开源、 Android设备上的媒体管理框架,它遵循BSD、MIT以及Apache 2.0协议发布。Glide具有获取、解码和展示视频剧照、图片、动画等功能,它还有灵活的API,这些API使开发者能够将Glide应用在几乎任何网络协议栈里。创建Glide的主要目的有两个,一个是实现平滑的图片列表滚动效果,另一个是支持远程图片的获取、大小调整和展示。近日,Glide 3.0发布,现已提供 jar包下载 ,同时还支持使用Gradle以及Maven进行构建。该版本包括很多值得关注的新功能,如支持Gif 动画和视频剧照解码、智能的暂停和重新开始请求、支持缩略图等,具体新增功能如下如下: GIF 动画的解码 :通过调用Glide.with(context).load(“图片路径“)方法,GIF动画图片可以自动显示为动画效果。如果想有更多的控制,还可以使用Glide.with(context).load(“图片路径“).asBitmap()方法加载静态图片,使用Glide.with(context).load(“图片路径“).asGif()方法加载动画图片 本地视频剧照的解码: 通过调用Glide.with(context).load(“图片路径“)方法,Glide能够支持Android设备中的所有视频剧照的加载和展示 缩略图的支持: 为了减少在同一个view组件里同时加载多张图片的时间,可以调用Glide.with(context).load(“图片路径“).thumbnail(“缩略比例“).into(“view组件“)方法加载一个缩略图,还可以控制thumbnail()中的参数的大小,以控制显示不同比例大小的缩略图 Activity 生命周期的集成: 当Activity暂停和重启时,Glide能够做到智能的暂停和重新开始请求,并且当Android设备的连接状态变化时,所有失败的请求能够自动重新请求 转码的支持: Glide的toBytes() 和transcode() 两个方法可以用来获取、解码和变换背景图片,并且transcode() 方法还能够改变图片的样式 动画的支持: 新增支持图片的淡入淡出动画效果(调用crossFade()方法)和查看动画的属性的功能 OkHttp 和Volley 的支持: 默认选择HttpUrlConnection作为网络协议栈,还可以选择OkHttp和Volley作为网络协议栈 其他功能: 如在图片加载过程中,使用Drawables对象作为占位符、图片请求的优化、图片的度和度可重新设定、缩略图和原图的缓存等功能

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值