positioning.service.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { Injectable, ElementRef, RendererFactory2, Inject, PLATFORM_ID, NgZone } from '@angular/core';
  6. import { isPlatformBrowser } from '@angular/common';
  7. import { positionElements } from './ng-positioning';
  8. import { fromEvent, merge, of, animationFrameScheduler, Subject } from 'rxjs';
  9. /**
  10. * @record
  11. */
  12. export function PositioningOptions() { }
  13. if (false) {
  14. /**
  15. * The DOM element, ElementRef, or a selector string of an element which will be moved
  16. * @type {?|undefined}
  17. */
  18. PositioningOptions.prototype.element;
  19. /**
  20. * The DOM element, ElementRef, or a selector string of an element which the element will be attached to
  21. * @type {?|undefined}
  22. */
  23. PositioningOptions.prototype.target;
  24. /**
  25. * A string of the form 'vert-attachment horiz-attachment' or 'placement'
  26. * - placement can be "top", "bottom", "left", "right"
  27. * not yet supported:
  28. * - vert-attachment can be any of 'top', 'middle', 'bottom'
  29. * - horiz-attachment can be any of 'left', 'center', 'right'
  30. * @type {?|undefined}
  31. */
  32. PositioningOptions.prototype.attachment;
  33. /**
  34. * A string similar to `attachment`. The one difference is that, if it's not provided,
  35. * `targetAttachment` will assume the mirror image of `attachment`.
  36. * @type {?|undefined}
  37. */
  38. PositioningOptions.prototype.targetAttachment;
  39. /**
  40. * A string of the form 'vert-offset horiz-offset'
  41. * - vert-offset and horiz-offset can be of the form "20px" or "55%"
  42. * @type {?|undefined}
  43. */
  44. PositioningOptions.prototype.offset;
  45. /**
  46. * A string similar to `offset`, but referring to the offset of the target
  47. * @type {?|undefined}
  48. */
  49. PositioningOptions.prototype.targetOffset;
  50. /**
  51. * If true component will be attached to body
  52. * @type {?|undefined}
  53. */
  54. PositioningOptions.prototype.appendToBody;
  55. }
  56. var PositioningService = /** @class */ (function () {
  57. function PositioningService(ngZone, rendererFactory, platformId) {
  58. var _this = this;
  59. this.update$$ = new Subject();
  60. this.positionElements = new Map();
  61. this.isDisabled = false;
  62. if (isPlatformBrowser(platformId)) {
  63. ngZone.runOutsideAngular((/**
  64. * @return {?}
  65. */
  66. function () {
  67. _this.triggerEvent$ = merge(fromEvent(window, 'scroll', { passive: true }), fromEvent(window, 'resize', { passive: true }),
  68. /* tslint:disable-next-line: deprecation */
  69. of(0, animationFrameScheduler), _this.update$$);
  70. _this.triggerEvent$.subscribe((/**
  71. * @return {?}
  72. */
  73. function () {
  74. if (_this.isDisabled) {
  75. return;
  76. }
  77. _this.positionElements
  78. /* tslint:disable-next-line: no-any */
  79. .forEach((/**
  80. * @param {?} positionElement
  81. * @return {?}
  82. */
  83. function (positionElement) {
  84. positionElements(_getHtmlElement(positionElement.target), _getHtmlElement(positionElement.element), positionElement.attachment, positionElement.appendToBody, _this.options, rendererFactory.createRenderer(null, null));
  85. }));
  86. }));
  87. }));
  88. }
  89. }
  90. /**
  91. * @param {?} options
  92. * @return {?}
  93. */
  94. PositioningService.prototype.position = /**
  95. * @param {?} options
  96. * @return {?}
  97. */
  98. function (options) {
  99. this.addPositionElement(options);
  100. };
  101. Object.defineProperty(PositioningService.prototype, "event$", {
  102. get: /**
  103. * @return {?}
  104. */
  105. function () {
  106. return this.triggerEvent$;
  107. },
  108. enumerable: true,
  109. configurable: true
  110. });
  111. /**
  112. * @return {?}
  113. */
  114. PositioningService.prototype.disable = /**
  115. * @return {?}
  116. */
  117. function () {
  118. this.isDisabled = true;
  119. };
  120. /**
  121. * @return {?}
  122. */
  123. PositioningService.prototype.enable = /**
  124. * @return {?}
  125. */
  126. function () {
  127. this.isDisabled = false;
  128. };
  129. /**
  130. * @param {?} options
  131. * @return {?}
  132. */
  133. PositioningService.prototype.addPositionElement = /**
  134. * @param {?} options
  135. * @return {?}
  136. */
  137. function (options) {
  138. this.positionElements.set(_getHtmlElement(options.element), options);
  139. };
  140. /**
  141. * @return {?}
  142. */
  143. PositioningService.prototype.calcPosition = /**
  144. * @return {?}
  145. */
  146. function () {
  147. this.update$$.next();
  148. };
  149. /**
  150. * @param {?} elRef
  151. * @return {?}
  152. */
  153. PositioningService.prototype.deletePositionElement = /**
  154. * @param {?} elRef
  155. * @return {?}
  156. */
  157. function (elRef) {
  158. this.positionElements.delete(_getHtmlElement(elRef));
  159. };
  160. /**
  161. * @param {?} options
  162. * @return {?}
  163. */
  164. PositioningService.prototype.setOptions = /**
  165. * @param {?} options
  166. * @return {?}
  167. */
  168. function (options) {
  169. this.options = options;
  170. };
  171. PositioningService.decorators = [
  172. { type: Injectable }
  173. ];
  174. /** @nocollapse */
  175. PositioningService.ctorParameters = function () { return [
  176. { type: NgZone },
  177. { type: RendererFactory2 },
  178. { type: Number, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
  179. ]; };
  180. return PositioningService;
  181. }());
  182. export { PositioningService };
  183. if (false) {
  184. /**
  185. * @type {?}
  186. * @private
  187. */
  188. PositioningService.prototype.options;
  189. /**
  190. * @type {?}
  191. * @private
  192. */
  193. PositioningService.prototype.update$$;
  194. /**
  195. * @type {?}
  196. * @private
  197. */
  198. PositioningService.prototype.positionElements;
  199. /**
  200. * @type {?}
  201. * @private
  202. */
  203. PositioningService.prototype.triggerEvent$;
  204. /**
  205. * @type {?}
  206. * @private
  207. */
  208. PositioningService.prototype.isDisabled;
  209. }
  210. /**
  211. * @param {?} element
  212. * @return {?}
  213. */
  214. function _getHtmlElement(element) {
  215. // it means that we got a selector
  216. if (typeof element === 'string') {
  217. return document.querySelector(element);
  218. }
  219. if (element instanceof ElementRef) {
  220. return element.nativeElement;
  221. }
  222. return element;
  223. }
  224. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"positioning.service.js","sourceRoot":"ng://ngx-bootstrap/positioning/","sources":["positioning.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, ElementRef, RendererFactory2, Inject, PLATFORM_ID, NgZone } from '@angular/core';\nimport { isPlatformBrowser } from '@angular/common';\n\nimport { positionElements } from './ng-positioning';\n\nimport { fromEvent, merge, of, animationFrameScheduler, Subject, Observable } from 'rxjs';\nimport { Options } from './models';\n\n\nexport interface PositioningOptions {\n  /** The DOM element, ElementRef, or a selector string of an element which will be moved */\n  element?: HTMLElement | ElementRef | string;\n\n  /** The DOM element, ElementRef, or a selector string of an element which the element will be attached to  */\n  target?: HTMLElement | ElementRef | string;\n\n  /**\n   * A string of the form 'vert-attachment horiz-attachment' or 'placement'\n   * - placement can be \"top\", \"bottom\", \"left\", \"right\"\n   * not yet supported:\n   * - vert-attachment can be any of 'top', 'middle', 'bottom'\n   * - horiz-attachment can be any of 'left', 'center', 'right'\n   */\n  attachment?: string;\n\n  /** A string similar to `attachment`. The one difference is that, if it's not provided,\n   * `targetAttachment` will assume the mirror image of `attachment`.\n   */\n  targetAttachment?: string;\n\n  /** A string of the form 'vert-offset horiz-offset'\n   * - vert-offset and horiz-offset can be of the form \"20px\" or \"55%\"\n   */\n  offset?: string;\n\n  /** A string similar to `offset`, but referring to the offset of the target */\n  targetOffset?: string;\n\n  /** If true component will be attached to body */\n  appendToBody?: boolean;\n}\n\n\n@Injectable()\nexport class PositioningService {\n  private options: Options;\n  private update$$ = new Subject<null>();\n  private positionElements = new Map();\n  private triggerEvent$: Observable<number|Event>;\n  private isDisabled = false;\n\n  constructor(\n    ngZone: NgZone,\n    rendererFactory: RendererFactory2,\n    @Inject(PLATFORM_ID) platformId: number\n  ) {\n\n    if (isPlatformBrowser(platformId)) {\n      ngZone.runOutsideAngular(() => {\n        this.triggerEvent$ = merge(\n          fromEvent(window, 'scroll', { passive: true }),\n          fromEvent(window, 'resize', { passive: true }),\n          /* tslint:disable-next-line: deprecation */\n          of(0, animationFrameScheduler),\n          this.update$$\n        );\n\n        this.triggerEvent$.subscribe(() => {\n          if (this.isDisabled) {\n            return;\n          }\n\n          this.positionElements\n          /* tslint:disable-next-line: no-any */\n            .forEach((positionElement: any) => {\n              positionElements(\n                _getHtmlElement(positionElement.target),\n                _getHtmlElement(positionElement.element),\n                positionElement.attachment,\n                positionElement.appendToBody,\n                this.options,\n                rendererFactory.createRenderer(null, null)\n              );\n            });\n        });\n      });\n    }\n  }\n\n  position(options: PositioningOptions): void {\n    this.addPositionElement(options);\n  }\n\n  get event$(): Observable<number|Event> {\n    return this.triggerEvent$;\n  }\n\n  disable(): void {\n    this.isDisabled = true;\n  }\n\n  enable(): void {\n    this.isDisabled = false;\n  }\n\n  addPositionElement(options: PositioningOptions): void {\n    this.positionElements.set(_getHtmlElement(options.element), options);\n  }\n\n  calcPosition(): void {\n    this.update$$.next();\n  }\n\n  deletePositionElement(elRef: ElementRef): void {\n    this.positionElements.delete(_getHtmlElement(elRef));\n  }\n\n  setOptions(options: Options) {\n    this.options = options;\n  }\n}\n\nfunction _getHtmlElement(element: HTMLElement | ElementRef | string): HTMLElement {\n  // it means that we got a selector\n  if (typeof element === 'string') {\n    return document.querySelector(element);\n  }\n\n  if (element instanceof ElementRef) {\n    return element.nativeElement;\n  }\n\n  return element;\n}\n"]}