最近项目上有一个需求,在显示图片的时候,需要传递自定义的头部就行认证。google了一番之后,发现没有现成的组件库可以使用【也可能是我没找到】,所以请求图片只能采用xhr方式来异步加载。下面就是在做这个组件库时的一些笔记,主要关注以下两个点:

  • 图片的等比例缩放处理
  • 在请求图片的过程中,由于是异步加载,如果后加载的一个图片太小,而前一个图片过大,就会有图片显示不正确的问题

图片的缩放处理

最开始想到的是使用CSS 属性background来显示图片,后来发现使用CSS的background-size实现按照比例缩放图片好像有点困难,具体如下:

  • 如果图片原始的尺寸小于外层容器的尺寸,我希望它居中显示
  • 如果图片原始尺寸大于外层尺寸
    • 如果ratio > 1 (imageWidth / imageHeight),图片应该按照宽度来进行缩放
    • 如果ratio = 1, 图片等比例缩放
    • 如果ratio < 1, 图片按照高度来缩放

因为要取到图片的原始尺寸,使用img标签显示也会有点问题。所以最终采用的是new Image()这个Web Api来创建的图片。具体代码如下:

export const getImage = (src: string) => (
new Promise((resolve, reject) => {
const image = new Image();
image.onload = () => resolve(image);
image.onerror = () => reject(new Error(NETWORK_ERROR));
image.src = src;
image.crossOrigin = '';
return image;
})
);

同时,图片缩放的部分代码如下:

if (ratio > 1) {
if (imageWidth > wrapperWidth) {
displayWidth = wrapperWidth;
displayHeight = parseInt(`${(1 / ratio) * wrapperWidth}`, 10);
}
} else if (ratio === 1) {
if (imageWidth > wrapperWidth) {
displayWidth = wrapperWidth;
displayHeight = wrapperWidth;
} else {
displayWidth = wrapperHeight;
displayHeight = wrapperHeight;
}
} else if (imageHeight > wrapperHeight) {
displayWidth = parseInt(`${ratio * wrapperHeight}`, 10);
displayHeight = wrapperHeight;
}

图片的覆盖问题

因为需要进行头部的认证,所以请求图片的方式统一使用了XHR的方式来进行请求,然后就会造成图片覆盖的问题。造成这个原因是,当出现了图片地址替换的时候,比如类似下面的代码:

const [src, setSrc] = useState(src1);
useEffect(() => { setTimeout(() => setSrc(src2)); }, [src]); return (
<div className="App">
<Image width={50} height={100} src={src} errorMessage="something bad happen" />
</div>
);

上述代码中的src2会后被加载,如果src1的加载速度比src2的加载速度快倒没有什么问题,但是反之,就会出现后加载的图片反而被先加载的图片进行覆盖。那么,怎么解决这个问题:

想到的办法是,当开始加载后一个图片时,首先进行判断是否存在上一个加载图片的请求,如果存在,则直接abort,类似于debounce的做法。具体的做法如下:

  • 声明一个图片请求的类,专门用来作图片请求

    export default class ImageRequest {
    xmlHttpRequest: XMLHttpRequest;
    url: string;
    headers: XMLHttpRequestHeaders; setHeaders() {
    if (this.headers) {
    const keys = Object.keys(this.headers);
    keys.forEach((key: string) => {
    this.xmlHttpRequest.setRequestHeader(key, this.headers[key]);
    });
    }
    } request(url: string, headers: XMLHttpRequestHeaders) {
    this.url = url;
    this.headers = headers; if (this.xmlHttpRequest) {
    this.xmlHttpRequest.abort();
    } this.xmlHttpRequest = new XMLHttpRequest();
    this.xmlHttpRequest.open('GET', this.url);
    this.xmlHttpRequest.responseType = 'blob';
    this.setHeaders();
    this.xmlHttpRequest.send(); return new Promise((resolve, reject) => {
    this.xmlHttpRequest.onload = () => {
    this.xmlHttpRequest = null;
    if (this.xmlHttpRequest.status === 200) {
    resolve(this.xmlHttpRequest.response);
    } else {
    reject(new Error(`${IMAGE_LOAD_ERROR}${this.xmlHttpRequest.statusText}`));
    }
    }; this.xmlHttpRequest.onerror = () => {
    reject(new Error(NETWORK_ERROR));
    };
    });
    }
    }

    在每个实例中维持一个XMLHttpRequest的引用,每当进行请求的时候,首先判断当前引用是否存在,如果存在,则直接abort,否则,则进行图片的请求。同时在组件中,需要创建一个实例

    // 记住,不能在组件外部声明实例,需要保存在每一个组件中,确保每一个组件都有一个新的请求实例
    // const imageRequest: ImageRequest = new ImageRequest(); const Image: React.FC<Props> = (props) => {
    const [request] = useState<ImageRequest>(new ImageRequest()); useEffect(() => {
    if (src) {
    setState(LOADING_STATE.LOADING);
    loadImage(request, src, headers).then((img: HTMLImageElement) => {
    const { displayWidth, displayHeight } = getDisplayImageSize(img, width, height);
    const displayImage = img;
    displayImage.width = displayWidth;
    displayImage.height = displayHeight;
    setState({ ...LOADING_STATE.SUCCESS, image: displayImage });
    }).catch(() => setState(LOADING_STATE.FAIL));
    }
    }, [loadImage, src]); // ...
    };

    注意,这里的ImageRequest实例只能保存在组件的state中,因为如果在组件开始使用const引入,如果一个页面中存在多个相同组件时,就会导致多个组件共享一个request实例中的xmlHttpRequest引用,就会出现前面的图片全部都会被abort掉的情况。

总结

看是简单的问题,做起来也会比较复杂,口说的没用,做起来才行。

最后,项目地址:https://github.com/Rynxiao/react-image,npm包地址:https://www.npmjs.com/package/rt-image,欢迎留言和star

做一个单纯的react-image显示组件的更多相关文章

  1. 使用React并做一个简单的to-do-list

    1. 前言 说到React,我从一年之前就开始试着了解并且看了相关的入门教程,而且还买过一本<React:引领未来的用户界面开发框架 >拜读.React的轻量组件化的思想及其virtual ...

  2. 4-13 Webpacker-React.js; 用React做一个下拉表格的功能: <详解>

    Rails5.1增加了Webpacker: Webpacker essentially is the decisions made by the Rails team and bundled up i ...

  3. 用vue的抽象组件来做一个防止img标签url为空或url地址出错的验证

    看了网上文章学习了下vue的抽象组件,感觉就跟react的高阶组件一样的使用场景,只是更加面向vue的底层编程 ,网上介绍的抽象组件一般有2种用法,1 用来加防抖和节流 2 用来控制按钮是否允许点击做 ...

  4. react实例之todo,做一个实时响应的列表操作

    react实例之todo, 做一个实时响应的列表操作 在所有的mvc框架中,最常见的例子不是hello world,而是todo,由于reactjs的简单性,在不引用flux和redux的情况下,我们 ...

  5. VUE2.0+VUE-Router做一个图片上传预览的组件

    之前发了一篇关于自己看待前端组件化的文章,但是由于学习和实践的业务逻辑差异,所以自己练习的一些demo逻辑比较简单,打算用vue重构现在公司做的项目,所以在一些小的功能页面上使用vue来做的,现在写的 ...

  6. 利用 React 高阶组件实现一个面包屑导航

    什么是 React 高阶组件 React 高阶组件就是以高阶函数的方式包裹需要修饰的 React 组件,并返回处理完成后的 React 组件.React 高阶组件在 React 生态中使用的非常频繁, ...

  7. 手把手做一个基于vue-cli的组件库(下篇)

    基于vue-cli4的ui组件库,上篇:如何做一个初步的组件.下篇:编写说明文档及页面优化.接上篇,开工. GitHub源码地址:https://github.com/sq-github/sq-ui ...

  8. 用struts2标签如何从数据库获取数据并在查询页面显示。最近做一个小项目,需要用到struts2标签从数据库查询数据,并且用迭代器iterator标签在查询页面显示,可是一开始,怎么也获取不到数据,想了许久,最后发现,是自己少定义了一个变量,也就是var变量。

    最近做一个小项目,需要用到struts2标签从数据库查询数据,并且用迭代器iterator标签在查询页面显示,可是一开始,怎么也获取不到数据,想了许久,最后发现,是自己少定义了一个变量,也就是var变 ...

  9. 用jQuery做一个三级菜单,鼠标移动到二级菜单的选项上,然后再迅速离开后,当鼠标再移动到该一级菜单或其他二级菜单选项,三级菜单也会显示。

    用jQuery做一个三级菜单,鼠标移动到二级菜单的选项上,然后再迅速离开后,当鼠标再移动到该一级菜单或其他二级菜单选项,三级菜单也会显示. 原因:在为一个元素绑定hover事件之后,用户把光标移入元素 ...

  10. Android UI组件----用相对布局RelativeLayout做一个登陆界面

    [声明] 欢迎转载,但请保留文章原始出处→_→ 生命壹号:http://www.cnblogs.com/smyhvae/ 文章来源:http://www.cnblogs.com/smyhvae/p/3 ...

随机推荐

  1. 设置WordPress文章关键词自动获取,文章所属分类名称,描述自动获取文章内容,给文章的图片自动加上AlT标签

    最近在优化网站,SEO优化标准:每一篇文章都要有关键词,关键词的个数为3到6个.每一篇文章都要有描述,描述的字数为汉字在70~80之间,在160个字符之间.每一篇文章的图片都要有Alt标签,自动给图片 ...

  2. Hugging Face 与 Wiz Research 合作提高人工智能安全性

    我们很高兴地宣布,我们正在与 Wiz 合作,目标是提高我们平台和整个 AI/ML 生态系统的安全性. Wiz 研究人员 与 Hugging Face 就我们平台的安全性进行合作并分享了他们的发现. W ...

  3. 【漏洞复现】CVE-2023-27372 RCE漏洞

    产品介绍 SPIP是一个互联网发布系统,其中非常重视协作工作,多语言环境和Web作者的易用性.它是自由软件,在GNU/GPL许可证下分发.这意味着它可以用于任何互联网站点,无论是个人的还是机构的,非营 ...

  4. 当装饰者模式遇上Read Through缓存,一场技术的浪漫邂逅

    在<经验之谈:我为什么选择了这样一个激进的缓存大Key治理方案>一文中,我提到在系统中使用的缓存是旁路缓存模式,有读者朋友问,有没有用到过其他的缓存模式,本文将结合一个我曾经工作中的案例, ...

  5. [数字华容道] Html+css+js 实现小游戏

    [数字华容道] Html+css+js 实现小游戏 效果图 代码预览 在线预览地址 代码示例 <!DOCTYPE html> <html> <head> <m ...

  6. Flutter(三):Flutter App 可行性分析

    一.生态建设 第三方Package https://pub.dev/packages?sort=popularity 截止2021年4月,第三方库达到17000+ 二.Devops 代码风格检查 An ...

  7. 7款优秀的AI搜索引擎工具推荐

    AI搜索引擎不仅能够理解复杂的查询语句,还能够通过学习用户的搜索习惯和偏好,提供更加个性化的搜索结果.本篇文章将介绍7款在这一领域表现出色的AI搜索引擎工具,它们各有特色,但都致力于为用户提供更加智能 ...

  8. 莫烦tensorflow学习记录 (5)什么是过拟合 (Overfitting)

    什么是过拟合 (Overfitting) 莫烦讲的非常通俗易懂可以看看https://mofanpy.com/tutorials/machine-learning/tensorflow/intro-o ...

  9. QShop商城-开发规范

    QShop商城-项目介绍 QShop商城,是全新推出的一款轻量级.高性能.前后端分离的电商系统,支持微信小程序,前后端源码100%开源,完美支持二次开发,让您快速搭建个性化独立商城. 技术架构:.Ne ...

  10. 【Effective C++】设计与声明——考虑写出一个不抛异常的swap函数

    wap是个有趣的函数.原本它只是STL的一部分,而后成为异常安全性编程的脊柱,以及用来实现自我赋值可能性的一个常见机制.所谓swap两对象值,就是将两对象的值交换. 典型实现 缺省情况下的swap动作 ...