/* vim: set expandtab sw=4 ts=4 sts=4: */ /** * function used in or for navigation panel * * @package phpMyAdmin-Navigation */ /** * Executed on page load */ $(function() { if (! $('#pma_navigation').length) { // Don't bother running any code if the navigation is not even on the page return; } // Do not let the page reload on submitting the fast filter $(document).on('submit', '.fast_filter', function(event) { event.preventDefault(); }); // Fire up the resize handlers new ResizeHandler(); /** * opens/closes (hides/shows) tree elements * loads data via ajax */ $('#pma_navigation_tree a.expander').live('click', function(event) { event.preventDefault(); event.stopImmediatePropagation(); var $icon = $(this).find('img'); if ($icon.is('.ic_b_plus')) { expandTreeNode($(this)); } else { collapseTreeNode($(this)); } }); /** * Register event handler for click on the reload * navigation icon at the top of the panel */ $('#pma_navigation_reload').live('click', function (event) { event.preventDefault(); $('#pma_navigation .throbber') .first() .css('visibility', 'visible'); PMA_reloadNavigation(); }); /** * Bind all "fast filter" events */ $('#pma_navigation_tree li.fast_filter span') .live('click', PMA_fastFilter.events.clear); $('#pma_navigation_tree li.fast_filter input.searchClause') .live('focus', PMA_fastFilter.events.focus) .live('blur', PMA_fastFilter.events.blur) .live('keyup', PMA_fastFilter.events.keyup); /** * Ajax handler for pagination */ $('#pma_navigation_tree div.pageselector a.ajax').live('click', function (event) { event.preventDefault(); PMA_navigationTreePagination($(this)); }); /** * Node highlighting */ $('#pma_navigation_tree.highlight li:not(.fast_filter)').live( 'mouseover', function () { if ($('li:visible', this).length == 0) { $(this).addClass('activePointer'); } } ); $('#pma_navigation_tree.highlight li:not(.fast_filter)').live( 'mouseout', function () { $(this).removeClass('activePointer'); } ); /** * Jump to recent table */ $('#recentTable').live('change', function() { if (this.value != '') { var arr = jQuery.parseJSON(this.value); var $form = $(this).closest('form'); $form.find('input[name=db]').val(arr['db']); $form.find('input[name=table]').val(arr['table']); $form.submit(); } }); /** Create a Routine, Trigger or Event */ $('li.new_procedure a.ajax, li.new_function a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('routine'); dialog.editorDialog(1, $(this)) }); $('li.new_trigger a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('trigger'); dialog.editorDialog(1, $(this)) }); $('li.new_event a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('event'); dialog.editorDialog(1, $(this)) }); /** Edit Routines, Triggers and Events */ $('li.procedure > a.ajax, li.function > a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('routine'); dialog.editorDialog(0, $(this)) }); $('li.trigger > a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('trigger'); dialog.editorDialog(0, $(this)) }); $('li.event > a.ajax').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object('event'); dialog.editorDialog(0, $(this)) }); /** Export Routines, Triggers and Events */ $('li.procedure a.ajax img, li.function a.ajax img, li.trigger a.ajax img, li.event a.ajax img').live('click', function (event) { event.preventDefault(); var dialog = new RTE.object(); dialog.exportDialog($(this).parent()) }); /** New index */ $('li.new_index a.ajax').live('click', function (event) { event.preventDefault(); var url = $(this).attr('href').substr( $(this).attr('href').indexOf('?') + 1 ) + '&ajax_request=true'; var title = PMA_messages['strAddIndex']; indexEditorDialog(url, title); }); /** Edit index */ $('li.index a.ajax').live('click', function (event) { event.preventDefault(); var url = $(this).attr('href').substr( $(this).attr('href').indexOf('?') + 1 ) + '&ajax_request=true'; var title = PMA_messages['strEditIndex']; indexEditorDialog(url, title); }); /** New view */ $('li.new_view a.ajax').live('click', function (event) { event.preventDefault(); PMA_createViewDialog($(this)); }); PMA_showCurrentNavigation(); }); /** * Expands a node in navigation tree. * * @param $expandElem expander * @param callback callback function * * @returns void */ function expandTreeNode($expandElem, callback) { var $children = $expandElem.closest('li').children('div.list_container'); var $icon = $expandElem.find('img'); if ($expandElem.hasClass('loaded')) { if ($icon.is('.ic_b_plus')) { $icon.removeClass('ic_b_plus').addClass('ic_b_minus'); $children.show('fast'); } if (callback && typeof callback == 'function') { callback.call(); } } else { var $throbber = $('#pma_navigation .throbber') .first() .clone() .css('visibility', 'visible') .click(false); $icon.hide(); $throbber.insertBefore($icon); loadChildNodes($expandElem, function(data) { if (data.success === true) { var $destination = $expandElem.closest('li'); $icon.removeClass('ic_b_plus').addClass('ic_b_minus'); $destination .children('div.list_container') .show('fast'); if ($destination.find('ul > li').length == 1) { $destination.find('ul > li') .find('a.expander.container') .click(); } if (callback && typeof callback == 'function') { callback.call(); } } else { PMA_ajaxShowMessage(data.error, false); } $icon.show(); $throbber.remove(); }); } $expandElem.blur(); } /** * Auto-scrolls the newly chosen database * * @param object $element The element to set to view * @param object $container The container srollable element * */ function scrollToView($element, $container) { var elementOffset = $element.offset(), containerOffset = $container.offset(); if(elementOffset != undefined && containerOffset != undefined) { var pushToOffset = elementOffset.top - containerOffset.top + $container.scrollTop(); $('#pma_navigation_tree_content').stop().animate({ scrollTop: pushToOffset }); } } /** * Collapses a node in navigation tree. * * @param $expandElem expander * * @returns void */ function collapseTreeNode($expandElem) { var $children = $expandElem.closest('li').children('div.list_container'); var $icon = $expandElem.find('img'); if ($expandElem.hasClass('loaded')) { if ($icon.is('.ic_b_minus')) { $icon.removeClass('ic_b_minus').addClass('ic_b_plus'); $children.hide('fast'); } } $expandElem.blur(); } /** * Loads child items of a node and executes a given callback * * @param $expandElem expander * @param callback callback function * * @returns void */ function loadChildNodes($expandElem, callback) { var $destination = $expandElem.closest('li'); var searchClause = PMA_fastFilter.getSearchClause(); var searchClause2 = PMA_fastFilter.getSearchClause2($expandElem); var params = { aPath: $expandElem.find('span.aPath').text(), vPath: $expandElem.find('span.vPath').text(), pos: $expandElem.find('span.pos').text(), pos2_name: $expandElem.find('span.pos2_name').text(), pos2_value: $expandElem.find('span.pos2_value').text(), searchClause: searchClause, searchClause2: searchClause2 }; var url = $('#pma_navigation').find('a.navigation_url').attr('href'); $.get(url, params, function (data) { if (data.success === true) { $expandElem.addClass('loaded'); $destination.find('div.list_container').remove(); // FIXME: Hack, there shouldn't be a list container there $destination.append(data.message); if (callback && typeof callback == 'function') { callback(data); } } }); } /** * Expand the navigation and highlight the current database or table/view * * @returns void */ function PMA_showCurrentNavigation() { var db = PMA_commonParams.get('db'); var table = PMA_commonParams.get('table'); $('#pma_navigation_tree') .find('li.selected') .removeClass('selected'); if (db && table) { // if we are at the table/view level // open the database in the tree var $dbItem = highlightLoadedItem( $('#pma_navigation_tree > div'), db, 'database', false, false ); if ($dbItem) { // open the table in the tree and select it var $expander = $dbItem.children('div:first').children('a.expander'); // if not loaded or loaded but collapsed if (! $expander.hasClass('loaded') || $expander.find('img').is('.ic_b_plus') ) { expandTreeNode($expander, function() { loadAndHighlightTableOrView($dbItem, table); }); } else { loadAndHighlightTableOrView($dbItem, table); } } } else if (db) { // if we are at the database level // open in the tree and select the database highlightLoadedItem( $('#pma_navigation_tree > div'), db, 'database', true, true ); } function highlightLoadedItem($container, name, clazz, doSelect, doOpen) { var ret = false; $container.children('ul').children('li').each(function() { var $li = $(this); // this is a navigation group, recurse if ($li.is('.navGroup')) { var $container = $li.children('div.list_container'); var $childRet = highlightLoadedItem( $container, name, clazz, doSelect, doOpen ); if ($childRet) { ret = $childRet; return false; } } else { // this is a real navigation item // name and class matches if ($li.is('.' + clazz) && $li.children('a').text() == name) { if (doSelect) { $li.addClass('selected'); if (! doOpen) { // if the node will be opened no point scrolling now scrollToView($li, $('#pma_navigation_tree_content')); } } if (doOpen) { var $expander = $li.find('div:first').children('a.expander'); if ($expander.length > 0) { expandTreeNode($expander, function() { scrollToView($li, $('#pma_navigation_tree_content')); }); } } // taverse up and expand and parent navigation groups $li.parents('.navGroup').each(function() { $cont = $(this).children('div.list_container'); if (! $cont.is(':visible')) { $(this) .children('div:first') .children('a.expander') .click(); } }); ret = $li; return false; } } }); return ret; } function loadAndHighlightTableOrView($dbItem, table) { var $container = $dbItem.children('div.list_container'); var $tableContainer = $container .children('ul') .children('li.tableContainer'); var $viewContainer = $container .children('ul') .children('li.viewContainer'); if ($tableContainer.length > 0) { var $expander = $tableContainer .children('div:first') .children('a.expander'); if (! $expander.hasClass('loaded') ) { loadChildNodes($expander, function(data) { highlightTableOrView($tableContainer, $viewContainer, table); }); } else { highlightTableOrView($tableContainer, $viewContainer, table); } } else if ($viewContainer.length > 0) { highlightView($viewContainer, table); } else { // no containers, highlight the item highlightLoadedItem($container, table, 'table', true, false); } } function highlightTableOrView($tableContainer, $viewContainer, table) { if (isItemInContainer($tableContainer, table, 'table')) { var $expander = $tableContainer .children('div:first') .children('a.expander'); if ($expander.find('img').is('.ic_b_plus')) { expandTreeNode($expander); } highlightLoadedItem( $tableContainer.children('div.list_container'), table, 'table', true, false ); } else if ($viewContainer.length > 0) { highlightView($viewContainer, table); } } function isItemInContainer($container, name, clazz) { $items = $container.find('li.' + clazz); var found = false; $items.each(function() { if ($(this).children('a').text() == name) { found = true; return false; } }); return found; } function highlightView($viewContainer, view) { var $expander = $viewContainer .children('div:first') .children('a.expander'); if (! $expander.hasClass('loaded') || $expander.find('img').is('.ic_b_plus') ) { expandTreeNode($expander, function() { highlightLoadedItem( $viewContainer.children('div.list_container'), view, 'view', true, false ); }); } else { highlightLoadedItem( $viewContainer.children('div.list_container'), view, 'view', true, false ); } } } /** * Reloads the whole navigation tree while preserving its state * * @param function the callback function * @return void */ function PMA_reloadNavigation(callback) { var params = { reload: true, pos: $('#pma_navigation_tree').find('a.expander:first > span.pos').text() }; // Traverse the navigation tree backwards to generate all the actual // and virtual paths, as well as the positions in the pagination at // various levels, if necessary. var count = 0; $('#pma_navigation_tree').find('a.expander:visible').each(function () { if ($(this).find('img').is('.ic_b_minus') && $(this).closest('li').find('div.list_container .ic_b_minus').length == 0 ) { params['n' + count + '_aPath'] = $(this).find('span.aPath').text(); params['n' + count + '_vPath'] = $(this).find('span.vPath').text(); var pos2_name = $(this).find('span.pos2_name').text(); if (! pos2_name) { pos2_name = $(this) .parent() .parent() .find('span.pos2_name:last') .text(); } var pos2_value = $(this).find('span.pos2_value').text(); if (! pos2_value) { pos2_value = $(this) .parent() .parent() .find('span.pos2_value:last') .text(); } params['n' + count + '_pos2_name'] = pos2_name; params['n' + count + '_pos2_value'] = pos2_value; params['n' + count + '_pos3_name'] = $(this).find('span.pos3_name').text(); params['n' + count + '_pos3_value'] = $(this).find('span.pos3_value').text(); count++; } }); var url = $('#pma_navigation').find('a.navigation_url').attr('href'); $.post(url, params, function (data) { // Hide throbber if it's visible $('#pma_navigation .throbber') .first() .css('visibility', 'hidden'); if (data.success) { $('#pma_navigation_tree').html(data.message).children('div').show(); PMA_showCurrentNavigation(); // Fire the callback, if any if (typeof callback === 'function') { callback.call(); } } else { PMA_ajaxShowMessage(data.error); } }); } /** * Handles any requests to change the page in a branch of a tree * * This can be called from link click or select change event handlers * * @param object $this A jQuery object that points to the element that * initiated the action of changing the page * * @return void */ function PMA_navigationTreePagination($this) { var $msgbox = PMA_ajaxShowMessage(); var isDbSelector = $this.closest('div.pageselector').is('.dbselector'); if ($this[0].tagName == 'A') { var url = $this.attr('href'); var params = 'ajax_request=true'; } else { // tagName == 'SELECT' var url = 'navigation.php'; var params = $this.closest("form").serialize() + '&ajax_request=true'; } var searchClause = PMA_fastFilter.getSearchClause(); if (searchClause) { params += '&searchClause=' + encodeURIComponent(searchClause); } if (isDbSelector) { params += '&full=true'; } else { var searchClause2 = PMA_fastFilter.getSearchClause2($this); if (searchClause2) { params += '&searchClause2=' + encodeURIComponent(searchClause2); } } $.post(url, params, function (data) { PMA_ajaxRemoveMessage($msgbox); if (data.success) { if (isDbSelector) { var val = PMA_fastFilter.getSearchClause(); $('#pma_navigation_tree') .html(data.message) .children('div') .show(); if (val) { $('#pma_navigation_tree') .find('li.fast_filter input.searchClause') .val(val); } } else { var $parent = $this.closest('div.list_container').parent(); var val = PMA_fastFilter.getSearchClause2($this); $this.closest('div.list_container').html( $(data.message).children().show() ); if (val) { $parent.find('li.fast_filter input.searchClause').val(val); } $parent.find('span.pos2_value:first').text( $parent.find('span.pos2_value:last').text() ); $parent.find('span.pos3_value:first').text( $parent.find('span.pos3_value:last').text() ); } } else { PMA_ajaxShowMessage(data.error); } }); } /** * @var ResizeHandler Custom object that manages the resizing of the navigation * * XXX: Must only be ever instanciated once * XXX: Inside event handlers the 'this' object is accessed as 'event.data.resize_handler' */ var ResizeHandler = function () { /** * Whether the user has initiated a resize operation */ this.active = false; /** * @var int panel_width Used by the collapser to know where to go * back to when uncollapsing the panel */ this.panel_width = 0; /** * @var string left Used to provide support for RTL languages */ this.left = $('html').attr('dir') == 'ltr' ? 'left' : 'right'; /** * Adjusts the width of the navigation panel to the specified value * * @param int pos Navigation width in pixels * * @return void */ this.setWidth = function (pos) { var $resizer = $('#pma_navigation_resizer'); var resizer_width = $resizer.width(); var $collapser = $('#pma_navigation_collapser'); $('#pma_navigation').width(pos); $('body').css('margin-' + this.left, pos + 'px'); $("#floating_menubar") .css('margin-' + this.left, (pos + resizer_width) + 'px'); $resizer.css(this.left, pos + 'px'); if (pos === 0) { $collapser .css(this.left, pos + resizer_width) .html(this.getSymbol(pos)) .prop('title', PMA_messages['strShowPanel']); } else { $collapser .css(this.left, pos) .html(this.getSymbol(pos)) .prop('title', PMA_messages['strHidePanel']); } setTimeout(function (){ $(window).trigger('resize'); }, 4); }; /** * Returns the horizontal position of the mouse, * relative to the outer side of the navigation panel * * @param int pos Navigation width in pixels * * @return void */ this.getPos = function (event) { var pos = event.pageX; var windowWidth = $(window).width(); if (this.left != 'left') { pos = windowWidth - event.pageX; } if (pos < 0) { pos = 0; } else if (pos + 100 >= windowWidth) { pos = windowWidth - 100; } else { this.panel_width = 0; } return pos; }; /** * Returns the HTML code for the arrow symbol used in the collapser * * @param int width The width of the panel * * @return string */ this.getSymbol = function (width) { if (this.left == 'left') { if (width == 0) { return '→'; } else { return '←'; } } else { if (width == 0) { return '←'; } else { return '→'; } } }; /** * Event handler for initiating a resize of the panel * * @param object e Event data (contains a reference to resizeHandler) * * @return void */ this.mousedown = function (event) { event.preventDefault(); event.data.resize_handler.active = true; $('body').css('cursor', 'col-resize'); }; /** * Event handler for terminating a resize of the panel * * @param object e Event data (contains a reference to resizeHandler) * * @return void */ this.mouseup = function (event) { if (event.data.resize_handler.active) { event.data.resize_handler.active = false; $('body').css('cursor', ''); $.cookie('pma_navi_width', event.data.resize_handler.getPos(event)); $('#topmenu').menuResizer('resize'); } }; /** * Event handler for updating the panel during a resize operation * * @param object e Event data (contains a reference to resizeHandler) * * @return void */ this.mousemove = function (event) { if (event.data && event.data.resize_handler && event.data.resize_handler.active) { event.preventDefault(); var pos = event.data.resize_handler.getPos(event); event.data.resize_handler.setWidth(pos); } }; /** * Event handler for collapsing the panel * * @param object e Event data (contains a reference to resizeHandler) * * @return void */ this.collapse = function (event) { event.preventDefault(); event.data.active = false; var panel_width = event.data.resize_handler.panel_width; var width = $('#pma_navigation').width(); if (width === 0 && panel_width === 0) { panel_width = 240; } event.data.resize_handler.setWidth(panel_width); event.data.resize_handler.panel_width = width; }; /** * Even thandler for resizing the navigation tree height on window resize * * @return void */ this.treeResize = function (event) { var $nav = $("#pma_navigation"), $nav_tree = $("#pma_navigation_tree"), $nav_header = $("#pma_navigation_header"), $nav_tree_content = $("#pma_navigation_tree_content"); $nav_tree.height($nav.height() - $nav_header.height()); $nav_tree_content.height($nav_tree.height() - $nav_tree_content.position().top); }; /* Initialisation section begins here */ if ($.cookie('pma_navi_width')) { // If we have a cookie, set the width of the panel to its value var pos = Math.abs(parseInt($.cookie('pma_navi_width'), 10) || 0); this.setWidth(pos); $('#topmenu').menuResizer('resize'); } // Register the events for the resizer and the collapser $('#pma_navigation_resizer') .live('mousedown', {'resize_handler':this}, this.mousedown); $(document) .bind('mouseup', {'resize_handler':this}, this.mouseup) .bind('mousemove', {'resize_handler':this}, $.throttle(this.mousemove, 4)); var $collapser = $('#pma_navigation_collapser'); $collapser.live('click', {'resize_handler':this}, this.collapse); // Add the correct arrow symbol to the collapser $collapser.html(this.getSymbol($('#pma_navigation').width())); // Fix navigation tree height $(window).on('resize', this.treeResize); // need to call this now and then, browser might decide // to show/hide horizontal scrollbars depending on page content width setInterval(this.treeResize, 2000); }; // End of ResizeHandler /** * @var object PMA_fastFilter Handles the functionality that allows filtering * of the items in a branch of the navigation tree */ var PMA_fastFilter = { /** * Construct for the asynchronous fast filter functionality * * @param object $this A jQuery object pointing to the list container * which is the nearest parent of the fast filter * @param string searchClause The query string for the filter * * @return new PMA_fastFilter.filter object */ filter: function ($this, searchClause) { /** * @var object $this A jQuery object pointing to the list container * which is the nearest parent of the fast filter */ this.$this = $this; /** * @var bool searchClause The query string for the filter */ this.searchClause = searchClause; /** * @var object $clone A clone of the original contents * of the navigation branch before * the fast filter was applied */ this.$clone = $this.clone(); /** * @var bool swapped Whether the user clicked on the "N other results" link */ this.swapped = false; /** * @var object xhr A reference to the ajax request that is currently running */ this.xhr = null; /** * @var int timeout Used to delay the request for asynchronous search */ this.timeout = null; var $filterInput = $this.find('li.fast_filter input.searchClause'); if ( $filterInput.length != 0 && $filterInput.val() != '' && $filterInput.val() != $filterInput[0].defaultValue ) { this.request(); } }, /** * Gets the query string from the database fast filter form * * @return string */ getSearchClause: function () { var retval = ''; var $input = $('#pma_navigation_tree') .find('li.fast_filter.db_fast_filter input.searchClause'); if ($input.length && $input.val() != $input[0].defaultValue) { retval = $input.val(); } return retval; }, /** * Gets the query string from a second level item's fast filter form * The retrieval is done by trasversing the navigation tree backwards * * @return string */ getSearchClause2: function ($this) { var $filterContainer = $this.closest('div.list_container'); var $filterInput = $([]); while (1) { if ($filterContainer.find('li.fast_filter:not(.db_fast_filter) input.searchClause').length != 0) { $filterInput = $filterContainer.find('li.fast_filter:not(.db_fast_filter) input.searchClause'); break; } else if (! $filterContainer.is('div.list_container')) { break; } $filterContainer = $filterContainer .parent() .closest('div.list_container'); } var searchClause2 = ''; if ($filterInput.length != 0 && $filterInput.first().val() != $filterInput[0].defaultValue ) { searchClause2 = $filterInput.val(); } return searchClause2; }, /** * @var hash events A list of functions that are bound to DOM events * at the top of this file */ events: { focus: function (event) { var $obj = $(this).closest('div.list_container'); if (! $obj.data('fastFilter')) { $obj.data( 'fastFilter', new PMA_fastFilter.filter($obj, $(this).val()) ); } if ($(this).val() == this.defaultValue) { $(this).val(''); } else { $(this).select(); } }, blur: function (event) { if ($(this).val() == '') { $(this).val(this.defaultValue); } var $obj = $(this).closest('div.list_container'); if ($(this).val() == this.defaultValue && $obj.data('fastFilter')) { $obj.data('fastFilter').restore(); } }, keyup: function (event) { var $obj = $(this).closest('div.list_container'); var str = ''; if ($(this).val() != this.defaultValue && $(this).val() != '') { $obj.find('div.pageselector').hide(); str = $(this).val().toLowerCase(); } $obj.find('li > a').not('.container').each(function () { if ($(this).text().toLowerCase().indexOf(str) != -1) { $(this).parent().show().removeClass('hidden'); } else { $(this).parent().hide().addClass('hidden'); } }); var container_filter = function ($curr, str) { $curr.children('ul').children('li.navGroup').each(function() { var $group = $(this); $group.children('div.list_container').each(function() { container_filter($(this)); // recursive }); $group.show().removeClass('hidden'); if ($group.children('div.list_container').children('ul') .children('li').not('.hidden').length === 0) { $group.hide().addClass('hidden'); } }); }; if ($(this).closest('li.fast_filter').is('.db_fast_filter')) { container_filter($('#pma_navigation_tree_content'), str); } else { container_filter($obj, str); } if ($(this).val() != this.defaultValue && $(this).val() != '') { if (! $obj.data('fastFilter')) { $obj.data( 'fastFilter', new PMA_fastFilter.filter($obj, $(this).val()) ); } else { $obj.data('fastFilter').update($(this).val()); } } else if ($obj.data('fastFilter')) { $obj.data('fastFilter').restore(true); } }, clear: function (event) { event.stopPropagation(); // Clear the input and apply the fast filter with empty input var filter = $(this).closest('div.list_container').data('fastFilter'); if (filter) { filter.restore(); } var value = $(this).prev()[0].defaultValue; $(this).prev().val(value).trigger('keyup'); } } }; /** * Handles a change in the search clause * * @param string searchClause The query string for the filter * * @return void */ PMA_fastFilter.filter.prototype.update = function (searchClause) { if (this.searchClause != searchClause) { this.searchClause = searchClause; this.$this.find('.moreResults').remove(); this.request(); } }; /** * After a delay of 250mS, initiates a request to retrieve search results * Multiple calls to this function will always abort the previous request * * @return void */ PMA_fastFilter.filter.prototype.request = function () { var self = this; clearTimeout(self.timeout); if (self.$this.find('li.fast_filter').find('img.throbber').length == 0) { self.$this.find('li.fast_filter').append( $('
').append( $('#pma_navigation_content') .find('img.throbber') .clone() .css('visibility', 'visible') ) ); } self.timeout = setTimeout(function () { if (self.xhr) { self.xhr.abort(); } var url = $('#pma_navigation').find('a.navigation_url').attr('href'); var results = self.$this.find('li:not(.hidden):not(.fast_filter):not(.navGroup)').not('[class^=new]').length; var params = self.$this.find('> ul > li > form.fast_filter').first().serialize() + "&results=" + results; if (self.$this.find('> ul > li > form.fast_filter:first input[name=searchClause]').length == 0) { var $input = $('#pma_navigation_tree').find('li.fast_filter.db_fast_filter input.searchClause'); if ($input.length && $input.val() != $input[0].defaultValue) { params += '&searchClause=' + encodeURIComponent($input.val()); } } self.xhr = $.ajax({ url: url, type: 'post', dataType: 'json', data: params, complete: function (jqXHR) { var data = $.parseJSON(jqXHR.responseText); self.$this.find('li.fast_filter').find('div.throbber').remove(); if (data && data.results) { var $listItem = $('
  • ', {'class':'moreResults'}) .appendTo(self.$this.find('li.fast_filter')); var $link = $('', {href:'#'}) .text(data.results) .appendTo($listItem) .click(function (event) { event.preventDefault(); self.swap.apply(self, [data.message]); }); } } }); }, 250); }; /** * Replaces the contents of the navigation branch with the search results * * @param string list The search results * * @return void */ PMA_fastFilter.filter.prototype.swap = function (list) { this.swapped = true; this.$this .html($(list).html()) .children() .show() .end() .find('li.fast_filter input.searchClause') .val(this.searchClause); this.$this.data('fastFilter', this); }; /** * Restores the navigation to the original state after the fast filter is cleared * * @param bool focus Whether to also focus the input box of the fast filter * * @return void */ PMA_fastFilter.filter.prototype.restore = function (focus) { if (this.swapped) { this.swapped = false; this.$this.html(this.$clone.html()).children().show(); this.$this.data('fastFilter', this); if (focus) { this.$this.find('li.fast_filter input.searchClause').focus(); } } this.searchClause = ''; this.$this.find('.moreResults').remove(); this.$this.find('div.pageselector').show(); this.$this.find('div.throbber').remove(); };