lucy-calendar
Version:
LucyCalendar is a powerful and flexible date picker library for Angular applications, specifically designed for Ethiopian dates. It provides a user-friendly interface for selecting dates and supports various customization options to fit your needs.
251 lines • 50.1 kB
JavaScript
import { CommonModule } from '@angular/common';
import { Component, Input, HostListener, Output, EventEmitter } from '@angular/core';
import { FormsModule } from '@angular/forms';
import { isEthiopianLeapYear, toEthiopian, toGregorian } from './date-convertor';
import { DropdownComponent } from './custom-dropdown/custom-dropdown.component';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
export class LucyCalendarComponent {
ngOnChanges(changes) {
// if (changes['value'] && !changes['value'].firstChange) {
// this.value = changes['value'].currentValue;
// this.parseDate();
// // this.valueChange.emit(this.value);
// this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: this.selectedDay });
// this.emitChange();
// }
// else
if (changes['dateValue'] && !changes['dateValue'].firstChange && changes['dateValue'].currentValue?.getTime() !== this.dateValue?.getTime()) {
this.dateValue = changes['dateValue'].currentValue;
if (this.dateValue === null)
return;
const et = toEthiopian(this.dateValue);
this.selectedYear = et.year;
this.selectedMonth = et.month;
this.selectedDay = et.day;
// this.dateValueChange.emit(this.dateValue);
// this.value = this.formatDate();
this.emitChange();
}
}
ngOnInit() {
if (this.dateValue) {
this.selectedYear = this.dateValue.getFullYear();
this.selectedMonth = this.dateValue.getMonth();
this.selectedDay = this.dateValue.getDate();
}
// if (this.value) {
// this.parseDate();
// }
// if (this.selectedDay !== 0) {
// this.selectDate(this.selectedDay);
// } else {
// this.value = '';
// }
this.filteredMonths = this.availableMonths.filter(m => !this.isMonthOptionDisabled(m));
this.filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));
}
label = 'Select Date';
// @Input() value: string | null = null; // New input for value
valueChange = new EventEmitter(); // Output event emitter for value
dateValue = null;
dateValueChange = new EventEmitter(); // Output event emitter for grValue
placeholder = null;
min = null;
max = null;
dateFormat = 'DD/MM/YYYY'; // New input for date format
// @Input() disabled: boolean = false; // New input for disabled state
// @Input() readonly: boolean = true; // New input for readonly state
calendarVisible = false;
currentDate = new Date();
selectedYear = toEthiopian(this.currentDate).year; // Start with 2015 (Ethiopian year 2008)
selectedMonth = 1; // Start with Meskerem (January in Ethiopian calendar)
selectedDay = 0;
monthNames = [
"መስከረም", "ጥቅምት", "ህዳር", "ታህሳስ", "ጥር", "የካቲት",
"መጋቢት", "ሚይዚያ", "ግንቦት", "ሰኔ", "ሐምሌ", "ነሐሴ", "ጳጉሜ"
];
dayNames = ["እሁድ", "ሰኞ", "ማክሰኞ", "ረቡዕ", "ሐሙስ", "ዓርብ", "ቅዳሜ"];
availableYears = Array.from({ length: 101 }, (_, i) => this.currentDate.getFullYear() - 50 + i);
filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));
availableMonths = Array.from({ length: 13 }, (_, i) => i + 1);
filteredMonths = this.availableMonths.filter(month => !this.isMonthOptionDisabled(month));
refreshMonthOptions() {
this.filteredMonths = this.availableMonths.filter(month => !this.isMonthOptionDisabled(month));
}
;
refreshYearOptions() {
this.filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));
}
emitChange() {
this.dateValueChange.emit(this.dateValue); // Emit the new date value
// this.valueChange.emit(this.value);
}
toggleCalendar() {
this.calendarVisible = !this.calendarVisible;
if (this.dateValue) {
const et = toEthiopian(this.dateValue);
this.selectedYear = et.year;
this.selectedMonth = et.month;
this.selectedDay = et.day;
}
// if (this.value) {
// this.parseDate();
// }
}
monthDisplay = (month) => this.monthNames[month - 1]; /* Month numbers are 1-indexed so adjust for array (0-indexed)*/
selectMonthYear(month, year) {
this.selectedMonth = month;
this.selectedYear = year;
this.dateValue = toGregorian({ year, month, day: 1 });
// this.value = this.formatDate();
this.emitChange();
}
onMonthChanges(month) {
this.selectMonthYear(month, this.selectedYear);
this.refreshYearOptions();
}
onYearChanges(year) {
this.selectMonthYear(this.selectedMonth, year);
this.refreshMonthOptions();
}
prevMonth() {
this.selectedMonth = (this.selectedMonth - 1 + 13) % 13 || 13;
this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });
// this.value = this.formatDate();
this.emitChange();
}
nextMonth() {
this.selectedMonth = (this.selectedMonth + 1) % 13 || 13;
this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });
// this.value = this.formatDate();
this.emitChange();
}
getLeadingEmptyDays() {
const firstDay = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });
return Array(firstDay.getDay()).fill(null);
}
get daysInMonth() {
const daysInEthiopianMonth = this.selectedMonth === 13 ? (isEthiopianLeapYear(this.selectedYear) ? 6 : 5) : 30; // Pagumē has 6 days in a leap year
return Array.from({ length: daysInEthiopianMonth }, (_, i) => i + 1);
}
selectDate(day) {
this.selectedDay = day;
this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: day });
// this.value = this.formatDate();
this.calendarVisible = false;
this.emitChange();
}
clearDate() {
// if (this.disabled) return; // Prevent clearing date if disabled
this.dateValue = null;
// this.value = null;
this.selectedDay = 0;
this.calendarVisible = false;
this.emitChange();
}
selectToday() {
const today = toEthiopian(new Date());
this.selectedYear = today.year;
this.selectedMonth = today.month;
this.selectedDay = today.day;
this.selectDate(today.day);
}
isDayDisabled(day) {
if (this.max === null)
return false;
const date = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: day });
return date > this.max;
}
isNextMonthDisabled() {
if (this.max === null)
return false;
const nextMonthDate = toGregorian({ year: this.selectedYear, month: this.selectedMonth + 1, day: 1 });
return nextMonthDate > this.max;
}
isPrevMonthDisabled() {
if (this.min === null)
return false;
const prevMonthDate = toGregorian({ year: this.selectedYear, month: this.selectedMonth - 1, day: 1 });
return prevMonthDate < this.min;
}
isMonthOptionDisabled(monthIndex) {
const monthDate = toGregorian({ year: this.selectedYear, month: monthIndex, day: 1 });
return (this.max !== null && (monthDate > this.max || (this.selectedYear === this.max.getFullYear() && monthIndex > this.max.getMonth())))
|| (this.min !== null && (monthDate < this.min || (this.selectedYear === this.min.getFullYear() && monthIndex < this.min.getMonth())));
}
isYearOptionDisabled(year) {
const yearDate = toGregorian({ year: year, month: 1, day: 1 });
return (this.max !== null && yearDate > this.max) || (this.min !== null && yearDate < this.min);
}
formatDate() {
const formattedMonth = this.padZero(this.selectedMonth);
const formattedDay = this.padZero(this.selectedDay);
return this.dateFormat
.replace(/YYYY/i, this.selectedYear.toString())
.replace(/MM/i, formattedMonth)
.replace(/dd/i, formattedDay);
}
// parseDate() {
// if (!this.value || !this.dateFormat) return;
// // Detect separator from dateFormat (supports /, -, ., and others)
// const separator = this.dateFormat.match(/[^a-zA-Z0-9]/)?.[0] || '/';
// const dateParts = this.value.split(separator).map(Number);
// const formatParts = this.dateFormat.split(separator);
// if (dateParts.length !== formatParts.length) {
// console.error('Date format mismatch');
// return;
// }
// formatParts.forEach((part, index) => {
// const partType = part.toUpperCase()[0];
// switch (partType) {
// case 'Y':
// this.selectedYear = dateParts[index];
// break;
// case 'M':
// this.selectedMonth = dateParts[index];
// break;
// case 'D':
// this.selectedDay = dateParts[index];
// break;
// }
// });
// }
padZero(num) {
return num.toString().padStart(2, '0');
}
onClickOutside(event) {
const target = event.target;
if (!target.closest('.relative') && !target.closest('lucy-calendar') && !target.classList.contains('calendar-icon') && !target.classList.contains('lucy-host')) {
this.calendarVisible = false;
}
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LucyCalendarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: LucyCalendarComponent, isStandalone: true, selector: "lucy-calendar", inputs: { label: "label", dateValue: "dateValue", placeholder: "placeholder", min: "min", max: "max", dateFormat: "dateFormat", calendarVisible: "calendarVisible" }, outputs: { valueChange: "valueChange", dateValueChange: "dateValueChange" }, host: { listeners: { "document:click": "onClickOutside($event)" } }, usesOnChanges: true, ngImport: i0, template: "<div class=\"relative\">\r\n <!-- <lucy-date [label]=\"label\" [placeholder]=\"placeholder\" [dateFormat]=\"dateFormat\" [selectedDay]=\"selectedDay\"\r\n [readonly]=\"readonly\" [disabled]=\"disabled\" [dateFormat]=\"dateFormat\"></lucy-date> -->\r\n <!-- Calendar Dropdown -->\r\n <div id=\"calendar\" class=\"absolute mt-1 w-auto bg-white border border-gray-300 rounded-lg shadow-lg p-4 z-10\"\r\n [ngClass]=\"{'hidden': !calendarVisible}\">\r\n <div class=\"flex justify-between items-center mb-2\">\r\n <button (click)=\"prevMonth()\"\r\n class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n [disabled]=\"isPrevMonthDisabled()\">\r\n <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M15 19l-7-7 7-7\"></path>\r\n </svg>\r\n </button>\r\n <!-- (click)=\"toggleMonthYearSelection()\" -->\r\n <button class=\"font-medium text-gray-700\">\r\n {{ monthNames[selectedMonth - 1] }} {{ selectedYear }}\r\n </button>\r\n <button (click)=\"nextMonth()\"\r\n class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n [disabled]=\"isNextMonthDisabled()\">\r\n <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M9 5l7 7-7 7\"></path>\r\n </svg>\r\n </button>\r\n </div>\r\n\r\n <div class=\"grid grid-cols-2 gap-2\">\r\n <lucy-dropdown [options]=\"filteredMonths\" [selected]=\"selectedMonth\" (selectedChange)=\"onMonthChanges($event)\"\r\n [displayFn]=\"monthDisplay\">\r\n </lucy-dropdown>\r\n <lucy-dropdown [options]=\"filteredYears\" [selected]=\"selectedYear\"\r\n (selectedChange)=\"onYearChanges($event)\"></lucy-dropdown>\r\n </div>\r\n\r\n <!-- Days of the Month -->\r\n <div class=\"grid grid-cols-7 gap-1 text-center text-sm\">\r\n <div *ngFor=\"let dayName of dayNames\" class=\"text-gray-500\">{{ dayName }}</div>\r\n </div>\r\n\r\n <div class=\"grid grid-cols-7 gap-1 text-center text-sm pt-2\">\r\n <div *ngFor=\"let empty of getLeadingEmptyDays()\"></div>\r\n <button *ngFor=\"let day of daysInMonth\" class=\"py-1 rounded hover:bg-indigo-100 focus:bg-indigo-200\"\r\n [ngClass]=\"{'bg-blue-200': day === selectedDay, 'bg-gray-200 text-gray-400 cursor-not-allowed': isDayDisabled(day)}\"\r\n [disabled]=\"isDayDisabled(day)\" (click)=\"selectDate(day)\">\r\n {{ day }}\r\n </button>\r\n </div>\r\n\r\n <div class=\"flex justify-between mt-2\">\r\n <button (click)=\"clearDate()\"\r\n class=\"px-2 py-1 border border-red-500 text-red-500 rounded text-xs hover:bg-red-100\">Clear</button>\r\n <button (click)=\"selectToday()\"\r\n class=\"px-2 py-1 border border-blue-500 text-blue-500 rounded text-xs hover:bg-blue-100\">Today</button>\r\n </div>\r\n </div>\r\n</div>", styles: [""], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "ngmodule", type: FormsModule }, { kind: "component", type: DropdownComponent, selector: "lucy-dropdown", inputs: ["options", "selected", "displayFn"], outputs: ["selectedChange"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: LucyCalendarComponent, decorators: [{
type: Component,
args: [{ selector: 'lucy-calendar', standalone: true, imports: [CommonModule, FormsModule, DropdownComponent], template: "<div class=\"relative\">\r\n <!-- <lucy-date [label]=\"label\" [placeholder]=\"placeholder\" [dateFormat]=\"dateFormat\" [selectedDay]=\"selectedDay\"\r\n [readonly]=\"readonly\" [disabled]=\"disabled\" [dateFormat]=\"dateFormat\"></lucy-date> -->\r\n <!-- Calendar Dropdown -->\r\n <div id=\"calendar\" class=\"absolute mt-1 w-auto bg-white border border-gray-300 rounded-lg shadow-lg p-4 z-10\"\r\n [ngClass]=\"{'hidden': !calendarVisible}\">\r\n <div class=\"flex justify-between items-center mb-2\">\r\n <button (click)=\"prevMonth()\"\r\n class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n [disabled]=\"isPrevMonthDisabled()\">\r\n <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M15 19l-7-7 7-7\"></path>\r\n </svg>\r\n </button>\r\n <!-- (click)=\"toggleMonthYearSelection()\" -->\r\n <button class=\"font-medium text-gray-700\">\r\n {{ monthNames[selectedMonth - 1] }} {{ selectedYear }}\r\n </button>\r\n <button (click)=\"nextMonth()\"\r\n class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n [disabled]=\"isNextMonthDisabled()\">\r\n <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M9 5l7 7-7 7\"></path>\r\n </svg>\r\n </button>\r\n </div>\r\n\r\n <div class=\"grid grid-cols-2 gap-2\">\r\n <lucy-dropdown [options]=\"filteredMonths\" [selected]=\"selectedMonth\" (selectedChange)=\"onMonthChanges($event)\"\r\n [displayFn]=\"monthDisplay\">\r\n </lucy-dropdown>\r\n <lucy-dropdown [options]=\"filteredYears\" [selected]=\"selectedYear\"\r\n (selectedChange)=\"onYearChanges($event)\"></lucy-dropdown>\r\n </div>\r\n\r\n <!-- Days of the Month -->\r\n <div class=\"grid grid-cols-7 gap-1 text-center text-sm\">\r\n <div *ngFor=\"let dayName of dayNames\" class=\"text-gray-500\">{{ dayName }}</div>\r\n </div>\r\n\r\n <div class=\"grid grid-cols-7 gap-1 text-center text-sm pt-2\">\r\n <div *ngFor=\"let empty of getLeadingEmptyDays()\"></div>\r\n <button *ngFor=\"let day of daysInMonth\" class=\"py-1 rounded hover:bg-indigo-100 focus:bg-indigo-200\"\r\n [ngClass]=\"{'bg-blue-200': day === selectedDay, 'bg-gray-200 text-gray-400 cursor-not-allowed': isDayDisabled(day)}\"\r\n [disabled]=\"isDayDisabled(day)\" (click)=\"selectDate(day)\">\r\n {{ day }}\r\n </button>\r\n </div>\r\n\r\n <div class=\"flex justify-between mt-2\">\r\n <button (click)=\"clearDate()\"\r\n class=\"px-2 py-1 border border-red-500 text-red-500 rounded text-xs hover:bg-red-100\">Clear</button>\r\n <button (click)=\"selectToday()\"\r\n class=\"px-2 py-1 border border-blue-500 text-blue-500 rounded text-xs hover:bg-blue-100\">Today</button>\r\n </div>\r\n </div>\r\n</div>" }]
}], propDecorators: { label: [{
type: Input
}], valueChange: [{
type: Output
}], dateValue: [{
type: Input
}], dateValueChange: [{
type: Output
}], placeholder: [{
type: Input
}], min: [{
type: Input
}], max: [{
type: Input
}], dateFormat: [{
type: Input
}], calendarVisible: [{
type: Input
}], onClickOutside: [{
type: HostListener,
args: ['document:click', ['$event']]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lucy-calendar.component.js","sourceRoot":"","sources":["../../../../projects/lucy-calendar/src/lib/lucy-calendar.component.ts","../../../../projects/lucy-calendar/src/lib/lucy-calendar.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, Input, OnInit, HostListener, Output, EventEmitter, OnChanges, SimpleChanges } from '@angular/core';\r\nimport { FormsModule } from '@angular/forms';\r\nimport { isEthiopianLeapYear, toEthiopian, toGregorian } from './date-convertor';\r\nimport { DropdownComponent } from './custom-dropdown/custom-dropdown.component';\r\nimport { LucyDateComponent } from './lucy-date/lucy-date.component';\r\n\r\n@Component({\r\n  selector: 'lucy-calendar',\r\n  standalone: true,\r\n  imports: [CommonModule, FormsModule, DropdownComponent],\r\n  templateUrl: './lucy-calendar.component.html',\r\n  styleUrl: './lucy-calendar.component.css'\r\n})\r\nexport class LucyCalendarComponent implements OnInit, OnChanges {\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // if (changes['value'] && !changes['value'].firstChange) {\r\n    //   this.value = changes['value'].currentValue;\r\n    //   this.parseDate();\r\n    //   // this.valueChange.emit(this.value);\r\n    //   this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: this.selectedDay });\r\n    //   this.emitChange();\r\n    // }\r\n    // else \r\n    if (changes['dateValue'] && !changes['dateValue'].firstChange && changes['dateValue'].currentValue?.getTime() !== this.dateValue?.getTime()) {\r\n      this.dateValue = changes['dateValue'].currentValue;\r\n      if (this.dateValue === null)\r\n        return;\r\n      const et = toEthiopian(this.dateValue);\r\n      this.selectedYear = et.year;\r\n      this.selectedMonth = et.month;\r\n      this.selectedDay = et.day;\r\n      // this.dateValueChange.emit(this.dateValue);\r\n      // this.value = this.formatDate();\r\n      this.emitChange();\r\n    }\r\n  }\r\n  ngOnInit(): void {\r\n    if (this.dateValue) {\r\n      this.selectedYear = this.dateValue.getFullYear();\r\n      this.selectedMonth = this.dateValue.getMonth();\r\n      this.selectedDay = this.dateValue.getDate();\r\n    }\r\n    // if (this.value) {\r\n    //   this.parseDate();\r\n    // }\r\n    // if (this.selectedDay !== 0) {\r\n    //   this.selectDate(this.selectedDay);\r\n    // } else {\r\n    //   this.value = '';\r\n    // }\r\n    this.filteredMonths = this.availableMonths.filter(m => !this.isMonthOptionDisabled(m));\r\n    this.filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));\r\n  }\r\n  @Input() label: string = 'Select Date';\r\n  // @Input() value: string | null = null; // New input for value\r\n  @Output() valueChange: EventEmitter<string | null> = new EventEmitter<string | null>(); // Output event emitter for value\r\n  @Input() dateValue: Date | null = null;\r\n  @Output() dateValueChange: EventEmitter<Date | null> = new EventEmitter<Date | null>(); // Output event emitter for grValue\r\n\r\n  @Input() placeholder: string | null = null;\r\n  @Input() min: Date | null = null;\r\n  @Input() max: Date | null = null;\r\n  @Input() dateFormat: string = 'DD/MM/YYYY'; // New input for date format\r\n  // @Input() disabled: boolean = false; // New input for disabled state\r\n  // @Input() readonly: boolean = true; // New input for readonly state\r\n\r\n  @Input() calendarVisible: boolean = false;\r\n\r\n  currentDate: Date = new Date();\r\n  selectedYear: number = toEthiopian(this.currentDate).year; // Start with 2015 (Ethiopian year 2008)\r\n  selectedMonth: number = 1; // Start with Meskerem (January in Ethiopian calendar)\r\n  selectedDay: number = 0;\r\n  monthNames: string[] = [\r\n    \"መስከረም\", \"ጥቅምት\", \"ህዳር\", \"ታህሳስ\", \"ጥር\", \"የካቲት\",\r\n    \"መጋቢት\", \"ሚይዚያ\", \"ግንቦት\", \"ሰኔ\", \"ሐምሌ\", \"ነሐሴ\", \"ጳጉሜ\"\r\n  ];\r\n  dayNames: string[] = [\"እሁድ\", \"ሰኞ\", \"ማክሰኞ\", \"ረቡዕ\", \"ሐሙስ\", \"ዓርብ\", \"ቅዳሜ\"];\r\n\r\n  availableYears: number[] = Array.from({ length: 101 }, (_, i) => this.currentDate.getFullYear() - 50 + i);\r\n  filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));\r\n  availableMonths = Array.from({ length: 13 }, (_, i) => i + 1);\r\n  filteredMonths = this.availableMonths.filter(month => !this.isMonthOptionDisabled(month));\r\n\r\n\r\n  refreshMonthOptions() {\r\n    this.filteredMonths = this.availableMonths.filter(month => !this.isMonthOptionDisabled(month));\r\n  };\r\n  refreshYearOptions(): void {\r\n    this.filteredYears = this.availableYears.filter(y => !this.isYearOptionDisabled(y));\r\n  }\r\n\r\n  emitChange() {\r\n    this.dateValueChange.emit(this.dateValue); // Emit the new date value\r\n    // this.valueChange.emit(this.value);\r\n  }\r\n\r\n  toggleCalendar() {\r\n    this.calendarVisible = !this.calendarVisible;\r\n    if (this.dateValue) {\r\n      const et = toEthiopian(this.dateValue);\r\n      this.selectedYear = et.year;\r\n      this.selectedMonth = et.month;\r\n      this.selectedDay = et.day;\r\n    }\r\n    // if (this.value) {\r\n    //   this.parseDate();\r\n    // }\r\n  }\r\n\r\n  monthDisplay = (month: number): string => this.monthNames[month - 1]; /* Month numbers are 1-indexed so adjust for array (0-indexed)*/\r\n\r\n  selectMonthYear(month: number, year: number) {\r\n    this.selectedMonth = month;\r\n    this.selectedYear = year;\r\n    this.dateValue = toGregorian({ year, month, day: 1 });\r\n    // this.value = this.formatDate();\r\n    this.emitChange();\r\n  }\r\n\r\n  onMonthChanges(month: number) {\r\n    this.selectMonthYear(month, this.selectedYear);\r\n    this.refreshYearOptions();\r\n  }\r\n\r\n  onYearChanges(year: number) {\r\n    this.selectMonthYear(this.selectedMonth, year);\r\n    this.refreshMonthOptions();\r\n  }\r\n\r\n  prevMonth() {\r\n    this.selectedMonth = (this.selectedMonth - 1 + 13) % 13 || 13;\r\n    this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });\r\n    // this.value = this.formatDate();\r\n    this.emitChange()\r\n  }\r\n\r\n  nextMonth() {\r\n    this.selectedMonth = (this.selectedMonth + 1) % 13 || 13;\r\n    this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });\r\n    // this.value = this.formatDate();\r\n    this.emitChange();\r\n  }\r\n\r\n  getLeadingEmptyDays(): any[] {\r\n    const firstDay = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: 1 });\r\n    return Array(firstDay.getDay()).fill(null);\r\n  }\r\n\r\n  get daysInMonth(): number[] {\r\n    const daysInEthiopianMonth = this.selectedMonth === 13 ? (isEthiopianLeapYear(this.selectedYear) ? 6 : 5) : 30; // Pagumē has 6 days in a leap year\r\n    return Array.from({ length: daysInEthiopianMonth }, (_, i) => i + 1);\r\n  }\r\n\r\n  selectDate(day: number) {\r\n    this.selectedDay = day;\r\n    this.dateValue = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: day });\r\n    // this.value = this.formatDate();\r\n    this.calendarVisible = false;\r\n    this.emitChange();\r\n  }\r\n\r\n  clearDate() {\r\n    // if (this.disabled) return; // Prevent clearing date if disabled\r\n    this.dateValue = null;\r\n    // this.value = null;\r\n    this.selectedDay = 0;\r\n    this.calendarVisible = false;\r\n    this.emitChange();\r\n  }\r\n\r\n  selectToday() {\r\n    const today = toEthiopian(new Date());\r\n    this.selectedYear = today.year;\r\n    this.selectedMonth = today.month;\r\n    this.selectedDay = today.day;\r\n    this.selectDate(today.day);\r\n  }\r\n\r\n  isDayDisabled(day: number): boolean {\r\n    if (this.max === null)\r\n      return false;\r\n    const date = toGregorian({ year: this.selectedYear, month: this.selectedMonth, day: day });\r\n    return date > this.max;\r\n  }\r\n\r\n  isNextMonthDisabled(): boolean {\r\n    if (this.max === null)\r\n      return false;\r\n    const nextMonthDate = toGregorian({ year: this.selectedYear, month: this.selectedMonth + 1, day: 1 });\r\n    return nextMonthDate > this.max;\r\n  }\r\n\r\n  isPrevMonthDisabled(): boolean {\r\n    if (this.min === null)\r\n      return false;\r\n    const prevMonthDate = toGregorian({ year: this.selectedYear, month: this.selectedMonth - 1, day: 1 });\r\n    return prevMonthDate < this.min;\r\n  }\r\n\r\n  isMonthOptionDisabled(monthIndex: number): boolean {\r\n    const monthDate = toGregorian({ year: this.selectedYear, month: monthIndex, day: 1 });\r\n    return (this.max !== null && (monthDate > this.max || (this.selectedYear === this.max.getFullYear() && monthIndex > this.max.getMonth())))\r\n      || (this.min !== null && (monthDate < this.min || (this.selectedYear === this.min.getFullYear() && monthIndex < this.min.getMonth())));\r\n  }\r\n\r\n  isYearOptionDisabled(year: number): boolean {\r\n    const yearDate = toGregorian({ year: year, month: 1, day: 1 });\r\n    return (this.max !== null && yearDate > this.max) || (this.min !== null && yearDate < this.min);\r\n  }\r\n\r\n  formatDate(): string {\r\n    const formattedMonth = this.padZero(this.selectedMonth);\r\n    const formattedDay = this.padZero(this.selectedDay);\r\n    return this.dateFormat\r\n      .replace(/YYYY/i, this.selectedYear.toString())\r\n      .replace(/MM/i, formattedMonth)\r\n      .replace(/dd/i, formattedDay);\r\n  }\r\n\r\n  // parseDate() {\r\n  //   if (!this.value || !this.dateFormat) return;\r\n\r\n  //   // Detect separator from dateFormat (supports /, -, ., and others)\r\n  //   const separator = this.dateFormat.match(/[^a-zA-Z0-9]/)?.[0] || '/';\r\n  //   const dateParts = this.value.split(separator).map(Number);\r\n  //   const formatParts = this.dateFormat.split(separator);\r\n\r\n  //   if (dateParts.length !== formatParts.length) {\r\n  //     console.error('Date format mismatch');\r\n  //     return;\r\n  //   }\r\n\r\n  //   formatParts.forEach((part, index) => {\r\n  //     const partType = part.toUpperCase()[0];\r\n  //     switch (partType) {\r\n  //       case 'Y':\r\n  //         this.selectedYear = dateParts[index];\r\n  //         break;\r\n  //       case 'M':\r\n  //         this.selectedMonth = dateParts[index];\r\n  //         break;\r\n  //       case 'D':\r\n  //         this.selectedDay = dateParts[index];\r\n  //         break;\r\n  //     }\r\n  //   });\r\n  // }\r\n\r\n  padZero(num: number): string {\r\n    return num.toString().padStart(2, '0');\r\n  }\r\n\r\n  @HostListener('document:click', ['$event'])\r\n  onClickOutside(event: MouseEvent) {\r\n    const target = event.target as HTMLElement;\r\n    if (!target.closest('.relative') && !target.closest('lucy-calendar') && !target.classList.contains('calendar-icon') && !target.classList.contains('lucy-host')) {\r\n      this.calendarVisible = false;\r\n    }\r\n  }\r\n}\r\n","<div class=\"relative\">\r\n  <!-- <lucy-date [label]=\"label\" [placeholder]=\"placeholder\" [dateFormat]=\"dateFormat\" [selectedDay]=\"selectedDay\"\r\n    [readonly]=\"readonly\" [disabled]=\"disabled\" [dateFormat]=\"dateFormat\"></lucy-date> -->\r\n  <!-- Calendar Dropdown -->\r\n  <div id=\"calendar\" class=\"absolute mt-1 w-auto bg-white border border-gray-300 rounded-lg shadow-lg p-4 z-10\"\r\n    [ngClass]=\"{'hidden': !calendarVisible}\">\r\n    <div class=\"flex justify-between items-center mb-2\">\r\n      <button (click)=\"prevMonth()\"\r\n        class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n        [disabled]=\"isPrevMonthDisabled()\">\r\n        <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n          viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n          <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M15 19l-7-7 7-7\"></path>\r\n        </svg>\r\n      </button>\r\n      <!-- (click)=\"toggleMonthYearSelection()\" -->\r\n      <button class=\"font-medium text-gray-700\">\r\n        {{ monthNames[selectedMonth - 1] }} {{ selectedYear }}\r\n      </button>\r\n      <button (click)=\"nextMonth()\"\r\n        class=\"text-gray-500 hover:text-gray-700 disabled:text-gray-300 disabled:cursor-not-allowed\"\r\n        [disabled]=\"isNextMonthDisabled()\">\r\n        <svg class=\"h-6 w-6 text-gray-500 hover:text-gray-700 disabled:text-gray-300\" fill=\"none\" stroke=\"currentColor\"\r\n          viewBox=\"0 0 24 24\" xmlns=\"http://www.w3.org/2000/svg\">\r\n          <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M9 5l7 7-7 7\"></path>\r\n        </svg>\r\n      </button>\r\n    </div>\r\n\r\n    <div class=\"grid grid-cols-2 gap-2\">\r\n      <lucy-dropdown [options]=\"filteredMonths\" [selected]=\"selectedMonth\" (selectedChange)=\"onMonthChanges($event)\"\r\n        [displayFn]=\"monthDisplay\">\r\n      </lucy-dropdown>\r\n      <lucy-dropdown [options]=\"filteredYears\" [selected]=\"selectedYear\"\r\n        (selectedChange)=\"onYearChanges($event)\"></lucy-dropdown>\r\n    </div>\r\n\r\n    <!-- Days of the Month -->\r\n    <div class=\"grid grid-cols-7 gap-1 text-center text-sm\">\r\n      <div *ngFor=\"let dayName of dayNames\" class=\"text-gray-500\">{{ dayName }}</div>\r\n    </div>\r\n\r\n    <div class=\"grid grid-cols-7 gap-1 text-center text-sm pt-2\">\r\n      <div *ngFor=\"let empty of getLeadingEmptyDays()\"></div>\r\n      <button *ngFor=\"let day of daysInMonth\" class=\"py-1 rounded hover:bg-indigo-100 focus:bg-indigo-200\"\r\n        [ngClass]=\"{'bg-blue-200': day === selectedDay, 'bg-gray-200 text-gray-400 cursor-not-allowed': isDayDisabled(day)}\"\r\n        [disabled]=\"isDayDisabled(day)\" (click)=\"selectDate(day)\">\r\n        {{ day }}\r\n      </button>\r\n    </div>\r\n\r\n    <div class=\"flex justify-between mt-2\">\r\n      <button (click)=\"clearDate()\"\r\n        class=\"px-2 py-1 border border-red-500 text-red-500 rounded text-xs hover:bg-red-100\">Clear</button>\r\n      <button (click)=\"selectToday()\"\r\n        class=\"px-2 py-1 border border-blue-500 text-blue-500 rounded text-xs hover:bg-blue-100\">Today</button>\r\n    </div>\r\n  </div>\r\n</div>"]}