0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2024-12-30 22:03:56 -05:00

fix: don't apply duplicate class to images (#12631)

* fix: don't apply duplicate class to images

* Fromat
This commit is contained in:
Matt Kane 2024-12-04 18:26:17 +00:00 committed by GitHub
parent 176fe9f113
commit dec0305b75
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 130 additions and 4 deletions

View file

@ -0,0 +1,5 @@
---
'astro': patch
---
Fixes a bug where the class attribute was rendered twice on the image component

View file

@ -47,7 +47,7 @@ if (import.meta.env.DEV) {
additionalAttributes['data-image-component'] = 'true'; additionalAttributes['data-image-component'] = 'true';
} }
const attributes = useResponsive const { class: className, ...attributes } = useResponsive
? applyResponsiveAttributes({ ? applyResponsiveAttributes({
layout, layout,
image, image,
@ -58,4 +58,4 @@ const attributes = useResponsive
--- ---
{/* Applying class outside of the spread prevents it from applying unnecessary astro-* classes */} {/* Applying class outside of the spread prevents it from applying unnecessary astro-* classes */}
<img src={image.src} {...attributes} class={attributes.class} /> <img src={image.src} {...attributes} class={className} />

View file

@ -102,7 +102,7 @@ if (fallbackImage.srcSet.values.length > 0) {
imgAdditionalAttributes.srcset = fallbackImage.srcSet.attribute; imgAdditionalAttributes.srcset = fallbackImage.srcSet.attribute;
} }
const attributes = useResponsive const { class: className, ...attributes } = useResponsive
? applyResponsiveAttributes({ ? applyResponsiveAttributes({
layout, layout,
image: fallbackImage, image: fallbackImage,
@ -133,5 +133,5 @@ if (import.meta.env.DEV) {
}) })
} }
{/* Applying class outside of the spread prevents it from applying unnecessary astro-* classes */} {/* Applying class outside of the spread prevents it from applying unnecessary astro-* classes */}
<img src={fallbackImage.src} {...attributes} class={attributes.class} /> <img src={fallbackImage.src} {...attributes} class={className} />
</picture> </picture>

View file

@ -112,6 +112,7 @@ describe('astro:image:layout', () => {
it('passes in a parent class', () => { it('passes in a parent class', () => {
let $img = $('#local-class img'); let $img = $('#local-class img');
assert.match($img.attr('class'), /green/); assert.match($img.attr('class'), /green/);
}); });
@ -576,4 +577,124 @@ describe('astro:image:layout', () => {
}); });
}); });
}); });
describe('build', () => {
before(async () => {
fixture = await loadFixture({
root: './fixtures/core-image-layout/',
image: {
service: testImageService({ foo: 'bar' }),
domains: ['avatars.githubusercontent.com'],
},
});
await fixture.build();
});
describe('basics', () => {
let $;
let html;
before(async () => {
html = await fixture.readFile('/index.html');
$ = cheerio.load(html);
});
it('Adds the <img> tag', () => {
let $img = $('#local img');
assert.equal($img.length, 1);
assert.ok($img.attr('src').startsWith('/_astro'));
});
it('includes lazy loading attributes', () => {
let $img = $('#local img');
assert.equal($img.attr('loading'), 'lazy');
assert.equal($img.attr('decoding'), 'async');
assert.equal($img.attr('fetchpriority'), 'auto');
});
it('includes priority loading attributes', () => {
let $img = $('#local-priority img');
assert.equal($img.attr('loading'), 'eager');
assert.equal($img.attr('decoding'), 'sync');
assert.equal($img.attr('fetchpriority'), 'high');
});
it('has width and height - no dimensions set', () => {
let $img = $('#local img');
assert.equal($img.attr('width'), '2316');
assert.equal($img.attr('height'), '1544');
});
it('has proper width and height - only width', () => {
let $img = $('#local-width img');
assert.equal($img.attr('width'), '350');
assert.equal($img.attr('height'), '233');
});
it('has proper width and height - only height', () => {
let $img = $('#local-height img');
assert.equal($img.attr('width'), '300');
assert.equal($img.attr('height'), '200');
});
it('has proper width and height - has both width and height', () => {
let $img = $('#local-both img');
assert.equal($img.attr('width'), '300');
assert.equal($img.attr('height'), '400');
});
it('sets the style', () => {
let $img = $('#local-both img');
assert.match($img.attr('style'), /--w: 300/);
assert.match($img.attr('style'), /--h: 400/);
assert.equal($img.data('astro-image'), 'responsive');
});
it('sets the style when no dimensions set', () => {
let $img = $('#local img');
assert.match($img.attr('style'), /--w: 2316/);
assert.match($img.attr('style'), /--h: 1544/);
assert.equal($img.data('astro-image'), 'responsive');
});
it('sets style for fixed image', () => {
let $img = $('#local-fixed img');
assert.match($img.attr('style'), /--w: 800/);
assert.match($img.attr('style'), /--h: 600/);
assert.equal($img.data('astro-image'), 'fixed');
});
it('sets style for full-width image', () => {
let $img = $('#local-full-width img');
assert.equal($img.data('astro-image'), 'full-width');
});
it('passes in a parent class', () => {
let $img = $('#local-class img');
assert.equal($img.prop('class'), 'green');
});
it('only passes the class in once', () => {
// Check that class="green" only appears once
// We can't use cheerio because it normalises the DOM, so we have to use a regex
const matches = html.match(/class="green"/g);
assert.equal(matches.length, 1);
});
it('passes in a parent style', () => {
let $img = $('#local-style img');
assert.match($img.attr('style'), /border: 2px red solid/);
});
it('passes in a parent style as an object', () => {
let $img = $('#local-style-object img');
assert.match($img.attr('style'), /border:2px red solid/);
});
it('injects a style tag', () => {
const style = $('style').text();
assert.match(style, /\[data-astro-image\]/);
});
});
});
}); });