summaryrefslogtreecommitdiffstats
path: root/app/assets/javascripts/application-legacy.js
diff options
context:
space:
mode:
Diffstat (limited to 'app/assets/javascripts/application-legacy.js')
-rw-r--r--app/assets/javascripts/application-legacy.js1435
1 files changed, 1435 insertions, 0 deletions
diff --git a/app/assets/javascripts/application-legacy.js b/app/assets/javascripts/application-legacy.js
new file mode 100644
index 000000000..1219e1ef8
--- /dev/null
+++ b/app/assets/javascripts/application-legacy.js
@@ -0,0 +1,1435 @@
+/**
+ * Redmine - project management software
+ * Copyright (C) 2006- Jean-Philippe Lang
+ * This code is released under the GNU General Public License.
+ */
+
+function sanitizeHTML(string) {
+ var temp = document.createElement('span');
+ temp.textContent = string;
+ return temp.innerHTML;
+}
+
+function checkAll(id, checked) {
+ $('#'+id).find('input[type=checkbox]:enabled').prop('checked', checked);
+}
+
+function toggleCheckboxesBySelector(selector) {
+ var all_checked = true;
+ $(selector).each(function(index) {
+ if (!$(this).is(':checked')) { all_checked = false; }
+ });
+ $(selector).prop('checked', !all_checked).trigger('change');
+}
+
+function showAndScrollTo(id, focus) {
+ $('#'+id).show();
+ if (focus !== null) {
+ $('#'+focus).focus();
+ }
+ $('html, body').animate({scrollTop: $('#'+id).offset().top}, 100);
+}
+
+function toggleRowGroup(el) {
+ var tr = $(el).parents('tr').first();
+ var n = tr.next();
+ tr.toggleClass('open');
+ $(el).toggleClass('icon-expanded icon-collapsed');
+ toggleExpendCollapseIcon(el)
+ while (n.length && !n.hasClass('group')) {
+ n.toggle();
+ n = n.next('tr');
+ }
+}
+
+function toggleExpendCollapseIcon(el) {
+ const svg = el.getElementsByTagName('svg').item(0)
+
+ if (svg === null) {
+ return false;
+ }
+
+ if (el.classList.contains('icon-expanded')) {
+ updateSVGIcon(svg, 'angle-down')
+ svg.classList.remove('icon-rtl')
+ } else {
+ updateSVGIcon(svg, 'angle-right')
+ svg.classList.add('icon-rtl')
+ }
+}
+
+function updateSVGIcon(element, icon) {
+ const iconElement = element.getElementsByTagName("use").item(0)
+
+ if (iconElement === null) {
+ return false;
+ }
+
+ const iconPath = iconElement.getAttribute('href');
+ iconElement.setAttribute('href', iconPath.replace(/#.*$/g, "#icon--" + icon))
+}
+
+function createSVGIcon(icon) {
+ const clonedIcon = document.querySelector('#icon-copy-source svg').cloneNode(true);
+ updateSVGIcon(clonedIcon, icon);
+ return clonedIcon
+}
+
+function collapseAllRowGroups(el) {
+ var tbody = $(el).parents('tbody').first();
+ tbody.children('tr').each(function(index) {
+ if ($(this).hasClass('group')) {
+ $(this).removeClass('open');
+ var expander = $(this).find('.expander');
+ expander.switchClass('icon-expanded', 'icon-collapsed');
+ toggleExpendCollapseIcon(expander[0]);
+ } else {
+ $(this).hide();
+ }
+ });
+}
+
+function expandAllRowGroups(el) {
+ var tbody = $(el).parents('tbody').first();
+ tbody.children('tr').each(function(index) {
+ if ($(this).hasClass('group')) {
+ $(this).addClass('open');
+ var expander = $(this).find('.expander');
+ expander.switchClass('icon-collapsed', 'icon-expanded');
+ toggleExpendCollapseIcon(expander[0]);
+ } else {
+ $(this).show();
+ }
+ });
+}
+
+function toggleAllRowGroups(el) {
+ var tr = $(el).parents('tr').first();
+ if (tr.hasClass('open')) {
+ collapseAllRowGroups(el);
+ } else {
+ expandAllRowGroups(el);
+ }
+}
+
+function toggleFieldset(el) {
+ var fieldset = $(el).parents('fieldset').first();
+ fieldset.toggleClass('collapsed');
+ fieldset.children('legend').toggleClass('icon-expanded icon-collapsed');
+ toggleExpendCollapseIcon(fieldset.children('legend')[0])
+ fieldset.children('div').toggle();
+}
+
+function hideFieldset(el) {
+ var fieldset = $(el).parents('fieldset').first();
+ fieldset.toggleClass('collapsed');
+ fieldset.children('div').hide();
+}
+
+// columns selection
+function moveOptions(theSelFrom, theSelTo) {
+ $(theSelFrom).find('option:selected').detach().prop("selected", false).appendTo($(theSelTo));
+}
+
+function moveOptionUp(theSel) {
+ $(theSel).find('option:selected').each(function(){
+ $(this).prev(':not(:selected)').detach().insertAfter($(this));
+ });
+}
+
+function moveOptionTop(theSel) {
+ $(theSel).find('option:selected').detach().prependTo($(theSel));
+}
+
+function moveOptionDown(theSel) {
+ $($(theSel).find('option:selected').get().reverse()).each(function(){
+ $(this).next(':not(:selected)').detach().insertBefore($(this));
+ });
+}
+
+function moveOptionBottom(theSel) {
+ $(theSel).find('option:selected').detach().appendTo($(theSel));
+}
+
+function initFilters() {
+ $('#add_filter_select').change(function() {
+ addFilter($(this).val(), '', []);
+ });
+ $('#filters-table .field input[type=checkbox]').each(function() {
+ toggleFilter($(this).val());
+ });
+ $('#filters-table').on('click', '.field input[type=checkbox]', function() {
+ toggleFilter($(this).val());
+ });
+ $('#filters-table').on('keypress', 'input[type=text]', function(e) {
+ if (e.keyCode == 13) $(this).closest('form').submit();
+ });
+}
+
+function addFilter(field, operator, values) {
+ var fieldId = field.replace('.', '_');
+ var tr = $('#tr_'+fieldId);
+
+ var filterOptions = availableFilters[field];
+ if (!filterOptions) return;
+
+ if (filterOptions['remote'] && filterOptions['values'] == null) {
+ $.getJSON(filtersUrl, {'name': field}).done(function(data) {
+ filterOptions['values'] = data;
+ addFilter(field, operator, values) ;
+ });
+ return;
+ }
+
+ if (tr.length > 0) {
+ tr.show();
+ } else {
+ buildFilterRow(field, operator, values);
+ }
+ $('#cb_'+fieldId).prop('checked', true);
+ toggleFilter(field);
+ toggleMultiSelectIconInit();
+ $('#add_filter_select').val('').find('option').each(function() {
+ if ($(this).attr('value') == field) {
+ $(this).attr('disabled', true);
+ }
+ });
+}
+
+function buildFilterRow(field, operator, values) {
+ var fieldId = field.replace('.', '_');
+ var filterTable = $("#filters-table");
+ var filterOptions = availableFilters[field];
+ if (!filterOptions) return;
+ var operators = operatorByType[filterOptions['type']];
+ var filterValues = filterOptions['values'];
+ var select;
+
+ var tr = $('<div class="filter">').attr('id', 'tr_'+fieldId).html(
+ '<div class="field"><input checked="checked" id="cb_'+fieldId+'" name="f[]" value="'+field+'" type="checkbox"><label for="cb_'+fieldId+'"> '+filterOptions['name']+'</label></div>' +
+ '<div class="operator"><select id="operators_'+fieldId+'" name="op['+field+']"></select></div>' +
+ '<div class="values"></div>'
+ );
+ filterTable.append(tr);
+
+ select = tr.find('.operator select');
+ operators.forEach(function(op) {
+ var option = $('<option>').val(op).text(operatorLabels[op]);
+ if (op == operator) { option.prop('selected', true); }
+ select.append(option);
+ });
+ select.change(function(){ toggleOperator(field); });
+
+ switch (filterOptions['type']) {
+ case "list":
+ case "list_with_history":
+ case "list_optional":
+ case "list_optional_with_history":
+ case "list_status":
+ case "list_subprojects":
+ const iconType = values.length > 1 ? 'toggle-minus' : 'toggle-plus';
+ const iconSvg = createSVGIcon(iconType)
+
+ tr.find('.values').append(
+ $('<span>', { style: 'display:none;' }).append(
+ $('<select>', {
+ class: 'value',
+ id: `values_${fieldId}_1`,
+ name: `v[${field}][]`,
+ }),
+ '\n',
+ $('<span>', { class: `toggle-multiselect icon-only icon-${iconType}` }).append(iconSvg)
+ )
+ );
+ select = tr.find('.values select');
+ if (values.length > 1) { select.attr('multiple', true); }
+ filterValues.forEach(function(filterValue) {
+ var option = $('<option>');
+ if ($.isArray(filterValue)) {
+ option.val(filterValue[1]).text(filterValue[0]);
+ if ($.inArray(filterValue[1], values) > -1) {option.prop('selected', true);}
+ if (filterValue.length == 3) {
+ var optgroup = select.find('optgroup').filter(function(){return $(this).attr('label') == filterValue[2]});
+ if (!optgroup.length) {optgroup = $('<optgroup>').attr('label', filterValue[2]);}
+ option = optgroup.append(option);
+ }
+ } else {
+ option.val(filterValue).text(filterValue);
+ if ($.inArray(filterValue, values) > -1) {option.prop('selected', true);}
+ }
+ select.append(option);
+ });
+ break;
+ case "date":
+ case "date_past":
+ tr.find('.values').append(
+ '<span style="display:none;"><input type="date" name="v['+field+'][]" id="values_'+fieldId+'_1" size="10" class="value date_value" /></span>' +
+ ' <span style="display:none;"><input type="date" name="v['+field+'][]" id="values_'+fieldId+'_2" size="10" class="value date_value" /></span>' +
+ ' <span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="3" class="value" /> '+labelDayPlural+'</span>'
+ );
+ $('#values_'+fieldId+'_1').val(values[0]).datepickerFallback(datepickerOptions);
+ $('#values_'+fieldId+'_2').val(values[1]).datepickerFallback(datepickerOptions);
+ $('#values_'+fieldId).val(values[0]);
+ break;
+ case "string":
+ case "text":
+ case "search":
+ tr.find('.values').append(
+ '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="30" class="value" /></span>'
+ );
+ $('#values_'+fieldId).val(values[0]);
+ break;
+ case "relation":
+ tr.find('.values').append(
+ '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'" size="6" class="value" /></span>' +
+ '<span style="display:none;"><select class="value" name="v['+field+'][]" id="values_'+fieldId+'_1"></select></span>'
+ );
+ $('#values_'+fieldId).val(values[0]);
+ select = tr.find('.values select');
+ filterValues.forEach(function(filterValue) {
+ var option = $('<option>');
+ option.val(filterValue[1]).text(filterValue[0]);
+ if (values[0] == filterValue[1]) { option.prop('selected', true); }
+ select.append(option);
+ });
+ break;
+ case "integer":
+ case "float":
+ case "tree":
+ tr.find('.values').append(
+ '<span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_1" size="14" class="value" /></span>' +
+ ' <span style="display:none;"><input type="text" name="v['+field+'][]" id="values_'+fieldId+'_2" size="14" class="value" /></span>'
+ );
+ $('#values_'+fieldId+'_1').val(values[0]);
+ $('#values_'+fieldId+'_2').val(values[1]);
+ break;
+ }
+}
+
+function toggleFilter(field) {
+ var fieldId = field.replace('.', '_');
+ if ($('#cb_' + fieldId).is(':checked')) {
+ $("#operators_" + fieldId).show().removeAttr('disabled');
+ toggleOperator(field);
+ } else {
+ $("#operators_" + fieldId).hide().attr('disabled', true);
+ enableValues(field, []);
+ }
+}
+
+function enableValues(field, indexes) {
+ var fieldId = field.replace('.', '_');
+ $('#tr_'+fieldId+' .values .value').each(function(index) {
+ if ($.inArray(index, indexes) >= 0) {
+ $(this).removeAttr('disabled');
+ $(this).parents('span').first().show();
+ } else {
+ $(this).val('');
+ $(this).attr('disabled', true);
+ $(this).parents('span').first().hide();
+ }
+
+ if ($(this).hasClass('group')) {
+ $(this).addClass('open');
+ } else {
+ $(this).show();
+ }
+ });
+}
+
+function toggleOperator(field) {
+ var fieldId = field.replace('.', '_');
+ var operator = $("#operators_" + fieldId);
+ switch (operator.val()) {
+ case "!*":
+ case "*":
+ case "nd":
+ case "t":
+ case "ld":
+ case "nw":
+ case "w":
+ case "lw":
+ case "l2w":
+ case "nm":
+ case "m":
+ case "lm":
+ case "y":
+ case "o":
+ case "c":
+ case "*o":
+ case "!o":
+ enableValues(field, []);
+ break;
+ case "><":
+ enableValues(field, [0,1]);
+ break;
+ case "<t+":
+ case ">t+":
+ case "><t+":
+ case "t+":
+ case ">t-":
+ case "<t-":
+ case "><t-":
+ case "t-":
+ enableValues(field, [2]);
+ break;
+ case "=p":
+ case "=!p":
+ case "!p":
+ enableValues(field, [1]);
+ break;
+ default:
+ enableValues(field, [0]);
+ break;
+ }
+}
+
+function toggleMultiSelect(el) {
+ var isWorkflow = el.closest('.controller-workflows');
+ if (el.attr('multiple')) {
+ el.removeAttr('multiple');
+ if (isWorkflow) { el.find("option[value=all]").show(); }
+ el.attr('size', 1);
+ } else {
+ el.attr('multiple', true);
+ if (isWorkflow) { el.find("option[value=all]").attr("selected", false).hide(); }
+ if (el.children().length > 10)
+ el.attr('size', 10);
+ else
+ el.attr('size', 4);
+ }
+}
+
+function showTab(name, url) {
+ $('#tab-content-' + name).parent().find('.tab-content').hide();
+ $('#tab-content-' + name).show();
+ $('#tab-' + name).closest('.tabs').find('a').removeClass('selected');
+ $('#tab-' + name).addClass('selected');
+
+ replaceInHistory(url)
+
+ return false;
+}
+
+function showIssueHistory(journal, url) {
+ tab_content = $('#tab-content-history');
+ tab_content.parent().find('.tab-content').hide();
+ tab_content.show();
+ tab_content.parent().children('div.tabs').find('a').removeClass('selected');
+
+ $('#tab-' + journal).addClass('selected');
+
+ replaceInHistory(url)
+
+ switch(journal) {
+ case 'notes':
+ tab_content.find('.journal').show();
+ tab_content.find('.journal:not(.has-notes)').hide();
+ tab_content.find('.journal .wiki').show();
+ tab_content.find('.journal .journal-actions > *').show();
+
+ // always show thumbnails in notes tab
+ var thumbnails = tab_content.find('.journal .thumbnails');
+ thumbnails.show();
+ // show journals without notes, but with thumbnails
+ thumbnails.parents('.journal').show();
+ break;
+ case 'properties':
+ tab_content.find('.journal').show();
+ tab_content.find('.journal:not(.has-details)').hide();
+ tab_content.find('.journal .wiki').hide();
+ tab_content.find('.journal .thumbnails').hide();
+ tab_content.find('.journal .journal-actions > *').hide();
+ // Show reaction button in properties tab
+ tab_content.find('.journal .journal-actions .reaction-button-wrapper').show();
+ break;
+ default:
+ tab_content.find('.journal').show();
+ tab_content.find('.journal .wiki').show();
+ tab_content.find('.journal .thumbnails').show();
+ tab_content.find('.journal .journal-actions > *').show();
+ }
+
+ return false;
+}
+
+function getRemoteTab(name, remote_url, url, load_always) {
+ load_always = load_always || false;
+ var tab_content = $('#tab-content-' + name);
+
+ tab_content.parent().find('.tab-content').hide();
+ tab_content.parent().children('div.tabs').find('a').removeClass('selected');
+ $('#tab-' + name).addClass('selected');
+
+ replaceInHistory(url);
+
+ if (tab_content.children().length == 0 && load_always == false) {
+ $.ajax({
+ url: remote_url,
+ type: 'get',
+ success: function(data){
+ tab_content.html(data)
+ }
+ });
+ }
+
+ tab_content.show();
+ return false;
+}
+
+//replaces current URL with the "href" attribute of the current link
+//(only triggered if supported by browser)
+function replaceInHistory(url) {
+ if ("replaceState" in window.history && url !== undefined) {
+ window.history.replaceState(null, document.title, url);
+ }
+}
+
+function moveTabRight(el) {
+ var lis = $(el).parents('div.tabs').first().find('ul').children();
+ var bw = $(el).parents('div.tabs-buttons').outerWidth(true);
+ var tabsWidth = 0;
+ var i = 0;
+ lis.each(function() {
+ if ($(this).is(':visible')) {
+ tabsWidth += $(this).outerWidth(true);
+ }
+ });
+ if (tabsWidth < $(el).parents('div.tabs').first().width() - bw) { return; }
+ $(el).siblings('.tab-left').removeClass('disabled');
+ while (i<lis.length && !lis.eq(i).is(':visible')) { i++; }
+ var w = lis.eq(i).width();
+ lis.eq(i).hide();
+ if (tabsWidth - w < $(el).parents('div.tabs').first().width() - bw) {
+ $(el).addClass('disabled');
+ }
+}
+
+function moveTabLeft(el) {
+ var lis = $(el).parents('div.tabs').first().find('ul').children();
+ var i = 0;
+ while (i < lis.length && !lis.eq(i).is(':visible')) { i++; }
+ if (i > 0) {
+ lis.eq(i-1).show();
+ $(el).siblings('.tab-right').removeClass('disabled');
+ }
+ if (i <= 1) {
+ $(el).addClass('disabled');
+ }
+}
+
+function displayTabsButtons() {
+ var lis;
+ var tabsWidth;
+ var el;
+ var numHidden;
+ $('div.tabs').each(function() {
+ el = $(this);
+ lis = el.find('ul').children();
+ tabsWidth = 0;
+ numHidden = 0;
+ lis.each(function(){
+ if ($(this).is(':visible')) {
+ tabsWidth += $(this).outerWidth(true);
+ } else {
+ numHidden++;
+ }
+ });
+ var bw = $(el).find('div.tabs-buttons').outerWidth(true);
+ if ((tabsWidth < el.width() - bw) && (lis.length === 0 || lis.first().is(':visible'))) {
+ el.find('div.tabs-buttons').hide();
+ } else {
+ el.find('div.tabs-buttons').show().children('button.tab-left').toggleClass('disabled', numHidden == 0);
+ }
+ });
+}
+
+function setPredecessorFieldsVisibility() {
+ var relationType = $('#relation_relation_type');
+ if (relationType.val() == "precedes" || relationType.val() == "follows") {
+ $('#predecessor_fields').show();
+ } else {
+ $('#predecessor_fields').hide();
+ }
+}
+
+function showModal(id, width, title) {
+ var el = $('#'+id).first();
+ if (el.length === 0 || el.is(':visible')) {return;}
+ if (!title) title = el.find('h3.title').text();
+ // moves existing modals behind the transparent background
+ $(".modal").css('zIndex',99);
+ el.dialog({
+ width: width,
+ modal: true,
+ resizable: false,
+ dialogClass: 'modal',
+ title: title
+ }).on('dialogclose', function(){
+ $(".modal").css('zIndex',101);
+ });
+ el.find("input[type=text], input[type=submit]").first().focus();
+}
+
+function hideModal(el) {
+ var modal;
+ if (el) {
+ modal = $(el).parents('.ui-dialog-content');
+ } else {
+ modal = $('#ajax-modal');
+ }
+ modal.dialog("close");
+}
+
+function collapseScmEntry(id) {
+ $('.'+id).each(function() {
+ if ($(this).hasClass('open')) {
+ collapseScmEntry($(this).attr('id'));
+ }
+ $(this).hide();
+ });
+ $('#'+id).removeClass('open');
+}
+
+function expandScmEntry(id) {
+ $('.'+id).each(function() {
+ $(this).show();
+ if ($(this).hasClass('loaded') && !$(this).hasClass('collapsed')) {
+ expandScmEntry($(this).attr('id'));
+ }
+ });
+ $('#'+id).addClass('open');
+}
+
+function scmEntryClick(id, url) {
+ var el = $('#'+id);
+ var expander = el.find('.expander');
+ var folder = el.find('.icon-folder');
+
+ if (el.hasClass('open')) {
+ collapseScmEntry(id);
+ el.find('.expander').switchClass('icon-expanded', 'icon-collapsed');
+ el.addClass('collapsed');
+ updateSVGIcon(folder[0], 'folder')
+ toggleExpendCollapseIcon(expander[0]);
+
+ return false;
+ } else if (el.hasClass('loaded')) {
+ expandScmEntry(id);
+ el.find('.expander').switchClass('icon-collapsed', 'icon-expanded');
+ el.removeClass('collapsed');
+ updateSVGIcon(folder[0], 'folder-open')
+ toggleExpendCollapseIcon(expander[0]);
+
+ return false;
+ }
+ if (el.hasClass('loading')) {
+ return false;
+ }
+ el.addClass('loading');
+ $.ajax({
+ url: url,
+ success: function(data) {
+ el.after(data);
+ el.addClass('open').addClass('loaded').removeClass('loading');
+ el.find('.expander').switchClass('icon-collapsed', 'icon-expanded');
+ updateSVGIcon(folder[0], 'folder-open')
+ toggleExpendCollapseIcon(expander[0]);
+ }
+ });
+ return true;
+}
+
+function randomKey(size) {
+ var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';
+ var key = '';
+ for (var i = 0; i < size; i++) {
+ key += chars.charAt(Math.floor(Math.random() * chars.length));
+ }
+ return key;
+}
+
+function copyToClipboard(text) {
+ if (navigator.clipboard) {
+ return navigator.clipboard.writeText(text).catch(() => {
+ return fallbackClipboardCopy(text);
+ });
+ } else {
+ return fallbackClipboardCopy(text);
+ }
+}
+
+function fallbackClipboardCopy(text) {
+ const temp = document.createElement('textarea');
+ temp.value = text;
+ temp.style.position = 'fixed';
+ temp.style.left = '-9999px';
+ document.body.appendChild(temp);
+ temp.select();
+ document.execCommand('copy');
+ document.body.removeChild(temp);
+ return Promise.resolve();
+}
+
+function copyDataClipboardTextToClipboard(target) {
+ copyToClipboard(target.getAttribute('data-clipboard-text'));
+
+ if ($(target).closest('.drdn.expanded').length) {
+ $(target).closest('.drdn.expanded').removeClass("expanded");
+ }
+ return false;
+}
+
+function setupCopyButtonsToPreElements() {
+ document.querySelectorAll('.wiki pre:not(.pre-wrapper pre)').forEach((pre) => {
+ // Wrap the <pre> element with a container and add a copy button
+ const wrapper = document.createElement("div");
+ wrapper.classList.add("pre-wrapper");
+
+ const copyButton = document.createElement("a");
+ copyButton.title = rm.I18n.buttonCopy;
+ copyButton.classList.add("copy-pre-content-link", "icon-only");
+ copyButton.append(createSVGIcon("copy-pre-content"));
+
+ wrapper.appendChild(copyButton);
+ wrapper.append(pre.cloneNode(true));
+ pre.replaceWith(wrapper);
+
+ // Copy the contents of the pre tag when copyButton is clicked
+ copyButton.addEventListener("click", (event) => {
+ event.preventDefault();
+ let textToCopy = (pre.querySelector("code") || pre).textContent.replace(/\n$/, '');
+ if (pre.querySelector("code.syntaxhl")) { textToCopy = textToCopy.replace(/ $/, ''); } // Workaround for half-width space issue in Textile's highlighted code
+ copyToClipboard(textToCopy).then(() => {
+ updateSVGIcon(copyButton, "checked");
+ setTimeout(() => updateSVGIcon(copyButton, "copy-pre-content"), 2000);
+ });
+ });
+ });
+}
+
+function updateIssueFrom(url, el) {
+ $('#all_attributes input, #all_attributes textarea, #all_attributes select').each(function(){
+ $(this).data('valuebeforeupdate', $(this).val());
+ });
+ if (el) {
+ $("#form_update_triggered_by").val($(el).attr('id'));
+ }
+ return $.ajax({
+ url: url,
+ type: 'post',
+ data: $('#issue-form').serialize()
+ });
+}
+
+function replaceIssueFormWith(html){
+ var replacement = $(html);
+ $('#all_attributes input, #all_attributes textarea, #all_attributes select').each(function(){
+ var object_id = $(this).attr('id');
+ if (object_id && $(this).data('valuebeforeupdate')!=$(this).val()) {
+ replacement.find('#'+object_id).val($(this).val());
+ }
+ });
+ $('#all_attributes').empty();
+ $('#all_attributes').prepend(replacement);
+}
+
+function updateBulkEditFrom(url) {
+ $.ajax({
+ url: url,
+ type: 'post',
+ data: $('#bulk_edit_form').serialize()
+ });
+}
+
+function observeAutocompleteField(fieldId, url, options) {
+ $(document).ready(function() {
+ $('#'+fieldId).autocomplete($.extend({
+ source: url,
+ minLength: 2,
+ position: {collision: "flipfit"},
+ search: function(){$('#'+fieldId).addClass('ajax-loading');},
+ response: function(){$('#'+fieldId).removeClass('ajax-loading');}
+ }, options));
+ $('#'+fieldId).addClass('autocomplete');
+ });
+}
+
+function multipleAutocompleteField(fieldId, url, options) {
+ function split(val) {
+ return val.split(/,\s*/);
+ }
+
+ function extractLast(term) {
+ return split(term).pop();
+ }
+
+ $(document).ready(function () {
+ $('#' + fieldId).autocomplete($.extend({
+ source: function (request, response) {
+ $.getJSON(url, {
+ term: extractLast(request.term)
+ }, response);
+ },
+ minLength: 2,
+ position: {collision: "flipfit"},
+ search: function () {
+ $('#' + fieldId).addClass('ajax-loading');
+ },
+ response: function () {
+ $('#' + fieldId).removeClass('ajax-loading');
+ },
+ select: function (event, ui) {
+ var terms = split(this.value);
+ // remove the current input
+ terms.pop();
+ // add the selected item
+ terms.push(ui.item.value);
+ // add placeholder to get the comma-and-space at the end
+ terms.push("");
+ this.value = terms.join(", ");
+ return false;
+ }
+ }, options));
+ $('#' + fieldId).addClass('autocomplete');
+ });
+}
+
+function observeSearchfield(fieldId, targetId, url) {
+ $('#'+fieldId).each(function() {
+ var $this = $(this);
+ $this.addClass('autocomplete');
+ $this.attr('data-value-was', $this.val());
+ var check = function() {
+ var val = $this.val();
+ if ($this.attr('data-value-was') != val){
+ $this.attr('data-value-was', val);
+ $.ajax({
+ url: url,
+ type: 'get',
+ data: {q: $this.val()},
+ success: function(data){ if(targetId) $('#'+targetId).html(data); },
+ beforeSend: function(){ $this.addClass('ajax-loading'); },
+ complete: function(){ $this.removeClass('ajax-loading'); }
+ });
+ }
+ };
+ var reset = function() {
+ if (timer) {
+ clearInterval(timer);
+ timer = setInterval(check, 300);
+ }
+ };
+ var timer = setInterval(check, 300);
+ $this.bind('keyup click mousemove', reset);
+ });
+}
+
+$(document).ready(function(){
+ $(".drdn .autocomplete").val('');
+
+ // This variable is used to focus selected project
+ var selected;
+ $(document).on('click', '.drdn-trigger', function(e){
+ var drdn = $(this).closest(".drdn");
+ if (drdn.hasClass("expanded")) {
+ drdn.removeClass("expanded");
+ } else {
+ $(".drdn").removeClass("expanded");
+ drdn.addClass("expanded");
+ if ($(this).parent('#project-jump').length) {
+ selected = $('.drdn-items a.selected'); // Store selected project
+ selected.first().focus(); // Calling focus to scroll to selected project
+ }
+ if (!isMobile()) {
+ drdn.find(".autocomplete").focus();
+ }
+ e.stopPropagation();
+ }
+ });
+ $(document).click(function(e){
+ if ($(e.target).closest(".drdn").length < 1) {
+ $(".drdn.expanded").removeClass("expanded");
+ }
+ });
+
+ observeSearchfield('projects-quick-search', null, $('#projects-quick-search').data('automcomplete-url'));
+
+ $(".drdn-content").keydown(function(event){
+ var items = $(this).find(".drdn-items");
+
+ // If a project is selected set focused to selected only once
+ if (selected && selected.length > 0) {
+ var focused = selected;
+ selected = undefined;
+ }
+ else {
+ var focused = items.find("a:focus");
+ }
+ switch (event.which) {
+ case 40: //down
+ if (focused.length > 0) {
+ focused.nextAll("a").first().focus();;
+ } else {
+ items.find("a").first().focus();;
+ }
+ event.preventDefault();
+ break;
+ case 38: //up
+ if (focused.length > 0) {
+ var prev = focused.prevAll("a");
+ if (prev.length > 0) {
+ prev.first().focus();
+ } else {
+ $(this).find(".autocomplete").focus();
+ }
+ event.preventDefault();
+ }
+ break;
+ case 35: //end
+ if (focused.length > 0) {
+ focused.nextAll("a").last().focus();
+ event.preventDefault();
+ }
+ break;
+ case 36: //home
+ if (focused.length > 0) {
+ focused.prevAll("a").last().focus();
+ event.preventDefault();
+ }
+ break;
+ }
+ });
+});
+
+function beforeShowDatePicker(input, inst) {
+ var default_date = null;
+ switch ($(input).attr("id")) {
+ case "issue_start_date" :
+ if ($("#issue_due_date").length > 0) {
+ default_date = $("#issue_due_date").val();
+ }
+ break;
+ case "issue_due_date" :
+ if ($("#issue_start_date").length > 0) {
+ var start_date = $("#issue_start_date").val();
+ if (start_date != "") {
+ start_date = new Date(Date.parse(start_date));
+ if (start_date > new Date()) {
+ default_date = $("#issue_start_date").val();
+ }
+ }
+ }
+ break;
+ }
+ $(input).datepickerFallback("option", "defaultDate", default_date);
+}
+
+(function($){
+ $.fn.positionedItems = function(sortableOptions, options){
+ var settings = $.extend({
+ firstPosition: 1
+ }, options );
+
+ return this.sortable($.extend({
+ axis: 'y',
+ handle: ".sort-handle",
+ helper: function(event, ui){
+ ui.children('td').each(function(){
+ $(this).width($(this).width());
+ });
+ return ui;
+ },
+ update: function(event, ui) {
+ var sortable = $(this);
+ var handle = ui.item.find(".sort-handle").addClass("ajax-loading");
+ var url = handle.data("reorder-url");
+ var param = handle.data("reorder-param");
+ var data = {};
+ data[param] = {position: ui.item.index() + settings['firstPosition']};
+ $.ajax({
+ url: url,
+ type: 'put',
+ dataType: 'script',
+ data: data,
+ error: function(jqXHR, textStatus, errorThrown){
+ alert(jqXHR.status);
+ sortable.sortable("cancel");
+ },
+ complete: function(jqXHR, textStatus, errorThrown){
+ handle.removeClass("ajax-loading");
+ }
+ });
+ },
+ }, sortableOptions));
+ }
+}( jQuery ));
+
+var warnLeavingUnsavedMessage;
+function warnLeavingUnsaved(message) {
+ warnLeavingUnsavedMessage = message;
+ $(document).on('submit', 'form', function(){
+ $('textarea').removeData('changed');
+ });
+ $(document).on('change', 'textarea', function(){
+ $(this).data('changed', 'changed');
+ });
+ window.onbeforeunload = function(){
+ var warn = false;
+ $('textarea').blur().each(function(){
+ if ($(this).data('changed')) {
+ warn = true;
+ }
+ });
+ if (warn) {return warnLeavingUnsavedMessage;}
+ };
+}
+
+function setupAjaxIndicator() {
+ $(document).bind('ajaxSend', function(event, xhr, settings) {
+ if ($('.ajax-loading').length === 0 && settings.contentType != 'application/octet-stream') {
+ $('#ajax-indicator').show();
+ }
+ });
+ $(document).bind('ajaxStop', function() {
+ $('#ajax-indicator').hide();
+ });
+}
+
+function setupTabs() {
+ if($('.tabs').length > 0) {
+ displayTabsButtons();
+ $(window).resize(displayTabsButtons);
+ }
+}
+
+function setupFilePreviewNavigation() {
+ // only bind arrow keys when preview navigation is present
+ const element = $('.pagination.filepreview').first();
+ if (element) {
+
+ const handleArrowKey = function(selector, e){
+ const href = $(element).find(selector).attr('href');
+ if (href) {
+ window.location = href;
+ e.preventDefault();
+ }
+ };
+
+ $(document).keydown(function(e) {
+ if(e.shiftKey || e.metaKey || e.ctrlKey || e.altKey) return;
+ switch(e.key) {
+ case 'ArrowLeft':
+ handleArrowKey('.previous a', e);
+ break;
+
+ case 'ArrowRight':
+ handleArrowKey('.next a', e);
+ break;
+ }
+ });
+ }
+}
+
+$(document).on('keydown', 'form textarea', function(e) {
+ // Submit the form with Ctrl + Enter or Command + Return
+ var targetForm = $(e.target).closest('form');
+ if(e.keyCode == 13 && ((e.ctrlKey && !e.metaKey) || (!e.ctrlKey && e.metaKey) && targetForm.length)) {
+ // For ajax, use click() instead of submit() to prevent "Invalid form authenticity token" error
+ if (targetForm.attr('data-remote') == 'true') {
+ if (targetForm.find('input[type=submit]').length === 0) { return false; }
+ targetForm.find('textarea').blur().removeData('changed');
+ targetForm.find('input[type=submit]').first().click();
+ } else {
+ targetForm.find('textarea').blur().removeData('changed');
+ targetForm.submit();
+ }
+ }
+});
+
+
+function hideOnLoad() {
+ $('.hol').hide();
+}
+
+function addFormObserversForDoubleSubmit() {
+ $('form[method=post]').each(function() {
+ if (!$(this).hasClass('multiple-submit')) {
+ $(this).submit(function(form_submission) {
+ if ($(form_submission.target).attr('data-submitted')) {
+ form_submission.preventDefault();
+ } else {
+ $(form_submission.target).attr('data-submitted', true);
+ }
+ });
+ }
+ });
+}
+
+function defaultFocus(){
+ if (($('#content :focus').length == 0) && (window.location.hash == '')) {
+ $('#content input[type=text]:visible, #content textarea:visible').first().focus();
+ }
+}
+
+function blockEventPropagation(event) {
+ event.stopPropagation();
+ event.preventDefault();
+}
+
+function toggleDisabledOnChange() {
+ var checked = $(this).is(':checked');
+ $($(this).data('disables')).attr('disabled', checked);
+ $($(this).data('enables')).attr('disabled', !checked);
+ $($(this).data('shows')).toggle(checked);
+}
+function toggleDisabledInit() {
+ $('input[data-disables], input[data-enables], input[data-shows]').each(toggleDisabledOnChange);
+}
+function toggleMultiSelectIconInit() {
+ $('.toggle-multiselect:not(.icon-toggle-minus):not(.icon-toggle-plus)').each(function(){
+ let iconType;
+ if ($(this).siblings('select').find('option:selected').length > 1) {
+ iconType = 'toggle-minus';
+ } else {
+ iconType = 'toggle-plus';
+ }
+
+ $(this).addClass(`icon-${iconType}`);
+ updateSVGIcon($(this).find('svg')[0], iconType);
+ });
+}
+
+function toggleNewObjectDropdown() {
+ var dropdown = $('#new-object + ul.menu-children');
+ if(dropdown.hasClass('visible')){
+ dropdown.removeClass('visible');
+ }else{
+ dropdown.addClass('visible');
+ }
+}
+
+(function ( $ ) {
+
+ // detect if native date input is supported
+ var nativeDateInputSupported = true;
+
+ var input = document.createElement('input');
+ input.setAttribute('type','date');
+ if (input.type === 'text') {
+ nativeDateInputSupported = false;
+ }
+
+ var notADateValue = 'not-a-date';
+ input.setAttribute('value', notADateValue);
+ if (input.value === notADateValue) {
+ nativeDateInputSupported = false;
+ }
+
+ $.fn.datepickerFallback = function( options ) {
+ if (nativeDateInputSupported) {
+ return this;
+ } else {
+ return this.datepicker( options );
+ }
+ };
+}( jQuery ));
+
+$(document).ready(function(){
+ $('#content').on('change', 'input[data-disables], input[data-enables], input[data-shows]', toggleDisabledOnChange);
+ toggleDisabledInit();
+
+ $('#content').on('click', '.toggle-multiselect', function() {
+ toggleMultiSelect($(this).siblings('select'));
+ $(this).toggleClass('icon-toggle-plus icon-toggle-minus');
+ updateSVGIcon($(this).find('svg')[0], $(this).hasClass('icon-toggle-plus') ? 'toggle-plus' : 'toggle-minus');
+ });
+ toggleMultiSelectIconInit();
+
+ $('#history .tabs').on('click', 'a', function(e){
+ var tab = $(e.target).attr('id').replace('tab-','');
+ document.cookie = 'history_last_tab=' + tab + '; SameSite=Lax'
+ });
+});
+
+$(document).ready(function(){
+ $('#content').on('click', 'div.jstTabs a.tab-preview', function(event){
+ var tab = $(event.target);
+
+ var url = tab.data('url');
+ var form = tab.parents('form');
+ var jstBlock = tab.parents('.jstBlock');
+
+ var element = encodeURIComponent(jstBlock.find('.wiki-edit').val());
+ var attachments = form.find('.attachments_fields input').serialize();
+
+ $.ajax({
+ url: url,
+ type: 'post',
+ data: "text=" + element + '&' + attachments,
+ success: function(data){
+ jstBlock.find('.wiki-preview').html(data);
+ setupWikiTableSortableHeader();
+ }
+ });
+ });
+});
+
+function keepAnchorOnSignIn(form){
+ var hash = decodeURIComponent(self.document.location.hash);
+ if (hash) {
+ if (hash.indexOf("#") === -1) {
+ hash = "#" + hash;
+ }
+ form.action = form.action + hash;
+ }
+ return true;
+}
+
+$(function ($) {
+ $('#auth_source_ldap_mode').change(function () {
+ $('.ldaps_warning').toggle($(this).val() != 'ldaps_verify_peer');
+ }).change();
+});
+
+function setFilecontentContainerHeight() {
+ var $filecontainer = $('.filecontent-container');
+ var fileTypeSelectors = ['.image', 'video'];
+
+ if($filecontainer.length > 0 && $filecontainer.find(fileTypeSelectors.join(',')).length === 1) {
+ var containerOffsetTop = $filecontainer.offset().top;
+ var containerMarginBottom = parseInt($filecontainer.css('marginBottom'));
+ var paginationHeight = $filecontainer.next('.pagination').height();
+ var diff = containerOffsetTop + containerMarginBottom + paginationHeight;
+
+ $filecontainer.css('height', 'calc(100vh - ' + diff + 'px)')
+ }
+}
+
+function setupAttachmentDetail() {
+ setFilecontentContainerHeight();
+ $(window).resize(setFilecontentContainerHeight);
+}
+
+function setupWikiTableSortableHeader() {
+ if (typeof Tablesort === 'undefined') { return; }
+ $('div.wiki table').each(function(i, table){
+ if (table.rows.length < 3) return true;
+ var tr = $(table.rows).first();
+ if (tr.find("TH").length > 0) {
+ tr.attr('data-sort-method', 'none');
+ tr.find("TD").attr('data-sort-method', 'none');
+ new Tablesort(table);
+ }
+ });
+}
+
+function setupHoverTooltips(container) {
+ $(container || 'body').find("[title]:not(.no-tooltip)").tooltip({
+ show: {
+ delay: 400
+ },
+ position: {
+ my: "center bottom-5",
+ at: "center top"
+ }
+ });
+}
+function removeHoverTooltips(container) {
+ $(container || 'body').find("[title]:not(.no-tooltip)").tooltip('destroy')
+}
+$(function() { setupHoverTooltips(); });
+
+function inlineAutoComplete(element) {
+ 'use strict';
+
+ // do not attach if Tribute is already initialized
+ if (element.dataset.tribute === 'true') {return};
+
+ const getDataSource = function(entity) {
+ const dataSources = rm.AutoComplete.dataSources;
+
+ if (dataSources[entity]) {
+ return dataSources[entity];
+ } else {
+ return false;
+ }
+ }
+
+ const remoteSearch = function(url, cb) {
+ const xhr = new XMLHttpRequest();
+ xhr.onreadystatechange = function ()
+ {
+ if (xhr.readyState === 4) {
+ if (xhr.status === 200) {
+ var data = JSON.parse(xhr.responseText);
+ cb(data);
+ } else if (xhr.status === 403) {
+ cb([]);
+ }
+ }
+ };
+ xhr.open("GET", url, true);
+ xhr.send();
+ };
+
+ const tribute = new Tribute({
+ collection: [
+ {
+ trigger: '#',
+ values: function (text, cb) {
+ if (event.target.type === 'text' && element.getAttribute('autocomplete') != 'off') {
+ element.setAttribute('autocomplete', 'off');
+ }
+ // When triggered with text starting with "##", like "##a", the search term will become "#a",
+ // causing the SQL query to fail in finding issues with "a" in the subject.
+ // To avoid this, remove the first "#" from the search term.
+ if (text) {
+ text = text.replace(/^#/, '');
+ }
+ remoteSearch(getDataSource('issues') + encodeURIComponent(text), function (issues) {
+ return cb(issues);
+ });
+ },
+ lookup: 'label',
+ fillAttr: 'label',
+ requireLeadingSpace: true,
+ selectTemplate: function (issue) {
+ let leadingHash = "#"
+ // keep ## syntax which is a valid issue syntax to show issue with title.
+ if (this.currentMentionTextSnapshot.charAt(0) === "#") {
+ leadingHash = "##"
+ }
+ return leadingHash + issue.original.id;
+ },
+ menuItemTemplate: function (issue) {
+ return sanitizeHTML(issue.original.label);
+ }
+ },
+ {
+ trigger: '[[',
+ values: function (text, cb) {
+ remoteSearch(getDataSource('wiki_pages') + encodeURIComponent(text), function (wikiPages) {
+ return cb(wikiPages);
+ });
+ },
+ lookup: 'label',
+ fillAttr: 'label',
+ requireLeadingSpace: true,
+ selectTemplate: function (wikiPage) {
+ return '[[' + wikiPage.original.value + ']]';
+ },
+ menuItemTemplate: function (wikiPage) {
+ return sanitizeHTML(wikiPage.original.label);
+ }
+ },
+ {
+ trigger: '@',
+ lookup: function (user, mentionText) {
+ return user.name + user.firstname + user.lastname + user.login;
+ },
+ values: function (text, cb) {
+ const url = getDataSource('users');
+ if (url) {
+ remoteSearch(url + encodeURIComponent(text), function (users) {
+ return cb(users);
+ });
+ }
+ },
+ menuItemTemplate: function (user) {
+ return user.original.name;
+ },
+ selectTemplate: function (user) {
+ return '@' + user.original.login;
+ }
+ }
+ ],
+ noMatchTemplate: ""
+ });
+
+ tribute.attach(element);
+}
+
+// collapsible sidebar jQuery plugin
+(function($) {
+ // main container this is applied to
+ var main;
+ // triggers show/hide
+ var button;
+ // the key to use in local storage
+ // this will later be expanded using the current controller and action to
+ // allow for different sidebar states for different pages
+ var localStorageKey = 'redmine-sidebar-state';
+ // true if local storage is available
+ var canUseLocalStorage = function(){
+ try {
+ if('localStorage' in window){
+ localStorage.setItem('redmine.test.storage', 'ok');
+ var item = localStorage.getItem('redmine.test.storage');
+ localStorage.removeItem('redmine.test.storage');
+ if(item === 'ok') return true;
+ }
+ } catch (err) {}
+ return false;
+ }();
+ // function to set current sidebar state
+ var setState = function(state){
+ if(canUseLocalStorage){
+ localStorage.setItem(localStorageKey, state);
+ }
+ };
+ var applyState = function(){
+ if(main.hasClass('collapsedsidebar')){
+ updateSVGIcon(document.getElementById('sidebar-switch-button'), 'chevrons-left')
+ setState('hidden');
+ } else {
+ updateSVGIcon(document.getElementById('sidebar-switch-button'), 'chevrons-right')
+ setState('visible');
+ }
+ };
+ var setupToggleButton = function(){
+ button = $('#sidebar-switch-button');
+ button.click(function(e){
+ main.addClass("animate");
+ main.toggleClass('collapsedsidebar');
+ applyState();
+ e.preventDefault();
+ return false;
+ });
+ applyState();
+ };
+ $.fn.collapsibleSidebar = function() {
+ main = this;
+ // determine previously stored sidebar state for this page
+ if(canUseLocalStorage) {
+ // determine current controller/action pair and use them as storage key
+ var bodyClass = $('body').attr('class');
+ if(bodyClass){
+ try {
+ localStorageKey += '-' + bodyClass.split(/\s+/).filter(function(s){
+ return s.match(/(action|controller)-.*/);
+ }).sort().join('-');
+ } catch(e) {
+ // in case of error (probably IE8), continue with the unmodified key
+ }
+ }
+ var storedState = localStorage.getItem(localStorageKey);
+ main.toggleClass('collapsedsidebar', storedState === 'hidden');
+ }
+ // draw the toggle button once the DOM is complete
+ $(document).ready(setupToggleButton);
+ };
+}(jQuery));
+
+$(document).ready(setupAjaxIndicator);
+$(document).ready(hideOnLoad);
+$(document).ready(addFormObserversForDoubleSubmit);
+$(document).ready(defaultFocus);
+$(document).ready(setupAttachmentDetail);
+$(document).ready(setupTabs);
+$(document).ready(setupFilePreviewNavigation);
+$(document).ready(setupWikiTableSortableHeader);
+$(document).on('focus', '[data-auto-complete=true]', function(event) {
+ inlineAutoComplete(event.target);
+});
+document.addEventListener("DOMContentLoaded", () => { setupCopyButtonsToPreElements(); });