1
0

Breadcrumb.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. <template>
  2. <div :class="[prefixCls, `${prefixCls}--${theme}`]">
  3. <a-breadcrumb :routes="routes">
  4. <template #itemRender="{ route, routes: routesMatched, paths }">
  5. <Icon :icon="getIcon(route)" v-if="getShowBreadCrumbIcon && getIcon(route)" />
  6. <span v-if="!hasRedirect(routesMatched, route)">
  7. {{ t(route.name || route.meta.title) }}
  8. </span>
  9. <router-link v-else to="" @click="handleClick(route, paths, $event)">
  10. {{ t(route.name || route.meta.title) }}
  11. </router-link>
  12. </template>
  13. </a-breadcrumb>
  14. </div>
  15. </template>
  16. <script lang="ts">
  17. import type { RouteLocationMatched } from 'vue-router';
  18. import { useRouter } from 'vue-router';
  19. import type { Menu } from '/@/router/types';
  20. import { defineComponent, ref, watchEffect } from 'vue';
  21. import { Breadcrumb } from 'ant-design-vue';
  22. import Icon from '/@/components/Icon';
  23. import { useDesign } from '/@/hooks/web/useDesign';
  24. import { useRootSetting } from '/@/hooks/setting/useRootSetting';
  25. import { useGo } from '/@/hooks/web/usePage';
  26. import { useI18n } from '/@/hooks/web/useI18n';
  27. import { propTypes } from '/@/utils/propTypes';
  28. import { isString } from '/@/utils/is';
  29. import { filter } from '/@/utils/helper/treeHelper';
  30. import { getMenus } from '/@/router/menus';
  31. import { REDIRECT_NAME } from '/@/router/constant';
  32. import { getAllParentPath } from '/@/router/helper/menuHelper';
  33. export default defineComponent({
  34. name: 'LayoutBreadcrumb',
  35. components: { Icon, [Breadcrumb.name]: Breadcrumb },
  36. props: {
  37. theme: propTypes.oneOf(['dark', 'light']),
  38. },
  39. setup() {
  40. const routes = ref<RouteLocationMatched[]>([]);
  41. const { currentRoute } = useRouter();
  42. const { prefixCls } = useDesign('layout-breadcrumb');
  43. const { getShowBreadCrumbIcon } = useRootSetting();
  44. const go = useGo();
  45. const { t } = useI18n();
  46. watchEffect(async () => {
  47. if (currentRoute.value.name === REDIRECT_NAME) return;
  48. const menus = await getMenus();
  49. const routeMatched = currentRoute.value.matched;
  50. const cur = routeMatched?.[routeMatched.length - 1];
  51. let path = currentRoute.value.path;
  52. if (cur && cur?.meta?.currentActiveMenu) {
  53. path = cur.meta.currentActiveMenu as string;
  54. }
  55. const parent = getAllParentPath(menus, path);
  56. const filterMenus = menus.filter((item) => item.path === parent[0]);
  57. const matched = getMatched(filterMenus, parent) as any;
  58. if (!matched || matched.length === 0) return;
  59. const breadcrumbList = filterItem(matched);
  60. if (currentRoute.value.meta?.currentActiveMenu) {
  61. breadcrumbList.push({
  62. ...currentRoute.value,
  63. name: currentRoute.value.meta?.title || currentRoute.value.name,
  64. } as unknown as RouteLocationMatched);
  65. }
  66. routes.value = breadcrumbList;
  67. });
  68. function getMatched(menus: Menu[], parent: string[]) {
  69. const metched: Menu[] = [];
  70. menus.forEach((item) => {
  71. if (parent.includes(item.path)) {
  72. metched.push({
  73. ...item,
  74. name: item.meta?.title || item.name,
  75. });
  76. }
  77. if (item.children?.length) {
  78. metched.push(...getMatched(item.children, parent));
  79. }
  80. });
  81. return metched;
  82. }
  83. function filterItem(list: RouteLocationMatched[]) {
  84. return filter(list, (item) => {
  85. const { meta, name } = item;
  86. if (!meta) {
  87. return !!name;
  88. }
  89. const { title, hideBreadcrumb } = meta;
  90. if (!title || hideBreadcrumb) {
  91. return false;
  92. }
  93. return true;
  94. }).filter((item) => !item.meta?.hideBreadcrumb);
  95. }
  96. function handleClick(route: RouteLocationMatched, paths: string[], e: Event) {
  97. e?.preventDefault();
  98. const { children, redirect, meta } = route;
  99. if (children?.length && !redirect) {
  100. e?.stopPropagation();
  101. return;
  102. }
  103. if (meta?.carryParam) {
  104. return;
  105. }
  106. if (redirect && isString(redirect)) {
  107. go(redirect);
  108. } else {
  109. let goPath = '';
  110. if (paths.length === 1) {
  111. goPath = paths[0];
  112. } else {
  113. const ps = paths.slice(1);
  114. const lastPath = ps.pop() || '';
  115. goPath = `${lastPath}`;
  116. }
  117. goPath = /^\//.test(goPath) ? goPath : `/${goPath}`;
  118. go(goPath);
  119. }
  120. }
  121. function hasRedirect(routes: RouteLocationMatched[], route: RouteLocationMatched) {
  122. return routes.indexOf(route) !== routes.length - 1;
  123. }
  124. function getIcon(route) {
  125. return route.icon || route.meta?.icon;
  126. }
  127. return { routes, t, prefixCls, getIcon, getShowBreadCrumbIcon, handleClick, hasRedirect };
  128. },
  129. });
  130. </script>
  131. <style lang="less">
  132. @prefix-cls: ~'@{namespace}-layout-breadcrumb';
  133. .@{prefix-cls} {
  134. display: flex;
  135. padding: 0 8px;
  136. align-items: center;
  137. .ant-breadcrumb-link {
  138. .anticon {
  139. margin-right: 4px;
  140. margin-bottom: 2px;
  141. }
  142. }
  143. &--light {
  144. .ant-breadcrumb-link {
  145. color: @breadcrumb-item-normal-color;
  146. a {
  147. color: rgb(0 0 0 / 65%);
  148. &:hover {
  149. color: @primary-color;
  150. }
  151. }
  152. }
  153. .ant-breadcrumb-separator {
  154. color: @breadcrumb-item-normal-color;
  155. }
  156. }
  157. &--dark {
  158. .ant-breadcrumb-link {
  159. color: rgb(255 255 255 / 60%);
  160. a {
  161. color: rgb(255 255 255 / 80%);
  162. &:hover {
  163. color: @white;
  164. }
  165. }
  166. }
  167. .ant-breadcrumb-separator,
  168. .anticon {
  169. color: rgb(255 255 255 / 80%);
  170. }
  171. }
  172. }
  173. </style>