slide.vue 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. <template>
  2. <div class="slide">
  3. <el-menu
  4. :default-active="active"
  5. class="slide-menu"
  6. @select="(val) => (active = active === val ? undefined : val)"
  7. collapse
  8. :popper-offset="0"
  9. :popper-class="childType || 'slide-menu-poper'"
  10. @open="openHandler"
  11. >
  12. <SlideItem v-for="menu in menus" :data="menu" :active="menu.active" />
  13. </el-menu>
  14. <div class="ext">
  15. <component
  16. v-if="activeMenu?.mount"
  17. :is="activeMenu?.mount"
  18. :draw="draw"
  19. @exit="active = undefined"
  20. />
  21. </div>
  22. </div>
  23. </template>
  24. <script lang="ts" setup>
  25. import { ElMenu } from "element-plus";
  26. import { getItem, getValue, MenuItem } from "./menu.ts";
  27. import SlideItem from "./slide-item.vue";
  28. import { computed, nextTick, ref, watch } from "vue";
  29. import { Draw } from "../container/use-draw.ts";
  30. const props = defineProps<{ menus: MenuItem[]; draw: Draw }>();
  31. const active = ref<string>();
  32. const activeMenu = computed(() =>
  33. active.value === undefined ? null : getItem(active.value, props.menus)
  34. );
  35. watch(activeMenu, (menu, oldMenu) => {
  36. if (oldMenu?.payload?.type) {
  37. props.draw.quitDrawShape();
  38. }
  39. if (!menu || menu.mount) {
  40. return;
  41. }
  42. if (menu.handler) {
  43. menu.handler(props.draw);
  44. nextTick(() => (active.value = undefined));
  45. } else {
  46. props.draw.enterDrawShape(menu.payload.type, menu.payload.preset, menu.single);
  47. }
  48. });
  49. watch(
  50. () => props.draw.presetAdd && getValue(props.draw.presetAdd, props.menus),
  51. (val) => {
  52. active.value = val;
  53. }
  54. );
  55. const childType = ref<string>();
  56. const openHandler = (value: string) => {
  57. const item = getItem(value, props.menus);
  58. childType.value = item?.type;
  59. };
  60. </script>
  61. <style lang="scss" scoped>
  62. .slide {
  63. flex: 0 0 auto;
  64. width: var(--slideSize);
  65. margin-left: var(--left);
  66. background: #fff;
  67. transition: margin-left 0.3s ease;
  68. &.hide {
  69. transform: translateX(-100%);
  70. }
  71. position: relative;
  72. }
  73. .slide-menu {
  74. padding: 20px 0;
  75. width: 100%;
  76. height: 100%;
  77. overflow-y: auto;
  78. }
  79. .ext {
  80. position: absolute;
  81. left: 100%;
  82. top: 0;
  83. bottom: 0;
  84. z-index: 999;
  85. }
  86. </style>
  87. <style lang="scss">
  88. .slide-popper .el-menu--popup {
  89. width: var(--slideSize);
  90. }
  91. .slide-popper .el-menu--popup,
  92. .slide-menu-poper .el-menu--popup {
  93. width: var(--slideSize);
  94. }
  95. .slide-menu,
  96. .slide-menu-poper .el-menu--popup {
  97. --el-menu-base-level-padding: 0;
  98. --el-menu-item-height: 70px;
  99. .el-menu-item,
  100. .el-sub-menu__title {
  101. background: none;
  102. position: relative;
  103. &::before {
  104. content: "";
  105. position: absolute;
  106. width: 56px;
  107. height: 56px;
  108. background: var(--el-menu-hover-bg-color);
  109. border-radius: 4px;
  110. left: 50%;
  111. top: 50%;
  112. transform: translate(-50%, -50%);
  113. z-index: 0;
  114. opacity: 0;
  115. transition: opacity 0.3s ease;
  116. }
  117. &:hover::before {
  118. opacity: 1;
  119. }
  120. }
  121. }
  122. .slide-menu-poper .el-menu--popup {
  123. min-width: auto;
  124. }
  125. .slide-menu .menu-layout,
  126. .slide-menu-poper .menu-layout {
  127. position: relative;
  128. z-index: 1;
  129. width: 56px;
  130. height: 56px;
  131. margin: auto;
  132. border-radius: 4px;
  133. overflow: hidden;
  134. color: #000;
  135. display: flex;
  136. flex-direction: column;
  137. align-items: center;
  138. justify-content: center;
  139. line-height: 1em;
  140. font-size: 22px;
  141. span {
  142. font-size: 14px;
  143. margin-top: 5px;
  144. }
  145. }
  146. .sub-menu-horizontal .el-menu--popup {
  147. --el-menu-base-level-padding: 0;
  148. min-width: auto;
  149. .menu-layout {
  150. padding-left: 20px;
  151. padding-right: 20px;
  152. font-size: 32px;
  153. display: flex;
  154. align-items: center;
  155. span {
  156. margin-left: 5px;
  157. font-size: 14px;
  158. color: var(--el-menu-text-color);
  159. }
  160. }
  161. }
  162. </style>