Initial commit
This commit is contained in:
commit
3454be6cf7
12 changed files with 302 additions and 0 deletions
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
*.xpi
|
BIN
icon.afdesign
Normal file
BIN
icon.afdesign
Normal file
Binary file not shown.
8
icon.svg
Normal file
8
icon.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 147 KiB |
30
manifest.json
Normal file
30
manifest.json
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
{
|
||||||
|
"background": {
|
||||||
|
"scripts": ["scripts/background.js"],
|
||||||
|
"type": "module"
|
||||||
|
},
|
||||||
|
"browser_specific_settings": {
|
||||||
|
"gecko": {
|
||||||
|
"id": "tb.spaceapper@mmk2410.org",
|
||||||
|
"strict_min_version": "115.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"description": "Add webpages to the spaces toolbar.",
|
||||||
|
"developer": {
|
||||||
|
"name": "Marcel Kapfer",
|
||||||
|
"url": "https://mmk2410.org"
|
||||||
|
},
|
||||||
|
"host_permissions": ["*://*/*"],
|
||||||
|
"icons": {
|
||||||
|
"48": "icon.svg",
|
||||||
|
"96": "icon.svg"
|
||||||
|
},
|
||||||
|
"manifest_version": 3,
|
||||||
|
"name": "SpaceApper",
|
||||||
|
"options_ui": {
|
||||||
|
"page": "options/options.html",
|
||||||
|
"browser_style": false
|
||||||
|
},
|
||||||
|
"permissions": ["storage", "webRequest", "webRequestBlocking"],
|
||||||
|
"version": "0.1.0"
|
||||||
|
}
|
64
options/options.css
Normal file
64
options/options.css
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
html {
|
||||||
|
font-family: sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
button {
|
||||||
|
background-color: #e7e7e7;
|
||||||
|
border-radius: 5px;
|
||||||
|
border: none;
|
||||||
|
padding: 5px 10px;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-primary {
|
||||||
|
color: #fff;
|
||||||
|
background-color: #7a01d0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces {
|
||||||
|
width: 100%;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces td {
|
||||||
|
border-bottom: 1px solid #999;
|
||||||
|
border-top: 1px solid #999;
|
||||||
|
border-left: 1px solid #ddd;
|
||||||
|
border-right: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces td:first-of-type {
|
||||||
|
border-left: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces td:last-of-type {
|
||||||
|
border-right: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces thead td {
|
||||||
|
font-weight: bold;
|
||||||
|
padding: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces tbody input {
|
||||||
|
border: none;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
font-size: inherit;
|
||||||
|
padding: 10px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
#spaces tbody button {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#actions {
|
||||||
|
width: 100%;
|
||||||
|
margin: 10px 0;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
#actions button:first-of-type {
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
31
options/options.html
Normal file
31
options/options.html
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
<!doctype html>
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<link rel="stylesheet" href="options.css" />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<form>
|
||||||
|
<table id="spaces">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<td>Name</td>
|
||||||
|
<td>URL</td>
|
||||||
|
<td>Icon URL</td>
|
||||||
|
<td></td>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody></tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<div id="actions">
|
||||||
|
<button id="add" type="button">Add</button>
|
||||||
|
<button type="submit" class="btn-primary">Save</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<script type="module" src="options.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
81
options/options.js
Normal file
81
options/options.js
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
import { createSpaces, removeSpaces, hashSpaceName } from "../scripts/spaces.mjs";
|
||||||
|
import { getApps, storeApps } from "../scripts/storage.mjs";
|
||||||
|
|
||||||
|
function buildInputRow(name, value) {
|
||||||
|
const td = document.createElement("td");
|
||||||
|
|
||||||
|
const input = document.createElement("input");
|
||||||
|
input.setAttribute("type", "text");
|
||||||
|
input.setAttribute("name", name);
|
||||||
|
input.setAttribute("value", value);
|
||||||
|
|
||||||
|
td.appendChild(input);
|
||||||
|
return td;
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildSpaceElement(app) {
|
||||||
|
const tr = document.createElement("tr");
|
||||||
|
tr.className = "space";
|
||||||
|
|
||||||
|
tr.appendChild(buildInputRow("title", app.title));
|
||||||
|
tr.appendChild(buildInputRow("url", app.url));
|
||||||
|
tr.appendChild(buildInputRow("icon", app.icon));
|
||||||
|
|
||||||
|
const removeTd = document.createElement("td");
|
||||||
|
const removeButton = document.createElement("button");
|
||||||
|
removeButton.setAttribute("type", "button");
|
||||||
|
removeButton.addEventListener("click", removeSpace);
|
||||||
|
removeButton.textContent = "Remove";
|
||||||
|
removeTd.appendChild(removeButton);
|
||||||
|
|
||||||
|
tr.appendChild(removeTd);
|
||||||
|
return tr;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function setupPage() {
|
||||||
|
const apps = await getApps();
|
||||||
|
const spaces = document.querySelector("#spaces > tbody");
|
||||||
|
|
||||||
|
apps.forEach((app) => {
|
||||||
|
spaces.appendChild(buildSpaceElement(app));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function addSpace() {
|
||||||
|
const spaces = document.querySelector("#spaces > tbody");
|
||||||
|
const newApp = {
|
||||||
|
title: "",
|
||||||
|
url: "",
|
||||||
|
icon: "",
|
||||||
|
};
|
||||||
|
spaces.appendChild(buildSpaceElement(newApp));
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeSpace(e) {
|
||||||
|
e.target.parentElement.parentElement.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveChanges(e) {
|
||||||
|
await e.preventDefault();
|
||||||
|
|
||||||
|
const apps = [];
|
||||||
|
|
||||||
|
const spaces = Array.from(document.getElementsByClassName("space"));
|
||||||
|
spaces.forEach((space) => {
|
||||||
|
const app = {
|
||||||
|
name: hashSpaceName(space.querySelector("input[name='url']").value),
|
||||||
|
title: space.querySelector("input[name='title']").value,
|
||||||
|
url: space.querySelector("input[name='url']").value,
|
||||||
|
icon: space.querySelector("input[name='icon']").value,
|
||||||
|
};
|
||||||
|
apps.push(app);
|
||||||
|
});
|
||||||
|
|
||||||
|
await removeSpaces();
|
||||||
|
await storeApps(apps);
|
||||||
|
await createSpaces();
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", setupPage);
|
||||||
|
document.getElementById("add").addEventListener("click", addSpace);
|
||||||
|
document.querySelector("form").addEventListener("submit", saveChanges);
|
7
readme.md
Normal file
7
readme.md
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
# SpaceApper
|
||||||
|
|
||||||
|
**A Thunderbird add-on**
|
||||||
|
|
||||||
|
Add webpages to the Thunderbird spaces toolbar.
|
||||||
|
|
||||||
|
*Attention: The author has no experience in creating a add-on for Thunderbird. Therefore, the add-on should not be considered stable.*
|
3
scripts/background.js
Normal file
3
scripts/background.js
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
import { createSpaces } from "./spaces.mjs";
|
||||||
|
|
||||||
|
createSpaces();
|
11
scripts/listeners.mjs
Normal file
11
scripts/listeners.mjs
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
export { changeUserAgent };
|
||||||
|
|
||||||
|
function changeUserAgent(details) {
|
||||||
|
for (const header of details.requestHeaders) {
|
||||||
|
if (header.name.toLowerCase() === "user-agent") {
|
||||||
|
header.value = "Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:109.0) Gecko/20100101 Firefox/115.0";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return { requestHeaders: details.requestHeaders };
|
||||||
|
}
|
49
scripts/spaces.mjs
Normal file
49
scripts/spaces.mjs
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
import { changeUserAgent } from "./listeners.mjs";
|
||||||
|
import { getApps, storeApps } from "./storage.mjs";
|
||||||
|
|
||||||
|
export { createSpaces, removeSpaces, hashSpaceName };
|
||||||
|
|
||||||
|
function hashSpaceName(url) {
|
||||||
|
return btoa(url).replaceAll("=", "");
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createSpaces() {
|
||||||
|
const apps = await getApps();
|
||||||
|
|
||||||
|
if (apps.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const urls = [];
|
||||||
|
|
||||||
|
apps.forEach(async (app) => {
|
||||||
|
urls.push(app.url);
|
||||||
|
await browser.spaces.create(app.name, app.url, {
|
||||||
|
title: app.title,
|
||||||
|
defaultIcons: app.icon,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const matchUrls = urls.map((url) => {
|
||||||
|
return url.replace(/^(https?:\/\/).*((?<=[\/\.])[^\/\.?]*\.[^\/\.?]*)[\/?]?.*$/g, "$1*.$2/*");
|
||||||
|
});
|
||||||
|
|
||||||
|
if (browser.webRequest.onBeforeSendHeaders.hasListener(changeUserAgent)) {
|
||||||
|
browser.webRequest.onBeforeSendHeaders.removeListener(changeUserAgent);
|
||||||
|
}
|
||||||
|
|
||||||
|
browser.webRequest.onBeforeSendHeaders.addListener(changeUserAgent, { urls: matchUrls }, [
|
||||||
|
"requestHeaders",
|
||||||
|
"blocking",
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function removeSpaces() {
|
||||||
|
const spaces = await browser.spaces.query({ isSelfOwned: true });
|
||||||
|
|
||||||
|
spaces.forEach(async (space) => {
|
||||||
|
await browser.spaces.remove(space.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
await storeApps([]);
|
||||||
|
}
|
17
scripts/storage.mjs
Normal file
17
scripts/storage.mjs
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
export { storeApps, getApps };
|
||||||
|
|
||||||
|
async function storeApps(apps) {
|
||||||
|
await browser.storage.sync.set({
|
||||||
|
apps: apps,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getApps() {
|
||||||
|
const res = await browser.storage.sync.get("apps");
|
||||||
|
|
||||||
|
if (res.apps === undefined) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.apps;
|
||||||
|
}
|
Loading…
Reference in a new issue