chrome 扩展的浏览器动作、后台脚本和内容脚本之间通信的上下文和方法?

IT技术 javascript google-chrome google-chrome-extension
2021-01-25 06:42:54

我认为 chrome 扩展总体上非常简单且非常强大,但总是让我感到困惑的一件事是尝试在可能运行代码的各种脚本之间进行通信。当从浏览器操作的“default_popup”页面引用时,就会运行代码,“背景”的“脚本”属性中的代码和内容脚本。

这些类别中的脚本在什么上下文中运行,每个脚本如何与其他脚本通信?

3个回答

三种不同的语境

作为 Chrome 扩展开发者,您可以区分三种不同的环境。

  1. 扩展代码,在你的 Chrome 扩展进程中运行
  2. 内容脚本,在选项卡的进程中运行。
  3. 在选项卡进程中运行的非扩展代码由内容脚本注入)。

请注意,<iframe src="chrome-extension://EXTENSIONID/page.htm">在非扩展页面中,过去被视为案例 2(内容脚本),因为框架是在非特权选项卡进程中加载​​的。由于在 Chrome 56 中为扩展启动了进程外 iframe,这些页面由扩展进程处理,因此它们可能使用相同的全套扩展 API。这种行为变化(允许扩展框架使用特权扩展 API)是有意的

window在扩展进程中访问对象

因为所有的扩展代码都运行在同一个进程中,所以它们可以互相访问全局window对象。这一特性并不广为人知,但允许在同一个扩展进程中直接操作 JavaScript 和 DOM 对象。通常最好不要使用此方法,而是使用消息传递API。

// To access the `window` of a background page, use
var bgWindowObject = chrome.extension.getBackgroundPage();
// To access the `window` of an event or background page, use:
chrome.runtime.getBackgroundPage(function(bgWindowObject) {
    // Do something with `bgWindow` if you want
});

// To access the `window` of the badge's popup page (only if it's open!!!), use
var popupWindowObject = chrome.extension.getViews({type:'popup'})[0];

// To access the `window` of the options page (called /options.html), use
var allWindowObjects = chrome.extension.getViews({type:'tab'});
var popupWindowObjects = allWindowObjects.filter(function(windowObject) {
    return windowObject.location.pathname == '/options.html';
});
// Example: Get the `window` object of the first options page:
var popupWindowObject = popupWindowObjects[0];

为了使本节简短,我有意将代码示例限制为访问其他全局window对象的演示您可以使用这些方法来定义全局方法、设置全局变量、调用全局函数等
……前提是页面处于打开状态。有人认为弹出窗口window总是可用的。这不是真的,当弹出窗口关闭时,全局对象就被释放了!

通过消息传递进行通信

消息通道总是有两端:发送者和接收者。
要成为接收者,请使用chrome.runtime.onMessage.addListener方法绑定事件侦听器这可以通过扩展代码和内容脚本来完成。

要在扩展中传递消息,请使用chrome.runtime.sendMessage. 如果要将消息发送到另一个选项卡,请调用chrome.tabs.sendMessage通过包含一个整数 ( tabId) 作为其第一个参数来指定目标选项卡请注意,后台页面只能向一个选项卡发送消息。要访问所有选项卡,必须为每个选项卡调用该方法。例如:

chrome.tabs.query({}, function(tabs) {
    for (var i=0; i<tabs.length; i++) {
        chrome.tabs.sendMessage(tabs[i].id, "some message");
    }
});

内容脚本只能调用chrome.runtime.sendMessage向扩展代码发送消息。如果您想从一个内容脚本向另一个内容脚本发送消息,则需要一个背景/事件页面,它接收一条消息并将其发送到所需的选项卡。有关示例,请参阅此答案

这些sendMessage方法接受一个可选函数,该函数作为onMessage事件的第三个参数被接收

chrome.runtime.onMessage.addListener(function(message, sender, sendResponse) {
    if (message === 'message') sendResponse('the response');
});
chrome.runtime.sendMessage('message', function(response) {
    console('sendResponse was called with: ' + response);
});

前面的示例显示了明显的行为。当您想要异步发送响应时,事情会变得更加棘手,例如,如果您想要执行 AJAX 请求以获取一些数据。onMessage函数没有调用就返回时sendResponse,Chrome 会立即调用sendResponse. 由于sendResponse只能调用一次,您将收到以下错误:

无法发送响应:如果您想在侦听器返回后发送响应,则 chrome.runtime.onMessage 侦听器必须返回 true(消息由扩展名EXTENSION ID HERE发送

按照错误提示,return true;在您的 onMessage 事件侦听器中添加

chrome.runtime.onMessage.addListener(function(message, sender, sendResponse) {
    setTimeout(function() { // Example: asynchronous invocation of sendResponse
        sendResponse('async response');
    }, 200);
    return true;
});

在本节中,我已经解释了简单的一次性消息传递的实际应用。如果您想了解更多有关长寿命消息通道或跨扩展消息的信息,请阅读官方文档中教程

消息传递 API 经历了几次名称更改。如果您阅读旧示例,请记住这一点。可以在此处找到历史记录和兼容性说明

内容脚本和页面之间的通信

可以与页面进行通信。Apsillers 创建了一个很好的答案,解释了如何在(非扩展)页面和内容脚本之间设置通信通道。阅读他在站点可以调用浏览器扩展吗?.

apsiller的在方法的优势从文档中一个是用于自定义事件。该文档用于window.postMessage向页面发送消息,但这可能会导致与不期望消息事件的错误编码页面发生冲突。

哦,伙计,这个答案值得接受和更多的赞成……很好的解释。
2021-04-09 06:42:54

Google 文档拥有一切,但很难将所有信息汇总在一起。脚本有两种主要类型:
1. 后台脚本可以完全访问 Chrome api,但不能与目标网页交互。
2. 内容脚本可以相互交互,也可以与网页的 DOM 交互(但不能与网页的 DOM 交互,与它隔离的脚本不能),但对 Chrome api 的访问权限有限。
每当您加载新页面时两者都会运行(除非您使用“匹配”来限制内容脚本的运行位置)。

您可以通过消息传递在两者之间进行通信这比从后台脚本更容易从内容脚本完成,因为您需要知道后者的选项卡的 id。

其他脚本 ( browserAction.js, pageAction.js, optionsPage.js) 仅在其相应的 html 页面打开时运行(就像您在浏览器窗口中打开网页一样,这就是您真正在做的事情)。它们在限制和能力上类似于后台脚本。

尽量避免与页面脚本交互的需要。我所知道的最好的方法是通过共享 DOM 进行交互(字面意思是在 html 注释中编写 javascript 代码)。但是您的扩展程序的目标不是为此而设计的,因此您必须将自己的脚本包含在网页中。使用内容脚本将脚本元素写入文档(它src
chrome.extension.getURL("myscript.js")
并且您需要
"web_accessible_resources": ["myscript.js"]
在清单中包含它)。

自从我不得不处理 chrome 的扩展已经有一段时间了。我记得在我了解事情的运作方式之前,这是一场相当大的斗争。为了让您的扩展程序与浏览器进行通信,您可以轻松地使用 javascript/背景文件,并与您需要使用的网页进行通信,chrome.tabs.executeScript但这真的很棘手,而且可能会很麻烦。我建议你参加谷歌的扩展之旅,并给他们的 api 一个很好的忽视,一切都在那里!祝你好运,希望这个回答对你有帮助!:P