lightbox-gallery/index.js

327 lines
10 KiB
JavaScript
Raw Normal View History

/* globals GalleryYTPlayer, YT, GalleryVKPlayer, VK */
2023-06-19 20:58:52 +03:00
import getScrollbarWidth from '@advdominion/get-scrollbar-width';
const getYTPlayer = () => {
const script = document.createElement('script');
script.src = 'https://www.youtube.com/player_api';
script.async = true;
document.body.append(script);
};
const getVKPlayer = () => {
const script = document.createElement('script');
script.src = 'https://vk.com/js/api/videoplayer.js';
script.async = true;
document.body.append(script);
};
2023-06-19 20:58:52 +03:00
const playVideo = (slide) => {
const youtube = slide.querySelector('.advdominion-lg__video_youtube');
const vk = slide.querySelector('.advdominion-lg__video_vk');
2023-06-19 20:58:52 +03:00
if (youtube) {
if (slide.YTPlayer) {
slide.YTPlayer.playVideo();
} else if (window.YT) {
if (GalleryYTPlayer.state) {
slide.YTPlayer = new YT.Player(youtube, {
videoId: youtube.dataset.id,
events: {
onReady(event) {
event.target.playVideo();
},
},
});
} else {
GalleryYTPlayer.addListener(() => {
playVideo(slide);
});
}
} else {
getYTPlayer();
GalleryYTPlayer.addListener(() => {
playVideo(slide);
});
}
} else if (vk) {
if (slide.VKPlayer) {
slide.VKPlayer.play();
} else if (window.VK?.VideoPlayer) {
if (GalleryVKPlayer.state) {
slide.VKPlayer = VK.VideoPlayer(vk);
slide.VKPlayer.on('inited', () => {
slide.VKPlayer.play();
});
} else {
GalleryVKPlayer.addListener(() => {
playVideo(slide);
});
}
} else {
getVKPlayer();
GalleryVKPlayer.addListener(() => {
playVideo(slide);
});
}
2023-06-19 20:58:52 +03:00
}
};
const pauseVideo = (slide) => {
const youtube = slide.querySelector('.advdominion-lg__video_youtube');
const vk = slide.querySelector('.advdominion-lg__video_vk');
2023-06-19 20:58:52 +03:00
if (youtube && slide.YTPlayer) {
slide.YTPlayer.pauseVideo();
} else if (vk && slide.VKPlayer) {
slide.VKPlayer.pause();
2023-06-19 20:58:52 +03:00
}
};
const setVideoSize = (gallery) => {
const items = gallery.querySelectorAll('.advdominion-lg__video-wrapper');
items[0].style = `
width: '';
`;
let height = (items[0].getBoundingClientRect().width * 9) / 16;
let width = '';
if (height > items[0].getBoundingClientRect().height) {
height = items[0].getBoundingClientRect().height;
width = `${(height * 16) / 9}px`;
}
for (const item of items) {
item.style = `
height: ${height}px;
width: ${width};
`;
}
};
const hide = async (duration = 500, easing = 'linear') => {
const gallery = document.body.querySelector('.advdominion-lg');
2025-01-21 14:32:52 +03:00
await gallery.animate([{ opacity: 1 }, { opacity: 0 }], {
2023-06-19 20:58:52 +03:00
duration,
easing,
}).finished;
gallery.remove();
document.body.style = `
overflow: '';
margin-right: '';
`;
};
const show = async (duration = 500, easing = 'linear') => {
const gallery = document.body.querySelector('.advdominion-lg');
document.body.style = `
overflow: hidden;
margin-right: ${getScrollbarWidth()}px
`;
2025-01-21 14:32:52 +03:00
await gallery.animate([{ opacity: 0 }, { opacity: 1 }], {
2023-06-19 20:58:52 +03:00
duration,
easing,
}).finished;
};
const init = async (items = [], options = {}, index = 0) => {
2025-01-21 14:32:52 +03:00
const { Swiper, Navigation, Pagination } = await import('swiper');
2023-06-19 20:58:52 +03:00
Swiper.use([Navigation, Pagination]);
const isVideoInGallery = {
yt: false,
vk: false,
};
2023-06-19 20:58:52 +03:00
items = items.map((item) => {
if (/youtube\.com\/watch\?v=/.test(item)) {
isVideoInGallery.yt = true;
2023-06-19 20:58:52 +03:00
return `
<div class="advdominion-lg__item advdominion-lg__item_video swiper-slide">
<div class="advdominion-lg__video-wrapper">
<div
class="advdominion-lg__video advdominion-lg__video_youtube"
data-id="${/v=([\w-]+)/i.exec(item)[1]}">
Загрузка YouTube-плеера...
</div>
</div>
</div>
`;
} else if (/vkvideo\.ru\/video/.test(item)) {
isVideoInGallery.vk = true;
const videoId = /vkvideo.ru\/video(-?\d+)_(\d+)/.exec(item);
if (videoId === null) {
return `
<div class="advdominion-lg__item advdominion-lg__item_video swiper-slide">
<div class="advdominion-lg__video-wrapper">
<div class="advdominion-lg__video advdominion-lg__video_vk">
Ошибка при загрузке VK-плеера
</div>
</div>
</div>
`;
}
return `
<div class="advdominion-lg__item advdominion-lg__item_video swiper-slide">
<div class="advdominion-lg__video-wrapper">
<iframe class="advdominion-lg__video advdominion-lg__video_vk" src="https://vkvideo.ru/video_ext.php?oid=${videoId[1]}&id=${videoId[2]}&hd=4&js_api=1" allow="autoplay; encrypted-media; fullscreen; picture-in-picture; screen-wake-lock;" allowfullscreen></iframe>
</div>
</div>
`;
2023-06-19 20:58:52 +03:00
}
return `
<div class="advdominion-lg__item swiper-slide">
<img class="advdominion-lg__image" src="${item}" alt="" loading="lazy">
</div>
`;
});
if (isVideoInGallery.yt && !window.GalleryYTPlayer) {
2023-06-19 20:58:52 +03:00
window.GalleryYTPlayer = {
state: false,
get init() {
return this.state;
},
set init(val) {
this.state = val;
this.callback.forEach((i) => {
i(val);
});
},
callback: [],
addListener(callback) {
this.callback.push(callback);
},
};
window.onYouTubeIframeAPIReady = () => {
window.GalleryYTPlayer.init = true;
};
}
if (isVideoInGallery.vk && !window.GalleryVKPlayer) {
window.GalleryVKPlayer = {
state: false,
get init() {
return this.state;
},
set init(val) {
this.state = val;
this.callback.forEach((i) => {
i(val);
});
},
callback: [],
addListener(callback) {
this.callback.push(callback);
},
};
const checkInterval = setInterval(() => {
if (window.VK?.VideoPlayer) {
clearInterval(checkInterval);
window.GalleryVKPlayer.init = true;
}
}, 100);
}
2023-06-19 20:58:52 +03:00
document.body.insertAdjacentHTML(
'beforeend',
`
<div class="advdominion-lg">
<button class="advdominion-lg__close" type="button">
<svg class="advdominion-lg__close-icon"><use href="${options.icons.close}" /></svg>
</button>
<div class="advdominion-lg__container swiper-container">
<div class="advdominion-lg__wrapper swiper-wrapper">
${items.join('')}
</div>
2024-03-26 13:13:30 +03:00
${options.pagination || ''}
${options.navigation || ''}
2023-06-19 20:58:52 +03:00
</div>
</div>
2025-01-21 14:32:52 +03:00
`,
2023-06-19 20:58:52 +03:00
);
const gallery = document.body.querySelector('.advdominion-lg');
2024-03-26 13:13:00 +03:00
gallery.querySelector('.advdominion-lg__container').style = `
visibility: hidden;
`;
2023-06-19 20:58:52 +03:00
await show(options?.animation?.show?.duration, options?.animation?.show?.easing);
const swiper = new Swiper(
gallery.querySelector('.advdominion-lg__container'),
2025-01-21 14:32:52 +03:00
Object.assign({ init: false, initialSlide: index }, options.swiper),
2023-06-19 20:58:52 +03:00
);
swiper.on('init', function () {
2024-03-26 13:13:00 +03:00
gallery.querySelector('.advdominion-lg__container').style = `
visibility: '';
`;
2023-06-19 20:58:52 +03:00
if (this.slides[this.activeIndex].classList.contains('advdominion-lg__item_video')) {
playVideo(this.slides[this.activeIndex]);
}
if (isVideoInGallery.yt || isVideoInGallery.vk) {
2023-06-19 20:58:52 +03:00
setVideoSize(gallery);
}
});
swiper.on('slideChange', function () {
if (this.slides[this.previousIndex].classList.contains('advdominion-lg__item_video')) {
pauseVideo(this.slides[this.previousIndex]);
}
if (this.slides[this.activeIndex].classList.contains('advdominion-lg__item_video')) {
playVideo(this.slides[this.activeIndex]);
}
});
swiper.on('resize', function () {
if (isVideoInGallery.yt || isVideoInGallery.vk) {
2023-06-19 20:58:52 +03:00
setVideoSize(gallery);
}
});
swiper.init();
gallery.querySelector('.advdominion-lg__close').addEventListener('click', () => {
hide(options?.animation?.hide?.duration, options?.animation?.hide?.easing);
});
document.querySelector('body').addEventListener(
'keydown',
(e) => {
if (e.key === 'Escape') {
hide(options?.animation?.hide?.duration, options?.animation?.hide?.easing);
}
},
{
once: true,
2025-01-21 14:32:52 +03:00
},
2023-06-19 20:58:52 +03:00
);
};
export default (container, options) => {
const items = [];
if (options.single) {
items.push(container.href);
container.addEventListener('click', (event) => {
2023-06-19 20:58:52 +03:00
event.preventDefault();
init(items, options);
2023-06-19 20:58:52 +03:00
});
} else {
for (const [index, element] of [...container.children].entries()) {
items.push(element.href);
element.addEventListener('click', (event) => {
event.preventDefault();
init(items, options, index);
});
}
2023-06-19 20:58:52 +03:00
}
};