mirror of
https://github.com/ialley-workshop-open/uni-halo.git
synced 2025-05-24 04:20:15 +09:00
251 lines
5.1 KiB
Vue
251 lines
5.1 KiB
Vue
<template>
|
|
<view style="overflow: hidden;position: fixed;width: 100%;height: 100%;pointer-events: none; top: 0;">
|
|
<view class="danmu-li" v-for="(item, index) in listData" :class="item.type" :style="item.style" :key="index">
|
|
<view class="danmu-inner">
|
|
<view class="user-box">
|
|
<view class="user-img">
|
|
<view class="img-box">
|
|
<image :src="item.avatar || 'https://ss0.bdstatic.com/70cFvHSh_Q1YnxGkpoWK1HF6hhy/it/u=317894666,3379114684&fm=26&gp=0.jpg'"></image>
|
|
</view>
|
|
</view>
|
|
<view class="user-text cl1">{{ item.nickName }}</view>
|
|
<view class="user-status cl1">{{ item.text }}</view>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
</view>
|
|
</template>
|
|
<script>
|
|
export default {
|
|
props: {
|
|
//rightToLeft leftToRight leftBottom
|
|
type: {
|
|
type: String,
|
|
default: 'rightToLeft'
|
|
},
|
|
minTime: {
|
|
type: Number,
|
|
default: 4
|
|
},
|
|
maxTime: {
|
|
type: Number,
|
|
default: 9
|
|
},
|
|
minTop: {
|
|
type: Number,
|
|
default: 0
|
|
},
|
|
maxTop: {
|
|
type: Number,
|
|
default: 240
|
|
},
|
|
hrackH: {
|
|
//轨道高度
|
|
type: Number,
|
|
default: 40
|
|
},
|
|
noStacked: {
|
|
//不允许堆叠(暂不可用)
|
|
type: Array,
|
|
default() {
|
|
return [];
|
|
}
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
listData: []
|
|
};
|
|
},
|
|
mounted() {
|
|
this.hrackNum = Math.floor((this.maxTop - this.minTop) / this.hrackH);
|
|
},
|
|
methods: {
|
|
add(obj) {
|
|
let data = {
|
|
item: obj.item,
|
|
id: Date.parse(new Date()),
|
|
time: Math.ceil(Math.floor(Math.random() * (this.maxTime - this.minTime + 1) + this.minTime)),
|
|
type: this.type
|
|
};
|
|
if (this.type === 'leftBottom') {
|
|
let objData = {
|
|
item: data.item,
|
|
type: 'leftBottomEnter',
|
|
style: {
|
|
transition: `all 0.5s`,
|
|
animationDuration: `0.5s`,
|
|
transform: `translateX(0%)`,
|
|
bottom: `${this.minTop}px`
|
|
}
|
|
};
|
|
let listLen = this.listData.length;
|
|
let hrackNum = this.hrackNum;
|
|
for (let i in this.listData) {
|
|
if (this.listData[i].status === 'reuse') {
|
|
//重用
|
|
this.$set(this.listData, i, objData);
|
|
} else if (this.listData[i].status === 'reset') {
|
|
//重置
|
|
this.listData[i].style.transition = 'none';
|
|
this.listData[i].style.bottom = 0;
|
|
this.listData[i].status = 'reuse';
|
|
} else if (this.listData[i].status === 'recycle') {
|
|
//回收
|
|
this.listData[i].type = 'leftBottomExit';
|
|
this.listData[i].status = 'reset';
|
|
} else {
|
|
this.listData[i].style.bottom = parseInt(this.listData[i].style.bottom) + this.hrackH + 'px';
|
|
}
|
|
if (parseInt(this.listData[i].style.bottom) >= this.maxTop - this.hrackH && this.listData[i].status !== 'reset') {
|
|
//需要回收
|
|
this.listData[i].status = 'recycle';
|
|
}
|
|
}
|
|
if (listLen < hrackNum + 2) {
|
|
this.listData.push(objData);
|
|
}
|
|
} else if (this.type === 'rightToLeft' || this.type === 'leftToRight') {
|
|
let objData = this.horStacked(data);
|
|
for (let i in this.listData) {
|
|
if (this.listData[i].delTime <= Date.parse(new Date())) {
|
|
this.repaint(i, objData.type);
|
|
objData.type = '';
|
|
this.$set(this.listData, i, objData);
|
|
return;
|
|
}
|
|
}
|
|
this.listData.push(objData);
|
|
}
|
|
},
|
|
horStacked(data) {
|
|
return {
|
|
item: data.item,
|
|
type: this.type,
|
|
style: {
|
|
animationDuration: `${data.time}s`,
|
|
top: `${Math.ceil(Math.random() * (this.maxTop - this.minTop + 1) + this.minTop)}px`
|
|
},
|
|
delTime: Date.parse(new Date()) + data.time * 1200
|
|
};
|
|
},
|
|
repaint(index, type) {
|
|
setTimeout(() => {
|
|
this.listData[index].type = type;
|
|
}, 100);
|
|
}
|
|
}
|
|
};
|
|
</script>
|
|
<style></style>
|
|
<style lang="scss">
|
|
@keyframes leftBottomEnter {
|
|
0% {
|
|
transform: translateY(100%);
|
|
opacity: 0;
|
|
}
|
|
|
|
100% {
|
|
transform: translateY(0%);
|
|
opacity: 1;
|
|
}
|
|
}
|
|
|
|
@keyframes leftBottomExit {
|
|
0% {
|
|
transform: translateY(0%);
|
|
opacity: 1;
|
|
}
|
|
|
|
100% {
|
|
transform: translateY(-200%);
|
|
opacity: 0;
|
|
}
|
|
}
|
|
|
|
@keyframes leftToRight {
|
|
0% {
|
|
transform: translateX(-100%);
|
|
}
|
|
|
|
100% {
|
|
transform: translateX(100%);
|
|
}
|
|
}
|
|
|
|
@keyframes rightToLeft {
|
|
0% {
|
|
transform: translateX(100%);
|
|
}
|
|
|
|
100% {
|
|
transform: translateX(-100%);
|
|
}
|
|
}
|
|
|
|
.danmu-li {
|
|
position: absolute;
|
|
width: 100%;
|
|
transform: translateX(100%);
|
|
animation-timing-function: linear;
|
|
|
|
&.leftBottomEnter {
|
|
animation-name: leftBottomEnter;
|
|
}
|
|
&.leftBottomExit {
|
|
animation-name: leftBottomExit;
|
|
animation-fill-mode: forwards;
|
|
}
|
|
|
|
&.rightToLeft {
|
|
animation-name: rightToLeft;
|
|
}
|
|
|
|
&.leftToRight {
|
|
animation-name: leftToRight;
|
|
}
|
|
|
|
.danmu-inner {
|
|
display: inline-block;
|
|
|
|
.user-box {
|
|
display: flex;
|
|
padding: 3rpx 40rpx 3rpx 10rpx;
|
|
background: rgba(0, 0, 0, 0.3);
|
|
border-radius: 32rpx;
|
|
align-items: center;
|
|
|
|
.user-img {
|
|
.img-box {
|
|
display: flex;
|
|
|
|
image {
|
|
width: 58rpx;
|
|
height: 58rpx;
|
|
background: rgba(55, 55, 55, 1);
|
|
border-radius: 50%;
|
|
}
|
|
}
|
|
}
|
|
|
|
.user-status {
|
|
margin-left: 10rpx;
|
|
white-space: nowrap;
|
|
font-size: 28rpx;
|
|
font-weight: 400;
|
|
color: rgba(255, 255, 255, 1);
|
|
}
|
|
|
|
.user-text {
|
|
margin-left: 10rpx;
|
|
// white-space: nowrap;
|
|
font-size: 28rpx;
|
|
font-weight: 400;
|
|
width: 80rpx;
|
|
color: rgba(255, 255, 255, 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|