mirror of
https://github.com/Comfy-Org/ComfyUI_frontend.git
synced 2026-02-03 22:59:14 +00:00
## Summary Reorganized custom nodes manager functionality from scattered technical layers into a cohesive domain-focused module following [domain-driven design](https://en.wikipedia.org/wiki/Domain-driven_design) principles. ## Changes - **What**: Migrated all manager code from technical layers (`src/components/`, `src/stores/`, etc.) to unified domain structure at `src/workbench/extensions/manager/` - **Breaking**: Import paths changed for all manager-related modules (40+ files updated) ## Review Focus Verify all import path updates are correct and no circular dependencies introduced. Check that [Vue 3 composition API](https://vuejs.org/guide/reusability/composables.html) patterns remain consistent across relocated composables. ┆Issue is synchronized with this [Notion page](https://www.notion.so/PR-5662-refactor-Migrate-manager-code-to-DDD-structure-2736d73d3650812c87faf6ed0fffb196) by [Unito](https://www.unito.io)
323 lines
8.6 KiB
Vue
323 lines
8.6 KiB
Vue
<template>
|
|
<div
|
|
class="comfyui-logo-wrapper p-1 flex justify-center items-center cursor-pointer rounded-md mr-2"
|
|
:class="{
|
|
'comfyui-logo-menu-visible': menuRef?.visible
|
|
}"
|
|
:style="{
|
|
minWidth: isLargeSidebar ? '4rem' : 'auto'
|
|
}"
|
|
@click="menuRef?.toggle($event)"
|
|
>
|
|
<img
|
|
src="/assets/images/comfy-logo-mono.svg"
|
|
alt="ComfyUI Logo"
|
|
class="comfyui-logo h-7"
|
|
@contextmenu="showNativeSystemMenu"
|
|
/>
|
|
<i class="pi pi-angle-down ml-1 text-[10px]" />
|
|
</div>
|
|
<TieredMenu
|
|
ref="menuRef"
|
|
:model="translatedItems"
|
|
:popup="true"
|
|
class="comfy-command-menu"
|
|
:class="{
|
|
'comfy-command-menu-top': isTopMenu
|
|
}"
|
|
@show="onMenuShow"
|
|
>
|
|
<template #item="{ item, props }">
|
|
<a
|
|
class="p-menubar-item-link px-4 py-2"
|
|
v-bind="props.action"
|
|
:href="item.url"
|
|
target="_blank"
|
|
:class="typeof item.class === 'function' ? item.class() : item.class"
|
|
@mousedown="
|
|
isZoomCommand(item) ? handleZoomMouseDown(item, $event) : undefined
|
|
"
|
|
@click="handleItemClick(item, $event)"
|
|
>
|
|
<i
|
|
v-if="hasActiveStateSiblings(item)"
|
|
class="p-menubar-item-icon pi pi-check text-sm"
|
|
:class="{ invisible: !item.comfyCommand?.active?.() }"
|
|
/>
|
|
<span
|
|
v-else-if="
|
|
item.icon && item.comfyCommand?.id !== 'Comfy.NewBlankWorkflow'
|
|
"
|
|
class="p-menubar-item-icon"
|
|
:class="item.icon"
|
|
/>
|
|
<span class="p-menubar-item-label text-nowrap">{{ item.label }}</span>
|
|
<i
|
|
v-if="item.comfyCommand?.id === 'Comfy.NewBlankWorkflow'"
|
|
class="ml-auto"
|
|
:class="item.icon"
|
|
/>
|
|
<span
|
|
v-if="item?.comfyCommand?.keybinding"
|
|
class="ml-auto border border-surface rounded text-muted text-xs text-nowrap p-1 keybinding-tag"
|
|
>
|
|
{{ item.comfyCommand.keybinding.combo.toString() }}
|
|
</span>
|
|
<i v-if="item.items" class="ml-auto pi pi-angle-right" />
|
|
</a>
|
|
</template>
|
|
</TieredMenu>
|
|
|
|
<SubgraphBreadcrumb />
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import type { MenuItem } from 'primevue/menuitem'
|
|
import TieredMenu, {
|
|
type TieredMenuMethods,
|
|
type TieredMenuState
|
|
} from 'primevue/tieredmenu'
|
|
import { computed, nextTick, ref } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
|
|
import SubgraphBreadcrumb from '@/components/breadcrumb/SubgraphBreadcrumb.vue'
|
|
import SettingDialogHeader from '@/components/dialog/header/SettingDialogHeader.vue'
|
|
import SettingDialogContent from '@/platform/settings/components/SettingDialogContent.vue'
|
|
import { useSettingStore } from '@/platform/settings/settingStore'
|
|
import { useColorPaletteService } from '@/services/colorPaletteService'
|
|
import { useCommandStore } from '@/stores/commandStore'
|
|
import { useDialogStore } from '@/stores/dialogStore'
|
|
import { useMenuItemStore } from '@/stores/menuItemStore'
|
|
import { useColorPaletteStore } from '@/stores/workspace/colorPaletteStore'
|
|
import { showNativeSystemMenu } from '@/utils/envUtil'
|
|
import { normalizeI18nKey } from '@/utils/formatUtil'
|
|
import { whileMouseDown } from '@/utils/mouseDownUtil'
|
|
import { useManagerState } from '@/workbench/extensions/manager/composables/useManagerState'
|
|
import { ManagerTab } from '@/workbench/extensions/manager/types/comfyManagerTypes'
|
|
|
|
const colorPaletteStore = useColorPaletteStore()
|
|
const colorPaletteService = useColorPaletteService()
|
|
const menuItemsStore = useMenuItemStore()
|
|
const commandStore = useCommandStore()
|
|
const dialogStore = useDialogStore()
|
|
const settingStore = useSettingStore()
|
|
const { t } = useI18n()
|
|
|
|
const managerState = useManagerState()
|
|
|
|
const menuRef = ref<
|
|
({ dirty: boolean } & TieredMenuMethods & TieredMenuState) | null
|
|
>(null)
|
|
const isLargeSidebar = computed(
|
|
() => settingStore.get('Comfy.Sidebar.Size') !== 'small'
|
|
)
|
|
const isTopMenu = computed(() => settingStore.get('Comfy.UseNewMenu') === 'Top')
|
|
|
|
const translateMenuItem = (item: MenuItem): MenuItem => {
|
|
const label = typeof item.label === 'function' ? item.label() : item.label
|
|
const translatedLabel = label
|
|
? t(`menuLabels.${normalizeI18nKey(label)}`, label)
|
|
: undefined
|
|
|
|
return {
|
|
...item,
|
|
label: translatedLabel,
|
|
items: item.items?.map(translateMenuItem)
|
|
}
|
|
}
|
|
|
|
const showSettings = (defaultPanel?: string) => {
|
|
dialogStore.showDialog({
|
|
key: 'global-settings',
|
|
headerComponent: SettingDialogHeader,
|
|
component: SettingDialogContent,
|
|
props: {
|
|
defaultPanel
|
|
}
|
|
})
|
|
}
|
|
|
|
const showManageExtensions = async () => {
|
|
await managerState.openManager({
|
|
initialTab: ManagerTab.All,
|
|
showToastOnLegacyError: false
|
|
})
|
|
}
|
|
|
|
const themeMenuItems = computed(() => {
|
|
return colorPaletteStore.palettes.map((palette) => ({
|
|
key: `theme-${palette.id}`,
|
|
label: palette.name,
|
|
parentPath: 'theme',
|
|
comfyCommand: {
|
|
active: () => colorPaletteStore.activePaletteId === palette.id
|
|
},
|
|
command: async () => {
|
|
await colorPaletteService.loadColorPalette(palette.id)
|
|
}
|
|
}))
|
|
})
|
|
|
|
const extraMenuItems = computed(() => [
|
|
{ separator: true },
|
|
{
|
|
key: 'theme',
|
|
label: t('menu.theme'),
|
|
items: themeMenuItems.value
|
|
},
|
|
{ separator: true },
|
|
{
|
|
key: 'browse-templates',
|
|
label: t('menuLabels.Browse Templates'),
|
|
icon: 'icon-[comfy--template]',
|
|
command: () => commandStore.execute('Comfy.BrowseTemplates')
|
|
},
|
|
{
|
|
key: 'settings',
|
|
label: t('g.settings'),
|
|
icon: 'mdi mdi-cog-outline',
|
|
command: () => showSettings()
|
|
},
|
|
{
|
|
key: 'manage-extensions',
|
|
label: t('menu.manageExtensions'),
|
|
icon: 'mdi mdi-puzzle-outline',
|
|
command: showManageExtensions
|
|
}
|
|
])
|
|
|
|
const translatedItems = computed(() => {
|
|
const items = menuItemsStore.menuItems.map(translateMenuItem)
|
|
let helpIndex = items.findIndex((item) => item.key === 'Help')
|
|
let helpItem: MenuItem | undefined
|
|
|
|
if (helpIndex !== -1) {
|
|
items[helpIndex].icon = 'mdi mdi-help-circle-outline'
|
|
// If help is not the last item (i.e. we have extension commands), separate them
|
|
const isLastItem = helpIndex !== items.length - 1
|
|
helpItem = items.splice(
|
|
helpIndex,
|
|
1,
|
|
...(isLastItem
|
|
? [
|
|
{
|
|
separator: true
|
|
}
|
|
]
|
|
: [])
|
|
)[0]
|
|
}
|
|
helpIndex = items.length
|
|
|
|
items.splice(
|
|
helpIndex,
|
|
0,
|
|
...extraMenuItems.value,
|
|
...(helpItem
|
|
? [
|
|
{
|
|
separator: true
|
|
},
|
|
helpItem
|
|
]
|
|
: [])
|
|
)
|
|
|
|
return items
|
|
})
|
|
|
|
const onMenuShow = () => {
|
|
void nextTick(() => {
|
|
// Force the menu to show submenus on hover
|
|
if (menuRef.value) {
|
|
menuRef.value.dirty = true
|
|
}
|
|
})
|
|
}
|
|
|
|
const isZoomCommand = (item: MenuItem) => {
|
|
return (
|
|
item.comfyCommand?.id === 'Comfy.Canvas.ZoomIn' ||
|
|
item.comfyCommand?.id === 'Comfy.Canvas.ZoomOut'
|
|
)
|
|
}
|
|
|
|
const handleZoomMouseDown = (item: MenuItem, event: MouseEvent) => {
|
|
if (item.comfyCommand) {
|
|
whileMouseDown(
|
|
event,
|
|
async () => {
|
|
await commandStore.execute(item.comfyCommand!.id)
|
|
},
|
|
50
|
|
)
|
|
}
|
|
}
|
|
|
|
const handleItemClick = (item: MenuItem, event: MouseEvent) => {
|
|
// Prevent the menu from closing for zoom commands or commands that have active state
|
|
if (isZoomCommand(item) || item.comfyCommand?.active) {
|
|
event.preventDefault()
|
|
event.stopPropagation()
|
|
if (item.comfyCommand?.active) {
|
|
item.command?.({
|
|
item,
|
|
originalEvent: event
|
|
})
|
|
}
|
|
return false
|
|
}
|
|
}
|
|
|
|
const hasActiveStateSiblings = (item: MenuItem): boolean => {
|
|
// Check if this item has siblings with active state (either from store or theme items)
|
|
return (
|
|
item.parentPath &&
|
|
(item.parentPath === 'theme' ||
|
|
menuItemsStore.menuItemHasActiveStateChildren[item.parentPath])
|
|
)
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
@reference '../../assets/css/style.css';
|
|
|
|
:deep(.p-menubar-submenu.dropdown-direction-up) {
|
|
@apply top-auto bottom-full flex-col-reverse;
|
|
}
|
|
|
|
.keybinding-tag {
|
|
background: var(--p-content-hover-background);
|
|
border-color: var(--p-content-border-color);
|
|
border-style: solid;
|
|
}
|
|
|
|
.comfyui-logo-menu-visible,
|
|
.comfyui-logo-wrapper:hover {
|
|
background-color: color-mix(in srgb, var(--fg-color) 10%, transparent);
|
|
}
|
|
</style>
|
|
|
|
<style>
|
|
.comfy-command-menu {
|
|
--p-tieredmenu-item-focus-background: color-mix(
|
|
in srgb,
|
|
var(--fg-color) 15%,
|
|
transparent
|
|
);
|
|
--p-tieredmenu-item-active-background: color-mix(
|
|
in srgb,
|
|
var(--fg-color) 10%,
|
|
transparent
|
|
);
|
|
}
|
|
.comfy-command-menu ul {
|
|
background-color: var(--comfy-menu-secondary-bg) !important;
|
|
}
|
|
|
|
.comfy-command-menu-top .p-tieredmenu-submenu {
|
|
left: calc(100% + 15px) !important;
|
|
top: -4px !important;
|
|
}
|
|
</style>
|