You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

395 lines
10 KiB

6 months ago
  1. <template>
  2. <view v-if="show" class="uni-noticebar" :style="{ backgroundColor: backgroundColor }" @click="onClick">
  3. <!-- #ifdef MP-ALIPAY -->
  4. <view v-if="showClose === true || showClose === 'true'" class="uni-noticebar-close" @click="close">
  5. <uni-icons type="closeempty" :color="color" size="12" />
  6. </view>
  7. <view v-if="showIcon === true || showIcon === 'true'" class="uni-noticebar-icon">
  8. <uni-icons type="sound" :color="color" size="14" />
  9. </view>
  10. <!-- #endif -->
  11. <!-- #ifndef MP-ALIPAY -->
  12. <uni-icons v-if="showClose === true || showClose === 'true'" class="uni-noticebar-close" type="closeempty" :color="color"
  13. size="12" @click="close" />
  14. <uni-icons v-if="showIcon === true || showIcon === 'true'" class="uni-noticebar-icon" type="sound" :color="color"
  15. size="14" />
  16. <!-- #endif -->
  17. <view ref="textBox" class="uni-noticebar__content-wrapper" :class="{'uni-noticebar__content-wrapper--scrollable':scrollable, 'uni-noticebar__content-wrapper--single':!scrollable && (single || moreText)}">
  18. <view :id="elIdBox" class="uni-noticebar__content" :class="{'uni-noticebar__content--scrollable':scrollable, 'uni-noticebar__content--single':!scrollable && (single || moreText)}">
  19. <text :id="elId" ref="animationEle" class="uni-noticebar__content-text" :class="{'uni-noticebar__content-text--scrollable':scrollable,'uni-noticebar__content-text--single':!scrollable && (single || moreText)}"
  20. :style="{color:color, width:wrapWidth+'px', 'animationDuration': animationDuration, '-webkit-animationDuration': animationDuration ,animationPlayState: webviewHide?'paused':animationPlayState,'-webkit-animationPlayState':webviewHide?'paused':animationPlayState, animationDelay: animationDelay, '-webkit-animationDelay':animationDelay}">{{text}}</text>
  21. </view>
  22. </view>
  23. <view v-if="showGetMore === true || showGetMore === 'true'" class="uni-noticebar__more" @click="clickMore">
  24. <text v-if="moreText" :style="{ color: moreColor }" class="uni-noticebar__more-text">{{ moreText }}</text>
  25. <uni-icons type="arrowright" :color="moreColor" size="14" />
  26. </view>
  27. </view>
  28. </template>
  29. <script>
  30. import uniIcons from '../uni-icons/uni-icons.vue'
  31. // #ifdef APP-NVUE
  32. const dom = weex.requireModule('dom');
  33. const animation = weex.requireModule('animation');
  34. // #endif
  35. /**
  36. * NoticeBar 自定义导航栏
  37. * @description 通告栏组件
  38. * @tutorial https://ext.dcloud.net.cn/plugin?id=30
  39. * @property {Number} speed 文字滚动的速度默认100px/
  40. * @property {String} text 显示文字
  41. * @property {String} backgroundColor 背景颜色
  42. * @property {String} color 文字颜色
  43. * @property {String} moreColor 查看更多文字的颜色
  44. * @property {String} moreText 设置查看更多的文本
  45. * @property {Boolean} single = [true|false] 是否单行
  46. * @property {Boolean} scrollable = [true|false] 是否滚动为true时NoticeBar为单行
  47. * @property {Boolean} showIcon = [true|false] 是否显示左侧喇叭图标
  48. * @property {Boolean} showClose = [true|false] 是否显示左侧关闭按钮
  49. * @property {Boolean} showGetMore = [true|false] 是否显示右侧查看更多图标为true时NoticeBar为单行
  50. * @event {Function} click 点击 NoticeBar 触发事件
  51. * @event {Function} close 关闭 NoticeBar 触发事件
  52. * @event {Function} getmore 点击查看更多时触发事件
  53. */
  54. export default {
  55. name: 'UniNoticeBar',
  56. components: {
  57. uniIcons
  58. },
  59. props: {
  60. text: {
  61. type: String,
  62. default: ''
  63. },
  64. moreText: {
  65. type: String,
  66. default: ''
  67. },
  68. backgroundColor: {
  69. type: String,
  70. default: '#fffbe8'
  71. },
  72. speed: {
  73. // 默认1s滚动100px
  74. type: Number,
  75. default: 100
  76. },
  77. color: {
  78. type: String,
  79. default: '#de8c17'
  80. },
  81. moreColor: {
  82. type: String,
  83. default: '#999999'
  84. },
  85. single: {
  86. // 是否单行
  87. type: [Boolean, String],
  88. default: false
  89. },
  90. scrollable: {
  91. // 是否滚动,添加后控制单行效果取消
  92. type: [Boolean, String],
  93. default: false
  94. },
  95. showIcon: {
  96. // 是否显示左侧icon
  97. type: [Boolean, String],
  98. default: false
  99. },
  100. showGetMore: {
  101. // 是否显示右侧查看更多
  102. type: [Boolean, String],
  103. default: false
  104. },
  105. showClose: {
  106. // 是否显示左侧关闭按钮
  107. type: [Boolean, String],
  108. default: false
  109. }
  110. },
  111. data() {
  112. const elId = `Uni_${Math.ceil(Math.random() * 10e5).toString(36)}`
  113. const elIdBox = `Uni_${Math.ceil(Math.random() * 10e5).toString(36)}`
  114. return {
  115. textWidth: 0,
  116. boxWidth: 0,
  117. wrapWidth: '',
  118. webviewHide: false,
  119. // #ifdef APP-NVUE
  120. stopAnimation: false,
  121. // #endif
  122. elId: elId,
  123. elIdBox: elIdBox,
  124. show: true,
  125. animationDuration: 'none',
  126. animationPlayState: 'paused',
  127. animationDelay: '0s'
  128. }
  129. },
  130. mounted() {
  131. // #ifdef APP-PLUS
  132. var pages = getCurrentPages();
  133. var page = pages[pages.length - 1];
  134. var currentWebview = page.$getAppWebview();
  135. currentWebview.addEventListener('hide',()=>{
  136. this.webviewHide = true
  137. })
  138. currentWebview.addEventListener('show',()=>{
  139. this.webviewHide = false
  140. })
  141. // #endif
  142. this.$nextTick(() => {
  143. this.initSize()
  144. })
  145. },
  146. // #ifdef APP-NVUE
  147. beforeDestroy() {
  148. this.stopAnimation = true
  149. },
  150. // #endif
  151. methods: {
  152. initSize() {
  153. if (this.scrollable) {
  154. // #ifndef APP-NVUE
  155. let query = [],
  156. boxWidth = 0,
  157. textWidth = 0;
  158. let textQuery = new Promise((resolve, reject) => {
  159. uni.createSelectorQuery()
  160. // #ifndef MP-ALIPAY
  161. .in(this)
  162. // #endif
  163. .select(`#${this.elId}`)
  164. .boundingClientRect()
  165. .exec(ret => {
  166. this.textWidth = ret[0].width
  167. resolve()
  168. })
  169. })
  170. let boxQuery = new Promise((resolve, reject) => {
  171. uni.createSelectorQuery()
  172. // #ifndef MP-ALIPAY
  173. .in(this)
  174. // #endif
  175. .select(`#${this.elIdBox}`)
  176. .boundingClientRect()
  177. .exec(ret => {
  178. this.boxWidth = ret[0].width
  179. resolve()
  180. })
  181. })
  182. query.push(textQuery)
  183. query.push(boxQuery)
  184. Promise.all(query).then(() => {
  185. this.animationDuration = `${this.textWidth / this.speed}s`
  186. this.animationDelay = `-${this.boxWidth / this.speed}s`
  187. setTimeout(() => {
  188. this.animationPlayState = 'running'
  189. }, 1000)
  190. })
  191. // #endif
  192. // #ifdef APP-NVUE
  193. dom.getComponentRect(this.$refs['animationEle'], (res) => {
  194. let winWidth = uni.getSystemInfoSync().windowWidth
  195. this.textWidth = res.size.width
  196. animation.transition(this.$refs['animationEle'], {
  197. styles: {
  198. transform: `translateX(-${winWidth}px)`
  199. },
  200. duration: 0,
  201. timingFunction: 'linear',
  202. delay: 0
  203. }, () => {
  204. if (!this.stopAnimation) {
  205. animation.transition(this.$refs['animationEle'], {
  206. styles: {
  207. transform: `translateX(-${this.textWidth}px)`
  208. },
  209. timingFunction: 'linear',
  210. duration: (this.textWidth - winWidth) / this.speed * 1000,
  211. delay: 1000
  212. }, () => {
  213. if (!this.stopAnimation) {
  214. this.loopAnimation()
  215. }
  216. });
  217. }
  218. });
  219. })
  220. // #endif
  221. }
  222. // #ifdef APP-NVUE
  223. if (!this.scrollable && (this.single || this.moreText)) {
  224. dom.getComponentRect(this.$refs['textBox'], (res) => {
  225. this.wrapWidth = res.size.width
  226. })
  227. }
  228. // #endif
  229. },
  230. loopAnimation() {
  231. // #ifdef APP-NVUE
  232. animation.transition(this.$refs['animationEle'], {
  233. styles: {
  234. transform: `translateX(0px)`
  235. },
  236. duration: 0
  237. }, () => {
  238. if (!this.stopAnimation) {
  239. animation.transition(this.$refs['animationEle'], {
  240. styles: {
  241. transform: `translateX(-${this.textWidth}px)`
  242. },
  243. duration: this.textWidth / this.speed * 1000,
  244. timingFunction: 'linear',
  245. delay: 0
  246. }, () => {
  247. if (!this.stopAnimation) {
  248. this.loopAnimation()
  249. }
  250. });
  251. }
  252. });
  253. // #endif
  254. },
  255. clickMore() {
  256. this.$emit('getmore')
  257. },
  258. close() {
  259. this.show = false;
  260. this.$emit('close')
  261. },
  262. onClick() {
  263. this.$emit('click')
  264. }
  265. }
  266. }
  267. </script>
  268. <style lang="scss" scoped>
  269. .uni-noticebar {
  270. /* #ifndef APP-NVUE */
  271. display: flex;
  272. width: 100%;
  273. box-sizing: border-box;
  274. /* #endif */
  275. flex-direction: row;
  276. align-items: center;
  277. padding: 6px 12px;
  278. margin-bottom: 10px;
  279. }
  280. .uni-noticebar-close {
  281. margin-right: 5px;
  282. }
  283. .uni-noticebar-icon {
  284. margin-right: 5px;
  285. }
  286. .uni-noticebar__content-wrapper {
  287. flex: 1;
  288. flex-direction: column;
  289. overflow: hidden;
  290. }
  291. .uni-noticebar__content-wrapper--single {
  292. /* #ifndef APP-NVUE */
  293. line-height: 18px;
  294. /* #endif */
  295. }
  296. .uni-noticebar__content-wrapper--single,
  297. .uni-noticebar__content-wrapper--scrollable {
  298. flex-direction: row;
  299. }
  300. /* #ifndef APP-NVUE */
  301. .uni-noticebar__content-wrapper--scrollable {
  302. position: relative;
  303. height: 18px;
  304. }
  305. /* #endif */
  306. .uni-noticebar__content--scrollable {
  307. /* #ifdef APP-NVUE */
  308. flex: 0;
  309. /* #endif */
  310. /* #ifndef APP-NVUE */
  311. flex: 1;
  312. display: block;
  313. overflow: hidden;
  314. /* #endif */
  315. }
  316. .uni-noticebar__content--single {
  317. /* #ifndef APP-NVUE */
  318. display: flex;
  319. flex: none;
  320. width: 100%;
  321. justify-content: center;
  322. /* #endif */
  323. }
  324. .uni-noticebar__content-text {
  325. font-size: 14px;
  326. line-height: 18px;
  327. /* #ifndef APP-NVUE */
  328. word-break: break-all;
  329. /* #endif */
  330. }
  331. .uni-noticebar__content-text--single {
  332. /* #ifdef APP-NVUE */
  333. lines: 1;
  334. /* #endif */
  335. /* #ifndef APP-NVUE */
  336. display: block;
  337. width: 100%;
  338. white-space: nowrap;
  339. /* #endif */
  340. overflow: hidden;
  341. text-overflow: ellipsis;
  342. }
  343. .uni-noticebar__content-text--scrollable {
  344. /* #ifdef APP-NVUE */
  345. lines: 1;
  346. padding-left: 750rpx;
  347. /* #endif */
  348. /* #ifndef APP-NVUE */
  349. position: absolute;
  350. display: block;
  351. height: 18px;
  352. line-height: 18px;
  353. white-space: nowrap;
  354. padding-left: 100%;
  355. animation: notice 10s 0s linear infinite both;
  356. animation-play-state: paused;
  357. /* #endif */
  358. }
  359. .uni-noticebar__more {
  360. /* #ifndef APP-NVUE */
  361. display: inline-flex;
  362. /* #endif */
  363. flex-direction: row;
  364. flex-wrap: nowrap;
  365. align-items: center;
  366. padding-left: 5px;
  367. }
  368. .uni-noticebar__more-text {
  369. font-size: 14px;
  370. }
  371. @keyframes notice {
  372. 100% {
  373. transform: translate3d(-100%, 0, 0);
  374. }
  375. }
  376. </style>