extension: icon sets part 2 (enhancer + custom sets, sprites & spinners)

This commit is contained in:
dragonwocky 2021-10-30 22:31:49 +11:00
parent 96988d67e0
commit 805cbb836d
8 changed files with 207 additions and 643 deletions

View File

@ -1,411 +0,0 @@
/*
* notion-icons
* (c) 2019 jayhxmo (https://jaymo.io/)
* (c) 2020 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (c) 2020 CloudHill
* under the MIT license
*/
/* tab */
[hide-active-bar] > :nth-child(2) {
display: none;
}
.notion-icons--tab,
.notion-icons--tab > div {
color: var(--theme--text) !important;
}
#notion-icons--active-bar {
border-bottom: 2px solid var(--theme--text);
position: absolute;
bottom: -1px;
left: 8px;
right: 8px;
}
.notion-icons--restore-button svg {
width: 16px;
height: 16px;
fill: var(--theme--text_ui_info);
}
/* interactive hover */
.notion-icons--tab > div:hover,
.notion-icons--icon:hover,
.notion-icons--toggle:hover,
.notion-icons--restore-button:hover,
.notion-icons--removed-set:hover {
background: var(--theme--interactive_hover);
box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border) !important;
}
/* container */
#notion-icons {
position: absolute;
top: 1px;
left: 0;
right: 0;
bottom: 0;
z-index: 9999;
display: flex;
flex-direction: column;
align-items: stretch;
background: var(--theme--card);
border-radius: 3px;
overflow: hidden;
}
/* search */
.notion-icons--search {
flex-shrink: 0;
height: 28px;
min-width: 0px;
margin: 9px 14px 10px;
padding: 3px 6px;
border-radius: 3px;
display: flex;
align-items: center;
position: relative;
font-size: 14px;
line-height: 1.2;
background: var(--theme--tag_input);
box-shadow: rgba(15, 15, 15, 0.2) 0px 0px 0px 1px inset;
user-select: none;
cursor: text;
}
.notion-dark-theme .notion-icons--search {
background: rgba(15, 15, 15, 0.3);
box-shadow: rgba(15, 15, 15, 0.1) 0px 0px 0px 1px inset;
}
.notion-icons--search input {
font-size: inherit;
line-height: inherit;
border: none;
background: none;
width: 100%;
display: block;
resize: none;
padding: 0px;
}
.notion-icons--search svg {
flex-grow: 0;
flex-shrink: 0;
width: 14px;
height: 14px;
display: block;
fill: inherit;
backface-visibility: hidden;
margin-right: 6px;
color: rgba(55, 53, 47, 0.8);
}
.notion-dark-theme .notion-icons--search svg {
color: rgb(202, 204, 206);
}
/* scroller */
.notion-icons--scroller {
padding: 8px 12px;
overflow: hidden auto;
display: flex;
flex-direction: column;
}
/* divider */
.notion-icons--divider {
height: 1px;
margin-bottom: 9px;
border-bottom: 1px solid var(--theme--table-border);
}
/* icon set */
.notion-icons--icon-set {
margin-bottom: 8px;
color: var(--theme--text);
font-size: 11px;
line-height: 1.5;
letter-spacing: 1px;
font-weight: 600;
border-radius: 2px;
}
.notion-icons--icon-set.error {
color: var(--theme--text_red);
background: var(--theme--block_red);
border: 1px solid var(--theme--tag_red);
padding: 8px 16px;
}
.notion-icons--icon-set.error::after {
content: '!';
display: block;
font-size: 1.6em;
line-height: 0.9;
float: right;
}
/* icon set header/toggle */
.notion-icons--toggle {
display: flex;
align-items: center;
margin-bottom: 8px;
padding: 0.25em;
border-radius: 2px;
text-transform: uppercase;
user-select: none;
cursor: pointer;
transition: background 200ms, margin-bottom 200ms ease-in;
}
.notion-icons--icon-set.alert .notion-icons--toggle {
color: var(--theme--block_yellow-text);
background: var(--theme--block_yellow);
border: 1px solid var(--theme--tag_yellow);
margin-left: -1px;
margin-right: -1px;
}
.notion-icons--icon-set.alert .notion-icons--toggle:hover {
background: var(--theme--tag_yellow);
}
.notion-icons--toggle .triangle {
flex-grow: 0;
flex-shrink: 0;
width: 0.9em;
height: 1em;
margin: 0 0.75em 0 0.5em;
transition: transform 200ms ease-out 0s;
transform: rotateZ(180deg);
}
.notion-icons--author {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.notion-icons--author span,
.notion-icons--author a {
color: var(--theme--text_ui_info);
transition: color 20ms ease-in;
}
.notion-icons--toggle a:hover {
color: var(--theme--primary);
}
/* icon set body */
.notion-icons--body {
display: flex;
flex-wrap: wrap;
align-items: flex-start;
flex-grow: 1;
margin-left: 1.2em;
overflow: hidden;
transition: height 200ms ease-in, opacity 200ms ease-in;
}
/* hidden icon set */
.notion-icons--icon-set[hidden-set] {
padding-bottom: 0;
}
.notion-icons--icon-set[hidden-set] .notion-icons--toggle {
margin-bottom: 0;
}
.notion-icons--icon-set[hidden-set] .triangle {
transform: rotateZ(90deg);
}
.notion-icons--icon-set[hidden-set] .notion-icons--body {
opacity: 0;
}
/* icons */
.notion-icons--icon {
width: 40px;
height: 40px;
padding: 4px;
border-radius: 3px;
display: flex;
align-items: center;
justify-content: center;
position: relative;
user-select: none;
cursor: pointer;
transition: background 20ms ease-in;
}
.notion-icons--icon img {
width: 100%;
height: 100%;
pointer-events: none;
}
/* spritesheet */
.notion-icons--icon div {
width: 32px;
height: 32px;
background-size: 32px;
background-repeat: no-repeat;
pointer-events: none;
}
.notion-icons--icon.error {
font-size: 20px;
background: var(--theme--block_yellow);
border: 1px solid var(--theme--tag_yellow);
color: var(--theme--text_yellow);
}
.notion-icons--icon.error:hover {
background: var(--theme--tag_yellow);
}
/* tooltip */
.notion-icons--tooltip {
position: fixed;
pointer-events: none;
z-index: 99;
}
.notion-icons--tooltip > div {
position: absolute;
top: 0px;
left: 0px;
width: 40px;
height: 40px;
display: flex;
flex-direction: column;
justify-content: flex-end;
align-items: center;
}
.notion-icons--tooltip-text {
bottom: calc(100% + 6px);
padding: 4px 8px;
border-radius: 3px;
display: flex;
align-items: center;
flex-direction: column;
position: relative;
max-width: calc(100vw - 24px);
background: rgb(15, 15, 15);
color: rgba(255, 255, 255, 0.9);
box-shadow: rgba(0, 0, 0, 0.3) 0px 1px 4px;
font-size: 12px;
line-height: 1.4;
font-weight: 500;
white-space: nowrap;
overflow: hidden;
}
.notion-dark-theme .notion-icons--tooltip-text {
background: rgb(202, 204, 206);
color: rgb(15, 15, 15);
}
/* actions */
.notion-icons--actions {
flex-grow: 0;
flex-shrink: 0;
margin-left: auto;
display: flex;
align-items: center;
}
/* spinner */
.notion-icons--spinner {
width: 12px;
height: 12px;
}
.notion-icons--spinner img {
width: 100%;
height: 100%;
animation: rotation 1.3s infinite linear;
}
/* remove button */
.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: var(--theme--text_ui_info);
z-index: 1;
}
/* restore icon sets modal */
.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;
}
/* animation */
@keyframes rotation {
from {
transform: rotate(0deg);
}
to {
transform: rotate(359deg);
}
}

View File

@ -33,7 +33,7 @@
background: var(--theme--ui_interactive-active);
}
.icon_sets--scroller {
.icon_sets--view {
padding: 0;
overflow: hidden;
display: flex;
@ -46,6 +46,9 @@
padding: 10px 14px;
}
.icon_sets--actions > .notion-focusable-within {
flex-grow: 1;
}
.icon_sets--link_input {
flex-grow: 1;
font-size: 14px;
@ -105,36 +108,74 @@
}
.icon_sets--list {
/* scroller */
height: 100%;
word-break: break-all;
overflow: hidden auto;
padding: 0 14px 10px 14px;
}
.icon_sets--title {
.icon_sets--error {
color: var(--theme--accent_red);
}
.icon_sets--title,
.icon_sets--error {
margin: 6px 0 8px 0;
color: var(--theme--text_secondary);
font-size: 11px;
font-weight: 500;
line-height: 1.2;
user-select: none;
text-transform: uppercase;
cursor: pointer;
border-radius: 2px;
padding: 0.25em;
display: flex;
align-items: center;
}
.icon_sets--title {
cursor: pointer;
color: var(--theme--text_secondary);
}
.icon_sets--title:hover {
background: var(--theme--ui_interactive-hover);
}
.icon_sets--title:active {
background: var(--theme--ui_interactive-active);
}
.icon_sets--title .info {
/* tooltips */
height: 1em;
margin-left: 0.5em;
}
.icon_sets--spinner {
margin-left: 0.5em;
height: 1em;
width: 1em;
}
.icon_sets--spinner img {
width: 100%;
height: 100%;
animation: rotation 1.3s infinite linear;
}
@keyframes rotation {
from {
transform: rotate(0deg);
}
to {
transform: rotate(359deg);
}
}
.icon_sets--title a {
color: currentColor;
transition: color 100ms ease-in;
}
.icon_sets--title a:hover {
color: var(--theme--accent_blue);
}
.icon_sets--title .triangle {
height: 1em;
width: 0.9em;
@ -177,3 +218,16 @@
max-width: 24px;
max-height: 24px;
}
.icon_sets--sprite {
width: 24px;
height: 24px;
background-size: 24px;
background-repeat: no-repeat;
pointer-events: none;
}
.icon_sets--divider {
height: 1px;
margin: 1em 0;
border-bottom: 1px solid var(--theme--ui_divider);
}

View File

@ -6,12 +6,26 @@
* (https://notion-enhancer.github.io/) under the MIT license
*/
export default async function ({ web, components, notion }, db) {
export default async function ({ web, fs, components, notion }, db) {
const recentUploads = await db.get(['recent_uploads'], []),
$triangleSvg = web.html`<svg viewBox="0 0 100 100" class="triangle">
<polygon points="5.9,88.2 50,11.8 94.1,88.2" />
</svg>`;
const customIconSets = [],
customIconsFile = await db.get(['json']);
if (customIconsFile?.content) {
const iconsData = JSON.parse(customIconsFile.content);
customIconSets.push(...(iconsData.icons || iconsData));
}
const enhancerIconSets = [],
enhancerIconsUrl = 'https://raw.githubusercontent.com/notion-enhancer/icons/main/';
if (await db.get(['default_sets'])) {
const iconsData = await fs.getJSON(`${enhancerIconsUrl}/icons.json`);
enhancerIconSets.push(...(iconsData.icons || iconsData));
}
const mediaMenuSelector = '.notion-media-menu',
mediaScrollerSelector = '.notion-media-menu > .notion-scroller',
mediaFilterSelector = '.notion-media-menu > :first-child > :last-child',
@ -19,25 +33,35 @@ export default async function ({ web, components, notion }, db) {
tabBtnSelector = (n) =>
`.notion-media-menu > :first-child > :first-child > :nth-child(${n})`;
const renderSetTitle = async (id, title, $tooltip = undefined) => {
const isCollapsed = await db.get(['collapsed', id], false),
const renderSetTitle = async (title, loadPromises = [], $tooltip = undefined) => {
const isCollapsed = await db.get(['collapsed', title], false),
$title = web.html`<p class="icon_sets--title"
${isCollapsed ? 'data-collapsed="true"' : ''}></p>`;
${isCollapsed ? 'data-collapsed="true"' : ''}></p>`,
$spinner = web.html`<span class="icon_sets--spinner">
<img src="/images/loading-spinner.4dc19970.svg" />
</span>`;
web.render(
$title,
$triangleSvg.cloneNode(true),
web.html`<span>${web.escape(title)}</span>`
web.html`<span>${title}</span>`,
$spinner
);
$title.addEventListener('click', () => {
const newState = $title.dataset.collapsed !== 'true';
db.set(['collapsed', id], newState);
db.set(['collapsed', title], newState);
$title.dataset.collapsed = newState;
});
if ($tooltip) {
const $infoSvg = web.html`${await components.feather('info', { class: 'info' })}`;
components.setTooltip($infoSvg, $tooltip);
web.render($title, $infoSvg);
}
// hide spinner after all icons finish loading
// doesn't need to be waited on by renderers
(async () => {
await Promise.all(loadPromises);
$spinner.remove();
if ($tooltip) {
const $infoSvg = web.html`${await components.feather('info', { class: 'info' })}`;
components.setTooltip($infoSvg, $tooltip);
web.render($title, $infoSvg);
}
})();
return $title;
};
@ -57,26 +81,23 @@ export default async function ({ web, components, notion }, db) {
),
// sets
$setsList = web.html`<div class="icon_sets--list"></div>`,
$recentUploadsTitle = await renderSetTitle(
'recent_uploads',
'Recent',
web.html`<p><b>Click</b> to reuse an icon <br><b>Shift-click</b> to remove it</p>`
),
$recentUploads = web.html`<div class="icon_sets--set"></div>`,
// container
$iconsScroller = web.render(
web.html`<div class="icon_sets--scroller" style="display:none"></div>`,
$iconsView = web.render(
web.html`<div class="icon_sets--view" style="display:none"></div>`,
web.render(
web.html`<div class="icon_sets--actions"></div>`,
$iconsLinkInput,
$iconsLinkSubmit,
web.render(
web.html`<div class="notion-focusable-within" style="display:flex;border-radius:3px;"></div>`,
$iconsLinkInput,
$iconsLinkSubmit
),
$iconsUploadSubmit
),
web.render($setsList, $recentUploadsTitle, $recentUploads)
web.render($setsList)
);
let $mediaMenu, $activeTabUnderline;
const insertIconsTab = async (event) => {
const insertIconsTab = async () => {
if (document.contains($mediaMenu)) return;
// prevent injection into file upload menus
@ -94,10 +115,11 @@ export default async function ({ web, components, notion }, db) {
$activeTabUnderline =
$emojiTab.children[1] || $uploadTab.children[1] || $linkTab.children[1];
$emojiTab.after($iconsTab);
$emojiScroller.after($iconsScroller);
$emojiScroller.after($iconsView);
const renderRecentUploads = () => {
web.empty($recentUploads);
const renderRecentUploads = async () => {
const $recentUploads = web.html`<div class="icon_sets--set"></div>`,
loadPromises = [];
for (let i = recentUploads.length - 1; i >= 0; i--) {
const { signed, url } = recentUploads[i],
$icon = web.html`<span class="icon_sets--icon">
@ -111,23 +133,117 @@ export default async function ({ web, components, notion }, db) {
$icon.remove();
} else setIcon(url);
});
loadPromises.push(
new Promise((res, rej) => {
$icon.firstElementChild.onload = res;
$icon.firstElementChild.onerror = res;
})
);
}
$recentUploads.style.height = `${$recentUploads.scrollHeight}px`;
},
renderSets = async () => {};
const displayIconsTab = (force = false) => {
const $recentUploadsTitle = await renderSetTitle(
'Recent',
loadPromises,
web.html`<p><b>Click</b> to reuse an icon <br><b>Shift-click</b> to remove it</p>`
);
web.render($setsList, $recentUploadsTitle, $recentUploads);
},
renderIconSet = async (iconData, enhancerSet = false) => {
try {
const $set = web.html`<div class="icon_sets--set"></div>`;
if (iconData.sourceUrl?.endsWith?.('/')) {
iconData.sourceUrl = iconData.sourceUrl.slice(0, -1);
}
const loadPromises = [];
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],
sprite = enhancerSet
? `style="
background-image: url(${enhancerIconsUrl}${iconData.source}/sprite.png);
background-position: 0 -${i * 24}px;
"`
: '',
$img = sprite
? web.html`<div class="icon_sets--sprite" ${sprite}></div>`
: web.html`<img src="${web.escape(iconUrl)}">`,
$icon = web.render(web.html`<span class="icon_sets--icon"></span>`, $img);
web.render($set, $icon);
$icon.addEventListener('click', (event) => {
if (!event.shiftKey) setIcon(iconUrl);
});
if (!sprite) {
loadPromises.push(
new Promise((res, rej) => {
$img.onload = res;
$img.onerror = res;
})
);
}
}
const author = iconData.author
? iconData.authorUrl
? web.raw`by <a target="_blank" href="${web.escape(iconData.authorUrl)}">
${iconData.author}
</a>`
: web.raw`by ${web.escape(iconData.author)}`
: '',
$title = await renderSetTitle(
`${web.escape(iconData.name)} ${author}`,
loadPromises
);
web.render($setsList, $title, $set);
} catch (err) {
console.log(err);
web.render(
$setsList,
web.html`<div class="icon_sets--error">
Invalid set: ${web.escape(iconData?.name || 'Unknown')}
</div>`
);
}
},
renderCustomIconSets = async () => {
if (customIconSets.length) {
web.render($setsList, web.html`<div class="icon_sets--divider"></div>`);
}
await Promise.all(customIconSets.map((set) => renderIconSet(set)));
},
renderEnhancerIconSets = async () => {
if (enhancerIconSets.length) {
web.render($setsList, web.html`<div class="icon_sets--divider"></div>`);
}
await Promise.all(
enhancerIconSets.map((set) => {
set.sourceUrl = set.sourceUrl || enhancerIconsUrl + set.source;
return renderIconSet(set, true);
})
);
};
const displayIconsTab = async (force = false) => {
if ($iconsTab.contains($activeTabUnderline) && !force) return;
web.render($iconsTab, $activeTabUnderline);
$iconsScroller.style.display = '';
$iconsView.style.display = '';
$emojiScroller.style.display = 'none';
$emojiFilter.style.display = 'none';
renderRecentUploads();
web.empty($setsList);
await renderRecentUploads();
await renderCustomIconSets();
await renderEnhancerIconSets();
$iconsView.querySelectorAll('.icon_sets--set').forEach(($set) => {
$set.style.height = `${$set.scrollHeight}px`;
});
},
displayEmojiTab = (force = false) => {
if ($emojiTab.contains($activeTabUnderline) && !force) return;
web.render($emojiTab, $activeTabUnderline);
$iconsScroller.style.display = 'none';
$iconsView.style.display = 'none';
$emojiScroller.style.display = '';
$emojiFilter.style.display = '';
};

View File

@ -1,3 +0,0 @@
<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>

Before

Width:  |  Height:  |  Size: 251 B

View File

@ -1,3 +0,0 @@
<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>

Before

Width:  |  Height:  |  Size: 234 B

View File

@ -1,3 +0,0 @@
<svg xmlns="http://www.w3.org/2000/svg" width="17" height="17" viewBox="0 0 17 17">
<path d="M6.78027 13.6729C8.24805 13.6729 9.60156 13.1982 10.709 12.4072L14.875 16.5732C15.0684 16.7666 15.3232 16.8633 15.5957 16.8633C16.167 16.8633 16.5713 16.4238 16.5713 15.8613C16.5713 15.5977 16.4834 15.3516 16.29 15.1582L12.1504 11.0098C13.0205 9.86719 13.5391 8.45215 13.5391 6.91406C13.5391 3.19629 10.498 0.155273 6.78027 0.155273C3.0625 0.155273 0.0214844 3.19629 0.0214844 6.91406C0.0214844 10.6318 3.0625 13.6729 6.78027 13.6729ZM6.78027 12.2139C3.87988 12.2139 1.48047 9.81445 1.48047 6.91406C1.48047 4.01367 3.87988 1.61426 6.78027 1.61426C9.68066 1.61426 12.0801 4.01367 12.0801 6.91406C12.0801 9.81445 9.68066 12.2139 6.78027 12.2139Z" />
</svg>

Before

Width:  |  Height:  |  Size: 749 B

View File

@ -1 +0,0 @@
<svg viewBox="0 0 100 100" class="triangle"><polygon points="5.9,88.2 50,11.8 94.1,88.2" /></svg>

Before

Width:  |  Height:  |  Size: 97 B

View File

@ -1,185 +0,0 @@
// source => icon data
const enhancerIconSets = new Map();
getAsync(notionIconsUrl + 'icons.json', (iconsData) => {
const data = JSON.parse(iconsData);
(data.icons || data).forEach((set) => {
enhancerIconSets.set(set.source, set);
});
});
// array
let customIconSets;
if (store().json) {
const customData = JSON.parse(fs.readFileSync(store().json));
customIconSets = customData.icons || customData;
}
// convert icons data into renderable
function loadIconSets() {
const iconSets = new DocumentFragment();
if (customIconSets) {
customIconSets.forEach((i) => {
iconSets.appendChild(renderIconSet(i));
});
// divider
iconSets.appendChild(createElement('<div class="notion-icons--divider"></div>'));
}
if (enhancerIconSets.size > 0) {
enhancerIconSets.forEach((i, source) => {
// ignore removed icon sets
if (store().removedSets?.includes(source)) return;
i.sourceUrl = i.sourceUrl || notionIconsUrl + source;
iconSets.appendChild(renderIconSet(i, true));
});
}
return iconSets;
}
// returns icon set element
function renderIconSet(iconData, enhancerSet = false) {
const iconSet = createElement('<div class="notion-icons--icon-set"></div>');
try {
const author = iconData.author
? iconData.authorUrl
? ` by <a target="_blank" href="${iconData.authorUrl}">${iconData.author}</a>`
: ` by <span>${iconData.author}</span>`
: '';
const toggle = createElement(`
<div class="notion-icons--toggle">
${menuIcons.triangle}
<div class="notion-icons--author">${iconData.name}${author}</div>
<div class="notion-icons--actions">
<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.append(toggle, 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>`);
icon.innerHTML = enhancerSet
? // load sprite sheet
`<div style="background-image: url(${notionIconsUrl}${
iconData.source
}/sprite.png); background-position: 0 -${i * 32}px;"></div>`
: `<img src="${iconUrl}" />`;
// add filters to filterMap
const filters = [];
if (iconData.filter) {
if (iconData.filter === 'source') {
const filename = iconUrl.match(/.*\/(.+?)\./);
if (filename?.length > 1) {
filters.push(...filename[1].split(/[ \-_]/));
}
} else if (Array.isArray(iconData.filter)) {
filters.push(...iconData.filter[i]);
}
icon.setAttribute('filter', filters.join(' '));
}
// add set name and author to filters
filters.push(...iconData.name.toLowerCase().split(' '));
if (iconData.author) filters.push(...iconData.author.toLowerCase().split(' '));
filterMap.set(icon, filters);
// make sure icons load
if (!enhancerSet) {
promiseArray.push(
new Promise((resolve, reject) => {
icon.firstChild.onload = resolve;
icon.firstChild.onerror = () => {
reject();
icon.classList.add('error');
icon.innerHTML = '!';
};
})
);
}
garbageCollector.push(icon);
icon.addEventListener('click', () => setPageIcon(iconUrl));
iconSetBody.appendChild(icon);
}
// hide spinner after all icons finish loading
(async () => {
const spinner = toggle.querySelector('.notion-icons--spinner'),
loadPromise = Promise.all(promiseArray);
loadPromise.then(
() => spinner.remove(),
() => {
iconSet.classList.add('alert');
spinner.remove();
}
);
})();
// add remove icon set button
if (enhancerSet) {
const removeButton = createElement(
`<div class="notion-icons--remove-button">${menuIcons.remove}</div>`
);
removeButton.addEventListener('click', (e) => {
e.stopPropagation();
removeIconSet(iconData);
});
iconSet.querySelector('.notion-icons--actions').appendChild(removeButton);
}
// set up toggle
toggle.addEventListener('click', (e) => {
if (e.target.nodeName === 'A') return;
toggleIconSet(iconSet);
});
// hide by default?
if (store().hide) requestAnimationFrame(() => toggleIconSet(iconSet));
// tooltip
let timeout;
iconSetBody.addEventListener('mouseover', (e) => {
const el = e.target;
if (!el.hasAttribute('filter')) return;
document.querySelector('.notion-icons--tooltip')?.remove();
timeout = setTimeout(() => {
renderTooltip(el, el.getAttribute('filter'));
}, 300);
});
iconSetBody.addEventListener('mouseout', (e) => {
const el = e.target;
if (!el.hasAttribute('filter')) return;
document.querySelector('.notion-icons--tooltip')?.remove();
clearTimeout(timeout);
});
} catch (err) {
iconSet.classList.add('error');
iconSet.innerHTML = `Invalid Icon Set: ${iconData.name}`;
}
return iconSet;
}