Содержание параллакса в горизонтальном контейнере ScrollMagic для каждого слайда заикается

У меня есть закрепленный слайдер, который перемещается горизонтально, когда вы попадаете на него, обновляя значение X внутренней оболочки. Эта часть прекрасно работает.

Однако для каждого отдельного слайда мне бы хотелось иметь эффект параллакса для текста (чтобы двигаться медленнее при прокрутке - относительно текущего слайда).

Вот небольшой (упрощенный) тестовый пример, который я настроил: https://codepen.io/michaelpumo/pen/EJgWgd

К сожалению, хотя, по некоторым причинам, текст кажется шатким, а анимация не плавной. Это может быть причиной недопонимания API ScrollMagic (для меня это новость).

У меня есть 2 контроллера, потому что единственный способ получить часть "параллакса" состояла в том, чтобы установить контроллер в vertical: false для второго. Возможно, это как-то связано с этим?

Помощь очень ценится!

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>

Ответ 1

Чтобы получить желаемый эффект, используя методы ScrollMagic, вы должны установить refreshInterval: 1 для controller2.

Документация: http://scrollmagic.io/docs/ScrollMagic.Controller.html#constructor

Пример: 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)
})

Другой подход. Вы можете добавить tweenChanges: true к вашей сцене.

Документация: http://scrollmagic.io/docs/animation.GSAP.html#newScrollMagic.Sceneoptions

Пример: 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)
})

Ответ 2

Я не уверен, что понял суть вашего вопроса. Потому что мой родной язык другой. Но я хочу помочь тебе. Для плавного перемещения текста это поможет вам:

.El__title, .El__subtitle {
    position: relative;
    transition-property: all;
    transition-duration: 900ms;
    transition-timing-function: ease;
    transition-delay: 0s;
}

Если я вас неправильно понял, то покажите мне пример того, что вам нужно в конце. Тогда я постараюсь помочь.

Ответ 3

Я думаю, что ваш CSS просто нужно пару настроек. Во-первых, удалите добавленные браузером отступы в верхней части CSS, используя

*{margin:0px; 
  padding:0px;}

Похоже, основной проблемой является отступ в 50 пикселей. Я удалил его из _slide и добавил к _title в css, и отрегулировал ширину содержимого от width:100% до max-width:100%;

Еще одна вещь, которую я хотел бы предложить, это то, что вы можете добавить навигационные точки в верхний угол первого слайда (для плавного "перехода"/прокрутки между слайдами.) Возможно, подумайте об этом.

Вот кодекс (раздвоенный от твоего). Прокомментируйте, если проблема была неправильно истолкована, или вы чувствуете, что проблема по-прежнему существует.

Надеюсь это поможет

Ответ 4

Я проверил ваш код и выглядит все хорошо. Я могу порекомендовать вам, если вы хотите, чтобы плавное поведение добавляло переход к вашему CSS. На этом пути у вас будет плавный переход.

Просто добавьте эти строки:

&__title,
&__subtitle {
  position: relative;
}

/* Add the next lines */
&__title {
  transition: all .1s linear;
}

Вы можете изменить эффекты и время. Я отправляю вам ссылку на документы: Переход DOCS