/* * Copyright (C) Contributors to the Suwayomi project * * 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 * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ import CircularProgress from '@material-ui/core/CircularProgress'; import { makeStyles } from '@material-ui/core/styles'; import React, { useContext, useEffect, useState } from 'react'; import { useHistory, useParams } from 'react-router-dom'; import HorizontalPager from 'components/manga/reader/pager/HorizontalPager'; import PageNumber from 'components/manga/reader/PageNumber'; import WebtoonPager from 'components/manga/reader/pager/PagedPager'; import VerticalPager from 'components/manga/reader/pager/VerticalPager'; import ReaderNavBar, { defaultReaderSettings } from 'components/navbar/ReaderNavBar'; import NavbarContext from 'context/NavbarContext'; import client from 'util/client'; import useLocalStorage from 'util/useLocalStorage'; import cloneObject from 'util/cloneObject'; const useStyles = (settings: IReaderSettings) => makeStyles({ root: { width: settings.staticNav ? 'calc(100vw - 300px)' : '100vw', }, loading: { margin: '50px auto', }, }); const getReaderComponent = (readerType: ReaderType) => { switch (readerType) { case 'ContinuesVertical': return VerticalPager; break; case 'Webtoon': return VerticalPager; break; case 'SingleVertical': return WebtoonPager; break; case 'SingleRTL': return WebtoonPager; break; case 'SingleLTR': return WebtoonPager; break; case 'ContinuesHorizontal': return HorizontalPager; default: return VerticalPager; break; } }; const range = (n:number) => Array.from({ length: n }, (value, key) => key); const initialChapter = () => ({ pageCount: -1, index: -1, chapterCount: 0 }); export default function Reader() { const [settings, setSettings] = useLocalStorage('readerSettings', defaultReaderSettings); const classes = useStyles(settings)(); const history = useHistory(); const [serverAddress] = useLocalStorage('serverBaseURL', ''); const { chapterIndex, mangaId } = useParams<{ chapterIndex: string, mangaId: string }>(); const [manga, setManga] = useState({ id: +mangaId, title: '', thumbnailUrl: '' }); const [chapter, setChapter] = useState(initialChapter()); const [curPage, setCurPage] = useState(0); const { setOverride, setTitle } = useContext(NavbarContext); useEffect(() => { // make sure settings has all the keys const settingsClone = cloneObject(settings) as any; const defualtSettings = defaultReaderSettings(); let shouldUpdateSettings = false; Object.keys(defualtSettings).forEach((key) => { const keyOf = key as keyof IReaderSettings; if (settings[keyOf] === undefined) { settingsClone[keyOf] = defualtSettings[keyOf]; shouldUpdateSettings = true; } }); if (shouldUpdateSettings) { setSettings(settingsClone); } // set the custom navbar setOverride( { status: true, value: ( ), }, ); // clean up for when we leave the reader return () => setOverride({ status: false, value:
}); }, [manga, chapter, settings, curPage, chapterIndex]); useEffect(() => { setTitle('Reader'); client.get(`/api/v1/manga/${mangaId}/`) .then((response) => response.data) .then((data: IManga) => { setManga(data); setTitle(data.title); }); }, [chapterIndex]); useEffect(() => { setChapter(initialChapter); client.get(`/api/v1/manga/${mangaId}/chapter/${chapterIndex}`) .then((response) => response.data) .then((data:IChapter) => { setChapter(data); setCurPage(data.lastPageRead); }); }, [chapterIndex]); useEffect(() => { if (curPage !== -1) { const formData = new FormData(); formData.append('lastPageRead', curPage.toString()); client.patch(`/api/v1/manga/${manga.id}/chapter/${chapter.index}`, formData); } if (curPage === chapter.pageCount - 1) { const formDataRead = new FormData(); formDataRead.append('read', 'true'); client.patch(`/api/v1/manga/${manga.id}/chapter/${chapter.index}`, formDataRead); } }, [curPage]); // return spinner while chpater data is loading if (chapter.pageCount === -1) { return (
); } const nextChapter = () => { if (chapter.index < chapter.chapterCount) { const formData = new FormData(); formData.append('lastPageRead', `${chapter.pageCount - 1}`); formData.append('read', 'true'); client.patch(`/api/v1/manga/${manga.id}/chapter/${chapter.index}`, formData); history.push(`/manga/${manga.id}/chapter/${chapter.index + 1}`); } }; const prevChapter = () => { if (chapter.index > 1) { history.push(`/manga/${manga.id}/chapter/${chapter.index - 1}`); } }; const pages = range(chapter.pageCount).map((index) => ({ index, src: `${serverAddress}/api/v1/manga/${mangaId}/chapter/${chapterIndex}/page/${index}`, })); const ReaderComponent = getReaderComponent(settings.readerType); return (
); }