UNPKG

apeman-react-upload

Version:
282 lines (240 loc) 24.2 kB
/** * 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"]}