├── .gitignore ├── AboutConfigMultipleSelection.uc.js ├── AddBookmarkHere.uc.js ├── Bookmarkmenuplus.uc.xul ├── CustomAppMenu.uc.js ├── DefaultFullZoomLevel.uc.js ├── ExportHTMLFolder.uc.xul ├── GoogleSearchIcons.uc.js ├── README.md ├── SITEINFO_Writer.uc.js ├── SidebarModfix.uc.js ├── SimpleSessionManagerMod.uc.js ├── SnapLinksMod.uc.xul ├── UserAgentChangeModLite.uc.js ├── UserScriptLoader.uc.js ├── WordHighlightToolbar.uc.js ├── _uAutoPagerize.js ├── blur2revertMod.uc.js ├── bookmarkButtonLite.uc.xul ├── bookmarkButtonPlus.uc.xul ├── contextHistoryMenu.uc.xul ├── contextHistoryMenuLite.uc.xul ├── countryflags.js ├── externalAppButtonM.uc.js ├── externalAppButtonMEx.uc.js ├── externalFuncButtonM.uc.js ├── externalFuncMenu.uc.js ├── qrCreator.uc.js ├── qrCreatorOnline.uc.js ├── qrReader.uc.js ├── qrReaderOnline.uc.js ├── quickProxyModokiMod.uc.js ├── sendToGmail.uc.js ├── showLocationModEx.uc.js ├── showLocationModSpecial.uc.js ├── sideBookmarkBarMod.uc.js ├── uAutoPagerizeMod.uc.js └── urlfilterfast.uc.js /.gitignore: -------------------------------------------------------------------------------- 1 | # Build and Release Folders 2 | bin/ 3 | bin-debug/ 4 | bin-release/ 5 | 6 | # Other files and folders 7 | .settings/ 8 | 9 | # Project files, i.e. `.project`, `.actionScriptProperties` and `.flexProperties` 10 | # should NOT be excluded as they contain compiler settings and other important 11 | # information for Eclipse / Flash Builder. -------------------------------------------------------------------------------- /AboutConfigMultipleSelection.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name About:config Multiple Selection 3 | // @version 0.3 4 | // @description Allow multiple selection on about:config and more 5 | // @namespace https://github.com/nightson/ 6 | // @author NightsoN 7 | // @include chrome://browser/content/browser.xul 8 | // @note 0.3 add "Copy as Function" to the contextmenu. Now you can copy config entries in user_pref(xxx, xxx); format to be used in user.js 9 | // ==/UserScript== 10 | window.addEventListener('DOMContentLoaded', function (event) { 11 | var doc = event.target, 12 | win; 13 | if (!doc || !doc.location.href.startsWith('about:config')) return; 14 | doc.getElementById('configTree').setAttribute('seltype', 'multiple'); 15 | win = doc.defaultView; 16 | 17 | var contextMenu = doc.getElementById('configContext'), 18 | menuitem = contextMenu.insertBefore(doc.createElement('menuitem'), doc.getElementById('copyValue').nextSibling); 19 | menuitem.id = 'copyAsFunction'; 20 | menuitem.setAttribute('label', 'Copy as Function'); 21 | menuitem.setAttribute('accesskey', 'F'); 22 | menuitem.setAttribute('oncommand', 'copyAsFunction();'); 23 | 24 | win.getSelected = function () { 25 | var arr = [], 26 | i = 0, 27 | k = 0, 28 | j = win.view.selection.getRangeCount(), 29 | start = {}, 30 | end = {}; 31 | for (i; i < j; i++) { 32 | win.view.selection.getRangeAt(i, start, end); 33 | for (k = start.value; k <= end.value; k++) { 34 | arr.push(win.gPrefView[k]); 35 | } 36 | } 37 | return arr; 38 | } 39 | 40 | win.ResetSelected = function () { 41 | win.getSelected().forEach(function (i) { 42 | win.gPrefBranch.clearUserPref(i.prefCol); 43 | }) 44 | } 45 | 46 | win.copyPref = function () { 47 | var arr = []; 48 | win.getSelected().forEach(function (i) { 49 | arr.push(i.prefCol + ';' + i.valueCol); 50 | }); 51 | win.gClipboardHelper.copyString(arr.join('\n'), document); 52 | } 53 | 54 | win.copyName = function () { 55 | var arr = []; 56 | win.getSelected().forEach(function (i) { 57 | arr.push(i.prefCol); 58 | }); 59 | win.gClipboardHelper.copyString(arr.join('\n'), document); 60 | } 61 | 62 | win.copyValue = function () { 63 | var arr = []; 64 | win.getSelected().forEach(function (i) { 65 | arr.push(i.valueCol); 66 | }); 67 | win.gClipboardHelper.copyString(arr.join('\n'), document); 68 | } 69 | 70 | win.copyAsFunction = function () { 71 | var arr = []; 72 | win.getSelected().forEach(function (i) { 73 | if (i.typeCol === 32) { 74 | arr.push('user_pref("' + i.prefCol + '", "' + i.valueCol + '");'); 75 | } else { 76 | arr.push('user_pref("' + i.prefCol + '", ' + i.valueCol + ');'); 77 | } 78 | }); 79 | win.gClipboardHelper.copyString(arr.join('\n'), document); 80 | } 81 | 82 | }, true); 83 | -------------------------------------------------------------------------------- /AddBookmarkHere.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name Add Bookmark Here 3 | // @namespace http://script.bitcp.com/addbookmarkhere 4 | // @description add "Add Bookmark Here" contextmenu in places menu 5 | // @author zbinlin 6 | // @homepage http://bitcp.com 7 | // @include main 8 | // @include chrome://browser/content/places/places.xul 9 | // @include chrome://browser/content/bookmarks/bookmarksPanel.xul 10 | // @version 0.0.1 11 | // @note modify by lastdream 2013: support sidebar addbookmarkhere 12 | // ==/UserScript== 13 | 14 | (function () { 15 | if (window.AddBookmarkHere) return; 16 | var AddBookmarkHere = { 17 | init: function () { 18 | var placesContext = document.getElementById("placesContext"); 19 | if (!placesContext) return; 20 | var self = this; 21 | window.addEventListener("unload", function (e) { 22 | window.removeEventListener("unload", arguments.callee, false); 23 | self.uninit(); 24 | }, false); 25 | this.addContextMenu(placesContext, document.getElementById("placesContext_new:bookmark")); 26 | /* 27 | var node = document.getElementById("placesContext_createBookmark"); 28 | if (!node) return; 29 | node.removeAttribute("forcehideselection"); 30 | node.setAttribute("selection", "any"); 31 | node.removeAttribute("command"); 32 | node.setAttribute("oncommand", "AddBookmarkHere.addBookmark(event);"); 33 | */ 34 | }, 35 | addContextMenu: function (parentNode, afterNode) { 36 | var menuitem = document.createElement("menuitem"); 37 | menuitem.id = "placesContext_add:bookmark"; 38 | menuitem.setAttribute("label", "\u6DFB\u52A0\u5230\u8FD9\u91CC"); 39 | menuitem.setAttribute("accesskey", "h"); 40 | menuitem.setAttribute("selection", "any"); 41 | menuitem.addEventListener("command", this.addBookmark, false); 42 | parentNode.insertBefore(menuitem, afterNode); 43 | }, 44 | addBookmark: function (e) { 45 | var popupNode = e.currentTarget.parentNode.triggerNode; 46 | if (!popupNode) return; 47 | var view = PlacesUIUtils.getViewForNode(popupNode); 48 | if (!view) return; 49 | var bookmarks = Cc["@mozilla.org/browser/nav-bookmarks-service;1"].getService(Ci.nsINavBookmarksService); 50 | var selectedNode = view.selectedNode; 51 | var iid, aid; 52 | if (selectedNode) { 53 | if (PlacesUtils.nodeIsFolder(selectedNode) && !PlacesUtils.nodeIsLivemarkContainer(selectedNode) && !PlacesUtils.isReadonlyFolder(selectedNode)) { 54 | iid = selectedNode.itemId; 55 | aid = e.shiftKey ? 0 : bookmarks.DEFAULT_INDEX; 56 | } else { 57 | iid = bookmarks.getFolderIdForItem(selectedNode.itemId); 58 | var id = bookmarks.getItemIndex(selectedNode.itemId); 59 | aid = e.shiftKey ? id : id + 1; 60 | } 61 | } else { 62 | iid = view.result.root.folderItemId; 63 | aid = e.shiftKey ? 0 : bookmarks.DEFAULT_INDEX; 64 | } 65 | //var doc = gBrowser.getBrowserForTab(gBrowser.mCurrentTab).contentDocument; 66 | var mainWindow = window.QueryInterface(Components.interfaces.nsIInterfaceRequestor) 67 | .getInterface(Components.interfaces.nsIWebNavigation) 68 | .QueryInterface(Components.interfaces.nsIDocShellTreeItem) 69 | .rootTreeItem 70 | .QueryInterface(Components.interfaces.nsIInterfaceRequestor) 71 | .getInterface(Components.interfaces.nsIDOMWindow); 72 | var doc = mainWindow.gBrowser.getBrowserForTab(mainWindow.gBrowser.mCurrentTab).contentDocument; 73 | var uri = Services.io.newURI(doc.location.toString(), null, null); 74 | var title = doc.title; 75 | bookmarks.insertBookmark(iid, uri, aid, title); 76 | }, 77 | uninit: function () { 78 | try { 79 | var menuitem = document.getElementById("placesContext_add:bookmark"); 80 | menuitem.removeEventListener("command", this.addBookmark, false); 81 | } catch (ex) { 82 | } 83 | } 84 | }; 85 | 86 | AddBookmarkHere.init(); 87 | window.AddBookmarkHere = AddBookmarkHere; 88 | 89 | })(); 90 | -------------------------------------------------------------------------------- /Bookmarkmenuplus.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 10 | 11 | 13 | %browserDTD; 14 | 15 | %placesDTD; 16 | ]> 17 | 18 | 19 | 20 | 61 | 62 | 206 | 207 | 208 | 212 | 213 | 214 | 215 | 216 | -------------------------------------------------------------------------------- /GoogleSearchIcons.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name GoogleSearchIcons.uc.js 3 | // @note 20130614: modify by lastdream2013 for uAutoPagerize 4 | // @include main 5 | // ==/UserScript== 6 | 7 | (function () { 8 | var googleicon = function (doc) { 9 | if (!doc || !doc.location) return; 10 | if (doc.location.href.match(/^https?:\/\/.*\.google\..*\/search?.*$/i)) { 11 | 12 | // Check if already loaded 原来要注释掉了,为配合uAutoPagerize翻页 13 | //if(doc.getElementById("googleicon")) return; 14 | 15 | // Create Google Icon 16 | var div = doc.createElement("div"); 17 | div.id = "googleicon"; 18 | doc.body.appendChild(div); 19 | 20 | var results = doc.evaluate("//h3[contains(@class,'r')]/a", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); 21 | for (var i = results.snapshotLength - 1; i >= 0; i--) { 22 | var result = results.snapshotItem(i); 23 | 24 | if (result.firstChild.className == "googleicon") //Check if already loaded 25 | break; 26 | // Find url 27 | var url = result.href.replace(/<\S[^><]*>/g, ""); 28 | if (/^http:\/\/.*\.google\..*\/url\?.*$/i.test(url)) 29 | url = url.match(/&q=(.*?)&/)[1]; 30 | var safe = (url.indexOf("https://") < 0); 31 | if (url.match("google.com/interstitial?")) 32 | url = "chrome://googleicon/content/malware.png"; 33 | else 34 | url = "http://" + url.split('/')[2] + "/favicon.ico"; 35 | 36 | // Add icon 37 | (function (url) { 38 | var img = new Image(); 39 | img.width = 16; 40 | img.src = ""; 41 | img.className = "googleicon"; 42 | img.setAttribute("style", "border:0; padding-top:2px; padding-right:4px; display:block; float:left;"); 43 | result.insertBefore(img, result.firstChild); 44 | if (safe) { 45 | var newimg = new Image(); 46 | newimg.src = url; 47 | newimg.addEventListener("load", function (doc) { 48 | if (!img) return; 49 | img.src = url; 50 | }, false); 51 | } 52 | })(url); 53 | 54 | } 55 | } 56 | }; 57 | 58 | // Bind Google Icon 59 | var delay = function (aEvent) { 60 | var doc = aEvent.originalTarget; 61 | setTimeout(function () { 62 | googleicon(doc); 63 | }, 1); 64 | }; 65 | var load = function () { 66 | gBrowser.addEventListener("DOMContentLoaded", delay, true); 67 | gBrowser.addEventListener('GM_AutoPagerizeNextPageLoaded', delay, true); 68 | }; 69 | window.addEventListener("pageshow", load, false); 70 | 71 | })(); 72 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | userChromeJS 2 | ========== 3 | 4 | 这里的userChromeJS是我个人所使用过程中所原创,以及修改/bug修复别人的userChromeJS的版本。转载时请尊重所有原作者以及出处 5 | 如果您选择使用这些userChromeJS,请自行负责,作者不会为意外数据损害等情况承担责任。 6 | 7 | 20130509更新: 8 | 修复使用WordHighlightToolbar.uc.js中出现的问题,google搜索"JavaScript yield"之后,自动高亮条出现后,再点击跳转到: 9 | https://developer.mozilla.org/en-US/docs/JavaScript/New_in_JavaScript/1.7 10 | 页面代码框中出现不应该的字段。 11 | 12 | 20130510更新: 13 | SimpleSessionManagerMod.uc.js 14 | 原作者为nightson, 0510mod版加强了一下功能: 15 | 1. 启动时可选自动恢复上一次会话或从弹出菜单中选择恢复(水平有限,先做了个简陋的弹出菜单会话列表……); 16 | 2. 如果启动时恢复的是退出的自动恢复会话,可选自动删除已恢复的会话; 17 | 3. 退出时可选自动保存会话或确认手动保存; 18 | 4. 可选自定义出现在火狐橙菜单下或以小图标方式出现(打开脚本可自定义,小图标位置可自定义) 19 | 5. 增加删除所有会话的菜单选项。 20 | 21 | -------------------------------------------------------------------------------- /SITEINFO_Writer.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name SITEINFO_Writer.uc.js 3 | // @description AutoPagerize の SITEINFO を書くためのスクリプト 4 | // @namespace http://d.hatena.ne.jp/Griever/ 5 | // @author Griever 6 | // @include main 7 | // @compatibility Firefox 5 - firefox 23a1 8 | // @charset UTF-8 9 | // @version 下書き1 10 | // @note 同步:增加下一页中文选择xpath By ywzhaiqi@gmail.com 11 | // @note fix compatibility for firefox 23a1 by lastdream2013 12 | // @note まだこれからつくり込む段階 13 | // @note ツールメニューから起動する 14 | // ==/UserScript== 15 | 16 | (function(css){ 17 | 18 | if (window.siteinfo_writer) { 19 | window.siteinfo_writer.destroy(); 20 | delete window.siteinfo_writer; 21 | } 22 | 23 | window.siteinfo_writer = { 24 | init: function() { 25 | this.style = addStyle(css); 26 | 27 | var overlay = '\ 28 | \ 30 | \ 31 | \ 96 | \ 97 | '; 98 | overlay = "data:application/vnd.mozilla.xul+xml;charset=utf-8," + encodeURI(overlay); 99 | window.userChrome_js.loadOverlay(overlay, window.siteinfo_writer); 100 | }, 101 | observe : function (aSubject, aTopic, aData) { 102 | if (aTopic == "xul-overlay-merged") 103 | { 104 | this.popup = $("mainPopupSet").appendChild($C("menupopup", 105 | { 106 | id : "sw-popup", 107 | class : "sw-add-element", 108 | } 109 | )); 110 | 111 | var menuitem = $C("menuitem", 112 | { 113 | class : "sw-add-element", 114 | label : "辅助查找翻页规则", 115 | oncommand : "siteinfo_writer.show();", 116 | } 117 | ); 118 | $("devToolsSeparator").parentNode.insertBefore(menuitem, $("devToolsSeparator")); 119 | 120 | setTimeout(function () 121 | { 122 | if (!window.uAutoPagerize) 123 | { 124 | $("sw-launch").hidden = true; 125 | return; 126 | }; 127 | 128 | let aupPopup = $("uAutoPagerize-popup"); 129 | if (aupPopup) 130 | { 131 | aupPopup.appendChild(document.createElement("menuseparator")).setAttribute("class", "sw-add-element"); 132 | aupPopup.appendChild(menuitem.cloneNode(false)); 133 | } 134 | }, 2000); 135 | 136 | this.container = $("sw-container"); 137 | this.url = $("sw-url"); 138 | this.siteName = $("sw-siteName"); 139 | this.nextLink = $("sw-nextLink"); 140 | this.pageElement = $("sw-pageElement"); 141 | this.insertBefore = $("sw-insertBefore"); 142 | 143 | this.nextLink.addEventListener("popupshowing", siteinfo_writer.pps, false); 144 | this.pageElement.addEventListener("popupshowing", siteinfo_writer.pps, false); 145 | this.insertBefore.addEventListener("popupshowing", siteinfo_writer.pps, false); 146 | } 147 | }, 148 | uninit : function () { 149 | }, 150 | 151 | pps: function(event) { 152 | event.currentTarget.removeEventListener(event.type, arguments.callee, false); 153 | var popup = event.originalTarget; 154 | var type = event.currentTarget.id.replace("sw-", ""); 155 | 156 | popup.appendChild($C("menuseparator",{})); 157 | popup.appendChild($C("menuitem", { 158 | label: '@class="xxx" → contains()', 159 | oncommand: "siteinfo_writer.class2contains('"+ type +"');", 160 | })); 161 | popup.appendChild($C("menuitem",{ 162 | label: 'contains() → @class="xxx"', 163 | oncommand: "siteinfo_writer.contains2class('"+ type +"');", 164 | })); 165 | }, 166 | destroy: function() { 167 | $A(document.getElementsByClassName("sw-add-element")).forEach(function(e){ 168 | e.parentNode.removeChild(e); 169 | }) 170 | this.style.parentNode.removeChild(this.style); 171 | }, 172 | setUrl: function() { 173 | this.url.value = "^" + content.location.href.replace(/[()\[\]{}|+.,^$?\\]/g, '\\$&'); 174 | this.url.className = ""; 175 | }, 176 | show: function() { 177 | this.setUrl(); 178 | this.siteName.value = content.document.title; 179 | this.nextLink.value = ""; 180 | this.pageElement.value = ""; 181 | this.insertBefore.value = ""; 182 | this.container.hidden = false; 183 | }, 184 | hide: function() { 185 | this.container.hidden = true; 186 | }, 187 | toJSON: function() { 188 | var json = "\t{\n"; 189 | json += "\t\tsiteName : '" + this.siteName.value + "',\n"; 190 | json += "\t\turl : '" + this.url.value.replace(/\\/g, "\\\\") + "',\n"; 191 | json += "\t\tnextLink : '" + this.nextLink.value + "',\n"; 192 | json += "\t\tpageElement : '" + this.pageElement.value + "',\n"; 193 | if ( this.insertBefore.value != '' ) {json += "\t\tinsertBefore: '" + this.insertBefore.value + "',\n";} 194 | json += "\t\texampleUrl : '" + content.location.href + "',\n"; 195 | json += "\t},"; 196 | var r=confirm("翻页规则(按OK键将其复制到剪贴板):"+'\n\n' + json); 197 | if(r==true){ 198 | try{ 199 | Components.classes["@mozilla.org/widget/clipboardhelper;1"].getService(Components.interfaces.nsIClipboardHelper).copyString(json); 200 | } 201 | catch(e){ 202 | alert(e); 203 | } 204 | } 205 | }, 206 | toSuperPreLoaderFormat: function() { 207 | var spdb = "\t{\n"; 208 | spdb += "\t\tsiteName : '" + this.siteName.value + "',\n"; 209 | spdb += "\t\turl : /" + this.url.value.replace(/\//g, "\\\/") + "/i,\n"; 210 | spdb += "\t\tsiteExample : '" + content.location.href + "',\n"; 211 | spdb += "\t\tnextLink : '" + this.nextLink.value + "',\n"; 212 | spdb += "\t\tautopager: {\n"; 213 | spdb += "\t\t\tuseiframe : true,\n"; 214 | spdb += "\t\t\tpageElement : '" + this.pageElement.value + "',\n"; 215 | if ( this.insertBefore.value != '' ) {spdb += "\t\t\tHT_insert : ['" + this.insertBefore.value + "', 1],\n";} 216 | spdb += "\t\t}\n"; 217 | spdb += "\t},"; 218 | var r=confirm("翻页规则(SuperPreLoader格式)(按OK键将其复制到剪贴板):"+'\n\n' + spdb); 219 | if(r==true){ 220 | try{ 221 | Components.classes["@mozilla.org/widget/clipboardhelper;1"].getService(Components.interfaces.nsIClipboardHelper).copyString(spdb); 222 | } 223 | catch(e){ 224 | alert(e); 225 | } 226 | } 227 | }, 228 | launch: function() { 229 | if (content.ap) return alert("已经运行"); 230 | 231 | var i = {}; 232 | ["url", "nextLink", "pageElement", "insertBefore"].forEach(function(type) { 233 | if (this[type].value) 234 | i[type] = this[type].value 235 | }, this); 236 | if (!i.url || !i.nextLink || !i.pageElement) 237 | return alert("指定的值无效"); 238 | 239 | let [index, info] = uAutoPagerize.getInfo([i], content); 240 | 241 | if (index === 0) { 242 | if (content.AutoPagerize && content.AutoPagerize.launchAutoPager) 243 | content.AutoPagerize.launchAutoPager([i]); 244 | else alert("翻页规则语法正确,但uAutoPagerize无法执行,可能uAutoPagerize被禁用或没有安装脚本"); 245 | } else { 246 | alert("翻页规则不匹配"); 247 | } 248 | }, 249 | inspect: function(aType) { 250 | if (this._inspect) 251 | this._inspect.uninit(); 252 | var self = this; 253 | this._inspect = new Inspector(content, function(xpathArray) { 254 | if (xpathArray.length) { 255 | let range = document.createRange(); 256 | range.selectNodeContents(self.popup); 257 | range.deleteContents(); 258 | range.detach(); 259 | for (let [i, x] in Iterator(xpathArray)) { 260 | let menuitem = document.createElement("menuitem"); 261 | menuitem.setAttribute("label", x); 262 | menuitem.setAttribute("oncommand", "siteinfo_writer['"+ aType +"'].value = this.getAttribute('label')"); 263 | self.popup.appendChild(menuitem); 264 | } 265 | self.popup.openPopup(self.container, "before_start"); 266 | } 267 | self._inspect = null; 268 | }); 269 | }, 270 | xpathTest: function(aType) { 271 | var textbox = this[aType] 272 | if (!textbox || !textbox.value) return; 273 | try { 274 | var elements = getElementsByXPath(textbox.value, content.document); 275 | } catch (e) { 276 | alert(e); 277 | } 278 | 279 | for (let [i, elem] in Iterator(elements)) { 280 | if (!("orgcss" in elem)) 281 | elem.orgcss = elem.style.cssText; 282 | elem.style.backgroundImage = "-moz-linear-gradient(magenta, plum)"; 283 | elem.style.outline = "1px solid magenta"; 284 | } 285 | 286 | var self = this; 287 | if (this.timer) { 288 | clearTimeout(this.timer); 289 | this.timer = null; 290 | } 291 | this.timer = setTimeout(function() { 292 | for (let [i, elem] in Iterator(elements)) { 293 | if ("orgcss" in elem) { 294 | elem.orgcss? 295 | elem.style.cssText = elem.orgcss: 296 | elem.removeAttribute("style"); 297 | delete elem.orgcss; 298 | } 299 | } 300 | }, 5000); 301 | }, 302 | urlTest: function() { 303 | var url = this.url.value; 304 | try { 305 | var regexp = new RegExp(this.url.value); 306 | if (regexp.test(content.location.href)) 307 | this.url.classList.remove("error"); 308 | else 309 | this.url.classList.add("error"); 310 | } catch (e) { 311 | this.url.classList.add("error"); 312 | } 313 | }, 314 | inputTimer: null, 315 | onInput: function(event) { 316 | if (this.inputTimer) { 317 | clearTimeout(this.inputTimer); 318 | } 319 | var self = this; 320 | this.inputTimer = setTimeout(function() { 321 | self.urlTest(); 322 | }, 100); 323 | }, 324 | class2contains: function(aType) { 325 | this[aType].value = this.splitClass(this[aType].value); 326 | }, 327 | contains2class: function(aType) { 328 | this[aType].value = this.normalClass(this[aType].value); 329 | }, 330 | splitClass: function(xpath) { 331 | return xpath.replace(/@class=\"(.+?)\"/g, function(str, cls) { 332 | cls = cls.replace(/\s+/g, " ").replace(/^\s+|\s+$/g, "").split(" "); 333 | for (var i = 0, l = cls.length; i < l; i++) { 334 | cls[i] = 'contains(concat(" ",normalize-space(@class)," "), " '+ cls[i] +' ")'; 335 | } 336 | return cls.join(" and "); 337 | }); 338 | }, 339 | normalClass: function(xpath) { 340 | let r = /(?:contains\(concat\(\" \"\,normalize\-space\(@class\)\,\" \"\)\, \" .+? \"\)(?: and )?)+/g; 341 | return xpath.replace(r, function(str) { 342 | let cls = str.split(' and ').map(function(c) c.replace(/.*\" (.*) \".*/i, '$1') ); 343 | return '@class="'+ cls.join(' ') +'"'; 344 | }); 345 | }, 346 | }; 347 | 348 | window.siteinfo_writer.init(); 349 | 350 | function Inspector(aWindow, aCallback) { 351 | this.win = aWindow; 352 | this.doc = this.win.document; 353 | this.callback = aCallback; 354 | this.init(); 355 | } 356 | Inspector.prototype = { 357 | init : function(){ 358 | this.doc.addEventListener('mousedown', this, true); 359 | this.doc.addEventListener('click', this, true); 360 | this.doc.addEventListener('mouseover', this, true); 361 | }, 362 | uninit : function(){ 363 | var self = this; 364 | this.lowlight(); 365 | this.doc.removeEventListener('mouseover', this, true); 366 | setTimeout(function(){ 367 | self.doc.removeEventListener('mousedown', self, true); 368 | self.doc.removeEventListener('click', self, true); 369 | }, 500); 370 | XULBrowserWindow.statusTextField.label = ""; 371 | }, 372 | handleEvent : function(event){ 373 | switch(event.type){ 374 | case 'mousedown': 375 | event.preventDefault(); 376 | event.stopPropagation(); 377 | this.uninit(); 378 | if (event.button == 0) 379 | this.callback(this.getXPath(this.target)); 380 | break; 381 | case 'mouseover': 382 | if (this.target) 383 | this.lowlight(); 384 | this.target = event.target; 385 | this.highlight(); 386 | break; 387 | case "click": 388 | event.preventDefault(); 389 | event.stopPropagation(); 390 | break; 391 | } 392 | }, 393 | highlight : function(){ 394 | if (this.target.mozMatchesSelector('html, body')) 395 | return; 396 | if (!("orgcss" in this.target)) 397 | this.target.orgcss = this.target.style.cssText; 398 | this.target.style.outline = "magenta 2px solid"; 399 | this.target.style.outlineOffset = "-1px"; 400 | XULBrowserWindow.statusTextField.label = this.getElementXPath(this.target, this.ATTR_FULL); 401 | }, 402 | lowlight : function() { 403 | if ("orgcss" in this.target) { 404 | this.target.orgcss? 405 | this.target.style.cssText = this.target.orgcss: 406 | this.target.removeAttribute("style"); 407 | delete this.target.orgcss; 408 | } 409 | }, 410 | ATTR_CLASSID: 0, 411 | ATTR_ID: 1, 412 | ATTR_CLASS: 2, 413 | ATTR_NOT_CLASSID: 3, 414 | ATTR_FULL: 4, 415 | TEXT: 5, 416 | NEXT_REG: /[下后][一]?[页张个篇章节步]/, 417 | NEXT_REG_A: /^[下后][一]?[页张个篇章节步]$/, 418 | getXPath: function(originalTarget) { 419 | var nodes = getElementsByXPath( 420 | 'ancestor-or-self::*[not(local-name()="html" or local-name()="HTML" or local-name()="body" or local-name()="BODY")]', originalTarget).reverse(); 421 | if (nodes.length == 0) return []; 422 | 423 | var current = nodes.shift(); 424 | var obj = {}; 425 | obj.localnames = current.localName; 426 | obj.self_classid = this.getElementXPath(current, this.ATTR_CLASSID); 427 | obj.self_id = this.getElementXPath(current, this.ATTR_ID); 428 | obj.self_class = this.getElementXPath(current, this.ATTR_CLASS); 429 | obj.self_attr = this.getElementXPath(current, this.ATTR_NOT_CLASSID); 430 | obj.self_full = this.getElementXPath(current, this.ATTR_FULL); 431 | obj.self_text = this.getElementXPath(current, this.TEXT); 432 | obj.ancestor_classid = obj.self_classid; 433 | obj.ancestor_id = obj.self_id; 434 | obj.ancestor_class = obj.self_class; 435 | obj.ancestor_attr = obj.self_attr; 436 | obj.ancestor_full = obj.self_full; 437 | obj.ancestor_text = obj.self_text; 438 | 439 | var hasId = current.getAttribute("id"); 440 | for (let [i, elem] in Iterator(nodes)) { 441 | obj.localnames = elem.localName + "/" + obj.localnames; 442 | 443 | if (!hasId) { 444 | hasId = elem.getAttribute("id"); 445 | obj.ancestor_classid = this.getElementXPath(elem, this.ATTR_CLASSID) + "/" + obj.ancestor_classid; 446 | obj.ancestor_id = this.getElementXPath(elem, this.ATTR_ID) + "/" + obj.ancestor_id; 447 | obj.ancestor_full = this.getElementXPath(elem, this.ATTR_FULL) + "/" + obj.ancestor_full; 448 | obj.ancestor_text = this.getElementXPath(elem, this.ATTR_ID) + "/" + obj.ancestor_text; 449 | } 450 | obj.ancestor_class = this.getElementXPath(elem, this.ATTR_NOT_CLASS) + "/" + obj.ancestor_class; 451 | obj.ancestor_attr = this.getElementXPath(elem, this.ATTR_NOT_CLASSID) + "/" + obj.ancestor_attr; 452 | } 453 | 454 | for (let [key, val] in Iterator(obj)) { 455 | if (val.substr(0, 4) !== 'id("') 456 | obj[key] = val = "//" + val; 457 | } 458 | var res = [x for each(x in obj)].filter(function(e, i, a) a.indexOf(e) === i).sort(function(a, b){ 459 | let aa = a.substr(0, 4) == 'id("'; 460 | let bb = b.substr(0, 4) == 'id("'; 461 | if ((aa && bb) || (!aa && !bb)) 462 | return b.length - a.length; 463 | return bb? 1 : -1; 464 | }); 465 | //var res = [[x, obj[x]] for(x in obj)].sort(function([a,], [b,]) a >= b); 466 | ////inspectObject([x for each(x in res)]); 467 | //res = [x for each([,x] in res)].filter(function(e, i, a) a.indexOf(e) === i); 468 | ////inspectObject(res); 469 | return res; 470 | }, 471 | getElementXPath: function(elem, constant) { 472 | if (!elem.getAttribute) 473 | return ""; 474 | 475 | if (this.ATTR_CLASSID == constant) { 476 | if (elem.getAttribute("id")) 477 | return 'id("'+ elem.getAttribute('id') +'")'; 478 | if (elem.getAttribute("class")) 479 | return elem.nodeName.toLowerCase() + '[@class="' + elem.getAttribute("class") + '"]'; 480 | return elem.nodeName.toLowerCase(); 481 | } 482 | if (this.ATTR_ID == constant) { 483 | if (elem.getAttribute("id")) 484 | return 'id("'+ elem.getAttribute('id') +'")'; 485 | return elem.nodeName.toLowerCase(); 486 | } 487 | if (this.ATTR_CLASS == constant) { 488 | if (elem.getAttribute("class")) 489 | return elem.nodeName.toLowerCase() + '[@class="' + elem.getAttribute("class") + '"]'; 490 | return elem.nodeName.toLowerCase(); 491 | } 492 | if(this.TEXT == constant){ 493 | var elemHtml = elem.textContent; 494 | if(elemHtml.length < 20 && elemHtml.match(this.NEXT_REG)){ 495 | if(elemHtml.match(this.NEXT_REG_A)){ 496 | return elem.nodeName.toLowerCase() + '[text()="' + elemHtml + '"]'; 497 | } 498 | return elem.nodeName.toLowerCase() + '[contains(text(), "' + elemHtml + '")]'; 499 | } 500 | return elem.nodeName.toLowerCase(); 501 | } 502 | 503 | var xpath = elem.nodeName.toLowerCase(); 504 | if (this.ATTR_FULL == constant) { 505 | let x = []; 506 | if (elem.hasAttribute("id")) 507 | x[x.length] = '@id="' + elem.getAttribute("id") + '"'; 508 | if (elem.hasAttribute("class")) 509 | x[x.length] = '@class="' + elem.getAttribute("class") + '"'; 510 | if (x.length) 511 | xpath += '['+ x.join(" and ") +']'; 512 | } 513 | 514 | /* 515 | var attrs = elem.attributes; 516 | var arr = []; 517 | for (var i = 0, len = attrs.length; i < len; i++) { 518 | var name = attrs[i].nodeName; 519 | if (name === "style" || name === "id" || name === "class") continue; 520 | var value = attrs[i].nodeValue; 521 | arr[arr.length] = '@' + name + '="' + value + '"'; 522 | }; 523 | */ 524 | var arr = [ 525 | "@"+ x.nodeName +'="'+ x.nodeValue +'"' 526 | for each(x in $A(elem.attributes)) 527 | if (!/^(?:id|class|style)$/i.test(x.nodeName)) 528 | ]; 529 | if (arr.length > 0) 530 | xpath += '[' + arr.join(" and ") + ']'; 531 | return xpath; 532 | }, 533 | }; 534 | 535 | 536 | 537 | function log(){ Application.console.log($A(arguments)); } 538 | function $(id) document.getElementById(id); 539 | function $A(arr) Array.slice(arr); 540 | function $C(name, attr) { 541 | var el = document.createElement(name); 542 | if (attr) Object.keys(attr).forEach(function(n) el.setAttribute(n, attr[n])); 543 | return el; 544 | } 545 | 546 | function addStyle(css) { 547 | var pi = document.createProcessingInstruction( 548 | 'xml-stylesheet', 549 | 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(css) + '"' 550 | ); 551 | return document.insertBefore(pi, document.documentElement); 552 | } 553 | 554 | function getElementsByXPath(xpath, node) { 555 | var nodesSnapshot = getXPathResult(xpath, node, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE); 556 | var data = []; 557 | for (var i = 0, l = nodesSnapshot.snapshotLength; i < l; i++) { 558 | data[i] = nodesSnapshot.snapshotItem(i); 559 | } 560 | return data; 561 | } 562 | 563 | function getFirstElementByXPath(xpath, node) { 564 | var result = getXPathResult(xpath, node, XPathResult.FIRST_ORDERED_NODE_TYPE); 565 | return result.singleNodeValue; 566 | } 567 | 568 | function getXPathResult(xpath, node, resultType) { 569 | var doc = node.ownerDocument || node 570 | var resolver = doc.createNSResolver(node.documentElement || node) 571 | // Use |node.lookupNamespaceURI('')| for Opera 9.5 572 | var defaultNS = node.lookupNamespaceURI(null) 573 | if (defaultNS) { 574 | const defaultPrefix = '__default__' 575 | xpath = addDefaultPrefix(xpath, defaultPrefix) 576 | var defaultResolver = resolver 577 | resolver = function (prefix) { 578 | return (prefix == defaultPrefix) 579 | ? defaultNS : defaultResolver.lookupNamespaceURI(prefix) 580 | } 581 | } 582 | return doc.evaluate(xpath, node, resolver, resultType, null) 583 | } 584 | 585 | function addDefaultPrefix(xpath, prefix) { 586 | const tokenPattern = /([A-Za-z_\u00c0-\ufffd][\w\-.\u00b7-\ufffd]*|\*)\s*(::?|\()?|(".*?"|'.*?'|\d+(?:\.\d*)?|\.(?:\.|\d+)?|[\)\]])|(\/\/?|!=|[<>]=?|[\(\[|,=+-])|([@$])/g 587 | const TERM = 1, OPERATOR = 2, MODIFIER = 3 588 | var tokenType = OPERATOR 589 | prefix += ':' 590 | function replacer(token, identifier, suffix, term, operator, modifier) { 591 | if (suffix) { 592 | tokenType = 593 | (suffix == ':' || (suffix == '::' && 594 | (identifier == 'attribute' || identifier == 'namespace'))) 595 | ? MODIFIER : OPERATOR 596 | } 597 | else if (identifier) { 598 | if (tokenType == OPERATOR && identifier != '*') { 599 | token = prefix + token 600 | } 601 | tokenType = (tokenType == TERM) ? OPERATOR : TERM 602 | } 603 | else { 604 | tokenType = term ? TERM : operator ? OPERATOR : MODIFIER 605 | } 606 | return token 607 | } 608 | return xpath.replace(tokenPattern, replacer) 609 | }; 610 | 611 | 612 | })('\ 613 | \ 614 | #sw-grid row > label {\ 615 | margin: 1px !important;\ 616 | }\ 617 | #sw-grid row > textbox {\ 618 | -moz-appearance: none !important;\ 619 | margin: 1px !important;\ 620 | padding: 1px !important;\ 621 | border-width: 1px !important;\ 622 | }\ 623 | #sw-grid row > textbox[multiline="true"] {\ 624 | height: 4em;\ 625 | }\ 626 | \ 627 | #sw-grid .inspect {\ 628 | list-style-image: url("chrome://global/skin/icons/Search-glass.png");\ 629 | -moz-image-region: rect(0px, 16px, 16px, 0px);\ 630 | }\ 631 | \ 632 | #sw-grid .check {\ 633 | list-style-image: url("chrome://global/skin/icons/find.png");\ 634 | -moz-image-region: rect(0px, 48px, 16px, 32px);\ 635 | }\ 636 | \ 637 | #sw-url.error {\ 638 | outline: 1px solid red !important;\ 639 | }\ 640 | #sw-popup > * {\ 641 | max-width: none !important;\ 642 | }\ 643 | '); 644 | -------------------------------------------------------------------------------- /SnapLinksMod.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 3 | 17 | 25 | 40 | 41 | 42 | 43 | 45 | 49 | 53 | 83 | 84 | 91 | 92 | 93 | 94 | 433 | 434 | -------------------------------------------------------------------------------- /UserAgentChangeModLite.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name UserAgentChangeModLite.uc.js 3 | // @namespace http://www.sephiroth-j.de/mozilla/ 4 | // @charset utf-8 5 | // @note modify by lastdream2013 at 20130616 mino fix 6 | // @note modify by lastdream2013 at 20130409 sitelist : change SITELIST idx to Name 7 | // @note modify by lastdream2013 for navigator.userAgent https://g.mozest.com/thread-43428-1-2 8 | // @include chrome://browser/content/browser.xul 9 | // ==/UserScript== 10 | var ucjs_UAChanger = { 11 | 12 | DISPLAY_TYPE : 1, // 0显示列表为radiobox, 1显示为ua图标列表 13 | 14 | //(1)在url后面添加网站,注意用正则表达式 15 | SITE_LIST : [ 16 | 17 | // 在 http://www.google.co.jp/m 伪装成日本DoCoMo手机 18 | { 19 | url : "http://www\\.google\\.co\\.jp/m/", 20 | Name : "iPhone" 21 | }, //此处添加你需要的useragent的名称 22 | { 23 | url : "https?://www\\.icbc\\.com\\.cn/", 24 | Name : "Firefox10.0" 25 | }, { 26 | url : "https?://(?:mybank1?|b2c1)\\.icbc\\.com\\.cn/", 27 | Name : "Firefox10.0" 28 | },{ 29 | url : "http://vod\\.kankan\\.com/", 30 | Name : "Safari - Mac" 31 | }, //直接可以看kankan视频,无需高清组件 32 | 33 | //添加网站到此结束 34 | ], 35 | 36 | //现有版本firefox的图标 37 | NOW_UA_IMG : "", 38 | //其他版本firefox的图标 39 | EXT_FX_LIST_IMG : "", 40 | 41 | //自己在底下添加ua 42 | UA_LIST : [{ 43 | name : "分隔线", 44 | }, { 45 | name : "Firefox10.0", 46 | ua : "Mozilla/5.0 (Windows NT 6.1; rv:10.0.6) Gecko/20120716 Firefox/10.0.6", 47 | label : "Firefox10.0", 48 | img : "" 49 | }, { 50 | name : "分隔线", 51 | }, 52 | 53 | //伪装 IE8 - Windows7 54 | { 55 | name : "IE8 - Win7", //此处文字显示在右键菜单上,中文字符请转换成javascript编码,否则乱码(推荐http://rishida.net/tools/conversion/) 56 | ua : "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1; Trident/4.0)", 57 | label : "IE8", //此处文字显示在状态栏上,如果你设置状态栏不显示图标 58 | img : "" 59 | }, { 60 | name : "分隔线", 61 | }, 62 | 63 | // 伪装 Chrome22.0.1219.0 - Windows7 64 | { 65 | name : "Chrome - Win7", 66 | ua : "Mozilla/5.0 (Windows NT 6.1;) AppleWebKit/537.3 (KHTML, like Gecko) Chrome/22.0.1219.0 Safari/537.3", 67 | label : "Chrome Win7", 68 | img : "" 69 | }, { 70 | name : "分隔线", 71 | }, 72 | 73 | //伪装 Opera 10.60 74 | { 75 | name : "Opera", 76 | ua : "Opera/9.80 (Windows NT 6.1; U) Presto/2.6.30 Version/10.60", 77 | label : "Opera", 78 | img : "" 79 | }, { 80 | name : "分隔线", 81 | }, 82 | 83 | //伪装 Safari - Mac OS X 84 | { 85 | name : "Safari - Mac", 86 | ua : "Mozilla/5.0 (Macintosh; U; PPC Mac OS X 10_5_8; ja-jp) AppleWebKit/533.16 (KHTML, like Gecko) Version/5.0 Safari/533.16", 87 | label : "Safari", 88 | img : "" 89 | }, { 90 | name : "分隔线", 91 | }, 92 | 93 | //伪装 iPhone,查询http://www.zytrax.com/tech/web/mobile_ids.html 94 | { 95 | name : "iPhone", 96 | ua : "Mozilla/5.0 (iPhone; U; CPU iPhone OS 3_1_2 like Mac OS X; en-us) AppleWebKit/528.18 (KHTML, like Gecko) Version/4.0 Mobile/7D11 Safari/528.16", 97 | label : "iPhone", 98 | img : "" 99 | }, { 100 | name : "分隔线", 101 | }, 102 | 103 | //伪装 Apple iPad 2 104 | { 105 | name : "Apple iPad 2", 106 | ua : "Mozilla/5.0 (iPad; CPU OS 6_0 like Mac OS X) AppleWebKit/536.26 (KHTML, like Gecko) Version/6.0 Mobile/10A5355d Safari/8536.25", 107 | label : "Apple iPad 2", 108 | img : "" 109 | }, { 110 | name : "分隔线", 111 | }, 112 | 113 | //伪装 Google 爬虫 114 | { 115 | name : "Googlebot", 116 | ua : "Mozilla/5.0 (compatible; Googlebot/2.1; +http://www.google.com/bot.html)", 117 | label : "Googlebot", 118 | img : "" 119 | }, { 120 | name : "分隔线", 121 | }, 122 | // 添加ua,到此结束 123 | ], 124 | 125 | UANameIdxHash : [], 126 | 127 | // ----- 下面设置开始 ----- 128 | // defautl: ステータスバーの右端に表示する 129 | TARGET : null, // 定义一个target,用来调整状态栏顺序,null为空 130 | 131 | ADD_OTHER_FX : true, // true:自动添加其他版本firefox的ua false:不添加 132 | 133 | //2种版本firefox,下面请勿修改 134 | EXT_FX_LIST : [{ 135 | name : "Firefox4.0", 136 | ua : "Mozilla/5.0 (Windows; Windows NT 6.1; rv:2.0b2) Gecko/20100720 Firefox/4.0b2", 137 | label : "Fx4.0", 138 | img : "" 139 | }, { 140 | name : "Firefox3.6", 141 | ua : "Mozilla/5.0 (Windows; U; Windows NT 5.1; rv:1.9.2.8) Gecko/20100722 Firefox/3.6.8", 142 | label : "Fx3.6", 143 | img : "" 144 | }, 145 | ], 146 | // ---------------------- 147 | // UA リストのインデックス 148 | def_idx : 0, 149 | Current_idx : 0, 150 | 151 | // 初期化 152 | init : function () { 153 | this.mkData(); // UA データ(UA_LIST)を作る 154 | this.mkPanel(); // パネルとメニューを作る 155 | this.setSiteIdx(); 156 | // Observer 登録 157 | var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService); 158 | os.addObserver(this, "http-on-modify-request", false); 159 | os.addObserver(this.onDocumentCreated, "content-document-global-created", false); 160 | // イベント登録 161 | var contentArea = document.getElementById("appcontent"); 162 | contentArea.addEventListener("load", this, true); 163 | contentArea.addEventListener("select", this, false); 164 | var contentBrowser = this.getContentBrowser(); 165 | contentBrowser.tabContainer.addEventListener("TabClose", this, false); 166 | window.addEventListener("unload", this, false); 167 | }, 168 | onDocumentCreated : function (aSubject, aTopic, aData) { 169 | var aChannel = aSubject.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIWebNavigation).QueryInterface(Ci.nsIDocShell).currentDocumentChannel; 170 | if (aChannel instanceof Ci.nsIHttpChannel) { 171 | var navigator = aSubject.navigator; 172 | var userAgent = aChannel.getRequestHeader("User-Agent"); 173 | if (navigator.userAgent != userAgent) 174 | Object.defineProperty(XPCNativeWrapper.unwrap(navigator), "userAgent", { 175 | value : userAgent, 176 | enumerable : true 177 | }); 178 | } 179 | }, 180 | // UA データを作る 181 | mkData : function () { 182 | var ver = this.getVer(); // 現在使っている Firefox のバージョン 183 | // 現在使っている Firefox のデータを作る 184 | var tmp = []; 185 | tmp.name = "Firefox" + ver; 186 | tmp.ua = ""; 187 | tmp.img = this.NOW_UA_IMG; 188 | tmp.label = "Fx" + (this.ADD_OTHER_FX ? ver : ""); 189 | this.UA_LIST.unshift(tmp); 190 | // Fx のバージョンを見て UA を追加する 191 | if (this.ADD_OTHER_FX) { 192 | if (ver == 3.6) { // Fx3.6 の場合 Fx4 を追加する 193 | this.EXT_FX_LIST[0].img = this.EXT_FX_LIST_IMG; 194 | this.UA_LIST.push(this.EXT_FX_LIST[0]); 195 | } else { // Fx3.6 以外では Fx3.6 を追加する 196 | this.EXT_FX_LIST[1].img = this.EXT_FX_LIST_IMG; 197 | this.UA_LIST.push(this.EXT_FX_LIST[1]); 198 | } 199 | } 200 | // 起動時の UA を 初期化 (general.useragent.override の値が有るかチェック 07/03/02) 201 | var preferencesService = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefService).getBranch(""); 202 | if (preferencesService.getPrefType("general.useragent.override") != 0) { 203 | for (var i = 0; i < this.UA_LIST.length; i++) { 204 | if (preferencesService.getCharPref("general.useragent.override") == this.UA_LIST[i].ua) { 205 | this.def_idx = i; 206 | break; 207 | } 208 | } 209 | } 210 | }, 211 | // UA パネルを作る 212 | mkPanel : function () { 213 | var uacPanel = document.createElement("toolbarbutton"); 214 | uacPanel.setAttribute("id", "uac_statusbar_panel"); 215 | uacPanel.setAttribute("class", "toolbarbutton-1 chromeclass-toolbar-additional"); 216 | uacPanel.setAttribute("type", "menu"); 217 | // css 解决按钮定义在urlbar-icons撑大地址栏,变宽…… 218 | document.insertBefore(document.createProcessingInstruction('xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent( 219 | '\ 220 | #uac_statusbar_panel {\ 221 | -moz-appearance: none !important;\ 222 | border-style: none !important;\ 223 | border-radius: 0 !important;\ 224 | padding: 0 3px !important;\ 225 | margin: 0 !important;\ 226 | background: transparent !important;\ 227 | box-shadow: none !important;\ 228 | -moz-box-align: center !important;\ 229 | -moz-box-pack: center !important;\ 230 | min-width: 18px !important;\ 231 | min-height: 18px !important;\ 232 | }\ 233 | #uac_statusbar_panel > .toolbarbutton-icon {\ 234 | max-width: 18px !important;\ 235 | padding: 0 !important;\ 236 | margin: 0 !important;\ 237 | }\ 238 | #uac_statusbar_panel dropmarker{display: none !important;}\ 239 | ') + '"'), document.documentElement); 240 | 241 | uacPanel.setAttribute("image", this.UA_LIST[this.def_idx].img); 242 | uacPanel.style.padding = "0px 2px"; 243 | 244 | var toolbar = document.getElementById("urlbar-icons"); 245 | if (this.TARGET != null) { // default から書き換えている場合 246 | this.TARGET = document.getElementById(this.TARGET); 247 | } 248 | toolbar.insertBefore(uacPanel, this.TARGET); 249 | // UA パネルのコンテクストメニューを作る 250 | var PopupMenu = document.createElement("menupopup"); 251 | PopupMenu.setAttribute("id", "uac_statusbar_panel_popup"); 252 | for (var i = 0; i < this.UA_LIST.length; i++) { 253 | if (this.UA_LIST[i].name == "分隔线") { 254 | var mi = document.createElement("menuseparator"); 255 | PopupMenu.appendChild(mi); 256 | } else { 257 | var mi = document.createElement("menuitem"); 258 | 259 | mi.setAttribute('label', this.UA_LIST[i].name); 260 | mi.setAttribute('tooltiptext', this.UA_LIST[i].ua); 261 | mi.setAttribute('oncommand', "ucjs_UAChanger.setUA(" + i + ");"); 262 | 263 | if (this.DISPLAY_TYPE) { 264 | mi.setAttribute('class', 'menuitem-iconic'); 265 | mi.setAttribute('image', this.UA_LIST[i].img); 266 | } else { 267 | mi.setAttribute("type", "radio"); 268 | mi.setAttribute("checked", i == this.def_idx); 269 | } 270 | if (i == this.def_idx) { 271 | mi.setAttribute("style", 'font-weight: bold;'); 272 | mi.style.color = 'red'; 273 | } else { 274 | mi.setAttribute("style", 'font-weight: normal;'); 275 | mi.style.color = 'black'; 276 | } 277 | mi.setAttribute("uac-generated", true); 278 | PopupMenu.appendChild(mi); 279 | } 280 | } 281 | uacPanel.addEventListener("popupshowing", this, false); 282 | uacPanel.appendChild(PopupMenu); 283 | 284 | // パネルの変更を可能にする 285 | uacPanel.setAttribute("context", "uac_statusbar_panel_popup"); 286 | uacPanel.setAttribute("onclick", "event.stopPropagation();"); 287 | }, 288 | // URL 指定で User-Agent の書き換え(UserAgentSwitcher.uc.js より) 289 | observe : function (subject, topic, data) { 290 | if (topic != "http-on-modify-request") 291 | return; 292 | var http = subject.QueryInterface(Ci.nsIHttpChannel); 293 | for (var i = 0; i < this.SITE_LIST.length; i++) { 294 | if (http.URI && (new RegExp(this.SITE_LIST[i].url)).test(http.URI.spec)) { 295 | var idx = this.SITE_LIST[i].idx; 296 | http.setRequestHeader("User-Agent", this.UA_LIST[idx].ua, false); 297 | } 298 | } 299 | }, 300 | // イベント・ハンドラ 301 | handleEvent : function (aEvent) { 302 | var contentBrowser = this.getContentBrowser(); 303 | var uacPanel = document.getElementById("uac_statusbar_panel"); 304 | var uacMenu = document.getElementById("uac_statusbar_panel_popup"); 305 | switch (aEvent.type) { 306 | case "popupshowing": // コンテクスト・メニュー・ポップアップ時にチェック・マークを更新する 307 | var menu = aEvent.target; 308 | for (var i = 0; i < menu.childNodes.length; i++) { 309 | if (i == ucjs_UAChanger.Current_idx) { 310 | menu.childNodes[i].setAttribute("style", 'font-weight: bold;'); 311 | menu.childNodes[i].style.color = 'red'; 312 | if (!this.DISPLAY_TYPE) 313 | menu.childNodes[i].setAttribute("checked", true); 314 | } else { 315 | menu.childNodes[i].setAttribute("style", 'font-weight: normal;'); 316 | menu.childNodes[i].style.color = 'black'; 317 | } 318 | } 319 | break; 320 | case "load": // SITE_LIST に登録された URL の場合 321 | case "select": 322 | case "TabClose": 323 | for (var i = 0; i < ucjs_UAChanger.SITE_LIST.length; i++) { 324 | if ((new RegExp(this.SITE_LIST[i].url)).test(contentBrowser.currentURI.spec)) { 325 | var idx = this.SITE_LIST[i].idx; 326 | this.setImage(idx); 327 | return; 328 | } 329 | } 330 | this.setImage(this.def_idx); 331 | 332 | break; 333 | case "unload": // 終了処理 334 | var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService); 335 | os.removeObserver(this, "http-on-modify-request"); 336 | os.removeObserver(this.onDocumentCreated, "content-document-global-created"); 337 | var contentArea = document.getElementById("appcontent"); 338 | contentArea.removeEventListener("load", this, true); 339 | contentArea.removeEventListener("select", this, false); 340 | if (contentBrowser) 341 | contentBrowser.tabContainer.removeEventListener("TabClose", this, false); 342 | uacMenu.removeEventListener("popupshowing", this, false); 343 | window.removeEventListener("unload", this, false); 344 | break; 345 | } 346 | }, 347 | // 番号を指定して UA を設定 348 | setUA : function (i) { 349 | var preferencesService = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefService).getBranch(""); 350 | if (i == 0) { // オリジナル UA にする場合 351 | // 既にオリジナル UA の場合は何もしない 352 | if (preferencesService.getPrefType("general.useragent.override") == 0) 353 | return; 354 | preferencesService.clearUserPref("general.useragent.override"); 355 | } else { // 指定した UA にする場合 356 | preferencesService.setCharPref("general.useragent.override", this.UA_LIST[i].ua); 357 | } 358 | this.def_idx = i; 359 | this.setImage(i); 360 | }, 361 | // UA パネル画像とツールチップを設定 362 | setImage : function (i) { 363 | var uacPanel = document.getElementById("uac_statusbar_panel"); 364 | 365 | uacPanel.setAttribute("image", this.UA_LIST[i].img); 366 | uacPanel.style.padding = "0px 2px"; 367 | 368 | this.Current_idx = i; 369 | }, 370 | // アプリケーションのバージョンを取得する(Alice0775 氏のスクリプトから頂きました。) 371 | getVer : function () { 372 | var info = Cc["@mozilla.org/xre/app-info;1"].getService(Ci.nsIXULAppInfo); 373 | var ver = parseInt(info.version.substr(0, 3) * 10, 10) / 10; 374 | return ver; 375 | }, 376 | setSiteIdx : function () { 377 | for (let i = 0; i < this.UA_LIST.length; i++) { 378 | this.UANameIdxHash[this.UA_LIST[i].name] = i; 379 | } 380 | for (let j = 0; j < this.SITE_LIST.length; j++) { 381 | var uaName = this.SITE_LIST[j].Name; 382 | if (this.UANameIdxHash[uaName]) { 383 | this.SITE_LIST[j].idx = this.UANameIdxHash[uaName]; 384 | 385 | } else { 386 | this.SITE_LIST[j].idx = this.def_idx; 387 | 388 | } 389 | } 390 | }, 391 | // 現在のブラウザオブジェクトを得る。 392 | getContentBrowser : function () { 393 | var windowMediator = Cc["@mozilla.org/appshell/window-mediator;1"] 394 | .getService(Ci.nsIWindowMediator); 395 | var topWindowOfType = windowMediator.getMostRecentWindow("navigator:browser"); 396 | if (topWindowOfType) 397 | return topWindowOfType.document.getElementById("content"); 398 | return null; 399 | } 400 | } 401 | ucjs_UAChanger.init(); 402 | -------------------------------------------------------------------------------- /_uAutoPagerize.js: -------------------------------------------------------------------------------- 1 | // uAutoPagerize 的配置文件。Ver 0.2.2 以上专用。 2 | // 本体更新時に設定を書き換える手間を省くためのもので、無くても問題ない。 3 | 4 | // 排除列表 5 | var EXCLUDE = [ 6 | 'https://mail.google.com/*' 7 | ,'http://www.google.co.jp/reader/*' 8 | ,'http://b.hatena.ne.jp/*' 9 | ,'http://www.livedoor.com/*' 10 | ,'http://reader.livedoor.com/*' 11 | ,'http://fastladder.com/*' 12 | ,'http://**mail.yahoo.co.jp/*' 13 | ,'http://maps.google.co.jp/*' 14 | ,'*/archives/*' 15 | ]; 16 | 17 | // 自定义站点,优先级最高 18 | var MY_SITEINFO = [ 19 | { 20 | url : '^https://mobile\\.twitter\\.com/[^/]+/status(?:es)?/\\d', 21 | nextLink : 'id("tweets-list")/div[@class="list-tweet"][1]/div[@class="list-tweet-status permalink"]/a[@class="status_link"][2]', 22 | pageElement : 'id("tweets-list")', 23 | }, 24 | { 25 | url : '^http://www\\.dm5\\.com/m\\d+/', 26 | nextLink : 'id("s_next")/a', 27 | pageElement : 'id("showimage")', 28 | }, 29 | { 30 | siteName : '顶点小说', 31 | url : '^http://www\\.23us\\.com/html/.+\\.html', 32 | siteExample : 'http://www.23us.com/html/26/26627/16952316.html', 33 | nextLink : ' //dd[@id="footlink"]/descendant::a[text()="下一页"]', 34 | pageElement : 'id("amain")/dl/dd/h1 | id("contents")' 35 | }, 36 | { 37 | url : '^http://bbs\\.iiikl\\.net/forum\\.php\\?forum_id=.*', 38 | nextLink : '//a[@class="next"]', 39 | pageElement : '//tr[@class="topic_list_row"]', 40 | exampleUrl : 'http://bbs.iiikl.net/forum.php?forum_id=82&class_id=0&page=2' 41 | }, 42 | { 43 | "url" : "^http://[^.]+\\.getuploader\\.com/", 44 | "nextLink" : "//li[contains(concat(\" \", @class, \" \"), \" current \")]/following-sibling::li[contains(concat(\" \", @class, \" \"), \" page \")]/a", 45 | "pageElement" : "id(\"primary\")/table/tbody/tr|id(\"thumbnail\")" 46 | }, 47 | { 48 | url : '^http://list\\.tmall\\.com/search_product\\.htm', 49 | nextLink : '//a[@class="ui-page-next"]', 50 | pageElement : 'id("J_ItemList")', 51 | insertBefore : '', 52 | exampleUrl : 'http://list.tmall.com/search_product.htm?spm=3.1000473.332931.1.IYk3dO&q=&area_code=310000&sort=s&style=g&vmarket=0&from=sn_1_cat-qp&active=1&cat=50024400' 53 | }, 54 | { 55 | url : '^http://f\\.ppxclub\\.com/.*', 56 | nextLink : '//a[@class="nxt"]', 57 | pageElement : '//div[@class="pl bm"] | id("threadlist")/div', 58 | exampleUrl : 'http://f.ppxclub.com/147006-1-2' 59 | }, 60 | { 61 | siteName : '百度贴吧', 62 | url : '^http://tieba\\.baidu\\.(cn|com)/f', 63 | nextLink : '//div[@class="pager clearfix"]/descendant::a[@class="next"]', 64 | pageElement : '//ul[@id="thread_list"]', 65 | }, 66 | { 67 | siteName : '百度空间', 68 | url : '^http://hi\\.baidu\\.com', 69 | nextLink : 70 | { 71 | startAfter : '?page=', 72 | mFails : [/^http:\/\/hi\.baidu\.com\/.+/i, '?page=1'], 73 | inc : 1, 74 | isLast : function (doc, win, lhref) 75 | { 76 | var script = doc.querySelector("#pagerBar > script"); 77 | var m = script && script.textContent.match(/pageSize.*'(\d+)'[\s\S]*curPage.*'(\d+)'/); 78 | if (m && (m.length == 3)) 79 | { 80 | if (parseInt(m[2]) >= parseInt(m[1])) 81 | return true; 82 | } 83 | } 84 | }, 85 | pageElement : '//div[@class="mod-realcontent mod-cs-contentblock"]', 86 | exampleUrl : 'http://hi.baidu.com/gelida', 87 | }, 88 | { 89 | name : '水木社区', 90 | url : '^http://www\\.newsmth\\.net/nForum.*', 91 | nextLink : '//a[@title="下一页"]', 92 | pageElement : '//div[@class="b-content"] | //div[@class="b-content corner"]', 93 | exampleUrl : 'http://www.newsmth.net/nForum/#!board/TouHou' 94 | }, 95 | { 96 | siteName : '软件淘', 97 | url : '^http://www\\.65052424\\.com/', 98 | nextLink : '//a[@class="next"]', 99 | pageElement : '//div[@id="content"]', 100 | exampleUrl : 'http://www.65052424.com/page/7', 101 | }, 102 | { 103 | siteName : 'OperaChina', 104 | url : /http:\/\/(?:bbs\.operachina\.com|oc\.ls\.tl)/i, 105 | nextLink : 'auto;', 106 | pageElement : '//table | //article[@class="post blue"]' 107 | }, 108 | ]; 109 | 110 | // 本体に組み込まれている MICROFORMAT を利用するか? 111 | USE_MICROFORMAT = true; 112 | -------------------------------------------------------------------------------- /blur2revertMod.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name url bar blur to revert mod 3 | // @namespace blur2revert@zbinlin.gmail.com 4 | // @filename blur2revert.uc.js 5 | // @description url bar blur to revert mod ( Ctrl + 鼠标点击页面后,恢复原来的地址,规避原脚本和Noscript冲突 ) 6 | // @author zbinlin 7 | // @version 0.1.20110602.1 8 | // @updateURL https://j.mozest.com/ucscript/script/39.meta.js 9 | // ==/UserScript== 10 | 11 | if (location == "chrome://browser/content/browser.xul") { 12 | gBrowser.addEventListener("DOMWindowCreated", function () { 13 | window.content.document.addEventListener("click", function (e) { 14 | if (e.ctrlKey) { 15 | document.getElementById("urlbar").handleRevert(); 16 | } 17 | }, false); 18 | }, false); 19 | } 20 | -------------------------------------------------------------------------------- /bookmarkButtonLite.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | %mainwindowDTD; 6 | 7 | %placesDTD; 8 | ]> 9 | 10 | 11 | // ==UserScript== 12 | // @name bookmarkButtonlite.uc.xul 13 | // @description 14 | // @include main 15 | // @include chrome://browser/content/places/places.xul 16 | // @include chrome://browser/content/bookmarks/bookmarksPanel.xul 17 | // @author lastdream2013 18 | // @version 0.11 19 | // ==/UserScript== 20 | --> 21 | 35 | 36 | 37 | 38 | 52 | 62 | 67 | 72 | 73 | 74 | 75 | 77 | 81 | 82 | 83 | 84 | 85 | 86 | -------------------------------------------------------------------------------- /bookmarkButtonPlus.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | %mainwindowDTD; 6 | 7 | %placesDTD; 8 | ]> 9 | 10 | 11 | // ==UserScript== 12 | // @name bookmarkButtonPlus.uc.xul 13 | // @description 14 | // @include main 15 | // @include chrome://browser/content/places/places.xul 16 | // @include chrome://browser/content/bookmarks/bookmarksPanel.xul 17 | // @author lastdream2013 18 | // @version 0.11 19 | // ==/UserScript== 20 | --> 21 | 35 | 36 | 37 | 51 | 61 | 67 | 68 | 72 | 73 | 78 | 84 | 88 | 92 | 93 | 94 | 99 | 104 | 105 | 106 | 107 | 109 | 113 | 114 | 115 | 116 | 117 | 118 | -------------------------------------------------------------------------------- /contextHistoryMenu.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 4 | %browserDTD; 5 | ]> 6 | 7 | 8 | 9 | 10 | 14 | 17 | 18 | 22 | 23 | 28 | 32 | 34 | 35 | 40 | 43 | 44 | 45 | 46 | 47 | 48 | 49 | -------------------------------------------------------------------------------- /contextHistoryMenuLite.uc.xul: -------------------------------------------------------------------------------- 1 | 2 | 4 | %browserDTD; 5 | ]> 6 | 7 | 8 | 9 | 10 | 14 | 17 | 21 | 22 | 23 | 24 | 25 | 26 | -------------------------------------------------------------------------------- /externalAppButtonM.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name externalAppButtonM.uc.js 3 | // @namespace ithinc#mozine.cn 4 | // @description External Applications button 5 | // @include main 6 | // @author ithinc & lastdream2013 7 | // @charset UTF-8 8 | // @version 20130616.1.1.6 tidy and merge moveable code 9 | // @version 20130504.1.1.3 delay load exefile, may speedup firefox startup 10 | // @version 20130414.1.1.0 support submenu 11 | // @version 20130402.1.0.1 start modified by lastdream2013, add app menu 12 | // @version 20091216.1.0.0 Final release 13 | // @version 20091215.0.0.2 Handle toolbar apps and menu apps separately 14 | // @version 20091212.0.0.1 Initial release 15 | // ==/UserScript== 16 | 17 | var gExternalAppbuttonM = { 18 | autohideEmptySubDirs: true, //自动隐藏没有一个子项目的子目录菜单 19 | moveSubDirstoBottom: false, //把主菜单下的子目录移动到最下面 20 | moveablePositonOrInsertafter: false, //true : ToolbarPalette moveable button false: insert appbutton in "insertafter" 21 | insertafter: 'urlbar-icons', // useless if moveablePositonOrInsertafter is true; urlbar-icons addon-bar TabsToolbar alltabs-button 22 | toolbar: { 23 | //在这里定义好主菜单下子目录的名字,以及图标(可定义或留空,好图标不好找....); 可在中间加{name: 'separator'}建立一个目录与目录之间的分隔线 24 | subdirs: [ 25 | {name: 'windows工具', image: ""}, 26 | {name: '常用工具', image: "" }, 27 | {name: '网络工具', image: ""}, 28 | {name: '编辑工具', image: ""}, 29 | {name: '图像影音', image: ""}, 30 | {name: 'separator'}, 31 | {name: '常用目录' , image: ""}, 32 | {name: 'firefox目录', image: "" }, 33 | {name: 'firefox配置文件', image: "" }, 34 | ], 35 | 36 | //1.在这里定义好想要出现在主菜单下,或在主菜单子目录下的程序(subdir没有定义, 或在上面子目录名列表中找不到名称相匹配的[比如写错了], 就会出现在主菜单下面); 37 | //2. 可在中间加{name: 'separator'}, 分隔线如果定义了子目录名,就出现在子目录下面;没有定义就在主目录下面. 38 | //3. 这里定义的可以重复. 例如IE浏览器我想出现在windows工具下面, 又想出现在子菜单下面, 那么就定义二次: 39 | // {name: 'IE浏览器', path: 'C:\\Program Files\\Internet Explorer\\IEXPLORE.EXE', args: ['%u'], subdir: 'windows工具' }, 40 | // {name: 'IE浏览器', path: 'C:\\Program Files\\Internet Explorer\\IEXPLORE.EXE', args: ['%u'] }, 41 | // 就可以了, 建议先写完上面想要定义, 分类在子目录下的程序, 之后从中摘出你最常用的, 去掉后面的subdir定义, 放在最下面 42 | apps: [ 43 | {name: 'IE浏览器', path: 'C:\\Program Files\\Internet Explorer\\IEXPLORE.EXE', args: ['%u'], subdir: 'windows工具' }, 44 | {name: '记事本', path: 'C:\\Windows\\notepad.exe', subdir: 'windows工具'}, 45 | {name: '计算器', path: 'C:\\Windows\\System32\\calc.exe', subdir: 'windows工具'}, 46 | {name: '命令行', path: 'C:\\WINDOWS\\system32\\cmd.exe', subdir: 'windows工具'}, 47 | /* 48 | {name: 'HyperSnap 7', path: 'E:\\software\\HyperSnap 7\\HprSnap7.exe', subdir: '常用工具'}, 49 | {name: 'Photoshop', path: 'E:\\software\\Photoshop7.0\\Photoshop.exe', subdir: '常用工具'}, 50 | {name: 'separator', subdir: '常用工具'}, 51 | {name: 'Total Commander', path: 'E:\\software\\Total Commander\\TOTALCMD.EXE', subdir: '常用工具' }, 52 | {name: 'Everything', path: 'E:\\software\\Everything\\Everything.exe', subdir: '常用工具' }, 53 | {name: 'HostsX', path: 'E:\\software\\HostsX\\HostsX.exe', subdir: '常用工具'}, 54 | {name: 'Lingoes', path: 'E:\\software\\Lingoes\\Lingoes.exe', subdir: '常用工具'}, 55 | {name: 'PDFXCview', path: 'E:\\software\\PDF Viewer\\PDFXCview.exe', subdir: '常用工具'}, 56 | 57 | {name: 'Goagent', path: 'E:\\software\\ProxyTools\\goagent\\local\\goagent.exe', subdir: '网络工具'}, 58 | {name: 'opera', path: 'E:\\software\\Opera\\opera.exe', args: ['%u'], subdir: '网络工具'}, 59 | {name: 'Becky!', path: 'E:\\software\\Becky!\\B2.exe', subdir: '网络工具'}, 60 | {name: 'QQ', path: 'E:\\software\\QQ2013\\Bin\\QQ.exe', subdir: '网络工具'}, 61 | {name: 'AliIM', path: 'E:\\software\\AliWangWang\\AliIM.exe', subdir: '网络工具'}, 62 | {name: 'Fetion', path: 'E:\\software\\Fetion2012\\Fetion.exe', subdir: '网络工具'}, 63 | {name: 'Thunder', path: 'E:\\software\\Thunder7.2.11.3788JayXon\\Program\\Thunder.exe', subdir: '网络工具'}, 64 | 65 | {name: 'foobar2000', path: 'E:\\software\\Foobar2000\\foobar2000.exe', subdir: '图像影音'}, 66 | {name: 'PotPlayer', path: 'E:\\software\\PotPlayer\\PotPlayerMini.exe', subdir: '图像影音'}, 67 | {name: 'Imagine', path: 'E:\\software\\Imagine\\Imagine.exe', subdir: '图像影音'}, 68 | {name: 'i_view32', path: 'E:\\software\\IrfanView\\i_view32.exe', subdir: '图像影音'}, 69 | {name: 'Inpaint.exe', path: 'E:\\software\\Inpaint\\Inpaint.exe', subdir: '图像影音'}, 70 | {name: 'MangaMeeya', path: 'E:\\software\\MangaMeeya\\MangaMeeya.exe', subdir: '图像影音'}, 71 | {name: 'openCanvas', path: 'E:\\software\\openCanvas\\oC55.chs.exe', subdir: '图像影音'}, 72 | 73 | {name: 'EmEditor', path: 'E:\\software\\EmEditor\\EmEditor.exe', subdir: '编辑工具'}, 74 | {name: 'SublimeText2', path: 'E:\\software\\SublimeText2\\SublimeText.exe', subdir: '编辑工具'}, 75 | {name: 'AkelPad', path: 'E:\\software\\AkelPad\\AkelPad.exe', subdir: '编辑工具'}, 76 | {name: 'SourceInsight', path: 'E:\\software\\SourceInsight\\Insight3.exe', subdir: '编辑工具'}, 77 | {name: 'WinHex', path: 'E:\\software\\WinHex\\WinHex.exe', subdir: '编辑工具'}, 78 | 79 | {name: 'Software', path: 'E:\\software', subdir:'常用目录'}, 80 | {name: 'Downloads', path: 'I:\\Downloads', subdir:'常用目录'}, 81 | {name: 'TDDownloads', path: 'I:\\TDDownload', subdir:'常用目录'}, 82 | */ 83 | {name: 'profile', path: '\\', subdir:'firefox目录'}, 84 | {name: 'chrome', path: '\\chrome', subdir:'firefox目录'}, 85 | {name: 'CSS', path: '\\chrome\\CSS', subdir:'firefox目录'}, 86 | {name: 'SubScript', path: '\\chrome\\SubScript', subdir:'firefox目录'}, 87 | {name: 'UserScriptLoader', path: '\\chrome\\UserScriptLoader', subdir:'firefox目录'}, 88 | 89 | {name: 'userChrome.css', path: '\\chrome\\userChrome.css', subdir:'firefox配置文件'}, 90 | {name: 'userContent.css', path: '\\chrome\\userContent.css', subdir:'firefox配置文件'}, 91 | {name: 'prefs.js', path: '\\prefs.js', subdir:'firefox配置文件'}, 92 | {name: 'user.js', path: '\\user.js', subdir:'firefox配置文件'}, 93 | 94 | // 建议把要放在子目录下的程序定义在上面, 下面的定义放在主菜单下的最常用的程序 95 | /* {name: 'separator'}, 96 | {name: 'HyperSnap 7', path: 'E:\\software\\HyperSnap 7\\HprSnap7.exe'}, 97 | {name: 'Photoshop', path: 'E:\\software\\Photoshop7.0\\Photoshop.exe'}, 98 | {name: 'Everything', path: 'E:\\software\\Everything\\Everything.exe'}, 99 | {name: 'HostsX', path: 'E:\\software\\HostsX\\HostsX.exe'}, 100 | {name: 'separator'}, 101 | {name: 'Goagent', path: 'E:\\software\\ProxyTools\\goagent\\local\\goagent.exe'}, 102 | {name: 'Thunder', path: 'E:\\software\\Thunder7.2.11.3788JayXon\\Program\\Thunder.exe'}, 103 | {name: 'EmEditor', path: 'E:\\software\\EmEditor\\EmEditor.exe'}, 104 | {name: 'SourceInsight', path: 'E:\\software\\SourceInsight\\Insight3.exe'}, 105 | {name: 'separator'}, 106 | {name: 'Software', path: 'E:\\software'}, 107 | */ 108 | {name: 'chrome', path: '\\chrome'}, 109 | ], 110 | }, 111 | 112 | subdirPopupHash : [], 113 | subdirMenuHash : [], 114 | _externalAppPopup : null, 115 | _isready : false, 116 | init : function () { 117 | this.handleRelativePath(this.toolbar.apps); 118 | 119 | var ExternalAppBtn = document.createElement('toolbarbutton'); 120 | ExternalAppBtn.id = "gExternalAppbuttonM-ID"; 121 | ExternalAppBtn.setAttribute("label", "外部程序按钮"); 122 | ExternalAppBtn.setAttribute("onclick", "event.preventDefault();event.stopPropagation();"); 123 | ExternalAppBtn.setAttribute("tooltiptext", "外部程序按钮,可以自定义外部程序和路径"); 124 | ExternalAppBtn.setAttribute("class", "toolbarbutton-1 chromeclass-toolbar-additional"); 125 | ExternalAppBtn.setAttribute("type", "menu"); 126 | ExternalAppBtn.setAttribute("removable", "true"); 127 | ExternalAppBtn.setAttribute("image", ""); 128 | 129 | var ExternalAppPopup = document.createElement('menupopup'); 130 | ExternalAppPopup.setAttribute('onpopupshowing', 'event.stopPropagation();gExternalAppbuttonM.onpopupshowing();'); 131 | this._externalAppPopup = ExternalAppPopup; 132 | ExternalAppBtn.appendChild(ExternalAppPopup); 133 | setTimeout(function () { //延时加载菜单,不对启动造成影响,也不影响第一次打开菜单时的响应速度 134 | gExternalAppbuttonM.loadSubMenu(); 135 | }, 3000); 136 | document.insertBefore(document.createProcessingInstruction('xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent( 137 | '\ 138 | #gExternalAppbuttonM-ID {\ 139 | -moz-appearance: none !important;\ 140 | border-style: none !important;\ 141 | border-radius: 0 !important;\ 142 | padding: 0 3px !important;\ 143 | margin: 0 !important;\ 144 | background: transparent !important;\ 145 | box-shadow: none !important;\ 146 | -moz-box-align: center !important;\ 147 | -moz-box-pack: center !important;\ 148 | min-width: 18px !important;\ 149 | min-height: 18px !important;\ 150 | }\ 151 | #gExternalAppbuttonM-ID > .toolbarbutton-icon {\ 152 | max-width: 18px !important;\ 153 | padding: 0 !important;\ 154 | margin: 0 !important;\ 155 | }\ 156 | #gExternalAppbuttonM-ID dropmarker{display: none !important;}\ 157 | ') + '"'), document.documentElement); 158 | 159 | if (this.moveablePositonOrInsertafter) { 160 | var navigator = document.getElementById("navigator-toolbox"); 161 | if (!navigator || navigator.palette.id !== "BrowserToolbarPalette") 162 | return; 163 | navigator.palette.appendChild(ExternalAppBtn); 164 | this.updateToolbar(); 165 | } else { 166 | var navigator = document.getElementById(this.insertafter); 167 | if (!navigator) 168 | return; 169 | navigator.insertBefore(ExternalAppBtn, navigator.firstChild); 170 | } 171 | }, 172 | loadSubMenu : function () { 173 | if (this._isready) 174 | return; 175 | if (this._externalAppPopup == null) 176 | return; 177 | var ExternalAppPopup = this._externalAppPopup; 178 | for (var i = 0; i < this.toolbar.subdirs.length; i++) { 179 | if (this.toolbar.subdirs[i].name == 'separator') { 180 | ExternalAppPopup.appendChild(document.createElement('menuseparator')); 181 | } else { 182 | var subDirItem = ExternalAppPopup.appendChild(document.createElement('menu')); 183 | var subDirItemPopup = subDirItem.appendChild(document.createElement('menupopup')); 184 | subDirItem.setAttribute('class', 'menu-iconic'); 185 | subDirItem.setAttribute('label', this.toolbar.subdirs[i].name); 186 | subDirItem.setAttribute('image', this.toolbar.subdirs[i].image); 187 | gExternalAppbuttonM.subdirPopupHash[this.toolbar.subdirs[i].name] = subDirItemPopup; 188 | gExternalAppbuttonM.subdirMenuHash[this.toolbar.subdirs[i].name] = subDirItem; 189 | } 190 | } 191 | 192 | for (var i = 0; i < this.toolbar.apps.length; i++) { 193 | var appsItems; 194 | if (this.toolbar.apps[i].name == 'separator') { 195 | appsItems = document.createElement('menuseparator'); 196 | } else { 197 | appsItems = document.createElement('menuitem'); 198 | appsItems.setAttribute('class', 'menuitem-iconic'); 199 | appsItems.setAttribute('label', this.toolbar.apps[i].name); 200 | appsItems.setAttribute('image', 'moz-icon:file://' + this.toolbar.apps[i].path + '?size=16;'); 201 | appsItems.setAttribute('oncommand', "gExternalAppbuttonM.exec(this.path, this.args);"); 202 | appsItems.setAttribute('tooltiptext', this.toolbar.apps[i].name); 203 | appsItems.path = this.toolbar.apps[i].path; 204 | appsItems.args = this.toolbar.apps[i].args; 205 | } 206 | if (this.toolbar.apps[i].subdir && gExternalAppbuttonM.subdirPopupHash[this.toolbar.apps[i].subdir]) 207 | gExternalAppbuttonM.subdirPopupHash[this.toolbar.apps[i].subdir].appendChild(appsItems); 208 | else 209 | ExternalAppPopup.appendChild(appsItems); 210 | } 211 | 212 | if (this.autohideEmptySubDirs) { 213 | for (let[name, popup]in Iterator(gExternalAppbuttonM.subdirPopupHash)) { 214 | if (popup.hasChildNodes()) { 215 | continue; 216 | } else { 217 | gExternalAppbuttonM.subdirMenuHash[name].setAttribute("hidden", "true"); 218 | } 219 | } 220 | } 221 | 222 | if (this.moveSubDirstoBottom) { 223 | let i = ExternalAppPopup.childNodes.length; 224 | while (ExternalAppPopup.firstChild.getAttribute('class') != 'menuitem-iconic' && i-- != 0) { 225 | ExternalAppPopup.appendChild(ExternalAppPopup.firstChild); 226 | } 227 | } 228 | this._isready = true; 229 | }, 230 | onpopupshowing : function () { 231 | if (!this._isready) 232 | this.loadSubMenu(); 233 | }, 234 | 235 | handleRelativePath : function (apps) { 236 | for (var i = 0; i < apps.length; i++) { 237 | if (apps[i].path) { 238 | apps[i].path = apps[i].path.replace(/\//g, '\\').toLocaleLowerCase(); 239 | var ffdir = Components.classes['@mozilla.org/file/directory_service;1'].getService(Components.interfaces.nsIProperties).get("ProfD", Components.interfaces.nsILocalFile).path; 240 | if (/^(\\)/.test(apps[i].path)) { 241 | apps[i].path = ffdir + apps[i].path; 242 | } 243 | } 244 | } 245 | }, 246 | 247 | exec : function (path, args) { 248 | args = args || []; 249 | var args_t = args.slice(0); 250 | for (var i = 0; i < args_t.length; i++) { 251 | args_t[i] = args_t[i].replace(/%u/g, gBrowser.currentURI.spec); 252 | } 253 | 254 | var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsILocalFile); 255 | file.initWithPath(path); 256 | if (!file.exists()) { 257 | Cu.reportError('File Not Found: ' + path); 258 | return; 259 | } 260 | 261 | if (!file.isExecutable()) { 262 | file.launch(); 263 | } else { 264 | var process = Cc['@mozilla.org/process/util;1'].createInstance(Ci.nsIProcess); 265 | process.init(file); 266 | process.run(false, args_t, args_t.length); 267 | } 268 | }, 269 | updateToolbar : function () { 270 | let toolbars = Array.slice(document.querySelectorAll('#navigator-toolbox > toolbar')); 271 | toolbars.forEach(function (toolbar) { 272 | var currentset = toolbar.getAttribute("currentset"); 273 | if (currentset.split(",").indexOf("gExternalAppbuttonM-ID") < 0) 274 | return; 275 | toolbar.currentSet = currentset; 276 | try { 277 | BrowserToolboxCustomizeDone(true); 278 | } catch (ex) {} 279 | }); 280 | } 281 | }; 282 | gExternalAppbuttonM.init(); 283 | -------------------------------------------------------------------------------- /externalFuncButtonM.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name externalFuncButtonM.uc.js 3 | // @description External functions button 4 | // @include main 5 | // @author lastdream2013 6 | // @charset UTF-8 7 | // @version 20130616.0.13 minor fix 8 | // @version 20130511.0.11 tidy and merge moveable code 9 | // @version 20130507 0.1 first release 10 | // ==/UserScript== 11 | 12 | var gExternalFuncButtonM = { 13 | autohideEmptySubDirs: true, //自动隐藏没有一个子项目的子目录菜单 14 | moveSubDirstoBottom: false, //把主菜单下的子目录移动到最下面 15 | moveablePositonOrInsertafter: false, //true : ToolbarPalette moveable button false: insert appbutton in "insertafter" 16 | insertafter: 'alltabs-button', // useless if moveablePositonOrInsertafter is true; urlbar-icons addon-bar TabsToolbar alltabs-button 17 | toolbar : 18 | { 19 | //在这里定义好主菜单下子目录的名字,以及图标 可在中间加{name: 'separator'}建立一个目录与目录之间的分隔线 20 | subdirs : [ 21 | { 22 | name : 'firefox常用功能', 23 | image : "" 24 | }, 25 | { 26 | name : 'about', 27 | image : "" 28 | }, 29 | ], 30 | 31 | // 在这里定义firefox的功能按钮, command就是一小段程序, 可以从firefox api, 小书签或鼠标手势中摘取;可选自定义图标; 32 | // 同样, 建议先写完上面想要定义, 分类在子目录下的程序, 之后从中摘出你最常用的, 去掉后面的subdir定义, 放在最下面 33 | configs : [ 34 | { 35 | name : 'separator' 36 | }, 37 | { 38 | name : 'about:firefox', 39 | subdir : 'about', 40 | command : "openAboutDialog();", 41 | }, 42 | { 43 | name : 'about:about', 44 | subdir : 'about', 45 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:about')", 46 | }, 47 | { 48 | name : 'about:addons', 49 | subdir : 'about', 50 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:addons')", 51 | }, 52 | { 53 | name : 'about:cache', 54 | subdir : 'about', 55 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:cache')", 56 | }, 57 | { 58 | name : 'about:config', 59 | subdir : 'about', 60 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:config')", 61 | }, 62 | { 63 | name : 'about:memory', 64 | subdir : 'about', 65 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:memory')", 66 | }, 67 | { 68 | name : 'about:mozilla', 69 | subdir : 'about', 70 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:mozilla')", 71 | }, 72 | { 73 | name : 'about:plugins', 74 | subdir : 'about', 75 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:plugins')", 76 | }, 77 | { 78 | name : 'about:robots', 79 | subdir : 'about', 80 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:robots')", 81 | }, 82 | { 83 | name : 'about:support', 84 | subdir : 'about', 85 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:support')", 86 | }, 87 | { 88 | name : '选项', 89 | subdir : 'firefox常用功能', 90 | image : "", 91 | command : "openPreferences();", 92 | }, 93 | { 94 | name : '书签管理', 95 | subdir : 'firefox常用功能', 96 | image : "chrome://browser/skin/places/allBookmarks.png", 97 | command : "PlacesCommandHook.showPlacesOrganizer('AllBookmarks');", 98 | }, 99 | { 100 | name : '历史', 101 | subdir : 'firefox常用功能', 102 | image : "chrome://browser/content/abouthome/history.png", 103 | command : "PlacesCommandHook.showPlacesOrganizer('History');", 104 | }, 105 | { 106 | name : 'separator', 107 | subdir : 'firefox常用功能', 108 | }, 109 | { 110 | name : '重启(清除启动缓存)', 111 | subdir : 'firefox常用功能', 112 | command : "Services.appinfo.invalidateCachesOnRestart() || Application.restart();", 113 | }, 114 | 115 | // 建议把要放在子目录下的功能按钮,定义在上面, 下面的定义放在主菜单下的最常用的功能按钮, 116 | { 117 | name : '破解右键限制', 118 | subdir : '', 119 | image :"", 120 | command : 121 | function () 122 | { 123 | gBrowser.loadURI("javascript:(function(){var%20doc=document;var%20bd=doc.body;bd.onselectstart=bd.oncopy=bd.onpaste=bd.onkeydown=bd.oncontextmenu=bd.onmousemove=bd.onselectstart=bd.ondragstart=doc.onselectstart=doc.oncopy=doc.onpaste=doc.onkeydown=doc.oncontextmenu=null;doc.onselectstart=doc.oncontextmenu=doc.onmousedown=doc.onkeydown=function%20(){return%20true;};with(document.wrappedJSObject||document){onmouseup=null;onmousedown=null;oncontextmenu=null;}var%20arAllElements=document.getElementsByTagName('*');for(var%20i=arAllElements.length-1;i>=0;i--){var%20elmOne=arAllElements[i];with(elmOne.wrappedJSObject||elmOne){onmouseup=null;onmousedown=null;}}var%20head=document.getElementsByTagName('head')[0];if(head){var%20style=document.createElement('style');style.type='text/css';style.innerHTML=%22html,*{-moz-user-select:auto!important;}%22;head.appendChild(style);}void(0);})();") 124 | }, 125 | }, 126 | { 127 | name : '轻松下视频', 128 | subdir : '', 129 | image :"", 130 | command : 131 | function () 132 | { 133 | gBrowser.loadURI("javascript:if(typeof%20yXzyxe58==typeof%20alert)yXzyxe58();window.open('http://www.youtubexia.com/?url='%20+encodeURIComponent(document.location.href),'getA','toolbar=no,%20location=no,%20directories=no,%20status=no,%20menubar=no,%20scrollbars=yes,%20resizable=yes,%20copyhistory=yes,%20width=800,%20height=600');void(0);") 134 | }, 135 | }, 136 | { 137 | name : 'PrintWhatYouLike', 138 | subdir : '', 139 | image :"", 140 | command : 141 | function () 142 | { 143 | gBrowser.loadURI("javascript:(function(){if(window['ppw']&&ppw['bookmarklet']){ppw.bookmarklet.toggle();}else{window._pwyl_home='http://www.printwhatyoulike.com/';window._pwyl_pro_id=null;window._pwyl_bmkl=document.createElement('script');window._pwyl_bmkl.setAttribute('type','text/javascript');window._pwyl_bmkl.setAttribute('src',window._pwyl_home+'static/compressed/pwyl_bookmarklet_10.js');window._pwyl_bmkl.setAttribute('pwyl','true');document.getElementsByTagName('head')[0].appendChild(window._pwyl_bmkl);}})();") 144 | }, 145 | }, 146 | { 147 | name : '阅读助手', 148 | subdir : '', 149 | image :"", 150 | command : 151 | function () 152 | { 153 | gBrowser.loadURI("javascript:(function(){readStyle='style-newspaper';readSize='size-large';readMargin='margin-wide';_readability_script=document.createElement('SCRIPT');_readability_script.type='text/javascript';_readability_script.src='http://lab.arc90.com/experiments/readability/js/readability.js?x='+(Math.random());document.getElementsByTagName('head')%5B0%5D.appendChild(_readability_script);_readability_css=document.createElement('LINK');_readability_css.rel='stylesheet';_readability_css.href='http://lab.arc90.com/experiments/readability/css/readability.css';_readability_css.type='text/css';_readability_css.media='screen';document.getElementsByTagName('head')%5B0%5D.appendChild(_readability_css);_readability_print_css=document.createElement('LINK');_readability_print_css.rel='stylesheet';_readability_print_css.href='http://lab.arc90.com/experiments/readability/css/readability-print.css';_readability_print_css.media='print';_readability_print_css.type='text/css';document.getElementsByTagName('head')%5B0%5D.appendChild(_readability_print_css);})();") 154 | }, 155 | }, 156 | { 157 | name : 'separator', 158 | }, 159 | { 160 | name : '下载管理', 161 | image : "chrome://browser/skin/places/downloads.png", 162 | command : "BrowserDownloadsUI();", 163 | }, 164 | { 165 | name : '附加组件', 166 | image : "chrome://mozapps/skin/extensions/extensionGeneric-16.png", 167 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:addons')", 168 | }, 169 | { 170 | name : 'about:config', 171 | image : "", 172 | command : "getBrowser().selectedTab = getBrowser().addTab ('about:config')", 173 | }, 174 | { 175 | name : '错误控制台', 176 | image : "", 177 | command : "toJavaScriptConsole();", 178 | }, 179 | ], 180 | }, 181 | 182 | subdirPopupHash : [], 183 | subdirMenuHash : [], 184 | _externalFuncPopup : null, 185 | _isready : false, 186 | unescapeHTML : function(input) { 187 | return input.replace(/&/g, '&') 188 | .replace(/"/g, '\"') 189 | .replace(/</g, '<') 190 | .replace(/>/g, '>') 191 | .replace(/'/g, '\'') 192 | .replace(/([^\\])\\([^\\])/g, '$1\\\\$2'); 193 | }, 194 | init : function () { 195 | 196 | var ExternalFuncBtn = document.createElement('toolbarbutton'); 197 | ExternalFuncBtn.id = "ExternalFuncButtonM-ID"; 198 | ExternalFuncBtn.setAttribute("label", "扩展小功能按钮"); 199 | ExternalFuncBtn.setAttribute("onclick", "event.preventDefault();event.stopPropagation();"); 200 | ExternalFuncBtn.setAttribute("tooltiptext", "扩展小功能按钮,可以自定义小函数功能"); 201 | ExternalFuncBtn.setAttribute("class", "toolbarbutton-1 chromeclass-toolbar-additional"); 202 | ExternalFuncBtn.setAttribute("type", "menu"); 203 | ExternalFuncBtn.setAttribute("removable", "true"); 204 | ExternalFuncBtn.setAttribute("image", ""); 205 | 206 | 207 | var ExternalFuncPopup = document.createElement('menupopup'); 208 | ExternalFuncPopup.setAttribute('onpopupshowing', 'event.stopPropagation();gExternalFuncButtonM.onpopupshowing();'); 209 | this._externalFuncPopup = ExternalFuncPopup; 210 | ExternalFuncBtn.appendChild(ExternalFuncPopup); 211 | setTimeout(function () { //延时加载菜单,不对启动造成影响,也不影响第一次打开菜单时的响应速度 212 | gExternalFuncButtonM.loadSubMenu(); 213 | }, 3000); 214 | document.insertBefore(document.createProcessingInstruction('xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent( 215 | '\ 216 | #ExternalFuncButtonM-ID {\ 217 | -moz-appearance: none !important;\ 218 | border-style: none !important;\ 219 | border-radius: 0 !important;\ 220 | padding: 0 3px !important;\ 221 | margin: 0 !important;\ 222 | background: transparent !important;\ 223 | box-shadow: none !important;\ 224 | -moz-box-align: center !important;\ 225 | -moz-box-pack: center !important;\ 226 | min-width: 18px !important;\ 227 | min-height: 18px !important;\ 228 | }\ 229 | #ExternalFuncButtonM-ID > .toolbarbutton-icon {\ 230 | max-width: 18px !important;\ 231 | padding: 0 !important;\ 232 | margin: 0 !important;\ 233 | }\ 234 | #ExternalFuncButtonM-ID dropmarker{display: none !important;}\ 235 | ') + '"'), document.documentElement); 236 | 237 | if (this.moveablePositonOrInsertafter) { 238 | var navigator = document.getElementById("navigator-toolbox"); 239 | if (!navigator || navigator.palette.id !== "BrowserToolbarPalette") 240 | return; 241 | navigator.palette.appendChild(ExternalFuncBtn); 242 | this.updateToolbar(); 243 | } else { 244 | var navigator = document.getElementById(this.insertafter); 245 | if (!navigator) 246 | return; 247 | navigator.parentNode.insertBefore(ExternalFuncBtn, navigator.previousSibling); 248 | } 249 | }, 250 | loadSubMenu : function () { 251 | if (this._isready) 252 | return; 253 | if (this._externalFuncPopup == null) 254 | return; 255 | var ExternalFuncPopup = this._externalFuncPopup; 256 | for (var i = 0; i < this.toolbar.subdirs.length; i++) { 257 | if (this.toolbar.subdirs[i].name == 'separator') { 258 | ExternalFuncPopup.appendChild(document.createElement('menuseparator')); 259 | } else { 260 | var subDirItem = ExternalFuncPopup.appendChild(document.createElement('menu')); 261 | var subDirItemPopup = subDirItem.appendChild(document.createElement('menupopup')); 262 | subDirItem.setAttribute('class', 'menu-iconic'); 263 | subDirItem.setAttribute('label', this.toolbar.subdirs[i].name); 264 | subDirItem.setAttribute('image', this.toolbar.subdirs[i].image); 265 | gExternalFuncButtonM.subdirPopupHash[this.toolbar.subdirs[i].name] = subDirItemPopup; 266 | gExternalFuncButtonM.subdirMenuHash[this.toolbar.subdirs[i].name] = subDirItem; 267 | } 268 | } 269 | 270 | for (var i = 0; i < this.toolbar.configs.length; i++) { 271 | var configItems; 272 | if (this.toolbar.configs[i].name == 'separator') { 273 | configItems = document.createElement('menuseparator'); 274 | } else { 275 | configItems = ExternalFuncPopup.appendChild(document.createElement('menuitem')); 276 | configItems.setAttribute('class', 'menuitem-iconic'); 277 | configItems.setAttribute('label', this.toolbar.configs[i].name); 278 | configItems.setAttribute('image', this.toolbar.configs[i].image); 279 | if (typeof this.toolbar.configs[i].command == 'function') { 280 | configItems.setAttribute('oncommand', this.unescapeHTML(this.toolbar.configs[i].command.toSource()) + '.call(this, event);'); 281 | } else { 282 | configItems.setAttribute('oncommand', this.toolbar.configs[i].command); 283 | } 284 | configItems.setAttribute('tooltiptext', this.toolbar.configs[i].name); 285 | } 286 | if (this.toolbar.configs[i].subdir && gExternalFuncButtonM.subdirPopupHash[this.toolbar.configs[i].subdir]) 287 | gExternalFuncButtonM.subdirPopupHash[this.toolbar.configs[i].subdir].appendChild(configItems); 288 | else 289 | ExternalFuncPopup.appendChild(configItems); 290 | } 291 | 292 | if (this.autohideEmptySubDirs) { 293 | for (let[name, popup]in Iterator(gExternalFuncButtonM.subdirPopupHash)) { 294 | if (popup.hasChildNodes()) { 295 | continue; 296 | } else { 297 | gExternalFuncButtonM.subdirMenuHash[name].setAttribute("hidden", "true"); 298 | } 299 | } 300 | } 301 | 302 | if (this.moveSubDirstoBottom) { 303 | let i = ExternalFuncPopup.childNodes.length; 304 | while (ExternalFuncPopup.firstChild.getAttribute('class') != 'menuitem-iconic' && i-- != 0) { 305 | ExternalFuncPopup.appendChild(ExternalFuncPopup.firstChild); 306 | } 307 | } 308 | this._isready = true; 309 | }, 310 | onpopupshowing : function () { 311 | if (!this._isready) 312 | this.loadSubMenu(); 313 | }, 314 | updateToolbar: function () { 315 | let toolbars = Array.slice(document.querySelectorAll('#navigator-toolbox > toolbar')); 316 | toolbars.forEach(function (toolbar) { 317 | var currentset = toolbar.getAttribute("currentset"); 318 | if (currentset.split(",").indexOf("ExternalFuncButtonM-ID") < 0) 319 | return; 320 | toolbar.currentSet = currentset; 321 | try { 322 | BrowserToolboxCustomizeDone(true); 323 | } catch (ex) {} 324 | }); 325 | } 326 | }; 327 | 328 | gExternalFuncButtonM.init(); 329 | -------------------------------------------------------------------------------- /qrCreatorOnline.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @include chrome://browser/content/browser.xul 3 | // @name qrCreatorOnline.uc.js 4 | // @version 0.1 5 | // @charset UTF-8 6 | // @author lastdream2013 7 | // ==/UserScript== 8 | var qrCreatorOnline = { 9 | generateAddress: function(target_data) { 10 | return 'http://chart.apis.google.com/chart?cht=qr&chs=350x350&chl=' + encodeURIComponent(target_data); 11 | }, 12 | 13 | targetWindow: null, 14 | 15 | checkLength: function(arg) { 16 | if (arg ) { 17 | if ( arg.length == 0 ) { 18 | alert("没有要转化为二维码的内容!" ); 19 | return false; 20 | } 21 | else if (arg.length > 250) { 22 | alert("要转化为二维码的数据超长了!(大于250字节)" ); 23 | return false; 24 | }else { 25 | return true; 26 | } 27 | } 28 | else { 29 | return false; 30 | } 31 | }, 32 | 33 | popupImage: function(target_data, altText) { 34 | var img_node = content.document.getElementById('qrCreatorOnlineimageboxid'); 35 | if (img_node) { 36 | img_node.parentNode.removeChild(img_node); 37 | } 38 | img_node = this.pinImage(target_data, altText); 39 | 40 | content.document.body.appendChild(img_node); 41 | this.ImgDrag(img_node); 42 | content.document.addEventListener('click', function (e) { 43 | if(img_node && e.button==0 && e.target != img_node) { 44 | img_node.parentNode.removeChild(img_node); 45 | this.removeEventListener("click",arguments.callee,true); 46 | } 47 | }, true); 48 | }, 49 | 50 | pinImage: function(target_data, altText) { 51 | var img_node = new Image(); 52 | img_node.setAttribute('style', '-moz-box-shadow: 0 0 4px #000000'); 53 | with (img_node.style) { 54 | position = 'fixed'; 55 | left = '-moz-calc(50% - 183px)'; 56 | top = '-moz-calc(50% - 183px)'; 57 | zIndex = 99999; 58 | width = "350px"; 59 | height = "350px"; 60 | border = '8px solid rgba(0,0,0,.5)'; 61 | borderRadius = '5px'; 62 | background = 'transparent'; 63 | } 64 | img_node.setAttribute('id', 'qrCreatorOnlineimageboxid'); 65 | img_node.setAttribute('src', this.generateAddress(target_data)); 66 | img_node.setAttribute('alt', altText + ': ' + target_data); 67 | img_node.setAttribute('title', img_node.getAttribute('alt')); 68 | 69 | return img_node; 70 | }, 71 | 72 | onMenuItemCommand: function() { 73 | if(content.document.getElementById('qrCreatorOnlineimageboxid')) return; 74 | var target_data = ''; 75 | var altText = "QR码内容[网址]"; 76 | 77 | if (gContextMenu) { 78 | if (gContextMenu.isTextSelected) { 79 | target_data = content.getSelection().toString(); 80 | altText = "QR码内容[文本]"; 81 | } 82 | else if (gContextMenu.onLink) { 83 | target_data = gContextMenu.linkURL; 84 | } 85 | else if (gContextMenu.onImage) { 86 | target_data = gContextMenu.target.src; 87 | } 88 | else if ( ( content.document.location == "about:blank" || content.document.location == "about:newtab" ) ) { 89 | altText = "QR码内容[文本]"; 90 | target_data = prompt("请输入文本创建一个QR码(长度不超过250字节):", ""); 91 | } 92 | else { 93 | target_data = content.document.location; 94 | } 95 | } 96 | 97 | if (this.checkLength(target_data)) { 98 | this.popupImage(target_data, altText); 99 | } 100 | }, 101 | 102 | init: function() { 103 | this.ContextMenu(); 104 | var menu=document.getElementById("contentAreaContextMenu"); 105 | menu.addEventListener("popupshowing", this.optionsChangeLabel, false); 106 | }, 107 | 108 | ImgDrag: function (node){ 109 | var IsMousedown, LEFT, TOP, img_node = node; 110 | img_node.onmousedown = function(e) { 111 | IsMousedown = true; 112 | e = e||event; 113 | LEFT = e.clientX - img_node.offsetLeft; 114 | TOP = e.clientY - img_node.offsetTop; 115 | return false; 116 | } 117 | 118 | content.document.addEventListener("mousemove", function(e) { 119 | e = e||event; 120 | if (IsMousedown) { 121 | img_node.style.left = e.clientX - LEFT + "px"; 122 | img_node.style.top = e.clientY - TOP + "px"; 123 | } 124 | },false); 125 | 126 | content.document.addEventListener("mouseup", function(){ 127 | IsMousedown = false; 128 | },false); 129 | }, 130 | 131 | ContextMenu : function(){ 132 | var menu=document.getElementById("contentAreaContextMenu"); 133 | var menuItem=document.createElement("menuitem"); 134 | menuItem.setAttribute("id", "qrCreatorOnline.menu"); 135 | menuItem.addEventListener("command", function(){qrCreatorOnline.onMenuItemCommand();}, false); 136 | menuItem.setAttribute("label", "在线生成QR码"); 137 | menuItem.setAttribute("class", "menuitem-iconic"); 138 | menu.setAttribute("accesskey","S"); 139 | menuItem.setAttribute("image", ""); 140 | menu.insertBefore(menuItem, document.getElementById("context-openlinkintab")); 141 | 142 | }, 143 | optionsChangeLabel:function(){ 144 | var labelText; 145 | if (gContextMenu) { 146 | if (gContextMenu.isTextSelected) { 147 | labelText = "选区文本"; 148 | } 149 | else if (gContextMenu.onLink) { 150 | labelText = "链接地址"; 151 | } 152 | else if (gContextMenu.onImage) { 153 | labelText = "图象地址"; 154 | } 155 | else if ( content.document.location == "about:blank" || content.document.location == "about:newtab" ) { 156 | labelText = "手工输入"; 157 | } 158 | else { 159 | labelText = "当前网址"; 160 | } 161 | //Services.console.logStringMessage('[ content.document.location optionsChangeLabel ]: ' + content.document.location ); 162 | var currentEntry=document.getElementById("qrCreatorOnline.menu"); 163 | if(currentEntry){ let LABELTEXT = "在线生成QR码 : " + labelText ; currentEntry.setAttribute("label", LABELTEXT);} 164 | } 165 | }, 166 | }; 167 | 168 | qrCreatorOnline.init(); 169 | 170 | 171 | -------------------------------------------------------------------------------- /qrReaderOnline.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @include chrome://browser/content/browser.xul 3 | // @name qrReaderOnline.uc.js 4 | // @version 0.15 5 | // @charset UTF-8 6 | // @note version 20130627 7 | // @author lastdream2013 8 | // ==/UserScript== 9 | 10 | var qrReaderOnline={ 11 | unescapeHTML:function ( input ) { 12 | return String(input) 13 | .replace(/&/g, '&') 14 | .replace(/"/g, '"') 15 | .replace(/</g, '<') 16 | .replace(/>/g, '>') 17 | .replace(/'/g, '\''); 18 | }, 19 | 20 | initialize:function(){ 21 | var menu=document.getElementById("contentAreaContextMenu"); 22 | menu.addEventListener("popupshowing", qrReaderOnline.optionsShowHide, false); 23 | qrReaderOnline.createMenu(); 24 | }, 25 | getClickHandler:function(){ 26 | var url=qrReaderOnline.imgURL; 27 | let req = new XMLHttpRequest(); 28 | req.open("GET", 'http://zxing.org/w/decode?u=' + encodeURIComponent(url), true); 29 | req.send(null); 30 | req.onload = function () { 31 | if (req.status == 200) { 32 | var title = req.responseText.match(/(.+)<\/title>/i)[1]; 33 | if (title =="Decode Succeeded") { 34 | //Services.console.logStringMessage('[ req.responseText ]: ' + req.responseText ); 35 | try{ 36 | var resultstr = req.responseText.match(/Parsed Result<\/td><td>(.+)<\/td><\/tr><\/table><hr\/>/i)[1]; 37 | resultstr = resultstr.match(/<pre[^>]+>(.*)<\/pre>/i)[1]; 38 | resultstr = resultstr.replace(/<br\/>/ig, '\n'); 39 | qrReaderOnline.read(qrReaderOnline.unescapeHTML(resultstr)); 40 | } 41 | catch(e){ 42 | alert(e); 43 | } 44 | } 45 | else{ 46 | alert("解析失败:" + title); 47 | } 48 | } 49 | } 50 | }, 51 | 52 | read:function(d){ 53 | var p=/^(https?|ftp|file):\/\/[-_.~*'()|a-zA-Z0-9;:\/?,@&=+$%#]*[-_.~*)|a-zA-Z0-9;:\/?@&=+$%#]$/; 54 | if(p.test(d)){ 55 | var f=confirm("QR码为网络地址,确认打开?"+'\n\n'+d); 56 | if(f==true){ 57 | gBrowser.loadOneTab(d, null, null, null, true, false); 58 | } 59 | } 60 | else{ 61 | var r=confirm("QR代码值(按OK键将其复制到剪贴板):"+'\n\n'+d); 62 | if(r==true){ 63 | try{ 64 | Components.classes["@mozilla.org/widget/clipboardhelper;1"].getService(Components.interfaces.nsIClipboardHelper).copyString(d); 65 | } 66 | catch(e){ 67 | alert(e); 68 | } 69 | } 70 | } 71 | }, 72 | createMenu:function(){ 73 | var menu=document.getElementById("contentAreaContextMenu"); 74 | var menuItem=document.createElement("menuitem"); 75 | menuItem.setAttribute("id", "qrReaderOnline.menu"); 76 | menuItem.addEventListener("command", function(){qrReaderOnline.getClickHandler();}, false); 77 | menuItem.setAttribute("label", "在线解析图像QR码"); 78 | menuItem.setAttribute("class", "menuitem-iconic"); 79 | menuItem.setAttribute("image", ""); 80 | menu.insertBefore(menuItem, document.getElementById("context-openlinkintab")); 81 | 82 | }, 83 | optionsShowHide:function(){ 84 | if(gContextMenu){ 85 | var isViewableImage=false; 86 | qrReaderOnline.imgTarget=gContextMenu.target; 87 | var p=/^(https?|ftp):\/\/[-_.~*'()|a-zA-Z0-9;:\/?,@&=+$%#]*[-_.~*)|a-zA-Z0-9;:\/?@&=+$%#]$/; 88 | if(gContextMenu.onImage && ( p.test(gContextMenu.imageURL)) ){ 89 | // if(gContextMenu.onImage ){ 90 | isViewableImage=true; 91 | qrReaderOnline.imgURL=gContextMenu.imageURL; 92 | if(gContextMenu.onLink){qrReaderOnline.linkURL=gContextMenu.linkURL;} 93 | else{qrReaderOnline.linkURL='';} 94 | } 95 | var currentEntry=document.getElementById("qrReaderOnline.menu"); 96 | if(currentEntry){currentEntry.hidden=!isViewableImage;} 97 | 98 | } 99 | }, 100 | fun_OpenSite:function(url){ 101 | gBrowser.loadOneTab(url, null, null, null, true, false); 102 | }, 103 | }; 104 | 105 | qrReaderOnline.initialize(); 106 | 107 | 108 | -------------------------------------------------------------------------------- /quickProxyModokiMod.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name quickProxyModokiMod.uc.js 3 | // @namespace http://space.geocities.yahoo.co.jp/gl/alice0775 4 | // @description 快速代理设置 5 | // @include main 6 | // @compatibility Firefox 3.0 3.5 3.6 3.7a1pre WinXP 7 | // @charset UTF-8 8 | // @author Alice0775 9 | // @note fix compatibility for firefox23a1 10 | // @version 2012/01/31 11:00 by Alice0775 12.0a1 about:newtab 11 | // @version 2010/02/05 00:00 ウインドウが閉じてしまわないように 12 | // @version 2009/09/02 13:00 quickproxy.typeをセットしないで使おうとする方のため直前のproxy.typeをquickproxy.typeにするようにした。 13 | // @note 左クリック:オンオフ. 中クリック:起動時の設定 quickproxy.autooff, 右クリック:プロキシの設定 14 | // @note about:configのquickproxy.typeに値を予めセットしておく1(指定)または2(PAC) 15 | // @note 一発プロキシcontextProxySwitch.uc.xulを使うとより便利かも 16 | // ==/UserScript== 17 | 18 | (function (css) { 19 | if (window.quickProxy) { 20 | window.quickProxy.destroy(); 21 | delete window.quickProxy; 22 | } 23 | 24 | var quickProxy = { 25 | //-- config -- 26 | // 以下はFirefox3以上の場合に有効 27 | // [true]:オプションをタブで開く, false:ダイアログで開く 28 | optionInTab: true, 29 | // falseの時ダイアログを開く遅延 巧く開かないときは遅延を大きくする 30 | optionDelay: 800, 31 | //-- config -- 32 | _init: function(){ 33 | var overlay = '\ 34 | <overlay xmlns="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul" \ 35 | xmlns:html="http://www.w3.org/1999/xhtml"> \ 36 | <toolbarpalette id="urlbar-icons">\ 37 | <image id="quickproxy-status" label="quickproxySwitch" \ 38 | onclick="quickProxy._click(event);" \ 39 | tooltiptext="" >\ 40 | </image>\ 41 | </toolbarpalette>\ 42 | </overlay>'; 43 | overlay = "data:application/vnd.mozilla.xul+xml;charset=utf-8," + encodeURI(overlay); 44 | window.userChrome_js.loadOverlay(overlay, quickProxy); 45 | addStyle(css); 46 | 47 | quickProxy.addPrefListener(quickProxy.buttonPrefListener); // 登録処理 48 | window.addEventListener('unload',function(){ 49 | quickProxy.removePrefListener(quickProxy.buttonPrefListener); 50 | } 51 | ,false); 52 | }, 53 | 54 | observe: function (subject, topic, data) { 55 | if (topic == "xul-overlay-merged") { 56 | var icon = document.getElementById("quickproxy-status"); 57 | var qp_autooff = quickProxy.getPref('quickproxy.autooff', 'bool', false); 58 | var Is_Proxy_On = quickProxy.getPref("network.proxy.type", 'int', 0); 59 | quickProxy.addPrefListener(quickProxy.buttonPrefListener); // 登録処理 60 | window.addEventListener('unload',function(){ 61 | quickProxy.removePrefListener(quickProxy.buttonPrefListener); 62 | } 63 | ,false); 64 | if (qp_autooff == true && quickProxy.getNumberOfWindow() == 1){ 65 | Is_Proxy_On = 0 66 | } 67 | quickProxy.setPref("network.proxy.type", 'int', Is_Proxy_On); 68 | 69 | quickProxy._updateUI(); 70 | //Application.console.log("quickProxy界面加载完毕!"); 71 | } 72 | }, 73 | 74 | getNumberOfWindow: function(){ 75 | var mediator = Components.classes["@mozilla.org/appshell/window-mediator;1"] 76 | .getService(Components.interfaces.nsIWindowMediator); 77 | var enumerator = mediator.getEnumerator("navigator:browser"); 78 | var num = 0; 79 | while(enumerator.hasMoreElements()) { 80 | var win = enumerator.getNext(); 81 | num++; 82 | } 83 | return num; 84 | }, 85 | 86 | _switch: function() { 87 | var Is_Proxy_On = this.getPref("network.proxy.type", 'int', 0); 88 | var Proxy_Type = this.getPref("quickproxy.type", 'int', 1); 89 | if (Is_Proxy_On == 0){ 90 | Is_Proxy_On = Proxy_Type; 91 | }else{ 92 | Is_Proxy_On = 0; 93 | } 94 | this.setPref("network.proxy.type", 'int', Is_Proxy_On); 95 | this._updateUI(); 96 | }, 97 | 98 | _click: function(e){ 99 | if (e.button == 0){ 100 | this._switch(); 101 | } 102 | if (e.button == 2) { 103 | var aURL = "chrome://browser/content/preferences/connection.xul"; 104 | if (parseInt(Components.classes["@mozilla.org/xre/app-info;1"] 105 | .getService(Components.interfaces.nsIXULAppInfo) 106 | .version.substr(0,3) * 10,10) / 10 < 3) { 107 | //Fx2 108 | var aName =""; 109 | var aFeatures = 'modal,centerscreen,resizable=no'; 110 | var aParams = null; 111 | openDialog(aURL, aName, aFeatures, aParams); 112 | } else { 113 | if (this.optionInTab) { 114 | //Fx3 タブバージョン (タブが一つの時, 閉じるとFxも閉じてしまう) 115 | if ("isBlankPageURL" in window ? isBlankPageURL(gBrowser.currentURI.spec) : gBrowser.currentURI.spec == "about:blank") 116 | gBrowser.loadOneTab(aURL); 117 | else 118 | gBrowser.selectedTab = gBrowser.addTab(aURL); 119 | var Proxy_Type = this.getPref("network.proxy.type", 'int', 0); 120 | if(Proxy_Type > 0) this.setPref("quickproxy.type", 'int', Proxy_Type); 121 | this._updateUI(); 122 | e.preventDefault(); 123 | 124 | setTimeout(function(self){ 125 | if (gBrowser.mTabs.length == 1) 126 | gBrowser.addTab(); 127 | }, 200 , this); 128 | 129 | return; 130 | } else { 131 | //Fx3 ダイアログバージョン 132 | var prefBack = this.getPref("browser.preferences.instantApply", "bool", false); 133 | this.setPref("browser.preferences.instantApply", "bool", true); 134 | var w = openPreferences("paneAdvanced", {"advancedTab":"networkTab"}); 135 | setTimeout(function(self, w, prefBack){ 136 | self.optionOnload(w, prefBack); 137 | }, self.optionDelay, this, w, prefBack); 138 | 139 | var Proxy_Type = this.getPref("network.proxy.type", 'int', 0); 140 | if(Proxy_Type > 0) this.setPref("quickproxy.type", 'int', Proxy_Type); 141 | this._updateUI(); 142 | e.preventDefault(); 143 | return; 144 | } 145 | } 146 | } 147 | if (e.button == 1) { 148 | var qp_autooff = this.getPref('quickproxy.autooff', 'bool', true); 149 | var prompts = Components.classes["@mozilla.org/embedcomp/prompt-service;1"] 150 | .getService(Components.interfaces.nsIPromptService); 151 | var check = {value: qp_autooff}; 152 | prompts.alertCheck(window, 'Window title', '', 153 | "启动时设置为禁用代理", check); //起動時は常にプロキシはオフ 154 | this.setPref('quickproxy.autooff', 'bool', check.value); 155 | } 156 | e.preventDefault(); 157 | }, 158 | 159 | optionOnload: function(w, prefBack){alert(2) 160 | this.setPref("browser.preferences.instantApply", "bool", prefBack); 161 | w.gAdvancedPane.showConnections(); 162 | }, 163 | 164 | _updateUI: function() { 165 | var Is_Proxy_On = this.getPref("network.proxy.type", 'int', 0); 166 | var icon = document.getElementById("quickproxy-status"); 167 | var text1 = ""; 168 | switch (Is_Proxy_On) { 169 | case 0: 170 | case 3: 171 | icon.setAttribute("state", "disable"); 172 | icon.setAttribute("tooltiptext", "左键:开|关代理; 中键:启动设置; 右键:代理设置"); 173 | return; 174 | case 1: 175 | var ip = this.getPref("network.proxy.http", "str", ""); 176 | var port = this.getPref("network.proxy.http_port", "int", 0); 177 | text1 = "\n当前代理:" + ip + ":" + port; 178 | break; 179 | case 2: 180 | text1 = "\n当前状态:自动代理配置(PAC)"; 181 | break; 182 | case 4: 183 | text1 = "\n当前状态:自动检测此网络的代理设置"; 184 | break; 185 | case 5: 186 | text1 = "\n当前状态:使用系统代理设置"; 187 | break; 188 | } 189 | icon.setAttribute("state", "enable"); 190 | icon.setAttribute("tooltiptext", "左键:开|关代理; 中键:启动设置; 右键:代理设置" + text1); 191 | }, 192 | 193 | getPref: function(aPrefString, aPrefType, aDefault){ 194 | var xpPref = Components.classes["@mozilla.org/preferences-service;1"] 195 | .getService(Components.interfaces.nsIPrefBranch); 196 | try{ 197 | switch (aPrefType){ 198 | case "str": 199 | return xpPref.getCharPref(aPrefString).toString(); break; 200 | case "int": 201 | return xpPref.getIntPref(aPrefString); break; 202 | case "bool": 203 | default: 204 | return xpPref.getBoolPref(aPrefString); break; 205 | } 206 | }catch(e){ 207 | } 208 | return aDefault; 209 | }, 210 | 211 | setPref: function(aPrefString, aPrefType, aValue){ 212 | var xpPref = Components.classes["@mozilla.org/preferences-service;1"] 213 | .getService(Components.interfaces.nsIPrefBranch); 214 | try{ 215 | switch (aPrefType){ 216 | case "str": 217 | return xpPref.setCharPref(aPrefString, aValue); break; 218 | case "int": 219 | aValue = parseInt(aValue); 220 | return xpPref.setIntPref(aPrefString, aValue); break; 221 | case "bool": 222 | default: 223 | return xpPref.setBoolPref(aPrefString, aValue); break; 224 | } 225 | }catch(e){ 226 | } 227 | return null; 228 | }, 229 | // 監視を開始する 230 | addPrefListener: function(aObserver) { 231 | try { 232 | var pbi = Components.classes['@mozilla.org/preferences;1'].getService(Components.interfaces.nsIPrefBranch2); 233 | pbi.addObserver(aObserver.domain, aObserver, false); 234 | } catch(e) {} 235 | }, 236 | 237 | // 監視を終了する 238 | removePrefListener: function(aObserver) { 239 | try { 240 | var pbi = Components.classes['@mozilla.org/preferences;1'].getService(Components.interfaces.nsIPrefBranch2); 241 | pbi.removeObserver(aObserver.domain, aObserver); 242 | } catch(e) {} 243 | }, 244 | 245 | buttonPrefListener:{ 246 | domain : 'network.proxy.type', 247 | //"grabScroll.XXX"という名前の設定が変更された場合全てで処理を行う 248 | 249 | observe : function(aSubject, aTopic, aPrefstring) { 250 | if (aTopic == 'nsPref:changed') { 251 | // 設定が変更された時の処理 252 | var type = quickProxy.getPref(aPrefstring, 'int', 0); 253 | if (type != 0) 254 | quickProxy.setPref("quickproxy.type", 'int', type); 255 | quickProxy._updateUI(); 256 | } 257 | } 258 | } 259 | } 260 | quickProxy._init(); 261 | window.quickProxy = quickProxy; 262 | 263 | function addStyle(css) { 264 | var pi = document.createProcessingInstruction( 265 | 'xml-stylesheet', 266 | 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(css) + '"' 267 | ); 268 | return document.insertBefore(pi, document.documentElement); 269 | } 270 | 271 | })('\ 272 | #quickproxy-status {\ 273 | list-style-image: url("");\ 274 | padding: 0 2px !important;\ 275 | }\ 276 | \ 277 | #quickproxy-status[state="disable"] {\ 278 | filter: url("chrome://mozapps/skin/extensions/extensions.svg#greyscale");\ 279 | }\ 280 | '); 281 | -------------------------------------------------------------------------------- /sendToGmail.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name sendToGmail.uc.js 3 | // @namespace ithinc#mozine.cn 4 | // @description Send the selection to Gmail 5 | // @include main 6 | // @compatibility Firefox 3.x 7 | // @author ithinc 8 | // @version LastMod 2009/3/15 20:30 Improvment on sending images 9 | // @version LastMod 2009/3/14 20:30 Add support for all Gmail display languages 10 | // @version LastMod 2009/3/13 21:30 Initial release 11 | // @Note null 12 | // ==/UserScript== 13 | 14 | /* :::: Send the selection to Gmail :::: */ 15 | 16 | function SendToGmail(from, to, cc, bcc, subject, body) { 17 | var gmailComposeURL = 'https://mail.google.com/mail/?ui=html&v=b&pv=tl&cs=b'; 18 | 19 | var req = new XMLHttpRequest(); 20 | req.open('GET', gmailComposeURL, true); 21 | req.onreadystatechange = function() { 22 | if (req.readyState == 4) { 23 | if (req.status == 200 && req.responseText.match(/<form[\s]+action="[^>"]+"[\s]+name=(f|"f")[^>]*>/i)) { 24 | if (!to && /([\w.]+)@gmail.com/.test(req.responseText)) to = RegExp.$1 + "+note@gmail.com"; 25 | 26 | var gmailSendURL = req.responseText.match(/<base[^>]*>/i).toString().split(/"/)[1] + req.responseText.match(/<form[\s]+action="[^>"]+"[\s]+name=(f|"f")[^>]*>/i).toString().split(/"/)[1]; 27 | var gmailSendParams = 'redir=?&nvp_bu_send=Send'; 28 | //gmailSendParams += '&from=' + encodeURIComponent(from); 29 | gmailSendParams += '&to=' + encodeURIComponent(to); 30 | gmailSendParams += '&cc=' + encodeURIComponent(cc); 31 | gmailSendParams += '&bcc=' + encodeURIComponent(bcc); 32 | gmailSendParams += '&subject=' + encodeURIComponent(subject); 33 | gmailSendParams += '&body=' + encodeURIComponent(body); 34 | 35 | var req2 = new XMLHttpRequest(); 36 | req2.open('POST', gmailSendURL, true); 37 | req2.setRequestHeader("Content-Type", "application/x-www-form-urlencoded"); 38 | req2.onreadystatechange = function() { 39 | if (req2.readyState == 4) { 40 | if (req2.status == 200) { 41 | // document.getElementById("statusbar-display").label = //req2.responseText.match(/<td[\s]+bgcolor="?#(f|F)(a|A)(d|D)163"?>[^<>]*<b>[^<>]*<\/b>/i).toString().split(/<|>/i)[4]; 42 | var alertsService=Components.classes["@mozilla.org/alerts-service;1"].getService(Components.interfaces.nsIAlertsService); 43 | alertsService.showAlertNotification("chrome://global/skin/icons/information-32.png","SentToGmail",req2.responseText.match(/<td[\s]+bgcolor="?#(f|F)(a|A)(d|D)163"?>[^<>]*<b>[^<>]*<\/b>/i).toString().split(/<|>/i)[4],false,"",null,""); 44 | } 45 | else { 46 | alert("\u6D88\u606F\u53D1\u9001\u4E0D\u6210\u529F!"); 47 | } 48 | } 49 | } 50 | req2.send(gmailSendParams); 51 | } 52 | else { 53 | alert("\u60A8\u6CA1\u6709\u767B\u9646\u4F60\u7684Gmail!"); 54 | } 55 | } 56 | } 57 | req.send(null); 58 | } 59 | 60 | (function() { 61 | var mainContextMenu = document.getElementById("contentAreaContextMenu"); 62 | 63 | var menuitem = mainContextMenu.insertBefore(document.createElement("menuitem"), document.getElementById("context-sendpage")); 64 | menuitem.setAttribute("id", "context-sendpagetogmail"); 65 | menuitem.setAttribute("label", "\u53D1\u9001\u5F53\u524D\u7F51\u5740\u5230Gmail"); 66 | menuitem.setAttribute("accesskey", "G"); 67 | menuitem.setAttribute("oncommand", "SendToGmail('', '', '', '', gBrowser.selectedTab.label, gBrowser.currentURI.spec);"); 68 | 69 | var menuitem = mainContextMenu.insertBefore(document.createElement("menuitem"), document.getElementById("context-sendlink")); 70 | menuitem.setAttribute("id", "context-sendlinktogmail"); 71 | menuitem.setAttribute("label", "\u53D1\u9001\u94FE\u63A5\u5730\u5740\u5230Gmail"); 72 | menuitem.setAttribute("accesskey", "G"); 73 | menuitem.setAttribute("oncommand", "SendToGmail('', '', '', '', gContextMenu.linkText(), gContextMenu.linkURL);"); 74 | 75 | var menuitem = mainContextMenu.insertBefore(document.createElement("menuitem"), document.getElementById("context-sendimage")); 76 | menuitem.setAttribute("id", "context-sendimagetogmail"); 77 | menuitem.setAttribute("label", "\u53D1\u9001\u56FE\u8C61\u5730\u5740\u5230Gmail"); 78 | menuitem.setAttribute("accesskey", "G"); 79 | menuitem.setAttribute("oncommand", "SendToGmail('', '', '', '', gContextMenu.target.getAttribute('title') || gContextMenu.target.getAttribute('alt') || gContextMenu.target.currentURI.spec, gBrowser.currentURI.spec + '\\n\\n' + gContextMenu.target.currentURI.spec);"); 80 | 81 | var menuitem = mainContextMenu.insertBefore(document.createElement("menuitem"), document.getElementById("frame-sep")); 82 | menuitem.setAttribute("id", "context-sendselecttogmail"); 83 | menuitem.setAttribute("label", "\u53D1\u9001 ... \u5230Gmail"); 84 | menuitem.setAttribute("accesskey", "G"); 85 | menuitem.setAttribute("oncommand", "SendToGmail('', '', '', '', gBrowser.selectedTab.label, gBrowser.currentURI.spec + '\\n\\n' + document.commandDispatcher.focusedWindow.getSelection());"); 86 | 87 | mainContextMenu.addEventListener("popupshowing", function(e) { 88 | document.getElementById("context-sendpagetogmail").setAttribute("hidden", "true"); 89 | document.getElementById("context-sendlinktogmail").setAttribute("hidden", "true"); 90 | document.getElementById("context-sendimagetogmail").setAttribute("hidden", "true"); 91 | document.getElementById("context-sendselecttogmail").setAttribute("hidden", "true"); 92 | 93 | if (gContextMenu.onLink) { 94 | document.getElementById("context-sendlinktogmail").removeAttribute("hidden"); 95 | } 96 | 97 | if (gContextMenu.onImage) { 98 | document.getElementById("context-sendimagetogmail").removeAttribute("hidden"); 99 | } 100 | 101 | if (gContextMenu.isTextSelected) { 102 | var selection = getBrowserSelection(16); 103 | if (selection && selection.length > 15) 104 | selection = selection.substr(0,15) + "..."; 105 | 106 | document.getElementById("context-sendselecttogmail").setAttribute("label", '\u53D1\u9001 "' + selection + '" \u5230Gmail'); 107 | document.getElementById("context-sendselecttogmail").removeAttribute("hidden"); 108 | } 109 | 110 | if (!gContextMenu.onLink && !gContextMenu.onImage && !gContextMenu.isContentSelected && !gContextMenu.onTextInput) { 111 | document.getElementById("context-sendpagetogmail").removeAttribute("hidden"); 112 | } 113 | }, false); 114 | })(); 115 | -------------------------------------------------------------------------------- /showLocationModEx.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name showLocationModEx.uc.js 3 | // @charset UTF-8 4 | // @description 显示国旗与IP 5 | // @include chrome://browser/content/browser.xul 6 | // @author 紫云飞 7 | // @note version20130719: mod by lastdream2013 8 | // ==/UserScript== 9 | 10 | (function(){ 11 | //改这里选择是否加载本地国旗图标库,不存在或路径错误自动切换从网络中读国旗图标 12 | var localFlagPath = "lib\\countryflags.js"; // 注意是相对路径: profile\chrome\lib\countryflags.js 13 | //改这里选择显示国旗图标/IP位置,如果是identity-box为地址栏前端显示,会自动加载隐藏page-proxy-favicon css配合显示效果 14 | var showLocationPos = "identity-box"; // urlbar-icons identity-box addon-bar status-bar 等等 15 | 16 | //下面的不知道不要动 17 | var IsUserLocalFlag = false; 18 | localFlagPath.path = localFlagPath.replace(/\//g, '\\'); 19 | var FullPath = Cc["@mozilla.org/file/directory_service;1"].getService(Ci.nsIProperties).get("UChrm", Ci.nsILocalFile).path; 20 | if (/^(\\)/.test(localFlagPath)) { 21 | FullPath = FullPath +localFlagPath; 22 | } 23 | else{ 24 | FullPath = FullPath + "\\" + localFlagPath; 25 | } 26 | 27 | var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsILocalFile); 28 | file.initWithPath(FullPath); 29 | if (file.exists()) { 30 | IsUserLocalFlag = true; 31 | userChrome.import(localFlagPath, "UChrm"); 32 | } 33 | 34 | location == "chrome://browser/content/browser.xul" && gBrowser.addEventListener("DOMWindowCreated", function (event) { 35 | 36 | if ( showLocationPos == "identity-box" ){ 37 | var cssStr =('\ 38 | #page-proxy-favicon,\ 39 | #identity-icon-label,\ 40 | #identity-icon-country-label {\ 41 | visibility: collapse !important;\ 42 | }\ 43 | #urlbar {height:24px !important;}\ 44 | '); 45 | var style = document.createProcessingInstruction('xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(cssStr) + '"'); 46 | document.insertBefore(style, document.documentElement); 47 | } 48 | 49 | var self = arguments.callee; 50 | if (!self.showLocation) { 51 | window.addEventListener("TabSelect", self, false); 52 | self.showLocation = document.getElementById(showLocationPos); 53 | 54 | self.showFlag = self.showLocation.appendChild(document.createElement("image")); 55 | //单击时复制 56 | self.showFlag.addEventListener("click", function () { 57 | Components.classes['@mozilla.org/widget/clipboardhelper;1'].createInstance(Components.interfaces.nsIClipboardHelper).copyString(self.showFlag.tooltipText); 58 | }, false); 59 | 60 | //self.showFlag.style.width = "16px"; 61 | //self.showFlag.style.Maxheight = "16px"; 62 | if ( showLocationPos == "identity-box" ){ 63 | self.showFlag.style.marginLeft = "4px"; 64 | self.showFlag.style.marginRight = "2px"; 65 | } 66 | 67 | //设置等待时国旗图标 68 | self.showFlag.src = self.flag = ""; 69 | self.isReqLocationHash = []; 70 | self.isReqFlagHash = []; 71 | self.showFlagTooltipHash = []; 72 | self.showFlagHash = []; 73 | self.flagPath = 'http://www.razerzone.com/asset/images/icons/flags/' //备用:self.flagPath = 'http://www.1108.hk/images/ext/' 74 | } 75 | try { 76 | var host = (event.originalTarget.location || content.location).hostname; 77 | if (!/tp/.test(content.location.protocol)) { 78 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 79 | return 80 | } 81 | var ip = Components.classes["@mozilla.org/network/dns-service;1"].getService(Components.interfaces.nsIDNSService).resolve(host, 0).getNextAddrAsString(); 82 | var server = (gBrowser.mCurrentBrowser.webNavigation.currentDocumentChannel.QueryInterface(Components.interfaces.nsIHttpChannel).getResponseHeader("server").match(/\w+/) || ["\u672A\u77E5"])[0]; 83 | 84 | if (!self.showFlagTooltipHash[host]) { 85 | (event.type == "TabSelect" || event.originalTarget == content.document); 86 | self.isReqLocationHash[host] = true; 87 | let req = new XMLHttpRequest(); 88 | req.open("GET", 'http://www.cz88.net/ip/index.aspx?ip=' + ip, true); 89 | req.send(null); 90 | req.onload = function () { 91 | if (req.status == 200) { 92 | var lmsg = req.responseText.match(/"InputIPAddrMessage">([^<]+)/); 93 | if ( lmsg ) 94 | var location = lmsg[1].replace(/\s*CZ88.NET.*/, "") + " \n"; 95 | else 96 | var location = ""; 97 | //设置文字提示内容 98 | self.showFlagTooltipHash[host] = location + server + " \n" + ip; 99 | host == content.location.hostname && (self.showFlag.tooltipText = self.showFlagTooltipHash[host]); 100 | } 101 | self.isReqLocationHash[host] = false; 102 | } 103 | } else { 104 | host == content.location.hostname && (self.showFlag.tooltipText = self.showFlagTooltipHash[host]); 105 | } 106 | 107 | if (!self.showFlagHash[host]) { 108 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 109 | self.isReqFlagHash[host] = true; 110 | let req = new XMLHttpRequest(); 111 | req.open("GET", 'http://freegeoip.net/json/' + ip, true); 112 | req.send(null); 113 | req.onload = function () { 114 | if (req.status == 200) { 115 | var responseObj =JSON.parse(req.responseText); 116 | self.showFlagHash[host] =responseObj.country_code.toLocaleLowerCase(); 117 | 118 | host == content.location.hostname; 119 | if (IsUserLocalFlag) { self.showFlag.src = CountryFlags[self.showFlagHash[host]]; } 120 | else { self.showFlag.src = self.flagPath + self.showFlagHash[host] + ".gif"; } 121 | } 122 | self.isReqFlagHash[host] = false; 123 | } 124 | } else { 125 | host == content.location.hostname; 126 | if (IsUserLocalFlag) { self.showFlag.src = CountryFlags[self.showFlagHash[host]]; } 127 | else { self.showFlag.src = self.flagPath + self.showFlagHash[host] + ".gif"; } 128 | } 129 | } catch (e) { 130 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 131 | } 132 | }, false) 133 | })(); 134 | -------------------------------------------------------------------------------- /showLocationModSpecial.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name showLocationModSpecial.uc.js 3 | // @charset UTF-8 4 | // @description 只显示国旗在前端,不过和https时的前面的图标不冲突 5 | // @include chrome://browser/content/browser.xul 6 | // @author 紫云飞 7 | // @note version20130719: mod by lastdream2013 8 | // ==/UserScript== 9 | 10 | (function(){ 11 | //改这里选择是否加载本地国旗图标库,不存在或路径错误自动切换从网络中读国旗图标 12 | var localFlagPath = "lib\\countryflags.js"; // 注意是相对路径: profile\chrome\lib\countryflags.js 13 | 14 | //下面的不知道不要动 15 | var IsUserLocalFlag = false; 16 | localFlagPath.path = localFlagPath.replace(/\//g, '\\'); 17 | var FullPath = Cc["@mozilla.org/file/directory_service;1"].getService(Ci.nsIProperties).get("UChrm", Ci.nsILocalFile).path; 18 | if (/^(\\)/.test(localFlagPath)) { 19 | FullPath = FullPath +localFlagPath; 20 | } 21 | else{ 22 | FullPath = FullPath + "\\" + localFlagPath; 23 | } 24 | 25 | var file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsILocalFile); 26 | file.initWithPath(FullPath); 27 | if (file.exists()) { 28 | IsUserLocalFlag = true; 29 | userChrome.import(localFlagPath, "UChrm"); 30 | } 31 | 32 | location == "chrome://browser/content/browser.xul" && gBrowser.addEventListener("DOMWindowCreated", function (event) { 33 | var cssStr = ('\ 34 | #urlbar {height:24px !important;}\ 35 | '); 36 | var style = document.createProcessingInstruction('xml-stylesheet', 'type="text/css" href="data:text/css;utf-8,' + encodeURIComponent(cssStr) + '"'); 37 | document.insertBefore(style, document.documentElement); 38 | var self = arguments.callee; 39 | if (!self.showLocation) { 40 | window.addEventListener("TabSelect", self, false); 41 | self.showLocation = document.getElementById("identity-box"); 42 | 43 | self.showFlag = self.showLocation.appendChild(document.createElement("image")); 44 | self.showFlag.setAttribute("hidden", "true"); 45 | 46 | self.showFlag.style.marginLeft = "4px"; 47 | self.showFlag.style.marginRight = "2px"; 48 | 49 | //设置等待时国旗图标 50 | self.showFlag.src = self.flag = ""; 51 | self.isReqFlagHash = []; 52 | self.showFlagHash = []; 53 | self.flagPath = 'http://www.razerzone.com/asset/images/icons/flags/' //备用:self.flagPath = 'http://www.1108.hk/images/ext/' 54 | } 55 | 56 | try { 57 | var host = (event.originalTarget.location || content.location).hostname; 58 | if (!/tp/.test(content.location.protocol) ) { 59 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 60 | gProxyFavIcon.removeAttribute("src"); 61 | self.showFlag.setAttribute("hidden", "true"); 62 | return 63 | } 64 | var ip = Components.classes["@mozilla.org/network/dns-service;1"].getService(Components.interfaces.nsIDNSService).resolve(host, 0).getNextAddrAsString(); 65 | var server = (gBrowser.mCurrentBrowser.webNavigation.currentDocumentChannel.QueryInterface(Components.interfaces.nsIHttpChannel).getResponseHeader("server").match(/\w+/) || ["\u672A\u77E5"])[0]; 66 | 67 | if (!self.showFlagHash[host]) { 68 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 69 | self.isReqFlagHash[host] = true; 70 | let req = new XMLHttpRequest(); 71 | req.open("GET", 'http://freegeoip.net/json/' + ip, true); 72 | req.send(null); 73 | req.onload = function () { 74 | if (req.status == 200) { 75 | //self.showFlagHash[host] = (req.responseText.match(/"country_code": "([^"]+)/) || ["", "CN"])[1].toLocaleLowerCase(); 76 | var responseObj =JSON.parse(req.responseText); 77 | self.showFlagHash[host] =responseObj.country_code.toLocaleLowerCase(); 78 | host == content.location.hostname; 79 | if (IsUserLocalFlag) { self.showFlag.src = CountryFlags[self.showFlagHash[host]]; } 80 | else { self.showFlag.src = self.flagPath + self.showFlagHash[host] + ".gif"; } 81 | 82 | if ( gBrowser.currentURI.spec.indexOf("https://") >= 0 ) { 83 | gProxyFavIcon.removeAttribute("src"); 84 | self.showFlag.removeAttribute("hidden"); 85 | } 86 | else{ 87 | gProxyFavIcon.src = self.showFlag.src; 88 | self.showFlag.setAttribute("hidden", "true"); 89 | } 90 | } 91 | self.isReqFlagHash[host] = false; 92 | } 93 | } else { 94 | host == content.location.hostname; 95 | if (IsUserLocalFlag) { self.showFlag.src = CountryFlags[self.showFlagHash[host]]; } 96 | else { self.showFlag.src = self.flagPath + self.showFlagHash[host] + ".gif"; } 97 | if ( gBrowser.currentURI.spec.indexOf("https://") >= 0 ) { 98 | gProxyFavIcon.removeAttribute("src"); 99 | self.showFlag.removeAttribute("hidden"); 100 | } 101 | else{ 102 | gProxyFavIcon.src = self.showFlag.src; 103 | self.showFlag.setAttribute("hidden", "true"); 104 | } 105 | } 106 | 107 | } catch (e) { 108 | (event.type == "TabSelect" || event.originalTarget == content.document) && (self.showFlag.src = self.flag); 109 | } 110 | }, false) 111 | })(); 112 | -------------------------------------------------------------------------------- /sideBookmarkBarMod.uc.js: -------------------------------------------------------------------------------- 1 | // ==UserScript== 2 | // @name sideBookmarkBarMod.uc.js 3 | // @namespace http://www.slimeden.com 4 | // @author Xiao Shan 5 | // @note test version 20130515: modify by lastdream2013 6 | // @note fix combility for firefox24a1 and windows xp system 7 | // @description Customize bookmarkbar to left side, arrange bookmarks vertically 8 | // @version 0.4.1 - 20110218 9 | // ==/UserScript== 10 | (function () { 11 | var sideBookmarkBar = document.createElement("vbox"); 12 | sideBookmarkBar.setAttribute("id", "sideBookmarkBar"); 13 | sideBookmarkBar.setAttribute("minwidth", "20px"); 14 | sideBookmarkBar.setAttribute("maxwidth", "20px"); 15 | var bookmarks = document.getElementById("personal-bookmarks"); 16 | 17 | // get personal-bookmarks from BrowserToolbarPalette 18 | if(!bookmarks) { 19 | var templateNode = gNavToolbox.palette.firstChild; 20 | while (templateNode) { 21 | if (templateNode.id == "personal-bookmarks") { 22 | break; 23 | } else { 24 | templateNode = templateNode.nextSibling; 25 | } 26 | } 27 | 28 | if (templateNode && templateNode.id == "personal-bookmarks") { 29 | bookmarks = templateNode; 30 | } else { 31 | return; 32 | } 33 | } 34 | bookmarks.setAttribute("orient", "vertical"); 35 | sideBookmarkBar.appendChild(bookmarks); 36 | 37 | document.getElementById("browser").insertBefore(sideBookmarkBar, document.getElementById("appcontent")); 38 | 39 | // put the BMB in the bottom of the sidebar 40 | var BMB = document.getElementById("bookmarks-menu-button"); 41 | 42 | if (BMB){ 43 | bookmarks.insertBefore(BMB, bookmarks.childNodes[0]); 44 | 45 | var BMBobserver = new MutationObserver(function(mutations) { 46 | for (let m of mutations) { 47 | // stop BMB button removed from bookmarks 48 | for (let e of m.removedNodes) { 49 | // Application.console.log("E.id: " + e.id); 50 | if ( e.id != BMB.id ) continue; 51 | if(e.parentNode && e.parentNode != bookmarks) { 52 | bookmarks.insertBefore(e, bookmarks.childNodes[0]); 53 | } 54 | } 55 | } 56 | }); 57 | BMBobserver.observe(bookmarks, { childList: true } ); 58 | } 59 | 60 | var menupopupobserver = new MutationObserver(function(mutations) { 61 | for (let m of mutations) { 62 | // open bookmark folder right sideby of sidebookmarkbar 63 | for (let a of m.addedNodes) { 64 | if("toolbarbutton" == a.localName && "bookmark-item" == a.className && "true" == a.getAttribute("container")) { 65 | if (a.firstChild && a.firstChild.localName == "menupopup") { 66 | a.firstChild.setAttribute("position", "end_before"); 67 | } 68 | } 69 | } 70 | } 71 | }); 72 | menupopupobserver.observe(bookmarks, { childList: true, subtree: true } ); 73 | 74 | // make the bookmark items arranged vertically 75 | var bookmarkItems = document.getElementById("PlacesToolbarItems"); 76 | bookmarkItems.setAttribute("orient", "vertical"); 77 | bookmarkItems.parentNode.setAttribute("orient", "vertical"); 78 | bookmarkItems.setAttribute("style", "padding: 0px 0px 0px 0px !important;"); 79 | 80 | var placesToolbar = document.getElementById("PlacesToolbar"); 81 | placesToolbar.setAttribute("orient", "vertical"); 82 | 83 | bookmarks.appendChild(placesToolbar); 84 | PlacesToolbarHelper.init(); 85 | if ( typeof(BookmarksMenuButton) != 'undefined' ) BookmarksMenuButton.customizeDone(); 86 | 87 | // make the drop indicator horizontal 88 | try { 89 | eval("PlacesToolbar.prototype._onDragOver = " + PlacesToolbar.prototype._onDragOver.toString().replace(/ind\.clientWidth/g,'ind.clientHeight').replace(/left/g,'top').replace(/right/g,'bottom').replace(/translateX/g,'translateY').replace(/translate\(/,'matrix(0,1,-1,0, 40px,')); 90 | }catch(e) {} 91 | 92 | // enable bookmark folder tooltip 93 | try{ 94 | eval("BookmarksEventHandler.fillInBHTooltip = " + BookmarksEventHandler.fillInBHTooltip.toString().replace(/(!cropped) (&& !url)/, '$1 && aDocument.tooltipNode.localName == "treechildren" $2')); 95 | }catch(e) {} 96 | 97 | //css 98 | document.insertBefore(document.createProcessingInstruction('xml-stylesheet','type="text/css" href="data:text/css;utf-8,' + encodeURIComponent( 99 | '\ 100 | #bookmarks-menu-button .toolbarbutton-text { display:none !important; }\ 101 | #personal-bookmarks .toolbarbutton-text {display: none !important;}\ 102 | #personal-bookmarks .toolbarbutton-icon {display: -moz-box !important; margin:0px 1px 0px 0px!important;}\ 103 | #PlacesToolbarItems toolbarseparator {\ 104 | -moz-appearance:none !important;\ 105 | min-height: 1px !important;\ 106 | max-height: 1px !important;\ 107 | background: gray !important;\ 108 | margin:0px 1px 0px 0px!important;\ 109 | }\ 110 | #bookmarks-menu-button .toolbarbutton-menubutton-button {\ 111 | -moz-box-orient: vertical !important;\ 112 | margin:0px 1px 0px 0px!important;\ 113 | }\ 114 | #bookmarks-menu-button.toolbarbutton-1 {\ 115 | -moz-box-orient: vertical !important;\ 116 | margin:0px 1px 0px 0px!important;\ 117 | }\ 118 | ')+ '"'), document.documentElement); 119 | })(); 120 | --------------------------------------------------------------------------------