1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-11 18:24:43 +02:00

Desktop, Mobile: Fixes #10674: Fix sidebar performance regression with many nested notebooks (#10676)

This commit is contained in:
Henry Heino 2024-07-04 05:56:57 -07:00 committed by GitHub
parent f32fe63205
commit 320d0df60d
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 106 additions and 47 deletions

View File

@ -1,7 +1,7 @@
import { useMemo } from 'react'; import { useMemo } from 'react';
import { FolderListItem, HeaderId, HeaderListItem, ListItem, ListItemType, TagListItem } from '../types'; import { FolderListItem, HeaderId, HeaderListItem, ListItem, ListItemType, TagListItem } from '../types';
import { FolderEntity, TagsWithNoteCountEntity } from '@joplin/lib/services/database/types'; import { FolderEntity, TagsWithNoteCountEntity } from '@joplin/lib/services/database/types';
import { renderFolders, renderTags } from '@joplin/lib/components/shared/side-menu-shared'; import { buildFolderTree, renderFolders, renderTags } from '@joplin/lib/components/shared/side-menu-shared';
import { _ } from '@joplin/lib/locale'; import { _ } from '@joplin/lib/locale';
import CommandService from '@joplin/lib/services/CommandService'; import CommandService from '@joplin/lib/services/CommandService';
import Setting from '@joplin/lib/models/Setting'; import Setting from '@joplin/lib/models/Setting';
@ -35,10 +35,13 @@ const useSidebarListData = (props: Props): ListItem[] => {
}); });
}, [props.tags]); }, [props.tags]);
const folderTree = useMemo(() => {
return buildFolderTree(props.folders);
}, [props.folders]);
const folderItems = useMemo(() => { const folderItems = useMemo(() => {
const renderProps = { const renderProps = {
folders: props.folders, folderTree,
collapsedFolderIds: props.collapsedFolderIds, collapsedFolderIds: props.collapsedFolderIds,
}; };
return renderFolders<ListItem>(renderProps, (folder, hasChildren, depth): FolderListItem => { return renderFolders<ListItem>(renderProps, (folder, hasChildren, depth): FolderListItem => {
@ -50,7 +53,7 @@ const useSidebarListData = (props: Props): ListItem[] => {
key: folder.id, key: folder.id,
}; };
}); });
}, [props.folders, props.collapsedFolderIds]); }, [folderTree, props.collapsedFolderIds]);
return useMemo(() => { return useMemo(() => {
const foldersHeader: HeaderListItem = { const foldersHeader: HeaderListItem = {

View File

@ -8,7 +8,7 @@ import Synchronizer from '@joplin/lib/Synchronizer';
import NavService from '@joplin/lib/services/NavService'; import NavService from '@joplin/lib/services/NavService';
import { _ } from '@joplin/lib/locale'; import { _ } from '@joplin/lib/locale';
import { ThemeStyle, themeStyle } from './global-style'; import { ThemeStyle, themeStyle } from './global-style';
import { isFolderSelected, renderFolders } from '@joplin/lib/components/shared/side-menu-shared'; import { buildFolderTree, isFolderSelected, renderFolders } from '@joplin/lib/components/shared/side-menu-shared';
import { FolderEntity, FolderIcon, FolderIconType } from '@joplin/lib/services/database/types'; import { FolderEntity, FolderIcon, FolderIconType } from '@joplin/lib/services/database/types';
import { AppState } from '../utils/types'; import { AppState } from '../utils/types';
import Setting from '@joplin/lib/models/Setting'; import Setting from '@joplin/lib/models/Setting';
@ -560,8 +560,16 @@ const SideMenuContentComponent = (props: Props) => {
items.push(renderSidebarButton('folder_header', _('Notebooks'), 'folder')); items.push(renderSidebarButton('folder_header', _('Notebooks'), 'folder'));
const folderTree = useMemo(() => {
return buildFolderTree(props.folders);
}, [props.folders]);
if (props.folders.length) { if (props.folders.length) {
const result = renderFolders(props, renderFolderItem); const result = renderFolders({
folderTree,
collapsedFolderIds: props.collapsedFolderIds,
}, renderFolderItem);
const folderItems = result.items; const folderItems = result.items;
items = items.concat(folderItems); items = items.concat(folderItems);
} }

View File

@ -1,6 +1,6 @@
import { FolderEntity } from '../../services/database/types'; import { FolderEntity } from '../../services/database/types';
import { getTrashFolder, getTrashFolderId } from '../../services/trash'; import { getTrashFolder, getTrashFolderId } from '../../services/trash';
import { renderFolders } from './side-menu-shared'; import { buildFolderTree, renderFolders } from './side-menu-shared';
const renderItem = (folder: FolderEntity, hasChildren: boolean, depth: number) => { const renderItem = (folder: FolderEntity, hasChildren: boolean, depth: number) => {
return [folder.id, hasChildren, depth]; return [folder.id, hasChildren, depth];
@ -86,8 +86,52 @@ describe('side-menu-shared', () => {
order: ['1', getTrashFolderId(), '2'], order: ['1', getTrashFolderId(), '2'],
}, },
], ],
])('should render folders', (props, expected) => {
const actual = renderFolders(props, renderItem); // Should not render id: 4 because it's contained within the child of a collapsed folder.
[
{
collapsedFolderIds: ['2'],
folders: [
{
id: '1',
parent_id: '',
deleted_time: 0,
},
{
id: '2',
parent_id: '',
deleted_time: 0,
},
{
id: '3',
parent_id: '2',
deleted_time: 0,
},
{
id: '4',
parent_id: '3',
deleted_time: 0,
},
getTrashFolder(),
],
notesParentType: 'Folder',
selectedFolderId: '',
selectedTagId: '',
},
{
items: [
['1', false, 0],
['2', true, 0],
[getTrashFolderId(), false, 0],
],
order: ['1', '2', getTrashFolderId()],
},
],
])('should render folders (case %#)', (props, expected) => {
const actual = renderFolders({
folderTree: buildFolderTree(props.folders),
collapsedFolderIds: props.collapsedFolderIds,
}, renderItem);
expect(actual).toEqual(expected); expect(actual).toEqual(expected);
}); });

View File

@ -1,39 +1,10 @@
import Folder from '../../models/Folder';
import BaseModel from '../../BaseModel';
import { FolderEntity, TagEntity, TagsWithNoteCountEntity } from '../../services/database/types'; import { FolderEntity, TagEntity, TagsWithNoteCountEntity } from '../../services/database/types';
import { getDisplayParentId, getTrashFolderId } from '../../services/trash'; import { getDisplayParentId } from '../../services/trash';
import { getCollator } from '../../models/utils/getCollator'; import { getCollator } from '../../models/utils/getCollator';
export type RenderFolderItem<T> = (folder: FolderEntity, hasChildren: boolean, depth: number)=> T; export type RenderFolderItem<T> = (folder: FolderEntity, hasChildren: boolean, depth: number)=> T;
export type RenderTagItem<T> = (tag: TagsWithNoteCountEntity)=> T; export type RenderTagItem<T> = (tag: TagsWithNoteCountEntity)=> T;
function folderHasChildren_(folders: FolderEntity[], folderId: string) {
if (folderId === getTrashFolderId()) {
return !!folders.find(f => !!f.deleted_time);
}
for (let i = 0; i < folders.length; i++) {
const folder = folders[i];
const folderParentId = getDisplayParentId(folder, folders.find(f => f.id === folder.parent_id));
if (folderParentId === folderId) return true;
}
return false;
}
function folderIsCollapsed(folders: FolderEntity[], folderId: string, collapsedFolderIds: string[]) {
if (!collapsedFolderIds || !collapsedFolderIds.length) return false;
while (true) {
const folder: FolderEntity = BaseModel.byId(folders, folderId);
if (!folder) throw new Error(`No folder with id ${folder.id}`);
const folderParentId = getDisplayParentId(folder, folders.find(f => f.id === folder.parent_id));
if (!folderParentId) return false;
if (collapsedFolderIds.indexOf(folderParentId) >= 0) return true;
folderId = folderParentId;
}
}
interface FolderSelectedContext { interface FolderSelectedContext {
selectedFolderId: string; selectedFolderId: string;
notesParentType: string; notesParentType: string;
@ -48,21 +19,36 @@ type ItemsWithOrder<ItemType> = {
order: string[]; order: string[];
}; };
interface RenderFoldersProps { interface FolderTree {
folders: FolderEntity[]; folders: FolderEntity[];
parentIdToChildren: Map<string, FolderEntity[]>;
idToItem: Map<string, FolderEntity>;
}
interface RenderFoldersProps {
folderTree: FolderTree;
collapsedFolderIds: string[]; collapsedFolderIds: string[];
} }
function folderIsCollapsed(context: RenderFoldersProps, folderId: string) {
if (!context.collapsedFolderIds || !context.collapsedFolderIds.length) return false;
while (true) {
const folder = context.folderTree.idToItem.get(folderId);
const folderParentId = getDisplayParentId(folder, context.folderTree.idToItem.get(folder.parent_id));
if (!folderParentId) return false;
if (context.collapsedFolderIds.includes(folderParentId)) return true;
folderId = folderParentId;
}
}
function renderFoldersRecursive_<T>(props: RenderFoldersProps, renderItem: RenderFolderItem<T>, items: T[], parentId: string, depth: number, order: string[]): ItemsWithOrder<T> { function renderFoldersRecursive_<T>(props: RenderFoldersProps, renderItem: RenderFolderItem<T>, items: T[], parentId: string, depth: number, order: string[]): ItemsWithOrder<T> {
const folders = props.folders; const folders = props.folderTree.parentIdToChildren.get(parentId ?? '') ?? [];
for (let i = 0; i < folders.length; i++) { const parentIdToChildren = props.folderTree.parentIdToChildren;
const folder = folders[i]; for (const folder of folders) {
if (folderIsCollapsed(props, folder.id)) continue;
const folderParentId = getDisplayParentId(folder, props.folders.find(f => f.id === folder.parent_id)); const hasChildren = parentIdToChildren.has(folder.id);
if (!Folder.idsEqual(folderParentId, parentId)) continue;
if (folderIsCollapsed(props.folders, folder.id, props.collapsedFolderIds)) continue;
const hasChildren = folderHasChildren_(folders, folder.id);
order.push(folder.id); order.push(folder.id);
items.push(renderItem(folder, hasChildren, depth)); items.push(renderItem(folder, hasChildren, depth));
if (hasChildren) { if (hasChildren) {
@ -81,6 +67,24 @@ export const renderFolders = <T> (props: RenderFoldersProps, renderItem: RenderF
return renderFoldersRecursive_(props, renderItem, [], '', 0, []); return renderFoldersRecursive_(props, renderItem, [], '', 0, []);
}; };
export const buildFolderTree = (folders: FolderEntity[]): FolderTree => {
const idToItem = new Map<string, FolderEntity>();
for (const folder of folders) {
idToItem.set(folder.id, folder);
}
const parentIdToChildren = new Map<string, FolderEntity[]>();
for (const folder of folders) {
const displayParentId = getDisplayParentId(folder, idToItem.get(folder.parent_id)) ?? '';
if (!parentIdToChildren.has(displayParentId)) {
parentIdToChildren.set(displayParentId, []);
}
parentIdToChildren.get(displayParentId).push(folder);
}
return { folders, parentIdToChildren, idToItem };
};
const sortTags = (tags: TagEntity[]) => { const sortTags = (tags: TagEntity[]) => {
tags = tags.slice(); tags = tags.slice();
const collator = getCollator(); const collator = getCollator();