Files
myeasycms-v2/apps/e2e/tests/invitations/invitations.po.ts
Giancarlo Buomprisco fa2fa9a15c chore: improve invitation flow, update project dependencies and documentation for Next.js 16 (#408)
* chore: update project dependencies and documentation for Next.js 16

- Upgraded Next.js from version 15 to 16 across various documentation files and components.
- Updated references to Next.js 16 in AGENTS.md and CLAUDE.md for consistency.
- Incremented application version to 2.21.0 in package.json.
- Refactored identity setup components to improve user experience and added confirmation dialogs for authentication methods.
- Enhanced invitation flow with new logic for handling user redirection and token generation.

* refactor: streamline invitation flow in e2e tests

- Simplified the invitation flow test by using a predefined email instead of generating a random one.
- Removed unnecessary steps such as clearing cookies and reloading the page before user sign-up.
- Enhanced clarity by eliminating commented-out code related to identity verification and user membership checks.

* refactor: improve code readability in IdentitiesPage and UpdatePasswordForm components

- Enhanced formatting of JSX elements in IdentitiesPage and UpdatePasswordForm for better readability.
- Adjusted indentation and line breaks to maintain consistent coding style across components.

* refactor: enhance LinkAccountsList component with user redirection logic

- Updated the LinkAccountsList component to include a redirectToPath option in the useLinkIdentityWithProvider hook for improved user experience.
- Removed redundant user hook declaration to streamline the code structure.

* refactor: update account setup logic in JoinTeamAccountPage

- Introduced a check for email-only authentication support to streamline account setup requirements.
- Adjusted the conditions for determining if a new account should set up additional authentication methods, enhancing user experience for new users.
2025-11-05 11:39:08 +07:00

171 lines
4.3 KiB
TypeScript

import { Page, expect } from '@playwright/test';
import { AuthPageObject } from '../authentication/auth.po';
import { TeamAccountsPageObject } from '../team-accounts/team-accounts.po';
export class InvitationsPageObject {
private readonly page: Page;
public auth: AuthPageObject;
public teamAccounts: TeamAccountsPageObject;
constructor(page: Page) {
this.page = page;
this.auth = new AuthPageObject(page);
this.teamAccounts = new TeamAccountsPageObject(page);
}
setup() {
return this.teamAccounts.setup();
}
public async inviteMembers(
invites: Array<{
email: string;
role: string;
}>,
) {
const form = this.getInviteForm();
for (let index = 0; index < invites.length; index++) {
const invite = invites[index];
if (!invite) {
continue;
}
console.log(`Inviting ${invite.email} with role ${invite.role}...`);
const nth = index + 1;
await this.page.fill(
`[data-test="invite-member-form-item"]:nth-child(${nth}) [data-test="invite-email-input"]`,
invite.email,
);
await this.page.click(
`[data-test="invite-member-form-item"]:nth-child(${nth}) [data-test="role-selector-trigger"]`,
);
await this.page.click(`[data-test="role-option-${invite.role}"]`);
if (index < invites.length - 1) {
await form.locator('[data-test="add-new-invite-button"]').click();
}
}
await form.locator('button[type="submit"]').click();
}
navigateToMembers() {
return expect(async () => {
await this.page
.locator('a', {
hasText: 'Members',
})
.click();
await this.page.waitForURL('**/home/*/members');
}).toPass();
}
async openInviteForm() {
await expect(async () => {
await this.page.click('[data-test="invite-members-form-trigger"]');
return await expect(this.getInviteForm()).toBeVisible();
}).toPass();
}
getInvitations() {
return this.page.locator('[data-test="invitation-email"]');
}
async deleteInvitation(email: string) {
const actions = this.getInvitationRow(email).getByRole('button');
await actions.click();
await this.page.locator('[data-test="remove-invitation-trigger"]').click();
await this.page.click(
'[data-test="delete-invitation-form"] button[type="submit"]',
);
}
getInvitationRow(email: string) {
return this.page.getByRole('row', { name: email });
}
async updateInvitation(email: string, role: string) {
const row = this.getInvitationRow(email);
const actions = row.getByRole('button');
await actions.click();
await this.page.locator('[data-test="update-invitation-trigger"]').click();
await this.page.click(`[data-test="role-selector-trigger"]`);
await this.page.click(`[data-test="role-option-${role}"]`);
await this.page.click(
'[data-test="update-invitation-form"] button[type="submit"]',
);
}
async acceptInvitation() {
console.log('Accepting invitation...');
const click = this.page
.locator('[data-test="join-team-form"] button[type="submit"]')
.click();
const response = this.page.waitForResponse((response) => {
return (
response.url().includes('/join') &&
response.request().method() === 'POST'
);
});
await Promise.all([click, response]);
// wait for animation to complete
await this.page.waitForTimeout(500);
// skip authentication setup
const continueButton = this.page.locator(
'[data-test="continue-button"]',
);
if (
await continueButton.isVisible({
timeout: 1000,
})
) {
await continueButton.click();
// Handle confirmation dialog that appears when skipping without adding auth
const confirmationDialog = this.page.locator(
'[data-test="no-auth-method-dialog"]',
);
if (
await confirmationDialog.isVisible({
timeout: 2000,
})
) {
console.log('Confirmation dialog appeared, clicking Continue...');
await this.page
.locator('[data-test="no-auth-dialog-continue"]')
.click();
}
}
// wait for redirect to account home
await this.page.waitForURL(new RegExp('/home/[a-z0-9-]+'));
}
private getInviteForm() {
return this.page.locator('[data-test="invite-members-form"]');
}
}