index.vue 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. <template>
  2. <view class="van-picker custom-class">
  3. <!-- parse <template is="toolbar" v-if="toolbarPosition === 'top'" :data="showToolbar, cancelButtonText, title, confirmButtonText"></template> -->
  4. <block name="toolbar" v-if="toolbarPosition === 'top'">
  5. <view v-if="showToolbar" class="van-picker__toolbar van-hairline--top-bottom toolbar-class">
  6. <view class="van-picker__cancel" hover-class="van-picker__cancel--hover" hover-stay-time="70" data-type="cancel" @tap="emit">
  7. {{ cancelButtonText }}
  8. </view>
  9. <view v-if="title" class="van-picker__title van-ellipsis">{{ title }}</view>
  10. <view class="van-picker__confirm" hover-class="van-picker__confirm--hover" hover-stay-time="70" data-type="confirm" @tap="emit">
  11. {{ confirmButtonText }}
  12. </view>
  13. </view>
  14. </block>
  15. <view v-if="loading" class="van-picker__loading">
  16. <loading color="#1989fa" />
  17. </view>
  18. <view class="van-picker__columns" :style="'height: ' + itemHeight * visibleItemCount + 'px'" @touchmove.stop.prevent="noop">
  19. <picker-column
  20. class="van-picker__column"
  21. :data-index="index"
  22. custom-class="column-class"
  23. :value-key="valueKey"
  24. :initial-options="isSimple(columns) ? item : item.values"
  25. :default-index="item.defaultIndex || defaultIndex"
  26. :item-height="itemHeight"
  27. :visible-item-count="visibleItemCount"
  28. active-class="active-class"
  29. @change="onChange($event, { index })"
  30. v-for="(item, index) in isSimple(columns) ? [columns] : columns"
  31. :key="index"
  32. ></picker-column>
  33. <view class="van-picker__mask" :style="'background-size: 100% ' + (itemHeight * visibleItemCount - itemHeight) / 2 + 'px'" />
  34. <view class="van-picker__frame van-hairline--top-bottom" :style="'height: ' + itemHeight + 'px'" />
  35. </view>
  36. <!-- parse <template is="toolbar" v-if="toolbarPosition === 'bottom'" :data="showToolbar, cancelButtonText, title, confirmButtonText"></template> -->
  37. <block name="toolbar" v-if="false" v-if="toolbarPosition === 'bottom'">
  38. <view v-if="showToolbar" class="van-picker__toolbar van-hairline--top-bottom toolbar-class">
  39. <view class="van-picker__cancel" hover-class="van-picker__cancel--hover" hover-stay-time="70" data-type="cancel" @tap="emit">
  40. {{ cancelButtonText }}
  41. </view>
  42. <view v-if="title" class="van-picker__title van-ellipsis">{{ title }}</view>
  43. <view class="van-picker__confirm" hover-class="van-picker__confirm--hover" hover-stay-time="70" data-type="confirm" @tap="emit">
  44. {{ confirmButtonText }}
  45. </view>
  46. </view>
  47. </block>
  48. </view>
  49. </template>
  50. <script module="isSimple" lang="wxs">
  51. function isSimple(columns) {
  52. return columns.length && !columns[0].values;
  53. }
  54. module.exports = isSimple;
  55. </script>
  56. <script>
  57. import pickerColumn from '../picker-column/index';
  58. import loading from '../loading/index';
  59. import { VantComponent } from '../common/component';
  60. import { pickerProps } from './shared';
  61. export default {
  62. components: {
  63. pickerColumn,
  64. loading
  65. },
  66. data() {
  67. return {
  68. toolbarPosition: '',
  69. showToolbar: '',
  70. cancelButtonText: '',
  71. title: '',
  72. confirmButtonText: '',
  73. loading: '',
  74. visibleItemCount: 0,
  75. valueKey: '',
  76. defaultIndex: '',
  77. columns: ''
  78. };
  79. },
  80. classes: ['active-class', 'toolbar-class', 'column-class'],
  81. props: Object.assign(Object.assign({}, pickerProps), {
  82. valueKey: {
  83. type: String,
  84. value: 'text'
  85. },
  86. toolbarPosition: {
  87. type: String,
  88. value: 'top'
  89. },
  90. defaultIndex: {
  91. type: Number,
  92. value: 0
  93. },
  94. columns: {
  95. type: Array,
  96. value: [],
  97. observer(columns = []) {
  98. this.simple = columns.length && !columns[0].values;
  99. this.children = this.zpSelectAllComponents('.van-picker__column');
  100. if (Array.isArray(this.children) && this.children.length) {
  101. this.setColumns().catch(() => {});
  102. }
  103. }
  104. }
  105. }),
  106. beforeCreate() {
  107. this.children = [];
  108. },
  109. methods: {
  110. noop() {},
  111. setColumns() {
  112. const { data } = this;
  113. const columns = this.simple
  114. ? [
  115. {
  116. values: data.columns
  117. }
  118. ]
  119. : data.columns;
  120. const stack = columns.map((column, index) => this.setColumnValues(index, column.values));
  121. return Promise.all(stack);
  122. },
  123. emit(event) {
  124. const { type } = event.currentTarget.dataset;
  125. if (this.simple) {
  126. this.$emit(type, {
  127. value: this.getColumnValue(0),
  128. index: this.getColumnIndex(0)
  129. });
  130. } else {
  131. this.$emit(type, {
  132. value: this.getValues(),
  133. index: this.getIndexes()
  134. });
  135. }
  136. },
  137. onChange(event, _dataset) {
  138. /* ---处理dataset begin--- */
  139. this.handleDataset(event, _dataset);
  140. /* ---处理dataset end--- */
  141. if (this.simple) {
  142. this.$emit('change', {
  143. picker: this,
  144. value: this.getColumnValue(0),
  145. index: this.getColumnIndex(0)
  146. });
  147. } else {
  148. this.$emit('change', {
  149. picker: this,
  150. value: this.getValues(),
  151. index: event.currentTarget.dataset.index
  152. });
  153. }
  154. },
  155. // get column instance by index
  156. getColumn(index) {
  157. return this.children[index];
  158. },
  159. // get column value by index
  160. getColumnValue(index) {
  161. const column = this.getColumn(index);
  162. return column && column.getValue();
  163. },
  164. // set column value by index
  165. setColumnValue(index, value) {
  166. const column = this.getColumn(index);
  167. if (column == null) {
  168. return Promise.reject(new Error('setColumnValue: 对应列不存在'));
  169. }
  170. return column.setValue(value);
  171. },
  172. // get column option index by column index
  173. getColumnIndex(columnIndex) {
  174. return (this.getColumn(columnIndex) || {}).data.currentIndex;
  175. },
  176. // set column option index by column index
  177. setColumnIndex(columnIndex, optionIndex) {
  178. const column = this.getColumn(columnIndex);
  179. if (column == null) {
  180. return Promise.reject(new Error('setColumnIndex: 对应列不存在'));
  181. }
  182. return column.setIndex(optionIndex);
  183. },
  184. // get options of column by index
  185. getColumnValues(index) {
  186. return (this.children[index] || {}).data.options;
  187. },
  188. // set options of column by index
  189. setColumnValues(index, options, needReset = true) {
  190. const column = this.children[index];
  191. if (column == null) {
  192. return Promise.reject(new Error('setColumnValues: 对应列不存在'));
  193. }
  194. const isSame = JSON.stringify(column.data.options) === JSON.stringify(options);
  195. if (isSame) {
  196. return Promise.resolve();
  197. }
  198. return column
  199. .set({
  200. options
  201. })
  202. .then(() => {
  203. if (needReset) {
  204. column.setIndex(0);
  205. }
  206. });
  207. },
  208. // get values of all columns
  209. getValues() {
  210. return this.children.map((child) => child.getValue());
  211. },
  212. // set values of all columns
  213. setValues(values) {
  214. const stack = values.map((value, index) => this.setColumnValue(index, value));
  215. return Promise.all(stack);
  216. },
  217. // get indexes of all columns
  218. getIndexes() {
  219. return this.children.map((child) => child.data.currentIndex);
  220. },
  221. // set indexes of all columns
  222. setIndexes(indexes) {
  223. const stack = indexes.map((optionIndex, columnIndex) => this.setColumnIndex(columnIndex, optionIndex));
  224. return Promise.all(stack);
  225. }
  226. }
  227. };
  228. </script>
  229. <style>
  230. @import './index.css';
  231. @import 'undefined';
  232. </style>