-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(datepicker): initial version of datepicker component
- Loading branch information
1 parent
9556e4b
commit b5b1053
Showing
31 changed files
with
1,631 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
17 changes: 17 additions & 0 deletions
17
demo/src/app/components/datepicker/datepicker.component.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
} |
13 changes: 13 additions & 0 deletions
13
demo/src/app/components/datepicker/demos/basic/datepicker-basic.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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
19
demo/src/app/components/datepicker/demos/basic/datepicker-basic.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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()}; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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')} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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: []}; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 = () => {}; | ||
} |
Oops, something went wrong.