点击冒泡效果

新建组件like-button,
<template>
<view class="like-button">
<!-- #ifdef APP-NVUE -->
<list class="animate-wrap">
<cell
class="a-img"
v-for="(item,index) in viewList"
:key="item.elId"
:ref="item.elId"
:style="{
'right': site.x || site[0] + 'rpx',
'bottom': site.y || site[1] + 'rpx'
}">
<image :style="{
'width': imgWidth + 'rpx',
'height': imgHeight + 'rpx'
}" mode="widthFix" :src="item.src" :animation="item.animation"></image>
</cell>
</list>
<!-- #endif -->
<!-- #ifndef APP-NVUE -->
<view class="animate-wrap">
<view
class="a-img"
v-for="(item,index) in viewList"
:key="item.elId"
:ref="item.elId"
:style="{
'right': site.x || site[0] + 'rpx',
'bottom': site.y || site[1] + 'rpx'
}">
<image :style="{
'width': imgWidth + 'rpx',
'height': imgHeight + 'rpx'
}" mode="widthFix" :src="item.src" :animation="item.animation"></image>
</view>
</view>
<!-- #endif -->
<view class="on-button">
<image :src="src" mode="widthFix" :style="{
'width': width + 'rpx',
'height': height + 'rpx'
}" v-if="!$slots.default" @click="handleClick"></image>
<view class="el_like_btn" @click="handleClick">
<slot></slot>
</view>
</view>
</view>
</template>
<script>
export default {
props: {
src: {
type: String,
default: '/static/logo.png'
},
showImgs: { // 显示图标路径
type: Array,
default: () => {
return [
``,
'/static/logo.png',
]
}
},
duration: { // 动画效果时间
type: Number,
default: 5000
},
range: { // x 间隔幅度
type: Number,
default: 50
},
high: {
type: Number,
default: 360
},
width: { // 图标宽度
type: Number || String,
default: 52
},
height: { // 图标高度
type: Number || String,
default: 52
},
imgWidth: { // 图标宽度
type: Number || String,
default: 52
},
imgHeight: { // 图标高度
type: Number || String,
default: 52
},
throttle: { // 点击节流 ms
type: Number,
default: 200
},
site: { // x y 坐标 [x<Number>, y<Number>]
type: Array || Object,
default: () => {
return [30, 160] || { x: 30, y: 160 }
}
},
large: { // 是否缩放冒泡
type: [Number, Boolean],
default: false
},
alone: {
type: Boolean,
default: true
}
},
data() {
return {
viewList: [], // 渲染元素
elId: 0, // 元素渲染id
oldTime: 0, // 全局时间用于函数节流
timer: null, // 定时器
waitDeleteIndex: 0
}
},
watch: {
},
methods: {
handleClick (e) {
// 函数节流
let interval = e.timeStamp - this.oldTime
if(interval < this.throttle) return null;
this.oldTime = e.timeStamp
let animation = {}
// 创建animate配置
// #ifdef APP-NVUE
animation = weex.requireModule('animation')
// #endif
let randomImg = Math.floor(Math.random() * this.showImgs.length)
let _item = {
elId: 'el_likeicon_' + this.elId, // 生成元素ref
src: this.showImgs[randomImg], // 随机图标
animation: animation, // 每个盒子动画
x: Math.ceil(Math.random() * this.range), // 方向间距
q: Math.floor(Math.random() * 2), // 随机方向
}
// 动画
let _abs = ['-', '']
let _dirX = Number(_abs[_item.q] + _item.x) // 随机的方向和间距
let _dirY = this.high - Math.random() * 10
// 生成DOM
this.elId ++
this.viewList.push(_item)
// #ifndef APP-NVUE
_item.animation = uni.createAnimation({
duration: this.duration,
timingFunction: 'ease-out',
})
setTimeout(() => {
console.log('animation finished.')
// 完成后事件回调
this.$emit('finished')
// 逐渐消失
if (this.alone) {
this.waitDeleteIndex ++
this.onThrottle(this.deleteView, this.duration)
return null;
} else {
// 完成动画后在n秒后清空
clearTimeout(this.timer)
this.timer = setTimeout(() => {
this.viewList = []
}, this.duration)
return null;
}
}, this.duration)
// #endif
// 执行动画
setTimeout(() => {
this.$nextTick(() => {
let _n = 1
if (this.large) _n = typeof(this.large) === 'number' ? this.large : 2;
// #ifndef APP-NVUE
_item.animation.translateY(-_dirY).translateX(_dirX).scale(_n, _n).opacity(0).step()
_item.animation = _item.animation.export()
// #endif
// #ifdef APP-NVUE
let el = this.$refs[_item.elId][0];
clearTimeout(this.timer)
_item.animation.transition(el, {
styles: {
transform: `translate(${_dirX}rpx, -${_dirY}rpx) scale(${_n}, ${_n}])`,
transformOrigin: 'center center',
opacity: 0
},
duration: this.duration, // ms
timingFunction: 'ease-out',
delay: 0 // ms
}, () => {
console.log('animation finished.')
// 完成后事件回调
this.$emit('finished')
// 逐渐消失
if (this.alone) {
this.waitDeleteIndex ++
this.onThrottle(this.deleteView, this.duration)
return null
} else {
// 完成动画后在n秒后清空
clearTimeout(this.timer)
this.timer = setTimeout(() => {
this.viewList = []
}, this.duration)
}
})
// #endif
})
}, 0)
// 点击立即触发组件事件
this.$emit('handleClick', this.elId)
},
deleteView () {
this.viewList.splice(0, this.waitDeleteIndex)
this.waitDeleteIndex = 0
},
onThrottle (fn, delay) {
let verifi = true
return (function () {
if (!verifi) return false;
verifi = false
setTimeout(() => {
fn()
verifi = true
}, delay)
})()
}
}
}
</script>
<style lang="scss">
.a-img {
position: fixed;
}
</style>
1引入组件import likeButton from '@/components/like-button/like-button.vue'
export default {
components: {likeButton}
}
<like-button></like-button>


nvue中动画支持css animation动画
<template>
<view class="wrapper">
<view class="box" :class="{origin: !go, target: go}"></view>
</view>
</template>
<script>
export default {
data() {
return {
go: false
}
},
mounted() {
setTimeout(() => {
this.go = true
}, 100)
}
}
</script>
<style>
.box {
height: 200rpx;
width: 200rpx;
background-color: #4CD964;
}
.target {
transform: translateX(100px);
transition:transform 1s ;
}
.origin { // 这里可以简写哈
transform: translateX(0px);
transition-property:transform;
transition-duration: 1s;
}
</style>
网友回复


