mirror of
				https://github.com/telekom-security/tpotce.git
				synced 2025-07-02 01:27:27 -04:00 
			
		
		
		
	
		
			
	
	
		
			865 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			865 lines
		
	
	
		
			24 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | /* Redmine - project management software | ||
|  |    Copyright (C) 2006-2017  Jean-Philippe Lang */ | ||
|  | 
 | ||
|  | /* Fix for CVE-2015-9251, to be removed with JQuery >= 3.0 */ | ||
|  | $.ajaxPrefilter(function (s) { | ||
|  |   if (s.crossDomain) { | ||
|  |     s.contents.script = false; | ||
|  |   } | ||
|  | }); | ||
|  | 
 | ||
|  | 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); | ||
|  | } | ||
|  | 
 | ||
|  | 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'); | ||
|  |   while (n.length && !n.hasClass('group')) { | ||
|  |     n.toggle(); | ||
|  |     n = n.next('tr'); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function collapseAllRowGroups(el) { | ||
|  |   var tbody = $(el).parents('tbody').first(); | ||
|  |   tbody.children('tr').each(function(index) { | ||
|  |     if ($(this).hasClass('group')) { | ||
|  |       $(this).removeClass('open'); | ||
|  |     } 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'); | ||
|  |     } 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('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 td.field input[type=checkbox]').each(function() { | ||
|  |     toggleFilter($(this).val()); | ||
|  |   }); | ||
|  |   $('#filters-table').on('click', 'td.field input[type=checkbox]', function() { | ||
|  |     toggleFilter($(this).val()); | ||
|  |   }); | ||
|  |   $('#filters-table').on('click', '.toggle-multiselect', function() { | ||
|  |     toggleMultiSelect($(this).siblings('select')); | ||
|  |   }); | ||
|  |   $('#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); | ||
|  |   $('#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 i, select; | ||
|  | 
 | ||
|  |   var tr = $('<tr class="filter">').attr('id', 'tr_'+fieldId).html( | ||
|  |     '<td class="field"><input checked="checked" id="cb_'+fieldId+'" name="f[]" value="'+field+'" type="checkbox"><label for="cb_'+fieldId+'"> '+filterOptions['name']+'</label></td>' + | ||
|  |     '<td class="operator"><select id="operators_'+fieldId+'" name="op['+field+']"></td>' + | ||
|  |     '<td class="values"></td>' | ||
|  |   ); | ||
|  |   filterTable.append(tr); | ||
|  | 
 | ||
|  |   select = tr.find('td.operator select'); | ||
|  |   for (i = 0; i < operators.length; i++) { | ||
|  |     var option = $('<option>').val(operators[i]).text(operatorLabels[operators[i]]); | ||
|  |     if (operators[i] == operator) { option.attr('selected', true); } | ||
|  |     select.append(option); | ||
|  |   } | ||
|  |   select.change(function(){ toggleOperator(field); }); | ||
|  | 
 | ||
|  |   switch (filterOptions['type']) { | ||
|  |   case "list": | ||
|  |   case "list_optional": | ||
|  |   case "list_status": | ||
|  |   case "list_subprojects": | ||
|  |     tr.find('td.values').append( | ||
|  |       '<span style="display:none;"><select class="value" id="values_'+fieldId+'_1" name="v['+field+'][]"></select>' + | ||
|  |       ' <span class="toggle-multiselect"> </span></span>' | ||
|  |     ); | ||
|  |     select = tr.find('td.values select'); | ||
|  |     if (values.length > 1) { select.attr('multiple', true); } | ||
|  |     for (i = 0; i < filterValues.length; i++) { | ||
|  |       var filterValue = filterValues[i]; | ||
|  |       var option = $('<option>'); | ||
|  |       if ($.isArray(filterValue)) { | ||
|  |         option.val(filterValue[1]).text(filterValue[0]); | ||
|  |         if ($.inArray(filterValue[1], values) > -1) {option.attr('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.attr('selected', true);} | ||
|  |       } | ||
|  |       select.append(option); | ||
|  |     } | ||
|  |     break; | ||
|  |   case "date": | ||
|  |   case "date_past": | ||
|  |     tr.find('td.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": | ||
|  |     tr.find('td.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('td.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('td.values select'); | ||
|  |     for (i = 0; i < filterValues.length; i++) { | ||
|  |       var filterValue = filterValues[i]; | ||
|  |       var option = $('<option>'); | ||
|  |       option.val(filterValue[1]).text(filterValue[0]); | ||
|  |       if (values[0] == filterValue[1]) { option.attr('selected', true); } | ||
|  |       select.append(option); | ||
|  |     } | ||
|  |     break; | ||
|  |   case "integer": | ||
|  |   case "float": | ||
|  |   case "tree": | ||
|  |     tr.find('td.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+' td.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 "t": | ||
|  |     case "ld": | ||
|  |     case "w": | ||
|  |     case "lw": | ||
|  |     case "l2w": | ||
|  |     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) { | ||
|  |   if (el.attr('multiple')) { | ||
|  |     el.removeAttr('multiple'); | ||
|  |     el.attr('size', 1); | ||
|  |   } else { | ||
|  |     el.attr('multiple', true); | ||
|  |     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).parent().find('div.tabs a').removeClass('selected'); | ||
|  |   $('#tab-content-' + name).show(); | ||
|  |   $('#tab-' + name).addClass('selected'); | ||
|  |   //replaces current URL with the "href" attribute of the current link | ||
|  |   //(only triggered if supported by browser) | ||
|  |   if ("replaceState" in window.history) { | ||
|  |     window.history.replaceState(null, document.title, url); | ||
|  |   } | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | 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).parents('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").zIndex(99); | ||
|  |   el.dialog({ | ||
|  |     width: width, | ||
|  |     modal: true, | ||
|  |     resizable: false, | ||
|  |     dialogClass: 'modal', | ||
|  |     title: title | ||
|  |   }).on('dialogclose', function(){ | ||
|  |     $(".modal").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 submitPreview(url, form, target) { | ||
|  |   $.ajax({ | ||
|  |     url: url, | ||
|  |     type: 'post', | ||
|  |     data: $('#'+form).serialize(), | ||
|  |     success: function(data){ | ||
|  |       $('#'+target).html(data); | ||
|  |     } | ||
|  |   }); | ||
|  | } | ||
|  | 
 | ||
|  | 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); | ||
|  |     if (el.hasClass('open')) { | ||
|  |         collapseScmEntry(id); | ||
|  |         el.addClass('collapsed'); | ||
|  |         return false; | ||
|  |     } else if (el.hasClass('loaded')) { | ||
|  |         expandScmEntry(id); | ||
|  |         el.removeClass('collapsed'); | ||
|  |         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'); | ||
|  |       } | ||
|  |     }); | ||
|  |     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 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 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; | ||
|  |   $(".drdn-trigger").click(function(e){ | ||
|  |     var drdn = $(this).closest(".drdn"); | ||
|  |     if (drdn.hasClass("expanded")) { | ||
|  |       drdn.removeClass("expanded"); | ||
|  |     } else { | ||
|  |       $(".drdn").removeClass("expanded"); | ||
|  |       drdn.addClass("expanded"); | ||
|  |       selected = $('.drdn-items a.selected'); // Store selected project | ||
|  |       selected.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").size() > 0) { | ||
|  |         default_date = $("#issue_due_date").val(); | ||
|  |       } | ||
|  |       break; | ||
|  |     case "issue_due_date" : | ||
|  |       if ($("#issue_start_date").size() > 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 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], #content textarea').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 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(); | ||
|  | }); | ||
|  | 
 | ||
|  | 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; | ||
|  | } | ||
|  | 
 | ||
|  | $(document).ready(setupAjaxIndicator); | ||
|  | $(document).ready(hideOnLoad); | ||
|  | $(document).ready(addFormObserversForDoubleSubmit); | ||
|  | $(document).ready(defaultFocus); | ||
|  | $(document).ready(setupTabs); |