apeman-react-upload
Version:
apeman react package for file upload components.
268 lines (230 loc) • 21.2 kB
JavaScript
/**
* 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":";;;;;AAKA;;;;;;AAEA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;;;AACA;;AACA;;AACA;;;;;AAGA,IAAM,WAAW,gBAAM,WAAN,CAAkB;AAAA;;;;;;;AAMjC,aAAW;;AAET,UAAM,iBAAM,MAFH;;AAIT,QAAI,iBAAM,MAJD;;AAMT,cAAU,iBAAM,IANP;;AAQT,cAAU,iBAAM,IARP;;AAUT,YAAQ,iBAAM,IAVL;;AAYT,aAAS,iBAAM,IAZN;;AAcT,WAAO,iBAAM,MAdJ;;AAgBT,YAAQ,iBAAM,MAhBL;;AAkBT,UAAM,iBAAM,MAlBH;;AAoBT,YAAQ,iBAAM,MApBL;;AAsBT,UAAM,iBAAM,MAtBH;;AAwBT,eAAW,iBAAM,MAxBR;;AA0BT,aAAS,iBAAM,MA1BN;;AA4BT,WAAO,iBAAM,SAAN,CAAgB,CACrB,iBAAM,MADe,EAErB,iBAAM,KAFe,CAAhB;AA5BE,GANsB;;AAwCjC,UAAQ,EAxCyB;;AA0CjC,WAAS;AACP,YADO,oBACG,IADH,EACS,QADT,EACmB;AACxB,UAAI,SAAS,IAAI,OAAO,UAAX,EAAb;AACA,aAAO,OAAP,GAAiB,SAAS,OAAT,CAAkB,GAAlB,EAAuB;AACtC,iBAAS,GAAT;AACD,OAFD;AAGA,aAAO,MAAP,GAAgB,SAAS,MAAT,CAAiB,EAAjB,EAAqB;AACnC,iBAAS,IAAT,EAAe,GAAG,MAAH,CAAU,MAAzB;AACD,OAFD;AAGA,aAAO,aAAP,CAAqB,IAArB;AACD,KAVM;AAWP,cAXO,sBAWK,GAXL,EAWU;AACf,UAAM,kBAAkB,CACtB,MADsB,EAEtB,OAFsB,EAGtB,MAHsB,EAItB,MAJsB,EAKtB,MALsB,CAAxB;AAOA,aAAO,eAAc,IAAd,CAAmB,GAAnB,KAA2B,CAAC,EAAC,CAAC,gBAAgB,OAAhB,CAAwB,eAAK,OAAL,CAAa,GAAb,CAAxB;AAArC;AACD;AApBM,GA1CwB;;AAiEjC,iBAjEiC,6BAiEd;AACjB,QAAM,IAAI,IAAV;AADiB,QAEX,KAFW,GAED,CAFC,CAEX,KAFW;;AAGjB,QAAI,WAAW,MAAM,KAAN,IAAe,MAAM,KAAN,CAAY,MAAZ,GAAqB,CAAnD;AACA,WAAO;AACL,gBAAU,KADL;AAEL,aAAO,IAFF;AAGL,YAAM,WAAW,GAAG,MAAH,CAAU,MAAM,KAAhB,CAAX,GAAoC;AAHrC,KAAP;AAKD,GA1EgC;AA4EjC,iBA5EiC,6BA4Ed;AACjB,WAAO;AACL,YAAM,IADD;AAEL,yBAAiB,eAAK,EAAL,EAFZ;AAGL,gBAAU,KAHL;AAIL,aAAO,GAJF;AAKL,cAAQ,GALH;AAML,cAAQ,IANH;AAOL,YAAM,aAPD;AAQL,YAAM,oBARD;AASL,iBAAW,aATN;AAUL,mBAAa,8BAAU,aAVlB;AAWL,gBAAU,IAXL;AAYL,cAAQ,IAZH;AAaL,eAAS;AAbJ,KAAP;AAeD,GA5FgC;AA8FjC,QA9FiC,oBA8FvB;AACR,QAAM,IAAI,IAAV;AADQ,QAEF,KAFE,GAEe,CAFf,CAEF,KAFE;AAAA,QAEK,KAFL,GAEe,CAFf,CAEK,KAFL;AAAA,QAGF,KAHE,GAGgB,KAHhB,CAGF,KAHE;AAAA,QAGK,MAHL,GAGgB,KAHhB,CAGK,MAHL;;AAIR,WACE;AAAA;MAAA,EAAK,WAAW,0BAAW,WAAX,EAAwB,MAAM,SAA9B,CAAhB;AACK,eAAO,OAAO,MAAP,CAAc,EAAd,EAAkB,MAAM,KAAxB,CADZ;MAEE,yCAAO,MAAK,MAAZ;AACO,mBAAU,iBADjB;AAEO,kBAAW,MAAM,QAFxB;AAGO,cAAO,MAAM,IAHpB;AAIO,YAAK,MAAM,EAJlB;AAKO,gBAAS,MAAM,MALtB;AAMO,kBAAU,EAAE,YANnB;AAOO,eAAO,EAAC,YAAD,EAAQ,cAAR;AAPd,QAFF;MAWE;AAAA;QAAA,EAAO,WAAU,iBAAjB,EAAmC,SAAU,MAAM,EAAnD;QACY,wCAAM,WAAU,mBAAhB,GADZ;QAGY;AAAA;UAAA,EAAM,WAAU,uBAAhB;UACI,qCAAG,WAAY,0BAAW,gBAAX,EAA6B,MAAM,IAAnC,CAAf,GADJ;UAEI;AAAA;YAAA,EAAM,WAAU,gBAAhB;YAAkC,MAAM;AAAxC,WAFJ;UAGI,MAAM;AAHV;AAHZ,OAXF;MAoBI,EAAE,mBAAF,CAAsB,MAAM,IAA5B,EAAkC,KAAlC,EAAyC,MAAzC,CApBJ;MAqBI,EAAE,mBAAF,CAAsB,CAAC,EAAE,MAAM,IAAN,IAAc,MAAM,IAAN,CAAW,MAAX,GAAoB,CAApC,CAAvB,EAA+D,MAAM,SAArE,CArBJ;MAsBI,EAAE,cAAF,CAAiB,MAAM,QAAvB,EAAiC,MAAM,OAAvC;AAtBJ,KADF;AA0BD,GA5HgC;;;;;;;;;;;AAsIjC,cAtIiC,wBAsInB,CAtImB,EAsIhB;AACf,QAAM,IAAI,IAAV;AADe,QAET,KAFS,GAEC,CAFD,CAET,KAFS;AAAA,QAGT,MAHS,GAGE,CAHF,CAGT,MAHS;;AAIf,QAAI,QAAQ,MAAM,SAAN,CAAgB,KAAhB,CAAsB,IAAtB,CAA2B,OAAO,KAAlC,EAAyC,CAAzC,CAAZ;;AAJe,QAMT,QANS,GAMqB,KANrB,CAMT,QANS;AAAA,QAMC,OAND,GAMqB,KANrB,CAMC,OAND;AAAA,QAMU,MANV,GAMqB,KANrB,CAMU,MANV;;;AAQf,MAAE,QAAF,CAAW,EAAE,UAAU,IAAZ,EAAX;AACA,QAAI,QAAJ,EAAc;AACZ,eAAS,CAAT;AACD;AACD,oBAAM,MAAN,CAAa,KAAb,EAAoB,SAAS,QAA7B,EAAuC,UAAC,GAAD,EAAM,IAAN,EAAe;AACpD,QAAE,IAAF,GAAS,IAAT;AACA,QAAE,MAAF,GAAW,MAAX;AACA,QAAE,QAAF,CAAW;AACT,kBAAU,KADD;AAET,eAAO,GAFE;AAGT,cAAM;AAHG,OAAX;AAKA,UAAI,GAAJ,EAAS;AACP,YAAI,OAAJ,EAAa;AACX,kBAAQ,GAAR;AACD;AACF,OAJD,MAIO;AACL,YAAI,MAAJ,EAAY;AACV,iBAAO,CAAP;AACD;AACF;AACF,KAjBD;AAkBD,GApKgC;AAsKjC,cAtKiC,0BAsKjB;AACd,QAAM,IAAI,IAAV;AADc,QAER,KAFQ,GAEE,CAFF,CAER,KAFQ;AAAA,QAGR,MAHQ,GAGG,KAHH,CAGR,MAHQ;;AAId,MAAE,QAAF,CAAW;AACT,aAAO,IADE;AAET,YAAM;AAFG,KAAX;AAIA,QAAI,MAAJ,EAAY;AACV,aAAO,EAAP;AACD;AACF,GAjLgC;;;;;;;AAuLjC,gBAvLiC,0BAuLjB,QAvLiB,EAuLP,KAvLO,EAuLA;AAC/B,QAAM,IAAI,IAAV;AACA,WACE,+DAAW,SAAS,QAApB,EAA8B,OAAO,KAArC,GADF;AAID,GA7LgC;AA+LjC,qBA/LiC,+BA+LZ,SA/LY,EA+LD,IA/LC,EA+LK;AACpC,QAAM,IAAI,IAAV;AACA,QAAI,CAAC,SAAL,EAAgB;AACd,aAAO,IAAP;AACD;AACD,WACE;AAAA;MAAA,EAAU,OAAQ,EAAE,YAApB,EAAmC,WAAU,yBAA7C;MACE,qCAAG,WAAY,0BAAW,uBAAX,EAAoC,IAApC,CAAf;AADF,KADF;AAKD,GAzMgC;AA2MjC,qBA3MiC,+BA2MZ,IA3MY,EA2MN,KA3MM,EA2MC,MA3MD,EA2MS;AACxC,QAAI,CAAC,IAAL,EAAW;AACT,aAAO,IAAP;AACD;AACD,QAAM,IAAI,IAAV;AACA,WAAO,KACJ,MADI,CACG,UAAC,GAAD;AAAA,aAAS,SAAS,UAAT,CAAoB,GAApB,CAAT;AAAA,KADH,EAEJ,GAFI,CAEA,UAAC,GAAD,EAAM,CAAN;AAAA,aACH,2DAAS,KAAM,GAAf;AACS,aAAM,GADf;AAES,gBAAS,MAFlB;AAGS,eAAQ,KAHjB;AAIS,mBAAY,0BAAW,yBAAX,CAJrB;AAKS,eAAQ,EAAE,MAAS,IAAI,EAAb,MAAF,EAAsB,KAAQ,IAAI,EAAZ,MAAtB,EALjB;AAMS,eAAM,KANf,GADG;AAAA,KAFA,CAAP;AAYD;AA5NgC,CAAlB,CAAjB;;kBA+Ne,Q","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"]}