Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Release] Stage to Main #2186

Merged
merged 3 commits into from
Apr 23, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 17 additions & 7 deletions libs/blocks/global-navigation/global-navigation.js
Original file line number Diff line number Diff line change
Expand Up @@ -280,6 +280,9 @@
if (this.useUniversalNav) {
delete this.blocks.profile;
this.blocks.universalNav = toFragment`<div class="feds-utilities"></div>`;
this.blocks.universalNav.addEventListener('click', () => {
if (this.isToggleExpanded()) this.toggleMenuMobile();

Check warning on line 284 in libs/blocks/global-navigation/global-navigation.js

View check run for this annotation

Codecov / codecov/patch

libs/blocks/global-navigation/global-navigation.js#L284

Added line #L284 was not covered by tests
}, true);
}
};

Expand Down Expand Up @@ -615,6 +618,18 @@
});
};

isToggleExpanded = () => this.elements.mobileToggle?.getAttribute('aria-expanded') === 'true';

toggleMenuMobile = () => {
const toggle = this.elements.mobileToggle;
const isExpanded = this.isToggleExpanded();
toggle?.setAttribute('aria-expanded', !isExpanded);
this.elements.navWrapper?.classList?.toggle('feds-nav-wrapper--expanded', !isExpanded);
closeAllDropdowns();
setCurtainState(!isExpanded);
toggle?.setAttribute('daa-ll', `hamburgermenu|${isExpanded ? 'open' : 'close'}`);
};

decorateToggle = () => {
if (!this.mainNavItemCount) return '';

Expand All @@ -638,20 +653,15 @@
};

const onToggleClick = async () => {
const isExpanded = toggle.getAttribute('aria-expanded') === 'true';
toggle.setAttribute('aria-expanded', !isExpanded);
this.elements.navWrapper.classList.toggle('feds-nav-wrapper--expanded', !isExpanded);
closeAllDropdowns();
setCurtainState(!isExpanded);
toggle.setAttribute('daa-ll', `hamburgermenu|${isExpanded ? 'open' : 'close'}`);
this.toggleMenuMobile();

if (this.blocks?.search?.instance) {
this.blocks.search.instance.clearSearchForm();
} else {
await this.loadSearch();
}

if (isExpanded) setHamburgerPadding();
if (this.isToggleExpanded()) setHamburgerPadding();
};

toggle.addEventListener('click', () => logErrorFor(onToggleClick, 'Toggle click failed', 'errorType=error,module=gnav'));
Expand Down
29 changes: 22 additions & 7 deletions libs/blocks/merch-card-collection/merch-card-collection.js
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@ const LITERAL_SLOTS = [
// eslint-disable-next-line no-promise-executor-return
const makePause = async (timeout = 0) => new Promise((resolve) => setTimeout(resolve, timeout));

const BLOCK_NAME = 'merch-card-collection';

const fail = (el, err = '') => {
window.lana?.log(`Failed to initialize merch cards: ${err}`);
el.innerHTML = '';
Expand All @@ -53,11 +55,13 @@ async function getCardsRoot(config, html) {
return cardsRoot;
}

const fetchOverrideCard = (path, config) => new Promise((resolve, reject) => {
fetch(`${localizeLink(path, config)}.plain.html`).then((res) => {
const fetchOverrideCard = (action, config) => new Promise((resolve, reject) => {
fetch(`${localizeLink(action?.target, config)}.plain.html`).then((res) => {
if (res.ok) {
res.text().then((cardContent) => {
resolve({ path, cardContent: /^<div>(.*)<\/div>$/.exec(cardContent.replaceAll('\n', ''))[1] });
const response = { path: action.target, cardContent: /^<div>(.*)<\/div>$/.exec(cardContent.replaceAll('\n', ''))[1] };
if (config?.mep?.preview) response.manifestId = action.manifestId;
resolve(response);
});
} else {
reject(res.statusText
Expand All @@ -70,6 +74,7 @@ const fetchOverrideCard = (path, config) => new Promise((resolve, reject) => {
});

async function overrideCards(root, overridePromises, config) {
let overrideString = '';
try {
if (overridePromises?.length > 0) {
// Wait for all override cards to be fetched
Expand All @@ -84,12 +89,16 @@ async function overrideCards(root, overridePromises, config) {
card.replaceWith(overrideMap[card.name]);
}
});
if (config.mep.preview) {
overrideString = overrideData.map(({ manifestId, path }) => `${manifestId}:${path}`).join(',');
}
}
}
} catch (error) {
/* c8 ignore next */
window?.lana?.log('Failed to override cards', error);
}
return overrideString;
}

/**
Expand Down Expand Up @@ -180,7 +189,7 @@ export default async function init(el) {
import('../../deps/merch-card.js'),
];

const { base } = getConfig();
const { base, mep } = getConfig();
const merchStyles = new Promise((resolve) => {
loadStyle(`${base}/blocks/merch/merch.css`, resolve);
});
Expand Down Expand Up @@ -291,14 +300,20 @@ export default async function init(el) {
}

const cardsRoot = await cardsRootPromise;
const overrides = el.dataset[OVERRIDE_PATHS];
const overridePromises = overrides?.split(',').map(fetchOverrideCard);
await overrideCards(cardsRoot, overridePromises, config);
const overridePromises = mep?.custom?.[BLOCK_NAME]?.map(
(action) => fetchOverrideCard(action, config),
);
const overrides = await overrideCards(cardsRoot, overridePromises, config);
await initMerchCards(attributes.filtered, preferences, cardsRoot);
await Promise.all([merchStyles, merchCardStyles, ...deps]);

merchCardCollection.append(...cardsRoot.children);

merchCardCollection.displayResult = true;
if (config?.mep?.preview && overrides) {
merchCardCollection.dataset.overrides = overrides;
}

await merchCardCollection.updateComplete;
performance.measure(
'merch-card-collection-render',
Expand Down
15 changes: 15 additions & 0 deletions libs/features/personalization/personalization.js
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,8 @@ const DATA_TYPE = {
TEXT: 'text',
};

export const CUSTOM_SELECTOR_PREFIX = 'in-block:';

export const appendJsonExt = (path) => (path.endsWith('.json') ? path : `${path}.json`);

export const normalizePath = (p) => {
Expand Down Expand Up @@ -289,6 +291,15 @@ function getSection(rootEl, idx) {
: rootEl.querySelector(`:scope > div:nth-child(${idx})`);
}

function registerCustomAction(cmd, manifestId) {
const { action, selector, target } = cmd;
const config = getConfig();
const blockName = selector.substring(CUSTOM_SELECTOR_PREFIX.length);
config.mep.custom ??= {};
config.mep.custom[blockName] ??= [];
config.mep.custom[blockName].push({ manifestId, action, target });
}

function getSelectedElement(selector, action, rootEl) {
if (!selector) return null;
if ((action.includes('appendtosection') || action.includes('prependtosection'))) {
Expand Down Expand Up @@ -356,6 +367,10 @@ function getSelectedElement(selector, action, rootEl) {
function handleCommands(commands, manifestId, rootEl = document) {
commands.forEach((cmd) => {
const { action, selector, target } = cmd;
if (selector.startsWith(CUSTOM_SELECTOR_PREFIX)) {
registerCustomAction(cmd, manifestId);
return;
}
if (action in COMMANDS) {
const el = getSelectedElement(selector, action, rootEl);
COMMANDS[action](el, target, manifestId);
Expand Down
22 changes: 21 additions & 1 deletion test/blocks/merch-card-collection/merch-card-collection.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -161,7 +161,26 @@ describe('Merch Cards', async () => {

it('should override cards when asked to', async () => {
const el = document.getElementById('multipleFilters');
el.dataset.overrides = '/override-photoshop,/override-express';
setConfig({
...conf,
mep: {
preview: true,
custom: {
'merch-card-collection': [
{
action: 'replace',
manifestId: 'promo1.json',
target: '/override-photoshop',
},
{
action: 'replace',
manifestId: 'promo2.json',
target: '/override-express',
},
],
},
},
});
cards = [...document.querySelectorAll('#cards .merch-card')]
.map((merchCardEl) => ({ cardContent: merchCardEl.outerHTML })); // mock cards
const merchCards = await init(el);
Expand All @@ -171,6 +190,7 @@ describe('Merch Cards', async () => {
const express = merchCards.querySelector('merch-card[name="express"]');
expect(photoshop.title.indexOf('PROMOTION') > 0).to.be.true;
expect(express.title.indexOf('PROMOTION') > 0).to.be.true;
expect(merchCards.dataset.overrides).to.equal('promo1.json:/override-photoshop,promo2.json:/override-express');
});

it('should localize the query-index url', async () => {
Expand Down
26 changes: 26 additions & 0 deletions test/features/personalization/actions.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -219,3 +219,29 @@ describe('useBlockCode action', async () => {
expect(myBlock.textContent?.trim()).to.equal('My New Block!');
});
});

describe('custom actions', async () => {
it('should not add custom configuration if not needed', async () => {
let manifestJson = await readFile({ path: './mocks/actions/manifestReplace.json' });
manifestJson = JSON.parse(manifestJson);
setFetchResponse(manifestJson);
await applyPers([{ manifestPath: '/path/to/manifest.json' }]);
expect(getConfig().mep.custom).to.be.undefined;
});

it('should add a custom action configuration', async () => {
let manifestJson = await readFile({ path: './mocks/actions/manifestCustomAction.json' });
manifestJson = JSON.parse(manifestJson);
setFetchResponse(manifestJson);

await applyPers([{ manifestPath: '/path/to/manifest.json' }]);
expect(getConfig().mep.custom).to.deep.equal({
'my-block': [{
action: 'replace',
manifestId: 'manifest.json',
target: '/fragments/fragmentreplaced',
},
],
});
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
{
"total": 1,
"offset": 0,
"limit": 5,
"data": [
{
"action": "replace",
"selector": "in-block:my-block",
"page filter (optional)": "",
"param-newoffer=123": "",
"chrome": "/fragments/fragmentreplaced",
"firefox": "",
"android": "",
"ios": ""
}
],
":type": "sheet"
}
Loading