UNPKG

ng-rating-pro

Version:

A powerful and customizable Angular rating component that allows full and half ratings with support for read-only mode, dynamic scaling, and SVG customization.<br/>Perfect for use in reviews, feedback forms, and rating-based applications.

148 lines 21.2 kB
import { Component, ContentChildren, EventEmitter, HostListener, Input, Output, ViewChild, ViewContainerRef, } from '@angular/core'; import { CustomRatingDirective } from './rating-icon/custom-rating.directive'; import { StarIconComponent } from './rating-icon/star-icon.component'; import { HeartIconComponent } from './rating-icon/heart-icon.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export var State; (function (State) { State["Empty"] = "empty"; State["Half"] = "half"; State["Full"] = "full"; })(State || (State = {})); export class NgRatingProComponent { constructor() { this.scale = 5; this.rating = 0; this.allowHalf = true; this.size = 20; // Default size this.spacing = 8; // Spacing between stars in viewBox units this.readonly = false; this.iconName = 'star'; this.ratingChange = new EventEmitter(); this.states = []; this.starWidth = 24; this.starHeight = 24; this.containerWidth = 0; this.totalWidth = 0; } ngOnInit() { this.updateStates(this.allowHalf); } ngAfterContentInit() { if (this.ratingDirectives.length !== 3) { this.loadDynamicComponent(); } else { this.ratingDirectives.forEach((directive) => { directive.updateRating(this.iconName); }); this.starHeight = this.ratingDirectives.first.iconViewBox[3]; this.starWidth = this.ratingDirectives.first.iconViewBox[2]; } this.updateDimensions(); } loadDynamicComponent() { const component = this.getComponent(); this.dynamicContainer.clear(); this.dynamicContainer.createComponent(component); this.iconName = component.iconName; } getComponent() { switch (this.iconName) { case 'star': return StarIconComponent; case 'heart': return HeartIconComponent; default: return StarIconComponent; } } updateDimensions() { // Calculate total width in viewBox units this.totalWidth = this.starWidth * this.scale + this.spacing * (this.scale - 1); // Calculate container width maintaining aspect ratio const aspectRatio = this.totalWidth / this.starHeight; this.containerWidth = this.size * aspectRatio; } getStarPosition(index) { return index * (this.starWidth + this.spacing); } onClick(event) { if (this.readonly) return; if (event.target instanceof SVGElement && +event.target.id > 0) { const clickedIndex = +event.target.id - 1; this.toggleRating(clickedIndex); this.updateStates(this.allowHalf); } event.stopPropagation(); } toggleRating(index) { const isFull = this.rating === index + 1; const isHalf = this.rating === index + 0.5; if (this.allowHalf) { if (isFull) { this.rating = index; } else if (isHalf) { this.rating = index + 1; } else { this.rating = index + 0.5; } } else { if (isFull) { this.rating = index; } else { this.rating = index + 1; } } } updateStates(allowHalf) { this.ratingChange.emit(this.rating); this.rating = Math.round(this.rating * 2) / 2; this.states = Array.from({ length: this.scale }, (_, i) => { if (i < Math.floor(this.rating)) return State.Full; if (i === Math.floor(this.rating) && this.rating % 1 !== 0 && allowHalf) return State.Half; return State.Empty; }); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgRatingProComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NgRatingProComponent, selector: "ngRatingPro", inputs: { scale: "scale", rating: "rating", allowHalf: "allowHalf", size: "size", spacing: "spacing", readonly: "readonly", iconName: "iconName" }, outputs: { ratingChange: "ratingChange" }, host: { listeners: { "click": "onClick($event)" } }, queries: [{ propertyName: "ratingDirectives", predicate: CustomRatingDirective }], viewQueries: [{ propertyName: "dynamicContainer", first: true, predicate: ["dynamicContainer"], descendants: true, read: ViewContainerRef, static: true }], ngImport: i0, template: "<div\n [style.width]=\"containerWidth + 'px'\"\n [style.height]=\"size + 'px'\"\n class=\"rating-container\"\n>\n <!-- Define symbols -->\n <svg style=\"display: none\">\n <ng-container #dynamicContainer></ng-container>\n <ng-content select=\"half\"></ng-content>\n <ng-content select=\"full\"></ng-content>\n <ng-content select=\"empty\"></ng-content>\n </svg>\n\n <!-- Rating display -->\n <svg\n [attr.viewBox]=\"'0 0 ' + totalWidth + ' ' + starHeight\"\n [style.width]=\"'100%'\"\n [style.height]=\"'100%'\"\n aria-hidden=\"true\"\n focusable=\"false\"\n class=\"rating\"\n >\n <ng-container *ngFor=\"let state of states; let i = index\">\n <use\n [id]=\"i + 1\"\n [attr.x]=\"getStarPosition(i)\"\n [attr.y]=\"0\"\n [attr.width]=\"starWidth\"\n [attr.height]=\"starHeight\"\n [attr.xlink:href]=\"'#' + iconName + '-' + state\"\n [ngStyle]=\"{ cursor: !readonly ? 'pointer' : 'default' }\"\n />\n </ng-container>\n </svg>\n</div>\n", styles: [":host{-webkit-user-select:none;user-select:none}.rating-container{display:inline-block;position:relative}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgRatingProComponent, decorators: [{ type: Component, args: [{ selector: 'ngRatingPro', template: "<div\n [style.width]=\"containerWidth + 'px'\"\n [style.height]=\"size + 'px'\"\n class=\"rating-container\"\n>\n <!-- Define symbols -->\n <svg style=\"display: none\">\n <ng-container #dynamicContainer></ng-container>\n <ng-content select=\"half\"></ng-content>\n <ng-content select=\"full\"></ng-content>\n <ng-content select=\"empty\"></ng-content>\n </svg>\n\n <!-- Rating display -->\n <svg\n [attr.viewBox]=\"'0 0 ' + totalWidth + ' ' + starHeight\"\n [style.width]=\"'100%'\"\n [style.height]=\"'100%'\"\n aria-hidden=\"true\"\n focusable=\"false\"\n class=\"rating\"\n >\n <ng-container *ngFor=\"let state of states; let i = index\">\n <use\n [id]=\"i + 1\"\n [attr.x]=\"getStarPosition(i)\"\n [attr.y]=\"0\"\n [attr.width]=\"starWidth\"\n [attr.height]=\"starHeight\"\n [attr.xlink:href]=\"'#' + iconName + '-' + state\"\n [ngStyle]=\"{ cursor: !readonly ? 'pointer' : 'default' }\"\n />\n </ng-container>\n </svg>\n</div>\n", styles: [":host{-webkit-user-select:none;user-select:none}.rating-container{display:inline-block;position:relative}\n"] }] }], propDecorators: { scale: [{ type: Input }], rating: [{ type: Input }], allowHalf: [{ type: Input }], size: [{ type: Input }], spacing: [{ type: Input }], readonly: [{ type: Input }], iconName: [{ type: Input }], ratingDirectives: [{ type: ContentChildren, args: [CustomRatingDirective] }], dynamicContainer: [{ type: ViewChild, args: ['dynamicContainer', { read: ViewContainerRef, static: true }] }], ratingChange: [{ type: Output }], onClick: [{ type: HostListener, args: ['click', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-rating-pro.component.js","sourceRoot":"","sources":["../../../../projects/ng-rating-pro/src/lib/ng-rating-pro.component.ts","../../../../projects/ng-rating-pro/src/lib/ng-rating-pro.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ComponentFactoryResolver,\n  ContentChildren,\n  EventEmitter,\n  HostListener,\n  Input,\n  OnInit,\n  Output,\n  QueryList,\n  ViewChild,\n  ViewContainerRef,\n} from '@angular/core';\nimport { CustomRatingDirective } from './rating-icon/custom-rating.directive';\nimport { StarIconComponent } from './rating-icon/star-icon.component';\nimport { HeartIconComponent } from './rating-icon/heart-icon.component';\n\nexport enum State {\n  Empty = 'empty',\n  Half = 'half',\n  Full = 'full',\n}\n\n@Component({\n  selector: 'ngRatingPro',\n  templateUrl: './ng-rating-pro.component.html',\n  styleUrls: ['./ng-rating-pro.component.css'],\n})\nexport class NgRatingProComponent implements OnInit {\n  @Input() scale: number = 5;\n  @Input() rating: number = 0;\n  @Input() allowHalf: boolean = true;\n  @Input() size: number = 20; // Default size\n  @Input() spacing: number = 8; // Spacing between stars in viewBox units\n  @Input() readonly: boolean = false;\n  @Input() iconName: string = 'star';\n\n  @ContentChildren(CustomRatingDirective)\n  ratingDirectives!: QueryList<CustomRatingDirective>;\n  @ViewChild('dynamicContainer', { read: ViewContainerRef, static: true })\n  dynamicContainer!: ViewContainerRef;\n\n  @Output() ratingChange: EventEmitter<number> = new EventEmitter<number>();\n\n  public states: State[] = [];\n\n  starWidth: number = 24;\n  starHeight: number = 24;\n  containerWidth: number = 0;\n  totalWidth: number = 0;\n\n  ngOnInit() {\n    this.updateStates(this.allowHalf);\n  }\n\n  ngAfterContentInit() {\n    if (this.ratingDirectives.length !== 3) {\n      this.loadDynamicComponent();\n    } else {\n      this.ratingDirectives.forEach((directive) => {\n        directive.updateRating(this.iconName);\n      });\n      this.starHeight = this.ratingDirectives.first.iconViewBox[3];\n      this.starWidth = this.ratingDirectives.first.iconViewBox[2];\n    }\n    this.updateDimensions();\n  }\n\n  private loadDynamicComponent() {\n    const component = this.getComponent();\n    this.dynamicContainer.clear();\n    this.dynamicContainer.createComponent(component);\n    this.iconName = component.iconName;\n  }\n\n  private getComponent() {\n    switch (this.iconName) {\n      case 'star':\n        return StarIconComponent;\n      case 'heart':\n        return HeartIconComponent;\n      default:\n        return StarIconComponent;\n    }\n  }\n\n  updateDimensions() {\n    // Calculate total width in viewBox units\n    this.totalWidth =\n      this.starWidth * this.scale + this.spacing * (this.scale - 1);\n\n    // Calculate container width maintaining aspect ratio\n    const aspectRatio = this.totalWidth / this.starHeight;\n    this.containerWidth = this.size * aspectRatio;\n  }\n\n  getStarPosition(index: number): number {\n    return index * (this.starWidth + this.spacing);\n  }\n\n  @HostListener('click', ['$event'])\n  onClick(event: MouseEvent) {\n    if (this.readonly) return;\n    if (event.target instanceof SVGElement && +event.target.id > 0) {\n      const clickedIndex = +event.target.id - 1;\n      this.toggleRating(clickedIndex);\n      this.updateStates(this.allowHalf);\n    }\n    event.stopPropagation();\n  }\n\n  private toggleRating(index: number) {\n    const isFull = this.rating === index + 1;\n    const isHalf = this.rating === index + 0.5;\n\n    if (this.allowHalf) {\n      if (isFull) {\n        this.rating = index;\n      } else if (isHalf) {\n        this.rating = index + 1;\n      } else {\n        this.rating = index + 0.5;\n      }\n    } else {\n      if (isFull) {\n        this.rating = index;\n      } else {\n        this.rating = index + 1;\n      }\n    }\n  }\n\n  public updateStates(allowHalf: boolean) {\n    this.ratingChange.emit(this.rating);\n    this.rating = Math.round(this.rating * 2) / 2;\n    this.states = Array.from({ length: this.scale }, (_, i) => {\n      if (i < Math.floor(this.rating)) return State.Full;\n      if (i === Math.floor(this.rating) && this.rating % 1 !== 0 && allowHalf)\n        return State.Half;\n      return State.Empty;\n    });\n  }\n}\n","<div\n  [style.width]=\"containerWidth + 'px'\"\n  [style.height]=\"size + 'px'\"\n  class=\"rating-container\"\n>\n  <!-- Define symbols -->\n  <svg style=\"display: none\">\n    <ng-container #dynamicContainer></ng-container>\n    <ng-content select=\"half\"></ng-content>\n    <ng-content select=\"full\"></ng-content>\n    <ng-content select=\"empty\"></ng-content>\n  </svg>\n\n  <!-- Rating display -->\n  <svg\n    [attr.viewBox]=\"'0 0 ' + totalWidth + ' ' + starHeight\"\n    [style.width]=\"'100%'\"\n    [style.height]=\"'100%'\"\n    aria-hidden=\"true\"\n    focusable=\"false\"\n    class=\"rating\"\n  >\n    <ng-container *ngFor=\"let state of states; let i = index\">\n      <use\n        [id]=\"i + 1\"\n        [attr.x]=\"getStarPosition(i)\"\n        [attr.y]=\"0\"\n        [attr.width]=\"starWidth\"\n        [attr.height]=\"starHeight\"\n        [attr.xlink:href]=\"'#' + iconName + '-' + state\"\n        [ngStyle]=\"{ cursor: !readonly ? 'pointer' : 'default' }\"\n      />\n    </ng-container>\n  </svg>\n</div>\n"]}