From df3bfd37a7f14322ac463f17dbfba5050a8343b5 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 1 May 2022 14:56:47 -0400 Subject: [PATCH 01/17] fix range input issue, add unit test, better color scheme support --- integration-tests/regression.test.js | 41 +++++++++++++++++++ .../test-app/element-store-generator.ts | 34 +++++++++++++++ .../test-app/element-with-store.ts | 13 ++++++ integration-tests/test-app/index.html | 6 +-- integration-tests/test-app/index.ts | 3 ++ package-lock.json | 1 + src/observe-tag.ts | 18 +++++--- 7 files changed, 107 insertions(+), 9 deletions(-) create mode 100644 integration-tests/test-app/element-store-generator.ts create mode 100644 integration-tests/test-app/element-with-store.ts diff --git a/integration-tests/regression.test.js b/integration-tests/regression.test.js index d691d4d..466513f 100644 --- a/integration-tests/regression.test.js +++ b/integration-tests/regression.test.js @@ -239,4 +239,45 @@ describe('Tram-One', () => { expect(document.title).toEqual('Tram-One Testing App'); }); + + it('should keep focus on inputs without a start and end selection', async () => { + // start the app + const { container } = startApp(); + + // previously when interacting with an input of a different type (e.g. range) + // when reapplying focus Tram-One would throw an error because while the + // function for setting selection range exists, it does not work + + // focus on the input (the range input defaults to 0) + userEvent.click(getByLabelText(container, 'Store Generator')); + + // verify that the element has focus (before we start changing text) + await waitFor(() => { + expect(getByLabelText(container, 'Store Generator')).toHaveFocus(); + }); + + // hit the right arrow key to increment the value + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 1 } }); + // userEvent.type(getByLabelText(container, 'Store Generator'), '{arrowright}'); + + // verify the element has the new value + expect(getByLabelText(container, 'Store Generator')).toHaveValue('1'); + + // wait for mutation observer to re-attach focus + // expect the input to keep focus after the change event + await waitFor(() => { + expect(getByLabelText(container, 'Store Generator')).toHaveFocus(); + }); + }); + + it('should clean up stores for elements that are no longer rendered', async () => { + // start the app + const { container } = startApp(); + + // previously stores made for elements that had been removed stayed in the tram-observable-store + + // FOR JESSE, DEBUGGING NOTES: + // window['tram-space']['tram-observable-store'] + // look at target, notice that it grows, even when the stores should have been removed + }); }); diff --git a/integration-tests/test-app/element-store-generator.ts b/integration-tests/test-app/element-store-generator.ts new file mode 100644 index 0000000..ce6163e --- /dev/null +++ b/integration-tests/test-app/element-store-generator.ts @@ -0,0 +1,34 @@ +import { registerHtml, useStore, TramOneComponent } from '../../src/tram-one'; +import elementwithstore from './element-with-store'; + +const html = registerHtml({ + elementwithstore, +}); + +/** + * Element to verify non-standard input controls, and also verify memory leak type issues + */ +const elementstoregenerator: TramOneComponent = () => { + const storeGeneratorStore = useStore({ count: 0 }); + const incrementCount = (event: InputEvent) => { + const inputElement = event.target as HTMLInputElement; + storeGeneratorStore.count = parseInt(inputElement.value); + }; + const storeElements = [...new Array(storeGeneratorStore.count)].map((_, index) => { + return html``; + }); + return html`
+ + + ${storeElements} +
`; +}; + +export default elementstoregenerator; diff --git a/integration-tests/test-app/element-with-store.ts b/integration-tests/test-app/element-with-store.ts new file mode 100644 index 0000000..016d5b9 --- /dev/null +++ b/integration-tests/test-app/element-with-store.ts @@ -0,0 +1,13 @@ +import { registerHtml, useStore, TramOneComponent } from '../../src/tram-one'; + +const html = registerHtml(); + +/** + * Dynamicly generated component that could possibly cause memory leaks + */ +const elementwithstore: TramOneComponent = ({ value }) => { + const subElementStore = useStore({ active: value }); + return html` ${subElementStore.active}, `; +}; + +export default elementwithstore; diff --git a/integration-tests/test-app/index.html b/integration-tests/test-app/index.html index 856a987..8aed791 100644 --- a/integration-tests/test-app/index.html +++ b/integration-tests/test-app/index.html @@ -3,9 +3,9 @@ diff --git a/integration-tests/test-app/index.ts b/integration-tests/test-app/index.ts index 3f4c5ed..bfd4aaf 100644 --- a/integration-tests/test-app/index.ts +++ b/integration-tests/test-app/index.ts @@ -8,6 +8,7 @@ import account from './account'; import tasks from './tasks'; import mirrorinput from './mirror-input'; import documentTitleSetter from './document-title-setter'; +import elementstoregenerator from './element-store-generator'; const html = registerHtml({ title: title, @@ -19,6 +20,7 @@ const html = registerHtml({ tasks: tasks, 'mirror-input': mirrorinput, 'document-title-setter': documentTitleSetter, + 'element-store-generator': elementstoregenerator, }); /** @@ -47,6 +49,7 @@ export const app = () => { + `; }; diff --git a/package-lock.json b/package-lock.json index d261ca6..2a1021a 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5,6 +5,7 @@ "requires": true, "packages": { "": { + "name": "tram-one", "version": "11.0.1", "license": "MIT", "dependencies": { diff --git a/src/observe-tag.ts b/src/observe-tag.ts index bf3e6a0..b910021 100644 --- a/src/observe-tag.ts +++ b/src/observe-tag.ts @@ -120,12 +120,18 @@ export default (tagFunction: () => TramOneElement): TramOneElement => { elementToGiveFocus = allActiveLikeElements[elementIndexToGiveFocus] as ElementPotentiallyWithSelectionAndFocus; // also try to set the selection, if there is a selection for this element - if (elementToGiveFocus.setSelectionRange !== undefined) { - elementToGiveFocus.setSelectionRange( - removedElementWithFocusData.selectionStart, - removedElementWithFocusData.selectionEnd, - removedElementWithFocusData.selectionDirection - ); + try { + if (elementToGiveFocus.setSelectionRange !== undefined) { + elementToGiveFocus.setSelectionRange( + removedElementWithFocusData.selectionStart, + removedElementWithFocusData.selectionEnd, + removedElementWithFocusData.selectionDirection + ); + } + } catch (exception) { + // don't worry if we fail + // this can happen if the element has a `setSelectionRange` but it isn't supported + // e.g. input with type="range" } elementToGiveFocus.scrollLeft = removedElementWithFocusData.scrollLeft; From dc6a4361857d2e9f78dc849474830591cbba1a12 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 1 May 2022 15:02:59 -0400 Subject: [PATCH 02/17] fix comment --- src/mutation-observer.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index a8b87f9..42adad4 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -58,7 +58,7 @@ const cleanupEffects = (cleanupEffects: (() => void)[]) => { // unobserve the reaction tied to the node, and run all cleanup effects for the node const clearNode = (node: Node | TramOneElement) => { - // if this element doesn't have a Reaction, it is not be a Tram-One Element + // if this element doesn't have a Reaction, it is not a Tram-One Element if (!(TRAM_TAG in node)) { return; } From 9b7139eca0e559644b1d51e36de3c117716b27c5 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 1 May 2022 15:03:31 -0400 Subject: [PATCH 03/17] additional performance test-app (no tests... yet) --- .../test-app/element-switcher-store.ts | 11 ++++ .../test-app/element-switcher-sub-section.ts | 33 +++++++++++ .../test-app/element-switcher.ts | 57 +++++++++++++++++++ performance-tests/test-app/index.ts | 3 + 4 files changed, 104 insertions(+) create mode 100644 performance-tests/test-app/element-switcher-store.ts create mode 100644 performance-tests/test-app/element-switcher-sub-section.ts create mode 100644 performance-tests/test-app/element-switcher.ts diff --git a/performance-tests/test-app/element-switcher-store.ts b/performance-tests/test-app/element-switcher-store.ts new file mode 100644 index 0000000..a32d46d --- /dev/null +++ b/performance-tests/test-app/element-switcher-store.ts @@ -0,0 +1,11 @@ +const { registerHtml, useStore } = require('../../src/tram-one'); + +const html = registerHtml(); + +/** + * Dynamicly generated component that could possibly cause memory leaks + */ +export default () => { + const subElementStore = useStore({ active: 1 }); + return html` ${subElementStore.active}, `; +}; diff --git a/performance-tests/test-app/element-switcher-sub-section.ts b/performance-tests/test-app/element-switcher-sub-section.ts new file mode 100644 index 0000000..dd76ebc --- /dev/null +++ b/performance-tests/test-app/element-switcher-sub-section.ts @@ -0,0 +1,33 @@ +const { registerHtml, useGlobalStore, useEffect } = require('../../src/tram-one'); + +const html = registerHtml(); + +/** + * This component changes the presentation and document title based on global state + */ +export default () => { + const pageStore = useGlobalStore('STORE'); + useEffect(() => { + document.title = pageStore.selected; + }); + return html` +
+ ${0 * pageStore.selected} + ${1 * pageStore.selected} + ${2 * pageStore.selected} + ${3 * pageStore.selected} + ${4 * pageStore.selected} + ${5 * pageStore.selected} + ${6 * pageStore.selected} + ${7 * pageStore.selected} + ${8 * pageStore.selected} + ${9 * pageStore.selected} + ${10 * pageStore.selected} + ${11 * pageStore.selected} + ${12 * pageStore.selected} + ${13 * pageStore.selected} + ${14 * pageStore.selected} + ${15 * pageStore.selected} +
+ `; +}; diff --git a/performance-tests/test-app/element-switcher.ts b/performance-tests/test-app/element-switcher.ts new file mode 100644 index 0000000..8180ce3 --- /dev/null +++ b/performance-tests/test-app/element-switcher.ts @@ -0,0 +1,57 @@ +const { registerHtml, useGlobalStore, useStore } = require('../../src/tram-one'); +import ElementSwitcher from './element-switcher-sub-section'; +import ElementStore from './element-switcher-store'; + +const html = registerHtml({ + ElementSwitcher, + ElementStore, +}); + +/** + * This page controls and switches rendering based on what's selected + */ +export default () => { + const pageStore = useGlobalStore('STORE', { selected: 1 }); + const switchStore = useStore({ id: 0 }); + + const changeSelection = (newSelection) => () => { + pageStore.selected = newSelection; + }; + + const startAutoSwitch = () => { + switchStore.id = setInterval(() => { + ( + document.querySelector('nav button[selected] + button') || document.querySelector('nav button:first-of-type') + ).click(); + }, 500); + }; + + const stopAutoSwitch = () => { + clearInterval(switchStore.id); + switchStore.id = 0; + }; + + const cycle = switchStore.id + ? html`` + : html``; + + const storeElements = [...new Array(pageStore.selected)].map(() => { + return html``; + }); + + return html` +
+

Element Switching Example

+ ${cycle} + + ${storeElements} + +
+ `; +}; diff --git a/performance-tests/test-app/index.ts b/performance-tests/test-app/index.ts index b5d05a8..4f33afb 100644 --- a/performance-tests/test-app/index.ts +++ b/performance-tests/test-app/index.ts @@ -2,9 +2,11 @@ const useUrlParams = require('use-url-params'); import { registerHtml, start } from '../../src/tram-one'; import elementRendering from './element-rendering'; +import elementSwitcher from './element-switcher'; const html = registerHtml({ 'element-rendering': elementRendering, + 'element-switcher': elementSwitcher, }); /** @@ -12,6 +14,7 @@ const html = registerHtml({ */ export const app = () => { if (useUrlParams('/element-rendering').matches) return html`
`; + if (useUrlParams('/element-switcher').matches) return html`
`; return html`

Performance Test App

From 7b0e2e667733ba86058721ba4f5d42d69550dac3 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 21:58:47 -0400 Subject: [PATCH 04/17] remove old stores, lots of tests --- integration-tests/integration.test.js | 9 +- integration-tests/internals.test.js | 69 +++++++++ integration-tests/regression.test.js | 137 +++++++++++++++--- .../test-app/element-store-generator.ts | 2 +- .../test-app/element-with-store.ts | 7 +- integration-tests/test-app/index.ts | 7 + integration-tests/test-helpers.ts | 19 +++ integration-tests/warnings.test.js | 5 - src/dom.ts | 6 +- src/effect-store.ts | 4 +- src/engine-names.ts | 2 + src/key-queue.ts | 31 ++++ src/key-store.ts | 48 ++++++ src/mutation-observer.ts | 82 +++++++++-- src/node-names.ts | 1 + src/observable-hook.ts | 12 +- src/observe-tag.ts | 5 +- src/{process-effects.ts => process-hooks.ts} | 25 +++- src/start.ts | 13 ++ src/types.ts | 16 +- src/working-key.ts | 20 +-- 21 files changed, 446 insertions(+), 74 deletions(-) create mode 100644 integration-tests/internals.test.js create mode 100644 integration-tests/test-helpers.ts create mode 100644 src/key-queue.ts create mode 100644 src/key-store.ts rename src/{process-effects.ts => process-hooks.ts} (51%) diff --git a/integration-tests/integration.test.js b/integration-tests/integration.test.js index a8153ad..3c2bbfe 100644 --- a/integration-tests/integration.test.js +++ b/integration-tests/integration.test.js @@ -1,12 +1,11 @@ const { getByText, fireEvent, waitFor } = require('@testing-library/dom'); const { startApp } = require('./test-app'); -describe('Tram-One', () => { - beforeEach(() => { - // clean up any tram-one properties between tests - window['tram-space'] = undefined; - }); +/** + * The following tests are intentional test that validate the behavior of new features. + */ +describe('Tram-One', () => { it('should render on a Node', () => { // mount the app on the container const container = document.createElement('div'); diff --git a/integration-tests/internals.test.js b/integration-tests/internals.test.js new file mode 100644 index 0000000..393dc5a --- /dev/null +++ b/integration-tests/internals.test.js @@ -0,0 +1,69 @@ +const { queryByText, fireEvent, waitFor, getByLabelText } = require('@testing-library/dom'); +const { default: userEvent } = require('@testing-library/user-event'); +const { startAppAndWait } = require('./test-helpers'); + +/** + * The following suite of tests verify the behavior of the internals of Tram-One, more so than other tests might. + * They are often inpercievable to end-users, and verify the expected behavior of the behind-the-scenes design. + */ + +describe('Tram-One', () => { + it('should clean up stores for elements that are no longer rendered', async () => { + // start the app + const { container } = await startAppAndWait(); + + // previously stores made for elements that had been removed stayed in the tram-observable-store + + const initialStores = Object.keys(window['tram-space']['tram-observable-store']); + + // focus on the input (the range input defaults to 0) + userEvent.click(getByLabelText(container, 'Store Generator')); + + // change the value of the input + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 1 } }); + + await waitFor(() => { + // make sure the new control is in the document + // (additionally, we're doing this to make sure that all the mutation observers have had a chance to catch up) + expect(queryByText(container, '[0: 0]')).toBeVisible(); + }); + + // expect us to have one additional store now + const postChangeStores = Object.keys(window['tram-space']['tram-observable-store']); + expect(postChangeStores.length).toBe(initialStores.length + 1); + + // change the value of the input back to 0 + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 0 } }); + + await waitFor(() => { + // make sure the new control is in the document + // (additionally, we're doing this to make sure that all the mutation observers have had a chance to catch up) + expect(queryByText(container, '[0: 0]')).toBe(null); + }); + + // wait for mutation observer clean up removed stores + await waitFor(() => { + const postChangeStoresTwo = Object.keys(window['tram-space']['tram-observable-store']); + // check that the lists are the same (they may have shuffled, so sort them) + expect(postChangeStoresTwo.sort()).toEqual(initialStores.sort()); + }); + }); + + it('should not have recursive working-key branches', async () => { + // start the app + await startAppAndWait(); + + // previously the working branch indices would have long recursive chains of branches + const workingKeyBranches = Object.keys(window['tram-space']['tram-hook-key'].branchIndices); + + // verify that top-level elements exist + expect(workingKeyBranches).toEqual(expect.arrayContaining(['app[{}]'])); + expect(workingKeyBranches).toEqual(expect.arrayContaining(['app[{}]/logo[{}]'])); + expect(workingKeyBranches).toEqual(expect.arrayContaining(['app[{}]/tab[{}]'])); + + // verify that no element contains a duplicate of 'app[{}]' - this indicates an issue with the key generation + workingKeyBranches.forEach((branch) => { + expect(branch).not.toMatch(/app\[\{\}\].*app\[\{\}\]/); + }); + }); +}); diff --git a/integration-tests/regression.test.js b/integration-tests/regression.test.js index 466513f..e27c8dc 100644 --- a/integration-tests/regression.test.js +++ b/integration-tests/regression.test.js @@ -1,11 +1,16 @@ -const { getByText, queryAllByText, fireEvent, waitFor, getByLabelText } = require('@testing-library/dom'); +const { getByText, queryAllByText, fireEvent, waitFor, getByLabelText, queryByText } = require('@testing-library/dom'); const { default: userEvent } = require('@testing-library/user-event'); -const { startApp } = require('./test-app'); +const { startAppAndWait } = require('./test-helpers'); + +/** + * The following suite of tests are made retroactively for unexpected behaviors. + * They are not for any direct feature, but rather validate the behavior of framework as a whole. + */ describe('Tram-One', () => { it('should not call cleanups that are not functions', async () => { // start the app - const { container } = startApp(); + const { container } = await await startAppAndWait(); // previously this would fail because the cleanup was called, // even though it was not a function, and instead was a promise (the result of an async function) @@ -18,7 +23,7 @@ describe('Tram-One', () => { it('should call updated cleanups', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // verify that the tab is rendered and the lock button is there expect(getByText(container, 'Was Locked: false')).toBeVisible(); @@ -48,7 +53,7 @@ describe('Tram-One', () => { it('should process state as an array', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when state was being processed, it would be converted to an object // this test adds an element to a store to verify array methods work @@ -67,7 +72,7 @@ describe('Tram-One', () => { window.history.pushState({}, '', '/test_account'); // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // verify the account info is read correctly at startup expect(getByText(container, 'Account: test_account')).toBeVisible(); @@ -75,7 +80,7 @@ describe('Tram-One', () => { it('should keep focus on inputs when components would rerender', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when interacting with an input, if the component would rerender // focus would be removed from the component and put on the body of the page @@ -89,7 +94,7 @@ describe('Tram-One', () => { }); // clear the input - userEvent.type(getByLabelText(container, 'New Task Label'), '{selectall}{backspace}'); + userEvent.clear(getByLabelText(container, 'New Task Label')); // wait for mutation observer to reapply focus await waitFor(() => { @@ -111,7 +116,7 @@ describe('Tram-One', () => { it('should keep focus on the most recent input when components rerender', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when interacting with an input, if the component would rerender // focus would be removed from the component and put on the body of the page @@ -150,7 +155,7 @@ describe('Tram-One', () => { it('should keep focus when both the parent and child element would update', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when interacting with an input, if both a parent and child element // would update, then focus would not reattach, and/or the value would not update correctly @@ -200,7 +205,7 @@ describe('Tram-One', () => { it('should not error when resetting focus if the number of elements changed', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when interacting with an input, if the number of elements decreased // an error was thrown because the element to focus on no longer existed @@ -232,7 +237,7 @@ describe('Tram-One', () => { it('should trigger use-effects of the first resolved element', async () => { // start the app - startApp(); + await startAppAndWait(); // previously, useEffects on the first resolved element would not trigger // because the effect queue and effect store were pointed to the same object instance @@ -242,7 +247,7 @@ describe('Tram-One', () => { it('should keep focus on inputs without a start and end selection', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); // previously when interacting with an input of a different type (e.g. range) // when reapplying focus Tram-One would throw an error because while the @@ -251,14 +256,13 @@ describe('Tram-One', () => { // focus on the input (the range input defaults to 0) userEvent.click(getByLabelText(container, 'Store Generator')); - // verify that the element has focus (before we start changing text) + // verify that the element has focus (before changing the value) await waitFor(() => { expect(getByLabelText(container, 'Store Generator')).toHaveFocus(); }); - // hit the right arrow key to increment the value + // change the value of the input fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 1 } }); - // userEvent.type(getByLabelText(container, 'Store Generator'), '{arrowright}'); // verify the element has the new value expect(getByLabelText(container, 'Store Generator')).toHaveValue('1'); @@ -270,14 +274,103 @@ describe('Tram-One', () => { }); }); - it('should clean up stores for elements that are no longer rendered', async () => { + it('should not reset stores for elements that are still rendered', async () => { + // start the app + const { container } = await startAppAndWait(); + + // previously state would be blown away if a parent element changed state multiple + + // focus on the input (the range input defaults to 0) + userEvent.click(getByLabelText(container, 'Store Generator')); + + // change the value of the input + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 1 } }); + + // click on one of the new stores several times + userEvent.click(getByText(container, '[0: 0]')); + userEvent.click(getByText(container, '[0: 1]')); + userEvent.click(getByText(container, '[0: 2]')); + userEvent.click(getByText(container, '[0: 3]')); + // the button should now say "[0: 4]" + expect(getByText(container, '[0: 4]')).toBeVisible(); + + // update the number of stores (the parent store element) + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 2 } }); + + // wait for mutation observer clean up removed stores + await waitFor(() => { + // we should see the new buttons + expect(getByText(container, '[1: 0]')).toBeVisible(); + }); + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 3 } }); + // wait for mutation observer clean up removed stores + await waitFor(() => { + // we should see the new buttons + expect(getByText(container, '[2: 0]')).toBeVisible(); + }); + + // we should still see the button with "4," + expect(getByText(container, '[0: 4]')).toBeVisible(); + }); + + it('should reset stores for elements that have been removed', async () => { // start the app - const { container } = startApp(); + const { container } = await startAppAndWait(); - // previously stores made for elements that had been removed stayed in the tram-observable-store + // previously state would be blown away if a parent element changed state multiple - // FOR JESSE, DEBUGGING NOTES: - // window['tram-space']['tram-observable-store'] - // look at target, notice that it grows, even when the stores should have been removed + // focus on the input (the range input defaults to 0) + userEvent.click(getByLabelText(container, 'Store Generator')); + + // change the value of the input + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 5 } }); + + // expect to see all the stores with their initial values + await waitFor(() => { + expect(getByText(container, '[0: 0]')).toBeVisible(); + expect(getByText(container, '[1: 0]')).toBeVisible(); + expect(getByText(container, '[2: 0]')).toBeVisible(); + expect(getByText(container, '[3: 0]')).toBeVisible(); + expect(getByText(container, '[4: 0]')).toBeVisible(); + }); + + // click on each of the new stores + userEvent.click(getByText(container, '[0: 0]')); + userEvent.click(getByText(container, '[1: 0]')); + userEvent.click(getByText(container, '[2: 0]')); + userEvent.click(getByText(container, '[3: 0]')); + userEvent.click(getByText(container, '[4: 0]')); + + // expect to see all the stores with the new values + await waitFor(() => { + expect(getByText(container, '[0: 1]')).toBeVisible(); + expect(getByText(container, '[1: 1]')).toBeVisible(); + expect(getByText(container, '[2: 1]')).toBeVisible(); + expect(getByText(container, '[3: 1]')).toBeVisible(); + expect(getByText(container, '[4: 1]')).toBeVisible(); + }); + + // remove all of the stores by setting the value to 0 + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 0 } }); + + await waitFor(() => { + expect(queryByText(container, '[0: 1]')).toBe(null); + expect(queryByText(container, '[1: 1]')).toBe(null); + expect(queryByText(container, '[2: 1]')).toBe(null); + expect(queryByText(container, '[3: 1]')).toBe(null); + expect(queryByText(container, '[4: 1]')).toBe(null); + }); + + // re-add the stores by setting the value to 5 + fireEvent.change(getByLabelText(container, 'Store Generator'), { target: { value: 5 } }); + + // expect to see all the stores with their initial values + await waitFor(() => { + expect(getByText(container, '[0: 0]')).toBeVisible(); + expect(getByText(container, '[1: 0]')).toBeVisible(); + expect(getByText(container, '[2: 0]')).toBeVisible(); + expect(getByText(container, '[3: 0]')).toBeVisible(); + expect(getByText(container, '[4: 0]')).toBeVisible(); + }); }); }); diff --git a/integration-tests/test-app/element-store-generator.ts b/integration-tests/test-app/element-store-generator.ts index ce6163e..2b63d80 100644 --- a/integration-tests/test-app/element-store-generator.ts +++ b/integration-tests/test-app/element-store-generator.ts @@ -15,7 +15,7 @@ const elementstoregenerator: TramOneComponent = () => { storeGeneratorStore.count = parseInt(inputElement.value); }; const storeElements = [...new Array(storeGeneratorStore.count)].map((_, index) => { - return html``; + return html``; }); return html`
diff --git a/integration-tests/test-app/element-with-store.ts b/integration-tests/test-app/element-with-store.ts index 016d5b9..b392151 100644 --- a/integration-tests/test-app/element-with-store.ts +++ b/integration-tests/test-app/element-with-store.ts @@ -5,9 +5,10 @@ const html = registerHtml(); /** * Dynamicly generated component that could possibly cause memory leaks */ -const elementwithstore: TramOneComponent = ({ value }) => { - const subElementStore = useStore({ active: value }); - return html` ${subElementStore.active}, `; +const elementwithstore: TramOneComponent = ({ index }) => { + const subElementStore = useStore({ count: 0 }); + const onIncrement = () => subElementStore.count++; + return html` `; }; export default elementwithstore; diff --git a/integration-tests/test-app/index.ts b/integration-tests/test-app/index.ts index bfd4aaf..c3b313f 100644 --- a/integration-tests/test-app/index.ts +++ b/integration-tests/test-app/index.ts @@ -9,6 +9,7 @@ import tasks from './tasks'; import mirrorinput from './mirror-input'; import documentTitleSetter from './document-title-setter'; import elementstoregenerator from './element-store-generator'; +import { TramWindow } from '../../src/types'; const html = registerHtml({ title: title, @@ -70,6 +71,12 @@ export const startApp = (container: any) => { window.document.body.appendChild(appContainer); } + // remove all existing state in the tram-space (since the app does not run in an isolated way) + // TODO we may need to carry this logic over to prevent issues in Hot-Reloading + Object.keys((window as unknown as TramWindow)['tram-space'] || {}).forEach((globalStore) => { + delete (window as unknown as TramWindow)['tram-space'][globalStore]; + }); + start(app, appContainer); return { diff --git a/integration-tests/test-helpers.ts b/integration-tests/test-helpers.ts new file mode 100644 index 0000000..f566609 --- /dev/null +++ b/integration-tests/test-helpers.ts @@ -0,0 +1,19 @@ +import { TramWindow } from '../src/types'; + +const { waitFor } = require('@testing-library/dom'); +const { startApp } = require('./test-app'); + +/** + * decorated startApp function that ensures that the app's mutation observers + * have kicked in before starting to interact with the app + */ +export const startAppAndWait = async () => { + const app = startApp(); + + await waitFor(() => { + // this waitFor is required to have the initial mutation observer trigger + expect(Object.keys((window as unknown as TramWindow)['tram-space']['tram-key-store']).length).toBeGreaterThan(0); + }); + + return app; +}; diff --git a/integration-tests/warnings.test.js b/integration-tests/warnings.test.js index 7746768..bdc6d7a 100644 --- a/integration-tests/warnings.test.js +++ b/integration-tests/warnings.test.js @@ -8,11 +8,6 @@ const { startApp: startBrokenApp } = require('./broken-app'); */ describe('Tram-One', () => { - beforeEach(() => { - // clean up any tram-one properties between tests - window['tram-space'] = undefined; - }); - it('should warn if selector is not found', () => { expect(() => startApp('#app')).toThrowError('Tram-One: could not find target, is the element on the page yet?'); }); diff --git a/src/dom.ts b/src/dom.ts index 973aeab..63c9cb8 100644 --- a/src/dom.ts +++ b/src/dom.ts @@ -11,7 +11,7 @@ import { restoreWorkingKey, } from './working-key'; import observeTag from './observe-tag'; -import processEffects from './process-effects'; +import processHooks from './process-hooks'; import { TRAM_TAG, TRAM_TAG_NEW_EFFECTS, TRAM_TAG_CLEANUP_EFFECTS } from './node-names'; import { Registry, Props, DOMTaggedTemplateFunction } from './types'; @@ -49,8 +49,8 @@ export const registerDom = (namespace: string | null, registry: Registry = {}): }; // observe store usage and process any new effects that were called when building the component - const processEffectsAndBuildTagResult = () => processEffects(populatedTagFunction); - const tagResult = observeTag(processEffectsAndBuildTagResult); + const processHooksAndBuildTagResult = () => processHooks(populatedTagFunction); + const tagResult = observeTag(processHooksAndBuildTagResult); // pop the branch off (since we are done rendering this component) popWorkingKeyBranch(TRAM_HOOK_KEY); diff --git a/src/effect-store.ts b/src/effect-store.ts index b7bab6c..716e1db 100644 --- a/src/effect-store.ts +++ b/src/effect-store.ts @@ -23,8 +23,8 @@ export const { * clear the effect store * usually called when we want to empty the effect store */ -export const clearEffectStore = (effectName: string) => { - const effectStore = getEffectStore(effectName); +export const clearEffectStore = (effectStoreName: string) => { + const effectStore = getEffectStore(effectStoreName); Object.keys(effectStore).forEach((key) => delete effectStore[key]); }; diff --git a/src/engine-names.ts b/src/engine-names.ts index 8fca56a..41cddd8 100644 --- a/src/engine-names.ts +++ b/src/engine-names.ts @@ -9,5 +9,7 @@ export const TRAM_HOOK_KEY = 'tram-hook-key'; export const TRAM_EFFECT_STORE = 'tram-effect-store'; export const TRAM_EFFECT_QUEUE = 'tram-effect-queue'; +export const TRAM_KEY_STORE = 'tram-key-store'; +export const TRAM_KEY_QUEUE = 'tram-key-queue'; export const TRAM_OBSERVABLE_STORE = 'tram-observable-store'; export const TRAM_MUTATION_OBSERVER = 'tram-mutation-observer'; diff --git a/src/key-queue.ts b/src/key-queue.ts new file mode 100644 index 0000000..cab61c8 --- /dev/null +++ b/src/key-queue.ts @@ -0,0 +1,31 @@ +/* + * The KeyQueue in Tram-One is a basic list of keys + * that needs to be persisted in the globalSpace. + * + * Currently this is used with useStore to keep track of what + * stores need to be associated with generated elements + */ + +import { buildNamespace } from './namespace'; + +const newDefaultKeyQueue = () => { + return [] as string[]; +}; + +export const { setup: setupKeyQueue, get: getKeyQueue, set: setKeyQueue } = buildNamespace(newDefaultKeyQueue); + +/** + * clear the key queue + * usually called when we want to empty the key queue + */ +export const clearKeyQueue = (keyQueueName: string) => { + const keyQueue = getKeyQueue(keyQueueName); + + keyQueue.splice(0, keyQueue.length); +}; + +/** + * restore the key queue to a previous value + * usually used when we had to interrupt the processing of keys + */ +export const restoreKeyQueue = setKeyQueue; diff --git a/src/key-store.ts b/src/key-store.ts new file mode 100644 index 0000000..f589e19 --- /dev/null +++ b/src/key-store.ts @@ -0,0 +1,48 @@ +/* + * The KeyStore in Tram-One is a basic key-value object + * that needs to be persisted in the globalSpace. + * + * Currently this is used with useStore and useGlobalStore to keep + * track of what stores need to be cleaned up when removing elements + */ + +import { buildNamespace } from './namespace'; +import { KeyObservers } from './types'; + +const newDefaultKeyStore = () => { + return {} as KeyObservers; +}; + +export const { setup: setupKeyStore, get: getKeyStore, set: setKeyStore } = buildNamespace(newDefaultKeyStore); + +/** + * clear the key store + * usually called when we want to empty the key store + */ +export const clearKeyStore = (keyStoreName: string) => { + const keyStore = getKeyStore(keyStoreName); + + Object.keys(keyStore).forEach((key) => delete keyStore[key]); +}; + +/** + * increment (or set initial value) for the keyStore + */ +export const incrementKeyStoreValue = (keyStoreName: string, key: string) => { + const keyStore = getKeyStore(keyStoreName); + keyStore[key] = keyStore[key] + 1 || 1; +}; + +/** + * decrement a value in the keyStore + */ +export const decrementKeyStoreValue = (keyStoreName: string, key: string) => { + const keyStore = getKeyStore(keyStoreName); + keyStore[key]--; +}; + +/** + * restore the key store to a previous value + * usually used when we had to interrupt the processing of keys + */ +export const restoreKeyStore = setKeyStore; diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index 42adad4..ce646fc 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -8,17 +8,37 @@ const { observe, unobserve } = require('@nx-js/observer-util'); -import { TRAM_TAG, TRAM_TAG_REACTION, TRAM_TAG_NEW_EFFECTS, TRAM_TAG_CLEANUP_EFFECTS } from './node-names'; +import { + TRAM_TAG, + TRAM_TAG_REACTION, + TRAM_TAG_NEW_EFFECTS, + TRAM_TAG_CLEANUP_EFFECTS, + TRAM_TAG_STORE_KEYS, +} from './node-names'; import { buildNamespace } from './namespace'; import { TramOneElement } from './types'; +import { getObservableStore } from './observable-store'; +import { TRAM_OBSERVABLE_STORE, TRAM_KEY_STORE } from './engine-names'; +import { decrementKeyStoreValue, getKeyStore, incrementKeyStoreValue } from './key-store'; -// process new effects for new nodes -const processEffects = (node: Node | TramOneElement) => { - // if this element doesn't have new effects, it is not be a Tram-One Element - if (!(TRAM_TAG_NEW_EFFECTS in node)) { +/** + * process new effects for new nodes + */ +const processHooks = (node: Node | TramOneElement) => { + // if this element doesn't have a TRAM_TAG, it's not a Tram-One Element + if (!(TRAM_TAG in node)) { return; } + const hasStoreKeys = node[TRAM_TAG_STORE_KEYS]; + + if (hasStoreKeys) { + // increment the usage of store keys in the key store (so we know an element is observing it) + node[TRAM_TAG_STORE_KEYS].forEach((key) => { + incrementKeyStoreValue(TRAM_KEY_STORE, key); + }); + } + const hasEffects = node[TRAM_TAG_NEW_EFFECTS]; if (hasEffects) { @@ -51,20 +71,48 @@ const processEffects = (node: Node | TramOneElement) => { } }; -// call all cleanup effects on the node +/** + * call all cleanup effects on the node + */ const cleanupEffects = (cleanupEffects: (() => void)[]) => { cleanupEffects.forEach((cleanup) => cleanup()); }; -// unobserve the reaction tied to the node, and run all cleanup effects for the node +/** + * remove the association of the store with this specific element + */ +const removeStoreKeyAssociation = (storeKeys: string[]) => { + storeKeys.forEach((storeKey) => { + decrementKeyStoreValue(TRAM_KEY_STORE, storeKey); + }); +}; + +/** + * remove any stores that no longer have any elements associated with them + * see removeStoreKeyAssociation above + */ +const cleanUpObservableStores = () => { + const observableStore = getObservableStore(TRAM_OBSERVABLE_STORE); + const keyStore = getKeyStore(TRAM_KEY_STORE); + Object.entries(keyStore).forEach(([key, observers]) => { + if (observers === 0) { + delete observableStore[key]; + } + }); +}; + +/** + * unobserve the reaction tied to the node, and run all cleanup effects for the node + */ const clearNode = (node: Node | TramOneElement) => { - // if this element doesn't have a Reaction, it is not a Tram-One Element + // if this element doesn't have a TRAM_TAG, it's not a Tram-One Element if (!(TRAM_TAG in node)) { return; } unobserve(node[TRAM_TAG_REACTION]); cleanupEffects(node[TRAM_TAG_CLEANUP_EFFECTS]); + removeStoreKeyAssociation(node[TRAM_TAG_STORE_KEYS]); }; const isTramOneComponent = (node: Node | TramOneElement) => { @@ -74,8 +122,9 @@ const isTramOneComponent = (node: Node | TramOneElement) => { return nodeIsATramOneComponent ? NodeFilter.FILTER_ACCEPT : NodeFilter.FILTER_SKIP; }; -// function to get the children (as a list) of the node passed in -// this only needs to query tram-one components, so we can use the attribute `tram` +/** + * function to get the children (as a list) of the node passed in + */ const childrenComponents = (node: Node | TramOneElement) => { const componentWalker = document.createTreeWalker(node, NodeFilter.SHOW_ELEMENT, isTramOneComponent); const children = []; @@ -90,15 +139,20 @@ const mutationObserverNamespaceConstructor = () => new MutationObserver((mutationList) => { // cleanup orphaned nodes that are no longer on the DOM const removedNodesInMutation = (mutation: MutationRecord) => [...mutation.removedNodes]; - const removedNodes = mutationList.flatMap(removedNodesInMutation).flatMap(childrenComponents); + const removedNodes = mutationList.flatMap(removedNodesInMutation); + const removedChildNodes = removedNodes.flatMap(childrenComponents); - removedNodes.forEach(clearNode); + removedChildNodes.forEach(clearNode); // call new effects on any new nodes const addedNodesInMutation = (mutation: MutationRecord) => [...mutation.addedNodes]; - const newNodes = mutationList.flatMap(addedNodesInMutation).flatMap(childrenComponents); + const newNodes = mutationList.flatMap(addedNodesInMutation); + const newChildNodes = newNodes.flatMap(childrenComponents); + + newChildNodes.forEach(processHooks); - newNodes.forEach(processEffects); + // clean up all local observable stores that have no observers + cleanUpObservableStores(); }); export const { setup: setupMutationObserver, get: getMutationObserver } = buildNamespace( diff --git a/src/node-names.ts b/src/node-names.ts index 1f6a41a..25213bb 100644 --- a/src/node-names.ts +++ b/src/node-names.ts @@ -8,5 +8,6 @@ export const TRAM_TAG = 'tram-tag'; export const TRAM_TAG_REACTION = 'tram-tag-reaction'; +export const TRAM_TAG_STORE_KEYS = 'tram-tag-store-keys'; export const TRAM_TAG_NEW_EFFECTS = 'tram-tag-new-effects'; export const TRAM_TAG_CLEANUP_EFFECTS = 'tram-tag-cleanup-effects'; diff --git a/src/observable-hook.ts b/src/observable-hook.ts index 41cf6c1..0de4583 100644 --- a/src/observable-hook.ts +++ b/src/observable-hook.ts @@ -1,8 +1,9 @@ -import { TRAM_OBSERVABLE_STORE, TRAM_HOOK_KEY } from './engine-names'; +import { TRAM_OBSERVABLE_STORE, TRAM_HOOK_KEY, TRAM_KEY_QUEUE } from './engine-names'; import { getObservableStore } from './observable-store'; import { getWorkingKeyValue, incrementWorkingKeyBranch } from './working-key'; import { StoreObject } from './types'; +import { getKeyQueue } from './key-queue'; /** * Shared source code for both observable hooks, useStore, and useGlobalStore. @@ -16,7 +17,7 @@ export default (key?: string, value?: Store): Store = const observableStore = getObservableStore(TRAM_OBSERVABLE_STORE); // increment the working key branch value - // this makes successive useEffects calls unique (until we reset the key) + // this makes successive hooks unique (until we reset the key) incrementWorkingKeyBranch(TRAM_HOOK_KEY); // if a key was passed in, use that, otherwise, generate a key @@ -32,6 +33,13 @@ export default (key?: string, value?: Store): Store = // get value for key const keyValue = observableStore[resolvedKey]; + // if we weren't passed in a key, this is a local obserable (not global), + const isLocalStore = !key; + if (isLocalStore) { + // if this is local, we should associate it with the element by putting it in the keyQueue + getKeyQueue(TRAM_KEY_QUEUE).push(resolvedKey); + } + // return value return keyValue; }; diff --git a/src/observe-tag.ts b/src/observe-tag.ts index b910021..cef5776 100644 --- a/src/observe-tag.ts +++ b/src/observe-tag.ts @@ -1,6 +1,6 @@ const { observe } = require('@nx-js/observer-util'); -import { TRAM_TAG_REACTION, TRAM_TAG_NEW_EFFECTS, TRAM_TAG_CLEANUP_EFFECTS } from './node-names'; +import { TRAM_TAG_REACTION, TRAM_TAG_NEW_EFFECTS, TRAM_TAG_CLEANUP_EFFECTS, TRAM_TAG } from './node-names'; import { TramOneElement, RemovedElementDataStore, Reaction, ElementPotentiallyWithSelectionAndFocus } from './types'; // functions to go to nodes or indices (made for .map) @@ -138,6 +138,9 @@ export default (tagFunction: () => TramOneElement): TramOneElement => { elementToGiveFocus.scrollTop = removedElementWithFocusData.scrollTop; } + // don't lose track that this is still a tram-one element + tagResult[TRAM_TAG] = true; + // copy the reaction and effects from the old tag to the new one tagResult[TRAM_TAG_REACTION] = oldTag[TRAM_TAG_REACTION]; tagResult[TRAM_TAG_NEW_EFFECTS] = oldTag[TRAM_TAG_NEW_EFFECTS]; diff --git a/src/process-effects.ts b/src/process-hooks.ts similarity index 51% rename from src/process-effects.ts rename to src/process-hooks.ts index 841f916..f0a95ac 100644 --- a/src/process-effects.ts +++ b/src/process-hooks.ts @@ -1,20 +1,25 @@ -import { TRAM_EFFECT_STORE, TRAM_EFFECT_QUEUE } from './engine-names'; -import { TRAM_TAG_NEW_EFFECTS } from './node-names'; +import { TRAM_EFFECT_STORE, TRAM_EFFECT_QUEUE, TRAM_KEY_QUEUE } from './engine-names'; +import { TRAM_TAG_NEW_EFFECTS, TRAM_TAG_STORE_KEYS } from './node-names'; import { getEffectStore, clearEffectStore, restoreEffectStore } from './effect-store'; import { TramOneElement } from './types'; +import { clearKeyQueue, getKeyQueue, restoreKeyQueue } from './key-queue'; /** * This is a helper function for the dom creation. - * This function stores any effects generated when building a tag in resulting node that is generated. + * This function stores any keys generated when building a tag in the resulting node that is generated. * * These are later processed by the mutation-observer, and cleaned up when the node is removed by the mutation-observer. + * + * This function is called every time state changes in an observable store */ export default (tagFunction: () => TramOneElement) => { - // save the existing effect queue for any components we are in the middle of building + // save the existing effect queue and key queue for any components we are in the middle of building const existingQueuedEffects = { ...getEffectStore(TRAM_EFFECT_QUEUE) }; + const existingQueuedKeys = [...getKeyQueue(TRAM_KEY_QUEUE)]; - // clear the effect queue (so we can listen for just new effects) + // clear the queues (so we can get just new effects and keys) clearEffectStore(TRAM_EFFECT_QUEUE); + clearKeyQueue(TRAM_KEY_QUEUE); // create the component, which will save new effects to the effect queue const tagResult = tagFunction(); @@ -23,12 +28,20 @@ export default (tagFunction: () => TramOneElement) => { const existingEffects = getEffectStore(TRAM_EFFECT_STORE); const queuedEffects = getEffectStore(TRAM_EFFECT_QUEUE); + // get all new keys + const newKeys = getKeyQueue(TRAM_KEY_QUEUE); + // store new effects in the node we just built + // TODO investigate if there are ever existing effects? const newEffects = Object.keys(queuedEffects).filter((effect) => !(effect in existingEffects)); tagResult[TRAM_TAG_NEW_EFFECTS] = newEffects.map((newEffectKey) => queuedEffects[newEffectKey]); - // restore the effect queue to what it was before we started + // store keys in the node we just built + tagResult[TRAM_TAG_STORE_KEYS] = newKeys; + + // restore the effect and key queues to what they were before we started restoreEffectStore(TRAM_EFFECT_QUEUE, existingQueuedEffects); + restoreKeyQueue(TRAM_KEY_QUEUE, existingQueuedKeys); return tagResult; }; diff --git a/src/start.ts b/src/start.ts index 4a1604b..5726dd8 100644 --- a/src/start.ts +++ b/src/start.ts @@ -6,6 +6,8 @@ import { TRAM_EFFECT_QUEUE, TRAM_OBSERVABLE_STORE, TRAM_MUTATION_OBSERVER, + TRAM_KEY_QUEUE, + TRAM_KEY_STORE, } from './engine-names'; import { setupTramOneSpace } from './namespace'; import { setupEffectStore } from './effect-store'; @@ -13,6 +15,8 @@ import { setupWorkingKey } from './working-key'; import { setupObservableStore } from './observable-store'; import { setupMutationObserver, startWatcher } from './mutation-observer'; import { ElementOrSelector, TramOneComponent } from './types'; +import { setupKeyQueue } from './key-queue'; +import { setupKeyStore } from './key-store'; /** * @name start @@ -32,6 +36,9 @@ export default (component: TramOneComponent, target: ElementOrSelector) => { // get the container to mount the app on const container = buildContainer(target); + // setup the window object to hold stores and queues + // in the future, we may allow this to be customized + // for multiple, sandboxed, instances of Tram-One setupTramOneSpace(); // setup store for effects @@ -46,6 +53,12 @@ export default (component: TramOneComponent, target: ElementOrSelector) => { // setup observable store for the useStore and useGlobalStore hooks setupObservableStore(TRAM_OBSERVABLE_STORE); + // setup key store for keeping track of stores to clean up + setupKeyStore(TRAM_KEY_STORE); + + // setup key queue for new observable stores when resolving mounts + setupKeyQueue(TRAM_KEY_QUEUE); + // setup a mutation observer for cleaning up removed elements and triggering effects setupMutationObserver(TRAM_MUTATION_OBSERVER); diff --git a/src/types.ts b/src/types.ts index afd7789..4a89fd5 100644 --- a/src/types.ts +++ b/src/types.ts @@ -1,4 +1,10 @@ -import { TRAM_TAG, TRAM_TAG_REACTION, TRAM_TAG_NEW_EFFECTS, TRAM_TAG_CLEANUP_EFFECTS } from './node-names'; +import { + TRAM_TAG, + TRAM_TAG_REACTION, + TRAM_TAG_NEW_EFFECTS, + TRAM_TAG_CLEANUP_EFFECTS, + TRAM_TAG_STORE_KEYS, +} from './node-names'; /* ============= PUBLIC TYPES ======================================== * A lot of the types here are wrapped using an array / index of 0. @@ -88,6 +94,7 @@ export interface TramOneElement extends Element { [TRAM_TAG_REACTION]: Reaction; [TRAM_TAG_NEW_EFFECTS]: Effect[]; [TRAM_TAG_CLEANUP_EFFECTS]: CleanupEffect[]; + [TRAM_TAG_STORE_KEYS]: string[]; } /* ============= INTERNAL TYPES ======================================== @@ -158,3 +165,10 @@ export interface ElementPotentiallyWithSelectionAndFocus extends Element { export interface EffectStore { [callLikeKey: string]: Effect; } + +/** + * Type for keeping track of the number of usages of a key + */ +export type KeyObservers = { + [key: string]: number; +}; diff --git a/src/working-key.ts b/src/working-key.ts index 71c3e26..5915827 100644 --- a/src/working-key.ts +++ b/src/working-key.ts @@ -8,16 +8,17 @@ import { WorkingkeyObject } from './types'; * values or effects to pull / trigger. */ -const defaultWorkingKey = { - // list of custom tags that we've stepped into - branch: [], - // map of branches to index value (used as a cursor for hooks) - branchIndices: { - '': 0, - }, -} as WorkingkeyObject; +const defaultWorkingKey = () => + ({ + // list of custom tags that we've stepped into + branch: [], + // map of branches to index value (used as a cursor for hooks) + branchIndices: { + '': 0, + }, + } as WorkingkeyObject); -export const { setup: setupWorkingKey, get: getWorkingKey } = buildNamespace(() => defaultWorkingKey); +export const { setup: setupWorkingKey, get: getWorkingKey } = buildNamespace(defaultWorkingKey); const getWorkingBranch = (keyName: string) => { const workingkeyObject = getWorkingKey(keyName); @@ -86,6 +87,7 @@ export const restoreWorkingKey = (keyName: string, restoreKey: WorkingkeyObject) key.branch = [...restoreKey.branch]; + // TODO investigate if this is doing what we need it to do const resetBranchValue = (branch: string) => { branches[branch] = restoreKey.branchIndices[branch] || 0; }; From 9efa357de7a086b04859abb21cdc5ccb4edb4f0d Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 22:05:20 -0400 Subject: [PATCH 05/17] fix type issues in performance tests --- performance-tests/test-app/element-switcher.ts | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/performance-tests/test-app/element-switcher.ts b/performance-tests/test-app/element-switcher.ts index 8180ce3..e3efa72 100644 --- a/performance-tests/test-app/element-switcher.ts +++ b/performance-tests/test-app/element-switcher.ts @@ -14,14 +14,15 @@ export default () => { const pageStore = useGlobalStore('STORE', { selected: 1 }); const switchStore = useStore({ id: 0 }); - const changeSelection = (newSelection) => () => { + const changeSelection = (newSelection: number) => () => { pageStore.selected = newSelection; }; const startAutoSwitch = () => { switchStore.id = setInterval(() => { ( - document.querySelector('nav button[selected] + button') || document.querySelector('nav button:first-of-type') + (document.querySelector('nav button[selected] + button') as HTMLButtonElement) || + (document.querySelector('nav button:first-of-type') as HTMLButtonElement) ).click(); }, 500); }; From 42c641b764e26ea745617caf4ba70cd468863ed2 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 22:52:34 -0400 Subject: [PATCH 06/17] remove extra await --- integration-tests/regression.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/integration-tests/regression.test.js b/integration-tests/regression.test.js index e27c8dc..0b76098 100644 --- a/integration-tests/regression.test.js +++ b/integration-tests/regression.test.js @@ -10,7 +10,7 @@ const { startAppAndWait } = require('./test-helpers'); describe('Tram-One', () => { it('should not call cleanups that are not functions', async () => { // start the app - const { container } = await await startAppAndWait(); + const { container } = await startAppAndWait(); // previously this would fail because the cleanup was called, // even though it was not a function, and instead was a promise (the result of an async function) From c08fa0f3e8f4fd775f9ca433441a8dc30d054ed3 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 22:58:06 -0400 Subject: [PATCH 07/17] fix some spacing --- integration-tests/internals.test.js | 1 + 1 file changed, 1 insertion(+) diff --git a/integration-tests/internals.test.js b/integration-tests/internals.test.js index 393dc5a..f00683f 100644 --- a/integration-tests/internals.test.js +++ b/integration-tests/internals.test.js @@ -54,6 +54,7 @@ describe('Tram-One', () => { await startAppAndWait(); // previously the working branch indices would have long recursive chains of branches + const workingKeyBranches = Object.keys(window['tram-space']['tram-hook-key'].branchIndices); // verify that top-level elements exist From d7e0ed4c0127a9714c863f1eb331f24fb4ecf676 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 23:23:37 -0400 Subject: [PATCH 08/17] update test descriptions --- integration-tests/regression.test.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/integration-tests/regression.test.js b/integration-tests/regression.test.js index 0b76098..baae48c 100644 --- a/integration-tests/regression.test.js +++ b/integration-tests/regression.test.js @@ -278,7 +278,7 @@ describe('Tram-One', () => { // start the app const { container } = await startAppAndWait(); - // previously state would be blown away if a parent element changed state multiple + // previously state would be blown away if a parent element changed state multiple times // focus on the input (the range input defaults to 0) userEvent.click(getByLabelText(container, 'Store Generator')); @@ -317,7 +317,7 @@ describe('Tram-One', () => { // start the app const { container } = await startAppAndWait(); - // previously state would be blown away if a parent element changed state multiple + // previously we would hold on to the local state of elements even if they had been removed // focus on the input (the range input defaults to 0) userEvent.click(getByLabelText(container, 'Store Generator')); From 9e8cffeb03ba3eb92a5d579dfe0bb6d68803a3c1 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 23:24:06 -0400 Subject: [PATCH 09/17] 12.0.0 --- package-lock.json | 4 ++-- package.json | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package-lock.json b/package-lock.json index 2a1021a..ede0f93 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "tram-one", - "version": "11.0.1", + "version": "12.0.0", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "tram-one", - "version": "11.0.1", + "version": "12.0.0", "license": "MIT", "dependencies": { "@nx-js/observer-util": "^4.2.2", diff --git a/package.json b/package.json index ac536e2..3323837 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "tram-one", - "version": "11.0.1", + "version": "12.0.0", "description": "🚋 Modern View Framework for Vanilla Javascript", "main": "dist/tram-one.cjs", "commonjs": "dist/tram-one.cjs", From 3c917ece5c17c053ac550d77cf32f9aa35187c13 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 23:48:42 -0400 Subject: [PATCH 10/17] remove TODOs --- integration-tests/test-app/index.ts | 1 - src/process-hooks.ts | 1 - src/working-key.ts | 1 - 3 files changed, 3 deletions(-) diff --git a/integration-tests/test-app/index.ts b/integration-tests/test-app/index.ts index c3b313f..25ab1be 100644 --- a/integration-tests/test-app/index.ts +++ b/integration-tests/test-app/index.ts @@ -72,7 +72,6 @@ export const startApp = (container: any) => { } // remove all existing state in the tram-space (since the app does not run in an isolated way) - // TODO we may need to carry this logic over to prevent issues in Hot-Reloading Object.keys((window as unknown as TramWindow)['tram-space'] || {}).forEach((globalStore) => { delete (window as unknown as TramWindow)['tram-space'][globalStore]; }); diff --git a/src/process-hooks.ts b/src/process-hooks.ts index f0a95ac..25afc85 100644 --- a/src/process-hooks.ts +++ b/src/process-hooks.ts @@ -32,7 +32,6 @@ export default (tagFunction: () => TramOneElement) => { const newKeys = getKeyQueue(TRAM_KEY_QUEUE); // store new effects in the node we just built - // TODO investigate if there are ever existing effects? const newEffects = Object.keys(queuedEffects).filter((effect) => !(effect in existingEffects)); tagResult[TRAM_TAG_NEW_EFFECTS] = newEffects.map((newEffectKey) => queuedEffects[newEffectKey]); diff --git a/src/working-key.ts b/src/working-key.ts index 5915827..be89d5e 100644 --- a/src/working-key.ts +++ b/src/working-key.ts @@ -87,7 +87,6 @@ export const restoreWorkingKey = (keyName: string, restoreKey: WorkingkeyObject) key.branch = [...restoreKey.branch]; - // TODO investigate if this is doing what we need it to do const resetBranchValue = (branch: string) => { branches[branch] = restoreKey.branchIndices[branch] || 0; }; From 28c6355db5c6bfb9590bd83056bb73e6a96e7176 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Sun, 8 May 2022 23:54:43 -0400 Subject: [PATCH 11/17] remove performance test page --- .../test-app/element-switcher-store.ts | 11 ---- .../test-app/element-switcher-sub-section.ts | 33 ----------- .../test-app/element-switcher.ts | 58 ------------------- performance-tests/test-app/index.ts | 3 - 4 files changed, 105 deletions(-) delete mode 100644 performance-tests/test-app/element-switcher-store.ts delete mode 100644 performance-tests/test-app/element-switcher-sub-section.ts delete mode 100644 performance-tests/test-app/element-switcher.ts diff --git a/performance-tests/test-app/element-switcher-store.ts b/performance-tests/test-app/element-switcher-store.ts deleted file mode 100644 index a32d46d..0000000 --- a/performance-tests/test-app/element-switcher-store.ts +++ /dev/null @@ -1,11 +0,0 @@ -const { registerHtml, useStore } = require('../../src/tram-one'); - -const html = registerHtml(); - -/** - * Dynamicly generated component that could possibly cause memory leaks - */ -export default () => { - const subElementStore = useStore({ active: 1 }); - return html` ${subElementStore.active}, `; -}; diff --git a/performance-tests/test-app/element-switcher-sub-section.ts b/performance-tests/test-app/element-switcher-sub-section.ts deleted file mode 100644 index dd76ebc..0000000 --- a/performance-tests/test-app/element-switcher-sub-section.ts +++ /dev/null @@ -1,33 +0,0 @@ -const { registerHtml, useGlobalStore, useEffect } = require('../../src/tram-one'); - -const html = registerHtml(); - -/** - * This component changes the presentation and document title based on global state - */ -export default () => { - const pageStore = useGlobalStore('STORE'); - useEffect(() => { - document.title = pageStore.selected; - }); - return html` -
- ${0 * pageStore.selected} - ${1 * pageStore.selected} - ${2 * pageStore.selected} - ${3 * pageStore.selected} - ${4 * pageStore.selected} - ${5 * pageStore.selected} - ${6 * pageStore.selected} - ${7 * pageStore.selected} - ${8 * pageStore.selected} - ${9 * pageStore.selected} - ${10 * pageStore.selected} - ${11 * pageStore.selected} - ${12 * pageStore.selected} - ${13 * pageStore.selected} - ${14 * pageStore.selected} - ${15 * pageStore.selected} -
- `; -}; diff --git a/performance-tests/test-app/element-switcher.ts b/performance-tests/test-app/element-switcher.ts deleted file mode 100644 index e3efa72..0000000 --- a/performance-tests/test-app/element-switcher.ts +++ /dev/null @@ -1,58 +0,0 @@ -const { registerHtml, useGlobalStore, useStore } = require('../../src/tram-one'); -import ElementSwitcher from './element-switcher-sub-section'; -import ElementStore from './element-switcher-store'; - -const html = registerHtml({ - ElementSwitcher, - ElementStore, -}); - -/** - * This page controls and switches rendering based on what's selected - */ -export default () => { - const pageStore = useGlobalStore('STORE', { selected: 1 }); - const switchStore = useStore({ id: 0 }); - - const changeSelection = (newSelection: number) => () => { - pageStore.selected = newSelection; - }; - - const startAutoSwitch = () => { - switchStore.id = setInterval(() => { - ( - (document.querySelector('nav button[selected] + button') as HTMLButtonElement) || - (document.querySelector('nav button:first-of-type') as HTMLButtonElement) - ).click(); - }, 500); - }; - - const stopAutoSwitch = () => { - clearInterval(switchStore.id); - switchStore.id = 0; - }; - - const cycle = switchStore.id - ? html`` - : html``; - - const storeElements = [...new Array(pageStore.selected)].map(() => { - return html``; - }); - - return html` -
-

Element Switching Example

- ${cycle} - - ${storeElements} - -
- `; -}; diff --git a/performance-tests/test-app/index.ts b/performance-tests/test-app/index.ts index 4f33afb..b5d05a8 100644 --- a/performance-tests/test-app/index.ts +++ b/performance-tests/test-app/index.ts @@ -2,11 +2,9 @@ const useUrlParams = require('use-url-params'); import { registerHtml, start } from '../../src/tram-one'; import elementRendering from './element-rendering'; -import elementSwitcher from './element-switcher'; const html = registerHtml({ 'element-rendering': elementRendering, - 'element-switcher': elementSwitcher, }); /** @@ -14,7 +12,6 @@ const html = registerHtml({ */ export const app = () => { if (useUrlParams('/element-rendering').matches) return html`
`; - if (useUrlParams('/element-switcher').matches) return html`
`; return html`

Performance Test App

From f40272fa5c4674aeda35a7a7659ad955a2c38e3d Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 00:16:20 -0400 Subject: [PATCH 12/17] remove unused functions from key-store --- src/key-store.ts | 16 ---------------- 1 file changed, 16 deletions(-) diff --git a/src/key-store.ts b/src/key-store.ts index f589e19..de13f78 100644 --- a/src/key-store.ts +++ b/src/key-store.ts @@ -15,16 +15,6 @@ const newDefaultKeyStore = () => { export const { setup: setupKeyStore, get: getKeyStore, set: setKeyStore } = buildNamespace(newDefaultKeyStore); -/** - * clear the key store - * usually called when we want to empty the key store - */ -export const clearKeyStore = (keyStoreName: string) => { - const keyStore = getKeyStore(keyStoreName); - - Object.keys(keyStore).forEach((key) => delete keyStore[key]); -}; - /** * increment (or set initial value) for the keyStore */ @@ -40,9 +30,3 @@ export const decrementKeyStoreValue = (keyStoreName: string, key: string) => { const keyStore = getKeyStore(keyStoreName); keyStore[key]--; }; - -/** - * restore the key store to a previous value - * usually used when we had to interrupt the processing of keys - */ -export const restoreKeyStore = setKeyStore; From f9182100fa383781b722b4eaf6c1a7f8cf8ff288 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 00:19:41 -0400 Subject: [PATCH 13/17] reanme function, new comment --- src/mutation-observer.ts | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index ce646fc..e5dee3d 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -22,9 +22,10 @@ import { TRAM_OBSERVABLE_STORE, TRAM_KEY_STORE } from './engine-names'; import { decrementKeyStoreValue, getKeyStore, incrementKeyStoreValue } from './key-store'; /** - * process new effects for new nodes + * process side-effects for new tram-one nodes + * (this includes calling useEffects, and keeping track of stores) */ -const processHooks = (node: Node | TramOneElement) => { +const processTramTags = (node: Node | TramOneElement) => { // if this element doesn't have a TRAM_TAG, it's not a Tram-One Element if (!(TRAM_TAG in node)) { return; @@ -149,7 +150,7 @@ const mutationObserverNamespaceConstructor = () => const newNodes = mutationList.flatMap(addedNodesInMutation); const newChildNodes = newNodes.flatMap(childrenComponents); - newChildNodes.forEach(processHooks); + newChildNodes.forEach(processTramTags); // clean up all local observable stores that have no observers cleanUpObservableStores(); From 5618fd384fd92cb15923fc672e79687e280e2afb Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 00:20:26 -0400 Subject: [PATCH 14/17] better comment --- src/mutation-observer.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index e5dee3d..edd7fd9 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -23,7 +23,7 @@ import { decrementKeyStoreValue, getKeyStore, incrementKeyStoreValue } from './k /** * process side-effects for new tram-one nodes - * (this includes calling useEffects, and keeping track of stores) + * (this includes calling effects, and keeping track of stores) */ const processTramTags = (node: Node | TramOneElement) => { // if this element doesn't have a TRAM_TAG, it's not a Tram-One Element From c9f9a5a1dc03b3695fae411cb37085582d7f427e Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 00:23:08 -0400 Subject: [PATCH 15/17] better comment for incrementKeyStoreValue --- src/mutation-observer.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index edd7fd9..59ca655 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -34,7 +34,8 @@ const processTramTags = (node: Node | TramOneElement) => { const hasStoreKeys = node[TRAM_TAG_STORE_KEYS]; if (hasStoreKeys) { - // increment the usage of store keys in the key store (so we know an element is observing it) + // for every store associated with this element, increment the count + // - this ensures that it doesn't get blown away when we clean up old stores node[TRAM_TAG_STORE_KEYS].forEach((key) => { incrementKeyStoreValue(TRAM_KEY_STORE, key); }); From e7e184ce0a9bf2c2294d378cce950d893c1cf0e0 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 00:29:30 -0400 Subject: [PATCH 16/17] update type comment --- src/types.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/types.ts b/src/types.ts index 4a89fd5..0ad1b93 100644 --- a/src/types.ts +++ b/src/types.ts @@ -167,7 +167,7 @@ export interface EffectStore { } /** - * Type for keeping track of the number of usages of a key + * Type for keeping track of the number of observers for a store */ export type KeyObservers = { [key: string]: number; From 56f368f35ac73ed437cfcf78e4021849a5ff5cf9 Mon Sep 17 00:00:00 2001 From: JRJurman Date: Mon, 9 May 2022 19:23:34 -0400 Subject: [PATCH 17/17] also delete entries from keyStore --- src/mutation-observer.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/src/mutation-observer.ts b/src/mutation-observer.ts index 59ca655..cdb438b 100644 --- a/src/mutation-observer.ts +++ b/src/mutation-observer.ts @@ -99,6 +99,7 @@ const cleanUpObservableStores = () => { Object.entries(keyStore).forEach(([key, observers]) => { if (observers === 0) { delete observableStore[key]; + delete keyStore[key]; } }); };