Skip to content

Commit

Permalink
📦 Upgrade to manifest v3
Browse files Browse the repository at this point in the history
  • Loading branch information
devmount committed Nov 6, 2024
1 parent 7a4a160 commit 5dbb08c
Show file tree
Hide file tree
Showing 5 changed files with 51 additions and 39 deletions.
10 changes: 5 additions & 5 deletions public/js/background.js
Original file line number Diff line number Diff line change
@@ -1,10 +1,11 @@
// this script holds all background activity not related to stats building

function main() {
const main = async () => {
// add icon to spaces toolbar
if (messenger.spacesToolbar) {
messenger.spacesToolbar.addButton(
if (messenger.spaces) {
await messenger.spaces.create(
'third_stats',
'../index.stats.html',
{
badgeBackgroundColor: '#e64db9',
badgeText: '',
Expand All @@ -22,10 +23,9 @@ function main() {
}
],
title: 'ThirdStats',
url: '../index.stats.html'
}
);
}
}
};

main();
29 changes: 16 additions & 13 deletions public/manifest.json
Original file line number Diff line number Diff line change
@@ -1,41 +1,44 @@
{
"name": "ThirdStats",
"version": "1.10.0",
"manifest_version": 2,
"manifest_version": 3,
"description": "__MSG_extensionDescription__",
"author": "Andreas Müller",
"homepage_url": "https://github.com/devmount/third-stats",
"default_locale": "en",
"applications": {
"browser_specific_settings": {
"gecko": {
"id": "thirdstats@devmount.de",
"strict_min_version": "91.0"
"strict_min_version": "128.0"
}
},
"browser_action": {
"action": {
"default_popup": "index.popup.html",
"default_title": "ThirdStats",
"default_icon": "./icon.svg"
"default_icon": "icon.svg"
},
"background": {
"scripts": [
"./js/background.js"
"js/background.js"
]
},
"options_ui": {
"page": "./index.options.html",
"open_in_tab": false,
"browser_style": true
"page": "index.options.html",
"open_in_tab": false
},
"permissions": [
"accountsRead",
"messagesRead",
"storage",
"downloads"
"downloads",
"messagesTagsList"
],
"icons": {
"64": "./icon.svg",
"32": "./icon.svg",
"16": "./icon.svg"
"64": "icon.svg",
"32": "icon.svg",
"16": "icon.svg"
},
"content_security_policy": {
"extension_pages": "default-src 'self'"
}
}
15 changes: 10 additions & 5 deletions src/Popup.vue
Original file line number Diff line number Diff line change
Expand Up @@ -33,20 +33,22 @@
<!-- list of all accounts -->
<section class="accounts">
<div
v-for="a in accounts"
:key="a.id"
class="background-hover-accent2 text-hover-highlight cursor-pointer shadow position-relative"
:class="{
'background-gray': options.dark,
'background-highlight-contrast': !options.dark,
}"
v-for="a in accounts"
:key="a.id"
@click.prevent="openTab('index.stats.html', a.id)"
>
<div class="position-relative z-5">
<h4>{{ a.name }}</h4>
<div class="text-small text-secondary">
{{ t("popup.nFolders", a.folderCount, [a.folderCount]) }}
<div v-if="a.hasOwnProperty('messageCount')">{{ t("popup.nMessages", a.messageCount, [a.messageCount]) }}</div>
<div>{{ t("popup.nFolders", [a.folderCount], a.folderCount) }}</div>
<div v-if="a.hasOwnProperty('messageCount')">
{{ t("popup.nMessages", [a.messageCount], a.messageCount) }}
</div>
</div>
</div>
<LineChart
Expand Down Expand Up @@ -113,7 +115,7 @@ const getAccounts = async () => {
// expand account object with additional data
await Promise.all(accountList.map(async a => {
// calculate folder count and append to account object
const folders = traverseAccount(a);
const folders = await traverseAccount(a);
a.folderCount = folders.length;
// get overall message count when cache is enabled
if (options.cache) {
Expand Down Expand Up @@ -148,10 +150,13 @@ const getAccounts = async () => {
onMounted(async () => {
// start loading indication
loading.value = true;
// get stored options
await getOptions();
// start account processing
await getAccounts();
// stop loading indication
loading.value = false;
});
Expand Down
10 changes: 5 additions & 5 deletions src/Stats.vue
Original file line number Diff line number Diff line change
Expand Up @@ -1122,7 +1122,7 @@ const getOptions = async () => {
// retrieve list of tags that can be set on messages
// their human-friendly name, color, and sort order
const getTags = async () => {
tags.value = await messenger.messages.listTags();
tags.value = await messenger.messages.tags.list();
};
// retrieve accounts and identities list
Expand Down Expand Up @@ -1303,7 +1303,7 @@ const analyzeMessage = (data, m, identityList) => {
// store results in <data> object
const processMessages = async (data, folder, identityList) => {
if (folder) {
for await (let m of queryMessages(folder, active.period.start, active.period.end)) {
for await (let m of queryMessages(folder.id, active.period.start, active.period.end)) {
analyzeMessage(data, m, identityList);
}
}
Expand All @@ -1315,7 +1315,7 @@ const processAccount = async (a) => {
// get identities from account, or from preferences if it's a local account
const identities = a.type != "none" ? a.identities.map(i => i.email.toLowerCase()) : options.addresses;
// get all folders and subfolders from given account or selected folder of active account (filter field)
const foldersList = active.folder ? [JSON.parse(JSON.stringify(active.folder))] : traverseAccount(a);
const foldersList = active.folder ? [JSON.parse(JSON.stringify(active.folder))] : await traverseAccount(a);
// build folder list for filter selection, if not already present
if (!folders.value.length) {
folders.value = foldersList;
Expand Down Expand Up @@ -1402,7 +1402,7 @@ const loadAccount = async (id, refresh, auto=false) => {
let accountsData = [];
// init progress indicator
progress.current = 1;
progress.max = activeAccounts.reduce((p,c) => p+traverseAccount(c).length, 0);
progress.max = activeAccounts.reduce(async (p,c) => p + (await traverseAccount(c).length), 0);
// when auto processing remember displayed account key and disable live counts
let displayedAccountKey = null;
let liveCountUpDisabled = false;
Expand Down Expand Up @@ -1539,7 +1539,7 @@ const loadAccount = async (id, refresh, auto=false) => {
// set tab title
document.title = "ThirdStats: " + account.name;
// (re)calculate list of folders
folders.value = traverseAccount(account);
folders.value = await traverseAccount(account);
// only check storage if no refresh was requested cache is enabled
const result = options.cache ? await messenger.storage.local.get("stats-" + id) : null;
if (!refresh && result && result["stats-" + id]) {
Expand Down
26 changes: 15 additions & 11 deletions src/utils.js
Original file line number Diff line number Diff line change
Expand Up @@ -107,14 +107,14 @@ const isSelfMessage = (message, identities) => {
};

// generator to query messages of given folder
const queryMessages = async function* (folder, fromDate, toDate) {
const queryMessages = async function* (folderId, fromDate, toDate) {
// handle date filter
const dateFilterActive = fromDate && toDate;
const from = new Date(fromDate);
const to = new Date(toDate);
try {
// paginate messages
let page = await messenger.messages.list(folder);
let page = await messenger.messages.list(folderId);
for (let message of page.messages) {
const messagesOutsideDateFilter = message.date < from || message.date > to;
if (!(dateFilterActive && messagesOutsideDateFilter)) {
Expand All @@ -137,18 +137,22 @@ const queryMessages = async function* (folder, fromDate, toDate) {
};

// flatten folder hierarchie of given account
const traverseAccount = (account) => {
let arrayOfFolders = [];
// recursive function to traverse all subfolders
const traverseAccount = async (account) => {
const foldersList = [];
// Recursive function to traverse all subfolders
function traverse(folders) {
if (!folders) return;
for (let f of folders) {
arrayOfFolders.push(f);
if (!folders?.length) return;
folders.forEach(f => {
if (!f.isRoot) foldersList.push(f);
traverse(f.subFolders);
}
});
}
traverse(account.folders);
return arrayOfFolders;

// Start with root
const rootFolder = await messenger.folders.get(account.rootFolder.id, true);
traverse([rootFolder]);

return foldersList;
};

// extract an email address from a given string
Expand Down

0 comments on commit 5dbb08c

Please sign in to comment.