diff --git a/libs/blocks/table/table.css b/libs/blocks/table/table.css
index d2476a42ce..19eb5beed9 100644
--- a/libs/blocks/table/table.css
+++ b/libs/blocks/table/table.css
@@ -360,7 +360,6 @@
width: 15px;
height: 15px;
cursor: pointer;
- margin-inline: unset;
}
.table .section-head-title:hover .icon.expand {
diff --git a/libs/blocks/text/text.css b/libs/blocks/text/text.css
index f8cbefd365..6242f50132 100644
--- a/libs/blocks/text/text.css
+++ b/libs/blocks/text/text.css
@@ -100,7 +100,7 @@
position: relative;
}
-.text-block .icon-list-item .icon.node-index-first {
+.text-block .icon-list-item .icon.margin-right:not(.margin-left) { /* target first node only */
position: absolute;
inset: 0 100% auto auto;
}
@@ -122,6 +122,7 @@
.text-block .icon-area {
display: flex;
+ column-gap: var(--spacing-xs);
}
.text-block p.icon-area { /* NOT tags with icons in them */
@@ -217,6 +218,10 @@
max-width: unset;
}
+.text-block .icon-area.con-button {
+ column-gap: unset;
+}
+
.text-block .icon-area picture {
line-height: 0em;
height: inherit; /* Safari + FF bug fix */
diff --git a/libs/features/georoutingv2/georoutingv2.css b/libs/features/georoutingv2/georoutingv2.css
index 09df2cba8b..255ba3f301 100644
--- a/libs/features/georoutingv2/georoutingv2.css
+++ b/libs/features/georoutingv2/georoutingv2.css
@@ -83,7 +83,6 @@
}
.dialog-modal.locale-modal-v2 span.icon {
- display: inline;
vertical-align: middle;
}
diff --git a/libs/features/georoutingv2/georoutingv2.js b/libs/features/georoutingv2/georoutingv2.js
index 236e9b685a..7213696f63 100644
--- a/libs/features/georoutingv2/georoutingv2.js
+++ b/libs/features/georoutingv2/georoutingv2.js
@@ -194,7 +194,7 @@ function buildContent(currentPage, locale, geoData, locales) {
{ once: true },
);
img.src = `${config.miloLibs || config.codeRoot}/img/georouting/${flagFile}`;
- const span = createTag('span', { class: 'icon node-index-first' }, img);
+ const span = createTag('span', { class: 'icon margin-inline-end' }, img);
const mainAction = createTag('a', {
class: 'con-button blue button-l', lang, role: 'button', 'aria-haspopup': !!locales, 'aria-expanded': false, href: '#',
}, span);
diff --git a/libs/features/icons/icons.css b/libs/features/icons/icons.css
index 1b095e97dc..f1a8ce53f9 100644
--- a/libs/features/icons/icons.css
+++ b/libs/features/icons/icons.css
@@ -4,7 +4,7 @@
border-bottom: none;
}
-.milo-tooltip::before {
+.milo-tooltip::before {
content: attr(data-tooltip);
position: absolute;
top: 50%;
diff --git a/libs/features/icons/icons.js b/libs/features/icons/icons.js
index a39ff61e06..975c0625dc 100644
--- a/libs/features/icons/icons.js
+++ b/libs/features/icons/icons.js
@@ -1,9 +1,5 @@
-import { getFederatedContentRoot } from '../../utils/federated.js';
-import { loadLink, loadStyle } from '../../utils/utils.js';
-
let fetchedIcons;
let fetched = false;
-const federalIcons = {};
async function getSVGsfromFile(path) {
/* c8 ignore next */
@@ -26,7 +22,6 @@ async function getSVGsfromFile(path) {
return miloIcons;
}
-// TODO: remove after all consumers have stopped calling this method
// eslint-disable-next-line no-async-promise-executor
export const fetchIcons = (config) => new Promise(async (resolve) => {
/* c8 ignore next */
@@ -39,10 +34,10 @@ export const fetchIcons = (config) => new Promise(async (resolve) => {
resolve(fetchedIcons);
});
-async function decorateToolTip(icon) {
+function decorateToolTip(icon) {
const wrapper = icon.closest('em');
- if (!wrapper) return;
wrapper.className = 'tooltip-wrapper';
+ if (!wrapper) return;
const conf = wrapper.textContent.split('|');
// Text is the last part of a tooltip
const content = conf.pop().trim();
@@ -50,101 +45,30 @@ async function decorateToolTip(icon) {
icon.dataset.tooltip = content;
// Position is the next to last part of a tooltip
const place = conf.pop()?.trim().toLowerCase() || 'right';
- const defaultIcon = 'info-outline';
- icon.className = `icon icon-${defaultIcon} milo-tooltip ${place}`;
- icon.dataset.name = defaultIcon;
+ icon.className = `icon icon-info milo-tooltip ${place}`;
wrapper.parentElement.replaceChild(icon, wrapper);
}
-export function getIconData(icon) {
- const fedRoot = getFederatedContentRoot();
- const name = [...icon.classList].find((c) => c.startsWith('icon-'))?.substring(5);
- const path = `${fedRoot}/federal/assets/icons/svgs/${name}.svg`;
- return { path, name };
-}
-
-function preloadInViewIconResources(config) {
- const { base } = config;
- loadStyle(`${base}/features/icons/icons.css`);
-}
-
-const preloadInViewIcons = async (icons = []) => icons.forEach((icon) => {
- const { path } = getIconData(icon);
- loadLink(path, { rel: 'preload', as: 'fetch', crossorigin: 'anonymous' });
-});
-
-function filterDuplicatedIcons(icons) {
- if (!icons.length) return [];
- const uniqueIconKeys = new Set();
- const uniqueIcons = [];
- for (const icon of icons) {
- const key = [...icon.classList].find((c) => c.startsWith('icon-'))?.substring(5);
- if (!uniqueIconKeys.has(key)) {
- uniqueIconKeys.add(key);
- uniqueIcons.push(icon);
- }
- }
- return uniqueIcons;
-}
-
-export async function decorateIcons(area, icons, config) {
- if (!icons.length) return;
- const uniqueIcons = filterDuplicatedIcons(icons);
- if (!uniqueIcons.length) return;
- preloadInViewIcons(uniqueIcons);
- preloadInViewIconResources(config);
- icons.forEach((icon) => {
- const iconName = [...icon.classList].find((c) => c.startsWith('icon-'))?.substring(5);
- if (!iconName) return;
- icon.dataset.name = iconName;
- });
-}
-
-export default async function loadIcons(icons) {
- const fedRoot = getFederatedContentRoot();
- const iconRequests = [];
- const iconsToFetch = new Map();
-
+export default async function loadIcons(icons, config) {
+ const iconSVGs = await fetchIcons(config);
+ if (!iconSVGs) return;
icons.forEach(async (icon) => {
- const isToolTip = icon.classList.contains('icon-tooltip');
- if (isToolTip) decorateToolTip(icon);
- const iconName = icon.dataset.name;
- if (icon.dataset.svgInjected || !iconName) return;
- if (!federalIcons[iconName] && !iconsToFetch.has(iconName)) {
- const url = `${fedRoot}/federal/assets/icons/svgs/${iconName}.svg`;
- iconsToFetch.set(iconName, fetch(url)
- .then(async (res) => {
- if (!res.ok) throw new Error(`Failed to fetch SVG for ${iconName}: ${res.statusText}`);
- const text = await res.text();
- const parser = new DOMParser();
- const svgDoc = parser.parseFromString(text, 'image/svg+xml');
- const svgElement = svgDoc.querySelector('svg');
- if (!svgElement) {
- window.lana?.log(`No SVG element found in fetched content for ${iconName}`);
- return;
- }
- const svgClone = svgElement.cloneNode(true);
- svgClone.classList.add('icon-milo', `icon-milo-${iconName}`);
- federalIcons[iconName] = svgClone;
- })
- /* c8 ignore next 3 */
- .catch((error) => {
- window.lana?.log(`Error fetching SVG for ${iconName}:`, error);
- }));
- }
- iconRequests.push(iconsToFetch.get(iconName));
+ const { classList } = icon;
+ if (classList.contains('icon-tooltip')) decorateToolTip(icon);
+ const iconName = icon.classList[1].replace('icon-', '');
+ const existingIcon = icon.querySelector('svg');
+ if (!iconSVGs[iconName] || existingIcon) return;
const parent = icon.parentElement;
- if (parent && parent.parentElement.tagName === 'LI') parent.parentElement.classList.add('icon-list-item');
- });
-
- await Promise.all(iconRequests);
-
- icons.forEach((icon) => {
- const iconName = icon.dataset.name;
- if (iconName && federalIcons[iconName] && !icon.dataset.svgInjected) {
- const svgClone = federalIcons[iconName].cloneNode(true);
- icon.appendChild(svgClone);
- icon.dataset.svgInjected = 'true';
+ if (parent.childNodes.length > 1) {
+ if (parent.lastChild === icon) {
+ icon.classList.add('margin-inline-start');
+ } else if (parent.firstChild === icon) {
+ icon.classList.add('margin-inline-end');
+ if (parent.parentElement.tagName === 'LI') parent.parentElement.classList.add('icon-list-item');
+ } else {
+ icon.classList.add('margin-inline-start', 'margin-inline-end');
+ }
}
+ icon.insertAdjacentHTML('afterbegin', iconSVGs[iconName].outerHTML);
});
}
diff --git a/libs/styles/styles.css b/libs/styles/styles.css
index 66893a7c18..839966a67e 100644
--- a/libs/styles/styles.css
+++ b/libs/styles/styles.css
@@ -128,7 +128,6 @@
--icon-size-s: 32px;
--icon-size-xs: 24px;
--icon-size-xxs: 16px;
- --icon-spacing: 8px;
/* z-index */
--above-all: 9000; /* Used for page tools that overlay page content */
@@ -350,7 +349,6 @@
line-height: 20px;
min-height: 21px;
padding: 7px 18px 8px;
- --icon-spacing: 12px;
}
.xl-button .con-button,
@@ -360,7 +358,6 @@
line-height: 24px;
min-height: 28px;
padding: 10px 24px 8px;
- --icon-spacing: 14px;
}
.xxl-button .con-button,
@@ -370,7 +367,6 @@
line-height: 27px;
min-height: 27px;
padding: 14px 30px 15px;
- --icon-spacing: 14px;
}
.con-button.button-justified {
@@ -563,23 +559,19 @@ div[data-failed="true"]::before {
color: var(--color-gray-300);
}
-span.icon {
- width: 1em;
- display: inline-block;
- margin-inline: var(--icon-spacing);
-}
+span.icon.margin-right { margin-right: 8px; }
-span.icon.node-index-first { margin-inline-start: unset; }
-span.icon.node-index-middle { margin-inline: var(--icon-spacing); }
-span.icon.node-index-last { margin-inline-end: unset; }
-span.icon.node-index-only { margin-inline: unset; }
+span.icon.margin-left { margin-left: 8px; }
-span.icon svg {
- height: 1em;
- position: relative;
- top: .1em;
- width: auto;
-}
+span.icon.margin-inline-end { margin-inline-end: 8px; }
+
+span.icon.margin-inline-start { margin-inline-start: 8px; }
+
+.button-l .con-button span.icon.margin-left,
+.con-button.button-l span.icon.margin-left { margin-left: 12px; }
+
+.button-xl .con-button span.icon.margin-left,
+.con-button.button-xl span.icon.margin-left { margin-left: 14px; }
/* Con Block Utils */
.con-block.xs-spacing { padding: var(--spacing-xs) 0; }
diff --git a/libs/utils/utils.js b/libs/utils/utils.js
index c0c6ddefbb..3c9fe2e194 100644
--- a/libs/utils/utils.js
+++ b/libs/utils/utils.js
@@ -789,6 +789,16 @@ function decorateHeader() {
if (promo?.length) header.classList.add('has-promo');
}
+async function decorateIcons(area, config) {
+ const icons = area.querySelectorAll('span.icon');
+ if (icons.length === 0) return;
+ const { base } = config;
+ loadStyle(`${base}/features/icons/icons.css`);
+ loadLink(`${base}/img/icons/icons.svg`, { rel: 'preload', as: 'fetch', crossorigin: 'anonymous' });
+ const { default: loadIcons } = await import('../features/icons/icons.js');
+ await loadIcons(icons, config);
+}
+
export async function customFetch({ resource, withCacheRules }) {
const options = {};
if (withCacheRules) {
@@ -1265,8 +1275,9 @@ function decorateDocumentExtras() {
decorateHeader();
}
-async function documentPostSectionLoading(area, config) {
+async function documentPostSectionLoading(config) {
decorateFooterPromo();
+
const appendage = getMetadata('title-append');
if (appendage) {
import('../features/title-append/title-append.js').then((module) => module.default(appendage));
@@ -1330,18 +1341,6 @@ async function resolveInlineFrags(section) {
section.preloadLinks = newlyDecoratedSection.preloadLinks;
}
-export function setIconsIndexClass(icons) {
- [...icons].forEach((icon) => {
- const parent = icon.parentNode;
- const children = parent.childNodes;
- const nodeIndex = [...children].indexOf.call(children, icon);
- let indexClass = (nodeIndex === children.length - 1) ? 'last' : 'middle';
- if (nodeIndex === 0) indexClass = 'first';
- if (children.length === 1) indexClass = 'only';
- icon.classList.add(`node-index-${indexClass}`);
- });
-}
-
async function processSection(section, config, isDoc) {
await resolveInlineFrags(section);
const firstSection = section.el.dataset.idx === '0';
@@ -1349,6 +1348,7 @@ async function processSection(section, config, isDoc) {
preloadBlockResources(section.preloadLinks);
await Promise.all([
decoratePlaceholders(section.el, config),
+ decorateIcons(section.el, config),
]);
const loadBlocks = [...stylePromises];
if (section.preloadLinks.length) {
@@ -1381,11 +1381,6 @@ export async function loadArea(area = document) {
decorateDocumentExtras();
}
- const allIcons = area.querySelectorAll('span.icon');
- if (allIcons.length) {
- setIconsIndexClass(allIcons);
- }
-
const sections = decorateSections(area, isDoc);
const areaBlocks = [];
@@ -1398,21 +1393,13 @@ export async function loadArea(area = document) {
});
}
- if (allIcons.length) {
- const { default: loadIcons, decorateIcons } = await import('../features/icons/icons.js');
- const areaIcons = area.querySelectorAll('span.icon');
- await decorateIcons(area, areaIcons, config);
- await loadIcons(areaIcons);
- }
-
const currentHash = window.location.hash;
if (currentHash) {
scrollToHashedElement(currentHash);
}
- if (isDoc) {
- await documentPostSectionLoading(area, config);
- }
+ if (isDoc) await documentPostSectionLoading(config);
+
await loadDeferred(area, areaBlocks, config);
}
diff --git a/test/features/icons/icons.test.js b/test/features/icons/icons.test.js
index cb3cb994db..cd355a0aff 100644
--- a/test/features/icons/icons.test.js
+++ b/test/features/icons/icons.test.js
@@ -1,56 +1,45 @@
import { readFile } from '@web/test-runner-commands';
import { expect } from '@esm-bundle/chai';
-import sinon, { stub } from 'sinon';
-import { waitForElement } from '../../helpers/waitfor.js';
+import { stub } from 'sinon';
+import { setConfig, getConfig, createTag } from '../../../libs/utils/utils.js';
-const { default: loadIcons, getIconData } = await import('../../../libs/features/icons/icons.js');
-const { setIconsIndexClass } = await import('../../../libs/utils/utils.js');
-const mockRes = ({ payload, status = 200, ok = true } = {}) => new Promise((resolve) => {
- resolve({
- status,
- ok,
- json: () => payload,
- text: () => payload,
- });
-});
+const { default: loadIcons } = await import('../../../libs/features/icons/icons.js');
+
+const codeRoot = '/libs';
+const conf = { codeRoot };
+setConfig(conf);
+const config = getConfig();
document.body.innerHTML = await readFile({ path: './mocks/body.html' });
let icons;
-const svgEx = `
-`;
describe('Icon Suppprt', () => {
- beforeEach(() => {
- stub(window, 'fetch').callsFake(() => mockRes({}));
- });
+ let paramsGetStub;
- afterEach(() => {
- sinon.restore();
+ before(() => {
+ paramsGetStub = stub(URLSearchParams.prototype, 'get');
+ paramsGetStub.withArgs('cache').returns('off');
});
- it('Replaces span.icon', async () => {
- const payload = svgEx;
- window.fetch.returns(mockRes({ payload }));
+ after(() => {
+ paramsGetStub.restore();
+ });
+ before(async () => {
icons = document.querySelectorAll('span.icon');
- icons.forEach((icon) => {
- const { name } = getIconData(icon);
- icon.dataset.name = name;
- });
- await loadIcons(icons);
+ await loadIcons(icons, config);
+ await loadIcons(icons, config); // Test duplicate icon not created if run twice
+ });
- const selector = await waitForElement('span.icon svg');
- expect(selector).to.exist;
+ it('Fetches successfully with cache control enabled', async () => {
+ const otherIcons = [createTag('span', { class: 'icon icon-play' })];
+ await loadIcons(otherIcons, config);
});
- it('Sets icon index class', async () => {
- icons = document.querySelectorAll('span.icon');
- setIconsIndexClass(icons);
- const secondIconHasIndexClass = icons[2].classList.contains('node-index-last');
- expect(secondIconHasIndexClass).to.be.true;
+ it('Replaces span.icon', async () => {
+ const selector = icons[0].querySelector(':scope svg');
+ expect(selector).to.exist;
});
it('No duplicate icon', async () => {
diff --git a/test/features/icons/mocks/body.html b/test/features/icons/mocks/body.html
index 2d908a81ca..586013bf4f 100644
--- a/test/features/icons/mocks/body.html
+++ b/test/features/icons/mocks/body.html
@@ -1,6 +1,4 @@
-
+