UNPKG

@bimeister/pupakit.calendar

Version:
76 lines 20.9 kB
import { CdkVirtualScrollViewport } from '@angular/cdk/scrolling'; import { ChangeDetectionStrategy, Component, EventEmitter, Output, ViewChild, ViewEncapsulation, } from '@angular/core'; import { isNil } from '@bimeister/utilities'; import { Subject } from 'rxjs'; import { map, switchMap } from 'rxjs/operators'; import { MONTHS_IN_YEAR } from '../../declarations/constants/months-in-year.const'; import { CalendarTextKey } from '../../declarations/enums/calendar-text-key.enum'; import '../../declarations/enums/month-index.enum'; import '../../declarations/interfaces/calendar-month.interface'; 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 "@bimeister/pupakit.common"; import * as i6 from "@angular/cdk/scrolling"; import * as i7 from "../calendar-header/calendar-header.component"; import * as i8 from "../calendar-selector-button/calendar-selector-button.component"; import * as i9 from "../../pipes/is-current-calendar-month.pipe"; const DIVIDER_HEIGHT_PX = 12; const YEAR_LABEL_HEIGHT_PX = 16; const YEAR_TABLE_HEIGHT_PX = 188; const ITEM_HEIGHT_PX = DIVIDER_HEIGHT_PX * 2 + YEAR_LABEL_HEIGHT_PX + YEAR_TABLE_HEIGHT_PX; export class CalendarMonthSelectorComponent { constructor(calendarTranslationService, calendarConfigService) { this.calendarTranslationService = calendarTranslationService; this.calendarConfigService = calendarConfigService; this.select = new EventEmitter(); this.virtualScrollViewport$ = new Subject(); this.headerTitle$ = this.calendarTranslationService.translation$.pipe(map((translation) => translation.texts[CalendarTextKey.SelectMonth])); this.itemHeight = ITEM_HEIGHT_PX; this.startYear = this.calendarConfigService.startYear; this.currentYearInScroll$ = this.virtualScrollViewport$.pipe(switchMap((viewport) => viewport.scrolledIndexChange), map((index) => index + this.startYear)); this.yearsIndexes = Array.from({ length: this.calendarConfigService.yearsRange, }); this.monthsIndexes = Array.from({ length: MONTHS_IN_YEAR }); this.monthNameByIndex$ = this.calendarTranslationService.translation$.pipe(map((translation) => translation.months)); } ngAfterViewInit() { this.scrollToCurrentYear(); if (!isNil(this.virtualScrollViewport)) { this.virtualScrollViewport$.next(this.virtualScrollViewport); } } selectMonth(year, month) { this.select.emit({ year, month, }); } scrollToCurrentYear() { if (isNil(this.virtualScrollViewport)) { return; } const currentYearIndex = new Date().getFullYear() - this.startYear; requestAnimationFrame(() => { this.virtualScrollViewport.scrollToIndex(currentYearIndex); }); } } CalendarMonthSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CalendarMonthSelectorComponent, deps: [{ token: i1.CalendarTranslationService }, { token: i2.CalendarConfigService }], target: i0.ɵɵFactoryTarget.Component }); CalendarMonthSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: CalendarMonthSelectorComponent, selector: "pupa-calendar-month-selector", outputs: { select: "select" }, viewQueries: [{ propertyName: "virtualScrollViewport", first: true, predicate: CdkVirtualScrollViewport, descendants: true }], ngImport: i0, template: "<pupa-calendar-header>\n <div class=\"header-content\">\n <h4 class=\"header-title\">\n {{ headerTitle$ | async }}\n </h4>\n\n <div class=\"year-label\">{{ currentYearInScroll$ | async }}</div>\n </div>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n <cdk-virtual-scroll-viewport\n *pupaLet=\"monthNameByIndex$ | async as monthNameByIndex\"\n pupaScrollableContent\n class=\"scroller\"\n [itemSize]=\"itemHeight\"\n >\n <div\n class=\"scroller__item\"\n *cdkVirtualFor=\"let _ of yearsIndexes; templateCacheSize: 10; let isLast = last; let yearIndex = index\"\n >\n <div class=\"scroller__divider\"></div>\n\n <div class=\"year-table\">\n <div class=\"year-table__month\" *ngFor=\"let __ of monthsIndexes; let month = index\">\n <pupa-calendar-selector-button\n [isCurrent]=\"{ year: startYear + yearIndex, month } | isCurrentCalendarMonth\"\n (click)=\"selectMonth(startYear + yearIndex, month)\"\n >\n {{ monthNameByIndex[month] }}\n </pupa-calendar-selector-button>\n </div>\n </div>\n\n <ng-container *ngIf=\"!isLast\">\n <div class=\"scroller__divider\"></div>\n <div class=\"year-label\">{{ yearIndex + startYear + 1 }}</div>\n </ng-container>\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{flex-direction:column;box-sizing:border-box;padding:0 4rem}.header-title{display:flex;align-items:center;margin:0;height:12rem}.scroller-wrapper{width:100%;height:100%;box-sizing:border-box;display:block}.scroller{width:100%;height:100%;display:block;box-sizing:border-box}.scroller__item{padding:0rem 4rem}.scroller__divider{display:block;width:100%;height:3rem}.year-label{font-family:NotoSans,sans-serif;font-weight:400;font-size:11px;line-height:16px}.year-table{display:grid;grid-template-columns:repeat(3,1fr);column-gap:2rem;row-gap:5rem}.year-table__month{width:100%}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.ScrollableContentDirective, selector: "[pupaScrollableContent]" }, { kind: "directive", type: i5.LetDirective, selector: "[pupaLet]", inputs: ["pupaLet"] }, { kind: "directive", type: i6.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i6.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i6.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }, { kind: "component", type: i7.CalendarHeaderComponent, selector: "pupa-calendar-header" }, { kind: "component", type: i8.CalendarSelectorButtonComponent, selector: "pupa-calendar-selector-button", inputs: ["isCurrent"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "pipe", type: i9.IsCurrentCalendarMonthPipe, name: "isCurrentCalendarMonth" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: CalendarMonthSelectorComponent, decorators: [{ type: Component, args: [{ selector: 'pupa-calendar-month-selector', encapsulation: ViewEncapsulation.Emulated, changeDetection: ChangeDetectionStrategy.OnPush, template: "<pupa-calendar-header>\n <div class=\"header-content\">\n <h4 class=\"header-title\">\n {{ headerTitle$ | async }}\n </h4>\n\n <div class=\"year-label\">{{ currentYearInScroll$ | async }}</div>\n </div>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n <cdk-virtual-scroll-viewport\n *pupaLet=\"monthNameByIndex$ | async as monthNameByIndex\"\n pupaScrollableContent\n class=\"scroller\"\n [itemSize]=\"itemHeight\"\n >\n <div\n class=\"scroller__item\"\n *cdkVirtualFor=\"let _ of yearsIndexes; templateCacheSize: 10; let isLast = last; let yearIndex = index\"\n >\n <div class=\"scroller__divider\"></div>\n\n <div class=\"year-table\">\n <div class=\"year-table__month\" *ngFor=\"let __ of monthsIndexes; let month = index\">\n <pupa-calendar-selector-button\n [isCurrent]=\"{ year: startYear + yearIndex, month } | isCurrentCalendarMonth\"\n (click)=\"selectMonth(startYear + yearIndex, month)\"\n >\n {{ monthNameByIndex[month] }}\n </pupa-calendar-selector-button>\n </div>\n </div>\n\n <ng-container *ngIf=\"!isLast\">\n <div class=\"scroller__divider\"></div>\n <div class=\"year-label\">{{ yearIndex + startYear + 1 }}</div>\n </ng-container>\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{flex-direction:column;box-sizing:border-box;padding:0 4rem}.header-title{display:flex;align-items:center;margin:0;height:12rem}.scroller-wrapper{width:100%;height:100%;box-sizing:border-box;display:block}.scroller{width:100%;height:100%;display:block;box-sizing:border-box}.scroller__item{padding:0rem 4rem}.scroller__divider{display:block;width:100%;height:3rem}.year-label{font-family:NotoSans,sans-serif;font-weight:400;font-size:11px;line-height:16px}.year-table{display:grid;grid-template-columns:repeat(3,1fr);column-gap:2rem;row-gap:5rem}.year-table__month{width:100%}\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-month-selector.component.js","sourceRoot":"","sources":["../../../../src/components/calendar-month-selector/calendar-month-selector.component.ts","../../../../src/components/calendar-month-selector/calendar-month-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, Subject } from 'rxjs';\nimport { map, switchMap } from 'rxjs/operators';\nimport { MONTHS_IN_YEAR } from '../../declarations/constants/months-in-year.const';\nimport { CalendarTextKey } from '../../declarations/enums/calendar-text-key.enum';\nimport { MonthIndex } from '../../declarations/enums/month-index.enum';\nimport { CalendarMonth } from '../../declarations/interfaces/calendar-month.interface';\nimport { CalendarTranslation } from '../../declarations/interfaces/calendar-translation.interface';\nimport { CalendarConfigService } from '../../services/calendar-config.service';\nimport { CalendarTranslationService } from '../../services/calendar-translation.service';\n\nconst DIVIDER_HEIGHT_PX: number = 12;\nconst YEAR_LABEL_HEIGHT_PX: number = 16;\nconst YEAR_TABLE_HEIGHT_PX: number = 188;\n\nconst ITEM_HEIGHT_PX: number = DIVIDER_HEIGHT_PX * 2 + YEAR_LABEL_HEIGHT_PX + YEAR_TABLE_HEIGHT_PX;\n\n@Component({\n  selector: 'pupa-calendar-month-selector',\n  templateUrl: './calendar-month-selector.component.html',\n  styleUrls: ['./calendar-month-selector.component.scss'],\n  encapsulation: ViewEncapsulation.Emulated,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CalendarMonthSelectorComponent implements AfterViewInit {\n  @Output() public select: EventEmitter<CalendarMonth> = new EventEmitter<CalendarMonth>();\n\n  @ViewChild(CdkVirtualScrollViewport)\n  private readonly virtualScrollViewport: CdkVirtualScrollViewport;\n\n  private readonly virtualScrollViewport$: Subject<CdkVirtualScrollViewport> = new Subject<CdkVirtualScrollViewport>();\n\n  public readonly headerTitle$: Observable<string> = this.calendarTranslationService.translation$.pipe(\n    map((translation: CalendarTranslation) => translation.texts[CalendarTextKey.SelectMonth])\n  );\n\n  public readonly itemHeight: number = ITEM_HEIGHT_PX;\n\n  public readonly startYear: number = this.calendarConfigService.startYear;\n\n  public readonly currentYearInScroll$: Observable<number> = this.virtualScrollViewport$.pipe(\n    switchMap((viewport: CdkVirtualScrollViewport) => viewport.scrolledIndexChange),\n    map((index: number) => index + this.startYear)\n  );\n\n  public readonly yearsIndexes: unknown[] = Array.from({\n    length: this.calendarConfigService.yearsRange,\n  });\n  public readonly monthsIndexes: unknown[] = Array.from({ length: MONTHS_IN_YEAR });\n\n  public readonly monthNameByIndex$: Observable<Record<MonthIndex, string>> =\n    this.calendarTranslationService.translation$.pipe(map((translation: CalendarTranslation) => translation.months));\n\n  constructor(\n    private readonly calendarTranslationService: CalendarTranslationService,\n    private readonly calendarConfigService: CalendarConfigService\n  ) {}\n\n  public ngAfterViewInit(): void {\n    this.scrollToCurrentYear();\n\n    if (!isNil(this.virtualScrollViewport)) {\n      this.virtualScrollViewport$.next(this.virtualScrollViewport);\n    }\n  }\n\n  public selectMonth(year: number, month: number): void {\n    this.select.emit({\n      year,\n      month,\n    });\n  }\n\n  private scrollToCurrentYear(): void {\n    if (isNil(this.virtualScrollViewport)) {\n      return;\n    }\n\n    const currentYearIndex: number = new Date().getFullYear() - this.startYear;\n\n    requestAnimationFrame(() => {\n      this.virtualScrollViewport.scrollToIndex(currentYearIndex);\n    });\n  }\n}\n","<pupa-calendar-header>\n  <div class=\"header-content\">\n    <h4 class=\"header-title\">\n      {{ headerTitle$ | async }}\n    </h4>\n\n    <div class=\"year-label\">{{ currentYearInScroll$ | async }}</div>\n  </div>\n</pupa-calendar-header>\n\n<div class=\"scroller-wrapper\">\n  <cdk-virtual-scroll-viewport\n    *pupaLet=\"monthNameByIndex$ | async as monthNameByIndex\"\n    pupaScrollableContent\n    class=\"scroller\"\n    [itemSize]=\"itemHeight\"\n  >\n    <div\n      class=\"scroller__item\"\n      *cdkVirtualFor=\"let _ of yearsIndexes; templateCacheSize: 10; let isLast = last; let yearIndex = index\"\n    >\n      <div class=\"scroller__divider\"></div>\n\n      <div class=\"year-table\">\n        <div class=\"year-table__month\" *ngFor=\"let __ of monthsIndexes; let month = index\">\n          <pupa-calendar-selector-button\n            [isCurrent]=\"{ year: startYear + yearIndex, month } | isCurrentCalendarMonth\"\n            (click)=\"selectMonth(startYear + yearIndex, month)\"\n          >\n            {{ monthNameByIndex[month] }}\n          </pupa-calendar-selector-button>\n        </div>\n      </div>\n\n      <ng-container *ngIf=\"!isLast\">\n        <div class=\"scroller__divider\"></div>\n        <div class=\"year-label\">{{ yearIndex + startYear + 1 }}</div>\n      </ng-container>\n    </div>\n  </cdk-virtual-scroll-viewport>\n</div>\n"]}