mirror of https://github.com/nocodb/nocodb
Raju Udava
10 months ago
committed by
GitHub
5 changed files with 835 additions and 46 deletions
@ -0,0 +1,318 @@
|
||||
// Fields
|
||||
|
||||
import BasePage from '../../Base'; |
||||
import { expect, Locator } from '@playwright/test'; |
||||
import { DetailsPage } from './index'; |
||||
import { UITypes } from 'nocodb-sdk'; |
||||
|
||||
export class FieldsPage extends BasePage { |
||||
readonly detailsPage: DetailsPage; |
||||
|
||||
readonly searchFieldInput: Locator; |
||||
readonly addNewFieldButton: Locator; |
||||
readonly resetFieldChangesButton: Locator; |
||||
readonly saveChangesButton: Locator; |
||||
|
||||
readonly addOrEditColumn: Locator; |
||||
readonly fieldListWrapper: Locator; |
||||
|
||||
constructor(details: DetailsPage) { |
||||
super(details.rootPage); |
||||
this.detailsPage = details; |
||||
this.searchFieldInput = this.get().getByTestId('nc-field-search-input'); |
||||
this.addNewFieldButton = this.get().getByTestId('nc-field-add-new'); |
||||
this.resetFieldChangesButton = this.get().getByTestId('nc-field-reset'); |
||||
this.saveChangesButton = this.get().getByTestId('nc-field-save-changes'); |
||||
this.addOrEditColumn = this.get().getByTestId('add-or-edit-column'); |
||||
this.fieldListWrapper = this.get().getByTestId('nc-field-list-wrapper'); |
||||
} |
||||
|
||||
get() { |
||||
return this.detailsPage.get().locator('.nc-fields-wrapper'); |
||||
} |
||||
|
||||
async fillSearch({ title }: { title: string }) { |
||||
const searchInput = this.get().getByTestId('nc-field-search-input'); |
||||
await searchInput.click(); |
||||
await searchInput.fill(title); |
||||
} |
||||
|
||||
async clearSearch() { |
||||
await this.get().getByTestId('nc-field-clear-search').click(); |
||||
} |
||||
|
||||
async clickNewField() { |
||||
await this.addNewFieldButton.click(); |
||||
} |
||||
|
||||
async clickRestoreField({ title }: { title: string }) { |
||||
await this.getField({ title }).getByTestId('nc-field-restore-changes').click(); |
||||
} |
||||
|
||||
async createOrUpdate({ |
||||
title, |
||||
type = UITypes.SingleLineText, |
||||
isUpdateMode = false, |
||||
saveChanges = true, |
||||
formula = '', |
||||
qrCodeValueColumnTitle = '', |
||||
barcodeValueColumnTitle = '', |
||||
barcodeFormat = '', |
||||
childTable = '', |
||||
childColumn = '', |
||||
relationType = '', |
||||
rollupType = '', |
||||
format = '', |
||||
dateFormat = 'YYYY-MM-DD', |
||||
timeFormat = 'HH:mm', |
||||
insertAboveColumnTitle, |
||||
insertBelowColumnTitle, |
||||
}: { |
||||
title: string; |
||||
type?: UITypes; |
||||
isUpdateMode?: boolean; |
||||
saveChanges?: boolean; |
||||
formula?: string; |
||||
qrCodeValueColumnTitle?: string; |
||||
barcodeValueColumnTitle?: string; |
||||
barcodeFormat?: string; |
||||
childTable?: string; |
||||
childColumn?: string; |
||||
relationType?: string; |
||||
rollupType?: string; |
||||
format?: string; |
||||
dateFormat?: string; |
||||
timeFormat?: string; |
||||
insertAboveColumnTitle?: string; |
||||
insertBelowColumnTitle?: string; |
||||
}) { |
||||
if (!isUpdateMode) { |
||||
if (insertAboveColumnTitle) { |
||||
await this.selectFieldAction({ title: insertAboveColumnTitle, action: 'insert-above' }); |
||||
} else if (insertBelowColumnTitle) { |
||||
await this.selectFieldAction({ title: insertBelowColumnTitle, action: 'insert-below' }); |
||||
} else { |
||||
await this.clickNewField(); |
||||
} |
||||
} |
||||
await this.addOrEditColumn.waitFor({ state: 'visible' }); |
||||
|
||||
await this.fillTitle({ title }); |
||||
|
||||
await this.selectType({ type }); |
||||
await this.rootPage.waitForTimeout(500); |
||||
|
||||
switch (type) { |
||||
case 'SingleSelect': |
||||
case 'MultiSelect': |
||||
break; |
||||
case 'Duration': |
||||
if (format) { |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`, { |
||||
hasText: format, |
||||
}) |
||||
.click(); |
||||
} |
||||
break; |
||||
case 'Date': |
||||
await this.addOrEditColumn.locator('.nc-date-select').click(); |
||||
await this.rootPage.locator('.nc-date-select').pressSequentially(dateFormat); |
||||
await this.rootPage.locator('.ant-select-item').locator(`text="${dateFormat}"`).click(); |
||||
break; |
||||
case 'DateTime': |
||||
// Date Format
|
||||
await this.addOrEditColumn.locator('.nc-date-select').click(); |
||||
await this.rootPage.locator('.ant-select-item').locator(`text="${dateFormat}"`).click(); |
||||
// Time Format
|
||||
await this.addOrEditColumn.locator('.nc-time-select').click(); |
||||
await this.rootPage.locator('.ant-select-item').locator(`text="${timeFormat}"`).click(); |
||||
break; |
||||
case 'Formula': |
||||
await this.addOrEditColumn.locator('.nc-formula-input').fill(formula); |
||||
break; |
||||
case 'QrCode': |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`) |
||||
.locator(`[data-testid="nc-qr-${qrCodeValueColumnTitle}"]`) |
||||
.click(); |
||||
break; |
||||
case 'Barcode': |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`, { |
||||
hasText: new RegExp(`^${barcodeValueColumnTitle}$`), |
||||
}) |
||||
.click(); |
||||
break; |
||||
case 'Lookup': |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`, { |
||||
hasText: childTable, |
||||
}) |
||||
.click(); |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(2).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`, { |
||||
hasText: childColumn, |
||||
}) |
||||
.last() |
||||
.click(); |
||||
break; |
||||
case 'Rollup': |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage |
||||
.locator(`.ant-select-item`, { |
||||
hasText: childTable, |
||||
}) |
||||
.click(); |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(2).click(); |
||||
await this.rootPage |
||||
.locator(`.nc-dropdown-relation-column >> .ant-select-item`, { |
||||
hasText: childColumn, |
||||
}) |
||||
.click(); |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(3).click(); |
||||
await this.rootPage |
||||
.locator(`.nc-dropdown-rollup-function >> .ant-select-item`, { |
||||
hasText: rollupType, |
||||
}) |
||||
.nth(0) |
||||
.click(); |
||||
break; |
||||
case 'Links': |
||||
await this.addOrEditColumn |
||||
.locator('.nc-ltar-relation-type >> .ant-radio') |
||||
.nth(relationType === 'Has Many' ? 0 : 1) |
||||
.click(); |
||||
await this.addOrEditColumn.locator('.ant-select-single').nth(1).click(); |
||||
await this.rootPage.locator(`.nc-ltar-child-table >> input[type="search"]`).fill(childTable); |
||||
await this.rootPage |
||||
.locator(`.nc-dropdown-ltar-child-table >> .ant-select-item`, { |
||||
hasText: childTable, |
||||
}) |
||||
.nth(0) |
||||
.click(); |
||||
break; |
||||
case 'User': |
||||
break; |
||||
default: |
||||
break; |
||||
} |
||||
|
||||
if (saveChanges) { |
||||
await this.saveChanges(); |
||||
|
||||
const fieldsText = await this.getAllFieldText(); |
||||
|
||||
if (insertAboveColumnTitle) { |
||||
// verify field inserted above the target field
|
||||
expect(fieldsText[fieldsText.findIndex(title => title.startsWith(insertAboveColumnTitle)) - 1]).toBe(title); |
||||
} else if (insertBelowColumnTitle) { |
||||
// verify field inserted below the target field
|
||||
expect(fieldsText[fieldsText.findIndex(title => title.startsWith(insertBelowColumnTitle)) + 1]).toBe(title); |
||||
} else { |
||||
// verify field inserted at the end
|
||||
expect(fieldsText[fieldsText.length - 1]).toBe(title); |
||||
} |
||||
} |
||||
} |
||||
|
||||
async fillTitle({ title }: { title: string }) { |
||||
const fieldTitleInput = this.addOrEditColumn.locator('.nc-fields-input'); |
||||
await fieldTitleInput.click(); |
||||
await fieldTitleInput.fill(title); |
||||
} |
||||
|
||||
async selectType({ type }: { type: string }) { |
||||
await this.addOrEditColumn.locator('.ant-select-selector > .ant-select-selection-item').click(); |
||||
|
||||
await this.addOrEditColumn.locator('.ant-select-selection-search-input[aria-expanded="true"]').waitFor(); |
||||
await this.addOrEditColumn.locator('.ant-select-selection-search-input[aria-expanded="true"]').fill(type); |
||||
|
||||
// Select column type
|
||||
await this.rootPage.locator('.rc-virtual-list-holder-inner > div').locator(`text="${type}"`).click(); |
||||
} |
||||
|
||||
async saveChanges() { |
||||
await this.waitForResponse({ |
||||
uiAction: async () => await this.saveChangesButton.click(), |
||||
requestUrlPathToMatch: 'api/v1/db/meta/views/', |
||||
httpMethodsToMatch: ['GET'], |
||||
responseJsonMatcher: json => json['list'], |
||||
}); |
||||
await this.rootPage.waitForTimeout(200); |
||||
} |
||||
|
||||
getField({ title }: { title: string }) { |
||||
return this.fieldListWrapper.getByTestId(`nc-field-item-${title}`); |
||||
} |
||||
|
||||
async getFieldVisibilityCheckbox({ title }: { title: string }) { |
||||
return this.getField({ title }).getByTestId('nc-field-visibility-checkbox'); |
||||
} |
||||
|
||||
async selectFieldAction({ |
||||
title, |
||||
action, |
||||
isDisplayValueField = false, |
||||
}: { |
||||
title: string; |
||||
action: 'copy-id' | 'duplicate' | 'insert-above' | 'insert-below' | 'delete'; |
||||
isDisplayValueField?: boolean; |
||||
}) { |
||||
const field = this.getField({ title }); |
||||
await field.scrollIntoViewIfNeeded(); |
||||
|
||||
await field.hover(); |
||||
// await field.getByTestId('nc-field-item-action-button').waitFor({ state: 'visible' });
|
||||
await field.getByTestId('nc-field-item-action-button').click(); |
||||
|
||||
const fieldActionDropdown = isDisplayValueField |
||||
? this.rootPage.locator('.nc-field-item-action-dropdown-display-column') |
||||
: this.rootPage.locator('.nc-field-item-action-dropdown'); |
||||
|
||||
await fieldActionDropdown.waitFor({ state: 'visible' }); |
||||
await fieldActionDropdown.getByTestId(`nc-field-item-action-${action}`).click(); |
||||
|
||||
if (action === 'copy-id') { |
||||
await field.getByTestId('nc-field-item-action-button').click(); |
||||
} |
||||
|
||||
await fieldActionDropdown.waitFor({ state: 'hidden' }); |
||||
} |
||||
|
||||
async getAllFieldText() { |
||||
const fieldsText = []; |
||||
const locator = this.fieldListWrapper.locator('>div'); |
||||
const count = await locator.count(); |
||||
for (let i = 0; i < count; i++) { |
||||
await locator.nth(i).scrollIntoViewIfNeeded(); |
||||
const text = await locator.nth(i).getByTestId('nc-field-title').textContent(); |
||||
fieldsText.push(text); |
||||
} |
||||
return fieldsText; |
||||
} |
||||
|
||||
async getFieldId({ title, isDisplayValueField = false }: { title: string; isDisplayValueField?: boolean }) { |
||||
const field = this.getField({ title }); |
||||
await field.scrollIntoViewIfNeeded(); |
||||
|
||||
await field.hover(); |
||||
await field.getByTestId('nc-field-item-action-button').waitFor({ state: 'visible' }); |
||||
await field.getByTestId('nc-field-item-action-button').click(); |
||||
|
||||
const fieldActionDropdown = isDisplayValueField |
||||
? this.rootPage.locator('.nc-field-item-action-dropdown-display-column') |
||||
: this.rootPage.locator('.nc-field-item-action-dropdown'); |
||||
|
||||
await fieldActionDropdown.waitFor({ state: 'visible' }); |
||||
const fieldId = await fieldActionDropdown.getByTestId('nc-field-item-id').textContent(); |
||||
await field.getByTestId('nc-field-item-action-button').click(); |
||||
await fieldActionDropdown.waitFor({ state: 'hidden' }); |
||||
return fieldId; |
||||
} |
||||
} |
@ -0,0 +1,380 @@
|
||||
import { expect, test } from '@playwright/test'; |
||||
import { DashboardPage } from '../../../pages/Dashboard'; |
||||
import setup, { unsetup } from '../../../setup'; |
||||
import { FieldsPage } from '../../../pages/Dashboard/Details/FieldsPage'; |
||||
import { getTextExcludeIconText } from '../../utils/general'; |
||||
import { UITypes } from 'nocodb-sdk'; |
||||
|
||||
const allFieldList = [ |
||||
{ |
||||
title: 'Single Line Text', |
||||
type: UITypes.SingleLineText, |
||||
}, |
||||
{ |
||||
title: 'Long Text', |
||||
type: UITypes.LongText, |
||||
}, |
||||
{ |
||||
title: 'Number', |
||||
type: UITypes.Number, |
||||
}, |
||||
{ |
||||
title: 'Decimal', |
||||
type: UITypes.Decimal, |
||||
}, |
||||
{ |
||||
title: 'Attachment', |
||||
type: UITypes.Attachment, |
||||
}, |
||||
{ |
||||
title: 'Checkbox', |
||||
type: UITypes.Checkbox, |
||||
}, |
||||
{ |
||||
title: 'MultiSelect', |
||||
type: UITypes.MultiSelect, |
||||
}, |
||||
{ |
||||
title: 'SingleSelect', |
||||
type: UITypes.SingleSelect, |
||||
}, |
||||
{ |
||||
title: 'Date', |
||||
type: UITypes.Date, |
||||
}, |
||||
{ |
||||
title: 'DateTime', |
||||
type: UITypes.DateTime, |
||||
}, |
||||
{ |
||||
title: 'Year', |
||||
type: UITypes.Year, |
||||
}, |
||||
{ |
||||
title: 'Time', |
||||
type: UITypes.Time, |
||||
}, |
||||
{ |
||||
title: 'PhoneNumber', |
||||
type: UITypes.PhoneNumber, |
||||
}, |
||||
{ |
||||
title: 'Email', |
||||
type: UITypes.Email, |
||||
}, |
||||
{ |
||||
title: 'URL', |
||||
type: UITypes.URL, |
||||
}, |
||||
{ |
||||
title: 'Currency', |
||||
type: UITypes.Currency, |
||||
}, |
||||
{ |
||||
title: 'Percent', |
||||
type: UITypes.Percent, |
||||
}, |
||||
{ |
||||
title: 'Duration', |
||||
type: UITypes.Duration, |
||||
}, |
||||
{ |
||||
title: 'Rating', |
||||
type: UITypes.Rating, |
||||
}, |
||||
{ |
||||
title: 'Formula', |
||||
type: UITypes.Formula, |
||||
formula: 'LEN({Title})', |
||||
}, |
||||
{ |
||||
title: 'QrCode', |
||||
type: UITypes.QrCode, |
||||
qrCodeValueColumnTitle: 'Title', |
||||
}, |
||||
{ |
||||
title: 'Barcode', |
||||
type: UITypes.Barcode, |
||||
barcodeValueColumnTitle: 'Title', |
||||
}, |
||||
{ |
||||
title: 'Geometry', |
||||
type: UITypes.Geometry, |
||||
}, |
||||
{ |
||||
title: 'JSON', |
||||
type: UITypes.JSON, |
||||
}, |
||||
{ |
||||
title: 'User', |
||||
type: UITypes.User, |
||||
}, |
||||
{ |
||||
title: 'Links', |
||||
type: UITypes.Links, |
||||
relationType: 'Has Many', |
||||
childTable: 'Multifield', |
||||
}, |
||||
]; |
||||
|
||||
test.describe('Multi Field Editor', () => { |
||||
let dashboard: DashboardPage, fields: FieldsPage; |
||||
let context: any; |
||||
const defaultFieldName = 'Multi Field Editor'; |
||||
|
||||
test.beforeEach(async ({ page }) => { |
||||
context = await setup({ page, isEmptyProject: true }); |
||||
dashboard = new DashboardPage(page, context.base); |
||||
fields = dashboard.details.fields; |
||||
|
||||
await dashboard.treeView.createTable({ title: 'Multifield', baseTitle: context.base.title }); |
||||
await openMultiFieldOfATable(); |
||||
}); |
||||
|
||||
test.afterEach(async () => { |
||||
await unsetup(context); |
||||
}); |
||||
|
||||
async function openMultiFieldOfATable() { |
||||
await dashboard.grid.topbar.openDetailedTab(); |
||||
await dashboard.details.clickFieldsTab(); |
||||
} |
||||
|
||||
async function toggleShowSystemFieldsFromDataTab() { |
||||
await dashboard.grid.topbar.openDataTab(); |
||||
await dashboard.grid.toolbar.fields.toggleShowSystemFields(); |
||||
await openMultiFieldOfATable(); |
||||
} |
||||
|
||||
const verifyGridColumnHeaders = async ({ fields = [] }: { fields: string[] }) => { |
||||
await dashboard.grid.topbar.openDataTab(); |
||||
|
||||
const locator = dashboard.grid.get().locator(`th`); |
||||
const count = await locator.count(); |
||||
|
||||
// exclude first checkbox and last add new column
|
||||
expect(count - 2).toBe(fields.length); |
||||
|
||||
for (let i = 1; i < count - 1; i++) { |
||||
const header = locator.nth(i); |
||||
const text = await getTextExcludeIconText(header); |
||||
expect(text).toBe(fields[i - 1]); |
||||
} |
||||
|
||||
await openMultiFieldOfATable(); |
||||
}; |
||||
|
||||
const searchAndVerifyFields = async ({ searchQuery, fieldList }: { searchQuery: string; fieldList: string[] }) => { |
||||
await fields.searchFieldInput.fill(searchQuery); |
||||
|
||||
const allFields = await fields.getAllFieldText(); |
||||
expect(allFields).toEqual( |
||||
searchQuery ? fieldList.filter(field => field.toLowerCase().includes(searchQuery.toLowerCase())) : fieldList |
||||
); |
||||
}; |
||||
|
||||
test('Verify system fields are not listed, Add New field, update & Restore, reset', async () => { |
||||
//Verify system fields are not listed
|
||||
await toggleShowSystemFieldsFromDataTab(); |
||||
let fieldsText = await fields.getAllFieldText(); |
||||
expect(fieldsText.length).toBe(1); |
||||
await toggleShowSystemFieldsFromDataTab(); |
||||
|
||||
// Add New Field
|
||||
await fields.createOrUpdate({ title: 'Name', saveChanges: false }); |
||||
await expect(fields.getField({ title: 'Name' })).toContainText('New field'); |
||||
await fields.saveChanges(); |
||||
|
||||
// Update Field title
|
||||
await fields.getField({ title: 'Name' }).click(); |
||||
await fields.createOrUpdate({ title: 'Updated Name', saveChanges: false, isUpdateMode: true }); |
||||
await expect(fields.getField({ title: 'Updated Name' })).toContainText('Updated field'); |
||||
await fields.saveChanges(); |
||||
|
||||
// Update and restore field changes
|
||||
await fields.getField({ title: 'Updated Name' }).click(); |
||||
await fields.createOrUpdate({ title: 'Updated Name to restore', saveChanges: false, isUpdateMode: true }); |
||||
await fields.clickRestoreField({ title: 'Updated Name to restore' }); |
||||
|
||||
// verify grid column header
|
||||
fieldsText = await fields.getAllFieldText(); |
||||
expect(fieldsText).toEqual(['Title', 'Updated Name']); |
||||
await verifyGridColumnHeaders({ fields: fieldsText }); |
||||
|
||||
// add new fields then reset changes and verify
|
||||
await fields.createOrUpdate({ title: 'field to reset', saveChanges: false }); |
||||
await fields.createOrUpdate({ title: 'Random', saveChanges: false }); |
||||
await fields.resetFieldChangesButton.click(); |
||||
|
||||
// verify with old fields
|
||||
await verifyGridColumnHeaders({ fields: fieldsText }); |
||||
}); |
||||
|
||||
// Todo: remove `skip`, if `optimized dependencies changed. reloading` issue is fixed
|
||||
test.skip('Add all fields and check status on clicking each field', async () => { |
||||
// Add all fields, verify status and save
|
||||
for (const field of allFieldList) { |
||||
await fields.createOrUpdate({ ...field, saveChanges: false }); |
||||
await expect(fields.getField({ title: field.title })).toContainText('New field'); |
||||
await fields.saveChanges(); |
||||
} |
||||
let fieldsText = await fields.getAllFieldText(); |
||||
|
||||
// verify all newly added field and its order
|
||||
expect(fieldsText).toEqual(['Title', ...allFieldList.map(field => field.title)]); |
||||
|
||||
// click on each field and check status
|
||||
fieldsText = await fields.getAllFieldText(); |
||||
for (const title of fieldsText) { |
||||
await fields.getField({ title }).click(); |
||||
await expect(fields.getField({ title })).not.toContainText(['New field', 'Updated field']); |
||||
} |
||||
}); |
||||
|
||||
test('Field operations: CopyId, Duplicate, InsertAbove, InsertBelow, Delete, Hide', async () => { |
||||
// Add New Field
|
||||
await fields.createOrUpdate({ title: defaultFieldName }); |
||||
|
||||
// copy-id and verify
|
||||
const fieldId = await fields.getFieldId({ title: defaultFieldName }); |
||||
await fields.selectFieldAction({ title: defaultFieldName, action: 'copy-id' }); |
||||
expect(fieldId).toBe(await dashboard.getClipboardText()); |
||||
|
||||
// duplicate and verify
|
||||
await fields.selectFieldAction({ title: defaultFieldName, action: 'duplicate' }); |
||||
await fields.saveChanges(); |
||||
|
||||
let fieldsText = await fields.getAllFieldText(); |
||||
expect(fieldsText[fieldsText.findIndex(field => field === defaultFieldName) + 1]).toBe(`${defaultFieldName}_copy`); |
||||
|
||||
// insert and verify
|
||||
await fields.createOrUpdate({ title: 'Above Inserted Field', insertAboveColumnTitle: defaultFieldName }); |
||||
await fields.createOrUpdate({ title: 'Below Inserted Field', insertBelowColumnTitle: defaultFieldName }); |
||||
|
||||
// delete and verify
|
||||
await fields.selectFieldAction({ title: `${defaultFieldName}_copy`, action: 'delete' }); |
||||
await expect(fields.getField({ title: `${defaultFieldName}_copy` })).toContainText('Deleted field'); |
||||
|
||||
await fields.saveChanges(); |
||||
|
||||
fieldsText = await fields.getAllFieldText(); |
||||
expect(!fieldsText.includes(`${defaultFieldName}_copy`)).toBeTruthy(); |
||||
|
||||
// verify grid column header
|
||||
await verifyGridColumnHeaders({ fields: fieldsText }); |
||||
|
||||
// Hide field and verify grid column header
|
||||
await (await fields.getFieldVisibilityCheckbox({ title: defaultFieldName })).click(); |
||||
await fields.saveChanges(); |
||||
|
||||
await verifyGridColumnHeaders({ fields: fieldsText.filter(field => field !== defaultFieldName) }); |
||||
}); |
||||
|
||||
test('Search field and verify', async () => { |
||||
const fieldList = ['Single Line Text', 'Long text', 'Rich text', 'Number', 'Percentage']; |
||||
|
||||
for (const field of fieldList) { |
||||
await fields.createOrUpdate({ title: field, saveChanges: false }); |
||||
} |
||||
await fields.saveChanges(); |
||||
|
||||
let searchQuery = 'text'; |
||||
await searchAndVerifyFields({ |
||||
searchQuery, |
||||
fieldList, |
||||
}); |
||||
|
||||
searchQuery = 'Rich text'; |
||||
await searchAndVerifyFields({ |
||||
searchQuery, |
||||
fieldList: ['Title', ...fieldList], |
||||
}); |
||||
|
||||
// clear search and verify
|
||||
await fields.clearSearch(); |
||||
await searchAndVerifyFields({ |
||||
searchQuery: '', |
||||
fieldList: ['Title', ...fieldList], |
||||
}); |
||||
}); |
||||
|
||||
test('Field Reorder and verify', async () => { |
||||
// default order: ['Title', 'Single Line Text', 'Long Text', 'Number', 'Percent','Links']
|
||||
const fieldList = [ |
||||
{ |
||||
title: 'Single Line Text', |
||||
type: UITypes.SingleLineText, |
||||
}, |
||||
{ |
||||
title: 'Long Text', |
||||
type: UITypes.LongText, |
||||
}, |
||||
{ |
||||
title: 'Number', |
||||
type: UITypes.Number, |
||||
}, |
||||
{ |
||||
title: 'Percent', |
||||
type: UITypes.Percent, |
||||
}, |
||||
{ |
||||
title: 'Links', |
||||
type: UITypes.Links, |
||||
relationType: 'Has Many', |
||||
childTable: 'Multifield', |
||||
}, |
||||
]; |
||||
|
||||
for (const field of fieldList) { |
||||
await fields.createOrUpdate({ ...field, saveChanges: false }); |
||||
} |
||||
await fields.saveChanges(); |
||||
// updated order : ['Title', 'Long Text','Single Line Text', 'Number', 'Percent','Links']
|
||||
await fields.getField({ title: fieldList[0].title }).dragTo(fields.getField({ title: fieldList[1].title })); |
||||
await expect(fields.getField({ title: fieldList[0].title })).toContainText('Updated field'); |
||||
|
||||
// updated order : ['Title', 'Long Text','Single Line Text', 'Number','Links', 'Percent']
|
||||
await fields.getField({ title: fieldList[4].title }).dragTo(fields.getField({ title: fieldList[3].title })); |
||||
await expect(fields.getField({ title: fieldList[4].title })).toContainText('Updated field'); |
||||
|
||||
await fields.saveChanges(); |
||||
const fieldsText = await fields.getAllFieldText(); |
||||
const expectedFieldText = ['Title', 'Long Text', 'Single Line Text', 'Number', 'Links', 'Percent']; |
||||
|
||||
expect(fieldsText).toEqual(expectedFieldText); |
||||
|
||||
await verifyGridColumnHeaders({ fields: expectedFieldText }); |
||||
}); |
||||
|
||||
test('Keyboard shortcuts: add new field, save and delete', async () => { |
||||
// add new field
|
||||
await dashboard.rootPage.keyboard.press('Alt+C'); |
||||
|
||||
// verify field is added and has `New field` status
|
||||
let fieldsText = await fields.getAllFieldText(); |
||||
await expect(fields.getField({ title: fieldsText[fieldsText.length - 1] })).toContainText('New field'); |
||||
|
||||
// update title
|
||||
await fields.createOrUpdate({ title: defaultFieldName, isUpdateMode: true }); |
||||
|
||||
// save the changes
|
||||
await dashboard.rootPage.keyboard.press((await dashboard.isMacOs()) ? 'Meta+S' : 'Control+S'); |
||||
await dashboard.rootPage.waitForTimeout(500); |
||||
|
||||
// verify result
|
||||
fieldsText = await fields.getAllFieldText(); |
||||
expect(fieldsText).toEqual(['Title', defaultFieldName]); |
||||
|
||||
// delete field
|
||||
await fields.getField({ title: defaultFieldName }).click(); |
||||
await dashboard.rootPage.keyboard.press((await dashboard.isMacOs()) ? 'Meta+Delete' : 'Delete'); |
||||
await expect(fields.getField({ title: defaultFieldName })).toContainText('Deleted field'); |
||||
|
||||
// save the changes
|
||||
await dashboard.rootPage.keyboard.press((await dashboard.isMacOs()) ? 'Meta+S' : 'Control+S'); |
||||
await dashboard.rootPage.waitForTimeout(500); |
||||
|
||||
fieldsText = await fields.getAllFieldText(); |
||||
expect(fieldsText).toEqual(['Title']); |
||||
}); |
||||
}); |
Loading…
Reference in new issue