gpt4 book ai didi

javascript - 如何从 XUL 中运行的脚本(例如在侧边栏中)中获取当前事件浏览器窗口/选项卡的 URL

转载 作者:行者123 更新时间:2023-12-03 06:11:23 24 4
gpt4 key购买 nike

我正在为 Firefox 创建一个简单的引导插件。我需要通过单击按钮通过侧边栏从浏览器捕获当前 URL。

我的bootstrap.js:

const {classes: Cc, interfaces: Ci, utils: Cu} = Components;
Cu.import('resource://gre/modules/Services.jsm');
Cu.import("resource://gre/modules/NetUtil.jsm");
Cu.import("resource://gre/modules/FileUtils.jsm");

/*start - windowlistener*/
var windowListener = {
//DO NOT EDIT HERE
onOpenWindow: function (aXULWindow) {
// Wait for the window to finish loading
let aDOMWindow =aXULWindow.QueryInterface(Ci.nsIInterfaceRequestor)
.getInterface(Ci.nsIDOMWindowInternal||Ci.nsIDOMWindow);
aDOMWindow.addEventListener("load", function () {
aDOMWindow.removeEventListener("load", arguments.callee, false);
windowListener.loadIntoWindow(aDOMWindow, aXULWindow);
}, false);
},
onCloseWindow: function (aXULWindow) {},
onWindowTitleChange: function (aXULWindow, aNewTitle) {},
register: function () {
// Load into any existing windows
let XULWindows = Services.wm.getXULWindowEnumerator(null);
while (XULWindows.hasMoreElements()) {
let aXULWindow = XULWindows.getNext();
let aDOMWindow = aXULWindow.QueryInterface(Ci.nsIInterfaceRequestor)
.getInterface(Ci.nsIDOMWindowInternal || Ci.nsIDOMWindow);
windowListener.loadIntoWindow(aDOMWindow, aXULWindow);
}
// Listen to new windows
Services.wm.addListener(windowListener);
},
unregister: function () {
// Unload from any existing windows
let XULWindows = Services.wm.getXULWindowEnumerator(null);
while (XULWindows.hasMoreElements()) {
let aXULWindow = XULWindows.getNext();
let aDOMWindow = aXULWindow.QueryInterface(Ci.nsIInterfaceRequestor)
.getInterface(Ci.nsIDOMWindowInternal || Ci.nsIDOMWindow);
windowListener.unloadFromWindow(aDOMWindow, aXULWindow);
}
//Stop listening so future added windows dont get this attached
Services.wm.removeListener(windowListener);
},
//END - DO NOT EDIT HERE
loadIntoWindow: function (aDOMWindow, aXULWindow) {
if (!aDOMWindow) {
return;
}

//START - EDIT BELOW HERE
var browser = aDOMWindow.document.querySelector('#browser')
if (browser) {
var splitter = aDOMWindow.document.createElement('splitter');
var propsToSet = {
id: 'demo-sidebar-with-html_splitter',
//I'm just copying what Mozilla does for their social sidebar splitter
// I left it out, but you can leave it in to see how you can style
// the splitter
class: 'sidebar-splitter'
}
for (var p in propsToSet) {
splitter.setAttribute(p, propsToSet[p]);
}

var sidebar = aDOMWindow.document.createElement('vbox');
var propsToSet = {
id: 'demo-sidebar-with-html_sidebar',
//Mozilla uses persist width here, I don't know what it does and can't
// see it how makes a difference so I left it out
//persist: 'width'
}
for (var p in propsToSet) {
sidebar.setAttribute(p, propsToSet[p]);
}
var htmlVal = loadJsonHTML(0);
var sidebarBrowser = aDOMWindow.document.createElement('browser');
var propsToSet = {
id: 'demo-sidebar-with-html_browser',
type: 'content',
context: 'contentAreaContextMenu',
disableglobalhistory: 'true',
tooltip: 'aHTMLTooltip',
autoscrollpopup: 'autoscroller',
flex: '1', //do not remove this
//you should change these widths to how you want
style: 'min-width: 14em; width: 18em; max-width: 36em;',
//or just set this to some URL like http://www.bing.com/
src: 'data:text/html,'+ htmlVal
}
for (var p in propsToSet) {
sidebarBrowser.setAttribute(p, propsToSet[p]);
}

browser.appendChild(splitter);

sidebar.appendChild(sidebarBrowser);
browser.appendChild(sidebar);
}
//END - EDIT BELOW HERE
},
unloadFromWindow: function (aDOMWindow, aXULWindow) {
if (!aDOMWindow) {
return;
}
//START - EDIT BELOW HERE
var splitter = aDOMWindow.document
.querySelector('#demo-sidebar-with-html_splitter');

if (splitter) {
var sidebar = aDOMWindow.document
.querySelector('#demo-sidebar-with-html_sidebar');
splitter.parentNode.removeChild(splitter);
sidebar.parentNode.removeChild(sidebar);
}
//END - EDIT BELOW HERE
}
};
/*end - windowlistener*/

function startup(aData, aReason) {
windowListener.register();
}

function shutdown(aData, aReason) {
if (aReason == APP_SHUTDOWN) return;
windowListener.unregister();
}

function loadJsonHTML(val=0){

var fileContent = "";
var localFile = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsILocalFile);

//full path is okay if directory exists
localFile.initWithPath("/Users/tinuy/Desktop/test_addodn/input.txt");

//otherwise specify directory, create it if necessary, and append leaf.
//localFile.initWithPath("C:\Users\tinuy\Documents\test\input.txt");

if ( localFile.exists() == false ) {
fileContent = "File does not exist";
}
var istream = Cc["@mozilla.org/network/file-input-stream;1"]
.createInstance(Ci.nsIFileInputStream);
istream.init(localFile, 0x01, 4, null);
var fileScriptableIO = Cc["@mozilla.org/scriptableinputstream;1"]
.createInstance(Ci.nsIScriptableInputStream);
fileScriptableIO.init(istream);
// parse the XML into our internal document
istream.QueryInterface(Ci.nsILineInputStream);
//fileContent = fileScriptableIO.read( '1' );
var csize = 0;
while ((csize = fileScriptableIO.available()) != 0) {
fileContent += fileScriptableIO.read( csize );
}
var array = fileContent.split("&");
fileScriptableIO.close();
istream.close();

return makeHTML(array[val], val);
}

function makeHTML(value, key){
var arrValues = value.split(",");
var htmlContent = '<div name="content" class="content">' +
'<p> Name :' + arrValues[0] + '</p>';
htmlContent += '<p> Price :' + arrValues[2] + '</p>';
htmlContent += '<p> Color :' + arrValues[3] + '</p>';
htmlContent += '<p> UID :' + arrValues[1] + '</p>';
htmlContent += '<p><input type="radio" name="valid" value="yes" />Yes &nbsp;&nbsp; ' +
'<input type="radio" name="valid" value="no" /> No</p>' +
'<p><input type="text" placeholder="Reason" name="checkreason"></p>' +
'<p><input type="text" placeholder="Feedback" name="feedback"></p>' +
'</div><div><button name="load" type="button" id="loadit" onclick="loadHtml()" ' +
'loadurl="'+arrValues[4]+'">Load</button> <button name="save" type="button">' +
'Save </button> <button name="next" type="button" key="'+key+'">Next </button> ' +
'</div> <script> function loadHtml() {' +
'var a = gBrowser.contentWindow.location.href ;alert(a);' +
'} </script>';
return htmlContent;
}

function install() {}

function uninstall() {}

我尝试了 Get current page URL from a firefox sidebar extension 的所有建议但没有任何效果。

最佳答案

但是,从您设置的事实来看:

var propsToSet = {
id: 'demo-sidebar-with-html_browser',
type: 'content', //<---------------------------This

<browser> type to content ,我假设以下两件事之一:

  1. 您实际上并没有尝试从加载到 <browser> 的代码中获取事件选项卡的 URI。通过src属性。
    • 如果是这种情况,请参阅我对“How to Get Active Tab Location by e10s add-on”的回答。在这种情况下,您运行的上下文应该允许您使用该答案中的代码。
    • 您的代码并不符合这种可能的假设 #1。
  2. 您想要访问 <browser> 内的 URI 表单代码您已加载到侧边栏,但不知道设置 <browser> 的限制 type to content 强加于您加载到 <browser> 中的代码。
    • 您似乎正在尝试从 <browser> 中的代码访问 currant 选项卡的 URL 。您应该考虑重构您的代码,以便不需要从 <browser> 中的内容访问 chrome 权限。 。换句话说,如果可能的话,您应该编写代码,这样您就不需要直接从 <browser> 中运行的代码访问 URL 或其他信息。你插入。然而,这样做最终可能会变得更加复杂。

获取<browser>内的chrome权限您要添加:

设置 <browser> type to content 用于:

A browser for content. The content that is loaded inside the browser is not allowed to access the chrome above it.

设置 <browser> typechrome用于:

(default behaviour): A browser, intended to be used for loading privileged content using a chrome:// URI. Don't use for content from web, as this may cause serious security problems!

设置 <browser> typechrome你可以这样做:

var propsToSet = {
id: 'demo-sidebar-with-html_browser',
type: 'chrome', //<---------------------------This

代码由Noitidart提供接近您所需要的。不过需要稍微修改一下,如 gBrowser.currentURI is a nsIURI, not a string )。因此,您需要使用gBrowser.currentURI.spec 。您可以通过更改 makeHTML() 来使其正常工作至:

function makeHTML(value, key){
var arrValues = value.split(",");
var htmlContent = '<html><head></head>'
+ '<body style="background-color: white;">'
+ ' <div name="content" class="content">'
+ ' <p> Name :' + arrValues[0] + '</p>';
htmlContent += '<p> Price :' + arrValues[2] + '</p>';
htmlContent += '<p> Color :' + arrValues[3] + '</p>';
htmlContent += '<p> UID :' + arrValues[1] + '</p>';
htmlContent +=
' <p><input type="radio" name="valid" value="yes" />Yes &nbsp;&nbsp; '
+ ' <input type="radio" name="valid" value="no" /> No</p>'
+ ' <p><input type="text" placeholder="Reason" name="checkreason" /></p>'
+ ' <p><input type="text" placeholder="Feedback" name="feedback" /></p>'
+ ' </div>'
+ ' <div>'
+ ' <button name="load" type="button" id="loadit" onclick="loadHtml()" '
+ ' loadurl="' + arrValues[4] + '">Load</button>'
+ ' <button name="save" type="button">Save </button>'
+ ' <button name="next" type="button" key="' + key + '">Next </button>'
+ ' </div>'
+ ' <script>'
+ ' function loadHtml() {'
+ ' const Cu = Components.utils;'
+ ' Cu.import("resource://gre/modules/Services.jsm");'
+ ' var win = Services.wm.getMostRecentWindow("navigator:browser");'
+ ' if (win) {'
+ ' var url = win.gBrowser.currentURI.spec;'
+ ' alert("URL=" + url);'
+ ' }'
+ ' }'
+ ' </script>'
+ '</body></html>';
return htmlContent;
}

安全问题:
您可能会遇到一些非常重要的安全问题。仅从您想要当前页面的 URL 的事实来看,就意味着您可能正在快速接近这些真正的问题。这将取决于您具体要做什么。鉴于您没有提供这些信息,我不会太深入地了解问题所在。但是,您应该阅读“Firefox Security Basics for Developers ”和“Security best practices in extensions

有关侧边栏的其他信息:
您应该认真考虑创建 Add-on SDK基于附加组件,而不是 Bootstrap 附加组件,并使用 ui/sidebar API。

如果您有兴趣,我创建了一些代码,可用于创建一个“侧边栏”,该侧边栏位于事件选项卡的顶部、左侧、右侧或底部,或者在单独的窗口中(类似于devtools 面板就可以了)。创建的侧边栏与当前事件的选项卡相关联,而不是与所有选项卡相关联(我想我应该将其作为一个选项,使其与所有选项卡相关联,或仅与当前选项卡相关联)在我的回答中:“How to create a bottom-docked panel with XUL in Firefox?”和“Firefox Extension, Window related sidebar

关于javascript - 如何从 XUL <browser> 中运行的脚本(例如在侧边栏中)中获取当前事件浏览器窗口/选项卡的 URL,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39299624/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com