Skip to content

Latest commit

 

History

History

addmenuPlus

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 

addMenuPlus.uc.js

addMenuPlus 是一个非常强大的定制菜单的 uc 脚本。通过配置文件可添加、修改、隐藏菜单,修改后无需重启生效。

基于 Griever/addMenu.uc.js 修改

  • 新增修改原有菜单的功能
  • 新增参数 %FAVICON_BASE64%:站点图标的 base64
  • 新增参数 %IMAGE_BASE64%:图片的 BASE64
  • 新增参数 %TITLES%:简短的标题

使用说明及技巧

  • _addmenu.js 文件为配置文件,默认放在 chrome 目录下。
  • about:config 中可通过 addMenu.FILE_PATH 设置配置文件的路径(如果没有手动新建一个)。例如local\_addMenu.js 为相对 chrome 下的路径(windows)。
  • 菜单栏的 "工具" 菜单中有个 "addMenu 的重新载入和编辑" 菜单,左键点击重新载入配置,右键打开文件编辑(需要首先设置 about:config 中 view_source.editor.path 编辑器的路径)
  • ID 为 addMenu-rebuild,可添加到 rebuild_userChrome.uc.xul 统一进行管理
  • 新增 载入配置出错提示,点击可直接定位到第几行,需要首先设置参数,详见 编辑器及参数说明
  • addMenu 脚本配置生成器

可参考的配置

成品

其它

firefox 32+ 右键错位的问题

修改 insertBefore: 'context-reload',insertBefore: 'context-bookmarkpage',insertBefore: 'context-openlinkincurrent',

配置的说明

可添加的范围

  • page: 页面右键菜单
  • tab: 标签右键
  • tool: 工具菜单
  • app: 左上角橙色菜单(firefox 29 以下版本)

二级子菜单

PageMenu, TabMenu, ToolMenu, AppMenu

标签的介绍

label       菜单的名称
accesskey   快捷键
exec        启动外部应用程序。(我新增相对路径。 \\ 代表当前配置的路径,例:\\Chrome 代表配置下的Chrome文件夹)
keyword     指定了关键字的书签或搜索引擎
text        复制你想要的字符串到剪贴板,可与 keyword, exec 一起使用
url         打开你想要的网址
where       打开的位置 (current, tab, tabshifted, window)
condition   菜单出现的条件 (select, link, mailto, image, media, input, noselect, nolink, nomailto, noimage, nomedia, noinput)
oncommand   自定义命令
command     命令的 id
onclick     点击的函数
image       添加图标 (对应图标 url 或 base64)
style       添加样式
...         Firefox 菜单的其它属性

id          标签的ID(我新增的,修改原菜单用)
position/insertBefore/insertAfter: 位置的设置(3选1),position: 1,  insertBefore: "id",  insertAfter: "id"
clone       false 为不克隆,直接改在原菜单上,还原必须重启生效或打开新窗口
onshowing   新增的,当页面右键显示时会执行该函数,可用于动态更改标签标题,详见下面的示例。

参考链接:

可利用的变量

%EOL%            换行(\r\n)
%TITLE%          标题
%TITLES%         简化标题(我新增的,来自 faviconContextMenu.uc.xul.css)
%URL%            地址
%SEL%            选取范围内的文字
%RLINK%          链接的地址
%IMAGE_URL%      图片的 URL
%IMAGE_BASE64%   图片的 Base64(我新增的,不支持 gif 动态图片)
%IMAGE_ALT%      图片的 alt 属性
%IMAGE_TITLE%    图片的 title 属性
%LINK%           链接的地址
%LINK_TEXT%      链接的文本
%RLINK_TEXT%     链接的文本(上面那个的别名)
%MEDIA_URL%      媒体 URL
%CLIPBOARD%      剪贴板的内容
%FAVICON%        Favicon(站点图标) 的 URL
%FAVICON_BASE64% Favicon 的 Base64(我新增的)
%EMAIL%          E-mail 链接
%HOST%           当前网页的域名
%LINK_HOST%      链接的域名
%RLINK_HOST%     链接的域名(同上)

%XXX_HTMLIFIED%  转义后的变量 (XXX 为 上面的 TITLE 等)
%XXX_HTML%       转义后的变量
%XXX_ENCODE%     encodeURIComponent 后的变量

简短的变量

%h               当前网页(域名)
%i               图片的 URL
%l               链接的 URL
%m               媒体的 URL
%p               剪贴板的内容
%s               选取的文字列
%t               标题
%u               URL

隐藏菜单右侧的 tab 提示

css('.addMenu .menu-iconic-accel[value="tab"] { display: none; }');
css('.addMenu .menu-iconic-accel[value="tabshifted"] { display: none; }');
css('.addMenu .menu-iconic-accel[value="window"] { display: none; }');
css('.addMenu .menu-iconic-accel[value="current"] { display: none; }');

示例

打开方式(默认当前页面),通过where 更改,具体tab(前台)、tabshifted(后台)、window(窗口)、current(当前页面)

示例:Google 相似图片搜索

page({
    label: 'Google 相似图片搜索',
    url : 'https://www.google.com/searchbyimage?image_url=%IMAGE_URL%',
    insertAfter: "context-viewimageinfo",
    condition: "image",
    where: 'tab',
});

示例:四引擎搜图

page({
    label: '四引擎搜图',
    condition: "image",
    image: 'http://www.tineye.com/favicon.ico',
    oncommand: function() {
        var url = encodeURIComponent(gContextMenu.mediaURL || gContextMenu.imageURL || gContextMenu.bgImageURL);
        gBrowser.addTab('https://www.google.com/searchbyimage?safe=off&image_url=' + url);
        gBrowser.addTab('http://www.tineye.com/search/?pluginver=firefox-1.0&sort=size&order=desc&url=' + url);
        gBrowser.addTab('http://stu.baidu.com/i?rt=0&rn=10&ct=1&tn=baiduimage&objurl=' + url);
        gBrowser.addTab('http://pic.sogou.com/ris?query=' + url);
    }
});

示例:短网址,分别为当前网页和链接上。

// addMenu 专用
page([{
    label: '短网址',
    condition: 'nolink',
    url: 'javascript:function iprl5(l){var d=document,z=d.createElement("scr"+"ipt"),b=d.body;try{if(!b){throw (0)}if(!l){alert("请输入网址!");return}d.title="(Shortening...) "+d.title;z.setAttribute("src","http://www.ruanyifeng.com/webapp/url_shortener_plugin.php?longUrl="+encodeURIComponent(l));b.appendChild(z)}catch(e){alert("请等待网页加载完毕!")}}iprl5("%URL%");void (0);'
},
{
    label: '短网址(链接)',
    condition: 'link',
    url: 'javascript:function iprl5(l){if(l.startsWith("javascript:")){alert("该网址无效:"+l);return;}var d=document,z=d.createElement("scr"+"ipt"),b=d.body;try{if(!b){throw (0)}if(!l){alert("请输入网址!");return}d.title="(Shortening...) "+d.title;z.setAttribute("src","http://www.ruanyifeng.com/webapp/url_shortener_plugin.php?longUrl="+encodeURIComponent(l));b.appendChild(z)}catch(e){alert("请等待网页加载完毕!")}}iprl5("%RLINK%");void (0);'
}
])

示例:页面右键添加一个复制链接文本的菜单

page({
    label: "复制链接文本",
    accesskey: "C",
    text: "%LINK_TEXT%",
    insertAfter: "context-copylink",
    condition: "link noimage"
});

示例:右键添加 Google Translate 菜单

page({label: "Google Translate",
    url: "http://translate.google.cn/translate?u=%u",
    accesskey: "t",
    where: "tab",
})

示例:右键添加 翻译整个页面 菜单(可用于 https),来源注:github.com 由于服务器限制,无法直接插入 js,而 google 翻译需要插入好几个 js,故无效。

page({
    label: "翻译整个页面",
    insertAfter: "context-selectall",
    image: "moz-anno:favicon:http://translate.google.cn/favicon.ico",
    oncommand: function(){
        var tab = document.getElementById('content');
        var win = tab.selectedBrowser.contentWindow.top.window;
        var d, b, o, v, p;
        d=win.document;b=d.body;o=d.createElement('scri'+'pt');o.setAttribute('src','https://translate.google.cn/translate_a/element.js?cb=googleTranslateElementInit');o.setAttribute('type','text/javascript');b.appendChild(o);v=b.insertBefore(d.createElement('div'),b.firstChild);v.id='google_translate_element';v.style.display='none';p=d.createElement('scri'+'pt');p.text='function googleTranslateElementInit(){new google.translate.TranslateElement({pageLanguage:\"\"},\"google_translate_element\");}';p.setAttribute('type','text/javascript');b.appendChild(p);
    }
});

示例:页面右键添加多个菜单

page([
    {
        label: "复制链接文本",
        text: "%LINK_TEXT%",
    },
    { },  // 分隔条
    {
        label: '复制图像base64',
        text: "%IMAGE_BASE64%",
        condition: "image",
    }
]);

示例:标签右键菜单

	tab([
	    {
	        label: "复制标题",
	        text: "%TITLE%",
	    },
	    {
	        label: "复制标题+URL",
	        text: "%TITLE%\n%URL%",
	    },
	    {
	        label: "复制标题(MD)",
	        accesskey: "D",
	        text: "[%TITLE%](%URL%)",
	    },
	    {
	        label: "复制标题(BBS)",
	        text: "[url=%URL%]%TITLE%[/url]",
	    },
	    {
	        label: "复制标题(Html)",
	        text: '<a href="%URL%">%TITLE%</a>',
	    },
	    {
            label: "复制标题(Link)",
            class: "copy",
            oncommand: function(){
                var url = addMenu.convertText('%URL%'),
                    label = addMenu.convertText('%TITLE%');
                addMenu.copyLink(url, label);
            }
        },
	    {},
	    {
	        label: "复制 Favicon 的 URL",
	        text: "%FAVICON%",
	    }, 
	    {
	        label: "复制 Favicon 的 Base64",
	        text: "%FAVICON_BASE64%",
	    },
	    {
	        label: "切换编码(gbk、utf-8)",
	        accesskey: "e",
	        oncommand: function () {
	            var charset = gBrowser.mCurrentBrowser._docShell.charset;
	            BrowserSetForcedCharacterSet(charset == "gbk" ? "utf-8" : "gbk");
	        }
	    },
	    {
	        label: "关闭所有标签页",
	        oncommand: "gBrowser.removeAllTabsBut(gBrowser.addTab('about:newtab'));",
	        insertAfter:"context_closeOtherTabs",
	        accesskey: "Q"
	    },
	    {
	        label: "复制所有标签标题+地址",
	        class: "copy",
	        oncommand: function(){
	            var text = "";
	            var tabs = gBrowser.mTabContainer.childNodes;
	            for (var i = 0, l = tabs.length, doc; i < l; i++) {
	                doc = tabs[i].linkedBrowser.contentDocument;
	                text += doc.title + "\n" + doc.location.href + "\n";
	            }

	            Cc["@mozilla.org/widget/clipboardhelper;1"].getService(Ci.nsIClipboardHelper).copyString(text);
	        }
	    },
	]);

示例:页面右键添加多功能子菜单

// 多功能菜单
var pagesub = PageMenu({ label: "多功能菜单", accesskey: "z", condition: "normal" , insertBefore: 'context-reload'});
pagesub([
    {
        label:"繁体转简体",
        image:"",
        oncommand: function(){
            content.document.documentElement.appendChild(content.document.createElement("script")).src = "http://tongwen.openfoundry.org/NewTongWen/tools/bookmarklet_cn2.js";
            content.document.documentElement.appendChild(content.document.createElement("style")).textContent = 'body { font-family: "微软雅黑" }';
        }
    },
    {
        label:"自动刷新",
        url: "javascript:(function(p)%7Bopen('','',p).document.write('%3Cbody%20id=1%3E%3Cnobr%20id=2%3E%3C/nobr%3E%3Chr%3E%3Cnobr%20id=3%3E%3C/nobr%3E%3Chr%3E%3Ca%20href=%22#%22onclick=%22return!(c=t)%22%3E%E7%82%B9%E5%87%BB%E5%BC%BA%E5%88%B6%E5%88%B7%E6%96%B0%3C/a%3E%3Cscript%3Efunction%20i(n)%7Breturn%20d.getElementById(n)%7Dfunction%20z()%7Bc+=0.2;if(c%3E=t)%7Bc=0;e.location=u;r++%7Dx()%7Dfunction%20x()%7Bs=t-Math.floor(c);m=Math.floor(s/60);s-=m*60;i(1).style.backgroundColor=(r==0%7C%7Cc/t%3E2/3?%22fcc%22:c/t%3C1/3?%22cfc%22:%22ffc%22);i(2).innerHTML=%22%E5%88%B7%E6%96%B0%E8%AE%A1%E6%95%B0:%20%22+r;i(3).innerHTML=%22%E5%88%B7%E6%96%B0%E5%80%92%E8%AE%A1%E6%97%B6:%20%22+m+%22:%22+(s%3C10?%220%22+s:s)%7Dc=r=0;d=document;e=opener.top;u=prompt(%22%E9%93%BE%E6%8E%A5%E5%9C%B0%E5%9D%80%22,e.location.href);t=u?prompt(%22%E5%88%B7%E6%96%B0%E9%97%B4%E9%9A%94/%E7%A7%92%EF%BC%9A%22,300):0;setInterval(%22z()%22,200);if(!t)%7Bwindow.close()%7D%3C/script%3E%3C/body%3E')%7D)('status=0,scrollbars=0,width=240,height=160,left=1,top=1')",
        image:""
    }, {
        label:"全页面截图",
        image:"",
        oncommand: function () {
            var canvas = document.createElementNS("http://www.w3.org/1999/xhtml", "canvas");
            canvas.width = content.document.documentElement.scrollWidth;
            canvas.height = content.document.documentElement.scrollHeight;
            var ctx = canvas.getContext("2d");
            ctx.drawWindow(content, 0, 0, canvas.width, canvas.height, "rgb(255,255,255)");
            saveImageURL(canvas.toDataURL(), content.document.title + ".png", null, null, null, null, document);
        }
    }, {
        label: "宽度匹配",
        url: "javascript:(function(){function%20t(f){a=d.createNodeIterator(d,1,f,false);while(a.nextNode()){}}var%20d=document;t(function(e){x=e.offsetLeft;l=e.offsetParent;while(l!=null){x+=l.offsetLeft;l=l.offsetParent}var%20w=d.documentElement.clientWidth-x;var%20s=e.style;if(s.marginLeft)w-=s.marginLeft;if(s.marginRight)w-=s.marginRight;if(s.paddingLeft)w-=s.paddingLeft;if(s.paddingRight)w-=s.paddingRight;if(s.borderSize)w-=s.borderSize;w-=d.defaultView.innerWidth-d.documentElement.offsetWidth;if(e.tagName=='IMG'){h=e.clientHeight*w/e.clientWidth;s.maxHeight=h}s.maxWidth=w+'px'})})();",
        image: "",
    }, {
        label: "破解右键防复制",
        url: "javascript:alert(document.body.oncontextmenu=document.body.onmouseup=document.body.onmousemove=document.body.onclick=document.body.onselectstart%20=document.body.oncopy=document.onmousedown%20=%20document.onkeydown%20=null)",
        image: "",
    }, 
    {
        label: "Google 站内搜索",
        accesskey: "s",
        url: "javascript:q%20=%20%22%22%20+%20(window.getSelection%20?%20window.getSelection()%20:%20document.getSelection%20?%20document.getSelection()%20:%20document.selection.createRange().text);%20if%20(!q)%20q%20=%20prompt(%22%E8%AF%B7%E8%BE%93%E5%85%A5%E5%85%B3%E9%94%AE%E8%AF%8D:%22,%20%22%22);%20if%20(q!=null)%20{var%20qlocation=%22%20%22;qlocation=('http://www.google.com/search?num=30&hl=zh-CN&newwindow=1&q='+q+'&sitesearch='+location.host+'');window.open(qlocation);}%20void%200"
    },
    {
        label: "明文显示密码",
        condition: "input",
        url: "javascript:(function()%7Bvar%20IN,F;IN=document.getElementsByTagName('input');for(var%20i=0;i<IN.length;i++)%7BF=IN%5Bi%5D;if(F.type.toLowerCase()=='password')%7Btry%7BF.type='text'%7Dcatch(r)%7Bvar%20n,Fa;n=document.createElement('input');Fa=F.attributes;for(var%20ii=0;ii<Fa.length;ii++)%7Bvar%20k,knn,knv;k=Fa%5Bii%5D;knn=k.nodeName;knv=k.nodeValue;if(knn.toLowerCase()!='type')%7Bif(knn!='height'&&knn!='width'&!!knv)n%5Bknn%5D=knv%7D%7D;F.parentNode.replaceChild(n,F)%7D%7D%7D%7D)()"
    },
    {},
    {
        label:"复制扩展清单",
        image:"chrome://mozapps/skin/extensions/extensionGeneric-16.png",
        oncommand: function () {
            Application.extensions ? Cc['@mozilla.org/widget/clipboardhelper;1'].getService(Ci.nsIClipboardHelper).copyString(Application.extensions.all.map(function (item, id) {
                    return id + 1 + ": " + item._item.name;
                }).join("\n")) : Application.getExtensions(function (extensions) {
                Cc['@mozilla.org/widget/clipboardhelper;1'].getService(Ci.nsIClipboardHelper).copyString(extensions.all.map(function (item, id) {
                        return id + 1 + ": " + item._item.name;
                    }).join("\n"));
            })
        }
    },
    {
        label: "复制用户脚本清单",
        oncommand: function () {
            Cu.import("resource://gre/modules/AddonManager.jsm");

            AddonManager.getAddonsByTypes(['greasemonkey-user-script', 'userscript'], function (aAddons) {
                var downURLs = [];
                aAddons.forEach(function (aAddon) {
                    var name, downURL;
                    if (aAddon._script) {  // Greasemonkey
                        name = aAddon._script.name;
                        downURL = aAddon._script._downloadURL;
                    } else {  // Scriptish
                        name = aAddon._name;
                        downURL = aAddon._downloadURL;
                        if (!downURL && item._updateURL) {
                            downURL = item._updateURL.replace(/\.meta\.js$/, '.user.js');
                        }
                        if (!downURL && item._homepageURL) {
                            downURL = item._homepageURL;
                        }
                    }

                    downURLs.push(name + '\n' + downURL);
                });
                Cc['@mozilla.org/widget/clipboardhelper;1'].getService(Ci.nsIClipboardHelper).
                    copyString(downURLs.join('\n\n'));
            });
        }
    },
    {label: "批量安装 GM 脚本",
        tooltiptext: "从剪贴板中的多个网址安装,是覆盖安装",
        oncommand: function() {
            if (!window.GM_BrowserUI) return;

            var scope = {};
            Cu.import('resource://greasemonkey/remoteScript.js', scope);

            var install_GM_script = function(url) {
                var rs = new scope.RemoteScript(url);
                rs.download(function(aSuccess, aType) {
                    if (aSuccess && 'dependencies' == aType) {
                        rs.install();
                    }
                });
            };

            var data = readFromClipboard();
            var m = data.match(/(https?:\/\/.*?\.user\.js)/g);
            if (m) {
                m.forEach(function(url) {
                    // 处理下 userscripts 的问题
                    url = url.replace(/^https?:\/\/userscripts\.org\//, 'http://userscripts.org:8080/');

                    install_GM_script(url);
                })
            }
        }
    },
    {
        label:"为此页搜索油侯脚本",
        url: "https://www.google.com/search?q=site:userscripts.org%20%HOST%",
        where: "tab",
        image:""
    },
    {},
    {label: "Google Web Cache",
        url: "http://webcache.googleusercontent.com/search?q=cache:%u",
        accesskey: "c",
        where: "tab",
    },
    {label: "Web Archive",
        url: "http://web.archive.org/web/*/%u",
        accesskey: "w",
        where: "tab",
    },
    {},
    {
        label: "Chrome",
        exec: "\\chrome",
    },
    {
        label: "gm_scripts",
        exec: "\\gm_scripts",
    },
    {
        label: "userChromeJS content",
        exec: "\\extensions\\[email protected]\\content",
    },
    {},
    {
        label: "重新加载配置",
        accesskey: "r",
        oncommand: "setTimeout(function(){ addMenu.rebuild(true); }, 10);"
    }
]);

示例:菜单出现的条件,排除了链接、图片、输入框、选择等多个条件

page({
    label: "snaplinks批量操作模式",
    condition: "nolink noimage noinput noselect",
    oncommand: "snapLinks.init();"
});

示例:隐藏 App 菜单的 Web 开发者

css("#appmenu_webDeveloper { display: none; }");

示例:打开相对路径程序或文件夹

page({
    label: "执行相对路径程序",
    // exec: "\\1\\base64.exe",  // 执行当前配置文件夹下的程序
    exec: "\\Chrome"  // 打开当前配置下的Chrome文件夹
});

示例:添加图标,如果是原有的菜单需要加上 class,menuitem-iconic 或 menu-iconic,前一个菜单项(无子菜单),后一个是是菜单(包含子菜单)。

page({
    label: "图标测试菜单",
    image: "chrome://browser/content/abouthome/bookmarks.png"
    // 或
    // image: ""
});

示例:输入框右键增加 "重做" 菜单

page({
    label: "重做",
    condition: "input",
    insertAfter: "context-undo",
    command: "cmd_redo",
    accesskey: "y"
});

示例:输入框右键增加 "粘贴并确定" 菜单

page({
    label: "粘贴并确定",
    condition: "input",
    insertAfter: "context-paste",
    oncommand: function(event) {
        goDoCommand("cmd_paste");

         window.QueryInterface(Ci.nsIInterfaceRequestor)
             .getInterface(Ci.nsIDOMWindowUtils)
             .sendKeyEvent("keypress", KeyEvent.DOM_VK_RETURN, 0, 0);
    }
})

示例:标签右键或链接右键增加 复制地址(BBS、MD) 菜单,左键复制 BBS 格式,中键原标题,右键 MD 格式,可去除标题一定内容。

function copyBBS_or_MD(event){
    var title = addMenu.convertText("%RLINK_TEXT%") || addMenu.convertText("%TITLE%"),
        url = addMenu.convertText("%RLINK%") || addMenu.convertText("%URL%");

    [" - 互助分享 - 大气谦和!", "_免费高速下载|百度云 网盘-分享无限制", " - Powered by Discuz!",
        "百度云 网盘-", "的分享", 
    ].forEach(function(r){ title = title.replace(r, ""); });

    switch(event.button){
        case 0:
            addMenu.copy("[url=" + url + "]" + title + "[/url]");
            break;
        case 1:
            addMenu.copy(title);
            event.target.parentNode.hidePopup();
            break;
        case 2:
            addMenu.copy("[" + title + "](" + url + ")");
            break;
    }
}

tab({
    label: "复制地址(BBS、MD)",
    tooltiptext: "左键复制 BBS 格式,右键 MD 格式",
    onclick: copyBBS_or_MD
});

page({
    label: "复制链接(BBS、MD)",
    condition: "link noimage",
    tooltiptext: "左键复制 BBS 格式,右键 MD 格式",
    insertBefore: "context-sep-copylink",
    onclick: copyBBS_or_MD
});

示例:输入框右键增加菜单,在光标处插入自定义字符。这种方式在百度贴吧无效,因其输入框较为特殊,可采用下面通用的复制粘贴的方式。

page({
    label: "在输入框光标处插入字符(测试)",
    condition: "input",
    insertAfter: "context-paste",
    oncommand: function(event) {
        var aText = "123";

        var input = gContextMenu.target;
        var aStart = aEnd = input.selectionStart;

        // 在光标处插入字符
        input.value = input.value.slice(0, aStart) + aText + input.value.slice(aEnd);

        // 移动光标到插入字符的后面
        var aOffset = aStart + aText.length;
        input.setSelectionRange(aOffset, aOffset);
    }
});

defpt 写的灌水的菜单 - 卡饭论坛

// 快捷回复
var Quickpostsub = PageMenu({
    label:"Quick Reply With...",
    condition:"input",
    insertBefore:"context-undo",
    oncommand: function(event){
        var text = event.target.getAttribute('txt');
        if(text) {
            addMenu.copy(text);
            goDoCommand("cmd_paste"); 
        }
    }
});

Quickpostsub([
    {label:"Outlook~~~",txt: "[email protected]",image:" "},
    {label:"Gmail~~~",txt: "[email protected]",image:" "},
    {},
    {label:"谢谢你的解答~~~", txt: "非常感谢你的解答!!!",image:" "},
    {label:"不用客气~~~", txt: "不用客气,大家互相帮助……\u256E\uFF08\u256F\u25C7\u2570\uFF09\u256D",image:" "},
    {label:"看起来很不错~~~", txt: "看起来很不错哦,收了~~~\n谢谢LZ啦!!!",image:" "},
    {label:"谢谢楼主分享~~~", txt: "谢谢楼主的分享!这个绝对要顶!!!",image:" "},
    {label:"楼上正解~~~", txt: "楼上正解……\u0285\uFF08\u00B4\u25D4\u0C6A\u25D4\uFF09\u0283",image:" "},
    {label:"坐等楼下解答~~~", txt: "坐等楼下高手解答……⊙_⊙",image:" "}
]);

示例:右键搜索图片二级菜单,来自卡饭论坛

var imagesub = PageMenu({
    label: "Reverse Image Search",
    accesskey: "I",
    condition: "image",
    insertAfter: "context-inspect",
    image: "",
});
imagesub([{
    label: 'Google Search',
    url: 'http://www.google.com/searchbyimage?image_url=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: "http://www.google.com/favicon.ico"
}, {
    label: 'Baidu Search',
    url: 'http://stu.baidu.com/i?rt=0&rn=10&ct=1&tn=baiduimage&objurl=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: ""
}, {
    label: 'Sougou Search',
    url: 'http://pic.sogou.com/ris?query=%IMAGE_URL%',
    condition: "image",
    where: "tab",
    image: ""
}, {
    label: 'TinEye Search',
    url: 'http://www.tineye.com/search?url=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: ""
}, {
    label: 'IQDB Search',
    url: 'http://iqdb.org/?url=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: ""
}, {
    label: 'Regex Search',
    url: 'http://regex.info/exif.cgi/?url=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: ""
}, {
    label: 'saucenao Search',
    url: 'http://saucenao.com/search.php?db=999&url=%IMAGE_URL%',
    condition: "image",
    where: 'tab',
    image: ""
}, ]);

示例:撤销关闭二级菜单 By feiruo

var undoMenu = PageMenu({
    label: '撤销关闭',
    position: 2,
    tooltiptext: "右键显示所有历史记录",
    onclick: "if (event.button == 2) {PlacesCommandHook.showPlacesOrganizer('History');}",
    onpopupshowing: function(e) {
        var popup = e.target;
        popup.setAttribute('id', 'addUndoMneun');
        var items = popup.querySelectorAll('.bookmark-item');
        [].forEach.call(items, function(item) {
            item.parentNode.removeChild(item);
        });
        var undoItems = JSON.parse(Cc['@mozilla.org/browser/sessionstore;1'].getService(Ci.nsISessionStore).getClosedTabData(window));
        if (undoItems.length == 0) {
            popup.setAttribute('oncommand', 'this.parentNode._placesView._onCommand(event);');
            if (!this.parentNode._placesView) new HistoryMenu(event);
        } else {
            undoItems.map(function(item, id) {
                var m = document.createElement('menuitem');
                m.setAttribute('label', item.title);
                m.setAttribute('image', item.image ? 'moz-anno:favicon:' + item.image : '');
                m.setAttribute('class', 'menuitem-iconic bookmark-item closedtab');
                m.setAttribute('oncommand', 'undoCloseTab(' + id + ')');
                m.setAttribute('type', 'unclose-menuitem');
                popup.appendChild(m);
            });
        }
    },
    image: ""
});
undoMenu([{
    label: "恢复上一次会话",
    command: "Browser:RestoreLastSession",
}]);

示例:保存所有图片到 zip

page({
    label: "保存所有图片到 zip",
    oncommand: function() {
        // 保存ディレクトリのパスがない場合は毎回ダイアログで決める
        //var path = "C:\\Users\\azu\\Downloads"; // エスケープしたディレクトリのパス
        var path = "";
        if (!path) {
            // ファイル保存ダイアログ
            var nsIFilePicker = Ci.nsIFilePicker;
            var FP = Cc['@mozilla.org/filepicker;1'].createInstance(nsIFilePicker);
            FP.init(window, 'Choose save folder.', nsIFilePicker.modeGetFolder);

            // ダイアログ表示
            if (FP.show() == nsIFilePicker.returnOK) {
                path = FP.file.path;
            } else {
                return false;
            }
        }
        // ダウンロードしたページを表示するために URI オブジェクト生成
        var hostURL = Components.classes['@mozilla.org/network/io-service;1'].getService(Components.interfaces.nsIIOService).newURI(location.href, null, null);
        // ページに貼り付けられた画像を保存する
        var links = content.document.images;
        var pack = [];
        for (var i = 0, length = links.length; i < length; i++) {
            // JPEG と PNG を保存する
            if (links[i].src.match(/\.jpe?g|\.png|img\.blogs\.yahoo(.*)folder[^thumb]/i)) {
                pack.push([links[i].src.split("/").pop(), links[i].src]);
            }
        }
        zipDeKure(pack, path);


        function zipDeKure(urls, savePath) {
            const ioService = Cc["@mozilla.org/network/io-service;1"].getService(Ci.nsIIOService);
            const zipWriter = Components.Constructor("@mozilla.org/zipwriter;1", "nsIZipWriter");
            var uri = content.window.location.href;
            var fileName = uri.substring(uri.lastIndexOf('://') + 3, uri.length);
            fileName = fileName.split(".").join("_");
            fileName = fileName.split("/").join("_");
            fileName = fileName.split("?").join("_");
            var path = savePath + "\\" + fileName + ".zip";
            var file = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsILocalFile);
            file.initWithPath(path);
            var zipW = new zipWriter();
            var ioFlag = 0x04 | 0x08 | 0x20;
            zipW.open(file, ioFlag);
            for (var i = 0, len = urls.length; i < len; i++) {
                var [name, url] = urls[i];
                var ch = ioService.newChannel(url, "UTF-8", null);
                var stream = ch.open();
                zipW.addEntryStream(name, Date.now() * 1000, Ci.nsIZipWriter.COMPRESS_DEFAULT, stream, false);
            }
            zipW.close();
        }
    },
    image: ""
})

示例:左键用傲游打开当前页,右键直接打开傲游,相对路径(上上层 parent)。

page({
    label: "傲游浏览器",
    tooltiptext: "左键:打开傲游\r\n右键:用傲游打开",
    onclick: function(e) {
        var maxthonPath = Services.dirsvc.get("ProfD", Ci.nsILocalFile).parent.parent.path + "\\Maxthon\\bin\\Maxthon.exe";
        switch (e.button) {
            case 0:
                var url = addMenu.convertText("%URL%");
                addMenu.exec(maxthonPath, url);
                break;
            case 2:
                addMenu.exec(maxthonPath);
                break;
        }
    },
})

示例:使用 Everything 搜索选中的文字,需要自行设置路径。

page({
    label: "Everything 搜索",
    oncommand: function(){
        var str = addMenu.convertText('%s');

        addMenu.exec("D:\\Program Files\\Everything\\Everything.exe", ['-search', str]);
    }
})

示例:快速保存选定文本为 txt 并打开。

page({
    label: '快速保存选定文本',
    condition: 'select',
    oncommand: function() {
        if (!window.NetUtil) Cu.import("resource://gre/modules/NetUtil.jsm");
        if (!window.FileUtils) Cu.import("resource://gre/modules/FileUtils.jsm");

        goDoCommand('cmd_copy');
        var data = readFromClipboard();

        var fp = Cc["@mozilla.org/filepicker;1"].createInstance(Ci.nsIFilePicker);
        fp.init(window, "另存为", Ci.nsIFilePicker.modeSave);
        fp.appendFilter("文本文件", "*.txt");
        fp.defaultString = content.document.title + '.txt';
  
        var res = fp.show();
        if (res != Ci.nsIFilePicker.returnCancel) {
            var aFile = fp.file;
  
            var ostream = FileUtils.openSafeFileOutputStream(aFile);
  
            var converter = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter);
            converter.charset = "gbk";
            var istream = converter.convertToInputStream(data);
  
            NetUtil.asyncCopy(istream, ostream, function(status) {
                if (!Components.isSuccessCode(status)) {
                    // Handle error!
                    return;
                }
  
                aFile.launch();
            });
        }
    }
})

示例:Firefox 31+ 横排菜单,在链接上和非链接上不相同

var openMenu = GroupMenu({
    label: '打开...',
    condition: 'noinput noselect nomailto nocanvas nomedia',
    insertBefore: 'context-sep-navigation'
});
openMenu([
    {
        label:"复制文本+链接",
        text:"%RLT_OR_UT%\n%RLINK_OR_URL%",
        image:""
    },
    {
        label:"在隐私窗打开",
        oncommand: "openLinkIn(addMenu.convertText('%RLINK_OR_URL%'), 'window',{private:true});",
        image:""
    },
    {
        label: "在 IE 中打开",
        text: "%RLINK_OR_URL%",
        exec: "C:\\Program Files\\Internet Explorer\\iexplore.exe",
    },
    {
        label: "在 Chrome 中打开",
        text: '%RLINK_OR_URL%',
        exec: Services.dirsvc.get("LocalAppData", Ci.nsILocalFile).path + "\\Google\\Chrome\\Application\\chrome.exe",
    },
    // {
    //     label: "在 Opera 中打开",
    //     text : "%RLINK_OR_URL%",
    //     exec : "D:\\Program Files\\Opera\\opera.exe",
    // },
]);

示例:子菜单中的 测试视频链接 只在 youku 页面显示,其它页面隐藏。

var testMenu = PageMenu({
    label: '测试子菜单',
    onpopupshowing: function (event){
        Array.slice(event.target.children).forEach(function(elem){
            if(elem.id == "mtest-video"){
                elem.hidden = content.location.host.indexOf("youku") == -1
            }
        });
    }
});
testMenu([
    {
        id: "mtest-video",
        label: "测试视频链接",
        hidden: true
    }
]);

示例:2014-9-8 版本之后新增的 onshowing 函数,在右键菜单出现时会执行该函数。例如下面这个例子只在卡饭论坛显示 插入 code 代码 右键菜单。

page({
    label: '插入 code 代码',
    id: 'addMenu-insert-bbcode',
    condition:"input",
    insertBefore: "context-undo",
    oncommand: function() {
        var str = addMenu.convertText('[code]%P[/code]');
        addMenu.copy(str);
        goDoCommand('cmd_paste'); 
    },
    onshowing: function(menuitem) {
        var isHidden = !(content.location.host == 'bbs.kafan.cn');
        this.hidden = isHidden;
    },
})

示例:动态显示标签标题,详见 怎么样用addmenuplus实现一个这样的菜单项

page({
    label: '复制: ...',
    text: '%SEL%',
    onshowing: function(menuitem) {
        var sel = getBrowserSelection(16);
        if (sel && sel.length > 15)
            sel = sel.substr(0,15) + "...";
        this.label = '复制: ' +  sel;
    },
})

示例:动态显示标签标题(二级菜单)。

var menu = PageMenu({
    label: '搜索',
    onshowing: function(menu) {
        var sel = getBrowserSelection(16);
        if (sel && sel.length > 15)
            sel = sel.substr(0,15) + "...";
        this.label = '搜索: ' +  sel;
    }
});
menu([
    {
        label: '百度',
        url: "http://www.baidu.com/s?ie=UTF-8&wd=%SEL%",
    },
    {
        label: 'Bing',
        url: "http://www.bing.com/search?q=%SEL%",
    },
]);

特殊的示例

示例:"字符编码" 移动到 "web开发者" 的位置

tab({
    id: "appmenu_developer_charsetMenu",
    insertAfter: "appmenu_webDeveloper",
    // clone: false,  // 不克隆,直接改在原来的菜单上面
});

示例:给 firebug 添加一个 accesskey

page({
    id: "menu_firebug_firebugInspect",
    accesskey: "R",
    clone: false
});

示例:修改错误控制台的按键(Ctrl + Shift + J)为以前的版本的控制台

page({
    id: "key_browserConsole",
    command: "",
    oncommand: "toJavaScriptConsole();",
    clone: false
})

示例:移动星星到书签栏后面,并修正图标大小(firefox 26)

page({
    id: 'star-button',
    insertAfter: 'personal-bookmarks',
    style: 'margin-top:5px;margin-bottom:5px;',
    clone: false
})

示例:给 flashgot 添加功能:右键点击一次性调用下载工具,并不改变默认的下载工具

page({
    id: "flashgot-menupopup-options",
    clone: false,
    onclick: function(event) {
        if (event.button == 2) {
            event.preventDefault();

            var defaultDM = gFlashGotService.defaultDM;
            gFlashGotService.defaultDM = event.target.label;
            window.setTimeout(function() {
                gFlashGot.download();
                gFlashGotService.defaultDM = defaultDM;
                document.getElementById("contentAreaContextMenu").hidePopup();
            }, 0);
        }
    },
});

使用了其它皮肤,右键错位的情况

把代码最后几行的css删除,请不要保留空行。

.addMenu > .menu-iconic-left {\
  -moz-appearance: menuimage;\
}\

可调用的方法

addMenu.copy()  // 复制
addMenu.exec(path, arg)  // 启动程序
addMenu.$$('a:not(:empty)', null, true)  // 获取所有被选中的链接
addMenu.convertText('%s');   // 得到选取的文字