From 1fa44e16f3cbb03795da6cae55ef719ab0c46275 Mon Sep 17 00:00:00 2001 From: Vjacheslav Trushkin Date: Sun, 2 May 2021 20:47:48 +0300 Subject: [PATCH] Test changing icon while icon is being loaded in React and Svelte components --- .../react/tests/api/30-changing-props.test.js | 104 ++++++++++++++++++ .../tests/api/30-changing-props.test.js | 96 ++++++++++++++++ 2 files changed, 200 insertions(+) diff --git a/packages/react/tests/api/30-changing-props.test.js b/packages/react/tests/api/30-changing-props.test.js index 8201afb..8130ffd 100644 --- a/packages/react/tests/api/30-changing-props.test.js +++ b/packages/react/tests/api/30-changing-props.test.js @@ -149,6 +149,110 @@ describe('Rendering icon', () => { }); }); + test('changing icon property while loading', (done) => { + const prefix = nextPrefix(); + const name = 'changing-prop'; + const name2 = 'changing-prop2'; + const iconName = `@${provider}:${prefix}:${name}`; + const iconName2 = `@${provider}:${prefix}:${name2}`; + let isSync = true; + + mockAPIData({ + provider, + prefix, + response: { + prefix, + icons: { + [name]: iconData, + }, + }, + delay: (next) => { + // Should have been called asynchronously, which means icon name has changed + expect(isSync).toEqual(false); + + // Send icon data + next(); + }, + }); + + mockAPIData({ + provider, + prefix, + response: { + prefix, + icons: { + [name2]: iconData2, + }, + }, + delay: (next) => { + // Should have been called asynchronously + expect(isSync).toEqual(false); + + // Icon should not have loaded yet + expect(iconExists(iconName2)).toEqual(false); + + // Send icon data + next(); + + // Test it again + expect(iconExists(iconName2)).toEqual(true); + + // Check if state was changed + // Wrapped in double setTimeout() because re-render takes 2 ticks + setTimeout(() => { + setTimeout(() => { + const tree = component.toJSON(); + + expect(tree).toMatchObject({ + type: 'svg', + props: { + 'xmlns': 'http://www.w3.org/2000/svg', + 'xmlnsXlink': 'http://www.w3.org/1999/xlink', + 'aria-hidden': true, + 'role': 'img', + 'style': {}, + 'dangerouslySetInnerHTML': { + __html: iconData2.body, + }, + 'width': '1em', + 'height': '1em', + 'preserveAspectRatio': 'xMidYMid meet', + 'viewBox': + '0 0 ' + + iconData2.width + + ' ' + + iconData2.height, + }, + children: null, + }); + + done(); + }, 0); + }, 0); + }, + }); + + // Check if icon has been loaded + expect(iconExists(iconName)).toEqual(false); + + // Render component + const component = renderer.create(); + const tree = component.toJSON(); + + // Should render placeholder + expect(tree).toMatchObject({ + type: 'span', + props: {}, + children: null, + }); + + // Change icon name + component.update(); + + // Async + isSync = false; + }); + test('changing multiple properties', (done) => { const prefix = nextPrefix(); const name = 'multiple-props'; diff --git a/packages/svelte/tests/api/30-changing-props.test.js b/packages/svelte/tests/api/30-changing-props.test.js index d34e7a2..0ed0206 100644 --- a/packages/svelte/tests/api/30-changing-props.test.js +++ b/packages/svelte/tests/api/30-changing-props.test.js @@ -123,6 +123,102 @@ describe('Rendering icon', () => { expect(html).toEqual('
'); }); + test('changing icon property while loading', (done) => { + const prefix = nextPrefix(); + const name = 'changing-prop'; + const name2 = 'changing-prop2'; + const iconName = `@${provider}:${prefix}:${name}`; + const iconName2 = `@${provider}:${prefix}:${name2}`; + let isSync = true; + let triggerSwap; + + mockAPIData({ + provider, + prefix, + response: { + prefix, + icons: { + [name]: iconData, + }, + }, + delay: (next) => { + // Should have been called asynchronously + expect(isSync).toEqual(false); + + // Icon should not have loaded yet + expect(iconExists(iconName)).toEqual(false); + + // Send icon data + next(); + }, + }); + + mockAPIData({ + provider, + prefix, + response: { + prefix, + icons: { + [name2]: iconData2, + }, + }, + delay: (next) => { + // Should have been called asynchronously + expect(isSync).toEqual(false); + + // Icon should not have loaded yet + expect(iconExists(iconName2)).toEqual(false); + + // Send icon data + next(); + + // Test it again + expect(iconExists(iconName2)).toEqual(true); + + // Check if state was changed + // Wrapped in double setTimeout() because re-render takes 2 ticks + setTimeout(() => { + setTimeout(() => { + const node = component.container.querySelector('svg'); + const html = node.parentNode.innerHTML; + + // Check HTML + expect(html).toEqual( + '' + ); + + done(); + }, 0); + }, 0); + }, + }); + + // Check if icon has been loaded + expect(iconExists(iconName)).toEqual(false); + + // Render component + const component = render(ChangeIcon, { + icon1: iconName, + icon2: iconName2, + expose: (swap) => { + triggerSwap = swap; + }, + }); + + // Should render empty icon + const html = component.container.innerHTML; + expect(html).toEqual('
'); + + // Fixture callback should have been called + expect(typeof triggerSwap).toEqual('function'); + + // Change property + triggerSwap(); + + // Async + isSync = false; + }); + test('changing multiple properties', (done) => { const prefix = nextPrefix(); const name = 'multiple-props';