Files
web-temp/layers/components/blocks/slide/Thumbnail.vue
2025-10-20 17:24:02 +09:00

183 lines
4.8 KiB
Vue

<script setup lang="ts">
import { Splide, SplideSlide } from '@splidejs/vue-splide'
import type { Splide as SplideType, Options } from '@splidejs/splide'
import type { PageDataTemplateComponentSet } from '#layers/types/api/pageData'
interface Props {
slideData: PageDataTemplateComponentSet[]
variant?: 'default' | 'media'
drag?: boolean
}
const props = withDefaults(defineProps<Props>(), {
variant: 'default',
drag: true,
})
let mainInst: SplideType | null = null
let thumbsInst: SplideType | null = null
defineExpose({
mainInst: computed(() => mainInst),
thumbsInst: computed(() => thumbsInst),
})
const breakpoints = useResponsiveBreakpointsReliable()
const mainRef = ref<InstanceType<typeof Splide> | null>(null)
const thumbsRef = ref<InstanceType<typeof Splide> | null>(null)
const mainOptions = computed<Options>(() => ({
type: 'fade',
rewind: true,
perPage: 1,
perMove: 1,
speed: 600,
arrows: false,
pagination: false,
drag: props.drag,
}))
const thumbOptions = computed<Options>(() => ({
type: 'slide',
rewind: true,
// focus: 'center',
autoWidth: true,
perMove: 1,
arrows: true,
pagination: false,
isNavigation: true,
updateOnMove: true,
classes: {
arrows: 'splide-arrows',
arrow: 'splide-arrow',
prev: 'arrow-prev',
next: 'arrow-next',
},
}))
const getThumbnailSrc = (item: PageDataTemplateComponentSet) => {
if (props.variant === 'media') {
const mediaComponent = getComponentGroup(item, 'media')
return mediaComponent ? getMediaImgSrc(mediaComponent) : ''
}
const pagenaviThumbnailComponent = getComponentGroup(
item,
'pagenaviThumbnail'
)
const pagenaviThumbnailSrc = getResponsiveSrc(
pagenaviThumbnailComponent?.res_path
)
return breakpoints.value.isMobile
? pagenaviThumbnailSrc?.mobileSrc
: pagenaviThumbnailSrc?.pcSrc || ''
}
onMounted(() => {
mainInst = mainRef.value?.splide ?? null
thumbsInst = thumbsRef.value?.splide ?? null
if (mainInst && thumbsInst) {
mainInst.sync(thumbsInst)
}
})
onBeforeUnmount(() => {
mainInst?.destroy?.()
thumbsInst?.destroy?.()
})
</script>
<template>
<div :class="['thumbnail-carousel', $attrs?.class, `thumbnail-${variant}`]">
<!-- 메인 슬라이드 -->
<Splide ref="mainRef" :options="mainOptions" class="main-splide">
<slot />
</Splide>
<!-- 썸네일 슬라이드 -->
<Splide
v-if="props.slideData.length > 0"
ref="thumbsRef"
:options="thumbOptions"
class="thumbnail-splide"
>
<SplideSlide
v-for="(item, index) in props.slideData"
:key="item.set_order || index"
class="thumbnail-slide"
>
<img
:src="getThumbnailSrc(item)"
alt="thumbnail image"
class="slide-image"
/>
</SplideSlide>
</Splide>
</div>
</template>
<style scoped>
.thumbnail-carousel {
}
.thumbnail-carousel:deep(img) {
@apply w-full h-full object-cover;
}
.thumbnail-splide {
@apply overflow-hidden flex justify-center;
}
.thumbnail-slide {
@apply overflow-hidden relative mr-[12px] !border-none rounded-[4px] md:mr-[16px]
after:content-[''] after:absolute after:top-0 after:left-0 after:w-full after:h-full
after:border after:border-white/60 after:rounded-[4px];
}
.thumbnail-slide:hover,
.thumbnail-slide.is-active {
@apply after:border-[var(--primary)];
}
/* 기본 버전 스타일 */
.thumbnail-carousel.thumbnail-default,
.thumbnail-carousel.thumbnail-default .main-splide,
.thumbnail-carousel.thumbnail-default .main-splide:deep(.splide__track) {
@apply h-full;
}
.thumbnail-carousel.thumbnail-default .thumbnail-splide {
@apply absolute bottom-[32px] left-1/2 -translate-x-1/2 max-w-[100%] md:bottom-[48px] md:max-w-[896px] md:px-[72px];
}
.thumbnail-carousel.thumbnail-default:deep(.arrow-prev) {
@apply left-0;
}
.thumbnail-carousel.thumbnail-default:deep(.arrow-next) {
@apply right-0;
}
.thumbnail-carousel.thumbnail-default .thumbnail-slide {
@apply aspect-[1/1] w-[8px] bg-red-500 md:w-[80px] md:bg-transparent
after:hidden md:after:block;
}
.thumbnail-carousel.thumbnail-default .thumbnail-slide.is-active {
@apply bg-blue-500 md:bg-transparent;
}
.thumbnail-carousel.thumbnail-default .thumbnail-slide img {
@apply hidden md:block;
}
/* 미디어 버전 스타일 */
.thumbnail-carousel.thumbnail-media .thumbnail-splide {
@apply w-screen mt-[20px] mx-[-20px] sm:mx-[-40px] md:w-fit md:max-w-[100%] md:mt-[28px] md:mx-auto md:px-[120px];
}
.thumbnail-carousel.thumbnail-media .thumbnail-splide:deep(.splide__track) {
@apply !px-[20px] sm:!px-[40px] md:!px-[0];
}
.thumbnail-carousel.thumbnail-media:deep(.arrow-prev) {
@apply left-[48px];
}
.thumbnail-carousel.thumbnail-media:deep(.arrow-next) {
@apply right-[48px];
}
.thumbnail-carousel.thumbnail-media .thumbnail-slide {
@apply aspect-[16/9] w-[92px] md:w-[128px];
}
</style>