index.vue 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. <template>
  2. <view
  3. class="van-picker-column custom-class"
  4. :style="'height: ' + itemHeight * visibleItemCount + 'px'"
  5. @touchstart="onTouchStart"
  6. @touchmove.stop.prevent="onTouchMove"
  7. @touchend="onTouchEnd"
  8. @touchcancel="onTouchEnd"
  9. >
  10. <view
  11. :style="
  12. 'transition: transform ' +
  13. duration +
  14. 'ms; line-height: ' +
  15. itemHeight +
  16. 'px; transform: translate3d(0, ' +
  17. (offset + (itemHeight * (visibleItemCount - 1)) / 2) +
  18. 'px, 0)'
  19. "
  20. >
  21. <view
  22. :data-index="index"
  23. :style="'height: ' + itemHeight + 'px'"
  24. :class="
  25. 'van-ellipsis van-picker-column__item ' +
  26. (option && option.disabled ? 'van-picker-column__item--disabled' : '') +
  27. ' ' +
  28. (index === currentIndex ? 'van-picker-column__item--selected active-class' : '')
  29. "
  30. @tap="onClickItem"
  31. v-for="(option, index) in options"
  32. :key="index"
  33. >
  34. {{ getOptionText(option, valueKey) }}
  35. </view>
  36. </view>
  37. </view>
  38. </template>
  39. <script module="getOptionText" lang="wxs" src="@/node_modules/@vant/weapp/dist/picker-column/index.wxs"></script>
  40. <script>
  41. import { VantComponent } from '../common/component';
  42. import { isObj, range } from '../common/utils';
  43. const DEFAULT_DURATION = 200;
  44. export default {
  45. data() {
  46. return {
  47. startY: 0,
  48. offset: 0,
  49. duration: 0,
  50. startOffset: 0,
  51. options: [],
  52. currentIndex: 0,
  53. option: {
  54. disabled: ''
  55. }
  56. };
  57. },
  58. classes: ['active-class'],
  59. props: {
  60. valueKey: String,
  61. className: String,
  62. itemHeight: Number,
  63. visibleItemCount: Number,
  64. initialOptions: {
  65. type: Array,
  66. default: () => []
  67. },
  68. defaultIndex: {
  69. type: Number,
  70. default: 0
  71. }
  72. },
  73. created() {
  74. const { defaultIndex, initialOptions } = this;
  75. this.set({
  76. currentIndex: defaultIndex,
  77. options: initialOptions
  78. }).then(() => {
  79. this.setIndex(defaultIndex);
  80. });
  81. },
  82. methods: {
  83. getCount() {
  84. return this.options.length;
  85. },
  86. onTouchStart(event) {
  87. this.setData({
  88. startY: event.touches[0].clientY,
  89. startOffset: this.offset,
  90. duration: 0
  91. });
  92. },
  93. onTouchMove(event) {
  94. const { data } = this;
  95. const deltaY = event.touches[0].clientY - data.startY;
  96. this.setData({
  97. offset: range(data.startOffset + deltaY, -(this.getCount() * data.itemHeight), data.itemHeight)
  98. });
  99. },
  100. onTouchEnd() {
  101. const { data } = this;
  102. if (data.offset !== data.startOffset) {
  103. this.setData({
  104. duration: DEFAULT_DURATION
  105. });
  106. const index = range(Math.round(-data.offset / data.itemHeight), 0, this.getCount() - 1);
  107. this.setIndex(index, true);
  108. }
  109. },
  110. onClickItem(event) {
  111. const { index } = event.currentTarget.dataset;
  112. this.setIndex(index, true);
  113. },
  114. adjustIndex(index) {
  115. const { data } = this;
  116. const count = this.getCount();
  117. index = range(index, 0, count);
  118. for (let i = index; i < count; i++) {
  119. if (!this.isDisabled(data.options[i])) {
  120. return i;
  121. }
  122. }
  123. for (let i = index - 1; i >= 0; i--) {
  124. if (!this.isDisabled(data.options[i])) {
  125. return i;
  126. }
  127. }
  128. },
  129. isDisabled(option) {
  130. return isObj(option) && option.disabled;
  131. },
  132. getOptionText(option) {
  133. const { data } = this;
  134. return isObj(option) && data.valueKey in option ? option[data.valueKey] : option;
  135. },
  136. setIndex(index, userAction) {
  137. const { data } = this;
  138. index = this.adjustIndex(index) || 0;
  139. const offset = -index * data.itemHeight;
  140. if (index !== data.currentIndex) {
  141. return this.set({
  142. offset,
  143. currentIndex: index
  144. }).then(() => {
  145. if (userAction) {
  146. this.$emit('change', index);
  147. }
  148. });
  149. }
  150. return this.set({
  151. offset
  152. });
  153. },
  154. setValue(value) {
  155. const { options } = this;
  156. for (let i = 0; i < options.length; i++) {
  157. if (this.getOptionText(options[i]) === value) {
  158. return this.setIndex(i);
  159. }
  160. }
  161. return Promise.resolve();
  162. },
  163. getValue() {
  164. const { data } = this;
  165. return data.options[data.currentIndex];
  166. }
  167. },
  168. watch: {
  169. defaultIndex: {
  170. handler: function (value) {
  171. this.setIndex(value);
  172. },
  173. immediate: true
  174. }
  175. }
  176. };
  177. </script>
  178. <style>
  179. @import './index.css';
  180. </style>