decide.nolog.cz/tests/acceptance/create-a-poll-test.js
Jeldrik Hanschke 0f94d81ad4
fix ember-cli-page-object deprecations (#674)
* fix ember-cli-page-object deprecations

* make linter happy
2023-10-15 23:26:18 +02:00

1351 lines
44 KiB
JavaScript

import {
click,
currentURL,
currentRouteName,
fillIn,
find,
findAll,
settled,
visit,
waitFor,
} from '@ember/test-helpers';
import { module, test } from 'qunit';
import { setupApplicationTest } from 'ember-qunit';
import { setupMirage } from 'ember-cli-mirage/test-support';
import { setupIntl, t } from 'ember-intl/test-support';
import {
setupBrowserNavigationButtons,
backButton,
} from 'ember-cli-browser-navigation-button-test-helper/test-support';
import { DateTime } from 'luxon';
import pageCreateIndex from 'croodle/tests/pages/create/index';
import pageCreateMeta from 'croodle/tests/pages/create/meta';
import pageCreateOptions from 'croodle/tests/pages/create/options';
import pageCreateOptionsDatetime from 'croodle/tests/pages/create/options-datetime';
import pageCreateSettings from 'croodle/tests/pages/create/settings';
import pagePollParticipation from 'croodle/tests/pages/poll/participation';
import asyncThrowsAssertion from '../assertions/async-throws';
import { calendarSelect } from 'ember-power-calendar/test-support';
module('Acceptance | create a poll', function (hooks) {
hooks.beforeEach(function () {
window.localStorage.setItem('locale', 'en');
});
setupApplicationTest(hooks);
setupMirage(hooks);
setupIntl(hooks);
hooks.beforeEach(function (assert) {
assert.asyncThrows = asyncThrowsAssertion;
});
test('create a default poll', async function (assert) {
const dates = [
DateTime.now().plus({ days: 1 }),
DateTime.now().plus({ weeks: 1 }),
];
await pageCreateIndex.visit();
assert.equal(currentRouteName(), 'create.index');
assert.equal(
pageCreateIndex.statusBar().active,
t('create.formStep.type').toString(),
'status bar shows correct item as current path (index)'
);
assert.deepEqual(
pageCreateIndex
.statusBar()
.toArray()
.map((el) => el.text),
[
t('create.formStep.type').toString(),
t('create.formStep.meta').toString(),
t('create.formStep.options.days').toString(),
t('create.formStep.options-datetime').toString(),
t('create.formStep.settings').toString(),
],
'status bar has correct items'
);
assert.deepEqual(
pageCreateIndex
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, true, true, true, true],
'status bar has correct items disabled (index)'
);
assert.ok(
pageCreateIndex.pollTypeHasFocus,
'poll type selection has autofocus'
);
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
assert.equal(
pageCreateMeta.statusBar().active,
t('create.formStep.meta').toString(),
'status bar shows correct item as current path (meta)'
);
assert.deepEqual(
pageCreateMeta
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, true, true, true],
'status bar has correct items disabled (meta)'
);
assert.ok(pageCreateMeta.titleHasFocus, 'title input has autofocus');
await pageCreateMeta.title('default poll').next();
assert.equal(currentRouteName(), 'create.options');
assert.equal(
pageCreateOptions.statusBar().active,
t('create.formStep.options.days').toString(),
'status bar shows correct item as current path (options.days)'
);
assert.deepEqual(
pageCreateOptions
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, false, true, true],
'status bar has correct items disabled (options)'
);
await pageCreateOptions.selectDates(dates);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.equal(
pageCreateOptionsDatetime.statusBar().active,
t('create.formStep.options-datetime').toString(),
'status bar shows correct item as current path (options-datetime)'
);
assert.deepEqual(
pageCreateOptionsDatetime
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, false, false, true],
'status bar has correct items disabled (options-datetime)'
);
assert.ok(
pageCreateOptionsDatetime.firstTime.inputHasFocus,
'first time input has autofocus'
);
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
assert.equal(
pageCreateSettings.statusBar().active,
t('create.formStep.settings').toString(),
'status bar shows correct item as current path (settings)'
);
assert.deepEqual(
pageCreateSettings
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, false, false, false],
'status bar has correct items disabled (settings)'
);
assert.ok(
pageCreateSettings.availableAnswersHasFocus,
'available answers selection has autofocus'
);
// simulate temporary server error
this.server.post('/polls', undefined, 503);
await assert.asyncThrows(async () => {
await pageCreateSettings.save();
}, 'Ember Data Request POST /api/index.php/polls returned a 503');
assert.equal(currentRouteName(), 'create.settings');
// simulate server is available again
// defer creation for testing loading spinner
let resolveSubmission;
let resolveSubmissionWith;
this.server.post('/polls', function (schema) {
return new Promise((resolve) => {
let attrs = this.normalizedRequestAttrs();
resolveSubmission = resolve;
resolveSubmissionWith = schema.polls.create(attrs);
});
});
pageCreateSettings.save();
// shows loading spinner while saving
await waitFor('[data-test-button="submit"] .spinner-border', {
timeoutMessage: 'timeout while waiting for loading spinner to appear',
});
assert.ok(true, 'loading spinner is shown');
resolveSubmission(resolveSubmissionWith);
await settled();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(
pagePollParticipation.urlIsValid(),
`poll url ${currentURL()} is valid`
);
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
dates.map((date) =>
Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(date)
),
'options are correctly labeled'
);
assert.deepEqual(
Array.from(
find('[data-test-form-element^="option"]').querySelectorAll(
'.radio label'
)
).map((el) => el.textContent.trim()),
[
t('answerTypes.yes.label').toString(),
t('answerTypes.no.label').toString(),
],
'answers are correctly labeled'
);
assert.ok(pagePollParticipation.nameHasFocus, 'name input has autofocus');
});
test('create a poll for answering a question', async function (assert) {
await pageCreateIndex.visit();
assert.equal(
pageCreateIndex.statusBar().active,
t('create.formStep.type').toString(),
'status bar shows correct item as current path (index)'
);
assert.deepEqual(
pageCreateIndex
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, true, true, true, true],
'status bar has correct items disabled'
);
await pageCreateIndex.pollType('MakeAPoll').next();
assert.equal(currentRouteName(), 'create.meta');
assert.equal(
pageCreateMeta.statusBar().active,
t('create.formStep.meta').toString(),
'status bar shows correct item as current path (meta)'
);
assert.deepEqual(
pageCreateMeta
.statusBar()
.toArray()
.map((el) => el.text),
[
t('create.formStep.type').toString(),
t('create.formStep.meta').toString(),
t('create.formStep.options.text').toString(),
t('create.formStep.settings').toString(),
],
'status bar has correct items'
);
assert.deepEqual(
pageCreateMeta
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, true, true],
'status bar has correct items disabled (meta)'
);
await pageCreateMeta.title('default poll').next();
assert.equal(currentRouteName(), 'create.options');
assert.equal(
pageCreateOptions.statusBar().active,
t('create.formStep.options.text').toString(),
'status bar shows correct item as current path (options.text)'
);
assert.deepEqual(
pageCreateOptions
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, false, true],
'status bar has correct items disabled (options)'
);
assert.ok(
pageCreateOptions.firstTextOption.inputHasFocus,
'first option input has autofocus'
);
assert.equal(
pageCreateOptions.textOptions().count,
2,
'there are two input fields as default'
);
await pageCreateOptions.next();
assert.equal(
currentRouteName(),
'create.options',
'validation errors prevents transition'
);
assert.equal(
pageCreateOptions.statusBar().active,
t('create.formStep.options.text').toString(),
'status bar shows correct item as current path (options.text)'
);
assert.ok(
pageCreateOptions.textOptions(0).hasError &&
pageCreateOptions.textOptions(1).hasError,
'validation errors are shown after submit'
);
await pageCreateOptions.textOptions(0).title('option a');
await pageCreateOptions.textOptions(1).title('option c');
await pageCreateOptions.textOptions(0).add();
assert.equal(pageCreateOptions.textOptions().count, 3, 'option was added');
await pageCreateOptions.textOptions(1).title('option b');
await pageCreateOptions.textOptions(2).add();
assert.equal(pageCreateOptions.textOptions().count, 4, 'option was added');
await pageCreateOptions.textOptions(3).title('to be deleted');
await pageCreateOptions.textOptions(3).delete();
assert.equal(
pageCreateOptions.textOptions().count,
3,
'option got deleted'
);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.settings');
assert.equal(
pageCreateSettings.statusBar().active,
t('create.formStep.settings').toString(),
'status bar shows correct item as current path (settings)'
);
assert.deepEqual(
pageCreateSettings
.statusBar()
.toArray()
.map((el) => el.isDisabled),
[false, false, false, false],
'status bar has correct items disabled (settings)'
);
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
['option a', 'option b', 'option c'],
'options are labeled correctly'
);
});
test('create a poll with times and description', async function (assert) {
const days = [
DateTime.now().plus({ days: 1 }),
DateTime.now().plus({ weeks: 1 }),
];
await pageCreateIndex.visit();
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta
.title('default poll')
.description('a sample description')
.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates(days);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.deepEqual(
findAll('[data-test-day] label').map((el) => el.textContent.trim()),
days.map((day) =>
Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)
),
'time inputs having days as label'
);
await pageCreateOptionsDatetime.times(0).time('10:00');
await pageCreateOptionsDatetime.times(0).add();
await pageCreateOptionsDatetime.times(1).time('18:00');
await pageCreateOptionsDatetime.times(2).time('12:00');
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'a sample description',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(days[0].set({ hour: 10, minutes: 0 })),
Intl.DateTimeFormat('en-US', { timeStyle: 'short' }).format(
days[0].set({ hours: 18, minutes: 0 })
),
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(days[1].set({ hour: 12, minutes: 0 })),
],
'options are correctly labeled'
);
});
test('create a poll with only one day and multiple times', async function (assert) {
const day = DateTime.now().plus({ days: 1 });
await pageCreateIndex.visit();
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta
.title('default poll')
.description('a sample description')
.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates([day]);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.deepEqual(
findAll('[data-test-day] label').map((el) => el.textContent.trim()),
[Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)],
'time inputs having days as label'
);
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.exists({ count: 1 }, 'shows one input field per day as default');
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.hasValue('', 'time input field is empty until user enters a time');
await pageCreateOptionsDatetime.times(0).time('08:00');
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.hasValue('08:00', 'time input field shows the time entered by the user');
await pageCreateOptionsDatetime.times(0).delete();
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.hasValue(
'',
'deleting clears time input value if there is only one time'
);
await pageCreateOptionsDatetime.times(0).time('10:00');
await pageCreateOptionsDatetime.times(0).add();
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.exists({ count: 2 }, 'user can add a second input field');
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[1]
)
.hasValue('', 'newly crated second input field is empty');
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[0]
)
.hasValue('10:00', 'value existing input field is not changed');
await pageCreateOptionsDatetime.times(1).time('14:00');
await pageCreateOptionsDatetime.times(1).add();
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.exists({ count: 3 }, 'user can add a third input field');
await pageCreateOptionsDatetime.times(2).time('18:00');
await pageCreateOptionsDatetime.times(1).delete();
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.exists({ count: 2 }, 'user can delete an input field');
assert.deepEqual(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`).map(
(el) => el.value
),
['10:00', '18:00'],
'correct input field is deleted'
);
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'a sample description',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(day.set({ hours: 10, minutes: 0 })),
Intl.DateTimeFormat('en-US', { timeStyle: 'short' }).format(
day.set({ hours: 18, minutes: 0 })
),
],
'options are correctly labeled'
);
});
test('create a poll with only one day (without time)', async function (assert) {
const day = DateTime.now().plus({ days: 1 });
await pageCreateIndex.visit();
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta
.title('default poll')
.description('a sample description')
.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates([day]);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.deepEqual(
findAll('[data-test-day] label').map((el) => el.textContent.trim()),
[Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)],
'time inputs having days as label'
);
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'a sample description',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)],
'options are correctly labeled'
);
});
test('create a poll with only one day (with time)', async function (assert) {
const day = DateTime.now().plus({ days: 1 });
await pageCreateIndex.visit();
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta
.title('default poll')
.description('a sample description')
.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates([day]);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.deepEqual(
findAll('[data-test-day] label').map((el) => el.textContent.trim()),
[Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)],
'time inputs having days as label'
);
await pageCreateOptionsDatetime.times(0).time('22:30');
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'a sample description',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(day.set({ hours: 22, minutes: 30 })),
],
'options are correctly labeled'
);
});
test('create a poll with times by adopting times of first day', async function (assert) {
const days = [
DateTime.now().plus({ days: 1 }),
DateTime.now().plus({ weeks: 1 }),
];
await visit('/create');
await click('button[type="submit"]');
assert.equal(currentRouteName(), 'create.meta');
await fillIn(
'[data-test-form-element="title"] input[type="text"]',
'example poll for to test time adopting workflow'
);
await click('button[type="submit"]');
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates(days);
await click('button[type="submit"]');
assert.equal(currentRouteName(), 'create.options-datetime');
for (let i = 1; i <= 3; i++) {
await click(
`[data-test-day="${days[0].toISODate()}"] button[data-test-action="add"]`
);
}
assert
.dom(
`[data-test-day="${days[0].toISODate()}"] button[data-test-action="add"]`
)
.exists({ count: 4 }, 'assumption: user created 4 time inputs');
for (const [index, inputEl] of findAll(
`[data-test-day="${days[0].toISODate()}"] input[type="time"]`
).entries()) {
await fillIn(inputEl, `${(index * 6).toString().padStart(2, '0')}:00`);
}
assert.deepEqual(
findAll(
`[data-test-day="${days[0].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['00:00', '06:00', '12:00', '18:00'],
'assumption: all 4 time inputs for first day are filled'
);
assert
.dom(`[data-test-day="${days[1].toISODate()}"] input[type="time"]`)
.exists({ count: 1 }, 'only one time input exists for second day')
.hasValue('', 'time input for second day is empty');
await click('button[data-test-action="adopt-times-of-first-day"]');
assert.deepEqual(
findAll(
`[data-test-day="${days[1].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['00:00', '06:00', '12:00', '18:00'],
'all 4 times from first day have been added to second day'
);
await click(
findAll(
`[data-test-day="${days[0].toISODate()}"] button[data-test-action="delete"]`
)[2]
);
assert.deepEqual(
findAll(
`[data-test-day="${days[0].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['00:00', '06:00', '18:00'],
'assumption: one time has been deleted from first day'
);
await click('button[data-test-action="adopt-times-of-first-day"]');
assert.deepEqual(
findAll(
`[data-test-day="${days[1].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['00:00', '06:00', '18:00'],
'second day has been updated with changed times from first day'
);
await fillIn(
findAll(`[data-test-day="${days[0].toISODate()}"] input[type="time"]`)[0],
'03:00'
);
await click(
findAll(
`[data-test-day="${days[0].toISODate()}"] button[data-test-action="add"]`
)[2]
);
await fillIn(
findAll(`[data-test-day="${days[0].toISODate()}"] input[type="time"]`)[3],
'22:00'
);
assert.deepEqual(
findAll(
`[data-test-day="${days[0].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['03:00', '06:00', '18:00', '22:00'],
'assumption: a fourth time has been added to the first day again'
);
await click('button[data-test-action="adopt-times-of-first-day"]');
assert.deepEqual(
findAll(
`[data-test-day="${days[1].toISODate()}"] input[type="time"]`
).map((el) => el.value),
['03:00', '06:00', '18:00', '22:00'],
'second day has been updated with times from first day as expected'
);
await click('button[type="submit"]');
assert.equal(currentRouteName(), 'create.settings');
await click('button[type="submit"]');
assert.equal(currentRouteName(), 'poll.participation');
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(days[0].set({ hour: 3, minutes: 0 })),
'6:00 AM',
'6:00 PM',
'10:00 PM',
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(days[1].set({ hour: 3, minutes: 0 })),
'6:00 AM',
'6:00 PM',
'10:00 PM',
],
'options are correctly labeled'
);
});
test('create a poll for answering a question with only one option', async function (assert) {
await pageCreateIndex.visit();
await pageCreateIndex.pollType('MakeAPoll').next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta.title('default poll').next();
assert.equal(currentRouteName(), 'create.options');
assert.equal(
pageCreateOptions.textOptions().count,
2,
'there are two input fields as default'
);
await pageCreateOptions.textOptions(0).title('option a');
await pageCreateOptions.textOptions(1).delete();
assert.equal(
pageCreateOptions.textOptions().count,
1,
'option was deleted'
);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
['option a'],
'options are labeled correctly'
);
});
test('create a poll and use back button (find a date)', async function (assert) {
let days = [DateTime.fromISO('2016-01-02'), DateTime.fromISO('2016-01-13')];
setupBrowserNavigationButtons();
await pageCreateIndex.visit();
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta
.title('default poll')
.description('a sample description')
.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates(days);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert.deepEqual(
findAll('[data-test-day] label').map((el) => el.textContent.trim()),
days.map((day) =>
Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(day)
),
'time inputs having days as label'
);
await fillIn('[data-test-day="2016-01-13"] input[type="time"]', '10:00');
assert
.dom('[data-test-day="2016-01-13"] input[type="time"]')
.hasValue('10:00', 'time input has the value entered by the user');
await backButton();
assert.equal(currentRouteName(), 'create.options');
assert.deepEqual(
findAll('.ember-power-calendar-day--selected').map(
(el) => el.dataset.date
),
days.map((day) => day.toISODate()),
'days are still present after back button is used'
);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
assert
.dom('[data-test-day="2016-01-02"] input[type="time"]')
.hasValue('', 'time input the user has not touched is still empty');
assert
.dom('[data-test-day="2016-01-13"] input[type="time"]')
.hasValue(
'10:00',
'time input is prefilled with the time user entered before using back button'
);
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.save();
assert.equal(currentRouteName(), 'poll.participation');
assert.true(pagePollParticipation.urlIsValid(), 'poll url is valid');
assert.equal(
pagePollParticipation.title,
'default poll',
'poll title is correct'
);
assert.equal(
pagePollParticipation.description,
'a sample description',
'poll description is correct'
);
assert.deepEqual(
findAll(
`[data-test-form-element^="option"] label:not(.custom-control-label)`
).map((el) => el.textContent.trim()),
[
Intl.DateTimeFormat('en-US', { dateStyle: 'full' }).format(days[0]),
Intl.DateTimeFormat('en-US', {
dateStyle: 'full',
timeStyle: 'short',
}).format(days[1].set({ hours: 10, minutes: 0 })),
],
'options are correctly labeled'
);
});
test('Start at first step is enforced', async function (assert) {
await pageCreateSettings.visit();
assert.equal(currentRouteName(), 'create.index');
});
test('back button', async function (assert) {
await pageCreateIndex.visit();
assert.equal(currentRouteName(), 'create.index');
await pageCreateIndex.next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta.title('foo').next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.selectDates([new Date()]);
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.options-datetime');
await pageCreateOptionsDatetime.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.back();
assert.equal(currentRouteName(), 'create.options-datetime');
await pageCreateOptionsDatetime.back();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.back();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta.back();
assert.equal(currentRouteName(), 'create.index');
await pageCreateIndex.pollType('MakeAPoll').next();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta.next();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.textOptions(0).add();
await pageCreateOptions.textOptions(1).title('bar');
await pageCreateOptions.next();
assert.equal(currentRouteName(), 'create.settings');
await pageCreateSettings.back();
assert.equal(currentRouteName(), 'create.options');
await pageCreateOptions.back();
assert.equal(currentRouteName(), 'create.meta');
await pageCreateMeta.back();
assert.equal(currentRouteName(), 'create.index');
});
module('validation', async function () {
test('validates user input when creating a poll with dates and times', async function (assert) {
const day = DateTime.now();
await visit('/create');
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/meta',
'assumption: user can go to next step after selecting poll type'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/meta',
'validation error prevents the user from going to option input before entering a title'
);
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-invalid',
'input field for poll type is shown as invalid'
);
assert
.dom('[data-test-form-element="title"] .invalid-feedback')
.hasText(
t('create.meta.input.title.validations.valueMissing'),
'shows a validation error for missing value'
);
assert
.dom('[data-test-form-element="description"] textarea')
.hasClass(
'is-valid',
'textarea for entering description is shown as valid even if user has not entered a value'
);
await fillIn('[data-test-form-element="title"] input', 'A');
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-invalid',
'input field for poll type is still shown as invalid after user entered a too short title'
);
assert
.dom('[data-test-form-element="title"] .invalid-feedback')
.hasText(
t('create.meta.input.title.validations.tooShort'),
'validation error message is updated to reflect too short value'
);
await fillIn(
'[data-test-form-element="title"] input',
'When to have our next hackathon?'
);
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-valid',
'input field for poll type is shown as valid after user entered a title'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options',
'assumption: user can go to next step after filling in poll title'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options',
'user can not skip options step without selecting at least one day'
);
assert
.dom('[data-test-form-element-for="days"] .form-control')
.hasClass(
'is-invalid',
'shows calendar for date selection as invalid if user has selected no day'
);
assert
.dom('[data-test-form-element-for="days"] .invalid-feedback')
.hasText(
t('create.options.error.notEnoughDates'),
'shows validation error that at least one day needs to be selected'
);
await calendarSelect(
'[data-test-form-element-for="days"]',
day.toJSDate()
);
assert
.dom('[data-test-form-element-for="days"] .form-control')
.hasClass(
'is-valid',
'shows calendar for date selection as valid after user selected a date'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options-datetime',
'user can process to next step after selecting at least one day'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/settings',
'user can skip time input for dates without entering any time'
);
await click('button[data-test-action="back"]');
assert.strictEqual(
currentURL(),
'/create/options-datetime',
'assumption: user can go back to time input step'
);
await click(
`[data-test-day="${day.toISODate()}"] button[data-test-action="add"]`
);
assert
.dom(`[data-test-day="${day.toISODate()}"] input[type="time"]`)
.exists(
{ count: 2 },
'assumption: user can add another time for the day'
);
for (const el of findAll(
`[data-test-day="${day.toISODate()}"] input[type="time"]`
)) {
await fillIn(el, '10:00');
}
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options-datetime',
'user can not go to next step when entering duplicated times for the same day'
);
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[0]
)
.hasClass('is-valid', 'first input is shown as valid');
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[1]
)
.hasClass(
'is-invalid',
'second input with same time is shown as invalid'
);
assert
.dom(`[data-test-day="${day.toISODate()}"] .invalid-feedback`)
.exists(
{ count: 1 },
'assumption: only one input has invalid feedback'
);
assert
.dom(`[data-test-day="${day.toISODate()}"] .invalid-feedback`)
.hasText(
t('create.options-datetime.error.duplicatedDate'),
'validation error message tells that times must be unique'
);
await fillIn(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[1],
'12:00'
);
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[0]
)
.hasClass(
'is-valid',
'first input is still shown as valid when user changes value of another input'
);
assert
.dom(
findAll(`[data-test-day="${day.toISODate()}"] input[type="time"]`)[1]
)
.hasClass(
'is-valid',
'second input is shown as valid after user filled in another time'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/settings',
'user can skip time input for dates without entering any time'
);
await click('button[type="submit"]');
assert.strictEqual(
currentRouteName(),
'poll.participation',
'user can finish the poll creation without changing any value on settings step'
);
});
test('validates user input when creating a poll for answering a question', async function (assert) {
await visit('/create');
await fillIn('[data-test-form-element="poll-type"] select', 'MakeAPoll');
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/meta',
'assumption: user can go to next step after selecting poll type'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/meta',
'validation error prevents the user from going to option input before entering a title'
);
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-invalid',
'input field for poll type is shown as invalid'
);
assert
.dom('[data-test-form-element="title"] .invalid-feedback')
.hasText(
t('create.meta.input.title.validations.valueMissing'),
'shows a validation error for missing value'
);
assert
.dom('[data-test-form-element="description"] textarea')
.hasClass(
'is-valid',
'textarea for entering description is shown as valid even if user has not entered a value'
);
await fillIn('[data-test-form-element="title"] input', 'A');
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-invalid',
'input field for poll type is still shown as invalid after user entered a too short title'
);
assert
.dom('[data-test-form-element="title"] .invalid-feedback')
.hasText(
t('create.meta.input.title.validations.tooShort'),
'validation error message is updated to reflect too short value'
);
await fillIn(
'[data-test-form-element="title"] input',
'What dessert should we have?'
);
assert
.dom('[data-test-form-element="title"] input')
.hasClass(
'is-valid',
'input field for poll type is shown as valid after user entered a title'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options',
'assumption: user can go to next step after filling in poll title'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/options',
'user can not skip options step without filling out at least one option'
);
assert
.dom('[data-test-form-element="option"] input[type="text"]')
.exists(
{ count: 2 },
'assumption: two input fields for options exists'
);
assert
.dom('[data-test-form-element="option"][data-test-option="0"] input')
.hasClass('is-invalid', 'input field for first option is invalid');
assert
.dom('[data-test-form-element="option"][data-test-option="1"] input')
.hasClass('is-invalid', 'input field for second option is invalid');
assert
.dom(
'[data-test-form-element="option"][data-test-option="0"] .invalid-feedback'
)
.hasText(
t('create.options.error.valueMissing'),
'shown value missing validation error for first input'
);
assert
.dom(
'[data-test-form-element="option"][data-test-option="1"] .invalid-feedback'
)
.hasText(
t('create.options.error.valueMissing'),
'shown value missing validation error for first input'
);
await fillIn(
'[data-test-form-element="option"][data-test-option="0"] input',
'Cheesecake'
);
assert
.dom('[data-test-form-element="option"][data-test-option="0"] input')
.hasClass(
'is-valid',
'input field for first option is valid after user entered a label'
);
assert
.dom('[data-test-form-element="option"][data-test-option="1"] input')
.hasClass(
'is-invalid',
'input field for second option is still invalid'
);
await fillIn(
'[data-test-form-element="option"][data-test-option="1"] input',
'Cheesecake'
);
assert
.dom('[data-test-form-element="option"][data-test-option="0"] input')
.hasClass(
'is-valid',
'input field for first option is valid after user entered a value'
);
assert
.dom('[data-test-form-element="option"][data-test-option="1"] input')
.hasClass(
'is-invalid',
'input field for second option is still invalid if entering duplicated error'
);
assert
.dom(
'[data-test-form-element="option"][data-test-option="1"] .invalid-feedback'
)
.hasText(
t('create.options.error.duplicatedOption'),
'validation error message is updated to duplicated label'
);
await fillIn(
'[data-test-form-element="option"][data-test-option="1"] input',
'Muffin'
);
assert
.dom('[data-test-form-element="option"][data-test-option="0"] input')
.hasClass(
'is-valid',
'input field for first option is valid after user entered a value'
);
assert
.dom('[data-test-form-element="option"][data-test-option="1"] input')
.hasClass(
'is-valid',
'input field for second option is valid after filling in a unique value'
);
await click('button[type="submit"]');
assert.strictEqual(
currentURL(),
'/create/settings',
'user can move to next step after entering valid values for the options'
);
await click('button[type="submit"]');
assert.strictEqual(
currentRouteName(),
'poll.participation',
'user can finish the poll creation without changing any value on settings step'
);
});
});
});