/** Get first element by selector * @param string * @param [HTMLElement] defaults to document * @return HTMLElement */ function qs(selector, context) { return (context || document).querySelector(selector); } /** Get last element by selector * @param string * @param [HTMLElement] defaults to document * @return HTMLElement */ function qsl(selector, context) { var els = qsa(selector, context); return els[els.length - 1]; } /** Get all elements by selector * @param string * @param [HTMLElement] defaults to document * @return NodeList */ function qsa(selector, context) { return (context || document).querySelectorAll(selector); } /** Return a function calling fn with the next arguments * @param function * @param ... * @return function with preserved this */ function partial(fn) { var args = Array.apply(null, arguments).slice(1); return function () { return fn.apply(this, args); }; } /** Return a function calling fn with the first parameter and then the next arguments * @param function * @param ... * @return function with preserved this */ function partialArg(fn) { var args = Array.apply(null, arguments); return function (arg) { args[0] = arg; return fn.apply(this, args); }; } /** Assign values from source to target * @param Object * @param Object */ function mixin(target, source) { for (var key in source) { target[key] = source[key]; } } /** Add or remove CSS class * @param HTMLElement * @param string * @param [bool] */ function alterClass(el, className, enable) { if (el) { el.className = el.className.replace(RegExp('(^|\\s)' + className + '(\\s|$)'), '$2') + (enable ? ' ' + className : ''); } } /** Toggle visibility * @param string * @return boolean false */ function toggle(id) { var el = qs('#' + id); el.className = (el.className == 'hidden' ? '' : 'hidden'); return false; } /** Set permanent cookie * @param string * @param number * @param string optional */ function cookie(assign, days) { var date = new Date(); date.setDate(date.getDate() + days); document.cookie = assign + '; expires=' + date; } /** Verify current Adminer version * @param string * @param string own URL base * @param string */ function verifyVersion(current, url, token) { cookie('adminer_version=0', 1); ajax('https://api.github.com/repos/adminerevo/adminerevo/releases/latest', function (request) { var data = window.JSON ? JSON.parse(request.responseText) : eval('(' + request.responseText + ')'); version = data.tag_name.replace(/[^\d.]/g, ''); if (version) { cookie('adminer_version=' + version, 1); var data = 'version=' + version; ajax(url + 'script=version', function () { }, data + '&token=' + token); if (version != current) { qs('#version').innerText = version; } } }); } /** Get value of select * @param HTMLElement * @return string */ function selectValue(select) { if (!select.selectedIndex) { return select.value; } var selected = select.options[select.selectedIndex]; return ((selected.attributes.value || {}).specified ? selected.value : selected.text); } /** Verify if element has a specified tag name * @param HTMLElement * @param string regular expression * @return bool */ function isTag(el, tag) { var re = new RegExp('^(' + tag + ')$', 'i'); return el && re.test(el.tagName); } /** Get parent node with specified tag name * @param HTMLElement * @param string regular expression * @return HTMLElement */ function parentTag(el, tag) { while (el && !isTag(el, tag)) { el = el.parentNode; } return el; } /** Set checked class * @param HTMLInputElement */ function trCheck(el) { var tr = parentTag(el, 'tr'); alterClass(tr, 'checked', el.checked); if (el.form && el.form['all'] && el.form['all'].onclick) { // Opera treats form.all as document.all el.form['all'].onclick(); } } /** Fill number of selected items * @param string * @param string * @uses thousandsSeparator */ function selectCount(id, count) { setHtml(id, (count === '' ? '' : '(' + (count + '').replace(/\B(?=(\d{3})+$)/g, thousandsSeparator) + ')')); var el = qs('#' + id); if (el) { var inputs = qsa('input', el.parentNode.parentNode); for (var i = 0; i < inputs.length; i++) { var input = inputs[i]; if (input.type == 'submit') { input.disabled = (count == '0'); } } } } /** Check all elements matching given name * @param RegExp * @this HTMLInputElement */ function formCheck(name) { var elems = this.form.elements; for (var i=0; i < elems.length; i++) { if (name.test(elems[i].name)) { elems[i].checked = this.checked; trCheck(elems[i]); } } } /** Check all rows in */ function tableCheck() { var inputs = qsa('table.checkable td:first-child input'); for (var i=0; i < inputs.length; i++) { trCheck(inputs[i]); } } /** Uncheck single element * @param string */ function formUncheck(id) { var el = qs('#' + id); el.checked = false; trCheck(el); } /** Get number of checked elements matching given name * @param HTMLInputElement * @param RegExp * @return number */ function formChecked(el, name) { var checked = 0; var elems = el.form.elements; for (var i=0; i < elems.length; i++) { if (name.test(elems[i].name) && elems[i].checked) { checked++; } } return checked; } /** Select clicked row * @param MouseEvent * @param [boolean] force click */ function tableClick(event, click) { var td = parentTag(getTarget(event), 'td'); var text; if (td && (text = td.getAttribute('data-text'))) { if (selectClick.call(td, event, +text, td.getAttribute('data-warning'))) { return; } } click = (click || !window.getSelection || getSelection().isCollapsed); var el = getTarget(event); while (!isTag(el, 'tr')) { if (isTag(el, 'table|a|input|textarea')) { if (el.type != 'checkbox') { return; } checkboxClick.call(el, event); click = false; } el = el.parentNode; if (!el) { // Ctrl+click on text fields hides the element return; } } el = el.firstChild.firstChild; if (click) { el.checked = !el.checked; el.onclick && el.onclick(); } if (el.name == 'check[]') { el.form['all'].checked = false; formUncheck('all-page'); } if (/^(tables|views)\[\]$/.test(el.name)) { formUncheck('check-all'); } trCheck(el); } var lastChecked; /** Shift-click on checkbox for multiple selection. * @param MouseEvent * @this HTMLInputElement */ function checkboxClick(event) { if (!this.name) { return; } if (event.shiftKey && (!lastChecked || lastChecked.name == this.name)) { var checked = (lastChecked ? lastChecked.checked : true); var inputs = qsa('input', parentTag(this, 'table')); var checking = !lastChecked; for (var i=0; i < inputs.length; i++) { var input = inputs[i]; if (input.name === this.name) { if (checking) { input.checked = checked; trCheck(input); } if (input === this || input === lastChecked) { if (checking) { break; } checking = true; } } } } else { lastChecked = this; } } /** Set HTML code of an element * @param string * @param string undefined to set parentNode to empty string */ function setHtml(id, html) { var el = qs('[id="' + id.replace(/[\\"]/g, '\\$&') + '"]'); // database name is used as ID if (el) { if (html == null) { el.parentNode.innerHTML = ''; } else { el.innerHTML = html; } } } /** Find node position * @param Node * @return number */ function nodePosition(el) { var pos = 0; while (el = el.previousSibling) { pos++; } return pos; } /** Go to the specified page * @param string * @param string */ function pageClick(href, page) { if (!isNaN(page) && page) { location.href = href + (page != 1 ? '&page=' + (page - 1) : ''); } } /** Display items in menu * @param MouseEvent * @this HTMLElement */ function menuOver(event) { var a = getTarget(event); if (isTag(a, 'a|span') && a.offsetLeft + a.offsetWidth > a.parentNode.offsetWidth - 15) { // 15 - ellipsis this.style.overflow = 'visible'; } } /** Hide items in menu * @this HTMLElement */ function menuOut() { this.style.overflow = 'auto'; } /** Add row in select fieldset * @this HTMLSelectElement */ function selectAddRow() { var field = this; var row = cloneNode(field.parentNode); field.onchange = selectFieldChange; field.onchange(); var selects = qsa('select', row); for (var i=0; i < selects.length; i++) { selects[i].name = selects[i].name.replace(/[a-z]\[\d+/, '$&1'); selects[i].selectedIndex = 0; } var inputs = qsa('input', row); for (var i=0; i < inputs.length; i++) { if (inputs[i].type === 'image') { // button continue; } inputs[i].name = inputs[i].name.replace(/[a-z]\[\d+/, '$&1'); inputs[i].className = ''; if (inputs[i].type == 'checkbox') { inputs[i].checked = false; } else { inputs[i].value = ''; } } var buttons = qsa('.icon', row); for (var i=0; i < buttons.length; i++) { buttons[i].onclick = selectRemoveRow; } field.parentNode.parentNode.appendChild(row); } /** Remove a row in select fieldset * @this HTMLInputElement */ function selectRemoveRow() { var button = this; var row = button.parentNode; var nextRow = row; while (nextRow = nextRow.nextSibling) { if (nextRow.tagName === row.tagName) { row.parentNode.removeChild(row); break; } } return false; } /** Prevent onsearch handler on Enter * @param KeyboardEvent * @this HTMLInputElement */ function selectSearchKeydown(event) { if (event.keyCode == 13 || event.keyCode == 10) { this.onsearch = function () { }; } } /** Clear column name after resetting search * @this HTMLInputElement */ function selectSearchSearch() { if (!this.value) { this.parentNode.firstChild.selectedIndex = 0; } } /** Toggles column context menu * @param [string] extra class name * @this HTMLElement */ function columnMouse(className) { var spans = qsa('span', this); for (var i=0; i < spans.length; i++) { if (/column/.test(spans[i].className)) { spans[i].className = 'column' + (className || ''); } } } /** Fill column in search field * @param string * @return boolean false */ function selectSearch(name) { var el = qs('#fieldset-search'); el.className = ''; var divs = qsa('div', el); for (var i=0; i < divs.length; i++) { var div = divs[i]; var el = qs('[name$="[col]"]', div); if (el && selectValue(el) == name) { break; } } if (i == divs.length) { div.firstChild.value = name; div.firstChild.onchange(); } qs('[name$="[val]"]', div).focus(); return false; } /** Check if Ctrl key (Command key on Mac) was pressed * @param KeyboardEvent|MouseEvent * @return boolean */ function isCtrl(event) { return (event.ctrlKey || event.metaKey) && !event.altKey; // shiftKey allowed } /** Return event target * @param Event * @return HTMLElement */ function getTarget(event) { return event.target || event.srcElement; } /** Send form by Ctrl+Enter on