Fix loading issue in galleries and redirect on gallery creation (#857)

* Fix loading issue in galleries and redirect on gallery creation
* Add error messages when image/galleries aren't found
* Clean up gallery/image/performer/scene view states
* Simplify error messages
This commit is contained in:
InfiniteTF
2020-10-17 00:57:02 +02:00
committed by GitHub
parent bbc6c43201
commit cf003a55bf
12 changed files with 171 additions and 105 deletions

View File

@@ -1,9 +1,8 @@
import { Tab, Nav, Dropdown } from "react-bootstrap"; import { Tab, Nav, Dropdown } from "react-bootstrap";
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
import { useParams, useHistory, Link } from "react-router-dom"; import { useParams, useHistory, Link } from "react-router-dom";
import * as GQL from "src/core/generated-graphql";
import { useFindGallery } from "src/core/StashService"; import { useFindGallery } from "src/core/StashService";
import { LoadingIndicator, Icon } from "src/components/Shared"; import { ErrorMessage, LoadingIndicator, Icon } from "src/components/Shared";
import { TextUtils } from "src/utils"; import { TextUtils } from "src/utils";
import * as Mousetrap from "mousetrap"; import * as Mousetrap from "mousetrap";
import { GalleryEditPanel } from "./GalleryEditPanel"; import { GalleryEditPanel } from "./GalleryEditPanel";
@@ -22,8 +21,8 @@ export const Gallery: React.FC = () => {
const history = useHistory(); const history = useHistory();
const isNew = id === "new"; const isNew = id === "new";
const [gallery, setGallery] = useState<Partial<GQL.GalleryDataFragment>>({});
const { data, error, loading } = useFindGallery(id); const { data, error, loading } = useFindGallery(id);
const gallery = data?.findGallery;
const [activeTabKey, setActiveTabKey] = useState("gallery-details-panel"); const [activeTabKey, setActiveTabKey] = useState("gallery-details-panel");
const activeRightTabKey = tab === "images" || tab === "add" ? tab : "images"; const activeRightTabKey = tab === "images" || tab === "add" ? tab : "images";
@@ -36,10 +35,6 @@ export const Gallery: React.FC = () => {
const [isDeleteAlertOpen, setIsDeleteAlertOpen] = useState<boolean>(false); const [isDeleteAlertOpen, setIsDeleteAlertOpen] = useState<boolean>(false);
useEffect(() => {
if (data?.findGallery) setGallery(data.findGallery);
}, [data]);
function onDeleteDialogClosed(deleted: boolean) { function onDeleteDialogClosed(deleted: boolean) {
setIsDeleteAlertOpen(false); setIsDeleteAlertOpen(false);
if (deleted) { if (deleted) {
@@ -125,8 +120,8 @@ export const Gallery: React.FC = () => {
<Tab.Pane eventKey="gallery-edit-panel" title="Edit"> <Tab.Pane eventKey="gallery-edit-panel" title="Edit">
<GalleryEditPanel <GalleryEditPanel
isVisible={activeTabKey === "gallery-edit-panel"} isVisible={activeTabKey === "gallery-edit-panel"}
isNew={false}
gallery={gallery} gallery={gallery}
onUpdate={(newGallery) => setGallery(newGallery)}
onDelete={() => setIsDeleteAlertOpen(true)} onDelete={() => setIsDeleteAlertOpen(true)}
/> />
</Tab.Pane> </Tab.Pane>
@@ -183,27 +178,29 @@ export const Gallery: React.FC = () => {
}; };
}); });
if (loading) {
return <LoadingIndicator />;
}
if (error) return <ErrorMessage error={error.message} />;
if (isNew) if (isNew)
return ( return (
<div className="row new-view"> <div className="row new-view">
<div className="col-6"> <div className="col-6">
<h2>Create Gallery</h2> <h2>Create Gallery</h2>
<GalleryEditPanel <GalleryEditPanel
gallery={gallery} isNew
gallery={undefined}
isVisible isVisible
isNew={isNew}
onUpdate={(newGallery) => setGallery(newGallery)}
onDelete={() => setIsDeleteAlertOpen(true)} onDelete={() => setIsDeleteAlertOpen(true)}
/> />
</div> </div>
</div> </div>
); );
if (loading || !gallery || !data?.findGallery) { if (!gallery)
return <LoadingIndicator />; return <ErrorMessage error={`No gallery with id ${id} found.`} />;
}
if (error) return <div>{error.message}</div>;
return ( return (
<div className="row"> <div className="row">

View File

@@ -1,4 +1,5 @@
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
import { useHistory } from "react-router-dom";
import { Button, Form, Col, Row } from "react-bootstrap"; import { Button, Form, Col, Row } from "react-bootstrap";
import * as GQL from "src/core/generated-graphql"; import * as GQL from "src/core/generated-graphql";
import { useGalleryCreate, useGalleryUpdate } from "src/core/StashService"; import { useGalleryCreate, useGalleryUpdate } from "src/core/StashService";
@@ -13,15 +14,25 @@ import { FormUtils, EditableTextUtils } from "src/utils";
import { RatingStars } from "src/components/Scenes/SceneDetails/RatingStars"; import { RatingStars } from "src/components/Scenes/SceneDetails/RatingStars";
interface IProps { interface IProps {
gallery: Partial<GQL.GalleryDataFragment>;
isVisible: boolean; isVisible: boolean;
isNew?: boolean;
onUpdate: (gallery: GQL.GalleryDataFragment) => void;
onDelete: () => void; onDelete: () => void;
} }
export const GalleryEditPanel: React.FC<IProps> = (props: IProps) => { interface INewProps {
isNew: true;
gallery: undefined;
}
interface IExistingProps {
isNew: false;
gallery: GQL.GalleryDataFragment;
}
export const GalleryEditPanel: React.FC<
IProps & (INewProps | IExistingProps)
> = (props) => {
const Toast = useToast(); const Toast = useToast();
const history = useHistory();
const [title, setTitle] = useState<string>(); const [title, setTitle] = useState<string>();
const [details, setDetails] = useState<string>(); const [details, setDetails] = useState<string>();
const [url, setUrl] = useState<string>(); const [url, setUrl] = useState<string>();
@@ -83,15 +94,15 @@ export const GalleryEditPanel: React.FC<IProps> = (props: IProps) => {
} }
}); });
function updateGalleryEditState(state: Partial<GQL.GalleryDataFragment>) { function updateGalleryEditState(state?: GQL.GalleryDataFragment) {
const perfIds = state.performers?.map((performer) => performer.id); const perfIds = state?.performers?.map((performer) => performer.id);
const tIds = state.tags ? state.tags.map((tag) => tag.id) : undefined; const tIds = state?.tags ? state?.tags.map((tag) => tag.id) : undefined;
setTitle(state.title ?? undefined); setTitle(state?.title ?? undefined);
setDetails(state.details ?? undefined); setDetails(state?.details ?? undefined);
setUrl(state.url ?? undefined); setUrl(state?.url ?? undefined);
setDate(state.date ?? undefined); setDate(state?.date ?? undefined);
setRating(state.rating === null ? NaN : state.rating); setRating(state?.rating === null ? NaN : state?.rating);
setStudioId(state?.studio?.id ?? undefined); setStudioId(state?.studio?.id ?? undefined);
setPerformerIds(perfIds); setPerformerIds(perfIds);
setTagIds(tIds); setTagIds(tIds);
@@ -104,7 +115,7 @@ export const GalleryEditPanel: React.FC<IProps> = (props: IProps) => {
function getGalleryInput() { function getGalleryInput() {
return { return {
id: props.isNew ? undefined : props.gallery.id!, id: props.isNew ? undefined : props.gallery.id,
title, title,
details, details,
url, url,
@@ -122,13 +133,12 @@ export const GalleryEditPanel: React.FC<IProps> = (props: IProps) => {
if (props.isNew) { if (props.isNew) {
const result = await createGallery(); const result = await createGallery();
if (result.data?.galleryCreate) { if (result.data?.galleryCreate) {
props.onUpdate(result.data.galleryCreate); history.push(`/galleries/${result.data.galleryCreate.id}`);
Toast.success({ content: "Created gallery" }); Toast.success({ content: "Created gallery" });
} }
} else { } else {
const result = await updateGallery(); const result = await updateGallery();
if (result.data?.galleryUpdate) { if (result.data?.galleryUpdate) {
props.onUpdate(result.data.galleryUpdate);
Toast.success({ content: "Updated gallery" }); Toast.success({ content: "Updated gallery" });
} }
} }

View File

@@ -8,7 +8,7 @@ import { showWhenSelected } from "src/hooks/ListHook";
import { useToast } from "src/hooks"; import { useToast } from "src/hooks";
interface IGalleryDetailsProps { interface IGalleryDetailsProps {
gallery: Partial<GQL.GalleryDataFragment>; gallery: GQL.GalleryDataFragment;
} }
export const GalleryImagesPanel: React.FC<IGalleryDetailsProps> = ({ export const GalleryImagesPanel: React.FC<IGalleryDetailsProps> = ({

View File

@@ -1,14 +1,13 @@
import { Tab, Nav, Dropdown } from "react-bootstrap"; import { Tab, Nav, Dropdown } from "react-bootstrap";
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
import { useParams, useHistory, Link } from "react-router-dom"; import { useParams, useHistory, Link } from "react-router-dom";
import * as GQL from "src/core/generated-graphql";
import { import {
useFindImage, useFindImage,
useImageIncrementO, useImageIncrementO,
useImageDecrementO, useImageDecrementO,
useImageResetO, useImageResetO,
} from "src/core/StashService"; } from "src/core/StashService";
import { LoadingIndicator, Icon } from "src/components/Shared"; import { ErrorMessage, LoadingIndicator, Icon } from "src/components/Shared";
import { useToast } from "src/hooks"; import { useToast } from "src/hooks";
import { TextUtils } from "src/utils"; import { TextUtils } from "src/utils";
import * as Mousetrap from "mousetrap"; import * as Mousetrap from "mousetrap";
@@ -27,8 +26,8 @@ export const Image: React.FC = () => {
const history = useHistory(); const history = useHistory();
const Toast = useToast(); const Toast = useToast();
const [image, setImage] = useState<GQL.ImageDataFragment | undefined>();
const { data, error, loading } = useFindImage(id); const { data, error, loading } = useFindImage(id);
const image = data?.findImage;
const [oLoading, setOLoading] = useState(false); const [oLoading, setOLoading] = useState(false);
const [incrementO] = useImageIncrementO(image?.id ?? "0"); const [incrementO] = useImageIncrementO(image?.id ?? "0");
const [decrementO] = useImageDecrementO(image?.id ?? "0"); const [decrementO] = useImageDecrementO(image?.id ?? "0");
@@ -38,21 +37,10 @@ export const Image: React.FC = () => {
const [isDeleteAlertOpen, setIsDeleteAlertOpen] = useState<boolean>(false); const [isDeleteAlertOpen, setIsDeleteAlertOpen] = useState<boolean>(false);
useEffect(() => {
if (data?.findImage) setImage(data.findImage);
}, [data]);
const updateOCounter = (newValue: number) => {
const modifiedImage = { ...image } as GQL.ImageDataFragment;
modifiedImage.o_counter = newValue;
setImage(modifiedImage);
};
const onIncrementClick = async () => { const onIncrementClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await incrementO(); await incrementO();
if (result.data) updateOCounter(result.data.imageIncrementO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -63,8 +51,7 @@ export const Image: React.FC = () => {
const onDecrementClick = async () => { const onDecrementClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await decrementO(); await decrementO();
if (result.data) updateOCounter(result.data.imageDecrementO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -75,8 +62,7 @@ export const Image: React.FC = () => {
const onResetClick = async () => { const onResetClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await resetO(); await resetO();
if (result.data) updateOCounter(result.data.imageResetO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -172,7 +158,6 @@ export const Image: React.FC = () => {
<ImageEditPanel <ImageEditPanel
isVisible={activeTabKey === "image-edit-panel"} isVisible={activeTabKey === "image-edit-panel"}
image={image} image={image}
onUpdate={(newImage) => setImage(newImage)}
onDelete={() => setIsDeleteAlertOpen(true)} onDelete={() => setIsDeleteAlertOpen(true)}
/> />
</Tab.Pane> </Tab.Pane>
@@ -196,11 +181,15 @@ export const Image: React.FC = () => {
}; };
}); });
if (loading || !image || !data?.findImage) { if (loading) {
return <LoadingIndicator />; return <LoadingIndicator />;
} }
if (error) return <div>{error.message}</div>; if (error) return <ErrorMessage error={error.message} />;
if (!image) {
return <ErrorMessage error={`No image found with id ${id}.`} />;
}
return ( return (
<div className="row"> <div className="row">

View File

@@ -15,7 +15,6 @@ import { RatingStars } from "src/components/Scenes/SceneDetails/RatingStars";
interface IProps { interface IProps {
image: GQL.ImageDataFragment; image: GQL.ImageDataFragment;
isVisible: boolean; isVisible: boolean;
onUpdate: (image: GQL.ImageDataFragment) => void;
onDelete: () => void; onDelete: () => void;
} }
@@ -107,7 +106,6 @@ export const ImageEditPanel: React.FC<IProps> = (props: IProps) => {
try { try {
const result = await updateImage(); const result = await updateImage();
if (result.data?.imageUpdate) { if (result.data?.imageUpdate) {
props.onUpdate(result.data.imageUpdate);
Toast.success({ content: "Updated image" }); Toast.success({ content: "Updated image" });
} }
} catch (e) { } catch (e) {

View File

@@ -9,7 +9,12 @@ import {
usePerformerCreate, usePerformerCreate,
usePerformerDestroy, usePerformerDestroy,
} from "src/core/StashService"; } from "src/core/StashService";
import { CountryFlag, Icon, LoadingIndicator } from "src/components/Shared"; import {
CountryFlag,
ErrorMessage,
Icon,
LoadingIndicator,
} from "src/components/Shared";
import { useToast } from "src/hooks"; import { useToast } from "src/hooks";
import { TextUtils } from "src/utils"; import { TextUtils } from "src/utils";
import FsLightbox from "fslightbox-react"; import FsLightbox from "fslightbox-react";
@@ -29,12 +34,11 @@ export const Performer: React.FC = () => {
const isNew = id === "new"; const isNew = id === "new";
// Performer state // Performer state
const [performer, setPerformer] = useState<
Partial<GQL.PerformerDataFragment>
>({});
const [imagePreview, setImagePreview] = useState<string | null>(); const [imagePreview, setImagePreview] = useState<string | null>();
const [imageEncoding, setImageEncoding] = useState<boolean>(false); const [imageEncoding, setImageEncoding] = useState<boolean>(false);
const [lightboxToggle, setLightboxToggle] = useState(false); const [lightboxToggle, setLightboxToggle] = useState(false);
const { data, loading: performerLoading, error } = useFindPerformer(id);
const performer = data?.findPerformer || ({} as Partial<GQL.Performer>);
// if undefined then get the existing image // if undefined then get the existing image
// if null then get the default (no) image // if null then get the default (no) image
@@ -45,9 +49,9 @@ export const Performer: React.FC = () => {
: imagePreview ?? `${performer.image_path}?default=true`; : imagePreview ?? `${performer.image_path}?default=true`;
// Network state // Network state
const [isLoading, setIsLoading] = useState(false); const [loading, setIsLoading] = useState(false);
const isLoading = performerLoading || loading;
const { data, error } = useFindPerformer(id);
const [updatePerformer] = usePerformerUpdate(); const [updatePerformer] = usePerformerUpdate();
const [createPerformer] = usePerformerCreate(); const [createPerformer] = usePerformerCreate();
const [deletePerformer] = usePerformerDestroy(); const [deletePerformer] = usePerformerDestroy();
@@ -63,11 +67,6 @@ export const Performer: React.FC = () => {
} }
}; };
useEffect(() => {
setIsLoading(false);
if (data?.findPerformer) setPerformer(data.findPerformer);
}, [data]);
const onImageChange = (image?: string | null) => setImagePreview(image); const onImageChange = (image?: string | null) => setImagePreview(image);
const onImageEncoding = (isEncoding = false) => setImageEncoding(isEncoding); const onImageEncoding = (isEncoding = false) => setImageEncoding(isEncoding);
@@ -89,10 +88,10 @@ export const Performer: React.FC = () => {
}; };
}); });
if ((!isNew && (!data || !data.findPerformer)) || isLoading) if (isLoading) return <LoadingIndicator />;
return <LoadingIndicator />; if (error) return <ErrorMessage error={error.message} />;
if (!performer.id && !isNew)
if (error) return <div>{error.message}</div>; return <ErrorMessage error={`No performer found with id ${id}.`} />;
async function onSave( async function onSave(
performerInput: performerInput:
@@ -102,21 +101,18 @@ export const Performer: React.FC = () => {
setIsLoading(true); setIsLoading(true);
try { try {
if (!isNew) { if (!isNew) {
const result = await updatePerformer({ await updatePerformer({
variables: performerInput as GQL.PerformerUpdateInput, variables: performerInput as GQL.PerformerUpdateInput,
}); });
if (performerInput.image) { if (performerInput.image) {
// Refetch image to bust browser cache // Refetch image to bust browser cache
await fetch(`/performer/${performer.id}/image`, { cache: "reload" }); await fetch(`/performer/${id}/image`, { cache: "reload" });
} }
if (result.data?.performerUpdate)
setPerformer(result.data?.performerUpdate);
} else { } else {
const result = await createPerformer({ const result = await createPerformer({
variables: performerInput as GQL.PerformerCreateInput, variables: performerInput as GQL.PerformerCreateInput,
}); });
if (result.data?.performerCreate) { if (result.data?.performerCreate) {
setPerformer(result.data.performerCreate);
history.push(`/performers/${result.data.performerCreate.id}`); history.push(`/performers/${result.data.performerCreate.id}`);
} }
} }
@@ -199,8 +195,7 @@ export const Performer: React.FC = () => {
} }
function setFavorite(v: boolean) { function setFavorite(v: boolean) {
performer.favorite = v; onSave({ ...performer, favorite: v });
onSave(performer);
} }
const renderIcons = () => ( const renderIcons = () => (

View File

@@ -12,7 +12,7 @@ import {
useSceneGenerateScreenshot, useSceneGenerateScreenshot,
} from "src/core/StashService"; } from "src/core/StashService";
import { GalleryViewer } from "src/components/Galleries/GalleryViewer"; import { GalleryViewer } from "src/components/Galleries/GalleryViewer";
import { LoadingIndicator, Icon } from "src/components/Shared"; import { ErrorMessage, LoadingIndicator, Icon } from "src/components/Shared";
import { useToast } from "src/hooks"; import { useToast } from "src/hooks";
import { ScenePlayer } from "src/components/ScenePlayer"; import { ScenePlayer } from "src/components/ScenePlayer";
import { TextUtils, JWUtils } from "src/utils"; import { TextUtils, JWUtils } from "src/utils";
@@ -39,8 +39,8 @@ export const Scene: React.FC = () => {
const [timestamp, setTimestamp] = useState<number>(getInitialTimestamp()); const [timestamp, setTimestamp] = useState<number>(getInitialTimestamp());
const [collapsed, setCollapsed] = useState(false); const [collapsed, setCollapsed] = useState(false);
const [scene, setScene] = useState<GQL.SceneDataFragment | undefined>();
const { data, error, loading } = useFindScene(id); const { data, error, loading } = useFindScene(id);
const scene = data?.findScene;
const { const {
data: sceneStreams, data: sceneStreams,
error: streamableError, error: streamableError,
@@ -59,10 +59,6 @@ export const Scene: React.FC = () => {
const queryParams = queryString.parse(location.search); const queryParams = queryString.parse(location.search);
const autoplay = queryParams?.autoplay === "true"; const autoplay = queryParams?.autoplay === "true";
useEffect(() => {
if (data?.findScene) setScene(data.findScene);
}, [data]);
function getInitialTimestamp() { function getInitialTimestamp() {
const params = queryString.parse(location.search); const params = queryString.parse(location.search);
const initialTimestamp = params?.t ?? "0"; const initialTimestamp = params?.t ?? "0";
@@ -72,17 +68,10 @@ export const Scene: React.FC = () => {
); );
} }
const updateOCounter = (newValue: number) => {
const modifiedScene = { ...scene } as GQL.SceneDataFragment;
modifiedScene.o_counter = newValue;
setScene(modifiedScene);
};
const onIncrementClick = async () => { const onIncrementClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await incrementO(); await incrementO();
if (result.data) updateOCounter(result.data.sceneIncrementO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -93,8 +82,7 @@ export const Scene: React.FC = () => {
const onDecrementClick = async () => { const onDecrementClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await decrementO(); await decrementO();
if (result.data) updateOCounter(result.data.sceneDecrementO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -105,8 +93,7 @@ export const Scene: React.FC = () => {
const onResetClick = async () => { const onResetClick = async () => {
try { try {
setOLoading(true); setOLoading(true);
const result = await resetO(); await resetO();
if (result.data) updateOCounter(result.data.sceneResetO);
} catch (e) { } catch (e) {
Toast.error(e); Toast.error(e);
} finally { } finally {
@@ -290,7 +277,6 @@ export const Scene: React.FC = () => {
<SceneEditPanel <SceneEditPanel
isVisible={activeTabKey === "scene-edit-panel"} isVisible={activeTabKey === "scene-edit-panel"}
scene={scene} scene={scene}
onUpdate={(newScene) => setScene(newScene)}
onDelete={() => setIsDeleteAlertOpen(true)} onDelete={() => setIsDeleteAlertOpen(true)}
/> />
</Tab.Pane> </Tab.Pane>
@@ -320,12 +306,10 @@ export const Scene: React.FC = () => {
return collapsed ? ">" : "<"; return collapsed ? ">" : "<";
} }
if (loading || streamableLoading || !scene || !data?.findScene) { if (loading || streamableLoading) return <LoadingIndicator />;
return <LoadingIndicator />; if (error) return <ErrorMessage error={error.message} />;
} if (streamableError) return <ErrorMessage error={streamableError.message} />;
if (!scene) return <ErrorMessage error={`No scene found with id ${id}.`} />;
if (error) return <div>{error.message}</div>;
if (streamableError) return <div>{streamableError.message}</div>;
return ( return (
<div className="row"> <div className="row">

View File

@@ -36,7 +36,6 @@ import { SceneScrapeDialog } from "./SceneScrapeDialog";
interface IProps { interface IProps {
scene: GQL.SceneDataFragment; scene: GQL.SceneDataFragment;
isVisible: boolean; isVisible: boolean;
onUpdate: (scene: GQL.SceneDataFragment) => void;
onDelete: () => void; onDelete: () => void;
} }
@@ -226,7 +225,6 @@ export const SceneEditPanel: React.FC<IProps> = (props: IProps) => {
try { try {
const result = await updateScene(); const result = await updateScene();
if (result.data?.sceneUpdate) { if (result.data?.sceneUpdate) {
props.onUpdate(result.data.sceneUpdate);
Toast.success({ content: "Updated scene" }); Toast.success({ content: "Updated scene" });
} }
} catch (e) { } catch (e) {

View File

@@ -0,0 +1,13 @@
import React, { ReactNode } from "react";
interface IProps {
error: string | ReactNode;
}
const ErrorMessage: React.FC<IProps> = ({ error }) => (
<div className="row ErrorMessage">
<h2 className="ErrorMessage-content">Error: {error}</h2>
</div>
);
export default ErrorMessage;

View File

@@ -19,3 +19,4 @@ export { default as LoadingIndicator } from "./LoadingIndicator";
export { ImageInput } from "./ImageInput"; export { ImageInput } from "./ImageInput";
export { SweatDrops } from "./SweatDrops"; export { SweatDrops } from "./SweatDrops";
export { default as CountryFlag } from "./CountryFlag"; export { default as CountryFlag } from "./CountryFlag";
export { default as ErrorMessage } from "./ErrorMessage";

View File

@@ -133,3 +133,13 @@ button.collapse-button.btn-primary:not(:disabled):not(.disabled):active {
max-width: 32rem; max-width: 32rem;
text-align: center; text-align: center;
} }
.ErrorMessage {
align-items: center;
height: 20rem;
justify-content: center;
&-content {
display: inline-block;
}
}

View File

@@ -309,19 +309,55 @@ export const useBulkSceneUpdate = (input: GQL.BulkSceneUpdateInput) =>
export const useScenesUpdate = (input: GQL.SceneUpdateInput[]) => export const useScenesUpdate = (input: GQL.SceneUpdateInput[]) =>
GQL.useScenesUpdateMutation({ variables: { input } }); GQL.useScenesUpdateMutation({ variables: { input } });
type SceneOMutation =
| GQL.SceneIncrementOMutation
| GQL.SceneDecrementOMutation
| GQL.SceneResetOMutation;
const updateSceneO = (
id: string,
cache: ApolloCache<SceneOMutation>,
updatedOCount?: number
) => {
const scene = cache.readQuery<
GQL.FindSceneQuery,
GQL.FindSceneQueryVariables
>({
query: GQL.FindSceneDocument,
variables: { id },
});
if (updatedOCount === undefined || !scene?.findScene) return;
cache.writeQuery<GQL.FindSceneQuery, GQL.FindSceneQueryVariables>({
query: GQL.FindSceneDocument,
variables: { id },
data: {
...scene,
findScene: {
...scene.findScene,
o_counter: updatedOCount,
},
},
});
};
export const useSceneIncrementO = (id: string) => export const useSceneIncrementO = (id: string) =>
GQL.useSceneIncrementOMutation({ GQL.useSceneIncrementOMutation({
variables: { id }, variables: { id },
update: (cache, data) =>
updateSceneO(id, cache, data.data?.sceneIncrementO),
}); });
export const useSceneDecrementO = (id: string) => export const useSceneDecrementO = (id: string) =>
GQL.useSceneDecrementOMutation({ GQL.useSceneDecrementOMutation({
variables: { id }, variables: { id },
update: (cache, data) =>
updateSceneO(id, cache, data.data?.sceneDecrementO),
}); });
export const useSceneResetO = (id: string) => export const useSceneResetO = (id: string) =>
GQL.useSceneResetOMutation({ GQL.useSceneResetOMutation({
variables: { id }, variables: { id },
update: (cache, data) => updateSceneO(id, cache, data.data?.sceneResetO),
}); });
export const useSceneDestroy = (input: GQL.SceneDestroyInput) => export const useSceneDestroy = (input: GQL.SceneDestroyInput) =>
@@ -372,19 +408,54 @@ export const useImagesDestroy = (input: GQL.ImagesDestroyInput) =>
update: deleteCache(imageMutationImpactedQueries), update: deleteCache(imageMutationImpactedQueries),
}); });
type ImageOMutation =
| GQL.ImageIncrementOMutation
| GQL.ImageDecrementOMutation
| GQL.ImageResetOMutation;
const updateImageO = (
id: string,
cache: ApolloCache<ImageOMutation>,
updatedOCount?: number
) => {
const image = cache.readQuery<
GQL.FindImageQuery,
GQL.FindImageQueryVariables
>({
query: GQL.FindImageDocument,
variables: { id },
});
if (updatedOCount === undefined || !image?.findImage) return;
cache.writeQuery<GQL.FindImageQuery, GQL.FindImageQueryVariables>({
query: GQL.FindImageDocument,
variables: { id },
data: {
findImage: {
...image.findImage,
o_counter: updatedOCount,
},
},
});
};
export const useImageIncrementO = (id: string) => export const useImageIncrementO = (id: string) =>
GQL.useImageIncrementOMutation({ GQL.useImageIncrementOMutation({
variables: { id }, variables: { id },
update: (cache, data) =>
updateImageO(id, cache, data.data?.imageIncrementO),
}); });
export const useImageDecrementO = (id: string) => export const useImageDecrementO = (id: string) =>
GQL.useImageDecrementOMutation({ GQL.useImageDecrementOMutation({
variables: { id }, variables: { id },
update: (cache, data) =>
updateImageO(id, cache, data.data?.imageDecrementO),
}); });
export const useImageResetO = (id: string) => export const useImageResetO = (id: string) =>
GQL.useImageResetOMutation({ GQL.useImageResetOMutation({
variables: { id }, variables: { id },
update: (cache, data) => updateImageO(id, cache, data.data?.imageResetO),
}); });
const galleryMutationImpactedQueries = [ const galleryMutationImpactedQueries = [