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:
parent
f32fe63205
commit
320d0df60d
@ -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 = {
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -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();
|
||||||
|
Loading…
Reference in New Issue
Block a user