@qeydar/datepicker
Version:
A comprehensive Date and Time Picker for Angular with Jalali calendar support
228 lines • 27.7 kB
JavaScript
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"]}