From f96fddab66cb36e347b787f94e24aacd29ed4f8f Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Wed, 21 Mar 2018 21:07:33 +1100 Subject: [PATCH 01/11] Updated to support React 16.0.0 - starting from 0.4.0, it requires react ^16.0.0 - refactored basic example - added file type checking in onDragEnter --- .babelrc | 2 +- examples/basic/client.js | 298 ++++++++++++++++++------------------ examples/basic/package.json | 7 +- package.json | 8 +- src/Receiver.js | 4 + 5 files changed, 159 insertions(+), 160 deletions(-) diff --git a/.babelrc b/.babelrc index 6306550..30366ba 100644 --- a/.babelrc +++ b/.babelrc @@ -1,3 +1,3 @@ { - "presets": ['es2015', 'react', 'stage-0'] + "presets": ['es2015', 'react'] } \ No newline at end of file diff --git a/examples/basic/client.js b/examples/basic/client.js index 80a5a79..ee77d88 100644 --- a/examples/basic/client.js +++ b/examples/basic/client.js @@ -1,162 +1,158 @@ -var React = require('react'); -var ReactDOM = require('react-dom'); -var FileUploader = require('../../lib'); -var _ = require('lodash'); - -var MyComponent = React.createClass({ - getInitialState: function() { - return { - isPanelOpen: false, - isDragOver: false, - files: [] - } - }, - - openPanel: function() { - this.setState({ isPanelOpen: true }); - }, - - closePanel: function() { - this.setState({ isPanelOpen: false }); - }, - - onDragOver: function(e) { - // your codes here: - // if you want to check if the files are dragged over - // a specific DOM node - //if (e.target === node) { - // this.setState({ - // isDragOver: true - // }); - //} - }, - - onFileDrop: function({ target }, files) { - let node = ReactDOM.findDOMNode(this.refs.uploadPanel); - if (target != node) { - return false; - } - - files.map(function(_file) { - if (_file.size > 1000 * 1000) { - _file.status = FileUploader.status.FAILED; - _file.error = 'file size exceeded maximum' - } - }); +import React, { Component } from 'react'; +import ReactDOM from 'react-dom'; +import _ from 'lodash'; +var FileUploader = require('react-file-uploader'); + +class MyComponent extends Component { + constructor(props) { + super(props); + + this.state = { + isPanelOpen: false, + isDragOver: false, + files: [], + }; + + this.uploadPanel = undefined; + } + + openPanel = () => { + this.setState({ isPanelOpen: true }); + } + + closePanel = () => { + this.setState({ isPanelOpen: false }); + } + + onDragOver = (e) => { + // your codes here: + // if you want to check if the files are dragged over + // a specific DOM node + const node = ReactDOM.findDOMNode(this.uploadPanel); - this.setState({ - files: this.state.files.concat(files) - }); + this.setState({ + isDragOver: e.target === node, + }); + } - // if you want to close the panel upon file drop - this.closePanel(); - }, + onFileDrop = ({ target }, files) => { + const node = ReactDOM.findDOMNode(this.uploadPanel); - onFileProgress: function(file) { - var files = this.state.files; + if (target !== node) { + this.closePanel(); + return false; + } - files.map(function(_file) { - if (_file.id === file.id) { - _file = file; - } + files.forEach(item => { + if (item.size > 1000 * 1000) { + item.status = FileUploader.status.FAILED; + item.error = 'file size exceeded maximum'; + } }); this.setState({ - files: files + files: this.state.files.concat(files), }); -}, - onFileUpdate: function(file) { - var files = this.state.files; + // if you want to close the panel upon file drop + this.closePanel(); + } - files.map(function(_file) { - if (_file.id === file.id) { - _file = file; - } - }); - - this.setState({ - files: files - }); - }, - - _getStatusString: function(status) { - switch (status) { - case -1: - return 'failed'; - break; - - case 0: - return 'pending'; - break; - - case 1: - return 'uploading'; - break; - - case 2: - return 'uploaded'; - break; - } - }, - - render: function() { - var _this = this; - - return ( -
-

{ this.props.title }

-

You can upload files with size with 1 MB at maximum

- -

- { - !this.state.isDragOver ? 'Drop here' : 'Files detected' - } -

-
-
-

Upload List

- - { - this.state.files.map(function(file, index) { - return ( - -
- { file.name } -
- { file.id } - { file.type } - { file.size / 1000 / 1000 } MB - { file.progress }% - - {_this._getStatusString(file.status)} - - { file.error } -
-
-
- ) - }) - } -
-
-
- ); - } -}); + onFileProgress = (file) => { + const { files = [] } = this.state; + const newFiles = files.map(item => item.id === file.id ? file : item); -ReactDOM.render(, document.getElementById('app')); \ No newline at end of file + this.setState({ + files: newFiles, + }); + } + + onFileUpdate = (file) => { + const { files = [] } = this.state; + const newFiles = files.map(item => item.id === file.id ? file : item); + + this.setState({ + files: newFiles, + }); + } + + setUploadPanelRef = (ref) => { + this.uploadPanel = ref; + } + + getStatusString(status) { + switch (status) { + case -1: + return 'failed'; + + case 0: + return 'pending'; + + case 1: + return 'uploading'; + + case 2: + return 'uploaded'; + + default: + return ''; + } + } + + render() { + return ( +
+

{ this.props.title }

+

You can upload files with size with 1 MB at maximum

+ +

+ { + !this.state.isDragOver ? 'Drop here' : 'Files detected' + } +

+
+
+

Upload List

+ + { + this.state.files.map((file, index) => ( + +
+ {file.name} +
+ {file.id} + {file.type} + {file.size / 1000 / 1000} MB + {file.progress}% + + {this.getStatusString(file.status)} + + {file.error} +
+
+
+ )) + } +
+
+
+ ); + } +} + +ReactDOM.render(, document.getElementById('app')); diff --git a/examples/basic/package.json b/examples/basic/package.json index dbfed66..acfcf24 100644 --- a/examples/basic/package.json +++ b/examples/basic/package.json @@ -1,19 +1,20 @@ { "scripts": { "test": "echo \"Error: no test specified\" && exit 1", - "build": "browserify client.js -o bundle.js -t [ babelify --presets [ es2015 react] ]" + "build": "browserify client.js -o bundle.js -t [ babelify --presets [ es2015 react] --plugins [ transform-class-properties ] ]" }, "dependencies": { "connect-multiparty": "^2.0.0", "express": "^4.13.3", "forever": "^0.15.1", "lodash": "^4.13.1", - "react": "^0.14.3", - "react-dom": "^0.14.3" + "react": "^16.0.0", + "react-dom": "^16.0.0" }, "devDependencies": { "babel-preset-es2015": "^6.1.18", "babel-preset-react": "^6.1.18", + "babel-plugin-transform-class-properties": "^6.24.1", "babelify": "^7.2.0", "browserify": "^12.0.1" } diff --git a/package.json b/package.json index 0cb7986..588f7d6 100644 --- a/package.json +++ b/package.json @@ -36,7 +36,6 @@ "babel-jest": "*", "babel-preset-es2015": "^6.1.18", "babel-preset-react": "^6.1.18", - "babel-preset-stage-0": "^6.1.18", "eslint": "^2.11.1", "eslint-config-airbnb": "^9.0.1", "eslint-plugin-import": "^1.8.1", @@ -46,12 +45,11 @@ "jsdom": "^7.0.2", "nock": "^8.0.0", "prop-types": "^15.5.10", - "react": "^0.14.8 || ^15.0.0", - "react-addons-test-utils": "^0.14.8 || ^15.0.0", - "react-dom": "^0.14.8 || ^15.0.0" + "react": "^16.0.0", + "react-dom": "^16.0.0" }, "peerDependencies": { - "react": "^0.14.8 || ^15.0.0" + "react": "^16.0.0" }, "jest": { "scriptPreprocessor": "/node_modules/babel-jest", diff --git a/src/Receiver.js b/src/Receiver.js index 64fafc1..e1e684b 100644 --- a/src/Receiver.js +++ b/src/Receiver.js @@ -42,6 +42,10 @@ class Receiver extends Component { } onDragEnter(e) { + if (!e.dataTransfer.types.includes('Files')) { + return; + } + const dragLevel = this.state.dragLevel + 1; this.setState({ dragLevel }); From 1fc5dba8e3e82a7d70ed466daa7baeb04c140e17 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Wed, 21 Mar 2018 21:56:49 +1100 Subject: [PATCH 02/11] fixed missing DragEvent on Safari --- src/Receiver.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Receiver.js b/src/Receiver.js index e1e684b..e7ac127 100644 --- a/src/Receiver.js +++ b/src/Receiver.js @@ -24,8 +24,8 @@ class Receiver extends Component { componentDidMount() { invariant( - !!window.DragEvent && !!window.DataTransfer, - 'Upload end point must be provided to upload files' + (!!window.DragEvent || !!window.Event) && !!window.DataTransfer, + 'Browser does not support DnD events or File API.' ); this.wrapper.addEventListener('dragenter', this.onDragEnter); From 693829b95e0a1b636004053283e93fb386e4c72a Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Thu, 22 Mar 2018 12:04:49 +1100 Subject: [PATCH 03/11] Exposed more request configs in UploadManager - added `reqConfigs` as a prop which accepts `accept`, `method`, `timeout` and `withCredentials` properties - added `formDataParser` as a prop which allows custom fields to be sent through in the request - referred to the suggestions in https://github.com/lionng429/react-file-uploader/pull/9/ --- src/UploadManager.js | 46 ++++++++++++++++++++++++++++++++++++++------ 1 file changed, 40 insertions(+), 6 deletions(-) diff --git a/src/UploadManager.js b/src/UploadManager.js index fffb115..f23538a 100644 --- a/src/UploadManager.js +++ b/src/UploadManager.js @@ -31,9 +31,16 @@ class UploadManager extends Component { upload(url, file) { const { + reqConfigs: { + accept = 'application/json', + method = 'post', + timeout, + withCredentials = false, + }, onUploadStart, onUploadEnd, onUploadProgress, + formDataParser, uploadErrorHandler, uploadHeader = {}, } = this.props; @@ -42,15 +49,28 @@ class UploadManager extends Component { onUploadStart(assign(file, { status: uploadStatus.UPLOADING })); } - const formData = new FormData(); - formData.append('file', file); + let formData = new FormData(); + formData = formDataParser(formData, file); + + let request = request[method.toLowerCase()](url) + .accept(accept) + .set(uploadHeader); + + if (file.type) { + request.type(file.type); + } + + if (timeout) { + request.timeout(timeout); + } + + if (withCredentials) { + request.withCredentials(); + } debug(`start uploading file#${file.id} to ${url}`, file); request - .post(url) - .accept('application/json') - .set(uploadHeader) .send(formData) .on('progress', ({ percent }) => { if (typeof onUploadProgress === 'function') { @@ -104,17 +124,31 @@ UploadManager.propTypes = { PropTypes.string, PropTypes.arrayOf(PropTypes.string), ]), + formDataParser: PropTypes.func, onUploadStart: PropTypes.func, onUploadProgress: PropTypes.func, onUploadEnd: PropTypes.func.isRequired, + reqConfigs: PropTypes.shape({ + accept: PropTypes.string, + method: PropTypes.string, + timeout: PropTypes.shape({ + response: PropTypes.number, + deadline: PropTypes.number, + }), + withCredentials: PropTypes.bool, + }), style: PropTypes.object, - uploadErrorHandler: PropTypes.func.isRequired, + uploadErrorHandler: PropTypes.func, uploadUrl: PropTypes.string.isRequired, uploadHeader: PropTypes.object, }; UploadManager.defaultProps = { component: 'ul', + formDataParser: (formData, file) => { + formData.append('file', file); + return formData; + }, uploadErrorHandler: (err, res) => { let error = null; const body = clone(res.body); From bd19cd0c61b20d17bd9d1548d7755b1dafbfc55b Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Thu, 22 Mar 2018 14:57:16 +1100 Subject: [PATCH 04/11] Updated ESLint settings - removed airbnb and related dependencies - fixed prop-types dependency - fixed exmaples --- .eslintrc | 20 +++++++++++++++++++- examples/basic/client.js | 25 ++++++++++++++++--------- examples/basic/package.json | 3 +-- package.json | 9 +++------ src/Receiver.js | 6 +++--- src/UploadHandler.js | 5 ++--- src/UploadManager.js | 9 +++------ src/__tests__/Receiver-test.js | 2 +- src/__tests__/UploadManager-test.js | 2 +- 9 files changed, 49 insertions(+), 32 deletions(-) diff --git a/.eslintrc b/.eslintrc index e89ba50..d6a0709 100644 --- a/.eslintrc +++ b/.eslintrc @@ -1,3 +1,21 @@ { - "extends": "airbnb" + "env": { + "browser": true, + "node": true, + "es6": true + }, + "parserOptions": { + "ecmaVersion": 6, + "sourceType": "module", + "ecmaFeatures": { + "jsx": true + } + }, + "plugins": [ + "react" + ], + "extends": [ + "eslint:recommended", + "plugin:react/recommended" + ] } \ No newline at end of file diff --git a/examples/basic/client.js b/examples/basic/client.js index ee77d88..06971ca 100644 --- a/examples/basic/client.js +++ b/examples/basic/client.js @@ -1,7 +1,7 @@ import React, { Component } from 'react'; import ReactDOM from 'react-dom'; import _ from 'lodash'; -var FileUploader = require('react-file-uploader'); +import * as FileUploader from '../../src/index'; class MyComponent extends Component { constructor(props) { @@ -14,17 +14,24 @@ class MyComponent extends Component { }; this.uploadPanel = undefined; + this.openPanel = this.openPanel.bind(this); + this.closePanel = this.closePanel.bind(this); + this.onDragOver = this.onDragOver.bind(this); + this.onFileDrop = this.onFileDrop.bind(this); + this.onFileProgress = this.onFileProgress.bind(this); + this.onFileUpdate = this.onFileUpdate.bind(this); + this.setUploadPanelRef = this.setUploadPanelRef.bind(this); } - openPanel = () => { + openPanel() { this.setState({ isPanelOpen: true }); } - closePanel = () => { + closePanel() { this.setState({ isPanelOpen: false }); } - onDragOver = (e) => { + onDragOver(e) { // your codes here: // if you want to check if the files are dragged over // a specific DOM node @@ -35,7 +42,7 @@ class MyComponent extends Component { }); } - onFileDrop = ({ target }, files) => { + onFileDrop({ target }, files) { const node = ReactDOM.findDOMNode(this.uploadPanel); if (target !== node) { @@ -58,7 +65,7 @@ class MyComponent extends Component { this.closePanel(); } - onFileProgress = (file) => { + onFileProgress(file) { const { files = [] } = this.state; const newFiles = files.map(item => item.id === file.id ? file : item); @@ -67,7 +74,7 @@ class MyComponent extends Component { }); } - onFileUpdate = (file) => { + onFileUpdate(file) { const { files = [] } = this.state; const newFiles = files.map(item => item.id === file.id ? file : item); @@ -76,7 +83,7 @@ class MyComponent extends Component { }); } - setUploadPanelRef = (ref) => { + setUploadPanelRef(ref) { this.uploadPanel = ref; } @@ -133,7 +140,7 @@ class MyComponent extends Component { this.state.files.map((file, index) => (
- {file.name} +
{file.name}
{file.id} {file.type} diff --git a/examples/basic/package.json b/examples/basic/package.json index acfcf24..5846c8d 100644 --- a/examples/basic/package.json +++ b/examples/basic/package.json @@ -1,7 +1,7 @@ { "scripts": { "test": "echo \"Error: no test specified\" && exit 1", - "build": "browserify client.js -o bundle.js -t [ babelify --presets [ es2015 react] --plugins [ transform-class-properties ] ]" + "build": "browserify client.js -o bundle.js -t [ babelify --presets [ es2015 react ] ]" }, "dependencies": { "connect-multiparty": "^2.0.0", @@ -14,7 +14,6 @@ "devDependencies": { "babel-preset-es2015": "^6.1.18", "babel-preset-react": "^6.1.18", - "babel-plugin-transform-class-properties": "^6.24.1", "babelify": "^7.2.0", "browserify": "^12.0.1" } diff --git a/package.json b/package.json index 588f7d6..b4d8b34 100644 --- a/package.json +++ b/package.json @@ -36,15 +36,11 @@ "babel-jest": "*", "babel-preset-es2015": "^6.1.18", "babel-preset-react": "^6.1.18", - "eslint": "^2.11.1", - "eslint-config-airbnb": "^9.0.1", - "eslint-plugin-import": "^1.8.1", - "eslint-plugin-jsx-a11y": "^1.3.0", - "eslint-plugin-react": "^5.1.1", + "eslint": "^4.19.1", + "eslint-plugin-react": "^7.7.0", "jest-cli": "*", "jsdom": "^7.0.2", "nock": "^8.0.0", - "prop-types": "^15.5.10", "react": "^16.0.0", "react-dom": "^16.0.0" }, @@ -72,6 +68,7 @@ "debug": "^2.2.0", "invariant": "^2.2.0", "lodash": ">=3.10.1", + "prop-types": "^15.5.10", "shortid": "^2.2.6", "superagent": "^1.4.0" } diff --git a/src/Receiver.js b/src/Receiver.js index e7ac127..01e9bf6 100644 --- a/src/Receiver.js +++ b/src/Receiver.js @@ -24,7 +24,7 @@ class Receiver extends Component { componentDidMount() { invariant( - (!!window.DragEvent || !!window.Event) && !!window.DataTransfer, + (window.DragEvent || window.Event) && window.DataTransfer, 'Browser does not support DnD events or File API.' ); @@ -77,10 +77,10 @@ class Receiver extends Component { const files = []; - if (!!e.dataTransfer) { + if (e.dataTransfer) { const fileList = e.dataTransfer.files || []; - for (let i = 0; i < fileList.length; i ++) { + for (let i = 0; i < fileList.length; i++) { fileList[i].id = shortid.generate(); fileList[i].status = status.PENDING; fileList[i].progress = 0; diff --git a/src/UploadHandler.js b/src/UploadHandler.js index bb015ae..14a0e69 100644 --- a/src/UploadHandler.js +++ b/src/UploadHandler.js @@ -35,11 +35,11 @@ class UploadHandler extends Component { } render() { - const { component, key, customClass, style } = this.props; + const { component, customClass, style } = this.props; return React.createElement( component, - { key, className: classNames(customClass), style }, + { className: classNames(customClass), style }, this.props.children ); } @@ -57,7 +57,6 @@ UploadHandler.propTypes = { PropTypes.arrayOf(PropTypes.string), ]), file: PropTypes.object.isRequired, - key: PropTypes.string, style: PropTypes.object, upload: PropTypes.func, }; diff --git a/src/UploadManager.js b/src/UploadManager.js index f23538a..38510a3 100644 --- a/src/UploadManager.js +++ b/src/UploadManager.js @@ -5,7 +5,7 @@ import classNames from 'classnames'; import assign from 'lodash/assign'; import bindKey from 'lodash/bindKey'; import clone from 'lodash/clone'; -import request from 'superagent'; +import superagent from 'superagent'; import uploadStatus from './constants/status'; const debug = require('debug')('react-file-upload:UploadManager'); @@ -52,14 +52,10 @@ class UploadManager extends Component { let formData = new FormData(); formData = formDataParser(formData, file); - let request = request[method.toLowerCase()](url) + const request = superagent[method.toLowerCase()](url) .accept(accept) .set(uploadHeader); - if (file.type) { - request.type(file.type); - } - if (timeout) { request.timeout(timeout); } @@ -149,6 +145,7 @@ UploadManager.defaultProps = { formData.append('file', file); return formData; }, + reqConfigs: {}, uploadErrorHandler: (err, res) => { let error = null; const body = clone(res.body); diff --git a/src/__tests__/Receiver-test.js b/src/__tests__/Receiver-test.js index 1b9ffc0..6e430e4 100644 --- a/src/__tests__/Receiver-test.js +++ b/src/__tests__/Receiver-test.js @@ -1,4 +1,4 @@ -/* eslint-disable no-undef, max-len */ +/* eslint-disable no-undef, max-len, react/no-deprecated, react/no-string-refs, react/no-find-dom-node */ jest.dontMock('../Receiver'); jest.dontMock('../index'); jest.dontMock('classnames'); diff --git a/src/__tests__/UploadManager-test.js b/src/__tests__/UploadManager-test.js index f610cf7..bb80b66 100644 --- a/src/__tests__/UploadManager-test.js +++ b/src/__tests__/UploadManager-test.js @@ -1,4 +1,4 @@ -/* eslint-disable no-undef, max-len */ +/* eslint-disable no-undef, max-len, react/no-deprecated, react/no-string-refs, react/no-find-dom-node */ jest.dontMock('../UploadManager'); jest.dontMock('../index'); jest.dontMock('classnames'); From b748e192a383e4b3e044768d3560742e680c0313 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Fri, 23 Mar 2018 15:29:09 +1100 Subject: [PATCH 05/11] Fixed several bugs and re-enabled unit tests - fixed invalid usage of `this.props.wrapperId` in `constructor` function - added wrapper HTMLElement checking in `Receiver` - fixed file object mutations in `UploadManager` - updated argument signature for `onUploadEnd` in `UploadManager` - fixed unit tests with `enzyme` --- package.json | 6 +- src/Receiver.js | 22 +- src/UploadHandler.js | 2 +- src/UploadManager.js | 27 +- src/__tests__/Receiver-test.js | 458 +++++++++++++++------------- src/__tests__/UploadManager-test.js | 200 ++++++------ src/__tests__/fake-test.js | 6 - 7 files changed, 377 insertions(+), 344 deletions(-) delete mode 100644 src/__tests__/fake-test.js diff --git a/package.json b/package.json index b4d8b34..594bf7e 100644 --- a/package.json +++ b/package.json @@ -5,8 +5,8 @@ "main": "lib/index.js", "scripts": { "clean": "rm -rf lib", - "test": "jest fake-test.js", - "test:report": "jest fake-test.js --coverage", + "test": "jest", + "test:report": "jest --coverage", "build:lib": "babel src --out-dir lib", "build": "npm run eslint && npm run test && npm run clean && npm run build:lib", "eslint": "eslint ./src/*.js ./src/**/*.js", @@ -36,6 +36,8 @@ "babel-jest": "*", "babel-preset-es2015": "^6.1.18", "babel-preset-react": "^6.1.18", + "enzyme": "^3.3.0", + "enzyme-adapter-react-16": "^1.1.1", "eslint": "^4.19.1", "eslint-plugin-react": "^7.7.0", "jest-cli": "*", diff --git a/src/Receiver.js b/src/Receiver.js index 01e9bf6..fcf10e8 100644 --- a/src/Receiver.js +++ b/src/Receiver.js @@ -9,7 +9,16 @@ class Receiver extends Component { constructor(props) { super(props); - this.wrapper = document.getElementById(this.props.wrapperId) || window; + this.wrapper = window; + + if (props.wrapperId) { + this.wrapper = document.getElementById(props.wrapperId); + } + + if (!this.wrapper) { + throw new Error(`wrapper element with Id ${props.wrapperId} not found.`); + } + this.onDragEnter = this.onDragEnter.bind(this); this.onDragOver = this.onDragOver.bind(this); this.onDragLeave = this.onDragLeave.bind(this); @@ -34,6 +43,13 @@ class Receiver extends Component { this.wrapper.addEventListener('drop', this.onFileDrop); } + componentWillReceiveProps(nextProps) { + if (nextProps.wrapperId !== this.props.wrapperId) { + // eslint-disable-next-line no-console + console.warn('[Receiver.js] Change in props.wrapperId is unexpected, no new event listeners will be created.'); + } + } + componentWillUnmount() { this.wrapper.removeEventListener('dragenter', this.onDragEnter); this.wrapper.removeEventListener('dragleave', this.onDragLeave); @@ -126,4 +142,8 @@ Receiver.propTypes = { wrapperId: PropTypes.string, }; +Receiver.defaultProps = { + isOpen: false +}; + export default Receiver; diff --git a/src/UploadHandler.js b/src/UploadHandler.js index 14a0e69..8183b37 100644 --- a/src/UploadHandler.js +++ b/src/UploadHandler.js @@ -3,7 +3,7 @@ import PropTypes from 'prop-types'; import classNames from 'classnames'; import uploadStatus from './constants/status'; -const debug = require('debug')('react-file-upload:UploadHandler'); +const debug = require('debug')('react-file-uploader:UploadHandler'); class UploadHandler extends Component { componentDidMount() { diff --git a/src/UploadManager.js b/src/UploadManager.js index 38510a3..b77c658 100644 --- a/src/UploadManager.js +++ b/src/UploadManager.js @@ -2,13 +2,12 @@ import React, { Component, cloneElement } from 'react'; import PropTypes from 'prop-types'; import invariant from 'invariant'; import classNames from 'classnames'; -import assign from 'lodash/assign'; import bindKey from 'lodash/bindKey'; import clone from 'lodash/clone'; import superagent from 'superagent'; import uploadStatus from './constants/status'; -const debug = require('debug')('react-file-upload:UploadManager'); +const debug = require('debug')('react-file-uploader:UploadManager'); class UploadManager extends Component { constructor(props) { @@ -46,7 +45,7 @@ class UploadManager extends Component { } = this.props; if (typeof onUploadStart === 'function') { - onUploadStart(assign(file, { status: uploadStatus.UPLOADING })); + onUploadStart(Object.assign({}, file, { status: uploadStatus.UPLOADING })); } let formData = new FormData(); @@ -70,7 +69,7 @@ class UploadManager extends Component { .send(formData) .on('progress', ({ percent }) => { if (typeof onUploadProgress === 'function') { - onUploadProgress(assign(file, { + onUploadProgress(Object.assign({}, file, { progress: percent, status: uploadStatus.UPLOADING, })); @@ -81,18 +80,16 @@ class UploadManager extends Component { if (error) { debug('failed to upload file', error); - - if (typeof onUploadEnd === 'function') { - onUploadEnd(assign(file, { error, status: uploadStatus.FAILED })); - } - - return; + } else { + debug('succeeded to upload file', result); } - debug('succeeded to upload file', res); - if (typeof onUploadEnd === 'function') { - onUploadEnd(assign(file, { result, status: uploadStatus.UPLOADED })); + onUploadEnd(Object.assign({}, file, { + error, + result, + status: error && uploadStatus.FAILED || uploadStatus.UPLOADED + })); } }); } @@ -103,7 +100,7 @@ class UploadManager extends Component { return React.createElement( component, { className: classNames(customClass), style }, - React.Children.map(children, child => cloneElement(child, assign({ + React.Children.map(children, child => cloneElement(child, Object.assign({ upload: bindKey(this, 'upload', uploadUrl, child.props.file), }, child.props))) ); @@ -146,7 +143,7 @@ UploadManager.defaultProps = { return formData; }, reqConfigs: {}, - uploadErrorHandler: (err, res) => { + uploadErrorHandler: (err, res = {}) => { let error = null; const body = clone(res.body); diff --git a/src/__tests__/Receiver-test.js b/src/__tests__/Receiver-test.js index 6e430e4..84a36dc 100644 --- a/src/__tests__/Receiver-test.js +++ b/src/__tests__/Receiver-test.js @@ -1,215 +1,257 @@ -/* eslint-disable no-undef, max-len, react/no-deprecated, react/no-string-refs, react/no-find-dom-node */ +/* eslint-disable no-undef, max-len */ jest.dontMock('../Receiver'); jest.dontMock('../index'); jest.dontMock('classnames'); import React from 'react'; -import ReactDOM from 'react-dom'; +import { shallow, configure } from 'enzyme'; +import Adapter from 'enzyme-adapter-react-16'; import { jsdom } from 'jsdom'; const FileUploader = require('../index'); const { PENDING } = FileUploader.status; const Receiver = FileUploader.Receiver; -const createComponent = (onDragEnter, onDragOver, onDragLeave, onFileDrop) => ( - -); - -// eslint-disable-next-line react/prefer-es6-class -const createTemplate = (initialState, props, Component) => React.createClass({ - getInitialState() { return initialState; }, - render() { - return ( - -

Test

-
- ); - }, -}); +configure({ adapter: new Adapter() }); + +const testFile = { + lastModified: 1465229147840, + lastModifiedDate: 'Tue Jun 07 2016 00:05:47 GMT+0800 (HKT)', + name: 'test.jpg', + size: 1024, + type: 'image/jpg', + webkitRelativePath: '', +}; + +const files = [testFile]; + +const createEvent = (eventName) => { + const event = document.createEvent('HTMLEvents'); + event.initEvent(eventName, false, true); + event.preventDefault = jest.genMockFn(); + event.dataTransfer = { + files, + setData: jest.genMockFunction(), + types: ['Files'] + }; + + return event; +}; describe('Receiver', () => { - beforeEach(function setting() { + let dragEnterEvent, + dragOverEvent, + dragLeaveEvent, + dropEvent, + stringClass = 'receiver', + arrayClass = ['react', 'receiver']; + + beforeEach(() => { global.document = jsdom(); global.window = document.parentWindow; + global.window.DragEvent = 'DragEvent'; + global.window.DataTransfer = 'DataTransfer'; + + dragEnterEvent = createEvent('dragenter'); + dragOverEvent = createEvent('dragover'); + dragLeaveEvent = createEvent('dragleave'); + dropEvent = createEvent('drop'); + }); + + describe('constructor', () => { + let emptyFn = () => {}, + component = ( + + ); + + it('should throw an error if DnD or File API is not supported', () => { + global.window.DragEvent = undefined; + global.window.DataTransfer = undefined; + + expect(() => shallow(component)).toThrow('Browser does not support DnD events or File API.'); + }); - const testFile = { - lastModified: 1465229147840, - lastModifiedDate: 'Tue Jun 07 2016 00:05:47 GMT+0800 (HKT)', - name: 'test.jpg', - size: 1024, - type: 'image/jpg', - webkitRelativePath: '', - }; - const files = [testFile]; - - const mockDT = { - files, - setData: jest.genMockFunction(), - }; - - this.stringClass = 'receiver'; - this.arrayClass = ['react', 'receiver']; - - this.dragEnterEvent = document.createEvent('HTMLEvents'); - this.dragEnterEvent.initEvent('dragenter', false, true); - - this.dragOverEvent = document.createEvent('HTMLEvents'); - this.dragOverEvent.initEvent('dragover', false, true); - this.dragOverEvent.preventDefault = jest.genMockFn(); - - this.dragLeaveEvent = document.createEvent('HTMLEvents'); - this.dragLeaveEvent.initEvent('dragleave', false, true); - - this.dropEvent = document.createEvent('HTMLEvents'); - this.dropEvent.initEvent('drop', false, true); - this.dropEvent.preventDefault = jest.genMockFn(); - this.dropEvent.dataTransfer = mockDT; + it('should assign window to this.wrapper if no wrapperId is provided', () => { + const receiver = shallow(component); + expect(receiver.instance().wrapper).toEqual(global.window); + }); + + it('should throw an error if wrapperId is given but element is not found', () => { + expect(() => shallow( + + )).toThrow(); + }); }); describe('state of dragLevel', () => { - beforeEach(function setting() { - const onDragEnter = jest.genMockFn(); - const onDragOver = jest.genMockFn(); - const onDragLeave = jest.genMockFn(); - const onFileDrop = jest.genMockFn(); - - this.onDragEnter = onDragEnter; - this.onDragOver = onDragOver; - this.onDragLeave = onDragLeave; - this.onFileDrop = onFileDrop; - - const Component = createComponent(onDragEnter, onDragOver, onDragLeave, onFileDrop); - const template = createTemplate({ isOpen: false, files: [] }, {}, Component); - - this.createTestParent = React.createFactory(template); - this.ParentComponent = this.createTestParent(); - this.container = document.createElement('div'); - this.instance = ReactDOM.render(this.ParentComponent, this.container); - this.receiver = this.instance.refs.receiver; + let receiver, + onDragEnter, + onDragOver, + onDragLeave, + onFileDrop; + + beforeEach(() => { + const mockOnDragEnter = jest.genMockFn(); + const mockOnDragOver = jest.genMockFn(); + const mockOnDragLeave = jest.genMockFn(); + const mockOnFileDrop = jest.genMockFn(); + + onDragEnter = mockOnDragEnter; + onDragOver = mockOnDragOver; + onDragLeave = mockOnDragLeave; + onFileDrop = mockOnFileDrop; + + const component = ( + + ); + + receiver = shallow(component); }); - it('should increase state of dragLevel by 1 with dragEnter event', function test() { - const oldDragLevel = this.receiver.state.dragLevel; - window.dispatchEvent(this.dragEnterEvent); - const newDragLevel = this.receiver.state.dragLevel; + it('should increase state of dragLevel by 1 with dragEnter event', () => { + const oldDragLevel = receiver.state().dragLevel; + window.dispatchEvent(dragEnterEvent); + const newDragLevel = receiver.state().dragLevel; expect(newDragLevel).toEqual(oldDragLevel + 1); }); - it('should call onDragEnter with dragEnter event if isOpen is false', function test() { - window.dispatchEvent(this.dragEnterEvent); - expect(this.onDragEnter).toBeCalled(); + it('should call onDragEnter with dragEnter event if isOpen is false', () => { + window.dispatchEvent(dragEnterEvent); + expect(onDragEnter).toBeCalled(); }); - it('should not call onDragEnter with dragEnter event if isOpen is true', function test() { - this.instance.setState({ isOpen: true }); - window.dispatchEvent(this.dragEnterEvent); - expect(this.onDragEnter).not.toBeCalled(); + it('should not call onDragEnter with dragEnter event if isOpen is true', () => { + receiver.setProps({ isOpen: true }); + window.dispatchEvent(dragEnterEvent); + expect(onDragEnter).not.toBeCalled(); }); - it('should call event.preventDefault with dragOver event', function test() { - window.dispatchEvent(this.dragOverEvent); - expect(this.dragOverEvent.preventDefault).toBeCalled(); + it('should not call onDragEnter callback with dragEnter event if dataTransfer.types does not include `Files`', () => { + onDragEnter = jest.genMockFn(); + dragEnterEvent.dataTransfer.types = ['HTMLElement']; + + receiver.setProps({ onDragEnter }); + + window.dispatchEvent(dragEnterEvent); + expect(onDragEnter).not.toBeCalled(); + }); + + it('should call event.preventDefault with dragOver event', () => { + window.dispatchEvent(dragOverEvent); + expect(dragOverEvent.preventDefault).toBeCalled(); }); - it('should call onDragOver with dragOver event', function test() { - window.dispatchEvent(this.dragOverEvent); - expect(this.onDragOver).toBeCalled(); + it('should call onDragOver with dragOver event', () => { + window.dispatchEvent(dragOverEvent); + expect(onDragOver).toBeCalled(); }); - it('should decrease state of dragLevel by 1 with dragLeave event', function test() { - const oldDragLevel = this.receiver.state.dragLevel; - window.dispatchEvent(this.dragEnterEvent); - const newDragLevel = this.receiver.state.dragLevel; + it('should decrease state of dragLevel by 1 with dragLeave event', () => { + const oldDragLevel = receiver.state().dragLevel; + window.dispatchEvent(dragEnterEvent); + const newDragLevel = receiver.state().dragLevel; expect(newDragLevel).toEqual(oldDragLevel + 1); - window.dispatchEvent(this.dragLeaveEvent); - const finalDragLevel = this.receiver.state.dragLevel; + window.dispatchEvent(dragLeaveEvent); + const finalDragLevel = receiver.state().dragLevel; expect(finalDragLevel).toEqual(newDragLevel - 1); - expect(this.onDragLeave).toBeCalled(); + expect(onDragLeave).toBeCalled(); }); - it('should call onDragLeave if state of dragLevel is not 0', function test() { - const oldDragLevel = this.receiver.state.dragLevel; - window.dispatchEvent(this.dragEnterEvent); - const newDragLevel = this.receiver.state.dragLevel; + it('should call onDragLeave if state of dragLevel is not 0', () => { + const oldDragLevel = receiver.state().dragLevel; + window.dispatchEvent(dragEnterEvent); + const newDragLevel = receiver.state().dragLevel; expect(newDragLevel).toEqual(oldDragLevel + 1); - window.dispatchEvent(this.dragEnterEvent); - const newerDragLevel = this.receiver.state.dragLevel; + window.dispatchEvent(dragEnterEvent); + const newerDragLevel = receiver.state().dragLevel; expect(newerDragLevel).toEqual(newDragLevel + 1); - window.dispatchEvent(this.dragLeaveEvent); - const finalDragLevel = this.receiver.state.dragLevel; + window.dispatchEvent(dragLeaveEvent); + const finalDragLevel = receiver.state().dragLevel; expect(finalDragLevel).toEqual(newerDragLevel - 1); - expect(this.onDragLeave).not.toBeCalled(); + expect(onDragLeave).not.toBeCalled(); - window.dispatchEvent(this.dragLeaveEvent); - const endDragLevel = this.receiver.state.dragLevel; + window.dispatchEvent(dragLeaveEvent); + const endDragLevel = receiver.state().dragLevel; expect(endDragLevel).toEqual(finalDragLevel - 1); - expect(this.onDragLeave).toBeCalled(); + expect(onDragLeave).toBeCalled(); }); - it('should call event.preventDefault with drop event', function test() { - window.dispatchEvent(this.dropEvent); + it('should call event.preventDefault with drop event', () => { + window.dispatchEvent(dropEvent); // eslint-disable-next-line no-undef - expect(this.dropEvent.preventDefault).toBeCalled(); + expect(dropEvent.preventDefault).toBeCalled(); }); - it('should call onFileDrop with drop event', function test() { - window.dispatchEvent(this.dropEvent); - expect(this.onFileDrop).toBeCalled(); + it('should call onFileDrop with drop event', () => { + window.dispatchEvent(dropEvent); + expect(onFileDrop).toBeCalled(); }); - it('should set state of dragLevel to 0 with dragEnter event', function test() { - const oldDragLevel = this.receiver.state.dragLevel; - window.dispatchEvent(this.dragEnterEvent); - const newDragLevel = this.receiver.state.dragLevel; + it('should set state of dragLevel to 0 with dragEnter event', () => { + const oldDragLevel = receiver.state().dragLevel; + window.dispatchEvent(dragEnterEvent); + const newDragLevel = receiver.state().dragLevel; expect(newDragLevel).toEqual(oldDragLevel + 1); - window.dispatchEvent(this.dropEvent); - const finalDragLevel = this.receiver.state.dragLevel; + window.dispatchEvent(dropEvent); + const finalDragLevel = receiver.state().dragLevel; expect(finalDragLevel).toEqual(0); }); - it('should not call any callback after Receiver did unmount', function test() { - ReactDOM.unmountComponentAtNode(this.container); - window.dispatchEvent(this.dragEnterEvent); - expect(this.onDragEnter).not.toBeCalled(); + it('should not call any callback after Receiver did unmount', () => { + receiver.unmount(); + window.dispatchEvent(dragEnterEvent); + expect(onDragEnter).not.toBeCalled(); - window.dispatchEvent(this.dragOverEvent); - expect(this.onDragOver).not.toBeCalled(); + window.dispatchEvent(dragOverEvent); + expect(onDragOver).not.toBeCalled(); - window.dispatchEvent(this.dragLeaveEvent); - expect(this.onDragLeave).not.toBeCalled(); + window.dispatchEvent(dragLeaveEvent); + expect(onDragLeave).not.toBeCalled(); - window.dispatchEvent(this.dropEvent); - expect(this.onFileDrop).not.toBeCalled(); + window.dispatchEvent(dropEvent); + expect(onFileDrop).not.toBeCalled(); }); }); describe('callbacks and callback arguments', () => { - beforeEach(function setting() { - const onDragEnter = (e) => { + let onDragEnter, + onDragOver, + onDragLeave, + onFileDrop; + + beforeEach(() => { + const mockOnDragEnter = (e) => { expect(e.type).toBe('dragenter'); }; - const onDragOver = (e) => { + const mockOnDragOver = (e) => { expect(e.type).toBe('dragover'); }; - const onDragLeave = (e) => { + const mockOnDragLeave = (e) => { expect(e.type).toBe('dragleave'); }; - const onFileDrop = (e, _files) => { + const mockOnFileDrop = (e, _files) => { expect(e.type).toBe('drop'); const file = _files[0]; expect(file.lastModified).toBe(testFile.lastModified); @@ -223,90 +265,90 @@ describe('Receiver', () => { expect(file.src).toBe(null); }; - this.onDragEnter = onDragEnter; - this.onDragOver = onDragOver; - this.onDragLeave = onDragLeave; - this.onFileDrop = onFileDrop; - - const Component = createComponent(onDragEnter, onDragOver, onDragLeave, onFileDrop); - const template = createTemplate({ isOpen: false, files: [] }, {}, Component); - - this.createTestParent = React.createFactory(template); - this.ParentComponent = this.createTestParent(); - this.container = document.createElement('div'); - this.instance = ReactDOM.render(this.ParentComponent, this.container); - this.receiver = this.instance.refs.receiver; + onDragEnter = mockOnDragEnter; + onDragOver = mockOnDragOver; + onDragLeave = mockOnDragLeave; + onFileDrop = mockOnFileDrop; + + const component = ( + + ); + + shallow(component); }); - it('should execute the onDragEnter callback with a DragEvent with type `dragenter` as argument', function test() { - window.dispatchEvent(this.dragEnterEvent); + it('should execute the onDragEnter callback with a DragEvent with type `dragenter` as argument', () => { + window.dispatchEvent(dragEnterEvent); }); - it('should execute the onDragOver callback with a DragEvent with type `dragover` as argument', function test() { - window.dispatchEvent(this.dragOverEvent); + it('should execute the onDragOver callback with a DragEvent with type `dragover` as argument', () => { + window.dispatchEvent(dragOverEvent); }); - it('should execute the onDragLeave callback with a DragEvent with type `dragleave` as argument', function test() { - window.dispatchEvent(this.dragLeaveEvent); + it('should execute the onDragLeave callback with a DragEvent with type `dragleave` as argument', () => { + window.dispatchEvent(dragLeaveEvent); }); - it('should execute the onFileDrop callback with a DragEvent with type `drop` as argument', function test() { - window.dispatchEvent(this.dropEvent); + it('should execute the onFileDrop callback with a DragEvent with type `drop` as argument', () => { + window.dispatchEvent(dropEvent); }); }); - describe('#render', () => { - beforeEach(function setting() { - const Component = createComponent(); - const template = createTemplate({ isOpen: false, files: [] }, {}, Component); - - this.createTestParent = React.createFactory(template); - this.ParentComponent = this.createTestParent(); - this.container = document.createElement('div'); - this.instance = ReactDOM.render(this.ParentComponent, this.container); - this.receiver = this.instance.refs.receiver; + describe('render()', () => { + let receiver, + childrenItems = Array(5).fill().map((value, index) => (
  • {index}
  • )); + + beforeEach(() => { + const mockOnDragEnter = jest.genMockFn(); + const mockOnDragOver = jest.genMockFn(); + const mockOnDragLeave = jest.genMockFn(); + const mockOnFileDrop = jest.genMockFn(); + + const component = ( + + {childrenItems} + + ); + + receiver = shallow(component); }); - it('should render nothing if isOpen is false', function test() { - const receiver = ReactDOM.findDOMNode(this.receiver); - expect(receiver).toBeNull(); - this.instance.setState({ isOpen: true }); + it('should render nothing if isOpen is false', () => { + expect(receiver.type()).toEqual(null); + expect(receiver.children().exists()).toBe(false); }); - it('should render a div wrapper with children if isOpen is true', function test() { - this.instance.setState({ isOpen: true }); - const receiver = ReactDOM.findDOMNode(this.receiver); - expect(receiver).toEqual(jasmine.any(HTMLDivElement)); - expect(receiver.firstElementChild).toEqual(jasmine.any(HTMLHeadingElement)); + it('should render a div wrapper with children if isOpen is true', () => { + receiver.setProps({ isOpen: true }); + expect(receiver.type()).toEqual('div'); + expect(receiver.children().length).toEqual(childrenItems.length); }); - it('should render a div wrapper with customClass in string', function test() { - const Component = createComponent(); - const template = createTemplate({ isOpen: true, files: [] }, { customClass: this.stringClass }, Component); - - this.createTestParent = React.createFactory(template); - this.ParentComponent = this.createTestParent(); - this.container = document.createElement('div'); - this.instance = ReactDOM.render(this.ParentComponent, this.container); - this.receiver = this.instance.refs.receiver; - - const receiver = ReactDOM.findDOMNode(this.receiver); - expect(receiver.className).toEqual(this.stringClass); + it('should render a div wrapper with customClass in string', () => { + receiver.setProps({ isOpen: true, customClass: stringClass }); + expect(receiver.hasClass(stringClass)).toBe(true); }); - it('should render a div wrapper with customClass in array', function test() { - const Component = createComponent(); - const template = createTemplate({ isOpen: true, files: [] }, { customClass: this.arrayClass }, Component); - - this.createTestParent = React.createFactory(template); - this.ParentComponent = this.createTestParent(); - this.container = document.createElement('div'); - this.instance = ReactDOM.render(this.ParentComponent, this.container); - this.receiver = this.instance.refs.receiver; - - const receiver = ReactDOM.findDOMNode(this.receiver); - expect(receiver.className).toEqual(this.arrayClass.join(' ')); + it('should render a div wrapper with customClass in array', () => { + receiver.setProps({ isOpen: true, customClass: arrayClass }); + arrayClass.forEach((classname) => { + expect(receiver.hasClass(classname)).toBe(true); + }); }); }); }); -/* eslint-enable no-undef */ +/* eslint-enable no-undef, max-len */ diff --git a/src/__tests__/UploadManager-test.js b/src/__tests__/UploadManager-test.js index bb80b66..17117df 100644 --- a/src/__tests__/UploadManager-test.js +++ b/src/__tests__/UploadManager-test.js @@ -1,11 +1,12 @@ -/* eslint-disable no-undef, max-len, react/no-deprecated, react/no-string-refs, react/no-find-dom-node */ +/* eslint-disable no-undef, max-len */ jest.dontMock('../UploadManager'); jest.dontMock('../index'); jest.dontMock('classnames'); jest.dontMock('lodash'); import React from 'react'; -import ReactDOM from 'react-dom'; +import { shallow, configure } from 'enzyme'; +import Adapter from 'enzyme-adapter-react-16'; import { jsdom } from 'jsdom'; import nock from 'nock'; @@ -13,142 +14,110 @@ const FileUploader = require('../index'); const UploadManager = FileUploader.UploadManager; const uploadStatus = FileUploader.status; +configure({ adapter: new Adapter() }); + describe('UploadManager', () => { - beforeEach(function setting() { + let stringClass = 'receiver', + arrayClass = ['react', 'receiver'], + uploadPath = 'http://localhost:3000/api/upload', + children = (

    children

    ), + uploadManager, + onUploadStart, + onUploadProgress, + onUploadEnd, + formDataParser, + err, + errorResponse, + successResponse, + errorHandler; + + beforeEach(() => { global.document = jsdom(); global.window = document.parentWindow; - this.stringClass = 'receiver'; - this.arrayClass = ['react', 'receiver']; - this.style = { display: 'block' }; - this.uploadPath = 'http://localhost:3000/api/upload'; - this.onUploadStart = jest.fn(); - this.onUploadProgress = jest.fn(); - this.onUploadEnd = jest.fn(); - - this.children =

    children

    ; - this.container = document.createElement('div'); - this.instance = ReactDOM.render( + onUploadStart = jest.genMockFn(); + onUploadProgress = jest.genMockFn(); + onUploadEnd = jest.genMockFn(); + formDataParser = jest.genMockFn(); + + err = new Error('not found'); + errorResponse = { body: { success: false, errors: { message: 'not found' } } }; + successResponse = { body: { success: true } }; + errorHandler = UploadManager.defaultProps.uploadErrorHandler; + + uploadManager = shallow( - {this.children} - , - this.container - ); + {children} + + ) }); - afterEach(function setting() { - this.container = null; - this.instance = null; + afterEach(() => { + uploadManager = null; }); - describe('#render()', () => { - it('should render ul element by default', function test() { - const node = ReactDOM.findDOMNode(this.instance); - expect(node).toEqual(jasmine.any(HTMLUListElement)); - expect(node.firstElementChild).toEqual(jasmine.any(HTMLParagraphElement)); + describe('render()', () => { + it('should render ul element by default', () => { + expect(uploadManager.type()).toEqual('ul'); + expect(uploadManager.childAt(0).type()).toEqual('p'); }); - it('should render wrapper element according to component props', function test() { - this.instance = ReactDOM.render( - - {this.children} - , - this.container - ); - const node = ReactDOM.findDOMNode(this.instance); - expect(node).toEqual(jasmine.any(HTMLDivElement)); + it('should render wrapper element according to component props', () => { + uploadManager.setProps({ component: 'div' }); + expect(uploadManager.type()).toEqual('div'); }); - it('should render a wrapper with customClass in string', function test() { - this.instance = ReactDOM.render( - - {this.children} - , - this.container - ); - const node = ReactDOM.findDOMNode(this.instance); - expect(node.className).toEqual(this.stringClass); + it('should render a wrapper with customClass in string', () => { + expect(uploadManager.hasClass(stringClass)).toBe(true); }); - it('should render a wrapper with customClass in array', function test() { - this.instance = ReactDOM.render( - - {this.children} - , - this.container - ); - const node = ReactDOM.findDOMNode(this.instance); - expect(node.className).toEqual(this.arrayClass.join(' ')); - }); - }); + it('should render a wrapper with customClass in array', () => { + uploadManager.setProps({ customClass: arrayClass }); - describe('#uploadErrorHandler()', () => { - beforeEach(function setting() { - this.err = new Error('not found'); - this.errorResponse = { body: { success: false, errors: { message: 'not found' } } }; - this.successResponse = { body: { success: true } }; - this.errorHandler = this.instance.props.uploadErrorHandler; + arrayClass.forEach((classname) => { + expect(uploadManager.hasClass(classname)).toBe(true); + }); }); + }); - it('should return an object contains key of `error` and `result`', function test() { - const result = this.errorHandler(null, this.successResponse); + describe('uploadErrorHandler()', () => { + it('should return an object contains key of `error` and `result`', () => { + const result = errorHandler(null, successResponse); expect(result.error).toBeNull(); - expect(result.result).toEqual(this.successResponse.body); + expect(result.result).toEqual(successResponse.body); }); - it('should return an object with key of `error` with value equals to the first argument if it is not empty', function test() { - const result = this.errorHandler(this.err, this.successResponse); - expect(result.error).toEqual(this.err.message); - expect(result.result).toEqual(this.successResponse.body); + it('should return an object with key of `error` with value equals to the first argument if it is not empty', () => { + const result = errorHandler(err, successResponse); + expect(result.error).toEqual(err.message); + expect(result.result).toEqual(successResponse.body); }); - it('should return an object with key of `error` with value equals to the value of `body.error` of the second argument if it is not empty', function test() { - const result = this.errorHandler(null, this.errorResponse); - expect(result.error).toEqual(this.errorResponse.body.errors); - delete this.errorResponse.body.errors; - expect(result.result).toEqual(this.errorResponse.body); + it('should return an object with key of `error` with value equals to the value of `body.error` of the second argument if it is not empty', () => { + const result = errorHandler(null, errorResponse); + expect(result.error).toEqual(errorResponse.body.errors); + delete errorResponse.body.errors; + expect(result.result).toEqual(errorResponse.body); }); }); - describe('#upload()', () => { - beforeEach(function setting() { + describe('upload()', () => { + let file; + + beforeEach(() => { + file = {}; + nock('http://localhost:3000') .filteringRequestBody(() => '*') .post('/api/upload', '*') - .reply(200, this.successResponse); - - this.instance = ReactDOM.render( - - {this.children} - , - this.container - ); - this.errorResponse = { success: false, errors: { message: 'not found' } }; - this.successResponse = { success: true }; + .reply(200, successResponse); }); afterEach(() => { @@ -156,9 +125,18 @@ describe('UploadManager', () => { nock.enableNetConnect(); }); - it('should call onUploadStart prop functions if it is given', function test() { - this.instance.upload(this.instance.props.uploadUrl, {}); - expect(this.onUploadStart).toBeCalledWith({ status: uploadStatus.UPLOADING }); + it('should call `props.onUploadStart` function if it is given', () => { + const instance = uploadManager.instance(); + instance.upload(instance.props.uploadUrl, file); + expect(onUploadStart).toBeCalledWith(Object.assign({}, file, { status: uploadStatus.UPLOADING })); + expect(file).toEqual({}); + }); + + it('should call `props.formDataParser` function if it is given', () => { + const instance = uploadManager.instance(); + instance.upload(instance.props.uploadUrl, {}); + expect(formDataParser).toBeCalledWith(new FormData(), {}); }); }); }); +/* eslint-enable no-undef, max-len */ diff --git a/src/__tests__/fake-test.js b/src/__tests__/fake-test.js deleted file mode 100644 index 3c8eb54..0000000 --- a/src/__tests__/fake-test.js +++ /dev/null @@ -1,6 +0,0 @@ -/* eslint-disable no-undef */ -describe('Fake Test', () => { - it('should pass this fake test', () => { - expect(1).toEqual(1); - }); -}); From 3f2d3b85fe67e98a889c6ee35b987c9d7e6c8490 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Fri, 23 Mar 2018 15:42:08 +1100 Subject: [PATCH 06/11] Downgraded react peer dependencies version --- package.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 594bf7e..26fe2ad 100644 --- a/package.json +++ b/package.json @@ -43,11 +43,10 @@ "jest-cli": "*", "jsdom": "^7.0.2", "nock": "^8.0.0", - "react": "^16.0.0", - "react-dom": "^16.0.0" + "react-dom": "^15.0.0 || ^16.0.0" }, "peerDependencies": { - "react": "^16.0.0" + "react": "^15.0.0 || ^16.0.0" }, "jest": { "scriptPreprocessor": "/node_modules/babel-jest", @@ -71,6 +70,7 @@ "invariant": "^2.2.0", "lodash": ">=3.10.1", "prop-types": "^15.5.10", + "react": "^15.0.0 || ^16.0.0", "shortid": "^2.2.6", "superagent": "^1.4.0" } From da9706ed8e3f1bd677f6f0b1759c08ef37315504 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Fri, 23 Mar 2018 15:46:15 +1100 Subject: [PATCH 07/11] bumped up version to 0.4.0 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 26fe2ad..ca26bc0 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "react-file-uploader", - "version": "0.3.7", + "version": "0.4.0", "description": "A set of file-upload-components with React.js.", "main": "lib/index.js", "scripts": { From 32b3acef5a424bcf5146bfac6f8a23547cadd130 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Sat, 24 Mar 2018 13:59:21 +1100 Subject: [PATCH 08/11] added property in onUploadEnd cb --- src/UploadManager.js | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/UploadManager.js b/src/UploadManager.js index b77c658..90dd150 100644 --- a/src/UploadManager.js +++ b/src/UploadManager.js @@ -45,7 +45,7 @@ class UploadManager extends Component { } = this.props; if (typeof onUploadStart === 'function') { - onUploadStart(Object.assign({}, file, { status: uploadStatus.UPLOADING })); + onUploadStart(Object.assign(file, { status: uploadStatus.UPLOADING })); } let formData = new FormData(); @@ -69,7 +69,7 @@ class UploadManager extends Component { .send(formData) .on('progress', ({ percent }) => { if (typeof onUploadProgress === 'function') { - onUploadProgress(Object.assign({}, file, { + onUploadProgress(Object.assign(file, { progress: percent, status: uploadStatus.UPLOADING, })); @@ -85,7 +85,8 @@ class UploadManager extends Component { } if (typeof onUploadEnd === 'function') { - onUploadEnd(Object.assign({}, file, { + onUploadEnd(Object.assign(file, { + progress: error && 0 || 100, error, result, status: error && uploadStatus.FAILED || uploadStatus.UPLOADED From a162b0bebd1e7f463b48876f7c3db9ff35d63f93 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Mon, 26 Mar 2018 10:50:57 +1100 Subject: [PATCH 09/11] improved performance issue in basic example --- examples/basic/client.js | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/examples/basic/client.js b/examples/basic/client.js index 06971ca..8a2129e 100644 --- a/examples/basic/client.js +++ b/examples/basic/client.js @@ -37,9 +37,11 @@ class MyComponent extends Component { // a specific DOM node const node = ReactDOM.findDOMNode(this.uploadPanel); - this.setState({ - isDragOver: e.target === node, - }); + if (this.state.isDragOver !== (e.target === node)) { + this.setState({ + isDragOver: e.target === node, + }); + } } onFileDrop({ target }, files) { From 713fa9a70e57b371ac17d76f3aa07904dbb94fc8 Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Mon, 26 Mar 2018 10:55:30 +1100 Subject: [PATCH 10/11] fixed unit test --- src/__tests__/UploadManager-test.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/__tests__/UploadManager-test.js b/src/__tests__/UploadManager-test.js index 17117df..4097477 100644 --- a/src/__tests__/UploadManager-test.js +++ b/src/__tests__/UploadManager-test.js @@ -129,13 +129,13 @@ describe('UploadManager', () => { const instance = uploadManager.instance(); instance.upload(instance.props.uploadUrl, file); expect(onUploadStart).toBeCalledWith(Object.assign({}, file, { status: uploadStatus.UPLOADING })); - expect(file).toEqual({}); + expect(file).toEqual({ status: uploadStatus.UPLOADING }); }); it('should call `props.formDataParser` function if it is given', () => { const instance = uploadManager.instance(); instance.upload(instance.props.uploadUrl, {}); - expect(formDataParser).toBeCalledWith(new FormData(), {}); + expect(formDataParser).toBeCalledWith(new FormData(), { status: uploadStatus.UPLOADING }); }); }); }); From e9cb8e3d542923bd7fa20c55b95a664169e53fca Mon Sep 17 00:00:00 2001 From: Marston Ng Date: Mon, 26 Mar 2018 11:16:41 +1100 Subject: [PATCH 11/11] fixed onUploadEnd argument property --- src/UploadManager.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/UploadManager.js b/src/UploadManager.js index 90dd150..7778c12 100644 --- a/src/UploadManager.js +++ b/src/UploadManager.js @@ -88,7 +88,7 @@ class UploadManager extends Component { onUploadEnd(Object.assign(file, { progress: error && 0 || 100, error, - result, + result: error && undefined || result, status: error && uploadStatus.FAILED || uploadStatus.UPLOADED })); }