280 lines
9.4 KiB
TypeScript
280 lines
9.4 KiB
TypeScript
import { KeyboardKey, KeyboardModifier } from '@/Services/IOService'
|
|
import { WebApplication } from '@/UIModels/Application'
|
|
import { AppState } from '@/UIModels/AppState'
|
|
import { PANEL_NAME_NOTES } from '@/Constants'
|
|
import { PrefKey, SystemViewId } from '@standardnotes/snjs'
|
|
import { observer } from 'mobx-react-lite'
|
|
import {
|
|
ChangeEventHandler,
|
|
FunctionComponent,
|
|
KeyboardEventHandler,
|
|
useCallback,
|
|
useEffect,
|
|
useMemo,
|
|
useRef,
|
|
useState,
|
|
} from 'react'
|
|
import ContentList from '@/Components/ContentListView/ContentList'
|
|
import NotesListOptionsMenu from '@/Components/ContentListView/NotesListOptionsMenu'
|
|
import NoAccountWarningWrapper from '@/Components/NoAccountWarning/NoAccountWarning'
|
|
import SearchOptions from '@/Components/SearchOptions/SearchOptions'
|
|
import PanelResizer, { PanelSide, ResizeFinishCallback, PanelResizeType } from '@/Components/PanelResizer/PanelResizer'
|
|
import { Disclosure, DisclosureButton, DisclosurePanel } from '@reach/disclosure'
|
|
import { useCloseOnBlur } from '@/Hooks/useCloseOnBlur'
|
|
|
|
type Props = {
|
|
application: WebApplication
|
|
appState: AppState
|
|
}
|
|
|
|
const ContentListView: FunctionComponent<Props> = ({ application, appState }) => {
|
|
const itemsViewPanelRef = useRef<HTMLDivElement>(null)
|
|
const displayOptionsMenuRef = useRef<HTMLDivElement>(null)
|
|
|
|
const {
|
|
completedFullSync,
|
|
noteFilterText,
|
|
optionsSubtitle,
|
|
panelTitle,
|
|
renderedItems,
|
|
setNoteFilterText,
|
|
searchBarElement,
|
|
selectNextItem,
|
|
selectPreviousItem,
|
|
onFilterEnter,
|
|
clearFilterText,
|
|
paginate,
|
|
panelWidth,
|
|
createNewNote,
|
|
} = appState.contentListView
|
|
|
|
const { selectedItems } = appState.selectedItems
|
|
|
|
const [showDisplayOptionsMenu, setShowDisplayOptionsMenu] = useState(false)
|
|
const [focusedSearch, setFocusedSearch] = useState(false)
|
|
|
|
const [closeDisplayOptMenuOnBlur] = useCloseOnBlur(displayOptionsMenuRef, setShowDisplayOptionsMenu)
|
|
|
|
const isFilesSmartView = useMemo(
|
|
() => appState.tags.selected?.uuid === SystemViewId.Files,
|
|
[appState.tags.selected?.uuid],
|
|
)
|
|
|
|
const addNewItem = useCallback(() => {
|
|
if (isFilesSmartView) {
|
|
void appState.files.uploadNewFile()
|
|
} else {
|
|
void createNewNote()
|
|
}
|
|
}, [appState.files, createNewNote, isFilesSmartView])
|
|
|
|
useEffect(() => {
|
|
/**
|
|
* In the browser we're not allowed to override cmd/ctrl + n, so we have to
|
|
* use Control modifier as well. These rules don't apply to desktop, but
|
|
* probably better to be consistent.
|
|
*/
|
|
const newNoteKeyObserver = application.io.addKeyObserver({
|
|
key: 'n',
|
|
modifiers: [KeyboardModifier.Meta, KeyboardModifier.Ctrl],
|
|
onKeyDown: (event) => {
|
|
event.preventDefault()
|
|
addNewItem()
|
|
},
|
|
})
|
|
|
|
const nextNoteKeyObserver = application.io.addKeyObserver({
|
|
key: KeyboardKey.Down,
|
|
elements: [document.body, ...(searchBarElement ? [searchBarElement] : [])],
|
|
onKeyDown: () => {
|
|
if (searchBarElement === document.activeElement) {
|
|
searchBarElement?.blur()
|
|
}
|
|
selectNextItem()
|
|
},
|
|
})
|
|
|
|
const previousNoteKeyObserver = application.io.addKeyObserver({
|
|
key: KeyboardKey.Up,
|
|
element: document.body,
|
|
onKeyDown: () => {
|
|
selectPreviousItem()
|
|
},
|
|
})
|
|
|
|
const searchKeyObserver = application.io.addKeyObserver({
|
|
key: 'f',
|
|
modifiers: [KeyboardModifier.Meta, KeyboardModifier.Shift],
|
|
onKeyDown: () => {
|
|
if (searchBarElement) {
|
|
searchBarElement.focus()
|
|
}
|
|
},
|
|
})
|
|
|
|
return () => {
|
|
newNoteKeyObserver()
|
|
nextNoteKeyObserver()
|
|
previousNoteKeyObserver()
|
|
searchKeyObserver()
|
|
}
|
|
}, [addNewItem, application.io, createNewNote, searchBarElement, selectNextItem, selectPreviousItem])
|
|
|
|
const onNoteFilterTextChange: ChangeEventHandler<HTMLInputElement> = useCallback(
|
|
(e) => {
|
|
setNoteFilterText(e.target.value)
|
|
},
|
|
[setNoteFilterText],
|
|
)
|
|
|
|
const onSearchFocused = useCallback(() => setFocusedSearch(true), [])
|
|
const onSearchBlurred = useCallback(() => setFocusedSearch(false), [])
|
|
|
|
const onNoteFilterKeyUp: KeyboardEventHandler = useCallback(
|
|
(e) => {
|
|
if (e.key === KeyboardKey.Enter) {
|
|
onFilterEnter()
|
|
}
|
|
},
|
|
[onFilterEnter],
|
|
)
|
|
|
|
const panelResizeFinishCallback: ResizeFinishCallback = useCallback(
|
|
(width, _lastLeft, _isMaxWidth, isCollapsed) => {
|
|
application.setPreference(PrefKey.NotesPanelWidth, width).catch(console.error)
|
|
appState.noteTags.reloadTagsContainerMaxWidth()
|
|
appState.panelDidResize(PANEL_NAME_NOTES, isCollapsed)
|
|
},
|
|
[appState, application],
|
|
)
|
|
|
|
const panelWidthEventCallback = useCallback(() => {
|
|
appState.noteTags.reloadTagsContainerMaxWidth()
|
|
}, [appState])
|
|
|
|
const toggleDisplayOptionsMenu = useCallback(() => {
|
|
setShowDisplayOptionsMenu(!showDisplayOptionsMenu)
|
|
}, [showDisplayOptionsMenu])
|
|
|
|
const addButtonLabel = useMemo(
|
|
() => (isFilesSmartView ? 'Upload file' : 'Create a new note in the selected tag'),
|
|
[isFilesSmartView],
|
|
)
|
|
|
|
return (
|
|
<div
|
|
id="items-column"
|
|
className="sn-component section app-column app-column-second"
|
|
aria-label={'Notes & Files'}
|
|
ref={itemsViewPanelRef}
|
|
>
|
|
<div className="content">
|
|
<div id="items-title-bar" className="section-title-bar">
|
|
<div id="items-title-bar-container">
|
|
<div className="section-title-bar-header">
|
|
<div className="sk-h2 font-semibold title">{panelTitle}</div>
|
|
<button
|
|
className="sk-button contrast wide"
|
|
title={addButtonLabel}
|
|
aria-label={addButtonLabel}
|
|
onClick={addNewItem}
|
|
>
|
|
<div className="sk-label">
|
|
<i className="ion-plus add-button" aria-hidden></i>
|
|
</div>
|
|
</button>
|
|
</div>
|
|
<div className="filter-section" role="search">
|
|
<div>
|
|
<input
|
|
type="text"
|
|
id="search-bar"
|
|
className="filter-bar"
|
|
placeholder="Search"
|
|
title="Searches notes and files in the currently selected tag"
|
|
value={noteFilterText}
|
|
onChange={onNoteFilterTextChange}
|
|
onKeyUp={onNoteFilterKeyUp}
|
|
onFocus={onSearchFocused}
|
|
onBlur={onSearchBlurred}
|
|
autoComplete="off"
|
|
/>
|
|
{noteFilterText && (
|
|
<button onClick={clearFilterText} id="search-clear-button">
|
|
✕
|
|
</button>
|
|
)}
|
|
</div>
|
|
|
|
{(focusedSearch || noteFilterText) && (
|
|
<div className="animate-fade-from-top">
|
|
<SearchOptions application={application} appState={appState} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
<NoAccountWarningWrapper appState={appState} />
|
|
</div>
|
|
<div id="items-menu-bar" className="sn-component" ref={displayOptionsMenuRef}>
|
|
<div className="sk-app-bar no-edges">
|
|
<div className="left">
|
|
<Disclosure open={showDisplayOptionsMenu} onChange={toggleDisplayOptionsMenu}>
|
|
<DisclosureButton
|
|
className={`sk-app-bar-item bg-contrast color-text border-0 focus:shadow-none ${
|
|
showDisplayOptionsMenu ? 'selected' : ''
|
|
}`}
|
|
onBlur={closeDisplayOptMenuOnBlur}
|
|
>
|
|
<div className="sk-app-bar-item-column">
|
|
<div className="sk-label">Options</div>
|
|
</div>
|
|
<div className="sk-app-bar-item-column">
|
|
<div className="sk-sublabel">{optionsSubtitle}</div>
|
|
</div>
|
|
</DisclosureButton>
|
|
<DisclosurePanel onBlur={closeDisplayOptMenuOnBlur}>
|
|
{showDisplayOptionsMenu && (
|
|
<NotesListOptionsMenu
|
|
application={application}
|
|
closeDisplayOptionsMenu={toggleDisplayOptionsMenu}
|
|
closeOnBlur={closeDisplayOptMenuOnBlur}
|
|
isOpen={showDisplayOptionsMenu}
|
|
/>
|
|
)}
|
|
</DisclosurePanel>
|
|
</Disclosure>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{completedFullSync && !renderedItems.length ? <p className="empty-items-list faded">No items.</p> : null}
|
|
{!completedFullSync && !renderedItems.length ? <p className="empty-items-list faded">Loading...</p> : null}
|
|
{renderedItems.length ? (
|
|
<ContentList
|
|
items={renderedItems}
|
|
selectedItems={selectedItems}
|
|
application={application}
|
|
appState={appState}
|
|
paginate={paginate}
|
|
/>
|
|
) : null}
|
|
</div>
|
|
{itemsViewPanelRef.current && (
|
|
<PanelResizer
|
|
collapsable={true}
|
|
hoverable={true}
|
|
defaultWidth={300}
|
|
panel={itemsViewPanelRef.current}
|
|
side={PanelSide.Right}
|
|
type={PanelResizeType.WidthOnly}
|
|
resizeFinishCallback={panelResizeFinishCallback}
|
|
widthEventCallback={panelWidthEventCallback}
|
|
width={panelWidth}
|
|
left={0}
|
|
/>
|
|
)}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default observer(ContentListView)
|