@frxjs/ngx-timeline 1.0.4 → 1.0.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.editorconfig +16 -0
- package/.github/workflows/CI-main.yml +48 -0
- package/README.md +136 -136
- package/angular.json +178 -0
- package/docs/3rdpartylicenses.txt +268 -0
- package/docs/assets/inter.png +0 -0
- package/docs/favicon.ico +0 -0
- package/docs/index.html +13 -0
- package/docs/main.6005a10000c87d7095d3.js +1 -0
- package/docs/polyfills.f8ddde825b13760c1743.js +1 -0
- package/docs/runtime.0e49e2b53282f40c8925.js +1 -0
- package/docs/styles.09e2c710755c8867a460.css +0 -0
- package/package.json +51 -13
- package/projects/demo-app/.browserslistrc +17 -0
- package/projects/demo-app/e2e/protractor.conf.js +37 -0
- package/projects/demo-app/e2e/src/app.e2e-spec.ts +23 -0
- package/projects/demo-app/e2e/src/app.po.ts +11 -0
- package/projects/demo-app/e2e/tsconfig.json +13 -0
- package/projects/demo-app/karma.conf.js +44 -0
- package/projects/demo-app/src/app/app.component.html +90 -0
- package/projects/demo-app/src/app/app.component.scss +417 -0
- package/projects/demo-app/src/app/app.component.spec.ts +24 -0
- package/projects/demo-app/src/app/app.component.ts +147 -0
- package/projects/demo-app/src/app/app.module.ts +20 -0
- package/projects/demo-app/src/assets/.gitkeep +0 -0
- package/projects/demo-app/src/assets/inter.png +0 -0
- package/projects/demo-app/src/environments/environment.prod.ts +3 -0
- package/projects/demo-app/src/environments/environment.ts +16 -0
- package/projects/demo-app/src/favicon.ico +0 -0
- package/projects/demo-app/src/index.html +13 -0
- package/projects/demo-app/src/main.ts +12 -0
- package/projects/demo-app/src/polyfills.ts +63 -0
- package/projects/demo-app/src/styles.scss +1 -0
- package/projects/demo-app/src/test.ts +25 -0
- package/projects/demo-app/tsconfig.app.json +26 -0
- package/projects/demo-app/tsconfig.spec.json +18 -0
- package/projects/demo-app/tslint.json +17 -0
- package/projects/ngx-timeline/README.md +136 -0
- package/projects/ngx-timeline/karma.conf.js +44 -0
- package/projects/ngx-timeline/ng-package.json +7 -0
- package/projects/ngx-timeline/package.json +20 -0
- package/{lib/components/index.d.ts → projects/ngx-timeline/src/lib/components/index.ts} +2 -2
- package/projects/ngx-timeline/src/lib/components/ngx-timeline-event/ngx-timeline-event.component.html +37 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline-event/ngx-timeline-event.component.scss +197 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline-event/ngx-timeline-event.component.spec.ts +89 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline-event/ngx-timeline-event.component.ts +66 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline.component.html +98 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline.component.spec.ts +160 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline.component.ts +187 -0
- package/projects/ngx-timeline/src/lib/components/ngx-timeline.scss +226 -0
- package/projects/ngx-timeline/src/lib/models/NgxConfigObj.ts +110 -0
- package/projects/ngx-timeline/src/lib/models/NgxTimelineEvent.ts +32 -0
- package/{lib/models/index.d.ts → projects/ngx-timeline/src/lib/models/index.ts} +2 -2
- package/projects/ngx-timeline/src/lib/ngx-timeline.module.ts +21 -0
- package/{lib/pipes/index.d.ts → projects/ngx-timeline/src/lib/pipes/index.ts} +1 -1
- package/projects/ngx-timeline/src/lib/pipes/ngx-date-pipe.ts +29 -0
- package/{public-api.d.ts → projects/ngx-timeline/src/public-api.ts} +7 -4
- package/projects/ngx-timeline/src/test.ts +26 -0
- package/projects/ngx-timeline/tsconfig.lib.json +25 -0
- package/projects/ngx-timeline/tsconfig.lib.prod.json +10 -0
- package/projects/ngx-timeline/tsconfig.spec.json +17 -0
- package/projects/ngx-timeline/tslint.json +17 -0
- package/tsconfig.json +35 -0
- package/tslint.json +140 -0
- package/bundles/frxjs-ngx-timeline.umd.js +0 -696
- package/bundles/frxjs-ngx-timeline.umd.js.map +0 -1
- package/bundles/frxjs-ngx-timeline.umd.min.js +0 -2
- package/bundles/frxjs-ngx-timeline.umd.min.js.map +0 -1
- package/esm2015/frxjs-ngx-timeline.js +0 -8
- package/esm2015/lib/components/index.js +0 -3
- package/esm2015/lib/components/ngx-timeline-event/ngx-timeline-event.component.js +0 -55
- package/esm2015/lib/components/ngx-timeline.component.js +0 -147
- package/esm2015/lib/models/NgxConfigObj.js +0 -93
- package/esm2015/lib/models/NgxTimelineEvent.js +0 -6
- package/esm2015/lib/models/index.js +0 -3
- package/esm2015/lib/ngx-timeline.module.js +0 -23
- package/esm2015/lib/pipes/index.js +0 -2
- package/esm2015/lib/pipes/ngx-date-pipe.js +0 -28
- package/esm2015/public-api.js +0 -8
- package/fesm2015/frxjs-ngx-timeline.js +0 -350
- package/fesm2015/frxjs-ngx-timeline.js.map +0 -1
- package/frxjs-ngx-timeline.d.ts +0 -7
- package/frxjs-ngx-timeline.metadata.json +0 -1
- package/lib/components/ngx-timeline-event/ngx-timeline-event.component.d.ts +0 -35
- package/lib/components/ngx-timeline.component.d.ts +0 -76
- package/lib/models/NgxConfigObj.d.ts +0 -59
- package/lib/models/NgxTimelineEvent.d.ts +0 -29
- package/lib/ngx-timeline.module.d.ts +0 -2
- package/lib/pipes/ngx-date-pipe.d.ts +0 -7
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
|
2
|
+
|
|
3
|
+
import { NgxTimelineEventComponent } from './ngx-timeline-event.component';
|
|
4
|
+
import { NgxTimelineItemPosition } from '../../models';
|
|
5
|
+
import localeIt from '@angular/common/locales/it';
|
|
6
|
+
import { registerLocaleData } from '@angular/common';
|
|
7
|
+
|
|
8
|
+
registerLocaleData(localeIt);
|
|
9
|
+
|
|
10
|
+
const year = 2021;
|
|
11
|
+
const monthIndex = 7;
|
|
12
|
+
const day = 19;
|
|
13
|
+
const event = {
|
|
14
|
+
type: "event",
|
|
15
|
+
position: NgxTimelineItemPosition.ON_LEFT,
|
|
16
|
+
periodInfo: {
|
|
17
|
+
periodKey: '08/2021',
|
|
18
|
+
year: year,
|
|
19
|
+
month: monthIndex + 1,
|
|
20
|
+
day: day,
|
|
21
|
+
firstDate: new Date(year, monthIndex, day)
|
|
22
|
+
},
|
|
23
|
+
eventInfo: {
|
|
24
|
+
timestamp: new Date(year, monthIndex, day),
|
|
25
|
+
title: 'event title',
|
|
26
|
+
description: 'event description',
|
|
27
|
+
id: '1'
|
|
28
|
+
}
|
|
29
|
+
};
|
|
30
|
+
describe('NgxTimelineEventComponent', () => {
|
|
31
|
+
let component: NgxTimelineEventComponent;
|
|
32
|
+
let fixture: ComponentFixture<NgxTimelineEventComponent>;
|
|
33
|
+
|
|
34
|
+
beforeEach(async () => {
|
|
35
|
+
await TestBed.configureTestingModule({
|
|
36
|
+
declarations: [ NgxTimelineEventComponent ]
|
|
37
|
+
})
|
|
38
|
+
.compileComponents();
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
beforeEach(() => {
|
|
42
|
+
fixture = TestBed.createComponent(NgxTimelineEventComponent);
|
|
43
|
+
component = fixture.componentInstance;
|
|
44
|
+
component.event = event;
|
|
45
|
+
fixture.detectChanges();
|
|
46
|
+
});
|
|
47
|
+
|
|
48
|
+
it('should create', () => {
|
|
49
|
+
expect(component).toBeTruthy();
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
describe('should getDateObj', () => {
|
|
53
|
+
it('without event', () => {
|
|
54
|
+
component.event = null;
|
|
55
|
+
const res = component.getDateObj();
|
|
56
|
+
expect(res).toEqual({day: undefined, month: undefined, year: undefined});
|
|
57
|
+
});
|
|
58
|
+
it('without event info', () => {
|
|
59
|
+
component.event = {};
|
|
60
|
+
const res = component.getDateObj();
|
|
61
|
+
expect(res).toEqual({day: undefined, month: undefined, year: undefined});
|
|
62
|
+
});
|
|
63
|
+
it('without event info timestamp', () => {
|
|
64
|
+
component.event = {eventInfo: null};
|
|
65
|
+
const res = component.getDateObj();
|
|
66
|
+
expect(res).toEqual({day: undefined, month: undefined, year: undefined});
|
|
67
|
+
});
|
|
68
|
+
it('should getDateObj without langCode', () => {
|
|
69
|
+
const res = component.getDateObj();
|
|
70
|
+
expect(res).toEqual({day: '19', month: 'Aug', year: 2021});
|
|
71
|
+
});
|
|
72
|
+
it('should getDateObj with supported langCode', () => {
|
|
73
|
+
component.langCode = 'en';
|
|
74
|
+
const res = component.getDateObj();
|
|
75
|
+
expect(res).toEqual({day: '19', month: 'Aug', year: 2021});
|
|
76
|
+
});
|
|
77
|
+
it('should getDateObj with another supported langCode', () => {
|
|
78
|
+
component.langCode = 'it';
|
|
79
|
+
const res = component.getDateObj();
|
|
80
|
+
expect(res).toEqual({day: '19', month: 'ago', year: 2021});
|
|
81
|
+
});
|
|
82
|
+
it('should getDateObj with unsupported langCode', () => {
|
|
83
|
+
component.langCode = 'xx';
|
|
84
|
+
const res = component.getDateObj();
|
|
85
|
+
expect(res).toEqual({day: '19', month: 'Aug', year: 2021});
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
});
|
package/projects/ngx-timeline/src/lib/components/ngx-timeline-event/ngx-timeline-event.component.ts
ADDED
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { Component, Input, Output, TemplateRef } from '@angular/core';
|
|
2
|
+
import { NgxTimelineItem, NgxTimelineItemPosition } from '../../models/NgxTimelineEvent';
|
|
3
|
+
import { DatePipe } from '@angular/common';
|
|
4
|
+
import { BehaviorSubject } from 'rxjs';
|
|
5
|
+
import { supportedLanguageCodes } from '../../models';
|
|
6
|
+
|
|
7
|
+
@Component({
|
|
8
|
+
selector: 'ngx-timeline-event',
|
|
9
|
+
templateUrl: './ngx-timeline-event.component.html',
|
|
10
|
+
styleUrls: ['./ngx-timeline-event.component.scss']
|
|
11
|
+
})
|
|
12
|
+
export class NgxTimelineEventComponent {
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Event to be displayed
|
|
16
|
+
*/
|
|
17
|
+
@Input() event: NgxTimelineItem;
|
|
18
|
+
/**
|
|
19
|
+
* Event position respect to the vertical line (LEFT or RIGHT)
|
|
20
|
+
*/
|
|
21
|
+
@Input() colSidePosition: NgxTimelineItemPosition;
|
|
22
|
+
/**
|
|
23
|
+
* Language code used to format the dates
|
|
24
|
+
*/
|
|
25
|
+
@Input() langCode?: string;
|
|
26
|
+
/**
|
|
27
|
+
* Inner custom template used to display the event detail
|
|
28
|
+
*/
|
|
29
|
+
@Input() innerEventCustomTemplate?: TemplateRef<any>;
|
|
30
|
+
/**
|
|
31
|
+
* Boolean used to enable or disable the animations
|
|
32
|
+
*/
|
|
33
|
+
@Input() enableAnimation = true;
|
|
34
|
+
/**
|
|
35
|
+
* Output click event emitter
|
|
36
|
+
*/
|
|
37
|
+
@Output() clickEmitter: BehaviorSubject<NgxTimelineItem> = new BehaviorSubject(null);
|
|
38
|
+
|
|
39
|
+
ngxTimelineItemPosition = NgxTimelineItemPosition;
|
|
40
|
+
|
|
41
|
+
private readonly monthAbbr = 'MMM';
|
|
42
|
+
private readonly dayFormat = 'dd';
|
|
43
|
+
|
|
44
|
+
constructor() { }
|
|
45
|
+
|
|
46
|
+
getDateObj(): any {
|
|
47
|
+
let day;
|
|
48
|
+
let month;
|
|
49
|
+
let year;
|
|
50
|
+
const dateTimestamp = this.event?.eventInfo?.timestamp;
|
|
51
|
+
if (dateTimestamp) {
|
|
52
|
+
const timestamp = new Date(dateTimestamp);
|
|
53
|
+
const langCode = this.getLangCode();
|
|
54
|
+
month = new DatePipe(langCode).transform(timestamp, this.monthAbbr);
|
|
55
|
+
day = new DatePipe(langCode).transform(timestamp, this.dayFormat);
|
|
56
|
+
year = timestamp.getFullYear();
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return {day, month, year};
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
protected getLangCode(): string {
|
|
63
|
+
return this.langCode && supportedLanguageCodes.includes(this.langCode) ? this.langCode : supportedLanguageCodes[0];
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
<div class="main-container">
|
|
2
|
+
<div class="items-container">
|
|
3
|
+
<div *ngFor="let item of items; let index = index;" class="row">
|
|
4
|
+
<!-- DESKTOP -->
|
|
5
|
+
<div class="col col-left desktop" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
6
|
+
<div class="event-outer-container" *ngIf="item.eventInfo && item.position === ngxTimelineItemPosition.ON_LEFT">
|
|
7
|
+
<ng-container *ngTemplateOutlet="eventCustomTemplate || eventTemplate; context: {event: item, colSidePosition: ngxTimelineItemPosition.ON_LEFT}"></ng-container>
|
|
8
|
+
</div>
|
|
9
|
+
<div class="hour left" *ngIf="item.eventInfo && item.position === ngxTimelineItemPosition.ON_RIGHT">
|
|
10
|
+
<ng-container *ngTemplateOutlet="dateInstantCustomTemplate || dateInstantTemplate; context: {item: item.eventInfo}" ></ng-container>
|
|
11
|
+
</div>
|
|
12
|
+
</div>
|
|
13
|
+
|
|
14
|
+
<div class="col col-center desktop" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
15
|
+
<div class="center-inner" *ngIf="item.periodInfo">
|
|
16
|
+
<ng-container *ngTemplateOutlet="periodCustomTemplate || periodTemplate; context: {period: item.periodInfo, index: index}"></ng-container>
|
|
17
|
+
</div>
|
|
18
|
+
<div class="center-inner no-period-key" *ngIf="!item.periodInfo">
|
|
19
|
+
<ng-container *ngTemplateOutlet="centerLinesIconTemplate; context: {index: index}"></ng-container>
|
|
20
|
+
</div>
|
|
21
|
+
</div>
|
|
22
|
+
|
|
23
|
+
<div class="col col-right desktop" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
24
|
+
<div class="event-outer-container" *ngIf="item.eventInfo && item.position === ngxTimelineItemPosition.ON_RIGHT">
|
|
25
|
+
<ng-container *ngTemplateOutlet="eventCustomTemplate || eventTemplate; context: {event: item, colSidePosition: ngxTimelineItemPosition.ON_RIGHT}"></ng-container>
|
|
26
|
+
</div>
|
|
27
|
+
<div class="hour right" *ngIf="item.eventInfo && item.position === ngxTimelineItemPosition.ON_LEFT">
|
|
28
|
+
<ng-container *ngTemplateOutlet="dateInstantCustomTemplate || dateInstantTemplate; context: {item: item.eventInfo}" ></ng-container>
|
|
29
|
+
</div>
|
|
30
|
+
</div>
|
|
31
|
+
|
|
32
|
+
<!-- MOBILE -->
|
|
33
|
+
<div class="col col-left mobile" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
34
|
+
<div class="hour left" *ngIf="item.eventInfo">
|
|
35
|
+
<div class="hour-inner-container">
|
|
36
|
+
<ng-container *ngTemplateOutlet="dateInstantCustomTemplate || dateInstantTemplate; context: {item: item.eventInfo}" ></ng-container>
|
|
37
|
+
</div>
|
|
38
|
+
</div>
|
|
39
|
+
</div>
|
|
40
|
+
|
|
41
|
+
<div class="col col-center mobile" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
42
|
+
<div class="center-inner" *ngIf="item.periodInfo;">
|
|
43
|
+
<ng-container *ngTemplateOutlet="periodCustomTemplate || periodTemplate; context: {period: item.periodInfo, index: index}"></ng-container>
|
|
44
|
+
</div>
|
|
45
|
+
<div class="center-inner no-period-key" *ngIf="!item.periodInfo;">
|
|
46
|
+
<ng-container *ngTemplateOutlet="centerLinesIconTemplate;context: {index: index}"></ng-container>
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
|
|
50
|
+
<div class="col col-right mobile" [ngClass]="item.periodInfo ? 'col-period' : 'col-event'">
|
|
51
|
+
<div class="event-outer-container" *ngIf="item.eventInfo">
|
|
52
|
+
<ng-container *ngTemplateOutlet="eventCustomTemplate || eventTemplate; context: {event: item, colSidePosition: ngxTimelineItemPosition.ON_RIGHT}"></ng-container>
|
|
53
|
+
</div>
|
|
54
|
+
</div>
|
|
55
|
+
</div>
|
|
56
|
+
</div>
|
|
57
|
+
</div>
|
|
58
|
+
|
|
59
|
+
|
|
60
|
+
<ng-template #centerLinesIconTemplate let-index=index>
|
|
61
|
+
<div class="line"></div>
|
|
62
|
+
<ng-container *ngTemplateOutlet="centerIconCustomTemplate || centerIconTemplate"></ng-container>
|
|
63
|
+
<div [ngClass]="{'transparent last-line': index === items.length - 1}" class="line"></div>
|
|
64
|
+
</ng-template>
|
|
65
|
+
|
|
66
|
+
<ng-template #centerIconTemplate>
|
|
67
|
+
<div class="center-icon-container">
|
|
68
|
+
<div class="icon"></div>
|
|
69
|
+
</div>
|
|
70
|
+
</ng-template>
|
|
71
|
+
|
|
72
|
+
<ng-template #dateInstantTemplate let-item=item>
|
|
73
|
+
<span>
|
|
74
|
+
{{item?.timestamp | ngxdate : ngxDateFormat.HOURS_MINUTES : langCode}}
|
|
75
|
+
</span>
|
|
76
|
+
</ng-template>
|
|
77
|
+
|
|
78
|
+
<ng-template #periodTemplate let-period=period let-index=index>
|
|
79
|
+
<div [ngClass]="{'transparent first-line': !index}" class="line"></div>
|
|
80
|
+
<div class="period-container">
|
|
81
|
+
<div class="period-inner-container">
|
|
82
|
+
<span>{{period?.firstDate | ngxdate : getPeriodKeyDateFormat() : langCode}}</span>
|
|
83
|
+
</div>
|
|
84
|
+
</div>
|
|
85
|
+
<div class="line"></div>
|
|
86
|
+
</ng-template>
|
|
87
|
+
|
|
88
|
+
<ng-template #eventTemplate let-event=event let-colSidePosition=colSidePosition>
|
|
89
|
+
<ngx-timeline-event
|
|
90
|
+
[event]="event"
|
|
91
|
+
[langCode]="langCode"
|
|
92
|
+
[enableAnimation]="enableAnimation"
|
|
93
|
+
[innerEventCustomTemplate]="innerEventCustomTemplate"
|
|
94
|
+
[colSidePosition]="colSidePosition"
|
|
95
|
+
(clickEmitter)="clickEmitter.next($event)">
|
|
96
|
+
</ngx-timeline-event>
|
|
97
|
+
</ng-template>
|
|
98
|
+
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
import { ComponentFixture, TestBed } from '@angular/core/testing';
|
|
2
|
+
|
|
3
|
+
import { NgxTimelineComponent } from './ngx-timeline.component';
|
|
4
|
+
import { NgxDateFormat, NgxTimelineEventGroup } from '../models';
|
|
5
|
+
|
|
6
|
+
describe('NgxTimelineComponent', () => {
|
|
7
|
+
let component: NgxTimelineComponent;
|
|
8
|
+
let fixture: ComponentFixture<NgxTimelineComponent>;
|
|
9
|
+
|
|
10
|
+
beforeEach(async () => {
|
|
11
|
+
await TestBed.configureTestingModule({
|
|
12
|
+
declarations: [ NgxTimelineComponent ]
|
|
13
|
+
})
|
|
14
|
+
.compileComponents();
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
fixture = TestBed.createComponent(NgxTimelineComponent);
|
|
19
|
+
component = fixture.componentInstance;
|
|
20
|
+
component.events = null;
|
|
21
|
+
fixture.detectChanges();
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
it('should create', () => {
|
|
25
|
+
expect(component).toBeTruthy();
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
it('should call the groupEvents when ngOnChanges', () => {
|
|
29
|
+
const spy = spyOn<any>(component, 'groupEvents');
|
|
30
|
+
component.ngOnChanges({});
|
|
31
|
+
expect(spy).toHaveBeenCalled();
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('should clear', () => {
|
|
35
|
+
component['clear']();
|
|
36
|
+
expect(component.groups).toEqual({});
|
|
37
|
+
expect(component.periods).toEqual([]);
|
|
38
|
+
expect(component.items).toEqual([]);
|
|
39
|
+
});
|
|
40
|
+
|
|
41
|
+
describe('should getPeriodKeyDateFormat', () => {
|
|
42
|
+
it('when default groupEvent is provided', () => {
|
|
43
|
+
const res = component.getPeriodKeyDateFormat();
|
|
44
|
+
expect(res).toEqual(NgxDateFormat.MONTH_YEAR);
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
[{groupEvent: NgxTimelineEventGroup.YEAR, formatDate: NgxDateFormat.YEAR},
|
|
48
|
+
{groupEvent: NgxTimelineEventGroup.DAY_MONTH_YEAR, formatDate: NgxDateFormat.DAY_MONTH_YEAR}
|
|
49
|
+
].forEach(elem => {
|
|
50
|
+
it(`when groupEvent ${elem.groupEvent} is provided`, () => {
|
|
51
|
+
component.groupEvent = elem.groupEvent;
|
|
52
|
+
expect(component.getPeriodKeyDateFormat()).toEqual(elem.formatDate);
|
|
53
|
+
});
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
describe('should groupEvents', ()=> {
|
|
60
|
+
it('when no events', () => {
|
|
61
|
+
const spies = [];
|
|
62
|
+
spies.push(spyOn<any>(component, 'clear'));
|
|
63
|
+
spies.push(spyOn<any>(component, 'sortEvents'));
|
|
64
|
+
spies.push(spyOn<any>(component, 'setGroups'));
|
|
65
|
+
spies.push(spyOn<any>(component, 'setPeriods'));
|
|
66
|
+
spies.push(spyOn<any>(component, 'setItems'));
|
|
67
|
+
component['groupEvents'](null);
|
|
68
|
+
spies.forEach(spy => expect(spy).not.toHaveBeenCalled());
|
|
69
|
+
});
|
|
70
|
+
it('when events', () => {
|
|
71
|
+
const spies = [];
|
|
72
|
+
spies.push(spyOn<any>(component, 'clear'));
|
|
73
|
+
spies.push(spyOn<any>(component, 'sortEvents'));
|
|
74
|
+
spies.push(spyOn<any>(component, 'setGroups'));
|
|
75
|
+
spies.push(spyOn<any>(component, 'setPeriods'));
|
|
76
|
+
spies.push(spyOn<any>(component, 'setItems'));
|
|
77
|
+
component['groupEvents']([]);
|
|
78
|
+
spies.forEach(spy => expect(spy).toHaveBeenCalled());
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
describe('should sortEvents', ()=> {
|
|
83
|
+
it('when events', () => {
|
|
84
|
+
const event = {timestamp: new Date(2021, 11, 10)};
|
|
85
|
+
const event2 = {timestamp: new Date(2021, 8, 10)};
|
|
86
|
+
const events = [event, event2];
|
|
87
|
+
component['sortEvents'](events);
|
|
88
|
+
expect(events[0]).toEqual(event2);
|
|
89
|
+
expect(events[1]).toEqual(event);
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
describe('should setGroups', ()=> {
|
|
94
|
+
it('when events', () => {
|
|
95
|
+
const event = {timestamp: new Date(2021, 11, 10)};
|
|
96
|
+
const event2 = {timestamp: new Date(2021, 8, 10)};
|
|
97
|
+
const event3 = {timestamp: new Date(2021, 8, 11)};
|
|
98
|
+
const events = [event, event2, event3];
|
|
99
|
+
component['setGroups'](events);
|
|
100
|
+
expect(Object.keys(component.groups).length).toEqual(2);
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
describe('should setItems', ()=> {
|
|
105
|
+
it('when events', () => {
|
|
106
|
+
const period = {periodInfo: {periodKey: '2021/7'}};
|
|
107
|
+
const period2 = {periodInfo: {periodKey: '2021/8'}};
|
|
108
|
+
component.periods = [period, period2];
|
|
109
|
+
const event = {timestamp: new Date(2021, 7, 10)};
|
|
110
|
+
const event2 = {timestamp: new Date(2021, 8, 10)};
|
|
111
|
+
const event3 = {timestamp: new Date(2021, 8, 11)};
|
|
112
|
+
component.groups['2021/7'] = [event];
|
|
113
|
+
component.groups['2021/8'] = [event2, event3];
|
|
114
|
+
component['setItems']();
|
|
115
|
+
expect(component.items.length).toEqual(5);
|
|
116
|
+
});
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
describe('should setPeriods', ()=> {
|
|
120
|
+
it('when events', () => {
|
|
121
|
+
const date = new Date(2021, 7, 10);
|
|
122
|
+
const date2 = new Date(2021, 8, 10);
|
|
123
|
+
const date3= new Date(2021, 8, 11);
|
|
124
|
+
const event = {timestamp: date}
|
|
125
|
+
const event2 = {timestamp: date2};
|
|
126
|
+
const event3 = {timestamp: date3};
|
|
127
|
+
component.groups['2021/7'] = [event];
|
|
128
|
+
component.groups['2021/8'] = [event2, event3];
|
|
129
|
+
component['setPeriods']();
|
|
130
|
+
expect(component.periods.length).toEqual(2);
|
|
131
|
+
expect(component.periods[0]).toEqual({periodInfo: {year: 2021, month: 7, day: NaN, periodKey: '2021/7', firstDate: date }})
|
|
132
|
+
expect(component.periods[1]).toEqual({periodInfo: {year: 2021, month: 8, day: NaN, periodKey: '2021/8', firstDate: date2 }})
|
|
133
|
+
});
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
describe('should getPeriodKeyFromEvent', ()=> {
|
|
137
|
+
it('when groupEvent by year', () => {
|
|
138
|
+
component.groupEvent = NgxTimelineEventGroup.YEAR;
|
|
139
|
+
const event = {timestamp: new Date(2021, 7, 10)};
|
|
140
|
+
const event2 = {timestamp: new Date(2021, 8, 9)};
|
|
141
|
+
expect(component['getPeriodKeyFromEvent'](event)).toBe('2021');
|
|
142
|
+
expect(component['getPeriodKeyFromEvent'](event2)).toBe('2021');
|
|
143
|
+
});
|
|
144
|
+
it('when groupEvent by month year', () => {
|
|
145
|
+
component.groupEvent = NgxTimelineEventGroup.MONTH_YEAR;
|
|
146
|
+
const event = {timestamp: new Date(2021, 7, 10)};
|
|
147
|
+
const event2 = {timestamp: new Date(2021, 8, 9)};
|
|
148
|
+
expect(component['getPeriodKeyFromEvent'](event)).toBe('2021/7');
|
|
149
|
+
expect(component['getPeriodKeyFromEvent'](event2)).toBe('2021/8');
|
|
150
|
+
});
|
|
151
|
+
it('when groupEvent by day month year', () => {
|
|
152
|
+
component.groupEvent = NgxTimelineEventGroup.DAY_MONTH_YEAR;
|
|
153
|
+
const event = {timestamp: new Date(2021, 7, 10)};
|
|
154
|
+
const event2 = {timestamp: new Date(2021, 8, 9)};
|
|
155
|
+
expect(component['getPeriodKeyFromEvent'](event)).toBe('2021/7/10');
|
|
156
|
+
expect(component['getPeriodKeyFromEvent'](event2)).toBe('2021/8/9');
|
|
157
|
+
});
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
});
|
|
@@ -0,0 +1,187 @@
|
|
|
1
|
+
import { Component, OnInit, Input, TemplateRef, OnChanges, SimpleChanges, Output } from '@angular/core';
|
|
2
|
+
import {
|
|
3
|
+
NgxTimelineEvent,
|
|
4
|
+
NgxTimelineItem,
|
|
5
|
+
NgxTimelineItemPosition,
|
|
6
|
+
NgxTimelinePeriodInfo,
|
|
7
|
+
NgxDateFormat,
|
|
8
|
+
NgxTimelineEventGroup,
|
|
9
|
+
NgxTimelineEventChangeSideInGroup,
|
|
10
|
+
periodKeyDateFormat,
|
|
11
|
+
fieldsToCheckEventChangeSideInGroup,
|
|
12
|
+
fieldsToAddEventGroup } from '../models';
|
|
13
|
+
import { BehaviorSubject } from 'rxjs';
|
|
14
|
+
|
|
15
|
+
@Component({
|
|
16
|
+
selector: 'ngx-timeline',
|
|
17
|
+
templateUrl: './ngx-timeline.component.html',
|
|
18
|
+
styleUrls: ['./ngx-timeline.scss'],
|
|
19
|
+
})
|
|
20
|
+
export class NgxTimelineComponent implements OnInit, OnChanges {
|
|
21
|
+
/**
|
|
22
|
+
* List of events to be displayed
|
|
23
|
+
*/
|
|
24
|
+
@Input() events: NgxTimelineEvent[];
|
|
25
|
+
/**
|
|
26
|
+
* Language code used to show the date formatted
|
|
27
|
+
*/
|
|
28
|
+
@Input() langCode?: string;
|
|
29
|
+
/**
|
|
30
|
+
* Boolean used to enable or disable the animations
|
|
31
|
+
*/
|
|
32
|
+
@Input() enableAnimation = true;
|
|
33
|
+
/**
|
|
34
|
+
* Logic to be applied in order to group events
|
|
35
|
+
*/
|
|
36
|
+
@Input() groupEvent?: NgxTimelineEventGroup = NgxTimelineEventGroup.MONTH_YEAR;
|
|
37
|
+
/**
|
|
38
|
+
* Logic to be applied in order to put evetns on LEFT or RIGHT
|
|
39
|
+
*/
|
|
40
|
+
@Input() changeSideInGroup?: NgxTimelineEventChangeSideInGroup = NgxTimelineEventChangeSideInGroup.ON_DIFFERENT_DAY;
|
|
41
|
+
/**
|
|
42
|
+
* Custom Template displayed before a group of events
|
|
43
|
+
*/
|
|
44
|
+
@Input() periodCustomTemplate: TemplateRef<any>;
|
|
45
|
+
/**
|
|
46
|
+
* Custom Template displayed to show a single event
|
|
47
|
+
*/
|
|
48
|
+
@Input() eventCustomTemplate: TemplateRef<any>;
|
|
49
|
+
/**
|
|
50
|
+
* Custom Template displayed to show an separator icon
|
|
51
|
+
*/
|
|
52
|
+
@Input() centerIconCustomTemplate: TemplateRef<any>;
|
|
53
|
+
/**
|
|
54
|
+
* Custom Template displayed to show the side date
|
|
55
|
+
*/
|
|
56
|
+
@Input() dateInstantCustomTemplate: TemplateRef<any>;
|
|
57
|
+
/**
|
|
58
|
+
* Custom Template displayed to show the inner event
|
|
59
|
+
*/
|
|
60
|
+
@Input() innerEventCustomTemplate: TemplateRef<any>;
|
|
61
|
+
/**
|
|
62
|
+
* Output click event emitter
|
|
63
|
+
*/
|
|
64
|
+
@Output()
|
|
65
|
+
clickEmitter: BehaviorSubject<NgxTimelineItem> = new BehaviorSubject(null);
|
|
66
|
+
|
|
67
|
+
groups: { [key: string]: NgxTimelineEvent[] } = {};
|
|
68
|
+
periods: NgxTimelineItem[] = [];
|
|
69
|
+
items: NgxTimelineItem[] = [];
|
|
70
|
+
ngxTimelineItemPosition = NgxTimelineItemPosition;
|
|
71
|
+
ngxDateFormat = NgxDateFormat;
|
|
72
|
+
|
|
73
|
+
private readonly separator = '/';
|
|
74
|
+
|
|
75
|
+
constructor() {}
|
|
76
|
+
|
|
77
|
+
|
|
78
|
+
ngOnInit(): void {
|
|
79
|
+
this.groupEvents(this.events);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
ngOnChanges(changes: SimpleChanges): void {
|
|
83
|
+
this.groupEvents(this.events);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
getPeriodKeyDateFormat(): string {
|
|
87
|
+
return periodKeyDateFormat[this.groupEvent];
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
protected clear(): void {
|
|
91
|
+
this.groups = {};
|
|
92
|
+
this.periods = [];
|
|
93
|
+
this.items = [];
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
protected groupEvents(events: NgxTimelineEvent[]): void {
|
|
97
|
+
if (events) {
|
|
98
|
+
this.clear();
|
|
99
|
+
this.sortEvents(events);
|
|
100
|
+
this.setGroups(events);
|
|
101
|
+
this.setPeriods();
|
|
102
|
+
this.setItems();
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
protected sortEvents(events: NgxTimelineEvent[]): void {
|
|
107
|
+
events.sort((a, b) => a.timestamp.getTime() - b.timestamp.getTime());
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
protected setGroups(events: NgxTimelineEvent[]): void {
|
|
111
|
+
events.forEach((event) => {
|
|
112
|
+
// conversion from string to actual Date
|
|
113
|
+
event.timestamp = new Date(event.timestamp);
|
|
114
|
+
const periodKey = this.getPeriodKeyFromEvent(event);
|
|
115
|
+
if (!this.groups[periodKey]) {
|
|
116
|
+
this.groups[periodKey] = [];
|
|
117
|
+
}
|
|
118
|
+
this.groups[periodKey].push(event);
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
protected setItems(): void {
|
|
123
|
+
this.periods.forEach((p) => {
|
|
124
|
+
// insert first the period
|
|
125
|
+
this.items.push(p);
|
|
126
|
+
// in each period putting items on left
|
|
127
|
+
const onLeft = true;
|
|
128
|
+
const periodInfo = p.periodInfo;
|
|
129
|
+
// insert then all the events in this period
|
|
130
|
+
this.addPeriodEvents(periodInfo, onLeft);
|
|
131
|
+
// onLeft = this.addEventItemsAndGetIfOnLeft(periodInfo, onLeft);
|
|
132
|
+
});
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
protected addPeriodEvents(periodInfo: NgxTimelinePeriodInfo, onLeft: boolean): void {
|
|
136
|
+
this.groups[periodInfo.periodKey].forEach((event, index) => {
|
|
137
|
+
const prevEvent = this.groups[periodInfo.periodKey][index - 1];
|
|
138
|
+
if (index > 0 && this.compareEvents(prevEvent, event)) {
|
|
139
|
+
onLeft = !onLeft;
|
|
140
|
+
}
|
|
141
|
+
this.pushEventOnItems(event, onLeft);
|
|
142
|
+
});
|
|
143
|
+
// return onLeft;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
protected pushEventOnItems(event: NgxTimelineEvent, onLeft: boolean): void {
|
|
147
|
+
this.items.push({
|
|
148
|
+
eventInfo: { ...event }, position: onLeft ?
|
|
149
|
+
this.ngxTimelineItemPosition.ON_LEFT : this.ngxTimelineItemPosition.ON_RIGHT
|
|
150
|
+
});
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
/**
|
|
154
|
+
* Compare the events inside the same group
|
|
155
|
+
*/
|
|
156
|
+
protected compareEvents(prevEvent: NgxTimelineEvent, event: NgxTimelineEvent): boolean {
|
|
157
|
+
return this.changeSideInGroup === NgxTimelineEventChangeSideInGroup.ALL ||
|
|
158
|
+
this.compareEventsField(prevEvent, event, ...fieldsToCheckEventChangeSideInGroup[this.changeSideInGroup]);
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
protected compareEventsField(prevEvent: NgxTimelineEvent, event: NgxTimelineEvent, ...fields: string[]): boolean {
|
|
162
|
+
return fields.reduce((res, field) => res = res || prevEvent.timestamp[field]() !== event.timestamp[field](), !!false);
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
protected setPeriods(): void {
|
|
166
|
+
this.periods = Object.keys(this.groups).map((periodKey) => {
|
|
167
|
+
const split = periodKey.split(this.separator);
|
|
168
|
+
return this.getPeriodInfo(split, periodKey);
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
private getPeriodInfo(split: string[], periodKey: string): { periodInfo: NgxTimelinePeriodInfo } {
|
|
173
|
+
return {
|
|
174
|
+
periodInfo: {
|
|
175
|
+
year: Number(split[0]),
|
|
176
|
+
month: Number(split[1]),
|
|
177
|
+
day: Number(split[2]),
|
|
178
|
+
periodKey,
|
|
179
|
+
firstDate: this.groups[periodKey][0].timestamp as Date,
|
|
180
|
+
},
|
|
181
|
+
};
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
protected getPeriodKeyFromEvent(event: NgxTimelineEvent): string {
|
|
185
|
+
return fieldsToAddEventGroup[this.groupEvent].map(field => event.timestamp[field]()).join(this.separator);
|
|
186
|
+
}
|
|
187
|
+
}
|