VUE2+Swiper(H5邀请函开发)

发布时间 2023-08-31 16:17:25作者: 丶凉雨拾忆

1、引入swiper

  • 安装固定版本
npm i swiper@5.3.6
npm i vue-awesome-swiper@4.0.4

2、下载

https://3.swiper.com.cn/download/index.html

  1. 下载 animate.min.css 或 animate.css
  2. 在main中引入 import '@/utils/animate.min.css'

3、由于Swiper动画js不适用VUE ,在utils下新增文件animate.js,重写Swiper动画js

/*
 * @Descripttion: 在Swiper实例中的幻灯片中添加动画效果
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 11:03:10
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 14:03:10
 */
// 将所有具有  .ani  类的元素隐藏,并将它们的样式缓存起来。
export function swiperAnimateCache() {
  const allBoxes = document.querySelectorAll('.ani')
  allBoxes.forEach((element) => {
    const styleValue = element.getAttribute('style') || ''
    element.setAttribute('swiper-animate-style-cache', styleValue)
    element.style.visibility = 'hidden'
  })
}

// 根据传入的 Swiper 实例  a ,为当前活动幻灯片中的所有具有  .ani  类的元素添加动画效果。
// 它会根据元素的属性设置动画的持续时间、延迟和效果,并将元素的样式设置为可见。
export function swiperAnimate(a) {
  clearSwiperAnimate()
  const activeSlide = a.slides[a.activeIndex]
  const animatedElements = activeSlide.querySelectorAll('.ani')
  animatedElements.forEach((element) => {
    element.style.visibility = 'visible'
    const effect = element.getAttribute('swiper-animate-effect') || ''
    element.classList.add('animated', effect)
    const duration = element.getAttribute('swiper-animate-duration') || ''
    const delay = element.getAttribute('swiper-animate-delay') || ''
    const style = `${
      element.getAttribute('style') || ''
    } animation-duration: ${duration}; -webkit-animation-duration: ${duration}; animation-delay: ${delay}; -webkit-animation-delay: ${delay};`
    element.setAttribute('style', style)
  })
}
// 清除所有具有  .ani  类的元素的动画效果。它会恢复元素的初始样式,并将元素隐藏起来。
export function clearSwiperAnimate() {
  const allBoxes = document.querySelectorAll('.ani')
  allBoxes.forEach((element) => {
    const styleCache = element.getAttribute('swiper-animate-style-cache')
    if (styleCache) {
      element.setAttribute('style', styleCache)
    }
    element.style.visibility = 'hidden'
    element.classList.remove('animated')
    const effectValue = element.getAttribute('swiper-animate-effect')
    if (effectValue) {
      element.classList.remove(effectValue)
    }
  })
}

4、创建路由

/*
 * @Descripttion: 文件说明
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 10:03:10
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 10:48:10
 */
import Vue from 'vue'
import Router from 'vue-router'
Vue.use(Router)

const routes = [
  // 默认
  {
    name: '/',
    component: () => import('@/view/Home'),
    meta: {
      loginRequest: true,
    },
  },
  // ------ 首页 ------
  {
    name: 'Home',
    component: () => import('@/view/Home'),
    meta: {
      title: '首页',
      keepAlive: false,
    },
  },
]

routes.forEach((route) => {
  route.path = route.path || '/' + (route.name || '')
})

const router = new Router({
  routes,
})

export { router }

5、创建Home页面并引入Swiper和animate.js

import { Swiper, SwiperSlide } from 'vue-awesome-swiper'
import 'swiper/css/swiper.css'
import { swiperAnimateCache, swiperAnimate } from '@/utils/animate.js'

6、Home 完整代码

/*
 * @Descripttion: 首页
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 09:22:12
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 09:57:00
 */
<template>
  <div class="HomePage">
    <swiper ref="mySwiper" :options="swiperOptions">
      <swiper-slide>
        <page1></page1>
      </swiper-slide>
      <swiper-slide>swiper2</swiper-slide>
      <swiper-slide>swiper3</swiper-slide>
      <swiper-slide>swiper4</swiper-slide>
      <swiper-slide>swiper5</swiper-slide>
      <swiper-slide>swiper6</swiper-slide>
    </swiper>
    <arrow></arrow>
    <music></music>
  </div>
</template>

<script>
import { Swiper, SwiperSlide } from 'vue-awesome-swiper'
import 'swiper/css/swiper.css'
import { swiperAnimateCache, swiperAnimate } from '@/utils/animate.js'
import arrow from '@/components/Arrow'
import music from '@/components/Music'
import page1 from '@/view/page1'
export default {
  name: 'HomePage',
  components: {
    Swiper,
    SwiperSlide,
    arrow,
    music,
    page1,
  },
  data() {
    return {
      // swiper 配置
      swiperOptions: {
        /*
         * 切换方向
         * 默认为"horizontal"(左右)
         * vertical"(上下)
         */
        direction: 'vertical',
        // 自动切换
        autoplay: false,
        // autoplay: {
        //   // 循环时间
        //   delay: 3000,
        //   // 最后一个slide时停止自动切换
        //   stopOnLastSlide: true,
        //   // 用户操作swiper之后,是否禁止autoplay。默认为true:停止。
        //   // 如果设置为false,用户操作swiper之后自动切换不会停止,每次都会重新启动autoplay。
        //   // 操作包括触碰,拖动,点击pagination等。
        //   disableOnInteraction: true,
        // },
        // 环路
        loop: false,
        /*
         * effect:slide的切换效果
         * 默认为"slide"(位移切换)
         * fade"(淡入)
         * cube"(方块)
         * coverflow"(3d流)
         * flip"(3d翻转)
         */
        effect: 'coverflow',
        cubeEffect: {
          slideShadows: true,
          shadow: true,
          shadowOffset: 100,
          shadowScale: 0.7,
        },
        on: {
          init: function () {
            swiperAnimateCache(this)
            swiperAnimate(this)
          },
          slideChangeTransitionStart: function () {},
          slideChangeTransitionEnd: function () {},
          slideChange: function () {
            swiperAnimate(this)
          },
        },
      },
    }
  },
}
</script>

<style lang="less" scoped>
.HomePage {
  width: 100%;
  height: 100%;
  background-color: #9d0b0a;
}
.swiper-container {
  width: 100%;
  height: 100%;
}
</style>

page1 首页组件, arrow 上滑提示箭头组件, music 播放音频组件

7、page1 完整代码

/*
 * @Descripttion: page1组件
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 12:11:10
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 13:42:10
 */
<template>
  <div
    class="page1"
    :style="{
      'background-image': `url(${require('@/assets/page.png')})`,
    }"
  >
    <div
      class="ani item item1"
      swiper-animate-effect="bounceInDown"
      swiper-animate-duration="2s"
      swiper-animate-delay="1s"
    >
      新
    </div>
    <div
      class="ani item item2"
      swiper-animate-effect="bounceInLeft"
      swiper-animate-duration="2s"
      swiper-animate-delay="2s"
    >
      年
    </div>
    <div
      class="ani item item3"
      swiper-animate-effect="bounceInRight"
      swiper-animate-duration="2s"
      swiper-animate-delay="2s"
    >
      快
    </div>
    <div
      class="ani item item4"
      swiper-animate-effect="bounceInDown"
      swiper-animate-duration="2s"
      swiper-animate-delay="1s"
    >
      乐
    </div>
    <div
      class="ani item5"
      swiper-animate-effect="flipInX"
      swiper-animate-duration="1s"
      swiper-animate-delay="5s"
    >
      <img :src="fuzi" />
    </div>
  </div>
</template>

<script>
export default {
  name: 'page1',
  data() {
    return {
      fuzi: require('@/assets/fuzi.png'),
    }
  },
}
</script>

<style lang="less" scoped>
.page1 {
  width: 100%;
  height: 100%;
  background-repeat: no-repeat;
  background-size: 100% 100%;
  .item {
    position: absolute;
    font-size: 52px;
    font-weight: bold;
  }
  .item1 {
    top: 80px;
    left: 11%;
  }
  .item2 {
    top: 80px;
    left: 32%;
  }
  .item3 {
    top: 80px;
    left: 54%;
  }
  .item4 {
    top: 80px;
    left: 75%;
  }
  .item5 {
    width: 70%;
    top: 25%;
    left: 15%;
    img {
      width: 100%;
    }
  }
}
</style>

8、Arrow 上滑组件完整代码

存在在 components 下

/*
 * @Descripttion: 上滑提示组件
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 12:11:10
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 13:42:10
 */
<template>
  <div class="scrolltips">
    <div
      class="sc"
      :style="{
        'background-image': `url(${require('@/assets/arrow.png')})`,
      }"
    ></div>
  </div>
</template>

<script>
export default {
  name: 'arrow',
}
</script>

<style lang="less" scoped>
.scrolltips {
  position: absolute;
  bottom: 0;
  left: 50%;
  z-index: 50;
  width: 32px;
  margin-left: -16px;
}
.sc {
  width: 32px;
  height: 20px;
  position: absolute;
  bottom: 0;
  right: 0;
  opacity: 0;
  background-size: 100% 100%;
  -webkit-transform: translateY(-20px);
  -ms-transform: translateY(-20px);
  transform: translateY(-20px);
  -webkit-animation: sc 2s 0.3s infinite;
  animation: sc 2s 0.3s infinite;
}
@-webkit-keyframes sc {
  0% {
    -webkit-transform: translateY(-20px);
    transform: translateY(-20px);
    opacity: 0;
  }
  30%,
  70% {
    -webkit-transform: translateY(-40px);
    transform: translateY(-40px);
    opacity: 1;
  }
  100% {
    -webkit-transform: translateY(-40px);
    transform: translateY(-40px);
    opacity: 0;
  }
}
@keyframes sc {
  0% {
    -webkit-transform: translateY(-20px);
    transform: translateY(-20px);
    opacity: 0;
  }
  30%,
  70% {
    -webkit-transform: translateY(-40px);
    transform: translateY(-40px);
    opacity: 1;
  }
  100% {
    -webkit-transform: translateY(-40px);
    transform: translateY(-40px);
    opacity: 0;
  }
}
</style>

8、音频组件完整代码

存在在 components 下

/*
 * @Descripttion: 音频组件
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 14:00:10
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 15:12:10
 */
<template>
  <div class="switch-music" @click="toggleAudio" :class="{ on: isPlay }">
    <div class="tips ts5" :class="{ on: showTips }">{{ tipsText }}</div>
    <div class="player"></div>
  </div>
</template>

<script>
export default {
  name: 'Music',
  data() {
    return {
      // 音频地址
      mp3: require('@/assets/test.mp3'),
      // mp3: '',
      // 新建一个 Audio 对象
      audio: null,
      // 打开、关闭
      isPlay: false,
      // 显示提示
      showTips: false,
      tipsText: '开始播放',
    }
  },
  mounted() {
    this.init()
  },
  methods: {
    /**
     *@Descripttion:初始化
     *@Author: PengShuai
     *@Date: 2023-08-31 11:29:29
     */
    init() {
      this.audio = new Audio(this.mp3)
      this.audio.loop = true
      this.audio.autoplay = true
      this.audio.addEventListener('play', this.handlePlay)
      this.audio.addEventListener('pause', this.handlePause)
    },
    /**
     *@Descripttion:点击事件
     *@Author: PengShuai
     *@Date: 2023-08-31 11:33:42
     */
    toggleAudio() {
      if (this.isPlay) {
        this.audio.pause()
      } else {
        this.audio.play()
      }
      this.isPlay = !this.isPlay
    },
    /**
     *@Descripttion:开始播放
     *@Author: PengShuai
     *@Date: 2023-08-31 11:30:58
     */
    handlePlay() {
      const _this = this
      _this.isPlay = true
      _this.showTips = true
      _this.tipsText = '开始播放'
      setTimeout(() => {
        _this.showTips = false
      }, 1000)
    },
    /**
     *@Descripttion:关闭播放
     *@Author: PengShuai
     *@Date: 2023-08-31 11:31:07
     */
    handlePause() {
      const _this = this
      _this.isPlay = false
      _this.showTips = true
      _this.tipsText = '暂停播放'
      setTimeout(() => {
        _this.showTips = false
      }, 1000)
    },
  },
}
</script>

<style lang="less" scoped>
.ts5 {
  -webkit-transition: 0.5s;
  -moz-transition: 0.5s;
  -ms-transition: 0.5s;
  -o-transition: 0.5s;
  transition: 0.5s;
}
.switch-music {
  position: absolute;
  z-index: 30;
  top: 0px;
  right: 0px;
  width: 56px;
  height: 56px;
  .tips {
    position: absolute;
    color: rgba(255, 255, 255, 0.75);
    top: 0;
    right: 60px;
    width: 100px;
    text-align: right;
    opacity: 0;
    line-height: 56px;
  }
  .on {
    opacity: 1;
    right: 50px;
  }
}
.player {
  width: 36px;
  height: 36px;
  margin: 10px;
  background-image: url();
  background-size: 100% 100%;
}
.on .player {
  background-image: url();
  -webkit-animation: player 2s linear infinite;
  animation: player 2s linear infinite;
}
@-webkit-keyframes player {
  0% {
    -webkit-transform: rotate(0);
    transform: rotate(0);
  }
  100% {
    -webkit-transform: rotate(360deg);
    transform: rotate(360deg);
  }
}
@keyframes player {
  0% {
    -webkit-transform: rotate(0);
    transform: rotate(0);
  }
  100% {
    -webkit-transform: rotate(360deg);
    transform: rotate(360deg);
  }
}
</style>

9、引入全局样式

main下引入import '@/utils/global.less'
注意:body下要设置overflow: hidden

/*
 * @Descripttion: 全局样式
 * @version: 0.0.1
 * @Author: PengShuai
 * @Date: 2023-08-31 09:01:12
 * @LastEditors: PengShuai
 * @LastEditTime: 2023-08-31 09:10:00
 */
html,
body {
    width: 100%;
    height: 100%;
    font-family:Helvetica Neue, Helvetica, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif;
    overflow: hidden;
}
*{
  margin: 0;
  padding: 0;
  box-sizing: border-box;
  position: relative;
}

10、文档地址

  • Swiper3
    https://3.swiper.com.cn/

  • CDN
    https://3.swiper.com.cn/download/index.html

  • css3动画
    https://animate.style/

11、实例


12、文件结构