123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532 |
- /**
- * menu - jQuery EasyUI
- *
- * Copyright (c) 2009-2013 www.jeasyui.com. All rights reserved.
- *
- * Licensed under the GPL or commercial licenses
- * To use it on other terms please contact us: info@jeasyui.com
- * http://www.gnu.org/licenses/gpl.txt
- * http://www.jeasyui.com/license_commercial.php
- */
- (function($){
-
- /**
- * initialize the target menu, the function can be invoked only once
- */
- function init(target){
- $(target).appendTo('body');
- $(target).addClass('menu-top'); // the top menu
-
- $(document).unbind('.menu').bind('mousedown.menu', function(e){
- var allMenu = $('body>div.menu:visible');
- var m = $(e.target).closest('div.menu', allMenu);
- if (m.length){return}
- $('body>div.menu-top:visible').menu('hide');
- });
-
- var menus = splitMenu($(target));
- for(var i=0; i<menus.length; i++){
- createMenu(menus[i]);
- }
-
- function splitMenu(menu){
- var menus = [];
- menu.addClass('menu');
- menus.push(menu);
- if (!menu.hasClass('menu-content')){
- menu.children('div').each(function(){
- var submenu = $(this).children('div');
- if (submenu.length){
- submenu.insertAfter(target);
- this.submenu = submenu; // point to the sub menu
- var mm = splitMenu(submenu);
- menus = menus.concat(mm);
- }
- });
- }
- return menus;
- }
-
- function createMenu(menu){
- var width = $.parser.parseOptions(menu[0], ['width']).width;
- if (menu.hasClass('menu-content')){
- menu[0].originalWidth = width || menu._outerWidth();
- } else {
- menu[0].originalWidth = width || 0;
- menu.children('div').each(function(){
- var item = $(this);
- if (item.hasClass('menu-sep')){
- // item.html(' ');
- } else {
- var itemOpts = $.extend({}, $.parser.parseOptions(this,['name','iconCls','href']), {
- disabled: (item.attr('disabled') ? true : undefined)
- });
- item[0].itemName = itemOpts.name || '';
- item[0].itemHref = itemOpts.href || '';
- // item.attr('name',itemOpts.name || '').attr('href',itemOpts.href || '');
-
- var text = item.addClass('menu-item').html();
- item.empty().append($('<div class="menu-text"></div>').html(text));
- if (itemOpts.iconCls){
- $('<div class="menu-icon"></div>').addClass(itemOpts.iconCls).appendTo(item);
- }
- if (itemOpts.disabled){
- setDisabled(target, item[0], true);
- }
- if (item[0].submenu){
- $('<div class="menu-rightarrow"></div>').appendTo(item); // has sub menu
- }
-
- bindMenuItemEvent(target, item);
- }
- });
- $('<div class="menu-line"></div>').prependTo(menu);
- }
- setMenuWidth(target, menu);
- menu.hide();
-
- bindMenuEvent(target, menu);
- }
- }
-
- function setMenuWidth(target, menu){
- var opts = $.data(target, 'menu').options;
- var d = menu.css('display');
- menu.css({
- display: 'block',
- left:-10000
- });
-
- // menu.find('div.menu-item')._outerHeight(22);
- var width = 0;
- menu.find('div.menu-text').each(function(){
- if (width < $(this)._outerWidth()){
- width = $(this)._outerWidth();
- }
- $(this).closest('div.menu-item')._outerHeight($(this)._outerHeight()+2);
- });
- width += 65;
- menu._outerWidth(Math.max((menu[0].originalWidth || 0), width, opts.minWidth));
-
- menu.css('display', d);
- }
-
- /**
- * bind menu event
- */
- function bindMenuEvent(target, menu){
- var state = $.data(target, 'menu');
- menu.unbind('.menu').bind('mouseenter.menu', function(){
- if (state.timer){
- clearTimeout(state.timer);
- state.timer = null;
- }
- }).bind('mouseleave.menu', function(){
- state.timer = setTimeout(function(){
- hideAll(target);
- }, 100);
- });
- }
-
- /**
- * bind menu item event
- */
- function bindMenuItemEvent(target, item){
- item.unbind('.menu');
- item.bind('click.menu', function(){
- if ($(this).hasClass('menu-item-disabled')){
- return;
- }
- // only the sub menu clicked can hide all menus
- if (!this.submenu){
- hideAll(target);
- var href = $(this).attr('href');
- if (href){
- location.href = href;
- }
- }
- var item = $(target).menu('getItem', this);
- $.data(target, 'menu').options.onClick.call(target, item);
- }).bind('mouseenter.menu', function(e){
- // hide other menu
- item.siblings().each(function(){
- if (this.submenu){
- hideMenu(this.submenu);
- }
- $(this).removeClass('menu-active');
- });
- // show this menu
- item.addClass('menu-active');
-
- if ($(this).hasClass('menu-item-disabled')){
- item.addClass('menu-active-disabled');
- return;
- }
-
- var submenu = item[0].submenu;
- if (submenu){
- $(target).menu('show', {
- menu: submenu,
- parent: item
- });
- }
- }).bind('mouseleave.menu', function(e){
- item.removeClass('menu-active menu-active-disabled');
- var submenu = item[0].submenu;
- if (submenu){
- if (e.pageX>=parseInt(submenu.css('left'))){
- item.addClass('menu-active');
- } else {
- hideMenu(submenu);
- }
-
- } else {
- item.removeClass('menu-active');
- }
- });
- }
-
- /**
- * hide top menu and it's all sub menus
- */
- function hideAll(target){
- var state = $.data(target, 'menu');
- if (state){
- if ($(target).is(':visible')){
- hideMenu($(target));
- state.options.onHide.call(target);
- }
- }
- return false;
- }
-
- /**
- * show the menu, the 'param' object has one or more properties:
- * left: the left position to display
- * top: the top position to display
- * menu: the menu to display, if not defined, the 'target menu' is used
- * parent: the parent menu item to align to
- * alignTo: the element object to align to
- */
- function showMenu(target, param){
- var left,top;
- param = param || {};
- var menu = $(param.menu || target);
- if (menu.hasClass('menu-top')){
- var opts = $.data(target, 'menu').options;
- $.extend(opts, param);
- left = opts.left;
- top = opts.top;
- if (opts.alignTo){
- var at = $(opts.alignTo);
- left = at.offset().left;
- top = at.offset().top + at._outerHeight();
- }
- // if (param.left != undefined){left = param.left}
- // if (param.top != undefined){top = param.top}
- if (left + menu.outerWidth() > $(window)._outerWidth() + $(document)._scrollLeft()){
- left = $(window)._outerWidth() + $(document).scrollLeft() - menu.outerWidth() - 5;
- }
- if (top + menu.outerHeight() > $(window)._outerHeight() + $(document).scrollTop()){
- top -= menu.outerHeight();
- }
- } else {
- var parent = param.parent; // the parent menu item
- left = parent.offset().left + parent.outerWidth() - 2;
- if (left + menu.outerWidth() + 5 > $(window)._outerWidth() + $(document).scrollLeft()){
- left = parent.offset().left - menu.outerWidth() + 2;
- }
- var top = parent.offset().top - 3;
- if (top + menu.outerHeight() > $(window)._outerHeight() + $(document).scrollTop()){
- top = $(window)._outerHeight() + $(document).scrollTop() - menu.outerHeight() - 5;
- }
- }
- menu.css({left:left,top:top});
- menu.show(0, function(){
- if (!menu[0].shadow){
- menu[0].shadow = $('<div class="menu-shadow"></div>').insertAfter(menu);
- }
- menu[0].shadow.css({
- display:'block',
- zIndex:$.fn.menu.defaults.zIndex++,
- left:menu.css('left'),
- top:menu.css('top'),
- width:menu.outerWidth(),
- height:menu.outerHeight()
- });
- menu.css('z-index', $.fn.menu.defaults.zIndex++);
- if (menu.hasClass('menu-top')){
- $.data(menu[0], 'menu').options.onShow.call(menu[0]);
- }
- });
- }
-
- function hideMenu(menu){
- if (!menu) return;
-
- hideit(menu);
- menu.find('div.menu-item').each(function(){
- if (this.submenu){
- hideMenu(this.submenu);
- }
- $(this).removeClass('menu-active');
- });
-
- function hideit(m){
- m.stop(true,true);
- if (m[0].shadow){
- m[0].shadow.hide();
- }
- m.hide();
- }
- }
-
- function findItem(target, text){
- var result = null;
- var tmp = $('<div></div>');
- function find(menu){
- menu.children('div.menu-item').each(function(){
- var item = $(target).menu('getItem', this);
- var s = tmp.empty().html(item.text).text();
- if (text == $.trim(s)) {
- result = item;
- } else if (this.submenu && !result){
- find(this.submenu);
- }
- });
- }
- find($(target));
- tmp.remove();
- return result;
- }
-
- function setDisabled(target, itemEl, disabled){
- var t = $(itemEl);
-
- if (disabled){
- t.addClass('menu-item-disabled');
- if (itemEl.onclick){
- itemEl.onclick1 = itemEl.onclick;
- itemEl.onclick = null;
- }
- } else {
- t.removeClass('menu-item-disabled');
- if (itemEl.onclick1){
- itemEl.onclick = itemEl.onclick1;
- itemEl.onclick1 = null;
- }
- }
- }
-
- function appendItem(target, param){
- var menu = $(target);
- if (param.parent){
- if (!param.parent.submenu){
- var submenu = $('<div class="menu"><div class="menu-line"></div></div>').appendTo('body');
- submenu.hide();
- param.parent.submenu = submenu;
- $('<div class="menu-rightarrow"></div>').appendTo(param.parent);
- }
- menu = param.parent.submenu;
- }
- var item = $('<div class="menu-item"></div>').appendTo(menu);
- $('<div class="menu-text"></div>').html(param.text).appendTo(item);
- if (param.iconCls) $('<div class="menu-icon"></div>').addClass(param.iconCls).appendTo(item);
- if (param.id) item.attr('id', param.id);
- // if (param.href) item.attr('href', param.href);
- // if (param.name) item.attr('name', param.name);
- if (param.name){item[0].itemName = param.name}
- if (param.href){item[0].itemHref = param.href}
- if (param.onclick){
- if (typeof param.onclick == 'string'){
- item.attr('onclick', param.onclick);
- } else {
- item[0].onclick = eval(param.onclick);
- }
- }
- if (param.handler) item[0].onclick = eval(param.handler);
-
- bindMenuItemEvent(target, item);
-
- if (param.disabled){
- setDisabled(target, item[0], true);
- }
- bindMenuEvent(target, menu);
- setMenuWidth(target, menu);
- }
-
- function removeItem(target, itemEl){
- function removeit(el){
- if (el.submenu){
- el.submenu.children('div.menu-item').each(function(){
- removeit(this);
- });
- var shadow = el.submenu[0].shadow;
- if (shadow) shadow.remove();
- el.submenu.remove();
- }
- $(el).remove();
- }
- removeit(itemEl);
- }
-
- function destroyMenu(target){
- $(target).children('div.menu-item').each(function(){
- removeItem(target, this);
- });
- if (target.shadow) target.shadow.remove();
- $(target).remove();
- }
-
- $.fn.menu = function(options, param){
- if (typeof options == 'string'){
- return $.fn.menu.methods[options](this, param);
- }
-
- options = options || {};
- return this.each(function(){
- var state = $.data(this, 'menu');
- if (state){
- $.extend(state.options, options);
- } else {
- state = $.data(this, 'menu', {
- options: $.extend({}, $.fn.menu.defaults, $.fn.menu.parseOptions(this), options)
- });
- init(this);
- }
- $(this).css({
- left: state.options.left,
- top: state.options.top
- });
- });
- };
-
- $.fn.menu.methods = {
- options: function(jq){
- return $.data(jq[0], 'menu').options;
- },
- show: function(jq, pos){
- return jq.each(function(){
- showMenu(this, pos);
- });
- },
- hide: function(jq){
- return jq.each(function(){
- hideAll(this);
- });
- },
- destroy: function(jq){
- return jq.each(function(){
- destroyMenu(this);
- });
- },
- /**
- * set the menu item text
- * param: {
- * target: DOM object, indicate the menu item
- * text: string, the new text
- * }
- */
- setText: function(jq, param){
- return jq.each(function(){
- $(param.target).children('div.menu-text').html(param.text);
- });
- },
- /**
- * set the menu icon class
- * param: {
- * target: DOM object, indicate the menu item
- * iconCls: the menu item icon class
- * }
- */
- setIcon: function(jq, param){
- return jq.each(function(){
- var item = $(this).menu('getItem', param.target);
- if (item.iconCls){
- $(item.target).children('div.menu-icon').removeClass(item.iconCls).addClass(param.iconCls);
- } else {
- $('<div class="menu-icon"></div>').addClass(param.iconCls).appendTo(param.target);
- }
- });
- },
- /**
- * get the menu item data that contains the following property:
- * {
- * target: DOM object, the menu item
- * id: the menu id
- * text: the menu item text
- * iconCls: the icon class
- * href: a remote address to redirect to
- * onclick: a function to be called when the item is clicked
- * }
- */
- getItem: function(jq, itemEl){
- var t = $(itemEl);
- var item = {
- target: itemEl,
- id: t.attr('id'),
- text: $.trim(t.children('div.menu-text').html()),
- disabled: t.hasClass('menu-item-disabled'),
- // href: t.attr('href'),
- // name: t.attr('name'),
- name: itemEl.itemName,
- href: itemEl.itemHref,
- onclick: itemEl.onclick
- }
- var icon = t.children('div.menu-icon');
- if (icon.length){
- var cc = [];
- var aa = icon.attr('class').split(' ');
- for(var i=0; i<aa.length; i++){
- if (aa[i] != 'menu-icon'){
- cc.push(aa[i]);
- }
- }
- item.iconCls = cc.join(' ');
- }
- return item;
- },
- findItem: function(jq, text){
- return findItem(jq[0], text);
- },
- /**
- * append menu item, the param contains following properties:
- * parent,id,text,iconCls,href,onclick
- * when parent property is assigned, append menu item to it
- */
- appendItem: function(jq, param){
- return jq.each(function(){
- appendItem(this, param);
- });
- },
- removeItem: function(jq, itemEl){
- return jq.each(function(){
- removeItem(this, itemEl);
- });
- },
- enableItem: function(jq, itemEl){
- return jq.each(function(){
- setDisabled(this, itemEl, false);
- });
- },
- disableItem: function(jq, itemEl){
- return jq.each(function(){
- setDisabled(this, itemEl, true);
- });
- }
- };
-
- $.fn.menu.parseOptions = function(target){
- return $.extend({}, $.parser.parseOptions(target, ['left','top',{minWidth:'number'}]));
- };
-
- $.fn.menu.defaults = {
- zIndex:110000,
- left: 0,
- top: 0,
- minWidth: 120,
- onShow: function(){},
- onHide: function(){},
- onClick: function(item){}
- };
- })(jQuery);
|