diff --git a/pages/api/media-type.ts b/pages/api/media-type.ts
index f64301f094..7489fecdf1 100644
--- a/pages/api/media-type.ts
+++ b/pages/api/media-type.ts
@@ -22,11 +22,13 @@ export default async function mediaTypeHandler(req: NextApiRequest, res: NextApi
return 'video';
}
+ if (contentType?.startsWith('image')) {
+ return 'image';
+ }
+
if (contentType?.startsWith('text/html')) {
return 'html';
}
-
- return 'image';
})();
res.status(200).json({ type: mediaType });
} catch (error) {
diff --git a/ui/address/tokens/NFTItem.tsx b/ui/address/tokens/NFTItem.tsx
index 50a33ab73b..cc61b70dd1 100644
--- a/ui/address/tokens/NFTItem.tsx
+++ b/ui/address/tokens/NFTItem.tsx
@@ -26,7 +26,8 @@ const NFTItem = ({ token, value, isLoading, withTokenLink, ...tokenInstance }: P
diff --git a/ui/shared/nft/NftMedia.pw.tsx b/ui/shared/nft/NftMedia.pw.tsx
index 71d09de045..4b91def420 100644
--- a/ui/shared/nft/NftMedia.pw.tsx
+++ b/ui/shared/nft/NftMedia.pw.tsx
@@ -10,7 +10,54 @@ test.describe('no url', () => {
test('preview +@dark-mode', async({ mount }) => {
const component = await mount(
-
+
+ ,
+ );
+
+ await expect(component).toHaveScreenshot();
+ });
+
+ test('with fallback', async({ mount, page }) => {
+ const IMAGE_URL = 'https://localhost:3000/my-image.jpg';
+
+ await page.route(IMAGE_URL, (route) => {
+ return route.fulfill({
+ status: 200,
+ path: './playwright/mocks/image_long.jpg',
+ });
+ });
+
+ const component = await mount(
+
+
+ ,
+ );
+
+ await expect(component).toHaveScreenshot();
+ });
+
+ test('non-media url and fallback', async({ mount, page }) => {
+ const ANIMATION_URL = 'https://localhost:3000/my-animation.m3u8';
+ const ANIMATION_MEDIA_TYPE_API_URL = `/node-api/media-type?url=${ encodeURIComponent(ANIMATION_URL) }`;
+ const IMAGE_URL = 'https://localhost:3000/my-image.jpg';
+
+ await page.route(ANIMATION_MEDIA_TYPE_API_URL, (route) => {
+ return route.fulfill({
+ status: 200,
+ body: JSON.stringify({ type: undefined }),
+ });
+ });
+
+ await page.route(IMAGE_URL, (route) => {
+ return route.fulfill({
+ status: 200,
+ path: './playwright/mocks/image_long.jpg',
+ });
+ });
+
+ const component = await mount(
+
+
,
);
@@ -35,7 +82,7 @@ test.describe('image', () => {
test('preview +@dark-mode', async({ mount }) => {
const component = await mount(
-
+
,
);
@@ -55,7 +102,7 @@ test('image preview hover', async({ mount, page }) => {
const component = await mount(
-
+
,
);
@@ -75,7 +122,7 @@ test('image fullscreen +@dark-mode +@mobile', async({ mount, page }) => {
});
const component = await mount(
-
+
,
);
@@ -107,7 +154,7 @@ test.describe('page', () => {
test('preview +@dark-mode', async({ mount }) => {
const component = await mount(
-
+
,
);
diff --git a/ui/shared/nft/NftMedia.tsx b/ui/shared/nft/NftMedia.tsx
index e01569c314..ace164692a 100644
--- a/ui/shared/nft/NftMedia.tsx
+++ b/ui/shared/nft/NftMedia.tsx
@@ -9,29 +9,31 @@ import NftImage from './NftImage';
import NftImageFullscreen from './NftImageFullscreen';
import NftVideo from './NftVideo';
import NftVideoFullscreen from './NftVideoFullscreen';
-import useNftMediaType from './useNftMediaType';
+import useNftMediaInfo from './useNftMediaInfo';
import { mediaStyleProps } from './utils';
interface Props {
- url: string | null;
+ imageUrl: string | null;
+ animationUrl: string | null;
className?: string;
isLoading?: boolean;
withFullscreen?: boolean;
}
-const NftMedia = ({ url, className, isLoading, withFullscreen }: Props) => {
+const NftMedia = ({ imageUrl, animationUrl, className, isLoading, withFullscreen }: Props) => {
const [ isMediaLoading, setIsMediaLoading ] = React.useState(true);
const [ isLoadingError, setIsLoadingError ] = React.useState(false);
const { ref, inView } = useInView({ triggerOnce: true });
- const type = useNftMediaType(url, !isLoading && inView);
+ const mediaInfo = useNftMediaInfo({ imageUrl, animationUrl, isEnabled: !isLoading && inView });
React.useEffect(() => {
- if (!isLoading) {
- setIsMediaLoading(Boolean(url));
+ if (!isLoading && !mediaInfo) {
+ setIsMediaLoading(false);
+ setIsLoadingError(true);
}
- }, [ isLoading, url ]);
+ }, [ isLoading, mediaInfo ]);
const handleMediaLoaded = React.useCallback(() => {
setIsMediaLoading(false);
@@ -45,11 +47,17 @@ const NftMedia = ({ url, className, isLoading, withFullscreen }: Props) => {
const { isOpen, onOpen, onClose } = useDisclosure();
const content = (() => {
- if (!url || isLoadingError) {
+ if (!mediaInfo || isLoadingError) {
const styleProps = withFullscreen ? {} : mediaStyleProps;
return ;
}
+ const { type, url } = mediaInfo;
+
+ if (!url) {
+ return null;
+ }
+
const props = {
src: url,
onLoad: handleMediaLoaded,
@@ -70,7 +78,13 @@ const NftMedia = ({ url, className, isLoading, withFullscreen }: Props) => {
})();
const modal = (() => {
- if (!url || !withFullscreen) {
+ if (!mediaInfo || !withFullscreen) {
+ return null;
+ }
+
+ const { type, url } = mediaInfo;
+
+ if (!url) {
return null;
}
diff --git a/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-non-media-url-and-fallback-1.png b/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-non-media-url-and-fallback-1.png
new file mode 100644
index 0000000000..4273139faf
Binary files /dev/null and b/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-non-media-url-and-fallback-1.png differ
diff --git a/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-with-fallback-1.png b/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-with-fallback-1.png
new file mode 100644
index 0000000000..4273139faf
Binary files /dev/null and b/ui/shared/nft/__screenshots__/NftMedia.pw.tsx_default_no-url-with-fallback-1.png differ
diff --git a/ui/shared/nft/useNftMediaInfo.tsx b/ui/shared/nft/useNftMediaInfo.tsx
new file mode 100644
index 0000000000..7c36f828e9
--- /dev/null
+++ b/ui/shared/nft/useNftMediaInfo.tsx
@@ -0,0 +1,98 @@
+import { useQuery } from '@tanstack/react-query';
+import React from 'react';
+
+import type { StaticRoute } from 'nextjs-routes';
+import { route } from 'nextjs-routes';
+
+import type { ResourceError } from 'lib/api/resources';
+import useFetch from 'lib/hooks/useFetch';
+
+import type { MediaType } from './utils';
+import { getPreliminaryMediaType } from './utils';
+
+interface Params {
+ imageUrl: string | null;
+ animationUrl: string | null;
+ isEnabled: boolean;
+}
+
+interface ReturnType {
+ type: MediaType | undefined;
+ url: string | null;
+}
+
+export default function useNftMediaInfo({ imageUrl, animationUrl, isEnabled }: Params): ReturnType | null {
+
+ const primaryQuery = useNftMediaTypeQuery(animationUrl, isEnabled);
+ const secondaryQuery = useNftMediaTypeQuery(imageUrl, !primaryQuery.isPending && !primaryQuery.data);
+
+ return React.useMemo(() => {
+ if (primaryQuery.isPending) {
+ return {
+ type: undefined,
+ url: animationUrl,
+ };
+ }
+
+ if (primaryQuery.data) {
+ return primaryQuery.data;
+ }
+
+ if (secondaryQuery.isPending) {
+ return {
+ type: undefined,
+ url: imageUrl,
+ };
+ }
+
+ if (secondaryQuery.data) {
+ return secondaryQuery.data;
+ }
+
+ return null;
+ }, [ animationUrl, imageUrl, primaryQuery.data, primaryQuery.isPending, secondaryQuery.data, secondaryQuery.isPending ]);
+}
+
+function useNftMediaTypeQuery(url: string | null, enabled: boolean) {
+ const fetch = useFetch();
+
+ return useQuery, ReturnType | null>({
+ queryKey: [ 'nft-media-type', url ],
+ queryFn: async() => {
+ if (!url) {
+ return null;
+ }
+
+ // media could be either image, gif, video or html-page
+ // so we pre-fetch the resources in order to get its content type
+ // have to do it via Node.js due to strict CSP for connect-src
+ // but in order not to abuse our server firstly we check file url extension
+ // and if it is valid we will trust it and display corresponding media component
+
+ const preliminaryType = getPreliminaryMediaType(url);
+
+ if (preliminaryType) {
+ return { type: preliminaryType, url };
+ }
+
+ const type = await (async() => {
+ try {
+ const mediaTypeResourceUrl = route({ pathname: '/node-api/media-type' as StaticRoute<'/api/media-type'>['pathname'], query: { url } });
+ const response = await fetch<{ type: MediaType | undefined }, ResourceError>(mediaTypeResourceUrl, undefined, { resource: 'media-type' });
+
+ return 'type' in response ? response.type : undefined;
+ } catch (error) {
+ return;
+ }
+ })();
+
+ if (!type) {
+ return null;
+ }
+
+ return { type, url };
+ },
+ enabled,
+ staleTime: Infinity,
+ });
+}
diff --git a/ui/shared/nft/useNftMediaType.tsx b/ui/shared/nft/useNftMediaType.tsx
deleted file mode 100644
index 4444b122e3..0000000000
--- a/ui/shared/nft/useNftMediaType.tsx
+++ /dev/null
@@ -1,49 +0,0 @@
-import { useQuery } from '@tanstack/react-query';
-
-import type { StaticRoute } from 'nextjs-routes';
-import { route } from 'nextjs-routes';
-
-import type { ResourceError } from 'lib/api/resources';
-import useFetch from 'lib/hooks/useFetch';
-
-import type { MediaType } from './utils';
-import { getPreliminaryMediaType } from './utils';
-
-export default function useNftMediaType(url: string | null, isEnabled: boolean) {
-
- const fetch = useFetch();
-
- const { data } = useQuery, MediaType>({
- queryKey: [ 'nft-media-type', url ],
- queryFn: async() => {
- if (!url) {
- return 'image';
- }
-
- // media could be either image, gif, video or html-page
- // so we pre-fetch the resources in order to get its content type
- // have to do it via Node.js due to strict CSP for connect-src
- // but in order not to abuse our server firstly we check file url extension
- // and if it is valid we will trust it and display corresponding media component
-
- const preliminaryType = getPreliminaryMediaType(url);
-
- if (preliminaryType) {
- return preliminaryType;
- }
-
- try {
- const mediaTypeResourceUrl = route({ pathname: '/node-api/media-type' as StaticRoute<'/api/media-type'>['pathname'], query: { url } });
- const response = await fetch<{ type: MediaType | undefined }, ResourceError>(mediaTypeResourceUrl, undefined, { resource: 'media-type' });
-
- return 'type' in response ? response.type ?? 'image' : 'image';
- } catch (error) {
- return 'image';
- }
- },
- enabled: isEnabled && Boolean(url),
- staleTime: Infinity,
- });
-
- return data;
-}
diff --git a/ui/token/TokenInventoryItem.tsx b/ui/token/TokenInventoryItem.tsx
index 72f650d2b8..6add374550 100644
--- a/ui/token/TokenInventoryItem.tsx
+++ b/ui/token/TokenInventoryItem.tsx
@@ -20,7 +20,8 @@ const TokenInventoryItem = ({ item, token, isLoading }: Props) => {
const mediaElement = (
);
diff --git a/ui/tokenInstance/TokenInstanceDetails.tsx b/ui/tokenInstance/TokenInstanceDetails.tsx
index 6896f1cc4c..dfff7aacc4 100644
--- a/ui/tokenInstance/TokenInstanceDetails.tsx
+++ b/ui/tokenInstance/TokenInstanceDetails.tsx
@@ -74,7 +74,8 @@ const TokenInstanceDetails = ({ data, token, scrollRef, isLoading }: Props) => {