import { expect } from '@playwright/test' import type { Palette } from '../src/types/colorPaletteTypes' import { comfyPageFixture as test } from './fixtures/ComfyPage' const customColorPalettes: Record = { obsidian: { version: 102, id: 'obsidian', name: 'Obsidian', colors: { node_slot: { CLIP: '#FFD500', CLIP_VISION: '#A8DADC', CLIP_VISION_OUTPUT: '#ad7452', CONDITIONING: '#FFA931', CONTROL_NET: '#6EE7B7', IMAGE: '#64B5F6', LATENT: '#FF9CF9', MASK: '#81C784', MODEL: '#B39DDB', STYLE_MODEL: '#C2FFAE', VAE: '#FF6E6E', TAESD: '#DCC274', PIPE_LINE: '#7737AA', PIPE_LINE_SDXL: '#7737AA', INT: '#29699C', XYPLOT: '#74DA5D', X_Y: '#38291f' }, litegraph_base: { BACKGROUND_IMAGE: '', CLEAR_BACKGROUND_COLOR: '#222222', NODE_TITLE_COLOR: 'rgba(255,255,255,.75)', NODE_SELECTED_TITLE_COLOR: '#FFF', NODE_TEXT_SIZE: 14, NODE_TEXT_COLOR: '#b8b8b8', NODE_SUBTEXT_SIZE: 12, NODE_DEFAULT_COLOR: 'rgba(0,0,0,.8)', NODE_DEFAULT_BGCOLOR: 'rgba(22,22,22,.8)', NODE_DEFAULT_BOXCOLOR: 'rgba(255,255,255,.75)', NODE_DEFAULT_SHAPE: 'box', NODE_BOX_OUTLINE_COLOR: '#236692', DEFAULT_SHADOW_COLOR: 'rgba(0,0,0,0)', DEFAULT_GROUP_FONT: 24, WIDGET_BGCOLOR: '#242424', WIDGET_OUTLINE_COLOR: '#333', WIDGET_TEXT_COLOR: '#a3a3a8', WIDGET_SECONDARY_TEXT_COLOR: '#97979c', LINK_COLOR: '#9A9', EVENT_LINK_COLOR: '#A86', CONNECTING_LINK_COLOR: '#AFA' }, comfy_base: { 'fg-color': '#fff', 'bg-color': '#242424', 'comfy-menu-bg': 'rgba(24,24,24,.9)', 'comfy-input-bg': '#262626', 'input-text': '#ddd', 'descrip-text': '#999', 'drag-text': '#ccc', 'error-text': '#ff4444', 'border-color': '#29292c', 'tr-even-bg-color': 'rgba(28,28,28,.9)', 'tr-odd-bg-color': 'rgba(19,19,19,.9)' } } }, obsidian_dark: { version: 102, id: 'obsidian_dark', name: 'Obsidian Dark', colors: { node_slot: { CLIP: '#FFD500', CLIP_VISION: '#A8DADC', CLIP_VISION_OUTPUT: '#ad7452', CONDITIONING: '#FFA931', CONTROL_NET: '#6EE7B7', IMAGE: '#64B5F6', LATENT: '#FF9CF9', MASK: '#81C784', MODEL: '#B39DDB', STYLE_MODEL: '#C2FFAE', VAE: '#FF6E6E', TAESD: '#DCC274', PIPE_LINE: '#7737AA', PIPE_LINE_SDXL: '#7737AA', INT: '#29699C', XYPLOT: '#74DA5D', X_Y: '#38291f' }, litegraph_base: { BACKGROUND_IMAGE: '', CLEAR_BACKGROUND_COLOR: '#000', NODE_TITLE_COLOR: 'rgba(255,255,255,.75)', NODE_SELECTED_TITLE_COLOR: '#FFF', NODE_TEXT_SIZE: 14, NODE_TEXT_COLOR: '#b8b8b8', NODE_SUBTEXT_SIZE: 12, NODE_DEFAULT_COLOR: 'rgba(0,0,0,.8)', NODE_DEFAULT_BGCOLOR: 'rgba(22,22,22,.8)', NODE_DEFAULT_BOXCOLOR: 'rgba(255,255,255,.75)', NODE_DEFAULT_SHAPE: 'box', NODE_BOX_OUTLINE_COLOR: '#236692', DEFAULT_SHADOW_COLOR: 'rgba(0,0,0,0)', DEFAULT_GROUP_FONT: 24, WIDGET_BGCOLOR: '#242424', WIDGET_OUTLINE_COLOR: '#333', WIDGET_TEXT_COLOR: '#a3a3a8', WIDGET_SECONDARY_TEXT_COLOR: '#97979c', LINK_COLOR: '#9A9', EVENT_LINK_COLOR: '#A86', CONNECTING_LINK_COLOR: '#AFA' }, comfy_base: { 'fg-color': '#fff', 'bg-color': '#242424', 'comfy-menu-bg': 'rgba(24,24,24,.9)', 'comfy-input-bg': '#262626', 'input-text': '#ddd', 'descrip-text': '#999', 'drag-text': '#ccc', 'error-text': '#ff4444', 'border-color': '#29292c', 'tr-even-bg-color': 'rgba(28,28,28,.9)', 'tr-odd-bg-color': 'rgba(19,19,19,.9)' } } }, // A custom light theme with fg color red light_red: { id: 'light_red', name: 'Light Red', light_theme: true, colors: { node_slot: {}, litegraph_base: {}, comfy_base: { 'fg-color': '#ff0000' } } } } test.describe('Color Palette', () => { test('Can show custom color palette', async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.CustomColorPalettes', customColorPalettes) // Reload to apply the new setting. Setting Comfy.CustomColorPalettes directly // doesn't update the store immediately. await comfyPage.setup() await comfyPage.setSetting('Comfy.ColorPalette', 'obsidian_dark') await expect(comfyPage.canvas).toHaveScreenshot( 'custom-color-palette-obsidian-dark.png' ) await comfyPage.setSetting('Comfy.ColorPalette', 'light_red') await comfyPage.nextFrame() await expect(comfyPage.canvas).toHaveScreenshot( 'custom-color-palette-light-red.png' ) await comfyPage.setSetting('Comfy.ColorPalette', 'dark') await comfyPage.nextFrame() await expect(comfyPage.canvas).toHaveScreenshot('default-color-palette.png') }) test('Can add custom color palette', async ({ comfyPage }) => { await comfyPage.page.evaluate((p) => { window['app'].extensionManager.colorPalette.addCustomColorPalette(p) }, customColorPalettes.obsidian_dark) expect(await comfyPage.getToastErrorCount()).toBe(0) await comfyPage.setSetting('Comfy.ColorPalette', 'obsidian_dark') await comfyPage.nextFrame() await expect(comfyPage.canvas).toHaveScreenshot( 'custom-color-palette-obsidian-dark.png' ) // Legacy `custom_` prefix is still supported await comfyPage.setSetting('Comfy.ColorPalette', 'custom_obsidian_dark') await comfyPage.nextFrame() await expect(comfyPage.canvas).toHaveScreenshot( 'custom-color-palette-obsidian-dark.png' ) }) }) test.describe('Node Color Adjustments', () => { test.beforeEach(async ({ comfyPage }) => { await comfyPage.loadWorkflow('every_node_color') }) test('should adjust opacity via node opacity setting', async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.Node.Opacity', 0.5) await comfyPage.page.waitForTimeout(128) // Drag mouse to force canvas to redraw await comfyPage.page.mouse.move(0, 0) await expect(comfyPage.canvas).toHaveScreenshot('node-opacity-0.5.png') await comfyPage.setSetting('Comfy.Node.Opacity', 1.0) await comfyPage.page.waitForTimeout(128) await comfyPage.page.mouse.move(8, 8) await expect(comfyPage.canvas).toHaveScreenshot('node-opacity-1.png') }) test('should persist color adjustments when changing themes', async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.Node.Opacity', 0.2) await comfyPage.setSetting('Comfy.ColorPalette', 'arc') await comfyPage.nextFrame() await comfyPage.page.mouse.move(0, 0) await expect(comfyPage.canvas).toHaveScreenshot( 'node-opacity-0.2-arc-theme.png' ) }) test('should not serialize color adjustments in workflow', async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.Node.Opacity', 0.5) await comfyPage.setSetting('Comfy.ColorPalette', 'light') const saveWorkflowInterval = 1000 await comfyPage.page.waitForTimeout(saveWorkflowInterval) const workflow = await comfyPage.page.evaluate(() => { return localStorage.getItem('workflow') }) for (const node of JSON.parse(workflow).nodes) { if (node.bgcolor) expect(node.bgcolor).not.toMatch(/hsla/) if (node.color) expect(node.color).not.toMatch(/hsla/) } }) test('should lighten node colors when switching to light theme', async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.ColorPalette', 'light') await comfyPage.nextFrame() await expect(comfyPage.canvas).toHaveScreenshot('node-lightened-colors.png') }) test.describe('Context menu color adjustments', () => { test.beforeEach(async ({ comfyPage }) => { await comfyPage.setSetting('Comfy.ColorPalette', 'light') await comfyPage.setSetting('Comfy.Node.Opacity', 0.3) const node = await comfyPage.getFirstNodeRef() await node?.clickContextMenuOption('Colors') }) test('should persist color adjustments when changing custom node colors', async ({ comfyPage }) => { await comfyPage.page .locator('.litemenu-entry.submenu span:has-text("red")') .click() await expect(comfyPage.canvas).toHaveScreenshot( 'node-opacity-0.3-color-changed.png' ) }) test('should persist color adjustments when removing custom node color', async ({ comfyPage }) => { await comfyPage.page .locator('.litemenu-entry.submenu span:has-text("No color")') .click() await expect(comfyPage.canvas).toHaveScreenshot( 'node-opacity-0.3-color-removed.png' ) }) }) })