├── .babelrc
├── .eslintrc.js
├── .flowconfig
├── .gitignore
├── .npmignore
├── .travis.yml
├── README.md
├── __tests__
└── index.js
├── demo
├── app.js
├── index.html
├── inline.js
└── webpack.config.js
├── package.json
├── prettier.config.js
├── src
├── index.d.ts
└── index.js
├── webpack.config.umd.js
└── yarn.lock
/.babelrc:
--------------------------------------------------------------------------------
1 | {
2 |
3 | "presets": ["@babel/preset-env", "@babel/preset-react", "@babel/preset-flow"],
4 | "plugins": [
5 | "@babel/plugin-proposal-class-properties",
6 | "@babel/plugin-proposal-object-rest-spread"
7 | ]
8 | }
9 |
--------------------------------------------------------------------------------
/.eslintrc.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | "extends": [
3 | "plugin:flowtype/recommended",
4 | "plugin:react/recommended",
5 | "prettier",
6 | "prettier/flowtype",
7 | "prettier/react"
8 | ],
9 | "plugins": [
10 | "flowtype",
11 | "react",
12 | "prettier"
13 | ],
14 | "parserOptions": {
15 | "sourceType": "module",
16 | "ecmaFeatures": {
17 | "jsx": true
18 | }
19 | },
20 | "env": {
21 | "es6": true,
22 | "node": true
23 | },
24 | "rules": {
25 | "prettier/prettier": "error"
26 | }
27 | }
--------------------------------------------------------------------------------
/.flowconfig:
--------------------------------------------------------------------------------
1 | [libs]
2 |
3 | [ignore]
4 | .*/node_modules/fbjs.*
5 | .*/node_modules/*
6 |
7 | [include]
8 | index.js
9 |
10 | [options]
11 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | node_modules
2 | npm-debug.log
3 | dist.js
4 | umd
5 | .DS_Store
6 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | demo
2 |
--------------------------------------------------------------------------------
/.travis.yml:
--------------------------------------------------------------------------------
1 | language: node_js
2 | node_js:
3 | - "6"
4 | - "8"
5 | dist: trusty # needs Ubuntu Trusty
6 | sudo: false # no need for virtualization.
7 | script:
8 | - yarn test
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # React Progressive Image
2 |
3 | [![Maintenance Status][maintenance-image]](#maintenance-status)
4 |
5 | [`react-progressive-image`](https://www.npmjs.com/package/react-progressive-image) React component for progressive image loading
6 |
7 | ### Install
8 |
9 | ```bash
10 | $ yarn add react-progressive-image
11 | ```
12 |
13 | The UMD build is also available on [unpkg](https://unpkg.com):
14 |
15 | ```html
16 |
17 | ```
18 |
19 | If you use the UMD build you can find the library on `window.ReactProgressiveImage`.
20 |
21 | ### Examples
22 |
23 | #### Simple
24 |
25 | ```jsx
26 |
27 | {src => }
28 |
29 | ```
30 |
31 | #### With Delay
32 |
33 | ```jsx
34 |
39 | {src => }
40 |
41 | ```
42 |
43 | #### With loading argument
44 |
45 | ```jsx
46 |
47 | {(src, loading) => (
48 |
49 | )}
50 |
51 | ```
52 |
53 | #### With srcSet
54 |
55 | ```jsx
56 |
64 | {(src, _loading, srcSetData) => (
65 |
71 | )}
72 |
73 | ```
74 |
75 | #### Component As Placeholder
76 |
77 | If you want to use a component, such as a loading spinner, as a placeholder, you can make use of the `loading` argument in the render callback. It will be true while the main image is loading and false once it has fully loaded. Keep in mind that the `placeholder` props is `required`, so you will need to explicitly declare an empty string as it's value if you plan on using a component in the render callback.
78 |
79 | ```jsx
80 | const dominantImageColor = '#86356B';
81 | const placeholder = (
82 |
85 | );
86 |
87 |
88 | {(src, loading) => {
89 | return loading ? placeholder : ;
90 | }}
91 | ;
92 | ```
93 |
94 | #### Progressive Enhancement and No JavaScript
95 |
96 | Since this component relies on JavaScript to replace the placeholder src with the full image src, you should use a fallback image if your application supports environments that do not have JavaScript enabled or is progressively enhanced.
97 |
98 | You can do this by adding the fallback image inside of a `` tag in the render callback you provide as the `ProgressiveImage` component's child.
99 |
100 | ```jsx
101 |
102 | {src => {
103 | return (
104 |
105 |
106 |
107 |
108 |
109 |
110 | );
111 | }}
112 |
113 | ```
114 |
115 | ### Props
116 |
117 | | Name | Type | Required | Description |
118 | | ----------- | -------------------------------------- | -------- | ----------------------------------------------- |
119 | | children | `function` | `true` | returns `src`, `loading`, and `srcSetData` |
120 | | delay | `number` | `false` | time in milliseconds before src image is loaded |
121 | | onError | `function` | `false` | returns error event |
122 | | placeholder | `string` | `true` | the src of the placeholder image |
123 | | src | `string` | `true` | the src of the main image |
124 | | srcSetData | `{srcSet: "string", sizes: "string" }` | `false` | srcset and sizes to be applied to the image |
125 |
126 | ## Maintenance Status
127 |
128 | **Archived:** This project is no longer maintained by Formidable. We are no longer responding to issues or pull requests unless they relate to security concerns. We encourage interested developers to fork this project and make it their own!
129 |
130 | [maintenance-image]: https://img.shields.io/badge/maintenance-archived-red.svg
--------------------------------------------------------------------------------
/__tests__/index.js:
--------------------------------------------------------------------------------
1 | import 'raf/polyfill';
2 | import React from 'react';
3 | import { configure, mount } from 'enzyme';
4 | import Adapter from 'enzyme-adapter-react-16';
5 | import ProgressiveImage from '../src';
6 |
7 | configure({ adapter: new Adapter() });
8 |
9 | const src = 'https://image.xyz/source';
10 | const placeholder = 'https://image.xyz/placeholder';
11 | const srcSetData = {
12 | srcSet: 'srcSet',
13 | sizes: 'sizes'
14 | };
15 |
16 | const mountProgressiveImage = (renderFn, delay) => {
17 | const defaultRender = image => {
18 | return ;
19 | };
20 | const render = renderFn || defaultRender;
21 | return mount(
22 |
28 | {render}
29 |
30 | );
31 | };
32 |
33 | describe('react-progressive-image', () => {
34 | beforeEach(() => {
35 | global.Image = Image;
36 | });
37 | it('exports a React component', () => {
38 | expect(typeof ProgressiveImage).toBe('function');
39 | });
40 | it('throws if not provided a function as a child', () => {
41 | /* eslint-disable no-console */
42 | const _error = console.error;
43 | console.error = jest.fn(() => {});
44 | try {
45 | expect(() => {
46 | mount(
47 |
48 | Uh oh!
49 |
50 | );
51 | }).toThrow(`ProgressiveImage requires a function as its only child`);
52 | } finally {
53 | console.error = _error;
54 | }
55 | /* eslint-enable no-console */
56 | });
57 | it('creates an instance of Image when mounted', () => {
58 | const wrapper = mountProgressiveImage();
59 | const instance = wrapper.instance();
60 | expect(instance.image.constructor).toBe(HTMLImageElement);
61 | });
62 | it('sets the onload property on the Image instance', () => {
63 | const wrapper = mountProgressiveImage();
64 | const instance = wrapper.instance();
65 | expect(instance.image.onload).toEqual(instance.onLoad);
66 | });
67 | it('sets the onerror property on the Image instance', () => {
68 | const wrapper = mountProgressiveImage();
69 | const instance = wrapper.instance();
70 | expect(instance.image.onerror).toEqual(instance.onError);
71 | });
72 | it('sets the src property on the Image instance', () => {
73 | const wrapper = mountProgressiveImage();
74 | const instance = wrapper.instance();
75 | expect(instance.image.src).toEqual(src);
76 | });
77 | it('sets the srcSet property on the Image instance', () => {
78 | const wrapper = mountProgressiveImage();
79 | const instance = wrapper.instance();
80 | expect(instance.image.srcset).toEqual(srcSetData.srcSet);
81 | });
82 | it('sets the sizes property on the Image instance', () => {
83 | const wrapper = mountProgressiveImage();
84 | const instance = wrapper.instance();
85 | expect(instance.image.sizes).toEqual(srcSetData.sizes);
86 | });
87 | it('renders placeholder image on initial render', () => {
88 | const render = jest.fn(src => );
89 | const wrapper = mountProgressiveImage(render);
90 | expect(render.mock.calls[0][0]).toEqual(placeholder);
91 | });
92 | it('renders src image on second render', () => {
93 | const render = jest.fn(src => );
94 | const wrapper = mountProgressiveImage(render);
95 | wrapper.instance().loadImage(src);
96 | wrapper.instance().onLoad();
97 | expect(render.mock.calls[1][0]).toEqual(src);
98 | });
99 | it('sets loading to false after src image is loaded', () => {
100 | const render = jest.fn(src => );
101 | const wrapper = mountProgressiveImage(render);
102 | expect(render.mock.calls[0][1]).toEqual(true);
103 | wrapper.instance().loadImage(src);
104 | wrapper.instance().onLoad();
105 | expect(render.mock.calls[1][1]).toEqual(false);
106 | });
107 | it('does not immediately set image if delay prop exists', () => {
108 | const delay = 3000;
109 | const render = jest.fn(src => );
110 | const wrapper = mountProgressiveImage(render, delay);
111 | wrapper.instance().loadImage(src);
112 | wrapper.instance().onLoad();
113 | expect(wrapper.instance().state.image).toEqual(placeholder);
114 | });
115 | it('sets image after delay passes if delay prop exists', () => {
116 | const delay = 3000;
117 | const render = jest.fn(src => );
118 | const wrapper = mountProgressiveImage(render, delay);
119 | wrapper.instance().loadImage(src);
120 | wrapper.instance().onLoad();
121 | setTimeout(() => {
122 | expect(wrapper.instance().state.image).toEqual(src);
123 | }, delay + 1);
124 | });
125 | });
126 |
--------------------------------------------------------------------------------
/demo/app.js:
--------------------------------------------------------------------------------
1 | import React from 'react';
2 | import ReactDOM from 'react-dom';
3 | import ProgressiveImage from '../src/index.js';
4 | import inline from './inline';
5 | const SM = 'https://farm2.staticflickr.com/1853/42944460370_e749cd18eb_b.jpg';
6 | const MD = 'https://farm2.staticflickr.com/1867/30884025408_7e6907d2e4_b.jpg';
7 | const LG = 'https://farm2.staticflickr.com/1875/42944459590_170ddf9fc8_b.jpg';
8 |
9 | const centerAlign = {
10 | alignItems: 'center',
11 | display: 'flex',
12 | flexDirection: 'column',
13 | justifyContent: 'center'
14 | };
15 |
16 | const containerStyle = {
17 | ...centerAlign,
18 | overflow: 'hidden',
19 | position: 'relative',
20 | maxWidth: 1440,
21 | margin: '0 auto'
22 | };
23 |
24 | const imageStyle = {
25 | height: '100vh',
26 | minWidth: '100%'
27 | };
28 |
29 | const textContainerStyle = {
30 | ...centerAlign,
31 | bottom: 0,
32 | left: 0,
33 | position: 'absolute',
34 | right: 0,
35 | top: 0
36 | };
37 |
38 | const textStyle = {
39 | color: '#fff',
40 | fontFamily: 'sans-serif',
41 | fontSize: '2.5em',
42 | textTransform: 'uppercase'
43 | };
44 |
45 | class App extends React.Component {
46 | render() {
47 | return (
48 |
49 |
50 |
51 | React
52 | Progressive
53 | Image
54 |
55 |
56 |
64 | {(image, loading, srcSetData) => {
65 | return (
66 |
72 | );
73 | }}
74 |
75 |
76 | );
77 | }
78 | }
79 |
80 | ReactDOM.render( , document.getElementById('content'));
81 |
--------------------------------------------------------------------------------
/demo/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
7 |
8 |
9 | Demo
10 |
11 |
12 |
13 |
14 |
17 |
18 |
22 |
23 |
33 |
34 |
35 |
--------------------------------------------------------------------------------
/demo/inline.js:
--------------------------------------------------------------------------------
1 | // eslint-disable max-len
2 |
3 | export default ``;
4 |
--------------------------------------------------------------------------------
/demo/webpack.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | mode: 'development',
3 | context: __dirname,
4 | entry: ['./app.js'],
5 | output: {
6 | path: __dirname,
7 | filename: 'bundle.js'
8 | },
9 | module: {
10 | rules: [
11 | {
12 | test: /\.js$/,
13 | exclude: /node_modules/,
14 | loader: 'babel-loader'
15 | }
16 | ]
17 | }
18 | };
19 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "react-progressive-image",
3 | "version": "0.6.0",
4 | "description": "Progressive image loading for React",
5 | "main": "dist.js",
6 | "typings": "src/index.d.ts",
7 | "scripts": {
8 | "build": "babel src/index.js --out-file dist.js",
9 | "umd": "webpack --config webpack.config.umd.js",
10 | "lint": "eslint {__tests__,src}/*.js",
11 | "flow": "flow",
12 | "test:only": "jest",
13 | "test": "npm run lint && npm run flow && npm run test:only",
14 | "prepublish": "npm run pretty && npm run test && npm run build && npm run umd",
15 | "demo": "webpack-dev-server --inline --hot --config demo/webpack.config.js --content-base demo/",
16 | "pretty": "prettier --write '{__tests__,src}/*.js'"
17 | },
18 | "keywords": [
19 | "react",
20 | "progressive",
21 | "images",
22 | "loading"
23 | ],
24 | "repository": "https://github.com/FormidableLabs/react-progressive-image.git",
25 | "author": "Brandon Dail ",
26 | "license": "MIT",
27 | "devDependencies": {
28 | "@babel/cli": "^7.0.0",
29 | "@babel/core": "^7.0.1",
30 | "@babel/plugin-proposal-class-properties": "^7.0.0",
31 | "@babel/plugin-proposal-object-rest-spread": "^7.0.0",
32 | "@babel/preset-env": "^7.0.0",
33 | "@babel/preset-flow": "^7.0.0",
34 | "@babel/preset-react": "^7.0.0",
35 | "babel-core": "^7.0.0-0",
36 | "babel-eslint": "^9.0.0",
37 | "babel-jest": "^23.6.0",
38 | "babel-loader": "^8.0.2",
39 | "enzyme": "^3.6.0",
40 | "enzyme-adapter-react-16": "^1.5.0",
41 | "eslint": "^5.6.0",
42 | "eslint-config-formidable": "^4.0.0",
43 | "eslint-config-prettier": "^3.0.1",
44 | "eslint-plugin-babel": "^5.2.0",
45 | "eslint-plugin-filenames": "^1.3.2",
46 | "eslint-plugin-flowtype": "^2.50.0",
47 | "eslint-plugin-import": "^2.14.0",
48 | "eslint-plugin-prettier": "^2.6.2",
49 | "eslint-plugin-react": "^7.11.1",
50 | "flow-bin": "^0.81.0",
51 | "jest": "^23.6.0",
52 | "prettier": "^1.14.2",
53 | "raf": "^3.4.0",
54 | "react": "^16.5.1",
55 | "react-dom": "^16.5.1",
56 | "react-test-renderer": "^16.5.1",
57 | "uglifyjs-webpack-plugin": "^2.0.0",
58 | "webpack": "^4.19.0",
59 | "webpack-cli": "^3.1.0",
60 | "webpack-dev-server": "^3.1.8"
61 | },
62 | "peerDependencies": {
63 | "react": "^15.0.0-0 || ^16.0.0-0",
64 | "react-dom": "^15.0.0-0 || ^16.0.0-0"
65 | },
66 | "sideEffects": false,
67 | "dependencies": {}
68 | }
69 |
--------------------------------------------------------------------------------
/prettier.config.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | singleQuote: true,
3 | trailingComma: "none"
4 | };
5 |
--------------------------------------------------------------------------------
/src/index.d.ts:
--------------------------------------------------------------------------------
1 | declare module 'react-progressive-image' {
2 | export interface ProgressiveImageProps {
3 | delay?: number;
4 | onError?: (errorEvent: Event) => void;
5 | placeholder: string;
6 | src: string;
7 | srcSetData?: {
8 | srcSet: string;
9 | sizes: string;
10 | };
11 | }
12 |
13 | export interface ProgressiveImageState {
14 | image: string;
15 | loading: boolean;
16 | srcSetData?: {
17 | srcSet: string;
18 | sizes: string;
19 | };
20 | }
21 |
22 | export default class ProgressiveImage extends React.Component<
23 | ProgressiveImageProps,
24 | ProgressiveImageState
25 | > {}
26 | }
27 |
--------------------------------------------------------------------------------
/src/index.js:
--------------------------------------------------------------------------------
1 | // @flow
2 |
3 | import * as React from 'react';
4 |
5 | type SrcSetData = {
6 | srcSet: string,
7 | sizes: string
8 | };
9 |
10 | type Props = {
11 | children: (string, boolean, SrcSetData) => React.Node,
12 | delay?: number,
13 | onError?: (errorEvent: Event) => void,
14 | placeholder: string,
15 | src: string,
16 | srcSetData?: SrcSetData
17 | };
18 |
19 | type State = {
20 | image: string,
21 | loading: boolean,
22 | srcSetData: SrcSetData
23 | };
24 |
25 | export default class ProgressiveImage extends React.Component {
26 | image: HTMLImageElement;
27 | constructor(props: Props) {
28 | super(props);
29 | this.state = {
30 | image: props.placeholder,
31 | loading: true,
32 | srcSetData: { srcSet: '', sizes: '' }
33 | };
34 | }
35 |
36 | componentDidMount() {
37 | const { src, srcSetData } = this.props;
38 | this.loadImage(src, srcSetData);
39 | }
40 |
41 | componentDidUpdate(prevProps: Props) {
42 | const { src, placeholder, srcSetData } = this.props;
43 | // We only invalidate the current image if the src has changed.
44 | if (src !== prevProps.src) {
45 | this.setState({ image: placeholder, loading: true }, () => {
46 | this.loadImage(src, srcSetData);
47 | });
48 | }
49 | }
50 |
51 | componentWillUnmount() {
52 | if (this.image) {
53 | this.image.onload = null;
54 | this.image.onerror = null;
55 | }
56 | }
57 |
58 | loadImage = (src: string, srcSetData?: SrcSetData) => {
59 | // If there is already an image we nullify the onload
60 | // and onerror props so it does not incorrectly set state
61 | // when it resolves
62 | if (this.image) {
63 | this.image.onload = null;
64 | this.image.onerror = null;
65 | }
66 | const image = new Image();
67 | this.image = image;
68 | image.onload = this.onLoad;
69 | image.onerror = this.onError;
70 | image.src = src;
71 | if (srcSetData) {
72 | image.srcset = srcSetData.srcSet;
73 | image.sizes = srcSetData.sizes;
74 | }
75 | };
76 |
77 | onLoad = () => {
78 | // use this.image.src instead of this.props.src to
79 | // avoid the possibility of props being updated and the
80 | // new image loading before the new props are available as
81 | // this.props.
82 |
83 | if (this.props.delay) {
84 | this.setImageWithDelay();
85 | } else {
86 | this.setImage();
87 | }
88 | };
89 |
90 | setImageWithDelay = () => {
91 | setTimeout(() => {
92 | this.setImage();
93 | }, this.props.delay);
94 | };
95 |
96 | setImage = () => {
97 | this.setState({
98 | image: this.image.src,
99 | loading: false,
100 | srcSetData: {
101 | srcSet: this.image.srcset || '',
102 | sizes: this.image.sizes || ''
103 | }
104 | });
105 | };
106 |
107 | onError = (errorEvent: Event) => {
108 | const { onError } = this.props;
109 | if (onError) {
110 | onError(errorEvent);
111 | }
112 | };
113 |
114 | render() {
115 | const { image, loading, srcSetData } = this.state;
116 | const { children } = this.props;
117 |
118 | if (!children || typeof children !== 'function') {
119 | throw new Error(`ProgressiveImage requires a function as its only child`);
120 | }
121 |
122 | return children(image, loading, srcSetData);
123 | }
124 | }
125 |
--------------------------------------------------------------------------------
/webpack.config.umd.js:
--------------------------------------------------------------------------------
1 | /* globals __dirname */
2 | const webpack = require('webpack');
3 | const path = require('path');
4 | const UglifyJsPlugin = require('uglifyjs-webpack-plugin');
5 |
6 | module.exports = {
7 | mode: 'production',
8 | entry: path.join(__dirname, 'src/index.js'),
9 | externals: [
10 | {
11 | 'react': {
12 | root: 'React',
13 | commonjs2: 'react',
14 | commonjs: 'react',
15 | amd: 'react'
16 | },
17 | 'react-dom': {
18 | root: 'ReactDom',
19 | commonjs2: 'react-dom',
20 | commonjs: 'react-dom',
21 | amd: 'react-dom'
22 | }
23 | }
24 | ],
25 | output: {
26 | library: 'ReactProgressiveImage',
27 | libraryTarget: 'umd',
28 | filename: 'react-progressive-image.min.js',
29 | path: path.join(__dirname, 'umd')
30 | },
31 | module: {
32 | rules: [{
33 | test: /\.js$/,
34 | exclude: /node_modules/,
35 | loader: 'babel-loader'
36 | }]
37 | },
38 | optimization: {
39 | minimizer: [
40 | new UglifyJsPlugin({
41 | uglifyOptions: {
42 | warnings: false
43 | }})
44 | ]
45 | },
46 | plugins: [
47 | new webpack.DefinePlugin({
48 | 'process.env.NODE_ENV': JSON.stringify('production')
49 | }),
50 | new webpack.SourceMapDevToolPlugin({
51 | filename: '[file].map'
52 | })
53 | ]
54 | };
55 |
--------------------------------------------------------------------------------