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

Allow infinite dependent keys when using withOnyx #355

Merged
merged 6 commits into from
Sep 21, 2023
Merged
Show file tree
Hide file tree
Changes from 2 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: 18 additions & 6 deletions lib/withOnyx.js
Original file line number Diff line number Diff line change
Expand Up @@ -92,16 +92,21 @@ export default function (mapOnyxToState, shouldDelayUpdates = false) {
this.checkEvictableKeys();
}

componentDidUpdate(prevProps, prevState) {
componentDidUpdate() {
// When the state is passed to the key functions with Str.result(), omit anything
// from state that was not part of the mapped keys.
const mappingPropNames = _.keys(mapOnyxToState);
const stateOnlyWithOnyxData = _.pick(this.state, mappingPropNames);

// If any of the mappings use data from the props, then when the props change, all the
// connections need to be reconnected with the new props
_.each(mapOnyxToState, (mapping, propertyName) => {
const previousKey = Str.result(mapping.key, {...prevProps, ...prevState});
const newKey = Str.result(mapping.key, {...this.props, ...this.state});
_.each(mapOnyxToState, (mapping, propName) => {
const previousKey = mapping.previousKey;
const newKey = Str.result(mapping.key, {...this.props, ...stateOnlyWithOnyxData});
if (previousKey !== newKey) {
Onyx.disconnect(this.activeConnectionIDs[previousKey], previousKey);
delete this.activeConnectionIDs[previousKey];
this.connectMappingToOnyx(mapping, propertyName);
this.connectMappingToOnyx(mapping, propName);
}
});
this.checkEvictableKeys();
Expand Down Expand Up @@ -238,7 +243,14 @@ export default function (mapOnyxToState, shouldDelayUpdates = false) {
* component
*/
connectMappingToOnyx(mapping, statePropertyName) {
const key = Str.result(mapping.key, {...this.props, ...this.state});
const mappingPropNames = _.keys(mapOnyxToState);
const stateOnlyWithOnyxData = _.pick(this.state, mappingPropNames);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would be better to move this to another function since its being used at top as well.

const key = Str.result(mapping.key, {...this.props, ...stateOnlyWithOnyxData});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably, this can also go into that same function.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sure, good idea. I've updated it to DRY up this logic and move it into a simple function defined at the top of the module.


// Remember the previous key so that if it ever changes, the component will reconnect to Onyx
// in componentDidUpdate
// eslint-disable-next-line no-param-reassign
mapOnyxToState[statePropertyName].previousKey = key;

// eslint-disable-next-line rulesdir/prefer-onyx-connect-in-libs
this.activeConnectionIDs[key] = Onyx.connect({
Expand Down
59 changes: 50 additions & 9 deletions tests/unit/withOnyxTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,10 @@ const ONYX_KEYS = {
TEST_KEY: 'test',
COLLECTION: {
TEST_KEY: 'test_',
RELATED_KEY: 'related_',
STATIC: 'static_',
DEPENDS_ON_STATIC: 'dependsOnStatic_',
DEPENDS_ON_DEPENDS_ON_STATIC: 'dependsOnDependsOnStatic_',
DEPENDS_ON_DEPENDS_ON_DEPENDS_ON_STATIC: 'dependsOnDependsOnDependsOnStatic_',
},
SIMPLE_KEY: 'simple',
SIMPLE_KEY_2: 'simple2',
Expand Down Expand Up @@ -242,27 +245,65 @@ describe('withOnyxTest', () => {
});

it('should pass a prop from one connected component to another', () => {
const collectionItemID = 1;
const onRender = jest.fn();
const markReadyForHydration = jest.fn();
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.TEST_KEY, {test_1: {id: 1}});
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.RELATED_KEY, {related_1: 'Test'});

// Given three collections with multiple items in each
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.STATIC, {
static_1: {name: 'Static 1', id: 1},
static_2: {name: 'Static 2', id: 2},
});

// And one collection will depends on data being loaded from the static collection
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.DEPENDS_ON_STATIC, {
dependsOnStatic_1: {name: 'dependsOnStatic 1', id: 3},
dependsOnStatic_2: {name: 'dependsOnStatic 2', id: 4},
});

// And one collection will depend on the data being loaded from the collection that depends on the static collection (multiple nested dependencies)
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.DEPENDS_ON_DEPENDS_ON_STATIC, {
dependsOnDependsOnStatic_3: {name: 'dependsOnDependsOnStatic 1', id: 5},
dependsOnDependsOnStatic_4: {name: 'dependsOnDependsOnStatic 2', id: 6},
});

// And another collection with one more layer of dependency just to prove it works
Onyx.mergeCollection(ONYX_KEYS.COLLECTION.DEPENDS_ON_DEPENDS_ON_DEPENDS_ON_STATIC, {
dependsOnDependsOnDependsOnStatic_5: {name: 'dependsOnDependsOnDependsOnStatic 1'},
dependsOnDependsOnDependsOnStatic_6: {name: 'dependsOnDependsOnDependsOnStatic 2'},
});

// When a component is rendered using withOnyx and several nested dependencies on the keys
return waitForPromisesToResolve()
.then(() => {
const TestComponentWithOnyx = withOnyx({
testObject: {
key: `${ONYX_KEYS.COLLECTION.TEST_KEY}${collectionItemID}`,
staticObject: {
key: `${ONYX_KEYS.COLLECTION.STATIC}1`,
},
testThing: {
key: ({testObject}) => `${ONYX_KEYS.COLLECTION.RELATED_KEY}${(testObject && testObject.id) || 0}`,
dependentObject: {
key: ({staticObject}) => `${ONYX_KEYS.COLLECTION.DEPENDS_ON_STATIC}${(staticObject && staticObject.id) || 0}`,
},
multiDependentObject: {
key: ({dependentObject}) => `${ONYX_KEYS.COLLECTION.DEPENDS_ON_DEPENDS_ON_STATIC}${(dependentObject && dependentObject.id) || 0}`,
},
extremeMultiDependentObject: {
key: ({multiDependentObject}) => `${ONYX_KEYS.COLLECTION.DEPENDS_ON_DEPENDS_ON_DEPENDS_ON_STATIC}${(multiDependentObject && multiDependentObject.id) || 0}`,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm lovin' it

},
})(ViewWithCollections);
render(<TestComponentWithOnyx markReadyForHydration={markReadyForHydration} onRender={onRender} />);
return waitForPromisesToResolve();
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I also found that the test wouldn't pass without this being added, which I think is normal. I'm not sure why it was removed as it existed when the test was originally written.

})

// Then all of the data gets properly loaded into the component as expected with the nested dependencies resolved
.then(() => {
expect(onRender).toHaveBeenLastCalledWith({
collections: {}, markReadyForHydration, onRender, testObject: {id: 1}, testThing: 'Test',
markReadyForHydration,
onRender,
collections: {},
testObject: {isDefaultProp: true},
staticObject: {name: 'Static 1', id: 1},
dependentObject: {name: 'dependsOnStatic 1', id: 3},
multiDependentObject: {name: 'dependsOnDependsOnStatic 1', id: 5},
extremeMultiDependentObject: {name: 'dependsOnDependsOnDependsOnStatic 1'},
});
});
});
Expand Down