effect-coverflow.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. import createShadow from '../../shared/create-shadow.js';
  2. import effectInit from '../../shared/effect-init.js';
  3. import effectTarget from '../../shared/effect-target.js';
  4. import { getSlideTransformEl } from '../../shared/utils.js';
  5. export default function EffectCoverflow({
  6. swiper,
  7. extendParams,
  8. on
  9. }) {
  10. extendParams({
  11. coverflowEffect: {
  12. rotate: 50,
  13. stretch: 0,
  14. depth: 100,
  15. scale: 1,
  16. modifier: 1,
  17. slideShadows: true
  18. }
  19. });
  20. const setTranslate = () => {
  21. const {
  22. width: swiperWidth,
  23. height: swiperHeight,
  24. slides,
  25. slidesSizesGrid
  26. } = swiper;
  27. const params = swiper.params.coverflowEffect;
  28. const isHorizontal = swiper.isHorizontal();
  29. const transform = swiper.translate;
  30. const center = isHorizontal ? -transform + swiperWidth / 2 : -transform + swiperHeight / 2;
  31. const rotate = isHorizontal ? params.rotate : -params.rotate;
  32. const translate = params.depth;
  33. // Each slide offset from center
  34. for (let i = 0, length = slides.length; i < length; i += 1) {
  35. const slideEl = slides[i];
  36. const slideSize = slidesSizesGrid[i];
  37. const slideOffset = slideEl.swiperSlideOffset;
  38. const centerOffset = (center - slideOffset - slideSize / 2) / slideSize;
  39. const offsetMultiplier = typeof params.modifier === 'function' ? params.modifier(centerOffset) : centerOffset * params.modifier;
  40. let rotateY = isHorizontal ? rotate * offsetMultiplier : 0;
  41. let rotateX = isHorizontal ? 0 : rotate * offsetMultiplier;
  42. // var rotateZ = 0
  43. let translateZ = -translate * Math.abs(offsetMultiplier);
  44. let stretch = params.stretch;
  45. // Allow percentage to make a relative stretch for responsive sliders
  46. if (typeof stretch === 'string' && stretch.indexOf('%') !== -1) {
  47. stretch = parseFloat(params.stretch) / 100 * slideSize;
  48. }
  49. let translateY = isHorizontal ? 0 : stretch * offsetMultiplier;
  50. let translateX = isHorizontal ? stretch * offsetMultiplier : 0;
  51. let scale = 1 - (1 - params.scale) * Math.abs(offsetMultiplier);
  52. // Fix for ultra small values
  53. if (Math.abs(translateX) < 0.001) translateX = 0;
  54. if (Math.abs(translateY) < 0.001) translateY = 0;
  55. if (Math.abs(translateZ) < 0.001) translateZ = 0;
  56. if (Math.abs(rotateY) < 0.001) rotateY = 0;
  57. if (Math.abs(rotateX) < 0.001) rotateX = 0;
  58. if (Math.abs(scale) < 0.001) scale = 0;
  59. const slideTransform = `translate3d(${translateX}px,${translateY}px,${translateZ}px) rotateX(${rotateX}deg) rotateY(${rotateY}deg) scale(${scale})`;
  60. const targetEl = effectTarget(params, slideEl);
  61. targetEl.style.transform = slideTransform;
  62. slideEl.style.zIndex = -Math.abs(Math.round(offsetMultiplier)) + 1;
  63. if (params.slideShadows) {
  64. // Set shadows
  65. let shadowBeforeEl = isHorizontal ? slideEl.querySelector('.swiper-slide-shadow-left') : slideEl.querySelector('.swiper-slide-shadow-top');
  66. let shadowAfterEl = isHorizontal ? slideEl.querySelector('.swiper-slide-shadow-right') : slideEl.querySelector('.swiper-slide-shadow-bottom');
  67. if (!shadowBeforeEl) {
  68. shadowBeforeEl = createShadow(params, slideEl, isHorizontal ? 'left' : 'top');
  69. }
  70. if (!shadowAfterEl) {
  71. shadowAfterEl = createShadow(params, slideEl, isHorizontal ? 'right' : 'bottom');
  72. }
  73. if (shadowBeforeEl) shadowBeforeEl.style.opacity = offsetMultiplier > 0 ? offsetMultiplier : 0;
  74. if (shadowAfterEl) shadowAfterEl.style.opacity = -offsetMultiplier > 0 ? -offsetMultiplier : 0;
  75. }
  76. }
  77. };
  78. const setTransition = duration => {
  79. const transformElements = swiper.slides.map(slideEl => getSlideTransformEl(slideEl));
  80. transformElements.forEach(el => {
  81. el.style.transitionDuration = `${duration}ms`;
  82. el.querySelectorAll('.swiper-slide-shadow-top, .swiper-slide-shadow-right, .swiper-slide-shadow-bottom, .swiper-slide-shadow-left').forEach(shadowEl => {
  83. shadowEl.style.transitionDuration = `${duration}ms`;
  84. });
  85. });
  86. };
  87. effectInit({
  88. effect: 'coverflow',
  89. swiper,
  90. on,
  91. setTranslate,
  92. setTransition,
  93. perspective: () => true,
  94. overwriteParams: () => ({
  95. watchSlidesProgress: true
  96. })
  97. });
  98. }