A Redux-based application holds a forest-like data structure, like Folders and Documents in a file system, but with many root Folders, like this:
In a React Component, DocumentList we display all Documents (all leafs of all trees) and when clicking on one Document, we need access to the Document's root (Folder) object (to read some information from it)
Example: When a User clicks on Document "4", we need to get the corresponding Root Folder "Abc"
👉 How would you handle this (receiving the Root folder) in your Redux applications?
👇 Find below some more informations about the state types
The data is stored in a normalized Redux Store, so Folder/Document objects have a pointer back to their parent object:
Folder and Documents are stored in separate "tables" in the redux state (one state/slice!)
Redux State:
type DocumentAndFolderState = {
documents: Array<Document>;
foldersById: Record<string, Folder>;
}
The types for our domain model and the state could look like this:
type Folder {
id: string;
label: string;
slug: string;
parentId?: string; // not set if this is a root folder
}
type Document {
id: string
label: string
parentId: string; // every document is inside a folder
}
Please consider all code example as "concepts" / dummy code, rathern then working real code
If we want to retrieve a parent Folder of a Document or Folder:
const myObject: Folder|Document = ...;
const parentFolder = myObject.parentId ? state.foldersById[myObject.parentId] : null;
If we want to retrieve the root Folder for a Document or Folder you can recursively traverse from a leaf to the root:
function getRootFolder(state, document: Document) {
let parentId = document.parentId;
let parent;
while (parentId) {
parent = state.foldersById[parentId];
parentId = parent.parentId;
}
return parent;
}
Imagine, we have a React component that shows all our Documents. No further information from the store is needed until some user clicks one one the Documents. In that case the root folder for this single Document is needed.
Think of this example: by clicking a Document the user should be navigated (by changing the URL), to a page where the root Folder is displayed. The url is part of the Folder object (slug
). This information (slug
) is not needed until the user clicks on a Document in the list.
The List component is part of a page that allows allows adding/chageing/removing documents, so the Store could change, while the List component is visible.
My question is, how and when do we determine the root folder of a Document in that component.
The component might look like this, and I wonder what is the best way to get the root folder in handleClick
.
(Note that the code is simplified, read it more like a concept, than real code)
function SelectDocument() {
const documents = useSelector(state => state.documents);
function handleClick(document: Document) {
// for some reason we need the root folder for the document here
const rootFolder = ???;
// We don't want to CHANGE the folder, only need some information, for example
// to navigate to a URL:
history.push(rootFolder.slug);
}
return <ul>
{documents.map(d => <li key={d.id} onClick={d => handleClick(d)}>{d.label}</li>)}
</ul>
}
For this task I came up with the following options:
function SelectDocument() {
const documents = useSelector(state => Object.values(state.documentsById);
const documentIdsToRootFolders: Record<string, Folder> = useSelector(state => {
const result = {};
documents.forEach(d => {
result[d.id] = getRootFolder(state, d);
})
return result;
}); // assume shallow equals here
function handleClick(document: Document) {
// for some reason we need the root folder for the document here
const rootFolder = documentIdsToRootFolders[document.id];
}
- Pro: classic Redux way, things change, component re-renders has all data it needs
- Con: determining the root folders for all docs might be expensive and is done anytime anything changes in the store
- Variation: maybe we could improve by using a persistent selctor (re-select)
We could add a rootFolderId
property to our Document and Folder types. This way we could simplify the selector:
function SelectDocument() {
const documents = useSelector(state => Object.values(state.documentsById);
const documentIdsToRootFolders: Record<string, Folder> = useSelector(state => {
const result = {};
documents.forEach(d => {
result[d.id] = state.foldersById[d.rootFolderId]
})
return result;
}); // assume shallow equals here
function handleClick(document: Document) {
// same as option 1
const rootFolder = documentIdsToRootFolders[document.id];
}
- Pro: might speed things up (depending on the depth of the tree)
- Con: need to make sure the
rootFolderId
is always up-to-date (what happens if a Document is moved to another folder?) - Variation: only select all root folder (those where
parentId
isundefined
) in the selector, than find the one that is searched in handleClick (Array.find
or something similiar)
Instead of a rootFolderId
(as in Option 2) we might store this connection in the state itself:
type DocumentAndFolderState = {
documents: Array<Document>;
foldersById: Record<string, Folder>;
// cached value!
documentIdToRootFolderId: Record<string, string>;
}
- Pro: This would reduce the selector even more
- Con: as in Option 2, we have to make sure, that the cache is always up-to-date
We could select the list of Documents, but not the list of Root Documents. Instead we read it from the state only when the Document is clicked (and we really need the Document):
function SelectDocument() {
const documents = useSelector(state => Object.values(state.documentsById);
const store = useStore();
function handleClick(document: Document) {
// same as option 1
const rootFolder = getRootFolder(store.getState(), document);
}
- Pro: we determine the root folders only in case we really need to. Not reactive, but remember: none of the informations from the root component is needed for rendering.
- Con: not a typical Redux usage?!
- Which option do you like most?
- are there any more ideas?
- thanks a lot for your hints, comments and thoughts!
You could do a combination of 1 and 4.
showRootFolder
or something