Skip to content

Commit

Permalink
fix(gatsby-image): Safari Downloading multiple resolutions
Browse files Browse the repository at this point in the history
Safari browser (Safari 12 / Mac OS) downloads multiple assets for a single `Img` tag.

The observed behavior is that the attributes of the `img` tag are queued early in the parsing of the document.

Resolved by removing the fallback `<source>` and placing those value back into the `<img>` tag.

Additionally the order of the attributes of `img` are important as well.   If `src` preceeds `srcSet` in the `<img>` Safari’s Pre-parser will enqueue both images to download.

Finally - added a note regarding a dependency on IntersectionObserver for lazy-loading
  • Loading branch information
Todd Brannam committed Feb 20, 2019
1 parent d33b3ec commit d909ded
Show file tree
Hide file tree
Showing 6 changed files with 21 additions and 21 deletions.
2 changes: 1 addition & 1 deletion e2e-tests/gatsby-image/src/pages/fixed.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export const pageQuery = graphql`
fruitsFixed: file(relativePath: { eq: "citrus-fruits.jpg" }) {
childImageSharp {
fixed(width: 500) {
...GatsbyImageSharpFixed
...GatsbyImageSharpFixed_withWebp

This comment has been minimized.

Copy link
@tbrannam

tbrannam Feb 20, 2019

Contributor

added withWebp to force a source to continue to be generated to satisfy e2e tests

}
}
}
Expand Down
2 changes: 1 addition & 1 deletion e2e-tests/gatsby-image/src/pages/fluid.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export const pageQuery = graphql`
fruitsFluid: file(relativePath: { eq: "citrus-fruits.jpg" }) {
childImageSharp {
fluid(maxWidth: 500) {
...GatsbyImageSharpFluid
...GatsbyImageSharpFluid_withWebp
}
}
}
Expand Down
2 changes: 2 additions & 0 deletions packages/gatsby-image/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -348,3 +348,5 @@ You will need to add it in your graphql query as is shown in the following snipp
- Gifs can't be resized the same way as pngs and jpegs, unfortunately—if you try
to use a gif with `gatsby-image`, it won't work. For now, the best workaround is
to [import the gif directly](/docs/adding-images-fonts-files).
- Lazy loading behavior is dependent on `IntersectionObserver` which is not available
in some fairly common browsers including Safari and IE. A polyfill is recommended.
14 changes: 5 additions & 9 deletions packages/gatsby-image/src/__tests__/__snapshots__/index.js.snap
Original file line number Diff line number Diff line change
Expand Up @@ -22,21 +22,19 @@ exports[`<Img /> should render fixed size images 1`] = `
srcset="some srcSetWebp"
type="image/webp"
/>
<source
srcset="some srcSet"
/>
<img
alt="Alt text for the image"
height="100"
itemprop="item-prop-for-the-image"
src="test_image.jpg"
srcset="some srcSet"
style="position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; opacity: 0;"
title="Title for the image"
width="100"
/>
</picture>
<noscript>
&lt;picture&gt;&lt;source type='image/webp' srcSet="some srcSetWebp" /&gt;&lt;source srcSet="some srcSet" /&gt;&lt;img width="100" height="100" src="test_image.jpg" alt="Alt text for the image" title="Title for the image" style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:0.5s;opacity:1;width:100%;height:100%;object-fit:cover;object-position:center"/&gt;&lt;/picture&gt;
&lt;picture&gt;&lt;source type='image/webp' srcset="some srcSetWebp" /&gt;&lt;img width="100" height="100" srcset="some srcSet" src="test_image.jpg" alt="Alt text for the image" title="Title for the image" style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:0.5s;opacity:1;width:100%;height:100%;object-fit:cover;object-position:center"/&gt;&lt;/picture&gt;
</noscript>
</div>
</div>
Expand Down Expand Up @@ -68,20 +66,18 @@ exports[`<Img /> should render fluid images 1`] = `
srcset="some srcSetWebp"
type="image/webp"
/>
<source
sizes="(max-width: 600px) 100vw, 600px"
srcset="some srcSet"
/>
<img
alt="Alt text for the image"
itemprop="item-prop-for-the-image"
sizes="(max-width: 600px) 100vw, 600px"
src="test_image.jpg"
srcset="some srcSet"
style="position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; opacity: 0;"
title="Title for the image"
/>
</picture>
<noscript>
&lt;picture&gt;&lt;source type='image/webp' srcSet="some srcSetWebp" sizes="(max-width: 600px) 100vw, 600px" /&gt;&lt;source srcSet="some srcSet" sizes="(max-width: 600px) 100vw, 600px" /&gt;&lt;img src="test_image.jpg" alt="Alt text for the image" title="Title for the image" style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:0.5s;opacity:1;width:100%;height:100%;object-fit:cover;object-position:center"/&gt;&lt;/picture&gt;
&lt;picture&gt;&lt;source type='image/webp' srcset="some srcSetWebp" sizes="(max-width: 600px) 100vw, 600px" /&gt;&lt;img sizes="(max-width: 600px) 100vw, 600px" srcset="some srcSet" src="test_image.jpg" alt="Alt text for the image" title="Title for the image" style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:0.5s;opacity:1;width:100%;height:100%;object-fit:cover;object-position:center"/&gt;&lt;/picture&gt;
</noscript>
</div>
</div>
Expand Down
1 change: 1 addition & 0 deletions packages/gatsby-image/src/__tests__/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,7 @@ describe(`<Img />`, () => {
it(`should have correct src, title and alt attributes`, () => {
const imageTag = setup().querySelector(`picture img`)
expect(imageTag.getAttribute(`src`)).toEqual(`test_image.jpg`)
expect(imageTag.getAttribute(`srcSet`)).toEqual(`some srcSet`)
expect(imageTag.getAttribute(`title`)).toEqual(`Title for the image`)
expect(imageTag.getAttribute(`alt`)).toEqual(`Alt text for the image`)
})
Expand Down
21 changes: 11 additions & 10 deletions packages/gatsby-image/src/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -80,25 +80,26 @@ const noscriptImg = props => {
const src = props.src ? `src="${props.src}" ` : `src="" ` // required attribute
const sizes = props.sizes ? `sizes="${props.sizes}" ` : ``
const srcSetWebp = props.srcSetWebp
? `<source type='image/webp' srcSet="${props.srcSetWebp}" ${sizes}/>`
: ``
const srcSet = props.srcSet
? `<source srcSet="${props.srcSet}" ${sizes}/>`
? `<source type='image/webp' srcset="${props.srcSetWebp}" ${sizes}/>`
: ``
const srcSet = props.srcSet ? `srcset="${props.srcSet}" ` : ``
const title = props.title ? `title="${props.title}" ` : ``
const alt = props.alt ? `alt="${props.alt}" ` : `alt="" ` // required attribute
const width = props.width ? `width="${props.width}" ` : ``
const height = props.height ? `height="${props.height}" ` : ``
const opacity = props.opacity ? props.opacity : `1`
const transitionDelay = props.transitionDelay ? props.transitionDelay : `0.5s`
return `<picture>${srcSetWebp}${srcSet}<img ${width}${height}${src}${alt}${title}style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:${transitionDelay};opacity:${opacity};width:100%;height:100%;object-fit:cover;object-position:center"/></picture>`
return `<picture>${srcSetWebp}<img ${width}${height}${sizes}${srcSet}${src}${alt}${title}style="position:absolute;top:0;left:0;transition:opacity 0.5s;transition-delay:${transitionDelay};opacity:${opacity};width:100%;height:100%;object-fit:cover;object-position:center"/></picture>`
}

const Img = React.forwardRef((props, ref) => {
const { style, onLoad, onError, ...otherProps } = props
const { sizes, srcSet, src, style, onLoad, onError, ...otherProps } = props

return (
<img
sizes={sizes}
srcSet={srcSet}

This comment has been minimized.

Copy link
@tbrannam

tbrannam Feb 20, 2019

Contributor

order matters to Safari - srcset should precede src - otherwise src and srcset image are both enqueued for download

src={src}
{...otherProps}
onLoad={onLoad}
onError={onError}
Expand Down Expand Up @@ -315,12 +316,12 @@ class Image extends React.Component {
/>
)}

<source srcSet={image.srcSet} sizes={image.sizes} />

<Img
alt={alt}
title={title}
sizes={image.sizes}
src={image.src}
srcSet={image.srcSet}
style={imageStyle}
ref={this.imageRef}
onLoad={this.handleImageLoaded}
Expand Down Expand Up @@ -399,14 +400,14 @@ class Image extends React.Component {
/>
)}

<source srcSet={image.srcSet} sizes={image.sizes} />

<Img
alt={alt}
title={title}
width={image.width}
height={image.height}
sizes={image.sizes}
src={image.src}
srcSet={image.srcSet}
style={imageStyle}
ref={this.imageRef}
onLoad={this.handleImageLoaded}
Expand Down

0 comments on commit d909ded

Please sign in to comment.