博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
谷歌扩展程序--------------Message
阅读量:4347 次
发布时间:2019-06-07

本文共 10915 字,大约阅读时间需要 36 分钟。

转载:https://developer.chrome.com/extensions/messaging#external-webpage

Message Passing

Since content scripts run in the context of a web page and not the extension, they often need some way of communicating with the rest of the extension. For example, an RSS reader extension might use content scripts to detect the presence of an RSS feed on a page, then notify the background page in order to display a page action icon for that page.

Communication between extensions and their content scripts works by using message passing. Either side can listen for messages sent from the other end, and respond on the same channel. A message can contain any valid JSON object (null, boolean, number, string, array, or object). There is a simple API for  and a more complex API that allows you to have  for exchanging multiple messages with a shared context. It is also possible to send a message to another extension if you know its ID, which is covered in the  section.

If you only need to send a single message to another part of your extension (and optionally get a response back), you should use the simplified  or  . This lets you send a one-time JSON-serializable message from a content script to extension , or vice versa, respectively . An optional callback parameter allows you handle the response from the other side, if there is one.

Sending a request from a content script looks like this:

chrome.runtime.sendMessage({ greeting: "hello"}, function(response) { console.log(response.farewell); });

Sending a request from the extension to a content script looks very similar, except that you need to specify which tab to send it to. This example demonstrates sending a message to the content script in the selected tab.

chrome.tabs.query({ active: true, currentWindow: true}, function(tabs) { chrome.tabs.sendMessage(tabs[0].id, { greeting: "hello"}, function(response) { console.log(response.farewell); }); });

On the receiving end, you need to set up an  event listener to handle the message. This looks the same from a content script or extension page.

chrome.runtime.onMessage.addListener( function(request, sender, sendResponse) { console.log(sender.tab ? "from a content script:" + sender.tab.url : "from the extension"); if (request.greeting == "hello") sendResponse({ farewell: "goodbye"}); });
In the above example, 
sendResponse was called synchronously. If you want to asynchronously use 
sendResponse, add 
return true; to the 
onMessage event handler.

Note: If multiple pages are listening for onMessage events, only the first to call sendResponse() for a particular event will succeed in sending the response. All other responses to that event will be ignored.

Note: The sendResponse callback is only valid if used synchronously, or if the event handler returns true to indicate that it will respond asynchronously. The sendMessagefunction's callback will be invoked automatically if no handlers return true or if thesendResponse callback is garbage-collected.

Sometimes it's useful to have a conversation that lasts longer than a single request and response. In this case, you can open a long-lived channel from your content script to an extension page , or vice versa, using  or , respectively . The channel can optionally have a name, allowing you to distinguish between different types of connections.

One use case might be an automatic form fill extension. The content script could open a channel to the extension page for a particular login, and send a message to the extension for each input element on the page to request the form data to fill in. The shared connection allows the extension to keep shared state linking the several messages coming from the content script.

When establishing a connection, each end is given a  object which is used for sending and receiving messages through that connection.

Here is how you open a channel from a content script, and send and listen for messages:

var port = chrome.runtime.connect({ name: "knockknock"}); port.postMessage({ joke: "Knock knock"}); port.onMessage.addListener(function(msg) { if (msg.question == "Who's there?") port.postMessage({ answer: "Madame"}); else if (msg.question == "Madame who?") port.postMessage({ answer: "Madame... Bovary"}); });

Sending a request from the extension to a content script looks very similar, except that you need to specify which tab to connect to. Simply replace the call to connect in the above example with .

In order to handle incoming connections, you need to set up a  event listener. This looks the same from a content script or an extension page. When another part of your extension calls "connect()", this event is fired, along with the  object you can use to send and receive messages through the connection. Here's what it looks like to respond to incoming connections:

chrome.runtime.onConnect.addListener(function(port) { console.assert(port.name == "knockknock"); port.onMessage.addListener(function(msg) { if (msg.joke == "Knock knock") port.postMessage({ question: "Who's there?"}); else if (msg.answer == "Madame") port.postMessage({ question: "Madame who?"}); else if (msg.answer == "Madame... Bovary") port.postMessage({ question: "I don't get it."}); }); });

Ports are designed as a two-way communication method between different parts of the extension, where a (top-level) frame is viewed as the smallest part. 

Upon calling ,  or , a  is created. This port can immediately be used for sending messages to the other end via .

If there are multiple frames in a tab, calling  results in multiple invocations of the  event (once for each frame in the tab). Similarly, if  is used, then the onConnect event may be fired multiple times (once for every frame in the extension process).

You may want to find out when a connection is closed, for example if you are maintaining separate state for each open port. For this you can listen to the  event. This event is fired when there are no valid ports at the other side of the channel. This happens in the following situations:

  • There are no listeners for  at the other end.
  • The tab containing the port is unloaded (e.g. if the tab is navigated).
  • The frame from where connect was called has unloaded.
  • All frames that received the port (via ) have unloaded.
  •  is called by the other end. Note that if a connect call results in multiple ports at the receiver's end, and disconnect() is called on any of these ports, then the onDisconnect event is only fired at the port of the sender, and not at the other ports.

In addition to sending messages between different components in your extension, you can use the messaging API to communicate with other extensions. This lets you expose a public API that other extensions can take advantage of.

Listening for incoming requests and connections is similar to the internal case, except you use the or  methods. Here's an example of each:

// For simple requests:chrome.runtime.onMessageExternal.addListener( function(request, sender, sendResponse) { if (sender.id == blocklistedExtension) return; // don't allow this extension access else if (request.getTargetData) sendResponse({ targetData: targetData}); else if (request.activateLasers) { var success = activateLasers(); sendResponse({ activateLasers: success}); } }); // For long-lived connections: chrome.runtime.onConnectExternal.addListener(function(port) { port.onMessage.addListener(function(msg) { // See other examples for sample onMessage handlers. }); });

Likewise, sending a message to another extension is similar to sending one within your extension. The only difference is that you must pass the ID of the extension you want to communicate with. For example:

// The ID of the extension we want to talk to.var laserExtensionId = "abcdefghijklmnoabcdefhijklmnoabc"; // Make a simple request: chrome.runtime.sendMessage(laserExtensionId, { getTargetData: true}, function(response) { if (targetInRange(response.targetData)) chrome.runtime.sendMessage(laserExtensionId, { activateLasers: true}); }); // Start a long-running conversation: var port = chrome.runtime.connect(laserExtensionId); port.postMessage(...);

Similar to , your app or extension can receive and respond to messages from regular web pages. To use this feature, you must first specify in your manifest.json which web sites you want to communicate with. For example:

"externally_connectable": {
"matches": ["*://*.example.com/*"] }

This will expose the messaging API to any page which matches the URL patterns you specify. The URL pattern must contain at least a  - that is, hostname patterns like "*", "*.com", "*.co.uk", and "*.appspot.com" are prohibited. From the web page, use the  or  APIs to send a message to a specific app or extension. For example:

// The ID of the extension we want to talk to.var editorExtensionId = "abcdefghijklmnoabcdefhijklmnoabc"; // Make a simple request: chrome.runtime.sendMessage(editorExtensionId, { openUrlInEditor: url}, function(response) { if (!response.success) handleError(url); });

From your app or extension, you may listen to messages from web pages via the or  APIs, similar to . Only the web page can initiate a connection. Here is an example:

chrome.runtime.onMessageExternal.addListener( function(request, sender, sendResponse) { if (sender.url == blocklistedWebsite) return; // don't allow this web page access if (request.openUrlInEditor) openUrl(request.openUrlInEditor); });

Extensions and apps  with native applications that are registered as a . To learn more about this feature, see .

When receiving a message from a content script or another extension, your background page should be careful not to fall victim to . Specifically, avoid using dangerous APIs such as the below:

chrome.tabs.sendMessage(tab.id, { greeting: "hello"}, function(response) { // WARNING! Might be evaluating an evil script! var resp = eval("(" + response.farewell + ")"); });
chrome.tabs.sendMessage(tab.id, { greeting: "hello"}, function(response) { // WARNING! Might be injecting a malicious script! document.getElementById("resp").innerHTML = response.farewell; });

Instead, prefer safer APIs that do not run scripts:

chrome.tabs.sendMessage(tab.id, { greeting: "hello"}, function(response) { // JSON.parse does not evaluate the attacker's scripts. var resp = JSON.parse(response.farewell); });
chrome.tabs.sendMessage(tab.id, { greeting: "hello"}, function(response) { // innerText does not let the attacker inject HTML elements. document.getElementById("resp").innerText = response.farewell; });

You can find simple examples of communication via messages in the  directory. The  demonstrates how a Chrome app can communicate with a native app. For more examples and for help in viewing the source code, see .

 

转载于:https://www.cnblogs.com/h2zZhou/p/9303863.html

你可能感兴趣的文章
PAT 1083 是否存在相等的差(20)(代码+思路)
查看>>
第三天冲刺
查看>>
ajax简介+原生ajax代码
查看>>
【C++基础】GDB调试C++程序
查看>>
noip 2014 总结
查看>>
Python自然语言处理学习笔记(55): 最大熵分类器
查看>>
UIView的动画效果
查看>>
Python学习目录
查看>>
bom
查看>>
HTML页面导出PDF——高清版
查看>>
[翻译] ALMoviePlayerController
查看>>
极客学院录制课程合集
查看>>
C语言的第一次实验报告
查看>>
机器学习算法(一):线性回归算法
查看>>
(06)mongodb 游标与分页
查看>>
spring JDBC 批量插入数据
查看>>
状态压缩题目小结
查看>>
Android WebView 开发具体解释(三)
查看>>
2016-2017-2 20155325实验二《Java面向对象程序设计》实验报告
查看>>
POJ.3145.Common Substrings(后缀数组 倍增 单调栈)
查看>>