mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-08 20:22:53 +03:00
89f9516f04
refs https://github.com/TryGhost/Team/issues/1390 - added `membersTimeFilters` feature flag and labs toggle - added `<GhDatePicker>` component that lifts functionality from the `<GhDateTimePicker>` component - `<GhDateTimePicker>` has not yet been refactored to use the new component internally as there are some odd/complex interactions with error handling with it's existing use-cases and they are in critical publishing paths so the refactor doesn't belong as part of this change - added "Created at" filter type to members filters - uses new date picker component for input value - has "before", "on or before", "after", "on or after" operators - "on" and "not on" operators were skipped as they require two NQL statements to represent, breaking the current 1:1 statement:filter approach used in the NQL-based query param parsing
240 lines
11 KiB
JavaScript
240 lines
11 KiB
JavaScript
import Service from '@ember/service';
|
|
import hbs from 'htmlbars-inline-precompile';
|
|
import moment from 'moment';
|
|
import sinon from 'sinon';
|
|
import {blur, click, fillIn, find, render} from '@ember/test-helpers';
|
|
import {datepickerSelect} from 'ember-power-datepicker/test-support';
|
|
import {describe, it} from 'mocha';
|
|
import {expect} from 'chai';
|
|
import {setupRenderingTest} from 'ember-mocha';
|
|
|
|
class SettingsStub extends Service {
|
|
timezone = 'Etc/UTC';
|
|
|
|
get(key) {
|
|
if (key === 'timezone') {
|
|
return this.timezone;
|
|
}
|
|
}
|
|
}
|
|
|
|
describe('Integration: Component: gh-date-time-picker', function () {
|
|
setupRenderingTest();
|
|
let clock;
|
|
|
|
beforeEach(async function () {
|
|
this.owner.register('service:settings', SettingsStub);
|
|
});
|
|
|
|
afterEach(function () {
|
|
clock?.restore();
|
|
});
|
|
|
|
it('renders', async function () {
|
|
await render(hbs`<GhDateTimePicker />`);
|
|
expect(find('[data-test-component="gh-date-time-picker"]'), 'component wrapper').to.exist;
|
|
expect(find('[data-test-date-time-picker-datepicker]'), 'datepicker trigger').to.exist;
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'datepicker input').to.exist;
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.exist;
|
|
expect(find('[data-test-date-time-picker-timezone]')).to.contain.text('UTC');
|
|
});
|
|
|
|
it('defaults to now when @date is empty', async function () {
|
|
clock = sinon.useFakeTimers({
|
|
now: moment('2022-02-22 22:22:22.000Z').toDate()
|
|
});
|
|
|
|
await render(hbs`<GhDateTimePicker />`);
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('22:22');
|
|
});
|
|
|
|
it('shows passed in @date value', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} />`);
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('22:22');
|
|
});
|
|
|
|
it('uses separate @time value', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} />`);
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('12:00');
|
|
});
|
|
|
|
it('can update date via date input', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '22:22');
|
|
|
|
this.set('updateDate', (newDate) => {
|
|
expect(moment(newDate).toISOString()).to.equal('2022-02-28T00:00:00.000Z');
|
|
this.set('date', newDate);
|
|
});
|
|
this.set('updateTime', (newTime) => {
|
|
expect(newTime).to.equal('22:22');
|
|
this.set('time', newTime);
|
|
});
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
await fillIn('[data-test-date-time-picker-date-input]', '2022-02-28');
|
|
await blur('[data-test-date-time-picker-date-input]');
|
|
});
|
|
|
|
it('can update time via time input', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '22:22');
|
|
|
|
this.set('updateDate', (newDate) => {
|
|
expect(moment(newDate).toISOString()).to.equal('2022-02-28T00:00:00.000Z');
|
|
this.set('date', newDate);
|
|
});
|
|
this.set('updateTime', (newTime) => {
|
|
expect(newTime).to.equal('18:00');
|
|
this.set('time', newTime);
|
|
});
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
await fillIn('[data-test-date-time-picker-time-input]', '18:00');
|
|
await blur('[data-test-date-time-picker-time-input]');
|
|
});
|
|
|
|
it('can update date via datepicker', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
|
|
this.set('updateDate', (newDate) => {
|
|
expect(moment(newDate).toISOString()).to.equal('2022-02-27T00:00:00.000Z');
|
|
this.set('date', newDate);
|
|
});
|
|
this.set('updateTime', (newTime) => {
|
|
expect(newTime).to.equal('12:00');
|
|
this.set('time', newTime);
|
|
});
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
await datepickerSelect('[data-test-date-time-picker-datepicker]', moment('2022-02-27T13:00:00.000Z').toDate());
|
|
});
|
|
|
|
it('updates when @date is changed externally', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} />`);
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('12:00');
|
|
|
|
this.set('date', moment('2022-02-28 10:00:00.000Z')).toDate();
|
|
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-28');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('12:00');
|
|
});
|
|
|
|
it('updates when @time is changed externally', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} />`);
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('12:00');
|
|
|
|
this.set('time', '08:00');
|
|
|
|
expect(find('[data-test-date-time-picker-date-input]'), 'date input').to.have.value('2022-02-22');
|
|
expect(find('[data-test-date-time-picker-time-input]'), 'time input').to.have.value('08:00');
|
|
});
|
|
|
|
it('handles invalid date input', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
|
|
const dateSpy = sinon.spy();
|
|
this.set('updateDate', dateSpy);
|
|
|
|
const timeSpy = sinon.spy();
|
|
this.set('updateTime', timeSpy);
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
await fillIn('[data-test-date-time-picker-date-input]', '2022-02-31');
|
|
await blur('[data-test-date-time-picker-date-input]');
|
|
|
|
expect(find('[data-test-date-time-picker-error]')).to.have.text('Invalid date');
|
|
|
|
await fillIn('[data-test-date-time-picker-date-input]', 'narp');
|
|
await blur('[data-test-date-time-picker-date-input]');
|
|
|
|
expect(find('[data-test-date-time-picker-error]')).to.contain.text('Invalid date format');
|
|
|
|
expect(dateSpy.callCount, '@setDate call count').to.equal(0);
|
|
expect(timeSpy.callCount, '@setTime call count').to.equal(0);
|
|
});
|
|
|
|
// TODO: move time format handling into component?
|
|
// it('handles invalid time input', async function () {
|
|
// this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
// this.set('time', '12:00');
|
|
|
|
// const dateSpy = sinon.spy();
|
|
// this.set('updateDate', dateSpy);
|
|
|
|
// const timeSpy = sinon.spy();
|
|
// this.set('updateTime', timeSpy);
|
|
|
|
// await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
// await fillIn('[data-test-date-time-picker-time-input]', '24:59');
|
|
// await blur('[data-test-date-time-picker-time-input]');
|
|
|
|
// expect(dateSpy.callCount, '@setDate call count').to.equal(0);
|
|
// expect(timeSpy.callCount, '@setTime call count').to.equal(0);
|
|
|
|
// expect(find('[data-test-date-time-picker-error]')).to.have.text('Invalid time');
|
|
|
|
// await fillIn('[data-test-date-time-picker-time-input]', 'narp');
|
|
// await blur('[data-test-date-time-picker-time-input]');
|
|
|
|
// expect(find('[data-test-date-time-picker-error]')).to.contain.text('Invalid time format');
|
|
// });
|
|
|
|
describe('min/max', function () {
|
|
it('disables datepicker dates outside of range', async function () {
|
|
this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
this.set('time', '12:00');
|
|
this.set('minDate', moment('2022-02-11 12:00:00.000Z').toDate());
|
|
this.set('maxDate', moment('2022-02-24 12:00:00.000Z').toDate());
|
|
|
|
await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @minDate={{this.minDate}} @maxDate={{this.maxDate}} />`);
|
|
await click('[data-test-date-time-picker-datepicker]');
|
|
|
|
expect(find('[data-date="2022-02-10"]')).to.have.attribute('disabled');
|
|
expect(find('[data-date="2022-02-25"]')).to.have.attribute('disabled');
|
|
});
|
|
|
|
// TODO: move date validation into component?
|
|
// it('errors when date input is earlier than min', async function () {
|
|
// this.set('date', moment('2022-02-22 22:22:22.000Z')).toDate();
|
|
// this.set('time', '12:00');
|
|
// this.set('minDate', moment('2022-02-11 12:00:00.000Z').toDate());
|
|
|
|
// const dateSpy = sinon.spy();
|
|
// this.set('updateDate', dateSpy);
|
|
|
|
// const timeSpy = sinon.spy();
|
|
// this.set('updateTime', timeSpy);
|
|
|
|
// await render(hbs`<GhDateTimePicker @date={{this.date}} @time={{this.time}} @minDate={{this.minDate}} @setDate={{this.updateDate}} @setTime={{this.updateTime}} />`);
|
|
// await fillIn('[data-test-date-time-picker-date-input]', '2022-02-10');
|
|
// await blur('[data-test-date-time-picker-date-input]');
|
|
|
|
// expect(dateSpy.callCount, '@setDate call count').to.equal(0);
|
|
// expect(timeSpy.callCount, '@setTime call count').to.equal(0);
|
|
// });
|
|
|
|
// it('errors when date input is later than max');
|
|
// it('errors when time input is earlier than min');
|
|
// it('errors when time input is later than max');
|
|
});
|
|
});
|