index.vue 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. <template>
  2. <view v-if="showWrapper" :class="utils.bem('dropdown-item', direction)" :style="wrapperStyle">
  3. <van-popup
  4. :show="showPopup"
  5. :custom-style="'position: absolute;' + popupStyle"
  6. overlay-style="position: absolute;"
  7. :overlay="overlay"
  8. :position="direction === 'down' ? 'top' : 'bottom'"
  9. :duration="transition ? duration : 0"
  10. :close-on-click-overlay="closeOnClickOverlay"
  11. @enter="onOpen"
  12. @leave="onClose"
  13. @close="toggle"
  14. @after-enter="onOpened"
  15. @after-leave="onClosed"
  16. >
  17. <van-cell
  18. :data-option="item"
  19. :class="utils.bem('dropdown-item__option', { active: item.value === valueClone })"
  20. clickable
  21. :icon="item.icon"
  22. @tap.native="onOptionTap($event, { option: item })"
  23. v-for="(item, index) in options"
  24. :key="index"
  25. >
  26. <view slot="title" class="van-dropdown-item__title" :style="item.value === valueClone ? 'color:' + activeColor : ''">
  27. {{ item.text }}
  28. </view>
  29. <van-icon v-if="item.value === valueClone" name="success" class="van-dropdown-item__icon" :color="activeColor" />
  30. </van-cell>
  31. <slot />
  32. </van-popup>
  33. </view>
  34. </template>
  35. <script module="utils" lang="wxs" src="@/node_modules/@vant/weapp/dist/wxs/utils.wxs"></script>
  36. <script>
  37. import { VantComponent } from '../common/component';
  38. export default {
  39. data() {
  40. return {
  41. transition: true,
  42. showPopup: false,
  43. showWrapper: false,
  44. displayTitle: '',
  45. overlay: '',
  46. duration: '',
  47. activeColor: '',
  48. closeOnClickOverlay: '',
  49. direction: '',
  50. wrapperStyle: '',
  51. valueClone: ''
  52. };
  53. },
  54. field: true,
  55. '../dropdown-menu/index': {
  56. name: 'dropdown-menu',
  57. type: 'ancestor',
  58. current: 'dropdown-item',
  59. linked() {
  60. this.updateDataFromParent();
  61. }
  62. },
  63. props: {
  64. value: {
  65. type: null
  66. },
  67. title: {
  68. type: String
  69. },
  70. disabled: Boolean,
  71. titleClass: {
  72. type: String
  73. },
  74. options: {
  75. type: Array,
  76. default: () => []
  77. },
  78. popupStyle: String
  79. },
  80. methods: {
  81. rerender() {
  82. this.valueClone = this.clone(this.value);
  83. this.$nextTick(() => {
  84. if (this.parent) {
  85. this.parent.updateItemListData();
  86. }
  87. });
  88. },
  89. updateDataFromParent() {
  90. if (this.parent) {
  91. const { overlay, duration, activeColor, closeOnClickOverlay, direction } = this.parent.data;
  92. this.setData({
  93. overlay,
  94. duration,
  95. activeColor,
  96. closeOnClickOverlay,
  97. direction
  98. });
  99. }
  100. },
  101. onOpen() {
  102. this.$emit('open');
  103. },
  104. onOpened() {
  105. this.$emit('opened');
  106. },
  107. onClose() {
  108. this.$emit('close');
  109. },
  110. onClosed() {
  111. this.$emit('closed');
  112. this.setData({
  113. showWrapper: false
  114. });
  115. },
  116. onOptionTap(event, _dataset) {
  117. /* ---处理dataset begin--- */
  118. this.handleDataset(event, _dataset);
  119. /* ---处理dataset end--- */
  120. const { option } = event.currentTarget.dataset;
  121. const { value } = option;
  122. const shouldEmitChange = this.value !== value;
  123. this.setData({
  124. showPopup: false,
  125. valueClone: value
  126. });
  127. this.$emit('close');
  128. this.rerender();
  129. if (shouldEmitChange) {
  130. this.$emit('change', value);
  131. }
  132. },
  133. toggle(show, options = {}) {
  134. const { showPopup } = this;
  135. if (typeof show !== 'boolean') {
  136. show = !showPopup;
  137. }
  138. if (show === showPopup) {
  139. return;
  140. }
  141. this.setData({
  142. transition: !options.immediate,
  143. showPopup: show
  144. });
  145. if (show) {
  146. this.parent.getChildWrapperStyle().then((wrapperStyle) => {
  147. this.setData({
  148. wrapperStyle,
  149. showWrapper: true
  150. });
  151. this.rerender();
  152. });
  153. } else {
  154. this.rerender();
  155. }
  156. }
  157. },
  158. watch: {
  159. value: {
  160. handler: function () {
  161. this.valueClone = this.clone(this.value);
  162. this.$nextTick(() => {
  163. if (this.parent) {
  164. this.parent.updateItemListData();
  165. }
  166. });
  167. },
  168. immediate: true
  169. },
  170. title: {
  171. handler: function () {
  172. this.valueClone = this.clone(this.value);
  173. this.$nextTick(() => {
  174. if (this.parent) {
  175. this.parent.updateItemListData();
  176. }
  177. });
  178. },
  179. immediate: true
  180. },
  181. titleClass: {
  182. handler: function () {
  183. this.valueClone = this.clone(this.value);
  184. this.$nextTick(() => {
  185. if (this.parent) {
  186. this.parent.updateItemListData();
  187. }
  188. });
  189. },
  190. immediate: true
  191. },
  192. options: {
  193. handler: function () {
  194. this.valueClone = this.clone(this.value);
  195. this.$nextTick(() => {
  196. if (this.parent) {
  197. this.parent.updateItemListData();
  198. }
  199. });
  200. },
  201. immediate: true,
  202. deep: true
  203. }
  204. }
  205. };
  206. </script>
  207. <style>
  208. @import './index.css';
  209. </style>