init menu with twind + notifications

This commit is contained in:
dragonwocky 2021-09-28 00:11:47 +10:00
parent b900c00641
commit a2b93cb513
15 changed files with 786 additions and 635 deletions

View File

@ -38,3 +38,13 @@ export const focusNotion = () => chrome.runtime.sendMessage({ action: 'focusNoti
/** reload all notion and enhancer menu tabs to apply changes */
export const reloadTabs = () => chrome.runtime.sendMessage({ action: 'reloadTabs' });
/** a notification displayed when the menu is opened for the first time */
export const welcomeNotification = {
id: '84e2d49b-c3dc-44b4-a154-cf589676bfa0',
color: 'blue',
icon: 'message-circle',
message: 'Welcome! Come chat with us on Discord.',
link: 'https://discord.gg/sFWPXtA',
version,
};

View File

@ -50,9 +50,7 @@ async function validate(mod) {
const test = await is(
type === 'file' && value ? `repo/${mod._dir}/${value}` : value,
type,
{
extension,
}
{ extension }
);
if (!test) {
if (optional && (await is(value, 'undefined'))) return true;
@ -288,3 +286,21 @@ export const optionDefault = async (id, key) => {
return undefined;
}
};
/**
* access the storage partition of a mod in the current profile
* @param {string} id - the uuid of the mod
* @returns {object} an object with the wrapped get/set functions
*/
export const db = async (id) => {
return storage.db(
['profiles', await storage.get(['currentprofile'], 'default'), id],
async (path, fallback = undefined) => {
if (path.length === 4) {
// profiles -> profile -> mod -> option
fallback = (await optionDefault(id, path[3])) ?? fallback;
}
return storage.get(path, fallback);
}
);
};

View File

@ -54,7 +54,7 @@ export const set = (path, value) => {
}
const pathClone = [...path],
namespace = path.shift();
chrome.storage.sync.get([], async (values) => {
chrome.storage.sync.get(async (values) => {
const update = values[namespace] ?? {};
let pointer = update,
old;

View File

@ -148,9 +148,7 @@ export const loadStylesheet = (path) => {
export const icon = (name, attrs = {}) => {
if (!_$featherStylesheet) {
_$featherStylesheet = html`<style data-enhancer-api-style="feather">
.enhancer-- {
width: 1.25em;
height: 1.25em;
.enhancer--feather {
stroke: currentColor;
stroke-width: 2;
stroke-linecap: round;

View File

@ -0,0 +1,58 @@
/**
* style-vendorizer v2.0.0
* @license MIT
* @source https://unpkg.com/style-vendorizer@^2.0.0?module
*/
var i = new Map([
['align-self', '-ms-grid-row-align'],
['color-adjust', '-webkit-print-color-adjust'],
['column-gap', 'grid-column-gap'],
['gap', 'grid-gap'],
['grid-template-columns', '-ms-grid-columns'],
['grid-template-rows', '-ms-grid-rows'],
['justify-self', '-ms-grid-column-align'],
['margin-inline-end', '-webkit-margin-end'],
['margin-inline-start', '-webkit-margin-start'],
['overflow-wrap', 'word-wrap'],
['padding-inline-end', '-webkit-padding-end'],
['padding-inline-start', '-webkit-padding-start'],
['row-gap', 'grid-row-gap'],
['scroll-margin-bottom', 'scroll-snap-margin-bottom'],
['scroll-margin-left', 'scroll-snap-margin-left'],
['scroll-margin-right', 'scroll-snap-margin-right'],
['scroll-margin-top', 'scroll-snap-margin-top'],
['scroll-margin', 'scroll-snap-margin'],
['text-combine-upright', '-ms-text-combine-horizontal'],
]);
function r(r) {
return i.get(r);
}
function n(i) {
var r =
/^(?:(text-(?:decoration$|e|or|si)|back(?:ground-cl|d|f)|box-d|(?:mask(?:$|-[ispro]|-cl)))|(tab-|column(?!-s)|text-align-l)|(ap)|(u|hy))/i.exec(
i
);
return r ? (r[1] ? 1 : r[2] ? 2 : r[3] ? 3 : 5) : 0;
}
function t(i, r) {
var n = /^(?:(pos)|(background-i)|((?:max-|min-)?(?:block-s|inl|he|widt))|(dis))/i.exec(i);
return n
? n[1]
? /^sti/i.test(r)
? 1
: 0
: n[2]
? /^image-/i.test(r)
? 1
: 0
: n[3]
? '-' === r[3]
? 2
: 0
: /^(inline-)?grid$/i.test(r)
? 4
: 0
: 0;
}
export { r as cssPropertyAlias, n as cssPropertyPrefixFlags, t as cssValuePrefixFlags };

View File

@ -12,25 +12,14 @@ if (
location.pathname.split(/[/-]/g).reverse()[0].length === 32
) {
import(chrome.runtime.getURL('api/_.mjs')).then(async (api) => {
const { registry, storage, web } = api,
profile = await storage.get(['currentprofile'], 'default');
const { registry, web } = api;
for (const mod of await registry.list((mod) => registry.enabled(mod.id))) {
const db = storage.db(
['profiles', profile, mod.id],
async (path, fallback = undefined) => {
if (path.length === 4) {
// profiles -> profile -> mod -> option
fallback = (await registry.optionDefault(mod.id, path[3])) ?? fallback;
}
return storage.get(path, fallback);
}
);
for (const sheet of mod.css?.client || []) {
web.loadStylesheet(`repo/${mod._dir}/${sheet}`);
}
for (let script of mod.js?.client || []) {
script = await import(chrome.runtime.getURL(`repo/${mod._dir}/${script}`));
script.default(api, db);
script.default(api, await registry.db(mod.id));
}
}
const errors = await registry.errors();

View File

@ -17,25 +17,32 @@ export default async function (api, db) {
<div><div>notion-enhancer</div></div>
</div>
</div>`;
$sidebarLink.addEventListener('click', env.openEnhancerMenu);
web.addHotkeyListener(await db.get(['hotkey']), env.openEnhancerMenu);
const setTheme = () =>
const updateTheme = () =>
db.set(['theme'], document.querySelector('.notion-dark-theme') ? 'dark' : 'light');
$sidebarLink.addEventListener('click', () => {
setTheme().then(env.openEnhancerMenu);
web.addDocumentObserver((mutation) => {
if (mutation.target === document.body) updateTheme();
});
window.addEventListener('focus', setTheme);
window.addEventListener('blur', setTheme);
setTheme();
updateTheme();
const errors = await registry.errors(),
notifications = {
cache: await db.get(['notifications'], []),
provider: await fs.getJSON('https://notion-enhancer.github.io/notifications.json'),
count: errors.length,
};
const notifications = {
cache: await db.get(['notifications'], []),
provider: [
env.welcomeNotification,
...(await fs.getJSON('https://notion-enhancer.github.io/notifications.json')),
],
count: (await registry.errors()).length,
};
for (const notification of notifications.provider) {
if (!notifications.cache.includes(notification.id)) notifications.count++;
if (
!notifications.cache.includes(notification.id) &&
notification.version === env.version &&
(!notification.environments || notification.environments.includes(env.name))
) {
notifications.count++;
}
}
if (notifications.count) {
$sidebarLink.dataset.hasNotifications = true;

View File

@ -0,0 +1,513 @@
/*
* notion-enhancer core: menu
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
* {
box-sizing: border-box;
word-break: break-word;
text-size-adjust: 100%;
-webkit-text-size-adjust: 100%;
font-size: inherit;
font-family: inherit;
fill: currentColor;
}
html {
transition: opacity 50ms ease-out;
font-size: var(--theme--font_body-size);
}
html[class] {
opacity: 1 !important;
}
body {
padding: 2rem 2.5rem;
position: relative;
margin: 0;
background: var(--theme--sidebar);
color: var(--theme--text);
font-family: var(--theme--font_sans);
min-height: 100vh;
}
body a {
color: inherit;
}
header {
display: flex;
flex-wrap: wrap;
margin-bottom: 1.25rem;
}
header > * {
margin: 0 1.25rem 0.1em 0;
font-size: var(--theme--font_heading1-size);
}
header h1 a {
text-decoration: none;
}
header h1 img {
width: 0.95em;
height: 0.95em;
margin-right: 0.5em;
margin-bottom: -0.1em;
}
header h1 svg {
width: 1em;
height: 1em;
padding-top: 0.3em;
margin-right: 0.2em;
}
header h1 svg[data-icon='info'] {
margin-right: 0em;
}
header h1 svg[data-icon='code'] {
margin-right: 0.3em;
}
img[data-view-target='notion'] {
cursor: pointer;
}
main {
display: grid;
grid-gap: 1.25em;
grid-template-columns: 1fr;
opacity: 1;
transition: opacity 200ms ease-out;
}
@media (min-width: 550px) {
main {
grid-template-columns: 1fr 1fr;
}
main > .action--buttons {
grid-column: span 2;
}
[data-view='mod'] main .library--card,
.documentation--body {
max-height: calc(100vh - 10rem);
overflow-y: auto;
}
[data-view='mod'] {
overflow: hidden;
height: 100vh;
}
}
@media (min-width: 850px) {
main {
grid-template-columns: 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 3;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 2;
}
}
@media (min-width: 1350px) {
main {
grid-template-columns: 1fr 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 4;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 3;
}
}
@media (min-width: 2050px) {
main {
grid-template-columns: 1fr 1fr 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 5;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 4;
}
}
main article {
border-radius: 5px;
box-shadow: rgb(0 0 0 / 10%) 0px 20px 25px -5px, rgb(0 0 0 / 4%) 0px 10px 10px -5px;
border: 1px solid var(--theme--divider);
background: var(--theme--page);
}
main article img {
max-width: 100%;
}
.action--buttons,
.library--expand {
margin: 0;
display: flex;
flex-wrap: wrap;
gap: 0.53em;
}
.library--expand a {
margin-left: auto;
}
.action--buttons a,
.library--expand a {
border-radius: 3px;
padding: 0.35rem 0.45rem;
text-decoration: none;
display: flex;
}
.action--buttons .action--alert {
cursor: pointer;
border-radius: 3px;
padding: 0.35rem 0.45rem;
background: var(--theme--block_grey);
color: var(--theme--block_grey-text);
border: none;
pointer-events: none;
opacity: 0;
transition: opacity 200ms ease-in-out;
}
.action--buttons .action--alert[data-triggered] {
pointer-events: all;
opacity: 1;
}
.action--buttons .action--alert[data-triggered]:hover {
background: none;
color: var(--theme--block_grey-text);
box-shadow: var(--theme--block_grey) 0px 0px 0px 1px inset;
}
.action--buttons span,
.library--expand span {
color: var(--theme--text_property);
}
.action--buttons a:hover,
.action--buttons a.action--active,
.library--expand a:hover {
background: var(--theme--button-hover);
}
.action--buttons svg,
.library--expand svg {
width: 1em;
height: 1em;
padding-top: 2px;
margin-right: 0.3rem;
}
.action--buttons svg *,
.library--expand svg * {
fill: var(--theme--text_property);
}
.library--preview {
border-bottom: 1px solid var(--theme--divider);
}
.library--version {
font-weight: normal;
font-size: var(--theme--font_ui-size);
border-radius: 5px;
padding: 0.125rem 0.25rem;
background: var(--theme--tag_default);
color: var(--theme--tag_default-text);
}
.library--description {
font-size: 1rem;
margin: 0.25rem 0;
}
.library--tags,
.library--authors {
padding: 0;
list-style-type: none;
display: flex;
flex-wrap: wrap;
margin: 0.5em 0;
}
.library--tags li {
font-size: var(--theme--font_ui-size);
margin: 0 0.5rem 0 0;
opacity: 0.7;
}
.library--authors li {
font-size: var(--theme--font_ui-size);
margin: 0 0.5rem 0 0;
}
.library--authors a {
font-size: var(--theme--font_ui-size);
text-decoration: none;
border-radius: 5px;
padding: 0.125rem 0.25rem 0.25rem 0;
}
.library--authors img {
height: 1em;
width: 1em;
margin-bottom: 0.15625em;
display: inline-block;
vertical-align: middle;
border-radius: 50%;
object-fit: cover;
}
.library--options {
border-top: 1px solid var(--theme--divider);
}
.library--toggle_label,
.library--select_label,
.library--text_label,
.library--number_label,
.library--color_label,
.library--file_label {
margin: 0.6rem 0;
display: block;
appearance: none;
font-size: var(--theme--font_ui-size);
}
.library--toggle_label *,
.library--select_label *,
.library--text_label *,
.library--number_label *,
.library--color_label *,
.library--file_label * {
appearance: none;
font-family: var(--theme--font_sans);
font-size: var(--theme--font_ui-size);
color: var(--theme--text);
}
label p:not([class]),
label p > span:not([class]),
label > span:not([class]) {
font-size: 1rem;
}
label [data-icon='fa/solid/question-circle'] {
height: var(--theme--font_ui_small-size);
width: var(--theme--font_ui_small-size);
margin-left: 0.25em;
margin-bottom: -1px;
}
.library--toggle_label > p,
.library--toggle_label > p,
.library--select_label > p,
.library--text_label > p,
.library--number_label > p,
.library--color_label > p,
.library--file_label > p {
margin: 0.6rem 0;
}
.library--toggle_label > :not(input) {
align-items: center;
display: flex;
--toggle--bg: var(--theme--toggle_off);
--toggle--translation: 0%;
}
.library--toggle_label > :not(input) .library--toggle {
position: relative;
margin: auto 0 auto auto;
min-width: 2.25rem;
width: 2.25rem;
height: 1.25rem;
display: block;
border-radius: 40px;
background: var(--toggle--bg);
cursor: pointer;
}
.library--toggle_label > :not(input) .library--toggle::after {
content: '';
transition: transform 200ms ease-out, background 200ms ease-out;
height: 0.8rem;
width: 0.8rem;
left: 0.325rem;
top: 0.2rem;
position: absolute;
border-radius: 100%;
background: var(--theme--toggle_dot);
transform: translateX(var(--toggle--translation));
}
.library--toggle_label input[type='checkbox']:checked + :not(input) {
--toggle--bg: var(--theme--toggle_on);
--toggle--translation: 100%;
}
.library--toggle_label:focus-within .library--toggle,
.library--file_label:focus-within .library--file,
.library--color_label:focus-within .library--color,
.library--select_label .library--select:focus-within {
outline: -webkit-focus-ring-color auto 1px;
}
.library--toggle_label input,
.library--file_label input {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border-width: 0;
}
.library--text_label textarea {
resize: none;
min-height: calc(1em + 16px);
height: var(--txt--scroll-height);
}
.library--text_label textarea,
.library--number_label input,
.library--select_label .library--select,
.library--color_label .library--color,
.library--file_label .library--file {
width: 100%;
padding: 6px 8px;
background: var(--theme--input);
border-radius: 3px;
border: none;
box-shadow: var(--theme--input-border) 0px 0px 0px 1px inset;
}
.library--select_label .library--select select,
.library--color_label .library--color input {
outline: none;
cursor: pointer;
}
.library--color_label .library--color input {
border-radius: 0;
}
.library--select_label .library--select select option {
background: var(--theme--tag_select);
}
.library--select_label .library--select,
.library--color_label .library--color,
.library--file_label .library--file {
padding: 0;
display: flex;
cursor: pointer;
}
.library--select_label .library--select > :first-child,
.library--color_label .library--color > :first-child,
.library--file_label .library--file > :first-child {
display: flex;
padding: 6px 8px;
background: var(--theme--input-border);
}
.library--select_label .library--select > :first-child svg,
.library--color_label .library--color > :first-child svg,
.library--file_label .library--file > :first-child svg {
width: 0.9em;
margin: auto 0;
}
.library--select_label .library--select > :first-child svg *,
.library--color_label .library--color > :first-child svg *,
.library--file_label .library--file > :first-child svg * {
color: var(--theme--input_icon);
}
.library--select_label .library--select > :last-child,
.library--color_label .library--color > :last-child,
.library--file_label .library--file > :last-child {
margin: auto 0;
padding: 6px 8px;
width: 100%;
height: 100%;
border: none;
background: none;
}
.library--file_label .library--file_title {
display: flex;
}
.library--file_title .library--file_remove {
margin-top: auto;
padding-top: 0.2em;
margin-left: auto;
}
.library--file_title .library--file_remove svg {
cursor: pointer;
width: 0.8em;
}
.library--title,
.library--title h2 {
margin: 0;
font-size: var(--theme--font_ui-size);
}
.library--title h2 > span {
font-size: var(--theme--font_heading2-size);
}
.library--card > div {
padding: 1rem;
}
[data-view='mod'] main .library--card {
overflow-x: auto;
}
.documentation--body {
padding: 1rem 2rem;
font-size: 1rem;
overflow-x: auto;
}
.notifications {
position: absolute;
bottom: 1.5rem;
right: 1.5rem;
}
.notification {
background: var(--theme--block_grey);
color: var(--theme--block_grey-text);
max-width: calc(100vw - 3rem);
width: 25rem;
font-size: var(--theme--font_ui-size);
border-top: 4px currentColor solid;
border-bottom-left-radius: 3px;
border-bottom-right-radius: 3px;
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06) !important;
padding: 1.25em 1.5em;
display: flex;
margin-top: 1rem;
position: relative;
transition: opacity 200ms ease-in-out, transform 115ms ease-out 200ms,
margin-top 115ms ease-out 200ms;
transform: scaleY(1);
transform-origin: 100% 100%;
opacity: 0;
}
.notification :not(.notification--dismiss) > svg {
height: 1.5rem;
width: 1.5rem;
margin-top: 0.25rem;
margin-right: 1rem;
}
.notification h3 {
margin: 0 0 0.25rem 0;
font-weight: bold;
}
.notification p {
margin: 0.25rem 0 0.5rem 0;
}
.notification .notification--dismiss {
position: absolute;
top: 0.5rem;
right: 0.75rem;
background: none;
border: none;
padding: 0.15rem 0 0.15rem 0.5rem;
width: var(--theme--font_body-size);
color: currentColor;
cursor: pointer;
transition: opacity 200ms ease-in-out;
opacity: 0;
}
.notification .notification--dismiss svg {
width: 100%;
}
.notification:hover .notification--dismiss,
.notification:focus-within .notification--dismiss {
opacity: 1;
}
::-webkit-scrollbar {
background: transparent;
width: 10px;
height: 10px;
}
::-webkit-scrollbar-track {
background: var(--theme--scrollbar_track);
}
::-webkit-scrollbar-corner {
background: var(--theme--scrollbar_track);
}
::-webkit-scrollbar-thumb {
background: var(--theme--scrollbar_thumb);
}
::-webkit-scrollbar-thumb:hover {
background: var(--theme--scrollbar_thumb-hover);
}

View File

@ -4,510 +4,18 @@
* (https://notion-enhancer.github.io/) under the MIT license
*/
* {
box-sizing: border-box;
word-break: break-word;
text-size-adjust: 100%;
-webkit-text-size-adjust: 100%;
font-size: inherit;
font-family: inherit;
fill: currentColor;
}
html {
transition: opacity 50ms ease-out;
font-size: var(--theme--font_body-size);
}
html[class] {
opacity: 1 !important;
}
body {
padding: 2rem 2.5rem;
position: relative;
margin: 0;
background: var(--theme--sidebar);
color: var(--theme--text);
font-family: var(--theme--font_sans);
min-height: 100vh;
}
body a {
color: inherit;
}
header {
display: flex;
flex-wrap: wrap;
margin-bottom: 1.25rem;
}
header > * {
margin: 0 1.25rem 0.1em 0;
font-size: var(--theme--font_heading1-size);
}
header h1 a {
text-decoration: none;
}
header h1 img {
width: 0.95em;
height: 0.95em;
margin-right: 0.5em;
margin-bottom: -0.1em;
}
header h1 svg {
width: 1em;
height: 1em;
padding-top: 0.3em;
margin-right: 0.2em;
}
header h1 svg[data-icon='info'] {
margin-right: 0em;
}
header h1 svg[data-icon='code'] {
margin-right: 0.3em;
}
img[data-view-target='notion'] {
cursor: pointer;
}
main {
display: grid;
grid-gap: 1.25em;
grid-template-columns: 1fr;
opacity: 1;
transition: opacity 200ms ease-out;
}
@media (min-width: 550px) {
main {
grid-template-columns: 1fr 1fr;
}
main > .action--buttons {
grid-column: span 2;
}
[data-view='mod'] main .library--card,
.documentation--body {
max-height: calc(100vh - 10rem);
overflow-y: auto;
}
[data-view='mod'] {
overflow: hidden;
height: 100vh;
}
}
@media (min-width: 850px) {
main {
grid-template-columns: 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 3;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 2;
}
}
@media (min-width: 1350px) {
main {
grid-template-columns: 1fr 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 4;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 3;
}
}
@media (min-width: 2050px) {
main {
grid-template-columns: 1fr 1fr 1fr 1fr 1fr;
}
main > .action--buttons {
grid-column: span 5;
}
[data-view='mod'] main > .documentation--body {
grid-column: span 4;
}
}
main article {
border-radius: 5px;
box-shadow: rgb(0 0 0 / 10%) 0px 20px 25px -5px, rgb(0 0 0 / 4%) 0px 10px 10px -5px;
border: 1px solid var(--theme--divider);
background: var(--theme--page);
}
main article img {
max-width: 100%;
}
.action--buttons,
.library--expand {
margin: 0;
display: flex;
flex-wrap: wrap;
gap: 0.53em;
}
.library--expand a {
margin-left: auto;
}
.action--buttons a,
.library--expand a {
border-radius: 3px;
padding: 0.35rem 0.45rem;
text-decoration: none;
display: flex;
}
.action--buttons .action--alert {
cursor: pointer;
border-radius: 3px;
padding: 0.35rem 0.45rem;
background: var(--theme--block_grey);
color: var(--theme--block_grey-text);
border: none;
pointer-events: none;
opacity: 0;
transition: opacity 200ms ease-in-out;
}
.action--buttons .action--alert[data-triggered] {
pointer-events: all;
opacity: 1;
}
.action--buttons .action--alert[data-triggered]:hover {
background: none;
color: var(--theme--block_grey-text);
box-shadow: var(--theme--block_grey) 0px 0px 0px 1px inset;
}
.action--buttons span,
.library--expand span {
color: var(--theme--text_property);
}
.action--buttons a:hover,
.action--buttons a.action--active,
.library--expand a:hover {
background: var(--theme--button-hover);
}
.action--buttons svg,
.library--expand svg {
width: 1em;
height: 1em;
padding-top: 2px;
margin-right: 0.3rem;
}
.action--buttons svg *,
.library--expand svg * {
fill: var(--theme--text_property);
}
.library--preview {
border-bottom: 1px solid var(--theme--divider);
}
.library--version {
font-weight: normal;
font-size: var(--theme--font_ui-size);
border-radius: 5px;
padding: 0.125rem 0.25rem;
background: var(--theme--tag_default);
color: var(--theme--tag_default-text);
}
.library--description {
font-size: 1rem;
margin: 0.25rem 0;
}
.library--tags,
.library--authors {
padding: 0;
list-style-type: none;
display: flex;
flex-wrap: wrap;
margin: 0.5em 0;
}
.library--tags li {
font-size: var(--theme--font_ui-size);
margin: 0 0.5rem 0 0;
opacity: 0.7;
}
.library--authors li {
font-size: var(--theme--font_ui-size);
margin: 0 0.5rem 0 0;
}
.library--authors a {
font-size: var(--theme--font_ui-size);
text-decoration: none;
border-radius: 5px;
padding: 0.125rem 0.25rem 0.25rem 0;
}
.library--authors img {
height: 1em;
width: 1em;
margin-bottom: 0.15625em;
display: inline-block;
vertical-align: middle;
border-radius: 50%;
object-fit: cover;
}
.library--options {
border-top: 1px solid var(--theme--divider);
}
.library--toggle_label,
.library--select_label,
.library--text_label,
.library--number_label,
.library--color_label,
.library--file_label {
margin: 0.6rem 0;
display: block;
appearance: none;
font-size: var(--theme--font_ui-size);
}
.library--toggle_label *,
.library--select_label *,
.library--text_label *,
.library--number_label *,
.library--color_label *,
.library--file_label * {
appearance: none;
font-family: var(--theme--font_sans);
font-size: var(--theme--font_ui-size);
color: var(--theme--text);
}
label p:not([class]),
label p > span:not([class]),
label > span:not([class]) {
font-size: 1rem;
}
label [data-icon='fa/solid/question-circle'] {
height: var(--theme--font_ui_small-size);
width: var(--theme--font_ui_small-size);
margin-left: 0.25em;
margin-bottom: -1px;
}
.library--toggle_label > p,
.library--toggle_label > p,
.library--select_label > p,
.library--text_label > p,
.library--number_label > p,
.library--color_label > p,
.library--file_label > p {
margin: 0.6rem 0;
}
.library--toggle_label > :not(input) {
align-items: center;
display: flex;
--toggle--bg: var(--theme--toggle_off);
--toggle--translation: 0%;
}
.library--toggle_label > :not(input) .library--toggle {
position: relative;
margin: auto 0 auto auto;
min-width: 2.25rem;
width: 2.25rem;
height: 1.25rem;
display: block;
border-radius: 40px;
background: var(--toggle--bg);
cursor: pointer;
}
.library--toggle_label > :not(input) .library--toggle::after {
content: '';
transition: transform 200ms ease-out, background 200ms ease-out;
height: 0.8rem;
width: 0.8rem;
left: 0.325rem;
top: 0.2rem;
position: absolute;
border-radius: 100%;
background: var(--theme--toggle_dot);
transform: translateX(var(--toggle--translation));
}
.library--toggle_label input[type='checkbox']:checked + :not(input) {
--toggle--bg: var(--theme--toggle_on);
--toggle--translation: 100%;
}
.library--toggle_label:focus-within .library--toggle,
.library--file_label:focus-within .library--file,
.library--color_label:focus-within .library--color,
.library--select_label .library--select:focus-within {
outline: -webkit-focus-ring-color auto 1px;
}
.library--toggle_label input,
.library--file_label input {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
white-space: nowrap;
border-width: 0;
}
.library--text_label textarea {
resize: none;
min-height: calc(1em + 16px);
height: var(--txt--scroll-height);
}
.library--text_label textarea,
.library--number_label input,
.library--select_label .library--select,
.library--color_label .library--color,
.library--file_label .library--file {
width: 100%;
padding: 6px 8px;
background: var(--theme--input);
border-radius: 3px;
border: none;
box-shadow: var(--theme--input-border) 0px 0px 0px 1px inset;
}
.library--select_label .library--select select,
.library--color_label .library--color input {
outline: none;
cursor: pointer;
}
.library--color_label .library--color input {
border-radius: 0;
}
.library--select_label .library--select select option {
background: var(--theme--tag_select);
}
.library--select_label .library--select,
.library--color_label .library--color,
.library--file_label .library--file {
padding: 0;
display: flex;
cursor: pointer;
}
.library--select_label .library--select > :first-child,
.library--color_label .library--color > :first-child,
.library--file_label .library--file > :first-child {
display: flex;
padding: 6px 8px;
background: var(--theme--input-border);
}
.library--select_label .library--select > :first-child svg,
.library--color_label .library--color > :first-child svg,
.library--file_label .library--file > :first-child svg {
width: 0.9em;
margin: auto 0;
}
.library--select_label .library--select > :first-child svg *,
.library--color_label .library--color > :first-child svg *,
.library--file_label .library--file > :first-child svg * {
color: var(--theme--input_icon);
}
.library--select_label .library--select > :last-child,
.library--color_label .library--color > :last-child,
.library--file_label .library--file > :last-child {
margin: auto 0;
padding: 6px 8px;
width: 100%;
height: 100%;
border: none;
background: none;
}
.library--file_label .library--file_title {
display: flex;
}
.library--file_title .library--file_remove {
margin-top: auto;
padding-top: 0.2em;
margin-left: auto;
}
.library--file_title .library--file_remove svg {
cursor: pointer;
width: 0.8em;
}
.library--title,
.library--title h2 {
margin: 0;
font-size: var(--theme--font_ui-size);
}
.library--title h2 > span {
font-size: var(--theme--font_heading2-size);
}
.library--card > div {
padding: 1rem;
}
[data-view='mod'] main .library--card {
overflow-x: auto;
}
.documentation--body {
padding: 1rem 2rem;
font-size: 1rem;
overflow-x: auto;
}
.notifications {
position: absolute;
bottom: 1.5rem;
right: 1.5rem;
}
.notification {
background: var(--theme--block_grey);
color: var(--theme--block_grey-text);
max-width: calc(100vw - 3rem);
width: 25rem;
font-size: var(--theme--font_ui-size);
border-top: 4px currentColor solid;
border-bottom-left-radius: 3px;
border-bottom-right-radius: 3px;
box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1), 0 2px 4px -1px rgba(0, 0, 0, 0.06) !important;
padding: 1.25em 1.5em;
display: flex;
margin-top: 1rem;
position: relative;
transition: opacity 200ms ease-in-out, transform 115ms ease-out 200ms,
margin-top 115ms ease-out 200ms;
transform: scaleY(1);
transform-origin: 100% 100%;
opacity: 0;
}
.notification :not(.notification--dismiss) > svg {
height: 1.5rem;
width: 1.5rem;
margin-top: 0.25rem;
margin-right: 1rem;
}
.notification h3 {
margin: 0 0 0.25rem 0;
font-weight: bold;
}
.notification p {
margin: 0.25rem 0 0.5rem 0;
}
.notification .notification--dismiss {
position: absolute;
top: 0.5rem;
right: 0.75rem;
background: none;
border: none;
padding: 0.15rem 0 0.15rem 0.5rem;
width: var(--theme--font_body-size);
color: currentColor;
cursor: pointer;
transition: opacity 200ms ease-in-out;
opacity: 0;
}
.notification .notification--dismiss svg {
width: 100%;
}
.notification:hover .notification--dismiss,
.notification:focus-within .notification--dismiss {
opacity: 1;
}
::-webkit-scrollbar {
background: transparent;
width: 10px;
height: 10px;
background: transparent;
}
::-webkit-scrollbar-track {
background: var(--theme--scrollbar_track);
}
::-webkit-scrollbar-track,
::-webkit-scrollbar-corner {
background: var(--theme--scrollbar_track);
background: var(--theme--scrollbar_track) !important;
}
::-webkit-scrollbar-thumb {
background: var(--theme--scrollbar_thumb);
background: var(--theme--scrollbar_thumb) !important;
}
::-webkit-scrollbar-thumb:hover {
background: var(--theme--scrollbar_thumb-hover);
background: var(--theme--scrollbar_thumb-hover) !important;
}

View File

@ -1,30 +1,11 @@
<!DOCTYPE html>
<html lang="en" style="opacity: 0">
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>notion-enhancer</title>
<title>notion-enhancer menu</title>
</head>
<body data-view>
<header>
<h1>
<img data-notion alt="" width="24" src="../../icons/colour.svg" />
<a href="?view=library">library</a>
</h1>
<h1>
<i data-icon="fa/solid/info"></i>
<a href="https://notion-enhancer.github.io/">website</a>
</h1>
<h1>
<i data-icon="fa/solid/code"></i>
<a href="https://github.com/notion-enhancer/extension">source code</a>
</h1>
<h1>
<i data-icon="fa/brands/discord"></i>
<a href="https://discord.gg/sFWPXtA">support</a>
</h1>
</header>
<main></main>
<script src="./menu.js" type="module"></script>
<body>
<script src="./menu.mjs" type="module"></script>
</body>
</html>

View File

@ -7,83 +7,9 @@
'use strict';
const _id = 'a6621988-551d-495a-97d8-3c568bca2e9e';
import { env, storage, web, fmt, fs, registry } from '../../api/_.mjs';
for (const mod of await registry.get((mod) => registry.isEnabled(mod.id))) {
for (const sheet of mod.css?.menu || []) {
web.loadStylesheet(`repo/${mod._dir}/${sheet}`);
}
}
async function loadTheme() {
document.documentElement.className = `notion-${
(await storage.get(_id, 'theme')) || 'dark'
}-theme`;
}
window.addEventListener('focus', loadTheme);
loadTheme();
import { env, storage, web, fmt, fs, registry, regexers } from '../../api/_.mjs';
document.querySelector('img[data-notion]').addEventListener('click', env.focusNotion);
web.addHotkeyListener(await storage.get(_id, 'hotkey.focustoggle'), env.focusNotion);
web.addDocumentObserver(web.loadIcons);
const notifications = {
$el: web.createElement(web.html`<footer class="notifications"></footer>`),
push({ heading, message, icon, color }, onDismiss = () => {}) {
const $notif = web.createElement(web.html`
<div role="alert" class="notification" style="
background: var(--theme--block_${web.escapeHtml(color)});
color: var(--theme--block_${web.escapeHtml(color)}-text);
">
<div><i data-icon="${web.escapeHtml(icon)}"></i></div>
<div>
<h3>${web.escapeHtml(heading)}</h3>
<p>${fmt.md.renderInline(message)}</p>
</div>
<button class="notification--dismiss"><i data-icon="fa/solid/times"></i></button>
</div>`);
this.$el.append($notif);
setTimeout(() => {
$notif.style.opacity = 1;
}, 100);
$notif.querySelector('.notification--dismiss').addEventListener('click', (event) => {
$notif.style.opacity = 0;
$notif.style.transform = 'scaleY(0)';
$notif.style.marginTop = `-${
$notif.offsetHeight / parseFloat(getComputedStyle(document.documentElement).fontSize)
}rem`;
setTimeout(() => $notif.remove(), 400);
onDismiss();
});
return $notif;
},
};
document.body.append(notifications.$el);
for (const error of await registry.errors()) {
notifications.push({
heading: `error: ${error.source}`,
message: error.message,
color: 'red',
icon: 'fa/solid/exclamation-triangle',
});
}
for (const notification of await (async () => {
const dismissed = await storage.get('_notifications', 'dismissed', []);
return (await fs.getJSON('https://notion-enhancer.github.io/notifications.json'))
.sort((a, b) => b.id - a.id)
.filter(({ id }) => !dismissed.includes(id));
})()) {
if (
(!notification.versions || notification.versions.includes(env.version)) &&
(!notification.environments || notification.environments.includes(env.name))
) {
notifications.push(notification, async () => {
const dismissed = await storage.get('_notifications', 'dismissed', []);
storage.set('_notifications', 'dismissed', [
...new Set([...dismissed, notification.id]),
]);
});
}
}
import * as router from './router.js';

View File

@ -0,0 +1,145 @@
/*
* notion-enhancer core: menu
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
'use strict';
// css-in-js for better component generation
import { tw, apply, setup } from '../../dep/twind.mjs';
const mapColorVariables = (color) => ({
'text': `var(--theme--text_${color})`,
'highlight': `var(--theme--highlight_${color})`,
'highlight-text': `var(--theme--highlight_${color}-text)`,
'block': `var(--theme--block_${color})`,
'block-text': `var(--theme--block_${color}-text)`,
'tag': `var(--theme--tag_${color})`,
'tag-text': `var(--theme--tag_${color}-text)`,
'callout': `var(--theme--callout_${color})`,
'callout-text': `var(--theme--callout_${color}-text)`,
});
setup({
preflight: {
body: apply`px-4 py-3 bg-notion-bg font-sans`,
},
theme: {
fontFamily: {
sans: ['var(--theme--font_sans)'],
mono: ['var(--theme--font_mono)'],
},
colors: {
'notion': {
'bg': 'var(--theme--bg)',
'secondary': 'var(--theme--bg_secondary)',
'popup': 'var(--theme--bg_popup)',
'divider': 'var(--theme--ui_divider)',
},
'icon': 'var(--theme--icon)',
'icon_ui': 'var(--theme--icon_ui)',
'foreground': 'var(--theme--text)',
'foreground_ui': 'var(--theme--text_ui)',
'interactive': 'var(--theme--ui_interactive)',
'interactive-hover': 'var(--theme--ui_interactive-hover)',
'toggle': {
'on': 'var(--theme--ui_toggle-on)',
'off': 'var(--theme--ui_toggle-off)',
'feature': 'var(--theme--ui_toggle-feature)',
},
'accent': {
'blue': 'var(--theme--accent_blue)',
'blue-contrast': 'var(--theme--accent_blue-text)',
'red': 'var(--theme--accent_red)',
'red-contrast': 'var(--theme--accent_red-text)',
},
'grey': mapColorVariables('grey'),
'brown': mapColorVariables('brown'),
'orange': mapColorVariables('orange'),
'yellow': mapColorVariables('yellow'),
'green': mapColorVariables('green'),
'blue': mapColorVariables('blue'),
'purple': mapColorVariables('purple'),
'pink': mapColorVariables('pink'),
'red': mapColorVariables('red'),
},
},
});
// initialisation and external interactions
import * as api from '../../api/_.mjs';
import { render } from '../../api/web.mjs';
const { env, fs, registry, web } = api,
db = await registry.db('a6621988-551d-495a-97d8-3c568bca2e9e');
web.addHotkeyListener(await db.get(['hotkey']), env.focusNotion);
for (const mod of await registry.list((mod) => registry.enabled(mod.id))) {
for (const sheet of mod.css?.menu || []) {
web.loadStylesheet(`repo/${mod._dir}/${sheet}`);
}
}
const loadTheme = async () => {
document.documentElement.className =
(await db.get(['theme'], 'light')) === 'dark' ? 'dark' : '';
};
document.addEventListener('visibilitychange', loadTheme);
loadTheme();
const notifications = {
$container: web.html`<div class="${tw`absolute bottom-3 right-4 w-80`}"></div>`,
cache: await db.get(['notifications'], []),
provider: [
env.welcomeNotification,
...(await fs.getJSON('https://notion-enhancer.github.io/notifications.json')),
],
add({ icon, message, id = undefined, color = undefined, link = undefined }) {
const style = tw`p-2 ${
color ? `bg-${color}-tag text-${color}-tag-text` : 'bg-notion-popup text-foreground'
} flex items-center rounded-full mt-3 shadow-md cursor-pointer`,
$notification = web.render(
link
? web.html`<a href="${web.escape(
link
)}" class="${style}" role="alert" target="_blank"></a>`
: web.html`<p class="${style}" role="alert"></p>`,
web.html`<span class="${tw`font-semibold mx-2 flex-auto`}">
${message}
</span>`,
web.html`${web.icon(icon, { class: tw`fill-current opacity-75 h-4 w-4 mx-2` })}`
);
$notification.addEventListener('click', async () => {
if (id !== undefined) {
notifications.cache.push(id);
await db.set(['notifications'], notifications.cache);
}
$notification.remove();
});
web.render(notifications.$container, $notification);
},
};
render(document.body, notifications.$container);
for (const notification of notifications.provider) {
if (
!notifications.cache.includes(notification.id) &&
notification.version === env.version &&
(!notification.environments || notification.environments.includes(env.name))
) {
notifications.add(notification);
}
}
const errors = await registry.errors();
if (errors.length) {
console.log('[notion-enhancer] registry errors:');
console.table(errors);
notifications.add({
icon: 'alert-circle',
message: 'Failed to load mods (check console).',
color: 'red',
});
}

View File

@ -14,7 +14,7 @@
],
"css": {
"client": ["client.css"],
"menu": ["menu.css", "markdown.css"]
"menu": ["menu.css"]
},
"js": {
"client": ["client.mjs"]

View File

@ -9,11 +9,12 @@
export default function (api, db) {
const { web } = api;
const $root = document.querySelector(':root');
$root.classList[document.body.classList.contains('dark') ? 'add' : 'remove']('dark');
const updateTheme = () =>
document.documentElement.classList[
document.body.classList.contains('dark') ? 'add' : 'remove'
]('dark');
updateTheme();
web.addDocumentObserver((mutation) => {
if (mutation.target === document.body) {
$root.classList[document.body.classList.contains('dark') ? 'add' : 'remove']('dark');
}
if (mutation.target === document.body) updateTheme();
});
}

View File

@ -61,7 +61,6 @@
--theme--text: rgb(55, 43, 47);
--theme--text_ui: rgba(55, 43, 47, 0.6);
--theme--text_placeholder: rgba(55, 43, 47, 0.4);
--theme--text_grey: rgb(155, 154, 151);
--theme--text_brown: rgb(100, 71, 58);
--theme--text_orange: rgb(217, 115, 13);