more.vue 6.3 KB

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