import { expect, Locator } from '@playwright/test';
import { StringValidationType, UITypes } from 'nocodb-sdk';
import { DashboardPage } from '..';
import BasePage from '../../Base';
import { ToolbarPage } from '../common/Toolbar';
import { TopbarPage } from '../common/Topbar';

export class FormPage extends BasePage {
  readonly dashboard: DashboardPage;
  readonly toolbar: ToolbarPage;
  readonly topbar: TopbarPage;

  // todo: All the locator should be private
  readonly addOrRemoveAllButton: Locator;
  readonly submitButton: Locator;

  readonly showAnotherFormRadioButton: Locator;
  readonly showAnotherFormAfter5SecRadioButton: Locator;
  readonly emailMeRadioButton: Locator;

  readonly formHeading: Locator;
  readonly formSubHeading: Locator;
  readonly afterSubmitMsg: Locator;

  readonly formFields: Locator;

  // validation
  readonly fieldPanel: Locator;
  readonly customValidationBtn: Locator;
  readonly customValidationDropdown: Locator;

  constructor(dashboard: DashboardPage) {
    super(dashboard.rootPage);
    this.dashboard = dashboard;
    this.toolbar = new ToolbarPage(this);
    this.topbar = new TopbarPage(this);

    this.addOrRemoveAllButton = dashboard
      .get()
      .locator('[data-testid="nc-form-show-all-fields"]')
      .locator('.nc-switch');
    this.submitButton = dashboard.get().locator('[data-testid="nc-form-submit"]');

    this.showAnotherFormRadioButton = dashboard.get().locator('[data-testid="nc-form-checkbox-submit-another-form"]');
    this.showAnotherFormAfter5SecRadioButton = dashboard
      .get()
      .locator('[data-testid="nc-form-checkbox-show-blank-form"]');
    this.emailMeRadioButton = dashboard.get().locator('[data-testid="nc-form-checkbox-send-email"]');
    this.formHeading = dashboard.get().locator('[data-testid="nc-form-heading"]');
    this.formSubHeading = dashboard.get().locator('[data-testid="nc-form-sub-heading"] .tiptap.ProseMirror');
    this.afterSubmitMsg = dashboard.get().locator('[data-testid="nc-form-after-submit-msg"] .tiptap.ProseMirror');

    this.formFields = dashboard.get().locator('.nc-form-fields-list');

    // validation
    this.fieldPanel = dashboard.get().locator('.nc-form-field-right-panel');
    this.customValidationBtn = this.fieldPanel.locator('.nc-custom-validation-btn');
    this.customValidationDropdown = this.rootPage.locator('.nc-custom-validator-dropdown');
  }

  get() {
    return this.dashboard.get().locator('[data-testid="nc-form-wrapper"]');
  }

  getFormAfterSubmit() {
    return this.dashboard.get().locator('[data-testid="nc-form-wrapper-submit"]');
  }

  getFormFields() {
    return this.get().locator('[data-testid="nc-form-fields"]');
  }

  getFormFieldsRequired() {
    return this.get().locator('[data-testid="nc-form-input-required"]');
  }

  getFormFieldsInputLabel() {
    return this.get().locator('textarea[data-testid="nc-form-input-label"]:visible');
  }

  getFormFieldsInputHelpText() {
    return this.get().locator('[data-testid="nc-form-input-help-text"] .tiptap.ProseMirror:visible');
  }

  async verifyFormFieldLabel({ index, label }: { index: number; label: string }) {
    await expect(this.getFormFields().nth(index).locator('[data-testid="nc-form-input-label"]')).toContainText(label);
  }

  async verifyFormFieldHelpText({ index, helpText }: { index: number; helpText: string }) {
    await expect(this.getFormFields().nth(index).locator('[data-testid="nc-form-help-text"]')).toContainText(helpText);
  }

  async verifyFieldsIsEditable({ index }: { index: number }) {
    await expect(this.getFormFields().nth(index)).toHaveClass(/nc-editable/);
  }

  async verifyAfterSubmitMsg({ msg }: { msg: string }) {
    expect((await this.afterSubmitMsg.textContent()).includes(msg)).toBeTruthy();
  }

  async verifyFormViewFieldsOrder({ fields }: { fields: string[] }) {
    const fieldLabels = this.get().locator('[data-testid="nc-form-input-label"]');
    await expect(fieldLabels).toHaveCount(fields.length);
    for (let i = 0; i < fields.length; i++) {
      await expect(fieldLabels.nth(i)).toContainText(fields[i]);
    }
  }

  async reorderFields({ sourceField, destinationField }: { sourceField: string; destinationField: string }) {
    // TODO: Otherwise form input boxes are not visible sometimes
    await this.rootPage.waitForTimeout(650);

    await expect(this.get().locator(`.nc-form-drag-${sourceField}`)).toBeVisible();
    await expect(this.get().locator(`.nc-form-drag-${destinationField}`)).toBeVisible();
    const src = this.get().locator(`.nc-form-drag-${sourceField.replace(' ', '')}`);
    const dst = this.get().locator(`.nc-form-drag-${destinationField.replace(' ', '')}`);
    await src.dragTo(dst);
  }

  async removeField({ field, mode }: { mode: 'dragDrop' | 'hideField'; field: string }) {
    // TODO: Otherwise form input boxes are not visible sometimes
    await this.rootPage.waitForTimeout(650);

    if (mode === 'dragDrop') {
      const src = this.get().locator(`.nc-form-drag-${field.replace(' ', '')}`);
      const dst = this.get().locator(`[data-testid="nc-drag-n-drop-to-hide"]`);
      await src.dragTo(dst);
    } else if (mode === 'hideField') {
      // in form-v2, hide field will be using right sidebar
      await this.formFields.locator(`[data-testid="nc-form-field-item-${field}"]`).locator('.nc-switch').click();
    }
  }

  async addField({ field, mode }: { mode: string; field: string }) {
    // TODO: Otherwise form input boxes are not visible sometimes
    await this.rootPage.waitForTimeout(650);

    if (mode === 'dragDrop') {
      const src = this.get().locator(`[data-testid="nc-form-hidden-column-${field}"] > div.ant-card-body`);
      const dst = this.get().locator(`[data-testid="nc-form-input-Country"]`);
      await src.waitFor({ state: 'visible' });
      await dst.waitFor({ state: 'visible' });
      await src.dragTo(dst, { trial: true });
      await src.dragTo(dst);
    } else if (mode === 'clickField') {
      await this.formFields.locator(`[data-testid="nc-form-field-item-${field}"]`).locator('.nc-switch').click();
    }
  }

  async removeAllFields() {
    if (await this.addOrRemoveAllButton.isChecked()) {
      await this.addOrRemoveAllButton.click();
    } else {
      await this.addOrRemoveAllButton.click();
      await this.addOrRemoveAllButton.click();
    }
  }

  async addAllFields() {
    if (!(await this.addOrRemoveAllButton.isChecked())) {
      await this.addOrRemoveAllButton.click();
    }
  }

  async configureHeader(param: { subtitle: string; title: string }) {
    await this.waitForResponse({
      uiAction: async () => {
        await this.formHeading.click();
        await this.formHeading.fill(param.title);
        await this.formSubHeading.click();
      },
      requestUrlPathToMatch: '/api/v1/db/meta/forms',
      httpMethodsToMatch: ['PATCH'],
    });
    await this.waitForResponse({
      uiAction: async () => {
        await this.formSubHeading.click();
        await this.formSubHeading.fill(param.subtitle);
        await this.formHeading.click();
      },
      requestUrlPathToMatch: '/api/v1/db/meta/forms',
      httpMethodsToMatch: ['PATCH'],
    });
  }

  async verifyHeader(param: { subtitle: string; title: string }) {
    await expect.poll(async () => await this.formHeading.inputValue()).toBe(param.title);
    await expect.poll(async () => await this.formSubHeading.textContent()).toBe(param.subtitle);
  }

  async fillForm(param: { field: string; value: string; type?: UITypes }[]) {
    for (let i = 0; i < param.length; i++) {
      await this.get()
        .locator(`[data-testid="nc-form-input-${param[i].field.replace(' ', '')}"]`)
        .click();

      switch (param[i].type) {
        case UITypes.LongText: {
          await this.get()
            .locator(`[data-testid="nc-form-input-${param[i].field.replace(' ', '')}"] >> textarea`)
            .fill(param[i].value);

          break;
        }
        default: {
          await this.get()
            .locator(`[data-testid="nc-form-input-${param[i].field.replace(' ', '')}"] >> input`)
            .fill(param[i].value);

          if ([UITypes.Date, UITypes.Time, UITypes.Year, UITypes.DateTime].includes(param[i].type)) {
            await this.rootPage.keyboard.press('Enter');
          }

          await this.getVisibleField({ title: param[i].field }).click();
        }
      }

      await this.rootPage.waitForTimeout(200);
    }
  }

  getVisibleField({ title }: { title: string }) {
    return this.get()
      .locator(`[data-testid="nc-form-fields"][data-title="${title}"]`)
      .locator('[data-testid="nc-form-input-label"]');
  }

  async selectVisibleField({ title }: { title: string }) {
    const field = this.getVisibleField({ title });

    await field.scrollIntoViewIfNeeded();
    await field.click();

    // Wait for field settings right pannel
    await this.fieldPanel.waitFor({ state: 'visible' });
  }

  async configureField({
    field,
    required,
    label,
    helpText,
  }: {
    field: string;
    required: boolean;
    label: string;
    helpText: string;
  }) {
    const waitForResponse = async (action: () => Promise<any>) =>
      await this.waitForResponse({
        uiAction: action,
        requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
        httpMethodsToMatch: ['PATCH'],
      });

    await this.selectVisibleField({ title: field });

    await waitForResponse(() => this.getFormFieldsInputLabel().fill(label));
    await waitForResponse(() => this.getFormFieldsInputHelpText().fill(helpText));
    if (required) {
      await waitForResponse(() => this.getFormFieldsRequired().click());
    }
    await this.formHeading.click();
  }

  async verifyField({
    field,
    required,
    label,
    helpText,
  }: {
    field: string;
    required: boolean;
    label: string;
    helpText: string;
  }) {
    let expectText = '';
    if (required) expectText = label + ' *';
    else expectText = label;

    const fieldLabel = this.getVisibleField({ title: field });
    await fieldLabel.scrollIntoViewIfNeeded();
    await expect(fieldLabel).toHaveText(expectText);

    const fieldHelpText = this.get()
      .locator(`.nc-form-drag-${field.replace(' ', '')}`)
      .locator('div[data-testid="nc-form-input-help-text-label"] .tiptap.ProseMirror');
    await expect(fieldHelpText).toHaveText(helpText);
  }

  async submitForm() {
    await this.submitButton.click();
  }

  async verifyStatePostSubmit(param: { message?: string; submitAnotherForm?: boolean; showBlankForm?: boolean }) {
    await this.rootPage.locator('.nc-form-success-msg').waitFor({ state: 'visible' });

    if (undefined !== param.message) {
      await expect(this.getFormAfterSubmit()).toContainText(param.message);
    }
    if (true === param.submitAnotherForm) {
      await expect(this.getFormAfterSubmit().locator('button:has-text("Submit Another Form")')).toBeVisible();
    }
    if (true === param.showBlankForm) {
      await this.get().waitFor();
    }
  }

  async configureSubmitMessage(param: { message: string }) {
    await this.waitForResponse({
      uiAction: async () => {
        await this.afterSubmitMsg.click();
        await this.afterSubmitMsg.fill(param.message);
      },
      requestUrlPathToMatch: '/api/v1/db/meta/forms',
      httpMethodsToMatch: ['PATCH'],
    });
  }

  submitAnotherForm() {
    return this.getFormAfterSubmit().locator('button:has-text("Submit Another Form")');
  }

  // todo: Wait for render to complete
  async waitLoading() {
    await this.rootPage.waitForTimeout(1000);
  }

  async verifyAfterSubmitMenuState(param: { showBlankForm?: boolean; submitAnotherForm?: boolean; emailMe?: boolean }) {
    if (true === param.showBlankForm) {
      await expect(
        this.get().locator('[data-testid="nc-form-checkbox-show-blank-form"][aria-checked="true"]')
      ).toBeVisible();
    }
    if (true === param.submitAnotherForm) {
      await expect(
        this.get().locator('[data-testid="nc-form-checkbox-submit-another-form"][aria-checked="true"]')
      ).toBeVisible();
    }
    if (true === param.emailMe) {
      await expect(
        this.get().locator('[data-testid="nc-form-checkbox-send-email"][aria-checked="true"]')
      ).toBeVisible();
    }
  }

  async getCustomValidationTypeOption({
    type,
    currValItem,
    index,
  }: {
    type: StringValidationType;
    currValItem: Locator;
    index: number;
  }) {
    await currValItem.locator('.nc-custom-validation-type-selector .ant-select-selector').click();

    const typeSelectorDropdown = this.rootPage.locator(`.nc-custom-validation-type-dropdown-${index}`);
    await typeSelectorDropdown.waitFor();

    const option = typeSelectorDropdown.getByTestId(`nc-custom-validation-type-option-${type}`);
    await option.scrollIntoViewIfNeeded();

    return {
      option: this.rootPage.getByTestId(`nc-custom-validation-type-option-${type}`),
      select: async () => {
        await option.click();
        await typeSelectorDropdown.waitFor({ state: 'hidden' });
      },
      closeSelector: async () =>
        await currValItem.locator('.nc-custom-validation-type-selector .ant-select-selector').click(),
      verify: async ({ isDisabled = false }: { isDisabled: boolean }) => {
        if (isDisabled) {
          await expect(option).toHaveClass(/ant-select-item-option-disabled/);
        } else {
          await expect(option).not.toHaveClass(/ant-select-item-option-disabled/);
        }
      },
    };
  }

  async addCustomValidation({
    type,
    value,
    errorMsg,
    index,
  }: {
    type: StringValidationType;
    value: string;
    errorMsg?: string;
    index: number;
  }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    await this.customValidationBtn.click();

    const dropdown = this.customValidationDropdown;
    await dropdown.waitFor({ state: 'visible' });

    await dropdown.locator('.nc-custom-validation-add-btn').click();

    const currValItem = this.customValidationDropdown.getByTestId(`nc-custom-validation-item-${index}`);

    await currValItem.waitFor({ state: 'visible' });

    // Select type
    const { select } = await this.getCustomValidationTypeOption({ type, currValItem, index });
    await select();

    // add value
    const valValueInput = currValItem.locator('.custom-validation-input >> input');
    await valValueInput.click();
    await valValueInput.fill(value);

    if (errorMsg !== undefined) {
      const valErrorMsgInput = currValItem.locator('.nc-custom-validation-error-message-input');
      await valErrorMsgInput.click();
      await valErrorMsgInput.fill(errorMsg);
    }

    //close dropdown
    await this.customValidationBtn.click();
    await dropdown.waitFor({ state: 'hidden' });
  }

  async updateCustomValidation({
    type,
    value,
    errorMsg,
    index,
  }: {
    type?: StringValidationType;
    value?: string;
    errorMsg?: string;
    index: number;
  }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    await this.customValidationBtn.click();

    const dropdown = this.customValidationDropdown;
    await dropdown.waitFor({ state: 'visible' });

    const currValItem = this.customValidationDropdown.getByTestId(`nc-custom-validation-item-${index}`);

    await currValItem.waitFor({ state: 'visible' });

    if (type) {
      // Select type
      const { select } = await this.getCustomValidationTypeOption({ type, currValItem, index });
      await select();
    }

    // update value
    if (value !== undefined) {
      const valValueInput = currValItem.locator('.custom-validation-input >> input');
      await valValueInput.click();
      await valValueInput.fill(value);
    }

    if (errorMsg !== undefined) {
      const valErrorMsgInput = currValItem.locator('.nc-custom-validation-error-message-input');
      await valErrorMsgInput.click();
      await valErrorMsgInput.fill(errorMsg);
    }

    //close dropdown
    await this.customValidationBtn.click();
    await dropdown.waitFor({ state: 'hidden' });
  }

  async verifyCustomValidationSelector({
    type,
    value: _value,
    errorMsg: _errorMsg,
    index,
  }: {
    type: StringValidationType;
    value?: string;
    errorMsg?: string;
    index: number;
  }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    await this.customValidationBtn.click();

    const dropdown = this.customValidationDropdown;
    await dropdown.waitFor({ state: 'visible' });

    const currValItem = this.customValidationDropdown.getByTestId(`nc-custom-validation-item-${index}`);
    await currValItem.waitFor({ state: 'visible' });

    const { verify } = await this.getCustomValidationTypeOption({
      type,
      currValItem,
      index,
    });
    await verify({ isDisabled: true });

    //close dropdown
    await this.customValidationBtn.click();
    await dropdown.waitFor({ state: 'hidden' });
  }
  async verifyCustomValidationValue({ value, hasError, index }: { value?: string; hasError?: boolean; index: number }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    await this.customValidationBtn.click();

    const dropdown = this.customValidationDropdown;
    await dropdown.waitFor({ state: 'visible' });

    const currValItem = this.customValidationDropdown.getByTestId(`nc-custom-validation-item-${index}`);
    await currValItem.waitFor({ state: 'visible' });

    //  value
    if (value !== undefined) {
      const valValueInput = currValItem.locator('.custom-validation-input >> input');
      await expect(valValueInput).toHaveValue(value);
    }

    if (hasError) {
      const valValueErr = currValItem.locator(
        '.nc-custom-validation-input-wrapper .nc-custom-validation-item-error-icon'
      );

      await expect(valValueErr).toBeVisible();
    }

    //close dropdown
    await this.customValidationBtn.click();
    await dropdown.waitFor({ state: 'hidden' });
  }

  async removeCustomValidationItem({ index }: { index: number }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    await this.customValidationBtn.click();

    const dropdown = this.customValidationDropdown;
    await dropdown.waitFor({ state: 'visible' });

    const currValItem = this.customValidationDropdown.getByTestId(`nc-custom-validation-item-${index}`);
    await currValItem.waitFor({ state: 'visible' });

    await currValItem.locator('.nc-custom-validation-delete-item').click();

    await currValItem.waitFor({ state: 'hidden' });

    //close dropdown
    await this.customValidationBtn.click();
    await dropdown.waitFor({ state: 'hidden' });
  }

  async verifyCustomValidationCount({ count }: { count: number }) {
    await this.customValidationBtn.waitFor({ state: 'visible' });
    const countStr = await this.customValidationBtn.locator('.nc-custom-validation-count').textContent();

    expect(parseInt(countStr) || 0).toEqual(count);
  }

  async getFormFieldsEmailPhoneUrlValidatorConfig({
    type,
  }: {
    type: UITypes.Email | UITypes.PhoneNumber | UITypes.URL;
  }) {
    const validateBtn = this.get().getByTestId(`nc-form-field-validate-${type}`);

    return {
      locator: validateBtn,
      click: async ({ enable }: { enable: boolean }) => {
        await validateBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateBtn.isChecked();

        if ((enable && !isEnabled) || (!enable && isEnabled)) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({ isEnabled, isDisabled }: { isEnabled?: boolean; isDisabled?: boolean }) => {
        await validateBtn.waitFor({ state: 'visible' });

        if (isEnabled !== undefined) {
          if (isEnabled) {
            await expect(validateBtn).toBeChecked();
          } else {
            await expect(validateBtn).not.toBeChecked();
          }
        }

        if (isDisabled !== undefined) {
          if (isDisabled) {
            await expect(validateBtn).toBeDisabled();
          } else {
            await expect(validateBtn).not.toBeDisabled();
          }
        }
      },
    };
  }

  async getFormFieldsValidateWorkEmailConfig() {
    const validateWorkEmailBtn = this.get().getByTestId('nc-form-field-allow-only-work-email');

    return {
      locator: validateWorkEmailBtn,
      click: async ({ enable }: { enable: boolean }) => {
        await validateWorkEmailBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateWorkEmailBtn.isChecked();

        if ((enable && !isEnabled) || (!enable && isEnabled)) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateWorkEmailBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({
        isEnabled,
        isDisabled,
        isVisible,
      }: {
        isEnabled?: boolean;
        isDisabled?: boolean;
        isVisible?: boolean;
      }) => {
        if (isEnabled !== undefined) {
          await validateWorkEmailBtn.waitFor({ state: 'visible' });

          if (isEnabled) {
            await expect(validateWorkEmailBtn).toBeChecked();
          } else {
            await expect(validateWorkEmailBtn).not.toBeChecked();
          }
        }

        if (isDisabled !== undefined) {
          await validateWorkEmailBtn.waitFor({ state: 'visible' });

          if (isDisabled) {
            await expect(validateWorkEmailBtn).toBeDisabled();
          } else {
            await expect(validateWorkEmailBtn).not.toBeDisabled();
          }
        }

        if (isVisible !== undefined) {
          if (isVisible) {
            await expect(validateWorkEmailBtn).toBeVisible();
          } else {
            await expect(validateWorkEmailBtn).not.toBeVisible();
          }
        }
      },
    };
  }

  async getFormFieldErrors({ title }: { title: string }) {
    // ant-form-item-explain
    const field = this.get().locator(`[data-testid="nc-form-fields"][data-title="${title}"]`);
    await field.scrollIntoViewIfNeeded();
    const fieldErrorEl = field.locator('.ant-form-item-explain');

    return {
      locator: fieldErrorEl,
      verify: async ({ hasError, hasErrorMsg }: { hasError?: boolean; hasErrorMsg?: string | RegExp }) => {
        if (hasError !== undefined) {
          if (hasError) {
            await expect(fieldErrorEl).toBeVisible();
          } else {
            await expect(fieldErrorEl).not.toBeVisible();
          }
        }

        if (hasErrorMsg !== undefined) {
          await expect(fieldErrorEl).toBeVisible();

          await expect(fieldErrorEl.locator('> div').filter({ hasText: hasErrorMsg }).first()).toHaveText(hasErrorMsg);
        }
      },
    };
  }

  async addLimitToRangeMinOrMax({ type, isMinValue, value }: { type: UITypes; isMinValue: boolean; value: string }) {
    const fieldLocator = this.get().getByTestId(`nc-limit-to-range-${isMinValue ? 'min' : 'max'}-${type}`);

    switch (type) {
      default: {
        await fieldLocator.locator('input:visible').waitFor();

        await this.waitForResponse({
          uiAction: async () => {
            await fieldLocator.locator(`input`).click();
            await fieldLocator.locator(`input`).fill(value);
            await this.rootPage.keyboard.press('Enter');
          },
          requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
          httpMethodsToMatch: ['PATCH'],
        });
      }
    }
  }

  async getFormFieldsValidateLimitToRange({ type }: { type: UITypes }) {
    const validateBtn = this.get().getByTestId(`nc-limit-to-range-${type}`);

    return {
      locator: validateBtn,
      click: async ({ enable, min, max }: { enable: boolean; min?: string; max?: string }) => {
        await validateBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateBtn.isChecked();

        if (enable && !isEnabled) {
          await validateBtn.click();
          await this.get().locator('.nc-limit-to-range-wrapper').first().waitFor({ state: 'visible' });

          if (min !== undefined) {
            await this.addLimitToRangeMinOrMax({ type, isMinValue: true, value: min });
          }

          if (max !== undefined) {
            await this.addLimitToRangeMinOrMax({ type, isMinValue: false, value: max });
          }
        } else if (!enable && isEnabled) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({
        isEnabled,
        isDisabled,
        isVisible,
      }: {
        isEnabled?: boolean;
        isDisabled?: boolean;
        isVisible?: boolean;
      }) => {
        if (isEnabled !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });

          if (isEnabled) {
            await expect(validateBtn).toBeChecked();
          } else {
            await expect(validateBtn).not.toBeChecked();
          }
        }

        if (isDisabled !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });

          if (isDisabled) {
            await expect(validateBtn).toBeDisabled();
          } else {
            await expect(validateBtn).not.toBeDisabled();
          }
        }

        if (isVisible !== undefined) {
          if (isVisible) {
            await expect(validateWorkEmailBtn).toBeVisible();
          } else {
            await expect(validateWorkEmailBtn).not.toBeVisible();
          }
        }
      },
    };
  }

  async getFormFieldsValidateAttFileType() {
    const validateBtn = this.get().getByTestId('nc-att-limit-file-type');
    const validationWrapper = this.get().locator('.nc-att-limit-file-type-wrapper');
    await validateBtn.scrollIntoViewIfNeeded();

    return {
      locator: validateBtn,
      click: async ({ enable, fillValue }: { enable: boolean; fillValue?: string }) => {
        await validateBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateBtn.isChecked();

        if (enable && !isEnabled) {
          await validateBtn.click();
        } else if (!enable && isEnabled) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
        if (enable && fillValue !== undefined) {
          await validationWrapper.locator('input').first().waitFor({ state: 'visible' });

          await this.waitForResponse({
            uiAction: async () => {
              await validationWrapper.locator('input').fill(fillValue);
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({ isEnabled, hasError }: { isEnabled?: boolean; hasError?: boolean }) => {
        if (isEnabled !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });

          if (isEnabled) {
            await expect(validateBtn).toBeChecked();
          } else {
            await expect(validateBtn).not.toBeChecked();
          }
        }

        if (hasError !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });
          await validationWrapper.waitFor({ state: 'visible' });

          if (hasError) {
            await expect(validationWrapper.locator('.validation-input-error')).toBeVisible();
          } else {
            await expect(validationWrapper.locator('.validation-input-error')).not.toBeVisible();
          }
        }
      },
    };
  }
  async getFormFieldsValidateAttFileCount() {
    const validateBtn = this.get().getByTestId('nc-att-limit-file-count');
    const validationWrapper = this.get().locator('.nc-att-limit-file-count-wrapper');
    await validateBtn.scrollIntoViewIfNeeded();

    return {
      locator: validateBtn,
      click: async ({ enable, fillValue }: { enable: boolean; fillValue?: string }) => {
        await validateBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateBtn.isChecked();

        if (enable && !isEnabled) {
          await validateBtn.click();
        } else if (!enable && isEnabled) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
        if (enable && fillValue !== undefined) {
          await validationWrapper.locator('input').first().waitFor({ state: 'visible' });

          await this.waitForResponse({
            uiAction: async () => {
              await validationWrapper.locator('input').fill(fillValue);
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({ isEnabled, hasError }: { isEnabled?: boolean; hasError?: boolean }) => {
        if (isEnabled !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });

          if (isEnabled) {
            await expect(validateBtn).toBeChecked();
          } else {
            await expect(validateBtn).not.toBeChecked();
          }
        }

        if (hasError !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });
          await validationWrapper.waitFor({ state: 'visible' });

          if (hasError) {
            await expect(validationWrapper.locator('.validation-input-error')).toBeVisible();
          } else {
            await expect(validationWrapper.locator('.validation-input-error')).not.toBeVisible();
          }
        }
      },
    };
  }
  async getFormFieldsValidateAttFileSize() {
    const validateBtn = this.get().getByTestId('nc-att-limit-file-size');
    const validationWrapper = this.get().locator('.nc-att-limit-file-size-wrapper');
    await validateBtn.scrollIntoViewIfNeeded();

    return {
      locator: validateBtn,
      click: async ({
        enable,
        fillValue,
        unit = 'KB',
      }: {
        enable: boolean;
        fillValue?: string;
        unit?: 'KB' | 'MB';
      }) => {
        await validateBtn.waitFor({ state: 'visible' });

        const isEnabled = await validateBtn.isChecked();

        if (enable && !isEnabled) {
          await validateBtn.click();
        } else if (!enable && isEnabled) {
          await this.waitForResponse({
            uiAction: async () => {
              await validateBtn.click();
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }

        if (enable && fillValue !== undefined) {
          await validationWrapper.locator('.nc-validation-input-wrapper input').first().waitFor({ state: 'visible' });

          await validationWrapper.locator('.ant-select-selector').click();
          const dropdown = this.rootPage.locator('.nc-att-limit-file-size-unit-selector-dropdown');
          await dropdown.waitFor({ state: 'visible' });
          const option = dropdown.locator('.ant-select-item-option').getByText(unit);
          await option.scrollIntoViewIfNeeded();
          await option.waitFor({ state: 'visible' });
          await option.click();

          await this.waitForResponse({
            uiAction: async () => {
              await validationWrapper.locator('.nc-validation-input-wrapper input').fill(fillValue);
            },
            requestUrlPathToMatch: '/api/v1/db/meta/form-columns',
            httpMethodsToMatch: ['PATCH'],
          });
        }
      },
      verify: async ({ isEnabled, hasError }: { isEnabled?: boolean; hasError?: boolean }) => {
        if (isEnabled !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });

          if (isEnabled) {
            await expect(validateBtn).toBeChecked();
          } else {
            await expect(validateBtn).not.toBeChecked();
          }
        }

        if (hasError !== undefined) {
          await validateBtn.waitFor({ state: 'visible' });
          await validationWrapper.waitFor({ state: 'visible' });

          if (hasError) {
            await expect(validationWrapper.locator('.validation-input-error')).toBeVisible();
          } else {
            await expect(validationWrapper.locator('.validation-input-error')).not.toBeVisible();
          }
        }
      },
    };
  }
}