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.
80 lines • 16 kB
JavaScript
import { CommonModule } from '@angular/common';
import { Component, EventEmitter, HostListener, Input, Output, ViewChild } from '@angular/core';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
export class DropdownComponent {
options = []; // List of dropdown options
selected; // Selected value
selectedChange = new EventEmitter(); // Event when selection changes
/**
* Optional function to convert an option to a displayable string.
*/
displayFn;
dropdownOpen = false;
dropdownContainer;
constructor() { }
ngOnInit() {
if (!this.selected && this.options.length > 0) {
this.selected = this.options[0]; // Default to first option if no selection
}
// Scroll to the selected option if it exists
setTimeout(() => {
const selectedIndex = this.options.indexOf(this.selected);
if (selectedIndex >= 0) {
const dropdownElement = this.dropdownContainer.nativeElement.querySelector('ul');
const selectedElement = dropdownElement?.children[selectedIndex];
if (selectedElement) {
selectedElement.scrollIntoView({ block: 'nearest' });
}
}
});
}
toggleDropdown(event) {
console.log('Dropdown clicked. Selected item:', this.selected);
this.dropdownOpen = !this.dropdownOpen;
if (this.dropdownOpen && this.selected) {
setTimeout(() => {
const selectedIndex = this.options.indexOf(this.selected);
if (selectedIndex >= 0) {
const dropdownElement = this.dropdownContainer.nativeElement.querySelector('ul');
const selectedElement = dropdownElement.children[selectedIndex];
if (selectedElement) {
selectedElement.scrollIntoView({ block: 'nearest' });
}
}
});
}
}
onOptionClick(option, index) {
this.selected = option;
this.selectedChange.emit(option);
this.dropdownOpen = false;
}
// Close dropdown when clicking outside
onClickOutside(event) {
if (this.dropdownContainer && !this.dropdownContainer.nativeElement.contains(event.target)) {
this.dropdownOpen = false;
}
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DropdownComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DropdownComponent, isStandalone: true, selector: "lucy-dropdown", inputs: { options: "options", selected: "selected", displayFn: "displayFn" }, outputs: { selectedChange: "selectedChange" }, host: { listeners: { "document:click": "onClickOutside($event)" } }, viewQueries: [{ propertyName: "dropdownContainer", first: true, predicate: ["dropdownContainer"], descendants: true }], ngImport: i0, template: "<div class=\"relative inline-block w-full\" #dropdownContainer>\r\n <!-- Dropdown Button -->\r\n <button (click)=\"toggleDropdown($event)\"\r\n class=\"w-full px-3 py-2 text-center text-gray-700 bg-white border border-gray-300 rounded-lg shadow-sm focus:outline-none\">\r\n {{ displayFn ? displayFn(selected) : selected }}\r\n <svg class=\"w-4 h-4 inline-block ml-2\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n </svg>\r\n </button>\r\n\r\n <!-- Dropdown List -->\r\n <div *ngIf=\"dropdownOpen\"\r\n class=\"absolute z-10 mt-1 w-full bg-white border border-gray-300 rounded-lg shadow-lg max-h-60 overflow-y-auto custom-scrollbar\">\r\n <ul>\r\n <li *ngFor=\"let option of options; let i = index\" (click)=\"onOptionClick(option, i)\"\r\n class=\"px-4 py-2 cursor-pointer hover:bg-gray-100\" [ngClass]=\"{'bg-blue-100': option === selected}\">\r\n {{ displayFn ? displayFn(option) : option }}\r\n </li>\r\n </ul>\r\n </div>\r\n</div>", styles: [".custom-scrollbar::-webkit-scrollbar{width:0px}.custom-scrollbar::-webkit-scrollbar-thumb{background-color:#0003;border-radius:4px;border:2px solid transparent;background-clip:content-box}.custom-scrollbar::-webkit-scrollbar-thumb:hover{background-color:#0006}.custom-scrollbar::-webkit-scrollbar-track{background:transparent}\n"], 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: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DropdownComponent, decorators: [{
type: Component,
args: [{ selector: 'lucy-dropdown', standalone: true, imports: [CommonModule], template: "<div class=\"relative inline-block w-full\" #dropdownContainer>\r\n <!-- Dropdown Button -->\r\n <button (click)=\"toggleDropdown($event)\"\r\n class=\"w-full px-3 py-2 text-center text-gray-700 bg-white border border-gray-300 rounded-lg shadow-sm focus:outline-none\">\r\n {{ displayFn ? displayFn(selected) : selected }}\r\n <svg class=\"w-4 h-4 inline-block ml-2\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\r\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n </svg>\r\n </button>\r\n\r\n <!-- Dropdown List -->\r\n <div *ngIf=\"dropdownOpen\"\r\n class=\"absolute z-10 mt-1 w-full bg-white border border-gray-300 rounded-lg shadow-lg max-h-60 overflow-y-auto custom-scrollbar\">\r\n <ul>\r\n <li *ngFor=\"let option of options; let i = index\" (click)=\"onOptionClick(option, i)\"\r\n class=\"px-4 py-2 cursor-pointer hover:bg-gray-100\" [ngClass]=\"{'bg-blue-100': option === selected}\">\r\n {{ displayFn ? displayFn(option) : option }}\r\n </li>\r\n </ul>\r\n </div>\r\n</div>", styles: [".custom-scrollbar::-webkit-scrollbar{width:0px}.custom-scrollbar::-webkit-scrollbar-thumb{background-color:#0003;border-radius:4px;border:2px solid transparent;background-clip:content-box}.custom-scrollbar::-webkit-scrollbar-thumb:hover{background-color:#0006}.custom-scrollbar::-webkit-scrollbar-track{background:transparent}\n"] }]
}], ctorParameters: () => [], propDecorators: { options: [{
type: Input
}], selected: [{
type: Input
}], selectedChange: [{
type: Output
}], displayFn: [{
type: Input
}], dropdownContainer: [{
type: ViewChild,
args: ['dropdownContainer']
}], onClickOutside: [{
type: HostListener,
args: ['document:click', ['$event']]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"custom-dropdown.component.js","sourceRoot":"","sources":["../../../../../projects/lucy-calendar/src/lib/custom-dropdown/custom-dropdown.component.ts","../../../../../projects/lucy-calendar/src/lib/custom-dropdown/custom-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, ElementRef, EventEmitter, HostListener, Input, OnInit, output, Output, ViewChild } from '@angular/core';\r\n\r\n@Component({\r\n  selector: 'lucy-dropdown',\r\n  standalone: true,\r\n  templateUrl: './custom-dropdown.component.html',\r\n  imports: [CommonModule],\r\n  styleUrls: ['./custom-dropdown.component.css']\r\n})\r\nexport class DropdownComponent<T> implements OnInit {\r\n  @Input() options: T[] = []; // List of dropdown options\r\n  @Input() selected!: T; // Selected value\r\n  @Output() selectedChange = new EventEmitter<T>(); // Event when selection changes\r\n\r\n  /**\r\n   * Optional function to convert an option to a displayable string.\r\n   */\r\n  @Input() displayFn?: (option: T) => string;\r\n\r\n  dropdownOpen: boolean = false;\r\n\r\n  @ViewChild('dropdownContainer') dropdownContainer!: ElementRef;\r\n\r\n  constructor() { }\r\n\r\n  ngOnInit(): void {\r\n    if (!this.selected && this.options.length > 0) {\r\n      this.selected = this.options[0]; // Default to first option if no selection\r\n    }\r\n    // Scroll to the selected option if it exists\r\n    setTimeout(() => {\r\n      const selectedIndex = this.options.indexOf(this.selected);\r\n      if (selectedIndex >= 0) {\r\n        const dropdownElement = this.dropdownContainer.nativeElement.querySelector('ul');\r\n        const selectedElement = dropdownElement?.children[selectedIndex];\r\n        if (selectedElement) {\r\n          selectedElement.scrollIntoView({ block: 'nearest' });\r\n        }\r\n      }\r\n    });\r\n  }\r\n\r\n  toggleDropdown(event: Event): void {\r\n    console.log('Dropdown clicked. Selected item:', this.selected);\r\n    this.dropdownOpen = !this.dropdownOpen;\r\n\r\n    if (this.dropdownOpen && this.selected) {\r\n      setTimeout(() => {\r\n        const selectedIndex = this.options.indexOf(this.selected);\r\n        if (selectedIndex >= 0) {\r\n          const dropdownElement = this.dropdownContainer.nativeElement.querySelector('ul');\r\n          const selectedElement = dropdownElement.children[selectedIndex];\r\n          if (selectedElement) {\r\n            selectedElement.scrollIntoView({ block: 'nearest' });\r\n          }\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  onOptionClick(option: T, index: number): void {\r\n    this.selected = option;\r\n    this.selectedChange.emit(option);\r\n    this.dropdownOpen = false;\r\n  }\r\n\r\n  // Close dropdown when clicking outside\r\n  @HostListener('document:click', ['$event'])\r\n  onClickOutside(event: Event) {\r\n    if (this.dropdownContainer && !this.dropdownContainer.nativeElement.contains(event.target)) {\r\n      this.dropdownOpen = false;\r\n    }\r\n  }\r\n}\r\n","<div class=\"relative inline-block w-full\" #dropdownContainer>\r\n    <!-- Dropdown Button -->\r\n    <button (click)=\"toggleDropdown($event)\"\r\n        class=\"w-full px-3 py-2 text-center text-gray-700 bg-white border border-gray-300 rounded-lg shadow-sm focus:outline-none\">\r\n        {{ displayFn ? displayFn(selected) : selected }}\r\n        <svg class=\"w-4 h-4 inline-block ml-2\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\r\n            <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M19 9l-7 7-7-7\" />\r\n        </svg>\r\n    </button>\r\n\r\n    <!-- Dropdown List -->\r\n    <div *ngIf=\"dropdownOpen\"\r\n        class=\"absolute z-10 mt-1 w-full bg-white border border-gray-300 rounded-lg shadow-lg max-h-60 overflow-y-auto custom-scrollbar\">\r\n        <ul>\r\n            <li *ngFor=\"let option of options; let i = index\" (click)=\"onOptionClick(option, i)\"\r\n                class=\"px-4 py-2 cursor-pointer hover:bg-gray-100\" [ngClass]=\"{'bg-blue-100': option === selected}\">\r\n                {{ displayFn ? displayFn(option) : option }}\r\n            </li>\r\n        </ul>\r\n    </div>\r\n</div>"]}