2023-10-10 14:38:59 -04:00
|
|
|
import { warn } from "@ember/debug";
|
|
|
|
import { action } from "@ember/object";
|
2019-10-30 16:28:29 -04:00
|
|
|
import { alias, oneWay } from "@ember/object/computed";
|
2019-10-30 16:13:48 -04:00
|
|
|
import Mixin from "@ember/object/mixin";
|
2023-10-10 14:38:59 -04:00
|
|
|
import { inject as service } from "@ember/service";
|
2020-04-30 16:41:02 -04:00
|
|
|
import { htmlSafe } from "@ember/template";
|
2023-10-10 14:38:59 -04:00
|
|
|
import { isNone } from "@ember/utils";
|
|
|
|
import { ajax } from "discourse/lib/ajax";
|
|
|
|
import { fmt, propertyNotEqual } from "discourse/lib/computed";
|
2021-09-27 04:43:26 -04:00
|
|
|
import { splitString } from "discourse/lib/utilities";
|
2023-10-10 14:38:59 -04:00
|
|
|
import discourseComputed, { bind } from "discourse-common/utils/decorators";
|
2023-10-18 06:07:09 -04:00
|
|
|
import I18n from "discourse-i18n";
|
2023-08-10 05:31:34 -04:00
|
|
|
import SiteSettingDefaultCategoriesModal from "../components/modal/site-setting-default-categories";
|
2018-03-04 19:04:23 -05:00
|
|
|
|
2018-03-13 15:59:12 -04:00
|
|
|
const CUSTOM_TYPES = [
|
|
|
|
"bool",
|
2023-10-06 13:21:01 -04:00
|
|
|
"integer",
|
2018-03-13 15:59:12 -04:00
|
|
|
"enum",
|
|
|
|
"list",
|
|
|
|
"url_list",
|
|
|
|
"host_list",
|
|
|
|
"category_list",
|
|
|
|
"value_list",
|
2018-07-18 06:57:43 -04:00
|
|
|
"category",
|
2018-08-03 16:41:37 -04:00
|
|
|
"uploaded_image_list",
|
2018-10-15 01:03:53 -04:00
|
|
|
"compact_list",
|
2018-11-14 02:03:02 -05:00
|
|
|
"secret_list",
|
2019-04-18 13:44:58 -04:00
|
|
|
"upload",
|
2019-11-01 03:10:13 -04:00
|
|
|
"group_list",
|
2020-03-09 05:07:03 -04:00
|
|
|
"tag_list",
|
2023-06-09 11:02:55 -04:00
|
|
|
"tag_group_list",
|
2020-06-04 10:44:54 -04:00
|
|
|
"color",
|
|
|
|
"simple_list",
|
2021-04-07 09:32:05 -04:00
|
|
|
"emoji_list",
|
2022-03-08 06:18:43 -05:00
|
|
|
"named_list",
|
2023-12-13 18:22:48 -05:00
|
|
|
"file_size_restriction",
|
2023-12-21 19:23:42 -05:00
|
|
|
"file_types_list",
|
2018-03-13 15:59:12 -04:00
|
|
|
];
|
2018-03-04 19:04:23 -05:00
|
|
|
|
2019-06-20 10:57:31 -04:00
|
|
|
const AUTO_REFRESH_ON_SAVE = ["logo", "logo_small", "large_icon"];
|
|
|
|
|
2023-03-01 04:12:39 -05:00
|
|
|
const DEFAULT_USER_PREFERENCES = [
|
|
|
|
"default_email_digest_frequency",
|
|
|
|
"default_include_tl0_in_digests",
|
|
|
|
"default_email_level",
|
|
|
|
"default_email_messages_level",
|
|
|
|
"default_email_mailing_list_mode",
|
|
|
|
"default_email_mailing_list_mode_frequency",
|
|
|
|
"default_email_previous_replies",
|
|
|
|
"default_email_in_reply_to",
|
|
|
|
"default_hide_profile_and_presence",
|
|
|
|
"default_other_new_topic_duration_minutes",
|
|
|
|
"default_other_auto_track_topics_after_msecs",
|
|
|
|
"default_other_notification_level_when_replying",
|
|
|
|
"default_other_external_links_in_new_tab",
|
|
|
|
"default_other_enable_quoting",
|
|
|
|
"default_other_enable_defer",
|
|
|
|
"default_other_dynamic_favicon",
|
|
|
|
"default_other_like_notification_frequency",
|
|
|
|
"default_other_skip_new_user_tips",
|
|
|
|
"default_topics_automatic_unpin",
|
|
|
|
"default_categories_watching",
|
|
|
|
"default_categories_tracking",
|
|
|
|
"default_categories_muted",
|
|
|
|
"default_categories_watching_first_post",
|
|
|
|
"default_categories_normal",
|
|
|
|
"default_tags_watching",
|
|
|
|
"default_tags_tracking",
|
|
|
|
"default_tags_muted",
|
|
|
|
"default_tags_watching_first_post",
|
|
|
|
"default_text_size",
|
|
|
|
"default_title_count_mode",
|
2023-06-14 19:31:28 -04:00
|
|
|
"default_navigation_menu_categories",
|
|
|
|
"default_navigation_menu_tags",
|
2023-07-06 21:52:10 -04:00
|
|
|
"default_sidebar_link_to_filtered_list",
|
|
|
|
"default_sidebar_show_count_of_new_items",
|
2023-03-01 04:12:39 -05:00
|
|
|
];
|
|
|
|
|
2019-10-30 15:03:08 -04:00
|
|
|
export default Mixin.create({
|
2023-08-10 05:31:34 -04:00
|
|
|
modal: service(),
|
|
|
|
site: service(),
|
2023-03-01 04:12:39 -05:00
|
|
|
attributeBindings: ["setting.setting:data-setting"],
|
2019-01-11 03:39:21 -05:00
|
|
|
classNameBindings: [":row", ":setting", "overridden", "typeClass"],
|
2018-03-04 19:04:23 -05:00
|
|
|
validationMessage: null,
|
2021-09-27 04:43:26 -04:00
|
|
|
setting: null,
|
2023-03-01 04:12:39 -05:00
|
|
|
|
|
|
|
content: alias("setting"),
|
|
|
|
isSecret: oneWay("setting.secret"),
|
|
|
|
componentName: fmt("typeClass", "site-settings/%@"),
|
|
|
|
overridden: propertyNotEqual("setting.default", "buffered.value"),
|
|
|
|
|
|
|
|
didInsertElement() {
|
|
|
|
this._super(...arguments);
|
|
|
|
this.element.addEventListener("keydown", this._handleKeydown);
|
|
|
|
},
|
|
|
|
|
|
|
|
willDestroyElement() {
|
|
|
|
this._super(...arguments);
|
|
|
|
this.element.removeEventListener("keydown", this._handleKeydown);
|
|
|
|
},
|
2018-03-04 19:04:23 -05:00
|
|
|
|
2019-11-07 16:38:28 -05:00
|
|
|
@discourseComputed("buffered.value", "setting.value")
|
2018-03-04 19:04:23 -05:00
|
|
|
dirty(bufferVal, settingVal) {
|
2021-09-27 04:43:26 -04:00
|
|
|
if (isNone(bufferVal)) {
|
2020-09-22 10:28:28 -04:00
|
|
|
bufferVal = "";
|
|
|
|
}
|
2021-09-27 04:43:26 -04:00
|
|
|
|
|
|
|
if (isNone(settingVal)) {
|
2020-09-22 10:28:28 -04:00
|
|
|
settingVal = "";
|
|
|
|
}
|
2018-03-04 19:04:23 -05:00
|
|
|
|
|
|
|
return bufferVal.toString() !== settingVal.toString();
|
|
|
|
},
|
|
|
|
|
2019-11-07 16:38:28 -05:00
|
|
|
@discourseComputed("setting", "buffered.value")
|
2018-03-04 19:04:23 -05:00
|
|
|
preview(setting, value) {
|
2021-09-27 04:43:26 -04:00
|
|
|
const preview = setting.preview;
|
2018-03-04 19:04:23 -05:00
|
|
|
if (preview) {
|
2021-09-27 04:43:26 -04:00
|
|
|
const escapedValue = preview.replace(/\{\{value\}\}/g, value);
|
|
|
|
return htmlSafe(`<div class='preview'>${escapedValue}</div>`);
|
2018-03-04 19:04:23 -05:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-11-07 16:38:28 -05:00
|
|
|
@discourseComputed("componentType")
|
2018-03-04 19:04:23 -05:00
|
|
|
typeClass(componentType) {
|
|
|
|
return componentType.replace(/\_/g, "-");
|
|
|
|
},
|
|
|
|
|
2019-11-28 00:19:01 -05:00
|
|
|
@discourseComputed("setting.setting", "setting.label")
|
|
|
|
settingName(setting, label) {
|
|
|
|
return label || setting.replace(/\_/g, " ");
|
2018-03-04 19:04:23 -05:00
|
|
|
},
|
|
|
|
|
2019-11-07 16:38:28 -05:00
|
|
|
@discourseComputed("type")
|
2018-03-04 19:04:23 -05:00
|
|
|
componentType(type) {
|
2022-07-17 14:48:36 -04:00
|
|
|
return CUSTOM_TYPES.includes(type) ? type : "string";
|
2018-03-04 19:04:23 -05:00
|
|
|
},
|
|
|
|
|
2019-11-07 16:38:28 -05:00
|
|
|
@discourseComputed("setting")
|
2018-08-03 16:41:37 -04:00
|
|
|
type(setting) {
|
|
|
|
if (setting.type === "list" && setting.list_type) {
|
|
|
|
return `${setting.list_type}_list`;
|
|
|
|
}
|
|
|
|
|
|
|
|
return setting.type;
|
|
|
|
},
|
|
|
|
|
2019-11-28 00:19:01 -05:00
|
|
|
@discourseComputed("setting.anyValue")
|
|
|
|
allowAny(anyValue) {
|
|
|
|
return anyValue !== false;
|
|
|
|
},
|
|
|
|
|
2019-12-04 01:13:41 -05:00
|
|
|
@discourseComputed("buffered.value")
|
2021-09-27 04:43:26 -04:00
|
|
|
bufferedValues(value) {
|
|
|
|
return splitString(value, "|");
|
|
|
|
},
|
2019-12-04 01:13:41 -05:00
|
|
|
|
|
|
|
@discourseComputed("setting.defaultValues")
|
2021-09-27 04:43:26 -04:00
|
|
|
defaultValues(value) {
|
|
|
|
return splitString(value, "|");
|
|
|
|
},
|
2019-12-04 01:13:41 -05:00
|
|
|
|
|
|
|
@discourseComputed("defaultValues", "bufferedValues")
|
|
|
|
defaultIsAvailable(defaultValues, bufferedValues) {
|
|
|
|
return (
|
2019-12-04 16:25:49 -05:00
|
|
|
defaultValues.length > 0 &&
|
2019-12-04 01:13:41 -05:00
|
|
|
!defaultValues.every((value) => bufferedValues.includes(value))
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2021-09-27 04:43:26 -04:00
|
|
|
@action
|
2023-03-01 04:12:39 -05:00
|
|
|
async update() {
|
2021-09-27 04:43:26 -04:00
|
|
|
const key = this.buffered.get("setting");
|
|
|
|
|
2023-03-01 04:12:39 -05:00
|
|
|
if (!DEFAULT_USER_PREFERENCES.includes(key)) {
|
|
|
|
await this.save();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const data = {
|
|
|
|
[key]: this.buffered.get("value"),
|
|
|
|
};
|
|
|
|
|
|
|
|
const result = await ajax(`/admin/site_settings/${key}/user_count.json`, {
|
|
|
|
type: "PUT",
|
|
|
|
data,
|
|
|
|
});
|
|
|
|
|
|
|
|
const count = result.user_count;
|
|
|
|
if (count > 0) {
|
2023-08-10 05:31:34 -04:00
|
|
|
await this.modal.show(SiteSettingDefaultCategoriesModal, {
|
|
|
|
model: {
|
|
|
|
siteSetting: { count, key: key.replaceAll("_", " ") },
|
|
|
|
setUpdateExistingUsers: this.setUpdateExistingUsers,
|
|
|
|
},
|
2021-09-27 04:43:26 -04:00
|
|
|
});
|
2023-08-10 05:31:34 -04:00
|
|
|
this.save();
|
2021-09-27 04:43:26 -04:00
|
|
|
} else {
|
2023-03-01 04:12:39 -05:00
|
|
|
await this.save();
|
2021-09-27 04:43:26 -04:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2023-08-10 05:31:34 -04:00
|
|
|
@action
|
|
|
|
setUpdateExistingUsers(value) {
|
|
|
|
this.updateExistingUsers = value;
|
|
|
|
},
|
|
|
|
|
2021-09-27 04:43:26 -04:00
|
|
|
@action
|
2023-03-01 04:12:39 -05:00
|
|
|
async save() {
|
|
|
|
try {
|
|
|
|
await this._save();
|
|
|
|
|
|
|
|
this.set("validationMessage", null);
|
|
|
|
this.commitBuffer();
|
|
|
|
if (AUTO_REFRESH_ON_SAVE.includes(this.setting.setting)) {
|
|
|
|
this.afterSave();
|
|
|
|
}
|
|
|
|
} catch (e) {
|
2023-05-04 16:01:19 -04:00
|
|
|
const json = e.jqXHR?.responseJSON;
|
|
|
|
if (json?.errors) {
|
|
|
|
let errorString = json.errors[0];
|
|
|
|
|
|
|
|
if (json.html_message) {
|
|
|
|
errorString = htmlSafe(errorString);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.set("validationMessage", errorString);
|
2023-03-01 04:12:39 -05:00
|
|
|
} else {
|
|
|
|
this.set("validationMessage", I18n.t("generic_error"));
|
|
|
|
}
|
|
|
|
}
|
2021-09-27 04:43:26 -04:00
|
|
|
},
|
|
|
|
|
2023-12-21 19:23:42 -05:00
|
|
|
@action
|
|
|
|
changeValueCallback(value) {
|
|
|
|
this.set("buffered.value", value);
|
|
|
|
},
|
|
|
|
|
2021-09-27 04:43:26 -04:00
|
|
|
@action
|
|
|
|
cancel() {
|
|
|
|
this.rollbackBuffer();
|
|
|
|
},
|
|
|
|
|
|
|
|
@action
|
|
|
|
resetDefault() {
|
|
|
|
this.set("buffered.value", this.get("setting.default"));
|
|
|
|
},
|
|
|
|
|
|
|
|
@action
|
|
|
|
toggleSecret() {
|
|
|
|
this.toggleProperty("isSecret");
|
|
|
|
},
|
|
|
|
|
|
|
|
@action
|
|
|
|
setDefaultValues() {
|
|
|
|
this.set(
|
|
|
|
"buffered.value",
|
|
|
|
this.bufferedValues.concat(this.defaultValues).uniq().join("|")
|
2018-03-04 19:04:23 -05:00
|
|
|
);
|
2021-09-27 04:43:26 -04:00
|
|
|
return false;
|
|
|
|
},
|
|
|
|
|
|
|
|
@bind
|
|
|
|
_handleKeydown(event) {
|
|
|
|
if (
|
|
|
|
event.key === "Enter" &&
|
|
|
|
event.target.classList.contains("input-setting-string")
|
|
|
|
) {
|
|
|
|
this.save();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2023-03-01 04:12:39 -05:00
|
|
|
async _save() {
|
2020-03-06 17:49:28 -05:00
|
|
|
warn("You should define a `_save` method", {
|
2019-01-16 14:53:13 -05:00
|
|
|
id: "discourse.setting-component.missing-save",
|
2018-03-04 19:04:23 -05:00
|
|
|
});
|
|
|
|
},
|
|
|
|
});
|