@immense/vue-pom-generator
v1.0.39
Published
Injects data-testid attributes for all interactive elements and generates page object models for every page.
Readme
@immense/vue-pom-generator
@immense/vue-pom-generator is a Vite plugin for Vue 3 that does two compile-time jobs:
- it injects a test-id attribute into interactive elements in
.vuetemplates - it turns the collected ids into an aggregated Page Object Model library for Playwright, with optional Playwright fixtures and optional C# output
If you already use Playwright with getByTestId, the point is simple: this package removes the repetitive work of keeping test ids in sync with Vue templates and then hand-writing page objects around those ids.
What this does
- Injects test ids during Vue compilation, not at runtime. It hooks into the Vue template compiler and rewrites the compiled template output.
- Uses real template signals to name ids and methods. Click handlers,
v-model,id/name,:to, wrapper configuration, and a few targeted fallbacks all feed the generated API. - Generates one aggregated TypeScript POM file plus a stable
index.tsbarrel. - Can generate Playwright fixtures so tests can request
userListPageinstead of constructingnew UserListPage(page)manually. - Can emit a single C# POM file for Playwright .NET consumers.
- Exposes
virtual:testidsso your app can import the current collected test-id manifest at runtime. - Ships ESLint rules to remove legacy manually-authored test ids and to discourage raw locator actions on generated getters.
What this does not do
- It is not a runtime DOM crawler. It only knows what it can learn from Vue SFC templates and router introspection.
- It does not fully understand arbitrary wrapper components automatically. There is some wrapper inference for simple local SFCs and some naming conventions, but serious design-system components still need
injection.nativeWrappers. - It does not fully fill route params for you. Generated
goTo()/goToSelf()methods use the discovered route template literally. A route like/users/:idstill needs a real/users/123when you actually navigate. - It does not auto-attach every file in
pom/custom. Custom helpers are imported, but they only affect generated classes when you explicitly configure attachments, or when they match the built-inToggle/Checkboxwidget conventions. - It does not make override classes globally replace generated classes.
pom/overridesonly changes generated fixture instantiation. Direct imports from the generated barrel still give you the generated class unless you import your override yourself. - The C# emitter is not feature-parity with the TypeScript emitter. It emits locator/action classes, but not Playwright fixtures, not helper attachments, and not the same route helper surface.
A minimal Vue example
Given a view like this:
<!-- src/views/UserEditorPage.vue -->
<template>
<form>
<input v-model="emailAddress" />
<button @click="save">Save</button>
</form>
</template>The current implementation will generate a surface in this shape:
<input data-testid="UserEditorPage-EmailAddress-input">
<button data-testid="UserEditorPage-Save-button">Save</button>export class UserEditorPage extends BasePage {
get EmailAddressInput() { /* Playwright locator */ }
get SaveButton() { /* Playwright locator */ }
async typeEmailAddress(text: string, annotationText = "") { /* ... */ }
async clickSave(wait: boolean = true) { /* ... */ }
}That example is intentionally small, but it shows the real contract:
- the Vue template stays close to normal app code
- the test ids are derived at compile time
- the generated class exposes both raw locators and typed action methods
How ids are actually derived today
The generator does not use one naming trick. It layers several signals.
- Click actions prefer semantic handler names such as
save,openDetails, orrunImport. - Inputs and wrapper components prefer
v-model, wrappervalueAttribute, or related model-like bindings. - Native elements also consider
id/nameattributes. - Router links /
:tobindings can contribute route-based naming and typed navigation return types when the target can be resolved. - Wrapper components can be explicit (
nativeWrappers) or inferred from simple local SFC templates. - Fallback naming exists, but it is intentionally conservative. That is why
generation.nameCollisionBehaviorexists.
Important limit: wrapper inference is helpful, not magical. The current implementation recursively inspects simple local SFC templates for the first inferable primitive (input, textarea, select, button, vselect, radio/checkbox inputs). It also recognizes some naming patterns like *Button. For anything more complex, configure nativeWrappers explicitly.
Playwright before/after examples
1) Raw selectors vs generated POM methods
Before
import { test, expect } from "@playwright/test";
test("saves a user", async ({ page }) => {
await page.goto("/users/new");
await page.getByTestId("UserEditorPage-EmailAddress-input").fill("[email protected]");
await page.getByTestId("UserEditorPage-Save-button").click();
await expect(page.getByText("Saved")).toBeVisible();
});After
import { test, expect } from "@playwright/test";
import { UserEditorPage } from "../tests/playwright/__generated__";
test("saves a user", async ({ page }) => {
const userEditorPage = new UserEditorPage(page);
await userEditorPage.typeEmailAddress("[email protected]");
await userEditorPage.clickSave();
await expect(page.getByText("Saved")).toBeVisible();
});Why this is better:
- selector strings stop leaking into tests
- refactors are localized to generated output instead of every test file
- you still keep access to raw locators when you need them
Without the generator, you still can write stable Playwright tests, but you keep manually maintaining both the selector strings and the page-object layer around them.
2) Manual page-object construction vs generated fixtures
Before
import { test, expect } from "@playwright/test";
import { UserListPage } from "../tests/playwright/__generated__";
test("shows the list", async ({ page }) => {
const userListPage = new UserListPage(page);
await userListPage.goTo();
await expect(userListPage.CreateButton).toBeVisible();
});After
import { test, expect } from "../tests/playwright/__generated__/fixtures.g";
test("shows the list", async ({ userListPage }) => {
await userListPage.goTo();
await expect(userListPage.CreateButton).toBeVisible();
});Why this is better:
- less setup noise in tests
- fixture types stay aligned with generated classes
- if
tests/playwright/pom/overrides/UserListPage.tsexists, the fixture will instantiate that override class automatically
Without fixtures, you still use the generated POMs, but every test has to construct them manually.
3) Nested helper calls vs flattened helper methods
Suppose you have a custom grid helper in tests/playwright/pom/custom/Grid.ts.
Without flatten
test("filters the grid", async ({ usersPage }) => {
await usersPage.grid.Search("[email protected]");
});With flatten: true
test("filters the grid", async ({ usersPage }) => {
await usersPage.Search("[email protected]");
});Why this is better:
- the test reads like the page API, not the internal helper graph
- the page still keeps the explicit
gridproperty when you want it
Without flatten, helper composition is still available; it is just more explicit and more nested.
Install
npm install @immense/vue-pom-generatorExports:
createVuePomGeneratorPlugins()vuePomGenerator()(alias)defineVuePomGeneratorConfig()@immense/vue-pom-generator/eslint
Basic Vite setup
import { defineConfig } from "vite";
import { defineVuePomGeneratorConfig, vuePomGenerator } from "@immense/vue-pom-generator";
const pomConfig = defineVuePomGeneratorConfig({
vueOptions: {
script: { defineModel: true, propsDestructure: true },
},
logging: { verbosity: "info" },
injection: {
attribute: "data-testid",
viewsDir: "src/views",
scanDirs: ["src"],
wrapperSearchRoots: ["../shared-ui/src/components"],
nativeWrappers: {
AppButton: { role: "button" },
AppTextField: { role: "input" },
AppRadioGroup: { role: "radio", requiresOptionDataTestIdPrefix: true },
},
excludeComponents: ["LegacyWidget"],
existingIdBehavior: "preserve",
},
generation: {
emit: ["ts", "csharp"],
csharp: {
namespace: "MyProject.Tests.Generated",
},
outDir: "tests/playwright/__generated__",
nameCollisionBehavior: "suffix",
router: {
entry: "src/router/index.ts",
moduleShims: {
"@/config/app-insights": {
getAppInsights: () => null,
},
"@/stores/alerts": ["useAlertsStore"],
},
},
playwright: {
fixtures: true,
customPoms: {
dir: "tests/playwright/pom/custom",
importAliases: {
Grid: "UserGridHelper",
},
attachments: [
{
className: "Grid",
propertyName: "grid",
attachWhenUsesComponents: ["DataGrid"],
attachTo: "both",
flatten: true,
},
],
},
},
},
});
export default defineConfig({
plugins: [...vuePomGenerator(pomConfig)],
});Important Vite ownership rule
By default, this package creates and returns its own @vitejs/plugin-vue instance.
That means:
- standard Vue app: spread
...vuePomGenerator(config)and do not separately passvue()into the same helper - app-owned Vue plugin / Nuxt / special Vite setup: set
vuePluginOwnership: "external", addvue()yourself, and let this package patch the resolved Vue plugin instead
Example:
import vue from "@vitejs/plugin-vue";
import { defineConfig } from "vite";
import { defineVuePomGeneratorConfig, vuePomGenerator } from "@immense/vue-pom-generator";
const pomConfig = defineVuePomGeneratorConfig({
vuePluginOwnership: "external",
});
export default defineConfig({
plugins: [
vue(),
...vuePomGenerator(pomConfig),
],
});Nuxt-style routing also uses the resolved app-owned Vue plugin. In practice, if you use generation.router.type: "nuxt", think in terms of external Vue plugin ownership.
What gets generated
By default, generation writes to tests/playwright/__generated__.
TypeScript output:
page-object-models.g.ts— aggregated generated classesindex.ts— stable barrel re-exportingpage-object-models.g_pom-runtime/— copied runtime support files used by the aggregated output
Optional Playwright fixture output:
fixtures.g.tsnext to the POMs by default- or a custom directory / file path when configured
Optional C# output:
page-object-models.g.cs
If you emit outside a __generated__ path, the generator also manages .gitattributes entries for generated files.
Actual Vite dev/build behavior
This is important if you are deciding whether the tool will fit into a real codebase.
- Dev server: on startup, it scans the configured
scanDirs, compiles each.vuefile into a snapshot, writes the aggregated outputs once, then batches add/change/delete events and regenerates incrementally. - Build: it generates from the richest build pass it sees, which matters because Vite can run multiple passes (for example SSR plus client). The generator avoids letting a thinner pass clobber a richer one.
- Always-on virtual module:
virtual:testidsis registered whether generation is enabled or disabled. - Generation can be disabled:
generation: falsestill keeps compile-time test-id injection and the virtual module, but skips emitted POM files.
Router-aware navigation: the real semantics
This package has two router-related behaviors, and they are easy to overstate.
1) Typed navigation methods from :to
When the generator can statically resolve a :to target, it can emit navigation methods that return the target POM type.
That enables patterns like:
await userListPage.goToCreateUser().typeEmailAddress("[email protected]");What is actually supported well today:
- literal string paths, such as
:to="'/users'" - object literals with
nameorpath, such as:to="{ name: 'users' }" - object literals with
paramskeys, enough for target-type resolution
What is not fully supported:
- arbitrary computed
:toexpressions - parameter-aware
goToSelf()URL filling - exposing rich route-param metadata on the generated POM surface
2) View-level route, goTo(), and goToSelf()
When generation.router is enabled, each view POM gets:
static readonly route: { template: string } | nullasync goTo()async goToSelf()
Important caveats:
goToSelf()literally doespage.goto(route.template)- a dynamic route template like
/users/:idstays/users/:id - if a component is matched by multiple routes, the generator currently picks one route template (the shortest one)
So the safe rule is:
- use generated
goTo()for simple/static routes - for dynamic routes, navigate with a real URL yourself or wrap that behavior in your override/custom code
Why moduleShims exists
Vue-router introspection SSR-loads your router entry through Vite. Real routers often import browser-only or application-only modules that do not belong in an introspection pass.
moduleShims exists so you can replace those imports just for route discovery.
string[]means “create no-op exported functions with these names”Record<string, fn>means “use these exact shim implementations”- wildcard
*exports are not supported
Without moduleShims, router introspection can fail even though your app itself runs fine.
pom/custom and pom/overrides: what those folders really mean
These two directories solve different problems.
pom/custom
Default: tests/playwright/pom/custom
This directory is for handwritten helper classes that the aggregated TypeScript output can import.
It is the default helper directory even if you omit generation.playwright.customPoms entirely.
Actual current behavior:
- the generator scans the directory non-recursively
- it imports top-level
.tsfiles only - it expects the file basename to match the exported class name (
Grid.ts→export class Grid {}) - the helper files are imported into the generated aggregate; they are not automatically attached everywhere
What it is for:
- wrappers around third-party widgets
- reusable page fragments with custom methods
- helper objects you want attached conditionally to generated pages/components
What it is not:
- a magic auto-discovery system that wires every helper into every page
- a replacement for
attachments
pom/overrides
Default convention: sibling overrides/ next to customPoms.dir
If customPoms.dir is:
tests/playwright/pom/customthen fixtures look for overrides in:
tests/playwright/pom/overridesActual current behavior:
- the override directory is inferred, not separately configurable
- generated fixtures check for
overrides/<ClassName>.ts - when the file exists, the fixture instantiates the override class instead of the generated class
- the generated POM barrel does not automatically re-export the override in place of the generated class
That means fixture override preference is real, but it is specifically a fixture-time constructor preference, not a global import replacement mechanism.
A typical override looks like this:
// tests/playwright/pom/overrides/UserListPage.ts
import { UserListPage as GeneratedUserListPage } from "../../__generated__/page-object-models.g";
export class UserListPage extends GeneratedUserListPage {
async openFirstUser() {
await this.clickOpen();
}
}Helper imports, aliasing, conditional wiring, and flatten
This is where most people need precision.
Helper imports
Every .ts file in customPoms.dir becomes an import in the aggregated TypeScript output.
Benefits:
- helpers are typechecked as normal TypeScript
- generated pages can compose them instead of duplicating logic
Without helper imports, generated classes only know about generated pages/components plus the built-in runtime support.
importAliases
importAliases changes the local import identifier used in generated output.
Example:
customPoms: {
importAliases: {
Grid: "UserGridHelper",
},
}This is useful when:
- you want a clearer local helper name
- a helper name would otherwise collide with a generated POM class name
Important semantic detail: attachments[].className still refers to the helper's real exported class / file basename (Grid), not the alias (UserGridHelper).
Without importAliases, the generator uses the basename as-is.
importNameCollisionBehavior
Current options:
"error"(default)"alias"
Why it exists:
- the aggregated file can import both generated classes and handwritten helpers
- collisions are easy when a helper and a generated class share the same name
What happens:
"error": generation fails and tells you to rename or alias the helper"alias": the generator auto-aliases the helper import (for exampleGridCustom)
Without this option, collisions would quietly create ambiguous generated code.
Conditional helper wiring (attachments)
Attachments decide when a handwritten helper becomes a property on a generated page/component.
Example:
attachments: [
{
className: "Grid",
propertyName: "grid",
attachWhenUsesComponents: ["DataGrid"],
attachTo: "both",
flatten: true,
},
]Actual semantics:
- the helper must exist in
customPoms.dir - the generated page/component must use at least one component named in
attachWhenUsesComponents - matching is based on the component usage collected from the Vue template, not runtime inspection
- the generated constructor instantiates the helper as
new Helper(page, this) attachTodefaults to"views"
Why it exists:
- you usually do not want every helper on every page
- helper attachment is often driven by the presence of a specific UI widget
Without attachments, your helper class can still exist, but generated POMs will not automatically expose it.
Important caveat: if the helper file is missing, the generator currently skips the attachment instead of failing.
flatten
flatten: true tells the generator to create pass-through methods on the generated class that forward to the attachment.
If Grid has:
export class Grid {
constructor(page: Page, owner: object) {}
Search(text: string) {}
}then the generated page can expose either:
page.grid.Search("[email protected]");or, with flatten: true:
page.Search("[email protected]");Actual current rules:
- only public instance methods are candidates
- method signatures are parsed from the helper class source
- complex/unsupported parameter shapes may prevent flattening for that method
- passthroughs are only emitted when the method name is unambiguous across flatten-enabled attachments
- passthroughs are skipped if they would collide with an existing generated method, attachment property, child component property, or widget property
flattenaffects methods, not fields/getters
Without flatten, helper composition still works; you just call through the helper property explicitly.
Playwright fixtures: actual behavior and caveats
When generation.playwright.fixtures is enabled, the generator emits a strongly typed Playwright fixture module.
What it gives you:
- lower-camel-case fixtures for views (
UserListPage→userListPage) - lower-camel-case fixtures for component classes too
pomFactory.create(Ctor)for ad-hoc page-object construction inside tests- an
animationoption that wires the generated runtime's pointer settings
Current caveats:
- there are no generated
openXPagehelpers; tests callgoTo()explicitly when available - override preference only affects fixture construction
- component fixtures are skipped when their lower-camel-case name would collide with reserved Playwright fixture names such as
page,context,browser, orrequest - an override class still needs a
new (page)-compatible constructor because that is what fixtures call
TypeScript vs C# output
TypeScript output
This is the main surface and the most complete implementation.
It includes:
- aggregated page/component classes
- child-component composition
- typed navigation return types from resolvable
:to - view-level
route/goTo()/goToSelf()when router generation is enabled - custom helper imports and attachments
- optional Playwright fixtures
C# output
Enable with:
generation: {
emit: ["ts", "csharp"],
}What the C# emitter currently does well:
- emits a single
page-object-models.g.cs - generates locator properties and action methods
- handles dynamic test-id interpolation
- supports navigation methods that return target page classes when the target is known
What it currently does not do:
- generate Playwright fixtures
- generate helper attachments / flattening
- generate the same view-level
route/goToSelf()helpers as TypeScript - provide feature parity for click instrumentation (
annotationTextis effectively a no-op there)
So if you need the full ergonomic surface, TypeScript is the first-class output today.
virtual:testids
This package registers a Vite virtual module named virtual:testids.
Usage:
import { testIdManifest } from "virtual:testids";
console.log(testIdManifest.UserEditorPage);What it contains:
- an object keyed by component name
- each value is a sorted array of collected test ids for that component
What it is good for:
- runtime inspection
- analytics / logging helpers that need the current generated ids
- debugging what the generator has collected
What it is not:
- a full metadata export
- a generated source file on disk
ESLint rules that actually ship
The package exports @immense/vue-pom-generator/eslint.
remove-existing-test-id-attributes
This is the migration rule that pairs with injection.existingIdBehavior.
What it does:
- removes explicit static attributes like
data-testid="save-button" - removes bound forms like
:data-testid="buttonId" - supports custom attribute names such as
data-qa - also handles object-literal cases inside Vue SFC expressions/scripts that represent test-id attrs
Why it exists:
- mixed manual/generated ids are hard to reason about
existingIdBehavior: "error"is much more usable when a fixer can clean existing code first
Recommended usage:
- run the ESLint rule with
--fix - switch
existingIdBehaviorto"error" - keep the rule in CI so manually-authored ids do not creep back in
Example flat config:
import vueParser from "vue-eslint-parser";
import { plugin as vuePomGeneratorEslint } from "@immense/vue-pom-generator/eslint";
export default [
{
files: ["**/*.vue"],
languageOptions: {
parser: vueParser,
ecmaVersion: 2022,
sourceType: "module",
},
plugins: {
"@immense/vue-pom-generator": vuePomGeneratorEslint,
},
rules: {
"@immense/vue-pom-generator/remove-existing-test-id-attributes": "error",
},
},
];If you use a custom attribute:
rules: {
"@immense/vue-pom-generator/remove-existing-test-id-attributes": ["error", { attribute: "data-qa" }],
}no-raw-locator-action
This rule exists too. It flags direct raw Playwright actions on generated PascalCase getters (for example calling .click() directly on a generated getter) so teams use the generated action methods instead.
Configuration reference
The sections below follow the actual VuePomGeneratorPluginOptions shape from plugin/types.ts.
Top-level options
vueOptions
What it does: Forwards options to
@vitejs/plugin-vue.Why it exists: You still need normal Vue compiler/plugin settings such as
defineModel,propsDestructure, or template compiler tweaks.Benefit: You do not lose ordinary Vue plugin configuration just because this package owns the Vue plugin by default.
Without it: the Vue plugin uses its normal defaults.
Example:
vueOptions: { script: { defineModel: true, propsDestructure: true }, }
vuePluginOwnership
What it does: Chooses whether this package creates
@vitejs/plugin-vueitself ("internal") or patches an app-owned plugin ("external").Why it exists: Some projects want a single explicit
vue()plugin in their Vite config, and Nuxt relies on the resolved app-owned plugin.Benefit: Avoids duplicate Vue-plugin setup and makes Nuxt/external ownership work.
Without it: standard Vue apps default to
"internal".Example:
vuePluginOwnership: "external"
logging.verbosity
What it does: Controls package log volume.
Why it exists: Generator startup scans and regen passes can be noisy when you are debugging, but you usually do not want that noise all the time.
Benefit: Lets you turn on useful lifecycle diagnostics without patching the package.
Without it: default is
"warn".Example:
logging: { verbosity: "debug" }
injection
injection controls compile-time test-id derivation and template rewriting.
injection.attribute
What it does: Sets the attribute name that is injected and later treated as the test id.
Why it exists: Some teams standardize on
data-testid, others ondata-qaordata-cy.Benefit: Keeps the app, Playwright, and generated POMs speaking the same attribute language.
Without it: the generator uses
data-testid.Example:
injection: { attribute: "data-qa" }
injection.viewsDir
What it does: Tells the generator which directory marks a Vue file as a “view” rather than a reusable component.
Why it exists: Views get page-specific behavior such as
goTo()/goToSelf()and are the main candidates for page fixtures.Benefit: Keeps page-level APIs separate from shared component APIs.
Without it: the generator treats
src/viewsas the view root.Example:
injection: { viewsDir: "app/pages" }
injection.nativeWrappers
What it does: Describes wrapper components so the generator can treat them like native controls.
Why it exists: Many Vue apps wrap buttons, inputs, selects, radio groups, or third-party widgets behind design-system components.
Benefit: You get stable control-specific ids and methods instead of generic component-shaped names.
Without it: the generator relies on native elements, limited wrapper inference, and a few naming conventions.
Example:
injection: { nativeWrappers: { AppButton: { role: "button" }, AppTextField: { role: "input" }, AppSelect: { role: "select", valueAttribute: "name" }, AppRadioGroup: { role: "radio", requiresOptionDataTestIdPrefix: true }, }, }
nativeWrappers[...].role
- What it does: Chooses the native behavior to emulate (
button,input,select,vselect,checkbox,toggle,radio,grid). - Why it exists: Role drives both test-id suffixes and generated POM method families (
click...,type...,select..., etc.). - Benefit: The generated API matches what the wrapped control actually does.
- Without it: wrapper components may be treated as generic tags unless they can be inferred.
nativeWrappers[...].valueAttribute
- What it does: Tells the generator which prop on the wrapper should provide the semantic value/name used in the generated test id.
- Why it exists: Some wrappers do not use
v-model, but still have a stable value prop such asname,value, orfield. - Benefit: You get meaningful ids and method names from wrapper props instead of fallback names.
- Without it: wrapper naming falls back to model bindings or other weaker signals.
nativeWrappers[...].requiresOptionDataTestIdPrefix
- What it does: Adds an
option-data-testid-prefixattribute for wrappers that need stable option-level ids. - Why it exists: Radio/select-style wrappers often need a root id plus consistent option ids.
- Benefit: Generated helper methods can target wrapper options consistently.
- Without it: option-level ids may be incomplete or ambiguous.
- Caveat: preserving an existing manual root id on these wrappers can be unsafe, and the current implementation will throw in that case.
injection.excludeComponents
What it does: Opts specific component names out of injection/collection.
Why it exists: Some components are better left alone, or are generated/third-party surfaces you do not want rewritten.
Benefit: Gives you a practical escape hatch without disabling the plugin globally.
Without it: all in-scope Vue components are eligible for injection.
Example:
injection: { excludeComponents: ["LegacyWidget"] }
injection.scanDirs
What it does: Sets which directories are scanned for
.vuefiles when building the POM graph.Why it exists: real projects rarely keep all SFCs under one folder, especially in Nuxt or monorepos.
Benefit: generation sees the same files your app actually uses.
Without it: the generator scans
src.Example:
injection: { scanDirs: ["src", "components", "layouts"] }
injection.wrapperSearchRoots
What it does: Adds extra roots for wrapper inference outside
scanDirs.Why it exists: wrapper components often live in sibling packages or shared UI workspaces.
Benefit: local wrapper inference can still work across package boundaries.
Without it: no extra wrapper lookup is done outside the scanned app directories.
Example:
injection: { wrapperSearchRoots: ["../shared-ui/src/components"] }
injection.existingIdBehavior
- What it does: Chooses what happens when a template already has the target attribute.
- Why it exists: migrations usually start from a mixed codebase with manual ids already present.
- Benefit: lets you migrate gradually (
preserve), force replacement (overwrite), or enforce cleanup (error). - Without it: default is
"preserve". - Current options:
"preserve"— keep the existing attribute"overwrite"— replace it with the generated one"error"— fail compilation
- Important caveat:
"preserve"can still throw for wrappers that require option prefixes, because preserving only the root id would leave nested option ids inconsistent.
generation
Set generation: false to keep injection and virtual:testids but skip emitted POM files.
generation.outDir
What it does: Sets the output directory for generated files.
Why it exists: some repos want generated code somewhere other than the default Playwright location.
Benefit: lets you fit the generator into your existing test layout.
Without it: output goes to
tests/playwright/__generated__.Example:
generation: { outDir: "e2e/generated" }
generation.emit
What it does: Chooses which languages to emit.
Why it exists: TypeScript is the main target, but some teams also want Playwright .NET classes.
Benefit: one collection pass can feed both outputs.
Without it: only TypeScript (
["ts"]) is emitted.Example:
generation: { emit: ["ts", "csharp"] }
generation.csharp.namespace
- What it does: Sets the namespace for generated C# classes.
- Why it exists: generated code needs to fit your test project's namespace conventions.
- Benefit: avoids immediate manual namespace edits.
- Without it: the namespace defaults to
Playwright.Generated.
generation.nameCollisionBehavior
- What it does: Controls what happens when two generated members inside the same class want the same name.
- Why it exists: collisions happen in real templates, especially when multiple elements share the same handler or weak fallback signals.
- Benefit: lets you decide between strictness and convenience.
- Without it: the generator silently suffixes (
"suffix"). - Current options:
"error"— fail fast"warn"— warn and suffix"suffix"— suffix silently
generation.basePageClassPath
- What it does: Points at the BasePage runtime template used for generated TypeScript output.
- Why it exists: some teams want to own or customize the generated runtime base class.
- Benefit: you can keep your own BasePage implementation while still using the generator.
- Without it: the package uses its bundled
class-generation/BasePage.ts.
generation.router
If omitted, router introspection is off.
generation.router.entry
- What it does: Points at the router entry file for standard Vue-router introspection.
- Why it exists: the generator SSR-loads your router to discover names, paths, and target components.
- Benefit: enables typed
:tonavigation targets plus view-level route helpers. - Without it: no Vue-router introspection happens.
- Important detail: the router module must export a default router factory function.
generation.router.type
- What it does: Chooses the router discovery strategy.
- Why it exists: standard Vue-router apps and Nuxt file-based routing need different discovery paths.
- Benefit: one config field covers both app styles.
- Without it: defaults to
"vue-router". - Current options:
"vue-router""nuxt"
generation.router.moduleShims
- What it does: Replaces selected imports only during router introspection.
- Why it exists: router modules often import app-only or browser-only dependencies that should not run during introspection.
- Benefit: keeps route discovery working without reshaping your real router module.
- Without it: the generator attempts to load your router module as-is.
- Allowed forms:
string[]for no-op exported functionsRecord<string, fn>for explicit shim implementations
- Not supported: wildcard
*exports
generation.playwright
This object holds Playwright-specific additions on top of the generated TypeScript classes.
generation.playwright.fixtures
- What it does: Enables emitted Playwright fixtures.
- Why it exists: tests become much cleaner when they can request generated page objects as fixtures.
- Benefit: less boilerplate and automatic override preference.
- Without it: you construct generated page objects manually.
- Accepted forms:
true— emitfixtures.g.tsnext to the generated POMs"path"— if the string ends in.ts/.tsx/.mts/.cts, it is treated as a file path; otherwise as an output directory{ outDir }— emit to a custom directory
generation.playwright.customPoms
- What it does: Configures handwritten helper imports and attachments.
- Why it exists: generated code alone is usually not enough for complex widgets or app-specific test abstractions.
- Benefit: lets generated classes compose handwritten helpers instead of forcing you to pick one approach or the other.
- Without it: the generator still uses the default helper directory (
tests/playwright/pom/custom), but with default naming/collision behavior and no explicit attachments.
generation.playwright.customPoms fields
customPoms.dir
- What it does: Sets the directory scanned for handwritten helper classes.
- Why it exists: helper code needs a predictable home.
- Benefit: generated output can import your handwritten helpers deterministically.
- Without it: defaults to
tests/playwright/pom/custom.
customPoms.importAliases
- What it does: Maps helper basenames to the local names used in generated imports.
- Why it exists: generated files may need clearer local names or collision avoidance.
- Benefit: keeps the aggregate readable and conflict-free.
- Without it: the basename is used directly, except for built-in alias defaults like
Toggle -> ToggleWidgetandCheckbox -> CheckboxWidget.
customPoms.importNameCollisionBehavior
- What it does: Controls how helper import names behave when they collide with generated class names.
- Why it exists: aggregated output shares one import namespace.
- Benefit: lets you fail hard or auto-alias based on team preference.
- Without it: the default is
"error".
customPoms.attachments
- What it does: Declares conditional helper attachments.
- Why it exists: helper composition is usually widget-driven, not universal.
- Benefit: pages/components only get the helpers they actually need.
- Without it: helper files are imported but not attached to generated classes.
customPoms.attachments[]
attachments[].className
- What it does: Names the helper class to attach.
- Why it exists: attachments need to reference a specific imported helper.
- Benefit: explicit helper wiring is easy to read and easy to diff.
- Without it: no helper is attached.
- Important detail: this is the real exported class / file basename, not any alias from
importAliases.
attachments[].propertyName
- What it does: Names the property exposed on the generated page/component.
- Why it exists: attached helpers need a stable public handle.
- Benefit: makes helper composition obvious at the call site.
- Without it: there is no property to access.
attachments[].attachWhenUsesComponents
- What it does: Lists template component names that trigger the attachment.
- Why it exists: helper attachment is based on component usage in the collected Vue template graph.
- Benefit: avoids attaching grid helpers to pages with no grid, modal helpers to pages with no modal, and so on.
- Without it: the helper never attaches.
attachments[].attachTo
- What it does: Limits attachment scope to views, components, or both.
- Why it exists: some helpers belong only on page objects, others also belong on reusable component POMs.
- Benefit: keeps the generated surface smaller and more intentional.
- Without it: the default is
"views". - Current options:
"views""components""both"
attachments[].flatten
- What it does: Generates direct pass-through methods on the generated class for eligible helper methods.
- Why it exists: sometimes nested helper syntax is too noisy in tests.
- Benefit: lets the page API read like a single surface while still being implemented by helpers.
- Without it: you always call through the helper property explicitly.
- Caveat: flattening is conservative; ambiguous or colliding method names are skipped.
Practical adoption advice
If you are evaluating this package critically, the most accurate short version is:
- it is strongest when your team already wants
getByTestId-style Playwright tests - it is strongest in TypeScript-first Playwright projects
- it is strongest when your Vue component library is either native-heavy or can be described with
nativeWrappers - it is useful with router-aware pages, but you should treat dynamic-route
goTo()support as partial and explicit - it becomes much more maintainable when you pair it with the ESLint cleanup rule and a small
pom/customfolder for the genuinely hard widgets
If that matches your codebase, the package removes a surprising amount of repetitive test maintenance. If it does not, the caveats above are the important ones to believe.
