electron/lib/renderer/content-scripts-injector.js

92 lines
2.9 KiB
JavaScript
Raw Normal View History

2016-05-28 07:41:12 +00:00
const {ipcRenderer} = require('electron')
2016-05-28 06:37:44 +00:00
const {runInThisContext} = require('vm')
// Check whether pattern matches.
// https://developer.chrome.com/extensions/match_patterns
const matchesPattern = function (pattern) {
2016-05-27 02:07:06 +00:00
if (pattern === '<all_urls>') return true
const regexp = new RegExp(`^${pattern.replace(/\*/g, '.*')}$`)
2017-07-10 21:50:59 +00:00
const url = `${location.protocol}//${location.host}${location.pathname}`
return url.match(regexp)
}
2016-05-28 06:37:44 +00:00
// Run the code with chrome API integrated.
const runContentScript = function (extensionId, url, code) {
const context = {}
require('./chrome-api').injectTo(extensionId, false, context)
2017-09-17 05:56:22 +00:00
const wrapper = `((chrome) => {\n ${code}\n })`
2016-05-28 06:37:44 +00:00
const compiledWrapper = runInThisContext(wrapper, {
filename: url,
lineOffset: 1,
displayErrors: true
})
return compiledWrapper.call(this, context.chrome)
2016-05-28 06:37:44 +00:00
}
const runStylesheet = function (url, code) {
2017-09-17 05:56:22 +00:00
const wrapper = `((code) => {
function init() {
2017-09-17 05:56:22 +00:00
const styleElement = document.createElement('style');
styleElement.textContent = code;
document.head.append(styleElement);
}
document.addEventListener('DOMContentLoaded', init);
2017-09-17 06:09:12 +00:00
})`
const compiledWrapper = runInThisContext(wrapper, {
filename: url,
lineOffset: 1,
2017-09-17 06:09:12 +00:00
displayErrors: true
})
return compiledWrapper.call(this, code)
}
// Run injected scripts.
// https://developer.chrome.com/extensions/content_scripts
const injectContentScript = function (extensionId, script) {
2017-07-20 18:01:49 +00:00
if (!script.matches.some(matchesPattern)) return
2017-09-17 05:56:22 +00:00
if (script.js) {
2017-07-20 18:24:39 +00:00
for (const {url, code} of script.js) {
const fire = runContentScript.bind(window, extensionId, url, code)
if (script.runAt === 'document_start') {
process.once('document-start', fire)
} else if (script.runAt === 'document_end') {
process.once('document-end', fire)
2017-09-17 05:56:22 +00:00
} else {
2017-07-20 18:24:39 +00:00
document.addEventListener('DOMContentLoaded', fire)
}
}
}
2017-09-17 05:56:22 +00:00
if (script.css) {
for (const {url, code} of script.css) {
const fire = runStylesheet.bind(window, url, code)
if (script.runAt === 'document_start') {
process.once('document-start', fire)
} else if (script.runAt === 'document_end') {
process.once('document-end', fire)
2017-09-17 05:56:22 +00:00
} else {
document.addEventListener('DOMContentLoaded', fire)
}
}
}
}
2016-05-28 07:41:12 +00:00
// Handle the request of chrome.tabs.executeJavaScript.
ipcRenderer.on('CHROME_TABS_EXECUTESCRIPT', function (event, senderWebContentsId, requestId, extensionId, url, code) {
2016-05-28 07:41:12 +00:00
const result = runContentScript.call(window, extensionId, url, code)
ipcRenderer.sendToAll(senderWebContentsId, `CHROME_TABS_EXECUTESCRIPT_RESULT_${requestId}`, result)
2016-05-28 07:41:12 +00:00
})
// Read the renderer process preferences.
2016-05-27 02:07:06 +00:00
const preferences = process.getRenderProcessPreferences()
if (preferences) {
for (const pref of preferences) {
if (pref.contentScripts) {
for (const script of pref.contentScripts) {
injectContentScript(pref.extensionId, script)
}
2016-05-27 02:07:06 +00:00
}
}
}