UNPKG

@rishovt/angular-nepali-datepicker

Version:

A lightweight Angular wrapper and fully customizable, non-typeable Nepali datepicker component for Angular with support for Unicode-rendered Nepali dates, BS↔AD conversion, and flexible date formats. Ideal for forms requiring calendar-based input with nat

161 lines 21.5 kB
import { Component, Input, Output, EventEmitter, forwardRef, ViewChild } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { RtcNepaliDatepickerService } from './angular-nepali-datepicker.service'; export class RtcNepaliDatepickerComponent { constructor(service) { this.service = service; this.options = {}; this.dateChange = new EventEmitter(); this.pickerId = 'angular-nepali-datepicker-default'; this.disabled = false; this.onChange = () => { }; this.onTouched = () => { }; this.todayDate = { year: 0, month: 0, day: 0, value: '' }; } ngOnInit() { this.service.BSGetCurrentDate().then(currentDate => { this.todayDate = currentDate; this.defaultOptions = { classes: 'form-control', placeholder: 'Select Nepali Date', dateFormat: 'YYYY-MM-DD', closeOnDateSelect: true, minDate: { year: 1800, month: 1, day: 1 }, maxDate: currentDate, disabled: false, unicodeDate: true, language: 'nepali', }; }); } onKeyDown(event) { event.preventDefault(); return false; } ngAfterViewInit() { this.service.loadLibrary().then(() => { this.service.BSGetCurrentDate().then(currentDate => { this.todayDate = Object.assign(Object.assign({}, currentDate), { value: this.service.formatDate(currentDate, this.options.dateFormat || 'YYYY-MM-DD') }); this.defaultOptions = { classes: 'form-control', placeholder: 'Select Nepali Date', dateFormat: 'YYYY-MM-DD', closeOnDateSelect: true, minDate: { year: 1800, month: 1, day: 1 }, maxDate: currentDate, disabled: false, unicodeDate: true, language: 'nepali', tooltip: 'Select Date' }; this.initializeDatePicker(); this.service.registerComponent(this.pickerId, this); }); }).catch((error) => { console.error('Failed to load Nepali Date Picker library:', error); }); } initializeDatePicker() { const input = this.inputElementRef.nativeElement; if (!input || typeof input.NepaliDatePicker !== 'function') { console.error('Nepali Date Picker is not loaded. Make sure the assets is included in consumer angular.json.'); return; } const config = Object.assign(Object.assign(Object.assign({}, this.defaultOptions), this.options), { onSelect: (bsDate) => { this.dateChange.emit(bsDate); input.dispatchEvent(new Event('change')); } }); input.NepaliDatePicker(config); } setDate(bsDate) { const input = this.inputElementRef.nativeElement; if (!input || typeof input.NepaliDatePicker !== 'function') return; try { input.NepaliDatePicker('destroy'); const updatedOptions = Object.assign(Object.assign(Object.assign({}, this.defaultOptions), this.options), { value: bsDate, onSelect: (selectedDate) => { this.dateChange.emit(selectedDate); input.dispatchEvent(new Event('change')); } }); const setValueAndInit = (displayValue) => { input.NepaliDatePicker(updatedOptions); input.value = displayValue; }; if (updatedOptions.unicodeDate) { const convertFullString = (str) => { return Promise.all(str.split('').map(d => this.service.ConvertToUnicode(+d))) .then(digits => digits.join('')); }; const [year, month, day] = bsDate.split('-'); Promise.all([ convertFullString(year), convertFullString(month), convertFullString(day) ]).then(([unicodeYear, unicodeMonth, unicodeDay]) => { setValueAndInit(`${unicodeYear}-${unicodeMonth}-${unicodeDay}`); }); } else { setValueAndInit(bsDate); } } catch (err) { console.warn('Failed to set date via NepaliDatePicker:', err); } } writeValue(value) { this.value = value; if (value) this.service.setDate(this.pickerId, value); } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(isDisabled) { this.disabled = isDisabled; } ngOnDestroy() { this.service.unregisterComponent(this.pickerId); const input = this.inputElementRef.nativeElement; if (input && typeof input.NepaliDatePicker === 'function') { input.NepaliDatePicker('destroy'); } } } RtcNepaliDatepickerComponent.decorators = [ { type: Component, args: [{ selector: 'rtc-nepali-datepicker', template: ` <input #nepaliInput type="text" [id]="pickerId" [class]="options.classes || 'form-control'" [placeholder]="options.placeholder || 'Select Nepali Date'" [disabled]="options.disabled || false" (keydown)="onKeyDown($event)" /> `, providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => RtcNepaliDatepickerComponent), multi: true } ] },] } ]; RtcNepaliDatepickerComponent.ctorParameters = () => [ { type: RtcNepaliDatepickerService } ]; RtcNepaliDatepickerComponent.propDecorators = { options: [{ type: Input }], dateChange: [{ type: Output }], value: [{ type: Input }], pickerId: [{ type: Input }], inputElementRef: [{ type: ViewChild, args: ['nepaliInput', { static: true },] }] }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"angular-nepali-datepicker.component.js","sourceRoot":"","sources":["../../../../../projects/angular-nepali-datepicker-ve/src/lib/angular-nepali-datepicker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, Output, EventEmitter, AfterViewInit, OnInit, forwardRef, ViewChild, ElementRef } from '@angular/core';\r\nimport { NG_VALUE_ACCESSOR } from '@angular/forms';\r\nimport { RtcNepaliDatepickerService } from './angular-nepali-datepicker.service';\r\nimport { DateObject } from './types';\r\n\r\n@Component({\r\n  selector: 'rtc-nepali-datepicker',\r\n  template: `\r\n    <input\r\n      #nepaliInput\r\n      type=\"text\"\r\n      [id]=\"pickerId\"\r\n      [class]=\"options.classes || 'form-control'\"\r\n      [placeholder]=\"options.placeholder || 'Select Nepali Date'\"\r\n      [disabled]=\"options.disabled || false\"\r\n      (keydown)=\"onKeyDown($event)\"\r\n    />\r\n  `,\r\n  providers: [\r\n    {\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => RtcNepaliDatepickerComponent),\r\n      multi: true\r\n    }\r\n  ]\r\n})\r\nexport class RtcNepaliDatepickerComponent implements AfterViewInit, OnInit {\r\n  @Input() options: {\r\n    classes?: string;\r\n    placeholder?: string;\r\n    dateFormat?: string;\r\n    closeOnDateSelect?: boolean;\r\n    minDate?: { year: number; month: number; day: number };\r\n    maxDate?: { year: number; month: number; day: number };\r\n    disabled?: boolean;\r\n    unicodeDate?: boolean;\r\n    language?: string;\r\n    onSelect?: (date: any) => void;\r\n  } = {};\r\n\r\n  @Output() dateChange = new EventEmitter<any>();\r\n  @Input() value: any;\r\n  @Input() pickerId: string = 'angular-nepali-datepicker-default';\r\n  public disabled: boolean = false;\r\n  private onChange: (value: any) => void = () => { };\r\n  private onTouched: () => void = () => { };\r\n\r\n  @ViewChild('nepaliInput', { static: true }) inputElementRef!: ElementRef;\r\n  defaultOptions: any;\r\n  todayDate: DateObject = { year: 0, month: 0, day: 0, value: '' };\r\n  constructor(private service: RtcNepaliDatepickerService) { }\r\n\r\n  ngOnInit(): void {\r\n    this.service.BSGetCurrentDate().then(currentDate => {\r\n      this.todayDate = currentDate;\r\n      this.defaultOptions = {\r\n        classes: 'form-control',\r\n        placeholder: 'Select Nepali Date',\r\n        dateFormat: 'YYYY-MM-DD',\r\n        closeOnDateSelect: true,\r\n        minDate: { year: 1800, month: 1, day: 1 },\r\n        maxDate: currentDate,\r\n        disabled: false,\r\n        unicodeDate: true,\r\n        language: 'nepali',\r\n      };\r\n    })\r\n  }\r\n\r\n  onKeyDown(event: KeyboardEvent): boolean {\r\n    event.preventDefault();\r\n    return false;\r\n  }\r\n  ngAfterViewInit(): void {\r\n    this.service.loadLibrary().then(() => {\r\n      this.service.BSGetCurrentDate().then(currentDate => {\r\n        this.todayDate = {\r\n          ...currentDate,\r\n          value: this.service.formatDate(currentDate, this.options.dateFormat || 'YYYY-MM-DD')\r\n        };\r\n        this.defaultOptions = {\r\n          classes: 'form-control',\r\n          placeholder: 'Select Nepali Date',\r\n          dateFormat: 'YYYY-MM-DD',\r\n          closeOnDateSelect: true,\r\n          minDate: { year: 1800, month: 1, day: 1 },\r\n          maxDate: currentDate,\r\n          disabled: false,\r\n          unicodeDate: true,\r\n          language: 'nepali',\r\n          tooltip: 'Select Date'\r\n        };\r\n        this.initializeDatePicker();\r\n        this.service.registerComponent(this.pickerId, this);\r\n      })\r\n    }).catch((error) => {\r\n      console.error('Failed to load Nepali Date Picker library:', error);\r\n    });\r\n  }\r\n\r\n  private initializeDatePicker(): void {\r\n    const input = this.inputElementRef.nativeElement;\r\n    if (!input || typeof input.NepaliDatePicker !== 'function') {\r\n      console.error('Nepali Date Picker is not loaded. Make sure the assets is included in consumer angular.json.');\r\n      return;\r\n    }\r\n    const config = {\r\n      ...this.defaultOptions,\r\n      ...this.options,\r\n      onSelect: (bsDate: any) => {\r\n        this.dateChange.emit(bsDate);\r\n        input.dispatchEvent(new Event('change'));\r\n      }\r\n    };\r\n    input.NepaliDatePicker(config);\r\n  }\r\n\r\n  public setDate(bsDate: string): void {\r\n    const input = this.inputElementRef.nativeElement;\r\n    if (!input || typeof input.NepaliDatePicker !== 'function') return;\r\n    try {\r\n      input.NepaliDatePicker('destroy');\r\n      const updatedOptions = {\r\n        ...this.defaultOptions,\r\n        ...this.options,\r\n        value: bsDate,\r\n        onSelect: (selectedDate: any) => {\r\n          this.dateChange.emit(selectedDate);\r\n          input.dispatchEvent(new Event('change'));\r\n        },\r\n      };\r\n\r\n      const setValueAndInit = (displayValue: string) => {\r\n        input.NepaliDatePicker(updatedOptions);\r\n        input.value = displayValue;\r\n      };\r\n      if (updatedOptions.unicodeDate) {\r\n        const convertFullString = (str: string): Promise<string> => {\r\n          return Promise.all(str.split('').map(d => this.service.ConvertToUnicode(+d)))\r\n            .then(digits => digits.join(''));\r\n        };\r\n        const [year, month, day] = bsDate.split('-');\r\n        Promise.all([\r\n          convertFullString(year),\r\n          convertFullString(month),\r\n          convertFullString(day)\r\n        ]).then(([unicodeYear, unicodeMonth, unicodeDay]) => {\r\n          setValueAndInit(`${unicodeYear}-${unicodeMonth}-${unicodeDay}`);\r\n        });\r\n      } else {\r\n        setValueAndInit(bsDate);\r\n      }\r\n    } catch (err) {\r\n      console.warn('Failed to set date via NepaliDatePicker:', err);\r\n    }\r\n  }\r\n\r\n  writeValue(value: any): void {\r\n    this.value = value;\r\n    if (value) this.service.setDate(this.pickerId, value);\r\n  }\r\n\r\n  registerOnChange(fn: any): void {\r\n    this.onChange = fn;\r\n  }\r\n\r\n  registerOnTouched(fn: any): void {\r\n    this.onTouched = fn;\r\n  }\r\n\r\n  setDisabledState(isDisabled: boolean): void {\r\n    this.disabled = isDisabled;\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.service.unregisterComponent(this.pickerId);\r\n    const input = this.inputElementRef.nativeElement;\r\n    if (input && typeof input.NepaliDatePicker === 'function') {\r\n      input.NepaliDatePicker('destroy');\r\n    }\r\n  }\r\n\r\n}"]}