slide.vue 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  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, _, onCleanup) => {
  36. if (!menu || menu.mount) return;
  37. if (menu.handler) {
  38. menu.handler(props.draw);
  39. nextTick(() => (active.value = undefined));
  40. } else {
  41. props.draw.enterDrawShape(menu.payload.type, menu.payload.preset);
  42. onCleanup(() => props.draw.quitDrawShape());
  43. }
  44. });
  45. watch(
  46. () => props.draw.presetAdd && getValue(props.draw.presetAdd, props.menus),
  47. (val) => {
  48. active.value = val;
  49. }
  50. );
  51. const childType = ref<string>();
  52. const openHandler = (value: string) => {
  53. const item = getItem(value, props.menus);
  54. childType.value = item?.type;
  55. };
  56. </script>
  57. <style lang="scss" scoped>
  58. @use '../../styles/global';
  59. .slide {
  60. margin-left: 0;
  61. &.hide {
  62. transform: translateX(-100%);
  63. }
  64. position: relative;
  65. }
  66. .slide-menu {
  67. padding: 20px 0;
  68. width: 100%;
  69. height: 100%;
  70. overflow-y: auto;
  71. }
  72. .ext {
  73. position: absolute;
  74. left: 100%;
  75. top: 0;
  76. bottom: 0;
  77. z-index: 999;
  78. }
  79. </style>
  80. <style lang="scss">
  81. @use '../../styles/global';
  82. .slide-popper .el-menu--popup {
  83. width: global.$slideSize;
  84. }
  85. .slide-popper .el-menu--popup,
  86. .slide-menu-poper .el-menu--popup {
  87. width: global.$slideSize;
  88. }
  89. .slide-menu,
  90. .slide-menu-poper .el-menu--popup {
  91. --el-menu-base-level-padding: 0;
  92. --el-menu-item-height: 70px;
  93. .el-menu-item,
  94. .el-sub-menu__title {
  95. background: none;
  96. position: relative;
  97. &::before {
  98. content: "";
  99. position: absolute;
  100. width: 56px;
  101. height: 56px;
  102. background: var(--el-menu-hover-bg-color);
  103. border-radius: 4px;
  104. left: 50%;
  105. top: 50%;
  106. transform: translate(-50%, -50%);
  107. z-index: 0;
  108. opacity: 0;
  109. transition: opacity 0.3s ease;
  110. }
  111. &:hover::before {
  112. opacity: 1;
  113. }
  114. }
  115. }
  116. .slide-menu-poper .el-menu--popup {
  117. min-width: auto;
  118. }
  119. .slide-menu .menu-layout,
  120. .slide-menu-poper .menu-layout {
  121. position: relative;
  122. z-index: 1;
  123. width: 56px;
  124. height: 56px;
  125. margin: auto;
  126. border-radius: 4px;
  127. overflow: hidden;
  128. color: #000;
  129. display: flex;
  130. flex-direction: column;
  131. align-items: center;
  132. justify-content: center;
  133. line-height: 1em;
  134. font-size: 22px;
  135. span {
  136. font-size: 14px;
  137. margin-top: 5px;
  138. }
  139. }
  140. .sub-menu-horizontal .el-menu--popup {
  141. min-width: auto;
  142. .menu-layout {
  143. font-size: 32px;
  144. display: flex;
  145. align-items: center;
  146. span {
  147. margin-left: 5px;
  148. font-size: 14px;
  149. }
  150. }
  151. }
  152. </style>