@ngfly/carousel
Version:
A smooth, customizable carousel component for Angular 17+ applications
206 lines • 24.4 kB
JavaScript
import { Injectable } from '@angular/core';
import { BehaviorSubject } from 'rxjs';
import * as i0 from "@angular/core";
/**
* Default carousel configuration values
*/
export const CAROUSEL_DEFAULTS = {
NAVIGATION_SIZE: '32px',
CONTENT_PADDING: '10px',
ANIMATION_DURATION: '300ms',
ANIMATION_TIMING: 'ease',
EMPTY_STATE_HEIGHT: '200px',
NAVIGATION_PREV_ICON: '❮',
NAVIGATION_NEXT_ICON: '❯',
INDICATOR_SIZE: '10px',
INDICATOR_SPACING: '5px',
INDICATOR_ACTIVE_COLOR: '#333',
INDICATOR_INACTIVE_COLOR: '#ccc',
INDICATOR_ACTIVE_OPACITY: '1',
INDICATOR_INACTIVE_OPACITY: '0.5',
INDICATOR_ANIMATION_DURATION: '250ms',
INDICATOR_ANIMATION_TIMING: 'ease',
INDICATOR_TRANSITION: 'all 0.3s ease-in-out'
};
/**
* Service for carousel-related functionality and state management
*/
export class CarouselService {
constructor() {
// Visibility state
this.isVisibleSubject = new BehaviorSubject(true);
this.isVisible$ = this.isVisibleSubject.asObservable();
}
/**
* Set carousel visibility
* @param visible Whether carousel is visible
*/
setVisibility(visible) {
this.isVisibleSubject.next(visible);
}
/**
* Get button shape styles based on the configured shape
* @param shape Button shape type
* @returns Style object
*/
getButtonShapeStyles(shape) {
const styles = {};
switch (shape) {
case 'circle':
styles['borderRadius'] = '50%';
break;
case 'rounded':
styles['borderRadius'] = '8px';
break;
case 'square':
default:
styles['borderRadius'] = '0';
}
return styles;
}
/**
* Get indicator styles
* @param config Indicator style configuration
* @param isActive Whether to get active or inactive styles
* @returns Style object
*/
getIndicatorStyles(config, isActive = false) {
// Base styles for both active and inactive
const baseStyles = {
width: CAROUSEL_DEFAULTS.INDICATOR_SIZE,
height: CAROUSEL_DEFAULTS.INDICATOR_SIZE,
display: 'inline-block',
transition: config?.transition || (config?.animation?.timing
? `all ${config.animation.duration || CAROUSEL_DEFAULTS.INDICATOR_ANIMATION_DURATION} ${config.animation.timing}`
: CAROUSEL_DEFAULTS.INDICATOR_TRANSITION),
cursor: 'pointer',
margin: `0 ${config?.spacing || CAROUSEL_DEFAULTS.INDICATOR_SPACING}`,
borderRadius: '50%' // Default circle shape
};
// Active/inactive specific styles
if (isActive) {
baseStyles['backgroundColor'] = CAROUSEL_DEFAULTS.INDICATOR_ACTIVE_COLOR;
baseStyles['opacity'] = CAROUSEL_DEFAULTS.INDICATOR_ACTIVE_OPACITY;
baseStyles['transform'] = 'scale(1.2)';
// Add animation if enabled and not explicitly disabled
const animEnabled = config?.animation?.enabled !== false;
const animType = config?.animation?.type || 'pulse';
if (animEnabled && animType !== 'none') {
if (animType === 'custom' && config?.animation?.custom) {
baseStyles['animation'] = config.animation.custom;
}
else if (animType === 'pulse') {
baseStyles['animation'] = `indicator-pulse 1s infinite alternate`;
}
}
// Apply custom active styles if provided (these override defaults)
if (config?.active) {
Object.assign(baseStyles, config.active);
}
}
else {
baseStyles['backgroundColor'] = CAROUSEL_DEFAULTS.INDICATOR_INACTIVE_COLOR;
baseStyles['opacity'] = CAROUSEL_DEFAULTS.INDICATOR_INACTIVE_OPACITY;
baseStyles['transform'] = 'scale(1)';
// Apply custom inactive styles if provided (these override defaults)
if (config?.inactive) {
Object.assign(baseStyles, config.inactive);
}
}
return baseStyles;
}
/**
* Get indicator container styles based on configuration
* @param config Indicator style configuration
* @returns Style object for container
*/
getIndicatorContainerStyles(config) {
const styles = {
position: 'absolute',
display: 'flex',
justifyContent: 'center',
alignItems: 'center',
zIndex: '100',
padding: '10px',
pointerEvents: 'auto'
};
// Default position (bottom center)
styles['bottom'] = config?.position?.bottom || '4px';
styles['left'] = config?.position?.left || '50%';
styles['transform'] = 'translateX(-50%)';
// Override with custom positions if provided
if (config?.position) {
Object.keys(config.position).forEach(key => {
const position = config.position;
if (position[key]) {
styles[key] = position[key];
}
});
// Handle transformations for centered positioning
if (config.position.left === '50%' && !styles['transform']) {
styles['transform'] = 'translateX(-50%)';
}
else if (config.position.top === '50%' && !styles['transform']) {
styles['transform'] = 'translateY(-50%)';
}
}
// Apply custom container styles if provided
if (config?.container) {
Object.assign(styles, config.container);
}
return styles;
}
/**
* Get navigation icons based on orientation
* @param isVertical Whether carousel is vertical
* @param icons Custom icon configuration
* @returns Previous and next icons
*/
getNavigationIcons(isVertical, icons) {
const defaultIcons = {
horizontal: {
prev: CAROUSEL_DEFAULTS.NAVIGATION_PREV_ICON,
next: CAROUSEL_DEFAULTS.NAVIGATION_NEXT_ICON
},
vertical: {
prev: CAROUSEL_DEFAULTS.NAVIGATION_PREV_ICON,
next: CAROUSEL_DEFAULTS.NAVIGATION_NEXT_ICON
}
};
const customIcons = icons || {};
const verticalIcons = customIcons.vertical || {};
return {
prev: isVertical
? (verticalIcons.prev || defaultIcons.vertical.prev)
: (customIcons.prev || defaultIcons.horizontal.prev),
next: isVertical
? (verticalIcons.next || defaultIcons.vertical.next)
: (customIcons.next || defaultIcons.horizontal.next)
};
}
/**
* Parse time string to milliseconds
* @param time Time string (e.g., '300ms', '0.5s')
* @returns Time in milliseconds
*/
parseTimeToMs(time) {
if (!time)
return 300; // Default 300ms
if (time.endsWith('ms')) {
return parseInt(time.slice(0, -2), 10);
}
if (time.endsWith('s')) {
return parseFloat(time.slice(0, -1)) * 1000;
}
return parseInt(time, 10);
}
}
CarouselService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CarouselService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
CarouselService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CarouselService, providedIn: 'root' });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: CarouselService, decorators: [{
type: Injectable,
args: [{
providedIn: 'root'
}]
}] });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"carousel.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/carousel.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { NavButtonShape, IndicatorStyle } from '../interfaces/carousel-config.interface';\n\n/**\n * Default carousel configuration values\n */\nexport const CAROUSEL_DEFAULTS = {\n  NAVIGATION_SIZE: '32px',\n  CONTENT_PADDING: '10px',\n  ANIMATION_DURATION: '300ms',\n  ANIMATION_TIMING: 'ease',\n  EMPTY_STATE_HEIGHT: '200px',\n  NAVIGATION_PREV_ICON: '❮',\n  NAVIGATION_NEXT_ICON: '❯',\n  INDICATOR_SIZE: '10px',\n  INDICATOR_SPACING: '5px',\n  INDICATOR_ACTIVE_COLOR: '#333',\n  INDICATOR_INACTIVE_COLOR: '#ccc',\n  INDICATOR_ACTIVE_OPACITY: '1',\n  INDICATOR_INACTIVE_OPACITY: '0.5',\n  INDICATOR_ANIMATION_DURATION: '250ms',\n  INDICATOR_ANIMATION_TIMING: 'ease',\n  INDICATOR_TRANSITION: 'all 0.3s ease-in-out'\n};\n\n/**\n * Service for carousel-related functionality and state management\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class CarouselService {\n  // Visibility state\n  private isVisibleSubject = new BehaviorSubject<boolean>(true);\n  isVisible$ = this.isVisibleSubject.asObservable();\n\n  /**\n   * Set carousel visibility\n   * @param visible Whether carousel is visible\n   */\n  setVisibility(visible: boolean): void {\n    this.isVisibleSubject.next(visible);\n  }\n\n  /**\n   * Get button shape styles based on the configured shape\n   * @param shape Button shape type\n   * @returns Style object\n   */\n  getButtonShapeStyles(shape?: NavButtonShape): Record<string, string> {\n    const styles: Record<string, string> = {};\n    \n    switch (shape) {\n      case 'circle':\n        styles['borderRadius'] = '50%';\n        break;\n      case 'rounded':\n        styles['borderRadius'] = '8px';\n        break;\n      case 'square':\n      default:\n        styles['borderRadius'] = '0';\n    }\n    \n    return styles;\n  }\n\n  /**\n   * Get indicator styles\n   * @param config Indicator style configuration\n   * @param isActive Whether to get active or inactive styles\n   * @returns Style object\n   */\n  getIndicatorStyles(config?: IndicatorStyle, isActive = false): Record<string, string> {\n    // Base styles for both active and inactive\n    const baseStyles: Record<string, string> = {\n      width: CAROUSEL_DEFAULTS.INDICATOR_SIZE,\n      height: CAROUSEL_DEFAULTS.INDICATOR_SIZE,\n      display: 'inline-block',\n      transition: config?.transition || (config?.animation?.timing \n        ? `all ${config.animation.duration || CAROUSEL_DEFAULTS.INDICATOR_ANIMATION_DURATION} ${config.animation.timing}`\n        : CAROUSEL_DEFAULTS.INDICATOR_TRANSITION),\n      cursor: 'pointer',\n      margin: `0 ${config?.spacing || CAROUSEL_DEFAULTS.INDICATOR_SPACING}`,\n      borderRadius: '50%' // Default circle shape\n    };\n    \n    // Active/inactive specific styles\n    if (isActive) {\n      baseStyles['backgroundColor'] = CAROUSEL_DEFAULTS.INDICATOR_ACTIVE_COLOR;\n      baseStyles['opacity'] = CAROUSEL_DEFAULTS.INDICATOR_ACTIVE_OPACITY;\n      baseStyles['transform'] = 'scale(1.2)';\n      \n      // Add animation if enabled and not explicitly disabled\n      const animEnabled = config?.animation?.enabled !== false;\n      const animType = config?.animation?.type || 'pulse';\n      \n      if (animEnabled && animType !== 'none') {\n        if (animType === 'custom' && config?.animation?.custom) {\n          baseStyles['animation'] = config.animation.custom;\n        } else if (animType === 'pulse') {\n          baseStyles['animation'] = `indicator-pulse 1s infinite alternate`;\n        }\n      }\n      \n      // Apply custom active styles if provided (these override defaults)\n      if (config?.active) {\n        Object.assign(baseStyles, config.active);\n      }\n    } else {\n      baseStyles['backgroundColor'] = CAROUSEL_DEFAULTS.INDICATOR_INACTIVE_COLOR;\n      baseStyles['opacity'] = CAROUSEL_DEFAULTS.INDICATOR_INACTIVE_OPACITY;\n      baseStyles['transform'] = 'scale(1)';\n      \n      // Apply custom inactive styles if provided (these override defaults)\n      if (config?.inactive) {\n        Object.assign(baseStyles, config.inactive);\n      }\n    }\n    \n    return baseStyles;\n  }\n\n  /**\n   * Get indicator container styles based on configuration\n   * @param config Indicator style configuration\n   * @returns Style object for container\n   */\n  getIndicatorContainerStyles(config?: IndicatorStyle): Record<string, string> {\n    const styles: Record<string, string> = {\n      position: 'absolute',\n      display: 'flex',\n      justifyContent: 'center',\n      alignItems: 'center',\n      zIndex: '100',\n      padding: '10px',\n      pointerEvents: 'auto'\n    };\n    \n    // Default position (bottom center)\n    styles['bottom'] = config?.position?.bottom || '4px';\n    styles['left'] = config?.position?.left || '50%';\n    styles['transform'] = 'translateX(-50%)';\n    \n    // Override with custom positions if provided\n    if (config?.position) {\n      Object.keys(config.position).forEach(key => {\n        const position = config.position as Record<string, string | undefined>;\n        if (position[key]) {\n          styles[key] = position[key] as string;\n        }\n      });\n      \n      // Handle transformations for centered positioning\n      if (config.position.left === '50%' && !styles['transform']) {\n        styles['transform'] = 'translateX(-50%)';\n      } else if (config.position.top === '50%' && !styles['transform']) {\n        styles['transform'] = 'translateY(-50%)';\n      }\n    }\n    \n    // Apply custom container styles if provided\n    if (config?.container) {\n      Object.assign(styles, config.container);\n    }\n    \n    return styles;\n  }\n\n  /**\n   * Get navigation icons based on orientation\n   * @param isVertical Whether carousel is vertical\n   * @param icons Custom icon configuration\n   * @returns Previous and next icons\n   */\n  getNavigationIcons(isVertical: boolean, icons?: any): { prev: string; next: string } {\n    const defaultIcons = {\n      horizontal: {\n        prev: CAROUSEL_DEFAULTS.NAVIGATION_PREV_ICON,\n        next: CAROUSEL_DEFAULTS.NAVIGATION_NEXT_ICON\n      },\n      vertical: {\n        prev: CAROUSEL_DEFAULTS.NAVIGATION_PREV_ICON, \n        next: CAROUSEL_DEFAULTS.NAVIGATION_NEXT_ICON\n      }\n    };\n\n    const customIcons = icons || {};\n    const verticalIcons = customIcons.vertical || {};\n\n    return {\n      prev: isVertical\n        ? (verticalIcons.prev || defaultIcons.vertical.prev)\n        : (customIcons.prev || defaultIcons.horizontal.prev),\n      next: isVertical\n        ? (verticalIcons.next || defaultIcons.vertical.next)\n        : (customIcons.next || defaultIcons.horizontal.next)\n    };\n  }\n\n  /**\n   * Parse time string to milliseconds\n   * @param time Time string (e.g., '300ms', '0.5s')\n   * @returns Time in milliseconds\n   */\n  parseTimeToMs(time: string): number {\n    if (!time) return 300; // Default 300ms\n    \n    if (time.endsWith('ms')) {\n      return parseInt(time.slice(0, -2), 10);\n    }\n    \n    if (time.endsWith('s')) {\n      return parseFloat(time.slice(0, -1)) * 1000;\n    }\n    \n    return parseInt(time, 10);\n  }\n} "]}