runtime.onMessage
Utilisez cet événement pour écouter les messages d'une autre partie de votre extension.
Voici quelques exemples de cas d'utilisation :
- dans un script de contenu, pour écouter les messages d'un script d'arrière-plan ;
- dans un script d'arrière-plan, pour écouter les messages d'un script de contenu ;
- dans une page d'options ou un script de popup, pour écouter les messages d'un script d'arrière-plan ;
- dans un script d'arrière-plan, pour écouter les messages d'une page d'options ou d'un script de popup.
Pour envoyer un message reçu par l'écouteur onMessage
, utilisez runtime.sendMessage()
ou (pour envoyer un message à un script de contenu) tabs.sendMessage()
.
Note :
Évitez de créer plusieurs écouteurs onMessage
pour le même type de message, car l'ordre de déclenchement des différents écouteurs ne sera pas garanti.
Lorsque vous voulez garantir la livraison d'un message à une terminaison spécifique, utilisez l'approche basée sur la connexion pour échanger des messages.
En plus du message, l'écouteur reçoit en paramètres :
- Un objet
sender
donnant les détails sur l'expéditeur du message ; - Une fonction
sendResponse()
qui peut être utilisée pour renvoyer une réponse à l'expéditeur.
Vous pouvez envoyer une réponse synchrone au message en appelant la fonction sendResponse()
dans votre écouteur. Voir un exemple.
Pour envoyer une réponse asynchrone, il existe deux options :
- Renvoyer
true
à partir de l'écouteur d'événement. Cela permet de conserver la fonctionsendResponse()
après le retour de l'écouteur pour éventuellement l'appeler plus tard. Voir un exemple. - Renvoyer une
Promise
depuis l'écouteur d'événement, et la résoudre lorsque vous avez la réponse (ou la rejeter en cas d'erreur). Voir un exemple.
Attention :
Retourner une promesse (Promise
) est désormais la méthode à privilégier car sendResponse()
sera retirée de la spécification W3C.
La bibliothèque populaire webextension-polyfill a déjà supprimé cette fonction de son implémentation.
Note : Vous pouvez également utiliser une approche basée sur la connexion pour échanger des messages.
Syntaxe
browser.runtime.onMessage.addListener(listener);
browser.runtime.onMessage.removeListener(listener);
browser.runtime.onMessage.hasListener(listener);
Les événements ont trois fonctions :
addListener(listener)
-
Ajoute un écouteur à cet événement.
removeListener(listener)
-
Cesse d'écouter cet événement. L'argument
listener
est l'écouteur à supprimer. hasListener(listener)
-
Vérifie si un
listener
est enregistré pour cet événement. Retournetrue
s'il écoute,false
sinon.
Syntaxe de addListener
Paramètres
listener
-
Une fonction d'écoute qui sera appelée lorsque cet événement se produira. La fonction recevra les arguments suivants :
message
-
Un objet qui est le message lui-même. C'est un objet sérialisable (voir l'algorithme de clonage de données).
sender
-
Un objet
runtime.MessageSender
représentant l'expéditeur du message.
sendResponse
-
Une fonction à appeler, au plus une fois, pour envoyer une réponse au
message
. La fonction prend un seul argument, qui peut être n'importe quel objet sérialisable (voir l'algorithme de clonage de données). Cet argument est renvoyé à l'expéditeur du message.Si vous avez plus d'un écouteur
onMessage()
dans le même document, alors un seul peut envoyer une réponse.Pour envoyer une réponse de manière synchrone, appelez
sendResponse()
avant le retour de la fonction d'écoute.Pour envoyer une réponse de manière asynchrone :
- soit on gardera une référence à l'argument
sendResponse()
et on retourneratrue
depuis la fonction listenener.sendResponse()
pourra être appelée après le retour de la fonction d'écoute. - ou on retournera
Promise
à partir de la fonction d'écoute et on résoudra la promesse lorsque la réponse sera prête. C'est la méthode à privilégier.
- soit on gardera une référence à l'argument
La fonction
listener
peut renvoyer un booléen ou unePromise
.Note : N'appelez pas
addListener()
en utilisant une fonctionasync
:js// ne faites pas ça browser.runtime.onMessage.addListener(async (data, sender) => { if (data.type === "handle_me") { return "done"; } });
L'écouteur consommera ainsi chaque message qu'il reçoit, ce qui empêchera effectivement tous les autres écouteurs de recevoir et de traiter des messages.
Si vous souhaitez adopter une approche asynchrone, utilisez plutôt une
Promise
, comme ceci :jsbrowser.runtime.onMessage.addListener((data, sender) => { if (data.type === "handle_me") { return Promise.resolve("done"); } });
Compatibilité des navigateurs
BCD tables only load in the browser
Exemples
Exemple simple
Ce script de contenu écoute les événements clic dans la page web. Si le clic a eu lieu sur un lien, il envoie un message à la page d'arrière-plan avec l'URL cible :
// content-script.js
window.addEventListener("click", notifyExtension);
function notifyExtension(e) {
if (e.target.tagName != "A") {
return;
}
browser.runtime.sendMessage({ url: e.target.href });
}
Le script d'arrière-plan écoute ces messages et affiche une notification à l'aide de l'API notifications
.
// background-script.js
browser.runtime.onMessage.addListener(notify);
function notify(message) {
browser.notifications.create({
type: "basic",
iconUrl: browser.extension.getURL("link.png"),
title: "Vous avez cliqué sur un lien !",
message: message.url,
});
}
Envoyer une réponse synchrone
Le script de contenu suivant envoie un message au script d'arrière-plan lorsque l'utilisateur ou l'utilisatrice clique sur la page. Il enregistre également toute réponse envoyée par le script d'arrière-plan :
// content-script.js
function handleResponse(message) {
console.log(`le script d’arrière-plan a répondu : ${message.response}`);
}
function handleError(error) {
console.log(`Erreur : ${error}`);
}
function sendMessage(e) {
var sending = browser.runtime.sendMessage({
content: "message du script de contenu",
});
sending.then(handleResponse, handleError);
}
window.addEventListener("click", sendMessage);
Voici une version du script d'arrière-plan correspondant, qui envoie une réponse de manière synchrone depuis l'intérieur de l'écouteur :
// background-script.js
function handleMessage(request, sender, sendResponse) {
console.log(`le script de contenu a envoyé un message : ${request.content}`);
sendResponse({ response: "réponse du script d’arrière-plan" });
}
browser.runtime.onMessage.addListener(handleMessage);
Et voici une autre version, qui utilise Promise.resolve()
:
// background-script.js
function handleMessage(request, sender, sendResponse) {
console.log(`le script de contenu a envoyé un message : ${request.content}`);
return Promise.resolve({ response: "réponse du script d’arrière-plan" });
}
browser.runtime.onMessage.addListener(handleMessage);
Envoi d'une réponse asynchrone à l'aide de sendResponse
Voici une autre version du script d'arrière-plan de l'exemple précédent. Il envoie une réponse de manière asynchrone, après le retour de l'écouteur. Remarquez le return true;
dans l'écouteur : cela indique au navigateur que vous avez l'intention d'utiliser l'argument sendResponse()
après le retour de l'écouteur.
// background-script.js
function handleMessage(request, sender, sendResponse) {
console.log(`le script de contenu a envoyé un message : ${request.content}`);
setTimeout(() => {
sendResponse({ response: "réponse asynchrone du script d’arrière-plan" });
}, 1000);
return true;
}
browser.runtime.onMessage.addListener(handleMessage);
Envoi d'une réponse asynchrone à l'aide d'une promesse
Ce script de contenu reçoit le premier lien <a>
dans la page, et envoie un message demandant si l'emplacement du lien fait partie des marque-pages. Il attend comme réponse un booléen
: true
si l'emplacement est dans les marque-pages, false
sinon.
// content-script.js
const firstLink = document.querySelector("a");
function handleResponse(isBookmarked) {
if (isBookmarked) {
firstLink.classList.add("bookmarked");
}
}
browser.runtime
.sendMessage({
url: firstLink.href,
})
.then(handleResponse);
Voici le script d'arrière-plan. Il utilise
pour voir si le lien est dans les marque-pages, ce qui renvoie une bookmarks.search()
promesse
:
// background-script.js
function isBookmarked(message, sender, response) {
return browser.bookmarks
.search({
url: message.url,
})
.then(function (results) {
return results.length > 0;
});
}
browser.runtime.onMessage.addListener(isBookmarked);
Si le gestionnaire asynchrone ne renvoie pas de promesse, vous pouvez explicitement construire une promesse. Cet exemple plutôt artificiel envoie une réponse après un délai d'une seconde, en utilisant Window.setTimeout()
:
// background-script.js
function handleMessage(request, sender, sendResponse) {
return new Promise((resolve) => {
setTimeout(() => {
resolve({ response: "réponse asynchrone du script d’arrière-plan" });
}, 1000);
});
}
browser.runtime.onMessage.addListener(handleMessage);
Example extensions
- beastify
- content-script-register
- cookie-bg-picker
- devtools-panels
- export-helpers
- find-across-tabs
- imagify
- mocha-client-tests
- notify-link-clicks-i18n
- store-collected-images
- user-script-register
- webpack-modules
Note :
Cette API est basée sur l'API Chromium chrome.runtime
. Cette documentation est dérivée de runtime.json
dans le code de Chromium code.
Les données de compatibilité relatives à Microsoft Edge sont fournies par Microsoft Corporation et incluses ici sous la licence Creative Commons Attribution 3.0 pour les États-Unis.