|
|
@ -1,5 +1,6 @@
|
|
|
|
import React, { useCallback, useMemo, useRef, useState } from 'react';
|
|
|
|
import React, { useCallback, useMemo, useRef, useState } from 'react';
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
|
|
|
|
import { SelectProvider } from 'App/SelectContext';
|
|
|
|
import { REFRESH_MOVIE, RSS_SYNC } from 'Commands/commandNames';
|
|
|
|
import { REFRESH_MOVIE, RSS_SYNC } from 'Commands/commandNames';
|
|
|
|
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
|
|
|
import LoadingIndicator from 'Components/Loading/LoadingIndicator';
|
|
|
|
import PageContent from 'Components/Page/PageContent';
|
|
|
|
import PageContent from 'Components/Page/PageContent';
|
|
|
@ -25,6 +26,7 @@ import scrollPositions from 'Store/scrollPositions';
|
|
|
|
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
|
|
|
import createCommandExecutingSelector from 'Store/Selectors/createCommandExecutingSelector';
|
|
|
|
import createDimensionsSelector from 'Store/Selectors/createDimensionsSelector';
|
|
|
|
import createDimensionsSelector from 'Store/Selectors/createDimensionsSelector';
|
|
|
|
import createMovieClientSideCollectionItemsSelector from 'Store/Selectors/createMovieClientSideCollectionItemsSelector';
|
|
|
|
import createMovieClientSideCollectionItemsSelector from 'Store/Selectors/createMovieClientSideCollectionItemsSelector';
|
|
|
|
|
|
|
|
import translate from 'Utilities/String/translate';
|
|
|
|
import MovieIndexFilterMenu from './Menus/MovieIndexFilterMenu';
|
|
|
|
import MovieIndexFilterMenu from './Menus/MovieIndexFilterMenu';
|
|
|
|
import MovieIndexSortMenu from './Menus/MovieIndexSortMenu';
|
|
|
|
import MovieIndexSortMenu from './Menus/MovieIndexSortMenu';
|
|
|
|
import MovieIndexViewMenu from './Menus/MovieIndexViewMenu';
|
|
|
|
import MovieIndexViewMenu from './Menus/MovieIndexViewMenu';
|
|
|
@ -33,6 +35,7 @@ import MovieIndexOverviews from './Overview/MovieIndexOverviews';
|
|
|
|
import MovieIndexOverviewOptionsModal from './Overview/Options/MovieIndexOverviewOptionsModal';
|
|
|
|
import MovieIndexOverviewOptionsModal from './Overview/Options/MovieIndexOverviewOptionsModal';
|
|
|
|
import MovieIndexPosters from './Posters/MovieIndexPosters';
|
|
|
|
import MovieIndexPosters from './Posters/MovieIndexPosters';
|
|
|
|
import MovieIndexPosterOptionsModal from './Posters/Options/MovieIndexPosterOptionsModal';
|
|
|
|
import MovieIndexPosterOptionsModal from './Posters/Options/MovieIndexPosterOptionsModal';
|
|
|
|
|
|
|
|
import MovieIndexSelectAllButton from './Select/MovieIndexSelectAllButton';
|
|
|
|
import MovieIndexTable from './Table/MovieIndexTable';
|
|
|
|
import MovieIndexTable from './Table/MovieIndexTable';
|
|
|
|
import MovieIndexTableOptions from './Table/MovieIndexTableOptions';
|
|
|
|
import MovieIndexTableOptions from './Table/MovieIndexTableOptions';
|
|
|
|
import styles from './MovieIndex.css';
|
|
|
|
import styles from './MovieIndex.css';
|
|
|
@ -53,7 +56,7 @@ interface MovieIndexProps {
|
|
|
|
initialScrollTop?: number;
|
|
|
|
initialScrollTop?: number;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const MovieIndex = withScrollPosition((props) => {
|
|
|
|
const MovieIndex = withScrollPosition((props: MovieIndexProps) => {
|
|
|
|
const {
|
|
|
|
const {
|
|
|
|
isFetching,
|
|
|
|
isFetching,
|
|
|
|
isPopulated,
|
|
|
|
isPopulated,
|
|
|
@ -80,6 +83,7 @@ const MovieIndex = withScrollPosition((props) => {
|
|
|
|
const scrollerRef = useRef<HTMLDivElement>();
|
|
|
|
const scrollerRef = useRef<HTMLDivElement>();
|
|
|
|
const [isOptionsModalOpen, setIsOptionsModalOpen] = useState(false);
|
|
|
|
const [isOptionsModalOpen, setIsOptionsModalOpen] = useState(false);
|
|
|
|
const [jumpToCharacter, setJumpToCharacter] = useState<string | null>(null);
|
|
|
|
const [jumpToCharacter, setJumpToCharacter] = useState<string | null>(null);
|
|
|
|
|
|
|
|
const [isSelectMode, setIsSelectMode] = useState(false);
|
|
|
|
|
|
|
|
|
|
|
|
const onRefreshMoviePress = useCallback(() => {
|
|
|
|
const onRefreshMoviePress = useCallback(() => {
|
|
|
|
dispatch(
|
|
|
|
dispatch(
|
|
|
@ -89,6 +93,10 @@ const MovieIndex = withScrollPosition((props) => {
|
|
|
|
);
|
|
|
|
);
|
|
|
|
}, [dispatch]);
|
|
|
|
}, [dispatch]);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const onSelectModePress = useCallback(() => {
|
|
|
|
|
|
|
|
setIsSelectMode(!isSelectMode);
|
|
|
|
|
|
|
|
}, [isSelectMode, setIsSelectMode]);
|
|
|
|
|
|
|
|
|
|
|
|
const onRssSyncPress = useCallback(() => {
|
|
|
|
const onRssSyncPress = useCallback(() => {
|
|
|
|
dispatch(
|
|
|
|
dispatch(
|
|
|
|
executeCommand({
|
|
|
|
executeCommand({
|
|
|
@ -194,118 +202,144 @@ const MovieIndex = withScrollPosition((props) => {
|
|
|
|
const hasNoMovie = !totalItems;
|
|
|
|
const hasNoMovie = !totalItems;
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
return (
|
|
|
|
<PageContent>
|
|
|
|
<SelectProvider isSelectMode={isSelectMode} items={items}>
|
|
|
|
<PageToolbar>
|
|
|
|
<PageContent>
|
|
|
|
<PageToolbarSection>
|
|
|
|
<PageToolbar>
|
|
|
|
<PageToolbarButton
|
|
|
|
<PageToolbarSection>
|
|
|
|
label="Update all"
|
|
|
|
<PageToolbarButton
|
|
|
|
iconName={icons.REFRESH}
|
|
|
|
label={translate('UpdateAll')}
|
|
|
|
spinningName={icons.REFRESH}
|
|
|
|
iconName={icons.REFRESH}
|
|
|
|
isSpinning={isRefreshingMovie}
|
|
|
|
spinningName={icons.REFRESH}
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
isSpinning={isRefreshingMovie}
|
|
|
|
onPress={onRefreshMoviePress}
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
/>
|
|
|
|
onPress={onRefreshMoviePress}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
<PageToolbarButton
|
|
|
|
|
|
|
|
label="RSS Sync"
|
|
|
|
|
|
|
|
iconName={icons.RSS}
|
|
|
|
|
|
|
|
isSpinning={isRssSyncExecuting}
|
|
|
|
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
|
|
|
|
onPress={onRssSyncPress}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
</PageToolbarSection>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<PageToolbarSection alignContent={align.RIGHT} collapseButtons={false}>
|
|
|
|
|
|
|
|
{view === 'table' ? (
|
|
|
|
|
|
|
|
<TableOptionsModalWrapper
|
|
|
|
|
|
|
|
columns={columns}
|
|
|
|
|
|
|
|
optionsComponent={MovieIndexTableOptions}
|
|
|
|
|
|
|
|
onTableOptionChange={onTableOptionChange}
|
|
|
|
|
|
|
|
>
|
|
|
|
|
|
|
|
<PageToolbarButton label="Options" iconName={icons.TABLE} />
|
|
|
|
|
|
|
|
</TableOptionsModalWrapper>
|
|
|
|
|
|
|
|
) : (
|
|
|
|
|
|
|
|
<PageToolbarButton
|
|
|
|
<PageToolbarButton
|
|
|
|
label="Options"
|
|
|
|
label={translate('RSSSync')}
|
|
|
|
iconName={view === 'posters' ? icons.POSTER : icons.OVERVIEW}
|
|
|
|
iconName={icons.RSS}
|
|
|
|
|
|
|
|
isSpinning={isRssSyncExecuting}
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
onPress={onOptionsPress}
|
|
|
|
onPress={onRssSyncPress}
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<PageToolbarSeparator />
|
|
|
|
<PageToolbarSeparator />
|
|
|
|
|
|
|
|
|
|
|
|
<MovieIndexViewMenu
|
|
|
|
<PageToolbarButton
|
|
|
|
view={view}
|
|
|
|
label={
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
isSelectMode
|
|
|
|
onViewSelect={onViewSelect}
|
|
|
|
? translate('StopSelecting')
|
|
|
|
/>
|
|
|
|
: translate('SelectMovie')
|
|
|
|
|
|
|
|
}
|
|
|
|
<MovieIndexSortMenu
|
|
|
|
iconName={isSelectMode ? icons.SERIES_ENDED : icons.CHECK}
|
|
|
|
sortKey={sortKey}
|
|
|
|
onPress={onSelectModePress}
|
|
|
|
sortDirection={sortDirection}
|
|
|
|
/>
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
|
|
|
|
onSortSelect={onSortSelect}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<MovieIndexFilterMenu
|
|
|
|
{isSelectMode ? <MovieIndexSelectAllButton /> : null}
|
|
|
|
selectedFilterKey={selectedFilterKey}
|
|
|
|
</PageToolbarSection>
|
|
|
|
filters={filters}
|
|
|
|
|
|
|
|
customFilters={customFilters}
|
|
|
|
<PageToolbarSection
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
alignContent={align.RIGHT}
|
|
|
|
onFilterSelect={onFilterSelect}
|
|
|
|
collapseButtons={false}
|
|
|
|
/>
|
|
|
|
>
|
|
|
|
</PageToolbarSection>
|
|
|
|
{view === 'table' ? (
|
|
|
|
</PageToolbar>
|
|
|
|
<TableOptionsModalWrapper
|
|
|
|
<div className={styles.pageContentBodyWrapper}>
|
|
|
|
columns={columns}
|
|
|
|
<PageContentBody
|
|
|
|
optionsComponent={MovieIndexTableOptions}
|
|
|
|
ref={scrollerRef}
|
|
|
|
onTableOptionChange={onTableOptionChange}
|
|
|
|
className={styles.contentBody}
|
|
|
|
>
|
|
|
|
innerClassName={styles[`${view}InnerContentBody`]}
|
|
|
|
<PageToolbarButton
|
|
|
|
initialScrollTop={props.initialScrollTop}
|
|
|
|
label={translate('Options')}
|
|
|
|
onScroll={onScroll}
|
|
|
|
iconName={icons.TABLE}
|
|
|
|
>
|
|
|
|
/>
|
|
|
|
{isFetching && !isPopulated ? <LoadingIndicator /> : null}
|
|
|
|
</TableOptionsModalWrapper>
|
|
|
|
|
|
|
|
) : (
|
|
|
|
{!isFetching && !!error ? <div>Unable to load movie</div> : null}
|
|
|
|
<PageToolbarButton
|
|
|
|
|
|
|
|
label={translate('Options')}
|
|
|
|
{isLoaded ? (
|
|
|
|
iconName={view === 'posters' ? icons.POSTER : icons.OVERVIEW}
|
|
|
|
<div className={styles.contentBodyContainer}>
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
<ViewComponent
|
|
|
|
onPress={onOptionsPress}
|
|
|
|
scrollerRef={scrollerRef}
|
|
|
|
|
|
|
|
items={items}
|
|
|
|
|
|
|
|
sortKey={sortKey}
|
|
|
|
|
|
|
|
sortDirection={sortDirection}
|
|
|
|
|
|
|
|
jumpToCharacter={jumpToCharacter}
|
|
|
|
|
|
|
|
isSmallScreen={isSmallScreen}
|
|
|
|
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
|
|
<MovieIndexFooter />
|
|
|
|
<PageToolbarSeparator />
|
|
|
|
</div>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{!error && isPopulated && !items.length ? (
|
|
|
|
<MovieIndexViewMenu
|
|
|
|
<NoMovie totalItems={totalItems} />
|
|
|
|
view={view}
|
|
|
|
) : null}
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
</PageContentBody>
|
|
|
|
onViewSelect={onViewSelect}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
{isLoaded && !!jumpBarItems.order.length ? (
|
|
|
|
<MovieIndexSortMenu
|
|
|
|
<PageJumpBar items={jumpBarItems} onItemPress={onJumpBarItemPress} />
|
|
|
|
sortKey={sortKey}
|
|
|
|
|
|
|
|
sortDirection={sortDirection}
|
|
|
|
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
|
|
|
|
onSortSelect={onSortSelect}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<MovieIndexFilterMenu
|
|
|
|
|
|
|
|
selectedFilterKey={selectedFilterKey}
|
|
|
|
|
|
|
|
filters={filters}
|
|
|
|
|
|
|
|
customFilters={customFilters}
|
|
|
|
|
|
|
|
isDisabled={hasNoMovie}
|
|
|
|
|
|
|
|
onFilterSelect={onFilterSelect}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
</PageToolbarSection>
|
|
|
|
|
|
|
|
</PageToolbar>
|
|
|
|
|
|
|
|
<div className={styles.pageContentBodyWrapper}>
|
|
|
|
|
|
|
|
<PageContentBody
|
|
|
|
|
|
|
|
ref={scrollerRef}
|
|
|
|
|
|
|
|
className={styles.contentBody}
|
|
|
|
|
|
|
|
innerClassName={styles[`${view}InnerContentBody`]}
|
|
|
|
|
|
|
|
initialScrollTop={props.initialScrollTop}
|
|
|
|
|
|
|
|
onScroll={onScroll}
|
|
|
|
|
|
|
|
>
|
|
|
|
|
|
|
|
{isFetching && !isPopulated ? <LoadingIndicator /> : null}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{!isFetching && !!error ? <div>Unable to load movie</div> : null}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{isLoaded ? (
|
|
|
|
|
|
|
|
<div className={styles.contentBodyContainer}>
|
|
|
|
|
|
|
|
<ViewComponent
|
|
|
|
|
|
|
|
scrollerRef={scrollerRef}
|
|
|
|
|
|
|
|
items={items}
|
|
|
|
|
|
|
|
sortKey={sortKey}
|
|
|
|
|
|
|
|
sortDirection={sortDirection}
|
|
|
|
|
|
|
|
jumpToCharacter={jumpToCharacter}
|
|
|
|
|
|
|
|
isSelectMode={isSelectMode}
|
|
|
|
|
|
|
|
isSmallScreen={isSmallScreen}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<MovieIndexFooter />
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{!error && isPopulated && !items.length ? (
|
|
|
|
|
|
|
|
<NoMovie totalItems={totalItems} />
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
</PageContentBody>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
{isLoaded && !!jumpBarItems.order.length ? (
|
|
|
|
|
|
|
|
<PageJumpBar
|
|
|
|
|
|
|
|
items={jumpBarItems}
|
|
|
|
|
|
|
|
onItemPress={onJumpBarItemPress}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
{view === 'posters' ? (
|
|
|
|
|
|
|
|
<MovieIndexPosterOptionsModal
|
|
|
|
|
|
|
|
isOpen={isOptionsModalOpen}
|
|
|
|
|
|
|
|
onModalClose={onOptionsModalClose}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
{view === 'overview' ? (
|
|
|
|
|
|
|
|
<MovieIndexOverviewOptionsModal
|
|
|
|
|
|
|
|
isOpen={isOptionsModalOpen}
|
|
|
|
|
|
|
|
onModalClose={onOptionsModalClose}
|
|
|
|
|
|
|
|
/>
|
|
|
|
) : null}
|
|
|
|
) : null}
|
|
|
|
</div>
|
|
|
|
</PageContent>
|
|
|
|
{view === 'posters' ? (
|
|
|
|
</SelectProvider>
|
|
|
|
<MovieIndexPosterOptionsModal
|
|
|
|
|
|
|
|
isOpen={isOptionsModalOpen}
|
|
|
|
|
|
|
|
onModalClose={onOptionsModalClose}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
{view === 'overview' ? (
|
|
|
|
|
|
|
|
<MovieIndexOverviewOptionsModal
|
|
|
|
|
|
|
|
isOpen={isOptionsModalOpen}
|
|
|
|
|
|
|
|
onModalClose={onOptionsModalClose}
|
|
|
|
|
|
|
|
/>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
</PageContent>
|
|
|
|
|
|
|
|
);
|
|
|
|
);
|
|
|
|
}, 'movieIndex');
|
|
|
|
}, 'movieIndex');
|
|
|
|
|
|
|
|
|
|
|
|