Aujourd’hui, j’ai cherché à injecter du code dans une page HTML depuis une extension Firefox. Ca permet de définir des objects dans le scope de la fenêtre, qui seront ensuite accessibles depuis le code javascript de la page. J’ai aussi cherché à communiquer dans l’autre sens : appeler des fonctions de mon extension depuis la page web.
Injection de script dans une page Web
L’injection se passe en deux étapes :
- Il faut savoir sur quelle page intervenir : on peut donc se binder sur l’évènement
DOMContentLoaded
du navigateur ; - Il faut ensuite injecter le code à proprement parler, et qu’il soit visible.
Etape 1
Voici le code commenté (à inclure dans browser.xul) :
- var Injector =
- {
- init: function()
- {
- var appcontent = document.getElementById('appcontent'); // On récupère le navigateur
- if(appcontent) // On se bind sur le bon évènement
- appcontent.addEventListener('DOMContentLoaded', Injector.onPageLoaded, true);
- },
- onPageLoaded: function(aEvent)
- {
- if(aEvent.target instanceOf HTMLDocument) // Si on est sur un document HTML
- {
- var doc = aEvent.originalTarget; // Voici le HTMLDocument
- var win = doc.defaultView; // Voici la DOMWindow
- }
- }
- }
2. Injection de code
Et voici comment injecter :
- /**
- * Permet d'injecter du javascript dans une page
- * @param win DOMWindow La fenêtre dans laquelle injecter le script
- * @param id string Un id unique pour identifier le script
- * @param src string Le code source
- * @param [optional] boolean bRemove Vrai si on veut enlever le script de la page après son inclusion
- */
- Injector.addScript = function(win, id, src, bRemove)
- {
- // On crée un élément script
- var element = win.document.createElementNS('http://www.w3.org/1999/xhtml', 'html:script');
- element.setAttribute('type', 'text/javascript');
- element.setAttribute('id', id);
- element.innerHTML = src;
- // Et on l'ajoute !
- win.document.documentElement.appendChild(element);
- // On supprime l'élément si demandé
- if(bRemove) element.parentNode.removeChild(element);
- }
Communiquer avec un script qui tourne dans le browser.xul
Communiquer dans l’autre sens a des intérêts : ça permet d’accéder à des fonctions plus avancée (accès à toutes les librairies XPCOM !).
La méthode est plutôt simple : on crée un élément dans le HTMLDocument
(utile pour faire passer des données, en utilisant des attributs auxquels on aura ensuite accès), et on envoie alors un évènement personnalisé, sur lequel écoute notre Injector
.
1. Ecoute des évènements de la page
On modifie le onPageLoad
pour rajouter le bind sur les évènements, que l’on va ensuite traiter avec la méthode onEvent
:
- [...]
- doc.addEventListener('injector-event', Injector.onEvent, false, true);
- [...]
- Injector.onEvent = function(aEvent)
- {
- // On peut récupérer l'élément sur lequel a été passé l'évènement
- var element = aEvent.target;
- // Et le document associé
- var doc = element.ownerDocument;
- // Faites ensuite ce que vous voulez. Vous pouvez utiliser la méthode element.getAttribute pour récupérer d'éventuels arguments, éventuellement encodés avec JSON.stringify puis décodés avec JSON.parse
- }
2. Envoi d’un évènement
Et voici le code à mettre dans votre page HTML :
- // Creation de l'évènement
- var ev = document.createEvent('Events');
- ev.initEvent('injector-event', true, false);
- // On utilise documentElement pour envoyer le message. Vous pouvez utiliser un élément personnalisé, n'importe où dans le code
- document.documentElement.dispatchEvent(ev);