index.vue 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. <script setup lang="ts">
  2. import { ref } from "vue";
  3. import { useEventListener, onClickOutside } from "@vueuse/core";
  4. import { emitter } from "/@/utils/mitt";
  5. let show = ref(false);
  6. const target = ref(null);
  7. onClickOutside(target, () => {
  8. show.value = false;
  9. });
  10. const addEventClick = (): void => {
  11. useEventListener("click", closeSidebar);
  12. };
  13. const closeSidebar = (evt: any): void => {
  14. const parent = evt.target.closest(".right-panel");
  15. if (!parent) {
  16. show.value = false;
  17. window.removeEventListener("click", closeSidebar);
  18. }
  19. };
  20. emitter.on("openPanel", () => {
  21. show.value = true;
  22. });
  23. defineExpose({
  24. addEventClick
  25. });
  26. </script>
  27. <template>
  28. <div :class="{ show: show }" class="right-panel-container">
  29. <div class="right-panel-background" />
  30. <div ref="target" class="right-panel">
  31. <div class="right-panel-items">
  32. <div class="project-configuration">
  33. <h3>项目配置</h3>
  34. <i class="el-icon-close" @click="show = !show"></i>
  35. </div>
  36. <div style="border-bottom: 1px solid #dcdfe6"></div>
  37. <slot />
  38. </div>
  39. </div>
  40. </div>
  41. </template>
  42. <style>
  43. .showright-panel {
  44. overflow: hidden;
  45. position: relative;
  46. width: calc(100% - 15px);
  47. }
  48. </style>
  49. <style lang="scss" scoped>
  50. .right-panel-background {
  51. position: fixed;
  52. top: 0;
  53. left: 0;
  54. opacity: 0;
  55. transition: opacity 0.3s cubic-bezier(0.7, 0.3, 0.1, 1);
  56. background: rgba(0, 0, 0, 0.2);
  57. z-index: -1;
  58. }
  59. .right-panel {
  60. width: 100%;
  61. max-width: 300px;
  62. height: 100vh;
  63. position: fixed;
  64. top: 0;
  65. right: 0;
  66. box-shadow: 0 0 15px 0 rgba(0, 0, 0, 0.05);
  67. transition: all 0.25s cubic-bezier(0.7, 0.3, 0.1, 1);
  68. transform: translate(100%);
  69. background: #fff;
  70. z-index: 40000;
  71. }
  72. .show {
  73. transition: all 0.3s cubic-bezier(0.7, 0.3, 0.1, 1);
  74. .right-panel-background {
  75. z-index: 20000;
  76. opacity: 1;
  77. width: 100%;
  78. height: 100%;
  79. }
  80. .right-panel {
  81. transform: translate(0);
  82. }
  83. }
  84. .handle-button {
  85. width: 48px;
  86. height: 48px;
  87. position: absolute;
  88. left: -48px;
  89. text-align: center;
  90. font-size: 24px;
  91. border-radius: 6px 0 0 6px !important;
  92. z-index: 0;
  93. pointer-events: auto;
  94. cursor: pointer;
  95. color: #fff;
  96. line-height: 48px;
  97. top: 45%;
  98. background: rgb(24, 144, 255);
  99. i {
  100. font-size: 24px;
  101. line-height: 48px;
  102. }
  103. }
  104. .right-panel-items {
  105. margin-top: 60px;
  106. height: 100vh;
  107. overflow: auto;
  108. }
  109. .project-configuration {
  110. display: flex;
  111. width: 100%;
  112. height: 30px;
  113. position: fixed;
  114. justify-content: space-between;
  115. align-items: center;
  116. top: 15px;
  117. margin-left: 10px;
  118. i {
  119. font-size: 20px;
  120. margin-right: 20px;
  121. &:hover {
  122. cursor: pointer;
  123. color: red;
  124. }
  125. }
  126. }
  127. :deep(.el-divider--horizontal) {
  128. width: 90%;
  129. margin: 20px auto 0 auto;
  130. }
  131. </style>