Instantly share code, notes, and snippets.
Created
November 20, 2012 10:21
-
Star
(0)
0
You must be signed in to star a gist -
Fork
(0)
0
You must be signed in to fork a gist
-
Save dzfl/4117137 to your computer and use it in GitHub Desktop.
addMenu.uc.js to Everything
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @name addMenu.uc.js | |
// @description メニューを拡張する userChromeJS 用のスクリプト | |
// @namespace http://d.hatena.ne.jp/Griever/ | |
// @author Griever | |
// @include main | |
// @license MIT License | |
// @compatibility Firefox 5 | |
// @charset UTF-8 | |
// @version 0.0.4 | |
// @note 0.0.4 アイコン用の CSS を追加 | |
// @note 0.0.4 設定ファイルから CSS を追加できるようにした | |
// @note 0.0.4 label の無い menu を splitmenu 風の動作にした | |
// @note 0.0.4 Vista でアイコンがズレる問題を修正…したかも | |
// @note 0.0.4 %SEL% の改行が消えてしまうのを修正 | |
// @note 0.0.3 keyword の新しい書式で古い書式が動かない場合があったのを修正 | |
// @note %URL_HTMLIFIED%, %EOL_ENCODE% が変換できなかったミスを修正 | |
// @note %LINK_OR_URL% 変数を作成(リンク URL がなければページの URL を返す) | |
// @note タブの右クリックメニューでは %URL% や %SEL% はそのタブのものを返すようにした | |
// @note keyword で "g %URL%" のような記述を可能にした | |
// @note ツールの再読み込みメニューの右クリックで設定ファイルを開くようにした | |
// ==/UserScript== | |
/***** 説明 ***** | |
◆ これは何? ◆ | |
メニューを拡張する userChromeJS 用のスクリプトです。 | |
作成に当たっては Copy URL Lite+ を参考にさせていただきました。 | |
・http://www.code-404.net/articles/browsers/copy-url-lite | |
◆ 使い方 ◆ | |
設定ファイル(_addmenu.js)を chrome フォルダにおいてください。 | |
.uc.js の方はどこでも構いません。 | |
ブラウザ起動後に設定ファイルが読み込まれ、メニューが追加されます。 | |
設定ファイルの再読み込みはツールメニューから行えます。 | |
◆ 書式 ◆ | |
page, tab, too, app 関数にメニューの素となるオブジェクトを渡す。 | |
オブジェクトのプロパティがそのまま menuitem の属性になります。 | |
○exec | |
外部アプリを起動します。 | |
パラメータは text プロパティを利用します。 | |
アプリのアイコンが自動で付きます。 | |
○keyword | |
ブックマークや検索エンジンのキーワードを指定します。 | |
text プロパティがあればそれを利用して検索などをします。 | |
検索エンジンなどのアイコンが自動で付きます。 | |
○text(変数が利用可能) | |
クリップボードにコピーしたい文字列を指定します。(Copy URL Lite+ 互換) | |
keyword, exec があればそれらの補助に使われます。 | |
○url(変数が利用可能) | |
開きたい URL を指定します。 | |
内容によっては自動的にアイコンが付きます。 | |
○where | |
keyword, url でのページの開き方を指定できます(current, tab, tabshifted, window) | |
省略するとブックマークのように左クリックと中クリックを使い分けられます。 | |
○condition | |
メニューを表示する条件を指定します。(Copy URL Lite+ 互換) | |
省略すると url や text プロパティから自動的に表示/非表示が決まります。 | |
select, link, mailto, image, media, input, noselect, nolink, nomailto, noimage, nomedia, noinput から組み合わせて使います。 | |
○oncommand, command | |
これらがある時は condition 以外の特殊なプロパティは無視されます。 | |
◆ サブメニュー ◆ | |
PageMenu, TabMenu, ToolMenu, AppMenu 関数を使って自由に追加できます。 | |
◆ 利用可能な変数 ◆ | |
%EOL% 改行(\r\n) | |
%TITLE% ページタイトル | |
%URL% URI | |
%SEL% 選択範囲の文字列 | |
%RLINK% リンクアンカー先の URL | |
%IMAGE_URL% 画像の URL | |
%IMAGE_ALT% 画像の alt 属性 | |
%IMAGE_TITLE% 画像の title 属性 | |
%LINK% リンクアンカー先の URL | |
%LINK_TEXT% リンクのテキスト | |
%RLINK_TEXT% リンクのテキスト | |
%MEDIA_URL% メディアの URL | |
%CLIPBOARD% クリップボードの内容 | |
%FAVICON% Favicon の URL | |
%EMAIL% リンク先の E-mail アドレス | |
%HOST% ページのホスト(ドメイン) | |
%LINK_HOST% リンクのホスト(ドメイン) | |
%RLINK_HOST% リンクのホスト(ドメイン) | |
%LINK_OR_URL% リンクの URL が取れなければページの URL | |
%RLINK_OR_URL% リンクの URL が取れなければページの URL | |
%XXX_HTMLIFIED% HTML エンコードされた上記変数(XXX → TITLE などに読み替える) | |
%XXX_HTML% HTML エンコードされた上記変数 | |
%XXX_ENCODE% URI エンコードされた上記変数 | |
◇ 簡易的な変数 ◇ | |
%h ページのホスト(ドメイン) | |
%i 画像の URL | |
%l リンクの URL | |
%m メディアの URL | |
%p クリップボードの内容 | |
%s 選択文字列 | |
%t ページのタイトル | |
%u ページの URL | |
基本的に Copy URL Lite+ の変数はそのまま使えます。 | |
大文字・小文字は区別しません。 | |
*/ | |
(function(css){ | |
let { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components; | |
if (window.addMenu) { | |
window.addMenu.destroy(); | |
delete window.addMenu; | |
} | |
window.addMenu = { | |
get prefs() { | |
delete this.prefs; | |
return this.prefs = Services.prefs.getBranch("addMenu.") | |
}, | |
get FILE() { | |
let aFile; | |
try { | |
// addMenu.FILE_PATH があればそれを使う | |
aFile = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsILocalFile) | |
aFile.initWithPath(this.prefs.getCharPref("FILE_PATH")); | |
} catch (e) { | |
aFile = Services.dirsvc.get("UChrm", Ci.nsILocalFile); | |
aFile.appendRelativePath("_addmenu.js"); | |
} | |
delete this.FILE; | |
return this.FILE = aFile; | |
}, | |
get focusedWindow() { | |
return gContextMenu && gContextMenu.target ? gContextMenu.target.ownerDocument.defaultView : content; | |
}, | |
init: function() { | |
let he = "(?:_HTML(?:IFIED)?|_ENCODE)?"; | |
let rTITLE = "%TITLE"+ he +"%|%t\\b"; | |
let rURL = "%(?:R?LINK_OR_)?URL"+ he +"%|%u\\b"; | |
let rHOST = "%HOST"+ he +"%|%h\\b"; | |
let rSEL = "%SEL"+ he +"%|%s\\b"; | |
let rLINK = "%R?LINK(?:_TEXT|_HOST)?"+ he +"%|%l\\b"; | |
let rIMAGE = "%IMAGE(?:_URL|_ALT|_TITLE)"+ he +"%|%i\\b"; | |
let rMEDIA = "%MEDIA_URL"+ he +"%|%m\\b"; | |
let rCLIPBOARD = "%CLIPBOARD"+ he +"%|%p\\b"; | |
let rFAVICON = "%FAVICON"+ he +"%"; | |
let rEMAIL = "%EMAIL"+ he +"%"; | |
let rExt = "%EOL"+ he +"%"; | |
this.rTITLE = new RegExp(rTITLE, "i"); | |
this.rURL = new RegExp(rURL, "i"); | |
this.rHOST = new RegExp(rHOST, "i"); | |
this.rSEL = new RegExp(rSEL, "i"); | |
this.rLINK = new RegExp(rLINK, "i"); | |
this.rIMAGE = new RegExp(rIMAGE, "i"); | |
this.rMEDIA = new RegExp(rMEDIA, "i"); | |
this.rCLIPBOARD = new RegExp(rCLIPBOARD, "i"); | |
this.rFAVICON = new RegExp(rFAVICON, "i"); | |
this.rEMAIL = new RegExp(rEMAIL, "i"); | |
this.rExt = new RegExp(rExt, "i"); | |
this.regexp = new RegExp( | |
[rTITLE, rURL, rHOST, rSEL, rLINK, rIMAGE, rMEDIA, rCLIPBOARD, rFAVICON, rEMAIL, rExt].join("|"), "ig"); | |
var ins; | |
ins = $("context-viewinfo"); | |
ins.parentNode.insertBefore($E(<menuseparator id="addMenu-page-insertpoint" class="addMenu-insert-point" />), ins.nextSibling); | |
ins = $("context_closeTab"); | |
ins.parentNode.insertBefore($E(<menuseparator id="addMenu-tab-insertpoint" class="addMenu-insert-point" />), ins.nextSibling); | |
ins = $("prefSep"); | |
ins.parentNode.insertBefore($E(<menuseparator id="addMenu-tool-insertpoint" class="addMenu-insert-point" />), ins.nextSibling); | |
ins = $("appmenu-quit"); | |
ins.parentNode.insertBefore($E(<menuseparator id="addMenu-app-insertpoint" class="addMenu-insert-point" />), ins); | |
ins = $("devToolsSeparator"); | |
ins.parentNode.insertBefore($E( | |
<menuitem id="addMenu-rebuild" | |
label={U("addMenu の再読み込み")} | |
oncommand="setTimeout(function(){ addMenu.rebuild(true); }, 10);" | |
onclick="if (event.button == 2) { event.preventDefault(); addMenu.edit(addMenu.FILE); }" /> | |
), ins); | |
$("contentAreaContextMenu").addEventListener("popupshowing", this, false); | |
this.style = addStyle(css); | |
this.rebuild(); | |
}, | |
uninit: function() { | |
$("contentAreaContextMenu").removeEventListener("popupshowing", this, false); | |
}, | |
destroy: function() { | |
this.uninit(); | |
this.removeMenuitem(); | |
$$('#addMenu-rebuild, .addMenu-insert-point').forEach(function(e) e.parentNode.removeChild(e)); | |
if (this.style && this.style.parentNode) this.style.parentNode.removeChild(this.style); | |
if (this.style2 && this.style2.parentNode) this.style2.parentNode.removeChild(this.style2); | |
}, | |
handleEvent: function(event) { | |
switch(event.type){ | |
case "popupshowing": | |
if (event.target != event.currentTarget) return; | |
var state = []; | |
if (gContextMenu.onTextInput) | |
state.push("input"); | |
if (gContextMenu.isTextSelected || | |
gContextMenu.onTextInput && this.getInputSelection(gContextMenu.target)) | |
state.push("select"); | |
if (gContextMenu.onLink) | |
state.push(gContextMenu.onMailtoLink ? "mailto" : "link"); | |
if (gContextMenu.onCanvas) | |
state.push("canvas image"); | |
if (gContextMenu.onImage) | |
state.push("image"); | |
if (gContextMenu.onVideo || gContextMenu.onAudio) | |
state.push("media"); | |
event.currentTarget.setAttribute("addMenu", state.join(" ")); | |
break; | |
} | |
}, | |
onCommand: function(event) { | |
var menuitem = event.target; | |
var text = menuitem.getAttribute("text") || ""; | |
var keyword = menuitem.getAttribute("keyword") || ""; | |
var url = menuitem.getAttribute("url") || ""; | |
var where = menuitem.getAttribute("where") || ""; | |
var exec = menuitem.getAttribute("exec") || ""; | |
if (keyword) { | |
let kw = keyword + (text? " " + (text = this.convertText(text)) : ""); | |
let newurl = getShortcutOrURI(kw); | |
if (newurl == kw && text) | |
return this.log(U("キーワードが見つかりません: ") + keyword); | |
this.openCommand(event, newurl, where); | |
} | |
else if (url) | |
this.openCommand(event, this.convertText(url), where); | |
else if (exec) | |
this.exec(exec, this.convertText(text)); | |
else if (text) | |
this.copy(this.convertText(text)); | |
}, | |
openCommand: function(event, url, where) { | |
var uri; | |
try { | |
uri = Services.io.newURI(url, null, null); | |
} catch (e) { | |
return this.log(U("URL が不正です: ") + url); | |
} | |
if (uri.scheme === "javascript") | |
loadURI(url); | |
else if (where) | |
openUILinkIn(uri.spec, where); | |
else if (event.button == 1) | |
openNewTabWith(uri.spec); | |
else openUILink(uri.spec, event); | |
}, | |
exec: function(path, arg){ | |
var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsILocalFile); | |
var process = Cc['@mozilla.org/process/util;1'].createInstance(Ci.nsIProcess); | |
try { | |
// to Everything | |
arg = arg.replace( | |
/\"(.*?)\"/g, | |
function (whole,s1) { | |
return( s1.replace(/\s+/g, "__SPACE__") ); | |
} | |
); | |
// to Everything end | |
var a = (typeof arg == 'string' || arg instanceof String) ? arg.split(/\s+/) : [arg]; | |
// to Everything | |
var UI = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter); | |
UI.charset = window.navigator.platform.toLowerCase().indexOf("win") >= 0? "Shift_JIS": "UTF-8"; | |
arg = UI.ConvertFromUnicode(arg) + UI.Finish(); | |
for(var i in a){ | |
a[i] = UI.ConvertFromUnicode(a[i]) | |
a[i] = a[i].replace(/__SPACE__/g, " "); | |
} | |
// to Everything end | |
file.initWithPath(path); | |
process.init(file); | |
process.run(false, a, a.length); | |
} catch(e) { | |
this.log(e); | |
} | |
}, | |
rebuild: function(isAlert) { | |
var aFile = this.FILE; | |
if (!aFile || !aFile.exists() || !aFile.isFile()) { | |
this.log(aFile? aFile.path : U("設定ファイル") + U(" が見つかりません")); | |
return; | |
} | |
var aiueo = [ | |
{ current: "page", submenu: "PageMenu", insertId: "addMenu-page-insertpoint" }, | |
{ current: "tab" , submenu: "TabMenu" , insertId: "addMenu-tab-insertpoint" }, | |
{ current: "tool", submenu: "ToolMenu", insertId: "addMenu-tool-insertpoint" }, | |
{ current: "app" , submenu: "AppMenu" , insertId: "addMenu-app-insertpoint" } | |
]; | |
var data = loadText(aFile); | |
var sandbox = new Cu.Sandbox( new XPCNativeWrapper(window) ); | |
sandbox.Components = Components; | |
sandbox.Cc = Cc; | |
sandbox.Ci = Ci; | |
sandbox.Cr = Cr; | |
sandbox.Cu = Cu; | |
sandbox.Services = Services; | |
sandbox.locale = Services.prefs.getCharPref("general.useragent.locale"); | |
sandbox.include = function(aLeafName) { | |
try { | |
let aFile = addMenu.FILE.parent; | |
aFile.QueryInterface(Ci.nsILocalFile); | |
aFile.appendRelativePath(aLeafName); | |
if (!aFile.exists() || !aFile.isFile()) | |
throw U(aLeafName + " が見つかりません"); | |
let fileURL = Services.io.getProtocolHandler("file").QueryInterface(Ci.nsIFileProtocolHandler).getURLSpecFromFile(aFile); | |
return Services.scriptloader.loadSubScript(fileURL + "?" + new Date().getTime(), sandbox, "UTF-8"); | |
} catch (e) { | |
Cu.reportError(e); | |
} | |
}; | |
sandbox._css = []; | |
aiueo.forEach(function({ current, submenu }){ | |
sandbox["_" + current] = []; | |
sandbox[current] = function(itemObj) { | |
ps(itemObj, sandbox["_" + current]); | |
} | |
sandbox[submenu] = function(menuObj) { | |
menuObj._items = [] | |
sandbox["_" + current].push(menuObj); | |
return function(itemObj) { | |
ps(itemObj, menuObj._items); | |
} | |
} | |
}, this); | |
function ps(item, array) { | |
("join" in item && "unshift" in item) ? | |
[].push.apply(array, item) : | |
array.push(item); | |
} | |
try { | |
Cu.evalInSandbox("function css(code){ this._css.push(code+'') };\n" + data, sandbox, "1.8"); | |
} catch (e) { | |
return this.log(e); | |
} | |
if (this.style2 && this.style2.parentNode) | |
this.style2.parentNode.removeChild(this.style2); | |
if (sandbox._css.length) | |
this.style2 = addStyle(sandbox._css.join("\n")); | |
this.removeMenuitem(); | |
aiueo.forEach(function({ current, submenu, insertId }){ | |
if (!sandbox["_" + current] || sandbox["_" + current].length == 0) return; | |
let insertPoint = $(insertId); | |
this.createMenuitem(sandbox["_" + current], insertPoint); | |
}, this); | |
if (isAlert) this.alert(U("設定を再読み込みしました")); | |
}, | |
newMenu: function(menuObj) { | |
var menu = document.createElement("menu"); | |
var popup = menu.appendChild(document.createElement("menupopup")); | |
for (let [key, val] in Iterator(menuObj)) { | |
if (key === "_items") continue; | |
if (typeof val == "function") | |
menuObj[key] = val = "(" + val.toSource() + ").call(this, event);" | |
menu.setAttribute(key, val); | |
} | |
let cls = menu.classList; | |
cls.add("addMenu"); | |
cls.add("menu-iconic"); | |
// 表示 / 非表示の設定 | |
if (menuObj.condition) | |
this.setCondition(menu, menuObj.condition); | |
menuObj._items.forEach(function(obj) { | |
popup.appendChild(this.newMenuitem(obj)); | |
}, this); | |
// menu に label が無い場合、最初の menuitem の label 等を持ってくる | |
// menu 部分をクリックで実行できるようにする(splitmenu みたいな感じ) | |
if (!menu.hasAttribute('label')) { | |
let firstItem = menu.querySelector('menuitem'); | |
if (firstItem) { | |
let command = firstItem.getAttribute('command'); | |
if (command) | |
firstItem = document.getElementById(command) || firstItem; | |
['label','accesskey','image','icon'].forEach(function(n){ | |
if (!menu.hasAttribute(n) && firstItem.hasAttribute(n)) | |
menu.setAttribute(n, firstItem.getAttribute(n)); | |
}, this); | |
menu.setAttribute('onclick', <![CDATA[ | |
if (event.target != event.currentTarget) return; | |
var firstItem = event.currentTarget.querySelector('menuitem'); | |
if (!firstItem) return; | |
if (event.button === 1) { | |
checkForMiddleClick(firstItem, event); | |
} else { | |
firstItem.doCommand(); | |
closeMenus(event.currentTarget); | |
} | |
]]>.toString()); | |
} | |
} | |
return menu; | |
}, | |
newMenuitem: function(obj) { | |
var menuitem; | |
// label == separator か必要なプロパティが足りない場合は区切りとみなす | |
if (obj.label === "separator" || | |
(!obj.label && !obj.text && !obj.keyword && !obj.url && !obj.oncommand && !obj.command)) { | |
menuitem = document.createElement("menuseparator"); | |
} else if (obj.oncommand || obj.command) { | |
let org = obj.command ? document.getElementById(obj.command) : null; | |
if (org && org.localName === "menuseparator") { | |
menuitem = document.createElement("menuseparator"); | |
} else { | |
menuitem = document.createElement("menuitem"); | |
if (obj.command) | |
menuitem.setAttribute("command", obj.command); | |
if (!obj.label) | |
obj.label = obj.command || obj.oncommand; | |
} | |
} else { | |
menuitem = document.createElement("menuitem"); | |
// property fix | |
if (!obj.label) | |
obj.label = obj.exec || obj.keyword || obj.url || obj.text; | |
if (obj.keyword && !obj.text) { | |
let index = obj.keyword.search(/\s+/); | |
if (index > 0) { | |
obj.text = obj.keyword.substr(index).trim(); | |
obj.keyword = obj.keyword.substr(0, index); | |
} | |
} | |
if (obj.where && /\b(tab|tabshifted|window|current)\b/i.test(obj.where)) | |
obj.where = RegExp.$1.toLowerCase(); | |
if (obj.where && !("acceltext" in obj)) | |
obj.acceltext = obj.where; | |
if (!obj.condition && (obj.url || obj.text)) { | |
// 表示 / 非表示の自動設定 | |
let condition = ""; | |
if (this.rSEL.test(obj.url || obj.text)) condition += " select"; | |
if (this.rLINK.test(obj.url || obj.text)) condition += " link"; | |
if (this.rEMAIL.test(obj.url || obj.text)) condition += " mailto"; | |
if (this.rIMAGE.test(obj.url || obj.text)) condition += " image"; | |
if (this.rMEDIA.test(obj.url || obj.text)) condition += " media"; | |
if (condition) | |
obj.condition = condition; | |
} | |
} | |
// obj を属性にする | |
for (let [key, val] in Iterator(obj)) { | |
if (key === "command") continue; | |
if (typeof val == "function") | |
obj[key] = val = "(" + val.toSource() + ").call(this, event);"; | |
menuitem.setAttribute(key, val); | |
} | |
var cls = menuitem.classList; | |
cls.add("addMenu"); | |
cls.add("menuitem-iconic"); | |
// 表示 / 非表示の設定 | |
if (obj.condition) | |
this.setCondition(menuitem, obj.condition); | |
// separator はここで終了 | |
if (menuitem.localName == "menuseparator") | |
return menuitem; | |
if (!obj.onclick) | |
menuitem.setAttribute("onclick", "checkForMiddleClick(this, event)"); | |
// oncommand, command はここで終了 | |
if (obj.oncommand || obj.command) | |
return menuitem; | |
menuitem.setAttribute("oncommand", "addMenu.onCommand(event);"); | |
// 可能ならばアイコンを付ける | |
this.setIcon(menuitem, obj); | |
return menuitem; | |
}, | |
createMenuitem: function(itemArray, insertPoint) { | |
var chldren = $A(insertPoint.parentNode.children); | |
for (let [, obj] in Iterator(itemArray)) { | |
if (!obj) continue; | |
let menuitem = obj._items ? this.newMenu(obj) : this.newMenuitem(obj); | |
let ins; | |
if (obj.insertAfter && (ins = $(obj.insertAfter))) { | |
ins.parentNode.insertBefore(menuitem, ins.nextSibling); | |
continue; | |
} | |
if (obj.insertBefore && (ins = $(obj.insertBefore))) { | |
ins.parentNode.insertBefore(menuitem, ins); | |
continue; | |
} | |
if (obj.position && parseInt(obj.position, 10) > 0) { | |
(ins = chldren[parseInt(obj.position, 10)-1]) ? | |
ins.parentNode.insertBefore(menuitem, ins): | |
insertPoint.parentNode.appendChild(menuitem); | |
continue; | |
} | |
insertPoint.parentNode.insertBefore(menuitem, insertPoint); | |
} | |
}, | |
removeMenuitem: function() { | |
$$('menu.addMenu').forEach(function(e) e.parentNode.removeChild(e) ); | |
$$('.addMenu').forEach(function(e) e.parentNode.removeChild(e) ); | |
}, | |
setIcon: function(menu, obj) { | |
if (menu.hasAttribute("src") || menu.hasAttribute("image") || menu.hasAttribute("icon")) | |
return; | |
if (obj.exec) { | |
var aFile = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsILocalFile); | |
try { | |
aFile.initWithPath(obj.exec); | |
} catch (e) { | |
return; | |
} | |
if (!aFile.exists() || !aFile.isExecutable()) { | |
menu.setAttribute("disabled", "true"); | |
} else { | |
let fileURL = Services.io.getProtocolHandler("file").QueryInterface(Ci.nsIFileProtocolHandler).getURLSpecFromFile(aFile); | |
menu.setAttribute("image", "moz-icon://" + fileURL + "?size=16"); | |
} | |
return; | |
} | |
if (obj.keyword) { | |
let engine = Services.search.getEngineByAlias(obj.keyword); | |
if (engine && engine.iconURI) { | |
menu.setAttribute("image", engine.iconURI.spec); | |
return; | |
} | |
} | |
let url = obj.keyword ? getShortcutOrURI(obj.keyword) : obj.url ? obj.url.replace(this.regexp, "") : ""; | |
if (!url) return; | |
let uri, iconURI; | |
try { | |
uri = Services.io.newURI(url, null, null); | |
} catch (e) { } | |
if (!uri) return; | |
menu.setAttribute("scheme", uri.scheme); | |
try { | |
iconURI = PlacesUtils.favicons.getFaviconForPage(uri); | |
} catch (e) { } | |
try { | |
// javascript: URI の host にアクセスするとエラー | |
menu.setAttribute("image", iconURI && iconURI.spec? | |
"moz-anno:favicon:" + iconURI.spec: | |
"moz-anno:favicon:" + uri.scheme + "://" + uri.host + "/favicon.ico"); | |
} catch (e) { } | |
}, | |
setCondition: function(menu, condition) { | |
if (/\bnormal\b/i.test(condition)) { | |
menu.setAttribute("condition", "normal"); | |
} else { | |
let match = condition.toLowerCase().match(/\b(?:no)?(?:select|link|mailto|image|canvas|media|input)\b/ig); | |
if (!match || !match[0]) | |
return; | |
match = match.filter(function(c,i,a) a.indexOf(c) === i); | |
menu.setAttribute("condition", match.join(" ")); | |
} | |
}, | |
convertText: function(text) { | |
var that = this; | |
var context = gContextMenu || { // とりあえずエラーにならないようにオブジェクトをでっち上げる | |
link: { href: "", host: "" }, | |
target: { alt: "", title: "" }, | |
__noSuchMethod__: function(id, args) "", | |
}; | |
var tab = document.popupNode && document.popupNode.localName == "tab" ? document.popupNode : null; | |
var win = tab ? tab.linkedBrowser.contentWindow : this.focusedWindow; | |
return text.replace(this.regexp, function(str){ | |
str = str.toUpperCase().replace("%LINK", "%RLINK"); | |
if (str.indexOf("_HTMLIFIED") >= 0) | |
return htmlEscape(convert(str.replace("_HTMLIFIED", ""))); | |
if (str.indexOf("_HTML") >= 0) | |
return htmlEscape(convert(str.replace("_HTML", ""))); | |
if (str.indexOf("_ENCODE") >= 0) | |
return encodeURIComponent(convert(str.replace("_ENCODE", ""))); | |
return convert(str); | |
}); | |
function convert(str) { | |
switch(str) { | |
case "%T" : return win.document.title; | |
case "%TITLE%" : return win.document.title; | |
case "%U" : return win.location.href; | |
case "%URL%" : return win.location.href; | |
case "%H" : return win.location.host; | |
case "%HOST%" : return win.location.host; | |
case "%S" : return that.getSelection(win) || ""; | |
case "%SEL%" : return that.getSelection(win) || ""; | |
case "%L" : return context.linkURL || ""; | |
case "%RLINK%" : return context.linkURL || ""; | |
case "%RLINK_HOST%" : return context.link.host || ""; | |
case "%RLINK_TEXT%" : return context.linkText() || ""; | |
case "%RLINK_OR_URL%": return context.linkURL || win.location.href; | |
case "%IMAGE_ALT%" : return context.target.alt || ""; | |
case "%IMAGE_TITLE%" : return context.target.title || ""; | |
case "%I" : return context.imageURL || ""; | |
case "%IMAGE_URL%" : return context.imageURL || ""; | |
case "%M" : return context.mediaURL || ""; | |
case "%MEDIA_URL%" : return context.mediaURL || ""; | |
case "%P" : return readFromClipboard() || ""; | |
case "%CLIPBOARD%" : return readFromClipboard() || ""; | |
case "%FAVICON%" : return gBrowser.getIcon(tab ? tab : null) || ""; | |
case "%EMAIL%" : return getEmailAddress() || ""; | |
case "%EOL%" : return "\r\n"; | |
} | |
return str; | |
} | |
function htmlEscape(s) { | |
return (s+"").replace(/&/g, "&").replace(/>/g, ">").replace(/</g, "<").replace(/\"/g, """).replace(/\'/g, "'"); | |
}; | |
function getEmailAddress() { | |
var url = context.linkURL; | |
if (!url || !/^mailto:([^?]+).*/i.test(url)) return ""; | |
var addresses = RegExp.$1; | |
try { | |
var characterSet = context.target.ownerDocument.characterSet; | |
const textToSubURI = Cc['@mozilla.org/intl/texttosuburi;1'].getService(Ci.nsITextToSubURI); | |
addresses = textToSubURI.unEscapeURIForUI(characterSet, addresses); | |
} catch (ex) { | |
} | |
return addresses; | |
} | |
}, | |
getSelection: function(win) { | |
win || (win = this.focusedWindow); | |
return this.getRangeAll(win).join(" ") || this.getInputSelection(win.document.activeElement); | |
}, | |
getRangeAll: function(win) { | |
win || (win = this.focusedWindow); | |
var sel = win.getSelection(); | |
var res = []; | |
for (var i = 0; i < sel.rangeCount; i++) { | |
res.push(sel.getRangeAt(i)); | |
}; | |
return res; | |
}, | |
getInputSelection: function(elem) { | |
if (elem instanceof HTMLTextAreaElement || elem instanceof HTMLInputElement && elem.mozIsTextField(false)) | |
return elem.value.substring(elem.selectionStart, elem.selectionEnd); | |
return ""; | |
}, | |
edit: function(aFile) { | |
if (!aFile || !aFile.exists() || !aFile.isFile()) return; | |
var editor = Services.prefs.getCharPref("view_source.editor.path"); | |
if (!editor) return this.log(U("エディタのパスが未設定です。\n view_source.editor.path を設定してください")); | |
try { | |
var UI = Cc["@mozilla.org/intl/scriptableunicodeconverter"].createInstance(Ci.nsIScriptableUnicodeConverter); | |
UI.charset = window.navigator.platform.toLowerCase().indexOf("win") >= 0? "Shift_JIS": "UTF-8"; | |
var path = UI.ConvertFromUnicode(aFile.path); | |
this.exec(editor, path); | |
} catch (e) {} | |
}, | |
copy: function(aText) { | |
Cc["@mozilla.org/widget/clipboardhelper;1"].getService(Ci.nsIClipboardHelper).copyString(aText); | |
XULBrowserWindow.statusTextField.label = "Copy: " + aText; | |
}, | |
alert: function(aString, aTitle) { | |
Cc['@mozilla.org/alerts-service;1'].getService(Ci.nsIAlertsService) | |
.showAlertNotification("", aTitle||"addMenu" , aString, false, "", null); | |
}, | |
$$: function(exp, context, aPartly) { | |
context || (context = this.focusedWindow.document); | |
var doc = context.ownerDocument || context; | |
var elements = $$(exp, doc); | |
if (arguments.length <= 2) | |
return elements; | |
var sel = doc.defaultView.getSelection(); | |
return elements.filter(function(q) sel.containsNode(q, aPartly)); | |
}, | |
log: log, | |
}; | |
window.addMenu.init(); | |
function $(id) { return document.getElementById(id); } | |
function $$(exp, doc) { return Array.prototype.slice.call((doc || document).querySelectorAll(exp)); } | |
function $A(args) { return Array.prototype.slice.call(args); } | |
function log() { Application.console.log(Array.slice(arguments)); } | |
function U(text) 1 < 'あ'.length ? decodeURIComponent(escape(text)) : text; | |
function $E(xml, doc) { | |
doc = doc || document; | |
xml = <root xmlns={doc.documentElement.namespaceURI}/>.appendChild(xml); | |
var settings = XML.settings(); | |
XML.prettyPrinting = false; | |
var root = new DOMParser().parseFromString(xml.toXMLString(), 'application/xml').documentElement; | |
XML.setSettings(settings); | |
doc.adoptNode(root); | |
var range = doc.createRange(); | |
range.selectNodeContents(root); | |
var frag = range.extractContents(); | |
range.detach(); | |
return frag.childNodes.length < 2 ? frag.firstChild : frag; | |
} | |
function addStyle(css) { | |
var pi = document.createProcessingInstruction( | |
'xml-stylesheet', | |
'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(css) + '"' | |
); | |
return document.insertBefore(pi, document.documentElement); | |
} | |
function loadText(aFile) { | |
var fstream = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(Ci.nsIFileInputStream); | |
var sstream = Cc["@mozilla.org/scriptableinputstream;1"].createInstance(Ci.nsIScriptableInputStream); | |
fstream.init(aFile, -1, 0, 0); | |
sstream.init(fstream); | |
var data = sstream.read(sstream.available()); | |
try { data = decodeURIComponent(escape(data)); } catch(e) {} | |
sstream.close(); | |
fstream.close(); | |
return data; | |
} | |
})(<![CDATA[ | |
#contentAreaContextMenu:not([addMenu~="select"]) .addMenu[condition~="select"], | |
#contentAreaContextMenu:not([addMenu~="link"]) .addMenu[condition~="link"], | |
#contentAreaContextMenu:not([addMenu~="mailto"]) .addMenu[condition~="mailto"], | |
#contentAreaContextMenu:not([addMenu~="image"]) .addMenu[condition~="image"], | |
#contentAreaContextMenu:not([addMenu~="canvas"]) .addMenu[condition~="canvas"], | |
#contentAreaContextMenu:not([addMenu~="media"]) .addMenu[condition~="media"], | |
#contentAreaContextMenu:not([addMenu~="input"]) .addMenu[condition~="input"], | |
#contentAreaContextMenu[addMenu~="select"] .addMenu[condition~="noselect"], | |
#contentAreaContextMenu[addMenu~="link"] .addMenu[condition~="nolink"], | |
#contentAreaContextMenu[addMenu~="mailto"] .addMenu[condition~="nomailto"], | |
#contentAreaContextMenu[addMenu~="image"] .addMenu[condition~="noimage"], | |
#contentAreaContextMenu[addMenu~="canvas"] .addMenu[condition~="nocanvas"], | |
#contentAreaContextMenu[addMenu~="media"] .addMenu[condition~="nomedia"], | |
#contentAreaContextMenu[addMenu~="input"] .addMenu[condition~="noinput"], | |
#contentAreaContextMenu:not([addMenu=""]) .addMenu[condition~="normal"] | |
{ display: none; } | |
.addMenu-insert-point | |
{ display: none !important; } | |
.addMenu[url] { | |
list-style-image: url("chrome://mozapps/skin/places/defaultFavicon.png"); | |
} | |
.addMenu.exec, | |
.addMenu[exec] { | |
list-style-image: url("chrome://browser/skin/aboutSessionRestore-window-icon.png"); | |
} | |
.addMenu.copy, | |
menuitem.addMenu[text]:not([url]):not([keyword]):not([exec]) | |
{ | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0pt, 32px, 16px, 16px); | |
} | |
.addMenu.checkbox .menu-iconic-icon { | |
-moz-appearance: checkbox; | |
} | |
.addMenu > .menu-iconic-left { | |
-moz-appearance: menuimage; | |
} | |
.addMenu:-moz-any([icon="back"], [command="context-back"], [command^="Browser:Back"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 18px, 18px, 0); | |
} | |
.addMenu:-moz-any([icon="forward"], [command="context-forward"], [command^="Browser:Forward"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 36px, 18px, 18px); | |
} | |
.addMenu:-moz-any([icon="stop"], [command="context-stop"], [command="Browser:Stop"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 54px, 18px, 36px); | |
} | |
.addMenu:-moz-any([icon="reload"], [command="context-reload"], [command^="Browser:Reload"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 72px, 18px, 54px); | |
} | |
.addMenu:-moz-any([icon="home"], [command="Browser:Home"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 90px, 18px, 72px); | |
} | |
.addMenu:-moz-any([icon="download"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 108px, 18px, 90px); | |
} | |
.addMenu:-moz-any([icon="history"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 126px, 18px, 108px); | |
} | |
.addMenu:-moz-any([icon="bookmark"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 144px, 18px, 126px); | |
} | |
.addMenu:-moz-any([icon="print"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 162px, 18px, 144px); | |
} | |
.addMenu:-moz-any([icon="newtab"], [command="context-openlinkintab"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 180px, 18px, 162px); | |
} | |
.addMenu:-moz-any([icon="newwindow"], [command="context-openlink"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 198px, 18px, 180px); | |
} | |
.addMenu:-moz-any([icon="copy"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 234px, 18px, 216px); | |
} | |
.addMenu:-moz-any([icon="paste"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 252px, 18px, 234px); | |
} | |
.addMenu:-moz-any([icon="fullscreen"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 270px, 18px, 252px); | |
} | |
.addMenu:-moz-any([icon="zoomout"], [icon="plus"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 288px, 18px, 270px); | |
} | |
.addMenu:-moz-any([icon="zoomin"], [icon="minus"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 306px, 18px, 288px); | |
} | |
.addMenu:-moz-any([icon="sync"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 324px, 18px, 306px); | |
} | |
.addMenu:-moz-any([icon="feed"]) { | |
list-style-image: url("chrome://browser/skin/Toolbar.png"); | |
-moz-image-region: rect(0, 342px, 18px, 324px); | |
} | |
.addMenu:-moz-any([icon="star"], [command^="context-bookmark"]) { | |
list-style-image: url("chrome://browser/skin/places/editBookmark.png"); | |
-moz-image-region: rect(0px, 16px, 16px, 0px); | |
} | |
.addMenu:-moz-any([icon="feed2"]) { | |
list-style-image: url("chrome://browser/skin/feeds/feedIcon16.png"); | |
} | |
.addMenu:-moz-any([icon="search"], [command="context-searchselect"]) { | |
list-style-image: url("chrome://global/skin/icons/Search-glass.png"); | |
-moz-image-region: rect(0px, 16px, 16px, 0px); | |
} | |
.addMenu:-moz-any([icon="starbutton"]) { | |
list-style-image: none; | |
} | |
.addMenu:-moz-any([icon="starbutton"]) > hbox > .menu-iconic-icon { | |
background: transparent no-repeat center center -moz-element(#star-button); | |
} | |
.addMenu:-moz-any([icon="close"]) { | |
list-style-image: url("chrome://global/skin/icons/close.png"); | |
-moz-image-region: rect(0, 16px, 16px, 0); | |
} | |
.addMenu:-moz-any([icon="close"]):hover { | |
-moz-image-region: rect(0, 32px, 16px, 16px); | |
} | |
.addMenu:-moz-any([icon="close"]):hover:active { | |
-moz-image-region: rect(0, 48px, 16px, 32px); | |
} | |
.addMenu:-moz-any([icon="newtab2"]) { | |
list-style-image: url("chrome://browser/skin/tabbrowser/newtab.png"); | |
} | |
.addMenu:-moz-any([icon="firefox"]) { | |
list-style-image: url("chrome://branding/content/icon16.png"); | |
} | |
.addMenu:-moz-any([icon="url"], [command="context-openlinkincurrent"]) { | |
list-style-image: url("chrome://mozapps/skin/places/defaultFavicon.png"); | |
} | |
.addMenu:-moz-any([icon="exec"]) { | |
list-style-image: url("chrome://browser/skin/aboutSessionRestore-window-icon.png"); | |
} | |
.addMenu:-moz-any([icon="checkbox"]) .menu-iconic-icon { | |
-moz-appearance: checkbox; | |
display: -moz-box; | |
} | |
.addMenu:-moz-any([icon="cut"], [command="context-cut"]) { | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0 16px 16px 0); | |
} | |
.addMenu:-moz-any([icon="copy2"], [command^="context-copy"]) { | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0, 32px, 16px, 16px); | |
} | |
.addMenu:-moz-any([icon="paste2"], [command="context-paste"]) { | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0 48px 16px 32px); | |
} | |
.addMenu:-moz-any([icon="print2"]) { | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0 64px 16px 48px); | |
} | |
.addMenu:-moz-any([icon="quit"]) { | |
list-style-image: url("chrome://browser/skin/appmenu-icons.png"); | |
-moz-image-region: rect(0 80px 16px 64px); | |
} | |
.addMenu:-moz-any([icon="privacy"], [command="Tools:PrivateBrowsing"]) { | |
list-style-image: url("chrome://browser/skin/Privacy-16.png"); | |
} | |
.addMenu:-moz-any([icon="addons"], [icon="addon"]) { | |
list-style-image: url("chrome://mozapps/skin/extensions/extensionGeneric-16.png"); | |
} | |
.addMenu:-moz-any([icon="folder"]) { | |
list-style-image: url("chrome://global/skin/icons/folder-item.png"); | |
-moz-image-region: rect(0px, 32px, 16px, 16px); | |
} | |
.addMenu:-moz-any([icon="livemark"]) { | |
list-style-image: url("chrome://browser/skin/livemark-folder.png"); | |
} | |
.addMenu:-moz-any([icon="query"]) { | |
list-style-image: url("chrome://browser/skin/places/query.png"); | |
} | |
.addMenu:-moz-any([icon="calendar"]) { | |
list-style-image: url("chrome://browser/skin/places/calendar.png"); | |
} | |
.addMenu:-moz-any([icon="menuback"]) { | |
list-style-image: url("chrome://browser/skin/menu-back.png"); | |
} | |
.addMenu:-moz-any([icon="menuforward"]) { | |
list-style-image: url("chrome://browser/skin/menu-forward.png"); | |
} | |
.addMenu:-moz-any([icon="tabview"]) { | |
list-style-image: url("chrome://browser/skin/tabview/tabview.png"); | |
-moz-image-region: rect(0, 90px, 18px, 72px); | |
} | |
.addMenu:-moz-any([icon="minimize"]) { | |
list-style-image: url("chrome://global/skin/icons/windowControls.png"); | |
-moz-image-region: rect(0, 16px, 16px, 0); | |
} | |
.addMenu:-moz-any([icon="minimize"]):hover { | |
-moz-image-region: rect(16px, 16px, 32px, 0); | |
} | |
.addMenu:-moz-any([icon="minimize"]):hover:active { | |
-moz-image-region: rect(32px, 16px, 48px, 0); | |
} | |
.addMenu:-moz-any([icon="restore"]) { | |
list-style-image: url("chrome://global/skin/icons/windowControls.png"); | |
-moz-image-region: rect(0, 32px, 16px, 16px); | |
} | |
.addMenu:-moz-any([icon="restore"]):hover { | |
-moz-image-region: rect(16px, 32px, 32px, 16px); | |
} | |
.addMenu:-moz-any([icon="restore"]):hover:active { | |
-moz-image-region: rect(32px, 32px, 48px, 16px); | |
} | |
.addMenu:-moz-any([icon="quit2"]) { | |
list-style-image: url("chrome://global/skin/icons/windowControls.png"); | |
-moz-image-region: rect(0, 48px, 16px, 32px); | |
} | |
.addMenu:-moz-any([icon="quit2"]):hover { | |
-moz-image-region: rect(16px, 48px, 32px, 32px); | |
} | |
.addMenu:-moz-any([icon="quit2"]):hover:active { | |
-moz-image-region: rect(32px, 48px, 48px, 32px); | |
} | |
.addMenu:-moz-any([icon="jpg"], [icon="jpeg"]) { | |
list-style-image: url("moz-icon://.jpg?size=16"); | |
} | |
.addMenu:-moz-any([icon="png"]) { | |
list-style-image: url("moz-icon://.png?size=16"); | |
} | |
.addMenu:-moz-any([icon="gif"]) { | |
list-style-image: url("moz-icon://.gif?size=16"); | |
} | |
.addMenu:-moz-any([icon="js"], [icon="javascript"]) { | |
list-style-image: url("moz-icon://.js?size=16"); | |
} | |
/* famfamfam */ | |
.addMenu:-moz-any([icon="save"], [command^="context-save"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="copy3"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="send"], [command^="context-send"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="edit"], [command="context-viewsource"], [command="context-viewpartialsource-selection"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="info"], [command="context-viewinfo"], [command="context-viewimageinfo"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="blank"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="application"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="image"], [command="context-viewimage"]) { | |
list-style-image: url(""); | |
} | |
.addMenu:-moz-any([icon="images"], [command="context-copyimage-contents"]) { | |
list-style-image: url(""); | |
} | |
]]>.toString()); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment