vue3+ts开发一个无缝轮播图插件

2022-02-09 17:05
椰子皮
2163
0
6
vue

基于 Vue3.0 + ts 开发 轻量、简易,支持无缝轮播

  • 基于 Vue3.0 + ts 开发
  • 轻量、简易,支持无缝轮播

 

思路:使用translateX改变横向位置,无缝衔接需要在列表最前面添加一张最后的图片,列表最后面添加第一张图片,滚动到最后一张或者前一张就结束过度动画,并把距离设置成下一张的距离

 

github地址

npm地址

 

觉得不错就点个star吧!

 

效果:

 

具体使用方法看上面github地址

 

新建一个文件:swiper.vue

 

js部分:

import { ref, reactive, onMounted, watch, computed } from 'vue'
interface SwiperItemConfig {
  title?: string,
  url?: string,
  id?: string,
  [x: string]: any,
}
interface Props {
  list: Array<SwiperItemConfig>
  duration?: number,
  interval?: number,
  autoplay?: boolean,
  direction?: string,
  showTitle?: boolean,
  showDots?: boolean,
  showArrow?: boolean,
}
const props = withDefaults(defineProps<Props>(), {
  /**
   * 列表数据
   */
  list: () => [],
  /**
   * 动画时长,毫秒
   */
  duration: 300,
  /**
   * 间隔时间,毫秒
   */
  interval: 3000,
  /**
   * 自动播放
   */
  autoplay: true,
  /**
   * 方向,prev:往右滚动,next:往左滚动
   */
  direction: 'next',
  /**
   * 显示标题
   */
  showTitle: true,
  /**
   * 显示滑动指示点, 标题和指示点二选一
   */
  showDots: false,
  /**
   * 显示箭头
   */
  showArrow: true,
})
const emit = defineEmits(['change', 'click'])
const newSwiperX = ref(-1) // 因为是translateX里面有异步的,所以用这个来替换衔接后的值
const swiperRefs = ref()
const swiperConfig = reactive({
  index: 0,
  arrow: true,
  inter: undefined as any,
  width: 0,
  direction: props.direction
})
const noAnimated = ref(true) // 防止刷新页面的时候有过度效果
const isAnimating = ref(false) // 点击间隔控制
const translateX = computed(() => {
  const { width, index } = swiperConfig
  return -width - (index * width)
})
// 监听索引改变
watch(() => swiperConfig.index, (index: number) => {
  const { width, direction } = swiperConfig
  const len = props.list.length
  const duration = props.duration
  const isNextEnd = (index === 0) && direction === 'next'
  const isPrevEnd = (index === len - 1) && direction === 'prev'
  let nextTimer = undefined
  let prevTimer = undefined
  clearTimeout(nextTimer)
  clearTimeout(prevTimer)
  // 滑动到最后一张判断
  if (isNextEnd) {
    // console.log('最后一张衔接了')
    newSwiperX.value = -(width * len) + (-width) // 因为末尾多添加了一张开头的,所以距离要多加一个个宽度
    isAnimating.value = true // 改变的时候禁止点击
    nextTimer = setTimeout(() => {
      noAnimated.value = true // 也不要有过渡动画
      isAnimating.value = false
      newSwiperX.value = -width // 就把距离替换成第二张的距离, 动画过渡完结束isAnimating
    }, duration)
  }
  // 如果是第一张开始往左, 距离设置为0
  if (isPrevEnd) {
    // console.log('第一张衔接了')
    newSwiperX.value = 0 // 因为开头多添加了一张结尾的,所以距离距离设置为0
    isAnimating.value = true // 改变的时候禁止点击
    prevTimer = setTimeout(() => {
      noAnimated.value = true // 也不要有过渡动画
      isAnimating.value = false
      newSwiperX.value = -width * (len) // 就把距离替换成倒数第二张的距离, 动画过渡完结束isAnimating
    }, duration)
  }
  if (!isNextEnd && !isPrevEnd) {
    newSwiperX.value = -1
    // console.log(newSwiperX.value)
  }
  emit('change', index)
})
// 循环
const loop = (type: string) => {
  swiperConfig.index = type == 'next' ? 0 : props.list.length - 1
}
// 往左
const next = () => {
  swiperConfig.direction = 'next'
  if (isAnimating.value) {
    return
  }
  noAnimated.value = false
  swiperConfig.index++
  if (swiperConfig.index === props.list.length) {
    loop('next')
  }
}
// 往右
const prev = () => {
  swiperConfig.direction = 'prev'
  if (isAnimating.value) {
    return
  }
  noAnimated.value = false
  swiperConfig.index--
  if (swiperConfig.index <= -1) {
    loop('prev')
  }
}
const stop = () => {
  if (props.autoplay) {
    clearInterval(swiperConfig.inter)
  }
}
// 初始化
const start = () => {
  if (props.autoplay) {
    clearInterval(swiperConfig.inter)
    swiperConfig.inter = setInterval(() => {
      props.direction === 'next' ? next() : prev()
    }, props.interval)
  }
}
const onItemClick = ((e: SwiperItemConfig) => {
  emit('click', e)
})
onMounted(() => {
  if (props.list.length) {
    setTimeout(() => {
      swiperConfig.width = swiperRefs.value.clientWidth
      start()
    }, 100)
  }
})

 

template部分:

<template>
  <div
    v-if="list.length"
    ref="swiperRefs"
    class="yzp-swiper-wrap"
    @mouseover="stop"
    @mouseout="start"
  >
    <ul
      :style="{
        transform: `translate3d(${newSwiperX === -1 ? translateX : newSwiperX}px,0,0)`,
        transition: noAnimated ? 'none' : `all ${duration / 1000}s`
      }"
      class="yzp-swiper-list"
    >
      <li v-show="swiperConfig.width" class="yzp-swiper-item">
        <div class="yzp-swiper-link">
          <slot name="swiperItem" :item="list[list.length -1]"></slot>
        </div>
      </li>
      <li
        v-for="(item, index) in list"
        :style="{ 'z-index': -index }"
        :key="index"
        class="yzp-swiper-item"
      >
        <div class="yzp-swiper-link" @click="onItemClick(item)">
          <slot name="swiperItem" :item="item"></slot>
        </div>
      </li>
      <li v-show="swiperConfig.width" class="yzp-swiper-item">
        <div class="yzp-swiper-link">
          <slot name="swiperItem" :item="list[0]"></slot>
        </div>
      </li>
    </ul>
    <!--swiper btn-->
    <div
      v-if="showArrow && !$slots.swiperCustomButton"
      class="yzp-swiper-btn"
    >
      <div :class="{ hasBtnSlot: !!$slots.swiperLeftButton }" class="yzp-swiper-left" @click="prev">
        <slot name="swiperLeftButton"></slot>
      </div>
      <div :class="{ hasBtnSlot: !!$slots.swiperRightButton }" class="yzp-swiper-right" @click="next">
        <slot name="swiperRightButton"></slot>
      </div>
    </div>
    <!--自定义箭头按钮-->
    <slot name="swiperCustomButton"></slot>
    <!--end swiper btn-->

    <!--swiper title-->
    <div v-if="showTitle && !showDots" class="yzp-swiper-text">
      <span class="yzp-swiper-title">{{ list[swiperConfig.index].title || '-' }}</span>
      <span class="yzp-swiper-index">{{ swiperConfig.index + 1 }} / {{ list.length }}</span>
    </div>
    <!--end swiper title-->

    <!--swiper dots-->
    <div v-if="showDots" class="yzp-swiper-dots">
      <ul class="yzp-swiper-dots-list">
        <li
          v-for="(dot, i) in list"
          :key="`dot-${i}`"
          :class="{ active: swiperConfig.index === i }"
          class="yzp-swiper-dots-item"
        ></li>
      </ul>
    </div>
    <!--end swiper dots-->
  </div>
</template>

 

css部分:

.yzp-swiper-wrap .yzp-swiper-btn > div {
  display: flex;
  align-items: center;
  justify-content: center;
  width: 40px;
  height: 40px;
  border-radius: 50%;
  background: rgba(255, 255, 255, 0.5);
  box-shadow: 0 0 10px rgba(0, 0, 0, 0.15);
  position: absolute;
  top: 0;
  bottom: 0;
  margin: auto;
  z-index: 2;
  cursor: pointer;
  transition: all 0.3s;
}
.yzp-swiper-wrap .yzp-swiper-btn > .hasBtnSlot {
  width: auto;
  height: auto;
  background: initial;
  box-shadow: none;
  border-radius: initial;
}
.yzp-swiper-wrap .yzp-swiper-btn > .hasBtnSlot:before,
.yzp-swiper-wrap .yzp-swiper-btn > .hasBtnSlot:after
{
  display: none;
}
.yzp-swiper-wrap .yzp-swiper-btn .yzp-swiper-left {
  left: -55px;
}
.yzp-swiper-wrap .yzp-swiper-btn .yzp-swiper-right {
  right: -55px;
}
.yzp-swiper-wrap:hover .yzp-swiper-btn .yzp-swiper-left {
  left: 15px;
}
.yzp-swiper-wrap:hover .yzp-swiper-btn .yzp-swiper-right {
  right: 15px;
}
.yzp-swiper-wrap .yzp-swiper-btn > div:after,
.yzp-swiper-wrap .yzp-swiper-btn > div:before {
  display: block;
  content: '';
  width: 1px;
  height: 12px;
  background: #666666;
  position: absolute;
  border-radius: 4px;
}
.yzp-swiper-wrap .yzp-swiper-btn > div:before {
  transform: rotate(-45deg);
  margin-top: -8px;
}
.yzp-swiper-wrap .yzp-swiper-btn > div:after {
  transform: rotate(45deg);
  margin-top: 8px;
}
.yzp-swiper-wrap .yzp-swiper-btn .yzp-swiper-left:before {
  transform: rotate(45deg);
}
.yzp-swiper-wrap .yzp-swiper-btn .yzp-swiper-left:after {
  transform: rotate(-45deg);
}
.yzp-swiper-wrap .yzp-swiper-list {
  height: 100%;
  width: 100%;
  display: flex;
  position: relative;
  padding: 0;
  margin: 0;
  list-style: none;
}
.yzp-swiper-wrap .yzp-swiper-list .yzp-swiper-item {
  flex-shrink: 0;
  width: 100%;
  height: 100%;
}
.yzp-swiper-wrap .yzp-swiper-list .yzp-swiper-item .yzp-swiper-link {
  display: block;
  width: 100%;
  height: 100%;
  background-size: cover !important;
  overflow: hidden;
}
.yzp-swiper-wrap
  .yzp-swiper-list
  .yzp-swiper-item
  .yzp-swiper-link
  .yzp-swiper-img {
  width: 100%;
  height: 100%;
  object-fit: cover;
}
.yzp-swiper-wrap {
  height: 100%;
  overflow: hidden;
  position: relative;
  width: 100%;
}
.yzp-swiper-wrap .yzp-swiper-text {
  position: absolute;
  left: 0;
  right: 0;
  bottom: 0;
  z-index: 2;
  background: rgba(0, 0, 0, 0.5);
  backdrop-filter: saturate(100%) blur(5px);
  color: #ffffff;
  font-size: 12px;
  display: flex;
  align-items: center;
  padding: 8px 10px;
  justify-content: space-between;
}
.yzp-swiper-wrap .yzp-swiper-text .yzp-swiper-title {
  display: inline-block;
  white-space: nowrap;
  text-overflow: ellipsis;
  overflow: hidden;
  margin-right: 5px;
}
.yzp-swiper-wrap .yzp-swiper-text .yzp-swiper-index {
  flex-shrink: 0;
}
.yzp-swiper-wrap .yzp-swiper-dots {
  position: absolute;
  left: 0;
  right: 0;
  bottom: 15px;
  z-index: 2;
  display: flex;
  align-items: center;
  justify-content: center;
}
.yzp-swiper-wrap .yzp-swiper-dots .yzp-swiper-dots-list {
  display: flex;
  align-items: center;
  padding: 0;
  margin: 0;
  list-style: none;
}
.yzp-swiper-wrap .yzp-swiper-dots .yzp-swiper-dots-item {
  height: 5px;
  width: 10px;
  border-radius: 10px;
  background: rgba(255, 255, 255, 0.5);
  transition: all 0.3s;
  margin: 0 5px;
}
.yzp-swiper-wrap .yzp-swiper-dots .yzp-swiper-dots-item.active {
  background: #ffffff;
  width: 20px;
}

 

 

支付宝微信
6
关注公众号获取更多内容
uniapp 长按longpress之后touchend touchcancel不触发不生效
结合lazyload实现文章页里面的图片预加载
暂无评论,快抢沙发吧
不支持canvas
春季
夏季
秋季
冬季
暗黑
简约
小清新