Search code examples
javascriptcssscrollmagic

Parallax content in a horizontal ScrollMagic container per slide is stuttered


I have a pinned slider that goes horizontally when you get to it by updating the X value of the inner wrapper. This part works great.

However, for each individual slide, I would then like to have a parallax effect for the text (to go slower as you scroll through - relative to the current slide).

Here's a small (simplified) test-case I set up: https://codepen.io/michaelpumo/pen/EJgWgd

Unfortunately though, for some reason, the text seems to stagger and the animation is not smooth. This could be with me misunderstanding ScrollMagic's API (it's new to me).

I have 2 controllers because the only way I could get the "parallax" part was to set the controller to vertical: false for the second one. Perhaps it's something to do with this?

Help is much appreciated!

JavaScript

const ease = window.Power4.easeInOut
const el = document.querySelector('#el')
const wrapper = document.querySelector('#wrapper')
const slides = el.querySelectorAll('.El__slide')
const amount = slides.length
const controller = new window.ScrollMagic.Controller()
const horizontalMovement = new window.TimelineMax()

const controller2 = new window.ScrollMagic.Controller({
  vertical: false
})

horizontalMovement
  .add([
    window.TweenMax.to(wrapper, 1, {
      x: `-${(100 / amount) * (amount - 1)}%`
    })
  ])

new window.ScrollMagic.Scene({
    triggerElement: el,
    triggerHook: 'onLeave',
    duration: `${amount * 100}%`
  })
  .setPin(el)
  .setTween(horizontalMovement)
  .addTo(controller)

slides.forEach((item, index) => {
  const title = item.querySelector('h1')
  const subtitle = item.querySelector('h2')
  const tween = new window.TimelineMax()

  tween
    .fromTo(title, 1, { x: 0 }, { x: 500 }, 0)
    .fromTo(subtitle, 1, { x: 600 }, { x: 500 }, 0)

  new window.ScrollMagic.Scene({
      triggerElement: item,
      triggerHook: 1,
      duration: '100%'
    })
    .setTween(tween)
    .addTo(controller2)
})

SCSS

.El {
  width: 100%;
  max-width: 100%;
  overflow: hidden;
  &__wrapper {
    display: flex;
    width: 400vw;
    height: 100vh;
  }
  &__slide {
    display: flex;
    align-items: center;
    width: 100vw;
    padding: 50px;
    &:nth-child(1) {
      background-color: salmon;
    }
    &:nth-child(2) {
      background-color: blue;
    }
    &:nth-child(3) {
      background-color: orange;
    }
    &:nth-child(4) {
      background-color: tomato;
    }
  }
  &__content {
    width: 100%;
  }
  &__title,
  &__subtitle {
    position: relative;
  }
}

HTML

<div id="el" class="El">
  <div id="wrapper" class="El__wrapper">
    <div class="El__slide">
      <div class="El__content">
        <h1 class="El__title">Title A</h1>
        <h2 class="El__subtitle">Subtitle A</h2>
      </div>
    </div>
    <div class="El__slide">
      <div class="El__content">
        <h1 class="El__title">Title B</h1>
        <h2 class="El__subtitle">Subtitle B</h2>
      </div>
    </div>
    <div class="El__slide">
      <div class="El__content">
        <h1 class="El__title">Title C</h1>
        <h2 class="El__subtitle">Subtitle C</h2>
      </div>
    </div>
    <div class="El__slide">
      <div class="El__content">
        <h1 class="El__title">Title D</h1>
        <h2 class="El__subtitle">Subtitle D</h2>
      </div>
    </div>
  </div>
</div>

Solution

  • To get the desired effect using ScrollMagic methods you have to set refreshInterval: 1 to the controller2.

    Documentantion: http://scrollmagic.io/docs/ScrollMagic.Controller.html#constructor

    Example: https://codepen.io/biomagic/pen/dLgzJO

    const ease = window.Power4.easeInOut
    const el = document.querySelector('#el')
    const wrapper = document.querySelector('#wrapper')
    const slides = el.querySelectorAll('.El__slide')
    const amount = slides.length
    const controller = new window.ScrollMagic.Controller()
    const horizontalMovement = new window.TimelineMax()
    
    const controller2 = new window.ScrollMagic.Controller({
      vertical: false,
      refreshInterval: 1
    })
    
    horizontalMovement
      .add([
        window.TweenMax.to(wrapper, 1, { x: `-${(100 / amount) * (amount - 1)}%` })
      ])
    
    new window.ScrollMagic.Scene({
      triggerElement: el,
      triggerHook: 'onLeave',
      duration: `${amount * 100}%`
    })
      .setPin(el)
      .setTween(horizontalMovement)
      .addTo(controller)
    
    slides.forEach((item, index) => {
      const title = item.querySelector('h1')
      const subtitle = item.querySelector('h2')
      const tween = new window.TimelineMax()
      tween
        .fromTo(title, 1, { x: 0 }, { x: 500 }, 0)
        .fromTo(subtitle, 1, { x: 600 }, { x: 500 }, 0)
      new window.ScrollMagic.Scene({
        triggerElement: item,
        triggerHook: 1,
        duration: '100%'
      })
      .setTween(tween)
      .addTo(controller2)
    })
    

    Another approach. You can to add tweenChanges: true to your scene.

    Documentation: http://scrollmagic.io/docs/animation.GSAP.html#newScrollMagic.Sceneoptions

    Example: https://codepen.io/biomagic/pen/rbqpMb

    const ease = window.Power4.easeInOut
    const el = document.querySelector('#el')
    const wrapper = document.querySelector('#wrapper')
    const slides = el.querySelectorAll('.El__slide')
    const amount = slides.length
    const controller = new window.ScrollMagic.Controller()
    const horizontalMovement = new window.TimelineMax()
    
    const controller2 = new window.ScrollMagic.Controller({
      vertical: false
    })
    
    horizontalMovement
      .add([
        window.TweenMax.to(wrapper, 1, { x: `-${(100 / amount) * (amount - 1)}%` })
      ])
    
    new window.ScrollMagic.Scene({
      triggerElement: el,
      triggerHook: 'onLeave',
      duration: `${amount * 100}%`
    })
      .setPin(el)
      .setTween(horizontalMovement)
      .addTo(controller)
    
    slides.forEach((item, index) => {
      const title = item.querySelector('h1')
      const subtitle = item.querySelector('h2')
      const tween = new window.TimelineMax()
      tween
        .fromTo(title, 1, { x: 0 }, { x: 500 }, 0)
        .fromTo(subtitle, 1, { x: 600 }, { x: 500 }, 0)
      new window.ScrollMagic.Scene({
        triggerElement: item,
        triggerHook: 1,
        duration: '100%',
        tweenChanges: true
      })
      .setTween(tween)
      .addTo(controller2)
    })