Reader settings are now saved in localStorage
This commit is contained in:
parent
89ad4620d6
commit
25d99ee294
|
@ -19,7 +19,7 @@ import { useLocalData } from "contexts/LocalDataContext";
|
||||||
|
|
||||||
export const MainPanel = (): JSX.Element => {
|
export const MainPanel = (): JSX.Element => {
|
||||||
const is3ColumnsLayout = useIs3ColumnsLayout();
|
const is3ColumnsLayout = useIs3ColumnsLayout();
|
||||||
const { mainPanelReduced = false, toggleMainPanelReduced, setConfigPanelOpen } = useAppLayout();
|
const { mainPanelReduced, toggleMainPanelReduced, setConfigPanelOpen } = useAppLayout();
|
||||||
const { langui } = useLocalData();
|
const { langui } = useLocalData();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -1,4 +1,12 @@
|
||||||
import React, { ReactNode, useContext, useEffect, useState } from "react";
|
import React, {
|
||||||
|
createContext,
|
||||||
|
Dispatch,
|
||||||
|
ReactNode,
|
||||||
|
SetStateAction,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { useLocalStorage, useSessionStorage } from "usehooks-ts";
|
import { useLocalStorage, useSessionStorage } from "usehooks-ts";
|
||||||
import { isDefined } from "helpers/others";
|
import { isDefined } from "helpers/others";
|
||||||
|
@ -9,27 +17,27 @@ import { useScrollIntoView } from "hooks/useScrollIntoView";
|
||||||
interface AppLayoutState {
|
interface AppLayoutState {
|
||||||
subPanelOpen: boolean;
|
subPanelOpen: boolean;
|
||||||
toggleSubPanelOpen: () => void;
|
toggleSubPanelOpen: () => void;
|
||||||
setSubPanelOpen: React.Dispatch<React.SetStateAction<AppLayoutState["subPanelOpen"]>>;
|
setSubPanelOpen: Dispatch<SetStateAction<AppLayoutState["subPanelOpen"]>>;
|
||||||
|
|
||||||
configPanelOpen: boolean;
|
configPanelOpen: boolean;
|
||||||
toggleConfigPanelOpen: () => void;
|
toggleConfigPanelOpen: () => void;
|
||||||
setConfigPanelOpen: React.Dispatch<React.SetStateAction<AppLayoutState["configPanelOpen"]>>;
|
setConfigPanelOpen: Dispatch<SetStateAction<AppLayoutState["configPanelOpen"]>>;
|
||||||
|
|
||||||
mainPanelReduced: boolean;
|
mainPanelReduced: boolean;
|
||||||
toggleMainPanelReduced: () => void;
|
toggleMainPanelReduced: () => void;
|
||||||
setMainPanelReduced: React.Dispatch<React.SetStateAction<AppLayoutState["mainPanelReduced"]>>;
|
setMainPanelReduced: Dispatch<SetStateAction<AppLayoutState["mainPanelReduced"]>>;
|
||||||
|
|
||||||
mainPanelOpen: boolean;
|
mainPanelOpen: boolean;
|
||||||
toggleMainPanelOpen: () => void;
|
toggleMainPanelOpen: () => void;
|
||||||
setMainPanelOpen: React.Dispatch<React.SetStateAction<AppLayoutState["mainPanelOpen"]>>;
|
setMainPanelOpen: Dispatch<SetStateAction<AppLayoutState["mainPanelOpen"]>>;
|
||||||
|
|
||||||
menuGestures: boolean;
|
menuGestures: boolean;
|
||||||
toggleMenuGestures: () => void;
|
toggleMenuGestures: () => void;
|
||||||
setMenuGestures: React.Dispatch<React.SetStateAction<AppLayoutState["menuGestures"]>>;
|
setMenuGestures: Dispatch<SetStateAction<AppLayoutState["menuGestures"]>>;
|
||||||
|
|
||||||
hasDisgardedSafariWarning: boolean;
|
hasDisgardedSafariWarning: boolean;
|
||||||
setHasDisgardedSafariWarning: React.Dispatch<
|
setHasDisgardedSafariWarning: Dispatch<
|
||||||
React.SetStateAction<AppLayoutState["hasDisgardedSafariWarning"]>
|
SetStateAction<AppLayoutState["hasDisgardedSafariWarning"]>
|
||||||
>;
|
>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -58,7 +66,7 @@ const initialState: RequiredNonNullable<AppLayoutState> = {
|
||||||
setHasDisgardedSafariWarning: () => null,
|
setHasDisgardedSafariWarning: () => null,
|
||||||
};
|
};
|
||||||
|
|
||||||
const AppLayoutContext = React.createContext<AppLayoutState>(initialState);
|
const AppLayoutContext = createContext<AppLayoutState>(initialState);
|
||||||
|
|
||||||
export const useAppLayout = (): AppLayoutState => useContext(AppLayoutContext);
|
export const useAppLayout = (): AppLayoutState => useContext(AppLayoutContext);
|
||||||
|
|
||||||
|
|
|
@ -1,17 +1,22 @@
|
||||||
import React, { ReactNode, useContext, useState } from "react";
|
import React, {
|
||||||
|
createContext,
|
||||||
|
Dispatch,
|
||||||
|
ReactNode,
|
||||||
|
SetStateAction,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import { RequiredNonNullable } from "types/types";
|
import { RequiredNonNullable } from "types/types";
|
||||||
|
|
||||||
interface ContainerQueriesState {
|
interface ContainerQueriesState {
|
||||||
screenWidth: number;
|
screenWidth: number;
|
||||||
setScreenWidth: React.Dispatch<React.SetStateAction<ContainerQueriesState["screenWidth"]>>;
|
setScreenWidth: Dispatch<SetStateAction<ContainerQueriesState["screenWidth"]>>;
|
||||||
|
|
||||||
contentPanelWidth: number;
|
contentPanelWidth: number;
|
||||||
setContentPanelWidth: React.Dispatch<
|
setContentPanelWidth: Dispatch<SetStateAction<ContainerQueriesState["contentPanelWidth"]>>;
|
||||||
React.SetStateAction<ContainerQueriesState["contentPanelWidth"]>
|
|
||||||
>;
|
|
||||||
|
|
||||||
subPanelWidth: number;
|
subPanelWidth: number;
|
||||||
setSubPanelWidth: React.Dispatch<React.SetStateAction<ContainerQueriesState["subPanelWidth"]>>;
|
setSubPanelWidth: Dispatch<SetStateAction<ContainerQueriesState["subPanelWidth"]>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialState: RequiredNonNullable<ContainerQueriesState> = {
|
const initialState: RequiredNonNullable<ContainerQueriesState> = {
|
||||||
|
@ -25,7 +30,7 @@ const initialState: RequiredNonNullable<ContainerQueriesState> = {
|
||||||
setSubPanelWidth: () => null,
|
setSubPanelWidth: () => null,
|
||||||
};
|
};
|
||||||
|
|
||||||
const ContainerQueriesContext = React.createContext<ContainerQueriesState>(initialState);
|
const ContainerQueriesContext = createContext<ContainerQueriesState>(initialState);
|
||||||
|
|
||||||
export const useContainerQueries = (): ContainerQueriesState => useContext(ContainerQueriesContext);
|
export const useContainerQueries = (): ContainerQueriesState => useContext(ContainerQueriesContext);
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import React, { ReactNode, useContext, useMemo } from "react";
|
import React, { createContext, ReactNode, useContext, useMemo } from "react";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { useFetch } from "usehooks-ts";
|
import { useFetch } from "usehooks-ts";
|
||||||
import {
|
import {
|
||||||
|
@ -29,7 +29,7 @@ const initialState: RequiredNonNullable<LocalDataState> = {
|
||||||
langui: {},
|
langui: {},
|
||||||
};
|
};
|
||||||
|
|
||||||
const LocalDataContext = React.createContext<LocalDataState>(initialState);
|
const LocalDataContext = createContext<LocalDataState>(initialState);
|
||||||
|
|
||||||
export const useLocalData = (): LocalDataState => useContext(LocalDataContext);
|
export const useLocalData = (): LocalDataState => useContext(LocalDataContext);
|
||||||
|
|
||||||
|
|
|
@ -1,11 +1,18 @@
|
||||||
import React, { ReactNode, useContext, useState } from "react";
|
import React, {
|
||||||
|
createContext,
|
||||||
|
Dispatch,
|
||||||
|
ReactNode,
|
||||||
|
SetStateAction,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import { RequiredNonNullable } from "types/types";
|
import { RequiredNonNullable } from "types/types";
|
||||||
|
|
||||||
interface TerminalState {
|
interface TerminalState {
|
||||||
previousLines: string[];
|
previousLines: string[];
|
||||||
previousCommands: string[];
|
previousCommands: string[];
|
||||||
setPreviousLines: React.Dispatch<React.SetStateAction<TerminalState["previousLines"]>>;
|
setPreviousLines: Dispatch<SetStateAction<TerminalState["previousLines"]>>;
|
||||||
setPreviousCommands: React.Dispatch<React.SetStateAction<TerminalState["previousCommands"]>>;
|
setPreviousCommands: Dispatch<SetStateAction<TerminalState["previousCommands"]>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialState: RequiredNonNullable<TerminalState> = {
|
const initialState: RequiredNonNullable<TerminalState> = {
|
||||||
|
@ -15,7 +22,7 @@ const initialState: RequiredNonNullable<TerminalState> = {
|
||||||
setPreviousCommands: () => null,
|
setPreviousCommands: () => null,
|
||||||
};
|
};
|
||||||
|
|
||||||
const TerminalContext = React.createContext<TerminalState>(initialState);
|
const TerminalContext = createContext<TerminalState>(initialState);
|
||||||
|
|
||||||
export const useTerminalContext = (): TerminalState => useContext(TerminalContext);
|
export const useTerminalContext = (): TerminalState => useContext(TerminalContext);
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,13 @@
|
||||||
import { useLocalStorage } from "usehooks-ts";
|
import { useLocalStorage } from "usehooks-ts";
|
||||||
import React, { ReactNode, useContext, useEffect, useLayoutEffect } from "react";
|
import React, {
|
||||||
|
createContext,
|
||||||
|
Dispatch,
|
||||||
|
ReactNode,
|
||||||
|
SetStateAction,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useLayoutEffect,
|
||||||
|
} from "react";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { isDefined, isDefinedAndNotEmpty } from "helpers/others";
|
import { isDefined, isDefinedAndNotEmpty } from "helpers/others";
|
||||||
import { RequiredNonNullable } from "types/types";
|
import { RequiredNonNullable } from "types/types";
|
||||||
|
@ -8,32 +16,28 @@ import { useDarkMode } from "hooks/useDarkMode";
|
||||||
|
|
||||||
interface UserSettingsState {
|
interface UserSettingsState {
|
||||||
fontSize: number;
|
fontSize: number;
|
||||||
setFontSize: React.Dispatch<React.SetStateAction<UserSettingsState["fontSize"]>>;
|
setFontSize: Dispatch<SetStateAction<UserSettingsState["fontSize"]>>;
|
||||||
|
|
||||||
darkMode: boolean;
|
darkMode: boolean;
|
||||||
toggleDarkMode: () => void;
|
toggleDarkMode: () => void;
|
||||||
setDarkMode: React.Dispatch<React.SetStateAction<UserSettingsState["darkMode"]>>;
|
setDarkMode: Dispatch<SetStateAction<UserSettingsState["darkMode"]>>;
|
||||||
|
|
||||||
selectedThemeMode: boolean;
|
selectedThemeMode: boolean;
|
||||||
toggleSelectedThemeMode: () => void;
|
toggleSelectedThemeMode: () => void;
|
||||||
setSelectedThemeMode: React.Dispatch<
|
setSelectedThemeMode: Dispatch<SetStateAction<UserSettingsState["selectedThemeMode"]>>;
|
||||||
React.SetStateAction<UserSettingsState["selectedThemeMode"]>
|
|
||||||
>;
|
|
||||||
|
|
||||||
dyslexic: boolean;
|
dyslexic: boolean;
|
||||||
toggleDyslexic: () => void;
|
toggleDyslexic: () => void;
|
||||||
setDyslexic: React.Dispatch<React.SetStateAction<UserSettingsState["dyslexic"]>>;
|
setDyslexic: Dispatch<SetStateAction<UserSettingsState["dyslexic"]>>;
|
||||||
|
|
||||||
currency: string;
|
currency: string;
|
||||||
setCurrency: React.Dispatch<React.SetStateAction<UserSettingsState["currency"]>>;
|
setCurrency: Dispatch<SetStateAction<UserSettingsState["currency"]>>;
|
||||||
|
|
||||||
playerName: string;
|
playerName: string;
|
||||||
setPlayerName: React.Dispatch<React.SetStateAction<UserSettingsState["playerName"]>>;
|
setPlayerName: Dispatch<SetStateAction<UserSettingsState["playerName"]>>;
|
||||||
|
|
||||||
preferredLanguages: string[];
|
preferredLanguages: string[];
|
||||||
setPreferredLanguages: React.Dispatch<
|
setPreferredLanguages: Dispatch<SetStateAction<UserSettingsState["preferredLanguages"]>>;
|
||||||
React.SetStateAction<UserSettingsState["preferredLanguages"]>
|
|
||||||
>;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialState: RequiredNonNullable<UserSettingsState> = {
|
const initialState: RequiredNonNullable<UserSettingsState> = {
|
||||||
|
@ -62,7 +66,7 @@ const initialState: RequiredNonNullable<UserSettingsState> = {
|
||||||
setPreferredLanguages: () => null,
|
setPreferredLanguages: () => null,
|
||||||
};
|
};
|
||||||
|
|
||||||
const UserSettingsContext = React.createContext<UserSettingsState>(initialState);
|
const UserSettingsContext = createContext<UserSettingsState>(initialState);
|
||||||
|
|
||||||
export const useUserSettings = (): UserSettingsState => useContext(UserSettingsContext);
|
export const useUserSettings = (): UserSettingsState => useContext(UserSettingsContext);
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,11 @@
|
||||||
import { useEffect } from "react";
|
import { Dispatch, SetStateAction, useEffect } from "react";
|
||||||
import { useLocalStorage } from "usehooks-ts";
|
import { useLocalStorage } from "usehooks-ts";
|
||||||
import { usePrefersDarkMode } from "./useMediaQuery";
|
import { usePrefersDarkMode } from "./useMediaQuery";
|
||||||
|
|
||||||
export const useDarkMode = (
|
export const useDarkMode = (
|
||||||
key: string,
|
key: string,
|
||||||
initialValue: boolean
|
initialValue: boolean
|
||||||
): [
|
): [boolean, boolean, Dispatch<SetStateAction<boolean>>, Dispatch<SetStateAction<boolean>>] => {
|
||||||
boolean,
|
|
||||||
boolean,
|
|
||||||
React.Dispatch<React.SetStateAction<boolean>>,
|
|
||||||
React.Dispatch<React.SetStateAction<boolean>>
|
|
||||||
] => {
|
|
||||||
const [darkMode, setDarkMode] = useLocalStorage(key, initialValue);
|
const [darkMode, setDarkMode] = useLocalStorage(key, initialValue);
|
||||||
const prefersDarkMode = usePrefersDarkMode();
|
const prefersDarkMode = usePrefersDarkMode();
|
||||||
const [selectedThemeMode, setSelectedThemeMode] = useLocalStorage("selectedThemeMode", false);
|
const [selectedThemeMode, setSelectedThemeMode] = useLocalStorage("selectedThemeMode", false);
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
import { useLocalStorage } from "usehooks-ts";
|
import { useLocalStorage } from "usehooks-ts";
|
||||||
|
import { Dispatch, SetStateAction } from "react";
|
||||||
import { LibraryItemUserStatus } from "types/types";
|
import { LibraryItemUserStatus } from "types/types";
|
||||||
|
|
||||||
export const useLibraryItemUserStatus = (): {
|
export const useLibraryItemUserStatus = (): {
|
||||||
libraryItemUserStatus: Record<string, LibraryItemUserStatus>;
|
libraryItemUserStatus: Record<string, LibraryItemUserStatus>;
|
||||||
setLibraryItemUserStatus: React.Dispatch<
|
setLibraryItemUserStatus: Dispatch<SetStateAction<Record<string, LibraryItemUserStatus>>>;
|
||||||
React.SetStateAction<Record<string, LibraryItemUserStatus>>
|
|
||||||
>;
|
|
||||||
} => {
|
} => {
|
||||||
const [libraryItemUserStatus, setLibraryItemUserStatus] = useLocalStorage(
|
const [libraryItemUserStatus, setLibraryItemUserStatus] = useLocalStorage(
|
||||||
"libraryItemUserStatus",
|
"libraryItemUserStatus",
|
||||||
|
|
|
@ -0,0 +1,115 @@
|
||||||
|
import { Dispatch, SetStateAction, useCallback, useMemo } from "react";
|
||||||
|
import { useLocalStorage } from "usehooks-ts";
|
||||||
|
import { ImageQuality } from "helpers/img";
|
||||||
|
|
||||||
|
export interface FilterSettings {
|
||||||
|
paperTexture: boolean;
|
||||||
|
bookFold: boolean;
|
||||||
|
lighting: boolean;
|
||||||
|
teint: number;
|
||||||
|
dropShadow: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ReaderSettings extends FilterSettings {
|
||||||
|
pageQuality: ImageQuality;
|
||||||
|
isSidePagesEnabled: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DEFAULT_READER_SETTINGS: ReaderSettings = {
|
||||||
|
bookFold: true,
|
||||||
|
lighting: true,
|
||||||
|
paperTexture: true,
|
||||||
|
teint: 0.1,
|
||||||
|
dropShadow: true,
|
||||||
|
pageQuality: ImageQuality.Large,
|
||||||
|
isSidePagesEnabled: true,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useReaderSettings = (): {
|
||||||
|
filterSettings: FilterSettings;
|
||||||
|
isSidePagesEnabled: ReaderSettings["isSidePagesEnabled"];
|
||||||
|
pageQuality: ReaderSettings["pageQuality"];
|
||||||
|
toggleBookFold: () => void;
|
||||||
|
toggleLighting: () => void;
|
||||||
|
togglePaperTexture: () => void;
|
||||||
|
toggleDropShadow: () => void;
|
||||||
|
toggleIsSidePagesEnabled: () => void;
|
||||||
|
setPageQuality: Dispatch<SetStateAction<ImageQuality>>;
|
||||||
|
setTeint: Dispatch<SetStateAction<number>>;
|
||||||
|
resetReaderSettings: () => void;
|
||||||
|
} => {
|
||||||
|
const [bookFold, setBookFold] = useLocalStorage(
|
||||||
|
"readerBookFold",
|
||||||
|
DEFAULT_READER_SETTINGS.bookFold
|
||||||
|
);
|
||||||
|
const [lighting, setLighting] = useLocalStorage(
|
||||||
|
"readerLighting",
|
||||||
|
DEFAULT_READER_SETTINGS.lighting
|
||||||
|
);
|
||||||
|
const [paperTexture, setPaperTexture] = useLocalStorage(
|
||||||
|
"readerPaperTexture",
|
||||||
|
DEFAULT_READER_SETTINGS.paperTexture
|
||||||
|
);
|
||||||
|
const [teint, setTeint] = useLocalStorage("readerTeint", DEFAULT_READER_SETTINGS.teint);
|
||||||
|
const [dropShadow, setDropShadow] = useLocalStorage(
|
||||||
|
"readerDropShadow",
|
||||||
|
DEFAULT_READER_SETTINGS.dropShadow
|
||||||
|
);
|
||||||
|
const [isSidePagesEnabled, setIsSidePagesEnabled] = useLocalStorage(
|
||||||
|
"readerIsSidePagesEnabled",
|
||||||
|
DEFAULT_READER_SETTINGS.isSidePagesEnabled
|
||||||
|
);
|
||||||
|
const [pageQuality, setPageQuality] = useLocalStorage(
|
||||||
|
"readerPageQuality",
|
||||||
|
DEFAULT_READER_SETTINGS.pageQuality
|
||||||
|
);
|
||||||
|
|
||||||
|
const toggleBookFold = useCallback(() => setBookFold((current) => !current), [setBookFold]);
|
||||||
|
const toggleLighting = useCallback(() => setLighting((current) => !current), [setLighting]);
|
||||||
|
const togglePaperTexture = useCallback(
|
||||||
|
() => setPaperTexture((current) => !current),
|
||||||
|
[setPaperTexture]
|
||||||
|
);
|
||||||
|
const toggleDropShadow = useCallback(() => setDropShadow((current) => !current), [setDropShadow]);
|
||||||
|
const toggleIsSidePagesEnabled = useCallback(
|
||||||
|
() => setIsSidePagesEnabled((current) => !current),
|
||||||
|
[setIsSidePagesEnabled]
|
||||||
|
);
|
||||||
|
|
||||||
|
const resetReaderSettings = useCallback(() => {
|
||||||
|
setBookFold(DEFAULT_READER_SETTINGS.bookFold);
|
||||||
|
setLighting(DEFAULT_READER_SETTINGS.lighting);
|
||||||
|
setPaperTexture(DEFAULT_READER_SETTINGS.paperTexture);
|
||||||
|
setTeint(DEFAULT_READER_SETTINGS.teint);
|
||||||
|
setDropShadow(DEFAULT_READER_SETTINGS.dropShadow);
|
||||||
|
setIsSidePagesEnabled(DEFAULT_READER_SETTINGS.isSidePagesEnabled);
|
||||||
|
setPageQuality(DEFAULT_READER_SETTINGS.pageQuality);
|
||||||
|
}, [
|
||||||
|
setBookFold,
|
||||||
|
setDropShadow,
|
||||||
|
setIsSidePagesEnabled,
|
||||||
|
setLighting,
|
||||||
|
setPageQuality,
|
||||||
|
setPaperTexture,
|
||||||
|
setTeint,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const filterSettings = useMemo(
|
||||||
|
() => ({ bookFold, lighting, paperTexture, teint, dropShadow }),
|
||||||
|
[bookFold, dropShadow, lighting, paperTexture, teint]
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
filterSettings,
|
||||||
|
isSidePagesEnabled,
|
||||||
|
pageQuality,
|
||||||
|
toggleBookFold,
|
||||||
|
toggleLighting,
|
||||||
|
togglePaperTexture,
|
||||||
|
toggleDropShadow,
|
||||||
|
toggleIsSidePagesEnabled,
|
||||||
|
setPageQuality,
|
||||||
|
setTeint,
|
||||||
|
resetReaderSettings,
|
||||||
|
};
|
||||||
|
};
|
|
@ -1,10 +1,10 @@
|
||||||
import { useEffect, useState } from "react";
|
import { Dispatch, SetStateAction, useEffect, useState } from "react";
|
||||||
import { isDefined } from "helpers/others";
|
import { isDefined } from "helpers/others";
|
||||||
|
|
||||||
export const useStateWithLocalStorage = <T>(
|
export const useStateWithLocalStorage = <T>(
|
||||||
key: string,
|
key: string,
|
||||||
initialValue: T
|
initialValue: T
|
||||||
): [T, React.Dispatch<React.SetStateAction<T>>] => {
|
): [T, Dispatch<SetStateAction<T>>] => {
|
||||||
const [value, setValue] = useState<T>(initialValue);
|
const [value, setValue] = useState<T>(initialValue);
|
||||||
const [isFromLocaleStorage, setFromLocaleStorage] = useState<boolean>(false);
|
const [isFromLocaleStorage, setFromLocaleStorage] = useState<boolean>(false);
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
import { GetStaticPaths, GetStaticPathsResult, GetStaticProps } from "next";
|
import { GetStaticPaths, GetStaticPathsResult, GetStaticProps } from "next";
|
||||||
import { Fragment, useCallback, useEffect, useMemo, useState } from "react";
|
import { Fragment, useCallback, useEffect, useMemo, useState } from "react";
|
||||||
import Hotkeys from "react-hot-keys";
|
import Hotkeys from "react-hot-keys";
|
||||||
import { useBoolean } from "usehooks-ts";
|
|
||||||
import Slider from "rc-slider";
|
import Slider from "rc-slider";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { TransformWrapper, TransformComponent } from "react-zoom-pan-pinch";
|
import { TransformWrapper, TransformComponent } from "react-zoom-pan-pinch";
|
||||||
|
@ -45,6 +44,7 @@ import { prettyInlineTitle, prettySlug } from "helpers/formatters";
|
||||||
import { useFullscreen } from "hooks/useFullscreen";
|
import { useFullscreen } from "hooks/useFullscreen";
|
||||||
import { useUserSettings } from "contexts/UserSettingsContext";
|
import { useUserSettings } from "contexts/UserSettingsContext";
|
||||||
import { useLocalData } from "contexts/LocalDataContext";
|
import { useLocalData } from "contexts/LocalDataContext";
|
||||||
|
import { FilterSettings, useReaderSettings } from "hooks/useReaderSettings";
|
||||||
|
|
||||||
const CUSTOM_DARK_DROPSHADOW = `
|
const CUSTOM_DARK_DROPSHADOW = `
|
||||||
drop-shadow(0 0 0.5em rgb(var(--theme-color-shade) / 30%))
|
drop-shadow(0 0 0.5em rgb(var(--theme-color-shade) / 30%))
|
||||||
|
@ -61,29 +61,9 @@ const CUSTOM_LIGHT_DROPSHADOW = `
|
||||||
const SIDEPAGES_PAGE_COUNT_ON_TEXTURE = 200;
|
const SIDEPAGES_PAGE_COUNT_ON_TEXTURE = 200;
|
||||||
const SIDEPAGES_PAGE_WIDTH = 0.02;
|
const SIDEPAGES_PAGE_WIDTH = 0.02;
|
||||||
|
|
||||||
const DEFAULT_READER_OPTIONS = {
|
|
||||||
pageQuality: ImageQuality.Large,
|
|
||||||
isSidePagesEnabled: true,
|
|
||||||
filterOptions: {
|
|
||||||
bookFold: true,
|
|
||||||
lighting: true,
|
|
||||||
paperTexture: true,
|
|
||||||
teint: 0.1,
|
|
||||||
dropShadow: true,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
type BookType = "book" | "manga";
|
type BookType = "book" | "manga";
|
||||||
type DisplayMode = "double" | "single";
|
type DisplayMode = "double" | "single";
|
||||||
|
|
||||||
interface FilterOptions {
|
|
||||||
paperTexture: boolean;
|
|
||||||
bookFold: boolean;
|
|
||||||
lighting: boolean;
|
|
||||||
teint: number;
|
|
||||||
dropShadow: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ╭────────╮
|
* ╭────────╮
|
||||||
* ──────────────────────────────────────────╯ PAGE ╰─────────────────────────────────────────────
|
* ──────────────────────────────────────────╯ PAGE ╰─────────────────────────────────────────────
|
||||||
|
@ -113,6 +93,19 @@ const LibrarySlug = ({
|
||||||
const is1ColumnLayout = useIs1ColumnLayout();
|
const is1ColumnLayout = useIs1ColumnLayout();
|
||||||
const { langui } = useLocalData();
|
const { langui } = useLocalData();
|
||||||
const { darkMode } = useUserSettings();
|
const { darkMode } = useUserSettings();
|
||||||
|
const {
|
||||||
|
filterSettings,
|
||||||
|
isSidePagesEnabled,
|
||||||
|
pageQuality,
|
||||||
|
toggleBookFold,
|
||||||
|
toggleLighting,
|
||||||
|
togglePaperTexture,
|
||||||
|
toggleDropShadow,
|
||||||
|
toggleIsSidePagesEnabled,
|
||||||
|
setPageQuality,
|
||||||
|
setTeint,
|
||||||
|
resetReaderSettings,
|
||||||
|
} = useReaderSettings();
|
||||||
const [currentPageIndex, setCurrentPageIndex] = useState(0);
|
const [currentPageIndex, setCurrentPageIndex] = useState(0);
|
||||||
const [currentZoom, setCurrentZoom] = useState(1);
|
const [currentZoom, setCurrentZoom] = useState(1);
|
||||||
const [isGalleryMode, setIsGalleryMode] = useState(false);
|
const [isGalleryMode, setIsGalleryMode] = useState(false);
|
||||||
|
@ -120,15 +113,7 @@ const LibrarySlug = ({
|
||||||
is1ColumnLayout ? "single" : "double"
|
is1ColumnLayout ? "single" : "double"
|
||||||
);
|
);
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const [filterOptions, setFilterOptions] = useState<FilterOptions>(
|
|
||||||
DEFAULT_READER_OPTIONS.filterOptions
|
|
||||||
);
|
|
||||||
const [pageQuality, setPageQuality] = useState<ImageQuality>(DEFAULT_READER_OPTIONS.pageQuality);
|
|
||||||
const {
|
|
||||||
value: isSidePagesEnabled,
|
|
||||||
toggle: toggleSidePagesEnabled,
|
|
||||||
setValue: setIsSidePagesEnabled,
|
|
||||||
} = useBoolean(DEFAULT_READER_OPTIONS.isSidePagesEnabled);
|
|
||||||
const { isFullscreen, toggleFullscreen } = useFullscreen(Ids.ContentPanel);
|
const { isFullscreen, toggleFullscreen } = useFullscreen(Ids.ContentPanel);
|
||||||
|
|
||||||
const effectiveDisplayMode = useMemo(
|
const effectiveDisplayMode = useMemo(
|
||||||
|
@ -254,7 +239,7 @@ const LibrarySlug = ({
|
||||||
}
|
}
|
||||||
70% 0%,
|
70% 0%,
|
||||||
${
|
${
|
||||||
filterOptions.bookFold
|
filterSettings.bookFold
|
||||||
? `90% .25%,
|
? `90% .25%,
|
||||||
95% .5%,
|
95% .5%,
|
||||||
98% .8%,
|
98% .8%,
|
||||||
|
@ -269,7 +254,7 @@ const LibrarySlug = ({
|
||||||
}
|
}
|
||||||
70% 100%
|
70% 100%
|
||||||
)`,
|
)`,
|
||||||
[filterOptions.bookFold, isSidePagesEnabled, leftSidePagesWidth]
|
[filterSettings.bookFold, isSidePagesEnabled, leftSidePagesWidth]
|
||||||
);
|
);
|
||||||
|
|
||||||
const rightSideClipPath = useMemo(
|
const rightSideClipPath = useMemo(
|
||||||
|
@ -284,7 +269,7 @@ const LibrarySlug = ({
|
||||||
}
|
}
|
||||||
30% 100%,
|
30% 100%,
|
||||||
${
|
${
|
||||||
filterOptions.bookFold
|
filterSettings.bookFold
|
||||||
? `10% 99.75%,
|
? `10% 99.75%,
|
||||||
5% 99.5%,
|
5% 99.5%,
|
||||||
2% 99.2%,
|
2% 99.2%,
|
||||||
|
@ -299,7 +284,7 @@ const LibrarySlug = ({
|
||||||
}
|
}
|
||||||
30% 0%
|
30% 0%
|
||||||
)`,
|
)`,
|
||||||
[filterOptions.bookFold, isSidePagesEnabled, rightSidePagesWidth]
|
[filterSettings.bookFold, isSidePagesEnabled, rightSidePagesWidth]
|
||||||
);
|
);
|
||||||
|
|
||||||
const pageHeight = useMemo(
|
const pageHeight = useMemo(
|
||||||
|
@ -314,43 +299,23 @@ const LibrarySlug = ({
|
||||||
|
|
||||||
<div className="mt-4 grid gap-2">
|
<div className="mt-4 grid gap-2">
|
||||||
<WithLabel label={langui.paper_texture}>
|
<WithLabel label={langui.paper_texture}>
|
||||||
<Switch
|
<Switch value={filterSettings.paperTexture} onClick={togglePaperTexture} />
|
||||||
value={filterOptions.paperTexture}
|
|
||||||
onClick={() =>
|
|
||||||
setFilterOptions((current) => ({ ...current, paperTexture: !current.paperTexture }))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</WithLabel>
|
</WithLabel>
|
||||||
|
|
||||||
<WithLabel label={langui.book_fold}>
|
<WithLabel label={langui.book_fold}>
|
||||||
<Switch
|
<Switch value={filterSettings.bookFold} onClick={toggleBookFold} />
|
||||||
value={filterOptions.bookFold}
|
|
||||||
onClick={() =>
|
|
||||||
setFilterOptions((current) => ({ ...current, bookFold: !current.bookFold }))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</WithLabel>
|
</WithLabel>
|
||||||
|
|
||||||
<WithLabel label={langui.lighting}>
|
<WithLabel label={langui.lighting}>
|
||||||
<Switch
|
<Switch value={filterSettings.lighting} onClick={toggleLighting} />
|
||||||
value={filterOptions.lighting}
|
|
||||||
onClick={() =>
|
|
||||||
setFilterOptions((current) => ({ ...current, lighting: !current.lighting }))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</WithLabel>
|
</WithLabel>
|
||||||
|
|
||||||
<WithLabel label={langui.side_pages}>
|
<WithLabel label={langui.side_pages}>
|
||||||
<Switch value={isSidePagesEnabled} onClick={toggleSidePagesEnabled} />
|
<Switch value={isSidePagesEnabled} onClick={toggleIsSidePagesEnabled} />
|
||||||
</WithLabel>
|
</WithLabel>
|
||||||
|
|
||||||
<WithLabel label={langui.shadow}>
|
<WithLabel label={langui.shadow}>
|
||||||
<Switch
|
<Switch value={filterSettings.dropShadow} onClick={toggleDropShadow} />
|
||||||
value={filterOptions.dropShadow}
|
|
||||||
onClick={() =>
|
|
||||||
setFilterOptions((current) => ({ ...current, dropShadow: !current.dropShadow }))
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</WithLabel>
|
</WithLabel>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -359,7 +324,7 @@ const LibrarySlug = ({
|
||||||
<Slider
|
<Slider
|
||||||
min={0}
|
min={0}
|
||||||
max={10}
|
max={10}
|
||||||
value={filterOptions.teint * 10}
|
value={filterSettings.teint * 10}
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
let value = 0;
|
let value = 0;
|
||||||
if (Array.isArray(event)) {
|
if (Array.isArray(event)) {
|
||||||
|
@ -367,10 +332,7 @@ const LibrarySlug = ({
|
||||||
} else {
|
} else {
|
||||||
value = event;
|
value = event;
|
||||||
}
|
}
|
||||||
setFilterOptions((current) => ({
|
setTeint(value / 10);
|
||||||
...current,
|
|
||||||
teint: value / 10,
|
|
||||||
}));
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
@ -418,9 +380,7 @@ const LibrarySlug = ({
|
||||||
text={langui.reset_all_options}
|
text={langui.reset_all_options}
|
||||||
icon={Icon.Replay}
|
icon={Icon.Replay}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setFilterOptions(DEFAULT_READER_OPTIONS.filterOptions);
|
resetReaderSettings();
|
||||||
setPageQuality(DEFAULT_READER_OPTIONS.pageQuality);
|
|
||||||
setIsSidePagesEnabled(DEFAULT_READER_OPTIONS.isSidePagesEnabled);
|
|
||||||
setDisplayMode(is1ColumnLayout ? "single" : "double");
|
setDisplayMode(is1ColumnLayout ? "single" : "double");
|
||||||
sendAnalytics("Reader", "Reset all options");
|
sendAnalytics("Reader", "Reset all options");
|
||||||
}}
|
}}
|
||||||
|
@ -428,19 +388,36 @@ const LibrarySlug = ({
|
||||||
</SubPanel>
|
</SubPanel>
|
||||||
),
|
),
|
||||||
[
|
[
|
||||||
langui,
|
langui.item,
|
||||||
|
langui.paper_texture,
|
||||||
|
langui.book_fold,
|
||||||
|
langui.lighting,
|
||||||
|
langui.side_pages,
|
||||||
|
langui.shadow,
|
||||||
|
langui.night_reader,
|
||||||
|
langui.reading_layout,
|
||||||
|
langui.single_page_view,
|
||||||
|
langui.double_page_view,
|
||||||
|
langui.quality,
|
||||||
|
langui.reset_all_options,
|
||||||
itemSlug,
|
itemSlug,
|
||||||
filterOptions.paperTexture,
|
filterSettings.paperTexture,
|
||||||
filterOptions.bookFold,
|
filterSettings.bookFold,
|
||||||
filterOptions.lighting,
|
filterSettings.lighting,
|
||||||
filterOptions.dropShadow,
|
filterSettings.dropShadow,
|
||||||
filterOptions.teint,
|
filterSettings.teint,
|
||||||
|
togglePaperTexture,
|
||||||
|
toggleBookFold,
|
||||||
|
toggleLighting,
|
||||||
isSidePagesEnabled,
|
isSidePagesEnabled,
|
||||||
toggleSidePagesEnabled,
|
toggleIsSidePagesEnabled,
|
||||||
|
toggleDropShadow,
|
||||||
displayMode,
|
displayMode,
|
||||||
pageQuality,
|
pageQuality,
|
||||||
|
setTeint,
|
||||||
changeDisplayMode,
|
changeDisplayMode,
|
||||||
setIsSidePagesEnabled,
|
setPageQuality,
|
||||||
|
resetReaderSettings,
|
||||||
is1ColumnLayout,
|
is1ColumnLayout,
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
|
@ -468,7 +445,7 @@ const LibrarySlug = ({
|
||||||
gridAutoFlow: "column",
|
gridAutoFlow: "column",
|
||||||
display: "grid",
|
display: "grid",
|
||||||
placeContent: "center",
|
placeContent: "center",
|
||||||
filter: filterOptions.dropShadow
|
filter: filterSettings.dropShadow
|
||||||
? darkMode
|
? darkMode
|
||||||
? CUSTOM_DARK_DROPSHADOW
|
? CUSTOM_DARK_DROPSHADOW
|
||||||
: CUSTOM_LIGHT_DROPSHADOW
|
: CUSTOM_LIGHT_DROPSHADOW
|
||||||
|
@ -485,7 +462,7 @@ const LibrarySlug = ({
|
||||||
src={firstPage}
|
src={firstPage}
|
||||||
quality={pageQuality}
|
quality={pageQuality}
|
||||||
/>
|
/>
|
||||||
<PageFilters page="single" bookType={bookType} options={filterOptions} />
|
<PageFilters page="single" bookType={bookType} options={filterSettings} />
|
||||||
<div
|
<div
|
||||||
className="absolute left-0 top-0 bottom-0 w-1/2"
|
className="absolute left-0 top-0 bottom-0 w-1/2"
|
||||||
onClick={() => currentZoom <= 1 && handlePageNavigation("left")}
|
onClick={() => currentZoom <= 1 && handlePageNavigation("left")}
|
||||||
|
@ -523,7 +500,7 @@ const LibrarySlug = ({
|
||||||
src={pageOrder === PageOrder.LeftToRight ? firstPage : secondPage}
|
src={pageOrder === PageOrder.LeftToRight ? firstPage : secondPage}
|
||||||
quality={pageQuality}
|
quality={pageQuality}
|
||||||
/>
|
/>
|
||||||
<PageFilters page="left" bookType={bookType} options={filterOptions} />
|
<PageFilters page="left" bookType={bookType} options={filterSettings} />
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
className={cJoin(
|
className={cJoin(
|
||||||
|
@ -557,7 +534,7 @@ const LibrarySlug = ({
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<PageFilters page="right" bookType={bookType} options={filterOptions} />
|
<PageFilters page="right" bookType={bookType} options={filterSettings} />
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
@ -647,7 +624,7 @@ const LibrarySlug = ({
|
||||||
[
|
[
|
||||||
is1ColumnLayout,
|
is1ColumnLayout,
|
||||||
currentZoom,
|
currentZoom,
|
||||||
filterOptions,
|
filterSettings,
|
||||||
darkMode,
|
darkMode,
|
||||||
pageHeight,
|
pageHeight,
|
||||||
effectiveDisplayMode,
|
effectiveDisplayMode,
|
||||||
|
@ -811,7 +788,7 @@ export const getStaticPaths: GetStaticPaths = async (context) => {
|
||||||
interface PageFiltersProps {
|
interface PageFiltersProps {
|
||||||
page: "left" | "right" | "single";
|
page: "left" | "right" | "single";
|
||||||
bookType: BookType;
|
bookType: BookType;
|
||||||
options: FilterOptions;
|
options: FilterSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
const PageFilters = ({ page, bookType, options }: PageFiltersProps) => {
|
const PageFilters = ({ page, bookType, options }: PageFiltersProps) => {
|
||||||
|
|
Loading…
Reference in New Issue