├── .babelrc
├── .eslintrc
├── .flowconfig
├── .gitignore
├── .npmignore
├── LICENSE
├── README.md
├── package.json
├── renovate.json
└── source
├── index.js
├── read-blob-file.js
└── spinner.js
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 | "presets": ["es2015", "es2016", "stage-0", "react"],
3 | "plugins": [
4 | "transform-merge-sibling-variables",
5 | "transform-minify-booleans",
6 | "transform-node-env-inline",
7 | "transform-react-constant-elements",
8 | "transform-react-inline-elements",
9 | "transform-remove-console",
10 | "transform-remove-debugger",
11 | "transform-simplify-comparison-operators",
12 | "transform-undefined-to-void"
13 | ]
14 | }
15 |
--------------------------------------------------------------------------------
/.eslintrc:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "airbnb",
3 | "parser": "babel-eslint",
4 | "plugins": [
5 | "import",
6 | "jsx-a11y",
7 | "react",
8 | "flowtype",
9 | ],
10 | "env": {
11 | "browser": true,
12 | "node": true,
13 | },
14 | "parserOptions": {
15 | "ecmaFeatures": {
16 | "jsx": true,
17 | "es6": true,
18 | "classes": true,
19 | },
20 | },
21 | "settings": {
22 | "react": {
23 | "pragma": "React",
24 | "version": "15.3",
25 | },
26 | "flowtype": {
27 | "onlyFilesWithFlowAnnotation": false,
28 | },
29 | },
30 | "rules": {
31 | "flowtype/require-parameter-type": 1,
32 | "flowtype/require-return-type": [
33 | 1,
34 | "always",
35 | {
36 | "annotateUndefined": "never"
37 | },
38 | ],
39 | "flowtype/space-after-type-colon": [
40 | 1,
41 | "always",
42 | ],
43 | "flowtype/space-before-type-colon": [
44 | 1,
45 | "never",
46 | ],
47 | "flowtype/type-id-match": [
48 | 1,
49 | "^([A-Z][a-z0-9]+)+Type$",
50 | ],
51 | },
52 | }
53 |
--------------------------------------------------------------------------------
/.flowconfig:
--------------------------------------------------------------------------------
1 | [include]
2 | source/
3 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Logs
2 | logs
3 | *.log
4 | npm-debug.log*
5 |
6 | # Runtime data
7 | pids
8 | *.pid
9 | *.seed
10 |
11 | # Directory for instrumented libs generated by jscoverage/JSCover
12 | lib-cov
13 |
14 | # Coverage directory used by tools like istanbul
15 | coverage
16 |
17 | # nyc test coverage
18 | .nyc_output
19 |
20 | # Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
21 | .grunt
22 |
23 | # node-waf configuration
24 | .lock-wscript
25 |
26 | # Compiled binary addons (http://nodejs.org/api/addons.html)
27 | build/Release
28 |
29 | # Dependency directories
30 | node_modules
31 | jspm_packages
32 |
33 | # Optional npm cache directory
34 | .npm
35 |
36 | # Optional REPL history
37 | .node_repl_history
38 |
39 | build
40 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | source
3 | .babelrc
4 | .eslintrc
5 | .gitignore
6 | logs
7 | *.log
8 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2016 Sergio Daniel Xalambrí
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 | # react-lazy-image
2 |
3 | Component to render images and lazyload them if are in the viewport (or near to them).
4 |
5 | This component extends from `React.PureComponent` so it needs React.js v15.3.0 or superior.
6 |
7 |
8 |
9 |
10 |
11 | ## Installation
12 | ```bash
13 | npm i -S react-lazy-image
14 | ```
15 |
16 | ## Usage example
17 | ```jsx
18 | import Image from 'react-lazy-image';
19 |
20 | const image = ;
21 | ```
22 |
23 | ## API
24 |
25 | ### `onLayout({ element: Object }): void`
26 | Called everytime the component is rendered or updated. Default: `() => {}`.
27 |
28 | ### `onError({ element: Object, error: Error }): void`
29 | Called if the request to load the image failed. Default: `() => {}`.
30 |
31 | ### `onLoad({ element: Object }): void`
32 | Called after the load ended (either successfully or not). Default: `() => {}`.
33 |
34 | ### `onLoadEnd({ element: Object }): void`
35 | Called after the load ended successfully. Default: `() => {}`.
36 |
37 | ### `onLoadStart({ element: Object }): void`
38 | Called when the request started. Default: `() => {}`.
39 |
40 | ### `onAbort({ element: Object }): void`
41 | Called if the load of the image was aborted. Default: `() => {}`.
42 |
43 | ### `onProgress({ element: Object }): void`
44 | Called everytime the AJAX progress event is dispatched. Default: `() => {}`.
45 |
46 | ### `offset: ?number`
47 | Set the amount of pixel near the viewport the component should be to start the image load. Default: `0`.
48 |
49 | ### `source: string`
50 | The image source path to load.
51 |
52 | ### `defaultSource: ?string`
53 | The default image source path or base64. If isn't defined then it uses a SVG animated spinner.
54 |
55 | ### `type: ?string`
56 | The format type of the image (`png`, `svg+xml`, `jpg` or `gif`). Default: `*`.
57 |
58 | ### `minLoaded: ?number`
59 | The minimum download percentaje to avoid aborting the request if the image leaves the viewport. Default: `50`.
60 |
61 | ## Common `img` attributes
62 | This component allow the usage of common `img` attributes like `alt`, `width`, `className`, etc. So you can use it as a normal `img` tag, just change `src` to `source` and (if you want to) add the other optional props.
63 |
64 | ## License
65 | The MIT License (MIT)
66 |
67 | Copyright (c) 2015 Sergio Daniel Xalambrí
68 |
69 | Permission is hereby granted, free of charge, to any person obtaining a copy
70 | of this software and associated documentation files (the "Software"), to deal
71 | in the Software without restriction, including without limitation the rights
72 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
73 | copies of the Software, and to permit persons to whom the Software is
74 | furnished to do so, subject to the following conditions:
75 |
76 | The above copyright notice and this permission notice shall be included in all
77 | copies or substantial portions of the Software.
78 |
79 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
80 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
81 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
82 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
83 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
84 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
85 | SOFTWARE.
86 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-lazy-image",
3 | "version": "1.1.0",
4 | "description": "Component to render images and lazyload them if are in the viewport (or near to them).",
5 | "main": "build/index.js",
6 | "scripts": {
7 | "lint": "eslint lib/index.js",
8 | "prebuild": "npm run lint",
9 | "build": "babel source -d build",
10 | "prepublish": "npm run build"
11 | },
12 | "repository": {
13 | "type": "git",
14 | "url": "git+https://github.com/sergiodxa/react-lazy-image.git"
15 | },
16 | "keywords": [
17 | "react",
18 | "lazyload",
19 | "image",
20 | "ajax"
21 | ],
22 | "author": "Sergio Daniel Xalambrí (http://sergio.xalambri.com.ar/)",
23 | "license": "MIT",
24 | "bugs": {
25 | "url": "https://github.com/sergiodxa/react-lazy-image/issues"
26 | },
27 | "homepage": "https://github.com/sergiodxa/react-lazy-image#readme",
28 | "peerDependencies": {
29 | "react": "^15.0.0"
30 | },
31 | "devDependencies": {
32 | "babel": "6.23.0",
33 | "babel-cli": "6.26.0",
34 | "babel-eslint": "6.1.2",
35 | "babel-plugin-transform-merge-sibling-variables": "6.8.0",
36 | "babel-plugin-transform-minify-booleans": "6.9.4",
37 | "babel-plugin-transform-node-env-inline": "6.8.0",
38 | "babel-plugin-transform-react-constant-elements": "6.23.0",
39 | "babel-plugin-transform-react-inline-elements": "6.22.0",
40 | "babel-plugin-transform-remove-console": "6.8.0",
41 | "babel-plugin-transform-remove-debugger": "6.9.4",
42 | "babel-plugin-transform-simplify-comparison-operators": "6.8.0",
43 | "babel-plugin-transform-undefined-to-void": "6.9.4",
44 | "babel-preset-es2015": "6.24.1",
45 | "babel-preset-es2016": "6.24.1",
46 | "babel-preset-react": "6.24.1",
47 | "babel-preset-stage-0": "6.24.1",
48 | "eslint": "3.2.2",
49 | "eslint-config-airbnb": "10.0.1",
50 | "eslint-plugin-flowtype": "2.4.0",
51 | "eslint-plugin-import": "1.16.0",
52 | "eslint-plugin-jsx-a11y": "2.0.1",
53 | "eslint-plugin-react": "6.0.0"
54 | }
55 | }
56 |
--------------------------------------------------------------------------------
/renovate.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": [
3 | "config:base"
4 | ]
5 | }
6 |
--------------------------------------------------------------------------------
/source/index.js:
--------------------------------------------------------------------------------
1 | import React, { Component } from 'react';
2 |
3 | import defaultSource from './spinner.js';
4 | import readBlobFile from './read-blob-file.js';
5 |
6 | type RectType = {
7 | bottom: number,
8 | height: number,
9 | left: number,
10 | right: number,
11 | top: number,
12 | width: number,
13 | };
14 |
15 | type PropType = {
16 | onLayout(event: { element: Object }): void,
17 | onError(event: { element: Object, error: Error }): void,
18 | onLoad(event: { element: Object }): void,
19 | onLoadEnd(event: { element: Object }): void,
20 | onLoadStart(event: { element: Object }): void,
21 | onAbort(event: { element: Object }): void,
22 | onProgress(event: { element: Object }): void,
23 | offset: ?number,
24 | source: string,
25 | defaultSource: ?string,
26 | type: ?string,
27 | minLoaded: ?number,
28 | };
29 |
30 | /**
31 | * Instance the default props
32 | * @type {PropTypes}
33 | */
34 | const defaultProps: PropType = {
35 | onLayout: () => {},
36 | onError: () => {},
37 | onLoad: () => {},
38 | onLoadEnd: () => {},
39 | onLoadStart: () => {},
40 | onAbort: () => {},
41 | onProgress: () => {},
42 | offset: 0,
43 | defaultSource,
44 | type: '*',
45 | minLoaded: 50,
46 | };
47 |
48 | /**
49 | * Component to render an image using LazyLoad to request it only if the component is in
50 | * the viewport and abort the load if the component leaves the viewport
51 | */
52 | class Image extends Component {
53 | static defaultProps = defaultProps;
54 |
55 | /**
56 | * Bind component methods to `this`
57 | * @param {PropType} props [description]
58 | */
59 | constructor(props: PropType) {
60 | super(props);
61 | // bind function methods
62 | this.checkViewport = this.checkViewport.bind(this);
63 | this.setRef = this.setRef.bind(this);
64 |
65 | this.handleLoadEnd = this.handleLoadEnd.bind(this);
66 | this.handleAbort = this.handleAbort.bind(this);
67 | this.handleProgress = this.handleProgress.bind(this);
68 | this.handleError = this.handleError.bind(this);
69 | this.handleLoadStart = this.handleLoadStart.bind(this);
70 | this.handleLoad = this.handleLoad.bind(this);
71 | }
72 |
73 |
74 | /**
75 | * The state contains the image base64 string to use
76 | * @type {Object}
77 | */
78 | state: { image: string } = {
79 | image: this.props.defaultSource,
80 | };
81 |
82 |
83 | /**
84 | * Call the onLayout after the component is renderer
85 | * Create the XHR object
86 | * Set the event listener for the scroll event to check if the component is in viewport
87 | * Check the viewport one time to now if it's already in it
88 | */
89 | componentDidMount() {
90 | this.props.onLayout({ element: this });
91 |
92 | this.request = new XMLHttpRequest();
93 | this.request.responseType = 'arraybuffer';
94 |
95 | window.addEventListener('scroll', this.checkViewport);
96 | this.checkViewport();
97 | }
98 |
99 |
100 | /**
101 | * Call the onLayout callback if the component is re-rendered
102 | */
103 | componentDidUpdate() {
104 | this.props.onLayout({ element: this });
105 | }
106 |
107 |
108 | /**
109 | * Remove scroll event listener if the component is unmounted
110 | */
111 | componentWillUnmount() {
112 | window.removeEventListener('scroll', this.checkViewport);
113 | this.request.abort();
114 | }
115 |
116 |
117 | /**
118 | * Get the reference to the image
119 | * @param {Element} element The DOM element to set the reference
120 | */
121 | setRef(element: Element): Object {
122 | this.element = element;
123 | return this;
124 | }
125 |
126 |
127 | /**
128 | * Fetch the image and save it in the state
129 | */
130 | fetch(): void {
131 | // set request event handlers
132 | this.request.onloadstart = this.handleLoadStart;
133 | this.request.onprogress = this.handleProgress;
134 | this.request.onload = this.handleLoad;
135 | this.request.onloadend = this.handleLoadEnd;
136 | this.request.onabort = this.handleAbort;
137 | this.request.onerror = this.handleError;
138 |
139 | // open AJAX request
140 | this.request.open('GET', this.props.source);
141 | // send request
142 | return this.request.send();
143 | }
144 |
145 |
146 | /**
147 | * Handle load start event
148 | * @param {Object} event Request start event object
149 | */
150 | handleLoadStart(event: Object): void {
151 | this.isRequesting = true;
152 | const element = this;
153 |
154 | if (event.lengthComputable) {
155 | this.progress.loaded = 0;
156 | this.progress.total = event.total;
157 | }
158 |
159 | return this.props.onLoadStart({ element });
160 | }
161 |
162 |
163 | /**
164 | * Set the loaded progress (and the total)
165 | * @param {Object} event The progress event data
166 | */
167 | handleProgress(event: Object): void {
168 | const element = this;
169 |
170 | if (event.lengthComputable) {
171 | this.progress.loaded = event.loaded;
172 | }
173 |
174 | return this.props.onProgress({ element });
175 | }
176 |
177 |
178 | /**
179 | * Handle the XHR load event
180 | * @param {Object} event The load event data
181 | */
182 | handleLoad(): void {
183 | const element: Image = this;
184 | this.isRequesting = false;
185 | return this.props.onLoad({ element });
186 | }
187 |
188 |
189 | /**
190 | * If the request ended successful get the response as a blob object, transform it to base64,
191 | * remove the scroll event listener and update the `state.image` value`
192 | * @param {Object} event The load end event data
193 | */
194 | handleLoadEnd(): void {
195 | const element: Image = this;
196 |
197 | // if the request status es between 200 and 300
198 | if (this.request.status >= 200 && this.request.status < 300) {
199 | // transform response to a blob
200 | const blob: Blob = new Blob([this.request.response], { type: `image/${this.props.type}` });
201 | // read blob as a base64 string
202 | return readBlobFile(blob)
203 | .then((image: string) => {
204 | // set the image base64 string in the state
205 | this.setState({ image }, (): void => {
206 | // remove event scroll listener
207 | window.removeEventListener('scroll', this.checkViewport);
208 | // set the component as not requesting anymore
209 | this.isRequesting = false;
210 | // call the `onLoadEnd` callback
211 | return this.props.onLoadEnd({ element });
212 | });
213 | })
214 | .catch((error: Object): void => {
215 | // set the component as not requesting anymore
216 | this.isRequesting = false;
217 | // if an error happens call the `onError` callback
218 | return this.props.onError({ error, element });
219 | });
220 | }
221 |
222 | return null;
223 | }
224 |
225 |
226 | /**
227 | * Handle request error event
228 | * @param {Object} event The error event data
229 | */
230 | handleError(event: Object): void {
231 | const element: Image = this;
232 | this.isRequesting = false;
233 | return this.props.onError({ element, error: new Error(event.response) });
234 | }
235 |
236 |
237 | /**
238 | * Handle the request abort event
239 | * @param {Object} event The abort event data
240 | */
241 | handleAbort(): void {
242 | const element = this;
243 | this.isRequesting = false;
244 | return this.props.onAbort({ element });
245 | }
246 |
247 |
248 | /**
249 | * Check if the component is in the current viewport and load the image
250 | */
251 | checkViewport(): Promise | null {
252 | if (this.isInViewport && !this.isRequesting) {
253 | // if is in viewport and is not requesting start fetching
254 | return this.fetch();
255 | }
256 | if (!this.isInViewport && this.isRequesting) {
257 | // if isn't in viewport and is requesting
258 | if (this.amountLoaded < this.props.minLoaded || isNaN(this.amountLoaded)) {
259 | // if the amount loaded is lower than the `this.props.minLoaded`
260 | // or is NaN abort the request
261 | return this.request.abort();
262 | }
263 | }
264 | return null;
265 | }
266 |
267 |
268 | /**
269 | * Define the prop types
270 | * @type {PropType}
271 | */
272 | props: PropType;
273 |
274 |
275 | /**
276 | * If the component is requesting an image or not
277 | * @type {Boolean}
278 | */
279 | isRequesting: boolean = false;
280 |
281 |
282 | /**
283 | * Progress loaded and total amount of bytes
284 | * @type {Object}
285 | */
286 | progress: { loaded: number, total: number } = {
287 | loaded: 0,
288 | total: 1,
289 | };
290 |
291 |
292 | /**
293 | * The progress of amount lodaded
294 | * @return {Number} The percentaje loaded
295 | */
296 | get amountLoaded(): number {
297 | return (this.progress.loaded * 100) / this.progress.total;
298 | }
299 |
300 |
301 | /**
302 | * Check if the component is in the viewport
303 | * @return {Boolean} If the component is in viewport
304 | */
305 | get isInViewport(): boolean {
306 | // get element position in viewport
307 | const rect: RectType = this.element.getBoundingClientRect();
308 | // get viewport height and width
309 | const viewportHeight: number = (window.innerHeight || document.documentElement.clientHeight);
310 | const viewportWidth: number = (window.innerWidth || document.documentElement.clientWidth);
311 | // check if the element is in the viewport (or near to them)
312 | return (
313 | rect.bottom >= (0 - this.props.offset) &&
314 | rect.right >= (0 - this.props.offset) &&
315 | rect.top < (viewportHeight + this.props.offset) &&
316 | rect.left < (viewportWidth + this.props.offset)
317 | );
318 | }
319 |
320 |
321 | /**
322 | * Get the images props without the component own props
323 | * @return {Object} The filtered props
324 | */
325 | get imgProps(): Object {
326 | const ownProps = [
327 | 'onLayout',
328 | 'onError',
329 | 'onLoad',
330 | 'onLoadEnd',
331 | 'onLoadStart',
332 | 'onAbort',
333 | 'onProgress',
334 | 'resizeMode',
335 | 'source',
336 | 'defaultSource',
337 | 'offset',
338 | 'minLoaded',
339 | ];
340 |
341 | return Object
342 | .keys(this.props)
343 | .filter((propName: string): boolean => ownProps.indexOf(propName) === -1)
344 | .reduce(
345 | (props: PropType, propName: string): PropType => ({
346 | ...props,
347 | [propName]: this.props[propName],
348 | }),
349 | {},
350 | );
351 | }
352 |
353 |
354 | /**
355 | * Component renderer method
356 | * @return {Object} The image JSX element
357 | */
358 | render(): Object {
359 | return (
360 |
365 | );
366 | }
367 | }
368 |
369 | export default Image;
370 |
--------------------------------------------------------------------------------
/source/read-blob-file.js:
--------------------------------------------------------------------------------
1 | /**
2 | * Read Blob file as a base64 string
3 | * @param {Blob} file The Blob file instance to read
4 | * @return {Promise} A reader promise
5 | */
6 | function readBlobFile(file: Blob): Promise {
7 | function promiseHandler(resolve: Function, reject: Function): FileReader {
8 | const reader: FileReader = new FileReader();
9 | reader.readAsDataURL(file);
10 | reader.onloadend = function onReadEnd(): void {
11 | if (reader.error) return reject(reader.error);
12 | return resolve(reader.result);
13 | };
14 | return reader;
15 | }
16 |
17 | return new Promise(promiseHandler);
18 | }
19 |
20 | export default readBlobFile;
21 |
--------------------------------------------------------------------------------
/source/spinner.js:
--------------------------------------------------------------------------------
1 | export default '';
2 |
--------------------------------------------------------------------------------