diff options
| author | Tristan Zur <tzur@web.web.ccwn.org> | 2014-03-27 22:27:47 +0100 |
|---|---|---|
| committer | Tristan Zur <tzur@web.web.ccwn.org> | 2014-03-27 22:27:47 +0100 |
| commit | b62676ca5d3d6f6ba3f019ea3f99722e165a98d8 (patch) | |
| tree | 86722cb80f07d4569f90088eeaea2fc2f6e2ef94 /hugo/js/navigation.js | |
Diffstat (limited to 'hugo/js/navigation.js')
| -rw-r--r-- | hugo/js/navigation.js | 1079 |
1 files changed, 1079 insertions, 0 deletions
diff --git a/hugo/js/navigation.js b/hugo/js/navigation.js new file mode 100644 index 0000000..4806f43 --- /dev/null +++ b/hugo/js/navigation.js @@ -0,0 +1,1079 @@ +/* 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('li').children('a.container').each(function () { + var $group = $(this).parent().children('ul'); + if ($group.children('li').children('a.container').length > 0) { + container_filter($group); // recursive + } + $group.parent().show().removeClass('hidden'); + if ($group.children().not('.hidden').length == 0) { + $group.parent().hide().addClass('hidden'); + } + }); + }; + 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( + $('<div class="throbber"></div>').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 = $('<li />', {'class':'moreResults'}) + .appendTo(self.$this.find('li.fast_filter')); + var $link = $('<a />', {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(); +}; |
