mirror of
https://github.com/FranP-code/spooky-spotify-showcase.git
synced 2025-10-13 00:02:36 +00:00
Scroll animations and error handling
This commit is contained in:
@@ -48,6 +48,9 @@ export function AlbumImage({
|
|||||||
error,
|
error,
|
||||||
}: AlbumImageProps) {
|
}: AlbumImageProps) {
|
||||||
const [spookyImageLoaded, setSpookyImageLoaded] = useState(false);
|
const [spookyImageLoaded, setSpookyImageLoaded] = useState(false);
|
||||||
|
const [loadError, setLoadError] = useState(false);
|
||||||
|
|
||||||
|
const errorValue = error || loadError;
|
||||||
|
|
||||||
const handleSaveImage = async () => {
|
const handleSaveImage = async () => {
|
||||||
if (!entry.data && !entry.isLoading) {
|
if (!entry.data && !entry.isLoading) {
|
||||||
@@ -87,7 +90,7 @@ export function AlbumImage({
|
|||||||
src={imageSource}
|
src={imageSource}
|
||||||
alt={album.name}
|
alt={album.name}
|
||||||
/>
|
/>
|
||||||
{loadGeneratedImage && spookyImageSource && !error && (
|
{loadGeneratedImage && spookyImageSource && !errorValue && (
|
||||||
<img
|
<img
|
||||||
className="h-36 w-36 cursor-pointer rounded"
|
className="h-36 w-36 cursor-pointer rounded"
|
||||||
style={{
|
style={{
|
||||||
@@ -98,10 +101,11 @@ export function AlbumImage({
|
|||||||
onLoad={onImageLoad}
|
onLoad={onImageLoad}
|
||||||
onError={() => {
|
onError={() => {
|
||||||
onImageLoad();
|
onImageLoad();
|
||||||
|
setLoadError(true);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{showSpookyImage && !spookyImageLoaded && !error && (
|
{showSpookyImage && !spookyImageLoaded && !errorValue && (
|
||||||
<div className="flex h-36 w-36 flex-col items-center justify-center rounded bg-slate-900 text-base backdrop-blur-3xl">
|
<div className="flex h-36 w-36 flex-col items-center justify-center rounded bg-slate-900 text-base backdrop-blur-3xl">
|
||||||
{(() => {
|
{(() => {
|
||||||
if (generateSpookyImageData) {
|
if (generateSpookyImageData) {
|
||||||
@@ -130,7 +134,7 @@ export function AlbumImage({
|
|||||||
})()}
|
})()}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
{error && showSpookyImage && (
|
{errorValue && showSpookyImage && (
|
||||||
<div className="flex h-36 w-36 items-center justify-center rounded-xl bg-slate-900">
|
<div className="flex h-36 w-36 items-center justify-center rounded-xl bg-slate-900">
|
||||||
<ErrorComponent />
|
<ErrorComponent />
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import { pulsar } from "ldrs";
|
|||||||
import Swiper from "./swiper";
|
import Swiper from "./swiper";
|
||||||
import { SwiperSlide } from "swiper/react";
|
import { SwiperSlide } from "swiper/react";
|
||||||
import AlbumImage from "./album-image";
|
import AlbumImage from "./album-image";
|
||||||
|
import ScrollFadeIn from "./scroll-fade-in";
|
||||||
|
|
||||||
pulsar.register();
|
pulsar.register();
|
||||||
ring2.register();
|
ring2.register();
|
||||||
@@ -112,13 +113,17 @@ export default function AlbumShowcase({
|
|||||||
handleSecondGenerateSpookyImage();
|
handleSecondGenerateSpookyImage();
|
||||||
}, [secondEntry.data, entry.data]);
|
}, [secondEntry.data, entry.data]);
|
||||||
|
|
||||||
const secondSpookyImageMatch = (
|
const secondSpookyImageSource = useMemo(() => {
|
||||||
(secondEntry.data?.value || generateSecondSpookyImage?.data || "") as string
|
const secondSpookyImageMatch = (
|
||||||
).match(/<img\s+src='([^']+)'[^>]*>/);
|
(secondEntry.data?.value ||
|
||||||
const secondSpookyImageSource =
|
generateSecondSpookyImage?.data ||
|
||||||
secondSpookyImageMatch && secondSpookyImageMatch[1]
|
"") as string
|
||||||
|
)?.match(/<img\s+src='([^']+)'[^>]*>/);
|
||||||
|
|
||||||
|
return secondSpookyImageMatch && secondSpookyImageMatch[1]
|
||||||
? secondSpookyImageMatch[1]
|
? secondSpookyImageMatch[1]
|
||||||
: "";
|
: "";
|
||||||
|
}, [secondEntry.data, generateSecondSpookyImage.data]);
|
||||||
|
|
||||||
const firstPlace = places[0] || 0;
|
const firstPlace = places[0] || 0;
|
||||||
const secondPlace = places[1] || 0;
|
const secondPlace = places[1] || 0;
|
||||||
@@ -129,10 +134,11 @@ export default function AlbumShowcase({
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<ScrollFadeIn className="flex w-full flex-wrap gap-3 rounded-xl bg-white bg-opacity-10 p-5 backdrop-blur-lg backdrop-filter">
|
||||||
key={album.id}
|
{/* <div
|
||||||
className="flex w-full flex-wrap gap-3 rounded-xl bg-white bg-opacity-10 p-5 backdrop-blur-lg backdrop-filter"
|
key={album.id}
|
||||||
>
|
className="flex w-full flex-wrap gap-3 rounded-xl bg-white bg-opacity-10 p-5 backdrop-blur-lg backdrop-filter"
|
||||||
|
> */}
|
||||||
<div
|
<div
|
||||||
onClick={() => setShowSpookyImage(!showSpookyImage)}
|
onClick={() => setShowSpookyImage(!showSpookyImage)}
|
||||||
className="mr-2 cursor-pointer *:select-none *:drag-none"
|
className="mr-2 cursor-pointer *:select-none *:drag-none"
|
||||||
@@ -233,6 +239,7 @@ export default function AlbumShowcase({
|
|||||||
))}
|
))}
|
||||||
</ul>
|
</ul>
|
||||||
</section>
|
</section>
|
||||||
</div>
|
{/* </div> */}
|
||||||
|
</ScrollFadeIn>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import {
|
|||||||
makeImageSpooky,
|
makeImageSpooky,
|
||||||
uploadImage,
|
uploadImage,
|
||||||
} from "@/server/utils";
|
} from "@/server/utils";
|
||||||
|
import { TRPCError } from "@trpc/server";
|
||||||
|
|
||||||
export type Entry = {
|
export type Entry = {
|
||||||
type: "artist" | "album";
|
type: "artist" | "album";
|
||||||
@@ -50,11 +51,17 @@ export const entryRouter = createTRPCRouter({
|
|||||||
try {
|
try {
|
||||||
const uploadedImage = await uploadImage(input.entry.image);
|
const uploadedImage = await uploadImage(input.entry.image);
|
||||||
if (!uploadedImage || typeof uploadedImage !== "string") {
|
if (!uploadedImage || typeof uploadedImage !== "string") {
|
||||||
throw new Error("Failed to upload image");
|
throw new TRPCError({
|
||||||
|
code: "INTERNAL_SERVER_ERROR",
|
||||||
|
message: "Failed to upload image",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
const spookyImage = makeImageSpooky(uploadedImage, input.entry.number);
|
const spookyImage = makeImageSpooky(uploadedImage, input.entry.number);
|
||||||
if (!spookyImage || typeof spookyImage !== "string") {
|
if (!spookyImage || typeof spookyImage !== "string") {
|
||||||
throw new Error("Failed to make image spooky");
|
throw new TRPCError({
|
||||||
|
code: "INTERNAL_SERVER_ERROR",
|
||||||
|
message: "Failed to make image spooky",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
await ctx.db.generatedImage.upsert({
|
await ctx.db.generatedImage.upsert({
|
||||||
where: {
|
where: {
|
||||||
@@ -69,6 +76,9 @@ export const entryRouter = createTRPCRouter({
|
|||||||
return spookyImage;
|
return spookyImage;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
|
if (error instanceof TRPCError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
@@ -93,7 +103,10 @@ export const entryRouter = createTRPCRouter({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
if (!generatedImage) {
|
if (!generatedImage) {
|
||||||
throw new Error("No generated image found");
|
throw new TRPCError({
|
||||||
|
code: "NOT_FOUND",
|
||||||
|
message: "No generated image found",
|
||||||
|
});
|
||||||
}
|
}
|
||||||
const entry = await ctx.db.spookyImage.create({
|
const entry = await ctx.db.spookyImage.create({
|
||||||
data: {
|
data: {
|
||||||
@@ -104,6 +117,9 @@ export const entryRouter = createTRPCRouter({
|
|||||||
return entry;
|
return entry;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
|
if (error instanceof TRPCError) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|||||||
Reference in New Issue
Block a user