index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. <template>
  2. <view :class="['lb-picker', inline ? 'lb-picker-inline' : '']">
  3. <!-- 默认插槽 -->
  4. <view v-if="!inline"
  5. class="lb-picker-default-slot"
  6. @tap="show">
  7. <slot></slot>
  8. </view>
  9. <!-- 遮罩层 -->
  10. <view v-if="visible && showMask && !inline"
  11. :class="['lb-picker-mask', animation ? 'lb-picker-mask-animation' : '']"
  12. :style="{
  13. backgroundColor: maskBgColor,
  14. zIndex: zIndex - 1
  15. }"
  16. @tap.stop="handleMaskTap"
  17. @touchmove.stop.prevent="moveHandle">
  18. </view>
  19. <view v-if="visible || inline"
  20. :class="[
  21. 'lb-picker-container',
  22. !inline ? 'lb-picker-container-fixed' : '',
  23. animation ? 'lb-picker-container-animation' : '',
  24. containerVisible ? 'lb-picker-container-show' : ''
  25. ]"
  26. :style="{
  27. borderRadius: `${radius} ${radius} 0 0`,
  28. zIndex: zIndex
  29. }">
  30. <view v-if="showHeader"
  31. class="lb-picker-header">
  32. <!-- 头部顶部插槽 -->
  33. <slot name="header-top"></slot>
  34. <view class="lb-picker-header-actions">
  35. <view class="lb-picker-action lb-picker-left">
  36. <!-- 取消 -->
  37. <view class="lb-picker-action-item lb-picker-action-cancel"
  38. @tap.stop="handleCancel">
  39. <slot v-if="$slots['cancel-text']"
  40. name="cancel-text">
  41. </slot>
  42. <text v-else
  43. class="lb-picker-action-cancel-text"
  44. :style="{ color: cancelColor }">
  45. {{ cancelText }}
  46. </text>
  47. </view>
  48. </view>
  49. <!-- 中间 -->
  50. <view class="lb-picker-action lb-picker-center"
  51. v-if="$slots['action-center']">
  52. <slot name="action-center"></slot>
  53. </view>
  54. <!-- 确定 -->
  55. <view class="lb-picker-action lb-picker-right">
  56. <view class="lb-picker-action-item lb-picker-action-confirm"
  57. @tap.stop="handleConfirm">
  58. <slot v-if="$slots['confirm-text']"
  59. name="confirm-text"> </slot>
  60. <text v-else
  61. class="lb-picker-action-confirm-text"
  62. :style="{ color: confirmColor }">
  63. {{ confirmText }}
  64. </text>
  65. </view>
  66. </view>
  67. </view>
  68. <!-- 头部底部插槽 -->
  69. <slot name="header-bottom"></slot>
  70. </view>
  71. <view class="lb-picker-content">
  72. <!-- 选择器顶部插槽 -->
  73. <slot name="picker-top"></slot>
  74. <view class="lb-picker-content-main"
  75. :style="{ height: pickerContentHeight }">
  76. <!-- loading -->
  77. <view v-if="loading"
  78. class="lb-picker-loading">
  79. <slot name="loading">
  80. <image class="lb-picker-loading-img"
  81. src="">
  82. </image>
  83. </slot>
  84. </view>
  85. <!-- 暂无数据 -->
  86. <view v-if="isEmpty && !loading"
  87. class="lb-picker-empty">
  88. <slot name="empty">
  89. <text class="lb-picker-empty-text"
  90. :style="{ color: emptyColor }">
  91. {{ $t('common.notData') }}
  92. </text>
  93. </slot>
  94. </view>
  95. <!-- 单选 -->
  96. <selector-picker v-if="mode === 'selector' && !loading && !isEmpty"
  97. :ref="mode"
  98. :value="value"
  99. :list="list"
  100. :mode="mode"
  101. :props="pickerProps"
  102. :height="pickerContentHeight"
  103. :inline="inline"
  104. @change="handleChange">
  105. </selector-picker>
  106. <!-- 多列联动 -->
  107. <multi-selector-picker v-if="mode === 'multiSelector' && !loading && !isEmpty"
  108. :ref="mode"
  109. :value="value"
  110. :list="list"
  111. :mode="mode"
  112. :level="level"
  113. :visible="visible"
  114. :props="pickerProps"
  115. :height="pickerContentHeight"
  116. :inline="inline"
  117. @change="handleChange">
  118. </multi-selector-picker>
  119. <!-- 非联动选择 -->
  120. <unlinked-selector-picker v-if="mode === 'unlinkedSelector' && !loading && !isEmpty"
  121. :ref="mode"
  122. :value="value"
  123. :list="list"
  124. :mode="mode"
  125. :visible="visible"
  126. :props="pickerProps"
  127. :height="pickerContentHeight"
  128. :inline="inline"
  129. @change="handleChange">
  130. </unlinked-selector-picker>
  131. </view>
  132. <!-- 选择器底部插槽 -->
  133. <slot name="picker-bottom"></slot>
  134. </view>
  135. </view>
  136. </view>
  137. </template>
  138. <script>
  139. const defaultProps = {
  140. label: 'label',
  141. value: 'value',
  142. children: 'children'
  143. }
  144. import { getColumns } from './utils'
  145. import SelectorPicker from './pickers/selector-picker'
  146. import MultiSelectorPicker from './pickers/multi-selector-picker'
  147. import UnlinkedSelectorPicker from './pickers/unlinked-selector-picker'
  148. export default {
  149. components: {
  150. SelectorPicker,
  151. MultiSelectorPicker,
  152. UnlinkedSelectorPicker
  153. },
  154. props:{
  155. value: [String, Number, Array],
  156. list: Array,
  157. mode: {
  158. type: String,
  159. default: 'selector',
  160. },
  161. level: {
  162. type: Number,
  163. default: 1
  164. },
  165. props: {
  166. type: Object
  167. },
  168. cancelText: {
  169. type: String,
  170. default: '取消'
  171. },
  172. cancelColor: String,
  173. confirmText: {
  174. type: String,
  175. default: '确定'
  176. },
  177. confirmColor: String,
  178. canHide: {
  179. type: Boolean,
  180. default: true
  181. },
  182. emptyColor: String,
  183. emptyText: {
  184. type: String,
  185. default: '暂无数据'
  186. },
  187. radius: String,
  188. columnNum: {
  189. type: Number,
  190. default: 5
  191. },
  192. loading: Boolean,
  193. closeOnClickMask: {
  194. type: Boolean,
  195. default: true
  196. },
  197. showMask: {
  198. type: Boolean,
  199. default: true
  200. },
  201. maskColor: {
  202. type: String,
  203. default: 'rgba(0, 0, 0, 0.4)'
  204. },
  205. dataset: Object,
  206. inline: Boolean,
  207. showHeader: {
  208. type: Boolean,
  209. default: true
  210. },
  211. animation: {
  212. type: Boolean,
  213. default: true
  214. },
  215. zIndex: {
  216. type: Number,
  217. default: 999
  218. }
  219. },
  220. data () {
  221. return {
  222. visible: false,
  223. containerVisible: false,
  224. maskBgColor: '',
  225. myValue: this.value,
  226. picker: {},
  227. pickerProps: Object.assign({}, defaultProps, this.props),
  228. pickerContentHeight: 34 * this.columnNum + 'px'
  229. }
  230. },
  231. computed: {
  232. isEmpty () {
  233. if (!this.list) return true
  234. if (this.list && !this.list.length) return true
  235. return false
  236. }
  237. },
  238. methods: {
  239. show () {
  240. if (this.inline) return
  241. this.visible = true
  242. setTimeout(() => {
  243. this.maskBgColor = this.maskColor
  244. this.containerVisible = true
  245. }, 20)
  246. },
  247. hide () {
  248. if (this.inline) return
  249. this.maskBgColor = ''
  250. this.containerVisible = false
  251. setTimeout(() => {
  252. this.visible = false
  253. }, 200)
  254. },
  255. handleCancel () {
  256. this.$emit('cancel', this.picker)
  257. if (this.canHide && !this.inline) {
  258. this.hide()
  259. }
  260. },
  261. handleConfirm () {
  262. if (this.isEmpty) {
  263. this.$emit('confirm', null)
  264. this.hide()
  265. } else {
  266. const picker = JSON.parse(JSON.stringify(this.picker))
  267. this.myValue = picker.value
  268. this.$emit('confirm', this.picker)
  269. this.$nextTick(() => {
  270. this.$refs[this.mode].isConfirmChange = true
  271. })
  272. if (this.canHide) this.hide()
  273. }
  274. },
  275. handleChange ({ value, item, index, change }) {
  276. this.picker.value = value
  277. this.picker.item = item
  278. this.picker.index = index
  279. this.picker.change = change
  280. this.picker.dataset = this.dataset || {}
  281. this.$emit('change', this.picker)
  282. this.$nextTick(() => {
  283. this.$refs[this.mode].isConfirmChange = false
  284. })
  285. },
  286. handleMaskTap () {
  287. if (this.closeOnClickMask) {
  288. this.hide()
  289. }
  290. },
  291. moveHandle () {},
  292. getColumnsInfo (value, type = 1) {
  293. let columnsInfo = getColumns(
  294. {
  295. value,
  296. list: this.list,
  297. mode: this.mode,
  298. props: this.pickerProps,
  299. level: this.level
  300. },
  301. type
  302. )
  303. if (columnsInfo) {
  304. if (this.mode === 'selector') {
  305. columnsInfo.index = columnsInfo.index[0]
  306. }
  307. } else {
  308. columnsInfo = {}
  309. }
  310. columnsInfo.dataset = this.dataset || {}
  311. return columnsInfo
  312. }
  313. },
  314. watch: {
  315. value (newVal) {
  316. this.myValue = newVal
  317. },
  318. myValue (newVal) {
  319. this.$emit('input', newVal)
  320. },
  321. visible (newVisible) {
  322. if (newVisible) {
  323. this.$emit('show')
  324. } else {
  325. this.$emit('hide')
  326. }
  327. },
  328. props (newProps) {
  329. this.pickerProps = Object.assign({}, defaultProps, newProps)
  330. }
  331. }
  332. }
  333. </script>
  334. <style lang="scss" scoped>
  335. @import "./style/picker.scss";
  336. </style>