aboutsummaryrefslogtreecommitdiffstats
path: root/sonar-server/src/main/js/application.js
diff options
context:
space:
mode:
Diffstat (limited to 'sonar-server/src/main/js/application.js')
-rw-r--r--sonar-server/src/main/js/application.js556
1 files changed, 556 insertions, 0 deletions
diff --git a/sonar-server/src/main/js/application.js b/sonar-server/src/main/js/application.js
new file mode 100644
index 00000000000..fbeef6d0c13
--- /dev/null
+++ b/sonar-server/src/main/js/application.js
@@ -0,0 +1,556 @@
+function showMessage(div_id, message) {
+ $j('#' + div_id + 'msg').html(message);
+ $j('#' + div_id).show();
+}
+function error(message) {
+ showMessage('error', message);
+}
+function warning(message) {
+ showMessage('warning', message);
+}
+function info(message) {
+ showMessage('info', message);
+}
+function toggleFav(resourceId, elt) {
+ $j.ajax({type: 'POST', dataType: 'json', url: baseUrl + '/favourites/toggle/' + resourceId,
+ success: function (data) {
+ var star = $j(elt);
+ star.removeClass('fav notfav');
+ star.addClass(data['css']);
+ star.attr('title', data['title']);
+ }});
+}
+
+function dashboardParameters() {
+ var queryString = window.location.search;
+ var parameters = "";
+
+ var matchDashboard = queryString.match(/did=\d+/);
+ if (matchDashboard && $j('#is-project-dashboard').length === 1) {
+ parameters += (matchDashboard[0] + "&");
+ }
+
+ var matchPeriod = queryString.match(/period=\d+/);
+ if (matchPeriod) {
+ // If we have a match for period, check that it is not project-specific
+ var period = parseInt(/period=(\d+)/.exec(queryString)[1]);
+ if (period <= 3) {
+ parameters += matchPeriod[0] + "&";
+ }
+ }
+
+ if (parameters !== "") {
+ parameters = "?" + parameters;
+ }
+ return parameters;
+}
+
+function resourceViewerOnBulkIssues() {
+ var issuesTab = 'tab=issues';
+ if (window.location.search.indexOf('tab=') >= 0) {
+ // If a tab is already selected
+ if (window.location.search.indexOf(issuesTab) >= 0) {
+ // If tab is issues, keep it and reload page
+ window.location.reload();
+ } else {
+ // Else, switch to issues tab
+ window.location.search = window.location.search.replace(/tab=\w+/, issuesTab);
+ }
+ } else {
+ // No tab selected, see how to add tab parameter
+ if (window.location.search.startsWith('?')) {
+ window.location.search += ('&' + issuesTab);
+ } else {
+ window.location.search += ('?' + issuesTab);
+ }
+ }
+}
+
+var SelectBox = {
+ cache: new Object(),
+ init: function (id) {
+ var box = document.getElementById(id);
+ var node;
+ SelectBox.cache[id] = [];
+ var cache = SelectBox.cache[id];
+ for (var i = 0; (node = box.options[i]); i++) {
+ cache.push({value: node.value, text: node.text, displayed: 1});
+ }
+ },
+ redisplay: function (id) {
+ // Repopulate HTML select box from cache
+ var box = document.getElementById(id);
+ // clear all options
+ box.options.length = 0;
+ for (var i = 0, j = SelectBox.cache[id].length; i < j; i++) {
+ var node = SelectBox.cache[id][i];
+ if (node.displayed) {
+ box.options[box.options.length] = new Option(node.text, node.value, false, false);
+ }
+ }
+ },
+ filter: function (id, text) {
+ // Redisplay the HTML select box, displaying only the choices containing ALL
+ // the words in text. (It's an AND search.)
+ var tokens = text.toLowerCase().split(/\s+/);
+ var node, token;
+ for (var i = 0; (node = SelectBox.cache[id][i]); i++) {
+ node.displayed = 1;
+ for (var j = 0; (token = tokens[j]); j++) {
+ if (node.text.toLowerCase().indexOf(token) == -1) {
+ node.displayed = 0;
+ }
+ }
+ }
+ SelectBox.redisplay(id);
+ },
+ delete_from_cache: function (id, value) {
+ var node, delete_index = null;
+ for (var i = 0; (node = SelectBox.cache[id][i]); i++) {
+ if (node.value == value) {
+ delete_index = i;
+ break;
+ }
+ }
+ var j = SelectBox.cache[id].length - 1;
+ for (var i = delete_index; i < j; i++) {
+ SelectBox.cache[id][i] = SelectBox.cache[id][i + 1];
+ }
+ SelectBox.cache[id].length--;
+ },
+ add_to_cache: function (id, option) {
+ SelectBox.cache[id].push({value: option.value, text: option.text, displayed: 1});
+ },
+ cache_contains: function (id, value) {
+ // Check if an item is contained in the cache
+ var node;
+ for (var i = 0; (node = SelectBox.cache[id][i]); i++) {
+ if (node.value == value) {
+ return true;
+ }
+ }
+ return false;
+ },
+ move: function (from, to) {
+ var from_box = document.getElementById(from);
+ var option;
+ for (var i = 0; (option = from_box.options[i]); i++) {
+ if (option.selected && SelectBox.cache_contains(from, option.value)) {
+ SelectBox.add_to_cache(to, {value: option.value, text: option.text, displayed: 1});
+ SelectBox.delete_from_cache(from, option.value);
+ }
+ }
+ SelectBox.redisplay(from);
+ SelectBox.redisplay(to);
+ },
+ move_all: function (from, to) {
+ var from_box = document.getElementById(from);
+ var option;
+ for (var i = 0; (option = from_box.options[i]); i++) {
+ if (SelectBox.cache_contains(from, option.value)) {
+ SelectBox.add_to_cache(to, {value: option.value, text: option.text, displayed: 1});
+ SelectBox.delete_from_cache(from, option.value);
+ }
+ }
+ SelectBox.redisplay(from);
+ SelectBox.redisplay(to);
+ },
+ sort: function (id) {
+ SelectBox.cache[id].sort(function (a, b) {
+ a = a.text.toLowerCase();
+ b = b.text.toLowerCase();
+ try {
+ if (a > b) {
+ return 1;
+ }
+ if (a < b) {
+ return -1;
+ }
+ }
+ catch (e) {
+ // silently fail on IE 'unknown' exception
+ }
+ return 0;
+ });
+ },
+ select_all: function (id) {
+ var box = document.getElementById(id);
+ for (var i = 0; i < box.options.length; i++) {
+ box.options[i].selected = 'selected';
+ }
+ }
+};
+
+
+var treemaps = {};
+
+function treemapById(id) {
+ return treemaps[id];
+}
+var TreemapContext = function (rid, label) {
+ this.rid = rid;
+ this.label = label;
+};
+
+/**
+ * HTML elements :
+ * tm-#{id} : required treemap container
+ * tm-bc-#{id} : required breadcrumb
+ * tm-loading-#{id} : optional loading icon
+ */
+var Treemap = function (id, sizeMetric, colorMetric, heightPercents) {
+ this.id = id;
+ this.sizeMetric = sizeMetric;
+ this.colorMetric = colorMetric;
+ this.breadcrumb = [];
+ treemaps[id] = this;
+ this.rootNode().height(this.rootNode().width() * heightPercents / 100);
+ this.initNodes();
+
+};
+Treemap.prototype.currentContext = function () {
+ if (this.breadcrumb.length > 0) {
+ return this.breadcrumb[this.breadcrumb.length - 1];
+ }
+ return null;
+};
+Treemap.prototype.load = function () {
+ var context = this.currentContext();
+ var output = '';
+ this.breadcrumb.forEach(function (ctx) {
+ output += ctx.label + '&nbsp;/&nbsp;';
+ });
+ $j('#tm-bc-' + this.id).html(output);
+ $j('#tm-loading-' + this.id).show();
+ var self = this;
+ $j.ajax({
+ type: "GET",
+ url: baseUrl + '/treemap/index?html_id=' + this.id + '&size_metric=' + this.sizeMetric + '&color_metric=' + this.colorMetric + '&resource=' + context.rid,
+ dataType: "html",
+ success: function (data) {
+ if (data.length > 1) {
+ self.rootNode().html(data);
+ self.initNodes();
+ } else {
+ // SONAR-3524
+ // When data is empty, do not display it, revert breadcrumb state and display a message to user
+ self.breadcrumb.pop();
+ $j("#tm-bottom-level-reached-msg-" + self.id).show();
+ }
+ $j("#tm-loading-" + self.id).hide();
+ }
+ });
+};
+Treemap.prototype.rootNode = function () {
+ return $j('#tm-' + this.id);
+};
+
+Treemap.prototype.initNodes = function () {
+ var self = this;
+ $j('#tm-' + this.id).find('a').each(function (index) {
+ $j(this).on("click", function (event) {
+ event.stopPropagation();
+ });
+ });
+ $j('#tm-' + this.id).find('[rid]').each(function (index) {
+ $j(this).on("contextmenu", function (event) {
+ event.stopPropagation();
+ event.preventDefault();
+ $j("#tm-bottom-level-reached-msg-" + self.id).hide();
+ // right click
+ if (self.breadcrumb.length > 1) {
+ self.breadcrumb.pop();
+ self.load();
+ } else if (self.breadcrumb.length == 1) {
+ $j("#tm-loading-" + self.id).show();
+ location.reload();
+ }
+ return false;
+ });
+ $j(this).on("click", function (event) {
+ var source = $j(this);
+ var rid = source.attr('rid');
+ var context = new TreemapContext(rid, source.text());
+ self.breadcrumb.push(context);
+ self.load();
+ }
+ );
+ });
+};
+
+function openModalWindow(url, options) {
+ var width = (options && options['width']) || 540;
+ var $dialog = $j('#modal');
+ if (!$dialog.length) {
+ $dialog = $j('<div id="modal" class="ui-widget-overlay"></div>').appendTo('body');
+ }
+ $j.get(url,function (html) {
+ $dialog.removeClass('ui-widget-overlay');
+ $dialog.html(html);
+ $dialog
+ .dialog({
+ dialogClass: "no-close",
+ width: width,
+ draggable: false,
+ autoOpen: false,
+ modal: true,
+ minHeight: 50,
+ resizable: false,
+ title: null,
+ close: function () {
+ $j('#modal').remove();
+ }
+ });
+ $dialog.dialog("open");
+ }).fail(function () {
+ alert("Server error. Please contact your administrator.");
+ }).always(function () {
+ $dialog.removeClass('ui-widget-overlay');
+ });
+ return false;
+}
+
+(function ($j) {
+ $j.fn.extend({
+ openModal: function() {
+ return this.each(function () {
+ var obj = $j(this);
+ var url = obj.attr('modal-url') || obj.attr('href');
+ return openModalWindow(url, {'width': obj.attr('modal-width')});
+ });
+ },
+ modal: function () {
+ return this.each(function () {
+ var obj = $j(this);
+ obj.unbind('click');
+ var $link = obj.bind('click', function () {
+ $link.openModal();
+ return false;
+ });
+ });
+ },
+ modalForm: function (ajax_options) {
+ return this.each(function () {
+ var obj = $j(this);
+ obj.submit(function (event) {
+ $j('input[type=submit]', this).attr('disabled', 'disabled');
+ $j.ajax($j.extend({
+ type: 'POST',
+ url: obj.attr('action'),
+ data: obj.serialize(),
+ success: function (data) {
+ window.location.reload();
+ },
+ error: function (xhr, textStatus, errorThrown) {
+ // If the modal window has defined a modal-error element, then returned text must be displayed in it
+ var errorElt = obj.find(".modal-error");
+ if (errorElt.length) {
+ // Hide all loading images
+ $j('.loading-image').addClass("hidden");
+ // Re activate submit button
+ $j('input[type=submit]', obj).removeAttr('disabled');
+ errorElt.show();
+ errorElt.html(xhr.responseText);
+ } else {
+ // otherwise replace modal window by the returned text
+ $j("#modal").html(xhr.responseText);
+ }
+ }
+ }, ajax_options));
+ return false;
+ });
+ });
+ }
+ });
+})(jQuery);
+
+function closeModalWindow() {
+ $j('#modal').dialog('close');
+ return false;
+}
+
+function supports_html5_storage() {
+ try {
+ return 'localStorage' in window && window['localStorage'] !== null;
+ } catch (e) {
+ return false;
+ }
+}
+
+//******************* HANDLING OF ACCORDION NAVIGATION [BEGIN] ******************* //
+
+function openAccordionItem(url, elt, updateCurrentElement) {
+ var htmlClass = 'accordion-item';
+ var currentElement = $j(elt).closest('.'+ htmlClass);
+
+ // Create loading image
+ var loadingImg = new Image();
+ loadingImg.src = baseUrl + "/images/loading.gif";
+ loadingImg.className = 'accordion-loading';
+ var loading = $j(loadingImg);
+ var existingLoading = currentElement.find('.accordion-loading');
+ if (updateCurrentElement && existingLoading.length) {
+ existingLoading.show();
+ loading.hide();
+ }
+
+ // Remove elements under current element
+ if (currentElement.length) {
+ // Fix the height in order to not change the position on the screen when removing elements under current element
+ var elementToRemove = currentElement.nextAll('.'+ htmlClass);
+ if (elementToRemove.height()) {
+ $j("#accordion-panel").height($j("#accordion-panel").height() + elementToRemove.height());
+ }
+ // Remove all accordion items after current element
+ elementToRemove.remove();
+ // Display loading image only if not already displayed (if previous call was not finished)
+ if (currentElement.next('.accordion-loading').length == 0) {
+ loading.insertAfter(currentElement);
+ }
+ } else {
+ // Current element is not in a working view, remove all working views
+ $j('.'+ htmlClass).remove();
+ // Display loading image only if not already displayed (if previous call was not finished)
+ if ($j("#accordion-panel").next('.accordion-loading').length == 0) {
+ loading.insertAfter($j("#accordion-panel"));
+ }
+ }
+
+ // Get content from url
+ var ajaxRequest = $j.ajax({
+ url: url
+ }).fail(function (jqXHR, textStatus) {
+ var error = "Server error. Please contact your administrator. The status of the error is : "+ jqXHR.status + ", textStatus is : "+ textStatus;
+ console.log(error);
+ $j("#accordion-panel").append($j('<div class="error">').append(error));
+ }).done(function (html) {
+ if (currentElement.length) {
+ var body = currentElement.find('.accordion-item-body');
+ if (!updateCurrentElement && !body.hasClass('accordion-item-body-medium')) {
+ body.addClass("accordion-item-body-medium");
+ elt.scrollIntoView(false);
+ }
+ } else {
+ $j("#accordion-panel").height('auto');
+
+ // Current element is not in a working view, remove again all working views to purge elements that could be added just before this one
+ $j('.'+ htmlClass).remove();
+ }
+
+ if (updateCurrentElement) {
+ // Fix the height in order to not change the position on the screen
+ var prevHeight = $j("#accordion-panel").height();
+ currentElement.html(html);
+ $j("#accordion-panel").height('auto');
+ var newHeight = $j("#accordion-panel").height();
+ if (prevHeight > newHeight) {
+ $j("#accordion-panel").height(prevHeight);
+ } else {
+ $j("#accordion-panel").height(newHeight);
+ }
+ } else {
+ // Add new item add the end of the panel and restore the height param
+ var nbElement = $j("."+htmlClass).size();
+ var newElement = $j('<div id="'+ htmlClass + nbElement +'" class="'+ htmlClass +'">');
+ $j("#accordion-panel").append(newElement);
+
+ // Add html after having adding the new element in the page in order to scripts (for instance for GWT) to be well executed
+ newElement.append(html);
+ $j("#accordion-panel").height('auto');
+
+ // Set the focus on the top of the current item with animation
+ if (currentElement.length) {
+ $j('html, body').animate({
+ scrollTop: currentElement.offset().top}, 500
+ );
+ }
+ }
+ loading.remove();
+ });
+ return ajaxRequest;
+}
+
+
+function expandAccordionItem(elt) {
+ var currentElement = $j(elt).closest('.accordion-item');
+ currentElement.find('.accordion-item-body').removeClass("accordion-item-body-medium");
+}
+
+//******************* HANDLING OF ACCORDION NAVIGATION [END] ******************* //
+
+
+//******************* HANDLING OF DROPDOWN MENUS [BEGIN] ******************* //
+
+var currentlyDisplayedDropdownMenu;
+
+var hideCurrentDropdownMenu = function () {
+ if (currentlyDisplayedDropdownMenu) {
+ currentlyDisplayedDropdownMenu.hide();
+ }
+ $j(document).unbind('mouseup', hideCurrentDropdownMenu);
+};
+
+var clickOnDropdownMenuLink = function (event) {
+ var link = $j(event.target).children('a');
+ if (link) {
+ var href = link.attr('href');
+ if (href && href.length > 1) {
+ // there's a real link, not a href="#"
+ window.location = href;
+ } else {
+ // otherwise, this means that the link is handled with an onclick event (for Ajax calls)
+ link.click();
+ }
+ }
+};
+
+function showDropdownMenu(menuId) {
+ showDropdownMenuOnElement($j('#' + menuId));
+}
+
+function showDropdownMenuOnElement(elt) {
+ var dropdownElt = $j(elt);
+
+ if (dropdownElt == currentlyDisplayedDropdownMenu) {
+ currentlyDisplayedDropdownMenu = "";
+ } else {
+ currentlyDisplayedDropdownMenu = dropdownElt;
+ $j(document).mouseup(hideCurrentDropdownMenu);
+
+ var dropdownChildren = dropdownElt.find('li');
+ dropdownChildren.unbind('click');
+ dropdownChildren.click(clickOnDropdownMenuLink);
+ dropdownElt.show();
+ }
+}
+
+//******************* HANDLING OF DROPDOWN MENUS [END] ******************* //
+
+function openPopup(url, popupId) {
+ window.open(url,popupId,'height=800,width=900,scrollbars=1,resizable=1');
+ return false;
+}
+
+
+jQuery(function() {
+
+ // Initialize top search
+ jQuery('#searchInput').topSearch({
+ minLength: 2,
+ results: '#searchResourcesResults',
+ spinner: '#searchingResources'
+ });
+
+
+ // Process login link in order to add the anchor
+ jQuery('#login-link').on('click', function(e) {
+ e.preventDefault();
+ var href = jQuery(this).prop('href'),
+ hash = window.location.hash;
+ if (hash.length > 0) {
+ href += decodeURIComponent(hash);
+ }
+ window.location = href;
+ });
+});
+