mirror of
https://github.com/stashapp/stash.git
synced 2025-12-17 20:34:37 +03:00
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:
@@ -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">
|
||||||
|
|||||||
@@ -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" });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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> = ({
|
||||||
|
|||||||
@@ -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">
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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 = () => (
|
||||||
|
|||||||
@@ -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">
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
13
ui/v2.5/src/components/Shared/ErrorMessage.tsx
Normal file
13
ui/v2.5/src/components/Shared/ErrorMessage.tsx
Normal 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;
|
||||||
@@ -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";
|
||||||
|
|||||||
@@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -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 = [
|
||||||
|
|||||||
Reference in New Issue
Block a user