2017-11-23 15:00:05 +01:00
|
|
|
var upload = (function () {
|
|
|
|
|
|
|
|
var exports = {};
|
|
|
|
|
|
|
|
function make_upload_absolute(uri) {
|
|
|
|
if (uri.indexOf(compose.uploads_path) === 0) {
|
|
|
|
// Rewrite the URI to a usable link
|
|
|
|
return compose.uploads_domain + uri;
|
|
|
|
}
|
|
|
|
return uri;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function resets an input type="file". Pass in the
|
|
|
|
// jquery object.
|
|
|
|
function clear_out_file_list(jq_file_list) {
|
|
|
|
if (compose.clone_file_input !== undefined) {
|
|
|
|
jq_file_list.replaceWith(compose.clone_file_input.clone(true));
|
|
|
|
}
|
|
|
|
// Hack explanation:
|
|
|
|
// IE won't let you do this (untested, but so says StackOverflow):
|
|
|
|
// $("#file_input").val("");
|
|
|
|
}
|
|
|
|
|
2018-02-13 23:26:19 +01:00
|
|
|
// Show the upload button only if the browser supports it.
|
|
|
|
exports.feature_check = function (upload_button) {
|
|
|
|
if (window.XMLHttpRequest && (new XMLHttpRequest()).upload) {
|
|
|
|
upload_button.removeClass("notdisplayed");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-13 22:48:48 +01:00
|
|
|
exports.options = function (config) {
|
|
|
|
var textarea;
|
|
|
|
var send_button;
|
|
|
|
var send_status;
|
|
|
|
var send_status_close;
|
|
|
|
var error_msg;
|
|
|
|
var upload_bar;
|
|
|
|
var file_input;
|
|
|
|
|
|
|
|
switch (config.mode) {
|
|
|
|
case 'compose':
|
|
|
|
textarea = $('#compose-textarea');
|
|
|
|
send_button = $('#compose-send-button');
|
|
|
|
send_status = $('#compose-send-status');
|
|
|
|
send_status_close = $('.compose-send-status-close');
|
|
|
|
error_msg = $('#compose-error-msg');
|
|
|
|
upload_bar = 'compose-upload-bar';
|
|
|
|
file_input = 'file_input';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw Error("Invalid upload mode!");
|
|
|
|
}
|
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
var uploadStarted = function () {
|
2018-02-13 22:48:48 +01:00
|
|
|
send_button.attr("disabled", "");
|
|
|
|
send_status.addClass("alert-info").show();
|
|
|
|
send_status_close.one('click', compose.abort_xhr);
|
|
|
|
error_msg.html($("<p>").text(i18n.t("Uploading…"))
|
2018-02-13 22:39:51 +01:00
|
|
|
.after('<div class="progress progress-striped active">' +
|
2018-02-13 22:48:48 +01:00
|
|
|
'<div class="bar" id="' + upload_bar + '" style="width: 00%;"></div>' +
|
2018-02-13 22:39:51 +01:00
|
|
|
'</div>'));
|
|
|
|
};
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
var progressUpdated = function (i, file, progress) {
|
2018-02-13 22:48:48 +01:00
|
|
|
$("#" + upload_bar).width(progress + "%");
|
2018-02-13 22:39:51 +01:00
|
|
|
};
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
var uploadError = function (error_code, server_response, file) {
|
|
|
|
var msg;
|
2018-02-13 22:48:48 +01:00
|
|
|
send_status.addClass("alert-error")
|
2018-02-13 22:39:51 +01:00
|
|
|
.removeClass("alert-info");
|
2018-02-13 22:48:48 +01:00
|
|
|
send_button.prop("disabled", false);
|
2018-02-13 22:39:51 +01:00
|
|
|
switch (error_code) {
|
2017-11-23 15:00:05 +01:00
|
|
|
case 'BrowserNotSupported':
|
|
|
|
msg = i18n.t("File upload is not yet available for your browser.");
|
|
|
|
break;
|
|
|
|
case 'TooManyFiles':
|
|
|
|
msg = i18n.t("Unable to upload that many files at once.");
|
|
|
|
break;
|
|
|
|
case 'FileTooLarge':
|
|
|
|
// sanitization not needed as the file name is not potentially parsed as HTML, etc.
|
|
|
|
var context = {
|
|
|
|
file_name: file.name,
|
|
|
|
};
|
|
|
|
msg = i18n.t('"__file_name__" was too large; the maximum file size is 25MiB.', context);
|
|
|
|
break;
|
2018-01-27 03:50:37 +01:00
|
|
|
case 413: // HTTP status "Request Entity Too Large"
|
2017-11-23 15:00:05 +01:00
|
|
|
msg = i18n.t("Sorry, the file was too large.");
|
|
|
|
break;
|
2018-01-27 04:00:32 +01:00
|
|
|
case 400:
|
|
|
|
var server_message = server_response && server_response.msg;
|
|
|
|
msg = server_message || i18n.t("An unknown error occurred.");
|
|
|
|
break;
|
2017-11-23 15:00:05 +01:00
|
|
|
default:
|
|
|
|
msg = i18n.t("An unknown error occurred.");
|
|
|
|
break;
|
2018-02-13 22:39:51 +01:00
|
|
|
}
|
2018-02-13 22:48:48 +01:00
|
|
|
error_msg.text(msg);
|
2018-02-13 22:39:51 +01:00
|
|
|
};
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
var uploadFinished = function (i, file, response) {
|
|
|
|
if (response.uri === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var split_uri = response.uri.split("/");
|
|
|
|
var filename = split_uri[split_uri.length - 1];
|
|
|
|
// Urgh, yet another hack to make sure we're "composing"
|
|
|
|
// when text gets added into the composebox.
|
|
|
|
if (!compose_state.composing()) {
|
|
|
|
compose_actions.start('stream');
|
|
|
|
}
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
var uri = make_upload_absolute(response.uri);
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
if (i === -1) {
|
|
|
|
// This is a paste, so there's no filename. Show the image directly
|
2018-02-13 22:48:48 +01:00
|
|
|
textarea.val(textarea.val() + "[pasted image](" + uri + ") ");
|
2018-02-13 22:39:51 +01:00
|
|
|
} else {
|
|
|
|
// This is a dropped file, so make the filename a link to the image
|
2018-02-13 22:48:48 +01:00
|
|
|
textarea.val(textarea.val() + "[" + filename + "](" + uri + ")" + " ");
|
2018-02-13 22:39:51 +01:00
|
|
|
}
|
|
|
|
compose_ui.autosize_textarea();
|
2018-02-13 22:48:48 +01:00
|
|
|
send_button.prop("disabled", false);
|
|
|
|
send_status.removeClass("alert-info").hide();
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
// In order to upload the same file twice in a row, we need to clear out
|
|
|
|
// the #file_input element, so that the next time we use the file dialog,
|
|
|
|
// an actual change event is fired. This is extracted to a function
|
|
|
|
// to abstract away some IE hacks.
|
2018-02-13 22:48:48 +01:00
|
|
|
clear_out_file_list($("#" + file_input));
|
2018-02-13 22:39:51 +01:00
|
|
|
};
|
2017-11-23 15:00:05 +01:00
|
|
|
|
2018-02-13 22:39:51 +01:00
|
|
|
return {
|
2017-11-23 15:00:05 +01:00
|
|
|
url: "/json/user_uploads",
|
2018-02-13 22:48:48 +01:00
|
|
|
fallback_id: file_input, // Target for standard file dialog
|
2017-11-23 15:00:05 +01:00
|
|
|
paramname: "file",
|
|
|
|
maxfilesize: page_params.maxfilesize,
|
|
|
|
data: {
|
|
|
|
// the token isn't automatically included in filedrop's post
|
|
|
|
csrfmiddlewaretoken: csrf_token,
|
|
|
|
},
|
|
|
|
raw_droppable: ['text/uri-list', 'text/plain'],
|
2018-02-13 22:39:51 +01:00
|
|
|
drop: uploadStarted,
|
|
|
|
progressUpdated: progressUpdated,
|
|
|
|
error: uploadError,
|
|
|
|
uploadFinished: uploadFinished,
|
2017-11-23 15:00:05 +01:00
|
|
|
rawDrop: function (contents) {
|
|
|
|
if (!compose_state.composing()) {
|
|
|
|
compose_actions.start('stream');
|
|
|
|
}
|
2018-02-13 22:48:48 +01:00
|
|
|
textarea.val(textarea.val() + contents);
|
2017-11-23 15:00:05 +01:00
|
|
|
compose_ui.autosize_textarea();
|
|
|
|
},
|
2018-02-13 22:39:51 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2017-11-23 15:00:05 +01:00
|
|
|
return exports;
|
|
|
|
}());
|
|
|
|
|
|
|
|
if (typeof module !== 'undefined') {
|
|
|
|
module.exports = upload;
|
|
|
|
}
|