The below code example represents the markup needed for the background image, page title, and surrounding content:
<div class="parallax">
<h1>This is a Parallax Scroll Example</h1>
</div>
<div class="content-outer">
<div class="content-inner">
<p>Paragraph 1</p>
<p>Paragraph 2</p>
<p>Paragraph 3</p>
<p>Paragraph 4</p>
<p>Paragraph 5</p>
</div>
</div>
The .parallax
element is the container where the background image will be stored using a pseudo-selector, and also where the page title will be displayed.
The .content-outer
and .content-inner
elements contain the page content that exists below the background image and page title.
For the CSS rules, we'll start by removing all padding and margin spacing around the edge of our document. This is just to clean up the display:
html, body {
padding: 0px;
margin: 0px;
}
Second, we'll remove all horizontal and vertical scrolling ability from the html
selector by setting its overflow
to hidden. This is required for the effect to work correctly:
html {
overflow: hidden;
}
And, third, we'll set the body width and height to the size of the viewport. We'll also enable vertical scrolling in case the page contents exceed the viewport height (which they will in this example) while keeping the horizontal scroll disabled.
We'll add a perspective
rule to give the element a 3D-space by adjusting the distance between the user and the element on the Z-axis. This output doesn't affect the element itself, only its child elements and how they will be displayed.
We'll then use the transform-style
rule with a value of preserve-3d to ensure that the child elements of body
will maintain its 3D position:
body {
width: 100vw;
height: 100vh;
overflow-x: hidden;
overflow-y: auto;
perspective: 1px;
-webkit-perspective: 1px;
transform-style: preserve-3d;
-webkit-transform-style: preserve-3d;
}
Now we'll set rules for our .parallax
element, the container that will house the background image displayed on the screen, as well as the page title.
Here, we set the element's size to 100% of the viewport width and height so that it covers the entire browser window.
Setting its position
to relative ensures that the background image positioning will be retained to its parent's boundaries.
And, again, we're setting transform-style
to preserve-3d to ensure that this container maintains its 3D positioning, as well:
div.parallax {
width: 100vw;
height: 100vh;
position: relative;
transform-style: preserve-3d;
-webkit-transform-style: preserve-3d;
}
Adding thetransform-style
rule to the.parallax
element was required for this effect to work properly in Firefox. This is because Firefox doesn't know to maintain 3D transformation rules within additional child elements and must be instructed to do so. You could also set this value to inherit for the same outcome.
Now, we have the .parallax:after
pseudo-element that provides the rules for our background image and how it will be positioned in the .parallax
container:
div.parallax:after {
background-image: url("/bg.jpg");
background-repeat: no-repeat;
background-position: center;
background-size: cover;
position: absolute;
top: 0px;
left: 0px;
right: 0px;
bottom: 0px;
z-index: -1;
content: "";
transform: translateZ(-1px) scale(2);
-webkit-transform: translateZ(-1px) scale(2);
}
Here, we've set the background image URL and its position within the .parallax
element. The positioning must be absolute and centered, which is accomplished by setting the top
, left
, right
, and bottom
rules to 0px. This sets the bounding box for the element and instructs the browser to fill all available space with the contents of that element.
Next, we've created a negative z-index
value to ensure that there is no overflow of content from other elements within the page, and to slow down the scroll speed which we'll also accomplish with our next CSS rule.
Setting negative Z values with the translateZ()
property ensures that the scroll speed will be slower by zooming out from its original position on the Z-axis. We then provide depth correction with the scale()
property, doubling the background image size so it renders at its original size for the user.
The scale factor is calculated as((translateZ * -1) + 1) / perspective
. So, if our viewportperspective
is set to 1px and we translate an element -1px along the Z-axis like we have in this example, the correctionscale
factor would be 2.
The caveat with Safari and iOS browsers is where the content below the .parallax
element overlaps the background image.
Generally, with the other browsers, you could set a CSS rule of overflow: hidden
to the .parallax
element to correct this issue. But, with Safari and iOS browsers, we need to take a different approach.
This can be corrected by simply increasing the perspective
value of the content below the .parallax
element to 2px, 1 pixel larger than the .parallax
element itself:
div.content-outer {
width: 100%;
background-color: white;
perspective: 2px;
-webkit-perspective: 2px;
}
The content will appear scaled as normal to the user but tricks the browser into thinking the content below the .parallax
element is actually higher in the Z-axis, and therefore covers up the background image as needed.
The tutorial explained how to create a parallax scroll effect with CSS in all modern browsers using a single set of rules.
You can download the complete source code from my GitHub repository. This codeset includes the assets from the examples above and will allow you to play around with different values as needed to further understand how each of the CSS rules works and interacts with each other.
Written by: Josh Rowe
Last Updated: March 18, 2024