extension: icon sets part 1 (save/reuse recent uploads)

This commit is contained in:
dragonwocky 2021-10-30 19:14:22 +11:00
parent 6a84fbec91
commit 96988d67e0
12 changed files with 1023 additions and 6 deletions

411
repo/icon-sets/app.css Normal file
View File

@ -0,0 +1,411 @@
/*
* 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);
}
}

179
repo/icon-sets/client.css Normal file
View File

@ -0,0 +1,179 @@
/*
* notion-enhancer: icon sets
* (c) 2019 jayhxmo (https://jaymo.io/)
* (c) 2020 CloudHill <rl.cloudhill@gmail.com> (https://github.com/CloudHill)
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
.icon_sets--tab_button {
position: relative;
padding-top: 6px;
padding-bottom: 6px;
flex-shrink: 0;
}
.icon_sets--tab_button > .notion-focusable {
user-select: none;
transition: background 20ms ease-in 0s;
cursor: pointer;
display: inline-flex;
align-items: center;
height: 28px;
border-radius: 3px;
font-size: 14px;
line-height: 1.2;
padding-left: 8px;
padding-right: 8px;
color: var(--theme--text);
}
.icon_sets--tab_button:hover > .notion-focusable {
background: var(--theme--ui_interactive-hover);
}
.icon_sets--tab_button:active > .notion-focusable {
background: var(--theme--ui_interactive-active);
}
.icon_sets--scroller {
padding: 0;
overflow: hidden;
display: flex;
flex-direction: column;
flex-grow: 1;
z-index: 1;
}
.icon_sets--actions {
display: flex;
padding: 10px 14px;
}
.icon_sets--link_input {
flex-grow: 1;
font-size: 14px;
line-height: 20px;
padding: 4px 6px;
border-top-left-radius: 3px;
border-bottom-left-radius: 3px;
box-shadow: var(--theme--ui_shadow) 0px 0px 0px 1px inset;
background: var(--theme--ui_input);
cursor: text;
height: 28px;
}
.icon_sets--link_input > input {
font-size: inherit;
line-height: inherit;
border: none;
background: none;
width: 100%;
display: block;
resize: none;
padding: 0px;
}
.icon_sets--link_submit {
border-top-left-radius: 0;
border-bottom-left-radius: 0;
border-top-right-radius: 3px;
border-bottom-right-radius: 3px;
}
.icon_sets--upload,
.icon_sets--link_submit {
user-select: none;
transition: background 20ms ease-in 0s;
cursor: pointer;
border: none;
background: var(--theme--accent_blue);
color: var(--theme--accent_blue-text);
line-height: 1.2;
padding: 6px 8px;
height: 28px;
font-size: 14px;
font-weight: 500;
}
.icon_sets--upload:hover,
.icon_sets--link_submit:hover {
background: var(--theme--accent_blue-hover);
}
.icon_sets--upload:active,
.icon_sets--link_submit:active {
background: var(--theme--accent_blue-active);
}
.icon_sets--upload {
margin-left: 0.5em;
border-radius: 3px;
}
.icon_sets--list {
height: 100%;
word-break: break-all;
overflow: hidden auto;
padding: 0 14px 10px 14px;
}
.icon_sets--title {
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:hover {
background: var(--theme--ui_interactive-hover);
}
.icon_sets--title:active {
background: var(--theme--ui_interactive-active);
}
.icon_sets--title .info {
height: 1em;
margin-left: 0.5em;
}
.icon_sets--title .triangle {
height: 1em;
width: 0.9em;
margin: 0 0.5em 0 0.25em;
transition: transform 200ms ease-out 0s;
transform: rotateZ(180deg);
}
.icon_sets--title[data-collapsed='true'] .triangle {
transform: rotateZ(90deg);
}
.icon_sets--title[data-collapsed='true'] + .icon_sets--set {
height: 0 !important;
}
.icon_sets--set {
display: flex;
flex-wrap: wrap;
overflow: hidden;
transition: height 200ms ease-out 0s;
}
.icon_sets--icon {
user-select: none;
transition: background 20ms ease-in 0s;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
border-radius: 3px;
width: 32px;
height: 32px;
font-size: 24px;
}
.icon_sets--icon:hover {
background: var(--theme--ui_interactive-hover);
}
.icon_sets--icon:active {
background: var(--theme--ui_interactive-active);
}
.icon_sets--icon > img {
max-width: 24px;
max-height: 24px;
}

182
repo/icon-sets/client.mjs Normal file
View File

@ -0,0 +1,182 @@
/*
* notion-enhancer: icon sets
* (c) 2019 jayhxmo (https://jaymo.io/)
* (c) 2020 CloudHill <rl.cloudhill@gmail.com> (https://github.com/CloudHill)
* (c) 2021 dragonwocky <thedragonring.bod@gmail.com> (https://dragonwocky.me/)
* (https://notion-enhancer.github.io/) under the MIT license
*/
export default async function ({ web, 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 mediaMenuSelector = '.notion-media-menu',
mediaScrollerSelector = '.notion-media-menu > .notion-scroller',
mediaFilterSelector = '.notion-media-menu > :first-child > :last-child',
mediaLinkInputSelector = '.notion-focusable-within > input[type=url]',
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),
$title = web.html`<p class="icon_sets--title"
${isCollapsed ? 'data-collapsed="true"' : ''}></p>`;
web.render(
$title,
$triangleSvg.cloneNode(true),
web.html`<span>${web.escape(title)}</span>`
);
$title.addEventListener('click', () => {
const newState = $title.dataset.collapsed !== 'true';
db.set(['collapsed', id], 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);
}
return $title;
};
const $iconsTab = web.html`<div class="icon_sets--tab_button">
<div class="notion-focusable" role="button" tabindex="0">Icons</div>
</div>`,
// actions
$iconsLinkInput = web.html`<div class="icon_sets--link_input">
<input placeholder="Paste an image link…" type="url">
</div>`,
$iconsLinkSubmit = web.html`<button class="icon_sets--link_submit">Submit</button>`,
$iconsUploadFile = web.html`<input type="file" accept="image/*" style="display:none">`,
$iconsUploadSubmit = web.render(
web.html`<button class="icon_sets--upload"></button>`,
'Upload an image',
$iconsUploadFile
),
// 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>`,
web.render(
web.html`<div class="icon_sets--actions"></div>`,
$iconsLinkInput,
$iconsLinkSubmit,
$iconsUploadSubmit
),
web.render($setsList, $recentUploadsTitle, $recentUploads)
);
let $mediaMenu, $activeTabUnderline;
const insertIconsTab = async (event) => {
if (document.contains($mediaMenu)) return;
// prevent injection into file upload menus
$mediaMenu = document.querySelector(mediaMenuSelector);
if (!$mediaMenu || !$mediaMenu.textContent.includes('Emoji')) return;
const $emojiTab = document.querySelector(tabBtnSelector(1)),
$emojiScroller = document.querySelector(mediaScrollerSelector),
$emojiFilter = document.querySelector(mediaFilterSelector),
$uploadTab = document.querySelector(tabBtnSelector(2)),
$linkTab = document.querySelector(tabBtnSelector(3));
$uploadTab.style.display = 'none';
$linkTab.style.display = 'none';
if ($activeTabUnderline) $activeTabUnderline.remove();
$activeTabUnderline =
$emojiTab.children[1] || $uploadTab.children[1] || $linkTab.children[1];
$emojiTab.after($iconsTab);
$emojiScroller.after($iconsScroller);
const renderRecentUploads = () => {
web.empty($recentUploads);
for (let i = recentUploads.length - 1; i >= 0; i--) {
const { signed, url } = recentUploads[i],
$icon = web.html`<span class="icon_sets--icon">
<img src="${web.escape(signed)}">
</span>`;
web.render($recentUploads, $icon);
$icon.addEventListener('click', (event) => {
if (event.shiftKey) {
recentUploads.splice(i, 1);
db.set(['recent_uploads'], recentUploads);
$icon.remove();
} else setIcon(url);
});
}
$recentUploads.style.height = `${$recentUploads.scrollHeight}px`;
},
renderSets = async () => {};
const displayIconsTab = (force = false) => {
if ($iconsTab.contains($activeTabUnderline) && !force) return;
web.render($iconsTab, $activeTabUnderline);
$iconsScroller.style.display = '';
$emojiScroller.style.display = 'none';
$emojiFilter.style.display = 'none';
renderRecentUploads();
},
displayEmojiTab = (force = false) => {
if ($emojiTab.contains($activeTabUnderline) && !force) return;
web.render($emojiTab, $activeTabUnderline);
$iconsScroller.style.display = 'none';
$emojiScroller.style.display = '';
$emojiFilter.style.display = '';
};
// use onclick instead of eventlistener to override prev
$iconsTab.onclick = displayIconsTab;
$emojiTab.onclick = displayEmojiTab;
// otherwise both may be visible on reopen
displayEmojiTab(true);
async function setIcon(iconUrl) {
// without this react gets upset
displayEmojiTab();
$linkTab.firstChild.click();
await new Promise(requestAnimationFrame);
// call native setter, imitate human input
const $notionLinkInput = $mediaMenu.querySelector(mediaLinkInputSelector),
proto = Object.getOwnPropertyDescriptor(window.HTMLInputElement.prototype, 'value');
proto.set.call($notionLinkInput, iconUrl);
const inputEvent = new Event('input', { bubbles: true }),
enterKeydownEvent = new KeyboardEvent('keydown', {
bubbles: true,
cancelable: true,
keyCode: 13,
});
$notionLinkInput.dispatchEvent(inputEvent);
$notionLinkInput.dispatchEvent(enterKeydownEvent);
}
const submitLinkIcon = () => {
const url = $iconsLinkInput.firstElementChild.value;
setIcon(url);
recentUploads.push({ signed: notion.sign(url, notion.getPageID()), url: url });
db.set(['recent_uploads'], recentUploads);
};
$iconsLinkInput.onkeyup = (event) => {
if (event.code === 13) submitLinkIcon();
};
$iconsLinkSubmit.onclick = submitLinkIcon;
// upload file to aws, then submit link
$iconsUploadSubmit.onclick = $iconsUploadFile.click;
$iconsUploadFile.onchange = async (event) => {
const file = event.target.files[0],
url = await notion.upload(file);
setIcon(url);
recentUploads.push({ signed: notion.sign(url, notion.getPageID()), url: url });
db.set(['recent_uploads'], recentUploads);
};
};
web.addDocumentObserver(insertIconsTab, [mediaMenuSelector]);
}

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

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

After

Width:  |  Height:  |  Size: 749 B

View File

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

After

Width:  |  Height:  |  Size: 97 B

185
repo/icon-sets/mod.js Normal file
View File

@ -0,0 +1,185 @@
// 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;
}

35
repo/icon-sets/mod.json Normal file
View File

@ -0,0 +1,35 @@
{
"name": "icon sets",
"id": "2d1f4809-9581-40dd-9bf3-4239db406483",
"version": "0.4.0",
"description": "upload, save and reuse custom icons directly from the icon picker.",
"tags": ["integration", "customisation"],
"authors": [
{
"name": "dragonwocky",
"email": "thedragonring.bod@gmail.com",
"homepage": "https://dragonwocky.me/",
"avatar": "https://dragonwocky.me/avatar.jpg"
}
],
"js": {
"client": ["client.mjs"]
},
"css": {
"client": ["client.css"]
},
"options": [
{
"type": "toggle",
"key": "default_sets",
"label": "load default icon sets from github",
"value": true
},
{
"type": "file",
"key": "json",
"label": "custom icon sets (.json)",
"extensions": ["json"]
}
]
}

View File

@ -283,7 +283,7 @@ const $modLists = {},
$search,
web.html`${await components.feather('search', { class: 'input-icon' })}`
),
message ? web.html`<p class="main-message">${web.escape(message)}</p>` : '',
message ? web.render(web.html`<p class="main-message"></p>`, message) : '',
$list
);
}
@ -322,7 +322,8 @@ $notionNavItem.addEventListener('click', env.focusNotion);
const $coreNavItem = web.html`<a href="?view=core" class="nav-item">core</a>`,
$extensionsNavItem = web.html`<a href="?view=extensions" class="nav-item">extensions</a>`,
$themesNavItem = web.html`<a href="?view=themes" class="nav-item">themes</a>`;
$themesNavItem = web.html`<a href="?view=themes" class="nav-item">themes</a>`,
$integrationsNavItem = web.html`<a href="?view=integrations" class="nav-item">integrations</a>`;
web.render(
document.body,
@ -336,6 +337,7 @@ web.render(
$coreNavItem,
$extensionsNavItem,
$themesNavItem,
$integrationsNavItem,
web.html`<a href="https://notion-enhancer.github.io" class="nav-item">docs</a>`,
web.html`<a href="https://discord.gg/sFWPXtA" class="nav-item">community</a>`
),
@ -367,10 +369,9 @@ router.addView('core', async () => {
await generators.modList(
'extension',
`Extensions modify and extend the functionality
or layout of the Notion client. They don't interfere
with Notion's data structures, so they can be safely
enabled or disabled at any time.`
`Extensions build on the functionality and layout of
the Notion client, modifying and interacting with
existing interfaces.`
);
router.addView('extensions', async () => {
web.empty($main);
@ -391,6 +392,18 @@ router.addView('themes', async () => {
return web.render($main, await generators.modList('theme'));
});
await generators.modList(
'integration',
web.html`<span class="danger">Integrations are extensions that use an unofficial API
to access and modify content. They are used just like
normal extensions, but may be more dangerous to use.</span>`
);
router.addView('integrations', async () => {
web.empty($main);
selectNavItem($integrationsNavItem);
return web.render($main, await generators.modList('integration'));
});
router.setDefaultView('extensions');
router.addQueryListener('id', openSidebarMenu);

View File

@ -86,6 +86,7 @@ const customClasses = {
'file-latest': apply`block w-full text-left text-foreground-secondary text-xs mt-2 hover:line-through cursor-pointer`,
'search-container': apply`block mx-2.5 my-2.5 relative`,
'search': apply`input pr-12`,
'danger': apply`bg-red-paragraph text-red-paragraph-text`,
};
setup({

View File

@ -24,6 +24,7 @@
"right-to-left",
"simpler-databases",
"emoji-sets",
"icon-sets",
"bypass-preview",
"topbar-icons",
"word-counter",