Make About page stacking card scroll effect

Using Motion One example, thanks to https://codepen.io/bramus/pen/rNdzpZK (by Bramus)
This commit is contained in:
2022-08-13 16:10:01 +02:00
parent 4a83ade94b
commit f688928757
6 changed files with 143 additions and 84 deletions

View File

@@ -20,6 +20,7 @@
"dayjs": "^1.11.4", "dayjs": "^1.11.4",
"embla-carousel": "^7.0.0", "embla-carousel": "^7.0.0",
"focus-visible": "^5.2.0", "focus-visible": "^5.2.0",
"motion": "^10.13.3",
"ogl": "^0.0.97", "ogl": "^0.0.97",
"sanitize.css": "^13.0.0", "sanitize.css": "^13.0.0",
"tweakpane": "^3.1.0" "tweakpane": "^3.1.0"

74
pnpm-lock.yaml generated
View File

@@ -17,6 +17,7 @@ specifiers:
eslint: ^8.21.0 eslint: ^8.21.0
eslint-plugin-svelte3: ^4.0.0 eslint-plugin-svelte3: ^4.0.0
focus-visible: ^5.2.0 focus-visible: ^5.2.0
motion: ^10.13.3
ogl: ^0.0.97 ogl: ^0.0.97
postcss: ^8.4.16 postcss: ^8.4.16
postcss-focus-visible: ^7.1.0 postcss-focus-visible: ^7.1.0
@@ -40,6 +41,7 @@ dependencies:
dayjs: 1.11.4 dayjs: 1.11.4
embla-carousel: 7.0.0 embla-carousel: 7.0.0
focus-visible: 5.2.0 focus-visible: 5.2.0
motion: 10.13.3
ogl: 0.0.97 ogl: 0.0.97
sanitize.css: 13.0.0 sanitize.css: 13.0.0
tweakpane: 3.1.0 tweakpane: 3.1.0
@@ -298,6 +300,67 @@ packages:
- supports-color - supports-color
dev: true dev: true
/@motionone/animation/10.13.2:
resolution: {integrity: sha512-YGWss58IR2X4lOjW89rv1Q+/Nq/QhfltaggI7i8sZTpKC1yUvM+XYDdvlRpWc6dk8LviMBrddBJAlLdbaqeRmw==}
dependencies:
'@motionone/easing': 10.13.2
'@motionone/types': 10.13.2
'@motionone/utils': 10.13.2
tslib: 2.4.0
dev: false
/@motionone/dom/10.13.2:
resolution: {integrity: sha512-THCRW+M7JUTbHRiWFJU13pju7bZlnPjZz9VDXA/h/Qj/Or7LvzKY/XfEy45rUlO0wXSa4lQ+1DGlrZNewamdDA==}
dependencies:
'@motionone/animation': 10.13.2
'@motionone/generators': 10.13.2
'@motionone/types': 10.13.2
'@motionone/utils': 10.13.2
hey-listen: 1.0.8
tslib: 2.4.0
dev: false
/@motionone/easing/10.13.2:
resolution: {integrity: sha512-3HqctS5NyDfDQ+8+cZqc3Pu7I6amFCt9zDUjcozHyFXHh4PKYHK4+GJDFjJIS8bCAF2BrJmpmduDQ2V7lFEYeQ==}
dependencies:
'@motionone/utils': 10.13.2
tslib: 2.4.0
dev: false
/@motionone/generators/10.13.2:
resolution: {integrity: sha512-QMoXV1MXEEhR6D3dct/RMMS1FwJlAsW+kMPbFGzBA4NbweblgeYQCft9DcDAVpV9wIwD6qvlBG9u99sOXLfHiA==}
dependencies:
'@motionone/types': 10.13.2
'@motionone/utils': 10.13.2
tslib: 2.4.0
dev: false
/@motionone/svelte/10.13.2:
resolution: {integrity: sha512-p6gH7oGhbonHzrPeUPzZiU3dx2yjDYQ7K00w9GaTbXBfmH/+rjUQ6vvG5SPvjY6BLtMK3/FMK512WvOMtvDQPQ==}
dependencies:
'@motionone/dom': 10.13.2
tslib: 2.4.0
dev: false
/@motionone/types/10.13.2:
resolution: {integrity: sha512-yYV4q5v5F0iADhab4wHfqaRJnM/eVtQLjUPhyEcS72aUz/xyOzi09GzD/Gu+K506BDfqn5eULIilUI77QNaqhw==}
dev: false
/@motionone/utils/10.13.2:
resolution: {integrity: sha512-6Lw5bDA/w7lrPmT/jYWQ76lkHlHs9fl2NZpJ22cVy1kKDdEH+Cl1U6hMTpdphO6VQktQ6v2APngag91WBKLqlA==}
dependencies:
'@motionone/types': 10.13.2
hey-listen: 1.0.8
tslib: 2.4.0
dev: false
/@motionone/vue/10.13.2:
resolution: {integrity: sha512-DfMzOUwKlzyjpwxF+RP1Q74ClmeoanPAeSGfD/JTvAyR1W6ARCOBMFpdgDvile1o7FNgHGx2RHt8210MsOE59g==}
dependencies:
'@motionone/dom': 10.13.2
tslib: 2.4.0
dev: false
/@nodelib/fs.scandir/2.1.5: /@nodelib/fs.scandir/2.1.5:
resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==} resolution: {integrity: sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==}
engines: {node: '>= 8'} engines: {node: '>= 8'}
@@ -1848,6 +1911,17 @@ packages:
hasBin: true hasBin: true
dev: true dev: true
/motion/10.13.3:
resolution: {integrity: sha512-lfBuoZL8xo0djAD3zGrMuSuc4J9MquE6hzRCsO9cVp+DgIUvNmvbM7+7SrZtpjLKxPWbppm+P56FRtVvMcGj1A==}
dependencies:
'@motionone/animation': 10.13.2
'@motionone/dom': 10.13.2
'@motionone/svelte': 10.13.2
'@motionone/types': 10.13.2
'@motionone/utils': 10.13.2
'@motionone/vue': 10.13.2
dev: false
/mri/1.2.0: /mri/1.2.0:
resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==} resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==}
engines: {node: '>=4'} engines: {node: '>=4'}

View File

@@ -3,7 +3,6 @@
</style> </style>
<script lang="ts"> <script lang="ts">
import { lerp } from '$utils/functions'
// Components // Components
import Image from '$components/atoms/Image.svelte' import Image from '$components/atoms/Image.svelte'
@@ -11,29 +10,11 @@
export let title: string export let title: string
export let text: string export let text: string
export let image: any = undefined export let image: any = undefined
export let progress: number = 0
let stepEl: HTMLElement
let scrollY: number, innerWidth: number, innerHeight: number
const imageRatio = image ? image.width / image.height : undefined const imageRatio = image ? image.width / image.height : undefined
const scale = lerp(0.925, 1, progress)
$: isMobile = innerWidth < 550
$: offsetTop = stepEl && stepEl.offsetTop
$: offsetStagger = lerp(isMobile ? 16 : 48, isMobile ? 64 : 120, progress)
$: isPinned = scrollY - innerHeight * 1.4 >= offsetTop
</script> </script>
<svelte:window bind:scrollY bind:innerWidth bind:innerHeight /> <div class="step" style:--index={index}>
<div bind:this={stepEl}
class="step"
class:is-pinned={isPinned}
style:--index={index}
style:--opacity-index={lerp(0.3, 0.05, progress)}
style:--scale={scale}
style:--offset-top="{offsetStagger}px"
>
<div class="step__card grid"> <div class="step__card grid">
{#if image} {#if image}
<Image <Image
@@ -48,11 +29,14 @@
alt={image.title} alt={image.title}
/> />
{/if} {/if}
<div class="content"> <div class="content">
<h3 class="title-medium">{title}</h3> <h3 class="title-medium">{title}</h3>
<div class="text text-small"> <div class="text text-small">
{@html text} {@html text}
</div> </div>
</div> </div>
<div class="overlay" />
</div> </div>
</div> </div>

View File

@@ -5,6 +5,7 @@
<script lang="ts"> <script lang="ts">
import { onMount, afterUpdate } from 'svelte' import { onMount, afterUpdate } from 'svelte'
import { map } from '$utils/functions' import { map } from '$utils/functions'
import { scroll, animate, type ScrollOptions } from 'motion'
// Components // Components
import Metas from '$components/Metas.svelte' import Metas from '$components/Metas.svelte'
import PageTransition from '$components/PageTransition.svelte' import PageTransition from '$components/PageTransition.svelte'
@@ -21,17 +22,16 @@
// console.log(data) // console.log(data)
let scrollY: number, innerWidth: number, innerHeight: number let scrollY: number, innerWidth: number, innerHeight: number
let stepsEl: HTMLElement
let photosGridEl: HTMLElement let photosGridEl: HTMLElement
let photosGridOffset: number = photosGridEl && photosGridEl.offsetTop let photosGridOffset: number = photosGridEl && photosGridEl.offsetTop
let sectionsObserver: IntersectionObserver let sectionsObserver: IntersectionObserver
// let stepsObserver: IntersectionObserver
$: parallaxPhotos = photosGridEl && map(scrollY, photosGridOffset - innerHeight, photosGridOffset + innerHeight / 1.5, 0, innerHeight * 0.15, true) $: parallaxPhotos = photosGridEl && map(scrollY, photosGridOffset - innerHeight, photosGridOffset + innerHeight / 1.5, 0, innerHeight * 0.15, true)
$: fadedPhotosIndexes = innerWidth > 768 $: fadedPhotosIndexes = innerWidth > 768
? [0, 2, 5, 7, 9, 12, 17, 20, 22, 26, 30, 32, 34] ? [0, 2, 5, 7, 9, 12, 17, 20, 22, 26, 30, 32, 34]
: [0] : [0]
onMount(() => { onMount(() => {
// Sections observer // Sections observer
sectionsObserver = new IntersectionObserver(([{ isIntersecting, target }]: [IntersectionObserverEntry] & [{ target: HTMLElement }]) => { sectionsObserver = new IntersectionObserver(([{ isIntersecting, target }]: [IntersectionObserverEntry] & [{ target: HTMLElement }]) => {
@@ -50,21 +50,34 @@
sections.forEach(section => sectionsObserver.observe(section)) sections.forEach(section => sectionsObserver.observe(section))
// Steps observer // Steps scroll animation
// stepsObserver = new IntersectionObserver(([{ intersectionRatio, target }]) => { const cards = stepsEl.querySelectorAll('.step')
// target.classList.toggle('is-pinned', intersectionRatio < 1) const cardsAmount = data.process_steps.length
// }, {
// threshold: 1, cards.forEach((card: HTMLElement, i: number) => {
// rootMargin: '-10% 0px 50%', const index = i + 1
// }) const reverseIndex = cardsAmount - index
// const steps = document.querySelectorAll('.about__process .steps > *') const overlay = card.querySelector('.overlay')
// steps.forEach(step => stepsObserver.observe(step)) const scrollOptions: ScrollOptions = {
target: stepsEl,
offset: [`${i / cardsAmount * 100}%`, `${index / cardsAmount * 100}%`],
}
// Card scale
scroll(animate(card, {
scale: [1, 1 - (0.02 * reverseIndex)]
}), scrollOptions)
// Overlay opacity
scroll(animate(overlay, {
opacity: [0, 0.2 + (0.05 * reverseIndex)]
}), scrollOptions)
})
// Destroy // Destroy
return () => { return () => {
// sectionsObserver && sectionsObserver.disconnect() sectionsObserver && sectionsObserver.disconnect()
// stepsObserver && stepsObserver.disconnect()
} }
}) })
@@ -106,16 +119,15 @@
<p class="text-normal">{data.process_subtitle}</p> <p class="text-normal">{data.process_subtitle}</p>
</div> </div>
<div class="steps"> <div class="steps" bind:this={stepsEl}
{#each data.process_steps as { title, text, image }, index} style:--cards-amount={data.process_steps.length}
<ProcessStep index={index + 1} >
{title} {text} {image} {#each data.process_steps as step, index}
progress={index / (data.process_steps.length - 1) * 1} <ProcessStep {...step} index={index} />
/>
{/each} {/each}
</div> </div>
<div class="intention" style:--offset-top="120px"> <div class="intention">
<p class="intention__content title-medium"> <p class="intention__content title-medium">
{data.process_intention} {data.process_intention}
</p> </p>

View File

@@ -1,17 +1,14 @@
// About page Step // About page Step
.step { .step {
position: sticky;
top: var(--offset-top);
// Card // Card
&__card { &__card {
position: relative;
display: block; display: block;
overflow: hidden; overflow: hidden;
padding: 56px 32px 32px; padding: 56px 32px 32px;
background: $color-primary-darker; background: $color-primary-darker;
border-radius: 12px; border-radius: 12px;
transform-origin: top center; transform: translateZ(0);
transition: transform 0.8s var(--ease-quart);
@include bp (sm) { @include bp (sm) {
--columns: 18; --columns: 18;
@@ -21,27 +18,12 @@
min-height: min(45vw, 720px); min-height: min(45vw, 720px);
border-radius: 16px; border-radius: 16px;
} }
// Overlay
&:before {
content: "";
display: block;
position: absolute;
z-index: -1;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: #000;
opacity: 0;
pointer-events: none;
transition: opacity 0.8s var(--ease-quart);
transform: translateZ(0);
}
} }
// Image // Image
:global(.image) { :global(.image) {
position: relative;
z-index: 2;
display: block; display: block;
overflow: hidden; overflow: hidden;
width: 70%; width: 70%;
@@ -66,6 +48,9 @@
// Content // Content
.content { .content {
position: relative;
z-index: 2;
@include bp (sm) { @include bp (sm) {
grid-column: 2 / span 7; grid-column: 2 / span 7;
grid-row: 2; grid-row: 2;
@@ -87,6 +72,22 @@
} }
} }
// Overlay
.overlay {
content: "";
display: block;
position: absolute;
z-index: 0;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: #000;
opacity: 0;
pointer-events: none;
transform: translateZ(0);
}
// Alternate content order // Alternate content order
&:nth-child(even) { &:nth-child(even) {
:global(.image) { :global(.image) {
@@ -100,19 +101,4 @@
} }
} }
} }
/*
** States
*/
// Is pinned
&:global(.is-pinned) {
.step__card {
transform: scale(var(--scale)) translateZ(0);
&:before {
opacity: var(--opacity-index);
}
}
}
} }

View File

@@ -100,19 +100,22 @@
// Steps grid // Steps grid
.steps { .steps {
--card-offset: 16px;
--card-margin: 40px;
grid-column: 1 / -1; grid-column: 1 / -1;
@include bp (sm) { @include bp (sm) {
grid-column: 4 / -4; grid-column: 4 / -4;
padding-bottom: calc(var(--cards-amount) * var(--card-offset));
} }
& > :global(*) { & > :global(*) {
margin-top: var(--offset-top); position: sticky;
margin-bottom: calc(-1 * var(--offset-top) + 8px); top: var(--card-margin);
padding-top: calc(var(--index) * var(--card-offset));
@include bp (sm) { padding-bottom: var(--card-margin);
margin-bottom: calc(-1 * var(--offset-top) * var(--scale) + 36px); margin-bottom: calc(-1 * var(--index) * var(--card-offset));
} transform-origin: center top;
} }
} }
@@ -126,7 +129,6 @@
justify-content: center; justify-content: center;
align-items: center; align-items: center;
min-height: min(45vw, 720px); min-height: min(45vw, 720px);
margin-top: 20px;
padding: 56px 32px; padding: 56px 32px;
border-radius: 12px; border-radius: 12px;
background: $color-primary-darker; background: $color-primary-darker;