index.vue 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. <template>
  2. <view class="van-dropdown-menu van-dropdown-menu--top-bottom">
  3. <view :data-index="index" :class="utils.bem('dropdown-menu__item', { disabled: item.disabled })" @tap="onTitleTap" v-for="(item, index) in itemListData" :key="index">
  4. <view
  5. :class="item.titleClass + ' ' + utils.bem('dropdown-menu__title', { active: item.showPopup, down: item.showPopup === (direction === 'down') })"
  6. :style="item.showPopup ? 'color:' + activeColor : ''"
  7. >
  8. <view class="van-ellipsis">
  9. {{ computed.displayTitle(item) }}
  10. </view>
  11. </view>
  12. </view>
  13. <slot />
  14. </view>
  15. </template>
  16. <script module="utils" lang="wxs" src="@/node_modules/@vant/weapp/dist/wxs/utils.wxs"></script>
  17. <script module="computed" lang="wxs" src="@/node_modules/@vant/weapp/dist/dropdown-menu/index.wxs"></script>
  18. <script>
  19. import { VantComponent } from '../common/component';
  20. import { addUnit } from '../common/utils';
  21. let ARRAY = [];
  22. export default {
  23. data() {
  24. return {
  25. itemListData: []
  26. };
  27. },
  28. field: true,
  29. '../dropdown-item/index': {
  30. name: 'dropdown-item',
  31. type: 'descendant',
  32. current: 'dropdown-menu',
  33. linked() {
  34. this.updateItemListData();
  35. },
  36. unlinked() {
  37. this.updateItemListData();
  38. }
  39. },
  40. props: {
  41. activeColor: {
  42. type: String
  43. },
  44. overlay: {
  45. type: Boolean,
  46. default: true
  47. },
  48. zIndex: {
  49. type: Number,
  50. default: 10
  51. },
  52. duration: {
  53. type: Number,
  54. default: 200
  55. },
  56. direction: {
  57. type: String,
  58. default: 'down'
  59. },
  60. closeOnClickOverlay: {
  61. type: Boolean,
  62. default: true
  63. },
  64. closeOnClickOutside: {
  65. type: Boolean,
  66. default: true
  67. }
  68. },
  69. beforeCreate() {
  70. const { windowHeight } = uni.getSystemInfoSync();
  71. this.windowHeight = windowHeight;
  72. ARRAY.push(this);
  73. },
  74. destroyed() {
  75. ARRAY = ARRAY.filter((item) => item !== this);
  76. },
  77. methods: {
  78. updateItemListData() {
  79. this.setData({
  80. itemListData: this.children.map((child) => child.data)
  81. });
  82. },
  83. updateChildrenData() {
  84. this.children.forEach((child) => {
  85. child.updateDataFromParent();
  86. });
  87. },
  88. toggleItem(active) {
  89. this.children.forEach((item, index) => {
  90. const { showPopup } = item.data;
  91. if (index === active) {
  92. item.toggle();
  93. } else if (showPopup) {
  94. item.toggle(false, {
  95. immediate: true
  96. });
  97. }
  98. });
  99. },
  100. close() {
  101. this.children.forEach((child) => {
  102. child.toggle(false, {
  103. immediate: true
  104. });
  105. });
  106. },
  107. getChildWrapperStyle() {
  108. const { zIndex, direction } = this;
  109. return this.getRect('.van-dropdown-menu').then((rect) => {
  110. const { top = 0, bottom = 0 } = rect;
  111. const offset = direction === 'down' ? bottom : this.windowHeight - top;
  112. let wrapperStyle = `z-index: ${zIndex};`;
  113. if (direction === 'down') {
  114. wrapperStyle += `top: ${addUnit(offset)};`;
  115. } else {
  116. wrapperStyle += `bottom: ${addUnit(offset)};`;
  117. }
  118. return wrapperStyle;
  119. });
  120. },
  121. onTitleTap(event) {
  122. const { index } = event.currentTarget.dataset;
  123. const child = this.children[index];
  124. if (!child.data.disabled) {
  125. ARRAY.forEach((menuItem) => {
  126. if (menuItem && menuItem.data.closeOnClickOutside && menuItem !== this) {
  127. menuItem.close();
  128. }
  129. });
  130. this.toggleItem(index);
  131. }
  132. }
  133. },
  134. watch: {
  135. activeColor: {
  136. handler: function () {
  137. this.children.forEach((child) => {
  138. child.updateDataFromParent();
  139. });
  140. },
  141. immediate: true
  142. },
  143. overlay: {
  144. handler: function () {
  145. this.children.forEach((child) => {
  146. child.updateDataFromParent();
  147. });
  148. },
  149. immediate: true
  150. },
  151. duration: {
  152. handler: function () {
  153. this.children.forEach((child) => {
  154. child.updateDataFromParent();
  155. });
  156. },
  157. immediate: true
  158. },
  159. direction: {
  160. handler: function () {
  161. this.children.forEach((child) => {
  162. child.updateDataFromParent();
  163. });
  164. },
  165. immediate: true
  166. },
  167. closeOnClickOverlay: {
  168. handler: function () {
  169. this.children.forEach((child) => {
  170. child.updateDataFromParent();
  171. });
  172. },
  173. immediate: true
  174. }
  175. }
  176. };
  177. </script>
  178. <style>
  179. @import './index.css';
  180. </style>