mirror of
				https://github.com/zulip/zulip.git
				synced 2025-10-31 03:53:50 +00:00 
			
		
		
		
	Add input to filter in sortablejs config. This prevents drag and drog from being called on clicking input field. Also set preventOnFilter to false. This prevents disabling the default behaviour on the click event. Fixes #17619
		
			
				
	
	
		
			495 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			495 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import $ from "jquery";
 | |
| import {Sortable} from "sortablejs";
 | |
| 
 | |
| import render_admin_profile_field_list from "../templates/admin_profile_field_list.hbs";
 | |
| import render_settings_profile_field_choice from "../templates/settings/profile_field_choice.hbs";
 | |
| 
 | |
| import * as channel from "./channel";
 | |
| import * as loading from "./loading";
 | |
| import * as settings_ui from "./settings_ui";
 | |
| 
 | |
| const meta = {
 | |
|     loaded: false,
 | |
| };
 | |
| 
 | |
| export function maybe_disable_widgets() {
 | |
|     if (page_params.is_admin) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     $(".organization-box [data-name='profile-field-settings']")
 | |
|         .find("input, button, select")
 | |
|         .prop("disabled", true);
 | |
| }
 | |
| 
 | |
| let order = [];
 | |
| const field_types = page_params.custom_profile_field_types;
 | |
| 
 | |
| export function field_type_id_to_string(type_id) {
 | |
|     for (const field_type of Object.values(field_types)) {
 | |
|         if (field_type.id === type_id) {
 | |
|             // Few necessary modifications in field-type-name for
 | |
|             // table-list view of custom fields UI in org settings
 | |
|             if (field_type.name === "Date picker") {
 | |
|                 return "Date";
 | |
|             } else if (field_type.name === "Person picker") {
 | |
|                 return "Person";
 | |
|             }
 | |
|             return field_type.name;
 | |
|         }
 | |
|     }
 | |
|     return undefined;
 | |
| }
 | |
| 
 | |
| function update_profile_fields_table_element() {
 | |
|     const profile_fields_table = $("#admin_profile_fields_table").expectOne();
 | |
| 
 | |
|     // If there are no custom fields, hide the table headers at the top
 | |
|     if (page_params.custom_profile_fields.length < 1) {
 | |
|         profile_fields_table.hide();
 | |
|     } else {
 | |
|         profile_fields_table.show();
 | |
|     }
 | |
| }
 | |
| 
 | |
| function delete_profile_field(e) {
 | |
|     e.preventDefault();
 | |
|     e.stopPropagation();
 | |
| 
 | |
|     settings_ui.do_settings_change(
 | |
|         channel.del,
 | |
|         "/json/realm/profile_fields/" + encodeURIComponent($(this).attr("data-profile-field-id")),
 | |
|         {},
 | |
|         $("#admin-profile-field-status").expectOne(),
 | |
|     );
 | |
|     update_profile_fields_table_element();
 | |
| }
 | |
| 
 | |
| function read_choice_field_data_from_form(field_elem) {
 | |
|     const field_data = {};
 | |
|     let field_order = 1;
 | |
|     $(field_elem)
 | |
|         .find("div.choice-row")
 | |
|         .each(function () {
 | |
|             const text = $(this).find("input")[0].value;
 | |
|             if (text) {
 | |
|                 field_data[field_order - 1] = {text, order: field_order.toString()};
 | |
|                 field_order += 1;
 | |
|             }
 | |
|         });
 | |
| 
 | |
|     return field_data;
 | |
| }
 | |
| 
 | |
| function read_external_account_field_data(field_elem) {
 | |
|     const field_data = {};
 | |
|     field_data.subtype = $(field_elem).find("select[name=external_acc_field_type]").val();
 | |
|     if (field_data.subtype === "custom") {
 | |
|         field_data.url_pattern = $(field_elem).find("input[name=url_pattern]").val();
 | |
|     }
 | |
|     return field_data;
 | |
| }
 | |
| 
 | |
| function update_choice_delete_btn(container, display_flag) {
 | |
|     const no_of_choice_row = container.find(".choice-row").length;
 | |
| 
 | |
|     // Disable delete button if there only one choice row
 | |
|     // Enable choice delete button more one than once choice
 | |
|     if (no_of_choice_row === 1) {
 | |
|         if (display_flag === true) {
 | |
|             container.find(".choice-row .delete-choice").show();
 | |
|         } else {
 | |
|             container.find(".choice-row .delete-choice").hide();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| function create_choice_row(container) {
 | |
|     const context = {};
 | |
|     const row = render_settings_profile_field_choice(context);
 | |
|     $(container).append(row);
 | |
| }
 | |
| 
 | |
| function clear_form_data() {
 | |
|     $("#profile_field_name").val("").closest(".control-group").show();
 | |
|     $("#profile_field_hint").val("").closest(".control-group").show();
 | |
|     // Set default type "Short Text" in field type dropdown
 | |
|     $("#profile_field_type").val(field_types.SHORT_TEXT.id);
 | |
|     // Clear data from choice field form
 | |
|     $("#profile_field_choices").html("");
 | |
|     create_choice_row($("#profile_field_choices"));
 | |
|     update_choice_delete_btn($("#profile_field_choices"), false);
 | |
|     $("#profile_field_choices_row").hide();
 | |
|     // Clear external account field form
 | |
|     $("#custom_field_url_pattern").val("");
 | |
|     $("#custom_external_account_url_pattern").hide();
 | |
|     $("#profile_field_external_accounts").hide();
 | |
|     $("#profile_field_external_accounts_type").val(
 | |
|         $("#profile_field_external_accounts_type option:first").val(),
 | |
|     );
 | |
| }
 | |
| 
 | |
| function set_up_create_field_form() {
 | |
|     const field_elem = $("#profile_field_external_accounts");
 | |
|     const field_url_pattern_elem = $("#custom_external_account_url_pattern");
 | |
| 
 | |
|     if (Number.parseInt($("#profile_field_type").val(), 10) === field_types.EXTERNAL_ACCOUNT.id) {
 | |
|         field_elem.show();
 | |
|         if ($("#profile_field_external_accounts_type").val() === "custom") {
 | |
|             field_url_pattern_elem.show();
 | |
|             $("#profile_field_name").val("").closest(".control-group").show();
 | |
|             $("#profile_field_hint").val("").closest(".control-group").show();
 | |
|         } else {
 | |
|             field_url_pattern_elem.hide();
 | |
|             $("#profile_field_name").closest(".control-group").hide();
 | |
|             $("#profile_field_hint").closest(".control-group").hide();
 | |
|         }
 | |
|     } else {
 | |
|         $("#profile_field_name").closest(".control-group").show();
 | |
|         $("#profile_field_hint").closest(".control-group").show();
 | |
|         field_url_pattern_elem.hide();
 | |
|         field_elem.hide();
 | |
|     }
 | |
| }
 | |
| 
 | |
| function read_field_data_from_form(field_type_id, field_elem) {
 | |
|     // Only read field data if we are creating a choice field
 | |
|     // or external account field.
 | |
|     if (field_type_id === field_types.CHOICE.id) {
 | |
|         return read_choice_field_data_from_form(field_elem);
 | |
|     } else if (field_type_id === field_types.EXTERNAL_ACCOUNT.id) {
 | |
|         return read_external_account_field_data(field_elem);
 | |
|     }
 | |
|     return undefined;
 | |
| }
 | |
| 
 | |
| function create_profile_field(e) {
 | |
|     e.preventDefault();
 | |
|     e.stopPropagation();
 | |
| 
 | |
|     let field_data = {};
 | |
|     const field_type = $("#profile_field_type").val();
 | |
|     const opts = {
 | |
|         success_continuation: clear_form_data,
 | |
|     };
 | |
|     field_data = read_field_data_from_form(
 | |
|         Number.parseInt(field_type, 10),
 | |
|         $(".new-profile-field-form"),
 | |
|     );
 | |
| 
 | |
|     const form_data = {
 | |
|         name: $("#profile_field_name").val(),
 | |
|         field_type,
 | |
|         hint: $("#profile_field_hint").val(),
 | |
|         field_data: JSON.stringify(field_data),
 | |
|     };
 | |
| 
 | |
|     settings_ui.do_settings_change(
 | |
|         channel.post,
 | |
|         "/json/realm/profile_fields",
 | |
|         form_data,
 | |
|         $("#admin-add-profile-field-status").expectOne(),
 | |
|         opts,
 | |
|     );
 | |
|     update_profile_fields_table_element();
 | |
| }
 | |
| 
 | |
| function add_choice_row(e) {
 | |
|     if ($(e.target).parent().next().hasClass("choice-row")) {
 | |
|         return;
 | |
|     }
 | |
|     const choices_div = e.delegateTarget;
 | |
|     update_choice_delete_btn($(choices_div), true);
 | |
|     create_choice_row(choices_div);
 | |
| }
 | |
| 
 | |
| function delete_choice_row(e) {
 | |
|     const row = $(e.currentTarget).parent();
 | |
|     const container = row.parent();
 | |
|     row.remove();
 | |
|     update_choice_delete_btn(container, false);
 | |
| }
 | |
| 
 | |
| function get_profile_field_info(id) {
 | |
|     const info = {};
 | |
|     info.row = $(`tr.profile-field-row[data-profile-field-id='${CSS.escape(id)}']`);
 | |
|     info.form = $(`tr.profile-field-form[data-profile-field-id='${CSS.escape(id)}']`);
 | |
|     return info;
 | |
| }
 | |
| 
 | |
| function get_profile_field(id) {
 | |
|     return page_params.custom_profile_fields.find((field) => field.id === id);
 | |
| }
 | |
| 
 | |
| export function parse_field_choices_from_field_data(field_data) {
 | |
|     const choices = [];
 | |
|     for (const [value, choice] of Object.entries(field_data)) {
 | |
|         choices.push({
 | |
|             value,
 | |
|             text: choice.text,
 | |
|             order: choice.order,
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     return choices;
 | |
| }
 | |
| 
 | |
| function set_up_external_account_field_edit_form(field_elem, url_pattern_val) {
 | |
|     if (field_elem.form.find("select[name=external_acc_field_type]").val() === "custom") {
 | |
|         field_elem.form.find("input[name=url_pattern]").val(url_pattern_val);
 | |
|         field_elem.form.find(".custom_external_account_detail").show();
 | |
|         field_elem.form.find("input[name=name]").val("").closest(".control-group").show();
 | |
|         field_elem.form.find("input[name=hint]").val("").closest(".control-group").show();
 | |
|     } else {
 | |
|         field_elem.form.find("input[name=name]").closest(".control-group").hide();
 | |
|         field_elem.form.find("input[name=hint]").closest(".control-group").hide();
 | |
|         field_elem.form.find(".custom_external_account_detail").hide();
 | |
|     }
 | |
| }
 | |
| 
 | |
| function set_up_choices_field_edit_form(profile_field, field_data) {
 | |
|     // Re-render field choices in edit form to load initial choice data
 | |
|     const choice_list = profile_field.form.find(".edit_profile_field_choices_container");
 | |
|     choice_list.off();
 | |
|     choice_list.html("");
 | |
| 
 | |
|     const choices_data = parse_field_choices_from_field_data(field_data);
 | |
| 
 | |
|     for (const choice of choices_data) {
 | |
|         choice_list.append(
 | |
|             render_settings_profile_field_choice({
 | |
|                 text: choice.text,
 | |
|             }),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     // Add blank choice at last
 | |
|     create_choice_row(choice_list);
 | |
|     update_choice_delete_btn(choice_list, false);
 | |
|     Sortable.create(choice_list[0], {
 | |
|         onUpdate() {},
 | |
|         filter: "input",
 | |
|         preventOnFilter: false,
 | |
|     });
 | |
| }
 | |
| 
 | |
| function open_edit_form(e) {
 | |
|     const field_id = Number.parseInt($(e.currentTarget).attr("data-profile-field-id"), 10);
 | |
|     const profile_field = get_profile_field_info(field_id);
 | |
| 
 | |
|     profile_field.row.hide();
 | |
|     profile_field.form.show();
 | |
|     const field = get_profile_field(field_id);
 | |
|     // Set initial value in edit form
 | |
|     profile_field.form.find("input[name=name]").val(field.name);
 | |
|     profile_field.form.find("input[name=hint]").val(field.hint);
 | |
|     let field_data = {};
 | |
|     if (field.field_data) {
 | |
|         field_data = JSON.parse(field.field_data);
 | |
|     }
 | |
| 
 | |
|     if (Number.parseInt(field.type, 10) === field_types.CHOICE.id) {
 | |
|         set_up_choices_field_edit_form(profile_field, field_data);
 | |
|     }
 | |
| 
 | |
|     if (Number.parseInt(field.type, 10) === field_types.EXTERNAL_ACCOUNT.id) {
 | |
|         profile_field.form.find("select[name=external_acc_field_type]").val(field_data.subtype);
 | |
|         set_up_external_account_field_edit_form(profile_field, field_data.url_pattern);
 | |
|     }
 | |
| 
 | |
|     profile_field.form.find(".reset").on("click", () => {
 | |
|         profile_field.form.hide();
 | |
|         profile_field.row.show();
 | |
|     });
 | |
| 
 | |
|     profile_field.form.find(".submit").on("click", () => {
 | |
|         e.preventDefault();
 | |
|         e.stopPropagation();
 | |
| 
 | |
|         const profile_field_status = $("#admin-profile-field-status").expectOne();
 | |
| 
 | |
|         // For some reason jQuery's serialize() is not working with
 | |
|         // channel.patch even though it is supported by $.ajax.
 | |
|         const data = {};
 | |
| 
 | |
|         data.name = profile_field.form.find("input[name=name]").val();
 | |
|         data.hint = profile_field.form.find("input[name=hint]").val();
 | |
|         data.field_data = JSON.stringify(
 | |
|             read_field_data_from_form(Number.parseInt(field.type, 10), profile_field.form),
 | |
|         );
 | |
| 
 | |
|         settings_ui.do_settings_change(
 | |
|             channel.patch,
 | |
|             "/json/realm/profile_fields/" + field_id,
 | |
|             data,
 | |
|             profile_field_status,
 | |
|         );
 | |
|     });
 | |
| 
 | |
|     profile_field.form
 | |
|         .find(".edit_profile_field_choices_container")
 | |
|         .on("input", ".choice-row input", add_choice_row);
 | |
|     profile_field.form
 | |
|         .find(".edit_profile_field_choices_container")
 | |
|         .on("click", "button.delete-choice", delete_choice_row);
 | |
|     $(".profile_field_external_accounts_edit select").on("change", (e) => {
 | |
|         const field_id = Number.parseInt(
 | |
|             $(e.target).closest(".profile-field-form").attr("data-profile-field-id"),
 | |
|             10,
 | |
|         );
 | |
|         const field_form = get_profile_field_info(field_id);
 | |
|         set_up_external_account_field_edit_form(field_form, "");
 | |
|     });
 | |
| }
 | |
| 
 | |
| export function reset() {
 | |
|     meta.loaded = false;
 | |
| }
 | |
| 
 | |
| function update_field_order() {
 | |
|     order = [];
 | |
|     $(".profile-field-row").each(function () {
 | |
|         order.push(Number.parseInt($(this).attr("data-profile-field-id"), 10));
 | |
|     });
 | |
|     settings_ui.do_settings_change(
 | |
|         channel.patch,
 | |
|         "/json/realm/profile_fields",
 | |
|         {order: JSON.stringify(order)},
 | |
|         $("#admin-profile-field-status").expectOne(),
 | |
|     );
 | |
| }
 | |
| 
 | |
| export function populate_profile_fields(profile_fields_data) {
 | |
|     if (!meta.loaded) {
 | |
|         // If outside callers call us when we're not loaded, just
 | |
|         // exit and we'll draw the widgets again during set_up().
 | |
|         return;
 | |
|     }
 | |
|     do_populate_profile_fields(profile_fields_data);
 | |
| }
 | |
| 
 | |
| export function do_populate_profile_fields(profile_fields_data) {
 | |
|     // We should only call this internally or from tests.
 | |
|     const profile_fields_table = $("#admin_profile_fields_table").expectOne();
 | |
| 
 | |
|     profile_fields_table.find("tr.profile-field-row").remove(); // Clear all rows.
 | |
|     profile_fields_table.find("tr.profile-field-form").remove(); // Clear all rows.
 | |
|     order = [];
 | |
| 
 | |
|     for (const profile_field of profile_fields_data) {
 | |
|         order.push(profile_field.id);
 | |
|         let field_data = {};
 | |
|         if (profile_field.field_data) {
 | |
|             field_data = JSON.parse(profile_field.field_data);
 | |
|         }
 | |
|         let choices = [];
 | |
|         if (profile_field.type === field_types.CHOICE.id) {
 | |
|             choices = parse_field_choices_from_field_data(field_data);
 | |
|         }
 | |
| 
 | |
|         profile_fields_table.append(
 | |
|             render_admin_profile_field_list({
 | |
|                 profile_field: {
 | |
|                     id: profile_field.id,
 | |
|                     name: profile_field.name,
 | |
|                     hint: profile_field.hint,
 | |
|                     type: field_type_id_to_string(profile_field.type),
 | |
|                     choices,
 | |
|                     is_choice_field: profile_field.type === field_types.CHOICE.id,
 | |
|                     is_external_account_field:
 | |
|                         profile_field.type === field_types.EXTERNAL_ACCOUNT.id,
 | |
|                 },
 | |
|                 can_modify: page_params.is_admin,
 | |
|                 realm_default_external_accounts: page_params.realm_default_external_accounts,
 | |
|             }),
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     if (page_params.is_admin) {
 | |
|         const field_list = $("#admin_profile_fields_table")[0];
 | |
|         Sortable.create(field_list, {
 | |
|             onUpdate: update_field_order,
 | |
|             filter: "input",
 | |
|             preventOnFilter: false,
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     update_profile_fields_table_element();
 | |
|     loading.destroy_indicator($("#admin_page_profile_fields_loading_indicator"));
 | |
| }
 | |
| 
 | |
| function set_up_choices_field() {
 | |
|     create_choice_row("#profile_field_choices");
 | |
|     update_choice_delete_btn($("#profile_field_choices"), false);
 | |
| 
 | |
|     if (page_params.is_admin) {
 | |
|         const choice_list = $("#profile_field_choices")[0];
 | |
|         Sortable.create(choice_list, {
 | |
|             onUpdate() {},
 | |
|             filter: "input",
 | |
|             preventOnFilter: false,
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     const field_type = $("#profile_field_type").val();
 | |
| 
 | |
|     if (Number.parseInt(field_type, 10) !== field_types.CHOICE.id) {
 | |
|         // If 'Choice' type is already selected, show choice row.
 | |
|         $("#profile_field_choices_row").hide();
 | |
|     }
 | |
| 
 | |
|     $("#profile_field_type").on("change", (e) => {
 | |
|         const selected_field_id = Number.parseInt($(e.target).val(), 10);
 | |
|         if (selected_field_id === field_types.CHOICE.id) {
 | |
|             $("#profile_field_choices_row").show();
 | |
|         } else {
 | |
|             $("#profile_field_choices_row").hide();
 | |
|         }
 | |
|     });
 | |
| 
 | |
|     $("#profile_field_choices").on("input", ".choice-row input", add_choice_row);
 | |
|     $("#profile_field_choices").on("click", "button.delete-choice", delete_choice_row);
 | |
| }
 | |
| 
 | |
| function set_up_external_account_field() {
 | |
|     $("#profile_field_type").on("change", () => {
 | |
|         set_up_create_field_form();
 | |
|     });
 | |
| 
 | |
|     $("#profile_field_external_accounts_type").on("change", () => {
 | |
|         set_up_create_field_form();
 | |
|     });
 | |
| }
 | |
| 
 | |
| export function get_external_account_link(field) {
 | |
|     const field_subtype = field.field_data.subtype;
 | |
|     let field_url_pattern;
 | |
| 
 | |
|     if (field_subtype === "custom") {
 | |
|         field_url_pattern = field.field_data.url_pattern;
 | |
|     } else {
 | |
|         field_url_pattern = page_params.realm_default_external_accounts[field_subtype].url_pattern;
 | |
|     }
 | |
|     return field_url_pattern.replace("%(username)s", field.value);
 | |
| }
 | |
| 
 | |
| export function set_up() {
 | |
|     build_page();
 | |
|     maybe_disable_widgets();
 | |
| }
 | |
| 
 | |
| export function build_page() {
 | |
|     // create loading indicators
 | |
|     loading.make_indicator($("#admin_page_profile_fields_loading_indicator"));
 | |
|     // Populate profile_fields table
 | |
|     do_populate_profile_fields(page_params.custom_profile_fields);
 | |
|     meta.loaded = true;
 | |
| 
 | |
|     $("#admin_profile_fields_table").on("click", ".delete", delete_profile_field);
 | |
|     $("#profile-field-settings").on("click", "#add-custom-profile-field-btn", create_profile_field);
 | |
|     $("#admin_profile_fields_table").on("click", ".open-edit-form", open_edit_form);
 | |
|     set_up_choices_field();
 | |
|     set_up_external_account_field();
 | |
|     clear_form_data();
 | |
| }
 |