Files
overleaf-cep/services/web/frontend/js/features/file-tree/components/file-tree-folder.tsx
David df03b5c8b4 Merge pull request #27389 from overleaf/dp-folder-click
Open folder on click of folder item

GitOrigin-RevId: 4af71533951ef183eec9dec4b4940470695f2f1d
2025-08-08 08:06:26 +00:00

106 lines
2.8 KiB
TypeScript

import { useCallback, useEffect } from 'react'
import classNames from 'classnames'
import {
useFileTreeSelectable,
useSelectableEntity,
} from '../contexts/file-tree-selectable'
import { useDroppable } from '../contexts/file-tree-draggable'
import FileTreeItemInner from './file-tree-item/file-tree-item-inner'
import FileTreeFolderList from './file-tree-folder-list'
import usePersistedState from '../../../shared/hooks/use-persisted-state'
import { Folder } from '../../../../../types/folder'
import { Doc } from '../../../../../types/doc'
import { FileRef } from '../../../../../types/file-ref'
import FileTreeFolderIcons from './file-tree-folder-icons'
import { useIsNewEditorEnabled } from '@/features/ide-redesign/utils/new-editor-utils'
function FileTreeFolder({
name,
id,
folders,
docs,
files,
}: {
name: string
id: string
folders: Folder[]
docs: Doc[]
files: FileRef[]
}) {
const newEditor = useIsNewEditorEnabled()
const { isSelected, props: selectableEntityProps } = useSelectableEntity(
id,
'folder'
)
const { selectedEntityParentIds } = useFileTreeSelectable()
const [expanded, setExpanded] = usePersistedState(
`folder.${id}.expanded`,
false
)
useEffect(() => {
if (selectedEntityParentIds.has(id)) {
setExpanded(true)
}
}, [id, selectedEntityParentIds, setExpanded])
const handleExpandCollapseClick = useCallback(() => {
setExpanded(expanded => !expanded)
}, [setExpanded])
const onClick = useCallback(() => {
if (newEditor) {
handleExpandCollapseClick()
}
}, [newEditor, handleExpandCollapseClick])
const { isOver: isOverRoot, dropRef: dropRefRoot } = useDroppable(id)
const { isOver: isOverList, dropRef: dropRefList } = useDroppable(id)
return (
<>
<li
// eslint-disable-next-line jsx-a11y/role-has-required-aria-props
role="treeitem"
// aria-selected is provided in selectableEntityProps
{...selectableEntityProps}
aria-expanded={expanded}
aria-label={name}
ref={dropRefRoot}
className={classNames(selectableEntityProps.className, {
'dnd-droppable-hover': isOverRoot || isOverList,
})}
translate="no"
>
<FileTreeItemInner
id={id}
name={name}
type="folder"
isSelected={isSelected}
onClick={onClick}
icons={
<FileTreeFolderIcons
expanded={expanded}
onExpandCollapseClick={handleExpandCollapseClick}
/>
}
/>
</li>
{expanded ? (
<FileTreeFolderList
folders={folders}
docs={docs}
files={files}
dropRef={dropRefList}
/>
) : null}
</>
)
}
export default FileTreeFolder