like-button.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. <template>
  2. <view class="like-button">
  3. <view class="animate-wrap">
  4. <view
  5. class="a-img"
  6. v-for="(item,index) in viewList"
  7. :key="item.elId"
  8. :ref="item.elId"
  9. :style="{
  10. 'right': site.x || site[0] + 'rpx',
  11. 'bottom': site.y || site[1] + 'rpx'
  12. }">
  13. <image :style="{
  14. 'width': imgWidth + 'rpx',
  15. 'height': imgHeight + 'rpx'
  16. }" mode="widthFix" :src="item.src" :animation="item.animation"></image>
  17. </view>
  18. </view>
  19. <view class="on-button">
  20. <image :src="src" mode="widthFix" :style="{
  21. 'width': width + 'rpx',
  22. 'height': height + 'rpx'
  23. }" v-if="!$slots.default" @click="handleClick"></image>
  24. <view class="el_like_btn" @click="handleClick">
  25. <slot></slot>
  26. </view>
  27. </view>
  28. </view>
  29. </template>
  30. <script>
  31. export default {
  32. props: {
  33. src: {
  34. type: String,
  35. default: '/static/logo.png'
  36. },
  37. showImgs: { // 显示图标路径
  38. type: Array,
  39. default: () => {
  40. return [
  41. ``,
  42. '/static/logo.png',
  43. ]
  44. }
  45. },
  46. duration: { // 动画效果时间
  47. type: Number,
  48. default: 5000
  49. },
  50. range: { // x 间隔幅度
  51. type: Number,
  52. default: 50
  53. },
  54. high: {
  55. type: Number,
  56. default: 360
  57. },
  58. width: { // 图标宽度
  59. type: Number || String,
  60. default: 52
  61. },
  62. height: { // 图标高度
  63. type: Number || String,
  64. default: 52
  65. },
  66. imgWidth: { // 图标宽度
  67. type: Number || String,
  68. default: 52
  69. },
  70. imgHeight: { // 图标高度
  71. type: Number || String,
  72. default: 52
  73. },
  74. throttle: { // 点击节流 ms
  75. type: Number,
  76. default: 200
  77. },
  78. site: { // x y 坐标 [x<Number>, y<Number>]
  79. type: Array || Object,
  80. default: () => {
  81. return [30, 160] || { x: 30, y: 160 }
  82. }
  83. },
  84. large: { // 是否缩放冒泡
  85. type: [Number, Boolean],
  86. default: false
  87. },
  88. alone: {
  89. type: Boolean,
  90. default: true
  91. }
  92. },
  93. data() {
  94. return {
  95. viewList: [], // 渲染元素
  96. elId: 0, // 元素渲染id
  97. oldTime: 0, // 全局时间用于函数节流
  98. timer: null // 定时器
  99. }
  100. },
  101. methods: {
  102. handleClick (e) {
  103. console.log(this.viewList.length)
  104. // 函数节流
  105. let interval = e.timeStamp - this.oldTime
  106. if(interval < this.throttle) return null;
  107. this.oldTime = e.timeStamp
  108. let animation = {}
  109. // 创建animate配置
  110. // #ifdef APP-NVUE
  111. animation = weex.requireModule('animation')
  112. // #endif
  113. let randomImg = Math.floor(Math.random() * this.showImgs.length)
  114. let _item = {
  115. elId: 'el_likeicon_' + this.elId, // 生成元素ref
  116. src: this.showImgs[randomImg], // 随机图标
  117. animation: animation, // 每个盒子动画
  118. x: Math.ceil(Math.random() * this.range), // 方向间距
  119. q: Math.floor(Math.random() * 2), // 随机方向
  120. }
  121. // 动画
  122. let _abs = ['-', '']
  123. let _dirX = Number(_abs[_item.q] + _item.x) // 随机的方向和间距
  124. let _dirY = this.high - Math.random() * 10
  125. // 生成DOM
  126. this.elId ++
  127. this.viewList.push(_item)
  128. // #ifndef APP-NVUE
  129. _item.animation = uni.createAnimation({
  130. duration: this.duration,
  131. timingFunction: 'ease-out',
  132. })
  133. setTimeout(() => {
  134. console.log('animation finished.')
  135. // 完成后事件回调
  136. this.$emit('finished')
  137. // 逐渐消失
  138. if (this.alone) return this.viewList.splice(0, 1);
  139. // 完成动画后在n秒后清空
  140. clearTimeout(this.timer)
  141. this.timer = setTimeout(() => {
  142. this.viewList = []
  143. }, this.duration)
  144. }, this.duration)
  145. // #endif
  146. // 执行动画
  147. setTimeout(() => {
  148. let _n = 1
  149. if (this.large) _n = typeof(this.large) === 'number' ? this.large : 2;
  150. // #ifndef APP-NVUE
  151. _item.animation.translateY(-_dirY).translateX(_dirX).scale(_n, _n).opacity(0).step()
  152. _item.animation = _item.animation.export()
  153. // #endif
  154. // #ifdef APP-NVUE
  155. let el = this.$refs[_item.elId][0];
  156. clearTimeout(this.timer)
  157. _item.animation.transition(el, {
  158. styles: {
  159. transform: `translate(${_dirX}rpx, -${_dirY}rpx) scale(${_n}, ${_n}])`,
  160. transformOrigin: 'center center',
  161. opacity: 0
  162. },
  163. duration: this.duration, // ms
  164. timingFunction: 'ease-out',
  165. delay: 0 // ms
  166. }, () => {
  167. console.log('animation finished.')
  168. // 完成后事件回调
  169. this.$emit('finished')
  170. // 逐渐消失
  171. if (this.alone) {
  172. setTimeout(() => {
  173. this.viewList.splice(0, 1)
  174. }, 0)
  175. return null
  176. } else {
  177. // 完成动画后在n秒后清空
  178. clearTimeout(this.timer)
  179. this.timer = setTimeout(() => {
  180. this.viewList = []
  181. }, this.duration)
  182. }
  183. })
  184. // #endif
  185. }, 100)
  186. // 点击立即触发组件事件
  187. this.$emit('handleClick', this.elId)
  188. }
  189. }
  190. }
  191. </script>
  192. <style lang="scss">
  193. .a-img {
  194. position: fixed;
  195. }
  196. </style>