%PDF- %PDF-
Direktori : /proc/309157/root/proc/self/root/www/loslex_o/tracker/js/ |
Current File : //proc/309157/root/proc/self/root/www/loslex_o/tracker/js/ace.js |
/* Ace Admin Theme v1.4 Copyright (c) 2016 Mohsen - (twitter.com/responsiweb) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. */ /*! * Ace v1.4.0 */ if (typeof jQuery === 'undefined') { throw new Error('Ace\'s JavaScript requires jQuery') } /** Required. Ace's Basic File to Initiliaze Different Parts and Some Variables. */ //some basic variables (function(undefined) { if( !('ace' in window) ) window['ace'] = {} if( !('helper' in window['ace']) ) window['ace'].helper = {} if( !('vars' in window['ace']) ) window['ace'].vars = {} window['ace'].vars['icon'] = ' ace-icon '; window['ace'].vars['.icon'] = '.ace-icon'; ace.vars['touch'] = ('ontouchstart' in window);//(('ontouchstart' in document.documentElement) || (window.DocumentTouch && document instanceof DocumentTouch)); //sometimes the only good way to work around browser's pecularities is to detect them using user-agents //though it's not accurate var agent = navigator.userAgent ace.vars['webkit'] = !!agent.match(/AppleWebKit/i) ace.vars['safari'] = !!agent.match(/Safari/i) && !agent.match(/Chrome/i); ace.vars['android'] = ace.vars['safari'] && !!agent.match(/Android/i) ace.vars['ios_safari'] = !!agent.match(/OS ([4-9])(_\d)+ like Mac OS X/i) && !agent.match(/CriOS/i) ace.vars['ie'] = window.navigator.msPointerEnabled || (document.all && document.querySelector);//8-11 ace.vars['old_ie'] = document.all && !document.addEventListener;//8 and below ace.vars['very_old_ie'] = document.all && !document.querySelector;//7 and below ace.vars['firefox'] = 'MozAppearance' in document.documentElement.style; ace.vars['non_auto_fixed'] = ace.vars['android'] || ace.vars['ios_safari']; //sometimes we try to use 'tap' event instead of 'click' if jquery mobile plugin is available ace['click_event'] = ace.vars['touch'] && jQuery.fn.tap ? 'tap' : 'click'; })(); //some ace helper functions (function($$ , undefined) {//$$ is ace.helper $$.unCamelCase = function(str) { return str.replace(/([a-z])([A-Z])/g, function(match, c1, c2){ return c1+'-'+c2.toLowerCase() }) } $$.strToVal = function(str) { var res = str.match(/^(?:(true)|(false)|(null)|(\-?[\d]+(?:\.[\d]+)?)|(\[.*\]|\{.*\}))$/i); var val = str; if(res) { if(res[1]) val = true; else if(res[2]) val = false; else if(res[3]) val = null; else if(res[4]) val = parseFloat(str); else if(res[5]) { try { val = JSON.parse(str) } catch (err) {} } } return val; } $$.getAttrSettings = function(elem, attr_list, prefix) { if(!elem) return; var list_type = attr_list instanceof Array ? 1 : 2; //attr_list can be Array or Object(key/value) var prefix = prefix ? prefix.replace(/([^\-])$/ , '$1-') : ''; prefix = 'data-' + prefix; var settings = {} for(var li in attr_list) if(attr_list.hasOwnProperty(li)) { var name = list_type == 1 ? attr_list[li] : li; var attr_val, attr_name = $$.unCamelCase(name.replace(/[^A-Za-z0-9]{1,}/g , '-')).toLowerCase() if( ! ((attr_val = elem.getAttribute(prefix + attr_name)) ) ) continue; settings[name] = $$.strToVal(attr_val); } return settings; } $$.scrollTop = function() { return document.scrollTop || document.documentElement.scrollTop || document.body.scrollTop } $$.winHeight = function() { return window.innerHeight || document.documentElement.clientHeight; } $$.redraw = function(elem, force) { if(!elem) return; var saved_val = elem.style['display']; elem.style.display = 'none'; elem.offsetHeight; if(force !== true) { elem.style.display = saved_val; } else { //force redraw for example in old IE setTimeout(function() { elem.style.display = saved_val; }, 10); } } })(ace.helper); /** <b>Scroll to top button</b>. */ (function($ , undefined) { //the scroll to top button var scroll_btn = $('.btn-scroll-up'); if(scroll_btn.length > 0) { var is_visible = false; $(window).on('scroll.scroll_btn', function() { var scroll = ace.helper.scrollTop(); var h = ace.helper.winHeight(); var body_sH = document.body.scrollHeight; if(scroll > parseInt(h / 4) || (scroll > 0 && body_sH >= h && h + scroll >= body_sH - 1)) {//|| for smaller pages, when reached end of page if(!is_visible) { scroll_btn.addClass('display'); is_visible = true; } } else { if(is_visible) { scroll_btn.removeClass('display'); is_visible = false; } } }).triggerHandler('scroll.scroll_btn'); scroll_btn.on(ace.click_event, function(){ var duration = Math.min(500, Math.max(100, parseInt(ace.helper.scrollTop() / 3))); $('html,body').animate({scrollTop: 0}, duration); return false; }); } })(window.jQuery); /** <b>Load content via Ajax </b>. For more information please refer to documentation #basics/ajax */ (function($ , undefined) { var ajax_loaded_scripts = {} function AceAjax(contentArea, settings) { var $contentArea = $(contentArea); var self = this; $contentArea.attr('data-ajax-content', 'true'); //get a list of 'data-*' attributes that override 'defaults' and 'settings' var attrib_values = ace.helper.getAttrSettings(contentArea, $.fn.ace_ajax.defaults); this.settings = $.extend({}, $.fn.ace_ajax.defaults, settings, attrib_values); var working = false; var $overlay = $();//empty set this.force_reload = false;//set jQuery ajax's cache option to 'false' to reload content this.loadUrl = function(hash, cache, manual_trigger) { var url = false; hash = hash.replace(/^(\#\!)?\#/, ''); this.force_reload = (cache === false) if(typeof this.settings.content_url === 'function') url = this.settings.content_url(hash); if(typeof url === 'string') this.getUrl(url, hash, manual_trigger); } this.loadAddr = function(url, hash, cache) { this.force_reload = (cache === false); this.getUrl(url, hash, false); } this.reload = function() { var hash = $.trim(window.location.hash); if(!hash && this.settings.default_url) hash = this.settings.default_url; this.loadUrl(hash, false); } this.post = function(url, data, updateView, extraParams) { var url = url || $.trim(location.href.replace(location.hash,'')); if(!url) return; var data = data || {} var updateView = updateView || false; this.getUrl(url, null, false, 'POST', data, updateView, extraParams); } this.getUrl = function(url, hash, manual_trigger, method, data, updateView, extraParams) { if(working) { return; } var method = method || 'GET'; var updateView = (method == 'GET') || (method == 'POST' && updateView == true) var data = data || null; var event $contentArea.trigger(event = $.Event('ajaxloadstart'), {url: url, hash: hash, method: method, data: data}) if (event.isDefaultPrevented()) return; self.startLoading(); var ajax_params = method == 'GET' ? {'url': url, 'cache': !this.force_reload} : {'url': url, 'method' : 'POST', 'data': data} if(method == 'POST' && typeof extraParams == 'object') ajax_params = $.extend({}, ajax_params, extraParams); $.ajax(ajax_params) .error(function() { $contentArea.trigger('ajaxloaderror', {url: url, hash: hash, method: method, data: data}); self.stopLoading(true); }) .done(function(result) { $contentArea.trigger('ajaxloaddone', {url: url, hash: hash, method: method, data: data}); if(method == 'POST') { var event $contentArea.trigger(event = $.Event('ajaxpostdone', {url: url, data: data, result: result})) if( event.isDefaultPrevented() ) updateView = false; } var link_element = null, link_text = ''; if(typeof self.settings.update_active === 'function') { link_element = self.settings.update_active.call(null, hash, url, method, updateView); } else if(self.settings.update_active === true && hash) { link_element = $('a[data-url="'+hash+'"]'); if(link_element.length > 0) { var nav = link_element.closest('.nav'); if(nav.length > 0) { nav.find('.active').each(function(){ var $class = 'active'; if( $(this).hasClass('hover') || self.settings.close_active ) $class += ' open'; $(this).removeClass($class); if(self.settings.close_active) { $(this).find(' > .submenu').css('display', ''); } }) var active_li = link_element.closest('li').addClass('active').parents('.nav li').addClass('active open'); nav.closest('.sidebar[data-sidebar-scroll=true]').each(function() { var $this = $(this); $this.ace_sidebar_scroll('reset'); if(manual_trigger == true) $this.ace_sidebar_scroll('scroll_to_active');//first time only }) } } } ///////// if(typeof self.settings.update_breadcrumbs === 'function') { link_text = self.settings.update_breadcrumbs.call(null, hash, url, link_element, method, updateView); } else if(self.settings.update_breadcrumbs === true && link_element != null && link_element.length > 0) { link_text = updateBreadcrumbs(link_element); } ///////// $overlay.addClass('content-loaded').detach(); if(updateView) { //convert "title" and "link" tags to "div" tags for later processing result = String(result) .replace(/<(title|link)([\s\>])/gi,'<div class="hidden ajax-append-$1"$2') .replace(/<\/(title|link)\>/gi,'</div>') $contentArea.empty().html(result); } $(self.settings.loading_overlay || $contentArea).append($overlay); //remove previous stylesheets inserted via ajax if(updateView) setTimeout(function() { $('head').find('link.ace-ajax-stylesheet').remove(); var main_selectors = ['link.ace-main-stylesheet', 'link#main-ace-style', 'link[href*="/ace.min.css"]', 'link[href*="/ace.css"]'] var ace_style = []; for(var m = 0; m < main_selectors.length; m++) { ace_style = $('head').find(main_selectors[m]).first(); if(ace_style.length > 0) break; } $contentArea.find('.ajax-append-link').each(function(e) { var $link = $(this); if ( $link.attr('href') ) { var new_link = jQuery('<link />', {type : 'text/css', rel: 'stylesheet', 'class': 'ace-ajax-stylesheet'}) if( ace_style.length > 0 ) new_link.insertBefore(ace_style); else new_link.appendTo('head'); new_link.attr('href', $link.attr('href'));//we set "href" after insertion, for IE to work } $link.remove(); }) }, 10); ////////////////////// if(typeof self.settings.update_title === 'function') { self.settings.update_title.call(null, hash, url, link_text, method, updateView); } else if(self.settings.update_title === true && method == 'GET') { updateTitle(link_text); } if( !manual_trigger && updateView ) { $('html,body').animate({scrollTop: 0}, 250); } ////////////////////// $contentArea.trigger('ajaxloadcomplete', {url: url, hash: hash, method: method, data:data}); ////////////////////// //if result contains call to "loadScripts" then don't stopLoading now var re = /\.(?:\s*)ace(?:_a|A)jax(?:\s*)\((?:\s*)(?:\'|\")loadScripts(?:\'|\")/; if(result.match(re)) self.stopLoading(); else self.stopLoading(true); }) } /////////////////////// var fixPos = false; var loadTimer = null; this.startLoading = function() { if(working) return; working = true; if(!this.settings.loading_overlay && $contentArea.css('position') == 'static') { $contentArea.css('position', 'relative');//for correct icon positioning fixPos = true; } $overlay.remove(); $overlay = $('<div class="ajax-loading-overlay"><i class="ajax-loading-icon '+(this.settings.loading_icon || '')+'"></i> '+this.settings.loading_text+'</div>') if(this.settings.loading_overlay == 'body') $('body').append($overlay.addClass('ajax-overlay-body')); else if(this.settings.loading_overlay) $(this.settings.loading_overlay).append($overlay); else $contentArea.append($overlay); if(this.settings.max_load_wait !== false) loadTimer = setTimeout(function() { loadTimer = null; if(!working) return; var event $contentArea.trigger(event = $.Event('ajaxloadlong')) if (event.isDefaultPrevented()) return; self.stopLoading(true); }, this.settings.max_load_wait * 1000); } this.stopLoading = function(stopNow) { if(stopNow === true) { working = false; $overlay.remove(); if(fixPos) { $contentArea.css('position', '');//restore previous 'position' value fixPos = false; } if(loadTimer != null) { clearTimeout(loadTimer); loadTimer = null; } } else { $overlay.addClass('almost-loaded'); $contentArea.one('ajaxscriptsloaded.inner_call', function() { self.stopLoading(true); /** if(window.Pace && Pace.running == true) { Pace.off('done'); Pace.once('done', function() { self.stopLoading(true) }) } else self.stopLoading(true); */ }) } } this.working = function() { return working; } /////////////////////// function updateBreadcrumbs(link_element) { var link_text = ''; //update breadcrumbs var breadcrumbs = $('.breadcrumb'); if(breadcrumbs.length > 0 && breadcrumbs.is(':visible')) { breadcrumbs.find('> li:not(:first-child)').remove(); var i = 0; link_element.parents('.nav li').each(function() { var link = $(this).find('> a'); var link_clone = link.clone(); link_clone.find('i,.fa,.glyphicon,.ace-icon,.menu-icon,.badge,.label').remove(); var text = link_clone.text(); link_clone.remove(); var href = link.attr('href'); if(i == 0) { var li = $('<li class="active"></li>').appendTo(breadcrumbs); li.text(text); link_text = text; } else { var li = $('<li><a /></li>').insertAfter(breadcrumbs.find('> li:first-child')); li.find('a').attr('href', href).text(text); } i++; }) } return link_text; } function updateTitle(link_text) { var $title = $contentArea.find('.ajax-append-title'); if($title.length > 0) { document.title = $title.text(); $title.remove(); } else if(link_text.length > 0) { var extra = $.trim(String(document.title).replace(/^(.*)[\-]/, ''));//for example like " - Ace Admin" if(extra) extra = ' - ' + extra; link_text = $.trim(link_text) + extra; } } this.loadScripts = function(scripts, callback) { var scripts = scripts || []; $.ajaxPrefilter('script', function(opts) {opts.cache = true}); setTimeout(function() { //let's keep a list of loaded scripts so that we don't load them more than once! function finishLoading() { if(typeof callback === 'function') callback(); $('.btn-group[data-toggle="buttons"] > .btn').button(); $contentArea.trigger('ajaxscriptsloaded'); } //var deferreds = []; var deferred_count = 0;//deferreds count var resolved = 0; for(var i = 0; i < scripts.length; i++) if(scripts[i]) { (function() { var script_name = "js-"+scripts[i].replace(/[^\w\d\-]/g, '-').replace(/\-\-/g, '-'); if( ajax_loaded_scripts[script_name] !== true ) deferred_count++; })() } function nextScript(index) { index += 1; if(index < scripts.length) loadScript(index); else { finishLoading(); } } function loadScript(index) { index = index || 0; if(!scripts[index]) {//could be null sometimes return nextScript(index); } var script_name = "js-"+scripts[index].replace(/[^\w\d\-]/g, '-').replace(/\-\-/g, '-'); //only load scripts that are not loaded yet! if( ajax_loaded_scripts[script_name] !== true ) { $.getScript(scripts[index]) .done(function() { ajax_loaded_scripts[script_name] = true; }) //.fail(function() { //}) .complete(function() { resolved++; if(resolved >= deferred_count && working) { finishLoading(); } else { nextScript(index); } }) } else {//script previoisly loaded nextScript(index); } } if (deferred_count > 0) { loadScript(); } else { finishLoading(); } }, 10) } ///////////////// $(window) .off('hashchange.ace_ajax') .on('hashchange.ace_ajax', function(e, manual_trigger) { var hash = $.trim(window.location.hash); if(!hash || hash.length == 0) return; if(self.settings.close_mobile_menu) { try {$(self.settings.close_mobile_menu).ace_sidebar('mobileHide')} catch(e){} } if(self.settings.close_dropdowns) { $('.dropdown.open .dropdown-toggle').dropdown('toggle'); } self.loadUrl(hash, null, manual_trigger); }).trigger('hashchange.ace_ajax', [true]); var hash = $.trim(window.location.hash); if(!hash && this.settings.default_url) window.location.hash = this.settings.default_url; }//AceAjax $.fn.aceAjax = $.fn.ace_ajax = function (option, value, value2, value3, value4) { var method_call; var $set = this.each(function () { var $this = $(this); var data = $this.data('ace_ajax'); var options = typeof option === 'object' && option; if (!data) $this.data('ace_ajax', (data = new AceAjax(this, options))); if (typeof option === 'string' && typeof data[option] === 'function') { if(value4 !== undefined) method_call = data[option](value, value2, value3, value4); else if(value3 !== undefined) method_call = data[option](value, value2, value3); else if(value2 !== undefined) method_call = data[option](value, value2); else method_call = data[option](value); } }); return (method_call === undefined) ? $set : method_call; } $.fn.aceAjax.defaults = $.fn.ace_ajax.defaults = { content_url: false, default_url: false, loading_icon: 'fa fa-spin fa-spinner fa-2x orange', loading_text: '', loading_overlay: null, update_breadcrumbs: true, update_title: true, update_active: true, close_active: false, max_load_wait: false, close_mobile_menu: false, close_dropdowns: false } })(window.jQuery); /** <b>Sidebar functions</b>. Collapsing/expanding, toggling mobile view menu and other sidebar functions. */ (function($ , undefined) { var sidebar_count = 0; function Sidebar(sidebar, settings) { var self = this; this.$sidebar = $(sidebar); this.$sidebar.attr('data-sidebar', 'true'); if( !this.$sidebar.attr('id') ) this.$sidebar.attr( 'id' , 'id-sidebar-'+(++sidebar_count) ) //get a list of 'data-*' attributes that override 'defaults' and 'settings' var attrib_values = ace.helper.getAttrSettings(sidebar, $.fn.ace_sidebar.defaults, 'sidebar-'); this.settings = $.extend({}, $.fn.ace_sidebar.defaults, settings, attrib_values); //some vars this.minimized = false;//will be initialized later this.collapsible = false;//... this.horizontal = false;//... this.mobile_view = false;// //return an array containing sidebar state variables this.vars = function() { return {'minimized': this.minimized, 'collapsible': this.collapsible, 'horizontal': this.horizontal, 'mobile_view': this.mobile_view} } this.get = function(name) { if(this.hasOwnProperty(name)) return this[name]; } this.set = function(name, value) { if(this.hasOwnProperty(name)) this[name] = value; } //return a reference to self (sidebar instance) this.ref = function() { return this; } //toggle icon for sidebar collapse/expand button var toggleIcon = function(minimized, save) { var icon = $(this).find(ace.vars['.icon']), icon1, icon2; if(icon.length > 0) { icon1 = icon.attr('data-icon1');//the icon for expanded state icon2 = icon.attr('data-icon2');//the icon for collapsed state if(typeof minimized !== "undefined") { if(minimized) icon.removeClass(icon1).addClass(icon2); else icon.removeClass(icon2).addClass(icon1); } else { icon.toggleClass(icon1).toggleClass(icon2); } try { if(save !== false) ace.settings.saveState(icon.get(0)); } catch(e) {} } } //if not specified, find the toggle button related to this sidebar var findToggleBtn = function() { var toggle_btn = self.$sidebar.find('.sidebar-collapse'); if(toggle_btn.length == 0) toggle_btn = $('.sidebar-collapse[data-target="#'+(self.$sidebar.attr('id')||'')+'"]'); if(toggle_btn.length != 0) toggle_btn = toggle_btn[0]; else toggle_btn = null; return toggle_btn; } //collapse/expand sidebar this.toggleMenu = function(toggle_btn, save) { if(this.collapsible) return false; this.minimized = !this.minimized; var save = !(toggle_btn === false || save === false); if(this.minimized) this.$sidebar.addClass('menu-min'); else this.$sidebar.removeClass('menu-min'); try { if(save) ace.settings.saveState(sidebar, 'class', 'menu-min', this.minimized); } catch(e) {} if( !toggle_btn ) { toggle_btn = findToggleBtn(); } if(toggle_btn) { toggleIcon.call(toggle_btn, this.minimized, save); } //force redraw for ie8 if(ace.vars['old_ie']) ace.helper.redraw(sidebar); $(document).trigger('settings.ace', ['sidebar_collapsed' , this.minimized, sidebar, save]); if( this.minimized ) this.$sidebar.trigger($.Event('collapse.ace.sidebar')); else this.$sidebar.trigger($.Event('expand.ace.sidebar')); return true; } this.collapse = function(toggle_btn, save) { if(this.collapsible) return; this.minimized = false; this.toggleMenu(toggle_btn, save) } this.expand = function(toggle_btn, save) { if(this.collapsible) return; this.minimized = true; this.toggleMenu(toggle_btn, save); } this.showResponsive = function() { this.$sidebar.removeClass(responsive_min_class).removeClass(responsive_max_class); } //collapse/expand in 2nd mobile style this.toggleResponsive = function(toggle_btn, showMenu) { if( !this.mobile_view || this.mobile_style != 3 ) return; if( this.$sidebar.hasClass('menu-min') ) { //remove menu-min because it interferes with responsive-max this.$sidebar.removeClass('menu-min'); var btn = findToggleBtn(); if(btn) toggleIcon.call(btn); } var showMenu = typeof showMenu === 'boolean' ? showMenu : (typeof toggle_btn === 'boolean' ? toggle_btn : this.$sidebar.hasClass(responsive_min_class)); if(showMenu) { this.$sidebar.addClass(responsive_max_class).removeClass(responsive_min_class); } else { this.$sidebar.removeClass(responsive_max_class).addClass(responsive_min_class); } this.minimized = !showMenu; if( !toggle_btn || typeof toggle_btn !== 'object' ) { toggle_btn = this.$sidebar.find('.sidebar-expand'); if(toggle_btn.length == 0) toggle_btn = $('.sidebar-expand[data-target="#'+(this.$sidebar.attr('id')||'')+'"]'); if(toggle_btn.length != 0) toggle_btn = toggle_btn[0]; else toggle_btn = null; } if(toggle_btn) { var icon = $(toggle_btn).find(ace.vars['.icon']), icon1, icon2; if(icon.length > 0) { icon1 = icon.attr('data-icon1');//the icon for expanded state icon2 = icon.attr('data-icon2');//the icon for collapsed state if(!showMenu) icon.removeClass(icon2).addClass(icon1); else icon.removeClass(icon1).addClass(icon2); } } if(showMenu) self.$sidebar.trigger($.Event('mobileShow.ace.sidebar')); else self.$sidebar.trigger($.Event('mobileHide.ace.sidebar')); $(document).triggerHandler('settings.ace', ['sidebar_collapsed' , this.minimized]); } //some helper functions //determine if we have 4th mobile style responsive sidebar and we are in mobile view this.is_collapsible = function() { var toggle return (this.$sidebar.hasClass('navbar-collapse')) && ((toggle = $('.navbar-toggle[data-target="#'+(this.$sidebar.attr('id')||'')+'"]').get(0)) != null) && toggle.scrollHeight > 0 //sidebar is collapsible and collapse button is visible? } //determine if we are in mobile view this.is_mobile_view = function() { var toggle return ((toggle = $('.menu-toggler[data-target="#'+(this.$sidebar.attr('id')||'')+'"]').get(0)) != null) && toggle.scrollHeight > 0 } var submenu_working = false; //show submenu this.show = function(sub, $duration, shouldWait) { //'shouldWait' indicates whether to wait for previous transition (submenu toggle) to be complete or not? shouldWait = (shouldWait !== false); if(shouldWait && submenu_working) return false; var $sub = $(sub); var event; $sub.trigger(event = $.Event('show.ace.submenu')) if (event.isDefaultPrevented()) { return false; } if(shouldWait) submenu_working = true; $duration = typeof $duration !== 'undefined' ? $duration : this.settings.duration; $sub.css({ height: 0, overflow: 'hidden', display: 'block' }) .removeClass('nav-hide').addClass('nav-show')//only for window < @grid-float-breakpoint and .navbar-collapse.menu-min .parent().addClass('open'); sub.scrollTop = 0;//this is for submenu_hover when sidebar is minimized and a submenu is scrollTop'ed using scrollbars ... var complete = function(ev, trigger) { ev && ev.stopPropagation(); $sub .css({'transition-property': '', 'transition-duration': '', overflow:'', height: ''}) //if(ace.vars['webkit']) ace.helper.redraw(sub);//little Chrome issue, force redraw ;) if(trigger !== false) $sub.trigger($.Event('shown.ace.submenu')) if(shouldWait) submenu_working = false; } var finalHeight = sub.scrollHeight; if($duration == 0 || finalHeight == 0 || !$.support.transition.end) { //(if duration is zero || element is hidden (scrollHeight == 0) || CSS3 transitions are not available) complete(); } else { $sub .css({ 'height': finalHeight, 'transition-property': 'height', 'transition-duration': ($duration/1000)+'s' } ) .one($.support.transition.end, complete); //there is sometimes a glitch, so maybe retry if(ace.vars['android'] ) { setTimeout(function() { complete(null, false); ace.helper.redraw(sub); }, $duration + 20); } } return true; } //hide submenu this.hide = function(sub, $duration, shouldWait) { //'shouldWait' indicates whether to wait for previous transition (submenu toggle) to be complete or not? shouldWait = (shouldWait !== false); if(shouldWait && submenu_working) return false; var $sub = $(sub); var event; $sub.trigger(event = $.Event('hide.ace.submenu')) if (event.isDefaultPrevented()) { return false; } if(shouldWait) submenu_working = true; $duration = typeof $duration !== 'undefined' ? $duration : this.settings.duration; var initialHeight = sub.scrollHeight; $sub.css({ height: initialHeight, overflow: 'hidden', display: 'block' }) .parent().removeClass('open'); sub.offsetHeight; //forces the "sub" to re-consider the new 'height' before transition var complete = function(ev, trigger) { ev && ev.stopPropagation(); $sub .css({display: 'none', overflow:'', height: '', 'transition-property': '', 'transition-duration': ''}) .removeClass('nav-show').addClass('nav-hide')//only for window < @grid-float-breakpoint and .navbar-collapse.menu-min if(trigger !== false) $sub.trigger($.Event('hidden.ace.submenu')) if(shouldWait) submenu_working = false; } if( $duration == 0 || initialHeight == 0 || !$.support.transition.end) { //(if duration is zero || element is hidden (scrollHeight == 0) || CSS3 transitions are not available) complete(); } else { $sub .css({ 'height': 0, 'transition-property': 'height', 'transition-duration': ($duration/1000)+'s' } ) .one($.support.transition.end, complete); //there is sometimes a glitch, so maybe retry if(ace.vars['android'] ) { setTimeout(function() { complete(null, false); ace.helper.redraw(sub); }, $duration + 20); } } return true; } //toggle submenu this.toggle = function(sub, $duration) { $duration = $duration || self.settings.duration; if( sub.scrollHeight == 0 ) {//if an element is hidden scrollHeight becomes 0 if( this.show(sub, $duration) ) return 1; } else { if( this.hide(sub, $duration) ) return -1; } return 0; } //toggle mobile menu this.mobileToggle = function(showMenu) { if(this.mobile_view) { if(this.mobile_style == 1 || this.mobile_style == 2) { this.toggleMobile(typeof showMenu === 'object' ? showMenu : null, typeof showMenu === 'boolean' ? showMenu : null); } else if(this.mobile_style == 3) { this.toggleResponsive(typeof showMenu === 'object' ? showMenu : null, typeof showMenu === 'boolean' ? showMenu : null); } //return true; } else if(this.collapsible) { this.toggleCollapsible(typeof showMenu === 'object' ? showMenu : null, typeof showMenu === 'boolean' ? showMenu : null); //return true; } //return true; } this.mobileShow = function() { this.mobileToggle(true); } this.mobileHide = function() { this.mobileToggle(false); } this.toggleMobile = function(toggle_btn, showMenu) { if(!(this.mobile_style == 1 || this.mobile_style == 2)) return; var showMenu = typeof showMenu === 'boolean' ? showMenu : (typeof toggle_btn === 'boolean' ? toggle_btn : !this.$sidebar.hasClass('display')); if( !toggle_btn || typeof toggle_btn !== 'object' ) { toggle_btn = $('.menu-toggler[data-target="#'+(this.$sidebar.attr('id')||'')+'"]'); if(toggle_btn.length != 0) toggle_btn = toggle_btn[0]; else toggle_btn = null; } if(showMenu) { this.$sidebar.addClass('display'); if(toggle_btn) $(toggle_btn).addClass('display'); } else { this.$sidebar.removeClass('display'); if(toggle_btn) $(toggle_btn).removeClass('display'); } if(showMenu) self.$sidebar.trigger($.Event('mobileShow.ace.sidebar')); else self.$sidebar.trigger($.Event('mobileHide.ace.sidebar')); } this.toggleCollapsible = function(toggle_btn, showMenu) { if(this.mobile_style != 4) return; var showMenu = typeof showMenu === 'boolean' ? showMenu : (typeof toggle_btn === 'boolean' ? toggle_btn : !this.$sidebar.hasClass('in')); if(showMenu) { this.$sidebar.collapse('show'); } else { this.$sidebar.removeClass('display'); this.$sidebar.collapse('hide'); } if(showMenu) self.$sidebar.trigger($.Event('mobileShow.ace.sidebar')); else self.$sidebar.trigger($.Event('mobileHide.ace.sidebar')); } //////////////// //private functions //sidebar vars var minimized_menu_class = 'menu-min'; var responsive_min_class = 'responsive-min'; var responsive_max_class = 'responsive-max'; var horizontal_menu_class = 'h-sidebar'; var sidebar_mobile_style = function() { //differnet mobile menu styles this.mobile_style = 1;//default responsive mode with toggle button inside navbar if(this.$sidebar.hasClass('responsive') && !$('.menu-toggler[data-target="#'+this.$sidebar.attr('id')+'"]').hasClass('navbar-toggle')) this.mobile_style = 2;//toggle button behind sidebar else if(this.$sidebar.hasClass(responsive_min_class)) this.mobile_style = 3;//minimized menu else if(this.$sidebar.hasClass('navbar-collapse')) this.mobile_style = 4;//collapsible (bootstrap style) } sidebar_mobile_style.call(self); function update_vars() { this.mobile_view = this.mobile_style < 4 && this.is_mobile_view(); this.collapsible = !this.mobile_view && this.is_collapsible(); this.minimized = (!this.collapsible && this.$sidebar.hasClass(minimized_menu_class)) || (this.mobile_style == 3 && this.mobile_view && this.$sidebar.hasClass(responsive_min_class)) this.horizontal = !(this.mobile_view || this.collapsible) && this.$sidebar.hasClass(horizontal_menu_class) } //update some basic variables $(window).on('resize.sidebar.vars' , function(){ update_vars.call(self); }).triggerHandler('resize.sidebar.vars') //toggling (show/hide) submenu elements this.$sidebar.on(ace.click_event+'.ace.submenu', '.nav-list', function (ev) { var nav_list = this; //check to see if we have clicked on an element which is inside a .dropdown-toggle element?! //if so, it means we should toggle a submenu var link_element = $(ev.target).closest('a'); if(!link_element || link_element.length == 0) return;//return if not clicked inside a link element var minimized = self.minimized && !self.collapsible; //if .sidebar is .navbar-collapse and in small device mode, then let minimized be uneffective if( !link_element.hasClass('dropdown-toggle') ) {//it doesn't have a submenu return //just one thing before we return //if sidebar is collapsed(minimized) and we click on a first level menu item //and the click is on the icon, not on the menu text then let's cancel event and cancel navigation //Good for touch devices, that when the icon is tapped to see the menu text, navigation is cancelled //navigation is only done when menu text is tapped if( ace.click_event == 'tap' && minimized && link_element.get(0).parentNode.parentNode == nav_list )//only level-1 links { var text = link_element.find('.menu-text').get(0); if( text != null && ev.target != text && !$.contains(text , ev.target) ) {//not clicking on the text or its children ev.preventDefault(); return false; } } //ios safari only has a bit of a problem not navigating to link address when scrolling down //specify data-link attribute to ignore this if(ace.vars['ios_safari'] && link_element.attr('data-link') !== 'false') { //only ios safari has a bit of a problem not navigating to link address when scrolling down //please see issues section in documentation document.location = link_element.attr('href'); ev.preventDefault(); return false; } return; } ev.preventDefault(); var sub = link_element.siblings('.submenu').get(0); if(!sub) return false; var $sub = $(sub); var height_change = 0;//the amount of height change in .nav-list var parent_ul = sub.parentNode.parentNode; if ( ( minimized && parent_ul == nav_list ) || ( ( $sub.parent().hasClass('hover') && $sub.css('position') == 'absolute' ) && !self.collapsible ) ) { return false; } var sub_hidden = (sub.scrollHeight == 0) //if not open and visible, let's open it and make it visible if( sub_hidden && self.settings.hide_open_subs ) {//being shown now $(parent_ul).find('> .open > .submenu').each(function() { //close all other open submenus except for the active one if(this != sub && !$(this.parentNode).hasClass('active')) { height_change -= this.scrollHeight; self.hide(this, self.settings.duration, false); } }) } if( sub_hidden ) {//being shown now self.show(sub, self.settings.duration); //if a submenu is being shown and another one previously started to hide, then we may need to update/hide scrollbars //but if no previous submenu is being hidden, then no need to check if we need to hide the scrollbars in advance if(height_change != 0) height_change += sub.scrollHeight;//we need new updated 'scrollHeight' here } else { self.hide(sub, self.settings.duration); height_change -= sub.scrollHeight; //== -1 means submenu is being hidden } //hide scrollbars if content is going to be small enough that scrollbars is not needed anymore //do this almost before submenu hiding begins //but when minimized submenu's toggle should have no effect if (height_change != 0) { if(self.$sidebar.attr('data-sidebar-scroll') == 'true' && !self.minimized) self.$sidebar.ace_sidebar_scroll('prehide', height_change) } return false; }); }//end of Sidebar //sidebar events //menu-toggler $(document) .on(ace.click_event+'.ace.menu', '.menu-toggler', function(e){ var btn = $(this); var sidebar = $(btn.attr('data-target')); if(sidebar.length == 0) return; e.preventDefault(); //sidebar.toggleClass('display'); //btn.toggleClass('display'); sidebar.ace_sidebar('mobileToggle', this); var click_event = ace.click_event+'.ace.autohide'; var auto_hide = sidebar.attr('data-auto-hide') === 'true'; if( btn.hasClass('display') ) { //hide menu if clicked outside of it! if(auto_hide) { $(document).on(click_event, function(ev) { if( sidebar.get(0) == ev.target || $.contains(sidebar.get(0), ev.target) ) { ev.stopPropagation(); return; } sidebar.ace_sidebar('mobileToggle', this, false); $(document).off(click_event); }) } if(sidebar.attr('data-sidebar-scroll') == 'true') sidebar.ace_sidebar_scroll('reset'); } else { if(auto_hide) $(document).off(click_event); } return false; }) //sidebar collapse/expand button .on(ace.click_event+'.ace.menu', '.sidebar-collapse', function(e){ var target = $(this).attr('data-target'), $sidebar = null; if(target) $sidebar = $(target); if($sidebar == null || $sidebar.length == 0) $sidebar = $(this).closest('.sidebar'); if($sidebar.length == 0) return; e.preventDefault(); $sidebar.ace_sidebar('toggleMenu', this); }) //this button is used in `mobile_style = 3` responsive menu style to expand minimized sidebar .on(ace.click_event+'.ace.menu', '.sidebar-expand', function(e){ var target = $(this).attr('data-target'), $sidebar = null; if(target) $sidebar = $(target); if($sidebar == null || $sidebar.length == 0) $sidebar = $(this).closest('.sidebar'); if($sidebar.length == 0) return; var btn = this; e.preventDefault(); $sidebar.ace_sidebar('toggleResponsive', this); var click_event = ace.click_event+'.ace.autohide'; if($sidebar.attr('data-auto-hide') === 'true') { if( $sidebar.hasClass(responsive_max_class) ) { $(document).on(click_event, function(ev) { if( $sidebar.get(0) == ev.target || $.contains($sidebar.get(0), ev.target) ) { ev.stopPropagation(); return; } $sidebar.ace_sidebar('toggleResponsive', btn); $(document).off(click_event); }) } else { $(document).off(click_event); } } }) $.fn.ace_sidebar = function (option, value, value2) { var method_call; var $set = this.each(function () { var $this = $(this); var data = $this.data('ace_sidebar'); var options = typeof option === 'object' && option; if (!data) $this.data('ace_sidebar', (data = new Sidebar(this, options))); if (typeof option === 'string' && typeof data[option] === 'function') { if(value instanceof Array) method_call = data[option].apply(data, value); else if(value2 !== undefined) method_call = data[option](value, value2); else method_call = data[option](value); } }); return (method_call === undefined) ? $set : method_call; }; $.fn.ace_sidebar.defaults = { 'duration': 300, 'hide_open_subs': true } })(window.jQuery); /** <b>Scrollbars for sidebar</b>. This approach can <span class="text-danger">only</span> be used on <u>fixed</u> sidebar. It doesn't use <u>"overflow:hidden"</u> CSS property and therefore can be used with <u>.hover</u> submenus and minimized sidebar. Except when in mobile view and menu toggle button is not in the navbar. */ (function($ , undefined) { //if( !$.fn.ace_scroll ) return; var old_safari = ace.vars['safari'] && navigator.userAgent.match(/version\/[1-5]/i) //NOTE //Safari on windows has not been updated for a long time. //And it has a problem when sidebar is fixed & scrollable and there is a CSS3 animation inside page content. //Very probably windows users of safari have migrated to another browser by now! var is_element_pos = 'getComputedStyle' in window ? //el.offsetHeight is used to force redraw and recalculate 'el.style.position' esp. for webkit! function(el, pos) { el.offsetHeight; return window.getComputedStyle(el).position == pos } : function(el, pos) { el.offsetHeight; return $(el).css('position') == pos } function Sidebar_Scroll(sidebar , settings) { var self = this; var $window = $(window); var $sidebar = $(sidebar), $nav, nav, $toggle, $shortcuts; $nav = $sidebar.find('.nav-list'); nav = $nav.get(0); if(!nav) return; var attrib_values = ace.helper.getAttrSettings(sidebar, $.fn.ace_sidebar_scroll.defaults); this.settings = $.extend({}, $.fn.ace_sidebar_scroll.defaults, settings, attrib_values); var scroll_to_active = self.settings.scroll_to_active; var ace_sidebar = $sidebar.ace_sidebar('ref'); $sidebar.attr('data-sidebar-scroll', 'true'); var scroll_div = null, scroll_content = null, scroll_content_div = null, bar = null, track = null, ace_scroll = null; this.is_scrolling = false; var _initiated = false; this.sidebar_fixed = is_element_pos(sidebar, 'fixed'); var $avail_height, $content_height; var available_height = function() { //available window space var offset = $nav.parent().offset();//because `$nav.offset()` considers the "scrolled top" amount as well if(self.sidebar_fixed) offset.top -= ace.helper.scrollTop(); return $window.innerHeight() - offset.top - ( self.settings.include_toggle ? 0 : $toggle.outerHeight() ) + 1; } var content_height = function() { return nav.clientHeight;//we don't use nav.scrollHeight here, because hover submenus are considered in calculating scrollHeight despite position=absolute! } var initiate = function(on_page_load) { if( _initiated ) return; if( !self.sidebar_fixed ) return;//eligible?? //return if we want scrollbars only on "fixed" sidebar and sidebar is not "fixed" yet! $nav = $sidebar.find('.nav-list'); $toggle = $sidebar.find('.sidebar-toggle').eq(0); $shortcuts = $sidebar.find('.sidebar-shortcuts').eq(0); nav = $nav.get(0); if(!nav) return; //initiate once $nav.wrap('<div class="nav-wrap-up pos-rel" />'); $nav.after('<div><div></div></div>'); $nav.wrap('<div class="nav-wrap" />'); if(!self.settings.include_toggle) $toggle.css({'z-index': 1}); if(!self.settings.include_shortcuts) $shortcuts.css({'z-index': 99}); scroll_div = $nav.parent().next() .ace_scroll({ size: available_height(), //reset: true, mouseWheelLock: true, hoverReset: false, dragEvent: true, styleClass: self.settings.scroll_style, touchDrag: false//disable touch drag event on scrollbars, we'll add a custom one later }) .closest('.ace-scroll').addClass('nav-scroll'); ace_scroll = scroll_div.data('ace_scroll'); scroll_content = scroll_div.find('.scroll-content').eq(0); scroll_content_div = scroll_content.find(' > div').eq(0); track = $(ace_scroll.get_track()); bar = track.find('.scroll-bar').eq(0); if(self.settings.include_shortcuts && $shortcuts.length != 0) { $nav.parent().prepend($shortcuts).wrapInner('<div />'); $nav = $nav.parent(); } if(self.settings.include_toggle && $toggle.length != 0) { $nav.append($toggle); $nav.closest('.nav-wrap').addClass('nav-wrap-t');//it just helps to remove toggle button's top border and restore li:last-child's bottom border } $nav.css({position: 'relative'}); if( self.settings.scroll_outside == true ) scroll_div.addClass('scrollout'); nav = $nav.get(0); nav.style.top = 0; scroll_content.on('scroll.nav', function() { nav.style.top = (-1 * this.scrollTop) + 'px'; }); //mousewheel library available? $nav.on(!!$.event.special.mousewheel ? 'mousewheel.ace_scroll' : 'mousewheel.ace_scroll DOMMouseScroll.ace_scroll', function(event){ if( !self.is_scrolling || !ace_scroll.is_active() ) { return !self.settings.lock_anyway; } //transfer $nav's mousewheel event to scrollbars return scroll_div.trigger(event); }); $nav.on('mouseenter.ace_scroll', function() { track.addClass('scroll-hover'); }).on('mouseleave.ace_scroll', function() { track.removeClass('scroll-hover'); }); /** $(document.body).on('touchmove.nav', function(event) { if( self.is_scrolling && $.contains(sidebar, event.target) ) { event.preventDefault(); return false; } }) */ //you can also use swipe event in a similar way //swipe.nav var content = scroll_content.get(0); $nav.on('ace_drag.nav', function(event) { if( !self.is_scrolling || !ace_scroll.is_active() ) { event.retval.cancel = true; return; } //if submenu hover is being scrolled let's cancel sidebar scroll! if( $(event.target).closest('.can-scroll').length != 0 ) { event.retval.cancel = true; return; } if(event.direction == 'up' || event.direction == 'down') { ace_scroll.move_bar(true); var distance = event.dy; distance = parseInt(Math.min($avail_height, distance)) if(Math.abs(distance) > 2) distance = distance * 2; if(distance != 0) { content.scrollTop = content.scrollTop + distance; nav.style.top = (-1 * content.scrollTop) + 'px'; } } }); //for drag only if(self.settings.smooth_scroll) { $nav .on('touchstart.nav MSPointerDown.nav pointerdown.nav', function(event) { $nav.css('transition-property', 'none'); bar.css('transition-property', 'none'); }) .on('touchend.nav touchcancel.nav MSPointerUp.nav MSPointerCancel.nav pointerup.nav pointercancel.nav', function(event) { $nav.css('transition-property', 'top'); bar.css('transition-property', 'top'); }); } if(old_safari && !self.settings.include_toggle) { var toggle = $toggle.get(0); if(toggle) scroll_content.on('scroll.safari', function() { ace.helper.redraw(toggle); }); } _initiated = true; //if the active item is not visible, scroll down so that it becomes visible //only the first time, on page load if(on_page_load == true) { self.reset();//try resetting at first if( scroll_to_active ) { self.scroll_to_active(); } scroll_to_active = false; } if( typeof self.settings.smooth_scroll === 'number' && self.settings.smooth_scroll > 0) { $nav.css({'transition-property': 'top', 'transition-duration': (self.settings.smooth_scroll / 1000).toFixed(2)+'s'}) bar.css({'transition-property': 'top', 'transition-duration': (self.settings.smooth_scroll / 1500).toFixed(2)+'s'}) scroll_div .on('drag.start', function(e) { e.stopPropagation(); $nav.css('transition-property', 'none') }) .on('drag.end', function(e) { e.stopPropagation(); $nav.css('transition-property', 'top') }); } if(ace.vars['android']) { //force hide address bar, because its changes don't trigger window resize and become kinda ugly var val = ace.helper.scrollTop(); if(val < 2) { window.scrollTo( val, 0 ); setTimeout( function() { self.reset(); }, 20 ); } var last_height = ace.helper.winHeight() , new_height; $(window).on('scroll.ace_scroll', function() { if(self.is_scrolling && ace_scroll.is_active()) { new_height = ace.helper.winHeight(); if(new_height != last_height) { last_height = new_height; self.reset(); } } }); } } this.scroll_to_active = function() { if( !ace_scroll || !ace_scroll.is_active() ) return; try { //sometimes there's no active item or not 'offsetTop' property var $active; var vars = ace_sidebar['vars']() var nav_list = $sidebar.find('.nav-list') if(vars['minimized'] && !vars['collapsible']) { $active = nav_list.find('> .active') } else { $active = $nav.find('> .active.hover') if($active.length == 0) $active = $nav.find('.active:not(.open)') } var top = $active.outerHeight(); nav_list = nav_list.get(0); var active = $active.get(0); while(active != nav_list) { top += active.offsetTop; active = active.parentNode; } var scroll_amount = top - scroll_div.height(); if(scroll_amount > 0) { nav.style.top = -scroll_amount + 'px'; scroll_content.scrollTop(scroll_amount); } }catch(e){} } this.reset = function(recalc) { if(recalc === true) { this.sidebar_fixed = is_element_pos(sidebar, 'fixed'); } if( !this.sidebar_fixed ) { this.disable(); return;//eligible?? } //return if we want scrollbars only on "fixed" sidebar and sidebar is not "fixed" yet! if( !_initiated ) initiate(); //initiate scrollbars if not yet var vars = ace_sidebar['vars'](); //enable if: //menu is not collapsible mode (responsive navbar-collapse mode which has default browser scrollbar) //menu is not horizontal or horizontal but mobile view (which is not navbar-collapse) //and available height is less than nav's height var enable_scroll = !vars['collapsible'] && !vars['horizontal'] && ($avail_height = available_height()) < ($content_height = nav.clientHeight); //we don't use nav.scrollHeight here, because hover submenus are considered in calculating scrollHeight despite position=absolute! this.is_scrolling = true; if( enable_scroll ) { scroll_content_div.css({height: $content_height, width: 8}); scroll_div.prev().css({'max-height' : $avail_height}) ace_scroll.update({size: $avail_height}) ace_scroll.enable(); ace_scroll.reset(); } if( !enable_scroll || !ace_scroll.is_active() ) { if(this.is_scrolling) this.disable(); } else { $sidebar.addClass('sidebar-scroll'); } //return this.is_scrolling; } this.disable = function() { this.is_scrolling = false; if(scroll_div) { scroll_div.css({'height' : '', 'max-height' : ''}); scroll_content_div.css({height: '', width: ''});//otherwise it will have height and takes up some space even when invisible scroll_div.prev().css({'max-height' : ''}) ace_scroll.disable(); } if(parseInt(nav.style.top) < 0 && self.settings.smooth_scroll && $.support.transition.end) { $nav.one($.support.transition.end, function() { $sidebar.removeClass('sidebar-scroll'); $nav.off('.trans'); }); } else { $sidebar.removeClass('sidebar-scroll'); } nav.style.top = 0; } this.prehide = function(height_change) { if(!this.is_scrolling || ace_sidebar.get('minimized')) return;//when minimized submenu's toggle should have no effect if(content_height() + height_change < available_height()) { this.disable(); } else if(height_change < 0) { //if content height is decreasing //let's move nav down while a submenu is being hidden var scroll_top = scroll_content.scrollTop() + height_change if(scroll_top < 0) return; nav.style.top = (-1 * scroll_top) + 'px'; } } this._reset = function(recalc) { if(recalc === true) { this.sidebar_fixed = is_element_pos(sidebar, 'fixed'); } if(ace.vars['webkit']) setTimeout(function() { self.reset() } , 0); else this.reset(); } this.set_hover = function() { if(track) track.addClass('scroll-hover'); } this.get = function(name) { if(this.hasOwnProperty(name)) return this[name]; } this.set = function(name, value) { if(this.hasOwnProperty(name)) this[name] = value; } this.ref = function() { //return a reference to self return this; } this.updateStyle = function(styleClass) { if(ace_scroll == null) return; ace_scroll.update({styleClass: styleClass}); } //change scrollbar size after a submenu is hidden/shown //but don't change if sidebar is minimized $sidebar.on('hidden.ace.submenu.sidebar_scroll shown.ace.submenu.sidebar_scroll', '.submenu', function(e) { e.stopPropagation(); if( !ace_sidebar.get('minimized') ) { //webkit has a little bit of a glitch!!! self._reset(); if( e.type == 'shown' ) self.set_hover(); } }); initiate(true);//true = on_page_load } //reset on document and window changes $(document).on('settings.ace.sidebar_scroll', function(ev, event_name, event_val){ $('.sidebar[data-sidebar-scroll=true]').each(function() { var $this = $(this); var sidebar_scroll = $this.ace_sidebar_scroll('ref'); if( event_name == 'sidebar_collapsed' && is_element_pos(this, 'fixed') ) { if( $this.attr('data-sidebar-hover') == 'true' ) $this.ace_sidebar_hover('reset'); sidebar_scroll._reset(); } else if( event_name === 'sidebar_fixed' || event_name === 'navbar_fixed' ) { var is_scrolling = sidebar_scroll.get('is_scrolling'); var sidebar_fixed = is_element_pos(this, 'fixed') sidebar_scroll.set('sidebar_fixed', sidebar_fixed); if(sidebar_fixed && !is_scrolling) { sidebar_scroll._reset(); } else if( !sidebar_fixed ) { sidebar_scroll.disable(); } } }); }); $(window).on('resize.ace.sidebar_scroll', function(){ $('.sidebar[data-sidebar-scroll=true]').each(function() { var $this = $(this); if( $this.attr('data-sidebar-hover') == 'true' ) $this.ace_sidebar_hover('reset'); ///////////// var sidebar_scroll = $(this).ace_sidebar_scroll('ref'); var sidebar_fixed = is_element_pos(this, 'fixed') sidebar_scroll.set('sidebar_fixed', sidebar_fixed); sidebar_scroll._reset(); }); }) ///////////////////////////////////////////// if(!$.fn.ace_sidebar_scroll) { $.fn.ace_sidebar_scroll = function (option, value) { var method_call; var $set = this.each(function () { var $this = $(this); var data = $this.data('ace_sidebar_scroll'); var options = typeof option === 'object' && option; if (!data) $this.data('ace_sidebar_scroll', (data = new Sidebar_Scroll(this, options))); if (typeof option === 'string' && typeof data[option] === 'function') { method_call = data[option](value); } }); return (method_call === undefined) ? $set : method_call; } $.fn.ace_sidebar_scroll.defaults = { 'scroll_to_active': true, 'include_shortcuts': true, 'include_toggle': false, 'smooth_scroll': 150, 'scroll_outside': false, 'scroll_style': '', 'lock_anyway': false } } })(window.jQuery); /** <b>Submenu hover adjustment</b>. Automatically move up a submenu to fit into screen when some part of it goes beneath window. Pass a "true" value as an argument and submenu will have native browser scrollbars when necessary. */ (function($ , undefined) { if( ace.vars['very_old_ie'] ) return; //ignore IE7 & below var hasTouch = ace.vars['touch']; var nativeScroll = ace.vars['old_ie'] || hasTouch; var is_element_pos = 'getComputedStyle' in window ? //el.offsetHeight is used to force redraw and recalculate 'el.style.position' esp. for webkit! function(el, pos) { el.offsetHeight; return window.getComputedStyle(el).position == pos } : function(el, pos) { el.offsetHeight; return $(el).css('position') == pos } $(window).on('resize.sidebar.ace_hover', function() { $('.sidebar[data-sidebar-hover=true]').ace_sidebar_hover('update_vars').ace_sidebar_hover('reset'); }) $(document).on('settings.ace.ace_hover', function(e, event_name, event_val) { if(event_name == 'sidebar_collapsed') $('.sidebar[data-sidebar-hover=true]').ace_sidebar_hover('reset'); else if(event_name == 'navbar_fixed') $('.sidebar[data-sidebar-hover=true]').ace_sidebar_hover('update_vars'); }) var sidebars = []; function Sidebar_Hover(sidebar , settings) { var self = this, that = this; var attrib_values = ace.helper.getAttrSettings(sidebar, $.fn.ace_sidebar_hover.defaults); this.settings = $.extend({}, $.fn.ace_sidebar_hover.defaults, settings, attrib_values); var $sidebar = $(sidebar), nav_list = $sidebar.find('.nav-list').get(0); $sidebar.attr('data-sidebar-hover', 'true'); sidebars.push($sidebar); var sidebar_vars = {}; var old_ie = ace.vars['old_ie']; var scroll_right = false; //scroll style class var hasHoverDelay = self.settings.sub_hover_delay || false; if(hasTouch && hasHoverDelay) self.settings.sub_hover_delay = parseInt(Math.max(self.settings.sub_hover_delay, 2500));//for touch device, delay is at least 2.5sec var $window = $(window); //navbar used for adding extra offset from top when adjusting submenu var $navbar = $('.navbar').eq(0); var navbar_fixed = $navbar.css('position') == 'fixed'; this.update_vars = function() { navbar_fixed = $navbar.css('position') == 'fixed'; } self.dirty = false; //on window resize or sidebar expand/collapse a previously "pulled up" submenu should be reset back to its default position //for example if "pulled up" in "responsive-min" mode, in "fullmode" should not remain "pulled up" this.reset = function() { if( self.dirty == false ) return; self.dirty = false;//so don't reset is not called multiple times in a row! $sidebar.find('.submenu').each(function() { var $sub = $(this), li = $sub.parent(); $sub.css({'top': '', 'bottom': '', 'max-height': ''}); if($sub.hasClass('ace-scroll')) { $sub.ace_scroll('disable'); } else { $sub.removeClass('sub-scroll'); } if( is_element_pos(this, 'absolute') ) $sub.addClass('can-scroll'); else $sub.removeClass('can-scroll'); li.removeClass('pull_up').find('.menu-text:first').css('margin-top', ''); }) $sidebar.find('.hover-show').removeClass('hover-show hover-shown hover-flip'); } this.updateStyle = function(newStyle) { sub_scroll_style = newStyle; $sidebar.find('.submenu.ace-scroll').ace_scroll('update', {styleClass: newStyle}); } this.changeDir = function(dir) { scroll_right = (dir === 'right'); } //update submenu scrollbars on submenu hide & show var lastScrollHeight = -1; //hide scrollbars if it's going to be not needed anymore! if(!nativeScroll) $sidebar.on('hide.ace.submenu.sidebar_hover', '.submenu', function(e) { if(lastScrollHeight < 1) return; e.stopPropagation(); var $sub = $(this).closest('.ace-scroll.can-scroll'); if($sub.length == 0 || !is_element_pos($sub[0], 'absolute')) return; if($sub[0].scrollHeight - this.scrollHeight < lastScrollHeight) { $sub.ace_scroll('disable'); } }); //reset scrollbars if(!nativeScroll) $sidebar.on('shown.ace.submenu.sidebar_hover hidden.ace.submenu.sidebar_hover', '.submenu', function(e) { if(lastScrollHeight < 1) return; var $sub = $(this).closest('.ace-scroll.can-scroll'); if($sub.length == 0 || !is_element_pos($sub[0], 'absolute') ) return; var sub_h = $sub[0].scrollHeight; if(lastScrollHeight > 14 && sub_h - lastScrollHeight > 4) { $sub.ace_scroll('enable').ace_scroll('reset');//don't update track position } else { $sub.ace_scroll('disable'); } }); /////////////////////// var currentScroll = -1; //some mobile browsers don't have mouseenter var event_1 = !hasTouch ? 'mouseenter.sub_hover' : 'touchstart.sub_hover';// pointerdown.sub_hover'; var event_2 = !hasTouch ? 'mouseleave.sub_hover' : 'touchend.sub_hover touchcancel.sub_hover';// pointerup.sub_hover pointercancel.sub_hover'; $sidebar.on(event_1, '.nav-list li, .sidebar-shortcuts', function (e) { sidebar_vars = $sidebar.ace_sidebar('vars'); //ignore if collapsible mode (mobile view .navbar-collapse) so it doesn't trigger submenu movements //or return if horizontal but not mobile_view (style 1&3) if( sidebar_vars['collapsible'] /**|| sidebar_vars['horizontal']*/ ) return; var $this = $(this); var shortcuts = false; var has_hover = $this.hasClass('hover'); var sub = $this.find('> .submenu').get(0); if( !(sub || ((this.parentNode == nav_list || has_hover || (shortcuts = $this.hasClass('sidebar-shortcuts'))) /**&& sidebar_vars['minimized']*/)) ) { if(sub) $(sub).removeClass('can-scroll'); return;//include .compact and .hover state as well? } var target_element = sub, is_abs = false; if( !target_element && this.parentNode == nav_list ) target_element = $this.find('> a > .menu-text').get(0); if( !target_element && shortcuts ) target_element = $this.find('.sidebar-shortcuts-large').get(0); if( (!target_element || !(is_abs = is_element_pos(target_element, 'absolute'))) && !has_hover ) { if(sub) $(sub).removeClass('can-scroll'); return; } var sub_hide = hasHoverDelay ? getSubHide(this) : null; //var show_sub = false; if(sub) { if(is_abs) { self.dirty = true; var newScroll = ace.helper.scrollTop(); //if submenu is becoming visible for first time or document has been scrolled, then adjust menu if( (hasHoverDelay && !sub_hide.is_visible()) || (!hasTouch && newScroll != currentScroll) || old_ie ) { //try to move/adjust submenu if the parent is a li.hover or if submenu is minimized //if( is_element_pos(sub, 'absolute') ) {//for example in small device .hover > .submenu may not be absolute anymore! $(sub).addClass('can-scroll'); //show_sub = true; if(!old_ie && !hasTouch) adjust_submenu.call(this, sub); else { //because ie8 needs some time for submenu to be displayed and real value of sub.scrollHeight be kicked in var that = this; setTimeout(function() { adjust_submenu.call(that, sub) }, 0) } //} //else $(sub).removeClass('can-scroll'); } currentScroll = newScroll; } else { $(sub).removeClass('can-scroll'); } } //if(show_sub) hasHoverDelay && sub_hide.show(); }).on(event_2, '.nav-list li, .sidebar-shortcuts', function (e) { sidebar_vars = $sidebar.ace_sidebar('vars'); if( sidebar_vars['collapsible'] /**|| sidebar_vars['horizontal']*/ ) return; if( !$(this).hasClass('hover-show') ) return; hasHoverDelay && getSubHide(this).hideDelay(); }); function subHide(li_sub) { var self = li_sub, $self = $(self); var timer = null; var visible = false; this.show = function() { if(timer != null) clearTimeout(timer); timer = null; $self.addClass('hover-show hover-shown'); visible = true; //let's hide .hover-show elements that are not .hover-shown anymore (i.e. marked for hiding in hideDelay) for(var i = 0; i < sidebars.length ; i++) { sidebars[i].find('.hover-show').not('.hover-shown').each(function() { getSubHide(this).hide(); }) } } this.hide = function() { visible = false; $self.removeClass('hover-show hover-shown hover-flip'); if(timer != null) clearTimeout(timer); timer = null; var sub = $self.find('> .submenu').get(0); if(sub) getSubScroll(sub, 'hide'); } this.hideDelay = function(callback) { if(timer != null) clearTimeout(timer); $self.removeClass('hover-shown');//somehow marked for hiding timer = setTimeout(function() { visible = false; $self.removeClass('hover-show hover-flip'); timer = null; var sub = $self.find('> .submenu').get(0); if(sub) getSubScroll(sub, 'hide'); if(typeof callback === 'function') callback.call(this); }, that.settings.sub_hover_delay); } this.is_visible = function() { return visible; } } function getSubHide(el) { var sub_hide = $(el).data('subHide'); if(!sub_hide) $(el).data('subHide', (sub_hide = new subHide(el))); return sub_hide; } function getSubScroll(el, func) { var sub_scroll = $(el).data('ace_scroll'); if(!sub_scroll) return false; if(typeof func === 'string') { sub_scroll[func](); return true; } return sub_scroll; } function adjust_submenu(sub) { var $li = $(this); var $sub = $(sub); sub.style.top = ''; sub.style.bottom = ''; var menu_text = null if( sidebar_vars['minimized'] && (menu_text = $li.find('.menu-text').get(0)) ) { //2nd level items don't have .menu-text menu_text.style.marginTop = ''; } var scroll = ace.helper.scrollTop(); var navbar_height = 0; var $scroll = scroll; if( navbar_fixed ) { navbar_height = sidebar.offsetTop;//$navbar.height(); $scroll += navbar_height + 1; //let's avoid our submenu from going below navbar //because of chrome z-index stacking issue and firefox's normal .submenu over fixed .navbar flicker issue } var off = $li.offset(); off.top = parseInt(off.top); var extra = 0, parent_height; sub.style.maxHeight = '';//otherwise scrollHeight won't be consistent in consecutive calls!? var sub_h = sub.scrollHeight; var parent_height = $li.height(); if(menu_text) { extra = parent_height; off.top += extra; } var sub_bottom = parseInt(off.top + sub_h) var move_up = 0; var winh = $window.height(); //if the bottom of menu is going to go below visible window var top_space = parseInt(off.top - $scroll - extra);//available space on top var win_space = winh;//available window space var horizontal = sidebar_vars['horizontal'], horizontal_sub = false; if(horizontal && this.parentNode == nav_list) { move_up = 0;//don't move up first level submenu in horizontal mode off.top += $li.height(); horizontal_sub = true;//first level submenu } if(!horizontal_sub && (move_up = (sub_bottom - (winh + scroll))) >= 0 ) { //don't move up more than available space move_up = move_up < top_space ? move_up : top_space; //move it up a bit more if there's empty space if(move_up == 0) move_up = 20; if(top_space - move_up > 10) { move_up += parseInt(Math.min(25, top_space - move_up)); } //move it down if submenu's bottom is going above parent LI if(off.top + (parent_height - extra) > (sub_bottom - move_up)) { move_up -= (off.top + (parent_height - extra) - (sub_bottom - move_up)); } if(move_up > 0) { sub.style.top = -(move_up) + 'px'; if( menu_text ) { menu_text.style.marginTop = -(move_up) + 'px'; } } } if(move_up < 0) move_up = 0;//when it goes below var pull_up = move_up > 0 && move_up > parent_height - 20; if(pull_up) { $li.addClass('pull_up'); } else $li.removeClass('pull_up'); //flip submenu if out of window width if(horizontal) { if($li.parent().parent().hasClass('hover-flip')) $li.addClass('hover-flip');//if a parent is already flipped, flip it then! else { var sub_off = $sub.offset(); var sub_w = $sub.width(); var win_w = $window.width(); if(sub_off.left + sub_w > win_w) { $li.addClass('hover-flip'); } } } //don't add scrollbars if it contains .hover menus var has_hover = $li.hasClass('hover') && !sidebar_vars['mobile_view']; if(has_hover && $sub.find('> li > .submenu').length > 0) return; //if( ) { var scroll_height = (win_space - (off.top - scroll)) + (move_up); //if after scroll, the submenu is above parent LI, then move it down var tmp = move_up - scroll_height; if(tmp > 0 && tmp < parent_height) scroll_height += parseInt(Math.max(parent_height, parent_height - tmp)); scroll_height -= 5; if(scroll_height < 90) { return; } var ace_scroll = false; if(!nativeScroll) { ace_scroll = getSubScroll(sub); if(ace_scroll == false) { $sub.ace_scroll({ //hideOnIdle: true, observeContent: true, detached: true, updatePos: false, reset: true, mouseWheelLock: true, styleClass: self.settings.sub_scroll_style }); ace_scroll = getSubScroll(sub); var track = ace_scroll.get_track(); if(track) { //detach it from body and insert it after submenu for better and cosistent positioning $sub.after(track); } } ace_scroll.update({size: scroll_height}); } else { $sub .addClass('sub-scroll') .css('max-height', (scroll_height)+'px') } lastScrollHeight = scroll_height; if(!nativeScroll && ace_scroll) { if(scroll_height > 14 && sub_h - scroll_height > 4) { ace_scroll.enable() ace_scroll.reset(); } else { ace_scroll.disable(); } ////////////////////////////////// var track = ace_scroll.get_track(); if(track) { track.style.top = -(move_up - extra - 1) + 'px'; var off = $sub.position(); var left = off.left if( !scroll_right ) { left += ($sub.outerWidth() - ace_scroll.track_size()); } else { left += 2; } track.style.left = parseInt(left) + 'px'; if(horizontal_sub) {//first level submenu track.style.left = parseInt(left - 2) + 'px'; track.style.top = parseInt(off.top) + (menu_text ? extra - 2 : 0) + 'px'; } } } //} //again force redraw for safari! if( ace.vars['safari'] ) { ace.helper.redraw(sub) } } } ///////////////////////////////////////////// $.fn.ace_sidebar_hover = function (option, value) { var method_call; var $set = this.each(function () { var $this = $(this); var data = $this.data('ace_sidebar_hover'); var options = typeof option === 'object' && option; if (!data) $this.data('ace_sidebar_hover', (data = new Sidebar_Hover(this, options))); if (typeof option === 'string' && typeof data[option] === 'function') { method_call = data[option](value); } }); return (method_call === undefined) ? $set : method_call; } $.fn.ace_sidebar_hover.defaults = { 'sub_hover_delay': 750, 'sub_scroll_style': 'no-track scroll-thin' } })(window.jQuery); /** <b>Widget boxes</b> */ (function($ , undefined) { var Widget_Box = function(box, options) { this.$box = $(box); var that = this; //this.options = $.extend({}, $.fn.widget_box.defaults, options); this.reload = function() { var $box = this.$box; var $remove_position = false; if($box.css('position') == 'static') { $remove_position = true; $box.addClass('position-relative'); } $box.append('<div class="widget-box-overlay"><i class="'+ ace.vars['icon'] + 'loading-icon fa fa-spinner fa-spin fa-2x white"></i></div>'); $box.one('reloaded.ace.widget', function() { $box.find('.widget-box-overlay').remove(); if($remove_position) $box.removeClass('position-relative'); }); } this.closeFast = function() { this.close(0); } this.close = function(closeSpeed) { var $box = this.$box; var closeSpeed = typeof closeSpeed === 'undefined' ? 300 : closeSpeed; $box.fadeOut(closeSpeed , function(){ $box.trigger('closed.ace.widget'); $box.remove(); } ) } this.toggleFast = function() { this.toggle(null, null, 0, 0); } this.toggle = function(type, button, expandSpeed, collapseSpeed) { var $box = this.$box; var $body = $box.find('.widget-body').eq(0); var $icon = null; var event_name = type || ($box.hasClass('collapsed') ? 'show' : 'hide'); var event_complete_name = event_name == 'show' ? 'shown' : 'hidden'; if( !button ) { button = $box.find('> .widget-header a[data-action=collapse]').eq(0); if(button.length == 0) button = null; } if(button) { $icon = button.find(ace.vars['.icon']).eq(0); var $match var $icon_down = null var $icon_up = null if( ($icon_down = $icon.attr('data-icon-show')) ) { $icon_up = $icon.attr('data-icon-hide') } else if( $match = $icon.attr('class').match(/fa\-(.*)\-(up|down)/) ) { $icon_down = 'fa-'+$match[1]+'-down' $icon_up = 'fa-'+$match[1]+'-up' } } var expandSpeed = typeof expandSpeed === 'undefined' ? 250 : expandSpeed; var collapseSpeed = typeof collapseSpeed === 'undefined' ? 200 : collapseSpeed; if( event_name == 'show' ) { if($icon) $icon.removeClass($icon_down).addClass($icon_up); $body.hide(); $box.removeClass('collapsed'); $body.slideDown(expandSpeed, function(){ $box.trigger(event_complete_name+'.ace.widget') }) } else { if($icon) $icon.removeClass($icon_up).addClass($icon_down); $body.slideUp(collapseSpeed, function(){ $box.addClass('collapsed') $box.trigger(event_complete_name+'.ace.widget') } ); } $box.trigger('toggled.ace.widget', [event_name]); } this.hide = function() { this.toggle('hide'); } this.show = function() { this.toggle('show'); } this.hideFast = function() { this.toggle('hide', null, 0, 0); } this.showFast = function() { this.toggle('show', null, 0, 0); } this.fullscreen = function(makeFullscreen) { var $icon = this.$box.find('> .widget-header a[data-action=fullscreen]').find(ace.vars['.icon']).eq(0); var $icon_expand = null var $icon_compress = null if( ($icon_expand = $icon.attr('data-icon1')) ) { $icon_compress = $icon.attr('data-icon2') } else { $icon_expand = 'fa-expand'; $icon_compress = 'fa-compress'; } var isAlreadyFull = this.$box.hasClass('fullscreen'); var noMakeFullscreenParam = makeFullscreen !== true && makeFullscreen !== false;//no true/false arguement provided for this function, so decide based on widget box classnames //make it fullscreen if: //1)we want to go full screen anyway //2)it is not fullscreen already if( makeFullscreen === true || (noMakeFullscreenParam && !isAlreadyFull) ) { $icon.removeClass($icon_expand).addClass($icon_compress); this.$box.addClass('fullscreen'); applyScrollbars(this.$box, true); } else if(makeFullscreen === false || (noMakeFullscreenParam && isAlreadyFull) ) { $icon.addClass($icon_expand).removeClass($icon_compress); this.$box.removeClass('fullscreen'); applyScrollbars(this.$box, false); } this.$box.trigger('fullscreened.ace.widget') } } $.fn.widget_box = function (option, value) { var method_call; var $set = this.each(function () { var $this = $(this); var data = $this.data('widget_box'); var options = typeof option === 'object' && option; if (!data) $this.data('widget_box', (data = new Widget_Box(this, options))); if (typeof option === 'string') method_call = data[option](value); }); return (method_call === undefined) ? $set : method_call; }; $(document).on(ace['click_event']+'.ace.widget', '.widget-header a[data-action]', function (ev) { ev.preventDefault(); var $this = $(this); var $box = $this.closest('.widget-box'); if( $box.length == 0 || $box.hasClass('ui-sortable-helper') ) return; var $widget_box = $box.data('widget_box'); if (!$widget_box) { $box.data('widget_box', ($widget_box = new Widget_Box($box.get(0)))); } var $action = $this.data('action'); if($action == 'collapse') { var event_name = $box.hasClass('collapsed') ? 'show' : 'hide'; var event $box.trigger(event = $.Event(event_name+'.ace.widget')) if (event.isDefaultPrevented()) return $box.trigger(event = $.Event('toggle.ace.widget'), [event_name]); $widget_box.toggle(event_name, $this); } else if($action == 'close') { var event $box.trigger(event = $.Event('close.ace.widget')) if (event.isDefaultPrevented()) return $widget_box.close(); } else if($action == 'reload') { $this.blur(); var event $box.trigger(event = $.Event('reload.ace.widget')) if (event.isDefaultPrevented()) return $widget_box.reload(); } else if($action == 'fullscreen') { var event $box.trigger(event = $.Event('fullscreen.ace.widget')) if (event.isDefaultPrevented()) return $widget_box.fullscreen(); } else if($action == 'settings') { $box.trigger('setting.ace.widget') } }); function applyScrollbars($widget, enable) { var $main = $widget.find('.widget-main').eq(0); $(window).off('resize.widget.scroll'); //IE8 has an unresolvable issue!!! re-scrollbaring with unknown values?! var nativeScrollbars = ace.vars['old_ie'] || ace.vars['touch']; if(enable) { var ace_scroll = $main.data('ace_scroll'); if( ace_scroll ) { $main.data('save_scroll', {size: ace_scroll['size'], lock: ace_scroll['lock'], lock_anyway: ace_scroll['lock_anyway']}); } var size = $widget.height() - $widget.find('.widget-header').height() - 10;//extra paddings size = parseInt(size); $main.css('min-height', size); if( !nativeScrollbars ) { if( ace_scroll ) { $main.ace_scroll('update', {'size': size, 'mouseWheelLock': true, 'lockAnyway': true}); } else { $main.ace_scroll({'size': size, 'mouseWheelLock': true, 'lockAnyway': true}); } $main.ace_scroll('enable').ace_scroll('reset'); } else { if( ace_scroll ) $main.ace_scroll('disable'); $main.css('max-height', size).addClass('overflow-scroll'); } $(window) .on('resize.widget.scroll', function() { var size = $widget.height() - $widget.find('.widget-header').height() - 10;//extra paddings size = parseInt(size); $main.css('min-height', size); if( !nativeScrollbars ) { $main.ace_scroll('update', {'size': size}).ace_scroll('reset'); } else { $main.css('max-height', size).addClass('overflow-scroll'); } }); } else { $main.css('min-height', ''); var saved_scroll = $main.data('save_scroll'); if(saved_scroll) { $main .ace_scroll('update', {'size': saved_scroll['size'], 'mouseWheelLock': saved_scroll['lock'], 'lockAnyway': saved_scroll['lock_anyway']}) .ace_scroll('enable') .ace_scroll('reset'); } if( !nativeScrollbars ) { if(!saved_scroll) $main.ace_scroll('disable'); } else { $main.css('max-height', '').removeClass('overflow-scroll'); } } } })(window.jQuery); /** <b>RTL</b> (right-to-left direction for Arabic, Hebrew, Persian languages). It's good for demo only. You should hard code RTL-specific changes inside your HTML/server-side code. Dynamically switching to RTL using Javascript is not a good idea. Please refer to documentation for more info. */ (function($ , undefined) { //Switching to RTL (right to left) Mode $('#ace-settings-rtl').removeAttr('checked').on('click', function(){ switch_direction(); }); //>>> you should hard code changes inside HTML for RTL direction //you shouldn't use this function to switch direction //this is only for dynamically switching for Ace's demo //take a look at this function to see what changes should be made //also take a look at docs for some tips var switch_direction = function() { applyChanges(); //in ajax when new content is loaded, we dynamically apply RTL changes again //please note that this is only for Ace demo //for info about RTL see Ace's docs $('.page-content-area[data-ajax-content=true]').on('ajaxscriptsloaded.rtl', function() { if( $('body').hasClass('rtl') ) { applyChanges(this); } }); ///////////////////////// function applyChanges(el) { var $body = $(document.body); if(!el) $body.toggleClass('rtl');//el is 'body' el = el || document.body; var $container = $(el); $container //toggle pull-right class on dropdown-menu .find('.dropdown-menu:not(.datepicker-dropdown,.colorpicker)').toggleClass('dropdown-menu-right') .end() //swap pull-left & pull-right .find('.pull-right:not(.dropdown-menu,blockquote,.profile-skills .pull-right)').removeClass('pull-right').addClass('tmp-rtl-pull-right') .end() .find('.pull-left:not(.dropdown-submenu,.profile-skills .pull-left)').removeClass('pull-left').addClass('pull-right') .end() .find('.tmp-rtl-pull-right').removeClass('tmp-rtl-pull-right').addClass('pull-left') .end() .find('.chosen-select').toggleClass('chosen-rtl').next().toggleClass('chosen-rtl'); function swap_classes(class1, class2) { $container .find('.'+class1).removeClass(class1).addClass('tmp-rtl-'+class1) .end() .find('.'+class2).removeClass(class2).addClass(class1) .end() .find('.tmp-rtl-'+class1).removeClass('tmp-rtl-'+class1).addClass(class2) } swap_classes('align-left', 'align-right'); swap_classes('no-padding-left', 'no-padding-right'); swap_classes('arrowed', 'arrowed-right'); swap_classes('arrowed-in', 'arrowed-in-right'); swap_classes('tabs-left', 'tabs-right'); swap_classes('messagebar-item-left', 'messagebar-item-right');//for inbox page $('.modal.aside-vc').ace_aside('flip').ace_aside('insideContainer'); //mirror all icons and attributes that have a "fa-*-right|left" attrobute $container.find('.fa').each(function() { if(this.className.match(/ui-icon/) || $(this).closest('.fc-button').length > 0) return; //skip mirroring icons of plugins that have built in RTL support var l = this.attributes.length; for(var i = 0 ; i < l ; i++) { var val = this.attributes[i].value; if(val.match(/fa\-(?:[\w\-]+)\-left/)) this.attributes[i].value = val.replace(/fa\-([\w\-]+)\-(left)/i , 'fa-$1-right') else if(val.match(/fa\-(?:[\w\-]+)\-right/)) this.attributes[i].value = val.replace(/fa\-([\w\-]+)\-(right)/i , 'fa-$1-left') } }); //browsers are incosistent with horizontal scroll and RTL //so let's make our scrollbars LTR and wrap the content inside RTL var rtl = $body.hasClass('rtl'); if(rtl) { $container.find('.scroll-hz').addClass('make-ltr') .find('.scroll-content') .wrapInner('<div class="make-rtl" />'); $('.sidebar[data-sidebar-hover=true]').ace_sidebar_hover('changeDir', 'right'); } else { //remove the wrap $container.find('.scroll-hz').removeClass('make-ltr') .find('.make-rtl').children().unwrap(); $('.sidebar[data-sidebar-hover=true]').ace_sidebar_hover('changeDir', 'left'); } if($.fn.ace_scroll) $container.find('.scroll-hz').ace_scroll('reset') //to reset scrollLeft //redraw the traffic pie chart on homepage with a different parameter try { var placeholder = $('#piechart-placeholder'); if(placeholder.length > 0) { var pos = $body.hasClass('rtl') ? 'nw' : 'ne';//draw on north-west or north-east? placeholder.data('draw').call(placeholder.get(0) , placeholder, placeholder.data('chart'), pos); } }catch(e) {} ace.helper.redraw(el, true); } } })(jQuery); /** <b>Auto content padding on fixed navbar & breadcrumbs</b>. Navbar's content and height is often predictable and when navbar is fixed we can add appropriate padding to content area using CSS. But when navbar is fixed and its content size and height is not predictable we may need to add the necessary padding to content area dynamically using Javascript. It's not often needed and you can have good results using CSS3 media queries to add necessary paddings based on window size. */ (function($ , undefined) { var navbar = $('.navbar').eq(0); var navbar_container = $('.navbar-container').eq(0); var sidebar = $('.sidebar').eq(0); var main_container = $('.main-container').get(0); var breadcrumbs = $('.breadcrumbs').eq(0); var page_content = $('.page-content').get(0); var default_padding = 8 if(navbar.length > 0) { $(window).on('resize.auto_padding', function() { if( navbar.css('position') == 'fixed' ) { var padding1 = !ace.vars['nav_collapse'] ? navbar.outerHeight() : navbar_container.outerHeight(); padding1 = parseInt(padding1); main_container.style.paddingTop = padding1 + 'px'; if(ace.vars['non_auto_fixed'] && sidebar.length > 0) { if(sidebar.css('position') == 'fixed') { sidebar.get(0).style.top = padding1 + 'px'; } else sidebar.get(0).style.top = ''; } if( breadcrumbs.length > 0 ) { if(breadcrumbs.css('position') == 'fixed') { var padding2 = default_padding + breadcrumbs.outerHeight() + parseInt(breadcrumbs.css('margin-top')); padding2 = parseInt(padding2); page_content.style['paddingTop'] = padding2 + 'px'; if(ace.vars['non_auto_fixed']) breadcrumbs.get(0).style.top = padding1 + 'px'; } else { page_content.style.paddingTop = ''; if(ace.vars['non_auto_fixed']) breadcrumbs.get(0).style.top = ''; } } } else { main_container.style.paddingTop = ''; page_content.style.paddingTop = ''; if(ace.vars['non_auto_fixed']) { if(sidebar.length > 0) { sidebar.get(0).style.top = ''; } if(breadcrumbs.length > 0) { breadcrumbs.get(0).style.top = ''; } } } }).triggerHandler('resize.auto_padding'); $(document).on('settings.ace.auto_padding', function(ev, event_name, event_val) { if(event_name == 'navbar_fixed' || event_name == 'breadcrumbs_fixed') { if(ace.vars['webkit']) { //force new 'css position' values to kick in navbar.get(0).offsetHeight; if(breadcrumbs.length > 0) breadcrumbs.get(0).offsetHeight; } $(window).triggerHandler('resize.auto_padding'); } }); /**$('#skin-colorpicker').on('change', function() { $(window).triggerHandler('resize.auto_padding'); });*/ } })(window.jQuery);