tabbar4.vue 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. <template>
  2. <view class="tabs" :class="[customClass]">
  3. <view v-if="tabs.length <= 4" class="flex-align">
  4. <view
  5. v-for="item in tabs"
  6. :key="item.value"
  7. class="cell flex-align-around"
  8. :class="{ active: tab.value === item.value }"
  9. @click="clickTab(item)"
  10. >
  11. {{ item.title }}
  12. </view>
  13. </view>
  14. <scroll-view
  15. v-else
  16. scroll-x
  17. class="scroll-content"
  18. :scroll-left="scrollLeft"
  19. :scroll-with-animation="true"
  20. >
  21. <view
  22. v-for="(item, index) in tabs"
  23. :key="item.value"
  24. class="cell"
  25. :class="{ active: tab.value === item.value }"
  26. @click="clickTab(item, index)"
  27. >
  28. {{ item.title }}
  29. </view>
  30. </scroll-view>
  31. </view>
  32. </template>
  33. <script>
  34. export default {
  35. props: {
  36. tabs: Array,
  37. initial: {
  38. default: 0,
  39. type: Number
  40. },
  41. customClass: {
  42. default: '',
  43. type: String
  44. }
  45. },
  46. data() {
  47. return {
  48. tab: this.tabs[this.initial],
  49. scrollLeft: 0,
  50. scrollW: 0,
  51. sizeWidth: []
  52. }
  53. },
  54. mounted() {
  55. if (this.scrollLeft <= 0 && this.tabs.length > 4) {
  56. this.getScrollW()
  57. }
  58. },
  59. methods: {
  60. getScrollW() {
  61. const query = uni.createSelectorQuery().in(this)
  62. query
  63. .select('.scroll-content')
  64. .boundingClientRect((data) => {
  65. this.scrollW = data.width
  66. })
  67. .exec()
  68. query
  69. .selectAll('.cell')
  70. .boundingClientRect((data) => {
  71. let dataLen = data.length
  72. for (let i = 0; i < dataLen; i++) {
  73. this.sizeWidth[i] = {
  74. left: data[i].left,
  75. width: data[i].width
  76. }
  77. }
  78. })
  79. .exec()
  80. },
  81. clickTab(item, index) {
  82. if (this.tab.value == item.value) return
  83. this.tab = item
  84. this.$emit('change', item)
  85. if (index) {
  86. this.scrollLeft =
  87. this.sizeWidth[index].left -
  88. this.scrollW / 2 +
  89. this.sizeWidth[index].width / 2
  90. }
  91. }
  92. }
  93. }
  94. </script>
  95. <style lang="scss" scoped>
  96. .tabs {
  97. height: 56rpx;
  98. .cell {
  99. height: 100%;
  100. width: 128rpx;
  101. position: relative;
  102. display: inline-block;
  103. font-size: 24rpx;
  104. color: #999999;
  105. background: #f3f3f8;
  106. border-radius: 24rpx 24rpx 0 0;
  107. text-align: center;
  108. line-height: 56rpx;
  109. &.active {
  110. color: #000;
  111. background-color: #fff;
  112. }
  113. }
  114. }
  115. .scroll-content {
  116. position: relative;
  117. height: 56rpx;
  118. white-space: nowrap;
  119. width: 100%;
  120. .cell {
  121. height: 100%;
  122. width: 128rpx;
  123. position: relative;
  124. display: inline-block;
  125. font-size: 24rpx;
  126. color: #999999;
  127. text-align: center;
  128. line-height: 56rpx;
  129. background: #f3f3f8;
  130. border-radius: 24rpx 24rpx 0 0;
  131. &.active {
  132. color: #000;
  133. background-color: #fff;
  134. }
  135. }
  136. }
  137. </style>