mirror of
https://github.com/n8n-io/n8n.git
synced 2025-03-05 20:50:17 -08:00
178 lines
4 KiB
Vue
178 lines
4 KiB
Vue
<template>
|
|
<div
|
|
:class="{
|
|
['menu-container']: true,
|
|
[$style.container]: true,
|
|
[$style.menuCollapsed]: collapsed,
|
|
[$style.transparentBackground]: transparentBackground,
|
|
}"
|
|
>
|
|
<div v-if="$slots.header" :class="$style.menuHeader">
|
|
<slot name="header"></slot>
|
|
</div>
|
|
<div :class="$style.menuContent">
|
|
<div :class="{ [$style.upperContent]: true, ['pt-xs']: $slots.menuPrefix }">
|
|
<div v-if="$slots.menuPrefix" :class="$style.menuPrefix">
|
|
<slot name="menuPrefix"></slot>
|
|
</div>
|
|
<ElMenu :default-active="defaultActive" :collapse="collapsed">
|
|
<N8nMenuItem
|
|
v-for="item in upperMenuItems"
|
|
:key="item.id"
|
|
:item="item"
|
|
:compact="collapsed"
|
|
:tooltip-delay="tooltipDelay"
|
|
:mode="mode"
|
|
:active-tab="activeTab"
|
|
:handle-select="onSelect"
|
|
/>
|
|
</ElMenu>
|
|
</div>
|
|
<div :class="[$style.lowerContent, 'pb-2xs']">
|
|
<slot name="beforeLowerMenu"></slot>
|
|
<ElMenu :default-active="defaultActive" :collapse="collapsed">
|
|
<N8nMenuItem
|
|
v-for="item in lowerMenuItems"
|
|
:key="item.id"
|
|
:item="item"
|
|
:compact="collapsed"
|
|
:tooltip-delay="tooltipDelay"
|
|
:mode="mode"
|
|
:active-tab="activeTab"
|
|
:handle-select="onSelect"
|
|
/>
|
|
</ElMenu>
|
|
<div v-if="$slots.menuSuffix" :class="$style.menuSuffix">
|
|
<slot name="menuSuffix"></slot>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div v-if="$slots.footer" :class="$style.menuFooter">
|
|
<slot name="footer"></slot>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { computed, onMounted, ref } from 'vue';
|
|
import { useRoute } from 'vue-router';
|
|
import { ElMenu } from 'element-plus';
|
|
import N8nMenuItem from '../N8nMenuItem';
|
|
import type { IMenuItem } from '../../types';
|
|
import { doesMenuItemMatchCurrentRoute } from '../N8nMenuItem/routerUtil';
|
|
|
|
interface MenuProps {
|
|
type?: 'primary' | 'secondary';
|
|
defaultActive?: string;
|
|
collapsed?: boolean;
|
|
transparentBackground?: boolean;
|
|
mode?: 'router' | 'tabs';
|
|
tooltipDelay?: number;
|
|
items?: IMenuItem[];
|
|
modelValue?: string;
|
|
}
|
|
|
|
const props = withDefaults(defineProps<MenuProps>(), {
|
|
type: 'primary',
|
|
collapsed: false,
|
|
transparentBackground: false,
|
|
mode: 'router',
|
|
tooltipDelay: 300,
|
|
items: () => [],
|
|
});
|
|
const $route = useRoute();
|
|
|
|
const emit = defineEmits<{
|
|
select: [itemId: string];
|
|
'update:modelValue': [itemId: string];
|
|
}>();
|
|
|
|
const activeTab = ref(props.modelValue);
|
|
|
|
const upperMenuItems = computed(() =>
|
|
props.items.filter((item: IMenuItem) => item.position === 'top' && item.available !== false),
|
|
);
|
|
|
|
const lowerMenuItems = computed(() =>
|
|
props.items.filter((item: IMenuItem) => item.position === 'bottom' && item.available !== false),
|
|
);
|
|
|
|
const currentRoute = computed(() => {
|
|
return $route ?? { name: '', path: '' };
|
|
});
|
|
|
|
onMounted(() => {
|
|
if (props.mode === 'router') {
|
|
const found = props.items.find((item) =>
|
|
doesMenuItemMatchCurrentRoute(item, currentRoute.value),
|
|
);
|
|
|
|
activeTab.value = found ? found.id : '';
|
|
} else {
|
|
activeTab.value = props.items.length > 0 ? props.items[0].id : '';
|
|
}
|
|
|
|
emit('update:modelValue', activeTab.value);
|
|
});
|
|
|
|
const onSelect = (item: IMenuItem): void => {
|
|
if (props.mode === 'tabs') {
|
|
activeTab.value = item.id;
|
|
}
|
|
|
|
emit('select', item.id);
|
|
emit('update:modelValue', item.id);
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss" module>
|
|
.container {
|
|
height: 100%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
background-color: var(--menu-background, var(--color-background-xlight));
|
|
}
|
|
|
|
.menuHeader {
|
|
display: flex;
|
|
flex-direction: column;
|
|
flex: 0 1 auto;
|
|
overflow-y: auto;
|
|
}
|
|
|
|
.menuContent {
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: space-between;
|
|
flex: 1 1 auto;
|
|
|
|
& > div > :global(.el-menu) {
|
|
background: none;
|
|
padding: var(--menu-padding, 12px);
|
|
}
|
|
}
|
|
|
|
.upperContent {
|
|
ul {
|
|
padding-top: 0 !important;
|
|
}
|
|
}
|
|
|
|
.lowerContent {
|
|
ul {
|
|
padding-bottom: 0 !important;
|
|
}
|
|
}
|
|
|
|
.menuCollapsed {
|
|
transition: width 150ms ease-in-out;
|
|
:global(.hideme) {
|
|
display: none !important;
|
|
}
|
|
}
|
|
|
|
.transparentBackground {
|
|
background-color: transparent;
|
|
}
|
|
</style>
|