UNPKG

apeman-react-upload

Version:
268 lines (230 loc) 21.2 kB
/** * apeman react package for file upload components. * @class ApUpload */ 'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); 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; var props = s.props; var width = props.width; var height = props.height; return _react2.default.createElement( 'div', { className: (0, _classnames2.default)('ap-upload', props.className), style: Object.assign({}, 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 // -------------------- // ------------------ // 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; var onError = props.onError; var onLoad = props.onLoad; s.setState({ spinning: true }); if (onChange) { onChange(e); } _async2.default.concat(files, ApUpload.readFile, function (err, urls) { e.urls = urls; e.target = target; s.setState({ spinning: false, error: err, urls: urls }); if (err) { if (onError) { onError(err); } } else { if (onLoad) { onLoad(e); } } }); }, 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":[],"mappings":"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","file":"ap_upload.js","sourceRoot":"/Users/okunishinishi/Projects/apeman-projects/apeman-react-upload/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  // ------------------\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\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, (err, urls) => {\n      e.urls = urls\n      e.target = target\n      s.setState({\n        spinning: false,\n        error: err,\n        urls: urls\n      })\n      if (err) {\n        if (onError) {\n          onError(err)\n        }\n      } else {\n        if (onLoad) {\n          onLoad(e)\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"]}