diff --git a/mods/notion-icons/app.css b/mods/notion-icons/app.css index 43a2dc9..0b49a68 100644 --- a/mods/notion-icons/app.css +++ b/mods/notion-icons/app.css @@ -6,7 +6,9 @@ * under the MIT license */ -.hide-active-bar > :nth-child(2) { +/* tab */ + +[hide-active-bar] > :nth-child(2) { display: none; } @@ -15,6 +17,22 @@ 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, @@ -24,6 +42,8 @@ box-shadow: 0 0 0 0.5px var(--theme--interactive_hover-border) !important; } +/* container */ + #notion-icons { position: absolute; top: 1px; @@ -36,17 +56,82 @@ align-items: stretch; background: var(--theme--card); border-radius: 3px; - padding: 8px 12px; - overflow-x: hidden; - overflow-y: scroll; + 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--select_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; - text-transform: uppercase; letter-spacing: 1px; font-weight: 600; border-radius: 2px; @@ -66,33 +151,19 @@ float: right; } +/* icon set header/toggle */ + .notion-icons--toggle { display: flex; align-items: center; - user-select: none; - cursor: pointer; 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--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--toggle a { - color: var(--theme-text); - transition: color 20ms ease-in; -} -.notion-icons--toggle a:hover { - color: var(--theme--primary); -} - .notion-icons--icon-set.alert .notion-icons--toggle { color: var(--theme--line_yellow-text); background: var(--theme--line_yellow); @@ -104,6 +175,32 @@ background: var(--theme--select_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; @@ -114,34 +211,49 @@ transition: height 200ms ease-in, opacity 200ms ease-in; } -.hidden-set { +/* hidden icon set */ + +.notion-icons--icon-set[hidden-set] { padding-bottom: 0; } -.hidden-set .notion-icons--toggle { +.notion-icons--icon-set[hidden-set] .notion-icons--toggle { margin-bottom: 0; } -.hidden-set .triangle { +.notion-icons--icon-set[hidden-set] .triangle { transform: rotateZ(90deg); } -.hidden-set .notion-icons--body { +.notion-icons--icon-set[hidden-set] .notion-icons--body { opacity: 0; } +/* icons */ + .notion-icons--icon { - cursor: pointer; - user-select: none; - transition: background 20ms ease-in; - display: flex; - align-items: center; - justify-content: center; - border-radius: 3px; 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 { @@ -154,8 +266,53 @@ background: var(--theme--select_yellow); } +/* tooltip */ -.notion-icons--extra { +.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; @@ -163,6 +320,8 @@ align-items: center; } +/* spinner */ + .notion-icons--spinner { width: 12px; height: 12px; @@ -173,6 +332,8 @@ animation: rotation 1.3s infinite linear; } +/* remove button */ + .notion-icons--remove-button { display: flex; justify-content: center; @@ -201,15 +362,11 @@ .notion-icons--remove-button svg { width: 100%; height: 100%; - fill: inherit; + fill: var(--theme--text_ui_info); z-index: 1; } -.notion-icons--restore-button svg { - width: 16px; - height: 16px; - fill: inherit; -} +/* restore icon sets modal */ .notion-icons--overlay-container { position: fixed; @@ -242,6 +399,8 @@ transition: background 0.4s ease; } +/* animation */ + @keyframes rotation { from { transform: rotate(0deg); diff --git a/mods/notion-icons/icons/search.svg b/mods/notion-icons/icons/search.svg new file mode 100644 index 0000000..e8c0216 --- /dev/null +++ b/mods/notion-icons/icons/search.svg @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/mods/notion-icons/icons/triangle.svg b/mods/notion-icons/icons/triangle.svg new file mode 100644 index 0000000..8615536 --- /dev/null +++ b/mods/notion-icons/icons/triangle.svg @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/mods/notion-icons/mod.js b/mods/notion-icons/mod.js index 8c1dfc7..2b77dfa 100644 --- a/mods/notion-icons/mod.js +++ b/mods/notion-icons/mod.js @@ -1,6 +1,6 @@ /* * notion-icons - * (c) 2019 jayhxmo (https://jaymo.io/) + * (c) 2020 jayhxmo (https://jaymo.io/) * (c) 2020 dragonwocky (https://dragonwocky.me/) * (c) 2020 CloudHill * under the MIT license @@ -10,7 +10,8 @@ const { createElement } = require('../../pkg/helpers.js'), fs = require('fs-extra'), - path = require('path'); + path = require('path'), + notionIconsUrl = 'https://raw.githubusercontent.com/notion-enhancer/icons/main/'; module.exports = { id: '2d1f4809-9581-40dd-9bf3-4239db406483', @@ -18,7 +19,7 @@ module.exports = { name: 'notion icons', desc: 'use custom icon sets directly in notion.', - version: '1.0.0', + version: '1.2.0', author: 'jayhxmo', options: [ { @@ -36,9 +37,9 @@ module.exports = { ], hacks: { 'renderer/preload.js'(store, __exports) { - let garbageCollector = []; - const iconsUrl = 'https://raw.githubusercontent.com/notion-enhancer/icons/main/'; - + let garbageCollector = [], + filterMap = new WeakMap(); + function getAsync(urlString, callback) { let httpReq = new XMLHttpRequest(); httpReq.onreadystatechange = function() { @@ -48,120 +49,358 @@ module.exports = { httpReq.send(null); } - let modalIcons; + const menuIcons = {}; (async () => { - modalIcons = { - remove: await fs.readFile( path.resolve(__dirname, 'icons/remove.svg') ), - restore: await fs.readFile( path.resolve(__dirname, 'icons/restore.svg') ), - } + menuIcons.triangle = await fs.readFile( path.resolve(__dirname, 'icons/triangle.svg') ); + menuIcons.remove = await fs.readFile( path.resolve(__dirname, 'icons/remove.svg' ) ); + menuIcons.restore = await fs.readFile( path.resolve(__dirname, 'icons/restore.svg' ) ); + menuIcons.search = await fs.readFile( path.resolve(__dirname, 'icons/search.svg' ) ); })(); - // Retrieve icons data - let notionIconsData; - getAsync(iconsUrl + 'icons.json', iconsData => { - notionIconsData = JSON.parse(iconsData); + // 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); + }) }); - // Retrieve custom icons data - let customIconsData; + // array + let customIconSets; if (store().json) { - customIconsData = JSON.parse( + const customData = JSON.parse( fs.readFileSync(store().json) ) + customIconSets = customData.icons || customData; } - function getTab(n, button = false) { - return document.querySelector( - `.notion-media-menu > :first-child > :first-child > :nth-child(${n}) ${button ? 'div' : ''}` - ); - } - - function isCurrentTab(n) { - return getTab(n).childNodes.length > 1; - } - - // Submits the icon's url as an image link - function setPageIcon(iconUrl) { - const input = document.querySelector('input[type=url]'); - - const nativeInputValueSetter = Object.getOwnPropertyDescriptor( - window.HTMLInputElement.prototype, 'value' - ).set; - nativeInputValueSetter.call(input, iconUrl); - - input.dispatchEvent( - new Event('input', { bubbles: true }) - ); - - input.dispatchEvent( - new KeyboardEvent('keydown', { bubbles: true, cancelable: true, keyCode: 13 }) - ); - - removeIcons(); - } + // notion icons overlay function addIconsTab() { - // Prevent icons tab duplication - if (getTab(5)) { - removeIcons(); - return; - } - // Change 'Upload an image' to 'Upload' + // prevent icons tab duplication + if (getTab(5)) + return removeIconsOverlay(); + + // change 'Upload an image' to 'Upload' getTab(2, true).innerText = 'Upload'; - // Initialize icons tab + // initialize icons tab const iconsTab = getTab(3).cloneNode(true); iconsTab.className = 'notion-icons--tab'; iconsTab.firstChild.innerText = 'Icons'; iconsTab.firstChild.addEventListener('click', renderIconsOverlay); - // Insert icons tab + // insert icons tab const tabStrip = getTab(1).parentElement; tabStrip.insertBefore(iconsTab, tabStrip.lastChild); - // Remove the icons overlay when clicking... - const closeTriggers = [ - // The fog layer + initCloseTriggers(); + } + + function renderIconsOverlay() { + if (!isCurrentTab(4)) { + // switch to 3rd tab so that the link can be input in the underlay + if (!isCurrentTab(3)) getTab(3, true).click(); + + if ( + store().removedSets?.length > 0 && + enhancerIconSets.size > 0 + ) + addRestoreButton(); + + // set active bar on icons tab + const iconsTab = getTab(4), + activeBar = createElement( + `
` + ); + iconsTab.style.position = 'relative'; + iconsTab.appendChild(activeBar); + getTab(3).setAttribute('hide-active-bar', ''); + + // create icons overlay + const notionIcons = createElement( + '
' + ); + + // render search bar + const search = createElement(` + + `), + searchInput = search.lastElementChild; + + searchInput.addEventListener('input', () => { + filterIcons(searchInput.value); + }); + + // render scroller and icon sets + const scroller = createElement(` +
+ `); + scroller.appendChild( loadIconSets() ); + + notionIcons.append(search, scroller); + + // insert icons overlay + document.querySelector('.notion-media-menu > .notion-scroller') + .appendChild(notionIcons); + + // focus on search bar + requestAnimationFrame(() => { + searchInput.focus(); + }); + } + } + + // convert icons data into renderable + function loadIconSets() { + const iconSets = new DocumentFragment(); + + if (customIconSets) { + customIconSets.forEach(i => { + iconSets.appendChild( renderIconSet(i) ); + }); + + // divider + iconSets.appendChild( + createElement('
') + ); + } + + 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( + '
' + ); + + try { + const author = iconData.author + ? iconData.authorUrl + ? ` by ${iconData.author}` + : ` by ${iconData.author}` + : ''; + + const toggle = createElement(` +
+ ${menuIcons.triangle} +
${iconData.name}${author}
+
+
+ +
+
+
+ `); + + const iconSetBody = createElement( + '
' + ); + + 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(`
`); + icon.innerHTML = enhancerSet + // load sprite sheet + ? `
` + : ``; + + // 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( + `
${menuIcons.remove}
` + ); + 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; + } + + function removeIconsOverlay() { + const elements = [ + document.getElementById('notion-icons'), + document.getElementById('notion-icons--active-bar'), + document.querySelector('.notion-icons--restore-button'), + document.querySelector('.notion-icons--tooltip'), + ] + elements.forEach(el => { + if (el) el.remove(); + }) + + getTab(4).style.position = ''; + + if (getTab(3)) + getTab(3).removeAttribute('hide-active-bar'); + + if ( + document.querySelector('.notion-icons--overlay-container') + ) closeRestoreOverlay(); + + if (garbageCollector.length) { + for (let i = 0; i < garbageCollector.length; i++) { + garbageCollector[i] = null; + } + garbageCollector = []; + } + } + + function initCloseTriggers() { + // remove the icons overlay when clicking... + const triggers = [ + // the fog layer document.querySelector('.notion-overlay-container [style*="width: 100vw; height: 100vh;"]'), - // The first three buttons - ...Array.from( Array(3), (e, i) => getTab(i + 1, true) ), - // The remove button - getTab(5).lastChild, + // the first three buttons + ...[1, 2, 3].map( n => getTab(n, true) ), + // the remove button + (getTab(5) || getTab(4)).lastElementChild, ]; - closeTriggers.forEach(trigger => { - trigger.addEventListener('click', removeIcons); - garbageCollector.push(trigger); + triggers.forEach(t => { + t.addEventListener('click', removeIconsOverlay); + garbageCollector.push(t); }) - // Remove the icons overlay when pressing the Escape key + // remove the icons overlay when pressing the Escape key document.querySelector('.notion-media-menu') .addEventListener('keydown', e => { - if (e.keyCode === 27) removeIcons(); + if (e.keyCode === 27) removeIconsOverlay(); }); } - + // restore overlay + function addRestoreButton() { - const buttons = getTab(5) ? getTab(5) : getTab(4); - const restoreButton = buttons.lastChild.cloneNode(true); + const buttons = getTab(1).parentElement.lastElementChild; + + const restoreButton = buttons.lastElementChild.cloneNode(true); restoreButton.className = 'notion-icons--restore-button'; - restoreButton.innerHTML = modalIcons.restore; - buttons.prepend(restoreButton); + restoreButton.innerHTML = menuIcons.restore; restoreButton.addEventListener('click', renderRestoreOverlay); + + buttons.prepend(restoreButton); } function renderRestoreOverlay() { if (!store().removedSets) return; - - store().removedSets.sort((a, b) => { - const setA = a.name.toLowerCase(), - setB = b.name.toLowerCase(); - - if (setA < setB) return -1; - if (setA > setB) return 1; - return 0; - }); + store().removedSets.sort(); const overlayContainer = createElement(`
@@ -181,23 +420,25 @@ module.exports = {
`) + store().removedSets.forEach(source => { + restoreOverlay.appendChild( renderRestoreItem(source) ); + }) + overlayContainer.appendChild(div); div.firstElementChild.appendChild(restoreOverlay); - // Fade in + // fade in restoreOverlay.animate( [ {opacity: 0}, {opacity: 1} ], { duration: 200 } ); - - store().removedSets.forEach(iconData => { - const restoreItem = renderRestoreItem(iconData); - restoreOverlay.appendChild(restoreItem); - }) } - function renderRestoreItem(iconData) { - const iconUrl = `${iconData.sourceUrl}/${iconData.source}_${0}.${iconData.extension}`; + function renderRestoreItem(source) { + const iconData = enhancerIconSets.get(source); + const iconUrl = ` + ${iconData.sourceUrl || (notionIconsUrl + source)}/${source}_${0}.${iconData.extension} + `; const restoreItem = createElement(`
@@ -213,245 +454,161 @@ module.exports = { function closeRestoreOverlay() { const overlayContainer = document.querySelector('.notion-icons--overlay-container'); overlayContainer.removeEventListener('click', closeRestoreOverlay); - // Fade out + // fade out document.querySelector('.notion-icons--restore').animate( [ {opacity: 1}, {opacity: 0} ], { duration: 200 } ).onfinish = () => overlayContainer.remove(); } - function renderIconsOverlay() { - if (!isCurrentTab(4)) { - // Switch to 3rd tab so that the link can be inputed in the underlay - if (!isCurrentTab(3)) getTab(3, true).click(); + // icon set actions - if ( - store().removedSets && - store().removedSets.length > 0 - ) { - addRestoreButton(); - } + function toggleIconSet(iconSet, hide) { + const isHidden = iconSet.hasAttribute('hidden-set'); + if (hide == null) hide = !isHidden; - // Set active bar on icons tab - const iconsTab = getTab(4); - const activeBar = createElement( - `
` - ) - activeBar.style = 'border-bottom: 2px solid var(--theme--text); position: absolute; bottom: -1px; left: 8px; right: 8px;'; - iconsTab.appendChild(activeBar); - getTab(4).style.position = 'relative'; - getTab(3).className = 'hide-active-bar'; - - // Convert icons data into renderable - const iconSets = []; - - if (customIconsData && customIconsData.icons) { - customIconsData.icons.forEach(i => { - iconSets.push( renderIconSet(i) ); - }); - - // Divider - iconSets.push( - createElement( - '
' - ) - ) - } - - if (notionIconsData && notionIconsData.icons) { - notionIconsData.icons.forEach(i => { - 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; - iconSets.push( renderIconSet(i) ); - }); - } - - // Create icons overlay - const notionIcons = createElement( - '
' + const body = iconSet.lastChild; + if (hide && !isHidden) { + iconSet.setAttribute('hidden-set', ''); + body.style.height = body.offsetHeight + 'px'; + requestAnimationFrame( + () => body.style.height = 0 ); - iconSets.forEach( set => notionIcons.appendChild(set) ); - - // Insert icons overlay - document.querySelector('.notion-media-menu > .notion-scroller') - .appendChild(notionIcons); } - } + else if (!hide && isHidden) { + iconSet.removeAttribute('hidden-set'); + // get height + body.style.height = ''; + const height = body.offsetHeight; + body.style.height = 0; - function renderIconSet(iconData) { - const iconSet = createElement('
') - - try { - const authorText = iconData.author - ? iconData.authorUrl - ? ` by ${iconData.author}` - : ` by ${iconData.author}` - : ''; - - const iconSetToggle = createElement( - `
- -
${iconData.name}${authorText}
-
-
- -
-
-
` + requestAnimationFrame( + () => body.style.height = height + 'px' ); - - const iconSetBody = createElement( - '
' + setTimeout( + () => body.style.height = '', 200 ); - - 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(`
`); - if (iconData.enhancerIcons) { - // Load sprite sheet - icon.innerHTML = ` -
-
- `; - } else { - icon.innerHTML = ``; - // 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( - '
' - ); - 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(); + if (!store().removedSets.includes(iconData.source)) + store().removedSets.push(iconData.source); + removeIconsOverlay(); 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(); + store().removedSets = store().removedSets + .filter(source => source !== iconData.source); + removeIconsOverlay(); renderIconsOverlay(); } - function removeIcons() { - const notionIcons = document.getElementById('notion-icons'), - activeBar = document.getElementById('notion-icons--active-bar'), - restoreButton = document.querySelector('.notion-icons--restore-button'), - overlayContainer = document.querySelector('.notion-icons--overlay-container'); + // other actions - if (notionIcons) - notionIcons.remove(); + // submit the icon's url as an image link + function setPageIcon(iconUrl) { + const input = document.querySelector('.notion-media-menu input[type=url]'); - if (activeBar) { - activeBar.remove(); - getTab(4).style.position = ''; - } - if (getTab(3)) getTab(3).className = ''; + const nativeInputValueSetter = Object.getOwnPropertyDescriptor( + window.HTMLInputElement.prototype, 'value' + ).set; + nativeInputValueSetter.call(input, iconUrl); - if (restoreButton) - restoreButton.remove(); + input.dispatchEvent( + new Event('input', { bubbles: true }) + ); - if (overlayContainer) - closeRestoreOverlay(); + input.dispatchEvent( + new KeyboardEvent('keydown', { bubbles: true, cancelable: true, keyCode: 13 }) + ); - if (garbageCollector.length) { - for (let i = 0; i < garbageCollector.length; i++) { - garbageCollector[i] = null; + removeIconsOverlay(); + } + + function filterIcons(input) { + const iconSets = document.querySelectorAll('.notion-icons--icon-set'); + if (!iconSets) return; + + // show all sets and icons + if (!input) return iconSets.forEach(set => { + set.style.display = ''; + set.querySelectorAll('.notion-icons--icon') + .forEach(i => i.style.display = ''); + }); + // split input into an array + else input = input.toLowerCase().trim().split(' '); + + const findMatch = icon => { + const iconFilters = filterMap.get(icon).slice(); + + // match whole words for the first terms + if (input.length > 1) { + let index; + for (let i of input.slice(0, -1)) { + if ( + ( index = iconFilters.indexOf(i) ) >= 0 + ) { + iconFilters.splice(index, 1); + continue; + } + return false; + } } - garbageCollector = []; + + // match partially for the last term + for (let iconFilter of iconFilters) { + if (iconFilter.includes(input[input.length - 1])) { + return true; + }; + } + + return false; } + + iconSets.forEach(set => { + let found = false; + + set.querySelectorAll('.notion-icons--icon') + .forEach(i => { + // hide icon set + if (!filterMap.has(i)) return; + + if (findMatch(i)) { + i.style.display = ''; + found = true; + } else i.style.display = 'none'; + }); + + if (!found) set.style.display = 'none'; + else { + set.style.display = ''; + toggleIconSet(set, false); + } + }) + } + + function renderTooltip(el, text) { + const rect = el.getBoundingClientRect(); + const overlayContainer = document.querySelector('.notion-overlay-container') + + const tooltip = createElement(` +
+
+
+ `), tooltipText = createElement( + `
${text}
` + ); + + tooltip.firstElementChild.appendChild(tooltipText); + overlayContainer.appendChild(tooltip); + + // prevent tooltip from rendering outside the window + const left = (tooltipText.offsetWidth / 2) - (rect.width / 2) - rect.left + 4; + if (left > 0) tooltipText.style.left = left + 'px'; } document.addEventListener('readystatechange', () => { @@ -465,16 +622,11 @@ module.exports = { const observer = new MutationObserver((list, observer) => { for ( let { addedNodes } of list) { if ( - addedNodes[0] && - addedNodes[0].style && - document.querySelector('.notion-media-menu') + addedNodes[0]?.querySelector?.('.notion-media-menu') && + /^pointer-events: auto; position: relative; z-index: \d;$/ + .test(addedNodes[0].style.cssText) ) { - for (let i = 0; i <= 3; i++) { - if (addedNodes[0].style.cssText === `pointer-events: auto; position: relative; z-index: ${i};`) { - addIconsTab(); - return; - } - } + addIconsTab(); } } }); @@ -484,6 +636,18 @@ module.exports = { }); } }); + + // helpers + + function getTab(n, button = false) { + return document.querySelector( + `.notion-media-menu > :first-child > :first-child > :nth-child(${n}) ${button ? 'div' : ''}` + ); + } + + function isCurrentTab(n) { + return getTab(n).childNodes.length > 1; + } }, }, };