@bimeister/pupakit.calendar
Version:
PupaKit Calendar
68 lines • 15.8 kB
JavaScript
import { CdkVirtualScrollViewport } from '@angular/cdk/scrolling';
import { ChangeDetectionStrategy, Component, EventEmitter, Output, ViewChild, ViewEncapsulation, } from '@angular/core';
import { isNil } from '@bimeister/utilities';
import 'rxjs';
import { map } from 'rxjs/operators';
import { CalendarTextKey } from '../../declarations/enums/calendar-text-key.enum';
import '../../declarations/interfaces/calendar-translation.interface';
import { CalendarConfigService } from '../../services/calendar-config.service';
import { CalendarTranslationService } from '../../services/calendar-translation.service';
import * as i0 from "@angular/core";
import * as i1 from "../../services/calendar-translation.service";
import * as i2 from "../../services/calendar-config.service";
import * as i3 from "@angular/common";
import * as i4 from "@bimeister/pupakit.kit";
import * as i5 from "@angular/cdk/scrolling";
import * as i6 from "../calendar-header/calendar-header.component";
import * as i7 from "../calendar-selector-button/calendar-selector-button.component";
const YEARS_IN_ROW = 3;
const ROW_HEIGHT_PX = 48;
function createYearsTable(startYear, endYear) {
return Array.from({
length: Math.ceil((endYear - startYear) / YEARS_IN_ROW),
}).map((_, rowIndex) => Array.from({
length: YEARS_IN_ROW,
})
.map((__, colIndex) => startYear + rowIndex * YEARS_IN_ROW + colIndex)
.filter((year) => year < endYear));
}
export class CalendarYearSelectorComponent {
constructor(calendarTranslationService, calendarConfigService) {
this.calendarTranslationService = calendarTranslationService;
this.calendarConfigService = calendarConfigService;
this.select = new EventEmitter();
this.headerTitle$ = this.calendarTranslationService.translation$.pipe(map((translation) => translation.texts[CalendarTextKey.SelectYear]));
this.rowHeightPx = ROW_HEIGHT_PX;
this.currentYear = new Date().getFullYear();
this.startYear = this.calendarConfigService.startYear;
this.endYear = this.calendarConfigService.endYear;
this.yearsTable = createYearsTable(this.startYear, this.endYear);
}
ngAfterViewInit() {
this.scrollToCurrentYear();
}
selectYear(year) {
this.select.emit(year);
}
scrollToCurrentYear() {
if (isNil(this.virtualScrollViewport)) {
return;
}
const currentYearIndex = Math.floor((this.currentYear - this.startYear) / YEARS_IN_ROW);
requestAnimationFrame(() => {
this.virtualScrollViewport.scrollToIndex(currentYearIndex);
});
}
}
CalendarYearSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CalendarYearSelectorComponent, deps: [{ token: i1.CalendarTranslationService }, { token: i2.CalendarConfigService }], target: i0.ɵɵFactoryTarget.Component });
CalendarYearSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: CalendarYearSelectorComponent, selector: "pupa-calendar-year-selector", outputs: { select: "select" }, viewQueries: [{ propertyName: "virtualScrollViewport", first: true, predicate: CdkVirtualScrollViewport, descendants: true }], ngImport: i0, template: "<pupa-calendar-header>\n <h4 class=\"header-content\">\n {{ headerTitle$ | async }}\n </h4>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n <cdk-virtual-scroll-viewport pupaScrollableContent class=\"scroller\" [itemSize]=\"rowHeightPx\">\n <div class=\"scroller__row\" *cdkVirtualFor=\"let yearsRow of yearsTable; templateCacheSize: 10\">\n <pupa-calendar-selector-button\n *ngFor=\"let year of yearsRow\"\n [isCurrent]=\"year === currentYear\"\n (click)=\"selectYear(year)\"\n >\n {{ year }}\n </pupa-calendar-selector-button>\n </div>\n </cdk-virtual-scroll-viewport>\n</div>\n", styles: [":host{display:flex;flex-direction:column;height:100%;width:100%;position:absolute;z-index:2;top:0;left:0;right:0;bottom:0;background-color:rgba(var(--semantic-color_surface-primary),var(--semantic-color-alpha_surface-primary))}.header-content{display:flex;align-items:center;box-sizing:border-box;height:12rem;padding:0 4rem;margin:0}.scroller-wrapper{width:100%;height:100%;box-sizing:border-box;display:block}.scroller{width:100%;height:100%;display:block;box-sizing:border-box}.scroller__row{display:grid;grid-template-columns:repeat(3,1fr);box-sizing:border-box;width:100%;padding:2rem 4rem;gap:1rem}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.ScrollableContentDirective, selector: "[pupaScrollableContent]" }, { kind: "directive", type: i5.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i5.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i5.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }, { kind: "component", type: i6.CalendarHeaderComponent, selector: "pupa-calendar-header" }, { kind: "component", type: i7.CalendarSelectorButtonComponent, selector: "pupa-calendar-selector-button", inputs: ["isCurrent"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CalendarYearSelectorComponent, decorators: [{
type: Component,
args: [{ selector: 'pupa-calendar-year-selector', encapsulation: ViewEncapsulation.Emulated, changeDetection: ChangeDetectionStrategy.OnPush, template: "<pupa-calendar-header>\n <h4 class=\"header-content\">\n {{ headerTitle$ | async }}\n </h4>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n <cdk-virtual-scroll-viewport pupaScrollableContent class=\"scroller\" [itemSize]=\"rowHeightPx\">\n <div class=\"scroller__row\" *cdkVirtualFor=\"let yearsRow of yearsTable; templateCacheSize: 10\">\n <pupa-calendar-selector-button\n *ngFor=\"let year of yearsRow\"\n [isCurrent]=\"year === currentYear\"\n (click)=\"selectYear(year)\"\n >\n {{ year }}\n </pupa-calendar-selector-button>\n </div>\n </cdk-virtual-scroll-viewport>\n</div>\n", styles: [":host{display:flex;flex-direction:column;height:100%;width:100%;position:absolute;z-index:2;top:0;left:0;right:0;bottom:0;background-color:rgba(var(--semantic-color_surface-primary),var(--semantic-color-alpha_surface-primary))}.header-content{display:flex;align-items:center;box-sizing:border-box;height:12rem;padding:0 4rem;margin:0}.scroller-wrapper{width:100%;height:100%;box-sizing:border-box;display:block}.scroller{width:100%;height:100%;display:block;box-sizing:border-box}.scroller__row{display:grid;grid-template-columns:repeat(3,1fr);box-sizing:border-box;width:100%;padding:2rem 4rem;gap:1rem}\n"] }]
}], ctorParameters: function () { return [{ type: i1.CalendarTranslationService }, { type: i2.CalendarConfigService }]; }, propDecorators: { select: [{
type: Output
}], virtualScrollViewport: [{
type: ViewChild,
args: [CdkVirtualScrollViewport]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"calendar-year-selector.component.js","sourceRoot":"","sources":["../../../../src/components/calendar-year-selector/calendar-year-selector.component.ts","../../../../src/components/calendar-year-selector/calendar-year-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkVirtualScrollViewport } from '@angular/cdk/scrolling';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  EventEmitter,\n  Output,\n  ViewChild,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { isNil } from '@bimeister/utilities';\nimport { Observable } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { CalendarTextKey } from '../../declarations/enums/calendar-text-key.enum';\nimport { CalendarTranslation } from '../../declarations/interfaces/calendar-translation.interface';\nimport { CalendarConfigService } from '../../services/calendar-config.service';\nimport { CalendarTranslationService } from '../../services/calendar-translation.service';\n\nconst YEARS_IN_ROW: number = 3;\nconst ROW_HEIGHT_PX: number = 48;\n\nfunction createYearsTable(startYear: number, endYear: number): number[][] {\n  return Array.from({\n    length: Math.ceil((endYear - startYear) / YEARS_IN_ROW),\n  }).map((_: unknown, rowIndex: number) =>\n    Array.from({\n      length: YEARS_IN_ROW,\n    })\n      .map((__: unknown, colIndex: number) => startYear + rowIndex * YEARS_IN_ROW + colIndex)\n      .filter((year: number) => year < endYear)\n  );\n}\n\n@Component({\n  selector: 'pupa-calendar-year-selector',\n  templateUrl: './calendar-year-selector.component.html',\n  styleUrls: ['./calendar-year-selector.component.scss'],\n  encapsulation: ViewEncapsulation.Emulated,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CalendarYearSelectorComponent implements AfterViewInit {\n  @Output()\n  public readonly select: EventEmitter<number> = new EventEmitter<number>();\n\n  @ViewChild(CdkVirtualScrollViewport)\n  private readonly virtualScrollViewport: CdkVirtualScrollViewport;\n\n  public readonly headerTitle$: Observable<string> = this.calendarTranslationService.translation$.pipe(\n    map((translation: CalendarTranslation) => translation.texts[CalendarTextKey.SelectYear])\n  );\n\n  public readonly rowHeightPx: number = ROW_HEIGHT_PX;\n\n  public readonly currentYear: number = new Date().getFullYear();\n\n  private readonly startYear: number = this.calendarConfigService.startYear;\n  private readonly endYear: number = this.calendarConfigService.endYear;\n\n  public readonly yearsTable: number[][] = createYearsTable(this.startYear, this.endYear);\n\n  constructor(\n    private readonly calendarTranslationService: CalendarTranslationService,\n    private readonly calendarConfigService: CalendarConfigService\n  ) {}\n\n  public ngAfterViewInit(): void {\n    this.scrollToCurrentYear();\n  }\n\n  public selectYear(year: number): void {\n    this.select.emit(year);\n  }\n\n  private scrollToCurrentYear(): void {\n    if (isNil(this.virtualScrollViewport)) {\n      return;\n    }\n\n    const currentYearIndex: number = Math.floor((this.currentYear - this.startYear) / YEARS_IN_ROW);\n\n    requestAnimationFrame(() => {\n      this.virtualScrollViewport.scrollToIndex(currentYearIndex);\n    });\n  }\n}\n","<pupa-calendar-header>\n  <h4 class=\"header-content\">\n    {{ headerTitle$ | async }}\n  </h4>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n  <cdk-virtual-scroll-viewport pupaScrollableContent class=\"scroller\" [itemSize]=\"rowHeightPx\">\n    <div class=\"scroller__row\" *cdkVirtualFor=\"let yearsRow of yearsTable; templateCacheSize: 10\">\n      <pupa-calendar-selector-button\n        *ngFor=\"let year of yearsRow\"\n        [isCurrent]=\"year === currentYear\"\n        (click)=\"selectYear(year)\"\n      >\n        {{ year }}\n      </pupa-calendar-selector-button>\n    </div>\n  </cdk-virtual-scroll-viewport>\n</div>\n"]}