Content-Length: 658154 | pFad | http://github.com/namecheap/ilc/pull/230/commits/fd40f1f23150caf476a7752fe90d394cc5668322

7E Feature/i18n/refactoring and tests by StyleT · Pull Request #230 · namecheap/ilc · GitHub
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

Feature/i18n/refactoring and tests #230

Merged
merged 64 commits into from
Nov 30, 2020
Merged
Changes from 30 commits
Commits
Show all changes
64 commits
Select commit Hold shift + click to select a range
3662bea
feat: basic i18n implementation at server side
StyleT Sep 18, 2020
1218f0a
chore: unit tests fix
StyleT Sep 21, 2020
6c637bf
chore: tests added
StyleT Sep 21, 2020
b54c36e
chore: args order fix
StyleT Sep 23, 2020
b31216c
Merge pull request #200 from namecheap/feature/i18n/basic_ssr_impleme…
StyleT Sep 23, 2020
1a45fb4
Merge branch 'master' into feature/i18n/master
StyleT Sep 25, 2020
eebec42
Merge branch 'master' into feature/i18n/master
StyleT Sep 25, 2020
86a84c9
initial client side intl implementation
StyleT Oct 1, 2020
16c0b23
client side intl implementation vol 2
StyleT Oct 6, 2020
b53b8ce
Fetching intl settings from config
StyleT Oct 6, 2020
65477b5
Bugfix & ability to disable i18n feature
StyleT Oct 6, 2020
ba42365
globalSpinner now is enabled by default & is synchronized with langua…
StyleT Oct 6, 2020
51c4a07
Removed location patch
StyleT Oct 9, 2020
8b43607
added x-request-host headers that will be sent to fragments
StyleT Oct 9, 2020
924b27b
typo fix
StyleT Oct 9, 2020
6a49945
tests fix for ClientRouter
StyleT Oct 9, 2020
ec07550
Adjustments to work with Localization from ILC side
StyleT Oct 9, 2020
da5105f
Better error handling during route change
StyleT Oct 9, 2020
2e18c02
fix: fixed performance tracking for special routes
StyleT Oct 9, 2020
838a83f
Fixed work with query params & hash during locale change
StyleT Oct 9, 2020
3c79ea9
Merge branch 'master' into feature/i18n/master
StyleT Oct 12, 2020
0ed629d
Merge branch 'feature/i18n/master' into feature/i18n/client_side_api
StyleT Oct 12, 2020
d9a5049
Added ability to correctly unmount apps
StyleT Oct 12, 2020
e0b6741
Removed unnecessary use of systemSdk
StyleT Oct 12, 2020
df191ad
fixed issue with infinite loop during locale change error
StyleT Oct 12, 2020
d6f4ef7
feat: currency can be changed together with locale
StyleT Oct 13, 2020
96e09bc
ilc-server-sdk -> ilc-sdk
StyleT Oct 13, 2020
5027be5
Fixed props for navbar
StyleT Oct 13, 2020
50a7278
Updated config
StyleT Oct 13, 2020
4f5e392
ilc-sdk update
StyleT Oct 13, 2020
d235fef
Code refactoring
StyleT Oct 15, 2020
3292a4a
iterablePromise fn moved to utils
StyleT Oct 15, 2020
29ee5e8
Code refactoring
StyleT Oct 16, 2020
0b325b8
tests fix
StyleT Oct 19, 2020
90f62be
client side tests fix
StyleT Oct 19, 2020
3e8133f
code refactoring
StyleT Oct 19, 2020
17fc8f5
chore: improved unit tests readability
StyleT Oct 20, 2020
fd40f1f
Merge pull request #215 from namecheap/feature/i18n/client_side_api
StyleT Oct 22, 2020
b025a12
Merge branch 'master' into feature/i18n/master
StyleT Oct 27, 2020
4ce2b1e
feat: global spinner config moved to registry (#218)
StyleT Oct 28, 2020
a63b7bb
feat: I18n settings were moved to Registry (#219)
StyleT Oct 28, 2020
69c252c
Merge branch 'master' into feature/i18n/master
StyleT Oct 29, 2020
303513b
fix: fixed handling of the localized A tags (#224)
StyleT Oct 30, 2020
8c23c9f
chore: better ilc-sdk import
StyleT Nov 5, 2020
690586a
Merge branch 'master' into feature/i18n/master
StyleT Nov 19, 2020
8fe0972
feat: added ability to export "mainSpa" callback as default one
StyleT Nov 19, 2020
4a16717
Merge pull request #229 from namecheap/feature/i18n/default_mainSpa_c…
StyleT Nov 19, 2020
384b9e7
chore: singleSpa.registerApplication() refactor
StyleT Nov 19, 2020
c55b39e
chore: test code removal
StyleT Nov 19, 2020
5dc518e
feat: added corect currency handling
StyleT Nov 23, 2020
b31fa3f
feat: added i18n.routingStrategy settings key
StyleT Nov 23, 2020
f82a80d
chore: registry tests fix
StyleT Nov 23, 2020
a755871
chore: code refactoring
StyleT Nov 23, 2020
0d05a07
chore: better file naming
StyleT Nov 23, 2020
59cae79
feat: Better interface for interaction with apps during language change
StyleT Nov 24, 2020
04b1513
ilc-sdk bump
StyleT Nov 24, 2020
bcea8be
chore: fixed TODO
StyleT Nov 26, 2020
0fd15a3
feat: added special fn for appId to appName/slotName conversion
StyleT Nov 26, 2020
747c1af
feat: improved error handling for i18n config change event
StyleT Nov 26, 2020
615341f
chore: refactoring & tests fix
StyleT Nov 27, 2020
b4c4f64
chore: unused code removal
StyleT Nov 27, 2020
7954c16
chore: centralized appId to name/slotName conversion
StyleT Nov 27, 2020
0addbae
chore: ilc-sdk version bump
StyleT Nov 27, 2020
38a9bbe
chore: code refactoring
StyleT Nov 27, 2020
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
39 changes: 33 additions & 6 deletions ilc/client.js
Original file line number Diff line number Diff line change
@@ -5,36 +5,44 @@ import Router from './client/ClientRouter';
import setupErrorHandlers from './client/errorHandler/setupErrorHandlers';
import {fragmentErrorHandlerFactory, crashIlc} from './client/errorHandler/fragmentErrorHandlerFactory';
import isActiveFactory from './client/isActiveFactory';
import initIlcConfig from './client/initIlcConfig';
import getIlcConfig from './client/ilcConfig';
import initIlcState from './client/initIlcState';
import setupPerformanceMonitoring from './client/performance';
import selectSlotsToRegister from './client/selectSlotsToRegister';
import {getSlotElement} from './client/utils';
import AsyncBootUp from './client/AsyncBootUp';
import IlcAppSdk from 'ilc-sdk/app';
import I18n from './client/i18n';

const System = window.System;
if (System === undefined) {
crashIlc();
throw new Error('ILC: can\'t find SystemJS on a page, crashing everything');
}

const registryConf = initIlcConfig();
const registryConf = getIlcConfig();
const state = initIlcState();
const router = new Router(registryConf, state, singleSpa);

const i18n = registryConf.settings.i18n ? new I18n(registryConf.settings.i18n, singleSpa) : null;
const router = new Router(registryConf, state, singleSpa, i18n ? i18n.unlocalizeUrl : undefined);
const asyncBootUp = new AsyncBootUp();

// Here we expose window.ILC.define also as window.define to ensure that regular AMD/UMD bundles work correctly by default
// See docs/umd_bundles_compatibility.md
if (!registryConf.settings.amdDefineCompatibilityMode) {
window.define = window.ILC.define;
}
window.ILC.getAppSdkAdapter = appId => ({appId, intl: i18n ? i18n.getAdapter() : null});

selectSlotsToRegister([...registryConf.routes, registryConf.specialRoutes['404']]).forEach((slots) => {
Object.keys(slots).forEach((slotName) => {
const appName = slots[slotName].appName;

const fragmentName = `${appName.replace('@portal/', '')}__at__${slotName}`;

const appSdk = new IlcAppSdk(window.ILC.getAppSdkAdapter(fragmentName));
const onUnmount = async () => appSdk.unmount();

singleSpa.registerApplication(
fragmentName,
async () => {
@@ -57,21 +65,40 @@ selectSlotsToRegister([...registryConf.routes, registryConf.specialRoutes['404']
}));
}

return Promise.all(waitTill)
.then(v => v[0].mainSpa !== undefined ? v[0].mainSpa(appConf.props || {}) : v[0]);
return Promise.all(waitTill).then(([spaBundle]) => {
const spaCallbacks = spaBundle.mainSpa !== undefined ? spaBundle.mainSpa(appConf.props || {}) : spaBundle;

let unmount = spaCallbacks.unmount;
if (Array.isArray(unmount)) {
unmount.unshift(onUnmount)
} else {
unmount = [onUnmount, unmount];
}

return {
bootstrap: spaCallbacks.bootstrap,
mount: spaCallbacks.mount,
unmount,
unload: spaCallbacks.unload,
};
});
},
isActiveFactory(router, appName, slotName),
{
domElementGetter: () => getSlotElement(slotName),
getCurrentPathProps: () => router.getCurrentRouteProps(appName, slotName),
getCurrentBasePath: () => router.getCurrentRoute().basePath,
appId: fragmentName, // Unique application ID, if same app will be rendered twice on a page - it will get different IDs
errorHandler: fragmentErrorHandlerFactory(registryConf, router.getCurrentRoute, appName, slotName)
errorHandler: fragmentErrorHandlerFactory(registryConf, router.getCurrentRoute, appName, slotName),
appSdk,
}
);
});
});

//TODO: to be removed
window.__IlcAppSdk = i18n ? new IlcAppSdk({appId: 'tst', intl: i18n.getAdapter()}) : null;

setupErrorHandlers(registryConf, router.getCurrentRoute);
setupPerformanceMonitoring(router.getCurrentRoute);

29 changes: 23 additions & 6 deletions ilc/client/ClientRouter.js
Original file line number Diff line number Diff line change
@@ -17,11 +17,20 @@ export default class ClientRouter {
#currentRoute;
#windowEventHandlers = {};
#forceSpecialRoute = null;

constructor(registryConf, state, singleSpa, location = window.location, logger = window.console) {
#unlocalizeUrl;

constructor(
registryConf,
state,
singleSpa,
unlocalizeUrl = (v) => v,
location = window.location,
logger = window.console
) {
this.#singleSpa = singleSpa;
this.#location = location;
this.#logger = logger;
this.#unlocalizeUrl = unlocalizeUrl;
this.#registryConf = registryConf;
this.#router = new Router(registryConf);
this.#currentUrl = this.#getCurrUrl();
@@ -106,7 +115,7 @@ export default class ClientRouter {
this.#prevRoute = this.#currentRoute;

const newUrl = this.#getCurrUrl();
if (this.#forceSpecialRoute !== null && this.#forceSpecialRoute.url === newUrl) {
if (this.#forceSpecialRoute !== null && this.#forceSpecialRoute.url === this.#getCurrUrl(true)) {
this.#currentRoute = this.#router.matchSpecial(newUrl, this.#forceSpecialRoute.id);
} else if (this.#forceSpecialRoute !== null) {
// Reset variable if it was set & now we go to different route
@@ -118,7 +127,7 @@ export default class ClientRouter {
// only hash will be changed so router.match will return error, since <base> tag has already been removed.
// so in this cases we shouldn't regenerate currentRoute
if (this.#currentUrl !== newUrl) {
this.#currentRoute = this.#router.match(this.#location.pathname + this.#location.search);
this.#currentRoute = this.#router.match(this.#getCurrUrl());
this.#currentUrl = newUrl;
}

@@ -166,9 +175,17 @@ export default class ClientRouter {
}

console.log(`ILC: Special route "${specialRouteId}" was triggered by "${appId}" app. Performing rerouting...`);
this.#forceSpecialRoute = {id: specialRouteId, url: this.#getCurrUrl()};
this.#forceSpecialRoute = {id: specialRouteId, url: this.#getCurrUrl(true)};
this.#singleSpa.triggerAppChange(); //This call would immediately invoke "single-spa:before-routing-event" and start apps mount/unmount process
};

#getCurrUrl = () => this.#location.pathname + this.#location.search;
#getCurrUrl = (withLocale = false) => {
const url = this.#location.pathname + this.#location.search;

if (withLocale) {
return url;
}

return this.#unlocalizeUrl(url);
}
}
10 changes: 5 additions & 5 deletions ilc/client/ClientRouter.spec.js
Original file line number Diff line number Diff line change
@@ -201,7 +201,7 @@ describe('client router', () => {
},
};

router = new ClientRouter(registryConfig, {}, singleSpa, location);
router = new ClientRouter(registryConfig, {}, singleSpa, undefined, location);

chai.expect(router.getCurrentRoute()).to.be.eql(expectedRoute);
chai.expect(router.getPrevRoute()).to.be.eql(expectedRoute);
@@ -259,7 +259,7 @@ describe('client router', () => {
},
};

router = new ClientRouter(registryConfig, {}, singleSpa, location, logger);
router = new ClientRouter(registryConfig, {}, singleSpa, undefined, location, logger);

chai.expect(mainRef.getElementsByTagName('base')).to.be.empty;

@@ -304,7 +304,7 @@ describe('client router', () => {
},
};

router = new ClientRouter(registryConfig, {}, singleSpa, location);
router = new ClientRouter(registryConfig, {}, singleSpa, undefined, location);

location.pathname = registryConfig.routes[2].route;
location.search = '?see=you';
@@ -345,7 +345,7 @@ describe('client router', () => {
},
};

router = new ClientRouter(registryConfig, {}, singleSpa, location);
router = new ClientRouter(registryConfig, {}, singleSpa, undefined, location);

window.dispatchEvent(singleSpaBeforeRoutingEvent);

@@ -362,7 +362,7 @@ describe('client router', () => {
search: '?hi=there',
};

router = new ClientRouter(registryConfig, {}, singleSpa, location);
router = new ClientRouter(registryConfig, {}, singleSpa, undefined, location);

const [eventName, eventListener] = addEventListener.getCall(0).args;

184 changes: 184 additions & 0 deletions ilc/client/TransactionManager.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,184 @@
import { getSlotElement } from './utils';
import getIlcConfig from './ilcConfig';

import scrollRestorer from '@mapbox/scroll-restorer';

export const slotWillBe = {
rendered: 'rendered',
removed: 'removed',
rerendered: 'rerendered',
default: null,
};

export class TransactionManager {
#registrySettings;
#globalSpinner;
#spinnerTimeout;

#fakeSlots = [];
#hiddenSlots = [];
#transactionBlockers = [];

constructor(registrySettings = {}) {
this.#registrySettings = Object.assign({
globalSpinner: true
}, registrySettings);

scrollRestorer.start({ autoRestore: false, captureScrollDebounce: 150 }); //TODO: add cleanup

window.addEventListener('ilc:crash', () => { //TODO: add cleanup
this.#removeGlobalSpinner();
});
}

handleAsyncAction(promise) {
if (this.#registrySettings.globalSpinner === false) {
return;
}

this.#runGlobalSpinner();
this.#transactionBlockers.push(promise);

const afterPromise = () => this.#removeTransactionBlocker(promise);
promise.then(afterPromise).catch(afterPromise)
}

handlePageTransaction(slotName, willBe) {
if (this.#registrySettings.globalSpinner === false) {
return;
}

if (!slotName) {
throw new Error('A slot name was not provided!');
}

switch (willBe) {
case slotWillBe.rendered:
this.#addContentListener(slotName);
break;
case slotWillBe.removed:
this.#renderFakeSlot(slotName);
break;
case slotWillBe.rerendered:
this.#renderFakeSlot(slotName);
this.#addContentListener(slotName);
break;
case slotWillBe.default:
break;
default:
throw new Error(`The slot action '${willBe}' did not match any possible values!`);
}
}

#addContentListener = slotName => {
this.#runGlobalSpinner();

if (window.location.hash) {
document.body.setAttribute('name', window.location.hash.slice(1));
}

const status = {
hasAddedNodes: false,
hasTextOrOpticNodes: false,
isAnyChildVisible: false,
};

const observer = new MutationObserver((mutationsList, observer) => {
if (!status.hasAddedNodes) {
status.hasAddedNodes = !!mutationsList.find(mutation => mutation.addedNodes.length);
}

// if we have rendered MS to DOM but meaningful content isn't rendered, e.g. due to essential data preload
if (!status.hasTextOrOpticNodes) {
const hasText = !!targetNode.innerText.trim().length
const hasOpticNodes = !!targetNode.querySelector(':not(div):not(span)');
status.hasTextOrOpticNodes = hasText || hasOpticNodes;
}

// if we have rendered MS to DOM but temporary hide it for some reason, e.g. to fetch data
if (!status.isAnyChildVisible) {
status.isAnyChildVisible = Array.from(targetNode.children).some(node => node.style.display !== 'none');
}

if (Object.values(status).some(n => !n)) return;

observer.disconnect();
this.#removeTransactionBlocker(observer);
});
this.#transactionBlockers.push(observer);
const targetNode = getSlotElement(slotName);
targetNode.style.display = 'none'; // we will show all new slots, only when all will be settled
this.#hiddenSlots.push(targetNode);
observer.observe(targetNode, { childList: true, subtree: true, attributeFilter: ['style'] });
};

#renderFakeSlot = slotName => {
const targetNode = getSlotElement(slotName);
const clonedNode = targetNode.cloneNode(true);
clonedNode.removeAttribute('id');
clonedNode.removeAttribute('class');
this.#fakeSlots.push(clonedNode);
targetNode.parentNode.insertBefore(clonedNode, targetNode.nextSibling);
targetNode.style.display = 'none'; // we hide old slot because fake already in the DOM.
this.#hiddenSlots.push(targetNode);
};

#onAllSlotsLoaded = () => {
this.#fakeSlots.forEach(node => node.remove());
this.#fakeSlots.length = 0;
this.#hiddenSlots.forEach(node => node.style.display = '');
this.#hiddenSlots.length = 0;
this.#removeGlobalSpinner();
document.body.removeAttribute('name');
scrollRestorer.restoreScroll(window.history.state ? window.history : {state: {scroll: {x: 0, y: 0}}});

window.dispatchEvent(new CustomEvent('ilc:all-slots-loaded'));
};

#runGlobalSpinner = () => {
const registrySettings = this.#registrySettings;
if (registrySettings.globalSpinner === false || this.#spinnerTimeout) {
return;
}

this.#spinnerTimeout = setTimeout(() => {
if (registrySettings.globalSpinner === true) {
this.#globalSpinner = document.createElement('dialog');
this.#globalSpinner.innerHTML = 'loading....';
document.body.appendChild(this.#globalSpinner);
this.#globalSpinner.showModal();
} else {
this.#globalSpinner = document.createElement('div');
this.#globalSpinner.innerHTML = registrySettings.globalSpinner;
document.body.appendChild(this.#globalSpinner);
}
}, 200);
};

#removeGlobalSpinner = () => {
if (this.#globalSpinner) {
this.#globalSpinner.remove();
this.#globalSpinner = null;
}

clearTimeout(this.#spinnerTimeout);
this.#spinnerTimeout = null;
};

#removeTransactionBlocker = (blocker) => {
this.#transactionBlockers.splice(this.#transactionBlockers.indexOf(blocker), 1);
!this.#transactionBlockers.length && this.#onAllSlotsLoaded();
}
}

let defaultInstance = null;
/**
* @return {TransactionManager}
*/
export default function defaultFactory() {
if (defaultInstance === null) {
defaultInstance = new TransactionManager(getIlcConfig().settings);
}

return defaultInstance;
}
Loading
Oops, something went wrong.








ApplySandwichStrip

pFad - (p)hone/(F)rame/(a)nonymizer/(d)eclutterfier!      Saves Data!


--- a PPN by Garber Painting Akron. With Image Size Reduction included!

Fetched URL: http://github.com/namecheap/ilc/pull/230/commits/fd40f1f23150caf476a7752fe90d394cc5668322

Alternative Proxies:

Alternative Proxy

pFad Proxy

pFad v3 Proxy

pFad v4 Proxy