<section>
<div class="product-slider">
<div class="slider-box">
<div class="slider">
<div class="slide">
<div class="image">
<div class="main_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_1.jpg" />
</div>
<div class="dop_imgs">
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_2.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_3.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_4.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_5.jpg" />
</div>
</div>
</div>
</div>
<div class="slide">
<div class="image">
<div class="main_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_1.jpg" />
</div>
<div class="dop_imgs">
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_2.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_3.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_4.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_5.jpg" />
</div>
</div>
</div>
</div>
<div class="slide">
<div class="image">
<div class="main_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_1.jpg" />
</div>
<div class="dop_imgs">
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_2.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_3.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_4.jpg" />
</div>
<div class="dop_img">
<img src="https://murom-mebel-spb.ru/sites/default/files/styles/full_product/public/products/krovat_bali_layt_5.jpg" />
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</section>
section {
display: flex;
align-items: center;
justify-content: center;
max-width: 360px;
border: 1px solid gray;
}
.product-slider {
display: flex;
width: 100%
}
.slider-box {
display: flex;
max-width: 360px;
height: 300px;
overflow: hidden;
}
.slider {
display: flex;
}
.slide {
max-width: 360px;
height: 270px;
border: 1px solid gray;
}
.slide img {
width: 360px;
height: auto;
}
.image {
display: flex;
overflow-x: scroll;
scroll-snap-type: x mandatory;
}
.dop_imgs {
display: flex;
}
.image > div {
scroll-snap-align: start;
}
class Slider {
constructor(slider) {
this.currentImg = 0;
this.speed = 500;
this.swipeStatus = this.swipeStatus.bind(this);
this.keyboardHandler = this.keyboardHandler.bind(this);
this.scrollImages = this.scrollImages.bind(this);
this.previousImage = this.previousImage.bind(this);
this.nextImage = this.nextImage.bind(this);
this.sliderWrapper = $(slider);
this.slider = this.sliderWrapper.find('.slider');
this.sliderBox = this.sliderWrapper.find('.slider-box');
this.sliderWrapper
.find('.slider-box')
.attr("tabIndex", 0)
.on("blur focus click", this.keyboardHandler)
.swipe({
triggerOnTouchEnd: true,
triggerOnTouchLeave: true,
swipeStatus: this.swipeStatus,
allowPageScroll: "vertical",
threshold: 75
});
this.maxImages = this.slider.find(">*").length;
this.visibleImages = this.sliderBox.width() / this.slider.find(".slide").width();
this.imagesOut = ( this.maxImages - this.sliderBox.width() / ( this.slider.find(".slide").width() ) );
if ( 0 < this.imagesOut ) {
this.sliderBox.append('<div class="slider-btn slider-previous-btn"></div><div class="slider-btn slider-next-btn"></div>');
}
this.prevBtn = this.sliderWrapper.find(".slider-previous-btn");
this.nextBtn = this.sliderWrapper.find(".slider-next-btn");
this.prevBtn.on("click", this.previousImage);
this.nextBtn.on("click", this.nextImage);
}
keyboardHandler(event) {
if (event.type === "focus") {
document.addEventListener("keydown", this.keyboardHandler);
this.constructor.activeInstance = this;
} else if (event.type === "blur") {
document.removeEventListener("keydown", this.keyboardHandler);
this.constructor.activeInstance = null;
} else if (event.type === "keydown") {
if (event.keyCode === 37) this.previousImage();
if (event.keyCode === 39) this.nextImage();
} else {
this.sliderBox.focus();
}
return true;
}
swipeStatus(event, phase, direction, distance) {
//If we are moving before swipe, and we are going L or R in X mode, or U or D in Y mode then drag.
if (phase == "move" && (direction == "left" || direction == "right")) {
var duration = 0;
if ( this.slider.hasClass( 'swiping' ) === false ) {
this.slider.addClass( 'swiping' );
}
if (direction == "left") {
this.scrollImages(
this.currentImg + 0.0025 * distance, duration
);
} else if (direction == "right") {
this.scrollImages(
this.currentImg - 0.0025 * distance, duration
);
}
} else if (phase == "cancel") {
this.scrollImages(this.currentImg, this.speed);
} else if (phase == "end") {
if (direction == "right") {
this.previousImage();
} else if (direction == "left") {
this.nextImage();
}
}
}
scrollImages(distance, duration) {
this.slider.css("transition-duration", (duration / 1000).toFixed(1) + "s");
this.slider.css("transform", "translate(" + (100 * -distance / this.maxImages) + "%,0)");
$('.slider-box a').on('click', function (ea) {
if( $('.slider').hasClass('swiping') ) {
ea.preventDefault();
} else {
console.log('clicked');
}
});
}
previousImage() {
if (this.currentImg != '0') {
this.currentImg = Math.max(this.currentImg - 1, 0);
this.scrollImages(this.currentImg, this.speed);
if ( this.slider.hasClass( 'swiping' ) ) {
setTimeout(function() {
$('.slider.swiping').removeClass('swiping');
}, 100 );
}
} else {
this.currentImg = this.maxImages - 1;
this.scrollImages(this.currentImg, this.speed);
if ( this.slider.hasClass( 'swiping' ) ) {
setTimeout(function() {
$('.slider.swiping').removeClass('swiping');
}, 100 );
}
}
}
nextImage() {
if (this.currentImg != this.maxImages - 1) {
this.currentImg = Math.min(this.currentImg + 1, this.maxImages - 1);
this.scrollImages(this.currentImg, this.speed);
if ( this.slider.hasClass( 'swiping' ) ) {
setTimeout(function() {
$('.slider.swiping').removeClass('swiping');
}, 100 );
}
} else {
this.currentImg = 0;
this.scrollImages(this.currentImg, this.speed);
if ( this.slider.hasClass( 'swiping' ) ) {
setTimeout(function() {
$('.slider.swiping').removeClass('swiping');
}, 100 );
}
}
}
}
$(() => {
for (const element of $(".product-slider")) {
new Slider(element);
}
});
This Pen doesn't use any external CSS resources.