slide.vue 3.5 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" />
  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 (!menu || menu.mount) {
  37. if (oldMenu?.payload?.type) {
  38. props.draw.quitDrawShape();
  39. }
  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);
  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. @use '../../styles/global';
  63. .slide {
  64. flex: 0 0 auto;
  65. width: global.$slideSize;
  66. margin-left: var(--left);
  67. background: #fff;
  68. transition: margin-left 0.3s ease;
  69. &.hide {
  70. transform: translateX(-100%);
  71. }
  72. position: relative;
  73. }
  74. .slide-menu {
  75. padding: 20px 0;
  76. width: 100%;
  77. height: 100%;
  78. overflow-y: auto;
  79. }
  80. .ext {
  81. position: absolute;
  82. left: 100%;
  83. top: 0;
  84. bottom: 0;
  85. z-index: 999;
  86. }
  87. </style>
  88. <style lang="scss">
  89. @use '../../styles/global';
  90. .slide-popper .el-menu--popup {
  91. width: global.$slideSize;
  92. }
  93. .slide-popper .el-menu--popup,
  94. .slide-menu-poper .el-menu--popup {
  95. width: global.$slideSize;
  96. }
  97. .slide-menu,
  98. .slide-menu-poper .el-menu--popup {
  99. --el-menu-base-level-padding: 0;
  100. --el-menu-item-height: 70px;
  101. .el-menu-item,
  102. .el-sub-menu__title {
  103. background: none;
  104. position: relative;
  105. &::before {
  106. content: "";
  107. position: absolute;
  108. width: 56px;
  109. height: 56px;
  110. background: var(--el-menu-hover-bg-color);
  111. border-radius: 4px;
  112. left: 50%;
  113. top: 50%;
  114. transform: translate(-50%, -50%);
  115. z-index: 0;
  116. opacity: 0;
  117. transition: opacity 0.3s ease;
  118. }
  119. &:hover::before {
  120. opacity: 1;
  121. }
  122. }
  123. }
  124. .slide-menu-poper .el-menu--popup {
  125. min-width: auto;
  126. }
  127. .slide-menu .menu-layout,
  128. .slide-menu-poper .menu-layout {
  129. position: relative;
  130. z-index: 1;
  131. width: 56px;
  132. height: 56px;
  133. margin: auto;
  134. border-radius: 4px;
  135. overflow: hidden;
  136. color: #000;
  137. display: flex;
  138. flex-direction: column;
  139. align-items: center;
  140. justify-content: center;
  141. line-height: 1em;
  142. font-size: 22px;
  143. span {
  144. font-size: 14px;
  145. margin-top: 5px;
  146. }
  147. }
  148. .sub-menu-horizontal .el-menu--popup {
  149. min-width: auto;
  150. .menu-layout {
  151. font-size: 32px;
  152. display: flex;
  153. align-items: center;
  154. span {
  155. margin-left: 5px;
  156. font-size: 14px;
  157. }
  158. }
  159. }
  160. </style>