chore: update onboarding disclaimer, unify telemetry handling, detect firefox vs chromium

This commit is contained in:
dragonwocky 2023-02-04 13:10:12 +11:00
parent 374efd3458
commit 809b59ebb1
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
12 changed files with 174 additions and 144 deletions

View File

@ -5,4 +5,4 @@ version=$(node -p "require('./package.json').version")
cd src cd src
mkdir -p ../dist mkdir -p ../dist
rm -f "../dist/notion-enhancer-$version.zip" rm -f "../dist/notion-enhancer-$version.zip"
zip -r9 "../dist/notion-enhancer-$version.zip" . -x electron/\* zip -r9 "../dist/notion-enhancer-$version.zip" .

View File

@ -6,7 +6,9 @@
"use strict"; "use strict";
const platform = "browser", const platform = navigator.userAgent.includes("Firefox")
? "firefox"
: "chromium",
version = chrome.runtime.getManifest().version, version = chrome.runtime.getManifest().version,
enhancerUrl = (target) => chrome.runtime.getURL(target); enhancerUrl = (target) => chrome.runtime.getURL(target);

View File

@ -25,18 +25,17 @@ const _isManifestValid = (modManifest) => {
}; };
let _mods; let _mods;
const getMods = async (category) => { const getMods = async (asyncFilter) => {
const { readJson } = globalThis.__enhancerApi; const { readJson } = globalThis.__enhancerApi;
// prettier-ignore // prettier-ignore
_mods ??= (await Promise.all((await readJson("registry.json")).map(async (_src) => { _mods ??= (await Promise.all((await readJson("registry.json")).map(async (_src) => {
const modManifest = { ...(await readJson(`${_src}/mod.json`)), _src }; const modManifest = { ...(await readJson(`${_src}/mod.json`)), _src };
return _isManifestValid(modManifest) ? modManifest : undefined; return _isManifestValid(modManifest) ? modManifest : undefined;
}))).filter((mod) => mod); }))).filter((mod) => mod);
return category // prettier-ignore
? _mods.filter(({ _src }) => { return (await Promise.all(_mods.map(async (mod) => {
return _src === category || _src.startsWith(`${category}/`); return !asyncFilter || (await asyncFilter(mod)) ? mod : undefined;
}) }))).filter((mod) => mod);
: _mods;
}, },
getProfile = async () => { getProfile = async () => {
const db = globalThis.__enhancerApi.initDatabase(); const db = globalThis.__enhancerApi.initDatabase();
@ -49,10 +48,10 @@ const isEnabled = async (id) => {
const { version, initDatabase } = globalThis.__enhancerApi, const { version, initDatabase } = globalThis.__enhancerApi,
mod = (await getMods()).find((mod) => mod.id === id); mod = (await getMods()).find((mod) => mod.id === id);
if (mod._src === "core") return true; if (mod._src === "core") return true;
// prettier-ignore
const agreedToTerms = await initDatabase().get("agreedToTerms"), const agreedToTerms = await initDatabase().get("agreedToTerms"),
enabledInProfile = await initDatabase([ enabledInProfile = await initDatabase([
await getProfile(), "enabledMods", await getProfile(),
"enabledMods",
]).get(id); ]).get(id);
return agreedToTerms === version && enabledInProfile; return agreedToTerms === version && enabledInProfile;
}, },
@ -63,10 +62,9 @@ const isEnabled = async (id) => {
}; };
const modDatabase = async (id) => { const modDatabase = async (id) => {
// prettier-ignore
const optionDefaults = (await getMods()) const optionDefaults = (await getMods())
.find((mod) => mod.id === id)?.options .find((mod) => mod.id === id)
.map((opt) => [opt.key, opt.value ?? opt.values?.[0]]) ?.options.map((opt) => [opt.key, opt.value ?? opt.values?.[0]])
.filter(([, value]) => typeof value !== "undefined"); .filter(([, value]) => typeof value !== "undefined");
return globalThis.__enhancerApi.initDatabase( return globalThis.__enhancerApi.initDatabase(
[await getProfile(), id], [await getProfile(), id],

View File

@ -5,21 +5,20 @@
*/ */
import { checkForUpdate } from "./update.mjs"; import { checkForUpdate } from "./update.mjs";
import { sendTelemetryPing } from "./telemetry.mjs";
import { Frame, Modal, Button } from "./components.mjs"; import { Frame, Modal, Button } from "./components.mjs";
// prettier-ignore
const asyncFilter = async (arr, predicate) => Promise.all(arr.map(predicate))
.then((results) => arr.filter((_v, index) => results[index]));
const doThemeOverride = async (db) => { const doThemeOverride = async (db) => {
const { getMods, isEnabled } = globalThis.__enhancerApi, const { getMods, isEnabled } = globalThis.__enhancerApi,
enabledFilter = (theme) => isEnabled(theme.id), loadThemeOverrides = await db.get("loadThemeOverrides");
overrideThemes = await db.get("loadThemeOverrides"), if (loadThemeOverrides === "Enabled") return true;
enabledThemes = await asyncFilter(await getMods("themes"), enabledFilter); if (loadThemeOverrides === "Disabled") return false;
return ( // prettier-ignore
overrideThemes === "Enabled" || return (await getMods(async (mod) => {
(overrideThemes === "Auto" && enabledThemes.length) // loadThemeOverrides === "Auto"
); if (!mod._src.startsWith("themes/")) return false;
return await isEnabled(mod.id);
})).length;
}, },
overrideThemes = async (db) => { overrideThemes = async (db) => {
const { html, enhancerUrl } = globalThis.__enhancerApi; const { html, enhancerUrl } = globalThis.__enhancerApi;
@ -39,86 +38,78 @@ const doThemeOverride = async (db) => {
}; };
const insertMenu = async (db) => { const insertMenu = async (db) => {
const notionSidebar = `.notion-sidebar-container .notion-sidebar > :nth-child(3) > div > :nth-child(2)`, const notionSidebar = `.notion-sidebar-container .notion-sidebar > :nth-child(3) > div > :nth-child(2)`,
{ html, addKeyListener, addMutationListener } = globalThis.__enhancerApi, { html, addKeyListener, addMutationListener } = globalThis.__enhancerApi,
{ platform, enhancerUrl, onMessage } = globalThis.__enhancerApi, { platform, enhancerUrl, onMessage } = globalThis.__enhancerApi,
menuButtonIconStyle = await db.get("menuButtonIconStyle"), menuButtonIconStyle = await db.get("menuButtonIconStyle"),
openMenuHotkey = await db.get("openMenuHotkey"), openMenuHotkey = await db.get("openMenuHotkey"),
renderPing = { renderPing = {
namespace: "notion-enhancer", namespace: "notion-enhancer",
hotkey: openMenuHotkey, hotkey: openMenuHotkey,
icon: menuButtonIconStyle, icon: menuButtonIconStyle,
}; };
let _contentWindow; let _contentWindow;
const sendThemePing = () => { const sendThemePing = () => {
const darkMode = document.body.classList.contains("dark"), const darkMode = document.body.classList.contains("dark"),
notionTheme = darkMode ? "dark" : "light"; notionTheme = darkMode ? "dark" : "light";
if (renderPing.theme === notionTheme) return; if (renderPing.theme === notionTheme) return;
renderPing.theme = notionTheme; renderPing.theme = notionTheme;
_contentWindow?.postMessage?.(renderPing, "*"); _contentWindow?.postMessage?.(renderPing, "*");
}, },
sendRenderPing = (contentWindow) => { sendRenderPing = (contentWindow) => {
_contentWindow ??= contentWindow; _contentWindow ??= contentWindow;
if (!$modal.hasAttribute("open")) return; if (!$modal.hasAttribute("open")) return;
delete renderPing.theme; delete renderPing.theme;
_contentWindow?.focus?.(); _contentWindow?.focus?.();
sendThemePing(); sendThemePing();
}; };
const $modal = html`<${Modal} onopen=${sendRenderPing}> const $modal = html`<${Modal} onopen=${sendRenderPing}>
<${Frame} <${Frame}
title="notion-enhancer menu" title="notion-enhancer menu"
src="${enhancerUrl("core/menu/index.html")}" src="${enhancerUrl("core/menu/index.html")}"
onload=${function () { onload=${function () {
// pass notion-enhancer api to electron menu process // pass notion-enhancer api to electron menu process
if (platform !== "browser") { if (["darwin", "win32", "linux"].includes(platform)) {
const apiKey = "__enhancerApi"; const apiKey = "__enhancerApi";
this.contentWindow[apiKey] = globalThis[apiKey]; this.contentWindow[apiKey] = globalThis[apiKey];
} }
sendRenderPing(this.contentWindow); sendRenderPing(this.contentWindow);
}} }}
/> />
<//>`, <//>`,
$button = html`<${Button} $button = html`<${Button}
onclick=${$modal.open} onclick=${$modal.open}
notifications=${(await checkForUpdate()) ? 1 : 0} notifications=${(await checkForUpdate()) ? 1 : 0}
themeOverridesLoaded=${await doThemeOverride(db)} themeOverridesLoaded=${await doThemeOverride(db)}
icon="notion-enhancer${menuButtonIconStyle === "Monochrome" icon="notion-enhancer${menuButtonIconStyle === "Monochrome"
? "?mask" ? "?mask"
: " text-[16px]"}" : " text-[16px]"}"
>notion-enhancer >notion-enhancer
<//>`; <//>`;
document.body.append($modal); document.body.append($modal);
addMutationListener(notionSidebar, () => { addMutationListener(notionSidebar, () => {
if (document.contains($button)) return; if (document.contains($button)) return;
document.querySelector(notionSidebar)?.append($button);
});
document.querySelector(notionSidebar)?.append($button); document.querySelector(notionSidebar)?.append($button);
addMutationListener("body", sendThemePing); });
window.addEventListener("focus", sendRenderPing); document.querySelector(notionSidebar)?.append($button);
addMutationListener("body", sendThemePing);
window.addEventListener("focus", sendRenderPing);
addKeyListener(openMenuHotkey, (event) => { addKeyListener(openMenuHotkey, (event) => {
event.preventDefault(); event.preventDefault();
$modal.open(); $modal.open();
}); });
window.addEventListener("message", (event) => { window.addEventListener("message", (event) => {
if (event.data?.namespace !== "notion-enhancer") return; if (event.data?.namespace !== "notion-enhancer") return;
if (event.data?.action === "close-menu") $modal.close(); if (event.data?.action === "close-menu") $modal.close();
if (event.data?.action === "open-menu") $modal.open(); if (event.data?.action === "open-menu") $modal.open();
}); });
onMessage("notion-enhancer", (message) => { onMessage("notion-enhancer", (message) => {
if (message === "open-menu") $modal.open(); if (message === "open-menu") $modal.open();
}); });
}, };
sendTelemetryPing = async () => {
const { version } = globalThis.__enhancerApi,
db = globalThis.__enhancerApi.initDatabase(),
agreedToTerms = await db.get("agreedToTerms"),
telemetryEnabled = await db.get("telemetryEnabled");
if (!telemetryEnabled || agreedToTerms !== version) return;
// telemetry
};
export default async (api, db) => { export default async (api, db) => {
await Promise.all([ await Promise.all([

View File

@ -23,16 +23,23 @@ function Onboarding() {
>Continue >Continue
<//>`, <//>`,
$agreeToTerms = html`<div class="mt-[32px]"> $agreeToTerms = html`<div class="mt-[32px]">
<${Heading} class="mb-[8px]">
Thanks for installing the notion-enhancer!
<//>
<${Description}> <${Description}>
Thanks for installing the notion-enhancer! It's been absolutely In order for the notion-enhancer to function, it may access, collect,
incredible to see how the notion-enhancer has grown from small process and/or store data on your device (including workspace content,
beginnings to something used today by over 11,000 people around the device metadata, and notion-enhancer configuration) according to its
world, now including you. Before you begin, please read the privacy privacy policy. Unless otherwise stated for telemetry purposes, the
policy to learn how the notion-enhancer uses your data and the terms & notion-enhancer will never transmit any of your data from your device.
conditions to understand what the notion-enhancer does and does not Telemetry can be disabled at any time through the menu.
offer. Ticking the box below and pressing <mark>Continue</mark> will <br />
unlock the notion-enhancer's full functionality, accessible through the <br />
sidebar. The notion-enhancer is free and open-source software distributed under
the <a href="${tsAndCs}#license">MIT License</a> without warranty of any
kind. In no event shall the authors be liable for any consequences of
the software's use. Before continuing, you must read and agree to the
notion-enhancer's privacy policy and terms & conditions.
<//> <//>
<div class="flex items-center my-[14px] gap-[8px]"> <div class="flex items-center my-[14px] gap-[8px]">
<${Checkbox} <${Checkbox}
@ -77,7 +84,7 @@ function Onboarding() {
</div>`, </div>`,
$featuredSponsors = html` $featuredSponsors = html`
<div class="mt-[32px]"> <div class="mt-[32px]">
<${Heading} class="mt-[32px] mb-[8px]">Featured Sponsors<//> <${Heading} class="mb-[8px]">Featured Sponsors<//>
<${Description}> <${Description}>
A few awesome companies out there have teamed up with me to provide A few awesome companies out there have teamed up with me to provide
you with the notion-enhancer, free forever. Check them out! you with the notion-enhancer, free forever. Check them out!

View File

@ -53,7 +53,7 @@ function Sidebar({ items, categories }) {
const { html, version } = globalThis.__enhancerApi, const { html, version } = globalThis.__enhancerApi,
{ initDatabase, isEnabled } = globalThis.__enhancerApi, { initDatabase, isEnabled } = globalThis.__enhancerApi,
$agreeToUnlock = html`<span $agreeToUnlock = html`<span
class="pt-[2px] pb-[5px] px-[15px] class="pt-[2px] pb-[5px] px-[15px] text-[12px]
inline-block text-[color:var(--theme--fg-red)]" inline-block text-[color:var(--theme--fg-red)]"
>To unlock the notion-enhancer's full functionality, agree to the privacy >To unlock the notion-enhancer's full functionality, agree to the privacy
policy and terms & conditions on the welcome page. policy and terms & conditions on the welcome page.

View File

@ -4,26 +4,14 @@
* (https://notion-enhancer.github.io/) under the MIT license * (https://notion-enhancer.github.io/) under the MIT license
*/ */
import { collectTelemetryData } from "../../telemetry.mjs";
import { useState, setState } from "../state.mjs"; import { useState, setState } from "../state.mjs";
import { Option } from "./Options.mjs"; import { Option } from "./Options.mjs";
const privacyPolicy = "https://notion-enhancer.github.io/about/privacy-policy/"; const privacyPolicy = "https://notion-enhancer.github.io/about/privacy-policy/";
function Telemetry() { function Telemetry() {
const { html, platform, version } = globalThis.__enhancerApi, const { html, initDatabase } = globalThis.__enhancerApi,
{ getMods, isEnabled, initDatabase } = globalThis.__enhancerApi, _get = async () => {
timezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
const $enabledMods = html`<code></code>`;
useState(["rerender"], async () => {
let enabledMods = [];
for (const mod of await getMods()) {
if (mod._src === "core") continue;
if (await isEnabled(mod.id)) enabledMods.push(mod.id);
}
$enabledMods.innerText = JSON.stringify(enabledMods);
});
const _get = async () => {
// defaults to true, must be explicitly set to false to disable // defaults to true, must be explicitly set to false to disable
return initDatabase().get("telemetryEnabled") ?? true; return initDatabase().get("telemetryEnabled") ?? true;
}, },
@ -32,6 +20,19 @@ function Telemetry() {
setState({ rerender: true }); setState({ rerender: true });
}; };
const $ = {
platform: html`<code></code>`,
version: html`<code></code>`,
timezone: html`<code></code>`,
enabledMods: html`<code></code>`,
};
useState(["rerender"], async () => {
const telemetryData = await collectTelemetryData();
for (const key in telemetryData) {
$[key].innerText = JSON.stringify(telemetryData[key]);
}
});
// todo: actually collect telemetry // todo: actually collect telemetry
return html`<${Option} return html`<${Option}
type="toggle" type="toggle"
@ -39,11 +40,10 @@ function Telemetry() {
description=${html`If telemetry is enabled, usage data will be collected description=${html`If telemetry is enabled, usage data will be collected
once a week from your device in order to better understand how and where once a week from your device in order to better understand how and where
the notion-enhancer is used. This data is anonymous and includes only your the notion-enhancer is used. This data is anonymous and includes only your
platform (<code>"${platform}"</code>), timezone platform (${$.platform}), notion-enhancer version (${$.version}), timezone
(<code>"${timezone}"</code>), notion-enhancer version (${$.timezone}), and enabled mods (${$.enabledMods}). You can opt in or
(<code>"${version}"</code>), and enabled mods (${$enabledMods}). You can out of telemetry at any time. This setting syncs across configuration
opt in or out of telemetry at any time. This setting syncs across profiles. For more information, read the notion-enhancer's
configuration profiles. For more information, read the notion-enhancer's
<a href=${privacyPolicy} class="ml-[3px]">privacy policy</a>.`} <a href=${privacyPolicy} class="ml-[3px]">privacy policy</a>.`}
...${{ _get, _set }} ...${{ _get, _set }}
/>`; />`;

27
src/core/telemetry.mjs Normal file
View File

@ -0,0 +1,27 @@
/**
* notion-enhancer
* (c) 2023 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
const collectTelemetryData = async () => {
const { platform, version } = globalThis.__enhancerApi,
{ getMods, isEnabled } = globalThis.__enhancerApi,
timezone = Intl.DateTimeFormat().resolvedOptions().timeZone,
// prettier-ignore
enabledMods = (await getMods(async (mod) => {
if (mod._src === "core") return false;
return await isEnabled(mod.id);
})).map(mod => mod.id);
return { platform, version, timezone, enabledMods };
},
sendTelemetryPing = async () => {
const db = globalThis.__enhancerApi.initDatabase(),
agreedToTerms = await db.get("agreedToTerms"),
telemetryEnabled = await db.get("telemetryEnabled");
if (!telemetryEnabled || agreedToTerms !== version) return;
// telemetry
const telemetryData = await collectTelemetryData();
};
export { collectTelemetryData, sendTelemetryPing };

View File

@ -8,8 +8,11 @@ let _release;
const repo = "notion-enhancer/notion-enhancer", const repo = "notion-enhancer/notion-enhancer",
endpoint = `https://api.github.com/repos/${repo}/releases/latest`, endpoint = `https://api.github.com/repos/${repo}/releases/latest`,
getRelease = async () => { getRelease = async () => {
const { readJson } = globalThis.__enhancerApi; const { version, readJson } = globalThis.__enhancerApi;
_release ??= (await readJson(endpoint))?.tag_name.replace(/^v/, ""); try {
_release ??= (await readJson(endpoint))?.tag_name.replace(/^v/, "");
} catch {}
_release ??= version;
return _release; return _release;
}; };

View File

@ -45,7 +45,10 @@ if (isElectron()) {
} }
} }
}; };
} else {
// clientStyles // clientStyles
// clientScripts // clientScripts
} else import("./api/browser.js").then(() => import("./load.mjs")); import(chrome.runtime.getURL("/api/browser.js")).then(() => {
import(chrome.runtime.getURL("/load.mjs"));
});
}

View File

@ -9,21 +9,21 @@
export default (async () => { export default (async () => {
// prettier-ignore // prettier-ignore
const { enhancerUrl } = globalThis.__enhancerApi, const { enhancerUrl } = globalThis.__enhancerApi,
isMenu = location.href.startsWith(enhancerUrl("/core/menu/index.html")), isMenu = location.href.startsWith(enhancerUrl("core/menu/index.html")),
pageLoaded = /(^\/$)|((-|\/)[0-9a-f]{32}((\?.+)|$))/.test(location.pathname), pageLoaded = /(^\/$)|((-|\/)[0-9a-f]{32}((\?.+)|$))/.test(location.pathname),
signedIn = localStorage["LRU:KeyValueStore2:current-user-id"]; signedIn = localStorage["LRU:KeyValueStore2:current-user-id"];
if (!isMenu && (!signedIn || !pageLoaded)) return; if (!isMenu && (!signedIn || !pageLoaded)) return;
if (!isMenu) console.log("notion-enhancer: loading..."); if (!isMenu) console.log("notion-enhancer: loading...");
await Promise.all([ await Promise.all([
import("./assets/icons.svg.js"), import(enhancerUrl("assets/icons.svg.js")),
import("./vendor/twind.min.js"), import(enhancerUrl("vendor/twind.min.js")),
import("./vendor/lucide.min.js"), import(enhancerUrl("vendor/lucide.min.js")),
import("./vendor/htm.min.js"), import(enhancerUrl("vendor/htm.min.js")),
import("./api/events.js"), import(enhancerUrl("api/events.js")),
import("./api/mods.js"), import(enhancerUrl("api/mods.js")),
]); ]);
await import("./api/interface.js"); await import(enhancerUrl("api/interface.js"));
const { getMods, isEnabled, modDatabase } = globalThis.__enhancerApi; const { getMods, isEnabled, modDatabase } = globalThis.__enhancerApi;
for (const mod of await getMods()) { for (const mod of await getMods()) {

View File

@ -7,7 +7,6 @@
"homepage_url": "https://notion-enhancer.github.io", "homepage_url": "https://notion-enhancer.github.io",
"content_scripts": [{ "matches": ["*://*.notion.so/*"], "js": ["/init.js"] }], "content_scripts": [{ "matches": ["*://*.notion.so/*"], "js": ["/init.js"] }],
"background": { "service_worker": "/worker.js" }, "background": { "service_worker": "/worker.js" },
"options_page": "/core/menu/index.html",
"action": {}, "action": {},
"icons": { "icons": {
"16": "/assets/colour-x16.png", "16": "/assets/colour-x16.png",