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
- 下载 animate.min.css 或 animate.css
- 在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、实例