UNPKG

@qeydar/datepicker

Version:

A comprehensive Date and Time Picker for Angular with Jalali calendar support

228 lines 27.7 kB
import { Injectable } from '@angular/core'; import * as i0 from "@angular/core"; export class SelectionStrategyService { /** * Check if a date is selected (for single selection) */ isSelected(date, selectedDate, dateAdapter) { return selectedDate && dateAdapter.isSameDay(date, selectedDate); } /** * Check if a date is the start of a range */ isRangeStart(date, selectedStartDate, dateAdapter) { return selectedStartDate && dateAdapter.isSameDay(date, selectedStartDate); } /** * Check if a date is the end of a range */ isRangeEnd(date, selectedEndDate, dateAdapter) { return selectedEndDate && dateAdapter.isSameDay(date, selectedEndDate); } /** * Check if a date is in range (between start and end) */ isInRange(date, selectedStartDate, selectedEndDate, tempEndDate, dateAdapter) { if (!selectedStartDate || (!selectedEndDate && !tempEndDate)) { return false; } const endDate = selectedEndDate || tempEndDate; return (dateAdapter.isAfter(date, selectedStartDate) && dateAdapter.isBefore(date, endDate)); } /** * Check if a date is selected (for range selection) */ isRangeSelected(date, selectedStartDate, selectedEndDate, dateAdapter) { return (this.isRangeStart(date, selectedStartDate, dateAdapter) || this.isRangeEnd(date, selectedEndDate, dateAdapter)); } /** * Handle single date selection */ handleSingleSelection(date, selectedDate, showTimePicker, existingTime) { let finalDate = date; if (showTimePicker && existingTime) { finalDate = this.applyTimeToDate(date, existingTime); } return { selectedDate: finalDate, shouldEmit: !showTimePicker, }; } /** * Handle range date selection */ handleRangeSelection(date, selectedStartDate, selectedEndDate, showTimePicker, existingTime) { let finalDate = date; if (showTimePicker && existingTime) { finalDate = this.applyTimeToDate(date, existingTime); } // If no start date or both dates are selected or new date is before start date if (!selectedStartDate || (selectedStartDate && selectedEndDate) || this.isDateBefore(date, selectedStartDate)) { return { selectedStartDate: finalDate, selectedEndDate: null, shouldEmit: true, activeInput: 'end', }; } else { return { selectedStartDate, selectedEndDate: finalDate, shouldEmit: true, activeInput: 'end', }; } } /** * Check if a period is active (for sidebar periods) */ isActivePeriod(period, selectedStartDate, selectedEndDate, dateAdapter, allPeriods) { if (!selectedStartDate || !selectedEndDate) return false; if (period.value === 'custom') { const otherPeriods = allPeriods.filter((p) => p.value !== 'custom'); const hasActiveOther = otherPeriods.some((p) => this.isPeriodMatch(p, selectedStartDate, selectedEndDate, dateAdapter)); return !hasActiveOther; } return this.isPeriodMatch(period, selectedStartDate, selectedEndDate, dateAdapter); } /** * Check if a period is matched (for sidebar periods) */ isPeriodMatch(period, selectedStartDate, selectedEndDate, dateAdapter) { const [start, end] = period.value; const sameStart = dateAdapter.isEqual(dateAdapter.startOfDay(start), dateAdapter.startOfDay(selectedStartDate)); const sameEnd = dateAdapter.isEqual(dateAdapter.startOfDay(end), dateAdapter.startOfDay(selectedEndDate)); return sameStart && sameEnd; } /** * Handle period selection */ selectPeriod(period) { if (period.value === 'custom') { return { selectedPeriod: 'custom', isCustom: true, }; } const [start, end] = period.value; return { selectedPeriod: period.value, dateRange: { start, end }, isCustom: false, }; } /** * Apply time to a date */ applyTimeToDate(date, timeDate) { const result = new Date(date); result.setHours(timeDate.getHours()); result.setMinutes(timeDate.getMinutes()); result.setSeconds(timeDate.getSeconds()); return result; } /** * Check if first date is before second date */ isDateBefore(date1, date2) { return date1 < date2; } /** * Create date range object */ createDateRange(start, end) { if (!start) return null; return { start, end }; } /** * Check if range selection is complete */ isRangeComplete(selectedStartDate, selectedEndDate) { return !!(selectedStartDate && selectedEndDate); } /** * Get the active date for range selection */ getActiveDateForRange(activeInput, selectedStartDate, selectedEndDate) { if (activeInput === 'start') { return selectedStartDate; } else if (activeInput === 'end') { return selectedEndDate; } return null; } /** * Update time for range selection */ updateRangeTime(timeDate, activeInput, selectedStartDate, selectedEndDate) { if (activeInput === 'start' && selectedStartDate) { const updatedDate = this.applyTimeToDate(selectedStartDate, timeDate); return { selectedStartDate: updatedDate, selectedEndDate, shouldEmit: true, }; } else if (activeInput === 'end' && selectedEndDate) { const updatedDate = this.applyTimeToDate(selectedEndDate, timeDate); return { selectedStartDate, selectedEndDate: updatedDate, shouldEmit: true, }; } return { selectedStartDate, selectedEndDate, shouldEmit: false, }; } /** * Update time for single selection */ updateSingleTime(timeDate, selectedDate) { if (!selectedDate) { return { selectedDate: new Date(), shouldEmit: false, }; } const updatedDate = this.applyTimeToDate(selectedDate, timeDate); return { selectedDate: updatedDate, shouldEmit: false, }; } /** * Handle mouse enter for range selection preview */ handleMouseEnter(date, selectedStartDate, selectedEndDate) { if (selectedStartDate && !selectedEndDate) { return date; // This will be used as tempEndDate } return null; } /** * Check if date is today */ isToday(date, dateAdapter, showToday) { return showToday && dateAdapter.isSameDay(date, dateAdapter.today()); } } SelectionStrategyService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectionStrategyService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); SelectionStrategyService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectionStrategyService, providedIn: 'root' }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectionStrategyService, decorators: [{ type: Injectable, args: [{ providedIn: 'root', }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selection-strategy.service.js","sourceRoot":"","sources":["../../../../../projects/qeydar-datepicker/src/date-picker-popup/services/selection-strategy.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\r\nimport { DateAdapter } from '../../date-adapter';\r\nimport { CustomLabels, DateRange } from '../../utils/models';\r\n\r\n@Injectable({\r\n  providedIn: 'root',\r\n})\r\nexport class SelectionStrategyService {\r\n  /**\r\n   * Check if a date is selected (for single selection)\r\n   */\r\n  isSelected(\r\n    date: Date,\r\n    selectedDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    return selectedDate && dateAdapter.isSameDay(date, selectedDate);\r\n  }\r\n\r\n  /**\r\n   * Check if a date is the start of a range\r\n   */\r\n  isRangeStart(\r\n    date: Date,\r\n    selectedStartDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    return selectedStartDate && dateAdapter.isSameDay(date, selectedStartDate);\r\n  }\r\n\r\n  /**\r\n   * Check if a date is the end of a range\r\n   */\r\n  isRangeEnd(\r\n    date: Date,\r\n    selectedEndDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    return selectedEndDate && dateAdapter.isSameDay(date, selectedEndDate);\r\n  }\r\n\r\n  /**\r\n   * Check if a date is in range (between start and end)\r\n   */\r\n  isInRange(\r\n    date: Date,\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null,\r\n    tempEndDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    if (!selectedStartDate || (!selectedEndDate && !tempEndDate)) {\r\n      return false;\r\n    }\r\n\r\n    const endDate = selectedEndDate || tempEndDate;\r\n    return (\r\n      dateAdapter.isAfter(date, selectedStartDate) &&\r\n      dateAdapter.isBefore(date, endDate)\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Check if a date is selected (for range selection)\r\n   */\r\n  isRangeSelected(\r\n    date: Date,\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    return (\r\n      this.isRangeStart(date, selectedStartDate, dateAdapter) ||\r\n      this.isRangeEnd(date, selectedEndDate, dateAdapter)\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Handle single date selection\r\n   */\r\n  handleSingleSelection(\r\n    date: Date,\r\n    selectedDate: Date | null,\r\n    showTimePicker: boolean,\r\n    existingTime?: Date\r\n  ): { selectedDate: Date; shouldEmit: boolean } {\r\n    let finalDate = date;\r\n\r\n    if (showTimePicker && existingTime) {\r\n      finalDate = this.applyTimeToDate(date, existingTime);\r\n    }\r\n\r\n    return {\r\n      selectedDate: finalDate,\r\n      shouldEmit: !showTimePicker,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Handle range date selection\r\n   */\r\n  handleRangeSelection(\r\n    date: Date,\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null,\r\n    showTimePicker: boolean,\r\n    existingTime?: Date\r\n  ): {\r\n    selectedStartDate: Date | null;\r\n    selectedEndDate: Date | null;\r\n    shouldEmit: boolean;\r\n    activeInput: 'start' | 'end';\r\n  } {\r\n    let finalDate = date;\r\n\r\n    if (showTimePicker && existingTime) {\r\n      finalDate = this.applyTimeToDate(date, existingTime);\r\n    }\r\n\r\n    // If no start date or both dates are selected or new date is before start date\r\n    if (\r\n      !selectedStartDate ||\r\n      (selectedStartDate && selectedEndDate) ||\r\n      this.isDateBefore(date, selectedStartDate)\r\n    ) {\r\n      return {\r\n        selectedStartDate: finalDate,\r\n        selectedEndDate: null,\r\n        shouldEmit: true,\r\n        activeInput: 'end',\r\n      };\r\n    } else {\r\n      return {\r\n        selectedStartDate,\r\n        selectedEndDate: finalDate,\r\n        shouldEmit: true,\r\n        activeInput: 'end',\r\n      };\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Check if a period is active (for sidebar periods)\r\n   */\r\n  isActivePeriod(\r\n    period: CustomLabels,\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null,\r\n    dateAdapter: DateAdapter<Date>,\r\n    allPeriods: CustomLabels[]\r\n  ): boolean {\r\n    if (!selectedStartDate || !selectedEndDate) return false;\r\n\r\n    if (period.value === 'custom') {\r\n      const otherPeriods = allPeriods.filter((p) => p.value !== 'custom');\r\n      const hasActiveOther = otherPeriods.some((p) =>\r\n        this.isPeriodMatch(p, selectedStartDate, selectedEndDate, dateAdapter)\r\n      );\r\n      return !hasActiveOther;\r\n    }\r\n\r\n    return this.isPeriodMatch(\r\n      period,\r\n      selectedStartDate,\r\n      selectedEndDate,\r\n      dateAdapter\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Check if a period is matched (for sidebar periods)\r\n   */\r\n  isPeriodMatch(\r\n    period: CustomLabels,\r\n    selectedStartDate: Date,\r\n    selectedEndDate: Date,\r\n    dateAdapter: DateAdapter<Date>\r\n  ): boolean {\r\n    const [start, end] = period.value as Date[];\r\n\r\n    const sameStart = dateAdapter.isEqual(\r\n      dateAdapter.startOfDay(start),\r\n      dateAdapter.startOfDay(selectedStartDate)\r\n    );\r\n\r\n    const sameEnd = dateAdapter.isEqual(\r\n      dateAdapter.startOfDay(end),\r\n      dateAdapter.startOfDay(selectedEndDate)\r\n    );\r\n\r\n    return sameStart && sameEnd;\r\n  }\r\n\r\n  /**\r\n   * Handle period selection\r\n   */\r\n  selectPeriod(period: CustomLabels): {\r\n    selectedPeriod: any;\r\n    dateRange?: DateRange;\r\n    isCustom: boolean;\r\n  } {\r\n    if (period.value === 'custom') {\r\n      return {\r\n        selectedPeriod: 'custom',\r\n        isCustom: true,\r\n      };\r\n    }\r\n\r\n    const [start, end] = period.value as Date[];\r\n    return {\r\n      selectedPeriod: period.value,\r\n      dateRange: { start, end },\r\n      isCustom: false,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Apply time to a date\r\n   */\r\n  applyTimeToDate(date: Date, timeDate: Date): Date {\r\n    const result = new Date(date);\r\n    result.setHours(timeDate.getHours());\r\n    result.setMinutes(timeDate.getMinutes());\r\n    result.setSeconds(timeDate.getSeconds());\r\n    return result;\r\n  }\r\n\r\n  /**\r\n   * Check if first date is before second date\r\n   */\r\n  private isDateBefore(date1: Date, date2: Date): boolean {\r\n    return date1 < date2;\r\n  }\r\n\r\n  /**\r\n   * Create date range object\r\n   */\r\n  createDateRange(start: Date | null, end: Date | null): DateRange | null {\r\n    if (!start) return null;\r\n    return { start, end };\r\n  }\r\n\r\n  /**\r\n   * Check if range selection is complete\r\n   */\r\n  isRangeComplete(\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null\r\n  ): boolean {\r\n    return !!(selectedStartDate && selectedEndDate);\r\n  }\r\n\r\n  /**\r\n   * Get the active date for range selection\r\n   */\r\n  getActiveDateForRange(\r\n    activeInput: 'start' | 'end' | '',\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null\r\n  ): Date | null {\r\n    if (activeInput === 'start') {\r\n      return selectedStartDate;\r\n    } else if (activeInput === 'end') {\r\n      return selectedEndDate;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  /**\r\n   * Update time for range selection\r\n   */\r\n  updateRangeTime(\r\n    timeDate: Date,\r\n    activeInput: 'start' | 'end' | '',\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null\r\n  ): {\r\n    selectedStartDate: Date | null;\r\n    selectedEndDate: Date | null;\r\n    shouldEmit: boolean;\r\n  } {\r\n    if (activeInput === 'start' && selectedStartDate) {\r\n      const updatedDate = this.applyTimeToDate(selectedStartDate, timeDate);\r\n      return {\r\n        selectedStartDate: updatedDate,\r\n        selectedEndDate,\r\n        shouldEmit: true,\r\n      };\r\n    } else if (activeInput === 'end' && selectedEndDate) {\r\n      const updatedDate = this.applyTimeToDate(selectedEndDate, timeDate);\r\n      return {\r\n        selectedStartDate,\r\n        selectedEndDate: updatedDate,\r\n        shouldEmit: true,\r\n      };\r\n    }\r\n\r\n    return {\r\n      selectedStartDate,\r\n      selectedEndDate,\r\n      shouldEmit: false,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Update time for single selection\r\n   */\r\n  updateSingleTime(\r\n    timeDate: Date,\r\n    selectedDate: Date | null\r\n  ): { selectedDate: Date; shouldEmit: boolean } {\r\n    if (!selectedDate) {\r\n      return {\r\n        selectedDate: new Date(),\r\n        shouldEmit: false,\r\n      };\r\n    }\r\n\r\n    const updatedDate = this.applyTimeToDate(selectedDate, timeDate);\r\n    return {\r\n      selectedDate: updatedDate,\r\n      shouldEmit: false,\r\n    };\r\n  }\r\n\r\n  /**\r\n   * Handle mouse enter for range selection preview\r\n   */\r\n  handleMouseEnter(\r\n    date: Date,\r\n    selectedStartDate: Date | null,\r\n    selectedEndDate: Date | null\r\n  ): Date | null {\r\n    if (selectedStartDate && !selectedEndDate) {\r\n      return date; // This will be used as tempEndDate\r\n    }\r\n    return null;\r\n  }\r\n\r\n  /**\r\n   * Check if date is today\r\n   */\r\n  isToday(\r\n    date: Date,\r\n    dateAdapter: DateAdapter<Date>,\r\n    showToday: boolean\r\n  ): boolean {\r\n    return showToday && dateAdapter.isSameDay(date, dateAdapter.today());\r\n  }\r\n}\r\n"]}