notion-icons: toggleable community icon packs

This commit is contained in:
CloudHill 2020-11-24 13:53:06 +07:00
parent f86e87a1ab
commit 0d37708ad8
4 changed files with 361 additions and 127 deletions

View File

@ -17,7 +17,9 @@
.notion-icons--tab > div:hover, .notion-icons--tab > div:hover,
.notion-icons--icon:hover, .notion-icons--icon:hover,
.notion-icons--toggle:hover { .notion-icons--toggle:hover,
.notion-icons--restore-button:hover,
.notion-icons--removed-set:hover {
background: var(--theme--interactive_hover); background: var(--theme--interactive_hover);
box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border) !important; box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border) !important;
} }
@ -50,7 +52,6 @@
border-radius: 2px; border-radius: 2px;
} }
.notion-icons--icon-set.error { .notion-icons--icon-set.error {
color: var(--theme--text_red); color: var(--theme--text_red);
background: var(--theme--line_red); background: var(--theme--line_red);
@ -65,17 +66,6 @@
float: right; float: right;
} }
.notion-icons--icon-set.alert .notion-icons--toggle {
color: var(--theme--line_yellow-text);
background: var(--theme--line_yellow);
border: 1px solid var(--theme--select_yellow);
margin-left: -1px;
margin-right: -1px;
}
.notion-icons--icon-set.alert .notion-icons--toggle:hover {
background: var(--theme--select_yellow);
}
.notion-icons--toggle { .notion-icons--toggle {
display: flex; display: flex;
align-items: center; align-items: center;
@ -86,7 +76,6 @@
border-radius: 2px; border-radius: 2px;
transition: background 200ms, margin-bottom 200ms ease-in; transition: background 200ms, margin-bottom 200ms ease-in;
} }
.notion-icons--toggle .triangle { .notion-icons--toggle .triangle {
width: 0.9em; width: 0.9em;
height: 1em; height: 1em;
@ -94,7 +83,6 @@
transition: transform 200ms ease-out 0s; transition: transform 200ms ease-out 0s;
transform: rotateZ(180deg); transform: rotateZ(180deg);
} }
.notion-icons--toggle a { .notion-icons--toggle a {
color: var(--theme-text); color: var(--theme-text);
transition: color 20ms ease-in; transition: color 20ms ease-in;
@ -103,6 +91,17 @@
color: var(--theme--primary); color: var(--theme--primary);
} }
.notion-icons--icon-set.alert .notion-icons--toggle {
color: var(--theme--line_yellow-text);
background: var(--theme--line_yellow);
border: 1px solid var(--theme--select_yellow);
margin-left: -1px;
margin-right: -1px;
}
.notion-icons--icon-set.alert .notion-icons--toggle:hover {
background: var(--theme--select_yellow);
}
.notion-icons--body { .notion-icons--body {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
@ -122,7 +121,6 @@
.hidden-set .triangle { .hidden-set .triangle {
transform: rotateZ(90deg); transform: rotateZ(90deg);
} }
.hidden-set .notion-icons--body { .hidden-set .notion-icons--body {
opacity: 0; opacity: 0;
} }
@ -139,6 +137,10 @@
height: 40px; height: 40px;
padding: 4px; padding: 4px;
} }
.notion-icons--icon img {
width: 100%;
height: 100%;
}
.notion-icons--icon.error { .notion-icons--icon.error {
font-size: 20px; font-size: 20px;
@ -146,20 +148,97 @@
border: 1px solid var(--theme--select_yellow); border: 1px solid var(--theme--select_yellow);
color: var(--theme--text_yellow); color: var(--theme--text_yellow);
} }
.notion-icons--icon.error:hover { .notion-icons--icon.error:hover {
background: var(--theme--select_yellow); background: var(--theme--select_yellow);
} }
.notion-icons--icon img {
width: 100%; .notion-icons--extra {
height: 100%; margin-left: auto;
margin-right: 8px;
display: flex;
align-items: center;
} }
.notion-icons--spinner {
width: 12px;
height: 12px;
}
.notion-icons--spinner img { .notion-icons--spinner img {
width: 100%;
height: 100%;
animation: rotation 1.3s infinite linear; animation: rotation 1.3s infinite linear;
} }
.notion-icons--remove-button {
display: flex;
justify-content: center;
align-items: center;
margin-left: 8px;
width: 16px;
height: 16px;
position: relative;
}
.notion-icons--remove-button::before {
content: 'Hide icon set';
position: absolute;
right: -3px;
padding: 4px 22px 4px 6px;
background: var(--theme--main);
box-shadow: var(--theme--box-shadow);
white-space: nowrap;
opacity: 0;
pointer-events: none;
transition: opacity 50ms ease-in;
}
.notion-icons--remove-button:hover::before {
opacity: 1;
pointer-events: auto;
}
.notion-icons--remove-button svg {
width: 100%;
height: 100%;
fill: inherit;
z-index: 1;
}
.notion-icons--restore-button svg {
width: 16px;
height: 16px;
fill: inherit;
}
.notion-icons--overlay-container {
position: fixed;
top: 0;
bottom: 0;
left: 0;
right: 0;
z-index: 999;
overflow: hidden;
}
.notion-icons--restore {
max-width: 320px;
max-height: 320px;
position: relative;
border-radius: 3px;
padding: 8px 0;
box-shadow: var(--theme--box-shadow_strong);
background: var(--theme--card);
overflow: hidden auto;
}
.notion-icons--removed-set {
display: flex;
align-items: center;
width: 100%;
padding: 8px 14px;
user-select: none;
cursor: pointer;
transition: background 0.4s ease;
}
@keyframes rotation { @keyframes rotation {
from { from {
transform: rotate(0deg); transform: rotate(0deg);

View File

@ -0,0 +1,3 @@
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24">
<polygon class="cls-1" points="18.72 16.6 14.12 12 18.72 7.4 16.6 5.28 12 9.88 7.4 5.28 5.28 7.4 9.88 12 5.28 16.6 7.4 18.72 12 14.12 16.6 18.72 18.72 16.6"/>
</svg>

After

Width:  |  Height:  |  Size: 251 B

View File

@ -0,0 +1,3 @@
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24">
<polygon class="cls-1" points="20 10.5 13.5 10.5 13.5 4 10.5 4 10.5 10.5 4 10.5 4 13.5 10.5 13.5 10.5 20 13.5 20 13.5 13.5 20 13.5 20 10.5"/>
</svg>

After

Width:  |  Height:  |  Size: 234 B

View File

@ -9,14 +9,15 @@
'use strict'; 'use strict';
const { createElement } = require('../../pkg/helpers.js'), const { createElement } = require('../../pkg/helpers.js'),
fs = require('fs-extra'); fs = require('fs-extra'),
path = require('path');
module.exports = { module.exports = {
id: '2d1f4809-9581-40dd-9bf3-4239db406483', id: '2d1f4809-9581-40dd-9bf3-4239db406483',
tags: ['extension'], tags: ['extension'],
name: 'notion icons', name: 'notion icons',
desc: desc:
'use custom icon sets directly in Notion.', 'use custom icon sets directly in notion.',
version: '1.0.0', version: '1.0.0',
author: 'jayhxmo', author: 'jayhxmo',
options: [ options: [
@ -47,13 +48,21 @@ module.exports = {
httpReq.send(null); httpReq.send(null);
} }
// Retrieve Icons data let modalIcons;
(async () => {
modalIcons = {
remove: await fs.readFile( path.resolve(__dirname, 'icons/remove.svg') ),
restore: await fs.readFile( path.resolve(__dirname, 'icons/restore.svg') ),
}
})();
// Retrieve icons data
let notionIconsData; let notionIconsData;
getAsync(iconsUrl + 'icons.json', iconsData => { getAsync(iconsUrl + 'icons.json', iconsData => {
notionIconsData = JSON.parse(iconsData); notionIconsData = JSON.parse(iconsData);
}); });
// Retrieve custom Icons data // Retrieve custom icons data
let customIconsData; let customIconsData;
if (store().json) { if (store().json) {
customIconsData = JSON.parse( customIconsData = JSON.parse(
@ -92,26 +101,25 @@ module.exports = {
} }
function addIconsTab() { function addIconsTab() {
// Prevent Icons tab duplication // Prevent icons tab duplication
if (getTab(5)) { if (getTab(5)) {
removeIcons(); removeIcons();
return; return;
} }
// Change 'Upload an image' to 'Upload' // Change 'Upload an image' to 'Upload'
getTab(2, true).innerText = 'Upload'; getTab(2, true).innerText = 'Upload';
// Initialize Icons tab // Initialize icons tab
const iconsTab = getTab(3).cloneNode(true); const iconsTab = getTab(3).cloneNode(true);
iconsTab.className = 'notion-icons--tab' iconsTab.className = 'notion-icons--tab';
iconsTab.firstChild.innerText = 'Icons'; iconsTab.firstChild.innerText = 'Icons';
iconsTab.firstChild.addEventListener('click', renderIconsOverlay); iconsTab.firstChild.addEventListener('click', renderIconsOverlay);
// Insert Icons tab // Insert icons tab
const tabStrip = getTab(1).parentElement; const tabStrip = getTab(1).parentElement;
tabStrip.insertBefore(iconsTab, tabStrip.lastChild); tabStrip.insertBefore(iconsTab, tabStrip.lastChild);
// Remove the Icons overlay when clicking... // Remove the icons overlay when clicking...
const closeTriggers = [ const closeTriggers = [
// The fog layer // The fog layer
document.querySelector('.notion-overlay-container [style*="width: 100vw; height: 100vh;"]'), document.querySelector('.notion-overlay-container [style*="width: 100vw; height: 100vh;"]'),
@ -126,111 +134,90 @@ module.exports = {
garbageCollector.push(trigger); garbageCollector.push(trigger);
}) })
// Remove the Icons overlay when pressing the Escape key // Remove the icons overlay when pressing the Escape key
document.querySelector('.notion-media-menu') document.querySelector('.notion-media-menu')
.addEventListener('keydown', e => { .addEventListener('keydown', e => {
if (e.keyCode === 27) removeIcons(); if (e.keyCode === 27) removeIcons();
}); });
} }
function renderIconSet(iconData) {
const iconSet = createElement(
'<div class="notion-icons--icon-set"></div>'
)
try { function addRestoreButton() {
const buttons = getTab(5) ? getTab(5) : getTab(4);
const restoreButton = buttons.lastChild.cloneNode(true);
restoreButton.className = 'notion-icons--restore-button';
restoreButton.innerHTML = modalIcons.restore;
buttons.prepend(restoreButton);
restoreButton.addEventListener('click', renderRestoreOverlay);
}
const authorText = iconData.author function renderRestoreOverlay() {
? iconData.authorUrl if (!store().removedSets) return;
? ` by <a target="_blank" href="${iconData.authorUrl}" style="opacity: 0.6;">${iconData.author}</a>`
: ` by <span style="opacity: 0.6;">${iconData.author}</span>`
: '';
const iconSetToggle = createElement( store().removedSets.sort((a, b) => {
`<div class="notion-icons--toggle"> const setA = a.name.toLowerCase(),
<svg viewBox="0 0 100 100" class="triangle"><polygon points="5.9,88.2 50,11.8 94.1,88.2"></polygon></svg> setB = b.name.toLowerCase();
<div style="white-space: nowrap; overflow: hidden; text-overflow: ellipsis;">${iconData.name}${authorText}</div>
<div class="notion-icons--spinner" style="margin-left: auto; margin-right: 8px; width: 1.2em; height: 1.2em;"> if (setA < setB) return -1;
<img src="/images/loading-spinner.4dc19970.svg" /> if (setA > setB) return 1;
</div> return 0;
</div>` });
);
const iconSetBody = createElement(
'<div class="notion-icons--body"></div>'
);
iconSet.append(iconSetToggle); const overlayContainer = createElement(`
iconSet.append(iconSetBody); <div class="notion-icons--overlay-container"></div>
`);
overlayContainer.addEventListener('click', closeRestoreOverlay);
document.querySelector('.notion-app-inner').appendChild(overlayContainer);
const promiseArray = []; const rect = document.querySelector('.notion-icons--restore-button')
// Render Icons .getBoundingClientRect();
for (let i = 0; i < (iconData.count || iconData.source.length); i++) { const div = createElement(`
<div style="position: fixed; top: ${rect.top}px; left: ${rect.left}px; height: ${rect.height}px;">
<div style="position: relative; top: 100%; pointer-events: auto;"></div>
</div>
`);
const iconUrl = iconData.sourceUrl const restoreOverlay = createElement(`
? Array.isArray(iconData.source) <div class="notion-icons--restore"></div>
? `${iconData.sourceUrl}/${iconData.source[i]}.${iconData.extension}` `)
: `${iconData.sourceUrl}/${iconData.source}_${i}.${iconData.extension}`
: iconData.source[i];
const icon = createElement(`<div class="notion-icons--icon"></div>`); overlayContainer.appendChild(div);
if (iconData.enhancerIcons) { div.firstElementChild.appendChild(restoreOverlay);
// Load sprite sheet
icon.innerHTML =
`<div style="width: 32px; height: 32px; background: url(${iconsUrl}${iconData.source}/sprite.png) 0 -${i * 32}px no-repeat; background-size: 32px;"></div>`;
} else {
icon.innerHTML = `<img src="${iconUrl}" />`;
// Make sure icons load
promiseArray.push(
new Promise((resolve, reject) => {
icon.firstChild.onload = resolve;
icon.firstChild.onerror = () => {
reject();
icon.classList.add('error');
icon.innerHTML = '!';
};
})
);
}
iconSetBody.append(icon); // Fade in
garbageCollector.push(icon); restoreOverlay.animate(
icon.addEventListener('click', () => setPageIcon(iconUrl)); [ {opacity: 0}, {opacity: 1} ],
} { duration: 200 }
);
// Hide spinner after all icons finish loading
(async () => {
const spinner = iconSetToggle.querySelector('.notion-icons--spinner'),
loadPromise = Promise.all(promiseArray);
loadPromise.then(
() => spinner.remove(),
() => {
iconSet.classList.add('alert')
spinner.remove();
}
)
})();
// Set up Toggle store().removedSets.forEach(iconData => {
requestAnimationFrame(() => { const restoreItem = renderRestoreItem(iconData);
iconSetBody.style.height = iconSetBody.style.maxHeight = `${iconSetBody.offsetHeight}px`; restoreOverlay.appendChild(restoreItem);
if (store().hide) iconSetToggle.click(); })
}); }
iconSetToggle.addEventListener('click', e => {
if (e.target.nodeName === 'A') return;
iconSet.classList.toggle('hidden-set')
iconSetBody.style.height = iconSet.classList.contains('hidden-set')
? 0
: iconSetBody.style.maxHeight;
});
} catch (err) {
iconSet.classList.add('error');
iconSet.innerHTML = `Invalid Icon Set: ${iconData.name}`;
}
return iconSet; function renderRestoreItem(iconData) {
const iconUrl = `${iconData.sourceUrl}/${iconData.source}_${0}.${iconData.extension}`;
const restoreItem = createElement(`
<div class="notion-icons--removed-set">
<div style="flex-grow: 0; flex-shrink: 0; width: 32px; height: 32px;">
<img style="width: 100%; height: 100%" src="${iconUrl}" />
</div>
<span style="margin: 0 8px;">${iconData.name}</span>
</div>
`)
restoreItem.addEventListener('click', () => restoreIconSet(iconData));
return restoreItem;
}
function closeRestoreOverlay() {
const overlayContainer = document.querySelector('.notion-icons--overlay-container');
overlayContainer.removeEventListener('click', closeRestoreOverlay);
// Fade out
document.querySelector('.notion-icons--restore').animate(
[ {opacity: 1}, {opacity: 0} ],
{ duration: 200 }
).onfinish = () => overlayContainer.remove();
} }
function renderIconsOverlay() { function renderIconsOverlay() {
@ -238,17 +225,24 @@ module.exports = {
// Switch to 3rd tab so that the link can be inputed in the underlay // Switch to 3rd tab so that the link can be inputed in the underlay
if (!isCurrentTab(3)) getTab(3, true).click(); if (!isCurrentTab(3)) getTab(3, true).click();
// Set active bar on Icons Tab if (
store().removedSets &&
store().removedSets.length > 0
) {
addRestoreButton();
}
// Set active bar on icons tab
const iconsTab = getTab(4); const iconsTab = getTab(4);
const activeBar = createElement( const activeBar = createElement(
`<div id="notion-icons--active-bar"></div>` `<div id="notion-icons--active-bar"></div>`
) )
activeBar.style = 'border-bottom: 2px solid var(--theme--text); position: absolute; bottom: -1px; left: 8px; right: 8px;'; activeBar.style = 'border-bottom: 2px solid var(--theme--text); position: absolute; bottom: -1px; left: 8px; right: 8px;';
iconsTab.append(activeBar); iconsTab.appendChild(activeBar);
getTab(4).style.position = 'relative'; getTab(4).style.position = 'relative';
getTab(3).className = 'hide-active-bar'; getTab(3).className = 'hide-active-bar';
// Convert Icons data into renderable // Convert icons data into renderable
const iconSets = []; const iconSets = [];
if (customIconsData && customIconsData.icons) { if (customIconsData && customIconsData.icons) {
@ -267,26 +261,175 @@ module.exports = {
if (notionIconsData && notionIconsData.icons) { if (notionIconsData && notionIconsData.icons) {
notionIconsData.icons.forEach(i => { notionIconsData.icons.forEach(i => {
i.sourceUrl = i.sourceUrl || (iconsUrl + i.source); i.sourceUrl = i.sourceUrl || (iconsUrl + i.source);
if ( store().removedSets ) {
for (let iconData of store().removedSets) {
if (iconData.source === i.source) return;
}
}
i.enhancerIcons = true; i.enhancerIcons = true;
iconSets.push( renderIconSet(i) ); iconSets.push( renderIconSet(i) );
}); });
} }
// Create Icons overlay // Create icons overlay
const notionIcons = createElement( const notionIcons = createElement(
'<div id="notion-icons"></div>' '<div id="notion-icons"></div>'
); );
iconSets.forEach( set => notionIcons.append(set) ); iconSets.forEach( set => notionIcons.appendChild(set) );
// Insert Icons overlay // Insert icons overlay
document.querySelector('.notion-media-menu > .notion-scroller') document.querySelector('.notion-media-menu > .notion-scroller')
.append(notionIcons); .appendChild(notionIcons);
} }
} }
function renderIconSet(iconData) {
const iconSet = createElement('<div class="notion-icons--icon-set"></div>')
try {
const authorText = iconData.author
? iconData.authorUrl
? ` by <a target="_blank" href="${iconData.authorUrl}" style="opacity: 0.6;">${iconData.author}</a>`
: ` by <span style="opacity: 0.6;">${iconData.author}</span>`
: '';
const iconSetToggle = createElement(
`<div class="notion-icons--toggle">
<svg viewBox="0 0 100 100" class="triangle"><polygon points="5.9,88.2 50,11.8 94.1,88.2"></polygon></svg>
<div style="white-space: nowrap; overflow: hidden; text-overflow: ellipsis;">${iconData.name}${authorText}</div>
<div class="notion-icons--extra">
<div class="notion-icons--spinner">
<img src="/images/loading-spinner.4dc19970.svg" />
</div>
</div>
</div>`
);
const iconSetBody = createElement(
'<div class="notion-icons--body"></div>'
);
iconSet.appendChild(iconSetToggle);
iconSet.appendChild(iconSetBody);
const promiseArray = [];
// Render icons
for (let i = 0; i < (iconData.count || iconData.source.length); i++) {
const iconUrl = iconData.sourceUrl
? Array.isArray(iconData.source)
? `${iconData.sourceUrl}/${iconData.source[i]}.${iconData.extension}`
: `${iconData.sourceUrl}/${iconData.source}_${i}.${iconData.extension}`
: iconData.source[i];
const icon = createElement(`<div class="notion-icons--icon"></div>`);
if (iconData.enhancerIcons) {
// Load sprite sheet
icon.innerHTML = `
<div style="width: 32px; height: 32px; background: url(${iconsUrl}${iconData.source}/sprite.png) 0 -${i * 32}px no-repeat; background-size: 32px;">
</div>
`;
} else {
icon.innerHTML = `<img src="${iconUrl}" />`;
// Make sure icons load
promiseArray.push(
new Promise((resolve, reject) => {
icon.firstChild.onload = resolve;
icon.firstChild.onerror = () => {
reject();
icon.classList.add('error');
icon.innerHTML = '!';
};
})
);
}
iconSetBody.appendChild(icon);
garbageCollector.push(icon);
icon.addEventListener('click', () => setPageIcon(iconUrl));
}
// Hide spinner after all icons finish loading
(async () => {
const spinner = iconSetToggle.querySelector('.notion-icons--spinner'),
loadPromise = Promise.all(promiseArray);
loadPromise.then(
() => spinner.remove(),
() => {
iconSet.classList.add('alert')
spinner.remove();
}
)
})();
// Add hide icon set button
if (iconData.enhancerIcons) {
const removeButton = createElement(
'<div class="notion-icons--remove-button"></div>'
);
removeButton.innerHTML = modalIcons.remove;
removeButton.addEventListener('click', e => {
e.stopPropagation();
removeIconSet(iconData)
});
iconSet.querySelector('.notion-icons--extra')
.appendChild(removeButton);
}
// Set up Toggle
requestAnimationFrame(() => {
iconSetBody.style.height = iconSetBody.style.maxHeight = `${iconSetBody.offsetHeight}px`;
if (store().removed) iconSetToggle.click();
});
iconSetToggle.addEventListener('click', e => {
if (e.target.nodeName === 'A') return;
toggleIconSet(iconSet);
});
} catch (err) {
iconSet.classList.add('error');
iconSet.innerHTML = `Invalid Icon Set: ${iconData.name}`;
}
return iconSet;
}
function toggleIconSet(iconSet) {
iconSet.classList.toggle('hidden-set');
const iconSetBody = iconSet.lastChild;
if (iconSetBody) {
iconSetBody.style.height = iconSet.classList.contains('hidden-set')
? 0
: iconSetBody.style.maxHeight;
}
}
function removeIconSet(iconData) {
if (!store().removedSets) store().removedSets = [];
for (const hiddenIconData of store().removedSets) {
if (hiddenIconData.source === iconData.source) return;
}
store().removedSets.push(iconData);
removeIcons();
renderIconsOverlay();
}
function restoreIconSet(iconData) {
if (!store().removedSets) return;
for (let i = 0; i < store().removedSets.length; i++) {
if (store().removedSets[i].source === iconData.source)
store().removedSets.splice(i, 1);
}
removeIcons();
renderIconsOverlay();
}
function removeIcons() { function removeIcons() {
const notionIcons = document.getElementById('notion-icons'), const notionIcons = document.getElementById('notion-icons'),
activeBar = document.getElementById('notion-icons--active-bar'); activeBar = document.getElementById('notion-icons--active-bar'),
restoreButton = document.querySelector('.notion-icons--restore-button'),
overlayContainer = document.querySelector('.notion-icons--overlay-container');
if (notionIcons) if (notionIcons)
notionIcons.remove(); notionIcons.remove();
@ -297,6 +440,12 @@ module.exports = {
} }
if (getTab(3)) getTab(3).className = ''; if (getTab(3)) getTab(3).className = '';
if (restoreButton)
restoreButton.remove();
if (overlayContainer)
closeRestoreOverlay();
if (garbageCollector.length) { if (garbageCollector.length) {
for (let i = 0; i < garbageCollector.length; i++) { for (let i = 0; i < garbageCollector.length; i++) {
garbageCollector[i] = null; garbageCollector[i] = null;