Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
158 changes: 158 additions & 0 deletions playgrounds/next/src/app/test/media/page.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,158 @@
'use client';

import { Media } from '@0xsequence/marketplace-sdk/react';
import { useState } from 'react';

// Mock assets for testing - using stable URLs that won't change
const TEST_ASSETS = {
// Using a data URL for truly stable testing
image:
'',
video:
'https://test-videos.co.uk/vids/bigbuckbunny/mp4/h264/720/Big_Buck_Bunny_720_10s_1MB.mp4',
brokenImage: 'https://broken-url-that-will-fail.com/image.jpg',
iframe: 'https://www.youtube.com/embed/dQw4w9WgXcQ',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This URL is not available

model3d: 'https://modelviewer.dev/shared-assets/models/Astronaut.gltf',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This page gives 404, can you change it as following?

Suggested change
model3d: 'https://modelviewer.dev/shared-assets/models/Astronaut.gltf',
model3d: 'https://modelviewer.dev/shared-assets/models/NeilArmstrong.glb',

};

export default function MediaTestPage() {
const [slowLoading, setSlowLoading] = useState(false);

// Check if slow loading is requested via query param
if (typeof window !== 'undefined') {
const urlParams = new URLSearchParams(window.location.search);
if (urlParams.get('slow') === 'true' && !slowLoading) {
setSlowLoading(true);
}
}

return (
<div className="container mx-auto p-8">
<h1 className="mb-8 font-bold text-2xl">Media Component Test Page</h1>

<div className="grid grid-cols-1 gap-8 md:grid-cols-2 lg:grid-cols-3">
{/* Image Test */}
<div className="space-y-2">
<h2 className="font-semibold">Image Media</h2>
<div data-testid="media-image">
<Media
name="Test Image"
assets={[TEST_ASSETS.image]}
containerClassName="aspect-square rounded-lg border"
isLoading={slowLoading}
/>
</div>
</div>

{/* Video Test */}
<div className="space-y-2">
<h2 className="font-semibold">Video Media</h2>
<div data-testid="media-video">
<Media
name="Test Video"
assets={[TEST_ASSETS.video]}
containerClassName="aspect-square rounded-lg border"
isLoading={slowLoading}
/>
</div>
</div>

{/* Fallback Test */}
<div className="space-y-2">
<h2 className="font-semibold">Fallback Media</h2>
<div data-testid="media-fallback">
<Media
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This keeps showing skeleton

name="Test Fallback"
assets={[TEST_ASSETS.brokenImage, TEST_ASSETS.image]}
containerClassName="aspect-square rounded-lg border"
/>
</div>
</div>

{/* Iframe Test */}
<div className="space-y-2">
<h2 className="font-semibold">Iframe Media</h2>
<div data-testid="media-iframe">
<Media
name="Test Iframe"
assets={[`${TEST_ASSETS.iframe}.html`]}
containerClassName="aspect-square rounded-lg border"
/>
</div>
</div>

{/* 3D Model Test */}
<div className="space-y-2">
<h2 className="font-semibold">3D Model Media</h2>
<div data-testid="media-3d-model">
<Media
name="Test 3D Model"
assets={[TEST_ASSETS.model3d]}
containerClassName="aspect-square rounded-lg border"
/>
</div>
</div>

{/* Custom Fallback Test */}
<div className="space-y-2">
<h2 className="font-semibold">Custom Fallback</h2>
<div data-testid="media-custom-fallback">
<Media
name="Test Custom Fallback"
assets={[TEST_ASSETS.brokenImage]}
containerClassName="aspect-square rounded-lg border"
fallbackContent={
<div className="flex h-full flex-col items-center justify-center">
<div className="mb-2 text-4xl">🎨</div>
<div className="font-medium text-sm">No Media Available</div>
</div>
}
/>
</div>
</div>
</div>

{/* Showcase for visual regression testing */}
<div className="mt-12" data-testid="media-showcase">
<h2 className="mb-4 font-bold text-xl">Media Showcase</h2>
<div className="grid grid-cols-4 gap-4">
<Media
name="Showcase 1"
assets={[TEST_ASSETS.image]}
containerClassName="aspect-square rounded border"
/>
<Media
name="Showcase 2"
assets={[TEST_ASSETS.video]}
containerClassName="aspect-square rounded border"
/>
<Media
name="Showcase 3"
assets={[]}
containerClassName="aspect-square rounded border"
/>
<Media
name="Showcase 4"
assets={[TEST_ASSETS.image]}
containerClassName="aspect-square rounded border"
isLoading={true}
/>
</div>
</div>

{/* SSR Test Section */}
<div className="mt-12 rounded bg-gray-100 p-4">
<h2 className="mb-4 font-bold text-xl">SSR Test</h2>
<p className="mb-4 text-gray-600 text-sm">
This section tests server-side rendering. Check the page source to
verify proper HTML generation.
</p>
<Media
name="SSR Test Media"
assets={[TEST_ASSETS.image]}
containerClassName="aspect-video rounded-lg border"
/>
</div>
</div>
);
}
200 changes: 200 additions & 0 deletions sdk/src/react/hooks/__tests__/useMediaLoad.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
import { act, renderHook } from '@testing-library/react';
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
import { useIframeLoad, useImageLoad, useVideoLoad } from '../useMediaLoad';

describe('useImageLoad', () => {
beforeEach(() => {
vi.useFakeTimers();
});

afterEach(() => {
vi.useRealTimers();
});

it('should handle already loaded images (SSR case)', async () => {
const onLoad = vi.fn();
const onError = vi.fn();

// Create a mock image element that's already loaded
const mockImg = {
complete: true,
naturalWidth: 100,
decode: vi.fn().mockResolvedValue(undefined),
parentElement: document.createElement('div'),
isConnected: true,
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
'data-loaded-src': undefined,
} as any;

const { result } = renderHook(() =>
useImageLoad({
onLoad,
onError,
src: 'https://example.com/image.jpg',
}),
);

// Call the ref callback with the mock image
act(() => {
result.current.imgRef(mockImg);
});

// Should call decode and then onLoad
expect(mockImg.decode).toHaveBeenCalled();
expect(result.current.isLoaded).toBe(false); // Will be true after decode resolves

// Wait for decode promise to resolve
await act(async () => {
await vi.runAllTimersAsync();
});

// Now check that isLoaded is true after decode resolves
expect(result.current.isLoaded).toBe(true);
expect(onLoad).toHaveBeenCalled();
});

it('should handle image load errors', async () => {
const onLoad = vi.fn();
const onError = vi.fn();

// Create a mock image element
const mockImg = {
complete: false,
decode: vi.fn().mockRejectedValue(new Error('Failed to decode')),
parentElement: document.createElement('div'),
isConnected: true,
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
'data-loaded-src': undefined,
src: '',
} as any;

const { result } = renderHook(() =>
useImageLoad({
onLoad,
onError,
src: 'https://example.com/image.jpg',
}),
);

// Call the ref callback with the mock image
act(() => {
result.current.imgRef(mockImg);
});

// Simulate error event
const errorHandler = mockImg.addEventListener.mock.calls.find(
(call: any) => call[0] === 'error',
)?.[1];

act(() => {
errorHandler?.();
});

// Should set error state
expect(result.current.hasError).toBe(true);
expect(onError).toHaveBeenCalled();
});

it('should not call handlers when disabled', () => {
const onLoad = vi.fn();
const onError = vi.fn();

const mockImg = {
complete: true,
naturalWidth: 100,
decode: vi.fn().mockResolvedValue(undefined),
parentElement: document.createElement('div'),
isConnected: true,
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
'data-loaded-src': undefined,
} as any;

const { result } = renderHook(() =>
useImageLoad({
onLoad,
onError,
src: 'https://example.com/image.jpg',
enabled: false,
}),
);

// Call the ref callback with the mock image
act(() => {
result.current.imgRef(mockImg);
});

// Should not call any handlers when disabled
expect(mockImg.decode).not.toHaveBeenCalled();
expect(onLoad).not.toHaveBeenCalled();
expect(onError).not.toHaveBeenCalled();
});
});

describe('useVideoLoad', () => {
it('should handle already loaded videos', () => {
const onLoad = vi.fn();
const onError = vi.fn();

// Create a mock video element that's already loaded
const mockVideo = {
readyState: 4, // HAVE_ENOUGH_DATA
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
} as unknown as HTMLVideoElement;

const { result } = renderHook(() =>
useVideoLoad({
onLoad,
onError,
src: 'https://example.com/video.mp4',
}),
);

// Call the ref callback with the mock video
act(() => {
result.current.videoRef(mockVideo);
});

// Should immediately set loaded state
expect(result.current.isLoaded).toBe(true);
expect(onLoad).toHaveBeenCalled();
});
});

describe('useIframeLoad', () => {
it('should handle iframe load events', () => {
const onLoad = vi.fn();
const onError = vi.fn();

const mockIframe = {
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
} as unknown as HTMLIFrameElement;

const { result } = renderHook(() =>
useIframeLoad({
onLoad,
onError,
src: 'https://example.com/iframe.html',
}),
);

// Call the ref callback with the mock iframe
act(() => {
result.current.iframeRef(mockIframe);
});

// Should set up event listeners
expect(mockIframe.addEventListener).toHaveBeenCalledWith(
'load',
expect.any(Function),
);
expect(mockIframe.addEventListener).toHaveBeenCalledWith(
'error',
expect.any(Function),
);
});
});
Loading