mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-30 10:36:35 +02:00
89 lines
3.0 KiB
TypeScript
89 lines
3.0 KiB
TypeScript
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||
|
import { ListItem, ListItemType } from '../types';
|
||
|
import { isFolderSelected, isTagSelected } from '@joplin/lib/components/shared/side-menu-shared';
|
||
|
import { Dispatch } from 'redux';
|
||
|
const { ALL_NOTES_FILTER_ID } = require('@joplin/lib/reserved-ids');
|
||
|
|
||
|
|
||
|
interface Props {
|
||
|
dispatch: Dispatch;
|
||
|
listItems: ListItem[];
|
||
|
|
||
|
notesParentType: string;
|
||
|
selectedTagId: string;
|
||
|
selectedFolderId: string;
|
||
|
selectedSmartFilterId: string;
|
||
|
}
|
||
|
|
||
|
const useSelectedSidebarIndex = (props: Props) => {
|
||
|
const appStateSelectedIndex = useMemo(() => {
|
||
|
for (let i = 0; i < props.listItems.length; i++) {
|
||
|
const listItem = props.listItems[i];
|
||
|
|
||
|
let selected = false;
|
||
|
if (listItem.kind === ListItemType.AllNotes) {
|
||
|
selected = props.selectedSmartFilterId === ALL_NOTES_FILTER_ID && props.notesParentType === 'SmartFilter';
|
||
|
} else if (listItem.kind === ListItemType.Header || listItem.kind === ListItemType.Spacer) {
|
||
|
selected = false;
|
||
|
} else if (listItem.kind === ListItemType.Folder) {
|
||
|
selected = isFolderSelected(listItem.folder, { selectedFolderId: props.selectedFolderId, notesParentType: props.notesParentType });
|
||
|
} else if (listItem.kind === ListItemType.Tag) {
|
||
|
selected = isTagSelected(listItem.tag, { selectedTagId: props.selectedTagId, notesParentType: props.notesParentType });
|
||
|
} else {
|
||
|
const exhaustivenessCheck: never = listItem;
|
||
|
return exhaustivenessCheck;
|
||
|
}
|
||
|
|
||
|
if (selected) {
|
||
|
return i;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}, [props.listItems, props.selectedFolderId, props.selectedTagId, props.selectedSmartFilterId, props.notesParentType]);
|
||
|
|
||
|
// Not all list items correspond with selectable Joplin folders/tags, but we want to
|
||
|
// be able to select them anyway. This is handled with selectedIndexOverride.
|
||
|
//
|
||
|
// When selectedIndexOverride >= 0, it corresponds to the index of a selected item with no
|
||
|
// specific note parent item (e.g. a header).
|
||
|
const [selectedIndexOverride, setSelectedIndexOverride] = useState(-1);
|
||
|
useEffect(() => {
|
||
|
setSelectedIndexOverride(-1);
|
||
|
}, [appStateSelectedIndex]);
|
||
|
|
||
|
const updateSelectedIndex = useCallback((newIndex: number) => {
|
||
|
if (newIndex < 0) {
|
||
|
newIndex = 0;
|
||
|
} else if (newIndex >= props.listItems.length) {
|
||
|
newIndex = props.listItems.length - 1;
|
||
|
}
|
||
|
|
||
|
const newItem = props.listItems[newIndex];
|
||
|
let newOverrideIndex = -1;
|
||
|
if (newItem.kind === ListItemType.AllNotes) {
|
||
|
props.dispatch({
|
||
|
type: 'SMART_FILTER_SELECT',
|
||
|
id: ALL_NOTES_FILTER_ID,
|
||
|
});
|
||
|
} else if (newItem.kind === ListItemType.Folder) {
|
||
|
props.dispatch({
|
||
|
type: 'FOLDER_SELECT',
|
||
|
id: newItem.folder.id,
|
||
|
});
|
||
|
} else if (newItem.kind === ListItemType.Tag) {
|
||
|
props.dispatch({
|
||
|
type: 'TAG_SELECT',
|
||
|
id: newItem.tag.id,
|
||
|
});
|
||
|
} else {
|
||
|
newOverrideIndex = newIndex;
|
||
|
}
|
||
|
setSelectedIndexOverride(newOverrideIndex);
|
||
|
}, [props.listItems, props.dispatch]);
|
||
|
|
||
|
const selectedIndex = selectedIndexOverride === -1 ? appStateSelectedIndex : selectedIndexOverride;
|
||
|
return { selectedIndex, updateSelectedIndex };
|
||
|
};
|
||
|
|
||
|
export default useSelectedSidebarIndex;
|