MainMenu.vue 4.6 KB

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