333ywb
火狐狸
火狐狸
  • UID27284
  • 注册日期2008-12-03
  • 最后登录2025-03-19
  • 发帖数124
  • 经验141枚
  • 威望0点
  • 贡献值148点
  • 好评度4点
  • 社区居民
  • 忠实会员
阅读:281回复:5

[已修复] OpenWith.uc脚本在136.0a1版本中失效了,有请大神修复

楼主#
更多 发布于:2025-02-03 21:16
OpenWith.uc脚本在136.0a1版本中失效了,有请大神修复
// ==UserScript==
// <a href="home.php?mod=space&uid=467097">@Name</a>           OpenWith.uc.js
// @description    用其他浏览器打开页面、链接、书签及标签
// <a href="home.php?mod=space&uid=321866">@author</a>         ding
// <a href="home.php?mod=space&uid=713796">@include</a>        main
// <a href="home.php?mod=space&uid=570682">@version</a>        2018.2.3.1
// @homepageURL    https://bbs.kafan.cn/thread-2114879-1-1.html
// @startup        window.OpenWithManager.init();
// <a href="home.php?mod=space&uid=748274">@Shutdown</a>       window.OpenWithManager.destroy();
// <a href="home.php?mod=space&uid=356229">@Note</a>           适配Firefox57+
// ==/UserScript==
location.href.startsWith('chrome://browser/content/browser.x') && (function () {
  
    const MENU_NAME = "用其它浏览器打开";
  
    //是否使用二级菜单
    const USE_MENU_AREA = true; //页面
    const USE_MENU_TAB = false; //标签
    const USE_MENU_PLACE = false; //书签
  
    function getFirefoxPath(){//firefox.exe所在路径
        return OS.Constants.Path.libDir;
    }
    function getRootPath() {//firefox所在盘路径
        var path = getFirefoxPath();
        var index = path.indexOf(":");
        return path.substring(0,index+1);
    }
    //修改内容后请将脚本改名来保证加载的是最新,或使用无缓存的userChrome.js
    var browsers = {
         
                Edge: {
            enable: true,
            name: "Edge",
            path: "C:\\Program Files (x86)\\Microsoft\\Edge\\Application\\msedge.exe",
            image: ""
        },
                Chrome: {
            enable: true,
            name: "Chrome",
            path: "D:\\GoogleChromePortable64\\GoogleChromePortable.exe",
            image: ""
        },
          
        Slimjet: {
            enable: true,
            name: "Slimjet",
            path: "D:\\Slimjet\\slimjet.exe",
            image: ""
        },
        FireFox: {
            enable: true,
            name: "Firefox Nightly",
            path: "D:\\FirefoxNightly\\FirefoxPortable.exe",
            image: ""
        },
  
        others: {
            enable: false,
            name: "",
            path: "",
            image: ""
        },
  
        all: {
            enable: true,
            name: "所有浏览器",
            image: ""
        }
    };
  
    if (window.OpenWithManager) {
        window.OpenWithManager.destroy();
        delete window.OpenWithManager;
    }
  
    function $(id) {
        return document.getElementById(id);
    }
  
    window.OpenWithManager = {
        useMenu: false,
        getTypeDesc: function (type) {
            switch (type) {
                case "page":
                    return "本页";
                case "link":
                    return "链接";
                //case "tab": return "标签";
                //case "bm": return "书签";
            }
            return "";
        },
        buildMenuName: function (label, type) {
            return "用 " + label + " 打开" + this.getTypeDesc(type);
        },
        attachNode: function (anchorNode, node) {
            if (this.useMenu) {
                anchorNode.appendChild(node);
            } else {
                anchorNode.parentNode.insertBefore(node, anchorNode);
            }
        },
        createMenuPopup: function (anchorNode, type) {
            let menu = document.createXULElement("menu");
            menu.setAttribute("label", MENU_NAME);
            menu.setAttribute("id", "openwith-menu-" + type);
            menu.setAttribute("class", "menu-iconic openwith-menu open-" + type);
            menu.setAttribute("image", browsers.all.image);
            anchorNode.parentNode.insertBefore(menu, anchorNode);
  
            let popup = document.createXULElement("menupopup");
            popup.id = "openwith-popup-" + type;
            menu.appendChild(popup);
            return popup;
        },
        createMenuItem: function (anchorNode, id, browser, type) {
            let menuitem = document.createXULElement("menuitem");
            menuitem.id = "openwith-m-" + type + "-" + id;
            menuitem.setAttribute("label", this.buildMenuName(browser.name, type));
            menuitem.setAttribute("oncommand", "OpenWithManager.openWithOtherBrowser(this,'" + id + "','" + type + "')");
            menuitem.setAttribute("class", "menuitem-iconic openwith-menuitem open-" + type);
            if (browser.image) {
                menuitem.setAttribute("image", browser.image);
            }
            this.attachNode(anchorNode, menuitem);
        },
        createBrowserMenu: function (anchorNode, type) {
            for (let key in browsers) {
                try {
                    if (browsers[key].enable) {
                        this.createMenuItem(anchorNode, key, browsers[key], type);
                    }
                } catch (e) {
                    alert(e.message);
                }
            }
        },
        //contentAreaContextMenu
        initContentAreaMenu: function () {
            var inspos = $("inspect-separator");
            let sep = document.createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            inspos.parentNode.insertBefore(sep, inspos);
            this.useMenu = USE_MENU_AREA;
  
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "area");
            } else {
                anchorNode = inspos;
            }
            //链接部分
            this.createBrowserMenu(anchorNode, "link");
  
            let sepLink = document.createXULElement("menuseparator");
            sepLink.setAttribute("class", "open-link openwith-menuitem");
            this.attachNode(anchorNode, sepLink);
  
            //控制链接菜单的显示
            inspos.parentNode.addEventListener("popupshowing", this, false);
  
            //页面部分
            this.createBrowserMenu(anchorNode, "page");
        },
        //tabContextMenu
        initTabContextMenu: function () {
  
            var inspos = $("context_closeTabOptions") || $("context_closeTabOptions") 
            let sep = document.createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            inspos.parentNode.insertBefore(sep, inspos.nextSibling);
  
            this.useMenu = USE_MENU_TAB;
  
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(sep.nextSibling, "tab");
            } else {
                anchorNode = sep.nextSibling;
            }
            this.createBrowserMenu(anchorNode, "tab");
  
        },
        //placesContext
        initPlacesContextMenu: function () {
  
            var inspos = $("placesContext_openSeparator");
            //let sep = document.createXULElement("menuseparator");
            //inspos.parentNode.insertBefore(sep, inspos);
  
            this.useMenu = USE_MENU_PLACE;
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "place");
            } else {
                anchorNode = inspos;
            }
            this.createBrowserMenu(anchorNode, "place");
  
            //文件夹显示
            inspos.parentNode.addEventListener("popupshowing",this, false);
        },
        handleEvent: function (event) {
            if(event.target.id=="placesContext"){
                var isFloder = false;
                try {
                    let selectedNode = PlacesUIUtils.getViewForNode(event.target.ownerDocument.popupNode).selectedNode;
                    isFloder = !selectedNode || selectedNode.hasChildren;
                } catch (e) {
                }
                let menus = $("placesContext").querySelectorAll(".open-place");
                for (let menu of menus) {
                    if (isFloder) {
                        menu.hidden = true;
                    } else {
                        menu.hidden = false;
                        menu.disabled = false;
                    }
                }
            }
            if(event.target.id=="contentAreaContextMenu"){
                let menus = $("contentAreaContextMenu").querySelectorAll(".open-link");
                for (let menu of menus) {
                    if (gContextMenu.onLink) {
                        menu.hidden = false;
                    } else {
                        menu.hidden = true;
                    }
                }
            }
  
        },
        init: function () {
  
            //contentAreaContextMenu
            this.initContentAreaMenu();
  
            //tabContextMenu
            this.initTabContextMenu();
  
            //placesContext
            this.initPlacesContextMenu();
  
        },
        destroy:function () {
            $("contentAreaContextMenu").removeEventListener("popupshowing", this, false);
            $("placesContext_openSeparator").removeEventListener("popupshowing", this, false);
  
            let menus = document.querySelectorAll(".openwith-menu"),
                menuitems = document.querySelectorAll(".openwith-menuitem");
            for(let menuitem of menuitems){
                menuitem.parentNode.removeChild(menuitem);
            }
            for(let menu of menus){
                menu.parentNode.removeChild(menu);
            }
        },
        openWithBrowser: function (url, path) {
            if (!path) {
                alert("浏览器路径未设置 ");
                return;
            }
            if (path == "microsoft-edge") {
                //edge用url打开
                var cmdUrl = path + ":" + url;
                let win = window.openDialog(cmdUrl, "", "chrome");
                win.close();
                return;
            }
  
            let clientApp = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile);
            clientApp.initWithPath(path);
            if (!clientApp.exists()) {
                alert("浏览器路径错误: " + path);
                return;
            }
            try {
                let ss = Cc["@mozilla.org/browser/shell-service;1"]
                    .getService(Ci.nsIShellService);
                ss.openApplicationWithURI(clientApp, url);
            } catch (e) {
                let p = Cc["@mozilla.org/process/util;1"]
                    .createInstance(Ci.nsIProcess);
                p.init(clientApp);
                p.run(false, [url], 1);
            }
        },
        openWithOtherBrowser : function (obj, id, type) {
            var url;
            switch (type) {
                case "page":
                    url = gBrowser.currentURI.spec;
                    break;
                case "link":
                    url = gContextMenu.linkURL;
                    break;
                case "tab": {
                    let tab = TabContextMenu.contextTab;
                    let bw = tab && tab.linkedBrowser;
                    url = bw && bw.currentURI.spec;
                }
                    break;
                case "place":
                    url = PlacesUIUtils.getViewForNode(PlacesUIUtils.lastContextMenuTriggerNode).selectedNode.uri;
                    break;
            }
            if (url) {
                if (id == "all") {
                    for (let key in browsers) {
                        let browser = browsers[key];
                        if (browser.enable && key != "all" && browser.path) {
                            this.openWithBrowser(url, browser.path);
                        }
                    }
                } else {
                    let browser = browsers[id];
                    this.openWithBrowser(url, browser.path);
                }
            }
        }
  
    };
    OpenWithManager.init();
})();
viewtheard
千年狐狸
千年狐狸
  • UID2383
  • 注册日期2005-01-20
  • 最后登录2025-03-15
  • 发帖数2042
  • 经验558枚
  • 威望1点
  • 贡献值428点
  • 好评度23点
  • 社区居民
  • 忠实会员
1楼#
发布于:2025-02-05 09:36


Mapping out compatibilityI thought through the compatibility issues with Firefox 136.0a1, noting that older XUL methods like "document.createXULElement" are deprecated. This script needs adapting or updating for the latest Firefox.
Updating XUL elementsI’m moving forward with replacing deprecated XUL methods with "document.createElementNS" for Firefox 136.0a1, ensuring tags like "menu", "menupopup", and "menuitem" align with the XUL namespace.
Addressing global objectsI'm considering updating gContextMenu references to "gBrowser.selectedBrowser" in the script. This adjustment is crucial to ensure compatibility with the latest Firefox updates.
Confirming compatibilityChecking that gContextMenu and gBrowser.currentURI.spec function as expected in Firefox 136.0a1. Also, I’m thinking about the existence and potential changes to "placesContext" and other element IDs.
Analyzing context menu changesOK, let me see. I’m gathering info on the availability of "contentAreaContextMenu" and "placesContext" in Firefox 136, considering updates to tab context menus like "context_closeTabOptions."
Tracking menu ID shiftsOK, let me see. If "context_closeTabOptions" is outdated in Firefox 136, updating the script to use "tabContextMenu" seems crucial for maintaining compatibility.
Tracking ID changesI'm looking into tab context menu IDs to ensure compatibility. In Firefox, element IDs like "context_closeTabOptions" are updated, so "tabContextMenu" might be the current identifier to use.
Transitioning to "tabContextMenu"OK, attaching the separator to "tabContextMenu" instead of "context_closeTabOptions" in Firefox 136. This aligns better with the recent ID changes.
Upgrading element creationI'm noting the switch to document.createElementNS for defining XUL elements. This change should streamline element creation throughout the codebase.
Maintaining contextI’m noting that "Cc" and "Ci" are likely legacy definitions from Components.classes and Components.interfaces. Their use in Chrome context scripts should remain stable, but we'll keep an eye on their status.
Advancing the scriptI’m wrapping up the script by defining createXULElement and updating all document.createXULElement calls to maintain XUL element creation. This introduces tabContextMenu updates.



下面给出一个修正后的脚本版本,主要修改内容是将所有对



document.createXULElement(...)
My Technical Blog: http://art-technical.blogspot.com/
viewtheard
千年狐狸
千年狐狸
  • UID2383
  • 注册日期2005-01-20
  • 最后登录2025-03-15
  • 发帖数2042
  • 经验558枚
  • 威望1点
  • 贡献值428点
  • 好评度23点
  • 社区居民
  • 忠实会员
2楼#
发布于:2025-02-05 09:36
的调用替换为使用 XUL 命名空间创建元素的方法,同时针对新版 Firefox(136+)的标签右键菜单做了适配(原来用 “context_closeTabOptions” 的做法可能已失效,现在改为直接在 “tabContextMenu” 内追加菜单项)。

请将下面代码保存为 OpenWith.uc.js 并在 Firefox 的 userChrome.js 配置中加载使用:
My Technical Blog: http://art-technical.blogspot.com/
viewtheard
千年狐狸
千年狐狸
  • UID2383
  • 注册日期2005-01-20
  • 最后登录2025-03-15
  • 发帖数2042
  • 经验558枚
  • 威望1点
  • 贡献值428点
  • 好评度23点
  • 社区居民
  • 忠实会员
3楼#
发布于:2025-02-05 09:37
// ==UserScript== 
// [url=home.php?mod=space&uid=467097]@Name[/url]           OpenWith.uc.js
// @description    用其他浏览器打开页面、链接、书签及标签
// [url=home.php?mod=space&uid=321866]@author[/url]         ding
// [url=home.php?mod=space&uid=713796]@include[/url]        main
// [url=home.php?mod=space&uid=570682]@version[/url]        2018.2.3.1 (修正新版Firefox)
// @homepageURL    https://bbs.kafan.cn/thread-2114879-1-1.html
// @startup        window.OpenWithManager.init();
// [url=home.php?mod=space&uid=748274]@Shutdown[/url]       window.OpenWithManager.destroy();
// [url=home.php?mod=space&uid=356229]@Note[/url]           适配Firefox57+及136+
   
location.href.startsWith('chrome://browser/content/browser.x') && (function () {
   
    const XUL_NS = "http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul";
    // 用于创建 XUL 元素
    function createXULElement(tag) {
        return document.createElementNS(XUL_NS, tag);
    }
   
    const MENU_NAME = "用其它浏览器打开";
   
    // 是否使用二级菜单
    const USE_MENU_AREA = true; // 页面
    const USE_MENU_TAB = false; // 标签(新版下直接追加到 tabContextMenu 中,建议关闭二级菜单)
    const USE_MENU_PLACE = false; // 书签
   
    function getFirefoxPath(){ // firefox.exe 所在路径
        return OS.Constants.Path.libDir;
    }
    function getRootPath() { // firefox 所在盘路径
        var path = getFirefoxPath();
        var index = path.indexOf(":");
        return path.substring(0,index+1);
    }
   
    // 修改内容后请将脚本改名来保证加载的是最新版本,或使用无缓存的 userChrome.js
    var browsers = {
        Edge: {
            enable: true,
            name: "Edge",
            path: "C:\\Program Files (x86)\\Microsoft\\Edge\\Application\\msedge.exe",
            image: ""
        },
        Chrome: {
            enable: true,
            name: "Chrome",
            path: "D:\\GoogleChromePortable64\\GoogleChromePortable.exe",
            image: ""
        },
        Slimjet: {
            enable: true,
            name: "Slimjet",
            path: "D:\\Slimjet\\slimjet.exe",
            image: ""
        },
        FireFox: {
            enable: true,
            name: "Firefox Nightly",
            path: "D:\\FirefoxNightly\\FirefoxPortable.exe",
            image: ""
        },
        others: {
            enable: false,
            name: "",
            path: "",
            image: ""
        },
        all: {
            enable: true,
            name: "所有浏览器",
            image: ""
        }
    };
   
    // 如果已存在旧的实例,则先清理
    if (window.OpenWithManager) {
        window.OpenWithManager.destroy();
        delete window.OpenWithManager;
    }
   
    function $(id) {
        return document.getElementById(id);
    }
   
    window.OpenWithManager = {
        useMenu: false,
        getTypeDesc: function (type) {
            switch (type) {
                case "page":
                    return "本页";
                case "link":
                    return "链接";
                // case "tab": return "标签";
                // case "bm": return "书签";
            }
            return "";
        },
        buildMenuName: function (label, type) {
            return "用 " + label + " 打开" + this.getTypeDesc(type);
        },
        attachNode: function (anchorNode, node) {
            if (this.useMenu) {
                anchorNode.appendChild(node);
            } else {
                anchorNode.parentNode.insertBefore(node, anchorNode);
            }
        },
        createMenuPopup: function (anchorNode, type) {
            let menu = createXULElement("menu");
            menu.setAttribute("label", MENU_NAME);
            menu.setAttribute("id", "openwith-menu-" + type);
            menu.setAttribute("class", "menu-iconic openwith-menu open-" + type);
            menu.setAttribute("image", browsers.all.image);
            anchorNode.parentNode.insertBefore(menu, anchorNode);
   
            let popup = createXULElement("menupopup");
            popup.id = "openwith-popup-" + type;
            menu.appendChild(popup);
            return popup;
        },
        createMenuItem: function (anchorNode, id, browser, type) {
            let menuitem = createXULElement("menuitem");
            menuitem.id = "openwith-m-" + type + "-" + id;
            menuitem.setAttribute("label", this.buildMenuName(browser.name, type));
            menuitem.setAttribute("oncommand", "OpenWithManager.openWithOtherBrowser(this, '" + id + "', '" + type + "')");
            menuitem.setAttribute("class", "menuitem-iconic openwith-menuitem open-" + type);
            if (browser.image) {
                menuitem.setAttribute("image", browser.image);
            }
            this.attachNode(anchorNode, menuitem);
        },
        createBrowserMenu: function (anchorNode, type) {
            for (let key in browsers) {
                try {
                    if (browsers[key].enable) {
                        this.createMenuItem(anchorNode, key, browsers[key], type);
                    }
                } catch (e) {
                    alert(e.message);
                }
            }
        },
        // contentAreaContextMenu(页面右键菜单)
        initContentAreaMenu: function () {
            var inspos = $("inspect-separator");
            let sep = createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            inspos.parentNode.insertBefore(sep, inspos);
            this.useMenu = USE_MENU_AREA;
   
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "area");
            } else {
                anchorNode = inspos;
            }
            // 添加“链接”相关菜单项
            this.createBrowserMenu(anchorNode, "link");
   
            let sepLink = createXULElement("menuseparator");
            sepLink.setAttribute("class", "open-link openwith-menuitem");
            this.attachNode(anchorNode, sepLink);
   
            // 根据当前右键环境控制链接菜单显示
            inspos.parentNode.addEventListener("popupshowing", this, false);
   
            // 添加“本页”菜单项
            this.createBrowserMenu(anchorNode, "page");
        },
        // tabContextMenu(标签右键菜单)
        initTabContextMenu: function () {
            // 在新版 Firefox 下,标签右键菜单的 id 为 "tabContextMenu"
            var tabContext = $("tabContextMenu");
            if (!tabContext) {
                return;
            }
            let sep = createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            // 直接追加到菜单末尾
            tabContext.appendChild(sep);
   
            this.useMenu = USE_MENU_TAB;
   
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(sep, "tab");
            } else {
                anchorNode = sep;
            }
            this.createBrowserMenu(anchorNode, "tab");
        },
        // placesContext(书签/历史右键菜单)
        initPlacesContextMenu: function () {
            var inspos = $("placesContext_openSeparator");
            if (!inspos) {
                return;
            }
            this.useMenu = USE_MENU_PLACE;
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "place");
            } else {
                anchorNode = inspos;
            }
            this.createBrowserMenu(anchorNode, "place");
   
            // 针对文件夹项控制显示
            inspos.parentNode.addEventListener("popupshowing", this, false);
        },
        handleEvent: function (event) {
            if (event.target.id == "placesContext") {
                var isFolder = false;
                try {
                    let selectedNode = PlacesUIUtils.getViewForNode(event.target.ownerDocument.popupNode).selectedNode;
                    isFolder = !selectedNode || selectedNode.hasChildren;
                } catch (e) { }
                let menus = $("placesContext").querySelectorAll(".open-place");
                for (let menu of menus) {
                    menu.hidden = isFolder;
                    if (!isFolder) {
                        menu.disabled = false;
                    }
                }
            }
            if (event.target.id == "contentAreaContextMenu") {
                let menus = $("contentAreaContextMenu").querySelectorAll(".open-link");
                for (let menu of menus) {
                    menu.hidden = !gContextMenu.onLink;
                }
            }
        },
        init: function () {
            // 初始化页面右键菜单
            this.initContentAreaMenu();
            // 初始化标签右键菜单
            this.initTabContextMenu();
            // 初始化书签/历史右键菜单
            this.initPlacesContextMenu();
        },
        destroy: function () {
            $("contentAreaContextMenu").removeEventListener("popupshowing", this, false);
            let placesSep = $("placesContext_openSeparator");
            if (placesSep)
                placesSep.removeEventListener("popupshowing", this, false);
   
            let menus = document.querySelectorAll(".openwith-menu"),
                menuitems = document.querySelectorAll(".openwith-menuitem");
            for (let menuitem of menuitems) {
                if (menuitem.parentNode)
                    menuitem.parentNode.removeChild(menuitem);
            }
            for (let menu of menus) {
                if (menu.parentNode)
                    menu.parentNode.removeChild(menu);
            }
        },
        openWithBrowser: function (url, path) {
            if (!path) {
                alert("浏览器路径未设置");
                return;
            }
            if (path == "microsoft-edge") {
                // Edge 使用 URL 打开
                var cmdUrl = path + ":" + url;
                let win = window.openDialog(cmdUrl, "", "chrome");
                win.close();
                return;
            }
   
            let clientApp = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile);
            clientApp.initWithPath(path);
            if (!clientApp.exists()) {
                alert("浏览器路径错误: " + path);
                return;
            }
            try {
                let ss = Cc["@mozilla.org/browser/shell-service;1"]
                    .getService(Ci.nsIShellService);
                ss.openApplicationWithURI(clientApp, url);
            } catch (e) {
                let p = Cc["@mozilla.org/process/util;1"]
                    .createInstance(Ci.nsIProcess);
                p.init(clientApp);
                p.run(false, [url], 1);
            }
        },
        openWithOtherBrowser: function (obj, id, type) {
            var url;
            switch (type) {
                case "page":
                    url = gBrowser.currentURI.spec;
                    break;
                case "link":
                    url = gContextMenu.linkURL;
                    break;
                case "tab": {
                    let tab = TabContextMenu.contextTab;
                    let bw = tab && tab.linkedBrowser;
                    url = bw && bw.currentURI.spec;
                }
                    break;
                case "place":
                    url = PlacesUIUtils.getViewForNode(PlacesUIUtils.lastContextMenuTriggerNode).selectedNode.uri;
                    break;
            }
            if (url) {
                if (id == "all") {
                    for (let key in browsers) {
                        let browser = browsers[key];
                        if (browser.enable && key != "all" && browser.path) {
                            this.openWithBrowser(url, browser.path);
                        }
                    }
                } else {
                    let browser = browsers[id];
                    this.openWithBrowser(url, browser.path);
                }
            }
        }
   
    };
    OpenWithManager.init();
})();
My Technical Blog: http://art-technical.blogspot.com/
viewtheard
千年狐狸
千年狐狸
  • UID2383
  • 注册日期2005-01-20
  • 最后登录2025-03-15
  • 发帖数2042
  • 经验558枚
  • 威望1点
  • 贡献值428点
  • 好评度23点
  • 社区居民
  • 忠实会员
4楼#
发布于:2025-02-05 09:41
说明
XUL 元素创建
为了适应 Firefox 136 及以后的版本,原脚本中直接调用 document.createXULElement 的方式已不可用,故增加了辅助函数 createXULElement(tag),该函数通过命名空间 http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul 创建 XUL 元素。

标签右键菜单适配
原脚本中查找 “context_closeTabOptions” 已不适用,新版本中标签右键菜单的容器 id 为 tabContextMenu,故在 initTabContextMenu() 中做了相应调整(你也可以根据需要调整是否使用二级菜单)。

其他部分
其他部分大体保持不变,仅做了少量兼容性处理。若仍有部分失效,请检查当前 Firefox UI 结构是否有进一步调整,可通过“检查元素”工具确认对应 id 和结构。

保存并重启 Firefox 后,即可在页面、链接、标签及书签右键菜单中看到“用其它浏览器打开”的相关菜单项。
My Technical Blog: http://art-technical.blogspot.com/
333ywb
火狐狸
火狐狸
  • UID27284
  • 注册日期2008-12-03
  • 最后登录2025-03-19
  • 发帖数124
  • 经验141枚
  • 威望0点
  • 贡献值148点
  • 好评度4点
  • 社区居民
  • 忠实会员
5楼#
发布于:2025-02-05 12:06
已修复,可以正常使用
// ==UserScript==
// @name           OpenWith.uc.js
// @description    用其他浏览器打开页面、链接、书签及标签
// @author         ding
// @include        main
// @version        2018.2.3.1
// @homepageURL    https://bbs.kafan.cn/thread-2114879-1-1.html
// @startup        window.OpenWithManager.init();
// @shutdown       window.OpenWithManager.destroy();
// @note           适配Firefox57+
// ==/UserScript==
location.href.startsWith('chrome://browser/content/browser.x') && (function () {
  
    const MENU_NAME = "用其它浏览器打开";
  
    //是否使用二级菜单
    const USE_MENU_AREA = true; //页面
    const USE_MENU_TAB = false; //标签
    const USE_MENU_PLACE = false; //书签
  
    function getFirefoxPath(){//firefox.exe所在路径
        return OS.Constants.Path.libDir;
    }
    function getRootPath() {//firefox所在盘路径
        var path = getFirefoxPath();
        var index = path.indexOf(":");
        return path.substring(0,index+1);
    }
    //修改内容后请将脚本改名来保证加载的是最新,或使用无缓存的userChrome.js
    var browsers = {
         
        Edge: {
            enable: true,
            name: "Edge",
            path: "C:\\Program Files (x86)\\Microsoft\\Edge\\Application\\msedge.exe",
            image: ""
        },
        Chrome: {
            enable: true,
            name: "Chrome",
            path: "D:\\GoogleChromePortable64\\GoogleChromePortable.exe",
            image: ""
        },
          
        Slimjet: {
            enable: true,
            name: "Slimjet",
            path: "D:\\Slimjet\\slimjet.exe",
            image: ""
        },
        FireFox: {
            enable: true,
            name: "Firefox Nightly",
            path: "D:\\FirefoxNightly\\FirefoxPortable.exe",
            image: ""
        },
  
        others: {
            enable: false,
            name: "",
            path: "",
            image: ""
        },
  
        all: {
            enable: true,
            name: "所有浏览器",
            image: ""
        }
    };
  
    if (window.OpenWithManager) {
        window.OpenWithManager.destroy();
        delete window.OpenWithManager;
    }
  
    function $(id) {
        return document.getElementById(id);
    }
  
    window.OpenWithManager = {
        useMenu: false,
        getTypeDesc: function (type) {
            switch (type) {
                case "page":
                    return "本页";
                case "link":
                    return "链接";
                //case "tab": return "标签";
                //case "bm": return "书签";
            }
            return "";
        },
        buildMenuName: function (label, type) {
            return "用 " + label + " 打开" + this.getTypeDesc(type);
        },
        attachNode: function (anchorNode, node) {
            if (this.useMenu) {
                anchorNode.appendChild(node);
            } else {
                anchorNode.parentNode.insertBefore(node, anchorNode);
            }
        },
        createMenuPopup: function (anchorNode, type) {
            let menu = document.createXULElement("menu");
            menu.setAttribute("label", MENU_NAME);
            menu.setAttribute("id", "openwith-menu-" + type);
            menu.setAttribute("class", "menu-iconic openwith-menu open-" + type);
            menu.setAttribute("image", browsers.all.image);
            anchorNode.parentNode.insertBefore(menu, anchorNode);
  
            let popup = document.createXULElement("menupopup");
            popup.id = "openwith-popup-" + type;
            menu.appendChild(popup);
            return popup;
        },
        createMenuItem: function (anchorNode, id, browser, type) {
            let menuitem = document.createXULElement("menuitem");
            menuitem.id = "openwith-m-" + type + "-" + id;
            menuitem.setAttribute("label", this.buildMenuName(browser.name, type));
              menuitem.addEventListener("command", function() {
              OpenWithManager.openWithOtherBrowser(this, id, type);
          });
            menuitem.setAttribute("class", "menuitem-iconic openwith-menuitem open-" + type);
            if (browser.image) {
                menuitem.setAttribute("image", browser.image);
            }
            this.attachNode(anchorNode, menuitem);
        },
        createBrowserMenu: function (anchorNode, type) {
            for (let key in browsers) {
                try {
                    if (browsers[key].enable) {
                        this.createMenuItem(anchorNode, key, browsers[key], type);
                    }
                } catch (e) {
                    alert(e.message);
                }
            }
        },
        //contentAreaContextMenu
        initContentAreaMenu: function () {
            var inspos = $("inspect-separator");
            let sep = document.createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            inspos.parentNode.insertBefore(sep, inspos);
            this.useMenu = USE_MENU_AREA;
  
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "area");
            } else {
                anchorNode = inspos;
            }
            //链接部分
            this.createBrowserMenu(anchorNode, "link");
  
            let sepLink = document.createXULElement("menuseparator");
            sepLink.setAttribute("class", "open-link openwith-menuitem");
            this.attachNode(anchorNode, sepLink);
  
            //控制链接菜单的显示
            inspos.parentNode.addEventListener("popupshowing", this, false);
  
            //页面部分
            this.createBrowserMenu(anchorNode, "page");
        },
        //tabContextMenu
        initTabContextMenu: function () {
  
            var inspos = $("context_closeTabOptions") || $("context_closeTabOptions") 
            let sep = document.createXULElement("menuseparator");
            sep.setAttribute("class", "openwith-menuitem");
            inspos.parentNode.insertBefore(sep, inspos.nextSibling);
  
            this.useMenu = USE_MENU_TAB;
  
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(sep.nextSibling, "tab");
            } else {
                anchorNode = sep.nextSibling;
            }
            this.createBrowserMenu(anchorNode, "tab");
  
        },
        //placesContext
        initPlacesContextMenu: function () {
  
            var inspos = $("placesContext_openSeparator");
            //let sep = document.createXULElement("menuseparator");
            //inspos.parentNode.insertBefore(sep, inspos);
  
            this.useMenu = USE_MENU_PLACE;
            var anchorNode;
            if (this.useMenu) {
                anchorNode = this.createMenuPopup(inspos, "place");
            } else {
                anchorNode = inspos;
            }
            this.createBrowserMenu(anchorNode, "place");
  
            //文件夹显示
            inspos.parentNode.addEventListener("popupshowing",this, false);
        },
        handleEvent: function (event) {
            if(event.target.id=="placesContext"){
                var isFloder = false;
                try {
                    let selectedNode = PlacesUIUtils.getViewForNode(event.target.ownerDocument.popupNode).selectedNode;
                    isFloder = !selectedNode || selectedNode.hasChildren;
                } catch (e) {
                }
                let menus = $("placesContext").querySelectorAll(".open-place");
                for (let menu of menus) {
                    if (isFloder) {
                        menu.hidden = true;
                    } else {
                        menu.hidden = false;
                        menu.disabled = false;
                    }
                }
            }
            if(event.target.id=="contentAreaContextMenu"){
                let menus = $("contentAreaContextMenu").querySelectorAll(".open-link");
                for (let menu of menus) {
                    if (gContextMenu.onLink) {
                        menu.hidden = false;
                    } else {
                        menu.hidden = true;
                    }
                }
            }
  
        },
        init: function () {
  
            //contentAreaContextMenu
            this.initContentAreaMenu();
  
            //tabContextMenu
            this.initTabContextMenu();
  
            //placesContext
            this.initPlacesContextMenu();
  
        },
        destroy:function () {
            $("contentAreaContextMenu").removeEventListener("popupshowing", this, false);
            $("placesContext_openSeparator").removeEventListener("popupshowing", this, false);
  
            let menus = document.querySelectorAll(".openwith-menu"),
                menuitems = document.querySelectorAll(".openwith-menuitem");
            for(let menuitem of menuitems){
                menuitem.parentNode.removeChild(menuitem);
            }
            for(let menu of menus){
                menu.parentNode.removeChild(menu);
            }
        },
        openWithBrowser: function (url, path) {
            if (!path) {
                alert("浏览器路径未设置 ");
                return;
            }
            if (path == "microsoft-edge") {
                //edge用url打开
                var cmdUrl = path + ":" + url;
                let win = window.openDialog(cmdUrl, "", "chrome");
                win.close();
                return;
            }
  
            let clientApp = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile);
            clientApp.initWithPath(path);
            if (!clientApp.exists()) {
                alert("浏览器路径错误: " + path);
                return;
            }
            try {
                let ss = Cc["@mozilla.org/browser/shell-service;1"]
                    .getService(Ci.nsIShellService);
                ss.openApplicationWithURI(clientApp, url);
            } catch (e) {
                let p = Cc["@mozilla.org/process/util;1"]
                    .createInstance(Ci.nsIProcess);
                p.init(clientApp);
                p.run(false, [url], 1);
            }
        },
        openWithOtherBrowser : function (obj, id, type) {
            var url;
            switch (type) {
                case "page":
                    url = gBrowser.currentURI.spec;
                    break;
                case "link":
                    url = gContextMenu.linkURL;
                    break;
                case "tab": {
                    let tab = TabContextMenu.contextTab;
                    let bw = tab && tab.linkedBrowser;
                    url = bw && bw.currentURI.spec;
                }
                    break;
                case "place":
                    url = PlacesUIUtils.getViewForNode(PlacesUIUtils.lastContextMenuTriggerNode).selectedNode.uri;
                    break;
            }
            if (url) {
                if (id == "all") {
                    for (let key in browsers) {
                        let browser = browsers[key];
                        if (browser.enable && key != "all" && browser.path) {
                            this.openWithBrowser(url, browser.path);
                        }
                    }
                } else {
                    let browser = browsers[id];
                    this.openWithBrowser(url, browser.path);
                }
            }
        }
  
    };
    OpenWithManager.init();
})();
游客

返回顶部