Skip to content

Commit

Permalink
Basic Thumbhash support for Solid component
Browse files Browse the repository at this point in the history
  • Loading branch information
simonihmig committed Dec 29, 2024
1 parent be8fad7 commit 0429d13
Show file tree
Hide file tree
Showing 3 changed files with 103 additions and 2 deletions.
34 changes: 33 additions & 1 deletion packages/solid/src/responsive-image.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
getDestinationWidthBySize,
env,
isLqipBlurhash,
isLqipThumbhash,
} from '@responsive-image/core';
import {
Component,
Expand Down Expand Up @@ -189,6 +190,36 @@ export const ResponsiveImage: Component<ResponsiveImageProps> = (props) => {
};
}

const thumbhashMeta = isLqipThumbhash(args.src.lqip)
? args.src.lqip
: undefined;
let thumbhashStyles: (() => JSX.CSSProperties | undefined) | undefined =
undefined;

if (!isServer && thumbhashMeta) {
const [blurhashLib] = createResource(() => {
return import('@responsive-image/core/thumbhash/decode');
});

thumbhashStyles = () => {
if (isLoaded()) {
return undefined;
}

const { hash } = thumbhashMeta;
const uri = blurhashLib()?.decode2url(hash);

if (!uri) {
return undefined;
}

return {
'background-image': `url("${uri}")`,
'background-size': 'cover',
};
};
}

return (
<picture>
{sourcesSorted.map((s) => (
Expand All @@ -205,7 +236,8 @@ export const ResponsiveImage: Component<ResponsiveImageProps> = (props) => {
data-ri-bh={blurhashMeta?.hash}
data-ri-bh-w={blurhashMeta?.width}
data-ri-bh-h={blurhashMeta?.height}
style={blurhashStyles?.()}
data-ri-th={thumbhashMeta?.hash}
style={blurhashStyles?.() ?? thumbhashStyles?.()}
on:load={() => setLoaded(true)}
/>
</picture>
Expand Down
38 changes: 38 additions & 0 deletions packages/solid/tests/client.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -486,4 +486,42 @@ describe('ResponsiveImage', () => {
'after image is loaded the background PNG is removed',
).to.equal('none');
});

test('it sets LQIP from thumbhash as background', async () => {
const { onload, loaded } = imageLoaded();
const imageData: ImageData = {
imageTypes: ['jpeg', 'webp'],
// to replicate the loading timing, we need to load a real existing image
imageUrlFor: () => `/test-assets/test-image.jpg?${cacheBreaker()}`,
aspectRatio: 1.5,
lqip: {
type: 'thumbhash',
hash: 'jJcFFYI1fIWHe4dweXlYeUaAmWj3',
},
};

const { container } = render(() => <ResponsiveImage src={imageData} />);
onload(container);

const imgEl = container.querySelector('img')!;
expect(imgEl).toBeDefined();
expect(imgEl?.complete).toBe(false);

await waitFor(
() =>
expect(imgEl.style.backgroundImage, 'it has a background PNG').to.match(
/data:image\/png/,
),
{ timeout: 5000 },
);
expect(imgEl).toHaveStyle({ backgroundSize: 'cover' });
expect(imgEl.style.backgroundImage).toMatchInlineSnapshot(`"url("")"`);

await loaded;

expect(
window.getComputedStyle(imgEl!).backgroundImage,
'after image is loaded the background PNG is removed',
).to.equal('none');
});
});
33 changes: 32 additions & 1 deletion packages/solid/tests/server.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,45 @@ describe('ResponsiveImage', () => {
imageUrlFor(width, type = 'jpeg') {
return `/provider/w${width}/image.${type}`;
},
aspectRatio: 1.5,
};

it('renders a responsive image', () => {
const string = renderToString(() => (
<ResponsiveImage src={defaultImageData} />
));
expect(string).toMatchInlineSnapshot(
`"<picture><source srcset="/provider/w640/image.avif 640w, /provider/w750/image.avif 750w, /provider/w828/image.avif 828w, /provider/w1080/image.avif 1080w, /provider/w1200/image.avif 1200w, /provider/w1920/image.avif 1920w, /provider/w2048/image.avif 2048w, /provider/w3840/image.avif 3840w" type="image/avif"><source srcset="/provider/w640/image.webp 640w, /provider/w750/image.webp 750w, /provider/w828/image.webp 828w, /provider/w1080/image.webp 1080w, /provider/w1200/image.webp 1200w, /provider/w1920/image.webp 1920w, /provider/w2048/image.webp 2048w, /provider/w3840/image.webp 3840w" type="image/webp"><source srcset="/provider/w640/image.jpeg 640w, /provider/w750/image.jpeg 750w, /provider/w828/image.jpeg 828w, /provider/w1080/image.jpeg 1080w, /provider/w1200/image.jpeg 1200w, /provider/w1920/image.jpeg 1920w, /provider/w2048/image.jpeg 2048w, /provider/w3840/image.jpeg 3840w" type="image/jpeg"><img width="320" class="ri-responsive " loading="lazy" decoding="async" src="/provider/w320/image.jpeg" style=""/></picture>"`,
`"<picture><source srcset="/provider/w640/image.avif 640w, /provider/w750/image.avif 750w, /provider/w828/image.avif 828w, /provider/w1080/image.avif 1080w, /provider/w1200/image.avif 1200w, /provider/w1920/image.avif 1920w, /provider/w2048/image.avif 2048w, /provider/w3840/image.avif 3840w" type="image/avif"><source srcset="/provider/w640/image.webp 640w, /provider/w750/image.webp 750w, /provider/w828/image.webp 828w, /provider/w1080/image.webp 1080w, /provider/w1200/image.webp 1200w, /provider/w1920/image.webp 1920w, /provider/w2048/image.webp 2048w, /provider/w3840/image.webp 3840w" type="image/webp"><source srcset="/provider/w640/image.jpeg 640w, /provider/w750/image.jpeg 750w, /provider/w828/image.jpeg 828w, /provider/w1080/image.jpeg 1080w, /provider/w1200/image.jpeg 1200w, /provider/w1920/image.jpeg 1920w, /provider/w2048/image.jpeg 2048w, /provider/w3840/image.jpeg 3840w" type="image/jpeg"><img width="320" height="213" class="ri-responsive " loading="lazy" decoding="async" src="/provider/w320/image.jpeg" style=""/></picture>"`,
);
});

it('renders blurhash LQIP', () => {
const imageData: ImageData = {
...defaultImageData,
lqip: {
type: 'blurhash',
hash: 'LEHV6nWB2yk8pyo0adR*.7kCMdnj',
width: 4,
height: 3,
},
};
const string = renderToString(() => <ResponsiveImage src={imageData} />);
expect(string).toMatchInlineSnapshot(
`"<picture><source srcset="/provider/w640/image.avif 640w, /provider/w750/image.avif 750w, /provider/w828/image.avif 828w, /provider/w1080/image.avif 1080w, /provider/w1200/image.avif 1200w, /provider/w1920/image.avif 1920w, /provider/w2048/image.avif 2048w, /provider/w3840/image.avif 3840w" type="image/avif"><source srcset="/provider/w640/image.webp 640w, /provider/w750/image.webp 750w, /provider/w828/image.webp 828w, /provider/w1080/image.webp 1080w, /provider/w1200/image.webp 1200w, /provider/w1920/image.webp 1920w, /provider/w2048/image.webp 2048w, /provider/w3840/image.webp 3840w" type="image/webp"><source srcset="/provider/w640/image.jpeg 640w, /provider/w750/image.jpeg 750w, /provider/w828/image.jpeg 828w, /provider/w1080/image.jpeg 1080w, /provider/w1200/image.jpeg 1200w, /provider/w1920/image.jpeg 1920w, /provider/w2048/image.jpeg 2048w, /provider/w3840/image.jpeg 3840w" type="image/jpeg"><img width="320" height="213" class="ri-responsive ri-lqip-blurhash " loading="lazy" decoding="async" src="/provider/w320/image.jpeg" data-ri-bh="LEHV6nWB2yk8pyo0adR*.7kCMdnj" data-ri-bh-w="4" data-ri-bh-h="3" style=""/></picture>"`,
);
});

it('renders thumbhash LQIP', () => {
const imageData: ImageData = {
...defaultImageData,
lqip: {
type: 'thumbhash',
hash: 'jJcFFYI1fIWHe4dweXlYeUaAmWj3',
},
};
const string = renderToString(() => <ResponsiveImage src={imageData} />);
expect(string).toMatchInlineSnapshot(
`"<picture><source srcset="/provider/w640/image.avif 640w, /provider/w750/image.avif 750w, /provider/w828/image.avif 828w, /provider/w1080/image.avif 1080w, /provider/w1200/image.avif 1200w, /provider/w1920/image.avif 1920w, /provider/w2048/image.avif 2048w, /provider/w3840/image.avif 3840w" type="image/avif"><source srcset="/provider/w640/image.webp 640w, /provider/w750/image.webp 750w, /provider/w828/image.webp 828w, /provider/w1080/image.webp 1080w, /provider/w1200/image.webp 1200w, /provider/w1920/image.webp 1920w, /provider/w2048/image.webp 2048w, /provider/w3840/image.webp 3840w" type="image/webp"><source srcset="/provider/w640/image.jpeg 640w, /provider/w750/image.jpeg 750w, /provider/w828/image.jpeg 828w, /provider/w1080/image.jpeg 1080w, /provider/w1200/image.jpeg 1200w, /provider/w1920/image.jpeg 1920w, /provider/w2048/image.jpeg 2048w, /provider/w3840/image.jpeg 3840w" type="image/jpeg"><img width="320" height="213" class="ri-responsive ri-lqip-thumbhash " loading="lazy" decoding="async" src="/provider/w320/image.jpeg" data-ri-th="jJcFFYI1fIWHe4dweXlYeUaAmWj3" style=""/></picture>"`,
);
});
});

0 comments on commit 0429d13

Please sign in to comment.