fix: updates <Picture /> to pass HTML attributes down to the <img /> element (#5038)

* fix: also pass attrs to underlying img

`class` for direct img styling, `width` & `height` to prevent layout shift, etc need to be passed to the underlying img tag to work.

* remove redundant attr from picture tag

* revert to previous commit

- seemed to break some tests
- can't deal with fixing that right now, maybe later

* only passing attributes to the img

* adding a note to the README

* chore: add changeset

Co-authored-by: Tony Sullivan <tony.f.sullivan@outlook.com>
This commit is contained in:
Emmanuel Chucks 2022-10-20 19:42:36 +00:00 committed by GitHub
parent 863391f10e
commit ed2dfdae5b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 46 additions and 27 deletions

View file

@ -0,0 +1,11 @@
---
'@astrojs/image': minor
---
HTML attributes included on the `<Picture />` component are now passed down to the underlying `<img />` element.
**Why?**
- when styling a `<picture>` the `class` and `style` attributes belong on the `<img>` itself
- `<picture>` elements [should not](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/picture#attributes) actually provide any `aria-` attributes
- `width` and `height` can be added to the `<img>` to help prevent layout shift

View file

@ -260,6 +260,10 @@ Position of the crop when fit is `cover` or `contain`.
### `<Picture />`
The built-in `<Picture />` component is used to create an optimized `<picture />` for both remote images hosted on other domains as well as local images imported from your project's `src` directory.
In addition to the component-specific properties, any valid HTML attribute for the `<img />` included in the `<Picture />` component will be included in the built `<img />`.
#### src
<p>

View file

@ -65,9 +65,9 @@ delete image.width;
delete image.height;
---
<picture {...attrs}>
<picture>
{sources.map((attrs) => <source {...attrs} {sizes} />)}
<img {...image} {loading} {decoding} {alt} />
<img {...image} {loading} {decoding} {alt} {...attrs} />
</picture>
<style>

View file

@ -78,10 +78,11 @@ describe('SSG pictures - dev', function () {
},
].forEach(({ title, id, url, query, alt }) => {
it(title, () => {
const sources = $(`${id} source`);
expect(sources.length).to.equal(3);
const image = $(`${id}`);
const picture = image.closest('picture');
const image = $(`${id} img`);
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const src = image.attr('src');
const [route, params] = src.split('?');
@ -158,10 +159,11 @@ describe('SSG pictures with subpath - dev', function () {
},
].forEach(({ title, id, url, query, alt }) => {
it(title, () => {
const sources = $(`${id} source`);
expect(sources.length).to.equal(3);
const image = $(`${id}`);
const picture = image.closest('picture');
const image = $(`${id} img`);
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const src = image.attr('src');
const [route, params] = src.split('?');
@ -250,10 +252,11 @@ describe('SSG pictures - build', function () {
},
].forEach(({ title, id, regex, size, alt }) => {
it(title, () => {
const sources = $(`${id} source`);
expect(sources.length).to.equal(3);
const image = $(`${id}`);
const picture = image.closest('picture');
const image = $(`${id} img`);
const sources = picture.children('source');
expect(sources.length).to.equal(3);
expect(image.attr('src')).to.match(regex);
expect(image.attr('alt')).to.equal(alt);
@ -349,10 +352,11 @@ describe('SSG pictures with subpath - build', function () {
},
].forEach(({ title, id, regex, size, alt }) => {
it(title, () => {
const sources = $(`${id} source`);
expect(sources.length).to.equal(3);
const image = $(`${id}`);
const picture = image.closest('picture');
const image = $(`${id} img`);
const sources = picture.children('source');
expect(sources.length).to.equal(3);
expect(image.attr('src')).to.match(regex);
expect(image.attr('alt')).to.equal(alt);

View file

@ -89,12 +89,12 @@ describe('SSR pictures - build', function () {
const html = await response.text();
const $ = cheerio.load(html);
const sources = $(`${id} source`);
const image = $(`${id}`);
const picture = image.closest('picture');
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const image = $(`${id} img`);
const src = image.attr('src');
const [route, params] = src.split('?');
@ -200,12 +200,12 @@ describe('SSR pictures with subpath - build', function () {
const html = await response.text();
const $ = cheerio.load(html);
const sources = $(`${id} source`);
const image = $(`${id}`);
const picture = image.closest('picture');
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const image = $(`${id} img`);
const src = image.attr('src');
const [route, params] = src.split('?');

View file

@ -104,12 +104,12 @@ describe('SSR pictures - dev', function () {
},
].forEach(({ title, id, url, query, alt, contentType }) => {
it(title, async () => {
const sources = $(`${id} source`);
const image = $(`${id}`);
const picture = image.closest('picture');
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const image = $(`${id} img`);
const src = image.attr('src');
const [route, params] = src.split('?');
@ -220,12 +220,12 @@ describe('SSR pictures with subpath - dev', function () {
},
].forEach(({ title, id, url, query, alt, contentType }) => {
it(title, async () => {
const sources = $(`${id} source`);
const image = $(`${id}`);
const picture = image.closest('picture');
const sources = picture.children('source');
expect(sources.length).to.equal(3);
const image = $(`${id} img`);
const src = image.attr('src');
const [route, params] = src.split('?');