2015-09-07 21:53:43 +00:00
|
|
|
/*
|
|
|
|
* vim: ts=4:sw=4:expandtab
|
2015-01-16 23:06:38 +00:00
|
|
|
*/
|
2014-10-25 01:44:30 +00:00
|
|
|
(function () {
|
|
|
|
'use strict';
|
2015-03-05 23:44:53 +00:00
|
|
|
window.Whisper = window.Whisper || {};
|
2015-01-19 01:43:25 +00:00
|
|
|
|
2015-03-03 00:31:04 +00:00
|
|
|
Whisper.FileSizeToast = Whisper.ToastView.extend({
|
2015-12-26 04:38:16 +00:00
|
|
|
templateName: 'file-size-modal',
|
|
|
|
render_attributes: function() {
|
|
|
|
return {
|
|
|
|
'file-size-warning': i18n('fileSizeWarning'),
|
|
|
|
limit: this.model.limit,
|
|
|
|
units: this.model.units
|
|
|
|
};
|
|
|
|
}
|
2015-03-03 00:31:04 +00:00
|
|
|
});
|
2015-12-21 06:12:57 +00:00
|
|
|
Whisper.UnsupportedFileTypeToast = Whisper.ToastView.extend({
|
2015-12-26 04:38:40 +00:00
|
|
|
template: i18n('unsupportedFileType')
|
2015-12-21 06:12:57 +00:00
|
|
|
});
|
2015-03-03 00:31:04 +00:00
|
|
|
|
2014-10-25 01:44:30 +00:00
|
|
|
Whisper.FileInputView = Backbone.View.extend({
|
|
|
|
tagName: 'span',
|
|
|
|
className: 'file-input',
|
2015-05-22 00:58:22 +00:00
|
|
|
initialize: function(options) {
|
2015-03-28 01:47:58 +00:00
|
|
|
this.$input = this.$('input[type=file]');
|
2017-03-11 00:36:13 +00:00
|
|
|
this.$input.click(function(e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
2015-01-16 23:39:01 +00:00
|
|
|
this.thumb = new Whisper.AttachmentPreviewView();
|
2015-02-06 06:42:16 +00:00
|
|
|
this.$el.addClass('file-input');
|
2015-05-22 00:58:22 +00:00
|
|
|
this.window = options.window;
|
2014-10-25 01:44:30 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
events: {
|
2016-04-15 21:57:42 +00:00
|
|
|
'change .choose-file': 'previewImages',
|
2015-01-24 20:00:56 +00:00
|
|
|
'click .close': 'deleteFiles',
|
2015-12-03 22:27:29 +00:00
|
|
|
'click .choose-file': 'open',
|
|
|
|
'drop': 'openDropped',
|
|
|
|
'dragover': 'showArea',
|
2017-02-05 10:21:35 +00:00
|
|
|
'dragleave': 'hideArea',
|
|
|
|
'paste': 'onPaste'
|
2015-01-24 20:00:56 +00:00
|
|
|
},
|
|
|
|
|
2016-04-15 21:39:02 +00:00
|
|
|
open: function(e) {
|
|
|
|
e.preventDefault();
|
2015-05-22 00:58:22 +00:00
|
|
|
// hack
|
|
|
|
if (this.window && this.window.chrome && this.window.chrome.fileSystem) {
|
|
|
|
this.window.chrome.fileSystem.chooseEntry({type: 'openFile'}, function(entry) {
|
2015-05-25 17:43:35 +00:00
|
|
|
if (!entry) {
|
|
|
|
return;
|
|
|
|
}
|
2015-05-22 00:58:22 +00:00
|
|
|
entry.file(function(file) {
|
|
|
|
this.file = file;
|
|
|
|
this.previewImages();
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
} else {
|
|
|
|
this.$input.click();
|
|
|
|
}
|
2014-10-25 01:44:30 +00:00
|
|
|
},
|
|
|
|
|
2015-03-03 02:27:14 +00:00
|
|
|
addThumb: function(src) {
|
2015-09-16 19:29:15 +00:00
|
|
|
this.$('.avatar').hide();
|
2015-03-03 02:27:14 +00:00
|
|
|
this.thumb.src = src;
|
2015-10-29 18:04:21 +00:00
|
|
|
this.$('.attachment-previews').append(this.thumb.render().el);
|
2015-10-30 01:19:51 +00:00
|
|
|
this.thumb.$('img')[0].onload = function() {
|
|
|
|
this.$el.trigger('force-resize');
|
|
|
|
}.bind(this);
|
2014-10-25 01:44:30 +00:00
|
|
|
},
|
|
|
|
|
2015-03-04 01:50:24 +00:00
|
|
|
autoScale: function(file) {
|
2015-03-12 03:20:54 +00:00
|
|
|
if (file.type.split('/')[0] !== 'image' || file.type === 'image/gif') {
|
2015-03-04 01:50:24 +00:00
|
|
|
// nothing to do
|
|
|
|
return Promise.resolve(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise(function(resolve, reject) {
|
2015-03-05 02:25:16 +00:00
|
|
|
var url = URL.createObjectURL(file);
|
|
|
|
var img = document.createElement('img');
|
|
|
|
img.onerror = reject;
|
|
|
|
img.onload = function () {
|
|
|
|
URL.revokeObjectURL(url);
|
2015-03-04 01:50:24 +00:00
|
|
|
|
2017-02-08 17:31:12 +00:00
|
|
|
var maxSize = 6000 * 1024;
|
|
|
|
var maxHeight = 4096;
|
|
|
|
var maxWidth = 4096;
|
2015-03-05 02:25:16 +00:00
|
|
|
if (img.width <= maxWidth && img.height <= maxHeight &&
|
|
|
|
file.size <= maxSize) {
|
|
|
|
resolve(file);
|
|
|
|
return;
|
|
|
|
}
|
2015-03-04 01:50:24 +00:00
|
|
|
|
2015-03-05 02:25:16 +00:00
|
|
|
// loadImage.scale -> components/blueimp-load-image
|
|
|
|
var canvas = loadImage.scale(img, {
|
2015-06-04 18:55:23 +00:00
|
|
|
canvas: true, maxWidth: maxWidth, maxHeight: maxHeight
|
2015-03-05 02:25:16 +00:00
|
|
|
});
|
2015-03-04 01:50:24 +00:00
|
|
|
|
2015-03-05 02:25:16 +00:00
|
|
|
var quality = 0.95;
|
|
|
|
var i = 4;
|
|
|
|
var blob;
|
|
|
|
do {
|
|
|
|
i = i - 1;
|
|
|
|
// dataURLtoBlob -> components/blueimp-canvas-to-blob
|
|
|
|
blob = dataURLtoBlob(
|
|
|
|
canvas.toDataURL('image/jpeg', quality)
|
|
|
|
);
|
|
|
|
quality = quality * maxSize / blob.size;
|
2016-04-09 11:57:52 +00:00
|
|
|
if (quality < 0.5) {
|
|
|
|
quality = 0.5;
|
2015-03-04 01:50:24 +00:00
|
|
|
}
|
2015-03-05 02:25:16 +00:00
|
|
|
} while (i > 0 && blob.size > maxSize);
|
|
|
|
|
|
|
|
resolve(blob);
|
|
|
|
};
|
|
|
|
img.src = url;
|
2015-03-04 01:50:24 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2014-10-25 01:44:30 +00:00
|
|
|
previewImages: function() {
|
2015-01-16 02:17:54 +00:00
|
|
|
this.clearForm();
|
2015-05-22 00:58:22 +00:00
|
|
|
var file = this.file || this.$input.prop('files')[0];
|
2015-03-04 01:50:24 +00:00
|
|
|
if (!file) { return; }
|
|
|
|
|
|
|
|
var type = file.type.split('/')[0];
|
|
|
|
switch (type) {
|
2017-03-08 00:54:46 +00:00
|
|
|
case 'audio': this.addThumb('images/audio.svg'); break;
|
|
|
|
case 'video': this.addThumb('images/video.svg'); break;
|
2015-03-04 01:50:24 +00:00
|
|
|
case 'image':
|
|
|
|
this.oUrl = URL.createObjectURL(file);
|
|
|
|
this.addThumb(this.oUrl);
|
|
|
|
break;
|
2015-12-21 06:12:57 +00:00
|
|
|
default:
|
2017-04-18 23:15:27 +00:00
|
|
|
this.addThumb('/images/file.svg'); break;
|
2015-03-04 01:50:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.autoScale(file).then(function(blob) {
|
2015-03-03 00:31:04 +00:00
|
|
|
var limitKb = 1000000;
|
2015-10-19 20:26:11 +00:00
|
|
|
var blobType = file.type === 'image/gif' ? 'gif' : type;
|
|
|
|
switch (blobType) {
|
|
|
|
case 'image':
|
2017-02-08 17:31:12 +00:00
|
|
|
limitKb = 6000; break;
|
2015-10-19 20:26:11 +00:00
|
|
|
case 'gif':
|
2017-04-16 18:52:41 +00:00
|
|
|
limitKb = 25000; break;
|
2015-10-19 20:26:11 +00:00
|
|
|
case 'audio':
|
|
|
|
limitKb = 100000; break;
|
|
|
|
case 'video':
|
|
|
|
limitKb = 100000; break;
|
2015-03-03 00:31:04 +00:00
|
|
|
}
|
2015-03-04 01:50:24 +00:00
|
|
|
if ((blob.size/1024).toFixed(4) >= limitKb) {
|
2015-10-19 20:26:11 +00:00
|
|
|
var units = ['kB','MB','GB'];
|
|
|
|
var u = -1;
|
|
|
|
var limit = limitKb * 1000;
|
|
|
|
do {
|
|
|
|
limit /= 1000;
|
|
|
|
++u;
|
|
|
|
} while (limit >= 1000 && u < units.length - 1);
|
2015-05-25 17:43:35 +00:00
|
|
|
var toast = new Whisper.FileSizeToast({
|
2015-10-19 20:26:11 +00:00
|
|
|
model: {limit: limit, units: units[u]}
|
2015-05-25 17:43:35 +00:00
|
|
|
});
|
|
|
|
toast.$el.insertAfter(this.$el);
|
|
|
|
toast.render();
|
2015-01-16 02:17:54 +00:00
|
|
|
this.deleteFiles();
|
|
|
|
}
|
2015-03-04 01:50:24 +00:00
|
|
|
}.bind(this));
|
2014-10-25 01:44:30 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
hasFiles: function() {
|
2015-05-22 02:01:37 +00:00
|
|
|
var files = this.file ? [this.file] : this.$input.prop('files');
|
2014-10-25 01:44:30 +00:00
|
|
|
return files && files.length && files.length > 0;
|
|
|
|
},
|
|
|
|
|
|
|
|
getFiles: function() {
|
|
|
|
var promises = [];
|
2015-05-22 02:01:37 +00:00
|
|
|
var files = this.file ? [this.file] : this.$input.prop('files');
|
2014-10-25 01:44:30 +00:00
|
|
|
for (var i = 0; i < files.length; i++) {
|
2015-03-04 01:50:24 +00:00
|
|
|
promises.push(this.getFile(files[i]));
|
2014-10-25 01:44:30 +00:00
|
|
|
}
|
2015-01-16 02:17:54 +00:00
|
|
|
this.clearForm();
|
2014-10-25 01:44:30 +00:00
|
|
|
return Promise.all(promises);
|
2015-01-16 02:17:54 +00:00
|
|
|
},
|
2014-10-25 01:44:30 +00:00
|
|
|
|
2015-03-04 01:50:24 +00:00
|
|
|
getFile: function(file) {
|
2015-05-22 00:58:22 +00:00
|
|
|
file = file || this.file || this.$input.prop('files')[0];
|
2015-03-24 02:24:26 +00:00
|
|
|
if (file === undefined) { return Promise.resolve(); }
|
2017-05-10 22:32:05 +00:00
|
|
|
var flags;
|
|
|
|
if (this.isVoiceNote) {
|
|
|
|
flags = textsecure.protobuf.AttachmentPointer.Flags.VOICE_MESSAGE;
|
|
|
|
}
|
|
|
|
return this.autoScale(file).then(this.readFile).then(function(attachment) {
|
|
|
|
if (flags) {
|
|
|
|
attachment.flags = flags;
|
|
|
|
}
|
|
|
|
return attachment;
|
|
|
|
}.bind(this));
|
2015-03-04 01:50:24 +00:00
|
|
|
},
|
|
|
|
|
2015-06-04 18:55:23 +00:00
|
|
|
getThumbnail: function() {
|
|
|
|
// Scale and crop an image to 256px square
|
|
|
|
var size = 256;
|
|
|
|
var file = this.file || this.$input.prop('files')[0];
|
2015-06-22 21:45:42 +00:00
|
|
|
if (file === undefined || file.type.split('/')[0] !== 'image' || file.type === 'image/gif') {
|
2015-06-04 18:55:23 +00:00
|
|
|
// nothing to do
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
var url = URL.createObjectURL(file);
|
|
|
|
var img = document.createElement('img');
|
|
|
|
img.onerror = reject;
|
|
|
|
img.onload = function () {
|
|
|
|
URL.revokeObjectURL(url);
|
|
|
|
// loadImage.scale -> components/blueimp-load-image
|
|
|
|
// scale, then crop.
|
|
|
|
var canvas = loadImage.scale(img, {
|
|
|
|
canvas: true, maxWidth: size, maxHeight: size,
|
|
|
|
cover: true, minWidth: size, minHeight: size
|
|
|
|
});
|
|
|
|
canvas = loadImage.scale(canvas, {
|
|
|
|
canvas: true, maxWidth: size, maxHeight: size,
|
|
|
|
crop: true, minWidth: size, minHeight: size
|
|
|
|
});
|
|
|
|
|
|
|
|
// dataURLtoBlob -> components/blueimp-canvas-to-blob
|
|
|
|
var blob = dataURLtoBlob(canvas.toDataURL('image/png'));
|
|
|
|
|
|
|
|
resolve(blob);
|
|
|
|
};
|
|
|
|
img.src = url;
|
|
|
|
}).then(this.readFile);
|
|
|
|
},
|
|
|
|
|
2015-02-19 08:20:22 +00:00
|
|
|
readFile: function(file) {
|
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
var FR = new FileReader();
|
|
|
|
FR.onload = function(e) {
|
2017-04-18 18:26:05 +00:00
|
|
|
resolve({
|
|
|
|
data: e.target.result,
|
|
|
|
contentType: file.type,
|
2017-04-18 18:44:03 +00:00
|
|
|
fileName: file.name,
|
|
|
|
size: file.size
|
2017-04-18 18:26:05 +00:00
|
|
|
});
|
2015-02-19 08:20:22 +00:00
|
|
|
};
|
2017-04-18 18:26:05 +00:00
|
|
|
FR.onerror = reject;
|
|
|
|
FR.onabort = reject;
|
2015-02-19 08:20:22 +00:00
|
|
|
FR.readAsArrayBuffer(file);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2015-01-16 02:17:54 +00:00
|
|
|
clearForm: function() {
|
2015-03-04 01:50:24 +00:00
|
|
|
if (this.oUrl) {
|
|
|
|
URL.revokeObjectURL(this.oUrl);
|
|
|
|
this.oUrl = null;
|
|
|
|
}
|
2015-01-16 23:39:01 +00:00
|
|
|
this.thumb.remove();
|
2015-09-16 19:29:15 +00:00
|
|
|
this.$('.avatar').show();
|
2015-10-30 01:19:51 +00:00
|
|
|
this.$el.trigger('force-resize');
|
2015-01-16 02:17:54 +00:00
|
|
|
},
|
|
|
|
|
2015-01-25 09:18:10 +00:00
|
|
|
deleteFiles: function(e) {
|
|
|
|
if (e) { e.stopPropagation(); }
|
2015-01-16 02:17:54 +00:00
|
|
|
this.clearForm();
|
|
|
|
this.$input.wrap('<form>').parent('form').trigger('reset');
|
|
|
|
this.$input.unwrap();
|
2015-05-24 23:02:22 +00:00
|
|
|
this.file = null;
|
2016-08-15 22:36:29 +00:00
|
|
|
this.$input.trigger('change');
|
2017-05-10 22:32:05 +00:00
|
|
|
this.isVoiceNote = false;
|
2015-12-03 22:27:29 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
openDropped: function(e) {
|
2016-02-18 01:08:50 +00:00
|
|
|
if (e.originalEvent.dataTransfer.types[0] != 'Files') {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-09 00:31:52 +00:00
|
|
|
|
2015-12-03 22:27:29 +00:00
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
this.file = e.originalEvent.dataTransfer.files[0];
|
|
|
|
this.previewImages();
|
2015-12-09 18:34:23 +00:00
|
|
|
this.$el.removeClass('dropoff');
|
2015-12-03 22:27:29 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
showArea: function(e) {
|
2016-02-18 01:08:50 +00:00
|
|
|
if (e.originalEvent.dataTransfer.types[0] != 'Files') {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-09 00:31:52 +00:00
|
|
|
|
2015-12-03 22:27:29 +00:00
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
2015-12-09 18:34:23 +00:00
|
|
|
this.$el.addClass('dropoff');
|
2015-12-03 22:27:29 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
hideArea: function(e) {
|
2016-02-18 01:08:50 +00:00
|
|
|
if (e.originalEvent.dataTransfer.types[0] != 'Files') {
|
|
|
|
return;
|
|
|
|
}
|
2015-12-09 00:31:52 +00:00
|
|
|
|
2015-12-03 22:27:29 +00:00
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
2015-12-09 18:34:23 +00:00
|
|
|
this.$el.removeClass('dropoff');
|
2017-02-05 10:21:35 +00:00
|
|
|
},
|
|
|
|
onPaste: function(e) {
|
|
|
|
var items = e.originalEvent.clipboardData.items;
|
|
|
|
var imgBlob = null;
|
|
|
|
for (var i = 0; i < items.length; i++) {
|
|
|
|
if (items[i].type.split('/')[0] === 'image') {
|
|
|
|
imgBlob = items[i].getAsFile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (imgBlob !== null) {
|
|
|
|
this.file = imgBlob;
|
|
|
|
this.previewImages();
|
|
|
|
}
|
2015-01-16 02:17:54 +00:00
|
|
|
}
|
2014-10-25 01:44:30 +00:00
|
|
|
});
|
|
|
|
})();
|