index.vue 15 KB

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