index.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. import baseComponent from '../helpers/baseComponent'
  2. import classNames from '../helpers/libs/classNames'
  3. import { useRect } from '../helpers/hooks/useDOM'
  4. const notice = ''
  5. const close = ''
  6. baseComponent({
  7. properties: {
  8. prefixCls: {
  9. type: String,
  10. value: 'wux-notice-bar',
  11. },
  12. icon: {
  13. type: String,
  14. value: notice,
  15. },
  16. content: {
  17. type: String,
  18. value: '',
  19. },
  20. mode: {
  21. type: String,
  22. value: '',
  23. },
  24. action: {
  25. type: String,
  26. value: close,
  27. },
  28. loop: {
  29. type: Boolean,
  30. value: false,
  31. },
  32. leading: {
  33. type: Number,
  34. value: 500,
  35. },
  36. trailing: {
  37. type: Number,
  38. value: 800,
  39. },
  40. speed: {
  41. type: Number,
  42. value: 25,
  43. },
  44. },
  45. data: {
  46. animatedWidth: 0,
  47. overflowWidth: 0,
  48. visible: true,
  49. },
  50. computed: {
  51. classes: ['prefixCls', function(prefixCls) {
  52. const wrap = classNames(prefixCls)
  53. const hd = `${prefixCls}__hd`
  54. const icon = `${prefixCls}__icon`
  55. const bd = `${prefixCls}__bd`
  56. const container = `${prefixCls}__marquee-container`
  57. const marquee = `${prefixCls}__marquee`
  58. const ft = `${prefixCls}__ft`
  59. const action = `${prefixCls}__action`
  60. return {
  61. wrap,
  62. hd,
  63. icon,
  64. bd,
  65. container,
  66. marquee,
  67. ft,
  68. action,
  69. }
  70. }],
  71. },
  72. observers: {
  73. content() {
  74. this.resetAnimation()
  75. },
  76. },
  77. methods: {
  78. clearMarqueeTimer() {
  79. if (this.marqueeTimer) {
  80. clearTimeout(this.marqueeTimer)
  81. this.marqueeTimer = null
  82. }
  83. },
  84. startAnimation() {
  85. this.clearMarqueeTimer()
  86. const { overflowWidth, loop, leading, trailing, speed } = this.data
  87. const isLeading = this.data.animatedWidth === 0
  88. const timeout = isLeading ? leading : speed
  89. const animate = () => {
  90. let animatedWidth = this.data.animatedWidth + 1
  91. const isRoundOver = animatedWidth > overflowWidth
  92. // 判断是否完成一次滚动
  93. if (isRoundOver) {
  94. if (!loop) return
  95. // 重置初始位置
  96. animatedWidth = 0
  97. }
  98. // 判断是否等待一段时间后进行下一次滚动
  99. if (isRoundOver && trailing) {
  100. setTimeout(() => {
  101. this.setData({ animatedWidth })
  102. this.marqueeTimer = setTimeout(animate, speed)
  103. }, trailing)
  104. } else {
  105. this.setData({ animatedWidth })
  106. this.marqueeTimer = setTimeout(animate, speed)
  107. }
  108. }
  109. if (this.data.overflowWidth !== 0) {
  110. this.marqueeTimer = setTimeout(animate, timeout)
  111. }
  112. },
  113. initAnimation(isForce) {
  114. const { prefixCls } = this.data
  115. useRect([`.${prefixCls}__marquee-container`, `.${prefixCls}__marquee`], this)
  116. .then((rects) => {
  117. if (rects.filter((n) => !n).length) return
  118. const [container, text] = rects
  119. const overflowWidth = text.width - container.width
  120. if (this.data.overflowWidth !== overflowWidth || isForce) {
  121. this.setData({ overflowWidth, animatedWidth: 0 }, () => {
  122. // 当文本内容存在且长度可滚动时,才触发动画
  123. if (text.width > 0 && overflowWidth > 0) {
  124. this.startAnimation()
  125. } else {
  126. this.clearMarqueeTimer()
  127. }
  128. })
  129. }
  130. })
  131. },
  132. resetAnimation() {
  133. this.initAnimation(true)
  134. },
  135. stopAnimation() {
  136. this.clearMarqueeTimer()
  137. },
  138. onAction() {
  139. if (this.data.mode === 'closable') {
  140. this.clearMarqueeTimer()
  141. this.setData({ visible: false })
  142. }
  143. this.triggerEvent('click')
  144. },
  145. onClick() {
  146. this.triggerEvent('click')
  147. },
  148. },
  149. ready() {
  150. this.initAnimation()
  151. },
  152. detached() {
  153. this.clearMarqueeTimer()
  154. },
  155. })