Chris@1296: /* Redmine - project management software
Chris@1296: Copyright (C) 2006-2012 Jean-Philippe Lang */
Chris@1296:
Chris@1296: function checkAll(id, checked) {
Chris@1296: if (checked) {
Chris@1296: $('#'+id).find('input[type=checkbox]').attr('checked', true);
Chris@1296: } else {
Chris@1296: $('#'+id).find('input[type=checkbox]').removeAttr('checked');
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function toggleCheckboxesBySelector(selector) {
Chris@1296: var all_checked = true;
Chris@1296: $(selector).each(function(index) {
Chris@1296: if (!$(this).is(':checked')) { all_checked = false; }
Chris@1296: });
Chris@1296: $(selector).attr('checked', !all_checked)
Chris@1296: }
Chris@1296:
Chris@1296: function showAndScrollTo(id, focus) {
Chris@1296: $('#'+id).show();
Chris@1296: if (focus!=null) {
Chris@1296: $('#'+focus).focus();
Chris@1296: }
Chris@1296: $('html, body').animate({scrollTop: $('#'+id).offset().top}, 100);
Chris@1296: }
Chris@1296:
Chris@1296: function toggleRowGroup(el) {
Chris@1296: var tr = $(el).parents('tr').first();
Chris@1296: var n = tr.next();
Chris@1296: tr.toggleClass('open');
Chris@1296: while (n.length && !n.hasClass('group')) {
Chris@1296: n.toggle();
Chris@1296: n = n.next('tr');
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function collapseAllRowGroups(el) {
Chris@1296: var tbody = $(el).parents('tbody').first();
Chris@1296: tbody.children('tr').each(function(index) {
Chris@1296: if ($(this).hasClass('group')) {
Chris@1296: $(this).removeClass('open');
Chris@1296: } else {
Chris@1296: $(this).hide();
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function expandAllRowGroups(el) {
Chris@1296: var tbody = $(el).parents('tbody').first();
Chris@1296: tbody.children('tr').each(function(index) {
Chris@1296: if ($(this).hasClass('group')) {
Chris@1296: $(this).addClass('open');
Chris@1296: } else {
Chris@1296: $(this).show();
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function toggleAllRowGroups(el) {
Chris@1296: var tr = $(el).parents('tr').first();
Chris@1296: if (tr.hasClass('open')) {
Chris@1296: collapseAllRowGroups(el);
Chris@1296: } else {
Chris@1296: expandAllRowGroups(el);
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function toggleFieldset(el) {
Chris@1296: var fieldset = $(el).parents('fieldset').first();
Chris@1296: fieldset.toggleClass('collapsed');
Chris@1296: fieldset.children('div').toggle();
Chris@1296: }
Chris@1296:
Chris@1296: function hideFieldset(el) {
Chris@1296: var fieldset = $(el).parents('fieldset').first();
Chris@1296: fieldset.toggleClass('collapsed');
Chris@1296: fieldset.children('div').hide();
Chris@1296: }
Chris@1296:
Chris@1296: function initFilters(){
Chris@1296: $('#add_filter_select').change(function(){
Chris@1296: addFilter($(this).val(), '', []);
Chris@1296: });
Chris@1296: $('#filters-table td.field input[type=checkbox]').each(function(){
Chris@1296: toggleFilter($(this).val());
Chris@1296: });
Chris@1296: $('#filters-table td.field input[type=checkbox]').live('click',function(){
Chris@1296: toggleFilter($(this).val());
Chris@1296: });
Chris@1296: $('#filters-table .toggle-multiselect').live('click',function(){
Chris@1296: toggleMultiSelect($(this).siblings('select'));
Chris@1296: });
Chris@1296: $('#filters-table input[type=text]').live('keypress', function(e){
Chris@1296: if (e.keyCode == 13) submit_query_form("query_form");
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function addFilter(field, operator, values) {
Chris@1296: var fieldId = field.replace('.', '_');
Chris@1296: var tr = $('#tr_'+fieldId);
Chris@1296: if (tr.length > 0) {
Chris@1296: tr.show();
Chris@1296: } else {
Chris@1296: buildFilterRow(field, operator, values);
Chris@1296: }
Chris@1296: $('#cb_'+fieldId).attr('checked', true);
Chris@1296: toggleFilter(field);
Chris@1296: $('#add_filter_select').val('').children('option').each(function(){
Chris@1296: if ($(this).attr('value') == field) {
Chris@1296: $(this).attr('disabled', true);
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function buildFilterRow(field, operator, values) {
Chris@1296: var fieldId = field.replace('.', '_');
Chris@1296: var filterTable = $("#filters-table");
Chris@1296: var filterOptions = availableFilters[field];
Chris@1296: var operators = operatorByType[filterOptions['type']];
Chris@1296: var filterValues = filterOptions['values'];
Chris@1296: var i, select;
Chris@1296:
Chris@1296: var tr = $('
').attr('id', 'tr_'+fieldId).html(
Chris@1296: ' | ' +
Chris@1296: ' | ' +
Chris@1296: ' | '
Chris@1296: );
Chris@1296: filterTable.append(tr);
Chris@1296:
Chris@1296: select = tr.find('td.operator select');
Chris@1296: for (i=0;i').val(operators[i]).text(operatorLabels[operators[i]]);
Chris@1296: if (operators[i] == operator) {option.attr('selected', true)};
Chris@1296: select.append(option);
Chris@1296: }
Chris@1296: select.change(function(){toggleOperator(field)});
Chris@1296:
Chris@1296: switch (filterOptions['type']){
Chris@1296: case "list":
Chris@1296: case "list_optional":
Chris@1296: case "list_status":
Chris@1296: case "list_subprojects":
Chris@1296: tr.find('td.values').append(
Chris@1296: '' +
Chris@1296: ' '
Chris@1296: );
Chris@1296: select = tr.find('td.values select');
Chris@1296: if (values.length > 1) {select.attr('multiple', true)};
Chris@1296: for (i=0;i');
Chris@1296: if ($.isArray(filterValue)) {
Chris@1296: option.val(filterValue[1]).text(filterValue[0]);
Chris@1296: if ($.inArray(filterValue[1], values) > -1) {option.attr('selected', true);}
Chris@1296: } else {
Chris@1296: option.val(filterValue).text(filterValue);
Chris@1296: if ($.inArray(filterValue, values) > -1) {option.attr('selected', true);}
Chris@1296: }
Chris@1296: select.append(option);
Chris@1296: }
Chris@1296: break;
Chris@1296: case "date":
Chris@1296: case "date_past":
Chris@1296: tr.find('td.values').append(
Chris@1296: '' +
Chris@1296: ' ' +
Chris@1296: ' '+labelDayPlural+''
Chris@1296: );
Chris@1296: $('#values_'+fieldId+'_1').val(values[0]).datepicker(datepickerOptions);
Chris@1296: $('#values_'+fieldId+'_2').val(values[1]).datepicker(datepickerOptions);
Chris@1296: $('#values_'+fieldId).val(values[0]);
Chris@1296: break;
Chris@1296: case "string":
Chris@1296: case "text":
Chris@1296: tr.find('td.values').append(
Chris@1296: ''
Chris@1296: );
Chris@1296: $('#values_'+fieldId).val(values[0]);
Chris@1296: break;
Chris@1296: case "relation":
Chris@1296: tr.find('td.values').append(
Chris@1296: '' +
Chris@1296: ''
Chris@1296: );
Chris@1296: $('#values_'+fieldId).val(values[0]);
Chris@1296: select = tr.find('td.values select');
Chris@1296: for (i=0;i');
Chris@1296: option.val(filterValue[1]).text(filterValue[0]);
Chris@1296: if (values[0] == filterValue[1]) {option.attr('selected', true)};
Chris@1296: select.append(option);
Chris@1296: }
Chris@1296: case "integer":
Chris@1296: case "float":
Chris@1296: tr.find('td.values').append(
Chris@1296: '' +
Chris@1296: ' '
Chris@1296: );
Chris@1296: $('#values_'+fieldId+'_1').val(values[0]);
Chris@1296: $('#values_'+fieldId+'_2').val(values[1]);
Chris@1296: break;
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function toggleFilter(field) {
Chris@1296: var fieldId = field.replace('.', '_');
Chris@1296: if ($('#cb_' + fieldId).is(':checked')) {
Chris@1296: $("#operators_" + fieldId).show().removeAttr('disabled');
Chris@1296: toggleOperator(field);
Chris@1296: } else {
Chris@1296: $("#operators_" + fieldId).hide().attr('disabled', true);
Chris@1296: enableValues(field, []);
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function enableValues(field, indexes) {
Chris@1296: var fieldId = field.replace('.', '_');
Chris@1296: $('#tr_'+fieldId+' td.values .value').each(function(index) {
Chris@1296: if ($.inArray(index, indexes) >= 0) {
Chris@1296: $(this).removeAttr('disabled');
Chris@1296: $(this).parents('span').first().show();
Chris@1296: } else {
Chris@1296: $(this).val('');
Chris@1296: $(this).attr('disabled', true);
Chris@1296: $(this).parents('span').first().hide();
Chris@1296: }
Chris@1296:
Chris@1296: if ($(this).hasClass('group')) {
Chris@1296: $(this).addClass('open');
Chris@1296: } else {
Chris@1296: $(this).show();
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function toggleOperator(field) {
Chris@1296: var fieldId = field.replace('.', '_');
Chris@1296: var operator = $("#operators_" + fieldId);
Chris@1296: switch (operator.val()) {
Chris@1296: case "!*":
Chris@1296: case "*":
Chris@1296: case "t":
Chris@1296: case "w":
Chris@1296: case "o":
Chris@1296: case "c":
Chris@1296: enableValues(field, []);
Chris@1296: break;
Chris@1296: case "><":
Chris@1296: enableValues(field, [0,1]);
Chris@1296: break;
Chris@1296: case "t+":
Chris@1296: case ">t-":
Chris@1296: case "= 10) return false;
Chris@1296: fileFieldCount++;
Chris@1296: var s = fields.children('span').first().clone();
Chris@1296: s.children('input.file').attr('name', "attachments[" + fileFieldCount + "][file]").val('');
Chris@1296: s.children('input.description').attr('name', "attachments[" + fileFieldCount + "][description]").val('');
Chris@1296: fields.append(s);
Chris@1296: }
Chris@1296:
Chris@1296: function removeFileField(el) {
Chris@1296: var fields = $('#attachments_fields');
Chris@1296: var s = $(el).parents('span').first();
Chris@1296: if (fields.children().length > 1) {
Chris@1296: s.remove();
Chris@1296: } else {
Chris@1296: s.children('input.file').val('');
Chris@1296: s.children('input.description').val('');
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function checkFileSize(el, maxSize, message) {
Chris@1296: var files = el.files;
Chris@1296: if (files) {
Chris@1296: for (var i=0; i maxSize) {
Chris@1296: alert(message);
Chris@1296: el.value = "";
Chris@1296: }
Chris@1296: }
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function showTab(name) {
Chris@1296: $('div#content .tab-content').hide();
Chris@1296: $('div.tabs a').removeClass('selected');
Chris@1296: $('#tab-content-' + name).show();
Chris@1296: $('#tab-' + name).addClass('selected');
Chris@1296: return false;
Chris@1296: }
Chris@1296:
Chris@1296: function moveTabRight(el) {
Chris@1296: var lis = $(el).parents('div.tabs').first().find('ul').children();
Chris@1296: var tabsWidth = 0;
Chris@1296: var i = 0;
Chris@1296: lis.each(function(){
Chris@1296: if ($(this).is(':visible')) {
Chris@1296: tabsWidth += $(this).width() + 6;
Chris@1296: }
Chris@1296: });
Chris@1296: if (tabsWidth < $(el).parents('div.tabs').first().width() - 60) { return; }
Chris@1296: while (i0) {
Chris@1296: lis.eq(i-1).show();
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function displayTabsButtons() {
Chris@1296: var lis;
Chris@1296: var tabsWidth = 0;
Chris@1296: var el;
Chris@1296: $('div.tabs').each(function() {
Chris@1296: el = $(this);
Chris@1296: lis = el.find('ul').children();
Chris@1296: lis.each(function(){
Chris@1296: if ($(this).is(':visible')) {
Chris@1296: tabsWidth += $(this).width() + 6;
Chris@1296: }
Chris@1296: });
Chris@1296: if ((tabsWidth < el.width() - 60) && (lis.first().is(':visible'))) {
Chris@1296: el.find('div.tabs-buttons').hide();
Chris@1296: } else {
Chris@1296: el.find('div.tabs-buttons').show();
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function setPredecessorFieldsVisibility() {
Chris@1296: var relationType = $('#relation_relation_type');
Chris@1296: if (relationType.val() == "precedes" || relationType.val() == "follows") {
Chris@1296: $('#predecessor_fields').show();
Chris@1296: } else {
Chris@1296: $('#predecessor_fields').hide();
Chris@1296: }
Chris@1296: }
Chris@1296:
Chris@1296: function showModal(id, width) {
Chris@1296: var el = $('#'+id).first();
Chris@1296: if (el.length == 0 || el.is(':visible')) {return;}
Chris@1296: var title = el.find('h3.title').text();
Chris@1296: el.dialog({
Chris@1296: width: width,
Chris@1296: modal: true,
Chris@1296: resizable: false,
Chris@1296: dialogClass: 'modal',
Chris@1296: title: title
Chris@1296: });
Chris@1296: el.find("input[type=text], input[type=submit]").first().focus();
Chris@1296: }
Chris@1296:
Chris@1296: function hideModal(el) {
Chris@1296: var modal;
Chris@1296: if (el) {
Chris@1296: modal = $(el).parents('.ui-dialog-content');
Chris@1296: } else {
Chris@1296: modal = $('#ajax-modal');
Chris@1296: }
Chris@1296: modal.dialog("close");
Chris@1296: }
Chris@1296:
Chris@1296: function submitPreview(url, form, target) {
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: type: 'post',
Chris@1296: data: $('#'+form).serialize(),
Chris@1296: success: function(data){
Chris@1296: $('#'+target).html(data);
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function collapseScmEntry(id) {
Chris@1296: $('.'+id).each(function() {
Chris@1296: if ($(this).hasClass('open')) {
Chris@1296: collapseScmEntry($(this).attr('id'));
Chris@1296: }
Chris@1296: $(this).hide();
Chris@1296: });
Chris@1296: $('#'+id).removeClass('open');
Chris@1296: }
Chris@1296:
Chris@1296: function expandScmEntry(id) {
Chris@1296: $('.'+id).each(function() {
Chris@1296: $(this).show();
Chris@1296: if ($(this).hasClass('loaded') && !$(this).hasClass('collapsed')) {
Chris@1296: expandScmEntry($(this).attr('id'));
Chris@1296: }
Chris@1296: });
Chris@1296: $('#'+id).addClass('open');
Chris@1296: }
Chris@1296:
Chris@1296: function scmEntryClick(id, url) {
Chris@1296: el = $('#'+id);
Chris@1296: if (el.hasClass('open')) {
Chris@1296: collapseScmEntry(id);
Chris@1296: el.addClass('collapsed');
Chris@1296: return false;
Chris@1296: } else if (el.hasClass('loaded')) {
Chris@1296: expandScmEntry(id);
Chris@1296: el.removeClass('collapsed');
Chris@1296: return false;
Chris@1296: }
Chris@1296: if (el.hasClass('loading')) {
Chris@1296: return false;
Chris@1296: }
Chris@1296: el.addClass('loading');
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: success: function(data){
Chris@1296: el.after(data);
Chris@1296: el.addClass('open').addClass('loaded').removeClass('loading');
Chris@1296: }
Chris@1296: });
Chris@1296: return true;
Chris@1296: }
Chris@1296:
Chris@1296: function randomKey(size) {
Chris@1296: var chars = new Array('0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z');
Chris@1296: var key = '';
Chris@1296: for (i = 0; i < size; i++) {
Chris@1296: key += chars[Math.floor(Math.random() * chars.length)];
Chris@1296: }
Chris@1296: return key;
Chris@1296: }
Chris@1296:
Chris@1296: // Can't use Rails' remote select because we need the form data
Chris@1296: function updateIssueFrom(url) {
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: type: 'post',
Chris@1296: data: $('#issue-form').serialize()
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function updateBulkEditFrom(url) {
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: type: 'post',
Chris@1296: data: $('#bulk_edit_form').serialize()
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function observeAutocompleteField(fieldId, url) {
Chris@1296: $(document).ready(function() {
Chris@1296: $('#'+fieldId).autocomplete({
Chris@1296: source: url,
Chris@1296: minLength: 2
Chris@1296: });
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function observeSearchfield(fieldId, targetId, url) {
Chris@1296: $('#'+fieldId).each(function() {
Chris@1296: var $this = $(this);
Chris@1296: $this.attr('data-value-was', $this.val());
Chris@1296: var check = function() {
Chris@1296: var val = $this.val();
Chris@1296: if ($this.attr('data-value-was') != val){
Chris@1296: $this.attr('data-value-was', val);
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: type: 'get',
Chris@1296: data: {q: $this.val()},
Chris@1296: success: function(data){ $('#'+targetId).html(data); },
Chris@1296: beforeSend: function(){ $this.addClass('ajax-loading'); },
Chris@1296: complete: function(){ $this.removeClass('ajax-loading'); }
Chris@1296: });
Chris@1296: }
Chris@1296: };
Chris@1296: var reset = function() {
Chris@1296: if (timer) {
Chris@1296: clearInterval(timer);
Chris@1296: timer = setInterval(check, 300);
Chris@1296: }
Chris@1296: };
Chris@1296: var timer = setInterval(check, 300);
Chris@1296: $this.bind('keyup click mousemove', reset);
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: function observeProjectModules() {
Chris@1296: var f = function() {
Chris@1296: /* Hides trackers and issues custom fields on the new project form when issue_tracking module is disabled */
Chris@1296: if ($('#project_enabled_module_names_issue_tracking').attr('checked')) {
Chris@1296: $('#project_trackers').show();
Chris@1296: }else{
Chris@1296: $('#project_trackers').hide();
Chris@1296: }
Chris@1296: };
Chris@1296:
Chris@1296: $(window).load(f);
Chris@1296: $('#project_enabled_module_names_issue_tracking').change(f);
Chris@1296: }
Chris@1296:
Chris@1296: function initMyPageSortable(list, url) {
Chris@1296: $('#list-'+list).sortable({
Chris@1296: connectWith: '.block-receiver',
Chris@1296: tolerance: 'pointer',
Chris@1296: update: function(){
Chris@1296: $.ajax({
Chris@1296: url: url,
Chris@1296: type: 'post',
Chris@1296: data: {'blocks': $.map($('#list-'+list).children(), function(el){return $(el).attr('id');})}
Chris@1296: });
Chris@1296: }
Chris@1296: });
Chris@1296: $("#list-top, #list-left, #list-right").disableSelection();
Chris@1296: }
Chris@1296:
Chris@1296: var warnLeavingUnsavedMessage;
Chris@1296: function warnLeavingUnsaved(message) {
Chris@1296: warnLeavingUnsavedMessage = message;
Chris@1296:
Chris@1296: $('form').submit(function(){
Chris@1296: $('textarea').removeData('changed');
Chris@1296: });
Chris@1296: $('textarea').change(function(){
Chris@1296: $(this).data('changed', 'changed');
Chris@1296: });
Chris@1296: window.onbeforeunload = function(){
Chris@1296: var warn = false;
Chris@1296: $('textarea').blur().each(function(){
Chris@1296: if ($(this).data('changed')) {
Chris@1296: warn = true;
Chris@1296: }
Chris@1296: });
Chris@1296: if (warn) {return warnLeavingUnsavedMessage;}
Chris@1296: };
Chris@1296: };
Chris@1296:
Chris@1296: $(document).ready(function(){
Chris@1296: $('#ajax-indicator').bind('ajaxSend', function(){
Chris@1296: if ($('.ajax-loading').length == 0) {
Chris@1296: $('#ajax-indicator').show();
Chris@1296: }
Chris@1296: });
Chris@1296: $('#ajax-indicator').bind('ajaxStop', function(){
Chris@1296: $('#ajax-indicator').hide();
Chris@1296: });
Chris@1296: });
Chris@1296:
Chris@1296: function hideOnLoad() {
Chris@1296: $('.hol').hide();
Chris@1296: }
Chris@1296:
Chris@1296: function addFormObserversForDoubleSubmit() {
Chris@1296: $('form[method=post]').each(function() {
Chris@1296: if (!$(this).hasClass('multiple-submit')) {
Chris@1296: $(this).submit(function(form_submission) {
Chris@1296: if ($(form_submission.target).attr('data-submitted')) {
Chris@1296: form_submission.preventDefault();
Chris@1296: } else {
Chris@1296: $(form_submission.target).attr('data-submitted', true);
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296: });
Chris@1296: }
Chris@1296:
Chris@1296: $(document).ready(hideOnLoad);
Chris@1296: $(document).ready(addFormObserversForDoubleSubmit);