index - 副本.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. <!-- 去掉头部下面空白之前的版本 -->
  2. <template>
  3. <view :class="['lb-picker', inline ? 'lb-picker-inline' : '']">
  4. <view class="lb-picker-mask"
  5. v-show="visible && !inline"
  6. :style="{ 'background-color': maskColor }"
  7. @tap.stop="handleMaskTap"
  8. @touchmove.stop.prevent="moveHandle">
  9. </view>
  10. <view :class="['lb-picker-container', visible ? 'lb-picker-toggle' : '']"
  11. :style="{ borderRadius: `${radius} ${radius} 0 0` }">
  12. <view v-if="showHeader"
  13. class="lb-picker-header"
  14. :style="{
  15. height: pickerHeaderHeight,
  16. 'line-height': pickerHeaderHeight
  17. }">
  18. <view class="lb-picker-action lb-picker-left">
  19. <view class="lb-picker-action-cancel"
  20. @tap.stop="handleCancel">
  21. <slot v-if="$slots['cancel-text']"
  22. name="cancel-text"> </slot>
  23. <view v-else
  24. class="action-cancel-text"
  25. :style="{ color: cancelColor }">
  26. {{ cancelText }}
  27. </view>
  28. </view>
  29. </view>
  30. <view class="lb-picker-action lb-picker-center"
  31. v-if="$slots['action-center']">
  32. <slot name="action-center"></slot>
  33. </view>
  34. <view class="lb-picker-action lb-picker-right">
  35. <view class="lb-picker-action-confirm"
  36. @tap.stop="handleConfirm">
  37. <slot v-if="$slots['confirm-text']"
  38. name="confirm-text"> </slot>
  39. <view v-else
  40. class="action-confirm-text"
  41. :style="{ color: confirmColor }">
  42. {{ confirmText }}
  43. </view>
  44. </view>
  45. </view>
  46. </view>
  47. <view class="lb-picker-content"
  48. :style="{ height: pickerContentHeight }">
  49. <!-- loading -->
  50. <view v-if="loading"
  51. class="lb-picker-loading">
  52. <slot name="loading">
  53. <view class="lb-picker-loading-img"></view>
  54. </slot>
  55. </view>
  56. <!-- 暂无数据 -->
  57. <view v-if="isEmpty && !loading"
  58. class="lb-picker-empty">
  59. <slot name="empty">
  60. <text class="lb-picker-empty-text"
  61. :style="{ color: emptyColor }">
  62. {{ emptyText }}
  63. </text>
  64. </slot>
  65. </view>
  66. <!-- 单选 -->
  67. <selector-picker v-if="mode === 'selector' && !loading && !isEmpty"
  68. :value="value"
  69. :list="list"
  70. :props="pickerProps"
  71. :height="pickerContentHeight"
  72. :inline="inline"
  73. @change="handleChange">
  74. </selector-picker>
  75. <!-- 多列联动 -->
  76. <multi-selector-picker v-if="mode === 'multiSelector' && !loading && !isEmpty"
  77. :value="value"
  78. :list="list"
  79. :level="level"
  80. :visible="visible"
  81. :props="pickerProps"
  82. :height="pickerContentHeight"
  83. :inline="inline"
  84. @change="handleChange">
  85. </multi-selector-picker>
  86. <!-- 非联动选择 -->
  87. <unlinked-selector-picker v-if="mode === 'unlinkedSelector' && !loading && !isEmpty"
  88. :value="value"
  89. :list="list"
  90. :visible="visible"
  91. :props="pickerProps"
  92. :height="pickerContentHeight"
  93. :inline="inline"
  94. @change="handleChange">
  95. </unlinked-selector-picker>
  96. </view>
  97. </view>
  98. </view>
  99. </template>
  100. <script>
  101. const defaultProps = {
  102. label: 'label',
  103. value: 'value',
  104. children: 'children'
  105. }
  106. import { getIndicatorHeight } from './utils'
  107. import SelectorPicker from './pickers/selector-picker'
  108. import MultiSelectorPicker from './pickers/multi-selector-picker'
  109. import UnlinkedSelectorPicker from './pickers/unlinked-selector-picker'
  110. const indicatorHeight = getIndicatorHeight()
  111. export default {
  112. components: {
  113. SelectorPicker,
  114. MultiSelectorPicker,
  115. UnlinkedSelectorPicker
  116. },
  117. props: {
  118. value: [String, Number, Array],
  119. list: Array,
  120. mode: {
  121. type: String,
  122. default: 'selector'
  123. },
  124. level: {
  125. type: Number,
  126. default: 1
  127. },
  128. props: {
  129. type: Object
  130. },
  131. cancelText: {
  132. type: String,
  133. default: '取消'
  134. },
  135. cancelColor: String,
  136. confirmText: {
  137. type: String,
  138. default: '确定'
  139. },
  140. confirmColor: String,
  141. canHide: {
  142. type: Boolean,
  143. default: true
  144. },
  145. emptyColor: String,
  146. emptyText: {
  147. type: String,
  148. default: '暂无数据'
  149. },
  150. radius: String,
  151. columnNum: {
  152. type: Number,
  153. default: 5
  154. },
  155. loading: Boolean,
  156. closeOnClickMask: {
  157. type: Boolean,
  158. default: true
  159. },
  160. maskColor: {
  161. type: String,
  162. default: 'rgba(0, 0, 0, 0.4)'
  163. },
  164. dataset: Object,
  165. inline: Boolean,
  166. showHeader: {
  167. type: Boolean,
  168. default: true
  169. }
  170. },
  171. data () {
  172. return {
  173. visible: false,
  174. myValue: this.value,
  175. picker: {},
  176. pickerProps: Object.assign({}, defaultProps, this.props),
  177. pickerHeaderHeight: indicatorHeight + 'px',
  178. pickerContentHeight: indicatorHeight * this.columnNum + 'px'
  179. }
  180. },
  181. computed: {
  182. isEmpty () {
  183. if (!this.list) return true
  184. if (this.list && !this.list.length) return true
  185. return false
  186. }
  187. },
  188. methods: {
  189. show () {
  190. if (this.inline) return
  191. this.visible = true
  192. },
  193. hide () {
  194. if (this.inline) return
  195. this.visible = false
  196. },
  197. handleCancel () {
  198. this.$emit('cancel', this.picker)
  199. if (this.canHide && !this.inline) {
  200. this.hide()
  201. }
  202. },
  203. handleConfirm () {
  204. if (this.isEmpty) {
  205. this.$emit('confirm', null)
  206. this.hide()
  207. } else {
  208. const picker = JSON.parse(JSON.stringify(this.picker))
  209. this.myValue = picker.value
  210. this.$emit('confirm', this.picker)
  211. if (this.canHide) this.hide()
  212. }
  213. },
  214. handleChange ({ value, item, index, change }) {
  215. this.picker.value = value
  216. this.picker.item = item
  217. this.picker.index = index
  218. this.picker.change = change
  219. this.picker.dataset = this.dataset || {}
  220. this.$emit('change', this.picker)
  221. },
  222. handleMaskTap () {
  223. if (this.closeOnClickMask) {
  224. this.visible = false
  225. }
  226. },
  227. moveHandle () {}
  228. },
  229. watch: {
  230. value (newVal) {
  231. this.myValue = newVal
  232. },
  233. myValue (newVal) {
  234. this.$emit('input', newVal)
  235. },
  236. visible (newVisible) {
  237. if (newVisible) {
  238. this.$emit('show')
  239. } else {
  240. this.$emit('hide')
  241. }
  242. }
  243. }
  244. }
  245. </script>
  246. <style lang="scss" scoped>
  247. @import "./style/picker.scss";
  248. </style>