mirror of
https://github.com/tachiyomiorg/tachiyomi-extensions-inspector.git
synced 2025-01-11 16:29:08 +01:00
reader ui changes
This commit is contained in:
parent
5d484b012c
commit
04837983fa
@ -66,7 +66,12 @@ export default function App() {
|
|||||||
<NavbarContext.Provider value={navBarContext}>
|
<NavbarContext.Provider value={navBarContext}>
|
||||||
<CssBaseline />
|
<CssBaseline />
|
||||||
<NavBar />
|
<NavBar />
|
||||||
<Container id="appMainContainer" maxWidth={false} disableGutters>
|
<Container
|
||||||
|
id="appMainContainer"
|
||||||
|
maxWidth={false}
|
||||||
|
disableGutters
|
||||||
|
style={{ paddingTop: '64px' }}
|
||||||
|
>
|
||||||
<Switch>
|
<Switch>
|
||||||
<Route path="/sources/:sourceId/search/">
|
<Route path="/sources/:sourceId/search/">
|
||||||
<Search />
|
<Search />
|
||||||
|
@ -18,9 +18,10 @@ import CategorySelect from './CategorySelect';
|
|||||||
const useStyles = (inLibrary: string) => makeStyles((theme: Theme) => ({
|
const useStyles = (inLibrary: string) => makeStyles((theme: Theme) => ({
|
||||||
root: {
|
root: {
|
||||||
width: '100%',
|
width: '100%',
|
||||||
// [theme.breakpoints.up('md')]: {
|
[theme.breakpoints.up('md')]: {
|
||||||
// display: 'flex',
|
position: 'fixed',
|
||||||
// },
|
width: '50vw',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
top: {
|
top: {
|
||||||
padding: '10px',
|
padding: '10px',
|
||||||
|
@ -39,7 +39,7 @@ export default function NavBar() {
|
|||||||
{!override.status
|
{!override.status
|
||||||
&& (
|
&& (
|
||||||
<div className={classes.root}>
|
<div className={classes.root}>
|
||||||
<AppBar position="static" color={darkTheme ? 'default' : 'primary'}>
|
<AppBar position="fixed" color={darkTheme ? 'default' : 'primary'}>
|
||||||
<Toolbar>
|
<Toolbar>
|
||||||
<IconButton
|
<IconButton
|
||||||
edge="start"
|
edge="start"
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable react/no-unused-prop-types */
|
||||||
/* eslint-disable @typescript-eslint/no-unused-vars */
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
/* This Source Code Form is subject to the terms of the Mozilla Public
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
@ -5,7 +6,7 @@
|
|||||||
|
|
||||||
import CircularProgress from '@material-ui/core/CircularProgress';
|
import CircularProgress from '@material-ui/core/CircularProgress';
|
||||||
import { makeStyles } from '@material-ui/core/styles';
|
import { makeStyles } from '@material-ui/core/styles';
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useRef, useState } from 'react';
|
||||||
import LazyLoad from 'react-lazyload';
|
import LazyLoad from 'react-lazyload';
|
||||||
|
|
||||||
const useStyles = makeStyles({
|
const useStyles = makeStyles({
|
||||||
@ -25,12 +26,31 @@ const useStyles = makeStyles({
|
|||||||
interface IProps {
|
interface IProps {
|
||||||
src: string
|
src: string
|
||||||
index: number
|
index: number
|
||||||
|
setCurPage: React.Dispatch<React.SetStateAction<number>>
|
||||||
}
|
}
|
||||||
|
|
||||||
function LazyImage(props: IProps) {
|
function LazyImage(props: IProps) {
|
||||||
const classes = useStyles();
|
const classes = useStyles();
|
||||||
const { src, index } = props;
|
const { src, index, setCurPage } = props;
|
||||||
const [imageSrc, setImagsrc] = useState<string>('');
|
const [imageSrc, setImagsrc] = useState<string>('');
|
||||||
|
const ref = useRef<HTMLImageElement>(null);
|
||||||
|
|
||||||
|
const handleScroll = () => {
|
||||||
|
if (ref.current) {
|
||||||
|
const rect = ref.current.getBoundingClientRect();
|
||||||
|
if (rect.y < 0 && rect.y + rect.height > 0) {
|
||||||
|
setCurPage(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
window.addEventListener('scroll', handleScroll);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('scroll', handleScroll);
|
||||||
|
};
|
||||||
|
}, [handleScroll]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const img = new Image();
|
const img = new Image();
|
||||||
@ -48,12 +68,17 @@ function LazyImage(props: IProps) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<img src={imageSrc} alt={`Page #${index}`} style={{ maxWidth: '100%' }} />
|
<img
|
||||||
|
ref={ref}
|
||||||
|
src={imageSrc}
|
||||||
|
alt={`Page #${index}`}
|
||||||
|
style={{ maxWidth: '100%' }}
|
||||||
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Page(props: IProps) {
|
export default function Page(props: IProps) {
|
||||||
const { src, index } = props;
|
const { src, index, setCurPage } = props;
|
||||||
const classes = useStyles();
|
const classes = useStyles();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@ -67,7 +92,7 @@ export default function Page(props: IProps) {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<LazyImage src={src} index={index} />
|
<LazyImage src={src} index={index} setCurPage={setCurPage} />
|
||||||
</LazyLoad>
|
</LazyLoad>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -14,10 +14,11 @@ import { useHistory } from 'react-router-dom';
|
|||||||
import Slide from '@material-ui/core/Slide';
|
import Slide from '@material-ui/core/Slide';
|
||||||
import Fade from '@material-ui/core/Fade';
|
import Fade from '@material-ui/core/Fade';
|
||||||
import Zoom from '@material-ui/core/Zoom';
|
import Zoom from '@material-ui/core/Zoom';
|
||||||
|
import { Switch } from '@material-ui/core';
|
||||||
import NavBarContext from '../context/NavbarContext';
|
import NavBarContext from '../context/NavbarContext';
|
||||||
import DarkTheme from '../context/DarkTheme';
|
import DarkTheme from '../context/DarkTheme';
|
||||||
|
|
||||||
const useStyles = makeStyles((theme: Theme) => ({
|
const useStyles = (settings: IReaderSettings) => makeStyles((theme: Theme) => ({
|
||||||
// main container and root div need to change classes...
|
// main container and root div need to change classes...
|
||||||
AppMainContainer: {
|
AppMainContainer: {
|
||||||
display: 'none',
|
display: 'none',
|
||||||
@ -27,7 +28,7 @@ const useStyles = makeStyles((theme: Theme) => ({
|
|||||||
},
|
},
|
||||||
|
|
||||||
root: {
|
root: {
|
||||||
position: 'fixed',
|
position: settings.staticNav ? 'sticky' : 'fixed',
|
||||||
top: 0,
|
top: 0,
|
||||||
left: 0,
|
left: 0,
|
||||||
minWidth: '300px',
|
minWidth: '300px',
|
||||||
@ -81,9 +82,15 @@ const useStyles = makeStyles((theme: Theme) => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
export interface IReaderSettings{
|
export interface IReaderSettings{
|
||||||
|
staticNav: boolean
|
||||||
|
showPageNumber: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const defaultReaderSettings = () => ({
|
||||||
|
staticNav: false,
|
||||||
|
showPageNumber: true,
|
||||||
|
} as IReaderSettings);
|
||||||
|
|
||||||
interface IProps {
|
interface IProps {
|
||||||
settings: IReaderSettings
|
settings: IReaderSettings
|
||||||
setSettings: React.Dispatch<React.SetStateAction<IReaderSettings>>
|
setSettings: React.Dispatch<React.SetStateAction<IReaderSettings>>
|
||||||
@ -96,14 +103,16 @@ export default function ReaderNavBar(props: IProps) {
|
|||||||
|
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
|
|
||||||
const [drawerOpen, setDrawerOpen] = useState(false);
|
const { settings, setSettings, manga } = props;
|
||||||
|
|
||||||
|
const [drawerOpen, setDrawerOpen] = useState(false || settings.staticNav);
|
||||||
const [hideOpenButton, setHideOpenButton] = useState(false);
|
const [hideOpenButton, setHideOpenButton] = useState(false);
|
||||||
const [prevScrollPos, setPrevScrollPos] = useState(0);
|
const [prevScrollPos, setPrevScrollPos] = useState(0);
|
||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
const classes = useStyles();
|
const classes = useStyles(settings)();
|
||||||
|
|
||||||
const { settings, setSettings, manga } = props;
|
const setSettingValue = (key: string, value: any) => setSettings({ ...settings, [key]: value });
|
||||||
|
|
||||||
const handleScroll = () => {
|
const handleScroll = () => {
|
||||||
const currentScrollPos = window.pageYOffset;
|
const currentScrollPos = window.pageYOffset;
|
||||||
@ -120,7 +129,7 @@ export default function ReaderNavBar(props: IProps) {
|
|||||||
return () => {
|
return () => {
|
||||||
window.removeEventListener('scroll', handleScroll);
|
window.removeEventListener('scroll', handleScroll);
|
||||||
};
|
};
|
||||||
}, [handleScroll]);
|
}, [handleScroll]);// handleScroll changes on every render
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
window.addEventListener('scroll', handleScroll);
|
window.addEventListener('scroll', handleScroll);
|
||||||
@ -135,7 +144,7 @@ export default function ReaderNavBar(props: IProps) {
|
|||||||
rootEl.classList.remove(classes.AppRootElment);
|
rootEl.classList.remove(classes.AppRootElment);
|
||||||
mainContainer.classList.remove(classes.AppMainContainer);
|
mainContainer.classList.remove(classes.AppMainContainer);
|
||||||
};
|
};
|
||||||
}, []);
|
}, [handleScroll]);// handleScroll changes on every render
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@ -154,16 +163,29 @@ export default function ReaderNavBar(props: IProps) {
|
|||||||
<Typography variant="h1">
|
<Typography variant="h1">
|
||||||
{title}
|
{title}
|
||||||
</Typography>
|
</Typography>
|
||||||
<IconButton
|
{!settings.staticNav
|
||||||
edge="start"
|
&& (
|
||||||
color="inherit"
|
<IconButton
|
||||||
aria-label="menu"
|
edge="start"
|
||||||
disableRipple
|
color="inherit"
|
||||||
onClick={() => setDrawerOpen(false)}
|
aria-label="menu"
|
||||||
>
|
disableRipple
|
||||||
<KeyboardArrowLeftIcon />
|
onClick={() => setDrawerOpen(false)}
|
||||||
</IconButton>
|
>
|
||||||
|
<KeyboardArrowLeftIcon />
|
||||||
|
</IconButton>
|
||||||
|
) }
|
||||||
</header>
|
</header>
|
||||||
|
<h3>Static Navigation</h3>
|
||||||
|
<Switch
|
||||||
|
checked={settings.staticNav}
|
||||||
|
onChange={(e) => setSettingValue('staticNav', e.target.checked)}
|
||||||
|
/>
|
||||||
|
<h3>Show page number</h3>
|
||||||
|
<Switch
|
||||||
|
checked={settings.showPageNumber}
|
||||||
|
onChange={(e) => setSettingValue('showPageNumber', e.target.checked)}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</Slide>
|
</Slide>
|
||||||
<Zoom in={!drawerOpen}>
|
<Zoom in={!drawerOpen}>
|
||||||
|
@ -19,6 +19,15 @@ const useStyles = makeStyles((theme: Theme) => ({
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
chapters: {
|
||||||
|
listStyle: 'none',
|
||||||
|
padding: 0,
|
||||||
|
[theme.breakpoints.up('md')]: {
|
||||||
|
width: '50%',
|
||||||
|
marginLeft: '50%',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
loading: {
|
loading: {
|
||||||
margin: '10px 0',
|
margin: '10px 0',
|
||||||
display: 'flex',
|
display: 'flex',
|
||||||
@ -53,7 +62,7 @@ export default function Manga() {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const chapterCards = (
|
const chapterCards = (
|
||||||
<ol style={{ listStyle: 'none', padding: 0, width: '100%' }}>
|
<ol className={classes.chapters}>
|
||||||
{chapters.length === 0
|
{chapters.length === 0
|
||||||
&& (
|
&& (
|
||||||
<div className={classes.loading}>
|
<div className={classes.loading}>
|
||||||
|
@ -8,12 +8,12 @@ import { makeStyles } from '@material-ui/core/styles';
|
|||||||
import React, { useContext, useEffect, useState } from 'react';
|
import React, { useContext, useEffect, useState } from 'react';
|
||||||
import { useParams } from 'react-router-dom';
|
import { useParams } from 'react-router-dom';
|
||||||
import Page from '../components/Page';
|
import Page from '../components/Page';
|
||||||
import ReaderNavBar, { IReaderSettings } from '../components/ReaderNavBar';
|
import ReaderNavBar, { defaultReaderSettings, IReaderSettings } from '../components/ReaderNavBar';
|
||||||
import NavbarContext from '../context/NavbarContext';
|
import NavbarContext from '../context/NavbarContext';
|
||||||
import client from '../util/client';
|
import client from '../util/client';
|
||||||
import useLocalStorage from '../util/useLocalStorage';
|
import useLocalStorage from '../util/useLocalStorage';
|
||||||
|
|
||||||
const useStyles = makeStyles({
|
const useStyles = (settings: IReaderSettings) => makeStyles({
|
||||||
reader: {
|
reader: {
|
||||||
display: 'flex',
|
display: 'flex',
|
||||||
flexDirection: 'column',
|
flexDirection: 'column',
|
||||||
@ -24,20 +24,32 @@ const useStyles = makeStyles({
|
|||||||
loading: {
|
loading: {
|
||||||
margin: '50px auto',
|
margin: '50px auto',
|
||||||
},
|
},
|
||||||
|
|
||||||
|
pageNumber: {
|
||||||
|
display: settings.showPageNumber ? 'block' : 'none',
|
||||||
|
position: 'fixed',
|
||||||
|
bottom: '50px',
|
||||||
|
right: settings.staticNav ? 'calc((100vw - 325px)/2)' : 'calc((100vw - 25px)/2)',
|
||||||
|
width: '50px',
|
||||||
|
textAlign: 'center',
|
||||||
|
backgroundColor: 'rgba(0, 0, 0, 0.3)',
|
||||||
|
borderRadius: '10px',
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const range = (n:number) => Array.from({ length: n }, (value, key) => key);
|
const range = (n:number) => Array.from({ length: n }, (value, key) => key);
|
||||||
|
|
||||||
export default function Reader() {
|
export default function Reader() {
|
||||||
const classes = useStyles();
|
const [settings, setSettings] = useLocalStorage<IReaderSettings>('readerSettings', defaultReaderSettings);
|
||||||
|
|
||||||
const [settings, setSettings] = useState<IReaderSettings>({});
|
const classes = useStyles(settings)();
|
||||||
|
|
||||||
const [serverAddress] = useLocalStorage<String>('serverBaseURL', '');
|
const [serverAddress] = useLocalStorage<String>('serverBaseURL', '');
|
||||||
|
|
||||||
const { chapterId, mangaId } = useParams<{chapterId: string, mangaId: string}>();
|
const { chapterId, mangaId } = useParams<{chapterId: string, mangaId: string}>();
|
||||||
const [manga, setManga] = useState<IMangaCard | IManga>({ id: +mangaId, title: '', thumbnailUrl: '' });
|
const [manga, setManga] = useState<IMangaCard | IManga>({ id: +mangaId, title: '', thumbnailUrl: '' });
|
||||||
const [pageCount, setPageCount] = useState<number>(-1);
|
const [pageCount, setPageCount] = useState<number>(-1);
|
||||||
|
const [curPage, setCurPage] = useState<number>(0);
|
||||||
|
|
||||||
const { setOverride, setTitle } = useContext(NavbarContext);
|
const { setOverride, setTitle } = useContext(NavbarContext);
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@ -54,7 +66,7 @@ export default function Reader() {
|
|||||||
|
|
||||||
// clean up for when we leave the reader
|
// clean up for when we leave the reader
|
||||||
return () => setOverride({ status: false, value: <div /> });
|
return () => setOverride({ status: false, value: <div /> });
|
||||||
}, [manga]);
|
}, [manga, settings]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setTitle('Reader');
|
setTitle('Reader');
|
||||||
@ -83,8 +95,16 @@ export default function Reader() {
|
|||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<div className={classes.reader}>
|
<div className={classes.reader}>
|
||||||
|
<div className={classes.pageNumber}>
|
||||||
|
{`${curPage + 1} / ${pageCount}`}
|
||||||
|
</div>
|
||||||
{range(pageCount).map((index) => (
|
{range(pageCount).map((index) => (
|
||||||
<Page key={index} index={index} src={`${serverAddress}/api/v1/manga/${mangaId}/chapter/${chapterId}/page/${index}`} />
|
<Page
|
||||||
|
key={index}
|
||||||
|
index={index}
|
||||||
|
src={`${serverAddress}/api/v1/manga/${mangaId}/chapter/${chapterId}/page/${index}`}
|
||||||
|
setCurPage={setCurPage}
|
||||||
|
/>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -2,19 +2,20 @@
|
|||||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||||
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
import { useState, Dispatch, SetStateAction } from 'react';
|
import React, { useState, Dispatch, SetStateAction } from 'react';
|
||||||
import storage from './localStorage';
|
import storage from './localStorage';
|
||||||
|
|
||||||
// eslint-disable-next-line max-len
|
// eslint-disable-next-line max-len
|
||||||
export default function useLocalStorage<T>(key: string, defaultValue: T) : [T, Dispatch<SetStateAction<T>>] {
|
export default function useLocalStorage<T>(key: string, defaultValue: T | (() => T)) : [T, Dispatch<SetStateAction<T>>] {
|
||||||
const [storedValue, setStoredValue] = useState<T>(storage.getItem(key, defaultValue));
|
const initialState = defaultValue instanceof Function ? defaultValue() : defaultValue;
|
||||||
|
const [storedValue, setStoredValue] = useState<T>(storage.getItem(key, initialState));
|
||||||
|
|
||||||
const setValue = (value: T | ((prevState: T) => T)) => {
|
const setValue = ((value: T | ((prevState: T) => T)) => {
|
||||||
// Allow value to be a function so we have same API as useState
|
// Allow value to be a function so we have same API as useState
|
||||||
const valueToStore = value instanceof Function ? value(storedValue) : value;
|
const valueToStore = value instanceof Function ? value(storedValue) : value;
|
||||||
setStoredValue(valueToStore);
|
setStoredValue(valueToStore);
|
||||||
storage.setItem(key, valueToStore);
|
storage.setItem(key, valueToStore);
|
||||||
};
|
}) as React.Dispatch<React.SetStateAction<T>>;
|
||||||
|
|
||||||
return [storedValue, setValue];
|
return [storedValue, setValue];
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user