mirror of
https://github.com/jellyfin/jellyfin-web
synced 2025-03-30 19:56:21 +00:00
115 lines
4.1 KiB
TypeScript
115 lines
4.1 KiB
TypeScript
![]() |
import React, { useCallback, useEffect, useState } from 'react';
|
||
|
import AppBar from '@mui/material/AppBar';
|
||
|
import Box from '@mui/material/Box';
|
||
|
import { ThemeProvider } from '@mui/material/styles';
|
||
|
import { Outlet, useLocation } from 'react-router-dom';
|
||
|
|
||
|
import AppHeader from 'components/AppHeader';
|
||
|
import Backdrop from 'components/Backdrop';
|
||
|
import { useApi } from 'hooks/useApi';
|
||
|
import { useLocalStorage } from 'hooks/useLocalStorage';
|
||
|
|
||
|
import AppToolbar from './components/AppToolbar';
|
||
|
import AppDrawer, { DRAWER_WIDTH, isDrawerPath } from './components/drawers/AppDrawer';
|
||
|
import ElevationScroll from './components/ElevationScroll';
|
||
|
import theme from './theme';
|
||
|
|
||
|
import './AppOverrides.scss';
|
||
|
|
||
|
interface ExperimentalAppSettings {
|
||
|
isDrawerPinned: boolean
|
||
|
}
|
||
|
|
||
|
const DEFAULT_EXPERIMENTAL_APP_SETTINGS: ExperimentalAppSettings = {
|
||
|
isDrawerPinned: false
|
||
|
};
|
||
|
|
||
|
const AppLayout = () => {
|
||
|
const [ appSettings, setAppSettings ] = useLocalStorage<ExperimentalAppSettings>('ExperimentalAppSettings', DEFAULT_EXPERIMENTAL_APP_SETTINGS);
|
||
|
const [ isDrawerActive, setIsDrawerActive ] = useState(appSettings.isDrawerPinned);
|
||
|
const { user } = useApi();
|
||
|
const location = useLocation();
|
||
|
|
||
|
const isDrawerAvailable = isDrawerPath(location.pathname);
|
||
|
const isDrawerOpen = isDrawerActive && isDrawerAvailable && Boolean(user);
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (isDrawerActive !== appSettings.isDrawerPinned) {
|
||
|
setAppSettings({
|
||
|
...appSettings,
|
||
|
isDrawerPinned: isDrawerActive
|
||
|
});
|
||
|
}
|
||
|
}, [ appSettings, isDrawerActive, setAppSettings ]);
|
||
|
|
||
|
const onToggleDrawer = useCallback(() => {
|
||
|
setIsDrawerActive(!isDrawerActive);
|
||
|
}, [ isDrawerActive, setIsDrawerActive ]);
|
||
|
|
||
|
return (
|
||
|
<ThemeProvider theme={theme}>
|
||
|
<Backdrop />
|
||
|
|
||
|
<div style={{ display: 'none' }}>
|
||
|
{/*
|
||
|
* TODO: These components are not used, but views interact with them directly so the need to be
|
||
|
* present in the dom. We add them in a hidden element to prevent errors.
|
||
|
*/}
|
||
|
<AppHeader />
|
||
|
</div>
|
||
|
|
||
|
<Box sx={{ display: 'flex' }}>
|
||
|
<ElevationScroll elevate={isDrawerOpen}>
|
||
|
<AppBar
|
||
|
position='fixed'
|
||
|
sx={{ zIndex: (muiTheme) => muiTheme.zIndex.drawer + 1 }}
|
||
|
>
|
||
|
<AppToolbar
|
||
|
isDrawerOpen={isDrawerOpen}
|
||
|
onDrawerButtonClick={onToggleDrawer}
|
||
|
/>
|
||
|
</AppBar>
|
||
|
</ElevationScroll>
|
||
|
|
||
|
<AppDrawer
|
||
|
open={isDrawerOpen}
|
||
|
onClose={onToggleDrawer}
|
||
|
onOpen={onToggleDrawer}
|
||
|
/>
|
||
|
|
||
|
<Box
|
||
|
component='main'
|
||
|
sx={{
|
||
|
width: '100%',
|
||
|
flexGrow: 1,
|
||
|
transition: theme.transitions.create('margin', {
|
||
|
easing: theme.transitions.easing.sharp,
|
||
|
duration: theme.transitions.duration.leavingScreen
|
||
|
}),
|
||
|
marginLeft: 0,
|
||
|
...(isDrawerAvailable && {
|
||
|
marginLeft: {
|
||
|
sm: `-${DRAWER_WIDTH}px`
|
||
|
}
|
||
|
}),
|
||
|
...(isDrawerActive && {
|
||
|
transition: theme.transitions.create('margin', {
|
||
|
easing: theme.transitions.easing.easeOut,
|
||
|
duration: theme.transitions.duration.enteringScreen
|
||
|
}),
|
||
|
marginLeft: 0
|
||
|
})
|
||
|
}}
|
||
|
>
|
||
|
<div className='mainAnimatedPages skinBody' />
|
||
|
<div className='skinBody'>
|
||
|
<Outlet />
|
||
|
</div>
|
||
|
</Box>
|
||
|
</Box>
|
||
|
</ThemeProvider>
|
||
|
);
|
||
|
};
|
||
|
|
||
|
export default AppLayout;
|