triggers.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { Trigger } from './trigger.class';
  6. /**
  7. * @record
  8. */
  9. export function ListenOptions() { }
  10. if (false) {
  11. /** @type {?|undefined} */
  12. ListenOptions.prototype.target;
  13. /** @type {?|undefined} */
  14. ListenOptions.prototype.targets;
  15. /** @type {?|undefined} */
  16. ListenOptions.prototype.triggers;
  17. /** @type {?|undefined} */
  18. ListenOptions.prototype.outsideClick;
  19. /** @type {?|undefined} */
  20. ListenOptions.prototype.outsideEsc;
  21. /** @type {?|undefined} */
  22. ListenOptions.prototype.show;
  23. /** @type {?|undefined} */
  24. ListenOptions.prototype.hide;
  25. /** @type {?|undefined} */
  26. ListenOptions.prototype.toggle;
  27. }
  28. /** @type {?} */
  29. const DEFAULT_ALIASES = {
  30. hover: ['mouseover', 'mouseout'],
  31. focus: ['focusin', 'focusout']
  32. };
  33. /* tslint:disable-next-line: no-any */
  34. /**
  35. * @param {?} triggers
  36. * @param {?=} aliases
  37. * @return {?}
  38. */
  39. export function parseTriggers(triggers, aliases = DEFAULT_ALIASES) {
  40. /** @type {?} */
  41. const trimmedTriggers = (triggers || '').trim();
  42. if (trimmedTriggers.length === 0) {
  43. return [];
  44. }
  45. /** @type {?} */
  46. const parsedTriggers = trimmedTriggers
  47. .split(/\s+/)
  48. .map((/**
  49. * @param {?} trigger
  50. * @return {?}
  51. */
  52. (trigger) => trigger.split(':')))
  53. .map((/**
  54. * @param {?} triggerPair
  55. * @return {?}
  56. */
  57. (triggerPair) => {
  58. /** @type {?} */
  59. const alias = aliases[triggerPair[0]] || triggerPair;
  60. return new Trigger(alias[0], alias[1]);
  61. }));
  62. /** @type {?} */
  63. const manualTriggers = parsedTriggers.filter((/**
  64. * @param {?} triggerPair
  65. * @return {?}
  66. */
  67. (triggerPair) => triggerPair.isManual()));
  68. if (manualTriggers.length > 1) {
  69. throw new Error('Triggers parse error: only one manual trigger is allowed');
  70. }
  71. if (manualTriggers.length === 1 && parsedTriggers.length > 1) {
  72. throw new Error('Triggers parse error: manual trigger can\'t be mixed with other triggers');
  73. }
  74. return parsedTriggers;
  75. }
  76. /**
  77. * @param {?} renderer
  78. * @param {?} target
  79. * @param {?} triggers
  80. * @param {?} showFn
  81. * @param {?} hideFn
  82. * @param {?} toggleFn
  83. * @return {?}
  84. */
  85. export function listenToTriggers(renderer,
  86. /* tslint:disable-next-line: no-any */
  87. target, triggers, showFn, hideFn, toggleFn) {
  88. /** @type {?} */
  89. const parsedTriggers = parseTriggers(triggers);
  90. /* tslint:disable-next-line: no-any */
  91. /** @type {?} */
  92. const listeners = [];
  93. if (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) {
  94. return Function.prototype;
  95. }
  96. parsedTriggers.forEach((/**
  97. * @param {?} trigger
  98. * @return {?}
  99. */
  100. (trigger) => {
  101. if (trigger.open === trigger.close) {
  102. listeners.push(renderer.listen(target, trigger.open, toggleFn));
  103. return;
  104. }
  105. listeners.push(renderer.listen(target, trigger.open, showFn), renderer.listen(target, trigger.close, hideFn));
  106. }));
  107. return (/**
  108. * @return {?}
  109. */
  110. () => {
  111. listeners.forEach((/**
  112. * @param {?} unsubscribeFn
  113. * @return {?}
  114. */
  115. (unsubscribeFn) => unsubscribeFn()));
  116. });
  117. }
  118. /**
  119. * @param {?} renderer
  120. * @param {?} options
  121. * @return {?}
  122. */
  123. export function listenToTriggersV2(renderer, options) {
  124. /** @type {?} */
  125. const parsedTriggers = parseTriggers(options.triggers);
  126. /** @type {?} */
  127. const target = options.target;
  128. // do nothing
  129. if (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) {
  130. return Function.prototype;
  131. }
  132. // all listeners
  133. /* tslint:disable-next-line: no-any */
  134. /** @type {?} */
  135. const listeners = [];
  136. // lazy listeners registration
  137. /** @type {?} */
  138. const _registerHide = [];
  139. /** @type {?} */
  140. const registerHide = (/**
  141. * @return {?}
  142. */
  143. () => {
  144. // add hide listeners to unregister array
  145. _registerHide.forEach((/**
  146. * @param {?} fn
  147. * @return {?}
  148. */
  149. (fn) => listeners.push(fn())));
  150. // register hide events only once
  151. _registerHide.length = 0;
  152. });
  153. // register open\close\toggle listeners
  154. parsedTriggers.forEach((/**
  155. * @param {?} trigger
  156. * @return {?}
  157. */
  158. (trigger) => {
  159. /** @type {?} */
  160. const useToggle = trigger.open === trigger.close;
  161. /** @type {?} */
  162. const showFn = useToggle ? options.toggle : options.show;
  163. if (!useToggle) {
  164. _registerHide.push((/**
  165. * @return {?}
  166. */
  167. () => renderer.listen(target, trigger.close, options.hide)));
  168. }
  169. listeners.push(renderer.listen(target, trigger.open, (/**
  170. * @return {?}
  171. */
  172. () => showFn(registerHide))));
  173. }));
  174. return (/**
  175. * @return {?}
  176. */
  177. () => {
  178. listeners.forEach((/**
  179. * @param {?} unsubscribeFn
  180. * @return {?}
  181. */
  182. (unsubscribeFn) => unsubscribeFn()));
  183. });
  184. }
  185. /**
  186. * @param {?} renderer
  187. * @param {?} options
  188. * @return {?}
  189. */
  190. export function registerOutsideClick(renderer, options) {
  191. if (!options.outsideClick) {
  192. return Function.prototype;
  193. }
  194. /* tslint:disable-next-line: no-any */
  195. return renderer.listen('document', 'click', (/**
  196. * @param {?} event
  197. * @return {?}
  198. */
  199. (event) => {
  200. if (options.target && options.target.contains(event.target)) {
  201. return undefined;
  202. }
  203. if (options.targets &&
  204. options.targets.some((/**
  205. * @param {?} target
  206. * @return {?}
  207. */
  208. target => target.contains(event.target)))) {
  209. return undefined;
  210. }
  211. options.hide();
  212. }));
  213. }
  214. /**
  215. * @param {?} renderer
  216. * @param {?} options
  217. * @return {?}
  218. */
  219. export function registerEscClick(renderer, options) {
  220. if (!options.outsideEsc) {
  221. return Function.prototype;
  222. }
  223. return renderer.listen('document', 'keyup.esc', (/**
  224. * @param {?} event
  225. * @return {?}
  226. */
  227. (event) => {
  228. if (options.target && options.target.contains(event.target)) {
  229. return undefined;
  230. }
  231. if (options.targets &&
  232. options.targets.some((/**
  233. * @param {?} target
  234. * @return {?}
  235. */
  236. target => target.contains(event.target)))) {
  237. return undefined;
  238. }
  239. options.hide();
  240. }));
  241. }
  242. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"triggers.js","sourceRoot":"ng://ngx-bootstrap/utils/","sources":["triggers.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @copyright Valor Software\n * @copyright Angular ng-bootstrap team\n */\nimport { Renderer2 } from '@angular/core';\nimport { Trigger } from './trigger.class';\n\n/* tslint:disable-next-line: no-any */\nexport type BsEventCallback = (event?: any) => boolean | void;\n\nexport interface ListenOptions {\n  target?: HTMLElement;\n  targets?: HTMLElement[];\n  triggers?: string;\n  outsideClick?: boolean;\n  outsideEsc?: boolean;\n  show?: BsEventCallback;\n  hide?: BsEventCallback;\n  toggle?: BsEventCallback;\n}\n\nconst DEFAULT_ALIASES = {\n  hover: ['mouseover', 'mouseout'],\n  focus: ['focusin', 'focusout']\n};\n\n/* tslint:disable-next-line: no-any */\nexport function parseTriggers(triggers: string, aliases: any = DEFAULT_ALIASES): Trigger[] {\n  const trimmedTriggers = (triggers || '').trim();\n\n  if (trimmedTriggers.length === 0) {\n    return [];\n  }\n\n  const parsedTriggers = trimmedTriggers\n    .split(/\\s+/)\n    .map((trigger: string) => trigger.split(':'))\n    .map((triggerPair: string[]) => {\n      const alias = aliases[triggerPair[0]] || triggerPair;\n\n      return new Trigger(alias[0], alias[1]);\n    });\n\n  const manualTriggers = parsedTriggers.filter((triggerPair: Trigger) =>\n    triggerPair.isManual()\n  );\n\n  if (manualTriggers.length > 1) {\n    throw new Error('Triggers parse error: only one manual trigger is allowed');\n  }\n\n  if (manualTriggers.length === 1 && parsedTriggers.length > 1) {\n    throw new Error('Triggers parse error: manual trigger can\\'t be mixed with other triggers');\n  }\n\n  return parsedTriggers;\n}\n\nexport function listenToTriggers(renderer: Renderer2,\n                                 /* tslint:disable-next-line: no-any */\n                                 target: any,\n                                 triggers: string,\n                                 showFn: BsEventCallback,\n                                 hideFn: BsEventCallback,\n                                 toggleFn: BsEventCallback): Function {\n  const parsedTriggers = parseTriggers(triggers);\n  /* tslint:disable-next-line: no-any */\n  const listeners: any[] = [];\n\n  if (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) {\n    return Function.prototype;\n  }\n\n  parsedTriggers.forEach((trigger: Trigger) => {\n    if (trigger.open === trigger.close) {\n      listeners.push(renderer.listen(target, trigger.open, toggleFn));\n\n      return;\n    }\n\n    listeners.push(\n      renderer.listen(target, trigger.open, showFn),\n      renderer.listen(target, trigger.close, hideFn)\n    );\n  });\n\n  return () => {\n    listeners.forEach((unsubscribeFn: Function) => unsubscribeFn());\n  };\n}\n\nexport function listenToTriggersV2(renderer: Renderer2,\n                                   options: ListenOptions): Function {\n  const parsedTriggers = parseTriggers(options.triggers);\n  const target = options.target;\n  // do nothing\n  if (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) {\n    return Function.prototype;\n  }\n\n  // all listeners\n  /* tslint:disable-next-line: no-any */\n  const listeners: any[] = [];\n\n  // lazy listeners registration\n  const _registerHide: Function[] = [];\n  const registerHide = () => {\n    // add hide listeners to unregister array\n    _registerHide.forEach((fn: Function) => listeners.push(fn()));\n    // register hide events only once\n    _registerHide.length = 0;\n  };\n\n  // register open\\close\\toggle listeners\n  parsedTriggers.forEach((trigger: Trigger) => {\n    const useToggle = trigger.open === trigger.close;\n    const showFn = useToggle ? options.toggle : options.show;\n\n    if (!useToggle) {\n      _registerHide.push(() =>\n        renderer.listen(target, trigger.close, options.hide)\n      );\n    }\n\n    listeners.push(\n      renderer.listen(target, trigger.open, () => showFn(registerHide))\n    );\n  });\n\n  return () => {\n    listeners.forEach((unsubscribeFn: Function) => unsubscribeFn());\n  };\n}\n\nexport function registerOutsideClick(renderer: Renderer2,\n                                     options: ListenOptions) {\n  if (!options.outsideClick) {\n    return Function.prototype;\n  }\n\n  /* tslint:disable-next-line: no-any */\n  return renderer.listen('document', 'click', (event: any) => {\n    if (options.target && options.target.contains(event.target)) {\n      return undefined;\n    }\n    if (\n      options.targets &&\n      options.targets.some(target => target.contains(event.target))\n    ) {\n      return undefined;\n    }\n\n    options.hide();\n  });\n}\n\nexport function registerEscClick(renderer: Renderer2,\n                                 options: ListenOptions) {\n  if (!options.outsideEsc) {\n    return Function.prototype;\n  }\n\n  return renderer.listen('document', 'keyup.esc', (event: any) => {\n    if (options.target && options.target.contains(event.target)) {\n      return undefined;\n    }\n    if (\n      options.targets &&\n      options.targets.some(target => target.contains(event.target))\n    ) {\n      return undefined;\n    }\n\n    options.hide();\n  });\n}\n"]}