mirror of
				https://github.com/zoriya/Kyoo.git
				synced 2025-10-26 08:12:35 -04:00 
			
		
		
		
	Handle invalid users on mobile too
This commit is contained in:
		
							parent
							
								
									b08bfeceb6
								
							
						
					
					
						commit
						44c88a885f
					
				| @ -55,7 +55,7 @@ import arrayShuffle from "array-shuffle"; | |||||||
| import { Tooltip } from "react-tooltip"; | import { Tooltip } from "react-tooltip"; | ||||||
| import { getCurrentAccount, readCookie, updateAccount } from "@kyoo/models/src/account-internal"; | import { getCurrentAccount, readCookie, updateAccount } from "@kyoo/models/src/account-internal"; | ||||||
| import { PortalProvider } from "@gorhom/portal"; | import { PortalProvider } from "@gorhom/portal"; | ||||||
| import { ConnectionError, ErrorContext } from "@kyoo/ui"; | import { ConnectionError } from "@kyoo/ui"; | ||||||
| 
 | 
 | ||||||
| const font = Poppins({ weight: ["300", "400", "900"], subsets: ["latin"], display: "swap" }); | const font = Poppins({ weight: ["300", "400", "900"], subsets: ["latin"], display: "swap" }); | ||||||
| 
 | 
 | ||||||
| @ -136,17 +136,7 @@ const WithLayout = ({ Component, ...props }: { Component: ComponentType }) => { | |||||||
| 	const layoutInfo = (Component as QueryPage).getLayout ?? (({ page }) => page); | 	const layoutInfo = (Component as QueryPage).getLayout ?? (({ page }) => page); | ||||||
| 	const { Layout, props: layoutProps } = | 	const { Layout, props: layoutProps } = | ||||||
| 		typeof layoutInfo === "function" ? { Layout: layoutInfo, props: {} } : layoutInfo; | 		typeof layoutInfo === "function" ? { Layout: layoutInfo, props: {} } : layoutInfo; | ||||||
| 	return ( | 	return <Layout page={<Component {...props} />} randomItems={[]} {...layoutProps} />; | ||||||
| 		<Layout |  | ||||||
| 			page={ |  | ||||||
| 				<ErrorContext> |  | ||||||
| 					<Component {...props} /> |  | ||||||
| 				</ErrorContext> |  | ||||||
| 			} |  | ||||||
| 			randomItems={[]} |  | ||||||
| 			{...layoutProps} |  | ||||||
| 		/> |  | ||||||
| 	); |  | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| const App = ({ Component, pageProps }: AppProps) => { | const App = ({ Component, pageProps }: AppProps) => { | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ | |||||||
|  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 |  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { ReactNode, createContext, useContext, useEffect, useMemo, useRef } from "react"; | import { ReactNode, createContext, useContext, useEffect, useMemo, useRef, useState } from "react"; | ||||||
| import { ServerInfoP, User, UserP } from "./resources"; | import { ServerInfoP, User, UserP } from "./resources"; | ||||||
| import { z } from "zod"; | import { z } from "zod"; | ||||||
| import { zdate } from "./utils"; | import { zdate } from "./utils"; | ||||||
| @ -69,7 +69,8 @@ export const ConnectionErrorContext = createContext<{ | |||||||
| 	error: KyooErrors | null; | 	error: KyooErrors | null; | ||||||
| 	loading: boolean; | 	loading: boolean; | ||||||
| 	retry?: () => void; | 	retry?: () => void; | ||||||
| }>({ error: null, loading: true }); | 	setError: (error: KyooErrors) => void; | ||||||
|  | }>({ error: null, loading: true, setError: () => {} }); | ||||||
| 
 | 
 | ||||||
| /* eslint-disable react-hooks/rules-of-hooks */ | /* eslint-disable react-hooks/rules-of-hooks */ | ||||||
| export const AccountProvider = ({ | export const AccountProvider = ({ | ||||||
| @ -96,6 +97,7 @@ export const AccountProvider = ({ | |||||||
| 						retry: () => { | 						retry: () => { | ||||||
| 							queryClient.resetQueries({ queryKey: ["auth", "me"] }); | 							queryClient.resetQueries({ queryKey: ["auth", "me"] }); | ||||||
| 						}, | 						}, | ||||||
|  | 						setError: () => {}, | ||||||
| 					}} | 					}} | ||||||
| 				> | 				> | ||||||
| 					{children} | 					{children} | ||||||
| @ -156,15 +158,18 @@ export const AccountProvider = ({ | |||||||
| 		} | 		} | ||||||
| 	}, [selected, queryClient]); | 	}, [selected, queryClient]); | ||||||
| 
 | 
 | ||||||
|  | 	const [permissionError, setPermissionError] = useState<KyooErrors | null>(null); | ||||||
|  | 
 | ||||||
| 	return ( | 	return ( | ||||||
| 		<AccountContext.Provider value={accounts}> | 		<AccountContext.Provider value={accounts}> | ||||||
| 			<ConnectionErrorContext.Provider | 			<ConnectionErrorContext.Provider | ||||||
| 				value={{ | 				value={{ | ||||||
| 					error: selected ? initialSsrError.current ?? user.error : null, | 					error: selected ? initialSsrError.current ?? user.error ?? permissionError : null, | ||||||
| 					loading: user.isLoading, | 					loading: user.isLoading, | ||||||
| 					retry: () => { | 					retry: () => { | ||||||
| 						queryClient.invalidateQueries({ queryKey: ["auth", "me"] }); | 						queryClient.invalidateQueries({ queryKey: ["auth", "me"] }); | ||||||
| 					}, | 					}, | ||||||
|  | 					setError: setPermissionError, | ||||||
| 				}} | 				}} | ||||||
| 			> | 			> | ||||||
| 				{children} | 				{children} | ||||||
|  | |||||||
| @ -18,21 +18,49 @@ | |||||||
|  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 |  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { ConnectionErrorContext } from "@kyoo/models"; | import { ConnectionErrorContext, useAccount } from "@kyoo/models"; | ||||||
| import { Button, H1, P, ts } from "@kyoo/primitives"; | import { Button, H1, Icon, Link, P, ts } from "@kyoo/primitives"; | ||||||
| import { useRouter } from "solito/router"; | import { useRouter } from "solito/router"; | ||||||
| import { useContext } from "react"; | import { useContext } from "react"; | ||||||
| import { useTranslation } from "react-i18next"; | import { useTranslation } from "react-i18next"; | ||||||
| import { View } from "react-native"; | import { View } from "react-native"; | ||||||
| import { useYoshiki } from "yoshiki/native"; | import { useYoshiki } from "yoshiki/native"; | ||||||
| import { DefaultLayout } from "../layout"; | import { DefaultLayout } from "../layout"; | ||||||
|  | import { ErrorView } from "./error"; | ||||||
|  | import Register from "@material-symbols/svg-400/rounded/app_registration.svg"; | ||||||
| 
 | 
 | ||||||
| export const ConnectionError = () => { | export const ConnectionError = () => { | ||||||
| 	const { css } = useYoshiki(); | 	const { css } = useYoshiki(); | ||||||
| 	const { t } = useTranslation(); | 	const { t } = useTranslation(); | ||||||
| 	const router = useRouter(); | 	const router = useRouter(); | ||||||
| 	const { error, retry } = useContext(ConnectionErrorContext); | 	const { error, retry } = useContext(ConnectionErrorContext); | ||||||
|  | 	const account = useAccount(); | ||||||
| 
 | 
 | ||||||
|  | 	if (error && (error.status === 401 || error.status == 403)) { | ||||||
|  | 		if (!account) { | ||||||
|  | 			return ( | ||||||
|  | 				<View | ||||||
|  | 					{...css({ flexGrow: 1, flexShrink: 1, justifyContent: "center", alignItems: "center" })} | ||||||
|  | 				> | ||||||
|  | 					<P>{t("errors.needAccount")}</P> | ||||||
|  | 					<Button | ||||||
|  | 						as={Link} | ||||||
|  | 						href={"/register"} | ||||||
|  | 						text={t("login.register")} | ||||||
|  | 						licon={<Icon icon={Register} {...css({ marginRight: ts(2) })} />} | ||||||
|  | 					/> | ||||||
|  | 				</View> | ||||||
|  | 			); | ||||||
|  | 		} | ||||||
|  | 		if (account.isVerified) return <ErrorView error={error} noBubble />; | ||||||
|  | 		return ( | ||||||
|  | 			<View | ||||||
|  | 				{...css({ flexGrow: 1, flexShrink: 1, justifyContent: "center", alignItems: "center" })} | ||||||
|  | 			> | ||||||
|  | 				<P>{t("errors.needVerification")}</P> | ||||||
|  | 			</View> | ||||||
|  | 		); | ||||||
|  | 	} | ||||||
| 	return ( | 	return ( | ||||||
| 		<View {...css({ padding: ts(2) })}> | 		<View {...css({ padding: ts(2) })}> | ||||||
| 			<H1 {...css({ textAlign: "center" })}>{t("errors.connection")}</H1> | 			<H1 {...css({ textAlign: "center" })}>{t("errors.connection")}</H1> | ||||||
|  | |||||||
| @ -18,19 +18,11 @@ | |||||||
|  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 |  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { KyooErrors, useAccount } from "@kyoo/models"; | import { ConnectionErrorContext, KyooErrors } from "@kyoo/models"; | ||||||
| import { P } from "@kyoo/primitives"; | import { P } from "@kyoo/primitives"; | ||||||
| import { | import { useContext, useLayoutEffect } from "react"; | ||||||
| 	ReactElement, |  | ||||||
| 	createContext, |  | ||||||
| 	useContext, |  | ||||||
| 	useEffect, |  | ||||||
| 	useLayoutEffect, |  | ||||||
| 	useState, |  | ||||||
| } from "react"; |  | ||||||
| import { View } from "react-native"; | import { View } from "react-native"; | ||||||
| import { useYoshiki } from "yoshiki/native"; | import { useYoshiki } from "yoshiki/native"; | ||||||
| import { PermissionError } from "./unauthorized"; |  | ||||||
| 
 | 
 | ||||||
| export const ErrorView = ({ | export const ErrorView = ({ | ||||||
| 	error, | 	error, | ||||||
| @ -40,7 +32,7 @@ export const ErrorView = ({ | |||||||
| 	noBubble?: boolean; | 	noBubble?: boolean; | ||||||
| }) => { | }) => { | ||||||
| 	const { css } = useYoshiki(); | 	const { css } = useYoshiki(); | ||||||
| 	const setError = useErrorContext(); | 	const { setError } = useContext(ConnectionErrorContext); | ||||||
| 
 | 
 | ||||||
| 	useLayoutEffect(() => { | 	useLayoutEffect(() => { | ||||||
| 		// if this is a permission error, make it go up the tree to have a whole page login screen.
 | 		// if this is a permission error, make it go up the tree to have a whole page login screen.
 | ||||||
| @ -65,22 +57,3 @@ export const ErrorView = ({ | |||||||
| 		</View> | 		</View> | ||||||
| 	); | 	); | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| const ErrorCtx = createContext<(val: KyooErrors | null) => void>(null!); |  | ||||||
| 
 |  | ||||||
| export const ErrorContext = ({ children }: { children: ReactElement }) => { |  | ||||||
| 	const [error, setError] = useState<KyooErrors | null>(null); |  | ||||||
| 	const account = useAccount(); |  | ||||||
| 
 |  | ||||||
| 	useEffect(() => { |  | ||||||
| 		setError(null); |  | ||||||
| 	}, [account, children]); |  | ||||||
| 
 |  | ||||||
| 	if (error && (error.status === 401 || error.status === 403)) |  | ||||||
| 		return <PermissionError error={error} />; |  | ||||||
| 	if (error) return <ErrorView error={error} noBubble />; |  | ||||||
| 	return <ErrorCtx.Provider value={setError}>{children}</ErrorCtx.Provider>; |  | ||||||
| }; |  | ||||||
| export const useErrorContext = () => { |  | ||||||
| 	return useContext(ErrorCtx); |  | ||||||
| }; |  | ||||||
|  | |||||||
| @ -18,13 +18,10 @@ | |||||||
|  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 |  * along with Kyoo. If not, see <https://www.gnu.org/licenses/>.
 | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| import { KyooErrors, useAccount } from "@kyoo/models"; | import { P } from "@kyoo/primitives"; | ||||||
| import { Button, Icon, Link, P, ts } from "@kyoo/primitives"; |  | ||||||
| import { useTranslation } from "react-i18next"; | import { useTranslation } from "react-i18next"; | ||||||
| import { View } from "react-native"; | import { View } from "react-native"; | ||||||
| import { rem, useYoshiki } from "yoshiki/native"; | import { useYoshiki } from "yoshiki/native"; | ||||||
| import { ErrorView } from "./error"; |  | ||||||
| import Register from "@material-symbols/svg-400/rounded/app_registration.svg"; |  | ||||||
| 
 | 
 | ||||||
| export const Unauthorized = ({ missing }: { missing: string[] }) => { | export const Unauthorized = ({ missing }: { missing: string[] }) => { | ||||||
| 	const { t } = useTranslation(); | 	const { t } = useTranslation(); | ||||||
| @ -43,31 +40,3 @@ export const Unauthorized = ({ missing }: { missing: string[] }) => { | |||||||
| 		</View> | 		</View> | ||||||
| 	); | 	); | ||||||
| }; | }; | ||||||
| 
 |  | ||||||
| export const PermissionError = ({ error }: { error: KyooErrors }) => { |  | ||||||
| 	const { t } = useTranslation(); |  | ||||||
| 	const { css } = useYoshiki(); |  | ||||||
| 	const account = useAccount(); |  | ||||||
| 
 |  | ||||||
| 	if (!account) { |  | ||||||
| 		return ( |  | ||||||
| 			<View |  | ||||||
| 				{...css({ flexGrow: 1, flexShrink: 1, justifyContent: "center", alignItems: "center" })} |  | ||||||
| 			> |  | ||||||
| 				<P>{t("errors.needAccount")}</P> |  | ||||||
| 				<Button |  | ||||||
| 					as={Link} |  | ||||||
| 					href={"/register"} |  | ||||||
| 					text={t("login.register")} |  | ||||||
| 					licon={<Icon icon={Register} {...css({ marginRight: ts(2) })} />} |  | ||||||
| 				/> |  | ||||||
| 			</View> |  | ||||||
| 		); |  | ||||||
| 	} |  | ||||||
| 	if (account.isVerified) return <ErrorView error={error} noBubble />; |  | ||||||
| 	return ( |  | ||||||
| 		<View {...css({ flexGrow: 1, flexShrink: 1, justifyContent: "center", alignItems: "center" })}> |  | ||||||
| 			<P>{t("errors.needVerification")}</P> |  | ||||||
| 		</View> |  | ||||||
| 	); |  | ||||||
| }; |  | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user