apeman-react-upload
Version:
apeman react package for file upload components.
282 lines (240 loc) • 24.2 kB
JavaScript
/**
* apeman react package for file upload components.
* @class ApUpload
*/
'use strict';
Object.defineProperty(exports, "__esModule", {
value: true
});
var _assign = require('babel-runtime/core-js/object/assign');
var _assign2 = _interopRequireDefault(_assign);
var _react = require('react');
var _react2 = _interopRequireDefault(_react);
var _classnames = require('classnames');
var _classnames2 = _interopRequireDefault(_classnames);
var _async = require('async');
var _async2 = _interopRequireDefault(_async);
var _path = require('path');
var _path2 = _interopRequireDefault(_path);
var _uuid = require('uuid');
var _uuid2 = _interopRequireDefault(_uuid);
var _apemanReactImage = require('apeman-react-image');
var _apemanReactSpinner = require('apeman-react-spinner');
var _apemanReactButton = require('apeman-react-button');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
/** @lends ApUpload */
var ApUpload = _react2.default.createClass({
displayName: 'ApUpload',
// --------------------
// Specs
// --------------------
propTypes: {
/** Name of input */
name: _react.PropTypes.string,
/** DOM id of input */
id: _react.PropTypes.string,
/** Allow multiple upload */
multiple: _react.PropTypes.bool,
/** Handler for change event */
onChange: _react.PropTypes.func,
/** Handler for load event */
onLoad: _react.PropTypes.func,
/** Handler for error event */
onError: _react.PropTypes.func,
/** Image width */
width: _react.PropTypes.number,
/** Image height */
height: _react.PropTypes.number,
/** Guide text */
text: _react.PropTypes.string,
/** Accept file type */
accept: _react.PropTypes.string,
/** Guide icon */
icon: _react.PropTypes.string,
/** Icon for close images */
closeIcon: _react.PropTypes.string,
/** Spinner theme */
spinner: _react.PropTypes.string,
/** Value of input */
value: _react.PropTypes.oneOfType([_react.PropTypes.string, _react.PropTypes.array])
},
mixins: [],
statics: {
readFile: function readFile(file, callback) {
var reader = new window.FileReader();
reader.onerror = function onerror(err) {
callback(err);
};
reader.onload = function onload(ev) {
callback(null, ev.target.result);
};
reader.readAsDataURL(file);
},
isImageUrl: function isImageUrl(url) {
var imageExtensions = ['.jpg', '.jpeg', '.svg', '.gif', '.png'];
return (/^data:image/.test(url) || !!~imageExtensions.indexOf(_path2.default.extname(url))
);
}
},
getInitialState: function getInitialState() {
var s = this;
var props = s.props;
var hasValue = props.value && props.value.length > 0;
return {
spinning: false,
error: null,
urls: hasValue ? [].concat(props.value) : null
};
},
getDefaultProps: function getDefaultProps() {
return {
name: null,
id: 'ap-upload-' + _uuid2.default.v4(),
multiple: false,
width: 180,
height: 180,
accept: null,
text: 'Upload file',
icon: 'fa fa-cloud-upload',
closeIcon: 'fa fa-close',
spinnerIcon: _apemanReactSpinner.ApSpinner.DEFAULT_THEME,
onChange: null,
onLoad: null,
onError: null
};
},
render: function render() {
var s = this;
var state = s.state,
props = s.props;
var width = props.width,
height = props.height;
return _react2.default.createElement(
'div',
{ className: (0, _classnames2.default)('ap-upload', props.className),
style: (0, _assign2.default)({}, props.style) },
_react2.default.createElement('input', { type: 'file',
className: 'ap-upload-input',
multiple: props.multiple,
name: props.name,
id: props.id,
accept: props.accept,
onChange: s.handleChange,
style: { width: width, height: height }
}),
_react2.default.createElement(
'label',
{ className: 'ap-upload-label', htmlFor: props.id },
_react2.default.createElement('span', { className: 'ap-upload-aligner' }),
_react2.default.createElement(
'span',
{ className: 'ap-upload-label-inner' },
_react2.default.createElement('i', { className: (0, _classnames2.default)('ap-upload-icon', props.icon) }),
_react2.default.createElement(
'span',
{ className: 'ap-upload-text' },
props.text
),
props.children
)
),
s._renderPreviewImage(state.urls, width, height),
s._renderRemoveButton(!!(state.urls && state.urls.length > 0), props.closeIcon),
s._renderSpinner(state.spinning, props.spinner)
);
},
// --------------------
// Lifecycle
// --------------------
componentWillReceiveProps: function componentWillReceiveProps(nextProps) {
var s = this;
var props = s.props;
var value = nextProps.value;
var hasValue = value && value.length > 0;
if (hasValue && props.value !== value) {
s.setState({ urls: [].concat(value) });
}
},
// ------------------
// Custom
// ------------------
handleChange: function handleChange(e) {
var s = this;
var props = s.props;
var target = e.target;
var files = Array.prototype.slice.call(target.files, 0);
var onChange = props.onChange,
onError = props.onError,
onLoad = props.onLoad;
s.setState({ spinning: true });
if (onChange) {
onChange(e);
}
_async2.default.concat(files, ApUpload.readFile, function (error, urls) {
s.setState({
spinning: false,
error: error,
urls: urls
});
if (error) {
if (onError) {
onError(error);
}
} else {
if (onLoad) {
var loaded = (0, _assign2.default)({ urls: urls, target: target });
onLoad(loaded);
}
}
});
},
handleRemove: function handleRemove() {
var s = this;
var props = s.props;
var onLoad = props.onLoad;
s.setState({
error: null,
urls: null
});
if (onLoad) {
onLoad([]);
}
},
// ------------------
// Private
// ------------------
_renderSpinner: function _renderSpinner(spinning, theme) {
var s = this;
return _react2.default.createElement(_apemanReactSpinner.ApSpinner, { enabled: spinning, theme: theme });
},
_renderRemoveButton: function _renderRemoveButton(removable, icon) {
var s = this;
if (!removable) {
return null;
}
return _react2.default.createElement(
_apemanReactButton.ApButton,
{ onTap: s.handleRemove, className: 'ap-upload-remove-button' },
_react2.default.createElement('i', { className: (0, _classnames2.default)('ap-upload-remove-icon', icon) })
);
},
_renderPreviewImage: function _renderPreviewImage(urls, width, height) {
if (!urls) {
return null;
}
var s = this;
return urls.filter(function (url) {
return ApUpload.isImageUrl(url);
}).map(function (url, i) {
return _react2.default.createElement(_apemanReactImage.ApImage, { key: url,
src: url,
height: height,
width: width,
className: (0, _classnames2.default)('ap-upload-preview-image'),
style: { left: i * 10 + '%', top: i * 10 + '%' },
scale: 'fit' });
});
}
});
exports.default = ApUpload;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["ap_upload.jsx"],"names":["ApUpload","createClass","propTypes","name","string","id","multiple","bool","onChange","func","onLoad","onError","width","number","height","text","accept","icon","closeIcon","spinner","value","oneOfType","array","mixins","statics","readFile","file","callback","reader","window","FileReader","onerror","err","onload","ev","target","result","readAsDataURL","isImageUrl","url","imageExtensions","test","indexOf","extname","getInitialState","s","props","hasValue","length","spinning","error","urls","concat","getDefaultProps","v4","spinnerIcon","DEFAULT_THEME","render","state","className","style","handleChange","children","_renderPreviewImage","_renderRemoveButton","_renderSpinner","componentWillReceiveProps","nextProps","setState","e","files","Array","prototype","slice","call","loaded","handleRemove","theme","removable","filter","map","i","left","top"],"mappings":"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","file":"ap_upload.jsx","sourceRoot":"lib","sourcesContent":["/**\n * apeman react package for file upload components.\n * @class ApUpload\n */\n\n'use strict'\n\nimport React, { PropTypes as types } from 'react'\nimport classnames from 'classnames'\nimport async from 'async'\nimport path from 'path'\nimport uuid from 'uuid'\nimport { ApImage } from 'apeman-react-image'\nimport { ApSpinner } from 'apeman-react-spinner'\nimport { ApButton } from 'apeman-react-button'\n\n/** @lends ApUpload */\nconst ApUpload = React.createClass({\n\n  // --------------------\n  // Specs\n  // --------------------\n\n  propTypes: {\n    /** Name of input */\n    name: types.string,\n    /** DOM id of input */\n    id: types.string,\n    /** Allow multiple upload */\n    multiple: types.bool,\n    /** Handler for change event */\n    onChange: types.func,\n    /** Handler for load event */\n    onLoad: types.func,\n    /** Handler for error event */\n    onError: types.func,\n    /** Image width */\n    width: types.number,\n    /** Image height */\n    height: types.number,\n    /** Guide text */\n    text: types.string,\n    /** Accept file type */\n    accept: types.string,\n    /** Guide icon */\n    icon: types.string,\n    /** Icon for close images */\n    closeIcon: types.string,\n    /** Spinner theme */\n    spinner: types.string,\n    /** Value of input */\n    value: types.oneOfType([\n      types.string,\n      types.array\n    ])\n  },\n\n  mixins: [],\n\n  statics: {\n    readFile (file, callback) {\n      let reader = new window.FileReader()\n      reader.onerror = function onerror (err) {\n        callback(err)\n      }\n      reader.onload = function onload (ev) {\n        callback(null, ev.target.result)\n      }\n      reader.readAsDataURL(file)\n    },\n    isImageUrl (url) {\n      const imageExtensions = [\n        '.jpg',\n        '.jpeg',\n        '.svg',\n        '.gif',\n        '.png'\n      ]\n      return /^data:image/.test(url) || !!~imageExtensions.indexOf(path.extname(url))\n    }\n  },\n\n  getInitialState () {\n    const s = this\n    let { props } = s\n    let hasValue = props.value && props.value.length > 0\n    return {\n      spinning: false,\n      error: null,\n      urls: hasValue ? [].concat(props.value) : null\n    }\n  },\n\n  getDefaultProps () {\n    return {\n      name: null,\n      id: `ap-upload-${uuid.v4()}`,\n      multiple: false,\n      width: 180,\n      height: 180,\n      accept: null,\n      text: 'Upload file',\n      icon: 'fa fa-cloud-upload',\n      closeIcon: 'fa fa-close',\n      spinnerIcon: ApSpinner.DEFAULT_THEME,\n      onChange: null,\n      onLoad: null,\n      onError: null\n    }\n  },\n\n  render () {\n    const s = this\n    let { state, props } = s\n    let { width, height } = props\n    return (\n      <div className={classnames('ap-upload', props.className)}\n           style={Object.assign({}, props.style)}>\n        <input type='file'\n               className='ap-upload-input'\n               multiple={ props.multiple }\n               name={ props.name }\n               id={ props.id }\n               accept={ props.accept }\n               onChange={s.handleChange}\n               style={{ width, height }}\n        />\n        <label className='ap-upload-label' htmlFor={ props.id }>\n                    <span className='ap-upload-aligner'>\n                    </span>\n          <span className='ap-upload-label-inner'>\n                        <i className={ classnames('ap-upload-icon', props.icon) }/>\n                        <span className='ap-upload-text'>{props.text}</span>\n            { props.children }\n                    </span>\n        </label>\n        { s._renderPreviewImage(state.urls, width, height) }\n        { s._renderRemoveButton(!!(state.urls && state.urls.length > 0), props.closeIcon) }\n        { s._renderSpinner(state.spinning, props.spinner) }\n      </div>\n    )\n  },\n\n  // --------------------\n  // Lifecycle\n  // --------------------\n\n  componentWillReceiveProps (nextProps) {\n    const s = this\n    const { props } = s\n    let { value } = nextProps\n    let hasValue = value && value.length > 0\n    if (hasValue && (props.value !== value)) {\n      s.setState({ urls: [].concat(value) })\n    }\n  },\n\n  // ------------------\n  // Custom\n  // ------------------\n\n  handleChange (e) {\n    const s = this\n    let { props } = s\n    let { target } = e\n    let files = Array.prototype.slice.call(target.files, 0)\n    let { onChange, onError, onLoad } = props\n\n    s.setState({ spinning: true })\n    if (onChange) {\n      onChange(e)\n    }\n    async.concat(files, ApUpload.readFile, (error, urls) => {\n      s.setState({\n        spinning: false,\n        error,\n        urls\n      })\n      if (error) {\n        if (onError) {\n          onError(error)\n        }\n      } else {\n        if (onLoad) {\n          let loaded = Object.assign({ urls, target })\n          onLoad(loaded)\n        }\n      }\n    })\n  },\n\n  handleRemove () {\n    const s = this\n    let { props } = s\n    let { onLoad } = props\n    s.setState({\n      error: null,\n      urls: null\n    })\n    if (onLoad) {\n      onLoad([])\n    }\n  },\n\n  // ------------------\n  // Private\n  // ------------------\n\n  _renderSpinner (spinning, theme) {\n    const s = this\n    return (\n      <ApSpinner enabled={spinning} theme={theme}>\n      </ApSpinner>\n    )\n  },\n\n  _renderRemoveButton (removable, icon) {\n    const s = this\n    if (!removable) {\n      return null\n    }\n    return (\n      <ApButton onTap={ s.handleRemove } className='ap-upload-remove-button'>\n        <i className={ classnames('ap-upload-remove-icon', icon) }/>\n      </ApButton>\n    )\n  },\n\n  _renderPreviewImage (urls, width, height) {\n    if (!urls) {\n      return null\n    }\n    const s = this\n    return urls\n      .filter((url) => ApUpload.isImageUrl(url))\n      .map((url, i) => (\n        <ApImage key={ url }\n                 src={ url }\n                 height={ height }\n                 width={ width }\n                 className={ classnames('ap-upload-preview-image') }\n                 style={ { left: `${i * 10}%`, top: `${i * 10}%` } }\n                 scale='fit'>\n        </ApImage>\n      ))\n  }\n})\n\nexport default ApUpload\n"]}