184 lines
6.4 KiB
JavaScript
184 lines
6.4 KiB
JavaScript
|
/**
|
||
|
* Bootstrap.js
|
||
|
*
|
||
|
* Created by Alexander Yuzhin on 5/27/14
|
||
|
* Copyright (c) 2014 Ascensio System SIA. All rights reserved.
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
function onDropDownKeyDown(e) {
|
||
|
var $this = $(this),
|
||
|
$parent = $this.parent(),
|
||
|
beforeEvent = jQuery.Event('keydown.before.bs.dropdown'),
|
||
|
afterEvent = jQuery.Event('keydown.after.bs.dropdown');
|
||
|
|
||
|
$parent.trigger(beforeEvent);
|
||
|
|
||
|
if ($parent.hasClass('no-stop-propagate') && arguments.length>1 && arguments[1] instanceof jQuery.Event) {
|
||
|
e = arguments[1];
|
||
|
if ( /^(38|40|27|13|9)$/.test(e.keyCode)) {
|
||
|
patchDropDownKeyDownAdditional.call(this, e);
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
}
|
||
|
} else if ( !$parent.hasClass('no-stop-propagate') || /^(38|40|27|13|9)$/.test(e.keyCode)) {
|
||
|
patchDropDownKeyDown.call(this, e);
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
}
|
||
|
|
||
|
$parent.trigger(afterEvent);
|
||
|
}
|
||
|
|
||
|
function patchDropDownKeyDown(e) {
|
||
|
if (!/(38|40|27|37|39)/.test(e.keyCode)) return;
|
||
|
|
||
|
var $this = $(this);
|
||
|
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
|
||
|
if ($this.is('.disabled, :disabled')) return;
|
||
|
|
||
|
var $parent = getParent($this);
|
||
|
var isActive = $parent.hasClass('open') || $parent.hasClass('over');
|
||
|
|
||
|
if (!isActive || (isActive && e.keyCode == 27)) {
|
||
|
if (e.which == 27) {
|
||
|
$items = $('[role=menu] li.dropdown-submenu.over:visible', $parent);
|
||
|
if ($items.size()) {
|
||
|
$items.eq($items.size()-1).removeClass('over');
|
||
|
return false;
|
||
|
} else if ($parent.hasClass('dropdown-submenu') && $parent.hasClass('over')) {
|
||
|
$parent.removeClass('over');
|
||
|
$parent.find('> a').focus();
|
||
|
} else {
|
||
|
$parent.find('[data-toggle=dropdown]').focus();
|
||
|
}
|
||
|
}
|
||
|
return (isActive) ? $this.click() : undefined;
|
||
|
}
|
||
|
|
||
|
// var $items = $('[role=menu] li:not(.divider):visible a', $parent) - original search function
|
||
|
var $items = $('> [role=menu] > li:not(.divider):not(.disabled):visible', $parent).find('> a');
|
||
|
|
||
|
if (!$items.length) return;
|
||
|
|
||
|
var index = $items.index($items.filter(':focus'));
|
||
|
if (e.keyCode == 39) { // right
|
||
|
if (index<0) return;
|
||
|
|
||
|
var li = $items.eq(index).parent();
|
||
|
if (li.hasClass('dropdown-submenu') && !li.hasClass('over')) {// open submenu and select first <li> item
|
||
|
li.mouseenter();
|
||
|
li.addClass('focused-submenu');
|
||
|
_.delay(function() {
|
||
|
var mnu = $('> [role=menu]', li),
|
||
|
$subitems = mnu.find('> li:not(.divider):not(.disabled):visible > a'),
|
||
|
focusIdx = 0;
|
||
|
if (mnu.find('> .menu-scroll').length>0) {
|
||
|
var offset = mnu.scrollTop();
|
||
|
for (var i=0; i<$subitems.length; i++) {
|
||
|
if ($subitems[i].offsetTop > offset) {
|
||
|
focusIdx = i; break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($subitems.length>0)
|
||
|
$subitems.eq(focusIdx).focus();
|
||
|
}, 250);
|
||
|
}
|
||
|
} else if (e.keyCode == 37) { // left
|
||
|
if ($parent.hasClass('dropdown-submenu') && $parent.hasClass('over')) { // close submenu
|
||
|
$parent.removeClass('over');
|
||
|
$parent.find('> a').focus();
|
||
|
}
|
||
|
} else {
|
||
|
if (e.keyCode == 38) { index > 0 ? index-- : ($this.hasClass('no-cyclic') ? (index = 0) : (index = $items.length - 1));} else // up
|
||
|
if (e.keyCode == 40) { index < $items.length - 1 ? index++ : ($this.hasClass('no-cyclic') ? (index = $items.length - 1) : (index = 0));} // down
|
||
|
if (!~index) index=0;
|
||
|
if ($parent.hasClass('dropdown-submenu') && $parent.hasClass('over'))
|
||
|
$parent.addClass('focused-submenu'); // for Safari. When focus go from parent menuItem to it's submenu don't hide this submenu
|
||
|
|
||
|
$items.eq(index).focus();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function patchDropDownKeyDownAdditional(e) { // only for formula menu when typing in cell editor
|
||
|
if (!/(38|40|27|37|39)/.test(e.keyCode)) return;
|
||
|
|
||
|
var $this = $(this);
|
||
|
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
|
||
|
if ($this.is('.disabled, :disabled')) return;
|
||
|
|
||
|
var $parent = getParent($this);
|
||
|
var isActive = $parent.hasClass('open') || $parent.hasClass('over');
|
||
|
|
||
|
if (!isActive || (isActive && e.keyCode == 27)) {
|
||
|
if (e.which == 27)
|
||
|
$parent.find('[data-toggle=dropdown]').focus();
|
||
|
return (isActive) ? $this.click() : undefined;
|
||
|
}
|
||
|
|
||
|
var $items = $('> [role=menu] > li:not(.divider):not(.disabled):visible', $parent).find('> a');
|
||
|
|
||
|
if (!$items.length) return;
|
||
|
|
||
|
var index = $items.index($items.filter('.focus'));
|
||
|
if (e.keyCode == 38) { index > 0 ? index-- : ($this.hasClass('no-cyclic') ? (index = 0) : (index = $items.length - 1));} else // up
|
||
|
if (e.keyCode == 40) { index < $items.length - 1 ? index++ : ($this.hasClass('no-cyclic') ? (index = $items.length - 1) : (index = 0));} // down
|
||
|
if (!~index) index=0;
|
||
|
|
||
|
$items.removeClass('focus');
|
||
|
$items.eq(index).addClass('focus');
|
||
|
}
|
||
|
|
||
|
function getParent($this) {
|
||
|
var selector = $this.attr('data-target');
|
||
|
|
||
|
if (!selector) {
|
||
|
selector = $this.attr('href');
|
||
|
selector = selector && /#/.test(selector) && selector.replace(/.*(?=#[^\s]*$)/, ''); //strip for ie7
|
||
|
}
|
||
|
|
||
|
var $parent = selector && $(selector);
|
||
|
|
||
|
return $parent && $parent.length ? $parent : $this.parent();
|
||
|
}
|
||
|
|
||
|
$(document)
|
||
|
.off('keydown.bs.dropdown.data-api')
|
||
|
.on('keydown.bs.dropdown.data-api', '[data-toggle=dropdown], [role=menu]' , onDropDownKeyDown);
|
||
|
|
||
|
/*
|
||
|
* workaround closing menu by right click
|
||
|
* */
|
||
|
(function () {
|
||
|
var eventsObj = $._data($(document).get(0), "events"), clickDefHandler;
|
||
|
if (eventsObj && eventsObj.click) {
|
||
|
eventsObj.click.every(function(e, i, a){
|
||
|
if (/click/.test(e.type) && !e.selector && /bs\..+\.dropdown/.test(e.namespace)) {
|
||
|
clickDefHandler = e.handler;
|
||
|
}
|
||
|
|
||
|
return !clickDefHandler;
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function onDropDownClick(e) {
|
||
|
if ((e.which == 1 || e.which == undefined) && !!clickDefHandler) {
|
||
|
clickDefHandler(e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!!clickDefHandler) {
|
||
|
$(document)
|
||
|
.off('click.bs.dropdown.data-api', clickDefHandler)
|
||
|
.on('click.bs.dropdown.data-api', onDropDownClick);
|
||
|
}
|
||
|
})();
|
||
|
/*
|
||
|
* */
|