Skip to content

Commit

Permalink
feat(datepicker): initial version of datepicker component
Browse files Browse the repository at this point in the history
  • Loading branch information
maxokorokov committed Aug 31, 2016
1 parent 9556e4b commit b5b1053
Show file tree
Hide file tree
Showing 31 changed files with 1,631 additions and 3 deletions.
2 changes: 2 additions & 0 deletions demo/src/app/app.routing.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {
NgbdButtons,
NgbdCarousel,
NgbdCollapse,
NgbdDatepicker,
NgbdDropdown,
NgbdModal,
NgbdPagination,
Expand All @@ -29,6 +30,7 @@ const routes: Routes = [
{path: 'components/buttons', component: NgbdButtons},
{path: 'components/carousel', component: NgbdCarousel},
{path: 'components/collapse', component: NgbdCollapse},
{path: 'components/datepicker', component: NgbdDatepicker},
{path: 'components/dropdown', component: NgbdDropdown},
{path: 'components/modal', component: NgbdModal},
{path: 'components/pagination', component: NgbdPagination},
Expand Down
17 changes: 17 additions & 0 deletions demo/src/app/components/datepicker/datepicker.component.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
import {Component} from '@angular/core';
import {DEMO_SNIPPETS} from './demos';

@Component({
selector: 'ngbd-datepicker',
template: `
<ngbd-content-wrapper component="Datepicker">
<ngbd-api-docs directive="NgbDatepicker"></ngbd-api-docs>
<ngbd-example-box demoTitle="Basic datepicker" [htmlSnippet]="snippets.basic.markup" [tsSnippet]="snippets.basic.code">
<ngbd-datepicker-basic></ngbd-datepicker-basic>
</ngbd-example-box>
</ngbd-content-wrapper>
`
})
export class NgbdDatepicker {
snippets = DEMO_SNIPPETS;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
<p>Simple datepicker</p>

<ngb-datepicker #dp [(ngModel)]="model" [minDate]="minDate" [maxDate]="maxDate"></ngb-datepicker>

<hr/>

<button class="btn btn-sm btn-outline-primary" (click)="selectToday()">Select Today</button>
<button class="btn btn-sm btn-outline-primary" (click)="dp.navigateTo()">To current month</button>
<button class="btn btn-sm btn-outline-primary" (click)="dp.navigateTo({year: 2013, month: 1})">To Feb 2013</button>

<hr/>

<pre>Model: {{ model | json }}</pre>
19 changes: 19 additions & 0 deletions demo/src/app/components/datepicker/demos/basic/datepicker-basic.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import {Component} from '@angular/core';

const now = new Date();

@Component({
selector: 'ngbd-datepicker-basic',
template: require('./datepicker-basic.html')
})
export class NgbdDatepickerBasic {

model;

minDate = {year: 2000, month: 0, date: 7};
maxDate = {year: 2020, month: 5, date: 15};

selectToday() {
this.model = {year: now.getFullYear(), month: now.getMonth(), date: now.getDate()};
}
}
9 changes: 9 additions & 0 deletions demo/src/app/components/datepicker/demos/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
import {NgbdDatepickerBasic} from './basic/datepicker-basic';

export const DEMO_DIRECTIVES = [NgbdDatepickerBasic];

export const DEMO_SNIPPETS = {
basic: {
code: require('!!prismjs?lang=typescript!./basic/datepicker-basic'),
markup: require('!!prismjs?lang=markup!./basic/datepicker-basic.html')}
};
14 changes: 14 additions & 0 deletions demo/src/app/components/datepicker/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
export * from './datepicker.component';

import {NgModule} from '@angular/core';
import {NgbdSharedModule} from '../../shared';
import {NgbdComponentsSharedModule} from '../shared';
import {NgbdDatepicker} from './datepicker.component';
import {DEMO_DIRECTIVES} from './demos';

@NgModule({
imports: [NgbdSharedModule, NgbdComponentsSharedModule],
exports: [NgbdDatepicker],
declarations: [NgbdDatepicker, ...DEMO_DIRECTIVES]
})
export class NgbdDatepickerModule {}
4 changes: 4 additions & 0 deletions demo/src/app/components/index.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
import {NgbdDatepickerModule} from './datepicker/index';
export * from './accordion';
export * from './alert';
export * from './buttons';
export * from './carousel';
export * from './collapse';
export * from './datepicker';
export * from './dropdown';
export * from './modal';
export * from './pagination';
Expand Down Expand Up @@ -42,6 +44,7 @@ import {NgbdTypeaheadModule} from './typeahead';
NgbdButtonsModule,
NgbdCarouselModule,
NgbdCollapseModule,
NgbdDatepickerModule,
NgbdDropdownModule,
NgbdModalModule,
NgbdPaginationModule,
Expand All @@ -59,6 +62,7 @@ import {NgbdTypeaheadModule} from './typeahead';
NgbdButtonsModule,
NgbdCarouselModule,
NgbdCollapseModule,
NgbdDatepickerModule,
NgbdDropdownModule,
NgbdModalModule,
NgbdPaginationModule,
Expand Down
1 change: 1 addition & 0 deletions demo/src/app/shared/side-nav/side-nav.component.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ export class SideNavComponent {
'Buttons',
'Carousel',
'Collapse',
'Datepicker',
'Dropdown',
'Modal',
'Pagination',
Expand Down
81 changes: 81 additions & 0 deletions src/datepicker/datepicker-day-view.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
import {TestBed, ComponentFixture} from '@angular/core/testing';
import {createGenericTestComponent} from '../util/tests';

import {Component} from '@angular/core';

import {NgbDatepickerModule} from './datepicker.module';
import {MonthViewModel, DayViewModel} from './datepicker-view-model';
import {NgbDatepickerDayView} from './datepicker-day-view';
import {NgbDate} from './ngb-date';

const createTestComponent = (html: string) =>
createGenericTestComponent(html, TestComponent) as ComponentFixture<TestComponent>;

function getElement(element: HTMLElement): HTMLElement {
return <HTMLElement>element.querySelector('[ngb-datepicker-day-view]');
}

describe('ngb-datepicker-day-view', () => {

beforeEach(() => {
TestBed.overrideModule(NgbDatepickerModule, {set: {exports: [NgbDatepickerDayView]}});
TestBed.configureTestingModule({declarations: [TestComponent], imports: [NgbDatepickerModule]});
});

it('should display date', () => {
const fixture =
createTestComponent('<div ngb-datepicker-day-view [day]="day" [month]="month" [selected]="selected"></div>');

const el = getElement(fixture.nativeElement);
expect(el.innerText).toBe('22');

fixture.componentInstance.day.date = new NgbDate(2016, 7, 25);
fixture.detectChanges();
expect(el.innerText).toBe('25');
});

it('should apply text-muted style for disabled days', () => {
const fixture =
createTestComponent('<div ngb-datepicker-day-view [day]="day" [month]="month" [selected]="selected"></div>');

const el = getElement(fixture.nativeElement);
expect(el).not.toHaveCssClass('text-muted');

fixture.componentInstance.day.disabled = true;
fixture.detectChanges();
expect(el).toHaveCssClass('text-muted');
});

it('should apply text-muted style for days of a different month', () => {
const fixture =
createTestComponent('<div ngb-datepicker-day-view [day]="day" [month]="month" [selected]="selected"></div>');

const el = getElement(fixture.nativeElement);
expect(el).not.toHaveCssClass('text-muted');

fixture.componentInstance.day.date = new NgbDate(2016, 8, 22);
fixture.detectChanges();
expect(el).toHaveCssClass('text-muted');
});

it('should apply selected style', () => {
const fixture =
createTestComponent('<div ngb-datepicker-day-view [day]="day" [month]="month" [selected]="selected"></div>');

const el = getElement(fixture.nativeElement);
expect(el).not.toHaveCssClass('bg-primary');

fixture.componentInstance.selected = true;
fixture.detectChanges();
expect(el).toHaveCssClass('bg-primary');
});
});

@Component({selector: 'test-cmp', template: ''})
class TestComponent {
day: DayViewModel = {date: new NgbDate(2016, 7, 22), disabled: false};

selected = false;

month: MonthViewModel = {year: 2016, number: 7, weekdays: [0], weeks: []};
}
20 changes: 20 additions & 0 deletions src/datepicker/datepicker-day-view.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import {Component, Input} from '@angular/core';
import {MonthViewModel, DayViewModel} from './datepicker-view-model';

@Component({
selector: '[ngb-datepicker-day-view]',
styles: [`
:host {
text-align: center;
padding: 0.185rem 0.25rem;
border-radius: 0.25rem;
}
`],
host: {'[class.bg-primary]': 'selected', '[class.text-muted]': 'day.date.month !== month.number || day.disabled'},
template: `{{ day.date.date }}`
})
export class NgbDatepickerDayView {
@Input() month: MonthViewModel;
@Input() day: DayViewModel;
@Input() selected: boolean;
}
151 changes: 151 additions & 0 deletions src/datepicker/datepicker-month-view.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,151 @@
import {TestBed, ComponentFixture} from '@angular/core/testing';
import {createGenericTestComponent} from '../util/tests';

import {Component} from '@angular/core';

import {NgbDatepickerModule} from './datepicker.module';
import {NgbDatepickerMonthView} from './datepicker-month-view';
import {MonthViewModel} from './datepicker-view-model';
import {NgbDate} from './ngb-date';
import {NgbDatepickerDayView} from './datepicker-day-view';

const createTestComponent = (html: string) =>
createGenericTestComponent(html, TestComponent) as ComponentFixture<TestComponent>;

function getWeekdays(element: HTMLElement): HTMLElement[] {
return <HTMLElement[]>Array.from(element.querySelectorAll('td.weekday'));
}

function getWeekNumbers(element: HTMLElement): HTMLElement[] {
return <HTMLElement[]>Array.from(element.querySelectorAll('td.weeknumber'));
}

function getDates(element: HTMLElement): HTMLElement[] {
return <HTMLElement[]>Array.from(element.querySelectorAll('td.day'));
}

function expectWeekdays(element: HTMLElement, weekdays: string[]) {
const result = getWeekdays(element).map(td => td.innerText.trim());
expect(result).toEqual(weekdays);
}

function expectWeekNumbers(element: HTMLElement, weeknumbers: string[]) {
const result = getWeekNumbers(element).map(td => td.innerText.trim());
expect(result).toEqual(weeknumbers);
}

function expectDates(element: HTMLElement, dates: string[]) {
const result = getDates(element).map(td => td.innerText.trim());
expect(result).toEqual(dates);
}

describe('ngb-datepicker-month-view', () => {

beforeEach(() => {
TestBed.overrideModule(NgbDatepickerModule, {set: {exports: [NgbDatepickerMonthView, NgbDatepickerDayView]}});
TestBed.configureTestingModule({declarations: [TestComponent], imports: [NgbDatepickerModule]});
});

it('should show/hide weekdays', () => {
const fixture =
createTestComponent('<tbody ngb-datepicker-month-view [month]="month" [showWeekdays]="showWeekdays"></tbody>');

expectWeekdays(fixture.nativeElement, ['Mo']);

fixture.componentInstance.showWeekdays = false;
fixture.detectChanges();
expectWeekdays(fixture.nativeElement, []);
});

it('should show/hide week numbers', () => {
const fixture = createTestComponent(
'<tbody ngb-datepicker-month-view [month]="month" [showWeekNumbers]="showWeekNumbers"></tbody>');

expectWeekNumbers(fixture.nativeElement, ['2']);

fixture.componentInstance.showWeekNumbers = false;
fixture.detectChanges();
expectWeekNumbers(fixture.nativeElement, []);
});

it('should use custom template to display dates', () => {
const fixture = createTestComponent(`
<template #tpl let-day="day">{{ day.date.date }}</template>
<tbody ngb-datepicker-month-view [month]="month" [dayTemplate]="tpl"></tbody>
`);
expectDates(fixture.nativeElement, ['22']);
});

it('should send date selection events', () => {
const fixture = createTestComponent(`
<template #tpl let-day="day">{{ day.date.date }}</template>
<tbody ngb-datepicker-month-view [month]="month" [dayTemplate]="tpl" (select)="onClick($event)"></tbody>
`);

spyOn(fixture.componentInstance, 'onClick');

const dates = getDates(fixture.nativeElement);
dates[0].click();

expect(fixture.componentInstance.onClick).toHaveBeenCalledWith(new NgbDate(2016, 7, 22));
});

it('should not send date selection events for disabled dates', () => {
const fixture = createTestComponent(`
<template #tpl let-day="day">{{ day.date.date }}</template>
<tbody ngb-datepicker-month-view [month]="month" [dayTemplate]="tpl" (select)="onClick($event)"></tbody>
`);

fixture.componentInstance.month.weeks[0].days[0].disabled = true;
fixture.detectChanges();

spyOn(fixture.componentInstance, 'onClick');

const dates = getDates(fixture.nativeElement);
dates[0].click();

expect(fixture.componentInstance.onClick).not.toHaveBeenCalledWith();
});

// CURSOR TESTS FAIL IN IE 9
// it('should set cursor to pointer', () => {
// const fixture = createTestComponent(`
// <template #tpl let-day="day">{{ day.date.date }}</template>
// <table><tbody ngb-datepicker-month-view [month]="month" [dayTemplate]="tpl"
// (change)="onClick($event)"></tbody></table>
// `);
//
// const dates = getDates(fixture.nativeElement);
// expect(window.getComputedStyle(dates[0]).getPropertyValue('cursor')).toBe('pointer');
// });
//
// it('should set not-allowed cursor for disabled dates', () => {
// const fixture = createTestComponent(`
// <template #tpl let-day="day">{{ day.date.date }}</template>
// <table><tbody ngb-datepicker-month-view [month]="month" [dayTemplate]="tpl"
// (change)="onClick($event)"></tbody></table>
// `);
//
// fixture.componentInstance.month.weeks[0].days[0].disabled = true;
// fixture.detectChanges();
//
// const dates = getDates(fixture.nativeElement);
// expect(window.getComputedStyle(dates[0]).getPropertyValue('cursor')).toBe('not-allowed');
// });

});

@Component({selector: 'test-cmp', template: ''})
class TestComponent {
month: MonthViewModel = {
year: 2016,
number: 7,
weekdays: [1],
weeks: [{number: 2, days: [{date: new NgbDate(2016, 7, 22), disabled: false}]}]
};

showWeekdays = true;
showWeekNumbers = true;

onClick = () => {};
}
Loading

0 comments on commit b5b1053

Please sign in to comment.