多维表格
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

209 lines
7.6 KiB

import { expect } from '@playwright/test';
import { DashboardPage } from '..';
import BasePage from '../../Base';
import { ToolbarPage } from '../common/Toolbar';
import { TopbarPage } from '../common/Topbar';
export class KanbanPage extends BasePage {
readonly dashboard: DashboardPage;
readonly toolbar: ToolbarPage;
readonly topbar: TopbarPage;
constructor(dashboard: DashboardPage) {
super(dashboard.rootPage);
this.dashboard = dashboard;
this.toolbar = new ToolbarPage(this);
this.topbar = new TopbarPage(this);
}
get() {
return this.dashboard.get().locator('[data-testid="nc-kanban-wrapper"]');
}
card(index: number) {
return this.get().locator(`.ant-card`).nth(index);
}
async openExpandedRow({ index }: { index: number }) {
await this.card(index).click();
await (await this.rootPage.locator('.ant-drawer-body').elementHandle())?.waitForElementState('stable');
}
// todo: Implement
async addOption() {}
async dragDropCard(param: { from: { stack: number; card: number }; to: { stack: number; card: number } }) {
const { from, to } = param;
const srcStack = this.get().locator(`.nc-kanban-stack`).nth(from.stack);
const dstStack = this.get().locator(`.nc-kanban-stack`).nth(to.stack);
const fromCard = srcStack.locator(`.nc-kanban-item`).nth(from.card);
const toCard = dstStack.locator(`.nc-kanban-item`).nth(to.card);
console.log(await fromCard.allTextContents());
console.log(await toCard.allTextContents());
await fromCard.dragTo(toCard, {
force: true,
sourcePosition: { x: 10, y: 10 },
targetPosition: { x: 10, y: 10 },
});
}
async dragDropStack(param: { from: number; to: number }) {
const { from, to } = param;
const [fromStack, toStack] = await Promise.all([
this.rootPage.locator(`.nc-kanban-stack-head >> .nc-kanban-stack-drag-handler`).nth(from),
this.rootPage.locator(`.nc-kanban-stack-head >> .nc-kanban-stack-drag-handler`).nth(to),
]);
await fromStack.dragTo(toStack);
}
async verifyStackCount(param: { count: number }) {
const { count } = param;
await expect(this.get().locator(`.nc-kanban-stack`)).toHaveCount(count);
}
async verifyStackOrder(param: { order: string[] }) {
await this.rootPage.waitForTimeout(1000);
const { order } = param;
const stacks = await this.get().locator(`.nc-kanban-stack`).count();
for (let i = 0; i < stacks; i++) {
const stack = this.get().locator(`.nc-kanban-stack`).nth(i);
await stack.scrollIntoViewIfNeeded();
// Since otherwise stack title will be repeated as title is in two divs, with one having hidden class
const stackTitle = stack.locator(`.nc-kanban-stack-head >> [data-testid="nc-kanban-stack-title"]`);
await stackTitle.waitFor({ state: 'visible' });
await expect(stackTitle).toHaveText(order[i], { ignoreCase: true });
}
}
async verifyStackFooter(param: { count: number[] }) {
const { count } = param;
const stacks = await this.get().locator(`.nc-kanban-stack`).count();
for (let i = 0; i < stacks; i++) {
const stack = this.get().locator(`.nc-kanban-stack`).nth(i);
await stack.scrollIntoViewIfNeeded();
const stackFooter = await stack.locator(`.nc-kanban-data-count`).innerText();
expect(stackFooter).toContain(`${count[i]} record${count[i] !== 1 ? 's' : ''}`);
}
}
async verifyCardCount(param: { count: number[] }) {
const { count } = param;
const stacks = await this.get().locator(`.nc-kanban-stack`).count();
for (let i = 0; i < stacks; i++) {
const stack = this.get().locator(`.nc-kanban-stack`).nth(i);
await stack.scrollIntoViewIfNeeded();
const stackCards = stack.locator(`.nc-kanban-item`);
await expect(stackCards).toHaveCount(count[i]);
}
}
async verifyCardOrder(param: { order: string[]; stackIndex: number }) {
const { order, stackIndex } = param;
const stack = this.get().locator(`.nc-kanban-stack`).nth(stackIndex);
for (let i = 0; i < order.length; i++) {
const card = stack.locator(`.nc-kanban-item`).nth(i);
await (await card.elementHandle())?.waitForElementState('stable');
await card.scrollIntoViewIfNeeded();
const cardTitle = card.locator(`.nc-cell`);
await expect(cardTitle).toHaveText(order[i]);
}
}
// todo: Wait for render to complete
async waitLoading() {
await this.rootPage.waitForTimeout(1000);
}
async addNewStack(param: { title: string }) {
const addNewStack = this.get().locator('.nc-kanban-add-new-stack');
await addNewStack.scrollIntoViewIfNeeded();
const addNewStackBtn = addNewStack.locator('.add-new-stack-btn');
await addNewStackBtn.waitFor({ state: 'visible' });
await addNewStackBtn.click();
const stackTitleInput = addNewStack.getByTestId('nc-kanban-stack-title-input');
await stackTitleInput.waitFor({ state: 'visible' });
await stackTitleInput.fill(param.title);
await this.rootPage.keyboard.press('Enter');
await this.get().getByTestId(`nc-kanban-stack-${param.title}`).waitFor({ state: 'visible' });
}
async renameStack(param: { index: number; title: string }) {
const stackHead = this.get().locator(`.nc-kanban-stack-head`).nth(param.index);
await stackHead.scrollIntoViewIfNeeded();
await this.stackContextMenu({ index: param.index, operation: 'rename-stack' });
const stackTitleInput = stackHead.getByTestId('nc-kanban-stack-title-input');
await stackTitleInput.waitFor({ state: 'visible' });
await stackTitleInput.fill(param.title);
await this.rootPage.keyboard.press('Enter');
await stackTitleInput.waitFor({ state: 'hidden' });
await stackHead.locator('.stack-rename-loader').waitFor({ state: 'hidden' });
await this.get().getByTestId(`nc-kanban-stack-${param.title}`).waitFor({ state: 'visible' });
}
async stackContextMenu({
index,
operation,
}: {
index: number;
operation:
| 'add-new-record'
| 'rename-stack'
| 'collapse-stack'
| 'collapse-all-stack'
| 'expand-all-stack'
| 'delete-stack';
}) {
await this.get().locator(`.nc-kanban-stack-head`).nth(index).getByTestId('nc-kanban-stack-context-menu').click();
const contextMenu = this.rootPage.locator('.nc-dropdown-kanban-stack-context-menu');
await contextMenu.waitFor({ state: 'visible' });
await contextMenu.getByTestId(`nc-kanban-context-menu-${operation}`).click();
await contextMenu.waitFor({ state: 'hidden' });
}
async collapseStack({ index }: { index: number }) {
await this.stackContextMenu({ index, operation: 'collapse-stack' });
}
async collapseAllStack({ index }: { index: number }) {
await this.stackContextMenu({ index, operation: 'collapse-all-stack' });
}
async expandStack(param: { index: number }) {
await this.rootPage.locator(`.nc-kanban-collapsed-stack`).nth(param.index).click();
}
async expandAllStack({ index }: { index: number }) {
await this.stackContextMenu({ index, operation: 'expand-all-stack' });
}
async verifyCollapseStackCount(param: { count: number }) {
await expect(this.rootPage.locator('.nc-kanban-collapsed-stack')).toHaveCount(param.count);
}
async addCard(param: { stackIndex: number }) {
await this.stackContextMenu({ index: param.stackIndex, operation: 'add-new-record' });
}
async deleteStack(param: { index: number }) {
await this.stackContextMenu({ index: param.index, operation: 'delete-stack' });
const confirmationModal = this.rootPage.locator(`div.ant-modal-content`);
await confirmationModal.locator(`button:has-text("Delete Stack")`).click();
await confirmationModal.waitFor({ state: 'hidden' });
}
}