├── .babelrc
├── .gitignore
├── .travis.yml
├── LICENSE
├── README.md
├── config
├── webpack.config.base.js
├── webpack.config.dev.js
└── webpack.config.prod.js
├── example
├── index.html
└── index.js
├── package.json
├── src
├── assets
│ └── loading.png
└── index.tsx
├── tea.yaml
└── tsconfig.json
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": [
3 | [
4 | "@babel/preset-env",
5 | {
6 | "targets": "> 0.25%, not dead"
7 | }
8 | ],
9 | "@babel/preset-react"
10 | ],
11 | "plugins": [
12 | "@babel/plugin-transform-runtime",
13 | "@babel/plugin-transform-modules-commonjs",
14 | [
15 | "@babel/plugin-proposal-decorators",
16 | {
17 | "legacy": true
18 | }
19 | ],
20 | "@babel/plugin-proposal-class-properties",
21 | "@babel/plugin-proposal-object-rest-spread"
22 | ]
23 | }
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
2 |
3 | # dependencies
4 | /node_modules
5 | /.pnp
6 | .pnp.js
7 | /dist
8 |
9 | # testing
10 | /coverage
11 |
12 | # production
13 | /build
14 |
15 | # misc
16 | .DS_Store
17 | .env.local
18 | .env.development.local
19 | .env.test.local
20 | .env.production.local
21 |
22 | npm-debug.log*
23 | yarn-debug.log*
24 | yarn-error.log*
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | sudo: required
2 |
3 | language: node_js
4 |
5 | node_js:
6 | - 8
7 |
8 | install:
9 | - npm install
10 | - npm install -g codecov
11 |
12 | before_script:
13 | - npm i -g npm
14 |
15 | script:
16 | - codecov
17 |
18 | after_success:
19 | - bash <(curl -s https://codecov.io/bash)
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2019 Jun
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 | react-hook-lazy-image 图片懒加载
14 |
15 | 一个应用React Hooks基于IntersectionObserver API实现的图片懒加载组件,具有如下特点:
16 |
17 | - 简单好用,配置灵活
18 | - 兼容主流浏览器
19 | - 使用React Hooks实现
20 | - 使用TypeScript
21 |
22 |
23 |
24 |
25 |
26 | ## 安装
27 | ```jsx
28 | // 使用yarn安装
29 | yarn add react-hook-lazy-image
30 |
31 | // 使用npm安装
32 | npm install react-hook-lazy-image -S
33 | ```
34 |
35 | ## 使用
36 | ```jsx
37 | import React from 'react';
38 | import LazyImage from 'react-hook-lazy-image';
39 |
40 | const list = [ // image src url ...];
41 |
42 | const LazyLoad:React.FC = () => {
43 | const clientHeight = window.innerHeight;
44 |
45 | const style = {height: 300, width: 400, border: '1px solid #000', margin: '10px'};
46 |
47 | return (
48 |
49 | {
50 | list.map((item, i) => (
51 |
52 | ))
53 | }
54 |
55 | )
56 | }
57 |
58 | export default LazyLoad;
59 | ```
60 |
61 |
62 | ## API
63 | | 属性 | 类型 | 是否必填 | 默认值 | 描述 |
64 | | --- | --- | --- | --- | --- |
65 | | src | string | 否 | - | 图片的真实src,不传默认显示占位图 |
66 | | defaultSrc | string | 否 | LazyImage占位图 | 默认渲染的占位图src地址 |
67 | | style | object | 否 | { width: 300, height: 200 } | 图片样式 |
68 | | options | object | 否 | - | 自定义配置,通过配置可以指定图片与特定的父级元素交叉时才去加载真实图片,祥见[IntersectionObserver](https://developer.mozilla.org/zh-CN/docs/Web/API/IntersectionObserver/IntersectionObserver) |
69 |
70 |
71 |
72 | ## 最后
73 | 如果觉得好用,请点个star支持一下哈~
74 | 如果在使用过程中有任何问题或者建议,可以在项目中提交issue,或者通过邮件与我取得联系,我会及时处理~
email:lujun5068@gmail.com
75 |
76 |
77 |
78 |
--------------------------------------------------------------------------------
/config/webpack.config.base.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | resolve: {
3 | extensions: ['.tsx', '.ts', '.js', '.jsx']
4 | },
5 | module: {
6 | rules: [
7 | {
8 | test: /\.jsx?$/,
9 | use: 'babel-loader',
10 | exclude: /node_modules/
11 | },
12 | {
13 | test: /\.tsx?$/,
14 | use: 'ts-loader',
15 | exclude: /node_modules/
16 | },
17 | {
18 | test: /\.(jpe?g|png|gif)/,
19 | use: {
20 | loader: 'file-loader',
21 | options: {
22 | name: 'images/[name][hash:8].[ext]'
23 | }
24 | }
25 | }
26 | ]
27 | }
28 | }
--------------------------------------------------------------------------------
/config/webpack.config.dev.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const merge = require('webpack-merge');
3 | const HtmlWebpackPlugin = require('html-webpack-plugin');
4 | const baseConfig = require('./webpack.config.base');
5 |
6 | const devConfig = {
7 | mode: 'development',
8 | entry: './example/index.js',
9 | output: {
10 | path: path.join(__dirname, '../example'),
11 | filename: '[name].js'
12 | },
13 | devtool: 'source-map',
14 | plugins: [
15 | new HtmlWebpackPlugin({
16 | template: './example/index.html'
17 | })
18 | ],
19 | devServer: {
20 | port: 4000,
21 | open: true,
22 | hot: true
23 | }
24 | }
25 |
26 | module.exports = merge(devConfig, baseConfig);
--------------------------------------------------------------------------------
/config/webpack.config.prod.js:
--------------------------------------------------------------------------------
1 | const path = require('path');
2 | const webpack = require('webpack');
3 | const merge = require('webpack-merge');
4 | const baseConfig = require('./webpack.config.base');
5 |
6 | const prodConfig = {
7 | mode: 'production',
8 | entry: './src/index.tsx',
9 | output: {
10 | publicPath: '/',
11 | filename: 'index.js',
12 | path: path.resolve(__dirname, '../dist'),
13 | libraryTarget: 'umd',
14 | libraryExport: 'default',
15 | },
16 | externals: {
17 | react: {
18 | root: "React",
19 | commonjs2: "react",
20 | commonjs: "react",
21 | amd: "react"
22 | },
23 | "react-dom": {
24 | root: "ReactDOM",
25 | commonjs2: "react-dom",
26 | commonjs: "react-dom",
27 | amd: "react-dom"
28 | }
29 | },
30 | plugins: [
31 | new webpack.optimize.ModuleConcatenationPlugin(),
32 | ],
33 | }
34 |
35 | module.exports = merge(prodConfig, baseConfig);
--------------------------------------------------------------------------------
/example/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 | lazy image
8 |
9 |
10 |
11 |
12 |
--------------------------------------------------------------------------------
/example/index.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import { render } from 'react-dom';
3 | import LazyImage from '../src/index';
4 |
5 | const list = [
6 | 'https://cdn.imgbin.com/1/6/10/imgbin-arctic-wolf-furry-fandom-carnivora-red-wolf-others-ZHnTC2tmQMPzs37cWQ7tPqp4R.jpg',
7 | 'https://cdn.imgbin.com/11/3/6/imgbin-london-soldier-icon-british-soldiers-two-royal-guards-illustration-DnXRqwwC7iWRUpyfneT3dqUpc.jpg',
8 | 'https://cdn.imgbin.com/17/18/0/imgbin-warrington-council-cheshire-east-london-borough-of-richmond-upon-thames-london-boroughs-others-hJL5XyKYPb4CjHKDUjJ1S9yWV.jpg',
9 | 'https://cdn.imgbin.com/21/6/22/imgbin-fermanagh-and-omagh-london-borough-of-lambeth-county-fermanagh-london-boroughs-others-LCDYu4VaxCS2kxqWHeaW4wgPf.jpg',
10 | 'https://cdn.imgbin.com/1/10/9/imgbin-london-borough-of-southwark-hayes-city-of-westminster-cities-of-london-and-westminster-geography-foreign-candidates-bwQZvxJYgVDGLtsWkkVrHgYY1.jpg',
11 | 'https://cdn.imgbin.com/12/10/17/imgbin-london-borough-of-hammersmith-and-fulham-silhouette-dog-yoga-sySpTkfUATdQ5A4xKzprWBTbG.jpg',
12 | 'https://cdn.imgbin.com/5/4/21/imgbin-london-borough-of-camden-hammersmith-commercial-building-facade-london-bridge-trading-bpv9mKugy0CnbgTak0Va2Wfy9.jpg',
13 | 'https://cdn.imgbin.com/17/2/24/imgbin-london-borough-of-ealing-london-boroughs-west-london-perceval-house-others-Qtw8cdkHXF7JUW9k7mezLELAR.jpg',
14 | 'https://cdn.imgbin.com/4/18/13/imgbin-kingston-upon-thames-royal-borough-of-kensington-and-chelsea-royal-borough-of-greenwich-london-borough-of-ealing-london-borough-of-hounslow-others-MvTGFyw0UDkbw6s6B5bWBTfZc.jpg',
15 | 'https://cdn.imgbin.com/25/22/16/imgbin-kingston-upon-thames-coat-of-arms-surrey-county-council-law-HA2zNQcxjxmMTM0FjHL4ZrsbD.jpg',
16 | 'https://cdn.imgbin.com/2/15/11/imgbin-county-armagh-counties-of-ireland-republic-of-ireland-coat-of-arms-of-ireland-creative-harp-fvtXjkya9NkurELWVaZndG6jL.jpg',
17 | 'https://cdn.imgbin.com/10/9/18/imgbin-wetsuit-yulex-counties-of-ireland-parthenium-argentatum-county-donegal-surfer-AUZFTFWmktMjJPSa05ZRJfcmB.jpg',
18 | 'https://cdn.imgbin.com/22/16/18/imgbin-counties-of-the-kingdom-of-hungary-zalaegerszeg-list-of-regions-of-hungary-town-with-county-rights-hungarian-Lzu3F4V6eSi0KDQWt53ydV9Fp.jpg',
19 | 'https://cdn.imgbin.com/22/9/1/imgbin-fifa-online-3-need-for-speed-edge-mabinogi-vindictus-nexus-the-kingdom-of-the-winds-festivals-cy4Q8uiTF95ihz6ZBjyCN10tZ.jpg',
20 | 'https://cdn.imgbin.com/25/8/24/imgbin-need-for-speed-shift-need-for-speed-most-wanted-shift-2-unleashed-xbox-360-need-for-speed-hot-pursuit-need-for-speed-8RjDqSUsEzgJFskmbVAR6QjWa.jpg',
21 | 'https://cdn.imgbin.com/23/20/11/imgbin-need-for-speed-hot-pursuit-2-need-for-speed-iii-hot-pursuit-need-for-speed-most-wanted-xbox-360-need-for-speed-69VBuefWmKD6c4P2d1hC2Hv9m.jpg',
22 | ]
23 |
24 |
25 |
26 | const LazyLoad = () => {
27 | const clientHeight = window.innerHeight;
28 |
29 | const style = {height: 300, width: 400, border: '1px solid #000', margin: '10px auto'};
30 |
31 | return (
32 |
33 | {
34 | list.map((item, i) => (
35 |
36 | ))
37 | }
38 |
39 | )
40 | }
41 |
42 | render( , document.getElementById('root'));
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-hook-lazy-image",
3 | "version": "1.1.1",
4 | "description": "A lazy loading image component that applies React Hooks based on IntersectionObserver API",
5 | "main": "dist/index.js",
6 | "files": [
7 | "dist"
8 | ],
9 | "scripts": {
10 | "build": "rm -rf dist && webpack --config ./config/webpack.config.prod.js",
11 | "pub": "npm run build && npm publish",
12 | "dev": "webpack-dev-server --config ./config/webpack.config.dev.js",
13 | "report-coverage": "codecov"
14 | },
15 | "repository": "https://github.com/lujun5068/lazy-image",
16 | "keywords": [
17 | "lazy",
18 | "lazy-image",
19 | "lazy-load-image",
20 | "load-image",
21 | "image",
22 | "懒加载",
23 | "图片懒加载"
24 | ],
25 | "author": "lujun_smile",
26 | "license": "MIT",
27 | "dependencies": {
28 | "react": "^16.12.0",
29 | "react-dom": "^16.12.0"
30 | },
31 | "devDependencies": {
32 | "@babel/cli": "^7.7.0",
33 | "@babel/core": "^7.7.2",
34 | "@babel/plugin-proposal-class-properties": "^7.0.0",
35 | "@babel/plugin-proposal-decorators": "^7.0.0",
36 | "@babel/plugin-transform-modules-commonjs": "^7.0.0",
37 | "@babel/plugin-transform-runtime": "^7.0.0",
38 | "@babel/preset-env": "^7.0.0",
39 | "@babel/preset-react": "^7.0.0",
40 | "@babel/runtime": "^7.7.4",
41 | "@types/node": "^12.12.8",
42 | "@types/react": "^16.9.11",
43 | "@types/react-dom": "^16.9.4",
44 | "babel-loader": "^8.0.6",
45 | "file-loader": "^4.2.0",
46 | "html-webpack-plugin": "next",
47 | "ts-loader": "^6.2.1",
48 | "typescript": "^3.7.2",
49 | "webpack": "^4.41.2",
50 | "webpack-cli": "^3.3.10",
51 | "webpack-dev-server": "^3.1.5",
52 | "webpack-merge": "^4.1.4"
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/src/assets/loading.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/lujun5068/lazy-image/c885d859337563129fde04f72c958dd99cc3e7de/src/assets/loading.png
--------------------------------------------------------------------------------
/src/index.tsx:
--------------------------------------------------------------------------------
1 | import React, {useEffect, useRef} from 'react';
2 | interface LazyImageProps {
3 | /**
4 | * 图片的真实src,不传默认显示占位图
5 | */
6 | src?: string;
7 |
8 | /**
9 | * 默认的占位图片,可以自己传,默认使用lazyImage的占位图
10 | */
11 | defaultSrc?: string;
12 |
13 | /**
14 | * 图片样式
15 | */
16 | style?: React.CSSProperties;
17 |
18 | /**
19 | * 自定义配置项
20 | */
21 | options?: IntersectionObserver
22 | }
23 |
24 | const defaultUrl = '';
25 |
26 | const defaultImgStyle:React.CSSProperties = {
27 | width: 300,
28 | height: 200
29 | }
30 |
31 | const LazyImage:React.FC = ({src= defaultUrl, style=defaultImgStyle, defaultSrc= defaultUrl, options={}}) => {
32 | const imgRef = useRef(null);
33 |
34 | const setDefaultSrc = () => {
35 | imgRef.current.src = defaultSrc;
36 | }
37 |
38 | const proxyImage = () => {
39 | const img = new Image();
40 | img.onload = () => {
41 | imgRef.current.src = src;
42 | }
43 | return {
44 | setSrc() {
45 | img.src = src;
46 | }
47 | }
48 | }
49 |
50 |
51 | useEffect(() => {
52 | setDefaultSrc();
53 |
54 | const observer = new IntersectionObserver((entries) => {
55 | entries.forEach(item => {
56 | if (item.isIntersecting) {
57 | proxyImage().setSrc();
58 | if (src) {
59 | observer.unobserve(item.target);
60 | }
61 | }
62 | })
63 | }, {...options})
64 |
65 | observer.observe(imgRef.current);
66 |
67 | return () => {
68 | observer.unobserve(imgRef.current);
69 | }
70 |
71 | }, [])
72 |
73 | return (
74 |
75 |
![]()
76 |
77 | )
78 | }
79 |
80 | export default LazyImage;
--------------------------------------------------------------------------------
/tea.yaml:
--------------------------------------------------------------------------------
1 | # https://tea.xyz/what-is-this-file
2 | ---
3 | version: 1.0.0
4 | codeOwners:
5 | - '0xee090f2d1Ab50b0115DE74600390e018ee895291'
6 | quorum: 1
7 |
--------------------------------------------------------------------------------
/tsconfig.json:
--------------------------------------------------------------------------------
1 | // @ts-ignore
2 | {
3 | "compilerOptions": {
4 | "outDir": "./dist",
5 | "target": "es5",
6 | "jsx": "react",
7 | "lib": ["dom", "esnext"],
8 | "module": "esnext",
9 | "sourceMap": true,
10 | "allowSyntheticDefaultImports": true,
11 | "moduleResolution": "node",
12 | "rootDir": "src",
13 | "forceConsistentCasingInFileNames": true,
14 | "noImplicitReturns": true,
15 | "suppressImplicitAnyIndexErrors": true,
16 | "noUnusedLocals": true,
17 | "experimentalDecorators": true,
18 | "declaration": true
19 | },
20 | "exclude": [
21 | "example",
22 | "node_modules",
23 | "build",
24 | "dist",
25 | "lib",
26 | "tests"
27 | ]
28 | }
29 |
30 |
--------------------------------------------------------------------------------