more.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. <template>
  2. <view class="zx-container car-function-set-more">
  3. <view class="fn-wrap">
  4. <view class="title">
  5. {{ $t('固定导航栏') }}
  6. <text class="sort-num">{{ realActiveTabs.length }} / {{ MAX_TAB_LEN }}</text>
  7. </view>
  8. <draggable v-model="activeTabs" :delay="0" :animation="200" class="tab-item-container">
  9. <view v-for="(tab, index) in activeTabs" :class="['tab-item']" :key="index" @longpress="toggleEdit"
  10. @tap="deleteTab(tab)">
  11. <image v-if="tab.isLock && isActiveEdit" :src="icons.lock" class="right-icon" />
  12. <image v-if="getMovableItem(tab)" :src="icons.edit" class="right-icon" />
  13. <image :src="tab.iconUrl" class="icon" />
  14. <view :class="['name', getMovableItem(tab) && 'shake']">{{ tab.name }}</view>
  15. </view>
  16. </draggable>
  17. </view>
  18. <view class="fn-wrap">
  19. <view class="title">{{ $t('其他功能') }}</view>
  20. <view class="tab-item-container is-to-be-select">
  21. <view v-for="tab in toBeSelectTabs" :key="tab.name" class="tab-item" @longpress="toggleEdit"
  22. @tap="addItemTab(tab)">
  23. <image v-if="showAddIcon" :src="icons.add" class="right-icon" />
  24. <image :src="tab.iconUrl" class="icon" />
  25. <view class="name">{{ tab.name }}</view>
  26. </view>
  27. </view>
  28. </view>
  29. <view class="tips">{{ $t('长按拖动可调整顺序,可增减固定导航栏内容') }}</view>
  30. <view class="un-bind-btn" @tap="toUnbind">{{ $t('解除绑定') }}</view>
  31. </view>
  32. </template>
  33. <script>
  34. import draggable from 'vuedraggable'
  35. import { QINIU_URL } from '@/common/constant'
  36. const ICONS = {
  37. lock: `${QINIU_URL}FgMCTZCySvEgLYgkzU65BUR4f4Ls`,
  38. edit: `${QINIU_URL}Fq3V1Zi5tKH7ibTwG1nO7N96CU8m`,
  39. add: `${QINIU_URL}FqLEWbMe8DcnQtNz6GdDDD87ObZK`,
  40. placeholder: `${QINIU_URL}FpKvfFNSDbx0d9RDwyGAQdFb7Kt6`
  41. }
  42. export default {
  43. components: {
  44. // draggable
  45. },
  46. data() {
  47. const MAX_TAB_LEN = 4
  48. return {
  49. QINIU_URL,
  50. MAX_TAB_LEN,
  51. icons: ICONS,
  52. isActiveEdit: false,
  53. placeholderTab: {
  54. name: '',
  55. icon: ICONS.placeholder,
  56. isLock: false,
  57. isPlaceholder: true
  58. },
  59. activeTabs: [],
  60. toBeSelectTabs: []
  61. }
  62. },
  63. computed: {
  64. realActiveTabs() {
  65. return this.activeTabs.filter(tab => tab.name)
  66. },
  67. showAddIcon() {
  68. return this.isActiveEdit && this.realActiveTabs.length < this.MAX_TAB_LEN
  69. }
  70. },
  71. created() {
  72. this.initializeTabs()
  73. },
  74. methods: {
  75. getMovableItem(tab) {
  76. return this.isActiveEdit && !tab.isLock && !tab.isPlaceholder
  77. },
  78. initializeTabs() {
  79. const lang = v => this.$t(v)
  80. this.activeTabs = [
  81. { name: lang('开机'), iconUrl: `${QINIU_URL}Fp5T9lSNakoiioji6S7W4DmFQ_ys`, isLock: true },
  82. { name: lang('闪灯鸣笛'), iconUrl: `${QINIU_URL}FpeQsDh2dbeTullNLI-HhWj4WAQS` },
  83. { name: lang('打开座桶'), iconUrl: `${QINIU_URL}FppwhbFzrEmDeJQgZJtDTu6WOoMZ` },
  84. { name: lang('打开尾箱'), iconUrl: `${QINIU_URL}Fv3KLuYWEeV5IM4_2sMbmur7yZtz` }
  85. ]
  86. this.toBeSelectTabs = [
  87. { name: lang('胎压'), iconUrl: `${QINIU_URL}FmbcjmvoB4J3CT1hrbjNX4kxv9Zq` },
  88. { name: lang('点出信息'), iconUrl: `${QINIU_URL}Fnx_4tLoq1ytvVA0UemepWisI73A` },
  89. { name: lang('导航'), iconUrl: `${QINIU_URL}FrA_ouJtDp-39g7rMBI0EYr2czVE` }
  90. ]
  91. this.addPlaceholders()
  92. },
  93. toggleEdit() {
  94. this.isActiveEdit = !this.isActiveEdit
  95. },
  96. deleteTab(tab) {
  97. if (tab.isLock) return
  98. const index = this.activeTabs.findIndex(t => t.name === tab.name)
  99. if (index !== -1) {
  100. const removedTab = this.activeTabs.splice(index, 1)[0]
  101. this.toBeSelectTabs.push(removedTab)
  102. this.addPlaceholders()
  103. }
  104. },
  105. addItemTab(tab) {
  106. const index = this.toBeSelectTabs.findIndex(t => t.name === tab.name)
  107. if (index !== -1) {
  108. const removedTab = this.toBeSelectTabs.splice(index, 1)[0]
  109. const placeholderIndex = this.activeTabs.findIndex(v => v.isPlaceholder)
  110. this.activeTabs.splice(placeholderIndex, 1, removedTab)
  111. if (this.realActiveTabs.length > this.MAX_TAB_LEN - 1) {
  112. this.isActiveEdit = false
  113. }
  114. }
  115. },
  116. addPlaceholders() {
  117. while (this.activeTabs.length < this.MAX_TAB_LEN) {
  118. this.activeTabs.push({ ...this.placeholderTab })
  119. }
  120. },
  121. toUnbind() {
  122. uni.navigateTo({
  123. url: '/pages/carFunctionSet/unbind'
  124. })
  125. }
  126. }
  127. }
  128. </script>
  129. <style lang="scss" scoped>
  130. @import "@/libs/css/layout.scss";
  131. .car-function-set-more {
  132. .fn-wrap {
  133. width: 100%;
  134. background: #FFFFFF;
  135. border-radius: 40rpx;
  136. margin-bottom: 20rpx;
  137. padding: 40rpx 0 48rpx;
  138. .title {
  139. font-family: PingFangSC, PingFang SC;
  140. font-weight: 400;
  141. font-size: 36rpx;
  142. color: #060809;
  143. margin-bottom: 16rpx;
  144. padding-left: 40rpx;
  145. .sort-num {
  146. margin-left: 12rpx;
  147. width: 36rpx;
  148. font-family: Futura, Futura;
  149. font-size: 30rpx;
  150. color: #060809;
  151. }
  152. }
  153. .tab-item-container {
  154. display: flex;
  155. &.is-to-be-select {
  156. flex-wrap: wrap;
  157. }
  158. }
  159. .tab-item {
  160. text-align: center;
  161. width: 25%;
  162. position: relative;
  163. margin-top: 24rpx;
  164. .right-icon {
  165. position: absolute;
  166. right: 20rpx;
  167. top: 0;
  168. width: 40rpx;
  169. height: 40rpx;
  170. z-index: 9;
  171. }
  172. .icon {
  173. width: 112rpx;
  174. height: 112rpx;
  175. margin-bottom: 20rpx;
  176. }
  177. .name {
  178. font-size: 28rpx;
  179. color: #060809;
  180. &.shake {
  181. animation: shake 0.2s ease-in-out infinite;
  182. }
  183. }
  184. }
  185. }
  186. .tips {
  187. font-size: 24rpx;
  188. color: #060809;
  189. text-align: center;
  190. margin-top: 32rpx;
  191. }
  192. .un-bind-btn {
  193. position: absolute;
  194. bottom: 56rpx;
  195. left: 50%;
  196. transform: translateX(-50%);
  197. width: 93%;
  198. padding: 24rpx;
  199. text-align: center;
  200. background: #E4E7EC;
  201. border-radius: 40rpx;
  202. font-family: PingFangSC, PingFang SC;
  203. font-weight: bold;
  204. font-size: 32rpx;
  205. color: #060809;
  206. &:active {
  207. opacity: .7;
  208. }
  209. }
  210. }
  211. </style>