swiper.js 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. function _extends() { _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; return _extends.apply(this, arguments); }
  2. import React, { useRef, useState, useEffect, forwardRef } from 'react';
  3. import { getParams } from './get-params.js';
  4. import { initSwiper, mountSwiper } from './init-swiper.js';
  5. import { needsScrollbar, needsNavigation, needsPagination, uniqueClasses, extend } from './utils.js';
  6. import { renderLoop, calcLoopedSlides } from './loop.js';
  7. import { getChangedParams } from './get-changed-params.js';
  8. import { getChildren } from './get-children.js';
  9. import { updateSwiper } from './update-swiper.js';
  10. import { renderVirtual, updateOnVirtualData } from './virtual.js';
  11. import { useIsomorphicLayoutEffect } from './use-isomorphic-layout-effect.js';
  12. const Swiper = /*#__PURE__*/forwardRef(({
  13. className,
  14. tag: Tag = 'div',
  15. wrapperTag: WrapperTag = 'div',
  16. children,
  17. onSwiper,
  18. ...rest
  19. } = {}, externalElRef) => {
  20. let eventsAssigned = false;
  21. const [containerClasses, setContainerClasses] = useState('swiper');
  22. const [virtualData, setVirtualData] = useState(null);
  23. const [breakpointChanged, setBreakpointChanged] = useState(false);
  24. const initializedRef = useRef(false);
  25. const swiperElRef = useRef(null);
  26. const swiperRef = useRef(null);
  27. const oldPassedParamsRef = useRef(null);
  28. const oldSlides = useRef(null);
  29. const nextElRef = useRef(null);
  30. const prevElRef = useRef(null);
  31. const paginationElRef = useRef(null);
  32. const scrollbarElRef = useRef(null);
  33. const {
  34. params: swiperParams,
  35. passedParams,
  36. rest: restProps,
  37. events
  38. } = getParams(rest);
  39. const {
  40. slides,
  41. slots
  42. } = getChildren(children);
  43. const onBeforeBreakpoint = () => {
  44. setBreakpointChanged(!breakpointChanged);
  45. };
  46. Object.assign(swiperParams.on, {
  47. _containerClasses(swiper, classes) {
  48. setContainerClasses(classes);
  49. }
  50. });
  51. if (!swiperElRef.current) {
  52. // init swiper
  53. Object.assign(swiperParams.on, events);
  54. eventsAssigned = true;
  55. swiperRef.current = initSwiper(swiperParams);
  56. swiperRef.current.loopCreate = () => {};
  57. swiperRef.current.loopDestroy = () => {};
  58. if (swiperParams.loop) {
  59. swiperRef.current.loopedSlides = calcLoopedSlides(slides, swiperParams);
  60. }
  61. if (swiperRef.current.virtual && swiperRef.current.params.virtual.enabled) {
  62. swiperRef.current.virtual.slides = slides;
  63. const extendWith = {
  64. cache: false,
  65. slides,
  66. renderExternal: setVirtualData,
  67. renderExternalUpdate: false
  68. };
  69. extend(swiperRef.current.params.virtual, extendWith);
  70. extend(swiperRef.current.originalParams.virtual, extendWith);
  71. }
  72. } // Listen for breakpoints change
  73. if (swiperRef.current) {
  74. swiperRef.current.on('_beforeBreakpoint', onBeforeBreakpoint);
  75. }
  76. const attachEvents = () => {
  77. if (eventsAssigned || !events || !swiperRef.current) return;
  78. Object.keys(events).forEach(eventName => {
  79. swiperRef.current.on(eventName, events[eventName]);
  80. });
  81. };
  82. const detachEvents = () => {
  83. if (!events || !swiperRef.current) return;
  84. Object.keys(events).forEach(eventName => {
  85. swiperRef.current.off(eventName, events[eventName]);
  86. });
  87. };
  88. useEffect(() => {
  89. return () => {
  90. if (swiperRef.current) swiperRef.current.off('_beforeBreakpoint', onBeforeBreakpoint);
  91. };
  92. }); // set initialized flag
  93. useEffect(() => {
  94. if (!initializedRef.current && swiperRef.current) {
  95. swiperRef.current.emitSlidesClasses();
  96. initializedRef.current = true;
  97. }
  98. }); // mount swiper
  99. useIsomorphicLayoutEffect(() => {
  100. if (externalElRef) {
  101. externalElRef.current = swiperElRef.current;
  102. }
  103. if (!swiperElRef.current) return;
  104. mountSwiper({
  105. el: swiperElRef.current,
  106. nextEl: nextElRef.current,
  107. prevEl: prevElRef.current,
  108. paginationEl: paginationElRef.current,
  109. scrollbarEl: scrollbarElRef.current,
  110. swiper: swiperRef.current
  111. }, swiperParams);
  112. if (onSwiper) onSwiper(swiperRef.current); // eslint-disable-next-line
  113. return () => {
  114. if (swiperRef.current && !swiperRef.current.destroyed) {
  115. swiperRef.current.destroy(true, false);
  116. }
  117. };
  118. }, []); // watch for params change
  119. useIsomorphicLayoutEffect(() => {
  120. attachEvents();
  121. const changedParams = getChangedParams(passedParams, oldPassedParamsRef.current, slides, oldSlides.current);
  122. oldPassedParamsRef.current = passedParams;
  123. oldSlides.current = slides;
  124. if (changedParams.length && swiperRef.current && !swiperRef.current.destroyed) {
  125. updateSwiper({
  126. swiper: swiperRef.current,
  127. slides,
  128. passedParams,
  129. changedParams,
  130. nextEl: nextElRef.current,
  131. prevEl: prevElRef.current,
  132. scrollbarEl: scrollbarElRef.current,
  133. paginationEl: paginationElRef.current
  134. });
  135. }
  136. return () => {
  137. detachEvents();
  138. };
  139. }); // update on virtual update
  140. useIsomorphicLayoutEffect(() => {
  141. updateOnVirtualData(swiperRef.current);
  142. }, [virtualData]); // bypass swiper instance to slides
  143. function renderSlides() {
  144. if (swiperParams.virtual) {
  145. return renderVirtual(swiperRef.current, slides, virtualData);
  146. }
  147. if (!swiperParams.loop || swiperRef.current && swiperRef.current.destroyed) {
  148. return slides.map(child => {
  149. return /*#__PURE__*/React.cloneElement(child, {
  150. swiper: swiperRef.current
  151. });
  152. });
  153. }
  154. return renderLoop(swiperRef.current, slides, swiperParams);
  155. }
  156. return /*#__PURE__*/React.createElement(Tag, _extends({
  157. ref: swiperElRef,
  158. className: uniqueClasses(`${containerClasses}${className ? ` ${className}` : ''}`)
  159. }, restProps), slots['container-start'], needsNavigation(swiperParams) && /*#__PURE__*/React.createElement(React.Fragment, null, /*#__PURE__*/React.createElement("div", {
  160. ref: prevElRef,
  161. className: "swiper-button-prev"
  162. }), /*#__PURE__*/React.createElement("div", {
  163. ref: nextElRef,
  164. className: "swiper-button-next"
  165. })), needsScrollbar(swiperParams) && /*#__PURE__*/React.createElement("div", {
  166. ref: scrollbarElRef,
  167. className: "swiper-scrollbar"
  168. }), needsPagination(swiperParams) && /*#__PURE__*/React.createElement("div", {
  169. ref: paginationElRef,
  170. className: "swiper-pagination"
  171. }), /*#__PURE__*/React.createElement(WrapperTag, {
  172. className: "swiper-wrapper"
  173. }, slots['wrapper-start'], renderSlides(), slots['wrapper-end']), slots['container-end']);
  174. });
  175. Swiper.displayName = 'Swiper';
  176. export { Swiper };