attache-rails 0.4.1 → 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: f691ca01b84ed9632ecbfd03562f5b6b945ad67b
4
- data.tar.gz: 609ef783d0cd979e6182f1ce3f6f6bb9ec4fabd5
3
+ metadata.gz: 9d102ed58a94d38425b805352694aa5011340006
4
+ data.tar.gz: 62d859eb1f12efe92f25f4a3ef2da1a28d00ed87
5
5
  SHA512:
6
- metadata.gz: 460e940fc24d692f513dd8d678dd04953ddb7910e8dd1e9b9dc631dd4ea6e985ad79ee78c764fdb6f6321b78b5509ca146b14912484415008deb4e593eb0e679
7
- data.tar.gz: 17422cee374be8e77da8a3b8720627ee0b402a71d2f51cdf4422ab524c0bc2634220f777706ab7fcc0662aa872175a1e2d3b7f723131d8cf623812c91487a926
6
+ metadata.gz: f2977a0f4f997292c63d7aa824209dc113c655385c50f23ca64007bbdca8b7ed38e2e1f674bd7bceb6140191003d2c1b76f162879fc6ac41b8f0185d4b1d6865
7
+ data.tar.gz: eeddf1577a1f8824800cdddb005a686748eec36f8784612e836fa125362b2ce5ef2f8d11ec5cf3844a3bcc8e8e34768289dda8504904a4a5fb34f61226ff922d
data/README.md CHANGED
@@ -31,7 +31,7 @@ Add the attache javascript to your `application.js`
31
31
  //= require attache
32
32
  ```
33
33
 
34
- If you want to customize the file upload look and feel, define your own react `<AttacheFilePreview/>` renderer *before* including the attache js. For example,
34
+ If you want to customize the file upload look and feel, define your own React `<AttacheFilePreview/>`, `<AttacheHeader/>`, `<AttachePlaceholder/>` renderer *before* including the attache js. For example,
35
35
 
36
36
  ``` javascript
37
37
  //= require ./my_attache_file_preview.js
@@ -1,4 +1,423 @@
1
- //= require attache/cors_upload
2
- //= require attache/bootstrap3
3
- //= require attache/file_input
4
- //= require attache/ujs
1
+ (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
2
+ 'use strict';
3
+
4
+ var _file_input = require('./attache/file_input');
5
+
6
+ var upgradeFileInput = function upgradeFileInput() {
7
+ var safeWords = { 'class': 'className', 'for': 'htmlFor' };
8
+ var sel = document.getElementsByClassName('enable-attache');
9
+ var ele, attrs, name, value;
10
+ for (var i = sel.length - 1; i >= 0; i--) {
11
+ ele = sel[i];
12
+ attrs = {};
13
+ for (var j = 0; j < ele.attributes.length; j++) {
14
+ name = ele.attributes[j].name;
15
+ value = ele.attributes[j].value;
16
+ if (name === 'class') value = value.replace('enable-attache', 'attache-enabled');
17
+ name = safeWords[name] || name;
18
+ attrs[name] = value;
19
+ }
20
+ var wrap = document.createElement('div');
21
+ ele.parentNode.replaceChild(wrap, ele);
22
+ ReactDOM.render(React.createElement(_file_input.AttacheFileInput, React.__spread({}, attrs)), wrap);
23
+ }
24
+ }; /*global $*/
25
+ /*global React*/
26
+ /*global ReactDOM*/
27
+
28
+ $(document).on('page:change', upgradeFileInput);
29
+ $(upgradeFileInput);
30
+
31
+ },{"./attache/file_input":4}],2:[function(require,module,exports){
32
+ 'use strict';
33
+
34
+ Object.defineProperty(exports, "__esModule", {
35
+ value: true
36
+ });
37
+ /*global $*/
38
+ /*global React*/
39
+
40
+ var Bootstrap3FilePreview = exports.Bootstrap3FilePreview = React.createClass({
41
+ displayName: 'Bootstrap3FilePreview',
42
+ getInitialState: function getInitialState() {
43
+ return { srcWas: '' };
44
+ },
45
+ onSrcLoaded: function onSrcLoaded(event) {
46
+ this.setState({ srcWas: this.props.src });
47
+ $(event.target).trigger('attache:imgload');
48
+ },
49
+ onSrcError: function onSrcError(event) {
50
+ $(event.target).trigger('attache:imgerror');
51
+ },
52
+ render: function render() {
53
+ var previewClassName = 'attache-file-preview';
54
+
55
+ // progressbar
56
+ if (this.state.srcWas !== this.props.src) {
57
+ previewClassName = previewClassName + ' attache-loading';
58
+ var className = this.props.className || 'progress-bar progress-bar-striped active' + (this.props.src ? ' progress-bar-success' : '');
59
+ var pctString = this.props.pctString || (this.props.src ? 100 : this.props.percentLoaded) + '%';
60
+ var pctDesc = this.props.pctDesc || (this.props.src ? 'Loading...' : pctString);
61
+ var pctStyle = { width: pctString, minWidth: '3em' };
62
+ var progress = React.createElement(
63
+ 'div',
64
+ { className: 'progress' },
65
+ React.createElement(
66
+ 'div',
67
+ {
68
+ className: className,
69
+ role: 'progressbar',
70
+ 'aria-valuenow': this.props.percentLoaded,
71
+ 'aria-valuemin': '0',
72
+ 'aria-valuemax': '100',
73
+ style: pctStyle },
74
+ pctDesc
75
+ )
76
+ );
77
+ }
78
+
79
+ // img tag
80
+ if (this.props.src) {
81
+ var img = React.createElement('img', { src: this.props.src, onLoad: this.onSrcLoaded, onError: this.onSrcError });
82
+ }
83
+
84
+ // combined
85
+ return React.createElement(
86
+ 'div',
87
+ { className: previewClassName },
88
+ progress,
89
+ img,
90
+ React.createElement(
91
+ 'div',
92
+ { className: 'clearfix' },
93
+ React.createElement(
94
+ 'div',
95
+ { className: 'pull-left' },
96
+ this.props.filename
97
+ ),
98
+ React.createElement(
99
+ 'a',
100
+ {
101
+ href: '#remove',
102
+ className: 'pull-right',
103
+ onClick: this.props.onRemove,
104
+ title: 'Click to remove' },
105
+ '×'
106
+ )
107
+ )
108
+ );
109
+ }
110
+ });
111
+
112
+ var Bootstrap3Placeholder = exports.Bootstrap3Placeholder = React.createClass({
113
+ displayName: 'Bootstrap3Placeholder',
114
+ render: function render() {
115
+ return React.createElement(
116
+ 'div',
117
+ { className: 'attache-file-preview' },
118
+ React.createElement('img', { src: this.props.src })
119
+ );
120
+ }
121
+ });
122
+
123
+ var Bootstrap3Header = exports.Bootstrap3Header = React.createClass({
124
+ displayName: 'Bootstrap3Header',
125
+ render: function render() {
126
+ return React.createElement('noscript', null);
127
+ }
128
+ });
129
+
130
+ },{}],3:[function(require,module,exports){
131
+ 'use strict';
132
+
133
+ var _createClass = (function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; })();
134
+
135
+ Object.defineProperty(exports, "__esModule", {
136
+ value: true
137
+ });
138
+
139
+ function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
140
+
141
+ /*global $*/
142
+ /*global alert*/
143
+ /*global XMLHttpRequest*/
144
+ /*global XDomainRequest*/
145
+
146
+ var counter = 0;
147
+
148
+ var CORSUpload = exports.CORSUpload = (function () {
149
+ function CORSUpload(options) {
150
+ _classCallCheck(this, CORSUpload);
151
+
152
+ if (options == null) options = {};
153
+ var option;
154
+ for (option in options) {
155
+ this[option] = options[option];
156
+ }
157
+ }
158
+
159
+ // for overwriting
160
+
161
+ _createClass(CORSUpload, [{
162
+ key: 'createLocalThumbnail',
163
+ value: function createLocalThumbnail() {}
164
+ }, {
165
+ key: 'onComplete',
166
+ value: function onComplete(uid, json) {}
167
+ }, {
168
+ key: 'onProgress',
169
+ value: function onProgress(uid, json) {}
170
+ }, {
171
+ key: 'onError',
172
+ value: function onError(uid, status) {
173
+ alert(status);
174
+ }
175
+ }, {
176
+ key: 'handleFileSelect',
177
+ value: function handleFileSelect() {
178
+ var f, _i, _len, _results, url, $ele, prefix;
179
+ $ele = $(this.file_element);
180
+ url = $ele.data('uploadurl');
181
+ if ($ele.data('hmac')) {
182
+ url = url + '?hmac=' + encodeURIComponent($ele.data('hmac')) + '&uuid=' + encodeURIComponent($ele.data('uuid')) + '&expiration=' + encodeURIComponent($ele.data('expiration')) + '';
183
+ }
184
+
185
+ prefix = Date.now() + '_';
186
+ _results = [];
187
+ for (_i = 0, _len = this.files.length; _i < _len; _i++) {
188
+ f = this.files[_i];
189
+ this.createLocalThumbnail(f); // if any
190
+ f.uid = prefix + counter++;
191
+ this.onProgress(f.uid, { src: f.src, filename: f.name, percentLoaded: 0, bytesLoaded: 0, bytesTotal: f.size });
192
+ _results.push(this.performUpload(f, url));
193
+ }
194
+ return _results;
195
+ }
196
+ }, {
197
+ key: 'createCORSRequest',
198
+ value: function createCORSRequest(method, url) {
199
+ var xhr;
200
+ xhr = new XMLHttpRequest();
201
+ if (xhr.withCredentials != null) {
202
+ xhr.open(method, url, true);
203
+ } else if (typeof XDomainRequest !== 'undefined') {
204
+ xhr = new XDomainRequest();
205
+ xhr.open(method, url);
206
+ } else {
207
+ xhr = null;
208
+ }
209
+ return xhr;
210
+ }
211
+ }, {
212
+ key: 'performUpload',
213
+ value: function performUpload(file, url) {
214
+ var this_s3upload, xhr;
215
+ this_s3upload = this;
216
+ url = url + (url.indexOf('?') === -1 ? '?' : '&') + 'file=' + encodeURIComponent(file.name);
217
+ xhr = this.createCORSRequest('PUT', url);
218
+ if (!xhr) {
219
+ this.onError(file.uid, 'CORS not supported');
220
+ } else {
221
+ xhr.onload = function (e) {
222
+ if (xhr.status === 200) {
223
+ this_s3upload.onComplete(file.uid, JSON.parse(e.target.responseText));
224
+ } else {
225
+ return this_s3upload.onError(file.uid, xhr.status + ' ' + xhr.statusText);
226
+ }
227
+ };
228
+ xhr.onerror = function () {
229
+ return this_s3upload.onError(file.uid, 'Unable to reach server');
230
+ };
231
+ xhr.upload.onprogress = function (e) {
232
+ var percentLoaded;
233
+ if (e.lengthComputable) {
234
+ percentLoaded = Math.round(e.loaded / e.total * 100);
235
+ return this_s3upload.onProgress(file.uid, { src: file.src, filename: file.name, percentLoaded: percentLoaded, bytesLoaded: e.loaded, bytesTotal: e.total });
236
+ }
237
+ };
238
+ }
239
+ return xhr.send(file);
240
+ }
241
+ }]);
242
+
243
+ return CORSUpload;
244
+ })();
245
+
246
+ },{}],4:[function(require,module,exports){
247
+ 'use strict';
248
+
249
+ var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; /*global $*/
250
+ /*global window*/
251
+ /*global React*/
252
+ /*global ReactDOM*/
253
+
254
+ Object.defineProperty(exports, "__esModule", {
255
+ value: true
256
+ });
257
+ exports.AttacheFileInput = undefined;
258
+
259
+ var _cors_upload = require('./cors_upload');
260
+
261
+ var _bootstrap = require('./bootstrap3');
262
+
263
+ var AttacheFileInput = exports.AttacheFileInput = React.createClass({
264
+ displayName: 'AttacheFileInput',
265
+ getInitialState: function getInitialState() {
266
+ var files = {};
267
+ if (this.props['data-value']) {
268
+ $.each(JSON.parse(this.props['data-value']), function (uid, json) {
269
+ if (json) files[uid] = json;
270
+ });
271
+ }
272
+ return { files: files, attaches_discarded: [], uploading: 0 };
273
+ },
274
+ onRemove: function onRemove(uid, e) {
275
+ e.preventDefault();
276
+ e.stopPropagation();
277
+
278
+ var fieldname = ReactDOM.findDOMNode(this).firstChild.name; // when 'user[avatar]'
279
+ var newfield = fieldname.replace(/\w+\](\[\]|)$/, 'attaches_discarded][]'); // become 'user[attaches_discarded][]'
280
+
281
+ this.state.attaches_discarded.push({ fieldname: newfield, path: this.state.files[uid].path });
282
+ delete this.state.files[uid];
283
+
284
+ this.setState(this.state);
285
+ },
286
+ performUpload: function performUpload(file_element, files) {
287
+ // user cancelled file chooser dialog. ignore
288
+ if (!files || files.length === 0) return;
289
+ if (!this.props.multiple) {
290
+ this.state.files = {};
291
+ files = [files[0]]; // array of 1 element
292
+ }
293
+
294
+ this.setState(this.state);
295
+ // upload the file via CORS
296
+ var that = this;
297
+
298
+ that.state.uploading = that.state.uploading + files.length;
299
+ if (!that.state.submit_buttons) that.state.submit_buttons = $("button,input[type='submit']", $(file_element).parents('form')[0]).filter(':not(:disabled)');
300
+
301
+ var upload = new _cors_upload.CORSUpload({
302
+ file_element: file_element,
303
+ files: files,
304
+ onProgress: this.setFileValue,
305
+ onComplete: function onComplete() {
306
+ that.state.uploading--;
307
+ that.setFileValue.apply(this, arguments);
308
+ },
309
+ onError: function onError(uid, status) {
310
+ that.state.uploading--;
311
+ that.setFileValue(uid, { pctString: '90%', pctDesc: status, className: 'progress-bar progress-bar-danger' });
312
+ }
313
+ });
314
+ upload.handleFileSelect();
315
+
316
+ // we don't want the file binary to be uploaded in the main form
317
+ // so the actual file input is neutered
318
+ file_element.value = '';
319
+ },
320
+ onChange: function onChange() {
321
+ var file_element = ReactDOM.findDOMNode(this).firstChild;
322
+ this.performUpload(file_element, file_element && file_element.files);
323
+ },
324
+ onDragOver: function onDragOver(e) {
325
+ e.stopPropagation();
326
+ e.preventDefault();
327
+ $(ReactDOM.findDOMNode(this)).addClass('attache-dragover');
328
+ },
329
+ onDragLeave: function onDragLeave(e) {
330
+ e.stopPropagation();
331
+ e.preventDefault();
332
+ $(ReactDOM.findDOMNode(this)).removeClass('attache-dragover');
333
+ },
334
+ onDrop: function onDrop(e) {
335
+ e.stopPropagation();
336
+ e.preventDefault();
337
+ var file_element = ReactDOM.findDOMNode(this).firstChild;
338
+ this.performUpload(file_element, e.target.files || e.dataTransfer.files);
339
+ $(ReactDOM.findDOMNode(this)).removeClass('attache-dragover');
340
+ },
341
+ setFileValue: function setFileValue(key, value) {
342
+ this.state.files[key] = value;
343
+ this.setState(this.state);
344
+ },
345
+ render: function render() {
346
+ var that = this;
347
+ var Header = window.AttacheHeader || _bootstrap.Bootstrap3Header;
348
+ var FilePreview = window.AttacheFilePreview || _bootstrap.Bootstrap3FilePreview;
349
+ var Placeholder = window.AttachePlaceholder || _bootstrap.Bootstrap3Placeholder;
350
+
351
+ if (that.state.uploading > 0) {
352
+ that.state.submit_buttons.attr('disabled', true);
353
+ } else if (that.state.submit_buttons) {
354
+ that.state.submit_buttons.attr('disabled', null);
355
+ }
356
+
357
+ var previews = [];
358
+ $.each(that.state.files, function (key, result) {
359
+ // json is input[value], drop non essential values
360
+ var copy = JSON.parse(JSON.stringify(result));
361
+ delete copy.src;
362
+ delete copy.filename;
363
+ var json = JSON.stringify(copy);
364
+ //
365
+ result.multiple = that.props.multiple;
366
+ if (result.path) {
367
+ var parts = result.path.split('/');
368
+ result.filename = parts.pop().split(/[#?]/).shift();
369
+ parts.push(encodeURIComponent(that.props['data-geometry'] || '128x128#'));
370
+ parts.push(encodeURIComponent(result.filename));
371
+ result.src = that.props['data-downloadurl'] + '/' + parts.join('/');
372
+ }
373
+ var previewKey = 'preview' + key;
374
+ previews.push(React.createElement(
375
+ 'div',
376
+ { key: previewKey, className: 'attache-file-input' },
377
+ React.createElement('input', {
378
+ type: 'hidden',
379
+ name: that.props.name,
380
+ value: json,
381
+ readOnly: 'true' }),
382
+ React.createElement(FilePreview, _extends({}, result, { key: key, onRemove: that.onRemove.bind(that, key) }))
383
+ ));
384
+ });
385
+
386
+ var placeholders = [];
387
+ if (previews.length === 0 && that.props['data-placeholder']) {
388
+ $.each(JSON.parse(that.props['data-placeholder']), function (uid, src) {
389
+ placeholders.push(React.createElement(Placeholder, _extends({ key: 'placeholder' }, that.props, { src: src })));
390
+ });
391
+ }
392
+
393
+ var discards = [];
394
+ $.each(that.state.attaches_discarded, function (index, discard) {
395
+ var discardKey = 'discard' + discard.path;
396
+ discards.push(React.createElement('input', {
397
+ key: discardKey,
398
+ type: 'hidden',
399
+ name: discard.fieldname,
400
+ value: discard.path }));
401
+ });
402
+
403
+ var className = ['attache-file-selector', 'attache-placeholders-count-' + placeholders.length, 'attache-previews-count-' + previews.length, this.props['data-classname']].join(' ').trim();
404
+ return React.createElement(
405
+ 'label',
406
+ {
407
+ htmlFor: that.props.id,
408
+ className: className,
409
+ onDragOver: this.onDragOver,
410
+ onDragLeave: this.onDragLeave,
411
+ onDrop: this.onDrop },
412
+ React.createElement('input', _extends({ type: 'file' }, that.props, { onChange: this.onChange })),
413
+ React.createElement('input', { type: 'hidden', name: that.props.name, value: '' }),
414
+ React.createElement(Header, that.props),
415
+ previews,
416
+ placeholders,
417
+ discards
418
+ );
419
+ }
420
+ });
421
+
422
+ },{"./bootstrap3":2,"./cors_upload":3}]},{},[1])
423
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../.nvm/v4.1.0/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","src/javascripts/attache.js","src/javascripts/attache/bootstrap3.js","src/javascripts/attache/cors_upload.js","src/javascripts/attache/file_input.js"],"names":[],"mappings":"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","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/*global $*/\n/*global React*/\n/*global ReactDOM*/\n\nimport { AttacheFileInput } from './attache/file_input'\n\nvar upgradeFileInput = function () {\n  var safeWords = { 'class': 'className', 'for': 'htmlFor' }\n  var sel = document.getElementsByClassName('enable-attache')\n  var ele, attrs, name, value\n  for (var i = sel.length - 1; i >= 0; i--) {\n    ele = sel[i]\n    attrs = {}\n    for (var j = 0; j < ele.attributes.length; j++) {\n      name = ele.attributes[j].name\n      value = ele.attributes[j].value\n      if (name === 'class') value = value.replace('enable-attache', 'attache-enabled')\n      name = safeWords[name] || name\n      attrs[name] = value\n    }\n    var wrap = document.createElement('div')\n    ele.parentNode.replaceChild(wrap, ele)\n    ReactDOM.render(React.createElement(AttacheFileInput, React.__spread({}, attrs)), wrap)\n  }\n}\n\n$(document).on('page:change', upgradeFileInput)\n$(upgradeFileInput)\n","/*global $*/\n/*global React*/\n\nexport var Bootstrap3FilePreview = React.createClass({\n  getInitialState () {\n    return { srcWas: '' }\n  },\n\n  onSrcLoaded (event) {\n    this.setState({ srcWas: this.props.src })\n    $(event.target).trigger('attache:imgload')\n  },\n\n  onSrcError (event) {\n    $(event.target).trigger('attache:imgerror')\n  },\n\n  render () {\n    var previewClassName = 'attache-file-preview'\n\n    // progressbar\n    if (this.state.srcWas !== this.props.src) {\n      previewClassName = previewClassName + ' attache-loading'\n      var className = this.props.className || 'progress-bar progress-bar-striped active' + (this.props.src ? ' progress-bar-success' : '')\n      var pctString = this.props.pctString || (this.props.src ? 100 : this.props.percentLoaded) + '%'\n      var pctDesc = this.props.pctDesc || (this.props.src ? 'Loading...' : pctString)\n      var pctStyle = { width: pctString, minWidth: '3em' }\n      var progress = (\n      <div className=\"progress\">\n        <div\n          className={className}\n          role=\"progressbar\"\n          aria-valuenow={this.props.percentLoaded}\n          aria-valuemin=\"0\"\n          aria-valuemax=\"100\"\n          style={pctStyle}>\n          {pctDesc}\n        </div>\n      </div>\n      )\n    }\n\n    // img tag\n    if (this.props.src) {\n      var img = <img src={this.props.src} onLoad={this.onSrcLoaded} onError={this.onSrcError} />\n    }\n\n    // combined\n    return (\n    <div className={previewClassName}>\n      {progress}\n      {img}\n      <div className=\"clearfix\">\n        <div className=\"pull-left\">\n          {this.props.filename}\n        </div>\n        <a\n          href=\"#remove\"\n          className=\"pull-right\"\n          onClick={this.props.onRemove}\n          title=\"Click to remove\">&times;</a>\n      </div>\n    </div>\n    )\n  }\n})\n\nexport var Bootstrap3Placeholder = React.createClass({\n  render () {\n    return (\n    <div className=\"attache-file-preview\">\n      <img src={this.props.src} />\n    </div>\n    )\n  }\n})\n\nexport var Bootstrap3Header = React.createClass({\n  render () {\n    return (\n    <noscript />\n    )\n  }\n})\n","/*global $*/\n/*global alert*/\n/*global XMLHttpRequest*/\n/*global XDomainRequest*/\n\nvar counter = 0\n\nexport class CORSUpload {\n  constructor (options) {\n    if (options == null) options = {}\n    var option\n    for (option in options) {\n      this[option] = options[option]\n    }\n  }\n\n  // for overwriting\n  createLocalThumbnail () { }\n  onComplete (uid, json) { }\n  onProgress (uid, json) { }\n  onError (uid, status) { alert(status) }\n\n  handleFileSelect () {\n    var f, _i, _len, _results, url, $ele, prefix\n    $ele = $(this.file_element)\n    url = $ele.data('uploadurl')\n    if ($ele.data('hmac')) {\n      url = url +\n        '?hmac=' + encodeURIComponent($ele.data('hmac')) +\n        '&uuid=' + encodeURIComponent($ele.data('uuid')) +\n        '&expiration=' + encodeURIComponent($ele.data('expiration')) +\n        ''\n    }\n\n    prefix = Date.now() + '_'\n    _results = []\n    for (_i = 0, _len = this.files.length; _i < _len; _i++) {\n      f = this.files[_i]\n      this.createLocalThumbnail(f) // if any\n      f.uid = prefix + (counter++)\n      this.onProgress(f.uid, { src: f.src, filename: f.name, percentLoaded: 0, bytesLoaded: 0, bytesTotal: f.size })\n      _results.push(this.performUpload(f, url))\n    }\n    return _results\n  }\n\n  createCORSRequest (method, url) {\n    var xhr\n    xhr = new XMLHttpRequest()\n    if (xhr.withCredentials != null) {\n      xhr.open(method, url, true)\n    } else if (typeof XDomainRequest !== 'undefined') {\n      xhr = new XDomainRequest()\n      xhr.open(method, url)\n    } else {\n      xhr = null\n    }\n    return xhr\n  }\n\n  performUpload (file, url) {\n    var this_s3upload, xhr\n    this_s3upload = this\n    url = url + (url.indexOf('?') === -1 ? '?' : '&') + 'file=' + encodeURIComponent(file.name)\n    xhr = this.createCORSRequest('PUT', url)\n    if (!xhr) {\n      this.onError(file.uid, 'CORS not supported')\n    } else {\n      xhr.onload = function (e) {\n        if (xhr.status === 200) {\n          this_s3upload.onComplete(file.uid, JSON.parse(e.target.responseText))\n        } else {\n          return this_s3upload.onError(file.uid, xhr.status + ' ' + xhr.statusText)\n        }\n      }\n      xhr.onerror = function () {\n        return this_s3upload.onError(file.uid, 'Unable to reach server')\n      }\n      xhr.upload.onprogress = function (e) {\n        var percentLoaded\n        if (e.lengthComputable) {\n          percentLoaded = Math.round((e.loaded / e.total) * 100)\n          return this_s3upload.onProgress(file.uid, { src: file.src, filename: file.name, percentLoaded: percentLoaded, bytesLoaded: e.loaded, bytesTotal: e.total })\n        }\n      }\n    }\n    return xhr.send(file)\n  }\n}\n","/*global $*/\n/*global window*/\n/*global React*/\n/*global ReactDOM*/\n\nimport { CORSUpload } from './cors_upload'\nimport { Bootstrap3Header, Bootstrap3FilePreview, Bootstrap3Placeholder } from './bootstrap3'\n\nexport var AttacheFileInput = React.createClass({\n  getInitialState () {\n    var files = {}\n    if (this.props['data-value']) {\n      $.each(JSON.parse(this.props['data-value']), function (uid, json) {\n        if (json) files[uid] = json\n      })\n    }\n    return { files: files, attaches_discarded: [], uploading: 0 }\n  },\n\n  onRemove (uid, e) {\n    e.preventDefault()\n    e.stopPropagation()\n\n    var fieldname = ReactDOM.findDOMNode(this).firstChild.name // when   'user[avatar]'\n    var newfield = fieldname.replace(/\\w+\\](\\[\\]|)$/, 'attaches_discarded][]') // become 'user[attaches_discarded][]'\n\n    this.state.attaches_discarded.push({ fieldname: newfield, path: this.state.files[uid].path })\n    delete this.state.files[uid]\n\n    this.setState(this.state)\n  },\n\n  performUpload (file_element, files) {\n    // user cancelled file chooser dialog. ignore\n    if (!files || files.length === 0) return\n    if (!this.props.multiple) {\n      this.state.files = {}\n      files = [files[0]] // array of 1 element\n    }\n\n    this.setState(this.state)\n    // upload the file via CORS\n    var that = this\n\n    that.state.uploading = that.state.uploading + files.length\n    if (!that.state.submit_buttons) that.state.submit_buttons = $(\"button,input[type='submit']\", $(file_element).parents('form')[0]).filter(':not(:disabled)')\n\n    var upload = new CORSUpload({\n      file_element: file_element,\n      files: files,\n      onProgress: this.setFileValue,\n      onComplete () {\n        that.state.uploading--\n        that.setFileValue.apply(this, arguments)\n      },\n      onError (uid, status) {\n        that.state.uploading--\n        that.setFileValue(uid, { pctString: '90%', pctDesc: status, className: 'progress-bar progress-bar-danger' })\n      }\n    })\n    upload.handleFileSelect()\n\n    // we don't want the file binary to be uploaded in the main form\n    // so the actual file input is neutered\n    file_element.value = ''\n  },\n\n  onChange () {\n    var file_element = ReactDOM.findDOMNode(this).firstChild\n    this.performUpload(file_element, file_element && file_element.files)\n  },\n\n  onDragOver (e) {\n    e.stopPropagation()\n    e.preventDefault()\n    $(ReactDOM.findDOMNode(this)).addClass('attache-dragover')\n  },\n\n  onDragLeave (e) {\n    e.stopPropagation()\n    e.preventDefault()\n    $(ReactDOM.findDOMNode(this)).removeClass('attache-dragover')\n  },\n\n  onDrop (e) {\n    e.stopPropagation()\n    e.preventDefault()\n    var file_element = ReactDOM.findDOMNode(this).firstChild\n    this.performUpload(file_element, e.target.files || e.dataTransfer.files)\n    $(ReactDOM.findDOMNode(this)).removeClass('attache-dragover')\n  },\n\n  setFileValue (key, value) {\n    this.state.files[key] = value\n    this.setState(this.state)\n  },\n\n  render () {\n    var that = this\n    var Header = window.AttacheHeader || Bootstrap3Header\n    var FilePreview = window.AttacheFilePreview || Bootstrap3FilePreview\n    var Placeholder = window.AttachePlaceholder || Bootstrap3Placeholder\n\n    if (that.state.uploading > 0) {\n      that.state.submit_buttons.attr('disabled', true)\n    } else if (that.state.submit_buttons) {\n      that.state.submit_buttons.attr('disabled', null)\n    }\n\n    var previews = []\n    $.each(that.state.files, function (key, result) {\n      // json is input[value], drop non essential values\n      var copy = JSON.parse(JSON.stringify(result))\n      delete copy.src\n      delete copy.filename\n      var json = JSON.stringify(copy)\n      //\n      result.multiple = that.props.multiple\n      if (result.path) {\n        var parts = result.path.split('/')\n        result.filename = parts.pop().split(/[#?]/).shift()\n        parts.push(encodeURIComponent(that.props['data-geometry'] || '128x128#'))\n        parts.push(encodeURIComponent(result.filename))\n        result.src = that.props['data-downloadurl'] + '/' + parts.join('/')\n      }\n      var previewKey = 'preview' + key\n      previews.push(\n        <div key={previewKey} className=\"attache-file-input\">\n          <input\n            type=\"hidden\"\n            name={that.props.name}\n            value={json}\n            readOnly=\"true\" />\n          <FilePreview {...result} key={key} onRemove={that.onRemove.bind(that, key)} />\n        </div>\n      )\n    })\n\n    var placeholders = []\n    if (previews.length === 0 && that.props['data-placeholder']) {\n      $.each(JSON.parse(that.props['data-placeholder']), function (uid, src) {\n        placeholders.push(\n          <Placeholder key=\"placeholder\" {...that.props} src={src} />\n        )\n      })\n    }\n\n    var discards = []\n    $.each(that.state.attaches_discarded, function (index, discard) {\n      var discardKey = 'discard' + discard.path\n      discards.push(\n        <input\n          key={discardKey}\n          type=\"hidden\"\n          name={discard.fieldname}\n          value={discard.path} />\n      )\n    })\n\n    var className = ['attache-file-selector', 'attache-placeholders-count-' + placeholders.length, 'attache-previews-count-' + previews.length, this.props['data-classname']].join(' ').trim()\n    return (\n    <label\n      htmlFor={that.props.id}\n      className={className}\n      onDragOver={this.onDragOver}\n      onDragLeave={this.onDragLeave}\n      onDrop={this.onDrop}>\n      <input type=\"file\" {...that.props} onChange={this.onChange} />\n      <input type=\"hidden\" name={that.props.name} value=\"\" />\n      <Header {...that.props} />\n      {previews}\n      {placeholders}\n      {discards}\n    </label>\n    )\n  }\n})\n"]}
data/app/index.html ADDED
@@ -0,0 +1,39 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="utf-8">
5
+ <meta http-equiv="X-UA-Compatible" content="IE=edge">
6
+ <meta name="viewport" content="width=device-width, initial-scale=1">
7
+ <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css" integrity="sha384-1q8mTJOASx8j1Au+a5WDVnPi2lkFfwwEAa8hDDdjZlpLegxhjVME1fgjWPGmkzs7" crossorigin="anonymous">
8
+ </head>
9
+ <body>
10
+ <div class="container">
11
+ <div class="row marketing">
12
+ <div class="col-sm-12">
13
+ <div class="page-header">
14
+ <h1>Test Form</h1>
15
+ </div>
16
+ <div class="panel">
17
+ <form class="form-horizontal">
18
+ <div class="form-group">
19
+ <label for="inputFile" class="col-sm-2 control-label">File</label>
20
+ <div class="col-sm-10">
21
+ <input type="file" class="form-control enable-attache" id="inputFile" placeholder="File">
22
+ </div>
23
+ </div>
24
+ <div class="form-group">
25
+ <div class="col-sm-offset-2 col-sm-10">
26
+ <button type="submit" class="btn btn-default">Sign in</button>
27
+ </div>
28
+ </div>
29
+ </form>
30
+ </div>
31
+ </div>
32
+ </div>
33
+ </div>
34
+ <script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.0.0-alpha1/jquery.js"></script>
35
+ <script src="https://cdnjs.cloudflare.com/ajax/libs/react/0.14.3/react.js"></script>
36
+ <script src="https://cdnjs.cloudflare.com/ajax/libs/react/0.14.3/react-dom.js"></script>
37
+ <script src="assets/javascripts/attache.js"></script>
38
+ </body>
39
+ </html>
@@ -14,9 +14,13 @@ module Attache
14
14
 
15
15
  # `discard` management
16
16
  base.class_eval do
17
+ attr_accessor :attaches_to_backup
17
18
  attr_accessor :attaches_discarded
18
- after_commit if: :attaches_discarded do |instance|
19
- instance.attaches_discard!(instance.attaches_discarded)
19
+ after_commit do |instance|
20
+ instance.attaches_discard!(instance.attaches_discarded) if instance.attaches_discarded.present?
21
+ instance.attaches_discarded = []
22
+ instance.attaches_backup!(instance.attaches_to_backup) if instance.attaches_to_backup.present?
23
+ instance.attaches_to_backup = []
20
24
  end
21
25
  end
22
26
  end
@@ -36,8 +40,8 @@ module Attache
36
40
  define_method "#{name}_url", -> (geometry) { attache_field_urls(self.send(name), geometry).try(:first) }
37
41
  define_method "#{name}_attributes", -> (geometry) { attache_field_attributes(self.send(name), geometry).try(:first) }
38
42
  define_method "#{name}=", -> (value) { super(attache_field_set(Array.wrap(value)).try(:first)) }
39
- after_update -> { self.attaches_discarded ||= []; attache_mark_for_discarding(self.send("#{name}_was"), self.send("#{name}"), self.attaches_discarded) }
40
- after_destroy -> { self.attaches_discarded ||= []; attache_mark_for_discarding(self.send("#{name}_was"), [], self.attaches_discarded) }
43
+ after_save -> { attache_update_pending_diffs(self.send("#{name}_was"), self.send("#{name}"), self.attaches_to_backup ||= [], self.attaches_discarded ||= []) }
44
+ after_destroy -> { attache_update_pending_diffs(self.send("#{name}_was"), [], self.attaches_to_backup ||= [], self.attaches_discarded ||= []) }
41
45
  end
42
46
 
43
47
  def has_many_attaches(name)
@@ -46,8 +50,8 @@ module Attache
46
50
  define_method "#{name}_urls", -> (geometry) { attache_field_urls(self.send(name), geometry) }
47
51
  define_method "#{name}_attributes", -> (geometry) { attache_field_attributes(self.send(name), geometry) }
48
52
  define_method "#{name}=", -> (value) { super(attache_field_set(Array.wrap(value))) }
49
- after_update -> { self.attaches_discarded ||= []; attache_mark_for_discarding(self.send("#{name}_was"), self.send("#{name}"), self.attaches_discarded) }
50
- after_destroy -> { self.attaches_discarded ||= []; attache_mark_for_discarding(self.send("#{name}_was"), [], self.attaches_discarded) }
53
+ after_save -> { attache_update_pending_diffs(self.send("#{name}_was"), self.send("#{name}"), self.attaches_to_backup ||= [], self.attaches_discarded ||= []) }
54
+ after_destroy -> { attache_update_pending_diffs(self.send("#{name}_was"), [], self.attaches_to_backup ||= [], self.attaches_discarded ||= []) }
51
55
  end
52
56
  end
53
57
  end
@@ -1,5 +1,5 @@
1
1
  module Attache
2
2
  module Rails
3
- VERSION = "0.4.1"
3
+ VERSION = "1.0.0"
4
4
  end
5
5
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: attache-rails
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.4.1
4
+ version: 1.0.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - choonkeat
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2015-11-14 00:00:00.000000000 Z
11
+ date: 2015-12-16 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rails
@@ -30,14 +30,14 @@ dependencies:
30
30
  requirements:
31
31
  - - "~>"
32
32
  - !ruby/object:Gem::Version
33
- version: 0.2.0
33
+ version: 1.0.0
34
34
  type: :runtime
35
35
  prerelease: false
36
36
  version_requirements: !ruby/object:Gem::Requirement
37
37
  requirements:
38
38
  - - "~>"
39
39
  - !ruby/object:Gem::Version
40
- version: 0.2.0
40
+ version: 1.0.0
41
41
  - !ruby/object:Gem::Dependency
42
42
  name: httpclient
43
43
  requirement: !ruby/object:Gem::Requirement
@@ -119,12 +119,7 @@ files:
119
119
  - README.md
120
120
  - Rakefile
121
121
  - app/assets/javascripts/attache.js
122
- - app/assets/javascripts/attache/bootstrap3.js
123
- - app/assets/javascripts/attache/bootstrap3.js.jsx
124
- - app/assets/javascripts/attache/cors_upload.js
125
- - app/assets/javascripts/attache/file_input.js
126
- - app/assets/javascripts/attache/file_input.js.jsx
127
- - app/assets/javascripts/attache/ujs.js
122
+ - app/index.html
128
123
  - lib/attache/rails.rb
129
124
  - lib/attache/rails/engine.rb
130
125
  - lib/attache/rails/model.rb
@@ -1,77 +0,0 @@
1
- if (typeof AttacheFilePreview === 'undefined') {
2
-
3
- var AttacheFilePreview = React.createClass({displayName: "AttacheFilePreview",
4
-
5
- getInitialState: function() {
6
- return { srcWas: '' };
7
- },
8
-
9
- onSrcLoaded: function() {
10
- this.setState({ srcWas: this.props.src });
11
- },
12
-
13
- render: function() {
14
- var previewClassName = "attache-file-preview";
15
-
16
- // progressbar
17
- if (this.state.srcWas != this.props.src) {
18
- previewClassName = previewClassName + " attache-loading";
19
- var className = this.props.className || "progress-bar progress-bar-striped active" + (this.props.src ? " progress-bar-success" : "");
20
- var pctString = this.props.pctString || (this.props.src ? 100 : this.props.percentLoaded) + "%";
21
- var pctDesc = this.props.pctDesc || (this.props.src ? 'Loading...' : pctString);
22
- var pctStyle = { width: pctString, minWidth: '3em' };
23
- var progress = (
24
- React.createElement("div", {className: "progress"},
25
- React.createElement("div", {className: className, role: "progressbar", "aria-valuenow": this.props.percentLoaded, "aria-valuemin": "0", "aria-valuemax": "100", style: pctStyle},
26
- pctDesc
27
- )
28
- )
29
- );
30
- }
31
-
32
- // img tag
33
- if (this.props.src) {
34
- var img = React.createElement("img", {src: this.props.src, onLoad: this.onSrcLoaded});
35
- }
36
-
37
- // combined
38
- return (
39
- React.createElement("div", {className: previewClassName},
40
- progress,
41
- img,
42
- React.createElement("div", {className: "clearfix"},
43
- React.createElement("div", {className: "pull-left"}, this.props.filename),
44
- React.createElement("a", {href: "#remove", className: "pull-right", onClick: this.props.onRemove, title: "Click to remove"}, "×")
45
- )
46
- )
47
- );
48
- }
49
- });
50
-
51
- }
52
-
53
- if (typeof AttachePlaceholder === 'undefined') {
54
-
55
- var AttachePlaceholder = React.createClass({displayName: "AttachePlaceholder",
56
- render: function() {
57
- return (
58
- React.createElement("div", {className: "attache-file-preview"},
59
- React.createElement("img", {src: this.props.src})
60
- )
61
- );
62
- }
63
- });
64
-
65
- }
66
-
67
- if (typeof AttacheHeader === 'undefined') {
68
-
69
- var AttacheHeader = React.createClass({displayName: "AttacheHeader",
70
- render: function() {
71
- return (
72
- React.createElement("noscript", null)
73
- );
74
- }
75
- });
76
-
77
- }
@@ -1,77 +0,0 @@
1
- if (typeof AttacheFilePreview === 'undefined') {
2
-
3
- var AttacheFilePreview = React.createClass({
4
-
5
- getInitialState: function() {
6
- return { srcWas: '' };
7
- },
8
-
9
- onSrcLoaded: function() {
10
- this.setState({ srcWas: this.props.src });
11
- },
12
-
13
- render: function() {
14
- var previewClassName = "attache-file-preview";
15
-
16
- // progressbar
17
- if (this.state.srcWas != this.props.src) {
18
- previewClassName = previewClassName + " attache-loading";
19
- var className = this.props.className || "progress-bar progress-bar-striped active" + (this.props.src ? " progress-bar-success" : "");
20
- var pctString = this.props.pctString || (this.props.src ? 100 : this.props.percentLoaded) + "%";
21
- var pctDesc = this.props.pctDesc || (this.props.src ? 'Loading...' : pctString);
22
- var pctStyle = { width: pctString, minWidth: '3em' };
23
- var progress = (
24
- <div className="progress">
25
- <div className={className} role="progressbar" aria-valuenow={this.props.percentLoaded} aria-valuemin="0" aria-valuemax="100" style={pctStyle}>
26
- {pctDesc}
27
- </div>
28
- </div>
29
- );
30
- }
31
-
32
- // img tag
33
- if (this.props.src) {
34
- var img = <img src={this.props.src} onLoad={this.onSrcLoaded} />;
35
- }
36
-
37
- // combined
38
- return (
39
- <div className={previewClassName}>
40
- {progress}
41
- {img}
42
- <div className="clearfix">
43
- <div className="pull-left">{this.props.filename}</div>
44
- <a href="#remove" className="pull-right" onClick={this.props.onRemove} title="Click to remove">&times;</a>
45
- </div>
46
- </div>
47
- );
48
- }
49
- });
50
-
51
- }
52
-
53
- if (typeof AttachePlaceholder === 'undefined') {
54
-
55
- var AttachePlaceholder = React.createClass({
56
- render: function() {
57
- return (
58
- <div className="attache-file-preview">
59
- <img src={this.props.src} />
60
- </div>
61
- );
62
- }
63
- });
64
-
65
- }
66
-
67
- if (typeof AttacheHeader === 'undefined') {
68
-
69
- var AttacheHeader = React.createClass({
70
- render: function() {
71
- return (
72
- <noscript />
73
- );
74
- }
75
- });
76
-
77
- }
@@ -1,87 +0,0 @@
1
- var AttacheCORSUpload = (function() {
2
- var counter = 0;
3
-
4
- AttacheCORSUpload.prototype.onComplete = function(uid, json) { };
5
- AttacheCORSUpload.prototype.onProgress = function(uid, json) { };
6
- AttacheCORSUpload.prototype.onError = function(uid, status) { alert(status); };
7
- AttacheCORSUpload.prototype.createLocalThumbnail = function() { }; // for overwriting
8
-
9
- function AttacheCORSUpload(options) {
10
- if (options == null) options = {};
11
- for (option in options) {
12
- this[option] = options[option];
13
- }
14
- this.handleFileSelect(options.file_element, options.files);
15
- }
16
-
17
- AttacheCORSUpload.prototype.handleFileSelect = function(file_element, files) {
18
- var f, output, _i, _len, _results, url, $ele, prefix;
19
- $ele = $(file_element);
20
- url = $ele.data('uploadurl');
21
- if ($ele.data('hmac')) {
22
- url = url +
23
- "?hmac=" + encodeURIComponent($ele.data('hmac')) +
24
- "&uuid=" + encodeURIComponent($ele.data('uuid')) +
25
- "&expiration=" + encodeURIComponent($ele.data('expiration')) +
26
- ""
27
- }
28
-
29
- prefix = Date.now() + "_";
30
- output = [];
31
- _results = [];
32
- for (_i = 0, _len = files.length; _i < _len; _i++) {
33
- f = files[_i];
34
- this.createLocalThumbnail(f); // if any
35
- f.uid = prefix + (counter++);
36
- this.onProgress(f.uid, { src: f.src, filename: f.name, percentLoaded: 0, bytesLoaded: 0, bytesTotal: f.size });
37
- _results.push(this.performUpload(f, url));
38
- }
39
- return _results;
40
- };
41
-
42
- AttacheCORSUpload.prototype.createCORSRequest = function(method, url) {
43
- var xhr;
44
- xhr = new XMLHttpRequest();
45
- if (xhr.withCredentials != null) {
46
- xhr.open(method, url, true);
47
- } else if (typeof XDomainRequest !== "undefined") {
48
- xhr = new XDomainRequest();
49
- xhr.open(method, url);
50
- } else {
51
- xhr = null;
52
- }
53
- return xhr;
54
- };
55
-
56
- AttacheCORSUpload.prototype.performUpload = function(file, url) {
57
- var this_s3upload, xhr;
58
- this_s3upload = this;
59
- url = url + (url.indexOf('?') == -1 ? '?' : '&') + 'file=' + encodeURIComponent(file.name);
60
- xhr = this.createCORSRequest('PUT', url);
61
- if (!xhr) {
62
- this.onError(file.uid, 'CORS not supported');
63
- } else {
64
- xhr.onload = function(e) {
65
- if (xhr.status === 200) {
66
- this_s3upload.onComplete(file.uid, JSON.parse(e.target.responseText));
67
- } else {
68
- return this_s3upload.onError(file.uid, xhr.status + ' ' + xhr.statusText);
69
- }
70
- };
71
- xhr.onerror = function() {
72
- return this_s3upload.onError(file.uid, 'Unable to reach server');
73
- };
74
- xhr.upload.onprogress = function(e) {
75
- var percentLoaded;
76
- if (e.lengthComputable) {
77
- percentLoaded = Math.round((e.loaded / e.total) * 100);
78
- return this_s3upload.onProgress(file.uid, { src: file.src, filename: file.name, percentLoaded: percentLoaded, bytesLoaded: e.loaded, bytesTotal: e.total });
79
- }
80
- };
81
- }
82
- return xhr.send(file);
83
- };
84
-
85
- return AttacheCORSUpload;
86
-
87
- })();
@@ -1,153 +0,0 @@
1
- var AttacheFileInput = React.createClass({displayName: "AttacheFileInput",
2
-
3
- getInitialState: function() {
4
- var files = {};
5
- if (this.props['data-value']) $.each(JSON.parse(this.props['data-value']), function(uid, json) {
6
- if (json) files[uid] = json;
7
- });
8
- return {files: files, attaches_discarded: [], uploading: 0 };
9
- },
10
-
11
- onRemove: function(uid, e) {
12
- e.preventDefault();
13
- e.stopPropagation();
14
-
15
- var fieldname = this.getDOMNode().firstChild.name; // when 'user[avatar]'
16
- var newfield = fieldname.replace(/\w+\](\[\]|)$/, 'attaches_discarded][]'); // become 'user[attaches_discarded][]'
17
-
18
- this.state.attaches_discarded.push({ fieldname: newfield, path: this.state.files[uid].path });
19
- delete this.state.files[uid];
20
-
21
- this.setState(this.state);
22
- },
23
-
24
- performUpload: function(file_element, files) {
25
- // user cancelled file chooser dialog. ignore
26
- if (! files || files.length == 0) return;
27
- if (! this.props.multiple) {
28
- this.state.files = {};
29
- files = [files[0]]; // array of 1 element
30
- }
31
-
32
- this.setState(this.state);
33
- // upload the file via CORS
34
- var that = this;
35
-
36
- that.state.uploading = that.state.uploading + files.length;
37
- if (! that.state.submit_buttons) that.state.submit_buttons = $("button,input[type='submit']", $(file_element).parents('form')[0]).filter(':not(:disabled)');
38
-
39
- new AttacheCORSUpload({
40
- file_element: file_element,
41
- files: files,
42
- onComplete: function() {
43
- that.state.uploading--;
44
- that.setFileValue.apply(this, arguments);
45
- },
46
- onProgress: this.setFileValue,
47
- onError: function(uid, status) {
48
- that.state.uploading--;
49
- that.setFileValue(uid, { pctString: '90%', pctDesc: status, className: 'progress-bar progress-bar-danger' });
50
- }
51
- });
52
-
53
- // we don't want the file binary to be uploaded in the main form
54
- // so the actual file input is neutered
55
- file_element.value = '';
56
- },
57
-
58
- onChange: function() {
59
- var file_element = this.getDOMNode().firstChild;
60
- this.performUpload(file_element, file_element && file_element.files);
61
- },
62
-
63
- onDragOver: function(e) {
64
- e.stopPropagation();
65
- e.preventDefault();
66
- $(this.getDOMNode()).addClass('attache-dragover');
67
- },
68
-
69
- onDragLeave: function(e) {
70
- e.stopPropagation();
71
- e.preventDefault();
72
- $(this.getDOMNode()).removeClass('attache-dragover');
73
- },
74
-
75
- onDrop: function(e) {
76
- e.stopPropagation();
77
- e.preventDefault();
78
- var file_element = this.getDOMNode().firstChild;
79
- this.performUpload(file_element, e.target.files || e.dataTransfer.files);
80
- $(this.getDOMNode()).removeClass('attache-dragover');
81
- },
82
-
83
- setFileValue: function(key, value) {
84
- this.state.files[key] = value;
85
- this.setState(this.state);
86
- },
87
-
88
- render: function() {
89
- var that = this;
90
-
91
- if (that.state.uploading > 0) {
92
- that.state.submit_buttons.attr('disabled', true);
93
- } else if (that.state.submit_buttons) {
94
- that.state.submit_buttons.attr('disabled', null);
95
- }
96
-
97
- var Header = eval(this.props['data-header-component'] || 'AttacheHeader');
98
- var Preview = eval(this.props['data-preview-component'] || 'AttacheFilePreview');
99
- var Placeholder = eval(this.props['data-placeholder-component'] || 'AttachePlaceholder');
100
-
101
- var previews = [];
102
- $.each(that.state.files, function(key, result) {
103
- // json is input[value], drop non essential values
104
- var copy = JSON.parse(JSON.stringify(result));
105
- delete copy.src;
106
- delete copy.filename;
107
- var json = JSON.stringify(copy);
108
- //
109
- result.multiple = that.props.multiple;
110
- if (result.path) {
111
- var parts = result.path.split('/');
112
- result.filename = parts.pop().split(/[#?]/).shift();
113
- parts.push(encodeURIComponent(that.props['data-geometry'] || '128x128#'));
114
- parts.push(encodeURIComponent(result.filename));
115
- result.src = that.props['data-downloadurl'] + '/' + parts.join('/');
116
- }
117
- var previewKey = "preview" + key;
118
- previews.push(
119
- React.createElement("div", {key: previewKey, className: "attache-file-input"},
120
- React.createElement("input", {type: "hidden", name: that.props.name, value: json, readOnly: "true"}),
121
- React.createElement(Preview, React.__spread({}, result, {key: key, onRemove: that.onRemove.bind(that, key)}))
122
- )
123
- );
124
- });
125
-
126
- var placeholders = [];
127
- if (previews.length == 0 && that.props['data-placeholder']) $.each(JSON.parse(that.props['data-placeholder']), function(uid, src) {
128
- placeholders.push(
129
- React.createElement(Placeholder, React.__spread({key: "placeholder"}, that.props, {src: src}))
130
- );
131
- });
132
-
133
- var discards = [];
134
- $.each(that.state.attaches_discarded, function(index, discard) {
135
- var discardKey = "discard" + discard.path;
136
- discards.push(
137
- React.createElement("input", {key: discardKey, type: "hidden", name: discard.fieldname, value: discard.path})
138
- );
139
- });
140
-
141
- var className = ["attache-file-selector", "attache-placeholders-count-" + placeholders.length, "attache-previews-count-" + previews.length, this.props['data-classname']].join(' ').trim();
142
- return (
143
- React.createElement("label", {htmlFor: that.props.id, className: className, onDragOver: this.onDragOver, onDragLeave: this.onDragLeave, onDrop: this.onDrop},
144
- React.createElement("input", React.__spread({type: "file"}, that.props, {onChange: this.onChange})),
145
- React.createElement("input", {type: "hidden", name: that.props.name, value: ""}),
146
- React.createElement(Header, React.__spread({}, that.props)),
147
- previews,
148
- placeholders,
149
- discards
150
- )
151
- );
152
- }
153
- });
@@ -1,153 +0,0 @@
1
- var AttacheFileInput = React.createClass({
2
-
3
- getInitialState: function() {
4
- var files = {};
5
- if (this.props['data-value']) $.each(JSON.parse(this.props['data-value']), function(uid, json) {
6
- if (json) files[uid] = json;
7
- });
8
- return {files: files, attaches_discarded: [], uploading: 0 };
9
- },
10
-
11
- onRemove: function(uid, e) {
12
- e.preventDefault();
13
- e.stopPropagation();
14
-
15
- var fieldname = this.getDOMNode().firstChild.name; // when 'user[avatar]'
16
- var newfield = fieldname.replace(/\w+\](\[\]|)$/, 'attaches_discarded][]'); // become 'user[attaches_discarded][]'
17
-
18
- this.state.attaches_discarded.push({ fieldname: newfield, path: this.state.files[uid].path });
19
- delete this.state.files[uid];
20
-
21
- this.setState(this.state);
22
- },
23
-
24
- performUpload: function(file_element, files) {
25
- // user cancelled file chooser dialog. ignore
26
- if (! files || files.length == 0) return;
27
- if (! this.props.multiple) {
28
- this.state.files = {};
29
- files = [files[0]]; // array of 1 element
30
- }
31
-
32
- this.setState(this.state);
33
- // upload the file via CORS
34
- var that = this;
35
-
36
- that.state.uploading = that.state.uploading + files.length;
37
- if (! that.state.submit_buttons) that.state.submit_buttons = $("button,input[type='submit']", $(file_element).parents('form')[0]).filter(':not(:disabled)');
38
-
39
- new AttacheCORSUpload({
40
- file_element: file_element,
41
- files: files,
42
- onComplete: function() {
43
- that.state.uploading--;
44
- that.setFileValue.apply(this, arguments);
45
- },
46
- onProgress: this.setFileValue,
47
- onError: function(uid, status) {
48
- that.state.uploading--;
49
- that.setFileValue(uid, { pctString: '90%', pctDesc: status, className: 'progress-bar progress-bar-danger' });
50
- }
51
- });
52
-
53
- // we don't want the file binary to be uploaded in the main form
54
- // so the actual file input is neutered
55
- file_element.value = '';
56
- },
57
-
58
- onChange: function() {
59
- var file_element = this.getDOMNode().firstChild;
60
- this.performUpload(file_element, file_element && file_element.files);
61
- },
62
-
63
- onDragOver: function(e) {
64
- e.stopPropagation();
65
- e.preventDefault();
66
- $(this.getDOMNode()).addClass('attache-dragover');
67
- },
68
-
69
- onDragLeave: function(e) {
70
- e.stopPropagation();
71
- e.preventDefault();
72
- $(this.getDOMNode()).removeClass('attache-dragover');
73
- },
74
-
75
- onDrop: function(e) {
76
- e.stopPropagation();
77
- e.preventDefault();
78
- var file_element = this.getDOMNode().firstChild;
79
- this.performUpload(file_element, e.target.files || e.dataTransfer.files);
80
- $(this.getDOMNode()).removeClass('attache-dragover');
81
- },
82
-
83
- setFileValue: function(key, value) {
84
- this.state.files[key] = value;
85
- this.setState(this.state);
86
- },
87
-
88
- render: function() {
89
- var that = this;
90
-
91
- if (that.state.uploading > 0) {
92
- that.state.submit_buttons.attr('disabled', true);
93
- } else if (that.state.submit_buttons) {
94
- that.state.submit_buttons.attr('disabled', null);
95
- }
96
-
97
- var Header = eval(this.props['data-header-component'] || 'AttacheHeader');
98
- var Preview = eval(this.props['data-preview-component'] || 'AttacheFilePreview');
99
- var Placeholder = eval(this.props['data-placeholder-component'] || 'AttachePlaceholder');
100
-
101
- var previews = [];
102
- $.each(that.state.files, function(key, result) {
103
- // json is input[value], drop non essential values
104
- var copy = JSON.parse(JSON.stringify(result));
105
- delete copy.src;
106
- delete copy.filename;
107
- var json = JSON.stringify(copy);
108
- //
109
- result.multiple = that.props.multiple;
110
- if (result.path) {
111
- var parts = result.path.split('/');
112
- result.filename = parts.pop().split(/[#?]/).shift();
113
- parts.push(encodeURIComponent(that.props['data-geometry'] || '128x128#'));
114
- parts.push(encodeURIComponent(result.filename));
115
- result.src = that.props['data-downloadurl'] + '/' + parts.join('/');
116
- }
117
- var previewKey = "preview" + key;
118
- previews.push(
119
- <div key={previewKey} className="attache-file-input">
120
- <input type="hidden" name={that.props.name} value={json} readOnly="true" />
121
- <Preview {...result} key={key} onRemove={that.onRemove.bind(that, key)}/>
122
- </div>
123
- );
124
- });
125
-
126
- var placeholders = [];
127
- if (previews.length == 0 && that.props['data-placeholder']) $.each(JSON.parse(that.props['data-placeholder']), function(uid, src) {
128
- placeholders.push(
129
- <Placeholder key="placeholder" {...that.props} src={src} />
130
- );
131
- });
132
-
133
- var discards = [];
134
- $.each(that.state.attaches_discarded, function(index, discard) {
135
- var discardKey = "discard" + discard.path;
136
- discards.push(
137
- <input key={discardKey} type="hidden" name={discard.fieldname} value={discard.path} />
138
- );
139
- });
140
-
141
- var className = ["attache-file-selector", "attache-placeholders-count-" + placeholders.length, "attache-previews-count-" + previews.length, this.props['data-classname']].join(' ').trim();
142
- return (
143
- <label htmlFor={that.props.id} className={className} onDragOver={this.onDragOver} onDragLeave={this.onDragLeave} onDrop={this.onDrop}>
144
- <input type="file" {...that.props} onChange={this.onChange}/>
145
- <input type="hidden" name={that.props.name} value="" />
146
- <Header {...that.props} />
147
- {previews}
148
- {placeholders}
149
- {discards}
150
- </label>
151
- );
152
- }
153
- });
@@ -1,23 +0,0 @@
1
- window.AttacheRails = {
2
- upgrade_fileinputs: function() {
3
- var safeWords = { 'class': 'className', 'for': 'htmlFor' };
4
- var sel = document.getElementsByClassName('enable-attache');
5
- var ele, attrs, name, value;
6
- for (var i = sel.length-1; i >= 0; i--) {
7
- ele = sel[i];
8
- attrs = {};
9
- for (var j = 0; j < ele.attributes.length; j++) {
10
- name = ele.attributes[j].name;
11
- value = ele.attributes[j].value;
12
- if (name === 'class') value = value.replace('enable-attache', 'attache-enabled');
13
- name = safeWords[name] || name;
14
- attrs[name] = value;
15
- }
16
- var wrap = document.createElement('div');
17
- ele.parentNode.replaceChild(wrap, ele); // ele.parentNode.insertBefore(wrap, ele.nextSibling);
18
- React.render(React.createElement(AttacheFileInput, React.__spread({}, attrs)), wrap);
19
- }
20
- }
21
- };
22
-
23
- $(document).on('page:change', AttacheRails.upgrade_fileinputs);