多维表格
 
 
 
 
 
 

321 lines
11 KiB

import { expect, Locator, Page } from '@playwright/test';
import BasePage from '../Base';
import { GridPage } from './Grid';
import { FormPage } from './Form';
import { ExpandedFormPage } from './ExpandedForm';
import { BulkUpdatePage } from './BulkUpdate';
import { ChildList } from './Grid/Column/LTAR/ChildList';
import { LinkRecord } from './Grid/Column/LTAR/LinkRecord';
import { TreeViewPage } from './TreeView';
import { SettingsPage } from './Settings';
import { ViewSidebarPage } from './ViewSidebar';
import { LeftSidebarPage } from './common/LeftSidebar';
import { ProjectViewPage } from './ProjectView';
import { GalleryPage } from './Gallery';
import { KanbanPage } from './Kanban';
import { MapPage } from './Map';
import { ImportAirtablePage } from './Import/Airtable';
import { ImportTemplatePage } from './Import/ImportTemplate';
import { WebhookFormPage } from './WebhookForm';
import { FindRowByScanOverlay } from './FindRowByScanOverlay';
import { SidebarPage } from './Sidebar';
import { DocsPageGroup } from './Docs';
import { ShareProjectButtonPage } from './ShareProjectButton';
import { ProjectTypes } from 'nocodb-sdk';
import { WorkspacePage } from '../WorkspacePage';
import { DetailsPage } from './Details';
import { WorkspaceSettingsObject } from './WorkspaceSettings';
import { CmdJ } from './Command/CmdJPage';
import { CmdK } from './Command/CmdKPage';
import { CmdL } from './Command/CmdLPage';
import { CalendarPage } from './Calendar';
export class DashboardPage extends BasePage {
readonly base: any;
readonly tablesSideBar: Locator;
readonly workspaceMenuLink: Locator;
readonly tabBar: Locator;
readonly treeView: TreeViewPage;
readonly grid: GridPage;
readonly gallery: GalleryPage;
readonly calendar: CalendarPage;
readonly form: FormPage;
readonly kanban: KanbanPage;
readonly map: MapPage;
readonly expandedForm: ExpandedFormPage;
readonly bulkUpdateForm: BulkUpdatePage;
readonly webhookForm: WebhookFormPage;
readonly findRowByScanOverlay: FindRowByScanOverlay;
readonly childList: ChildList;
readonly linkRecord: LinkRecord;
readonly settings: SettingsPage;
readonly viewSidebar: ViewSidebarPage;
readonly leftSidebar: LeftSidebarPage;
readonly baseView: ProjectViewPage;
readonly importAirtable: ImportAirtablePage;
readonly importTemplate = new ImportTemplatePage(this);
readonly docs: DocsPageGroup;
readonly sidebar: SidebarPage;
readonly shareProjectButton: ShareProjectButtonPage;
readonly details: DetailsPage;
readonly workspaceSettings: WorkspaceSettingsObject;
readonly cmdJ: CmdJ;
readonly cmdK: CmdK;
readonly cmdL: CmdL;
constructor(rootPage: Page, base: any) {
super(rootPage);
this.base = base;
this.tablesSideBar = rootPage.locator('.nc-treeview-container');
this.workspaceMenuLink = rootPage.getByTestId('nc-base-menu');
this.tabBar = rootPage.locator('.nc-tab-bar');
this.treeView = new TreeViewPage(this, base);
this.grid = new GridPage(this);
this.gallery = new GalleryPage(this);
this.calendar = new CalendarPage(this);
this.form = new FormPage(this);
this.kanban = new KanbanPage(this);
this.map = new MapPage(this);
this.expandedForm = new ExpandedFormPage(this);
this.bulkUpdateForm = new BulkUpdatePage(this);
this.webhookForm = new WebhookFormPage(this);
this.findRowByScanOverlay = new FindRowByScanOverlay(this);
this.childList = new ChildList(this);
this.linkRecord = new LinkRecord(this);
this.settings = new SettingsPage(this);
this.viewSidebar = new ViewSidebarPage(this);
this.leftSidebar = new LeftSidebarPage(this);
this.baseView = new ProjectViewPage(this);
this.importAirtable = new ImportAirtablePage(this);
this.sidebar = new SidebarPage(this);
this.docs = new DocsPageGroup(this);
this.shareProjectButton = new ShareProjectButtonPage(this);
this.details = new DetailsPage(this);
this.workspaceSettings = new WorkspaceSettingsObject(this);
this.cmdJ = new CmdJ(this);
this.cmdK = new CmdK(this);
this.cmdL = new CmdL(this);
}
get() {
return this.rootPage.locator('html');
}
async goto() {
await this.rootPage.goto(`/#/${this.base.fk_workspace_id}/${this.base.id}`);
}
getProjectMenuLink({ title }: { title: string }) {
return this.rootPage.locator(`div.nc-base-menu-item:has-text("${title}")`);
}
async clickOnBaseMenuLink() {
const baseMenuLocator = this.rootPage.locator(`.base-title-node:has-text("${this.base.title}")`).first();
await baseMenuLocator.waitFor({ state: 'visible' });
await baseMenuLocator.scrollIntoViewIfNeeded();
await baseMenuLocator.hover();
await baseMenuLocator.locator('[data-testid="nc-sidebar-context-menu"]').first().click();
}
async verifyTeamAndSettingsLinkIsVisible() {
await this.clickOnBaseMenuLink();
const teamAndSettingsLink = this.getProjectMenuLink({ title: ' Team & Settings' });
await expect(teamAndSettingsLink).toBeVisible();
await this.clickOnBaseMenuLink();
}
async verifyTeamAndSettingsLinkIsNotVisible() {
await this.clickOnBaseMenuLink();
const teamAndSettingsLink = this.getProjectMenuLink({ title: ' Team & Settings' });
await expect(teamAndSettingsLink).not.toBeVisible();
await this.clickOnBaseMenuLink();
}
async gotoSettings() {
await this.clickOnBaseMenuLink();
await this.rootPage.locator('.ant-dropdown').locator(`.nc-menu-item:has-text("Settings")`).click();
}
async gotoProjectSubMenu({ title }: { title: string }) {
await this.clickOnBaseMenuLink();
await this.rootPage.locator(`div.nc-base-menu-item:has-text("${title}")`).click();
}
async verifyInTabBar({ title }: { title: string }) {
await this.tabBar.textContent().then(text => expect(text).toContain(title));
}
async closeTab({ title }: { title: string }) {}
async clickHome() {
await this.leftSidebar.clickHome();
// wait for workspace page to render
const workspacePage = new WorkspacePage(this.rootPage);
await workspacePage.waitFor({ state: 'visible' });
}
async verifyOpenedTab({ title, mode = 'standard', emoji }: { title: string; mode?: string; emoji?: string }) {
await this.tabBar.locator(`.ant-tabs-tab-active:has-text("${title}")`).isVisible();
if (emoji) {
await expect(
this.tabBar.locator(`.ant-tabs-tab-active:has-text("${title}")`).getByTestId(`nc-tab-icon-emojione:${emoji}`)
).toBeVisible();
}
}
async verifyTabIsNotOpened({ title }: { title: string }) {
await expect(this.tabBar.locator(`.ant-tabs-tab:has-text("${title}")`)).not.toBeVisible();
}
// Hence will wait till contents are visible
async waitForTabRender({
title,
mode = 'standard',
type = ProjectTypes.DATABASE,
}: {
title: string;
mode?: string;
type?: ProjectTypes;
}) {}
// When a tab is opened, it is not always immediately visible.
async toggleMobileMode() {
await this.clickOnBaseMenuLink();
const projMenu = this.rootPage.locator('.nc-dropdown-base-menu');
await projMenu.locator('[data-menu-id="mobile-mode"]:visible').click();
await this.clickOnBaseMenuLink();
}
async signOut() {
await this.sidebar.userMenu.click();
await this.rootPage.waitForTimeout(1000);
await this.rootPage.getByTestId('nc-sidebar-user-logout').waitFor({ state: 'visible' });
await this.sidebar.userMenu.clickLogout();
await this.rootPage.waitForTimeout(1000);
await this.rootPage.locator('[data-testid="nc-form-signin"]:visible').waitFor();
await new Promise(resolve => setTimeout(resolve, 150));
}
async validateProjectMenu(param: { role: string; mode?: string }) {
await this.rootPage.locator('[data-testid="nc-base-menu"]').click();
const pMenu = this.rootPage.locator(`.nc-dropdown-base-menu:visible`);
// menu items
let menuItems = {
creator: [
'Copy Base Info',
'Swagger: REST APIs',
'Copy Auth Token',
'Team & Settings',
'Themes',
'Preview as',
'Language',
'Account',
],
editor: ['Copy Base Info', 'Swagger: REST APIs', 'Copy Auth Token', 'Language', 'Account'],
commenter: ['Copy Base Info', 'Copy Auth Token', 'Language', 'Account'],
viewer: ['Copy Base Info', 'Copy Auth Token', 'Language', 'Account'],
};
if (param?.mode === 'shareBase') {
menuItems = {
creator: [],
commenter: [],
editor: ['Language'],
viewer: ['Language'],
};
}
// common items
for (const item of menuItems[param.role]) {
await expect(pMenu).toContainText(item);
}
await this.rootPage.locator('[data-testid="nc-base-menu"]').click();
}
// Wait for the loader i.e the loader than appears when rows are being fetched, saved etc on the top right of dashboard
async waitForLoaderToDisappear() {
await this.rootPage.locator('[data-testid="nc-loading"]').waitFor({ state: 'hidden' });
}
async closeAllTabs() {
const tab = this.tabBar.locator(`.ant-tabs-tab`);
const tabCount = await tab.count();
for (let i = 0; i < tabCount; i++) {
await tab.nth(i).locator('button.ant-tabs-tab-remove').click();
await this.rootPage.waitForTimeout(200);
}
}
/* async closeAllTabs() {
await this.tabBar.locator(`.ant-tabs-tab`).waitFor({ state: 'visible' });
const tab = await this.tabBar.locator(`.ant-tabs-tab`);
const tabCount = await tab.count();
for (let i = 0; i < tabCount; i++) {
await tab.nth(i).locator('button.ant-tabs-tab-remove').click();
await tab.nth(i).waitFor({ state: 'detached' });
}
}*/
async validateWorkspaceMenu(param: { role: string; mode?: string }) {
await this.grid.workspaceMenu.toggle();
await this.grid.workspaceMenu.get().waitFor({ state: 'visible' });
const pMenu = this.grid.workspaceMenu.get();
// menu items
let menuItems = {
creator: ['Collaborators', 'Settings', 'Copy Auth Token', 'Themes', 'Preview as', 'Language', 'Account'],
editor: ['Collaborators', 'Settings', 'Copy Auth Token', 'Language', 'Account'],
commenter: ['Collaborators', 'Settings', 'Copy Auth Token', 'Language', 'Account'],
viewer: ['Collaborators', 'Settings', 'Copy Auth Token', 'Language', 'Account'],
};
if (param?.mode === 'shareBase') {
menuItems = {
creator: [],
commenter: [],
editor: ['Language'],
viewer: ['Language'],
};
}
// common items
for (const item of menuItems[param.role]) {
await expect(pMenu).toContainText(item);
}
// menuItems.creator is a super set. validate if the corresponding items missing in editor, commenter, viewer are not present
for (const item of menuItems.creator) {
if (!menuItems[param.role].includes(item)) {
await expect(pMenu).not.toContainText(item);
}
}
await this.grid.workspaceMenu.toggle();
}
private async _waitForDocsTabRender({ title, mode }: { title: string; mode: string }) {
await this.tabBar.locator(`.ant-tabs-tab-active:has-text("${title}")`).waitFor();
// wait active tab animation to finish
await expect
.poll(async () => {
return await this.tabBar.getByTestId(`nc-root-tabs-${title}`).evaluate(el => {
return window.getComputedStyle(el).getPropertyValue('color');
});
})
.toBe('rgb(67, 81, 232)');
await this.rootPage.waitForTimeout(500);
}
}