@nextsparkjs/testing 0.1.0-beta.53 → 0.1.0-beta.55
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/dist/index.d.ts +1 -1
- package/dist/index.js +10 -1
- package/dist/index.js.map +1 -1
- package/dist/utils/index.d.ts +25 -2
- package/dist/utils/index.js +10 -1
- package/dist/utils/index.js.map +1 -1
- package/package.json +7 -7
package/LICENSE
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2025 NextSpark
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/dist/index.d.ts
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
export { createAriaLabel, createPriorityAttr, createStateAttr, createTestingProps, keyboardHelpers } from './utils/index.js';
|
|
1
|
+
export { createAriaLabel, createCyId, createPriorityAttr, createStateAttr, createTestId, createTestingProps, keyboardHelpers, sel } from './utils/index.js';
|
|
2
2
|
export { BasePOMCore, DashboardEntityPOMCore, EntityConfig } from './pom/index.js';
|
|
3
3
|
export { ApiInterceptor, ApiInterceptorConfig } from './helpers/index.js';
|
package/dist/index.js
CHANGED
|
@@ -1,4 +1,13 @@
|
|
|
1
1
|
// src/utils/utils.ts
|
|
2
|
+
function createCyId(...parts) {
|
|
3
|
+
return parts.filter(Boolean).join("-");
|
|
4
|
+
}
|
|
5
|
+
function createTestId(...parts) {
|
|
6
|
+
return parts.filter(Boolean).join("-");
|
|
7
|
+
}
|
|
8
|
+
function sel(...parts) {
|
|
9
|
+
return parts.filter(Boolean).join(".");
|
|
10
|
+
}
|
|
2
11
|
var isDevelopment = process.env.NODE_ENV === "development";
|
|
3
12
|
var isTest = process.env.NODE_ENV === "test";
|
|
4
13
|
var enableTestingAttributes = isDevelopment || isTest;
|
|
@@ -886,6 +895,6 @@ var DashboardEntityPOMCore = class extends BasePOMCore {
|
|
|
886
895
|
}
|
|
887
896
|
};
|
|
888
897
|
|
|
889
|
-
export { ApiInterceptor, BasePOMCore, DashboardEntityPOMCore, createAriaLabel, createPriorityAttr, createStateAttr, createTestingProps, keyboardHelpers };
|
|
898
|
+
export { ApiInterceptor, BasePOMCore, DashboardEntityPOMCore, createAriaLabel, createCyId, createPriorityAttr, createStateAttr, createTestId, createTestingProps, keyboardHelpers, sel };
|
|
890
899
|
//# sourceMappingURL=index.js.map
|
|
891
900
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/utils/utils.ts","../src/pom/BasePOMCore.ts","../src/helpers/ApiInterceptor.ts","../src/pom/DashboardEntityPOMCore.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Testing Utilities\n * Helper functions for consistent testing attribute generation\n *\n * NOTE: For data-cy selectors, use sel() from '@nextsparkjs/core/selectors'\n * This file contains runtime utilities for accessibility and keyboard handling.\n */\n\n// =============================================================================\n// ENVIRONMENT DETECTION\n// =============================================================================\n\n/**\n * Environment-based testing mode\n */\nconst isDevelopment = process.env.NODE_ENV === 'development'\nconst isTest = process.env.NODE_ENV === 'test'\nconst enableTestingAttributes = isDevelopment || isTest\n\n// =============================================================================\n// ATTRIBUTE GENERATORS\n// =============================================================================\n\n/**\n * Create state-based data attributes for conditional testing\n *\n * @param state - Current state value\n * @returns State attribute value\n */\nexport function createStateAttr(\n state: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n): string {\n return state\n}\n\n/**\n * Create priority-based data attributes\n *\n * @param priority - Priority level\n * @returns Priority attribute value\n */\nexport function createPriorityAttr(priority: 'low' | 'medium' | 'high'): string {\n return priority\n}\n\n/**\n * Generate testing props object for components\n *\n * @param config - Testing configuration\n * @returns Testing props object\n */\nexport function createTestingProps(config: {\n testId?: string\n cyId?: string\n state?: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n priority?: 'low' | 'medium' | 'high'\n taskId?: string\n userId?: string\n}) {\n const props: Record<string, string | undefined> = {}\n\n if (config.testId) {\n props['data-testid'] = enableTestingAttributes ? config.testId : undefined\n }\n\n if (config.cyId) {\n props['data-cy'] = enableTestingAttributes ? config.cyId : undefined\n }\n\n if (config.state) {\n props['data-state'] = config.state\n }\n\n if (config.priority) {\n props['data-priority'] = config.priority\n }\n\n if (config.taskId) {\n props['data-task-id'] = config.taskId\n }\n\n if (config.userId) {\n props['data-user-id'] = config.userId\n }\n\n // Filter out undefined values\n return Object.fromEntries(\n Object.entries(props).filter((entry) => entry[1] !== undefined)\n )\n}\n\n/**\n * Accessibility helper for dynamic aria-label generation\n *\n * @param template - Label template with placeholders\n * @param values - Values to replace placeholders\n * @returns Formatted aria-label\n */\nexport function createAriaLabel(\n template: string,\n values: Record<string, string | number | boolean>\n): string {\n return template.replace(/\\{(\\w+)\\}/g, (match, key) => {\n return String(values[key] ?? match)\n })\n}\n\n// =============================================================================\n// KEYBOARD HELPERS\n// =============================================================================\n\n/**\n * Keyboard navigation helpers\n */\nexport const keyboardHelpers = {\n /**\n * Handle Enter and Space key activation\n */\n createActivationHandler: (onActivate: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n onActivate()\n }\n }\n },\n\n /**\n * Handle Escape key for closing\n */\n createEscapeHandler: (onClose: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Escape') {\n e.preventDefault()\n onClose()\n }\n }\n },\n\n /**\n * Handle arrow navigation in lists\n */\n createArrowNavigationHandler: (\n currentIndex: number,\n maxIndex: number,\n onIndexChange: (index: number) => void\n ) => {\n return (e: React.KeyboardEvent) => {\n switch (e.key) {\n case 'ArrowDown':\n e.preventDefault()\n onIndexChange(currentIndex < maxIndex ? currentIndex + 1 : 0)\n break\n case 'ArrowUp':\n e.preventDefault()\n onIndexChange(currentIndex > 0 ? currentIndex - 1 : maxIndex)\n break\n }\n }\n },\n}\n","/**\n * BasePOMCore - Base class with common utilities for all Page Object Models\n *\n * This is the CORE version that provides generic methods.\n * Themes should extend this class and implement the abstract `cySelector` method\n * to provide theme-specific selector resolution.\n *\n * @example Theme extension:\n * ```ts\n * // theme/tests/cypress/src/core/BasePOM.ts\n * import { BasePOMCore } from '@nextsparkjs/testing/pom'\n * import { cySelector } from '../selectors'\n *\n * export abstract class BasePOM extends BasePOMCore {\n * protected cySelector(path: string, replacements?: Record<string, string>): string {\n * return cySelector(path, replacements)\n * }\n * }\n * ```\n */\n\n/**\n * Type for placeholder replacements in selectors\n * Mirrors @nextsparkjs/core/selectors Replacements type\n */\nexport type Replacements = Record<string, string | number>\n\nexport abstract class BasePOMCore {\n /**\n * Abstract method - themes must implement to provide their cySelector\n * This allows themes to use their extended THEME_SELECTORS\n *\n * @param path - Dot-notation path to the selector (e.g., \"auth.login.form\")\n * @param replacements - Optional placeholder replacements\n * @returns Cypress selector string like [data-cy=\"selector-value\"]\n */\n protected abstract cySelector(path: string, replacements?: Replacements): string\n\n /**\n * Get a Cypress selector using the centralized selectors\n * Wrapper for cySelector with a shorter name\n *\n * @example\n * this.cy('auth.login.form')\n * // Returns: '[data-cy=\"login-form\"]'\n *\n * this.cy('entities.table.row', { slug: 'tasks', id: '123' })\n * // Returns: '[data-cy=\"tasks-row-123\"]'\n */\n protected cy(path: string, replacements?: Replacements): string {\n return this.cySelector(path, replacements)\n }\n\n /**\n * Replaces placeholders in a selector pattern and wraps with data-cy attribute\n *\n * @param pattern - Selector pattern with {placeholder} syntax\n * @param replacements - Object with placeholder values\n * @returns Formatted data-cy selector string\n *\n * @example\n * selector('{slug}-row-{id}', { slug: 'tasks', id: '123' })\n * // Returns: '[data-cy=\"tasks-row-123\"]'\n */\n protected selector(pattern: string, replacements: Replacements = {}): string {\n let result = pattern\n for (const [key, value] of Object.entries(replacements)) {\n result = result.replaceAll(`{${key}}`, String(value))\n }\n return `[data-cy=\"${result}\"]`\n }\n\n /**\n * Wrapper for cy.get with selector pattern support\n * @param pattern - Selector pattern or direct selector\n * @param replacements - Optional placeholder replacements\n */\n protected get(pattern: string, replacements: Replacements = {}) {\n return cy.get(this.selector(pattern, replacements))\n }\n\n /**\n * Generic wait with configurable timeout\n * @param selector - CSS selector to wait for\n * @param timeout - Max wait time in ms (default: 15000)\n */\n protected waitFor(selector: string, timeout = 15000) {\n cy.get(selector, { timeout }).should('be.visible')\n return this\n }\n\n /**\n * Wait for URL to contain a specific path\n * @param path - Path segment to check for\n */\n protected waitForUrl(path: string) {\n cy.url().should('include', path)\n return this\n }\n\n /**\n * Wait for URL to match a regex pattern\n * @param pattern - RegExp to match against URL\n */\n protected waitForUrlMatch(pattern: RegExp) {\n cy.url().should('match', pattern)\n return this\n }\n\n /**\n * Visit a URL and return self for chaining\n * @param url - URL to visit\n */\n visit(url: string) {\n cy.visit(url)\n return this\n }\n\n /**\n * Wait for page to load (checks for body visible)\n */\n waitForPageLoad() {\n cy.get('body').should('be.visible')\n return this\n }\n\n /**\n * Get an element by selector\n * @param selector - CSS selector\n */\n getElement(selector: string) {\n return cy.get(selector)\n }\n\n /**\n * Click on an element\n * @param selector - CSS selector\n */\n click(selector: string) {\n cy.get(selector).click()\n return this\n }\n\n /**\n * Type text into an input\n * @param selector - CSS selector\n * @param text - Text to type\n */\n type(selector: string, text: string) {\n cy.get(selector).clear().type(text)\n return this\n }\n\n /**\n * Check if element exists\n * @param selector - CSS selector\n */\n exists(selector: string) {\n return cy.get(selector).should('exist')\n }\n\n /**\n * Check if element is visible\n * @param selector - CSS selector\n */\n isVisible(selector: string) {\n return cy.get(selector).should('be.visible')\n }\n\n /**\n * Check if element does not exist\n * @param selector - CSS selector\n */\n notExists(selector: string) {\n return cy.get(selector).should('not.exist')\n }\n}\n\nexport default BasePOMCore\n","/**\n * ApiInterceptor - Helper for deterministic waits in Cypress\n *\n * Replaces unreliable cy.wait(ms) with cy.intercept() based waits\n * that wait for actual API responses.\n *\n * @example Basic usage:\n * ```ts\n * const api = new ApiInterceptor('customers')\n * api.setupCrudIntercepts()\n * cy.visit('/dashboard/customers')\n * api.waitForList()\n * ```\n *\n * @example Custom path:\n * ```ts\n * const api = new ApiInterceptor({\n * slug: 'categories',\n * customPath: '/api/v1/post-categories'\n * })\n * ```\n */\n\nexport interface ApiInterceptorConfig {\n /** Entity slug - used to generate aliases */\n slug: string\n /** Custom API path (e.g., '/api/v1/post-categories') */\n customPath?: string\n}\n\nexport class ApiInterceptor {\n private slug: string\n private endpoint: string\n\n constructor(slugOrConfig: string | ApiInterceptorConfig) {\n if (typeof slugOrConfig === 'string') {\n this.slug = slugOrConfig\n this.endpoint = `/api/v1/${slugOrConfig}`\n } else {\n this.slug = slugOrConfig.slug\n this.endpoint = slugOrConfig.customPath || `/api/v1/${slugOrConfig.slug}`\n }\n }\n\n // ============================================\n // ACCESSORS\n // ============================================\n\n /** Get the API endpoint path */\n get path(): string {\n return this.endpoint\n }\n\n /** Get the entity slug */\n get entitySlug(): string {\n return this.slug\n }\n\n /** Get alias names for all operations */\n get aliases() {\n return {\n list: `${this.slug}List`,\n create: `${this.slug}Create`,\n update: `${this.slug}Update`,\n delete: `${this.slug}Delete`,\n }\n }\n\n // ============================================\n // INTERCEPT SETUP\n // ============================================\n\n /**\n * Setup intercepts for all CRUD operations\n * Call this BEFORE navigation in beforeEach or at test start\n *\n * Note: We intercept both PUT and PATCH for updates since different\n * APIs may use different HTTP methods for updates.\n */\n setupCrudIntercepts(): this {\n cy.intercept('GET', `${this.endpoint}*`).as(this.aliases.list)\n cy.intercept('POST', this.endpoint).as(this.aliases.create)\n // Intercept both PUT and PATCH for updates (APIs may use either)\n cy.intercept('PUT', `${this.endpoint}/*`).as(this.aliases.update)\n cy.intercept('PATCH', `${this.endpoint}/*`).as(`${this.aliases.update}Patch`)\n cy.intercept('DELETE', `${this.endpoint}/*`).as(this.aliases.delete)\n return this\n }\n\n /**\n * Setup only list + create intercepts\n * Useful for list pages with inline create\n */\n setupListIntercepts(): this {\n cy.intercept('GET', `${this.endpoint}*`).as(this.aliases.list)\n cy.intercept('POST', this.endpoint).as(this.aliases.create)\n return this\n }\n\n // ============================================\n // WAIT METHODS\n // ============================================\n\n /**\n * Wait for list response (GET)\n * Use after navigation or after mutations to wait for refresh\n */\n waitForList(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.list}`, { timeout })\n }\n\n /**\n * Wait for create response (POST) and validate success status\n */\n waitForCreate(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.create}`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 201])\n }\n\n /**\n * Wait for update response (PATCH or PUT) and validate success status\n * Waits for PATCH first (more common), falls back to PUT\n */\n waitForUpdate(timeout = 10000): Cypress.Chainable {\n // Try PATCH first (more common in modern APIs), fall back to PUT\n return cy.wait(`@${this.aliases.update}Patch`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 201])\n }\n\n /**\n * Wait for delete response (DELETE) and validate success status\n */\n waitForDelete(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.delete}`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 204])\n }\n\n // ============================================\n // CONVENIENCE METHODS\n // ============================================\n\n /**\n * Wait for list refresh (alias for waitForList)\n * Semantic name for use after create/update/delete\n */\n waitForRefresh(timeout = 10000): Cypress.Chainable {\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for create + list refresh\n * Common pattern: create entity, wait for success, wait for list to refresh\n */\n waitForCreateAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForCreate(timeout)\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for update + list refresh\n * Common pattern: update entity, wait for success, wait for list to refresh\n */\n waitForUpdateAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForUpdate(timeout)\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for delete + list refresh\n * Common pattern: delete entity, wait for success, wait for list to refresh\n */\n waitForDeleteAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForDelete(timeout)\n return this.waitForList(timeout)\n }\n}\n\nexport default ApiInterceptor\n","/**\n * DashboardEntityPOMCore - Base class for all entity Page Object Models\n *\n * Provides standard CRUD operations for dashboard entities:\n * - Navigation (list, create, edit, detail pages)\n * - Table interactions (search, filters, pagination, row actions)\n * - Form operations (fill fields, submit, cancel)\n * - API interceptor integration for deterministic waits\n * - Bulk actions\n * - Delete confirmation dialogs\n *\n * This is the CORE version. Themes should extend this class and implement\n * the abstract `cySelector` method from BasePOMCore.\n *\n * @example Theme extension:\n * ```ts\n * // theme/tests/cypress/src/core/DashboardEntityPOM.ts\n * import { DashboardEntityPOMCore } from '@nextsparkjs/testing/pom'\n * import { cySelector } from '../selectors'\n *\n * export abstract class DashboardEntityPOM extends DashboardEntityPOMCore {\n * protected cySelector(path: string, replacements?: Record<string, string>): string {\n * return cySelector(path, replacements)\n * }\n * }\n * ```\n */\n\nimport { BasePOMCore, type Replacements } from './BasePOMCore'\nimport { ApiInterceptor } from '../helpers/ApiInterceptor'\n\nexport interface EntityConfig {\n slug: string\n singular?: string\n plural?: string\n tableName?: string\n fields?: string[]\n filters?: string[]\n}\n\nexport abstract class DashboardEntityPOMCore extends BasePOMCore {\n protected slug: string\n protected entityConfig: EntityConfig\n protected _api: ApiInterceptor | null = null\n\n /**\n * Get the entity slug (public accessor)\n * Useful for building dynamic selectors and URLs in tests\n */\n get entitySlug(): string {\n return this.slug\n }\n\n constructor(entitySlugOrConfig: string | EntityConfig) {\n super()\n if (typeof entitySlugOrConfig === 'string') {\n this.slug = entitySlugOrConfig\n this.entityConfig = { slug: entitySlugOrConfig }\n } else {\n this.slug = entitySlugOrConfig.slug\n this.entityConfig = entitySlugOrConfig\n }\n }\n\n // ============================================\n // API INTERCEPTOR\n // ============================================\n\n /**\n * Get or create ApiInterceptor instance for this entity\n */\n get api(): ApiInterceptor {\n if (!this._api) {\n this._api = new ApiInterceptor(this.slug)\n }\n return this._api\n }\n\n /**\n * Setup API intercepts for all CRUD operations\n * Call this BEFORE navigation\n */\n setupApiIntercepts() {\n this.api.setupCrudIntercepts()\n return this\n }\n\n // ============================================\n // SELECTORS (uses cySelector from theme)\n // ============================================\n\n /**\n * Get all selectors for this entity, with placeholders replaced\n * Uses the abstract cySelector method which themes implement\n */\n get selectors() {\n const slug = this.slug\n\n return {\n // Page\n page: this.cySelector('entities.page.container', { slug }),\n\n // List - Table\n tableContainer: this.cySelector('entities.list.table.container', { slug }),\n table: this.cySelector('entities.list.table.element', { slug }),\n addButton: this.cySelector('entities.list.addButton', { slug }),\n selectAll: this.cySelector('entities.list.table.selectAll', { slug }),\n selectionCount: this.cySelector('entities.list.selectionCount', { slug }),\n row: (id: string) => this.cySelector('entities.list.table.row.element', { slug, id }),\n rowSelect: (id: string) => this.cySelector('entities.list.table.row.checkbox', { slug, id }),\n rowMenu: (id: string) => this.cySelector('entities.list.table.row.menu', { slug, id }),\n rowAction: (action: string, id: string) => this.cySelector('entities.list.table.row.action', { slug, action, id }),\n cell: (name: string, id: string) => this.cySelector('entities.list.table.cell.element', { slug, name, id }),\n rowGeneric: `[data-cy^=\"${slug}-row-\"]`,\n\n // List - Search\n search: this.cySelector('entities.list.search.input', { slug }),\n searchContainer: this.cySelector('entities.list.search.container', { slug }),\n searchClear: this.cySelector('entities.list.search.clear', { slug }),\n\n // List - Pagination\n pagination: this.cySelector('entities.list.pagination.container', { slug }),\n pageSize: this.cySelector('entities.list.pagination.pageSize', { slug }),\n pageSizeOption: (size: string) => this.cySelector('entities.list.pagination.pageSizeOption', { slug, size }),\n pageInfo: this.cySelector('entities.list.pagination.info', { slug }),\n pageFirst: this.cySelector('entities.list.pagination.first', { slug }),\n pagePrev: this.cySelector('entities.list.pagination.prev', { slug }),\n pageNext: this.cySelector('entities.list.pagination.next', { slug }),\n pageLast: this.cySelector('entities.list.pagination.last', { slug }),\n\n // List - Filters\n filter: (field: string) => this.cySelector('entities.list.filters.container', { slug, field }),\n filterTrigger: (field: string) => this.cySelector('entities.list.filters.trigger', { slug, field }),\n filterContent: (field: string) => this.cySelector('entities.list.filters.content', { slug, field }),\n filterOption: (field: string, value: string) => this.cySelector('entities.list.filters.option', { slug, field, value }),\n filterBadge: (field: string, value: string) => this.cySelector('entities.list.filters.badge', { slug, field, value }),\n filterRemoveBadge: (field: string, value: string) => this.cySelector('entities.list.filters.removeBadge', { slug, field, value }),\n filterClearAll: (field: string) => this.cySelector('entities.list.filters.clearAll', { slug, field }),\n\n // List - Bulk actions\n bulkBar: this.cySelector('entities.list.bulk.bar', { slug }),\n bulkCount: this.cySelector('entities.list.bulk.count', { slug }),\n bulkSelectAll: this.cySelector('entities.list.bulk.selectAll', { slug }),\n bulkStatus: this.cySelector('entities.list.bulk.statusButton', { slug }),\n bulkDelete: this.cySelector('entities.list.bulk.deleteButton', { slug }),\n bulkClear: this.cySelector('entities.list.bulk.clearButton', { slug }),\n\n // List - Bulk status dialog\n bulkStatusDialog: this.cySelector('entities.list.bulk.statusDialog', { slug }),\n bulkStatusSelect: this.cySelector('entities.list.bulk.statusSelect', { slug }),\n bulkStatusOption: (value: string) => this.cySelector('entities.list.bulk.statusOption', { slug, value }),\n bulkStatusCancel: this.cySelector('entities.list.bulk.statusCancel', { slug }),\n bulkStatusConfirm: this.cySelector('entities.list.bulk.statusConfirm', { slug }),\n\n // List - Bulk delete dialog\n bulkDeleteDialog: this.cySelector('entities.list.bulk.deleteDialog', { slug }),\n bulkDeleteCancel: this.cySelector('entities.list.bulk.deleteCancel', { slug }),\n bulkDeleteConfirm: this.cySelector('entities.list.bulk.deleteConfirm', { slug }),\n\n // List - Confirm dialogs (for row actions)\n confirmDialog: this.cySelector('entities.list.confirm.dialog', { slug }),\n confirmCancel: this.cySelector('entities.list.confirm.cancel', { slug }),\n confirmAction: this.cySelector('entities.list.confirm.action', { slug }),\n\n // Header (detail pages) - modes: view, edit, create\n viewHeader: this.cySelector('entities.header.container', { slug, mode: 'view' }),\n editHeader: this.cySelector('entities.header.container', { slug, mode: 'edit' }),\n createHeader: this.cySelector('entities.header.container', { slug, mode: 'create' }),\n backButton: this.cySelector('entities.header.backButton', { slug }),\n editButton: this.cySelector('entities.header.editButton', { slug }),\n deleteButton: this.cySelector('entities.header.deleteButton', { slug }),\n title: this.cySelector('entities.header.title', { slug }),\n\n // Header - Delete confirmation\n deleteDialog: this.cySelector('entities.header.deleteDialog', { slug }),\n deleteCancel: this.cySelector('entities.header.deleteCancel', { slug }),\n deleteConfirm: this.cySelector('entities.header.deleteConfirm', { slug }),\n\n // Detail view\n detail: this.cySelector('entities.detail.container', { slug }),\n\n // Form\n form: this.cySelector('entities.form.container', { slug }),\n field: (name: string) => this.cySelector('entities.form.field', { slug, name }),\n submitButton: this.cySelector('entities.form.submitButton', { slug }),\n\n // Parent delete confirmation (EntityDetailWrapper - generic, no slug)\n parentDeleteConfirm: '[data-cy=\"confirm-delete\"]',\n parentDeleteCancel: '[data-cy=\"cancel-delete\"]',\n\n // Row action selectors (generic patterns for checking existence)\n rowActionEditGeneric: `[data-cy^=\"${slug}-action-edit-\"]`,\n rowActionDeleteGeneric: `[data-cy^=\"${slug}-action-delete-\"]`,\n }\n }\n\n // ============================================\n // NAVIGATION\n // ============================================\n\n /**\n * Navigate to entity list page\n */\n visitList() {\n cy.visit(`/dashboard/${this.slug}`)\n return this\n }\n\n /**\n * Navigate to create page\n */\n visitCreate() {\n cy.visit(`/dashboard/${this.slug}/create`)\n return this\n }\n\n /**\n * Navigate to edit page for specific entity\n */\n visitEdit(id: string) {\n cy.visit(`/dashboard/${this.slug}/${id}/edit`)\n return this\n }\n\n /**\n * Navigate to detail/view page for specific entity\n */\n visitDetail(id: string) {\n cy.visit(`/dashboard/${this.slug}/${id}`)\n return this\n }\n\n // ============================================\n // API-AWARE NAVIGATION\n // ============================================\n\n /**\n * Navigate to list and wait for API response\n */\n visitListWithApiWait() {\n this.setupApiIntercepts()\n this.visitList()\n this.api.waitForList()\n return this\n }\n\n /**\n * Navigate to edit page and wait for form to be visible\n */\n visitEditWithApiWait(id: string) {\n this.setupApiIntercepts()\n this.visitEdit(id)\n this.waitForForm()\n return this\n }\n\n /**\n * Navigate to detail page and wait for content\n */\n visitDetailWithApiWait(id: string) {\n this.setupApiIntercepts()\n this.visitDetail(id)\n this.waitForDetail()\n return this\n }\n\n // ============================================\n // WAITS\n // ============================================\n\n /**\n * Wait for list page to be fully loaded\n */\n waitForList() {\n cy.url().should('include', `/dashboard/${this.slug}`)\n cy.get(this.selectors.tableContainer, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n /**\n * Wait for form to be visible\n */\n waitForForm() {\n cy.get(this.selectors.form, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n /**\n * Wait for detail page to be loaded\n */\n waitForDetail() {\n cy.url().should('match', new RegExp(`/dashboard/${this.slug}/[a-z0-9-]+$`))\n cy.get(this.selectors.editButton, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n // ============================================\n // TABLE ACTIONS\n // ============================================\n\n /**\n * Click the Add/Create button\n */\n clickAdd() {\n cy.get(this.selectors.addButton).click()\n return this\n }\n\n /**\n * Type in the search input\n */\n search(term: string) {\n cy.get(this.selectors.search).clear().type(term)\n return this\n }\n\n /**\n * Clear the search input\n */\n clearSearch() {\n cy.get(this.selectors.searchClear).click()\n return this\n }\n\n /**\n * Click a specific row by ID\n */\n clickRow(id: string) {\n cy.get(this.selectors.row(id)).click()\n return this\n }\n\n /**\n * Find and click a row containing specific text\n */\n clickRowByText(text: string) {\n cy.contains(this.selectors.rowGeneric, text).click()\n return this\n }\n\n /**\n * Select a row checkbox\n */\n selectRow(id: string) {\n cy.get(this.selectors.rowSelect(id)).click()\n return this\n }\n\n /**\n * Open the row menu (three dots)\n */\n openRowMenu(id: string) {\n cy.get(this.selectors.rowMenu(id)).click()\n return this\n }\n\n /**\n * Click an action in the row menu\n */\n clickRowAction(action: string, id: string) {\n cy.get(this.selectors.rowAction(action, id)).click()\n return this\n }\n\n // ============================================\n // FILTERS\n // ============================================\n\n /**\n * Open a filter dropdown\n */\n openFilter(field: string) {\n cy.get(this.selectors.filterTrigger(field)).click()\n return this\n }\n\n /**\n * Select a filter option\n */\n selectFilterOption(field: string, value: string) {\n cy.get(this.selectors.filterOption(field, value)).click()\n return this\n }\n\n /**\n * Open filter and select option (convenience method)\n */\n selectFilter(field: string, value: string) {\n this.openFilter(field)\n this.selectFilterOption(field, value)\n return this\n }\n\n /**\n * Clear all selected options for a specific filter\n * NOTE: Clear button only appears when >1 option is selected\n */\n clearFilter(field: string) {\n cy.get(this.selectors.filterClearAll(field)).click()\n return this\n }\n\n // ============================================\n // PAGINATION\n // ============================================\n\n /**\n * Go to next page\n */\n nextPage() {\n cy.get(this.selectors.pageNext).click()\n return this\n }\n\n /**\n * Go to previous page\n */\n prevPage() {\n cy.get(this.selectors.pagePrev).click()\n return this\n }\n\n /**\n * Go to first page\n */\n firstPage() {\n cy.get(this.selectors.pageFirst).click()\n return this\n }\n\n /**\n * Go to last page\n */\n lastPage() {\n cy.get(this.selectors.pageLast).click()\n return this\n }\n\n /**\n * Change page size\n */\n setPageSize(size: string) {\n cy.get(this.selectors.pageSize).click()\n cy.get(this.selectors.pageSizeOption(size)).click()\n return this\n }\n\n // ============================================\n // FORM ACTIONS\n // ============================================\n\n /**\n * Fill a text input field\n */\n fillTextField(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('input').clear().type(value)\n return this\n }\n\n /**\n * Fill a textarea field\n */\n fillTextarea(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('textarea').clear().type(value)\n return this\n }\n\n /**\n * Select an option in a combobox/select field\n */\n selectOption(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('[role=\"combobox\"]').click()\n cy.get(`[data-cy=\"${this.slug}-field-${name}-option-${value}\"]`).click()\n return this\n }\n\n /**\n * Submit the form\n */\n submitForm() {\n cy.get(this.selectors.submitButton).click()\n return this\n }\n\n // ============================================\n // HEADER/DETAIL ACTIONS\n // ============================================\n\n /**\n * Click back button\n */\n clickBack() {\n cy.get(this.selectors.backButton).click()\n return this\n }\n\n /**\n * Click edit button\n */\n clickEdit() {\n cy.get(this.selectors.editButton).click()\n return this\n }\n\n /**\n * Click delete button\n */\n clickDelete() {\n cy.get(this.selectors.deleteButton).click()\n return this\n }\n\n /**\n * Confirm delete in dialog\n */\n confirmDelete() {\n cy.get(this.selectors.deleteConfirm).click()\n return this\n }\n\n /**\n * Cancel delete in dialog\n */\n cancelDelete() {\n cy.get(this.selectors.deleteCancel).click()\n return this\n }\n\n // ============================================\n // BULK ACTIONS\n // ============================================\n\n /**\n * Select all items using table header checkbox\n */\n selectAll() {\n cy.get(this.selectors.selectAll).click()\n return this\n }\n\n /**\n * Click bulk delete button\n */\n bulkDelete() {\n cy.get(this.selectors.bulkDelete).click()\n return this\n }\n\n /**\n * Confirm bulk delete\n */\n confirmBulkDelete() {\n cy.get(this.selectors.bulkDeleteConfirm).click()\n return this\n }\n\n /**\n * Cancel bulk delete\n */\n cancelBulkDelete() {\n cy.get(this.selectors.bulkDeleteCancel).click()\n return this\n }\n\n /**\n * Click bulk status button\n */\n bulkChangeStatus() {\n cy.get(this.selectors.bulkStatus).click()\n return this\n }\n\n /**\n * Select status in bulk status dialog\n */\n selectBulkStatus(value: string) {\n cy.get(this.selectors.bulkStatusSelect).click()\n cy.get(this.selectors.bulkStatusOption(value)).click()\n return this\n }\n\n /**\n * Confirm bulk status change\n */\n confirmBulkStatus() {\n cy.get(this.selectors.bulkStatusConfirm).click()\n return this\n }\n\n /**\n * Clear selection\n */\n clearSelection() {\n cy.get(this.selectors.bulkClear).click()\n return this\n }\n\n // ============================================\n // ASSERTIONS\n // ============================================\n\n /**\n * Assert text is visible in the list\n */\n assertInList(text: string) {\n cy.contains(text).should('be.visible')\n return this\n }\n\n /**\n * Assert text is not in the list\n */\n assertNotInList(text: string) {\n cy.contains(text).should('not.exist')\n return this\n }\n\n /**\n * Assert table is visible\n */\n assertTableVisible() {\n cy.get(this.selectors.table).should('be.visible')\n return this\n }\n\n /**\n * Assert form is visible\n */\n assertFormVisible() {\n cy.get(this.selectors.form).should('be.visible')\n return this\n }\n\n /**\n * Assert page title contains text\n */\n assertPageTitle(expected: string) {\n cy.get(this.selectors.title).should('contain.text', expected)\n return this\n }\n\n /**\n * Assert row exists\n */\n assertRowExists(id: string) {\n cy.get(this.selectors.row(id)).should('exist')\n return this\n }\n\n /**\n * Assert row does not exist\n */\n assertRowNotExists(id: string) {\n cy.get(this.selectors.row(id)).should('not.exist')\n return this\n }\n\n /**\n * Assert selection count\n */\n assertSelectionCount(count: number) {\n cy.get(this.selectors.selectionCount).should('contain.text', count.toString())\n return this\n }\n\n /**\n * Assert bulk bar is visible\n */\n assertBulkBarVisible() {\n cy.get(this.selectors.bulkBar).should('be.visible')\n return this\n }\n\n /**\n * Assert bulk bar is hidden\n */\n assertBulkBarHidden() {\n cy.get(this.selectors.bulkBar).should('not.be.visible')\n return this\n }\n}\n\nexport default DashboardEntityPOMCore\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/utils/utils.ts","../src/pom/BasePOMCore.ts","../src/helpers/ApiInterceptor.ts","../src/pom/DashboardEntityPOMCore.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Testing Utilities\n * Helper functions for consistent testing attribute generation\n *\n * For advanced selector patterns, use sel() from '@nextsparkjs/core/lib/selectors'\n * This file contains runtime utilities for accessibility and keyboard handling.\n */\n\n// =============================================================================\n// SELECTOR HELPERS\n// =============================================================================\n\n/**\n * Create a Cypress selector ID by joining parts with hyphens\n *\n * @example\n * createCyId('verify-email', 'loading') // 'verify-email-loading'\n * createCyId('settings', 'billing', 'plans') // 'settings-billing-plans'\n */\nexport function createCyId(...parts: string[]): string {\n return parts.filter(Boolean).join('-')\n}\n\n/**\n * Create a test ID by joining parts with hyphens\n *\n * @example\n * createTestId('settings', 'billing') // 'settings-billing'\n */\nexport function createTestId(...parts: string[]): string {\n return parts.filter(Boolean).join('-')\n}\n\n/**\n * Simple selector function that joins parts with a dot\n * For advanced patterns use sel() from '@nextsparkjs/core/lib/selectors'\n *\n * @example\n * sel('auth', 'login', 'form') // 'auth.login.form'\n */\nexport function sel(...parts: string[]): string {\n return parts.filter(Boolean).join('.')\n}\n\n// =============================================================================\n// ENVIRONMENT DETECTION\n// =============================================================================\n\n/**\n * Environment-based testing mode\n */\nconst isDevelopment = process.env.NODE_ENV === 'development'\nconst isTest = process.env.NODE_ENV === 'test'\nconst enableTestingAttributes = isDevelopment || isTest\n\n// =============================================================================\n// ATTRIBUTE GENERATORS\n// =============================================================================\n\n/**\n * Create state-based data attributes for conditional testing\n *\n * @param state - Current state value\n * @returns State attribute value\n */\nexport function createStateAttr(\n state: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n): string {\n return state\n}\n\n/**\n * Create priority-based data attributes\n *\n * @param priority - Priority level\n * @returns Priority attribute value\n */\nexport function createPriorityAttr(priority: 'low' | 'medium' | 'high'): string {\n return priority\n}\n\n/**\n * Generate testing props object for components\n *\n * @param config - Testing configuration\n * @returns Testing props object\n */\nexport function createTestingProps(config: {\n testId?: string\n cyId?: string\n state?: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n priority?: 'low' | 'medium' | 'high'\n taskId?: string\n userId?: string\n}) {\n const props: Record<string, string | undefined> = {}\n\n if (config.testId) {\n props['data-testid'] = enableTestingAttributes ? config.testId : undefined\n }\n\n if (config.cyId) {\n props['data-cy'] = enableTestingAttributes ? config.cyId : undefined\n }\n\n if (config.state) {\n props['data-state'] = config.state\n }\n\n if (config.priority) {\n props['data-priority'] = config.priority\n }\n\n if (config.taskId) {\n props['data-task-id'] = config.taskId\n }\n\n if (config.userId) {\n props['data-user-id'] = config.userId\n }\n\n // Filter out undefined values\n return Object.fromEntries(\n Object.entries(props).filter((entry) => entry[1] !== undefined)\n )\n}\n\n/**\n * Accessibility helper for dynamic aria-label generation\n *\n * @param template - Label template with placeholders\n * @param values - Values to replace placeholders\n * @returns Formatted aria-label\n */\nexport function createAriaLabel(\n template: string,\n values: Record<string, string | number | boolean>\n): string {\n return template.replace(/\\{(\\w+)\\}/g, (match, key) => {\n return String(values[key] ?? match)\n })\n}\n\n// =============================================================================\n// KEYBOARD HELPERS\n// =============================================================================\n\n/**\n * Keyboard navigation helpers\n */\nexport const keyboardHelpers = {\n /**\n * Handle Enter and Space key activation\n */\n createActivationHandler: (onActivate: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n onActivate()\n }\n }\n },\n\n /**\n * Handle Escape key for closing\n */\n createEscapeHandler: (onClose: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Escape') {\n e.preventDefault()\n onClose()\n }\n }\n },\n\n /**\n * Handle arrow navigation in lists\n */\n createArrowNavigationHandler: (\n currentIndex: number,\n maxIndex: number,\n onIndexChange: (index: number) => void\n ) => {\n return (e: React.KeyboardEvent) => {\n switch (e.key) {\n case 'ArrowDown':\n e.preventDefault()\n onIndexChange(currentIndex < maxIndex ? currentIndex + 1 : 0)\n break\n case 'ArrowUp':\n e.preventDefault()\n onIndexChange(currentIndex > 0 ? currentIndex - 1 : maxIndex)\n break\n }\n }\n },\n}\n","/**\n * BasePOMCore - Base class with common utilities for all Page Object Models\n *\n * This is the CORE version that provides generic methods.\n * Themes should extend this class and implement the abstract `cySelector` method\n * to provide theme-specific selector resolution.\n *\n * @example Theme extension:\n * ```ts\n * // theme/tests/cypress/src/core/BasePOM.ts\n * import { BasePOMCore } from '@nextsparkjs/testing/pom'\n * import { cySelector } from '../selectors'\n *\n * export abstract class BasePOM extends BasePOMCore {\n * protected cySelector(path: string, replacements?: Record<string, string>): string {\n * return cySelector(path, replacements)\n * }\n * }\n * ```\n */\n\n/**\n * Type for placeholder replacements in selectors\n * Mirrors @nextsparkjs/core/selectors Replacements type\n */\nexport type Replacements = Record<string, string | number>\n\nexport abstract class BasePOMCore {\n /**\n * Abstract method - themes must implement to provide their cySelector\n * This allows themes to use their extended THEME_SELECTORS\n *\n * @param path - Dot-notation path to the selector (e.g., \"auth.login.form\")\n * @param replacements - Optional placeholder replacements\n * @returns Cypress selector string like [data-cy=\"selector-value\"]\n */\n protected abstract cySelector(path: string, replacements?: Replacements): string\n\n /**\n * Get a Cypress selector using the centralized selectors\n * Wrapper for cySelector with a shorter name\n *\n * @example\n * this.cy('auth.login.form')\n * // Returns: '[data-cy=\"login-form\"]'\n *\n * this.cy('entities.table.row', { slug: 'tasks', id: '123' })\n * // Returns: '[data-cy=\"tasks-row-123\"]'\n */\n protected cy(path: string, replacements?: Replacements): string {\n return this.cySelector(path, replacements)\n }\n\n /**\n * Replaces placeholders in a selector pattern and wraps with data-cy attribute\n *\n * @param pattern - Selector pattern with {placeholder} syntax\n * @param replacements - Object with placeholder values\n * @returns Formatted data-cy selector string\n *\n * @example\n * selector('{slug}-row-{id}', { slug: 'tasks', id: '123' })\n * // Returns: '[data-cy=\"tasks-row-123\"]'\n */\n protected selector(pattern: string, replacements: Replacements = {}): string {\n let result = pattern\n for (const [key, value] of Object.entries(replacements)) {\n result = result.replaceAll(`{${key}}`, String(value))\n }\n return `[data-cy=\"${result}\"]`\n }\n\n /**\n * Wrapper for cy.get with selector pattern support\n * @param pattern - Selector pattern or direct selector\n * @param replacements - Optional placeholder replacements\n */\n protected get(pattern: string, replacements: Replacements = {}) {\n return cy.get(this.selector(pattern, replacements))\n }\n\n /**\n * Generic wait with configurable timeout\n * @param selector - CSS selector to wait for\n * @param timeout - Max wait time in ms (default: 15000)\n */\n protected waitFor(selector: string, timeout = 15000) {\n cy.get(selector, { timeout }).should('be.visible')\n return this\n }\n\n /**\n * Wait for URL to contain a specific path\n * @param path - Path segment to check for\n */\n protected waitForUrl(path: string) {\n cy.url().should('include', path)\n return this\n }\n\n /**\n * Wait for URL to match a regex pattern\n * @param pattern - RegExp to match against URL\n */\n protected waitForUrlMatch(pattern: RegExp) {\n cy.url().should('match', pattern)\n return this\n }\n\n /**\n * Visit a URL and return self for chaining\n * @param url - URL to visit\n */\n visit(url: string) {\n cy.visit(url)\n return this\n }\n\n /**\n * Wait for page to load (checks for body visible)\n */\n waitForPageLoad() {\n cy.get('body').should('be.visible')\n return this\n }\n\n /**\n * Get an element by selector\n * @param selector - CSS selector\n */\n getElement(selector: string) {\n return cy.get(selector)\n }\n\n /**\n * Click on an element\n * @param selector - CSS selector\n */\n click(selector: string) {\n cy.get(selector).click()\n return this\n }\n\n /**\n * Type text into an input\n * @param selector - CSS selector\n * @param text - Text to type\n */\n type(selector: string, text: string) {\n cy.get(selector).clear().type(text)\n return this\n }\n\n /**\n * Check if element exists\n * @param selector - CSS selector\n */\n exists(selector: string) {\n return cy.get(selector).should('exist')\n }\n\n /**\n * Check if element is visible\n * @param selector - CSS selector\n */\n isVisible(selector: string) {\n return cy.get(selector).should('be.visible')\n }\n\n /**\n * Check if element does not exist\n * @param selector - CSS selector\n */\n notExists(selector: string) {\n return cy.get(selector).should('not.exist')\n }\n}\n\nexport default BasePOMCore\n","/**\n * ApiInterceptor - Helper for deterministic waits in Cypress\n *\n * Replaces unreliable cy.wait(ms) with cy.intercept() based waits\n * that wait for actual API responses.\n *\n * @example Basic usage:\n * ```ts\n * const api = new ApiInterceptor('customers')\n * api.setupCrudIntercepts()\n * cy.visit('/dashboard/customers')\n * api.waitForList()\n * ```\n *\n * @example Custom path:\n * ```ts\n * const api = new ApiInterceptor({\n * slug: 'categories',\n * customPath: '/api/v1/post-categories'\n * })\n * ```\n */\n\nexport interface ApiInterceptorConfig {\n /** Entity slug - used to generate aliases */\n slug: string\n /** Custom API path (e.g., '/api/v1/post-categories') */\n customPath?: string\n}\n\nexport class ApiInterceptor {\n private slug: string\n private endpoint: string\n\n constructor(slugOrConfig: string | ApiInterceptorConfig) {\n if (typeof slugOrConfig === 'string') {\n this.slug = slugOrConfig\n this.endpoint = `/api/v1/${slugOrConfig}`\n } else {\n this.slug = slugOrConfig.slug\n this.endpoint = slugOrConfig.customPath || `/api/v1/${slugOrConfig.slug}`\n }\n }\n\n // ============================================\n // ACCESSORS\n // ============================================\n\n /** Get the API endpoint path */\n get path(): string {\n return this.endpoint\n }\n\n /** Get the entity slug */\n get entitySlug(): string {\n return this.slug\n }\n\n /** Get alias names for all operations */\n get aliases() {\n return {\n list: `${this.slug}List`,\n create: `${this.slug}Create`,\n update: `${this.slug}Update`,\n delete: `${this.slug}Delete`,\n }\n }\n\n // ============================================\n // INTERCEPT SETUP\n // ============================================\n\n /**\n * Setup intercepts for all CRUD operations\n * Call this BEFORE navigation in beforeEach or at test start\n *\n * Note: We intercept both PUT and PATCH for updates since different\n * APIs may use different HTTP methods for updates.\n */\n setupCrudIntercepts(): this {\n cy.intercept('GET', `${this.endpoint}*`).as(this.aliases.list)\n cy.intercept('POST', this.endpoint).as(this.aliases.create)\n // Intercept both PUT and PATCH for updates (APIs may use either)\n cy.intercept('PUT', `${this.endpoint}/*`).as(this.aliases.update)\n cy.intercept('PATCH', `${this.endpoint}/*`).as(`${this.aliases.update}Patch`)\n cy.intercept('DELETE', `${this.endpoint}/*`).as(this.aliases.delete)\n return this\n }\n\n /**\n * Setup only list + create intercepts\n * Useful for list pages with inline create\n */\n setupListIntercepts(): this {\n cy.intercept('GET', `${this.endpoint}*`).as(this.aliases.list)\n cy.intercept('POST', this.endpoint).as(this.aliases.create)\n return this\n }\n\n // ============================================\n // WAIT METHODS\n // ============================================\n\n /**\n * Wait for list response (GET)\n * Use after navigation or after mutations to wait for refresh\n */\n waitForList(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.list}`, { timeout })\n }\n\n /**\n * Wait for create response (POST) and validate success status\n */\n waitForCreate(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.create}`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 201])\n }\n\n /**\n * Wait for update response (PATCH or PUT) and validate success status\n * Waits for PATCH first (more common), falls back to PUT\n */\n waitForUpdate(timeout = 10000): Cypress.Chainable {\n // Try PATCH first (more common in modern APIs), fall back to PUT\n return cy.wait(`@${this.aliases.update}Patch`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 201])\n }\n\n /**\n * Wait for delete response (DELETE) and validate success status\n */\n waitForDelete(timeout = 10000): Cypress.Chainable {\n return cy.wait(`@${this.aliases.delete}`, { timeout })\n .its('response.statusCode')\n .should('be.oneOf', [200, 204])\n }\n\n // ============================================\n // CONVENIENCE METHODS\n // ============================================\n\n /**\n * Wait for list refresh (alias for waitForList)\n * Semantic name for use after create/update/delete\n */\n waitForRefresh(timeout = 10000): Cypress.Chainable {\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for create + list refresh\n * Common pattern: create entity, wait for success, wait for list to refresh\n */\n waitForCreateAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForCreate(timeout)\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for update + list refresh\n * Common pattern: update entity, wait for success, wait for list to refresh\n */\n waitForUpdateAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForUpdate(timeout)\n return this.waitForList(timeout)\n }\n\n /**\n * Wait for delete + list refresh\n * Common pattern: delete entity, wait for success, wait for list to refresh\n */\n waitForDeleteAndRefresh(timeout = 10000): Cypress.Chainable {\n this.waitForDelete(timeout)\n return this.waitForList(timeout)\n }\n}\n\nexport default ApiInterceptor\n","/**\n * DashboardEntityPOMCore - Base class for all entity Page Object Models\n *\n * Provides standard CRUD operations for dashboard entities:\n * - Navigation (list, create, edit, detail pages)\n * - Table interactions (search, filters, pagination, row actions)\n * - Form operations (fill fields, submit, cancel)\n * - API interceptor integration for deterministic waits\n * - Bulk actions\n * - Delete confirmation dialogs\n *\n * This is the CORE version. Themes should extend this class and implement\n * the abstract `cySelector` method from BasePOMCore.\n *\n * @example Theme extension:\n * ```ts\n * // theme/tests/cypress/src/core/DashboardEntityPOM.ts\n * import { DashboardEntityPOMCore } from '@nextsparkjs/testing/pom'\n * import { cySelector } from '../selectors'\n *\n * export abstract class DashboardEntityPOM extends DashboardEntityPOMCore {\n * protected cySelector(path: string, replacements?: Record<string, string>): string {\n * return cySelector(path, replacements)\n * }\n * }\n * ```\n */\n\nimport { BasePOMCore, type Replacements } from './BasePOMCore'\nimport { ApiInterceptor } from '../helpers/ApiInterceptor'\n\nexport interface EntityConfig {\n slug: string\n singular?: string\n plural?: string\n tableName?: string\n fields?: string[]\n filters?: string[]\n}\n\nexport abstract class DashboardEntityPOMCore extends BasePOMCore {\n protected slug: string\n protected entityConfig: EntityConfig\n protected _api: ApiInterceptor | null = null\n\n /**\n * Get the entity slug (public accessor)\n * Useful for building dynamic selectors and URLs in tests\n */\n get entitySlug(): string {\n return this.slug\n }\n\n constructor(entitySlugOrConfig: string | EntityConfig) {\n super()\n if (typeof entitySlugOrConfig === 'string') {\n this.slug = entitySlugOrConfig\n this.entityConfig = { slug: entitySlugOrConfig }\n } else {\n this.slug = entitySlugOrConfig.slug\n this.entityConfig = entitySlugOrConfig\n }\n }\n\n // ============================================\n // API INTERCEPTOR\n // ============================================\n\n /**\n * Get or create ApiInterceptor instance for this entity\n */\n get api(): ApiInterceptor {\n if (!this._api) {\n this._api = new ApiInterceptor(this.slug)\n }\n return this._api\n }\n\n /**\n * Setup API intercepts for all CRUD operations\n * Call this BEFORE navigation\n */\n setupApiIntercepts() {\n this.api.setupCrudIntercepts()\n return this\n }\n\n // ============================================\n // SELECTORS (uses cySelector from theme)\n // ============================================\n\n /**\n * Get all selectors for this entity, with placeholders replaced\n * Uses the abstract cySelector method which themes implement\n */\n get selectors() {\n const slug = this.slug\n\n return {\n // Page\n page: this.cySelector('entities.page.container', { slug }),\n\n // List - Table\n tableContainer: this.cySelector('entities.list.table.container', { slug }),\n table: this.cySelector('entities.list.table.element', { slug }),\n addButton: this.cySelector('entities.list.addButton', { slug }),\n selectAll: this.cySelector('entities.list.table.selectAll', { slug }),\n selectionCount: this.cySelector('entities.list.selectionCount', { slug }),\n row: (id: string) => this.cySelector('entities.list.table.row.element', { slug, id }),\n rowSelect: (id: string) => this.cySelector('entities.list.table.row.checkbox', { slug, id }),\n rowMenu: (id: string) => this.cySelector('entities.list.table.row.menu', { slug, id }),\n rowAction: (action: string, id: string) => this.cySelector('entities.list.table.row.action', { slug, action, id }),\n cell: (name: string, id: string) => this.cySelector('entities.list.table.cell.element', { slug, name, id }),\n rowGeneric: `[data-cy^=\"${slug}-row-\"]`,\n\n // List - Search\n search: this.cySelector('entities.list.search.input', { slug }),\n searchContainer: this.cySelector('entities.list.search.container', { slug }),\n searchClear: this.cySelector('entities.list.search.clear', { slug }),\n\n // List - Pagination\n pagination: this.cySelector('entities.list.pagination.container', { slug }),\n pageSize: this.cySelector('entities.list.pagination.pageSize', { slug }),\n pageSizeOption: (size: string) => this.cySelector('entities.list.pagination.pageSizeOption', { slug, size }),\n pageInfo: this.cySelector('entities.list.pagination.info', { slug }),\n pageFirst: this.cySelector('entities.list.pagination.first', { slug }),\n pagePrev: this.cySelector('entities.list.pagination.prev', { slug }),\n pageNext: this.cySelector('entities.list.pagination.next', { slug }),\n pageLast: this.cySelector('entities.list.pagination.last', { slug }),\n\n // List - Filters\n filter: (field: string) => this.cySelector('entities.list.filters.container', { slug, field }),\n filterTrigger: (field: string) => this.cySelector('entities.list.filters.trigger', { slug, field }),\n filterContent: (field: string) => this.cySelector('entities.list.filters.content', { slug, field }),\n filterOption: (field: string, value: string) => this.cySelector('entities.list.filters.option', { slug, field, value }),\n filterBadge: (field: string, value: string) => this.cySelector('entities.list.filters.badge', { slug, field, value }),\n filterRemoveBadge: (field: string, value: string) => this.cySelector('entities.list.filters.removeBadge', { slug, field, value }),\n filterClearAll: (field: string) => this.cySelector('entities.list.filters.clearAll', { slug, field }),\n\n // List - Bulk actions\n bulkBar: this.cySelector('entities.list.bulk.bar', { slug }),\n bulkCount: this.cySelector('entities.list.bulk.count', { slug }),\n bulkSelectAll: this.cySelector('entities.list.bulk.selectAll', { slug }),\n bulkStatus: this.cySelector('entities.list.bulk.statusButton', { slug }),\n bulkDelete: this.cySelector('entities.list.bulk.deleteButton', { slug }),\n bulkClear: this.cySelector('entities.list.bulk.clearButton', { slug }),\n\n // List - Bulk status dialog\n bulkStatusDialog: this.cySelector('entities.list.bulk.statusDialog', { slug }),\n bulkStatusSelect: this.cySelector('entities.list.bulk.statusSelect', { slug }),\n bulkStatusOption: (value: string) => this.cySelector('entities.list.bulk.statusOption', { slug, value }),\n bulkStatusCancel: this.cySelector('entities.list.bulk.statusCancel', { slug }),\n bulkStatusConfirm: this.cySelector('entities.list.bulk.statusConfirm', { slug }),\n\n // List - Bulk delete dialog\n bulkDeleteDialog: this.cySelector('entities.list.bulk.deleteDialog', { slug }),\n bulkDeleteCancel: this.cySelector('entities.list.bulk.deleteCancel', { slug }),\n bulkDeleteConfirm: this.cySelector('entities.list.bulk.deleteConfirm', { slug }),\n\n // List - Confirm dialogs (for row actions)\n confirmDialog: this.cySelector('entities.list.confirm.dialog', { slug }),\n confirmCancel: this.cySelector('entities.list.confirm.cancel', { slug }),\n confirmAction: this.cySelector('entities.list.confirm.action', { slug }),\n\n // Header (detail pages) - modes: view, edit, create\n viewHeader: this.cySelector('entities.header.container', { slug, mode: 'view' }),\n editHeader: this.cySelector('entities.header.container', { slug, mode: 'edit' }),\n createHeader: this.cySelector('entities.header.container', { slug, mode: 'create' }),\n backButton: this.cySelector('entities.header.backButton', { slug }),\n editButton: this.cySelector('entities.header.editButton', { slug }),\n deleteButton: this.cySelector('entities.header.deleteButton', { slug }),\n title: this.cySelector('entities.header.title', { slug }),\n\n // Header - Delete confirmation\n deleteDialog: this.cySelector('entities.header.deleteDialog', { slug }),\n deleteCancel: this.cySelector('entities.header.deleteCancel', { slug }),\n deleteConfirm: this.cySelector('entities.header.deleteConfirm', { slug }),\n\n // Detail view\n detail: this.cySelector('entities.detail.container', { slug }),\n\n // Form\n form: this.cySelector('entities.form.container', { slug }),\n field: (name: string) => this.cySelector('entities.form.field', { slug, name }),\n submitButton: this.cySelector('entities.form.submitButton', { slug }),\n\n // Parent delete confirmation (EntityDetailWrapper - generic, no slug)\n parentDeleteConfirm: '[data-cy=\"confirm-delete\"]',\n parentDeleteCancel: '[data-cy=\"cancel-delete\"]',\n\n // Row action selectors (generic patterns for checking existence)\n rowActionEditGeneric: `[data-cy^=\"${slug}-action-edit-\"]`,\n rowActionDeleteGeneric: `[data-cy^=\"${slug}-action-delete-\"]`,\n }\n }\n\n // ============================================\n // NAVIGATION\n // ============================================\n\n /**\n * Navigate to entity list page\n */\n visitList() {\n cy.visit(`/dashboard/${this.slug}`)\n return this\n }\n\n /**\n * Navigate to create page\n */\n visitCreate() {\n cy.visit(`/dashboard/${this.slug}/create`)\n return this\n }\n\n /**\n * Navigate to edit page for specific entity\n */\n visitEdit(id: string) {\n cy.visit(`/dashboard/${this.slug}/${id}/edit`)\n return this\n }\n\n /**\n * Navigate to detail/view page for specific entity\n */\n visitDetail(id: string) {\n cy.visit(`/dashboard/${this.slug}/${id}`)\n return this\n }\n\n // ============================================\n // API-AWARE NAVIGATION\n // ============================================\n\n /**\n * Navigate to list and wait for API response\n */\n visitListWithApiWait() {\n this.setupApiIntercepts()\n this.visitList()\n this.api.waitForList()\n return this\n }\n\n /**\n * Navigate to edit page and wait for form to be visible\n */\n visitEditWithApiWait(id: string) {\n this.setupApiIntercepts()\n this.visitEdit(id)\n this.waitForForm()\n return this\n }\n\n /**\n * Navigate to detail page and wait for content\n */\n visitDetailWithApiWait(id: string) {\n this.setupApiIntercepts()\n this.visitDetail(id)\n this.waitForDetail()\n return this\n }\n\n // ============================================\n // WAITS\n // ============================================\n\n /**\n * Wait for list page to be fully loaded\n */\n waitForList() {\n cy.url().should('include', `/dashboard/${this.slug}`)\n cy.get(this.selectors.tableContainer, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n /**\n * Wait for form to be visible\n */\n waitForForm() {\n cy.get(this.selectors.form, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n /**\n * Wait for detail page to be loaded\n */\n waitForDetail() {\n cy.url().should('match', new RegExp(`/dashboard/${this.slug}/[a-z0-9-]+$`))\n cy.get(this.selectors.editButton, { timeout: 15000 }).should('be.visible')\n return this\n }\n\n // ============================================\n // TABLE ACTIONS\n // ============================================\n\n /**\n * Click the Add/Create button\n */\n clickAdd() {\n cy.get(this.selectors.addButton).click()\n return this\n }\n\n /**\n * Type in the search input\n */\n search(term: string) {\n cy.get(this.selectors.search).clear().type(term)\n return this\n }\n\n /**\n * Clear the search input\n */\n clearSearch() {\n cy.get(this.selectors.searchClear).click()\n return this\n }\n\n /**\n * Click a specific row by ID\n */\n clickRow(id: string) {\n cy.get(this.selectors.row(id)).click()\n return this\n }\n\n /**\n * Find and click a row containing specific text\n */\n clickRowByText(text: string) {\n cy.contains(this.selectors.rowGeneric, text).click()\n return this\n }\n\n /**\n * Select a row checkbox\n */\n selectRow(id: string) {\n cy.get(this.selectors.rowSelect(id)).click()\n return this\n }\n\n /**\n * Open the row menu (three dots)\n */\n openRowMenu(id: string) {\n cy.get(this.selectors.rowMenu(id)).click()\n return this\n }\n\n /**\n * Click an action in the row menu\n */\n clickRowAction(action: string, id: string) {\n cy.get(this.selectors.rowAction(action, id)).click()\n return this\n }\n\n // ============================================\n // FILTERS\n // ============================================\n\n /**\n * Open a filter dropdown\n */\n openFilter(field: string) {\n cy.get(this.selectors.filterTrigger(field)).click()\n return this\n }\n\n /**\n * Select a filter option\n */\n selectFilterOption(field: string, value: string) {\n cy.get(this.selectors.filterOption(field, value)).click()\n return this\n }\n\n /**\n * Open filter and select option (convenience method)\n */\n selectFilter(field: string, value: string) {\n this.openFilter(field)\n this.selectFilterOption(field, value)\n return this\n }\n\n /**\n * Clear all selected options for a specific filter\n * NOTE: Clear button only appears when >1 option is selected\n */\n clearFilter(field: string) {\n cy.get(this.selectors.filterClearAll(field)).click()\n return this\n }\n\n // ============================================\n // PAGINATION\n // ============================================\n\n /**\n * Go to next page\n */\n nextPage() {\n cy.get(this.selectors.pageNext).click()\n return this\n }\n\n /**\n * Go to previous page\n */\n prevPage() {\n cy.get(this.selectors.pagePrev).click()\n return this\n }\n\n /**\n * Go to first page\n */\n firstPage() {\n cy.get(this.selectors.pageFirst).click()\n return this\n }\n\n /**\n * Go to last page\n */\n lastPage() {\n cy.get(this.selectors.pageLast).click()\n return this\n }\n\n /**\n * Change page size\n */\n setPageSize(size: string) {\n cy.get(this.selectors.pageSize).click()\n cy.get(this.selectors.pageSizeOption(size)).click()\n return this\n }\n\n // ============================================\n // FORM ACTIONS\n // ============================================\n\n /**\n * Fill a text input field\n */\n fillTextField(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('input').clear().type(value)\n return this\n }\n\n /**\n * Fill a textarea field\n */\n fillTextarea(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('textarea').clear().type(value)\n return this\n }\n\n /**\n * Select an option in a combobox/select field\n */\n selectOption(name: string, value: string) {\n cy.get(this.selectors.field(name)).find('[role=\"combobox\"]').click()\n cy.get(`[data-cy=\"${this.slug}-field-${name}-option-${value}\"]`).click()\n return this\n }\n\n /**\n * Submit the form\n */\n submitForm() {\n cy.get(this.selectors.submitButton).click()\n return this\n }\n\n // ============================================\n // HEADER/DETAIL ACTIONS\n // ============================================\n\n /**\n * Click back button\n */\n clickBack() {\n cy.get(this.selectors.backButton).click()\n return this\n }\n\n /**\n * Click edit button\n */\n clickEdit() {\n cy.get(this.selectors.editButton).click()\n return this\n }\n\n /**\n * Click delete button\n */\n clickDelete() {\n cy.get(this.selectors.deleteButton).click()\n return this\n }\n\n /**\n * Confirm delete in dialog\n */\n confirmDelete() {\n cy.get(this.selectors.deleteConfirm).click()\n return this\n }\n\n /**\n * Cancel delete in dialog\n */\n cancelDelete() {\n cy.get(this.selectors.deleteCancel).click()\n return this\n }\n\n // ============================================\n // BULK ACTIONS\n // ============================================\n\n /**\n * Select all items using table header checkbox\n */\n selectAll() {\n cy.get(this.selectors.selectAll).click()\n return this\n }\n\n /**\n * Click bulk delete button\n */\n bulkDelete() {\n cy.get(this.selectors.bulkDelete).click()\n return this\n }\n\n /**\n * Confirm bulk delete\n */\n confirmBulkDelete() {\n cy.get(this.selectors.bulkDeleteConfirm).click()\n return this\n }\n\n /**\n * Cancel bulk delete\n */\n cancelBulkDelete() {\n cy.get(this.selectors.bulkDeleteCancel).click()\n return this\n }\n\n /**\n * Click bulk status button\n */\n bulkChangeStatus() {\n cy.get(this.selectors.bulkStatus).click()\n return this\n }\n\n /**\n * Select status in bulk status dialog\n */\n selectBulkStatus(value: string) {\n cy.get(this.selectors.bulkStatusSelect).click()\n cy.get(this.selectors.bulkStatusOption(value)).click()\n return this\n }\n\n /**\n * Confirm bulk status change\n */\n confirmBulkStatus() {\n cy.get(this.selectors.bulkStatusConfirm).click()\n return this\n }\n\n /**\n * Clear selection\n */\n clearSelection() {\n cy.get(this.selectors.bulkClear).click()\n return this\n }\n\n // ============================================\n // ASSERTIONS\n // ============================================\n\n /**\n * Assert text is visible in the list\n */\n assertInList(text: string) {\n cy.contains(text).should('be.visible')\n return this\n }\n\n /**\n * Assert text is not in the list\n */\n assertNotInList(text: string) {\n cy.contains(text).should('not.exist')\n return this\n }\n\n /**\n * Assert table is visible\n */\n assertTableVisible() {\n cy.get(this.selectors.table).should('be.visible')\n return this\n }\n\n /**\n * Assert form is visible\n */\n assertFormVisible() {\n cy.get(this.selectors.form).should('be.visible')\n return this\n }\n\n /**\n * Assert page title contains text\n */\n assertPageTitle(expected: string) {\n cy.get(this.selectors.title).should('contain.text', expected)\n return this\n }\n\n /**\n * Assert row exists\n */\n assertRowExists(id: string) {\n cy.get(this.selectors.row(id)).should('exist')\n return this\n }\n\n /**\n * Assert row does not exist\n */\n assertRowNotExists(id: string) {\n cy.get(this.selectors.row(id)).should('not.exist')\n return this\n }\n\n /**\n * Assert selection count\n */\n assertSelectionCount(count: number) {\n cy.get(this.selectors.selectionCount).should('contain.text', count.toString())\n return this\n }\n\n /**\n * Assert bulk bar is visible\n */\n assertBulkBarVisible() {\n cy.get(this.selectors.bulkBar).should('be.visible')\n return this\n }\n\n /**\n * Assert bulk bar is hidden\n */\n assertBulkBarHidden() {\n cy.get(this.selectors.bulkBar).should('not.be.visible')\n return this\n }\n}\n\nexport default DashboardEntityPOMCore\n"]}
|
package/dist/utils/index.d.ts
CHANGED
|
@@ -2,9 +2,32 @@
|
|
|
2
2
|
* Testing Utilities
|
|
3
3
|
* Helper functions for consistent testing attribute generation
|
|
4
4
|
*
|
|
5
|
-
*
|
|
5
|
+
* For advanced selector patterns, use sel() from '@nextsparkjs/core/lib/selectors'
|
|
6
6
|
* This file contains runtime utilities for accessibility and keyboard handling.
|
|
7
7
|
*/
|
|
8
|
+
/**
|
|
9
|
+
* Create a Cypress selector ID by joining parts with hyphens
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* createCyId('verify-email', 'loading') // 'verify-email-loading'
|
|
13
|
+
* createCyId('settings', 'billing', 'plans') // 'settings-billing-plans'
|
|
14
|
+
*/
|
|
15
|
+
declare function createCyId(...parts: string[]): string;
|
|
16
|
+
/**
|
|
17
|
+
* Create a test ID by joining parts with hyphens
|
|
18
|
+
*
|
|
19
|
+
* @example
|
|
20
|
+
* createTestId('settings', 'billing') // 'settings-billing'
|
|
21
|
+
*/
|
|
22
|
+
declare function createTestId(...parts: string[]): string;
|
|
23
|
+
/**
|
|
24
|
+
* Simple selector function that joins parts with a dot
|
|
25
|
+
* For advanced patterns use sel() from '@nextsparkjs/core/lib/selectors'
|
|
26
|
+
*
|
|
27
|
+
* @example
|
|
28
|
+
* sel('auth', 'login', 'form') // 'auth.login.form'
|
|
29
|
+
*/
|
|
30
|
+
declare function sel(...parts: string[]): string;
|
|
8
31
|
/**
|
|
9
32
|
* Create state-based data attributes for conditional testing
|
|
10
33
|
*
|
|
@@ -61,4 +84,4 @@ declare const keyboardHelpers: {
|
|
|
61
84
|
createArrowNavigationHandler: (currentIndex: number, maxIndex: number, onIndexChange: (index: number) => void) => (e: React.KeyboardEvent) => void;
|
|
62
85
|
};
|
|
63
86
|
|
|
64
|
-
export { createAriaLabel, createPriorityAttr, createStateAttr, createTestingProps, keyboardHelpers };
|
|
87
|
+
export { createAriaLabel, createCyId, createPriorityAttr, createStateAttr, createTestId, createTestingProps, keyboardHelpers, sel };
|
package/dist/utils/index.js
CHANGED
|
@@ -1,4 +1,13 @@
|
|
|
1
1
|
// src/utils/utils.ts
|
|
2
|
+
function createCyId(...parts) {
|
|
3
|
+
return parts.filter(Boolean).join("-");
|
|
4
|
+
}
|
|
5
|
+
function createTestId(...parts) {
|
|
6
|
+
return parts.filter(Boolean).join("-");
|
|
7
|
+
}
|
|
8
|
+
function sel(...parts) {
|
|
9
|
+
return parts.filter(Boolean).join(".");
|
|
10
|
+
}
|
|
2
11
|
var isDevelopment = process.env.NODE_ENV === "development";
|
|
3
12
|
var isTest = process.env.NODE_ENV === "test";
|
|
4
13
|
var enableTestingAttributes = isDevelopment || isTest;
|
|
@@ -79,6 +88,6 @@ var keyboardHelpers = {
|
|
|
79
88
|
}
|
|
80
89
|
};
|
|
81
90
|
|
|
82
|
-
export { createAriaLabel, createPriorityAttr, createStateAttr, createTestingProps, keyboardHelpers };
|
|
91
|
+
export { createAriaLabel, createCyId, createPriorityAttr, createStateAttr, createTestId, createTestingProps, keyboardHelpers, sel };
|
|
83
92
|
//# sourceMappingURL=index.js.map
|
|
84
93
|
//# sourceMappingURL=index.js.map
|
package/dist/utils/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/utils/utils.ts"],"names":[],"mappings":";
|
|
1
|
+
{"version":3,"sources":["../../src/utils/utils.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\n * Testing Utilities\n * Helper functions for consistent testing attribute generation\n *\n * For advanced selector patterns, use sel() from '@nextsparkjs/core/lib/selectors'\n * This file contains runtime utilities for accessibility and keyboard handling.\n */\n\n// =============================================================================\n// SELECTOR HELPERS\n// =============================================================================\n\n/**\n * Create a Cypress selector ID by joining parts with hyphens\n *\n * @example\n * createCyId('verify-email', 'loading') // 'verify-email-loading'\n * createCyId('settings', 'billing', 'plans') // 'settings-billing-plans'\n */\nexport function createCyId(...parts: string[]): string {\n return parts.filter(Boolean).join('-')\n}\n\n/**\n * Create a test ID by joining parts with hyphens\n *\n * @example\n * createTestId('settings', 'billing') // 'settings-billing'\n */\nexport function createTestId(...parts: string[]): string {\n return parts.filter(Boolean).join('-')\n}\n\n/**\n * Simple selector function that joins parts with a dot\n * For advanced patterns use sel() from '@nextsparkjs/core/lib/selectors'\n *\n * @example\n * sel('auth', 'login', 'form') // 'auth.login.form'\n */\nexport function sel(...parts: string[]): string {\n return parts.filter(Boolean).join('.')\n}\n\n// =============================================================================\n// ENVIRONMENT DETECTION\n// =============================================================================\n\n/**\n * Environment-based testing mode\n */\nconst isDevelopment = process.env.NODE_ENV === 'development'\nconst isTest = process.env.NODE_ENV === 'test'\nconst enableTestingAttributes = isDevelopment || isTest\n\n// =============================================================================\n// ATTRIBUTE GENERATORS\n// =============================================================================\n\n/**\n * Create state-based data attributes for conditional testing\n *\n * @param state - Current state value\n * @returns State attribute value\n */\nexport function createStateAttr(\n state: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n): string {\n return state\n}\n\n/**\n * Create priority-based data attributes\n *\n * @param priority - Priority level\n * @returns Priority attribute value\n */\nexport function createPriorityAttr(priority: 'low' | 'medium' | 'high'): string {\n return priority\n}\n\n/**\n * Generate testing props object for components\n *\n * @param config - Testing configuration\n * @returns Testing props object\n */\nexport function createTestingProps(config: {\n testId?: string\n cyId?: string\n state?: 'active' | 'completed' | 'pending' | 'loading' | 'error'\n priority?: 'low' | 'medium' | 'high'\n taskId?: string\n userId?: string\n}) {\n const props: Record<string, string | undefined> = {}\n\n if (config.testId) {\n props['data-testid'] = enableTestingAttributes ? config.testId : undefined\n }\n\n if (config.cyId) {\n props['data-cy'] = enableTestingAttributes ? config.cyId : undefined\n }\n\n if (config.state) {\n props['data-state'] = config.state\n }\n\n if (config.priority) {\n props['data-priority'] = config.priority\n }\n\n if (config.taskId) {\n props['data-task-id'] = config.taskId\n }\n\n if (config.userId) {\n props['data-user-id'] = config.userId\n }\n\n // Filter out undefined values\n return Object.fromEntries(\n Object.entries(props).filter((entry) => entry[1] !== undefined)\n )\n}\n\n/**\n * Accessibility helper for dynamic aria-label generation\n *\n * @param template - Label template with placeholders\n * @param values - Values to replace placeholders\n * @returns Formatted aria-label\n */\nexport function createAriaLabel(\n template: string,\n values: Record<string, string | number | boolean>\n): string {\n return template.replace(/\\{(\\w+)\\}/g, (match, key) => {\n return String(values[key] ?? match)\n })\n}\n\n// =============================================================================\n// KEYBOARD HELPERS\n// =============================================================================\n\n/**\n * Keyboard navigation helpers\n */\nexport const keyboardHelpers = {\n /**\n * Handle Enter and Space key activation\n */\n createActivationHandler: (onActivate: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Enter' || e.key === ' ') {\n e.preventDefault()\n onActivate()\n }\n }\n },\n\n /**\n * Handle Escape key for closing\n */\n createEscapeHandler: (onClose: () => void) => {\n return (e: React.KeyboardEvent) => {\n if (e.key === 'Escape') {\n e.preventDefault()\n onClose()\n }\n }\n },\n\n /**\n * Handle arrow navigation in lists\n */\n createArrowNavigationHandler: (\n currentIndex: number,\n maxIndex: number,\n onIndexChange: (index: number) => void\n ) => {\n return (e: React.KeyboardEvent) => {\n switch (e.key) {\n case 'ArrowDown':\n e.preventDefault()\n onIndexChange(currentIndex < maxIndex ? currentIndex + 1 : 0)\n break\n case 'ArrowUp':\n e.preventDefault()\n onIndexChange(currentIndex > 0 ? currentIndex - 1 : maxIndex)\n break\n }\n }\n },\n}\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@nextsparkjs/testing",
|
|
3
|
-
"version": "0.1.0-beta.
|
|
3
|
+
"version": "0.1.0-beta.55",
|
|
4
4
|
"description": "Testing utilities for NextSpark applications - Selectors, POMs, and Cypress helpers",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"author": "NextSpark <hello@nextspark.dev>",
|
|
@@ -49,11 +49,6 @@
|
|
|
49
49
|
"dist",
|
|
50
50
|
"README.md"
|
|
51
51
|
],
|
|
52
|
-
"scripts": {
|
|
53
|
-
"build": "tsup",
|
|
54
|
-
"dev": "tsup --watch",
|
|
55
|
-
"clean": "rm -rf dist"
|
|
56
|
-
},
|
|
57
52
|
"peerDependencies": {
|
|
58
53
|
"cypress": ">=13.0.0"
|
|
59
54
|
},
|
|
@@ -68,5 +63,10 @@
|
|
|
68
63
|
"cypress": "^14.0.0",
|
|
69
64
|
"tsup": "^8.5.1",
|
|
70
65
|
"typescript": "^5"
|
|
66
|
+
},
|
|
67
|
+
"scripts": {
|
|
68
|
+
"build": "tsup",
|
|
69
|
+
"dev": "tsup --watch",
|
|
70
|
+
"clean": "rm -rf dist"
|
|
71
71
|
}
|
|
72
|
-
}
|
|
72
|
+
}
|