2021-03-11 05:43:45 +01:00
|
|
|
import $ from "jquery";
|
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
import pygments_data from "../generated/pygments_data.json";
|
2021-06-21 10:52:02 +02:00
|
|
|
import render_settings_deactivate_realm_modal from "../templates/confirm_dialog/confirm_deactivate_realm.hbs";
|
2021-02-28 01:20:46 +01:00
|
|
|
import render_settings_admin_auth_methods_list from "../templates/settings/admin_auth_methods_list.hbs";
|
|
|
|
|
2021-03-16 23:38:59 +01:00
|
|
|
import * as blueslip from "./blueslip";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as channel from "./channel";
|
2021-05-26 19:31:13 +02:00
|
|
|
import * as confirm_dialog from "./confirm_dialog";
|
2021-03-25 23:20:18 +01:00
|
|
|
import {csrf_token} from "./csrf";
|
2021-07-27 14:05:22 +02:00
|
|
|
import {DropdownListWidget} from "./dropdown_list_widget";
|
2022-05-03 08:06:40 +02:00
|
|
|
import {$t, $t_html, get_language_name} from "./i18n";
|
2022-09-28 08:27:24 +02:00
|
|
|
import * as keydown_util from "./keydown_util";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as loading from "./loading";
|
2021-03-25 22:35:45 +01:00
|
|
|
import {page_params} from "./page_params";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as realm_icon from "./realm_icon";
|
|
|
|
import * as realm_logo from "./realm_logo";
|
2021-09-28 09:09:52 +02:00
|
|
|
import {realm_user_settings_defaults} from "./realm_user_settings_defaults";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as settings_config from "./settings_config";
|
|
|
|
import * as settings_notifications from "./settings_notifications";
|
2021-07-20 14:51:48 +02:00
|
|
|
import * as settings_realm_domains from "./settings_realm_domains";
|
2021-09-30 12:00:57 +02:00
|
|
|
import * as settings_realm_user_settings_defaults from "./settings_realm_user_settings_defaults";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as settings_ui from "./settings_ui";
|
2021-04-04 15:15:18 +02:00
|
|
|
import * as stream_settings_data from "./stream_settings_data";
|
2021-02-28 01:20:46 +01:00
|
|
|
import * as ui_report from "./ui_report";
|
|
|
|
|
|
|
|
export let parse_time_limit;
|
2019-07-09 21:24:00 +02:00
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
const meta = {
|
2017-04-08 18:13:39 +02:00
|
|
|
loaded: false,
|
|
|
|
};
|
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function reset() {
|
2017-04-17 16:51:27 +02:00
|
|
|
meta.loaded = false;
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2017-04-17 16:51:27 +02:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function maybe_disable_widgets() {
|
2020-06-11 13:26:27 +02:00
|
|
|
if (page_params.is_owner) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-06-11 16:24:15 +02:00
|
|
|
$(".organization-box [data-name='auth-methods']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find("input, button, select, checked")
|
2020-07-22 02:59:06 +02:00
|
|
|
.prop("disabled", true);
|
2020-06-11 16:24:15 +02:00
|
|
|
|
2018-12-08 17:37:57 +01:00
|
|
|
if (page_params.is_admin) {
|
2020-07-22 02:59:06 +02:00
|
|
|
$("#deactivate_realm_button").prop("disabled", true);
|
|
|
|
$("#org-message-retention").find("input, select").prop("disabled", true);
|
2021-07-29 13:53:27 +02:00
|
|
|
$("#org-join").find("input, select").prop("disabled", true);
|
|
|
|
$("#id_realm_invite_required_label").parent().addClass("control-label-disabled");
|
2018-12-08 17:37:57 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-profile']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find("input, textarea, button, select")
|
2020-07-22 02:59:06 +02:00
|
|
|
.prop("disabled", true);
|
2018-12-08 17:37:57 +01:00
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-settings']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find("input, textarea, button, select")
|
2020-07-22 02:59:06 +02:00
|
|
|
.prop("disabled", true);
|
2018-12-08 17:37:57 +01:00
|
|
|
|
2021-11-16 10:40:36 +01:00
|
|
|
$(".organization-box [data-name='organization-settings']")
|
|
|
|
.find(".dropdown_list_reset_button")
|
|
|
|
.hide();
|
|
|
|
|
2018-12-08 17:37:57 +01:00
|
|
|
$(".organization-box [data-name='organization-settings']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find(".control-label-disabled")
|
|
|
|
.addClass("enabled");
|
2018-12-08 17:37:57 +01:00
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-permissions']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find("input, textarea, button, select")
|
2020-07-22 02:59:06 +02:00
|
|
|
.prop("disabled", true);
|
2018-12-08 17:37:57 +01:00
|
|
|
|
|
|
|
$(".organization-box [data-name='organization-permissions']")
|
2020-07-15 00:34:28 +02:00
|
|
|
.find(".control-label-disabled")
|
|
|
|
.addClass("enabled");
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2018-12-08 17:37:57 +01:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function get_sorted_options_list(option_values_object) {
|
2020-07-02 01:39:34 +02:00
|
|
|
const options_list = Object.keys(option_values_object).map((key) => ({
|
2020-02-09 04:15:38 +01:00
|
|
|
...option_values_object[key],
|
2020-07-20 22:18:43 +02:00
|
|
|
key,
|
2020-02-09 04:15:38 +01:00
|
|
|
}));
|
2020-01-23 07:01:34 +01:00
|
|
|
let comparator = (x, y) => x.order - y.order;
|
|
|
|
if (!options_list[0].order) {
|
|
|
|
comparator = (x, y) => {
|
|
|
|
const key_x = x.key.toUpperCase();
|
|
|
|
const key_y = y.key.toUpperCase();
|
|
|
|
if (key_x < key_y) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (key_x > key_y) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
options_list.sort(comparator);
|
|
|
|
return options_list;
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-01-23 07:01:34 +01:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function get_organization_settings_options() {
|
2020-01-23 07:01:34 +01:00
|
|
|
const options = {};
|
2021-04-28 20:39:24 +02:00
|
|
|
options.common_policy_values = get_sorted_options_list(settings_config.common_policy_values);
|
2021-02-28 01:20:46 +01:00
|
|
|
options.private_message_policy_values = get_sorted_options_list(
|
2020-07-15 00:34:28 +02:00
|
|
|
settings_config.private_message_policy_values,
|
|
|
|
);
|
2021-02-28 01:20:46 +01:00
|
|
|
options.wildcard_mention_policy_values = get_sorted_options_list(
|
2020-09-14 19:26:42 +02:00
|
|
|
settings_config.wildcard_mention_policy_values,
|
|
|
|
);
|
2021-05-26 12:21:37 +02:00
|
|
|
options.common_message_policy_values = get_sorted_options_list(
|
|
|
|
settings_config.common_message_policy_values,
|
|
|
|
);
|
2021-07-18 18:18:28 +02:00
|
|
|
options.invite_to_realm_policy_values = get_sorted_options_list(
|
|
|
|
settings_config.invite_to_realm_policy_values,
|
|
|
|
);
|
2020-01-23 07:01:34 +01:00
|
|
|
return options;
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-01-23 07:01:34 +01:00
|
|
|
|
2022-04-12 17:41:07 +02:00
|
|
|
export function get_org_type_dropdown_options() {
|
|
|
|
const current_org_type = page_params.realm_org_type;
|
|
|
|
if (current_org_type !== 0) {
|
|
|
|
return settings_config.defined_org_type_values;
|
|
|
|
}
|
|
|
|
return settings_config.all_org_type_values;
|
|
|
|
}
|
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function get_realm_time_limits_in_minutes(property) {
|
2022-06-03 19:20:32 +02:00
|
|
|
if (page_params[property] === null) {
|
|
|
|
// This represents "Anytime" case.
|
|
|
|
return null;
|
|
|
|
}
|
2019-11-02 00:06:25 +01:00
|
|
|
let val = (page_params[property] / 60).toFixed(1);
|
2020-10-07 09:17:30 +02:00
|
|
|
if (Number.parseFloat(val, 10) === Number.parseInt(val, 10)) {
|
|
|
|
val = Number.parseInt(val, 10);
|
2018-04-28 15:35:14 +02:00
|
|
|
}
|
|
|
|
return val.toString();
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2018-04-28 15:35:14 +02:00
|
|
|
|
2021-09-28 09:09:52 +02:00
|
|
|
function get_property_value(property_name, for_realm_default_settings) {
|
|
|
|
if (for_realm_default_settings) {
|
|
|
|
// realm_user_default_settings are stored in a separate object.
|
|
|
|
if (property_name === "twenty_four_hour_time") {
|
|
|
|
return JSON.stringify(realm_user_settings_defaults.twenty_four_hour_time);
|
|
|
|
}
|
2021-12-02 15:56:34 +01:00
|
|
|
if (
|
|
|
|
property_name === "email_notifications_batching_period_seconds" ||
|
|
|
|
property_name === "email_notification_batching_period_edit_minutes"
|
|
|
|
) {
|
|
|
|
return realm_user_settings_defaults.email_notifications_batching_period_seconds;
|
|
|
|
}
|
2021-09-28 09:09:52 +02:00
|
|
|
return realm_user_settings_defaults[property_name];
|
|
|
|
}
|
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
if (property_name === "realm_waiting_period_setting") {
|
2019-05-07 19:06:05 +02:00
|
|
|
if (page_params.realm_waiting_period_threshold === 0) {
|
|
|
|
return "none";
|
|
|
|
}
|
|
|
|
if (page_params.realm_waiting_period_threshold === 3) {
|
|
|
|
return "three_days";
|
|
|
|
}
|
|
|
|
return "custom_days";
|
|
|
|
}
|
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
if (property_name === "realm_message_retention_setting") {
|
2020-06-25 23:44:45 +02:00
|
|
|
if (page_params.realm_message_retention_days === settings_config.retain_message_forever) {
|
2020-05-07 13:19:54 +02:00
|
|
|
return "retain_forever";
|
|
|
|
}
|
|
|
|
return "retain_for_period";
|
|
|
|
}
|
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
if (property_name === "realm_org_join_restrictions") {
|
2018-07-27 23:26:29 +02:00
|
|
|
if (page_params.realm_emails_restricted_to_domains) {
|
2018-04-05 00:54:31 +02:00
|
|
|
return "only_selected_domain";
|
|
|
|
}
|
|
|
|
if (page_params.realm_disallow_disposable_email_addresses) {
|
|
|
|
return "no_disposable_email";
|
|
|
|
}
|
|
|
|
return "no_restriction";
|
2018-06-16 13:27:56 +02:00
|
|
|
}
|
|
|
|
|
2018-04-19 21:45:47 +02:00
|
|
|
return page_params[property_name];
|
2018-03-29 12:52:57 +02:00
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
export function extract_property_name($elem, for_realm_default_settings) {
|
2021-09-28 09:09:52 +02:00
|
|
|
if (for_realm_default_settings) {
|
|
|
|
// We use the name attribute, rather than the ID attribute,
|
|
|
|
// for realm_user_default_settings. This is because the
|
|
|
|
// display/notification settings elements do not always have
|
|
|
|
// IDs, and also the emojiset input is not compatible with the
|
|
|
|
// ID approach.
|
2022-01-25 11:36:19 +01:00
|
|
|
return $elem.attr("name");
|
2021-09-28 09:09:52 +02:00
|
|
|
}
|
2022-07-07 10:12:16 +02:00
|
|
|
|
|
|
|
if ($elem.attr("id").startsWith("id_authmethod")) {
|
|
|
|
// Authentication Method component IDs include authentication method name
|
|
|
|
// for uniqueness, anchored to "id_authmethod" prefix, e.g. "id_authmethodapple_<property_name>".
|
|
|
|
// We need to strip that whole construct down to extract the actual property name.
|
|
|
|
// The [\da-z]+ part of the regexp covers the auth method name itself.
|
|
|
|
// We assume it's not an empty string and can contain only digits and lowercase ASCII letters,
|
|
|
|
// this is ensured by a respective allowlist-based filter in populate_auth_methods().
|
|
|
|
return /^id_authmethod[\da-z]+_(.*)$/.exec($elem.attr("id"))[1];
|
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
return /^id_(.*)$/.exec($elem.attr("id").replace(/-/g, "_"))[1];
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2018-03-29 12:52:57 +02:00
|
|
|
|
2019-05-05 22:37:09 +02:00
|
|
|
function get_subsection_property_elements(element) {
|
2022-01-25 11:36:19 +01:00
|
|
|
const $subsection = $(element).closest(".org-subsection-parent");
|
|
|
|
if ($subsection.hasClass("theme-settings")) {
|
2021-09-28 09:09:52 +02:00
|
|
|
// Because the emojiset widget has a unique radio button
|
|
|
|
// structure, it needs custom code.
|
2022-01-25 11:36:19 +01:00
|
|
|
const $color_scheme_elem = $subsection.find(".setting_color_scheme");
|
2022-06-01 22:25:44 +02:00
|
|
|
const $display_emoji_reaction_users_elem = $subsection.find(
|
|
|
|
".display_emoji_reaction_users",
|
|
|
|
);
|
2022-01-25 11:36:19 +01:00
|
|
|
const $emojiset_elem = $subsection.find("input[name='emojiset']:checked");
|
2022-08-12 22:41:06 +02:00
|
|
|
const $user_list_style_elem = $subsection.find("input[name='user_list_style']:checked");
|
2022-01-25 11:36:19 +01:00
|
|
|
const $translate_emoticons_elem = $subsection.find(".translate_emoticons");
|
2022-08-12 22:41:06 +02:00
|
|
|
return [
|
|
|
|
$color_scheme_elem,
|
|
|
|
$emojiset_elem,
|
|
|
|
$user_list_style_elem,
|
|
|
|
$translate_emoticons_elem,
|
2022-06-01 22:25:44 +02:00
|
|
|
$display_emoji_reaction_users_elem,
|
2022-08-12 22:41:06 +02:00
|
|
|
];
|
2021-09-28 09:09:52 +02:00
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
return Array.from($subsection.find(".prop-element"));
|
2019-05-05 22:37:09 +02:00
|
|
|
}
|
|
|
|
|
2020-07-15 00:34:28 +02:00
|
|
|
const simple_dropdown_properties = [
|
2021-03-27 05:48:37 +01:00
|
|
|
"realm_create_private_stream_policy",
|
|
|
|
"realm_create_public_stream_policy",
|
2021-11-19 10:29:39 +01:00
|
|
|
"realm_create_web_public_stream_policy",
|
2020-07-15 00:34:28 +02:00
|
|
|
"realm_invite_to_stream_policy",
|
|
|
|
"realm_user_group_edit_policy",
|
|
|
|
"realm_private_message_policy",
|
2021-05-04 19:02:24 +02:00
|
|
|
"realm_add_custom_emoji_policy",
|
2021-05-23 21:40:20 +02:00
|
|
|
"realm_invite_to_realm_policy",
|
2020-09-14 19:26:42 +02:00
|
|
|
"realm_wildcard_mention_policy",
|
2021-04-30 09:35:20 +02:00
|
|
|
"realm_move_messages_between_streams_policy",
|
2021-05-26 12:21:37 +02:00
|
|
|
"realm_edit_topic_policy",
|
2022-04-12 17:41:07 +02:00
|
|
|
"realm_org_type",
|
2020-07-15 00:34:28 +02:00
|
|
|
];
|
2020-02-03 12:40:36 +01:00
|
|
|
|
|
|
|
function set_property_dropdown_value(property_name) {
|
2021-02-03 23:23:32 +01:00
|
|
|
$(`#id_${CSS.escape(property_name)}`).val(get_property_value(property_name));
|
2020-02-03 12:40:36 +01:00
|
|
|
}
|
|
|
|
|
2021-12-02 14:36:09 +01:00
|
|
|
export function change_element_block_display_property(elem_id, show_element) {
|
2022-01-25 11:36:19 +01:00
|
|
|
const $elem = $(`#${CSS.escape(elem_id)}`);
|
2020-02-03 13:03:53 +01:00
|
|
|
if (show_element) {
|
2022-01-25 11:36:19 +01:00
|
|
|
$elem.parent().show();
|
2020-02-03 13:03:53 +01:00
|
|
|
} else {
|
2022-01-25 11:36:19 +01:00
|
|
|
$elem.parent().hide();
|
2020-02-03 13:03:53 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-07 19:06:05 +02:00
|
|
|
function set_realm_waiting_period_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_waiting_period_setting");
|
2019-05-07 19:06:05 +02:00
|
|
|
$("#id_realm_waiting_period_setting").val(value);
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_waiting_period_threshold",
|
|
|
|
value === "custom_days",
|
|
|
|
);
|
2018-04-01 08:49:10 +02:00
|
|
|
}
|
2018-01-05 15:34:10 +01:00
|
|
|
|
2018-04-23 14:51:30 +02:00
|
|
|
function set_video_chat_provider_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const chat_provider_id = page_params.realm_video_chat_provider;
|
2019-05-09 09:54:38 +02:00
|
|
|
$("#id_realm_video_chat_provider").val(chat_provider_id);
|
2018-04-23 14:51:30 +02:00
|
|
|
}
|
|
|
|
|
2021-04-27 21:58:19 +02:00
|
|
|
function set_giphy_rating_dropdown() {
|
|
|
|
const rating_id = page_params.realm_giphy_rating;
|
|
|
|
$("#id_realm_giphy_rating").val(rating_id);
|
|
|
|
}
|
|
|
|
|
2022-04-12 09:56:58 +02:00
|
|
|
function update_message_edit_sub_settings(is_checked) {
|
2022-08-19 16:14:15 +02:00
|
|
|
settings_ui.disable_sub_setting_onchange(
|
|
|
|
is_checked,
|
|
|
|
"id_realm_message_content_edit_limit_seconds",
|
|
|
|
true,
|
|
|
|
);
|
2022-04-12 09:56:58 +02:00
|
|
|
settings_ui.disable_sub_setting_onchange(
|
|
|
|
is_checked,
|
|
|
|
"id_realm_message_content_edit_limit_minutes",
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
settings_ui.disable_sub_setting_onchange(is_checked, "id_realm_edit_topic_policy", true);
|
|
|
|
}
|
|
|
|
|
2022-09-02 13:58:03 +02:00
|
|
|
function update_custom_value_input(property_name) {
|
|
|
|
const $dropdown_elem = $(`#id_${CSS.escape(property_name)}`);
|
|
|
|
const custom_input_elem_id = $dropdown_elem
|
|
|
|
.parent()
|
|
|
|
.find(".admin-realm-time-limit-input")
|
|
|
|
.attr("id");
|
|
|
|
|
|
|
|
const show_custom_limit_input = $dropdown_elem.val() === "custom_period";
|
|
|
|
change_element_block_display_property(custom_input_elem_id, show_custom_limit_input);
|
|
|
|
if (show_custom_limit_input) {
|
|
|
|
$(`#${CSS.escape(custom_input_elem_id)}`).val(
|
|
|
|
get_realm_time_limits_in_minutes(property_name),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-02 13:07:11 +02:00
|
|
|
function get_time_limit_dropdown_setting_value(property_name) {
|
|
|
|
if (page_params[property_name] === null) {
|
|
|
|
return "any_time";
|
|
|
|
}
|
|
|
|
|
|
|
|
const valid_limit_values = settings_config.time_limit_dropdown_values.map((x) => x.value);
|
|
|
|
if (valid_limit_values.includes(page_params[property_name])) {
|
|
|
|
return page_params[property_name].toString();
|
|
|
|
}
|
|
|
|
|
|
|
|
return "custom_period";
|
|
|
|
}
|
|
|
|
|
2022-09-08 12:19:16 +02:00
|
|
|
function set_time_limit_setting(property_name) {
|
|
|
|
const dropdown_elem_val = get_time_limit_dropdown_setting_value(property_name);
|
|
|
|
$(`#id_${CSS.escape(property_name)}`).val(dropdown_elem_val);
|
|
|
|
|
|
|
|
const $custom_input = $(`#id_${CSS.escape(property_name)}`)
|
|
|
|
.parent()
|
|
|
|
.find(".admin-realm-time-limit-input");
|
|
|
|
$custom_input.val(get_realm_time_limits_in_minutes(property_name));
|
|
|
|
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
2022-09-08 12:19:16 +02:00
|
|
|
$custom_input.attr("id"),
|
2022-09-02 13:07:11 +02:00
|
|
|
dropdown_elem_val === "custom_period",
|
2020-07-15 00:34:28 +02:00
|
|
|
);
|
2018-04-02 11:18:40 +02:00
|
|
|
}
|
|
|
|
|
2022-09-08 12:19:16 +02:00
|
|
|
function set_msg_edit_limit_dropdown() {
|
|
|
|
set_time_limit_setting("realm_message_content_edit_limit_seconds");
|
|
|
|
}
|
|
|
|
|
2022-08-18 16:14:05 +02:00
|
|
|
function message_delete_limit_setting_enabled(setting_value) {
|
2021-06-23 12:53:38 +02:00
|
|
|
// This function is used to check whether the time-limit setting
|
|
|
|
// should be enabled. The setting is disabled when delete_own_message_policy
|
|
|
|
// is set to 'admins only' as admins can delete messages irrespective of
|
|
|
|
// time limit.
|
2022-08-18 16:14:05 +02:00
|
|
|
if (setting_value === settings_config.common_message_policy_values.by_admins_only.code) {
|
2021-06-23 12:53:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-08-18 16:14:05 +02:00
|
|
|
function set_delete_own_message_policy_dropdown(setting_value) {
|
|
|
|
$("#id_realm_delete_own_message_policy").val(setting_value);
|
2021-06-08 13:45:14 +02:00
|
|
|
settings_ui.disable_sub_setting_onchange(
|
2022-08-18 16:14:05 +02:00
|
|
|
message_delete_limit_setting_enabled(setting_value),
|
2022-08-19 16:47:26 +02:00
|
|
|
"id_realm_message_content_delete_limit_seconds",
|
2021-06-08 13:45:14 +02:00
|
|
|
true,
|
|
|
|
);
|
2022-09-02 13:07:11 +02:00
|
|
|
const limit_setting_dropdown_value = get_time_limit_dropdown_setting_value(
|
|
|
|
"realm_message_content_delete_limit_seconds",
|
|
|
|
);
|
|
|
|
if (limit_setting_dropdown_value === "custom_period") {
|
2021-06-08 13:45:14 +02:00
|
|
|
settings_ui.disable_sub_setting_onchange(
|
2022-08-18 16:14:05 +02:00
|
|
|
message_delete_limit_setting_enabled(setting_value),
|
2021-06-08 13:45:14 +02:00
|
|
|
"id_realm_message_content_delete_limit_minutes",
|
|
|
|
true,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-26 09:12:10 +01:00
|
|
|
function set_msg_delete_limit_dropdown() {
|
2022-09-08 12:19:16 +02:00
|
|
|
set_time_limit_setting("realm_message_content_delete_limit_seconds");
|
2017-11-26 09:12:10 +01:00
|
|
|
}
|
|
|
|
|
2020-05-07 13:19:54 +02:00
|
|
|
function set_message_retention_setting_dropdown() {
|
|
|
|
const value = get_property_value("realm_message_retention_setting");
|
|
|
|
$("#id_realm_message_retention_setting").val(value);
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_message_retention_days",
|
|
|
|
value === "retain_for_period",
|
|
|
|
);
|
|
|
|
if (
|
|
|
|
get_property_value("realm_message_retention_days") ===
|
|
|
|
settings_config.retain_message_forever
|
|
|
|
) {
|
2020-07-15 01:29:15 +02:00
|
|
|
$("#id_realm_message_retention_days").val("");
|
2020-06-21 11:35:02 +02:00
|
|
|
}
|
2020-05-07 13:19:54 +02:00
|
|
|
}
|
|
|
|
|
2018-04-05 00:54:31 +02:00
|
|
|
function set_org_join_restrictions_dropdown() {
|
2019-11-02 00:06:25 +01:00
|
|
|
const value = get_property_value("realm_org_join_restrictions");
|
2018-04-05 00:54:31 +02:00
|
|
|
$("#id_realm_org_join_restrictions").val(value);
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"allowed_domains_label",
|
|
|
|
value === "only_selected_domain",
|
|
|
|
);
|
2018-04-05 00:54:31 +02:00
|
|
|
}
|
|
|
|
|
2019-01-14 14:04:08 +01:00
|
|
|
function set_message_content_in_email_notifications_visiblity() {
|
2020-02-03 13:03:53 +01:00
|
|
|
change_element_block_display_property(
|
2020-07-15 01:29:15 +02:00
|
|
|
"message_content_in_email_notifications_label",
|
2020-07-15 00:34:28 +02:00
|
|
|
page_params.realm_message_content_allowed_in_email_notifications,
|
|
|
|
);
|
2019-01-14 14:04:08 +01:00
|
|
|
}
|
|
|
|
|
2019-05-08 08:11:30 +02:00
|
|
|
function set_digest_emails_weekday_visibility() {
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_digest_weekday",
|
|
|
|
page_params.realm_digest_emails_enabled,
|
|
|
|
);
|
2019-05-08 08:11:30 +02:00
|
|
|
}
|
|
|
|
|
2021-11-19 10:29:39 +01:00
|
|
|
function set_create_web_public_stream_dropdown_visibility() {
|
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_create_web_public_stream_policy",
|
2022-05-26 11:43:42 +02:00
|
|
|
page_params.server_web_public_streams_enabled &&
|
|
|
|
page_params.zulip_plan_is_not_limited &&
|
|
|
|
page_params.realm_enable_spectator_access,
|
2021-11-19 10:29:39 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-07-20 14:51:48 +02:00
|
|
|
export function populate_realm_domains_label(realm_domains) {
|
2017-04-08 18:13:39 +02:00
|
|
|
if (!meta.loaded) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-15 00:34:28 +02:00
|
|
|
const domains_list = realm_domains.map((realm_domain) =>
|
|
|
|
realm_domain.allow_subdomains ? "*." + realm_domain.domain : realm_domain.domain,
|
js: Convert _.map(a, …) to a.map(…).
And convert the corresponding function expressions to arrow style
while we’re here.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitCallExpression(path) {
const { callee, arguments: args } = path.node;
if (
n.MemberExpression.check(callee) &&
!callee.computed &&
n.Identifier.check(callee.object) &&
callee.object.name === "_" &&
n.Identifier.check(callee.property) &&
callee.property.name === "map" &&
args.length === 2 &&
checkExpression(args[0]) &&
checkExpression(args[1])
) {
const [arr, fn] = args;
path.replace(
b.callExpression(b.memberExpression(arr, b.identifier("map")), [
n.FunctionExpression.check(fn) ||
n.ArrowFunctionExpression.check(fn)
? b.arrowFunctionExpression(
fn.params,
n.BlockStatement.check(fn.body) &&
fn.body.body.length === 1 &&
n.ReturnStatement.check(fn.body.body[0])
? fn.body.body[0].argument || b.identifier("undefined")
: fn.body
)
: fn,
])
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 02:43:49 +01:00
|
|
|
);
|
2020-07-15 01:29:15 +02:00
|
|
|
let domains = domains_list.join(", ");
|
2017-04-08 18:13:39 +02:00
|
|
|
if (domains.length === 0) {
|
2021-04-13 06:51:54 +02:00
|
|
|
domains = $t({defaultMessage: "None"});
|
2017-04-08 18:13:39 +02:00
|
|
|
}
|
2021-04-13 06:51:54 +02:00
|
|
|
$("#allowed_domains_label").text($t({defaultMessage: "Allowed domains: {domains}"}, {domains}));
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
|
|
|
|
2019-04-07 18:56:32 +02:00
|
|
|
function sort_object_by_key(obj) {
|
2020-02-09 04:45:48 +01:00
|
|
|
const keys = Object.keys(obj).sort();
|
2019-11-02 00:06:25 +01:00
|
|
|
const new_obj = {};
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
|
|
|
for (const key of keys) {
|
2019-04-07 18:56:32 +02:00
|
|
|
new_obj[key] = obj[key];
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
|
|
|
|
2019-04-07 18:56:32 +02:00
|
|
|
return new_obj;
|
|
|
|
}
|
2021-02-28 01:20:46 +01:00
|
|
|
|
|
|
|
export function populate_auth_methods(auth_methods) {
|
2017-04-08 18:13:39 +02:00
|
|
|
if (!meta.loaded) {
|
|
|
|
return;
|
|
|
|
}
|
2022-07-07 10:14:13 +02:00
|
|
|
const $auth_methods_list = $("#id_realm_authentication_methods").expectOne();
|
2019-04-07 18:56:32 +02:00
|
|
|
auth_methods = sort_object_by_key(auth_methods);
|
2019-11-02 00:06:25 +01:00
|
|
|
let rendered_auth_method_rows = "";
|
2020-02-06 04:27:31 +01:00
|
|
|
for (const [auth_method, value] of Object.entries(auth_methods)) {
|
2019-07-09 21:24:00 +02:00
|
|
|
rendered_auth_method_rows += render_settings_admin_auth_methods_list({
|
2019-04-06 08:43:34 +02:00
|
|
|
method: auth_method,
|
|
|
|
enabled: value,
|
2020-06-11 16:24:15 +02:00
|
|
|
is_owner: page_params.is_owner,
|
2022-07-07 10:12:16 +02:00
|
|
|
// The negated character class regexp serves as an allowlist - the replace() will
|
|
|
|
// remove *all* symbols *but* digits (\d) and lowecase letters (a-z),
|
|
|
|
// so that we can make assumptions on this string elsewhere in the code.
|
|
|
|
// As a result, the only two "incoming" assumptions on the auth method name are:
|
|
|
|
// 1) It contains at least one allowed symbol
|
|
|
|
// 2) No two auth method names are identical after this allowlist filtering
|
|
|
|
prefix: "id_authmethod" + auth_method.toLowerCase().replace(/[^\da-z]/g, "") + "_",
|
2019-04-06 08:43:34 +02:00
|
|
|
});
|
2020-02-06 04:27:31 +01:00
|
|
|
}
|
2022-07-07 10:14:13 +02:00
|
|
|
$auth_methods_list.html(rendered_auth_method_rows);
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2017-04-08 18:13:39 +02:00
|
|
|
|
2018-04-01 08:56:18 +02:00
|
|
|
function update_dependent_subsettings(property_name) {
|
js: Convert a.indexOf(…) !== -1 to a.includes(…).
Babel polyfills this for us for Internet Explorer.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import K from "ast-types/gen/kinds";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
recast.visit(ast, {
visitBinaryExpression(path) {
const { operator, left, right } = path.node;
if (
n.CallExpression.check(left) &&
n.MemberExpression.check(left.callee) &&
!left.callee.computed &&
n.Identifier.check(left.callee.property) &&
left.callee.property.name === "indexOf" &&
left.arguments.length === 1 &&
checkExpression(left.arguments[0]) &&
((["===", "!==", "==", "!=", ">", "<="].includes(operator) &&
n.UnaryExpression.check(right) &&
right.operator == "-" &&
n.Literal.check(right.argument) &&
right.argument.value === 1) ||
([">=", "<"].includes(operator) &&
n.Literal.check(right) &&
right.value === 0))
) {
const test = b.callExpression(
b.memberExpression(left.callee.object, b.identifier("includes")),
[left.arguments[0]]
);
path.replace(
["!==", "!=", ">", ">="].includes(operator)
? test
: b.unaryExpression("!", test)
);
changed = true;
}
this.traverse(path);
},
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-08 04:55:06 +01:00
|
|
|
if (simple_dropdown_properties.includes(property_name)) {
|
2020-02-03 12:40:36 +01:00
|
|
|
set_property_dropdown_value(property_name);
|
2021-05-09 23:36:58 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (property_name) {
|
|
|
|
case "realm_waiting_period_threshold":
|
|
|
|
set_realm_waiting_period_dropdown();
|
|
|
|
break;
|
|
|
|
case "realm_video_chat_provider":
|
|
|
|
set_video_chat_provider_dropdown();
|
|
|
|
break;
|
2022-04-12 09:56:58 +02:00
|
|
|
case "realm_allow_message_editing":
|
|
|
|
update_message_edit_sub_settings(page_params.realm_allow_message_editing);
|
|
|
|
break;
|
2021-05-09 23:36:58 +02:00
|
|
|
case "realm_message_retention_days":
|
|
|
|
set_message_retention_setting_dropdown();
|
|
|
|
break;
|
2021-06-08 13:45:14 +02:00
|
|
|
case "realm_delete_own_message_policy":
|
2022-08-18 16:14:05 +02:00
|
|
|
set_delete_own_message_policy_dropdown(page_params.realm_delete_own_message_policy);
|
2021-06-08 13:45:14 +02:00
|
|
|
break;
|
2021-05-09 23:36:58 +02:00
|
|
|
case "realm_org_join_restrictions":
|
|
|
|
set_org_join_restrictions_dropdown();
|
|
|
|
break;
|
|
|
|
case "realm_message_content_allowed_in_email_notifications":
|
|
|
|
set_message_content_in_email_notifications_visiblity();
|
|
|
|
break;
|
|
|
|
case "realm_digest_emails_enabled":
|
2021-08-26 13:04:03 +02:00
|
|
|
settings_notifications.set_enable_digest_emails_visibility(
|
2021-09-30 12:00:57 +02:00
|
|
|
settings_notifications.user_settings_panel,
|
2021-08-26 13:04:03 +02:00
|
|
|
);
|
2021-09-16 12:47:12 +02:00
|
|
|
settings_notifications.set_enable_digest_emails_visibility(
|
2021-09-30 12:00:57 +02:00
|
|
|
settings_realm_user_settings_defaults.realm_default_settings_panel,
|
2021-09-16 12:47:12 +02:00
|
|
|
);
|
2021-05-09 23:36:58 +02:00
|
|
|
set_digest_emails_weekday_visibility();
|
|
|
|
break;
|
2021-11-19 10:29:39 +01:00
|
|
|
case "realm_enable_spectator_access":
|
|
|
|
set_create_web_public_stream_dropdown_visibility();
|
|
|
|
break;
|
2018-03-29 12:52:57 +02:00
|
|
|
}
|
2018-04-01 08:49:10 +02:00
|
|
|
}
|
2018-03-29 12:52:57 +02:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export let default_code_language_widget = null;
|
|
|
|
export let notifications_stream_widget = null;
|
|
|
|
export let signup_notifications_stream_widget = null;
|
2021-02-28 01:38:10 +01:00
|
|
|
|
2021-09-28 09:09:52 +02:00
|
|
|
function discard_property_element_changes(elem, for_realm_default_settings) {
|
2022-01-25 11:36:19 +01:00
|
|
|
const $elem = $(elem);
|
|
|
|
const property_name = extract_property_name($elem, for_realm_default_settings);
|
2021-09-28 09:09:52 +02:00
|
|
|
const property_value = get_property_value(property_name, for_realm_default_settings);
|
2018-03-29 12:58:35 +02:00
|
|
|
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (property_name) {
|
|
|
|
case "realm_authentication_methods":
|
|
|
|
populate_auth_methods(property_value);
|
|
|
|
break;
|
|
|
|
case "realm_notifications_stream_id":
|
|
|
|
notifications_stream_widget.render(property_value);
|
|
|
|
break;
|
|
|
|
case "realm_signup_notifications_stream_id":
|
|
|
|
signup_notifications_stream_widget.render(property_value);
|
|
|
|
break;
|
|
|
|
case "realm_default_code_block_language":
|
|
|
|
default_code_language_widget.render(property_value);
|
|
|
|
break;
|
2022-05-03 08:06:40 +02:00
|
|
|
case "realm_default_language":
|
|
|
|
$("#org-notifications .language_selection_widget .language_selection_button span").attr(
|
|
|
|
"data-language-code",
|
|
|
|
property_value,
|
|
|
|
);
|
|
|
|
$("#org-notifications .language_selection_widget .language_selection_button span").text(
|
|
|
|
get_language_name(property_value),
|
|
|
|
);
|
|
|
|
break;
|
2021-09-28 09:09:52 +02:00
|
|
|
case "emojiset":
|
2022-08-12 22:41:06 +02:00
|
|
|
// Because this widget has a radio button structure, it
|
|
|
|
// needs custom reset code.
|
2022-01-25 11:36:19 +01:00
|
|
|
$elem
|
|
|
|
.closest(".org-subsection-parent")
|
2021-09-28 09:09:52 +02:00
|
|
|
.find(`.setting_emojiset_choice[value='${CSS.escape(property_value)}'`)
|
|
|
|
.prop("checked", true);
|
|
|
|
break;
|
2022-08-12 22:41:06 +02:00
|
|
|
case "user_list_style":
|
|
|
|
// Because this widget has a radio button structure, it
|
|
|
|
// needs custom reset code.
|
|
|
|
$elem
|
|
|
|
.closest(".org-subsection-parent")
|
2022-09-30 13:32:38 +02:00
|
|
|
.find(`.setting_user_list_style_choice[value='${CSS.escape(property_value)}']`)
|
2022-08-12 22:41:06 +02:00
|
|
|
.prop("checked", true);
|
|
|
|
break;
|
2021-12-02 15:56:34 +01:00
|
|
|
case "email_notifications_batching_period_seconds":
|
|
|
|
case "email_notification_batching_period_edit_minutes":
|
|
|
|
settings_notifications.set_notification_batching_ui(
|
|
|
|
$("#realm-user-default-settings"),
|
|
|
|
realm_user_settings_defaults.email_notifications_batching_period_seconds,
|
|
|
|
);
|
|
|
|
break;
|
2022-04-12 17:41:07 +02:00
|
|
|
case "realm_org_type":
|
|
|
|
set_input_element_value($elem, property_value);
|
|
|
|
// Remove 'unspecified' option (value=0) from realm_org_type
|
|
|
|
// dropdown menu options whenever page_params.realm_org_type
|
|
|
|
// returns another value.
|
|
|
|
if (property_value !== 0) {
|
|
|
|
$("#id_realm_org_type option[value=0]").remove();
|
|
|
|
}
|
|
|
|
break;
|
2022-09-02 13:07:11 +02:00
|
|
|
case "realm_message_content_edit_limit_seconds":
|
|
|
|
case "realm_message_content_delete_limit_seconds":
|
2022-09-08 12:19:16 +02:00
|
|
|
set_time_limit_setting(property_name);
|
2022-06-03 19:20:32 +02:00
|
|
|
break;
|
2021-05-09 23:36:58 +02:00
|
|
|
default:
|
|
|
|
if (property_value !== undefined) {
|
2022-01-25 11:36:19 +01:00
|
|
|
set_input_element_value($elem, property_value);
|
2021-05-09 23:36:58 +02:00
|
|
|
} else {
|
|
|
|
blueslip.error("Element refers to unknown property " + property_name);
|
|
|
|
}
|
2018-03-29 12:58:35 +02:00
|
|
|
}
|
|
|
|
|
2018-04-01 08:56:18 +02:00
|
|
|
update_dependent_subsettings(property_name);
|
2018-03-29 12:58:35 +02:00
|
|
|
}
|
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function sync_realm_settings(property) {
|
2022-11-02 19:39:58 +01:00
|
|
|
if (!meta.loaded) {
|
2018-03-29 13:15:50 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-05-13 10:39:36 +02:00
|
|
|
const value = page_params[`realm_${property}`];
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (property) {
|
|
|
|
case "notifications_stream_id":
|
|
|
|
notifications_stream_widget.render(value);
|
|
|
|
break;
|
|
|
|
case "signup_notifications_stream_id":
|
|
|
|
signup_notifications_stream_widget.render(value);
|
|
|
|
break;
|
|
|
|
case "default_code_block_language":
|
|
|
|
default_code_language_widget.render(value);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (property) {
|
|
|
|
case "emails_restricted_to_domains":
|
|
|
|
case "disallow_disposable_email_addresses":
|
|
|
|
property = "org_join_restrictions";
|
|
|
|
break;
|
2020-07-15 01:29:15 +02:00
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
const $element = $(`#id_realm_${CSS.escape(property)}`);
|
|
|
|
if ($element.length) {
|
|
|
|
discard_property_element_changes($element);
|
2018-03-29 13:15:50 +02:00
|
|
|
}
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2019-03-07 15:18:10 +01:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function change_save_button_state($element, state) {
|
2022-09-15 13:22:47 +02:00
|
|
|
function show_hide_element($element, show, fadeout_delay, fadeout_callback) {
|
2019-03-07 15:18:10 +01:00
|
|
|
if (show) {
|
2020-07-15 01:29:15 +02:00
|
|
|
$element.removeClass("hide").addClass(".show").fadeIn(300);
|
2019-03-07 15:18:10 +01:00
|
|
|
return;
|
|
|
|
}
|
2020-07-02 01:45:54 +02:00
|
|
|
setTimeout(() => {
|
2022-09-15 13:22:47 +02:00
|
|
|
$element.fadeOut(300, fadeout_callback);
|
2019-03-07 15:18:10 +01:00
|
|
|
}, fadeout_delay);
|
|
|
|
}
|
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
const $saveBtn = $element.find(".save-button");
|
2020-09-01 07:54:40 +02:00
|
|
|
const $textEl = $saveBtn.find(".save-discard-widget-button-text");
|
2019-03-07 15:18:10 +01:00
|
|
|
|
2018-03-29 02:02:01 +02:00
|
|
|
if (state !== "saving") {
|
2020-07-15 01:29:15 +02:00
|
|
|
$saveBtn.removeClass("saving");
|
2018-03-29 02:02:01 +02:00
|
|
|
}
|
2019-03-07 15:18:10 +01:00
|
|
|
|
|
|
|
if (state === "discarded") {
|
2022-09-15 13:22:47 +02:00
|
|
|
show_hide_element($element, false, 0, () =>
|
|
|
|
enable_or_disable_save_button($element.closest(".org-subsection-parent")),
|
|
|
|
);
|
2019-03-07 15:18:10 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-02 00:06:25 +01:00
|
|
|
let button_text;
|
|
|
|
let data_status;
|
|
|
|
let is_show;
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (state) {
|
|
|
|
case "unsaved":
|
|
|
|
button_text = $t({defaultMessage: "Save changes"});
|
|
|
|
data_status = "unsaved";
|
|
|
|
is_show = true;
|
|
|
|
|
|
|
|
$element.find(".discard-button").show();
|
|
|
|
break;
|
|
|
|
case "saved":
|
|
|
|
button_text = $t({defaultMessage: "Save changes"});
|
|
|
|
data_status = "";
|
|
|
|
is_show = false;
|
|
|
|
break;
|
|
|
|
case "saving":
|
|
|
|
button_text = $t({defaultMessage: "Saving"});
|
|
|
|
data_status = "saving";
|
|
|
|
is_show = true;
|
|
|
|
|
|
|
|
$element.find(".discard-button").hide();
|
|
|
|
$saveBtn.addClass("saving");
|
|
|
|
break;
|
|
|
|
case "failed":
|
|
|
|
button_text = $t({defaultMessage: "Save changes"});
|
|
|
|
data_status = "failed";
|
|
|
|
is_show = true;
|
|
|
|
break;
|
|
|
|
case "succeeded":
|
|
|
|
button_text = $t({defaultMessage: "Saved"});
|
|
|
|
data_status = "saved";
|
|
|
|
is_show = false;
|
|
|
|
break;
|
2019-03-09 08:34:53 +01:00
|
|
|
}
|
|
|
|
|
2019-03-07 15:18:10 +01:00
|
|
|
$textEl.text(button_text);
|
|
|
|
$saveBtn.attr("data-status", data_status);
|
2022-09-15 13:22:47 +02:00
|
|
|
if (state === "unsaved") {
|
|
|
|
enable_or_disable_save_button($element.closest(".org-subsection-parent"));
|
|
|
|
}
|
2019-03-07 15:18:10 +01:00
|
|
|
show_hide_element($element, is_show, 800);
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2018-03-29 02:02:01 +02:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
export function save_organization_settings(data, $save_button, patch_url) {
|
|
|
|
const $subsection_parent = $save_button.closest(".org-subsection-parent");
|
|
|
|
const $save_btn_container = $subsection_parent.find(".save-button-controls");
|
|
|
|
const $failed_alert_elem = $subsection_parent.find(".subsection-failed-status p");
|
|
|
|
change_save_button_state($save_btn_container, "saving");
|
2021-09-28 15:56:47 +02:00
|
|
|
channel.patch({
|
2021-09-29 11:25:28 +02:00
|
|
|
url: patch_url,
|
2021-09-28 15:56:47 +02:00
|
|
|
data,
|
|
|
|
success() {
|
2022-01-25 11:36:19 +01:00
|
|
|
$failed_alert_elem.hide();
|
|
|
|
change_save_button_state($save_btn_container, "succeeded");
|
2021-09-28 15:56:47 +02:00
|
|
|
},
|
|
|
|
error(xhr) {
|
2022-01-25 11:36:19 +01:00
|
|
|
change_save_button_state($save_btn_container, "failed");
|
|
|
|
$save_button.hide();
|
|
|
|
ui_report.error($t_html({defaultMessage: "Save failed"}), xhr, $failed_alert_elem);
|
2021-09-28 15:56:47 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
function get_input_type($input_elem, input_type) {
|
2020-05-11 15:11:29 +02:00
|
|
|
if (["boolean", "string", "number"].includes(input_type)) {
|
|
|
|
return input_type;
|
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
return $input_elem.data("setting-widget-type");
|
2020-05-11 15:11:29 +02:00
|
|
|
}
|
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function get_input_element_value(input_elem, input_type) {
|
2022-01-25 11:36:19 +01:00
|
|
|
const $input_elem = $(input_elem);
|
|
|
|
input_type = get_input_type($input_elem, input_type);
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (input_type) {
|
|
|
|
case "boolean":
|
2022-01-25 11:36:19 +01:00
|
|
|
return $input_elem.prop("checked");
|
2021-05-09 23:36:58 +02:00
|
|
|
case "string":
|
2022-01-25 11:36:19 +01:00
|
|
|
return $input_elem.val().trim();
|
2021-05-09 23:36:58 +02:00
|
|
|
case "number":
|
2022-01-25 11:36:19 +01:00
|
|
|
return Number.parseInt($input_elem.val().trim(), 10);
|
2021-09-28 09:09:52 +02:00
|
|
|
case "radio-group":
|
2022-01-25 11:36:19 +01:00
|
|
|
if ($input_elem.prop("checked")) {
|
|
|
|
return $input_elem.val().trim();
|
2021-09-28 09:09:52 +02:00
|
|
|
}
|
|
|
|
return undefined;
|
2022-08-25 10:02:54 +02:00
|
|
|
case "time-limit":
|
2022-09-20 16:40:19 +02:00
|
|
|
return get_time_limit_setting_value($input_elem);
|
2021-05-09 23:36:58 +02:00
|
|
|
default:
|
|
|
|
return undefined;
|
2020-03-22 16:13:50 +01:00
|
|
|
}
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-03-22 16:13:50 +01:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
export function set_input_element_value($input_elem, value) {
|
|
|
|
const input_type = get_input_type($input_elem, typeof value);
|
2020-05-11 15:44:01 +02:00
|
|
|
if (input_type) {
|
2020-07-15 01:29:15 +02:00
|
|
|
if (input_type === "boolean") {
|
2022-01-25 11:36:19 +01:00
|
|
|
return $input_elem.prop("checked", value);
|
2021-05-09 23:36:58 +02:00
|
|
|
} else if (input_type === "string" || input_type === "number") {
|
2022-01-25 11:36:19 +01:00
|
|
|
return $input_elem.val(value);
|
2020-05-11 15:44:01 +02:00
|
|
|
}
|
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
blueslip.error(`Failed to set value of property ${extract_property_name($input_elem)}`);
|
2020-09-24 07:50:36 +02:00
|
|
|
return undefined;
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-05-11 15:44:01 +02:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function set_up() {
|
|
|
|
build_page();
|
|
|
|
maybe_disable_widgets();
|
|
|
|
}
|
2018-12-08 18:16:37 +01:00
|
|
|
|
2022-07-07 10:14:13 +02:00
|
|
|
function get_auth_method_list_data() {
|
2020-04-10 04:43:37 +02:00
|
|
|
const new_auth_methods = {};
|
2022-07-07 10:12:16 +02:00
|
|
|
const $auth_method_rows = $("#id_realm_authentication_methods").find("div.method_row");
|
2020-04-10 04:43:37 +02:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
for (const method_row of $auth_method_rows) {
|
2020-07-15 00:34:28 +02:00
|
|
|
new_auth_methods[$(method_row).data("method")] = $(method_row)
|
|
|
|
.find("input")
|
|
|
|
.prop("checked");
|
2020-04-10 04:43:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return new_auth_methods;
|
|
|
|
}
|
|
|
|
|
2022-09-20 16:40:19 +02:00
|
|
|
function get_time_limit_setting_value($input_elem, for_api_data = true) {
|
2022-08-25 10:02:54 +02:00
|
|
|
const select_elem_val = $input_elem.val();
|
|
|
|
|
|
|
|
if (select_elem_val === "any_time") {
|
2022-09-02 13:07:11 +02:00
|
|
|
// "unlimited" is sent to API when a user wants to set the setting to
|
|
|
|
// "Any time" and the message_content_edit_limit_seconds field is "null"
|
|
|
|
// for that case.
|
|
|
|
if (!for_api_data) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-08-25 10:02:54 +02:00
|
|
|
return JSON.stringify("unlimited");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (select_elem_val !== "custom_period") {
|
|
|
|
return Number.parseInt(select_elem_val, 10);
|
|
|
|
}
|
|
|
|
|
2022-09-02 13:07:11 +02:00
|
|
|
const $custom_input_elem = $input_elem.parent().find(".admin-realm-time-limit-input");
|
|
|
|
if ($custom_input_elem.val().length === 0) {
|
|
|
|
// This handles the case where the initial setting value is "Any time" and then
|
|
|
|
// dropdown is changed to "Custom" where the input box is empty initially and
|
|
|
|
// thus we do not show the save-discard widget until something is typed in the
|
|
|
|
// input box.
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return parse_time_limit($custom_input_elem);
|
2022-08-25 10:02:54 +02:00
|
|
|
}
|
|
|
|
|
2021-09-28 09:09:52 +02:00
|
|
|
function check_property_changed(elem, for_realm_default_settings) {
|
2022-01-25 11:36:19 +01:00
|
|
|
const $elem = $(elem);
|
|
|
|
const property_name = extract_property_name($elem, for_realm_default_settings);
|
2021-09-28 09:09:52 +02:00
|
|
|
let current_val = get_property_value(property_name, for_realm_default_settings);
|
2022-08-25 15:44:55 +02:00
|
|
|
let proposed_val;
|
2020-04-10 04:43:37 +02:00
|
|
|
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (property_name) {
|
|
|
|
case "realm_authentication_methods":
|
|
|
|
current_val = sort_object_by_key(current_val);
|
|
|
|
current_val = JSON.stringify(current_val);
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = get_auth_method_list_data();
|
|
|
|
proposed_val = JSON.stringify(proposed_val);
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
|
|
|
case "realm_notifications_stream_id":
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = Number.parseInt(notifications_stream_widget.value(), 10);
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
|
|
|
case "realm_signup_notifications_stream_id":
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = Number.parseInt(signup_notifications_stream_widget.value(), 10);
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
|
|
|
case "realm_default_code_block_language":
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = default_code_language_widget.value();
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
2021-12-02 15:56:34 +01:00
|
|
|
case "email_notifications_batching_period_seconds":
|
2022-09-20 19:49:30 +02:00
|
|
|
proposed_val = get_time_limit_setting_value($elem, false);
|
2021-12-02 15:56:34 +01:00
|
|
|
break;
|
2022-08-25 10:02:54 +02:00
|
|
|
case "realm_message_content_edit_limit_seconds":
|
|
|
|
case "realm_message_content_delete_limit_seconds":
|
2022-09-20 16:40:19 +02:00
|
|
|
proposed_val = get_time_limit_setting_value($elem, false);
|
2022-08-22 12:19:50 +02:00
|
|
|
break;
|
2022-05-03 08:06:40 +02:00
|
|
|
case "realm_default_language":
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = $(
|
2022-05-03 08:06:40 +02:00
|
|
|
"#org-notifications .language_selection_widget .language_selection_button span",
|
|
|
|
).attr("data-language-code");
|
|
|
|
break;
|
2021-05-09 23:36:58 +02:00
|
|
|
default:
|
|
|
|
if (current_val !== undefined) {
|
2022-08-25 15:44:55 +02:00
|
|
|
proposed_val = get_input_element_value($elem, typeof current_val);
|
2021-05-09 23:36:58 +02:00
|
|
|
} else {
|
|
|
|
blueslip.error("Element refers to unknown property " + property_name);
|
|
|
|
}
|
2020-04-10 04:43:37 +02:00
|
|
|
}
|
2022-08-25 15:44:55 +02:00
|
|
|
return current_val !== proposed_val;
|
2020-04-10 04:43:37 +02:00
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
export function save_discard_widget_status_handler($subsection, for_realm_default_settings) {
|
|
|
|
$subsection.find(".subsection-failed-status p").hide();
|
|
|
|
$subsection.find(".save-button").show();
|
|
|
|
const properties_elements = get_subsection_property_elements($subsection);
|
2021-01-23 02:36:54 +01:00
|
|
|
const show_change_process_button = properties_elements.some((elem) =>
|
2021-09-28 09:09:52 +02:00
|
|
|
check_property_changed(elem, for_realm_default_settings),
|
2021-01-23 02:36:54 +01:00
|
|
|
);
|
2020-04-10 04:43:37 +02:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
const $save_btn_controls = $subsection.find(".subsection-header .save-button-controls");
|
2020-04-10 04:43:37 +02:00
|
|
|
const button_state = show_change_process_button ? "unsaved" : "discarded";
|
2022-01-25 11:36:19 +01:00
|
|
|
change_save_button_state($save_btn_controls, button_state);
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-04-10 06:30:14 +02:00
|
|
|
|
2021-02-28 01:20:46 +01:00
|
|
|
export function init_dropdown_widgets() {
|
2021-04-04 15:15:18 +02:00
|
|
|
const streams = stream_settings_data.get_streams_for_settings_page();
|
2020-04-10 09:38:55 +02:00
|
|
|
const notification_stream_options = {
|
2020-12-22 00:37:36 +01:00
|
|
|
data: streams.map((x) => ({
|
|
|
|
name: x.name,
|
|
|
|
value: x.stream_id.toString(),
|
|
|
|
})),
|
2020-04-30 13:09:48 +02:00
|
|
|
on_update: () => {
|
2021-02-28 01:20:46 +01:00
|
|
|
save_discard_widget_status_handler($("#org-notifications"));
|
2020-04-30 13:09:48 +02:00
|
|
|
},
|
2021-04-13 06:51:54 +02:00
|
|
|
default_text: $t({defaultMessage: "Disabled"}),
|
2020-07-02 01:41:40 +02:00
|
|
|
render_text: (x) => `#${x}`,
|
2020-04-10 09:38:55 +02:00
|
|
|
null_value: -1,
|
|
|
|
};
|
2021-07-27 14:05:22 +02:00
|
|
|
notifications_stream_widget = new DropdownListWidget({
|
2020-12-22 00:37:36 +01:00
|
|
|
widget_name: "realm_notifications_stream_id",
|
|
|
|
value: page_params.realm_notifications_stream_id,
|
|
|
|
...notification_stream_options,
|
|
|
|
});
|
2022-04-28 00:12:57 +02:00
|
|
|
notifications_stream_widget.setup();
|
2021-07-27 14:05:22 +02:00
|
|
|
signup_notifications_stream_widget = new DropdownListWidget({
|
2020-12-22 00:37:36 +01:00
|
|
|
widget_name: "realm_signup_notifications_stream_id",
|
|
|
|
value: page_params.realm_signup_notifications_stream_id,
|
|
|
|
...notification_stream_options,
|
|
|
|
});
|
2022-04-28 00:12:57 +02:00
|
|
|
signup_notifications_stream_widget.setup();
|
2021-07-27 14:05:22 +02:00
|
|
|
default_code_language_widget = new DropdownListWidget({
|
2020-07-15 01:29:15 +02:00
|
|
|
widget_name: "realm_default_code_block_language",
|
2020-07-02 01:41:40 +02:00
|
|
|
data: Object.keys(pygments_data.langs).map((x) => ({
|
|
|
|
name: x,
|
|
|
|
value: x,
|
|
|
|
})),
|
2020-05-19 00:16:34 +02:00
|
|
|
value: page_params.realm_default_code_block_language,
|
2020-04-30 13:09:48 +02:00
|
|
|
on_update: () => {
|
2021-02-28 01:20:46 +01:00
|
|
|
save_discard_widget_status_handler($("#org-other-settings"));
|
2020-04-30 13:09:48 +02:00
|
|
|
},
|
2021-04-13 06:51:54 +02:00
|
|
|
default_text: $t({defaultMessage: "No language set"}),
|
2020-04-10 09:38:55 +02:00
|
|
|
});
|
2022-04-28 00:12:57 +02:00
|
|
|
default_code_language_widget.setup();
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|
2020-04-10 04:43:37 +02:00
|
|
|
|
2022-09-15 13:22:47 +02:00
|
|
|
function enable_or_disable_save_button($subsection_elem) {
|
2022-09-02 19:03:01 +02:00
|
|
|
const time_limit_settings = Array.from($subsection_elem.find(".time-limit-setting"));
|
|
|
|
let disable_save_btn = false;
|
|
|
|
for (const setting_elem of time_limit_settings) {
|
|
|
|
const dropdown_elem_val = $(setting_elem).find("select").val();
|
|
|
|
const custom_input_elem_val = Number.parseInt(
|
2022-09-23 12:23:06 +02:00
|
|
|
Number($(setting_elem).find(".admin-realm-time-limit-input").val()),
|
2022-09-02 19:03:01 +02:00
|
|
|
10,
|
|
|
|
);
|
|
|
|
|
|
|
|
disable_save_btn =
|
2022-09-03 09:12:40 +02:00
|
|
|
dropdown_elem_val === "custom_period" &&
|
2022-09-02 19:03:01 +02:00
|
|
|
(custom_input_elem_val <= 0 || Number.isNaN(custom_input_elem_val));
|
|
|
|
if (disable_save_btn) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$subsection_elem.find(".subsection-changes-save button").prop("disabled", disable_save_btn);
|
|
|
|
}
|
|
|
|
|
2021-09-28 09:09:52 +02:00
|
|
|
export function register_save_discard_widget_handlers(
|
2022-01-25 11:36:19 +01:00
|
|
|
$container,
|
2021-09-28 09:09:52 +02:00
|
|
|
patch_url,
|
|
|
|
for_realm_default_settings,
|
|
|
|
) {
|
2022-01-25 11:36:19 +01:00
|
|
|
$container.on("change input", "input, select, textarea", (e) => {
|
2018-03-14 23:50:17 +01:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
|
2019-08-22 01:48:38 +02:00
|
|
|
// This event handler detects whether after these input
|
|
|
|
// changes, any fields have different values from the current
|
|
|
|
// official values stored in the database and page_params. If
|
|
|
|
// they do, we transition to the "unsaved" state showing the
|
|
|
|
// save/discard widget; otherwise, we hide that widget (the
|
|
|
|
// "discarded" state).
|
2019-08-05 21:11:30 +02:00
|
|
|
|
|
|
|
if ($(e.target).hasClass("no-input-change-detection")) {
|
|
|
|
// This is to prevent input changes detection in elements
|
|
|
|
// within a subsection whose changes should not affect the
|
|
|
|
// visibility of the discard button
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-12-02 15:56:34 +01:00
|
|
|
if ($(e.target).hasClass("setting_email_notifications_batching_period_seconds")) {
|
|
|
|
const show_elem = $(e.target).val() === "custom_period";
|
|
|
|
change_element_block_display_property(
|
|
|
|
"realm_email_notification_batching_period_edit_minutes",
|
|
|
|
show_elem,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
const $subsection = $(e.target).closest(".org-subsection-parent");
|
|
|
|
save_discard_widget_status_handler($subsection, for_realm_default_settings);
|
2020-09-24 07:50:36 +02:00
|
|
|
return undefined;
|
2018-03-14 23:50:17 +01:00
|
|
|
});
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
$container.on("click", ".subsection-header .subsection-changes-discard button", (e) => {
|
2021-06-03 18:15:17 +02:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
for (const elem of get_subsection_property_elements(e.target)) {
|
2021-09-28 09:09:52 +02:00
|
|
|
discard_property_element_changes(elem, for_realm_default_settings);
|
2021-06-03 18:15:17 +02:00
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
const $save_btn_controls = $(e.target).closest(".save-button-controls");
|
|
|
|
change_save_button_state($save_btn_controls, "discarded");
|
2021-06-03 18:15:17 +02:00
|
|
|
});
|
2018-03-14 23:52:36 +01:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
parse_time_limit = function parse_time_limit($elem) {
|
2022-09-23 12:30:28 +02:00
|
|
|
return Math.floor(Number.parseFloat(Number($elem.val()), 10).toFixed(1) * 60);
|
2018-08-04 08:18:21 +02:00
|
|
|
};
|
2018-04-28 15:35:14 +02:00
|
|
|
|
2018-03-21 00:50:00 +01:00
|
|
|
function get_complete_data_for_subsection(subsection) {
|
2019-11-02 00:06:25 +01:00
|
|
|
let data = {};
|
2020-02-21 21:45:34 +01:00
|
|
|
|
2021-05-09 23:36:58 +02:00
|
|
|
switch (subsection) {
|
|
|
|
case "notifications":
|
|
|
|
data.notifications_stream_id = Number.parseInt(
|
|
|
|
notifications_stream_widget.value(),
|
|
|
|
10,
|
2020-07-15 00:34:28 +02:00
|
|
|
);
|
2021-05-09 23:36:58 +02:00
|
|
|
data.signup_notifications_stream_id = Number.parseInt(
|
|
|
|
signup_notifications_stream_widget.value(),
|
|
|
|
10,
|
2020-07-15 00:34:28 +02:00
|
|
|
);
|
2022-05-03 08:06:40 +02:00
|
|
|
data.default_language = $(
|
|
|
|
"#org-notifications .language_selection_widget .language_selection_button span",
|
|
|
|
).attr("data-language-code");
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
|
|
|
case "message_retention": {
|
|
|
|
const message_retention_setting_value = $(
|
|
|
|
"#id_realm_message_retention_setting",
|
|
|
|
).val();
|
|
|
|
if (message_retention_setting_value === "retain_forever") {
|
2021-08-02 18:43:08 +02:00
|
|
|
data.message_retention_days = JSON.stringify("unlimited");
|
2021-05-09 23:36:58 +02:00
|
|
|
} else {
|
|
|
|
data.message_retention_days = JSON.stringify(
|
|
|
|
get_input_element_value($("#id_realm_message_retention_days")),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
break;
|
2018-03-22 14:07:52 +01:00
|
|
|
}
|
2021-05-09 23:36:58 +02:00
|
|
|
case "other_settings": {
|
|
|
|
const code_block_language_value = default_code_language_widget.value();
|
|
|
|
// No need to JSON-encode, since this value is already a string.
|
|
|
|
data.default_code_block_language = code_block_language_value;
|
|
|
|
break;
|
2018-03-22 14:07:52 +01:00
|
|
|
}
|
2021-05-09 23:36:58 +02:00
|
|
|
case "org_join": {
|
|
|
|
const org_join_restrictions = $("#id_realm_org_join_restrictions").val();
|
|
|
|
switch (org_join_restrictions) {
|
|
|
|
case "only_selected_domain":
|
|
|
|
data.emails_restricted_to_domains = true;
|
|
|
|
data.disallow_disposable_email_addresses = false;
|
|
|
|
break;
|
|
|
|
case "no_disposable_email":
|
|
|
|
data.emails_restricted_to_domains = false;
|
|
|
|
data.disallow_disposable_email_addresses = true;
|
|
|
|
break;
|
|
|
|
case "no_restriction":
|
|
|
|
data.disallow_disposable_email_addresses = false;
|
|
|
|
data.emails_restricted_to_domains = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const waiting_period_threshold = $("#id_realm_waiting_period_setting").val();
|
|
|
|
switch (waiting_period_threshold) {
|
|
|
|
case "none":
|
|
|
|
data.waiting_period_threshold = 0;
|
|
|
|
break;
|
|
|
|
case "three_days":
|
|
|
|
data.waiting_period_threshold = 3;
|
|
|
|
break;
|
|
|
|
case "custom_days":
|
|
|
|
data.waiting_period_threshold = $(
|
|
|
|
"#id_realm_waiting_period_threshold",
|
|
|
|
).val();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2018-04-30 18:26:59 +02:00
|
|
|
}
|
2021-05-09 23:36:58 +02:00
|
|
|
case "auth_settings":
|
|
|
|
data = {};
|
2022-07-07 10:14:13 +02:00
|
|
|
data.authentication_methods = JSON.stringify(get_auth_method_list_data());
|
2021-05-09 23:36:58 +02:00
|
|
|
break;
|
2018-03-14 23:45:42 +01:00
|
|
|
}
|
2019-03-25 11:52:55 +01:00
|
|
|
return data;
|
2018-03-21 00:50:00 +01:00
|
|
|
}
|
|
|
|
|
2019-05-05 15:13:52 +02:00
|
|
|
function populate_data_for_request(subsection) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const data = {};
|
|
|
|
const properties_elements = get_subsection_property_elements(subsection);
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
for (const input_elem of properties_elements) {
|
|
|
|
const $input_elem = $(input_elem);
|
|
|
|
if (check_property_changed($input_elem, for_realm_default_settings)) {
|
|
|
|
const input_value = get_input_element_value($input_elem);
|
2020-05-07 13:30:24 +02:00
|
|
|
if (input_value !== undefined) {
|
2021-09-28 09:09:52 +02:00
|
|
|
let property_name;
|
|
|
|
if (for_realm_default_settings) {
|
|
|
|
// We use the name attribute, rather than the ID attribute,
|
|
|
|
// for realm_user_default_settings. This is because the
|
|
|
|
// display/notification settings elements do not always have
|
|
|
|
// IDs, and also the emojiset input is not compatible with the
|
|
|
|
// ID approach.
|
2022-01-25 11:36:19 +01:00
|
|
|
property_name = $input_elem.attr("name");
|
2022-07-07 10:12:16 +02:00
|
|
|
} else if ($input_elem.attr("id").startsWith("id_authmethod")) {
|
|
|
|
// Authentication Method component IDs include authentication method name
|
|
|
|
// for uniqueness, anchored to "id_authmethod" prefix, e.g. "id_authmethodapple_<property_name>".
|
|
|
|
// We need to strip that whole construct down to extract the actual property name.
|
|
|
|
// The [\da-z]+ part of the regexp covers the auth method name itself.
|
|
|
|
// We assume it's not an empty string and can contain only digits and lowercase ASCII letters,
|
|
|
|
// this is ensured by a respective allowlist-based filter in populate_auth_methods().
|
|
|
|
[, property_name] = /^id_authmethod[\da-z]+_(.*)$/.exec(
|
|
|
|
$input_elem.attr("id"),
|
|
|
|
);
|
2021-09-28 09:09:52 +02:00
|
|
|
} else {
|
2022-01-25 11:36:19 +01:00
|
|
|
[, property_name] = /^id_realm_(.*)$/.exec($input_elem.attr("id"));
|
2021-09-28 09:09:52 +02:00
|
|
|
}
|
2021-04-07 22:00:40 +02:00
|
|
|
data[property_name] = input_value;
|
2019-05-05 15:13:52 +02:00
|
|
|
}
|
|
|
|
}
|
js: Automatically convert _.each to for…of.
This commit was automatically generated by the following script,
followed by lint --fix and a few small manual lint-related cleanups.
import * as babelParser from "recast/parsers/babel";
import * as recast from "recast";
import * as tsParser from "recast/parsers/typescript";
import { builders as b, namedTypes as n } from "ast-types";
import { Context } from "ast-types/lib/path-visitor";
import K from "ast-types/gen/kinds";
import { NodePath } from "ast-types/lib/node-path";
import assert from "assert";
import fs from "fs";
import path from "path";
import process from "process";
const checkExpression = (node: n.Node): node is K.ExpressionKind =>
n.Expression.check(node);
const checkStatement = (node: n.Node): node is K.StatementKind =>
n.Statement.check(node);
for (const file of process.argv.slice(2)) {
console.log("Parsing", file);
const ast = recast.parse(fs.readFileSync(file, { encoding: "utf8" }), {
parser: path.extname(file) === ".ts" ? tsParser : babelParser,
});
let changed = false;
let inLoop = false;
let replaceReturn = false;
const visitLoop = (...args: string[]) =>
function(this: Context, path: NodePath) {
for (const arg of args) {
this.visit(path.get(arg));
}
const old = { inLoop };
inLoop = true;
this.visit(path.get("body"));
inLoop = old.inLoop;
return false;
};
recast.visit(ast, {
visitDoWhileStatement: visitLoop("test"),
visitExpressionStatement(path) {
const { expression, comments } = path.node;
let valueOnly;
if (
n.CallExpression.check(expression) &&
n.MemberExpression.check(expression.callee) &&
!expression.callee.computed &&
n.Identifier.check(expression.callee.object) &&
expression.callee.object.name === "_" &&
n.Identifier.check(expression.callee.property) &&
["each", "forEach"].includes(expression.callee.property.name) &&
[2, 3].includes(expression.arguments.length) &&
checkExpression(expression.arguments[0]) &&
(n.FunctionExpression.check(expression.arguments[1]) ||
n.ArrowFunctionExpression.check(expression.arguments[1])) &&
[1, 2].includes(expression.arguments[1].params.length) &&
n.Identifier.check(expression.arguments[1].params[0]) &&
((valueOnly = expression.arguments[1].params[1] === undefined) ||
n.Identifier.check(expression.arguments[1].params[1])) &&
(expression.arguments[2] === undefined ||
n.ThisExpression.check(expression.arguments[2]))
) {
const old = { inLoop, replaceReturn };
inLoop = false;
replaceReturn = true;
this.visit(
path
.get("expression")
.get("arguments")
.get(1)
.get("body")
);
inLoop = old.inLoop;
replaceReturn = old.replaceReturn;
const [right, { body, params }] = expression.arguments;
const loop = b.forOfStatement(
b.variableDeclaration("let", [
b.variableDeclarator(
valueOnly ? params[0] : b.arrayPattern([params[1], params[0]])
),
]),
valueOnly
? right
: b.callExpression(
b.memberExpression(right, b.identifier("entries")),
[]
),
checkStatement(body) ? body : b.expressionStatement(body)
);
loop.comments = comments;
path.replace(loop);
changed = true;
}
this.traverse(path);
},
visitForStatement: visitLoop("init", "test", "update"),
visitForInStatement: visitLoop("left", "right"),
visitForOfStatement: visitLoop("left", "right"),
visitFunction(path) {
this.visit(path.get("params"));
const old = { replaceReturn };
replaceReturn = false;
this.visit(path.get("body"));
replaceReturn = old.replaceReturn;
return false;
},
visitReturnStatement(path) {
if (replaceReturn) {
assert(!inLoop); // could use labeled continue if this ever fires
const { argument, comments } = path.node;
if (argument === null) {
const s = b.continueStatement();
s.comments = comments;
path.replace(s);
} else {
const s = b.expressionStatement(argument);
s.comments = comments;
path.replace(s, b.continueStatement());
}
return false;
}
this.traverse(path);
},
visitWhileStatement: visitLoop("test"),
});
if (changed) {
console.log("Writing", file);
fs.writeFileSync(file, recast.print(ast).code, { encoding: "utf8" });
}
}
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-02-06 06:19:47 +01:00
|
|
|
}
|
|
|
|
|
2019-05-05 15:13:52 +02:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
$container.on("click", ".subsection-header .subsection-changes-save button", (e) => {
|
2018-03-21 00:50:00 +01:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2022-01-25 11:36:19 +01:00
|
|
|
const $save_button = $(e.currentTarget);
|
|
|
|
const $subsection_elem = $save_button.closest(".org-subsection-parent");
|
2021-09-28 09:09:52 +02:00
|
|
|
let extra_data = {};
|
|
|
|
|
|
|
|
if (!for_realm_default_settings) {
|
|
|
|
// The organization settings system has some coupled
|
|
|
|
// fields that must be submitted together, which is
|
|
|
|
// managed by the get_complete_data_for_subsection function.
|
2022-01-25 11:36:19 +01:00
|
|
|
const [, subsection_id] = /^org-submit-(.*)$/.exec($save_button.attr("id"));
|
2022-03-02 04:10:04 +01:00
|
|
|
const subsection = subsection_id.replace(/-/g, "_");
|
2021-09-28 09:09:52 +02:00
|
|
|
extra_data = get_complete_data_for_subsection(subsection);
|
|
|
|
}
|
2018-03-21 00:50:00 +01:00
|
|
|
|
2020-02-09 04:15:38 +01:00
|
|
|
const data = {
|
2022-01-25 11:36:19 +01:00
|
|
|
...populate_data_for_request($subsection_elem),
|
2021-09-28 09:09:52 +02:00
|
|
|
...extra_data,
|
2020-02-09 04:15:38 +01:00
|
|
|
};
|
2022-01-25 11:36:19 +01:00
|
|
|
save_organization_settings(data, $save_button, patch_url);
|
2017-05-25 00:50:07 +02:00
|
|
|
});
|
2021-09-29 13:03:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export function build_page() {
|
|
|
|
meta.loaded = true;
|
|
|
|
|
|
|
|
loading.make_indicator($("#admin_page_auth_methods_loading_indicator"));
|
|
|
|
|
|
|
|
// Initialize all the dropdown list widgets.
|
|
|
|
init_dropdown_widgets();
|
|
|
|
// Populate realm domains
|
2021-07-20 14:51:48 +02:00
|
|
|
populate_realm_domains_label(page_params.realm_domains);
|
2021-09-29 13:03:35 +02:00
|
|
|
|
|
|
|
// Populate authentication methods table
|
|
|
|
populate_auth_methods(page_params.realm_authentication_methods);
|
|
|
|
|
|
|
|
for (const property_name of simple_dropdown_properties) {
|
|
|
|
set_property_dropdown_value(property_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
set_realm_waiting_period_dropdown();
|
|
|
|
set_video_chat_provider_dropdown();
|
|
|
|
set_giphy_rating_dropdown();
|
|
|
|
set_msg_edit_limit_dropdown();
|
|
|
|
set_msg_delete_limit_dropdown();
|
2022-08-18 16:14:05 +02:00
|
|
|
set_delete_own_message_policy_dropdown(page_params.realm_delete_own_message_policy);
|
2021-09-29 13:03:35 +02:00
|
|
|
set_message_retention_setting_dropdown();
|
|
|
|
set_org_join_restrictions_dropdown();
|
|
|
|
set_message_content_in_email_notifications_visiblity();
|
|
|
|
set_digest_emails_weekday_visibility();
|
2021-11-19 10:29:39 +01:00
|
|
|
set_create_web_public_stream_dropdown_visibility();
|
2021-09-29 13:03:35 +02:00
|
|
|
|
2021-09-28 09:09:52 +02:00
|
|
|
register_save_discard_widget_handlers($(".admin-realm-form"), "/json/realm", false);
|
2017-08-18 01:23:55 +02:00
|
|
|
|
2020-07-02 01:45:54 +02:00
|
|
|
$(".org-subsection-parent").on("keydown", "input", (e) => {
|
2018-03-22 14:16:53 +01:00
|
|
|
e.stopPropagation();
|
2022-09-28 08:27:24 +02:00
|
|
|
if (keydown_util.is_enter_event(e)) {
|
2018-03-22 14:16:53 +01:00
|
|
|
e.preventDefault();
|
2020-07-15 00:34:28 +02:00
|
|
|
$(e.target)
|
|
|
|
.closest(".org-subsection-parent")
|
2021-06-03 18:15:17 +02:00
|
|
|
.find(".subsection-changes-save button")
|
2020-07-20 21:24:26 +02:00
|
|
|
.trigger("click");
|
2018-03-22 14:16:53 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-09-02 13:58:03 +02:00
|
|
|
$("#id_realm_message_content_edit_limit_seconds").on("change", () => {
|
|
|
|
update_custom_value_input("realm_message_content_edit_limit_seconds");
|
2018-04-02 11:18:40 +02:00
|
|
|
});
|
|
|
|
|
2022-09-02 13:58:03 +02:00
|
|
|
$("#id_realm_message_content_delete_limit_seconds").on("change", () => {
|
|
|
|
update_custom_value_input("realm_message_content_delete_limit_seconds");
|
2017-11-26 09:12:10 +01:00
|
|
|
});
|
|
|
|
|
2020-07-20 21:26:58 +02:00
|
|
|
$("#id_realm_message_retention_setting").on("change", (e) => {
|
2020-05-07 13:19:54 +02:00
|
|
|
const message_retention_setting_dropdown_value = e.target.value;
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_message_retention_days",
|
|
|
|
message_retention_setting_dropdown_value === "retain_for_period",
|
|
|
|
);
|
2020-05-07 13:19:54 +02:00
|
|
|
});
|
|
|
|
|
2020-07-20 21:26:58 +02:00
|
|
|
$("#id_realm_waiting_period_setting").on("change", function () {
|
2019-11-02 00:06:25 +01:00
|
|
|
const waiting_period_threshold = this.value;
|
2020-07-15 00:34:28 +02:00
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_waiting_period_threshold",
|
|
|
|
waiting_period_threshold === "custom_days",
|
|
|
|
);
|
2018-01-05 15:34:10 +01:00
|
|
|
});
|
|
|
|
|
2022-02-20 12:06:19 +01:00
|
|
|
$("#id_realm_digest_emails_enabled").on("change", (e) => {
|
|
|
|
const digest_emails_enabled = $(e.target).is(":checked");
|
|
|
|
change_element_block_display_property(
|
|
|
|
"id_realm_digest_weekday",
|
|
|
|
digest_emails_enabled === true,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2020-07-20 21:26:58 +02:00
|
|
|
$("#id_realm_org_join_restrictions").on("change", (e) => {
|
2019-11-02 00:06:25 +01:00
|
|
|
const org_join_restrictions = e.target.value;
|
2022-01-25 11:36:19 +01:00
|
|
|
const $node = $("#allowed_domains_label").parent();
|
2020-07-15 01:29:15 +02:00
|
|
|
if (org_join_restrictions === "only_selected_domain") {
|
2022-01-25 11:36:19 +01:00
|
|
|
$node.show();
|
2020-02-08 06:23:59 +01:00
|
|
|
if (page_params.realm_domains.length === 0) {
|
2021-07-20 14:51:48 +02:00
|
|
|
settings_realm_domains.show_realm_domains_modal();
|
2018-04-05 00:54:31 +02:00
|
|
|
}
|
|
|
|
} else {
|
2022-01-25 11:36:19 +01:00
|
|
|
$node.hide();
|
2018-04-05 00:54:31 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2022-04-12 09:56:58 +02:00
|
|
|
$("#id_realm_allow_message_editing").on("change", (e) => {
|
|
|
|
const is_checked = $(e.target).prop("checked");
|
|
|
|
update_message_edit_sub_settings(is_checked);
|
|
|
|
});
|
|
|
|
|
2022-04-25 12:58:35 +02:00
|
|
|
$("#id_realm_delete_own_message_policy").on("change", (e) => {
|
|
|
|
const setting_value = Number.parseInt($(e.target).val(), 10);
|
2022-08-18 16:22:46 +02:00
|
|
|
set_delete_own_message_policy_dropdown(setting_value);
|
2022-04-25 12:58:35 +02:00
|
|
|
});
|
|
|
|
|
2020-07-20 21:26:58 +02:00
|
|
|
$("#id_realm_org_join_restrictions").on("click", (e) => {
|
2018-04-05 00:54:31 +02:00
|
|
|
// This prevents the disappearance of modal when there are
|
|
|
|
// no allowed domains otherwise it gets closed due to
|
|
|
|
// the click event handler attached to `#settings_overlay_container`
|
|
|
|
e.stopPropagation();
|
|
|
|
});
|
|
|
|
|
2021-07-20 14:51:48 +02:00
|
|
|
$("#show_realm_domains_modal").on("click", (e) => {
|
2017-04-08 18:13:39 +02:00
|
|
|
e.stopPropagation();
|
2021-07-20 14:51:48 +02:00
|
|
|
settings_realm_domains.show_realm_domains_modal();
|
2017-04-08 18:13:39 +02:00
|
|
|
});
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
function realm_icon_logo_upload_complete($spinner, $upload_text, $delete_button) {
|
|
|
|
$spinner.css({visibility: "hidden"});
|
|
|
|
$upload_text.show();
|
|
|
|
$delete_button.show();
|
2020-05-27 14:16:04 +02:00
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
function realm_icon_logo_upload_start($spinner, $upload_text, $delete_button) {
|
|
|
|
$spinner.css({visibility: "visible"});
|
|
|
|
$upload_text.hide();
|
|
|
|
$delete_button.hide();
|
2020-05-27 14:16:04 +02:00
|
|
|
}
|
|
|
|
|
2022-01-25 11:36:19 +01:00
|
|
|
function upload_realm_logo_or_icon($file_input, night, icon) {
|
2019-11-02 00:06:25 +01:00
|
|
|
const form_data = new FormData();
|
2020-06-17 08:02:33 +02:00
|
|
|
let widget;
|
2020-06-11 01:04:12 +02:00
|
|
|
let url;
|
2018-08-16 01:26:55 +02:00
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
form_data.append("csrfmiddlewaretoken", csrf_token);
|
2022-01-25 11:36:19 +01:00
|
|
|
for (const [i, file] of Array.prototype.entries.call($file_input[0].files)) {
|
2020-07-15 01:29:15 +02:00
|
|
|
form_data.append("file-" + i, file);
|
2020-02-08 01:53:49 +01:00
|
|
|
}
|
2020-06-11 01:04:12 +02:00
|
|
|
if (icon) {
|
2020-07-15 01:29:15 +02:00
|
|
|
url = "/json/realm/icon";
|
|
|
|
widget = "#realm-icon-upload-widget";
|
2019-01-27 08:25:10 +01:00
|
|
|
} else {
|
2020-06-11 01:04:12 +02:00
|
|
|
if (night) {
|
2020-07-15 01:29:15 +02:00
|
|
|
widget = "#realm-night-logo-upload-widget";
|
2020-06-11 01:04:12 +02:00
|
|
|
} else {
|
2020-07-15 01:29:15 +02:00
|
|
|
widget = "#realm-day-logo-upload-widget";
|
2020-06-11 01:04:12 +02:00
|
|
|
}
|
2020-07-15 01:29:15 +02:00
|
|
|
url = "/json/realm/logo";
|
|
|
|
form_data.append("night", JSON.stringify(night));
|
2019-01-27 08:25:10 +01:00
|
|
|
}
|
2022-01-25 11:36:19 +01:00
|
|
|
const $spinner = $(`${widget} .upload-spinner-background`).expectOne();
|
|
|
|
const $upload_text = $(`${widget} .image-upload-text`).expectOne();
|
|
|
|
const $delete_button = $(`${widget} .image-delete-button`).expectOne();
|
|
|
|
const $error_field = $(`${widget} .image_file_input_error`).expectOne();
|
|
|
|
realm_icon_logo_upload_start($spinner, $upload_text, $delete_button);
|
|
|
|
$error_field.hide();
|
2018-08-16 01:26:55 +02:00
|
|
|
channel.post({
|
2020-07-20 22:18:43 +02:00
|
|
|
url,
|
2018-08-16 01:26:55 +02:00
|
|
|
data: form_data,
|
|
|
|
cache: false,
|
|
|
|
processData: false,
|
|
|
|
contentType: false,
|
2020-07-20 22:18:43 +02:00
|
|
|
success() {
|
2022-01-25 11:36:19 +01:00
|
|
|
realm_icon_logo_upload_complete($spinner, $upload_text, $delete_button);
|
2018-08-16 01:26:55 +02:00
|
|
|
},
|
2020-07-20 22:18:43 +02:00
|
|
|
error(xhr) {
|
2022-01-25 11:36:19 +01:00
|
|
|
realm_icon_logo_upload_complete($spinner, $upload_text, $delete_button);
|
|
|
|
ui_report.error("", xhr, $error_field);
|
2018-08-16 01:26:55 +02:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2019-02-27 15:45:26 +01:00
|
|
|
|
2020-06-11 01:04:12 +02:00
|
|
|
realm_icon.build_realm_icon_widget(upload_realm_logo_or_icon, null, true);
|
2020-05-08 13:30:34 +02:00
|
|
|
if (page_params.zulip_plan_is_not_limited) {
|
2020-06-11 01:04:12 +02:00
|
|
|
realm_logo.build_realm_logo_widget(upload_realm_logo_or_icon, false);
|
|
|
|
realm_logo.build_realm_logo_widget(upload_realm_logo_or_icon, true);
|
2019-06-12 14:22:37 +02:00
|
|
|
}
|
2019-03-11 16:48:59 +01:00
|
|
|
|
2020-07-15 01:29:15 +02:00
|
|
|
$("#deactivate_realm_button").on("click", (e) => {
|
2021-05-26 19:31:13 +02:00
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
2018-01-30 14:58:50 +01:00
|
|
|
|
2021-05-26 19:31:13 +02:00
|
|
|
function do_deactivate_realm() {
|
|
|
|
channel.post({
|
|
|
|
url: "/json/realm/deactivate",
|
|
|
|
error(xhr) {
|
|
|
|
ui_report.error(
|
|
|
|
$t_html({defaultMessage: "Failed"}),
|
|
|
|
xhr,
|
|
|
|
$("#admin-realm-deactivation-status").expectOne(),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
});
|
2018-01-30 14:58:50 +01:00
|
|
|
}
|
2021-05-26 19:31:13 +02:00
|
|
|
|
|
|
|
const html_body = render_settings_deactivate_realm_modal();
|
|
|
|
|
|
|
|
confirm_dialog.launch({
|
|
|
|
html_heading: $t_html({defaultMessage: "Deactivate organization"}),
|
|
|
|
help_link: "/help/deactivate-your-organization",
|
|
|
|
html_body,
|
|
|
|
on_click: do_deactivate_realm,
|
2018-01-30 14:58:50 +01:00
|
|
|
});
|
|
|
|
});
|
2021-02-28 01:20:46 +01:00
|
|
|
}
|