mirror of
https://github.com/cytopia/devilbox.git
synced 2025-01-04 03:44:16 +00:00
1722 lines
61 KiB
JavaScript
1722 lines
61 KiB
JavaScript
/* vim: set expandtab sw=4 ts=4 sts=4: */
|
|
/**
|
|
* function used in or for navigation panel
|
|
*
|
|
* @package phpMyAdmin-Navigation
|
|
*/
|
|
|
|
/* global isStorageSupported, setupConfigTabs, setupRestoreField, setupValidation */ // js/config.js
|
|
/* global RTE */ // js/rte.js
|
|
|
|
var Navigation = {};
|
|
|
|
/**
|
|
* updates the tree state in sessionStorage
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.treeStateUpdate = function () {
|
|
// update if session storage is supported
|
|
if (isStorageSupported('sessionStorage')) {
|
|
var storage = window.sessionStorage;
|
|
// try catch necessary here to detect whether
|
|
// content to be stored exceeds storage capacity
|
|
try {
|
|
storage.setItem('navTreePaths', JSON.stringify(Navigation.traverseForPaths()));
|
|
storage.setItem('server', CommonParams.get('server'));
|
|
storage.setItem('token', CommonParams.get('token'));
|
|
} catch (error) {
|
|
// storage capacity exceeded & old navigation tree
|
|
// state is no more valid, so remove it
|
|
storage.removeItem('navTreePaths');
|
|
storage.removeItem('server');
|
|
storage.removeItem('token');
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* updates the filter state in sessionStorage
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.filterStateUpdate = function (filterName, filterValue) {
|
|
if (isStorageSupported('sessionStorage')) {
|
|
var storage = window.sessionStorage;
|
|
try {
|
|
var currentFilter = $.extend({}, JSON.parse(storage.getItem('navTreeSearchFilters')));
|
|
var filter = {};
|
|
filter[filterName] = filterValue;
|
|
currentFilter = $.extend(currentFilter, filter);
|
|
storage.setItem('navTreeSearchFilters', JSON.stringify(currentFilter));
|
|
} catch (error) {
|
|
storage.removeItem('navTreeSearchFilters');
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* restores the filter state on navigation reload
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.filterStateRestore = function () {
|
|
if (isStorageSupported('sessionStorage')
|
|
&& typeof window.sessionStorage.navTreeSearchFilters !== 'undefined'
|
|
) {
|
|
var searchClauses = JSON.parse(window.sessionStorage.navTreeSearchFilters);
|
|
if (Object.keys(searchClauses).length < 1) {
|
|
return;
|
|
}
|
|
// restore database filter if present and not empty
|
|
if (searchClauses.hasOwnProperty('dbFilter')
|
|
&& searchClauses.dbFilter.length
|
|
) {
|
|
var $obj = $('#pma_navigation_tree');
|
|
if (! $obj.data('fastFilter')) {
|
|
$obj.data(
|
|
'fastFilter',
|
|
new Navigation.FastFilter.Filter($obj, '')
|
|
);
|
|
}
|
|
$obj.find('li.fast_filter.db_fast_filter input.searchClause')
|
|
.val(searchClauses.dbFilter)
|
|
.trigger('keyup');
|
|
}
|
|
// find all table filters present in the tree
|
|
var $tableFilters = $('#pma_navigation_tree li.database')
|
|
.children('div.list_container')
|
|
.find('li.fast_filter input.searchClause');
|
|
// restore table filters
|
|
$tableFilters.each(function () {
|
|
$obj = $(this).closest('div.list_container');
|
|
// aPath associated with this filter
|
|
var filterName = $(this).siblings('input[name=aPath]').val();
|
|
// if this table's filter has a state stored in storage
|
|
if (searchClauses.hasOwnProperty(filterName)
|
|
&& searchClauses[filterName].length
|
|
) {
|
|
// clear state if item is not visible,
|
|
// happens when table filter becomes invisible
|
|
// as db filter has already been applied
|
|
if (! $obj.is(':visible')) {
|
|
Navigation.filterStateUpdate(filterName, '');
|
|
return true;
|
|
}
|
|
if (! $obj.data('fastFilter')) {
|
|
$obj.data(
|
|
'fastFilter',
|
|
new Navigation.FastFilter.Filter($obj, '')
|
|
);
|
|
}
|
|
$(this).val(searchClauses[filterName])
|
|
.trigger('keyup');
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Loads child items of a node and executes a given callback
|
|
*
|
|
* @param isNode
|
|
* @param $expandElem expander
|
|
* @param callback callback function
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.loadChildNodes = function (isNode, $expandElem, callback) {
|
|
var $destination = null;
|
|
var params = null;
|
|
|
|
if (isNode) {
|
|
if (!$expandElem.hasClass('expander')) {
|
|
return;
|
|
}
|
|
$destination = $expandElem.closest('li');
|
|
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': '',
|
|
'searchClause2': ''
|
|
};
|
|
if ($expandElem.closest('ul').hasClass('search_results')) {
|
|
params.searchClause = Navigation.FastFilter.getSearchClause();
|
|
params.searchClause2 = Navigation.FastFilter.getSearchClause2($expandElem);
|
|
}
|
|
} else {
|
|
$destination = $('#pma_navigation_tree_content');
|
|
params = {
|
|
'aPath': $expandElem.attr('aPath'),
|
|
'vPath': $expandElem.attr('vPath'),
|
|
'pos': $expandElem.attr('pos'),
|
|
'pos2_name': '',
|
|
'pos2_value': '',
|
|
'searchClause': '',
|
|
'searchClause2': ''
|
|
};
|
|
}
|
|
|
|
var url = $('#pma_navigation').find('a.navigation_url').attr('href');
|
|
$.get(url, params, function (data) {
|
|
if (typeof data !== 'undefined' && data.success === true) {
|
|
$destination.find('div.list_container').remove(); // FIXME: Hack, there shouldn't be a list container there
|
|
if (isNode) {
|
|
$destination.append(data.message);
|
|
$expandElem.addClass('loaded');
|
|
} else {
|
|
$destination.html(data.message);
|
|
$destination.children()
|
|
.first()
|
|
.css({
|
|
border: '0px',
|
|
margin: '0em',
|
|
padding : '0em'
|
|
})
|
|
.slideDown('slow');
|
|
}
|
|
if (data.errors) {
|
|
var $errors = $(data.errors);
|
|
if ($errors.children().length > 0) {
|
|
$('#pma_errors').replaceWith(data.errors);
|
|
}
|
|
}
|
|
if (callback && typeof callback === 'function') {
|
|
callback(data);
|
|
}
|
|
} else if (data.redirect_flag === '1') {
|
|
if (window.location.href.indexOf('?') === -1) {
|
|
window.location.href += '?session_expired=1';
|
|
} else {
|
|
window.location.href += CommonParams.get('arg_separator') + 'session_expired=1';
|
|
}
|
|
window.location.reload();
|
|
} else {
|
|
var $throbber = $expandElem.find('img.throbber');
|
|
$throbber.hide();
|
|
var $icon = $expandElem.find('img.ic_b_plus');
|
|
$icon.show();
|
|
Functions.ajaxShowMessage(data.error, false);
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Collapses a node in navigation tree.
|
|
*
|
|
* @param $expandElem expander
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.collapseTreeNode = function ($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.slideUp('fast');
|
|
}
|
|
}
|
|
$expandElem.trigger('blur');
|
|
$children.promise().done(Navigation.treeStateUpdate);
|
|
};
|
|
|
|
/**
|
|
* 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.
|
|
*
|
|
* @return Object
|
|
*/
|
|
Navigation.traverseForPaths = function () {
|
|
var params = {
|
|
pos: $('#pma_navigation_tree').find('div.dbselector select').val()
|
|
};
|
|
if ($('#navi_db_select').length) {
|
|
return params;
|
|
}
|
|
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 pos2Name = $(this).find('span.pos2_name').text();
|
|
if (! pos2Name) {
|
|
pos2Name = $(this)
|
|
.parent()
|
|
.parent()
|
|
.find('span.pos2_name:last')
|
|
.text();
|
|
}
|
|
var pos2Value = $(this).find('span.pos2_value').text();
|
|
if (! pos2Value) {
|
|
pos2Value = $(this)
|
|
.parent()
|
|
.parent()
|
|
.find('span.pos2_value:last')
|
|
.text();
|
|
}
|
|
|
|
params['n' + count + '_pos2_name'] = pos2Name;
|
|
params['n' + count + '_pos2_value'] = pos2Value;
|
|
|
|
params['n' + count + '_pos3_name'] = $(this).find('span.pos3_name').text();
|
|
params['n' + count + '_pos3_value'] = $(this).find('span.pos3_value').text();
|
|
count++;
|
|
}
|
|
});
|
|
return params;
|
|
};
|
|
|
|
/**
|
|
* 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 Navigation.ResizeHandler();
|
|
|
|
/**
|
|
* opens/closes (hides/shows) tree elements
|
|
* loads data via ajax
|
|
*/
|
|
$(document).on('click', '#pma_navigation_tree a.expander', function (event) {
|
|
event.preventDefault();
|
|
event.stopImmediatePropagation();
|
|
var $icon = $(this).find('img');
|
|
if ($icon.is('.ic_b_plus')) {
|
|
Navigation.expandTreeNode($(this));
|
|
} else {
|
|
Navigation.collapseTreeNode($(this));
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Register event handler for click on the reload
|
|
* navigation icon at the top of the panel
|
|
*/
|
|
$(document).on('click', '#pma_navigation_reload', function (event) {
|
|
event.preventDefault();
|
|
|
|
// Find the loading symbol and show it
|
|
var $iconThrobberSrc = $('#pma_navigation').find('.throbber');
|
|
$iconThrobberSrc.show();
|
|
// TODO Why is a loading symbol both hidden, and invisible?
|
|
$iconThrobberSrc.css('visibility', '');
|
|
|
|
// Callback to be used to hide the loading symbol when done reloading
|
|
function hideNav () {
|
|
$iconThrobberSrc.hide();
|
|
}
|
|
|
|
// Reload the navigation
|
|
Navigation.reload(hideNav);
|
|
});
|
|
|
|
$(document).on('change', '#navi_db_select', function () {
|
|
if (! $(this).val()) {
|
|
CommonParams.set('db', '');
|
|
Navigation.reload();
|
|
}
|
|
$(this).closest('form').trigger('submit');
|
|
});
|
|
|
|
/**
|
|
* Register event handler for click on the collapse all
|
|
* navigation icon at the top of the navigation tree
|
|
*/
|
|
$(document).on('click', '#pma_navigation_collapse', function (event) {
|
|
event.preventDefault();
|
|
$('#pma_navigation_tree').find('a.expander').each(function () {
|
|
var $icon = $(this).find('img');
|
|
if ($icon.is('.ic_b_minus')) {
|
|
$(this).trigger('click');
|
|
}
|
|
});
|
|
});
|
|
|
|
/**
|
|
* Register event handler to toggle
|
|
* the 'link with main panel' icon on mouseenter.
|
|
*/
|
|
$(document).on('mouseenter', '#pma_navigation_sync', function (event) {
|
|
event.preventDefault();
|
|
var synced = $('#pma_navigation_tree').hasClass('synced');
|
|
var $img = $('#pma_navigation_sync').children('img');
|
|
if (synced) {
|
|
$img.removeClass('ic_s_link').addClass('ic_s_unlink');
|
|
} else {
|
|
$img.removeClass('ic_s_unlink').addClass('ic_s_link');
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Register event handler to toggle
|
|
* the 'link with main panel' icon on mouseout.
|
|
*/
|
|
$(document).on('mouseout', '#pma_navigation_sync', function (event) {
|
|
event.preventDefault();
|
|
var synced = $('#pma_navigation_tree').hasClass('synced');
|
|
var $img = $('#pma_navigation_sync').children('img');
|
|
if (synced) {
|
|
$img.removeClass('ic_s_unlink').addClass('ic_s_link');
|
|
} else {
|
|
$img.removeClass('ic_s_link').addClass('ic_s_unlink');
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Register event handler to toggle
|
|
* the linking with main panel behavior
|
|
*/
|
|
$(document).on('click', '#pma_navigation_sync', function (event) {
|
|
event.preventDefault();
|
|
var synced = $('#pma_navigation_tree').hasClass('synced');
|
|
var $img = $('#pma_navigation_sync').children('img');
|
|
if (synced) {
|
|
$img
|
|
.removeClass('ic_s_unlink')
|
|
.addClass('ic_s_link')
|
|
.attr('alt', Messages.linkWithMain)
|
|
.attr('title', Messages.linkWithMain);
|
|
$('#pma_navigation_tree')
|
|
.removeClass('synced')
|
|
.find('li.selected')
|
|
.removeClass('selected');
|
|
} else {
|
|
$img
|
|
.removeClass('ic_s_link')
|
|
.addClass('ic_s_unlink')
|
|
.attr('alt', Messages.unlinkWithMain)
|
|
.attr('title', Messages.unlinkWithMain);
|
|
$('#pma_navigation_tree').addClass('synced');
|
|
Navigation.showCurrent();
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Bind all "fast filter" events
|
|
*/
|
|
$(document).on('click', '#pma_navigation_tree li.fast_filter span', Navigation.FastFilter.events.clear);
|
|
$(document).on('focus', '#pma_navigation_tree li.fast_filter input.searchClause', Navigation.FastFilter.events.focus);
|
|
$(document).on('blur', '#pma_navigation_tree li.fast_filter input.searchClause', Navigation.FastFilter.events.blur);
|
|
$(document).on('keyup', '#pma_navigation_tree li.fast_filter input.searchClause', Navigation.FastFilter.events.keyup);
|
|
|
|
/**
|
|
* Ajax handler for pagination
|
|
*/
|
|
$(document).on('click', '#pma_navigation_tree div.pageselector a.ajax', function (event) {
|
|
event.preventDefault();
|
|
Navigation.treePagination($(this));
|
|
});
|
|
|
|
/**
|
|
* Node highlighting
|
|
*/
|
|
$(document).on(
|
|
'mouseover',
|
|
'#pma_navigation_tree.highlight li:not(.fast_filter)',
|
|
function () {
|
|
if ($('li:visible', this).length === 0) {
|
|
$(this).addClass('activePointer');
|
|
}
|
|
}
|
|
);
|
|
$(document).on(
|
|
'mouseout',
|
|
'#pma_navigation_tree.highlight li:not(.fast_filter)',
|
|
function () {
|
|
$(this).removeClass('activePointer');
|
|
}
|
|
);
|
|
|
|
/** Create a Routine, Trigger or Event */
|
|
$(document).on('click', 'li.new_procedure a.ajax, li.new_function a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('routine');
|
|
dialog.editorDialog(1, $(this));
|
|
});
|
|
$(document).on('click', 'li.new_trigger a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('trigger');
|
|
dialog.editorDialog(1, $(this));
|
|
});
|
|
$(document).on('click', 'li.new_event a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('event');
|
|
dialog.editorDialog(1, $(this));
|
|
});
|
|
|
|
/** Edit Routines, Triggers or Events */
|
|
$(document).on('click', 'li.procedure > a.ajax, li.function > a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('routine');
|
|
dialog.editorDialog(0, $(this));
|
|
});
|
|
$(document).on('click', 'li.trigger > a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('trigger');
|
|
dialog.editorDialog(0, $(this));
|
|
});
|
|
$(document).on('click', 'li.event > a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('event');
|
|
dialog.editorDialog(0, $(this));
|
|
});
|
|
|
|
/** Execute Routines */
|
|
$(document).on('click', 'li.procedure div a.ajax img,' +
|
|
' li.function div a.ajax img', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object('routine');
|
|
dialog.executeDialog($(this).parent());
|
|
});
|
|
/** Export Triggers and Events */
|
|
$(document).on('click', 'li.trigger div:eq(1) a.ajax img,' +
|
|
' li.event div:eq(1) a.ajax img', function (event) {
|
|
event.preventDefault();
|
|
var dialog = new RTE.Object();
|
|
dialog.exportDialog($(this).parent());
|
|
});
|
|
|
|
/** New index */
|
|
$(document).on('click', '#pma_navigation_tree li.new_index a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var url = $(this).attr('href').substr(
|
|
$(this).attr('href').indexOf('?') + 1
|
|
) + CommonParams.get('arg_separator') + 'ajax_request=true';
|
|
var title = Messages.strAddIndex;
|
|
Functions.indexEditorDialog(url, title);
|
|
});
|
|
|
|
/** Edit index */
|
|
$(document).on('click', 'li.index a.ajax', function (event) {
|
|
event.preventDefault();
|
|
var url = $(this).attr('href').substr(
|
|
$(this).attr('href').indexOf('?') + 1
|
|
) + CommonParams.get('arg_separator') + 'ajax_request=true';
|
|
var title = Messages.strEditIndex;
|
|
Functions.indexEditorDialog(url, title);
|
|
});
|
|
|
|
/** New view */
|
|
$(document).on('click', 'li.new_view a.ajax', function (event) {
|
|
event.preventDefault();
|
|
Functions.createViewDialog($(this));
|
|
});
|
|
|
|
/** Hide navigation tree item */
|
|
$(document).on('click', 'a.hideNavItem.ajax', function (event) {
|
|
event.preventDefault();
|
|
var argSep = CommonParams.get('arg_separator');
|
|
var params = $(this).getPostData();
|
|
params += argSep + 'ajax_request=true' + argSep + 'server=' + CommonParams.get('server');
|
|
$.ajax({
|
|
type: 'POST',
|
|
data: params,
|
|
url: $(this).attr('href'),
|
|
success: function (data) {
|
|
if (typeof data !== 'undefined' && data.success === true) {
|
|
Navigation.reload();
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
/** Display a dialog to choose hidden navigation items to show */
|
|
$(document).on('click', 'a.showUnhide.ajax', function (event) {
|
|
event.preventDefault();
|
|
var $msg = Functions.ajaxShowMessage();
|
|
var argSep = CommonParams.get('arg_separator');
|
|
var params = $(this).getPostData();
|
|
params += argSep + 'ajax_request=true';
|
|
$.post($(this).attr('href'), params, function (data) {
|
|
if (typeof data !== 'undefined' && data.success === true) {
|
|
Functions.ajaxRemoveMessage($msg);
|
|
var buttonOptions = {};
|
|
buttonOptions[Messages.strClose] = function () {
|
|
$(this).dialog('close');
|
|
};
|
|
$('<div></div>')
|
|
.attr('id', 'unhideNavItemDialog')
|
|
.append(data.message)
|
|
.dialog({
|
|
width: 400,
|
|
minWidth: 200,
|
|
modal: true,
|
|
buttons: buttonOptions,
|
|
title: Messages.strUnhideNavItem,
|
|
close: function () {
|
|
$(this).remove();
|
|
}
|
|
});
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error);
|
|
}
|
|
});
|
|
});
|
|
|
|
/** Show a hidden navigation tree item */
|
|
$(document).on('click', 'a.unhideNavItem.ajax', function (event) {
|
|
event.preventDefault();
|
|
var $tr = $(this).parents('tr');
|
|
var $hiddenTableCount = $tr.parents('tbody').children().length;
|
|
var $hideDialogBox = $tr.closest('div.ui-dialog');
|
|
var $msg = Functions.ajaxShowMessage();
|
|
var argSep = CommonParams.get('arg_separator');
|
|
var params = $(this).getPostData();
|
|
params += argSep + 'ajax_request=true' + argSep + 'server=' + CommonParams.get('server');
|
|
$.ajax({
|
|
type: 'POST',
|
|
data: params,
|
|
url: $(this).attr('href'),
|
|
success: function (data) {
|
|
Functions.ajaxRemoveMessage($msg);
|
|
if (typeof data !== 'undefined' && data.success === true) {
|
|
$tr.remove();
|
|
if ($hiddenTableCount === 1) {
|
|
$hideDialogBox.remove();
|
|
}
|
|
Navigation.reload();
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
|
|
// Add/Remove favorite table using Ajax.
|
|
$(document).on('click', '.favorite_table_anchor', function (event) {
|
|
event.preventDefault();
|
|
var $self = $(this);
|
|
var anchorId = $self.attr('id');
|
|
if ($self.data('favtargetn') !== null) {
|
|
if ($('a[data-favtargets="' + $self.data('favtargetn') + '"]').length > 0) {
|
|
$('a[data-favtargets="' + $self.data('favtargetn') + '"]').trigger('click');
|
|
return;
|
|
}
|
|
}
|
|
|
|
$.ajax({
|
|
url: $self.attr('href'),
|
|
cache: false,
|
|
type: 'POST',
|
|
data: {
|
|
'favoriteTables': (isStorageSupported('localStorage') && typeof window.localStorage.favoriteTables !== 'undefined')
|
|
? window.localStorage.favoriteTables
|
|
: '',
|
|
'server': CommonParams.get('server'),
|
|
},
|
|
success: function (data) {
|
|
if (data.changes) {
|
|
$('#pma_favorite_list').html(data.list);
|
|
$('#' + anchorId).parent().html(data.anchor);
|
|
Functions.tooltip(
|
|
$('#' + anchorId),
|
|
'a',
|
|
$('#' + anchorId).attr('title')
|
|
);
|
|
// Update localStorage.
|
|
if (isStorageSupported('localStorage')) {
|
|
window.localStorage.favoriteTables = data.favoriteTables;
|
|
}
|
|
} else {
|
|
Functions.ajaxShowMessage(data.message);
|
|
}
|
|
}
|
|
});
|
|
});
|
|
// Check if session storage is supported
|
|
if (isStorageSupported('sessionStorage')) {
|
|
var storage = window.sessionStorage;
|
|
// remove tree from storage if Navi_panel config form is submitted
|
|
$(document).on('submit', 'form.config-form', function () {
|
|
storage.removeItem('navTreePaths');
|
|
});
|
|
// Initialize if no previous state is defined
|
|
if ($('#pma_navigation_tree_content').length &&
|
|
typeof storage.navTreePaths === 'undefined'
|
|
) {
|
|
Navigation.reload();
|
|
} else if (CommonParams.get('server') === storage.server &&
|
|
CommonParams.get('token') === storage.token
|
|
) {
|
|
// Reload the tree to the state before page refresh
|
|
Navigation.reload(Navigation.filterStateRestore, JSON.parse(storage.navTreePaths));
|
|
} else {
|
|
// If the user is different
|
|
Navigation.treeStateUpdate();
|
|
Navigation.reload();
|
|
}
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Expands a node in navigation tree.
|
|
*
|
|
* @param $expandElem expander
|
|
* @param callback callback function
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.expandTreeNode = function ($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.slideDown('fast');
|
|
}
|
|
if (callback && typeof callback === 'function') {
|
|
callback.call();
|
|
}
|
|
$children.promise().done(Navigation.treeStateUpdate);
|
|
} else {
|
|
var $throbber = $('#pma_navigation').find('.throbber')
|
|
.first()
|
|
.clone()
|
|
.css({ visibility: 'visible', display: 'block' })
|
|
.on('click', false);
|
|
$icon.hide();
|
|
$throbber.insertBefore($icon);
|
|
|
|
Navigation.loadChildNodes(true, $expandElem, function (data) {
|
|
if (typeof data !== 'undefined' && data.success === true) {
|
|
var $destination = $expandElem.closest('li');
|
|
$icon.removeClass('ic_b_plus').addClass('ic_b_minus');
|
|
$children = $destination.children('div.list_container');
|
|
$children.slideDown('fast');
|
|
if ($destination.find('ul > li').length === 1) {
|
|
$destination.find('ul > li')
|
|
.find('a.expander.container')
|
|
.trigger('click');
|
|
}
|
|
if (callback && typeof callback === 'function') {
|
|
callback.call();
|
|
}
|
|
Navigation.showFullName($destination);
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error, false);
|
|
}
|
|
$icon.show();
|
|
$throbber.remove();
|
|
$children.promise().done(Navigation.treeStateUpdate);
|
|
});
|
|
}
|
|
$expandElem.trigger('blur');
|
|
};
|
|
|
|
/**
|
|
* Auto-scrolls the newly chosen database
|
|
*
|
|
* @param object $element The element to set to view
|
|
* @param boolean $forceToTop Whether to force scroll to top
|
|
*
|
|
*/
|
|
Navigation.scrollToView = function ($element, $forceToTop) {
|
|
Navigation.filterStateRestore();
|
|
var $container = $('#pma_navigation_tree_content');
|
|
var elemTop = $element.offset().top - $container.offset().top;
|
|
var textHeight = 20;
|
|
var scrollPadding = 20; // extra padding from top of bottom when scrolling to view
|
|
if (elemTop < 0 || $forceToTop) {
|
|
$container.stop().animate({
|
|
scrollTop: elemTop + $container.scrollTop() - scrollPadding
|
|
});
|
|
} else if (elemTop + textHeight > $container.height()) {
|
|
$container.stop().animate({
|
|
scrollTop: elemTop + textHeight - $container.height() + $container.scrollTop() + scrollPadding
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Expand the navigation and highlight the current database or table/view
|
|
*
|
|
* @returns void
|
|
*/
|
|
Navigation.showCurrent = function () {
|
|
var db = CommonParams.get('db');
|
|
var table = CommonParams.get('table');
|
|
|
|
var autoexpand = $('#pma_navigation_tree').hasClass('autoexpand');
|
|
|
|
$('#pma_navigation_tree')
|
|
.find('li.selected')
|
|
.removeClass('selected');
|
|
var $dbItem;
|
|
if (db) {
|
|
$dbItem = findLoadedItem(
|
|
$('#pma_navigation_tree').find('> div'), db, 'database', !table
|
|
);
|
|
if ($('#navi_db_select').length &&
|
|
$('option:selected', $('#navi_db_select')).length
|
|
) {
|
|
if (! Navigation.selectCurrentDatabase()) {
|
|
return;
|
|
}
|
|
// If loaded database in navigation is not same as current one
|
|
if ($('#pma_navigation_tree_content').find('span.loaded_db:first').text()
|
|
!== $('#navi_db_select').val()
|
|
) {
|
|
Navigation.loadChildNodes(false, $('option:selected', $('#navi_db_select')), function () {
|
|
handleTableOrDb(table, $('#pma_navigation_tree_content'));
|
|
var $children = $('#pma_navigation_tree_content').children('div.list_container');
|
|
$children.promise().done(Navigation.treeStateUpdate);
|
|
});
|
|
} else {
|
|
handleTableOrDb(table, $('#pma_navigation_tree_content'));
|
|
}
|
|
} else if ($dbItem) {
|
|
fullExpand(table, $dbItem);
|
|
}
|
|
} else if ($('#navi_db_select').length && $('#navi_db_select').val()) {
|
|
$('#navi_db_select').val('').hide().trigger('change');
|
|
} else if (autoexpand && $('#pma_navigation_tree_content > ul > li.database').length === 1) {
|
|
// automatically expand the list if there is only single database
|
|
|
|
// find the name of the database
|
|
var dbItemName = '';
|
|
|
|
$('#pma_navigation_tree_content > ul > li.database').children('a').each(function () {
|
|
var name = $(this).text();
|
|
if (!dbItemName && name.trim()) { // if the name is not empty, it is the desired element
|
|
dbItemName = name;
|
|
}
|
|
});
|
|
|
|
$dbItem = findLoadedItem(
|
|
$('#pma_navigation_tree').find('> div'), dbItemName, 'database', !table
|
|
);
|
|
|
|
fullExpand(table, $dbItem);
|
|
}
|
|
Navigation.showFullName($('#pma_navigation_tree'));
|
|
|
|
function fullExpand (table, $dbItem) {
|
|
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')
|
|
) {
|
|
Navigation.expandTreeNode($expander, function () {
|
|
handleTableOrDb(table, $dbItem);
|
|
});
|
|
} else {
|
|
handleTableOrDb(table, $dbItem);
|
|
}
|
|
}
|
|
|
|
function handleTableOrDb (table, $dbItem) {
|
|
if (table) {
|
|
loadAndHighlightTableOrView($dbItem, table);
|
|
} else {
|
|
var $container = $dbItem.children('div.list_container');
|
|
var $tableContainer = $container.children('ul').children('li.tableContainer');
|
|
if ($tableContainer.length > 0) {
|
|
var $expander = $tableContainer.children('div:first').children('a.expander');
|
|
$tableContainer.addClass('selected');
|
|
Navigation.expandTreeNode($expander, function () {
|
|
Navigation.scrollToView($dbItem, true);
|
|
});
|
|
} else {
|
|
Navigation.scrollToView($dbItem, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
function findLoadedItem ($container, name, clazz, doSelect) {
|
|
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 = findLoadedItem(
|
|
$container, name, clazz, doSelect
|
|
);
|
|
if ($childRet) {
|
|
ret = $childRet;
|
|
return false;
|
|
}
|
|
} else { // this is a real navigation item
|
|
// name and class matches
|
|
if (((clazz && $li.is('.' + clazz)) || ! clazz) &&
|
|
$li.children('a').text() === name) {
|
|
if (doSelect) {
|
|
$li.addClass('selected');
|
|
}
|
|
// taverse up and expand and parent navigation groups
|
|
$li.parents('.navGroup').each(function () {
|
|
var $cont = $(this).children('div.list_container');
|
|
if (! $cont.is(':visible')) {
|
|
$(this)
|
|
.children('div:first')
|
|
.children('a.expander')
|
|
.trigger('click');
|
|
}
|
|
});
|
|
ret = $li;
|
|
return false;
|
|
}
|
|
}
|
|
});
|
|
return ret;
|
|
}
|
|
|
|
function loadAndHighlightTableOrView ($dbItem, itemName) {
|
|
var $container = $dbItem.children('div.list_container');
|
|
var $expander;
|
|
var $whichItem = isItemInContainer($container, itemName, 'li.table, li.view');
|
|
// If item already there in some container
|
|
if ($whichItem) {
|
|
// get the relevant container while may also be a subcontainer
|
|
var $relatedContainer = $whichItem.closest('li.subContainer').length
|
|
? $whichItem.closest('li.subContainer')
|
|
: $dbItem;
|
|
$whichItem = findLoadedItem(
|
|
$relatedContainer.children('div.list_container'),
|
|
itemName, null, true
|
|
);
|
|
// Show directly
|
|
showTableOrView($whichItem, $relatedContainer.children('div:first').children('a.expander'));
|
|
// else if item not there, try loading once
|
|
} else {
|
|
var $subContainers = $dbItem.find('.subContainer');
|
|
// If there are subContainers i.e. tableContainer or viewContainer
|
|
if ($subContainers.length > 0) {
|
|
var $containers = [];
|
|
$subContainers.each(function (index) {
|
|
$containers[index] = $(this);
|
|
$expander = $containers[index]
|
|
.children('div:first')
|
|
.children('a.expander');
|
|
if (! $expander.hasClass('loaded')) {
|
|
loadAndShowTableOrView($expander, $containers[index], itemName);
|
|
}
|
|
});
|
|
// else if no subContainers
|
|
} else {
|
|
$expander = $dbItem
|
|
.children('div:first')
|
|
.children('a.expander');
|
|
if (! $expander.hasClass('loaded')) {
|
|
loadAndShowTableOrView($expander, $dbItem, itemName);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function loadAndShowTableOrView ($expander, $relatedContainer, itemName) {
|
|
Navigation.loadChildNodes(true, $expander, function () {
|
|
var $whichItem = findLoadedItem(
|
|
$relatedContainer.children('div.list_container'),
|
|
itemName, null, true
|
|
);
|
|
if ($whichItem) {
|
|
showTableOrView($whichItem, $expander);
|
|
}
|
|
});
|
|
}
|
|
|
|
function showTableOrView ($whichItem, $expander) {
|
|
Navigation.expandTreeNode($expander, function () {
|
|
if ($whichItem) {
|
|
Navigation.scrollToView($whichItem, false);
|
|
}
|
|
});
|
|
}
|
|
|
|
function isItemInContainer ($container, name, clazz) {
|
|
var $whichItem = null;
|
|
var $items = $container.find(clazz);
|
|
$items.each(function () {
|
|
if ($(this).children('a').text() === name) {
|
|
$whichItem = $(this);
|
|
return false;
|
|
}
|
|
});
|
|
return $whichItem;
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Disable navigation panel settings
|
|
*
|
|
* @return void
|
|
*/
|
|
Navigation.disableSettings = function () {
|
|
$('#pma_navigation_settings_icon').addClass('hide');
|
|
$('#pma_navigation_settings').remove();
|
|
};
|
|
|
|
/**
|
|
* Ensure that navigation panel settings is properly setup.
|
|
* If not, set it up
|
|
*
|
|
* @return void
|
|
*/
|
|
Navigation.ensureSettings = function (selflink) {
|
|
$('#pma_navigation_settings_icon').removeClass('hide');
|
|
|
|
if (!$('#pma_navigation_settings').length) {
|
|
var params = {
|
|
getNaviSettings: true,
|
|
server: CommonParams.get('server'),
|
|
};
|
|
var url = $('#pma_navigation').find('a.navigation_url').attr('href');
|
|
$.post(url, params, function (data) {
|
|
if (typeof data !== 'undefined' && data.success) {
|
|
$('#pma_navi_settings_container').html(data.message);
|
|
setupRestoreField();
|
|
setupValidation();
|
|
setupConfigTabs();
|
|
$('#pma_navigation_settings').find('form').attr('action', selflink);
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error);
|
|
}
|
|
});
|
|
} else {
|
|
$('#pma_navigation_settings').find('form').attr('action', selflink);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Reloads the whole navigation tree while preserving its state
|
|
*
|
|
* @param function the callback function
|
|
* @param Object stored navigation paths
|
|
*
|
|
* @return void
|
|
*/
|
|
Navigation.reload = function (callback, paths) {
|
|
var params = {
|
|
'reload': true,
|
|
'no_debug': true,
|
|
'server': CommonParams.get('server'),
|
|
};
|
|
var pathsLocal = paths || Navigation.traverseForPaths();
|
|
$.extend(params, pathsLocal);
|
|
if ($('#navi_db_select').length) {
|
|
params.db = CommonParams.get('db');
|
|
requestNaviReload(params);
|
|
return;
|
|
}
|
|
requestNaviReload(params);
|
|
|
|
function requestNaviReload (params) {
|
|
var url = $('#pma_navigation').find('a.navigation_url').attr('href');
|
|
$.post(url, params, function (data) {
|
|
if (typeof data !== 'undefined' && data.success) {
|
|
$('#pma_navigation_tree').html(data.message).children('div').show();
|
|
if ($('#pma_navigation_tree').hasClass('synced')) {
|
|
Navigation.selectCurrentDatabase();
|
|
Navigation.showCurrent();
|
|
}
|
|
// Fire the callback, if any
|
|
if (typeof callback === 'function') {
|
|
callback.call();
|
|
}
|
|
Navigation.treeStateUpdate();
|
|
} else {
|
|
Functions.ajaxShowMessage(data.error);
|
|
}
|
|
});
|
|
}
|
|
};
|
|
|
|
Navigation.selectCurrentDatabase = function () {
|
|
var $naviDbSelect = $('#navi_db_select');
|
|
|
|
if (!$naviDbSelect.length) {
|
|
return false;
|
|
}
|
|
|
|
if (CommonParams.get('db')) { // db selected
|
|
$naviDbSelect.show();
|
|
}
|
|
|
|
$naviDbSelect.val(CommonParams.get('db'));
|
|
return $naviDbSelect.val() === CommonParams.get('db');
|
|
};
|
|
|
|
/**
|
|
* 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
|
|
*/
|
|
Navigation.treePagination = function ($this) {
|
|
var $msgbox = Functions.ajaxShowMessage();
|
|
var isDbSelector = $this.closest('div.pageselector').is('.dbselector');
|
|
var url;
|
|
var params;
|
|
if ($this[0].tagName === 'A') {
|
|
url = $this.attr('href');
|
|
params = 'ajax_request=true';
|
|
} else { // tagName === 'SELECT'
|
|
url = 'navigation.php';
|
|
params = $this.closest('form').serialize() + CommonParams.get('arg_separator') + 'ajax_request=true';
|
|
}
|
|
var searchClause = Navigation.FastFilter.getSearchClause();
|
|
if (searchClause) {
|
|
params += CommonParams.get('arg_separator') + 'searchClause=' + encodeURIComponent(searchClause);
|
|
}
|
|
if (isDbSelector) {
|
|
params += CommonParams.get('arg_separator') + 'full=true';
|
|
} else {
|
|
var searchClause2 = Navigation.FastFilter.getSearchClause2($this);
|
|
if (searchClause2) {
|
|
params += CommonParams.get('arg_separator') + 'searchClause2=' + encodeURIComponent(searchClause2);
|
|
}
|
|
}
|
|
$.post(url, params, function (data) {
|
|
if (typeof data !== 'undefined' && data.success) {
|
|
Functions.ajaxRemoveMessage($msgbox);
|
|
var val;
|
|
if (isDbSelector) {
|
|
val = Navigation.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();
|
|
val = Navigation.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 {
|
|
Functions.ajaxShowMessage(data.error);
|
|
Functions.handleRedirectAndReload(data);
|
|
}
|
|
Navigation.treeStateUpdate();
|
|
});
|
|
};
|
|
|
|
/**
|
|
* @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'
|
|
*/
|
|
Navigation.ResizeHandler = function () {
|
|
/**
|
|
* @var int panelWidth Used by the collapser to know where to go
|
|
* back to when uncollapsing the panel
|
|
*/
|
|
this.panelWidth = 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} position Navigation width in pixels
|
|
*
|
|
* @return void
|
|
*/
|
|
this.setWidth = function (position) {
|
|
var pos = position;
|
|
if (typeof pos !== 'number') {
|
|
pos = 240;
|
|
}
|
|
var $resizer = $('#pma_navigation_resizer');
|
|
var resizerWidth = $resizer.width();
|
|
var $collapser = $('#pma_navigation_collapser');
|
|
var windowWidth = $(window).width();
|
|
$('#pma_navigation').width(pos);
|
|
$('body').css('margin-' + this.left, pos + 'px');
|
|
// Issue #15127 : Adding fixed positioning to menubar
|
|
// Issue #15570 : Panels on homescreen go underneath of floating menubar
|
|
$('#floating_menubar')
|
|
.css('margin-' + this.left, $('#pma_navigation').width() + $('#pma_navigation_resizer').width())
|
|
.css(this.left, 0)
|
|
.css({
|
|
'position': 'fixed',
|
|
'top': 0,
|
|
'width': '100%',
|
|
'z-index': 99
|
|
})
|
|
.append($('#serverinfo'))
|
|
.append($('#topmenucontainer'));
|
|
// Allow the DOM to render, then adjust the padding on the body
|
|
setTimeout(function () {
|
|
$('body').css(
|
|
'padding-top',
|
|
$('#floating_menubar').outerHeight(true)
|
|
);
|
|
}, 2);
|
|
$('#pma_console')
|
|
.css('margin-' + this.left, (pos + resizerWidth) + 'px');
|
|
$resizer.css(this.left, pos + 'px');
|
|
if (pos === 0) {
|
|
$collapser
|
|
.css(this.left, pos + resizerWidth)
|
|
.html(this.getSymbol(pos))
|
|
.prop('title', Messages.strShowPanel);
|
|
} else if (windowWidth > 768) {
|
|
$collapser
|
|
.css(this.left, pos)
|
|
.html(this.getSymbol(pos))
|
|
.prop('title', Messages.strHidePanel);
|
|
$('#pma_navigation_resizer').css({ 'width': '3px' });
|
|
} else {
|
|
$collapser
|
|
.css(this.left, windowWidth - 22)
|
|
.html(this.getSymbol(100))
|
|
.prop('title', Messages.strHidePanel);
|
|
$('#pma_navigation').width(windowWidth);
|
|
$('body').css('margin-' + this.left, '0px');
|
|
$('#pma_navigation_resizer').css({ 'width': '0px' });
|
|
}
|
|
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();
|
|
var windowScroll = $(window).scrollLeft();
|
|
pos = pos - windowScroll;
|
|
if (this.left !== 'left') {
|
|
pos = windowWidth - event.pageX;
|
|
}
|
|
if (pos < 0) {
|
|
pos = 0;
|
|
} else if (pos + 100 >= windowWidth) {
|
|
pos = windowWidth - 100;
|
|
} else {
|
|
this.panelWidth = 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 Navigation.ResizeHandler)
|
|
*
|
|
* @return void
|
|
*/
|
|
this.mousedown = function (event) {
|
|
event.preventDefault();
|
|
$(document)
|
|
.on('mousemove', { 'resize_handler': event.data.resize_handler },
|
|
$.throttle(event.data.resize_handler.mousemove, 4))
|
|
.on('mouseup', { 'resize_handler': event.data.resize_handler },
|
|
event.data.resize_handler.mouseup);
|
|
$('body').css('cursor', 'col-resize');
|
|
};
|
|
/**
|
|
* Event handler for terminating a resize of the panel
|
|
*
|
|
* @param object e Event data (contains a reference to Navigation.ResizeHandler)
|
|
*
|
|
* @return void
|
|
*/
|
|
this.mouseup = function (event) {
|
|
$('body').css('cursor', '');
|
|
Functions.configSet('NavigationWidth', event.data.resize_handler.getPos(event));
|
|
$('#topmenu').menuResizer('resize');
|
|
$(document)
|
|
.off('mousemove')
|
|
.off('mouseup');
|
|
};
|
|
/**
|
|
* Event handler for updating the panel during a resize operation
|
|
*
|
|
* @param object e Event data (contains a reference to Navigation.ResizeHandler)
|
|
*
|
|
* @return void
|
|
*/
|
|
this.mousemove = function (event) {
|
|
event.preventDefault();
|
|
if (event.data && event.data.resize_handler) {
|
|
var pos = event.data.resize_handler.getPos(event);
|
|
event.data.resize_handler.setWidth(pos);
|
|
}
|
|
if ($('.sticky_columns').length !== 0) {
|
|
Sql.handleAllStickyColumns();
|
|
}
|
|
};
|
|
/**
|
|
* Event handler for collapsing the panel
|
|
*
|
|
* @param object e Event data (contains a reference to Navigation.ResizeHandler)
|
|
*
|
|
* @return void
|
|
*/
|
|
this.collapse = function (event) {
|
|
event.preventDefault();
|
|
var panelWidth = event.data.resize_handler.panelWidth;
|
|
var width = $('#pma_navigation').width();
|
|
if (width === 0 && panelWidth === 0) {
|
|
panelWidth = 240;
|
|
}
|
|
Functions.configSet('NavigationWidth', panelWidth);
|
|
event.data.resize_handler.setWidth(panelWidth);
|
|
event.data.resize_handler.panelWidth = width;
|
|
};
|
|
/**
|
|
* Event handler for resizing the navigation tree height on window resize
|
|
*
|
|
* @return void
|
|
*/
|
|
this.treeResize = function () {
|
|
var $nav = $('#pma_navigation');
|
|
var $navTree = $('#pma_navigation_tree');
|
|
var $navHeader = $('#pma_navigation_header');
|
|
var $navTreeContent = $('#pma_navigation_tree_content');
|
|
var height = ($nav.height() - $navHeader.height());
|
|
|
|
height = height > 50 ? height : 800; // keep min. height
|
|
$navTree.height(height);
|
|
if ($navTreeContent.length > 0) {
|
|
$navTreeContent.height(height - $navTreeContent.position().top);
|
|
} else {
|
|
// TODO: in fast filter search response there is no #pma_navigation_tree_content, needs to be added in php
|
|
$navTree.css({
|
|
'overflow-y': 'auto'
|
|
});
|
|
}
|
|
// Set content bottom space beacuse of console
|
|
$('body').css('margin-bottom', $('#pma_console').height() + 'px');
|
|
};
|
|
// Hide the pma_navigation initially when loaded on mobile
|
|
if ($(window).width() < 768) {
|
|
this.setWidth(0);
|
|
} else {
|
|
this.setWidth(Functions.configGet('NavigationWidth', false));
|
|
$('#topmenu').menuResizer('resize');
|
|
}
|
|
// Register the events for the resizer and the collapser
|
|
$(document).on('mousedown', '#pma_navigation_resizer', { 'resize_handler': this }, this.mousedown);
|
|
$(document).on('click', '#pma_navigation_collapser', { 'resize_handler': this }, this.collapse);
|
|
|
|
// Add the correct arrow symbol to the collapser
|
|
$('#pma_navigation_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);
|
|
this.treeResize();
|
|
};
|
|
|
|
/**
|
|
* @var object FastFilter Handles the functionality that allows filtering
|
|
* of the items in a branch of the navigation tree
|
|
*/
|
|
Navigation.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 Navigation.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 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 = $([]);
|
|
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');
|
|
}
|
|
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 () {
|
|
var $obj = $(this).closest('div.list_container');
|
|
if (! $obj.data('fastFilter')) {
|
|
$obj.data(
|
|
'fastFilter',
|
|
new Navigation.FastFilter.Filter($obj, $(this).val())
|
|
);
|
|
}
|
|
if ($(this).val() === this.defaultValue) {
|
|
$(this).val('');
|
|
} else {
|
|
$(this).trigger('select');
|
|
}
|
|
},
|
|
blur: function () {
|
|
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();
|
|
}
|
|
|
|
/**
|
|
* FIXME at the server level a value match is done while on
|
|
* the client side it is a regex match. These two should be aligned
|
|
*/
|
|
|
|
// regex used for filtering.
|
|
var regex;
|
|
try {
|
|
regex = new RegExp(str, 'i');
|
|
} catch (err) {
|
|
return;
|
|
}
|
|
|
|
// this is the div that houses the items to be filtered by this filter.
|
|
var outerContainer;
|
|
if ($(this).closest('li.fast_filter').is('.db_fast_filter')) {
|
|
outerContainer = $('#pma_navigation_tree_content');
|
|
} else {
|
|
outerContainer = $obj;
|
|
}
|
|
|
|
// filters items that are directly under the div as well as grouped in
|
|
// groups. Does not filter child items (i.e. a database search does
|
|
// not filter tables)
|
|
var itemFilter = function ($curr) {
|
|
$curr.children('ul').children('li.navGroup').each(function () {
|
|
$(this).children('div.list_container').each(function () {
|
|
itemFilter($(this)); // recursive
|
|
});
|
|
});
|
|
$curr.children('ul').children('li').children('a').not('.container').each(function () {
|
|
if (regex.test($(this).text())) {
|
|
$(this).parent().show().removeClass('hidden');
|
|
} else {
|
|
$(this).parent().hide().addClass('hidden');
|
|
}
|
|
});
|
|
};
|
|
itemFilter(outerContainer);
|
|
|
|
// hides containers that does not have any visible children
|
|
var containerFilter = function ($curr) {
|
|
$curr.children('ul').children('li.navGroup').each(function () {
|
|
var $group = $(this);
|
|
$group.children('div.list_container').each(function () {
|
|
containerFilter($(this)); // recursive
|
|
});
|
|
$group.show().removeClass('hidden');
|
|
if ($group.children('div.list_container').children('ul')
|
|
.children('li').not('.hidden').length === 0) {
|
|
$group.hide().addClass('hidden');
|
|
}
|
|
});
|
|
};
|
|
containerFilter(outerContainer);
|
|
|
|
if ($(this).val() !== this.defaultValue && $(this).val() !== '') {
|
|
if (! $obj.data('fastFilter')) {
|
|
$obj.data(
|
|
'fastFilter',
|
|
new Navigation.FastFilter.Filter($obj, $(this).val())
|
|
);
|
|
} else {
|
|
if (event.keyCode === 13) {
|
|
$obj.data('fastFilter').update($(this).val());
|
|
}
|
|
}
|
|
} else if ($obj.data('fastFilter')) {
|
|
$obj.data('fastFilter').restore(true);
|
|
}
|
|
// update filter state
|
|
var filterName;
|
|
if ($(this).attr('name') === 'searchClause2') {
|
|
filterName = $(this).siblings('input[name=aPath]').val();
|
|
} else {
|
|
filterName = 'dbFilter';
|
|
}
|
|
Navigation.filterStateUpdate(filterName, $(this).val());
|
|
},
|
|
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
|
|
*/
|
|
Navigation.FastFilter.Filter.prototype.update = function (searchClause) {
|
|
if (this.searchClause !== searchClause) {
|
|
this.searchClause = searchClause;
|
|
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
|
|
*/
|
|
Navigation.FastFilter.Filter.prototype.request = function () {
|
|
var self = this;
|
|
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', display: 'block' })
|
|
)
|
|
);
|
|
}
|
|
if (self.xhr) {
|
|
self.xhr.abort();
|
|
}
|
|
var url = $('#pma_navigation').find('a.navigation_url').attr('href');
|
|
var params = self.$this.find('> ul > li > form.fast_filter').first().serialize();
|
|
|
|
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 += CommonParams.get('arg_separator') + 'searchClause=' + encodeURIComponent($input.val());
|
|
}
|
|
}
|
|
self.xhr = $.ajax({
|
|
url: url,
|
|
type: 'post',
|
|
dataType: 'json',
|
|
data: params,
|
|
complete: function (jqXHR, status) {
|
|
if (status !== 'abort') {
|
|
var data = JSON.parse(jqXHR.responseText);
|
|
self.$this.find('li.fast_filter').find('div.throbber').remove();
|
|
if (data && data.results) {
|
|
self.swap.apply(self, [data.message]);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
};
|
|
/**
|
|
* Replaces the contents of the navigation branch with the search results
|
|
*
|
|
* @param string list The search results
|
|
*
|
|
* @return void
|
|
*/
|
|
Navigation.FastFilter.Filter.prototype.swap = function (list) {
|
|
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
|
|
*/
|
|
Navigation.FastFilter.Filter.prototype.restore = function (focus) {
|
|
if (this.$this.children('ul').first().hasClass('search_results')) {
|
|
this.$this.html(this.$clone.html()).children().show();
|
|
this.$this.data('fastFilter', this);
|
|
if (focus) {
|
|
this.$this.find('li.fast_filter input.searchClause').trigger('focus');
|
|
}
|
|
}
|
|
this.searchClause = '';
|
|
this.$this.find('div.pageselector').show();
|
|
this.$this.find('div.throbber').remove();
|
|
};
|
|
|
|
/**
|
|
* Show full name when cursor hover and name not shown completely
|
|
*
|
|
* @param object $containerELem Container element
|
|
*
|
|
* @return void
|
|
*/
|
|
Navigation.showFullName = function ($containerELem) {
|
|
$containerELem.find('.hover_show_full').on('mouseenter', function () {
|
|
/** mouseenter */
|
|
var $this = $(this);
|
|
var thisOffset = $this.offset();
|
|
if ($this.text() === '') {
|
|
return;
|
|
}
|
|
var $parent = $this.parent();
|
|
if (($parent.offset().left + $parent.outerWidth())
|
|
< (thisOffset.left + $this.outerWidth())) {
|
|
var $fullNameLayer = $('#full_name_layer');
|
|
if ($fullNameLayer.length === 0) {
|
|
$('body').append('<div id="full_name_layer" class="hide"></div>');
|
|
$('#full_name_layer').mouseleave(function () {
|
|
/** mouseleave */
|
|
$(this).addClass('hide')
|
|
.removeClass('hovering');
|
|
}).on('mouseenter', function () {
|
|
/** mouseenter */
|
|
$(this).addClass('hovering');
|
|
});
|
|
$fullNameLayer = $('#full_name_layer');
|
|
}
|
|
$fullNameLayer.removeClass('hide');
|
|
$fullNameLayer.css({ left: thisOffset.left, top: thisOffset.top });
|
|
$fullNameLayer.html($this.clone());
|
|
setTimeout(function () {
|
|
if (! $fullNameLayer.hasClass('hovering')) {
|
|
$fullNameLayer.trigger('mouseleave');
|
|
}
|
|
}, 200);
|
|
}
|
|
});
|
|
};
|