MainMenu.vue 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. <!--
  2. Menu principal de l'application
  3. Prend en paramètre une liste de ItemMenu et les met en forme
  4. -->
  5. <template>
  6. <v-navigation-drawer
  7. v-model="displayMenu"
  8. :rail="isRail"
  9. :disable-resize-watcher="true"
  10. class="theme-secondary main-menu"
  11. >
  12. <template #prepend>
  13. <slot name="title"></slot>
  14. </template>
  15. <v-list
  16. open-strategy="single"
  17. active-class="active"
  18. class="left-menu"
  19. >
  20. <!-- TODO: que se passe-t-il si le menu ne comprend qu'un seul MenuItem? -->
  21. <div v-for="(item, i) in items" :key="i">
  22. <!-- Cas 1 : l'item n'a pas d'enfants, c'est un lien (ou le menu est en mode réduit) -->
  23. <v-list-item
  24. v-if="!item.children || isRail"
  25. :title="$t(item.label)"
  26. :prepend-icon="item.icon.name"
  27. :href="!isInternalLink(item) ? item.to : undefined"
  28. :to="isInternalLink(item) ? item.to : undefined"
  29. exact
  30. height="48px"
  31. class="menu-item"
  32. />
  33. <!-- Cas 2 : l'item a des enfants, c'est un groupe -->
  34. <v-list-group
  35. v-else
  36. expand-icon="fas fa-angle-down"
  37. collapse-icon="fas fa-angle-up"
  38. >
  39. <template #activator="{ props }">
  40. <v-list-item
  41. v-bind="props"
  42. :prepend-icon="item.icon.name"
  43. :title="$t(item.label)"
  44. class="theme-secondary menu-item"
  45. height="48px"
  46. />
  47. </template>
  48. <v-list-item
  49. v-for="child in item.children"
  50. :key="$t(child.label)"
  51. :title="$t(child.label)"
  52. :prepend-icon="child.icon.name"
  53. :href="!isInternalLink(child) ? child.to : undefined"
  54. :to="isInternalLink(child) ? child.to : undefined"
  55. exact
  56. height="38px"
  57. class="theme-secondary-alt"
  58. />
  59. </v-list-group>
  60. </div>
  61. </v-list>
  62. <template #append>
  63. <slot name="foot"></slot>
  64. </template>
  65. </v-navigation-drawer>
  66. </template>
  67. <script setup lang="ts">
  68. import {useMenu} from "~/composables/layout/useMenu";
  69. import {computed} from "@vue/reactivity";
  70. import { useDisplay } from 'vuetify'
  71. import { MenuGroup, MenuItem } from "~/types/layout";
  72. const { getMenu, hasMenu, isInternalLink, setMenuState, isMenuOpened } = useMenu()
  73. const { mdAndUp, lgAndUp } = useDisplay()
  74. const menu = getMenu('Main')
  75. const isOpened = computed(() => isMenuOpened('Main'))
  76. let items: Array<MenuGroup | MenuItem>
  77. if (menu === null) {
  78. items = []
  79. } else if (menu.hasOwnProperty('children')) {
  80. items = (menu as MenuGroup).children ?? []
  81. } else {
  82. items = [menu]
  83. }
  84. // En vue lg+, on affiche toujours le menu
  85. const displayMenu = computed(() => {
  86. return menu !== null && hasMenu('Main') && (lgAndUp.value || isOpened.value)
  87. })
  88. // En vue md+, fermer le menu le passe simplement en mode rail
  89. // Sinon, le fermer le masque complètement
  90. const isRail = computed(() => {
  91. return menu !== null && mdAndUp.value && !isOpened.value && !items.some((item) => item.expanded)
  92. })
  93. const unwatch = watch(lgAndUp, (newValue, oldValue) => {
  94. // Par défaut si l'écran est trop petit au chargement de la page, le menu doit rester fermé.
  95. if (process.client && menu !== null) {
  96. setMenuState('Main', lgAndUp.value)
  97. }
  98. })
  99. onUnmounted(() => {
  100. unwatch()
  101. })
  102. </script>
  103. <style scoped lang="scss">
  104. .v-list-item {
  105. min-height: 10px !important;
  106. }
  107. :deep(.v-list-item-title),
  108. :deep(.v-icon),
  109. {
  110. font-size: 14px;
  111. color: rgb(var(--v-theme-on-secondary));
  112. }
  113. .v-list-item__prepend {
  114. margin: 10px 0;
  115. margin-right: 10px !important;
  116. }
  117. .v-application--is-ltr .v-list-group--no-action > .v-list-group__header {
  118. margin-left: 0;
  119. padding-left: 0;
  120. }
  121. .v-application--is-ltr .v-list-group--no-action > .v-list-group__items > .v-list-item {
  122. padding-left: 30px;
  123. }
  124. .v-list-item__content {
  125. padding: 8px 0;
  126. }
  127. .v-list-group__items .v-list-item {
  128. padding-inline-start: 30px !important;
  129. }
  130. .v-list-group--no-action > .v-list-group__header,
  131. .v-list-item
  132. {
  133. border-left: 3px solid rgb(var(--v-theme-secondary));
  134. height: 48px;
  135. }
  136. .v-list-item:hover,
  137. .v-list-item.active,
  138. :deep(.v-list-group__items .v-list-item)
  139. {
  140. border-left: 3px solid rgb(var(--v-theme-primary));
  141. background-color: rgb(var(--v-theme-secondary-alt)) !important;
  142. color: rgb(var(--v-theme-on-secondary-alt)) !important;
  143. }
  144. :deep(.v-list-group__items .v-list-item-title) {
  145. color: rgb(var(--v-theme-on-secondary-alt));
  146. }
  147. :deep(.v-list-group__items .v-icon) {
  148. color: rgb(var(--v-theme-on-secondary-alt));
  149. }
  150. :deep(.v-list-item .v-icon) {
  151. margin-right: 10px;
  152. }
  153. :deep(.menu-item .fa) {
  154. text-align: center;
  155. }
  156. </style>