rating.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, EventEmitter, forwardRef, Input, Output, TemplateRef, ViewEncapsulation } from '@angular/core';
  6. import { NgbRatingConfig } from './rating-config';
  7. import { getValueInRange } from '../util/util';
  8. import { Key } from '../util/key';
  9. import { NG_VALUE_ACCESSOR } from '@angular/forms';
  10. /**
  11. * The context for the custom star display template defined in the `starTemplate`.
  12. * @record
  13. */
  14. export function StarTemplateContext() { }
  15. if (false) {
  16. /**
  17. * The star fill percentage, an integer in the `[0, 100]` range.
  18. * @type {?}
  19. */
  20. StarTemplateContext.prototype.fill;
  21. /**
  22. * Index of the star, starts with `0`.
  23. * @type {?}
  24. */
  25. StarTemplateContext.prototype.index;
  26. }
  27. /** @type {?} */
  28. const NGB_RATING_VALUE_ACCESSOR = {
  29. provide: NG_VALUE_ACCESSOR,
  30. useExisting: forwardRef((/**
  31. * @return {?}
  32. */
  33. () => NgbRating)),
  34. multi: true
  35. };
  36. /**
  37. * A directive that helps visualising and interacting with a star rating bar.
  38. */
  39. export class NgbRating {
  40. /**
  41. * @param {?} config
  42. * @param {?} _changeDetectorRef
  43. */
  44. constructor(config, _changeDetectorRef) {
  45. this._changeDetectorRef = _changeDetectorRef;
  46. this.contexts = [];
  47. this.disabled = false;
  48. /**
  49. * An event emitted when the user is hovering over a given rating.
  50. *
  51. * Event payload equals to the rating being hovered over.
  52. */
  53. this.hover = new EventEmitter();
  54. /**
  55. * An event emitted when the user stops hovering over a given rating.
  56. *
  57. * Event payload equals to the rating of the last item being hovered over.
  58. */
  59. this.leave = new EventEmitter();
  60. /**
  61. * An event emitted when the user selects a new rating.
  62. *
  63. * Event payload equals to the newly selected rating.
  64. */
  65. this.rateChange = new EventEmitter(true);
  66. this.onChange = (/**
  67. * @param {?} _
  68. * @return {?}
  69. */
  70. (_) => { });
  71. this.onTouched = (/**
  72. * @return {?}
  73. */
  74. () => { });
  75. this.max = config.max;
  76. this.readonly = config.readonly;
  77. }
  78. /**
  79. * @return {?}
  80. */
  81. ariaValueText() { return `${this.nextRate} out of ${this.max}`; }
  82. /**
  83. * @param {?} value
  84. * @return {?}
  85. */
  86. enter(value) {
  87. if (!this.readonly && !this.disabled) {
  88. this._updateState(value);
  89. }
  90. this.hover.emit(value);
  91. }
  92. /**
  93. * @return {?}
  94. */
  95. handleBlur() { this.onTouched(); }
  96. /**
  97. * @param {?} value
  98. * @return {?}
  99. */
  100. handleClick(value) { this.update(this.resettable && this.rate === value ? 0 : value); }
  101. /**
  102. * @param {?} event
  103. * @return {?}
  104. */
  105. handleKeyDown(event) {
  106. // tslint:disable-next-line:deprecation
  107. switch (event.which) {
  108. case Key.ArrowDown:
  109. case Key.ArrowLeft:
  110. this.update(this.rate - 1);
  111. break;
  112. case Key.ArrowUp:
  113. case Key.ArrowRight:
  114. this.update(this.rate + 1);
  115. break;
  116. case Key.Home:
  117. this.update(0);
  118. break;
  119. case Key.End:
  120. this.update(this.max);
  121. break;
  122. default:
  123. return;
  124. }
  125. // note 'return' in default case
  126. event.preventDefault();
  127. }
  128. /**
  129. * @param {?} changes
  130. * @return {?}
  131. */
  132. ngOnChanges(changes) {
  133. if (changes['rate']) {
  134. this.update(this.rate);
  135. }
  136. }
  137. /**
  138. * @return {?}
  139. */
  140. ngOnInit() {
  141. this.contexts = Array.from({ length: this.max }, (/**
  142. * @param {?} v
  143. * @param {?} k
  144. * @return {?}
  145. */
  146. (v, k) => ({ fill: 0, index: k })));
  147. this._updateState(this.rate);
  148. }
  149. /**
  150. * @param {?} fn
  151. * @return {?}
  152. */
  153. registerOnChange(fn) { this.onChange = fn; }
  154. /**
  155. * @param {?} fn
  156. * @return {?}
  157. */
  158. registerOnTouched(fn) { this.onTouched = fn; }
  159. /**
  160. * @return {?}
  161. */
  162. reset() {
  163. this.leave.emit(this.nextRate);
  164. this._updateState(this.rate);
  165. }
  166. /**
  167. * @param {?} isDisabled
  168. * @return {?}
  169. */
  170. setDisabledState(isDisabled) { this.disabled = isDisabled; }
  171. /**
  172. * @param {?} value
  173. * @param {?=} internalChange
  174. * @return {?}
  175. */
  176. update(value, internalChange = true) {
  177. /** @type {?} */
  178. const newRate = getValueInRange(value, this.max, 0);
  179. if (!this.readonly && !this.disabled && this.rate !== newRate) {
  180. this.rate = newRate;
  181. this.rateChange.emit(this.rate);
  182. }
  183. if (internalChange) {
  184. this.onChange(this.rate);
  185. this.onTouched();
  186. }
  187. this._updateState(this.rate);
  188. }
  189. /**
  190. * @param {?} value
  191. * @return {?}
  192. */
  193. writeValue(value) {
  194. this.update(value, false);
  195. this._changeDetectorRef.markForCheck();
  196. }
  197. /**
  198. * @private
  199. * @param {?} index
  200. * @return {?}
  201. */
  202. _getFillValue(index) {
  203. /** @type {?} */
  204. const diff = this.nextRate - index;
  205. if (diff >= 1) {
  206. return 100;
  207. }
  208. if (diff < 1 && diff > 0) {
  209. return parseInt((diff * 100).toFixed(2), 10);
  210. }
  211. return 0;
  212. }
  213. /**
  214. * @private
  215. * @param {?} nextValue
  216. * @return {?}
  217. */
  218. _updateState(nextValue) {
  219. this.nextRate = nextValue;
  220. this.contexts.forEach((/**
  221. * @param {?} context
  222. * @param {?} index
  223. * @return {?}
  224. */
  225. (context, index) => context.fill = this._getFillValue(index)));
  226. }
  227. }
  228. NgbRating.decorators = [
  229. { type: Component, args: [{
  230. selector: 'ngb-rating',
  231. changeDetection: ChangeDetectionStrategy.OnPush,
  232. encapsulation: ViewEncapsulation.None,
  233. host: {
  234. 'class': 'd-inline-flex',
  235. 'tabindex': '0',
  236. 'role': 'slider',
  237. 'aria-valuemin': '0',
  238. '[attr.aria-valuemax]': 'max',
  239. '[attr.aria-valuenow]': 'nextRate',
  240. '[attr.aria-valuetext]': 'ariaValueText()',
  241. '[attr.aria-disabled]': 'readonly ? true : null',
  242. '(blur)': 'handleBlur()',
  243. '(keydown)': 'handleKeyDown($event)',
  244. '(mouseleave)': 'reset()'
  245. },
  246. template: `
  247. <ng-template #t let-fill="fill">{{ fill === 100 ? '&#9733;' : '&#9734;' }}</ng-template>
  248. <ng-template ngFor [ngForOf]="contexts" let-index="index">
  249. <span class="sr-only">({{ index < nextRate ? '*' : ' ' }})</span>
  250. <span (mouseenter)="enter(index + 1)" (click)="handleClick(index + 1)" [style.cursor]="readonly || disabled ? 'default' : 'pointer'">
  251. <ng-template [ngTemplateOutlet]="starTemplate || starTemplateFromContent || t" [ngTemplateOutletContext]="contexts[index]">
  252. </ng-template>
  253. </span>
  254. </ng-template>
  255. `,
  256. providers: [NGB_RATING_VALUE_ACCESSOR]
  257. }] }
  258. ];
  259. /** @nocollapse */
  260. NgbRating.ctorParameters = () => [
  261. { type: NgbRatingConfig },
  262. { type: ChangeDetectorRef }
  263. ];
  264. NgbRating.propDecorators = {
  265. max: [{ type: Input }],
  266. rate: [{ type: Input }],
  267. readonly: [{ type: Input }],
  268. resettable: [{ type: Input }],
  269. starTemplate: [{ type: Input }],
  270. starTemplateFromContent: [{ type: ContentChild, args: [TemplateRef, { static: false },] }],
  271. hover: [{ type: Output }],
  272. leave: [{ type: Output }],
  273. rateChange: [{ type: Output }]
  274. };
  275. if (false) {
  276. /** @type {?} */
  277. NgbRating.prototype.contexts;
  278. /** @type {?} */
  279. NgbRating.prototype.disabled;
  280. /** @type {?} */
  281. NgbRating.prototype.nextRate;
  282. /**
  283. * The maximal rating that can be given.
  284. * @type {?}
  285. */
  286. NgbRating.prototype.max;
  287. /**
  288. * The current rating. Could be a decimal value like `3.75`.
  289. * @type {?}
  290. */
  291. NgbRating.prototype.rate;
  292. /**
  293. * If `true`, the rating can't be changed.
  294. * @type {?}
  295. */
  296. NgbRating.prototype.readonly;
  297. /**
  298. * If `true`, the rating can be reset to `0` by mouse clicking currently set rating.
  299. * @type {?}
  300. */
  301. NgbRating.prototype.resettable;
  302. /**
  303. * The template to override the way each star is displayed.
  304. *
  305. * Alternatively put an `<ng-template>` as the only child of your `<ngb-rating>` element
  306. * @type {?}
  307. */
  308. NgbRating.prototype.starTemplate;
  309. /** @type {?} */
  310. NgbRating.prototype.starTemplateFromContent;
  311. /**
  312. * An event emitted when the user is hovering over a given rating.
  313. *
  314. * Event payload equals to the rating being hovered over.
  315. * @type {?}
  316. */
  317. NgbRating.prototype.hover;
  318. /**
  319. * An event emitted when the user stops hovering over a given rating.
  320. *
  321. * Event payload equals to the rating of the last item being hovered over.
  322. * @type {?}
  323. */
  324. NgbRating.prototype.leave;
  325. /**
  326. * An event emitted when the user selects a new rating.
  327. *
  328. * Event payload equals to the newly selected rating.
  329. * @type {?}
  330. */
  331. NgbRating.prototype.rateChange;
  332. /** @type {?} */
  333. NgbRating.prototype.onChange;
  334. /** @type {?} */
  335. NgbRating.prototype.onTouched;
  336. /**
  337. * @type {?}
  338. * @private
  339. */
  340. NgbRating.prototype._changeDetectorRef;
  341. }
  342. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rating.js","sourceRoot":"ng://@ng-bootstrap/ng-bootstrap/","sources":["rating/rating.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChild,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n  ViewEncapsulation\n} from '@angular/core';\nimport {NgbRatingConfig} from './rating-config';\nimport {getValueInRange} from '../util/util';\nimport {Key} from '../util/key';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\n\n/**\n * The context for the custom star display template defined in the `starTemplate`.\n */\nexport interface StarTemplateContext {\n  /**\n   * The star fill percentage, an integer in the `[0, 100]` range.\n   */\n  fill: number;\n\n  /**\n   * Index of the star, starts with `0`.\n   */\n  index: number;\n}\n\nconst NGB_RATING_VALUE_ACCESSOR = {\n  provide: NG_VALUE_ACCESSOR,\n  useExisting: forwardRef(() => NgbRating),\n  multi: true\n};\n\n/**\n * A directive that helps visualising and interacting with a star rating bar.\n */\n@Component({\n  selector: 'ngb-rating',\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  host: {\n    'class': 'd-inline-flex',\n    'tabindex': '0',\n    'role': 'slider',\n    'aria-valuemin': '0',\n    '[attr.aria-valuemax]': 'max',\n    '[attr.aria-valuenow]': 'nextRate',\n    '[attr.aria-valuetext]': 'ariaValueText()',\n    '[attr.aria-disabled]': 'readonly ? true : null',\n    '(blur)': 'handleBlur()',\n    '(keydown)': 'handleKeyDown($event)',\n    '(mouseleave)': 'reset()'\n  },\n  template: `\n    <ng-template #t let-fill=\"fill\">{{ fill === 100 ? '&#9733;' : '&#9734;' }}</ng-template>\n    <ng-template ngFor [ngForOf]=\"contexts\" let-index=\"index\">\n      <span class=\"sr-only\">({{ index < nextRate ? '*' : ' ' }})</span>\n      <span (mouseenter)=\"enter(index + 1)\" (click)=\"handleClick(index + 1)\" [style.cursor]=\"readonly || disabled ? 'default' : 'pointer'\">\n        <ng-template [ngTemplateOutlet]=\"starTemplate || starTemplateFromContent || t\" [ngTemplateOutletContext]=\"contexts[index]\">\n        </ng-template>\n      </span>\n    </ng-template>\n  `,\n  providers: [NGB_RATING_VALUE_ACCESSOR]\n})\nexport class NgbRating implements ControlValueAccessor,\n    OnInit, OnChanges {\n  contexts: StarTemplateContext[] = [];\n  disabled = false;\n  nextRate: number;\n\n\n  /**\n   * The maximal rating that can be given.\n   */\n  @Input() max: number;\n\n  /**\n   * The current rating. Could be a decimal value like `3.75`.\n   */\n  @Input() rate: number;\n\n  /**\n   * If `true`, the rating can't be changed.\n   */\n  @Input() readonly: boolean;\n\n  /**\n   * If `true`, the rating can be reset to `0` by mouse clicking currently set rating.\n   */\n  @Input() resettable: boolean;\n\n  /**\n   * The template to override the way each star is displayed.\n   *\n   * Alternatively put an `<ng-template>` as the only child of your `<ngb-rating>` element\n   */\n  @Input() starTemplate: TemplateRef<StarTemplateContext>;\n  @ContentChild(TemplateRef, {static: false}) starTemplateFromContent: TemplateRef<StarTemplateContext>;\n\n  /**\n   * An event emitted when the user is hovering over a given rating.\n   *\n   * Event payload equals to the rating being hovered over.\n   */\n  @Output() hover = new EventEmitter<number>();\n\n  /**\n   * An event emitted when the user stops hovering over a given rating.\n   *\n   * Event payload equals to the rating of the last item being hovered over.\n   */\n  @Output() leave = new EventEmitter<number>();\n\n  /**\n   * An event emitted when the user selects a new rating.\n   *\n   * Event payload equals to the newly selected rating.\n   */\n  @Output() rateChange = new EventEmitter<number>(true);\n\n  onChange = (_: any) => {};\n  onTouched = () => {};\n\n  constructor(config: NgbRatingConfig, private _changeDetectorRef: ChangeDetectorRef) {\n    this.max = config.max;\n    this.readonly = config.readonly;\n  }\n\n  ariaValueText() { return `${this.nextRate} out of ${this.max}`; }\n\n  enter(value: number): void {\n    if (!this.readonly && !this.disabled) {\n      this._updateState(value);\n    }\n    this.hover.emit(value);\n  }\n\n  handleBlur() { this.onTouched(); }\n\n  handleClick(value: number) { this.update(this.resettable && this.rate === value ? 0 : value); }\n\n  handleKeyDown(event: KeyboardEvent) {\n    // tslint:disable-next-line:deprecation\n    switch (event.which) {\n      case Key.ArrowDown:\n      case Key.ArrowLeft:\n        this.update(this.rate - 1);\n        break;\n      case Key.ArrowUp:\n      case Key.ArrowRight:\n        this.update(this.rate + 1);\n        break;\n      case Key.Home:\n        this.update(0);\n        break;\n      case Key.End:\n        this.update(this.max);\n        break;\n      default:\n        return;\n    }\n\n    // note 'return' in default case\n    event.preventDefault();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['rate']) {\n      this.update(this.rate);\n    }\n  }\n\n  ngOnInit(): void {\n    this.contexts = Array.from({length: this.max}, (v, k) => ({fill: 0, index: k}));\n    this._updateState(this.rate);\n  }\n\n  registerOnChange(fn: (value: any) => any): void { this.onChange = fn; }\n\n  registerOnTouched(fn: () => any): void { this.onTouched = fn; }\n\n  reset(): void {\n    this.leave.emit(this.nextRate);\n    this._updateState(this.rate);\n  }\n\n  setDisabledState(isDisabled: boolean) { this.disabled = isDisabled; }\n\n  update(value: number, internalChange = true): void {\n    const newRate = getValueInRange(value, this.max, 0);\n    if (!this.readonly && !this.disabled && this.rate !== newRate) {\n      this.rate = newRate;\n      this.rateChange.emit(this.rate);\n    }\n    if (internalChange) {\n      this.onChange(this.rate);\n      this.onTouched();\n    }\n    this._updateState(this.rate);\n  }\n\n  writeValue(value) {\n    this.update(value, false);\n    this._changeDetectorRef.markForCheck();\n  }\n\n  private _getFillValue(index: number): number {\n    const diff = this.nextRate - index;\n\n    if (diff >= 1) {\n      return 100;\n    }\n    if (diff < 1 && diff > 0) {\n      return parseInt((diff * 100).toFixed(2), 10);\n    }\n\n    return 0;\n  }\n\n  private _updateState(nextValue: number) {\n    this.nextRate = nextValue;\n    this.contexts.forEach((context, index) => context.fill = this._getFillValue(index));\n  }\n}\n"]}