Skip to content

Commit

Permalink
feat(password): add tests about odsChange emit on first render
Browse files Browse the repository at this point in the history
  • Loading branch information
dpellier committed Nov 28, 2024
1 parent af559b0 commit 93085eb
Show file tree
Hide file tree
Showing 4 changed files with 184 additions and 7 deletions.
Original file line number Diff line number Diff line change
@@ -1,11 +1,9 @@
import { AttachInternals, Component, Event, type EventEmitter, type FunctionalComponent, Host, Listen, Method, Prop, State, h } from '@stencil/core';
import { submitFormOnEnter } from '../../../../../utils/dom';
import { type OdsInput, type OdsInputChangeEvent } from '../../../../input/src';
import { updateInternals } from '../../controller/ods-password';
import { VALUE_DEFAULT_VALUE, getInitialValue, updateInternals } from '../../controller/ods-password';
import { type OdsPasswordChangeEventDetail } from '../../interfaces/events';

const VALUE_DEFAULT_VALUE = null;

@Component({
formAssociated: true,
shadow: {
Expand Down Expand Up @@ -104,9 +102,7 @@ export class OdsPassword {
}

componentWillLoad(): void {
if (!this.value && (this.value !== VALUE_DEFAULT_VALUE || this.defaultValue)) {
this.value = this.defaultValue ?? null;
}
this.value = getInitialValue(this.value, this.defaultValue);
}

async componentDidLoad(): Promise<void> {
Expand Down Expand Up @@ -135,7 +131,8 @@ export class OdsPassword {

render(): FunctionalComponent {
return (
<Host class="ods-password"
<Host
class="ods-password"
disabled={ this.isDisabled }
readonly={ this.isReadonly }>
<ods-input
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,16 @@
import type { OdsInput } from '../../../input/src';
import { setInternalsValidityFromOdsComponent } from '../../../../utils/dom';

const VALUE_DEFAULT_VALUE = null;

function getInitialValue(value: string | null, defaultValue?: string): string | null {
if (defaultValue !== undefined && value === VALUE_DEFAULT_VALUE) {
return defaultValue;
}

return value;
}

async function updateInternals(internals: ElementInternals, value: string | null, inputEl?: HTMLElement & OdsInput): Promise<void> {
internals.setFormValue(value?.toString() ?? '');

Expand All @@ -10,5 +20,7 @@ async function updateInternals(internals: ElementInternals, value: string | null
}

export {
getInitialValue,
updateInternals,
VALUE_DEFAULT_VALUE,
};
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { type E2EElement, type E2EPage, newE2EPage } from '@stencil/core/testing';
import { type OdsPasswordChangeEventDetail } from '../../src';

describe('ods-password behaviour', () => {
let el: E2EElement;
Expand All @@ -15,6 +16,105 @@ describe('ods-password behaviour', () => {

beforeEach(jest.clearAllMocks);

describe('initialization', () => {
let odsChangeEventCount = 0;
let odsChangeEventDetail = {};

async function setupWithSpy(content: string): Promise<void> {
odsChangeEventCount = 0;
odsChangeEventDetail = {};
page = await newE2EPage();

// page.spyOnEvent doesn't seems to work to observe event emitted on first render, before any action happens
// so we spy manually
await page.exposeFunction('onOdsChangeEvent', (detail: OdsPasswordChangeEventDetail) => {
odsChangeEventCount++;
odsChangeEventDetail = detail;
});

await page.evaluateOnNewDocument(() => {
window.addEventListener('odsChange', (event: Event) => {
// @ts-ignore function is exposed manually
window.onOdsChangeEvent((event as CustomEvent<OdsPasswordChangeEventDetail>).detail);
});
});

await page.setContent(content);
}

describe('with no value attribute defined', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: null,
});
});
});

describe('with empty string value', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password value=""></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: '',
});
});
});

describe('with no value but empty default-value', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password default-value=""></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: '',
});
});
});

describe('with no value but default-value defined', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password default-value="default"></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: 'default',
});
});
});

describe('with defined value', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password value="value"></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: 'value',
});
});
});

describe('with defined value and default value', () => {
it('should trigger a uniq odsChange event', async() => {
await setupWithSpy('<ods-password default-value="default" value="value"></ods-password>');

expect(odsChangeEventCount).toBe(1);
expect(odsChangeEventDetail).toEqual({
validity: {},
value: 'value',
});
});
});
});

describe('methods', () => {
describe('clear', () => {
it('should receive odsClear event', async() => {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
jest.mock('../../../../utils/dom');

import { setInternalsValidityFromOdsComponent } from '../../../../utils/dom';
import { type OdsInput } from '../../../input/src';
import { getInitialValue, updateInternals } from '../../src/controller/ods-password';

describe('ods-password controller', () => {
beforeEach(jest.clearAllMocks);

describe('getInitialValue', () => {
it('should return null if value is null and no default value', () => {
expect(getInitialValue(null)).toBe(null);
});

it('should return string if value is set regarding of default value', () => {
expect(getInitialValue('')).toBe('');
expect(getInitialValue('value')).toBe('value');
expect(getInitialValue('value', 'default')).toBe('value');
// @ts-ignore for test purpose
expect(getInitialValue('value', null)).toBe('value');
});

it('should return default value if value is null', () => {
expect(getInitialValue(null, '')).toBe('');
expect(getInitialValue(null, 'default')).toBe('default');
});
});

describe('updateInternals', () => {
const dummyInput = { dummy: 'input' };
const dummyInternal = {
setFormValue: jest.fn(),
} as unknown as ElementInternals;

it('should set internal value with empty string', async() => {
// @ts-ignore for test purpose
await updateInternals(dummyInternal);
expect(dummyInternal.setFormValue).toHaveBeenCalledWith('');

// @ts-ignore for test purpose
await updateInternals(dummyInternal, undefined, {} as HTMLElement & OdsInput);
expect(dummyInternal.setFormValue).toHaveBeenCalledWith('');

await updateInternals(dummyInternal, null, {} as HTMLElement & OdsInput);
expect(dummyInternal.setFormValue).toHaveBeenCalledWith('');
});

it('should set internal value with string value', async() => {
const dummyValue = 'dummy value';

await updateInternals(dummyInternal, dummyValue, {} as HTMLElement & OdsInput);

expect(dummyInternal.setFormValue).toHaveBeenCalledWith(dummyValue);
});

it('should not set internal validity if no input element is defined', async() => {
await updateInternals(dummyInternal, 'dummyValue');

expect(setInternalsValidityFromOdsComponent).not.toHaveBeenCalled();
});

it('should set internal validity if input element is defined', async() => {
await updateInternals(dummyInternal, 'dummyValue', dummyInput as unknown as HTMLElement & OdsInput);

expect(setInternalsValidityFromOdsComponent).toHaveBeenCalledWith(dummyInput, dummyInternal);
});
});
});

0 comments on commit 93085eb

Please sign in to comment.