In this lesson, you will get an introduction to creating a Chrome extension. Extensions are fairly compatible across other browsers with minor tweaking to the main manifest.json file, but adjustments may be necessary for browser API calls, which we will get to later. For this lesson, we will be using Google Chrome as it provides a quick workflow to loading changes to an extension file.
- First, create a new folder in a place that you'll remember.
- Navigate to this folder, create a new file within it, and name the file "manifest.json".
- Open this file in a text editor, and paste the following contents:
{
"manifest_version": 3,
"name": "Hello World",
"version": "1.0",
"description": "HacKSU extension lesson"
}
At this point, we have a functional extension. The manifest.json file is the heart of your Chrome extension, essentially a config file. Web browsers are a constantly evolving software, and behaviors that used to behave a certain way will eventually become deprecated. Manifest version 2 is scheduled to be deprecated in June of 2024.
You may have heard about the anti-adblock changes coming to Chrome later this year (2024). As of now, Adblockers currently run on manifest version 2, and currently rely on large sets of rules (upwards of 300k) which determine whether content on a page is an ad. You can view some example rule lists here. The forced upgrade to "manifest_version": 3
significantly limits the number of rules that an extension can run (up to 30k), which greatly limits the effectiveness of adblockers.
For our purposes, Manifest version 3 signifies that our web extension is up to date with the latest features and web standards, but this is not necessary to develop an extension locally.
As an optional step, we can add an icon to our extension. Add an icon image file to your project somewhere within your folder (commonly within a subfolder, /images/
), and add the following to the manifest.json file.
"icons": {
"128": "[your-image-path-here]"
},
In my case, I use "128": "images/icon128.png"
You can specify multiple icon sizes if you wish (128, 48, 32, 16).
- Save the manifest.json file.
- Paste
chrome://extensions
in your address bar and enter. This is the main page we will be visiting to manage our extensions.
- Enable the "Developer mode" toggle on the top right. If this is not enabled, you cannot load custom extensions.
- Click the now visibile "Load unpacked" button on the top left. In the file explorer, locate the folder that you created in step 1.
We have now enabled our extension. Currently, this extension is doing nothing. But we'll add some functionality soon.
You can disable this extension at any time by clicking the toggle at the bottom right of our extension item.
Let's create a script that runs on a webpage. We will first need to make some modifications to our manifest.json file to tell Chrome to give us this permission.
- Place a comma after the last key-value pair item of our manifest, and then paste the following key-value pairs, before the ending bracket.
"content_scripts": [
{
"matches": [
"https://www.example.com/"
],
"js": [
"script.js"
]
}
]
"content_scripts" allows your extension to inject scripts into existing web pages. You specify what script file you will use within "js", and all pages that this script should modify within "matches." With these key-value pairs, we are telling Chrome that we want to run all JavaScript files in the list within the key "js", on all websites listed within the "matches" key.
Note: Despite these keys having one value each, these square brackets are NOT optional. JSON syntax does not support type coersion between one-element lists and strings.
Now, it is time to create our script.
-
Create a new file within the folder you created at the start of the lesson, and name the file
script.js
. -
Place the following JavaScript code within the file, and save.
alert("buzz buzz");
Our extension is ready for testing.
- Navigate to the extensions page and refresh the extension (Bottom right of the extension, next to the toggle).
We will need to hit the refresh button each time that we make a change to our extension. I'd recommend keeping this extensions page open in one of your tabs.
For now, we've only configured our extension to run on https://www.example.com. In the future, if you wish to run a script on all websites, you can replace the URL with the token "<all urls>"
- Visit https://www.example.com, you will be greeted by an alert.
In addition to dialogues, we can also inject HTML elements into the page, or alter the appearance of existing content. Append the following to the JavaScript file:
// hide p element
const hideMe = document.getElementsByTagName("p");
for (const e of hideMe) {
e.style.display = 'none';
}
Saving our changes, the p elements in our document are no longer visible.
Let's try something more fun. CSS has a couple of built-in filters, which we can choose to apply to the entirety of the loaded webpage.
// invert the page colors
document.documentElement.style.filter = 'invert(100%)';
After saving our extension, the entire page inverts colors once the page loads.
Let's inject an image into the page.
- Add the following to manifest.json, and be sure to replace
[image-path-here]
with whatever path your image uses.
"web_accessible_resources": [
{
"resources": [
"[image-path-here]"
],
"matches": [
"<all_urls>"
]
}
]
In this example, we are using the <all_urls>
keyword to modify the appearance of all pages. If you do not wish to do this, you can replace this with a specific URL as done earlier in the lesson.
- Add the following to
script.js
, again, replace[image-path-here]
with the path to your image.${chrome.runtime.id}
brings you to the base folder of your extension.
// overlay an image on the center of the page
let img = document.createElement('img');
img.src = `chrome-extension://${chrome.runtime.id}/[image-path-here]`;
img.style.cssText = `
position: fixed;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
z-index: 9999;
`;
document.body.appendChild(img);
Save and reload the extension.
Many extensions have a dialog that appears when an extension's icon is clicked.
- We first need to add a new key-value pair in our
manifest.json
file.
"action": {
"default_popup": "popup.html"
}
This specifies the page that opens when the extension icon is clicked in the browser toolbar. Now, we need to create the popup page.
- Create a new file (within the folder created at the beginning) and name it
popup.html
. Open the file with any text editor, and add the following HTML:
<h1>:)</h1>
Make sure all of your files are saved. When reloading our extension and clicking on the icon (you may need to find your extension by clicking the "puzzle piece" icon and pinning it to your toolbar) we should see the following:
Let's add a second page to navigate to within our popup.
- Append
popup.html
with the following:
<a href="popup2.html">Go to a new page</a>
- Create new file, and name it
popup2.html
. Open the file and add the following HTML:
<p>You found me!</p>
<a href="popup.html">Go back</a>
We now are able to switch between two pages of our extension.
- Let's try adding the following link (it will not work just yet)
<a href="https://hacksu.com/cute_puppy.jpg">Feel better!</a>
We can open a link in a new tab by adding target="_blank"
to our HTML element, however, maybe we want to open a link in the current browser tab?
To do this, we need access the tabs Chrome extension API.
- Add the following to the
manifest.json
file to grant permission to use the "tabs" API.
"permissions": [
"tabs"
],
This allows us to use methods such as chrome.tabs.create()
, which requires JavaScript to execute.
- Create a new file,
popup.js
, and reference this new file at the bottom of popup.html.
<h1>:)</h1>
<a href="popup2.html">Go to a new page</a>
<a href="https://hacksu.com/cute_puppy.jpg">Feel better!</a>
<script src="popup.js"></script>
- Within popup.js, add the following code:
// bind the link element to Chrome new tab event
window.addEventListener('click', (e) => {
if (e.target.href !== undefined) {
chrome.tabs.update({ url: e.target.href })
}
})
You don't need to understand all of this code. In a nutshell, this code is checking if a clicked element has an href attribute, and if so, passes that value to the current active Chrome tab.
However, a side effect of this code is that it causes unwanted side effects to the functionality of our previous link. One way to differentiate this behavior is by qualifying classes to your elements, and checking for the existence of this qualifier.
- Update the JavaScript code in
popup.js
as follows:
// bind the link element to Chrome new tab event
window.addEventListener('click', (e) => {
if (e.target.href !== undefined && e.target.classList.contains("current-tab")) {
chrome.tabs.update({ url: e.target.href })
}
})
and our HTML element:
<a href="https://hacksu.com/cute_puppy.jpg" class="current-tab">Feel better!</a>
Now, we are able to differentiate our links depending on behavior needed.
- Modify the permissions to your manifest.json file.
"permissions": [
"tabs",
"activeTab",
"scripting"
],
- Add a button element to your popup.html file, and give it an id
<button id="invert">Invert</button>
- Add the following code to your popup.js
// add functionality to invert button
document.getElementById('invert').addEventListener('click', () => {
// query active tabs
chrome.tabs.query({ active: true, currentWindow: true }, tabs => {
// execute script to invert colors
chrome.scripting.executeScript({
// target the active tab
target: { tabId: tabs[0].id },
function: () => {
let filterValue = document.documentElement.style.filter;
if (filterValue.includes('invert')) {
document.documentElement.style.filter = 'none';
} else {
document.documentElement.style.filter = 'invert(100%)';
}
}
});
});
});
Now, we can press a button on our popup and have it modify the current tab appearance.
If you wish to explore extension development further, one of the easiest ways to do so is to mess with existing extensions. You can find some example extensions on the Chrome Extensions documentation page.
-
Find a project that seems interesting to you.
-
Clone the repo here: https://github.com/GoogleChrome/chrome-extensions-samples.git
-
Load the unpacked extension in your browser, look at the code behind it, see if you can follow along with what makes it work, and experiment with modifications to change its functionality.
Happy hacking!