REFACTOR: Move upload utilities to their own file
This commit is contained in:
@@ -31,14 +31,17 @@ import { findRawTemplate } from "discourse/lib/raw-templates";
|
||||
import { iconHTML } from "discourse-common/lib/icon-library";
|
||||
import {
|
||||
tinyAvatar,
|
||||
displayErrorForUpload,
|
||||
getUploadMarkdown,
|
||||
validateUploadedFiles,
|
||||
authorizesOneOrMoreImageExtensions,
|
||||
formatUsername,
|
||||
clipboardData,
|
||||
safariHacksDisabled
|
||||
} from "discourse/lib/utilities";
|
||||
import {
|
||||
validateUploadedFiles,
|
||||
authorizesOneOrMoreImageExtensions,
|
||||
getUploadMarkdown,
|
||||
displayErrorForUpload
|
||||
} from "discourse/lib/uploads";
|
||||
|
||||
import {
|
||||
cacheShortUploadUrl,
|
||||
resolveAllShortUrls
|
||||
@@ -82,7 +85,7 @@ export default Component.extend({
|
||||
if (requiredCategoryMissing) {
|
||||
return "composer.reply_placeholder_choose_category";
|
||||
} else {
|
||||
const key = authorizesOneOrMoreImageExtensions()
|
||||
const key = authorizesOneOrMoreImageExtensions(this.currentUser.staff)
|
||||
? "reply_placeholder"
|
||||
: "reply_placeholder_no_images";
|
||||
return `composer.${key}`;
|
||||
@@ -700,6 +703,7 @@ export default Component.extend({
|
||||
if (this._pasted) data.formData.pasted = true;
|
||||
|
||||
const opts = {
|
||||
user: this.currentUser,
|
||||
isPrivateMessage,
|
||||
allowStaffToUploadAnyFileInPm: this.siteSettings
|
||||
.allow_staff_to_upload_any_file_in_pm
|
||||
|
||||
@@ -2,7 +2,7 @@ import discourseComputed from "discourse-common/utils/decorators";
|
||||
import Controller from "@ember/controller";
|
||||
import ModalFunctionality from "discourse/mixins/modal-functionality";
|
||||
import { ajax } from "discourse/lib/ajax";
|
||||
import { allowsImages } from "discourse/lib/utilities";
|
||||
import { allowsImages } from "discourse/lib/uploads";
|
||||
import { popupAjaxError } from "discourse/lib/ajax-error";
|
||||
|
||||
export default Controller.extend(ModalFunctionality, {
|
||||
@@ -42,7 +42,10 @@ export default Controller.extend(ModalFunctionality, {
|
||||
|
||||
@discourseComputed()
|
||||
allowAvatarUpload() {
|
||||
return this.siteSettings.allow_uploaded_avatars && allowsImages();
|
||||
return (
|
||||
this.siteSettings.allow_uploaded_avatars &&
|
||||
allowsImages(this.currentUser.staff)
|
||||
);
|
||||
},
|
||||
|
||||
actions: {
|
||||
|
||||
@@ -14,12 +14,11 @@ import {
|
||||
on
|
||||
} from "discourse-common/utils/decorators";
|
||||
import { getOwner } from "discourse-common/lib/get-owner";
|
||||
import { escapeExpression, safariHacksDisabled } from "discourse/lib/utilities";
|
||||
import {
|
||||
escapeExpression,
|
||||
uploadIcon,
|
||||
authorizesOneOrMoreExtensions,
|
||||
safariHacksDisabled
|
||||
} from "discourse/lib/utilities";
|
||||
uploadIcon
|
||||
} from "discourse/lib/uploads";
|
||||
import { emojiUnescape } from "discourse/lib/text";
|
||||
import { shortDate } from "discourse/lib/formatter";
|
||||
import { SAVE_LABELS, SAVE_ICONS } from "discourse/models/composer";
|
||||
@@ -322,11 +321,13 @@ export default Controller.extend({
|
||||
|
||||
@discourseComputed
|
||||
allowUpload() {
|
||||
return authorizesOneOrMoreExtensions();
|
||||
return authorizesOneOrMoreExtensions(this.currentUser.staff);
|
||||
},
|
||||
|
||||
@discourseComputed()
|
||||
uploadIcon: () => uploadIcon(),
|
||||
uploadIcon() {
|
||||
return uploadIcon(this.currentUser.staff);
|
||||
},
|
||||
|
||||
actions: {
|
||||
togglePreview() {
|
||||
|
||||
@@ -7,13 +7,13 @@ import {
|
||||
} from "discourse-common/utils/decorators";
|
||||
import {
|
||||
allowsAttachments,
|
||||
authorizesAllExtensions,
|
||||
authorizedExtensions,
|
||||
authorizesAllExtensions,
|
||||
uploadIcon
|
||||
} from "discourse/lib/utilities";
|
||||
} from "discourse/lib/uploads";
|
||||
|
||||
function uploadTranslate(key) {
|
||||
if (allowsAttachments()) {
|
||||
function uploadTranslate(key, user) {
|
||||
if (allowsAttachments(user.staff)) {
|
||||
key += "_with_attachments";
|
||||
}
|
||||
return `upload_selector.${key}`;
|
||||
@@ -28,17 +28,23 @@ export default Controller.extend(ModalFunctionality, {
|
||||
selection: "local",
|
||||
|
||||
@discourseComputed()
|
||||
uploadIcon: () => uploadIcon(),
|
||||
uploadIcon() {
|
||||
return uploadIcon(this.currentUser.staff);
|
||||
},
|
||||
|
||||
@discourseComputed()
|
||||
title: () => uploadTranslate("title"),
|
||||
title() {
|
||||
return uploadTranslate("title", this.currentUser);
|
||||
},
|
||||
|
||||
@discourseComputed("selection")
|
||||
tip(selection) {
|
||||
const authorized_extensions = authorizesAllExtensions()
|
||||
const authorized_extensions = authorizesAllExtensions(
|
||||
this.currentUser.staff
|
||||
)
|
||||
? ""
|
||||
: `(${authorizedExtensions()})`;
|
||||
return I18n.t(uploadTranslate(`${selection}_tip`), {
|
||||
: `(${authorizedExtensions(this.currentUser.staff)})`;
|
||||
return I18n.t(uploadTranslate(`${selection}_tip`, this.currentUser), {
|
||||
authorized_extensions
|
||||
});
|
||||
},
|
||||
|
||||
@@ -0,0 +1,285 @@
|
||||
import { isAppleDevice } from "discourse/lib/utilities";
|
||||
|
||||
function isGUID(value) {
|
||||
return /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(
|
||||
value
|
||||
);
|
||||
}
|
||||
|
||||
function imageNameFromFileName(fileName) {
|
||||
const split = fileName.split(".");
|
||||
let name = split[split.length - 2];
|
||||
|
||||
if (isAppleDevice() && isGUID(name)) {
|
||||
name = I18n.t("upload_selector.default_image_alt_text");
|
||||
}
|
||||
|
||||
return encodeURIComponent(name);
|
||||
}
|
||||
|
||||
export function validateUploadedFiles(files, opts) {
|
||||
if (!files || files.length === 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (files.length > 1) {
|
||||
bootbox.alert(I18n.t("post.errors.too_many_uploads"));
|
||||
return false;
|
||||
}
|
||||
|
||||
const upload = files[0];
|
||||
|
||||
// CHROME ONLY: if the image was pasted, sets its name to a default one
|
||||
if (typeof Blob !== "undefined" && typeof File !== "undefined") {
|
||||
if (
|
||||
upload instanceof Blob &&
|
||||
!(upload instanceof File) &&
|
||||
upload.type === "image/png"
|
||||
) {
|
||||
upload.name = "image.png";
|
||||
}
|
||||
}
|
||||
|
||||
opts = opts || {};
|
||||
opts.type = uploadTypeFromFileName(upload.name);
|
||||
|
||||
return validateUploadedFile(upload, opts);
|
||||
}
|
||||
|
||||
function validateUploadedFile(file, opts) {
|
||||
if (opts.skipValidation) return true;
|
||||
|
||||
opts = opts || {};
|
||||
let user = opts.user;
|
||||
let staff = user && user.staff;
|
||||
|
||||
if (!authorizesOneOrMoreExtensions(staff)) return false;
|
||||
|
||||
const name = file && file.name;
|
||||
|
||||
if (!name) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// check that the uploaded file is authorized
|
||||
if (opts.allowStaffToUploadAnyFileInPm && opts.isPrivateMessage) {
|
||||
if (staff) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
if (opts.imagesOnly) {
|
||||
if (!isAnImage(name) && !isAuthorizedImage(name, staff)) {
|
||||
bootbox.alert(
|
||||
I18n.t("post.errors.upload_not_authorized", {
|
||||
authorized_extensions: authorizedImagesExtensions(staff)
|
||||
})
|
||||
);
|
||||
return false;
|
||||
}
|
||||
} else if (opts.csvOnly) {
|
||||
if (!/\.csv$/i.test(name)) {
|
||||
bootbox.alert(I18n.t("user.invited.bulk_invite.error"));
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
if (!authorizesAllExtensions(staff) && !isAuthorizedFile(name, staff)) {
|
||||
bootbox.alert(
|
||||
I18n.t("post.errors.upload_not_authorized", {
|
||||
authorized_extensions: authorizedExtensions(staff)
|
||||
})
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
if (!opts.bypassNewUserRestriction) {
|
||||
// ensures that new users can upload a file
|
||||
if (user && !user.isAllowedToUploadAFile(opts.type)) {
|
||||
bootbox.alert(
|
||||
I18n.t(`post.errors.${opts.type}_upload_not_allowed_for_new_user`)
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// everything went fine
|
||||
return true;
|
||||
}
|
||||
|
||||
const IMAGES_EXTENSIONS_REGEX = /(png|jpe?g|gif|svg|ico)/i;
|
||||
|
||||
function extensionsToArray(exts) {
|
||||
return exts
|
||||
.toLowerCase()
|
||||
.replace(/[\s\.]+/g, "")
|
||||
.split("|")
|
||||
.filter(ext => ext.indexOf("*") === -1);
|
||||
}
|
||||
|
||||
function extensions() {
|
||||
return extensionsToArray(Discourse.SiteSettings.authorized_extensions);
|
||||
}
|
||||
|
||||
function staffExtensions() {
|
||||
return extensionsToArray(
|
||||
Discourse.SiteSettings.authorized_extensions_for_staff
|
||||
);
|
||||
}
|
||||
|
||||
function imagesExtensions(staff) {
|
||||
let exts = extensions().filter(ext => IMAGES_EXTENSIONS_REGEX.test(ext));
|
||||
if (staff) {
|
||||
const staffExts = staffExtensions().filter(ext =>
|
||||
IMAGES_EXTENSIONS_REGEX.test(ext)
|
||||
);
|
||||
exts = _.union(exts, staffExts);
|
||||
}
|
||||
return exts;
|
||||
}
|
||||
|
||||
function extensionsRegex() {
|
||||
return new RegExp("\\.(" + extensions().join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function imagesExtensionsRegex(staff) {
|
||||
return new RegExp("\\.(" + imagesExtensions(staff).join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function staffExtensionsRegex() {
|
||||
return new RegExp("\\.(" + staffExtensions().join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function isAuthorizedFile(fileName, staff) {
|
||||
if (staff && staffExtensionsRegex().test(fileName)) {
|
||||
return true;
|
||||
}
|
||||
return extensionsRegex().test(fileName);
|
||||
}
|
||||
|
||||
function isAuthorizedImage(fileName, staff) {
|
||||
return imagesExtensionsRegex(staff).test(fileName);
|
||||
}
|
||||
|
||||
export function authorizedExtensions(staff) {
|
||||
const exts = staff ? [...extensions(), ...staffExtensions()] : extensions();
|
||||
return exts.filter(ext => ext.length > 0).join(", ");
|
||||
}
|
||||
|
||||
function authorizedImagesExtensions(staff) {
|
||||
return authorizesAllExtensions(staff)
|
||||
? "png, jpg, jpeg, gif, svg, ico"
|
||||
: imagesExtensions(staff).join(", ");
|
||||
}
|
||||
|
||||
export function authorizesAllExtensions(staff) {
|
||||
return (
|
||||
Discourse.SiteSettings.authorized_extensions.indexOf("*") >= 0 ||
|
||||
(Discourse.SiteSettings.authorized_extensions_for_staff.indexOf("*") >= 0 &&
|
||||
staff)
|
||||
);
|
||||
}
|
||||
|
||||
export function authorizesOneOrMoreExtensions(staff) {
|
||||
if (authorizesAllExtensions(staff)) return true;
|
||||
|
||||
return (
|
||||
Discourse.SiteSettings.authorized_extensions.split("|").filter(ext => ext)
|
||||
.length > 0
|
||||
);
|
||||
}
|
||||
|
||||
export function authorizesOneOrMoreImageExtensions(staff) {
|
||||
if (authorizesAllExtensions(staff)) return true;
|
||||
return imagesExtensions(staff).length > 0;
|
||||
}
|
||||
|
||||
export function isAnImage(path) {
|
||||
return /\.(png|jpe?g|gif|svg|ico)$/i.test(path);
|
||||
}
|
||||
|
||||
function uploadTypeFromFileName(fileName) {
|
||||
return isAnImage(fileName) ? "image" : "attachment";
|
||||
}
|
||||
|
||||
export function allowsImages(staff) {
|
||||
return (
|
||||
authorizesAllExtensions(staff) ||
|
||||
IMAGES_EXTENSIONS_REGEX.test(authorizedExtensions(staff))
|
||||
);
|
||||
}
|
||||
|
||||
export function allowsAttachments(staff) {
|
||||
return (
|
||||
authorizesAllExtensions(staff) ||
|
||||
authorizedExtensions(staff).split(", ").length >
|
||||
imagesExtensions(staff).length
|
||||
);
|
||||
}
|
||||
|
||||
export function uploadIcon(staff) {
|
||||
return allowsAttachments(staff) ? "upload" : "far-image";
|
||||
}
|
||||
|
||||
function uploadLocation(url) {
|
||||
if (Discourse.CDN) {
|
||||
url = Discourse.getURLWithCDN(url);
|
||||
return /^\/\//.test(url) ? "http:" + url : url;
|
||||
} else if (Discourse.S3BaseUrl) {
|
||||
return "https:" + url;
|
||||
} else {
|
||||
var protocol = window.location.protocol + "//",
|
||||
hostname = window.location.hostname,
|
||||
port = window.location.port ? ":" + window.location.port : "";
|
||||
return protocol + hostname + port + url;
|
||||
}
|
||||
}
|
||||
|
||||
export function getUploadMarkdown(upload) {
|
||||
if (isAnImage(upload.original_filename)) {
|
||||
const name = imageNameFromFileName(upload.original_filename);
|
||||
return ``;
|
||||
} else if (
|
||||
!Discourse.SiteSettings.prevent_anons_from_downloading_files &&
|
||||
/\.(mov|mp4|webm|ogv|mp3|ogg|wav|m4a)$/i.test(upload.original_filename)
|
||||
) {
|
||||
return uploadLocation(upload.url);
|
||||
} else {
|
||||
return `[${upload.original_filename}|attachment](${
|
||||
upload.short_url
|
||||
}) (${I18n.toHumanSize(upload.filesize)})`;
|
||||
}
|
||||
}
|
||||
|
||||
export function displayErrorForUpload(data) {
|
||||
if (data.jqXHR) {
|
||||
switch (data.jqXHR.status) {
|
||||
// cancelled by the user
|
||||
case 0:
|
||||
return;
|
||||
|
||||
// entity too large, usually returned from the web server
|
||||
case 413:
|
||||
const type = uploadTypeFromFileName(data.files[0].name);
|
||||
const max_size_kb = Discourse.SiteSettings[`max_${type}_size_kb`];
|
||||
bootbox.alert(I18n.t("post.errors.file_too_large", { max_size_kb }));
|
||||
return;
|
||||
|
||||
// the error message is provided by the server
|
||||
case 422:
|
||||
if (data.jqXHR.responseJSON.message) {
|
||||
bootbox.alert(data.jqXHR.responseJSON.message);
|
||||
} else {
|
||||
bootbox.alert(data.jqXHR.responseJSON.errors.join("\n"));
|
||||
}
|
||||
return;
|
||||
}
|
||||
} else if (data.errors && data.errors.length > 0) {
|
||||
bootbox.alert(data.errors.join("\n"));
|
||||
return;
|
||||
}
|
||||
// otherwise, display a generic error message
|
||||
bootbox.alert(I18n.t("post.errors.upload"));
|
||||
}
|
||||
@@ -195,292 +195,6 @@ export function setCaretPosition(ctrl, pos) {
|
||||
}
|
||||
}
|
||||
|
||||
export function validateUploadedFiles(files, opts) {
|
||||
if (!files || files.length === 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (files.length > 1) {
|
||||
bootbox.alert(I18n.t("post.errors.too_many_uploads"));
|
||||
return false;
|
||||
}
|
||||
|
||||
const upload = files[0];
|
||||
|
||||
// CHROME ONLY: if the image was pasted, sets its name to a default one
|
||||
if (typeof Blob !== "undefined" && typeof File !== "undefined") {
|
||||
if (
|
||||
upload instanceof Blob &&
|
||||
!(upload instanceof File) &&
|
||||
upload.type === "image/png"
|
||||
) {
|
||||
upload.name = "image.png";
|
||||
}
|
||||
}
|
||||
|
||||
opts = opts || {};
|
||||
opts.type = uploadTypeFromFileName(upload.name);
|
||||
|
||||
return validateUploadedFile(upload, opts);
|
||||
}
|
||||
|
||||
export function validateUploadedFile(file, opts) {
|
||||
if (opts.skipValidation) return true;
|
||||
if (!authorizesOneOrMoreExtensions()) return false;
|
||||
|
||||
opts = opts || {};
|
||||
|
||||
const name = file && file.name;
|
||||
|
||||
if (!name) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// check that the uploaded file is authorized
|
||||
if (opts.allowStaffToUploadAnyFileInPm && opts.isPrivateMessage) {
|
||||
if (Discourse.User.currentProp("staff")) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
if (opts.imagesOnly) {
|
||||
if (!isAnImage(name) && !isAuthorizedImage(name)) {
|
||||
bootbox.alert(
|
||||
I18n.t("post.errors.upload_not_authorized", {
|
||||
authorized_extensions: authorizedImagesExtensions()
|
||||
})
|
||||
);
|
||||
return false;
|
||||
}
|
||||
} else if (opts.csvOnly) {
|
||||
if (!/\.csv$/i.test(name)) {
|
||||
bootbox.alert(I18n.t("user.invited.bulk_invite.error"));
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
if (!authorizesAllExtensions() && !isAuthorizedFile(name)) {
|
||||
bootbox.alert(
|
||||
I18n.t("post.errors.upload_not_authorized", {
|
||||
authorized_extensions: authorizedExtensions()
|
||||
})
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
if (!opts.bypassNewUserRestriction) {
|
||||
// ensures that new users can upload a file
|
||||
if (!Discourse.User.current().isAllowedToUploadAFile(opts.type)) {
|
||||
bootbox.alert(
|
||||
I18n.t(`post.errors.${opts.type}_upload_not_allowed_for_new_user`)
|
||||
);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// everything went fine
|
||||
return true;
|
||||
}
|
||||
|
||||
const IMAGES_EXTENSIONS_REGEX = /(png|jpe?g|gif|svg|ico)/i;
|
||||
|
||||
function extensionsToArray(exts) {
|
||||
return exts
|
||||
.toLowerCase()
|
||||
.replace(/[\s\.]+/g, "")
|
||||
.split("|")
|
||||
.filter(ext => ext.indexOf("*") === -1);
|
||||
}
|
||||
|
||||
function extensions() {
|
||||
return extensionsToArray(Discourse.SiteSettings.authorized_extensions);
|
||||
}
|
||||
|
||||
function staffExtensions() {
|
||||
return extensionsToArray(
|
||||
Discourse.SiteSettings.authorized_extensions_for_staff
|
||||
);
|
||||
}
|
||||
|
||||
function imagesExtensions() {
|
||||
let exts = extensions().filter(ext => IMAGES_EXTENSIONS_REGEX.test(ext));
|
||||
if (Discourse.User.currentProp("staff")) {
|
||||
const staffExts = staffExtensions().filter(ext =>
|
||||
IMAGES_EXTENSIONS_REGEX.test(ext)
|
||||
);
|
||||
exts = _.union(exts, staffExts);
|
||||
}
|
||||
return exts;
|
||||
}
|
||||
|
||||
function extensionsRegex() {
|
||||
return new RegExp("\\.(" + extensions().join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function imagesExtensionsRegex() {
|
||||
return new RegExp("\\.(" + imagesExtensions().join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function staffExtensionsRegex() {
|
||||
return new RegExp("\\.(" + staffExtensions().join("|") + ")$", "i");
|
||||
}
|
||||
|
||||
function isAuthorizedFile(fileName) {
|
||||
if (
|
||||
Discourse.User.currentProp("staff") &&
|
||||
staffExtensionsRegex().test(fileName)
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
return extensionsRegex().test(fileName);
|
||||
}
|
||||
|
||||
function isAuthorizedImage(fileName) {
|
||||
return imagesExtensionsRegex().test(fileName);
|
||||
}
|
||||
|
||||
export function authorizedExtensions() {
|
||||
const exts = Discourse.User.currentProp("staff")
|
||||
? [...extensions(), ...staffExtensions()]
|
||||
: extensions();
|
||||
return exts.filter(ext => ext.length > 0).join(", ");
|
||||
}
|
||||
|
||||
export function authorizedImagesExtensions() {
|
||||
return authorizesAllExtensions()
|
||||
? "png, jpg, jpeg, gif, svg, ico"
|
||||
: imagesExtensions().join(", ");
|
||||
}
|
||||
|
||||
export function authorizesAllExtensions() {
|
||||
return (
|
||||
Discourse.SiteSettings.authorized_extensions.indexOf("*") >= 0 ||
|
||||
(Discourse.SiteSettings.authorized_extensions_for_staff.indexOf("*") >= 0 &&
|
||||
Discourse.User.currentProp("staff"))
|
||||
);
|
||||
}
|
||||
|
||||
export function authorizesOneOrMoreExtensions() {
|
||||
if (authorizesAllExtensions()) return true;
|
||||
|
||||
return (
|
||||
Discourse.SiteSettings.authorized_extensions.split("|").filter(ext => ext)
|
||||
.length > 0
|
||||
);
|
||||
}
|
||||
|
||||
export function authorizesOneOrMoreImageExtensions() {
|
||||
if (authorizesAllExtensions()) return true;
|
||||
|
||||
return imagesExtensions().length > 0;
|
||||
}
|
||||
|
||||
export function isAnImage(path) {
|
||||
return /\.(png|jpe?g|gif|svg|ico)$/i.test(path);
|
||||
}
|
||||
|
||||
function uploadTypeFromFileName(fileName) {
|
||||
return isAnImage(fileName) ? "image" : "attachment";
|
||||
}
|
||||
|
||||
function isGUID(value) {
|
||||
return /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(
|
||||
value
|
||||
);
|
||||
}
|
||||
|
||||
function imageNameFromFileName(fileName) {
|
||||
const split = fileName.split(".");
|
||||
let name = split[split.length - 2];
|
||||
|
||||
if (exports.isAppleDevice() && isGUID(name)) {
|
||||
name = I18n.t("upload_selector.default_image_alt_text");
|
||||
}
|
||||
|
||||
return encodeURIComponent(name);
|
||||
}
|
||||
|
||||
export function allowsImages() {
|
||||
return (
|
||||
authorizesAllExtensions() ||
|
||||
IMAGES_EXTENSIONS_REGEX.test(authorizedExtensions())
|
||||
);
|
||||
}
|
||||
|
||||
export function allowsAttachments() {
|
||||
return (
|
||||
authorizesAllExtensions() ||
|
||||
authorizedExtensions().split(", ").length > imagesExtensions().length
|
||||
);
|
||||
}
|
||||
|
||||
export function uploadIcon() {
|
||||
return allowsAttachments() ? "upload" : "far-image";
|
||||
}
|
||||
|
||||
export function uploadLocation(url) {
|
||||
if (Discourse.CDN) {
|
||||
url = Discourse.getURLWithCDN(url);
|
||||
return /^\/\//.test(url) ? "http:" + url : url;
|
||||
} else if (Discourse.S3BaseUrl) {
|
||||
return "https:" + url;
|
||||
} else {
|
||||
var protocol = window.location.protocol + "//",
|
||||
hostname = window.location.hostname,
|
||||
port = window.location.port ? ":" + window.location.port : "";
|
||||
return protocol + hostname + port + url;
|
||||
}
|
||||
}
|
||||
|
||||
export function getUploadMarkdown(upload) {
|
||||
if (isAnImage(upload.original_filename)) {
|
||||
const name = imageNameFromFileName(upload.original_filename);
|
||||
return ``;
|
||||
} else if (
|
||||
!Discourse.SiteSettings.prevent_anons_from_downloading_files &&
|
||||
/\.(mov|mp4|webm|ogv|mp3|ogg|wav|m4a)$/i.test(upload.original_filename)
|
||||
) {
|
||||
return uploadLocation(upload.url);
|
||||
} else {
|
||||
return `[${upload.original_filename}|attachment](${
|
||||
upload.short_url
|
||||
}) (${I18n.toHumanSize(upload.filesize)})`;
|
||||
}
|
||||
}
|
||||
|
||||
export function displayErrorForUpload(data) {
|
||||
if (data.jqXHR) {
|
||||
switch (data.jqXHR.status) {
|
||||
// cancelled by the user
|
||||
case 0:
|
||||
return;
|
||||
|
||||
// entity too large, usually returned from the web server
|
||||
case 413:
|
||||
const type = uploadTypeFromFileName(data.files[0].name);
|
||||
const max_size_kb = Discourse.SiteSettings[`max_${type}_size_kb`];
|
||||
bootbox.alert(I18n.t("post.errors.file_too_large", { max_size_kb }));
|
||||
return;
|
||||
|
||||
// the error message is provided by the server
|
||||
case 422:
|
||||
if (data.jqXHR.responseJSON.message) {
|
||||
bootbox.alert(data.jqXHR.responseJSON.message);
|
||||
} else {
|
||||
bootbox.alert(data.jqXHR.responseJSON.errors.join("\n"));
|
||||
}
|
||||
return;
|
||||
}
|
||||
} else if (data.errors && data.errors.length > 0) {
|
||||
bootbox.alert(data.errors.join("\n"));
|
||||
return;
|
||||
}
|
||||
// otherwise, display a generic error message
|
||||
bootbox.alert(I18n.t("post.errors.upload"));
|
||||
}
|
||||
|
||||
export function defaultHomepage() {
|
||||
let homepage = null;
|
||||
let elem = _.first($(homepageSelector));
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
import {
|
||||
displayErrorForUpload,
|
||||
validateUploadedFiles
|
||||
} from "discourse/lib/utilities";
|
||||
} from "discourse/lib/uploads";
|
||||
import getUrl from "discourse-common/lib/get-url";
|
||||
import { on } from "@ember/object/evented";
|
||||
import Mixin from "@ember/object/mixin";
|
||||
@@ -78,7 +78,7 @@ export default Mixin.create({
|
||||
|
||||
$upload.on("fileuploadsubmit", (e, data) => {
|
||||
const opts = _.merge(
|
||||
{ bypassNewUserRestriction: true },
|
||||
{ bypassNewUserRestriction: true, user: this.currentUser },
|
||||
this.validateUploadedFilesOptions()
|
||||
);
|
||||
const isValid = validateUploadedFiles(data.files, opts);
|
||||
|
||||
Reference in New Issue
Block a user