mirror of https://github.com/nocodb/nocodb
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.
223 lines
8.0 KiB
223 lines
8.0 KiB
import { expect, Locator } from '@playwright/test'; |
|
import BasePage from '../../Base'; |
|
import { DashboardPage } from '..'; |
|
import { getTextExcludeIconText } from '../../../tests/utils/general'; |
|
|
|
export class BulkUpdatePage extends BasePage { |
|
readonly dashboard: DashboardPage; |
|
readonly bulkUpdateButton: Locator; |
|
readonly formHeader: Locator; |
|
readonly columnsDrawer: Locator; |
|
readonly form: Locator; |
|
|
|
constructor(dashboard: DashboardPage) { |
|
super(dashboard.rootPage); |
|
this.dashboard = dashboard; |
|
this.bulkUpdateButton = this.dashboard.get().locator('.nc-bulk-update-save-btn'); |
|
this.formHeader = this.dashboard.get().locator('.nc-bulk-update-bulk-update-header'); |
|
this.columnsDrawer = this.dashboard.get().locator('.nc-columns-drawer'); |
|
this.form = this.dashboard.get().locator('div.form'); |
|
} |
|
|
|
get() { |
|
return this.dashboard.get().locator(`.nc-drawer-bulk-update`); |
|
} |
|
|
|
async close() { |
|
return this.dashboard.rootPage.keyboard.press('Escape'); |
|
} |
|
|
|
async getInactiveColumn(index: number) { |
|
const inactiveColumns = this.columnsDrawer.locator('.ant-card'); |
|
return inactiveColumns.nth(index); |
|
} |
|
|
|
async getActiveColumn(index: number) { |
|
const activeColumns = this.form.locator('[data-testid="nc-bulk-update-fields"]'); |
|
return activeColumns.nth(index); |
|
} |
|
|
|
async getInactiveColumns() { |
|
const inactiveColumns = this.columnsDrawer.locator('.ant-card'); |
|
const inactiveColumnsCount = await inactiveColumns.count(); |
|
const inactiveColumnsTitles = []; |
|
// get title for each inactive column |
|
for (let i = 0; i < inactiveColumnsCount; i++) { |
|
const title = await getTextExcludeIconText(inactiveColumns.nth(i).locator('.ant-card-body')); |
|
inactiveColumnsTitles.push(title); |
|
} |
|
|
|
return inactiveColumnsTitles; |
|
} |
|
|
|
async getActiveColumns() { |
|
const activeColumns = this.form.locator('[data-testid="nc-bulk-update-fields"]'); |
|
const activeColumnsCount = await activeColumns.count(); |
|
const activeColumnsTitles = []; |
|
// get title for each active column |
|
for (let i = 0; i < activeColumnsCount; i++) { |
|
const title = await getTextExcludeIconText( |
|
activeColumns.nth(i).locator('[data-testid="nc-bulk-update-input-label"]') |
|
); |
|
activeColumnsTitles.push(title); |
|
} |
|
|
|
return activeColumnsTitles; |
|
} |
|
|
|
async removeField(index: number) { |
|
const removeFieldButton = this.form.locator('[data-testid="nc-bulk-update-fields"]'); |
|
const removeFieldButtonCount = await removeFieldButton.count(); |
|
await removeFieldButton.nth(index).locator('[data-testid="nc-bulk-update-fields-remove-icon"]').click(); |
|
const newRemoveFieldButtonCount = await removeFieldButton.count(); |
|
expect(newRemoveFieldButtonCount).toBe(removeFieldButtonCount - 1); |
|
} |
|
|
|
async addField(index: number) { |
|
const addFieldButton = this.columnsDrawer.locator('.ant-card'); |
|
const addFieldButtonCount = await addFieldButton.count(); |
|
await addFieldButton.nth(index).click(); |
|
const newAddFieldButtonCount = await addFieldButton.count(); |
|
expect(newAddFieldButtonCount).toBe(addFieldButtonCount - 1); |
|
} |
|
|
|
////////////////////////////////////////////////////////////////////////////// |
|
|
|
async fillField({ columnTitle, value, type = 'text' }: { columnTitle: string; value: string; type?: string }) { |
|
let picker = null; |
|
const field = this.form.locator(`[data-testid="nc-bulk-update-input-${columnTitle}"]`); |
|
await field.scrollIntoViewIfNeeded(); |
|
await field.hover(); |
|
if (type !== 'checkbox' && type !== 'attachment') { |
|
await field.click(); |
|
} |
|
switch (type) { |
|
case 'text': |
|
await field.locator('input').waitFor(); |
|
await field.locator('input').fill(value); |
|
break; |
|
case 'longText': |
|
await field.locator('textarea').waitFor(); |
|
await field.locator('textarea').fill(value); |
|
break; |
|
case 'rating': |
|
await field |
|
.locator('.ant-rate-star') |
|
.nth(Number(value) - 1) |
|
.click(); |
|
break; |
|
case 'year': |
|
await field.locator('input').click(); |
|
picker = this.rootPage.locator('.nc-picker-year.active'); |
|
await picker.waitFor(); |
|
await this.configureYear(value); |
|
break; |
|
case 'time': |
|
// eslint-disable-next-line no-case-declarations |
|
const timeInput = field.locator('.nc-time-input'); |
|
await timeInput.click(); |
|
// eslint-disable-next-line no-case-declarations |
|
const dropdown = this.rootPage.locator('.nc-picker-time.active'); |
|
await dropdown.waitFor({ state: 'visible' }); |
|
|
|
await timeInput.fill(value); |
|
await this.rootPage.keyboard.press('Enter'); |
|
|
|
await dropdown.waitFor({ state: 'hidden' }); |
|
|
|
break; |
|
case 'singleSelect': |
|
picker = this.rootPage.locator('.ant-select-dropdown.active'); |
|
await picker.waitFor(); |
|
await picker.locator(`.nc-select-option-SingleSelect-${value}`).click(); |
|
break; |
|
case 'multiSelect': |
|
picker = this.rootPage.locator('.ant-select-dropdown.active'); |
|
await picker.waitFor(); |
|
for (const val of value.split(',')) { |
|
await picker.locator(`.nc-select-option-MultiSelect-${val}`).click(); |
|
} |
|
break; |
|
case 'checkbox': |
|
if (value === 'true') { |
|
await field.locator('.nc-checkbox').click(); |
|
} |
|
break; |
|
case 'attachment': |
|
// eslint-disable-next-line no-case-declarations |
|
const attachFileAction = field.locator('[data-testid="attachment-cell-file-picker-button"]').click(); |
|
await this.attachFile({ filePickUIAction: attachFileAction, filePath: [value] }); |
|
break; |
|
case 'date': |
|
{ |
|
await field.locator('input').click(); |
|
const values = value.split('-'); |
|
const { year, month, day } = { year: values[0], month: values[1], day: values[2] }; |
|
picker = this.rootPage.locator('.nc-picker-date.active'); |
|
|
|
await picker.waitFor(); |
|
const yearBtn = picker.locator('.nc-year-picker-btn'); |
|
await yearBtn.waitFor(); |
|
await yearBtn.click(); |
|
await this.configureYear(year); |
|
|
|
const monthBtn = picker.locator('.nc-month-picker-btn'); |
|
|
|
await monthBtn.click(); |
|
await picker.waitFor(); |
|
await picker.locator(`span[title="${year}-${month}"]`).click(); |
|
|
|
await picker.waitFor(); |
|
await picker.locator(`span[title="${year}-${month}-${day}"]:visible`).click(); |
|
} |
|
break; |
|
} |
|
} |
|
|
|
async configureYear(year: string) { |
|
// configure year |
|
await this.rootPage.locator('.nc-year-picker-btn:visible').waitFor(); |
|
|
|
let flag = true; |
|
|
|
while (flag) { |
|
const firstVisibleYear = await this.rootPage.locator('.nc-year-item').first().textContent(); |
|
const lastVisibleYear = await this.rootPage.locator('.nc-year-item').last().textContent(); |
|
|
|
if (+year >= +firstVisibleYear && +year <= +lastVisibleYear) { |
|
flag = false; |
|
} else if (+year < +firstVisibleYear) { |
|
await this.rootPage.locator('.nc-prev-page-btn').click(); |
|
} else if (+year > +lastVisibleYear) { |
|
await this.rootPage.locator('.nc-next-page-btn').click(); |
|
} |
|
} |
|
|
|
await this.rootPage.locator(`span[title="${year}"]`).waitFor(); |
|
await this.rootPage.locator(`span[title="${year}"]`).click({ force: true }); |
|
} |
|
|
|
async save({ |
|
awaitResponse = true, |
|
}: { |
|
awaitResponse?: boolean; |
|
} = {}) { |
|
await this.bulkUpdateButton.click(); |
|
const confirmModal = this.rootPage.locator('.ant-modal'); |
|
|
|
const saveRowAction = () => confirmModal.locator('.ant-btn-primary').click(); |
|
if (!awaitResponse) { |
|
await saveRowAction(); |
|
} else { |
|
await this.waitForResponse({ |
|
uiAction: saveRowAction, |
|
requestUrlPathToMatch: 'api/v1/db/data/noco/', |
|
httpMethodsToMatch: ['GET'], |
|
// responseJsonMatcher: json => json['pageInfo'], |
|
}); |
|
} |
|
|
|
await this.get().waitFor({ state: 'hidden' }); |
|
await this.rootPage.locator('[data-testid="grid-load-spinner"]').waitFor({ state: 'hidden' }); |
|
} |
|
}
|
|
|