Redmine 3.4.4
This commit is contained in:
commit
64924a6376
2112 changed files with 259028 additions and 0 deletions
857
public/javascripts/application.js
Normal file
857
public/javascripts/application.js
Normal file
|
@ -0,0 +1,857 @@
|
|||
/* Redmine - project management software
|
||||
Copyright (C) 2006-2017 Jean-Philippe Lang */
|
||||
|
||||
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);
|
Loading…
Add table
Add a link
Reference in a new issue