SliderSiblingOnly.js 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. import React, {useRef, useEffect, useState} from 'react'
  2. import SliderImage from './SliderImage'
  3. export default ({images = [], imgWidth=377, className, onChange, onClick, ref, siblingOnly, current:propCurrent=0, ...props}) => {
  4. const containerRef = useRef()
  5. const autoscroll = useRef(false)
  6. const afterAutoscroll = useRef(false)
  7. const inTouch = useRef(false)
  8. const timeout = useRef(false)
  9. const afterTimeout = useRef(false)
  10. const imgs = useRef([])
  11. const currentRef = useRef(propCurrent)
  12. const diffRef = useRef(0)
  13. const getCurrentImages = (current) => {
  14. return [
  15. images[(current + images.length -1) % images.length],
  16. images[current],
  17. images[(current + 1) % images.length]
  18. ]
  19. }
  20. let currentImages = getCurrentImages(currentRef.current)
  21. useEffect(() => {
  22. if (containerRef.current)
  23. containerRef.current.scrollTo({left: imgWidth, behavior: 'auto'})
  24. }, [containerRef.current])
  25. const onTouchEnd = () => {
  26. if (!containerRef.current)
  27. setTimeout(onTouchEnd, 20)
  28. // console.log('TOUCH END', inTouch.current, autoscroll.current, afterAutoscroll.current,
  29. // containerRef.current.scrollLeft, containerRef.current.scrollWidth)
  30. if (inTouch.current){
  31. if (timeout.current) clearInterval(timeout.current)
  32. timeout.current = setTimeout(onTouchEnd, 200)
  33. return;
  34. }
  35. const {scrollLeft, scrollWidth} = containerRef.current;
  36. const viewPortWidth = containerRef.current.getBoundingClientRect().width
  37. console.log(scrollLeft, scrollWidth)
  38. autoscroll.current = true
  39. if (scrollLeft < imgWidth/2) {
  40. console.log('left')
  41. containerRef.current.scrollTo({left: 0, behavior: 'smooth'})
  42. diffRef.current = -1
  43. onScroll()
  44. }
  45. else if (scrollLeft >= scrollWidth - viewPortWidth*1.5) {
  46. console.log('right', (scrollLeft - imgWidth * 2))
  47. const toRight = imgWidth * 2
  48. diffRef.current = +1
  49. containerRef.current.scrollTo({left: toRight, behavior: 'smooth'})
  50. onScroll()
  51. }
  52. else {
  53. console.log('center', containerRef.current.scrollWidth)
  54. diffRef.current = 0
  55. const toCurrent = imgWidth
  56. if (toCurrent === scrollLeft){
  57. autoscroll.current = false
  58. }
  59. else {
  60. containerRef.current.scrollTo({left: toCurrent, behavior: 'smooth'})
  61. }
  62. }
  63. }
  64. const onScroll = () => {
  65. if (autoscroll.current){
  66. if (afterTimeout.current)
  67. clearInterval(afterTimeout.current)
  68. afterTimeout.current = setTimeout(() => {
  69. const {scrollLeft, scrollWidth} = containerRef.current;
  70. const {current} = currentRef
  71. console.log('AFTER', scrollLeft, scrollWidth)
  72. const newCurrent = ((currentRef.current + diffRef.current +images.length) % images.length)
  73. let currentImages = getCurrentImages(newCurrent)
  74. console.log('move from', diffRef.current, current, newCurrent, currentImages)
  75. if (newCurrent !== current){
  76. imgs.current[0].src = currentImages[0]
  77. imgs.current[1].src = currentImages[1]
  78. imgs.current[2].src = currentImages[2]
  79. containerRef.current.scrollTo({left: imgWidth, behavior: 'auto'})
  80. onChange(newCurrent)
  81. //afterAutoscroll.current = true
  82. }
  83. currentRef.current = newCurrent
  84. autoscroll.current = false
  85. diffRef.current = 0
  86. console.log(diffRef.current, newCurrent)
  87. }, 200)
  88. return;
  89. }
  90. if (timeout.current) clearInterval(timeout.current)
  91. timeout.current = setTimeout(onTouchEnd, 200)
  92. //e.preventDefault();
  93. }
  94. //useEffect(scrollToCurrent, [current])
  95. const isTouchDevice = () => (navigator.maxTouchPoints || 'ontouchstart' in document.documentElement);
  96. const onImgClick = (e, i) => {
  97. const touchDevice = isTouchDevice()
  98. if (touchDevice && typeof onClick === 'function'){
  99. onClick(i)
  100. }
  101. else {
  102. const viewPortWidth = containerRef.current.getBoundingClientRect().width
  103. const clickX = e.clientX -e.target.getBoundingClientRect().x
  104. const {scrollLeft} = containerRef.current;
  105. if (clickX > viewPortWidth * 0.75) {
  106. containerRef.current.scrollTo({left: scrollLeft + imgWidth*0.8, behavior: 'auto'})
  107. onTouchEnd()
  108. }
  109. else if (clickX < viewPortWidth * 0.25) {
  110. containerRef.current.scrollTo({left: scrollLeft - imgWidth*0.8, behavior: 'auto'})
  111. onTouchEnd()
  112. }
  113. else if (typeof onClick === 'function'){
  114. onClick(i)
  115. }
  116. }
  117. }
  118. const imgStyle = {
  119. maxWidth: imgWidth + 'px',
  120. minWidth: imgWidth + 'px',
  121. width: imgWidth + 'px',
  122. boxSizing: 'border-box'
  123. }
  124. return (
  125. <div className={className}
  126. style={{cssText: `overflow: auto; overflow-y: hidden; -ms-overflow-style: none; scrollbar-width: none; overscroll-behavior-x: none;`}}
  127. onScroll={onScroll} //TODO: do it only on desktop, on touch devices use onTouchEnd
  128. ref={containerRef}
  129. {...isTouchDevice() ? {
  130. onTouchStart(){
  131. inTouch.current = true
  132. autoscroll.current = false
  133. },
  134. onTouchEnd(){
  135. inTouch.current = false
  136. }
  137. }: {}}
  138. >
  139. <div style={{minWidth: imgWidth*3 + 'px'}}>
  140. <img src={currentImages[0]}
  141. style={imgStyle}
  142. key={`image_${(currentRef.current -1 +images.length) % images.length}`}
  143. alt={`image ${(currentRef.current -1 +images.length) % images.length}`}
  144. ref={img => imgs.current[0] = img}
  145. />
  146. <img src={currentImages[1]}
  147. style={imgStyle}
  148. key={`image_${currentRef.current}`}
  149. alt={`image ${currentRef.current}`}
  150. ref={img => imgs.current[1] = img}
  151. onClick={e => onImgClick(e, currentRef.current)}
  152. />
  153. <img src={currentImages[2]}
  154. style={imgStyle}
  155. key={`image_${(currentRef.current +1 +images.length) % images.length}`}
  156. alt={`image ${(currentRef.current +1 +images.length) % images.length}`}
  157. ref={img => imgs.current[2] = img}
  158. />
  159. </div>
  160. </div>
  161. )
  162. }