mirror of
https://github.com/n8n-io/n8n.git
synced 2026-05-25 13:55:18 +02:00
105 lines
2.7 KiB
TypeScript
105 lines
2.7 KiB
TypeScript
import type { Locator, Page } from '@playwright/test';
|
|
import { expect } from '@playwright/test';
|
|
|
|
/**
|
|
* Node Creator component for adding nodes to workflows.
|
|
* Used within CanvasPage as `n8n.canvas.nodeCreator.*`
|
|
*
|
|
* @example
|
|
* // Access via canvas page
|
|
* await n8n.canvas.nodeCreator.open();
|
|
* await n8n.canvas.nodeCreator.searchFor('Gmail');
|
|
* await n8n.canvas.nodeCreator.selectItem('Gmail');
|
|
*/
|
|
export class NodeCreator {
|
|
constructor(private page: Page) {}
|
|
|
|
// Core locators
|
|
getRoot(): Locator {
|
|
return this.page.getByTestId('node-creator');
|
|
}
|
|
|
|
getSearchBar(): Locator {
|
|
return this.page.getByTestId('node-creator-search-bar');
|
|
}
|
|
|
|
getNodeItems(): Locator {
|
|
return this.page.getByTestId('item-iterator-item');
|
|
}
|
|
|
|
getCategoryItems(): Locator {
|
|
return this.page.getByTestId('node-creator-category-item');
|
|
}
|
|
|
|
getActiveSubcategory(): Locator {
|
|
return this.page.getByTestId('nodes-list-header').first();
|
|
}
|
|
|
|
getNoResults(): Locator {
|
|
return this.page.getByTestId('node-creator-no-results');
|
|
}
|
|
|
|
getNoTriggersCallout(): Locator {
|
|
return this.page.getByTestId('actions-panel-no-triggers-callout');
|
|
}
|
|
|
|
getActivationCallout(): Locator {
|
|
return this.page.getByTestId('actions-panel-activation-callout');
|
|
}
|
|
|
|
getTriggerText(): Locator {
|
|
return this.page.getByText('What triggers this workflow?');
|
|
}
|
|
|
|
getNextText(): Locator {
|
|
return this.page.getByText('What happens next?');
|
|
}
|
|
|
|
// Item getters
|
|
getItem(text: string, options: { exact?: boolean } = {}): Locator {
|
|
if (options.exact) {
|
|
return this.getNodeItems().filter({ has: this.page.getByText(text, { exact: true }) });
|
|
}
|
|
return this.getNodeItems().filter({ hasText: text }).first();
|
|
}
|
|
|
|
getCategoryItem(text: string): Locator {
|
|
return this.getCategoryItems().filter({ hasText: text });
|
|
}
|
|
|
|
// Actions
|
|
async open(): Promise<void> {
|
|
await this.page.getByTestId('node-creator-plus-button').click();
|
|
await expect(this.getRoot()).toBeVisible();
|
|
}
|
|
|
|
async close(): Promise<void> {
|
|
await this.page.keyboard.press('Escape');
|
|
}
|
|
|
|
async searchFor(text: string): Promise<void> {
|
|
await this.getSearchBar().fill(text);
|
|
}
|
|
|
|
async clearSearch(): Promise<void> {
|
|
await this.getSearchBar().clear();
|
|
}
|
|
|
|
async selectItem(text: string): Promise<void> {
|
|
await this.getItem(text).click();
|
|
}
|
|
|
|
async selectCategoryItem(text: string): Promise<void> {
|
|
await this.getCategoryItem(text).click();
|
|
}
|
|
|
|
async navigateToSubcategory(category: string, options: { exact?: boolean } = {}): Promise<void> {
|
|
await this.getItem(category, options).click();
|
|
await expect(this.getActiveSubcategory()).toContainText(category);
|
|
}
|
|
|
|
async goBackFromSubcategory(): Promise<void> {
|
|
await this.getActiveSubcategory().locator('button').click();
|
|
}
|
|
}
|