UNPKG

@ng-bootstrap/ng-bootstrap

Version:
95 lines 15.2 kB
import { Observable, merge } from 'rxjs'; import { share, filter, delay, map } from 'rxjs/operators'; export class Trigger { constructor(open, close) { this.open = open; this.close = close; if (!close) { this.close = open; } } isManual() { return this.open === 'manual' || this.close === 'manual'; } } const DEFAULT_ALIASES = { hover: ['mouseenter', 'mouseleave'], focus: ['focusin', 'focusout'], }; export function parseTriggers(triggers, aliases = DEFAULT_ALIASES) { const trimmedTriggers = (triggers || '').trim(); if (trimmedTriggers.length === 0) { return []; } const parsedTriggers = trimmedTriggers .split(/\s+/) .map((trigger) => trigger.split(':')) .map((triggerPair) => { let alias = aliases[triggerPair[0]] || triggerPair; return new Trigger(alias[0], alias[1]); }); const manualTriggers = parsedTriggers.filter((triggerPair) => triggerPair.isManual()); if (manualTriggers.length > 1) { throw 'Triggers parse error: only one manual trigger is allowed'; } if (manualTriggers.length === 1 && parsedTriggers.length > 1) { throw "Triggers parse error: manual trigger can't be mixed with other triggers"; } return parsedTriggers; } export function observeTriggers(renderer, nativeElement, triggers, isOpenedFn) { return new Observable((subscriber) => { const listeners = []; const openFn = () => subscriber.next(true); const closeFn = () => subscriber.next(false); const toggleFn = () => subscriber.next(!isOpenedFn()); triggers.forEach((trigger) => { if (trigger.open === trigger.close) { listeners.push(renderer.listen(nativeElement, trigger.open, toggleFn)); } else { listeners.push(renderer.listen(nativeElement, trigger.open, openFn), renderer.listen(nativeElement, trigger.close, closeFn)); } }); return () => { listeners.forEach((unsubscribeFn) => unsubscribeFn()); }; }); } const delayOrNoop = (time) => (time > 0 ? delay(time) : (a) => a); export function triggerDelay(openDelay, closeDelay, isOpenedFn) { return (input$) => { let pending = null; const filteredInput$ = input$.pipe(map((open) => ({ open })), filter((event) => { const currentlyOpen = isOpenedFn(); if (currentlyOpen !== event.open && (!pending || pending.open === currentlyOpen)) { pending = event; return true; } if (pending && pending.open !== event.open) { pending = null; } return false; }), share()); const delayedOpen$ = filteredInput$.pipe(filter((event) => event.open), delayOrNoop(openDelay)); const delayedClose$ = filteredInput$.pipe(filter((event) => !event.open), delayOrNoop(closeDelay)); return merge(delayedOpen$, delayedClose$).pipe(filter((event) => { if (event === pending) { pending = null; return event.open !== isOpenedFn(); } return false; }), map((event) => event.open)); }; } export function listenToTriggers(renderer, nativeElement, triggers, isOpenedFn, openFn, closeFn, openDelay = 0, closeDelay = 0) { const parsedTriggers = parseTriggers(triggers); if (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) { return () => { }; } const subscription = observeTriggers(renderer, nativeElement, parsedTriggers, isOpenedFn) .pipe(triggerDelay(openDelay, closeDelay, isOpenedFn)) .subscribe((open) => (open ? openFn() : closeFn())); return () => subscription.unsubscribe(); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"triggers.js","sourceRoot":"","sources":["../../../../src/util/triggers.ts"],"names":[],"mappings":"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","sourcesContent":["import { Observable, merge } from 'rxjs';\nimport { share, filter, delay, map } from 'rxjs/operators';\nimport { Renderer2 } from '@angular/core';\n\nexport class Trigger {\n\tconstructor(public open: string, public close?: string) {\n\t\tif (!close) {\n\t\t\tthis.close = open;\n\t\t}\n\t}\n\n\tisManual() {\n\t\treturn this.open === 'manual' || this.close === 'manual';\n\t}\n}\n\nconst DEFAULT_ALIASES = {\n\thover: ['mouseenter', 'mouseleave'],\n\tfocus: ['focusin', 'focusout'],\n};\n\nexport function parseTriggers(triggers: string, aliases = DEFAULT_ALIASES): Trigger[] {\n\tconst trimmedTriggers = (triggers || '').trim();\n\n\tif (trimmedTriggers.length === 0) {\n\t\treturn [];\n\t}\n\n\tconst parsedTriggers = trimmedTriggers\n\t\t.split(/\\s+/)\n\t\t.map((trigger) => trigger.split(':'))\n\t\t.map((triggerPair) => {\n\t\t\tlet alias = aliases[triggerPair[0]] || triggerPair;\n\t\t\treturn new Trigger(alias[0], alias[1]);\n\t\t});\n\n\tconst manualTriggers = parsedTriggers.filter((triggerPair) => triggerPair.isManual());\n\n\tif (manualTriggers.length > 1) {\n\t\tthrow 'Triggers parse error: only one manual trigger is allowed';\n\t}\n\n\tif (manualTriggers.length === 1 && parsedTriggers.length > 1) {\n\t\tthrow \"Triggers parse error: manual trigger can't be mixed with other triggers\";\n\t}\n\n\treturn parsedTriggers;\n}\n\nexport function observeTriggers(\n\trenderer: Renderer2,\n\tnativeElement: HTMLElement,\n\ttriggers: Trigger[],\n\tisOpenedFn: () => boolean,\n) {\n\treturn new Observable<boolean>((subscriber) => {\n\t\tconst listeners: (() => void)[] = [];\n\t\tconst openFn = () => subscriber.next(true);\n\t\tconst closeFn = () => subscriber.next(false);\n\t\tconst toggleFn = () => subscriber.next(!isOpenedFn());\n\n\t\ttriggers.forEach((trigger: Trigger) => {\n\t\t\tif (trigger.open === trigger.close) {\n\t\t\t\tlisteners.push(renderer.listen(nativeElement, trigger.open, toggleFn));\n\t\t\t} else {\n\t\t\t\tlisteners.push(\n\t\t\t\t\trenderer.listen(nativeElement, trigger.open, openFn),\n\t\t\t\t\trenderer.listen(nativeElement, trigger.close!, closeFn),\n\t\t\t\t);\n\t\t\t}\n\t\t});\n\n\t\treturn () => {\n\t\t\tlisteners.forEach((unsubscribeFn) => unsubscribeFn());\n\t\t};\n\t});\n}\n\nconst delayOrNoop = <T>(time: number) => (time > 0 ? delay<T>(time) : (a: Observable<T>) => a);\n\nexport function triggerDelay(openDelay: number, closeDelay: number, isOpenedFn: () => boolean) {\n\treturn (input$: Observable<boolean>) => {\n\t\tlet pending: { open: boolean } | null = null;\n\t\tconst filteredInput$ = input$.pipe(\n\t\t\tmap((open) => ({ open })),\n\t\t\tfilter((event) => {\n\t\t\t\tconst currentlyOpen = isOpenedFn();\n\t\t\t\tif (currentlyOpen !== event.open && (!pending || pending.open === currentlyOpen)) {\n\t\t\t\t\tpending = event;\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tif (pending && pending.open !== event.open) {\n\t\t\t\t\tpending = null;\n\t\t\t\t}\n\t\t\t\treturn false;\n\t\t\t}),\n\t\t\tshare(),\n\t\t);\n\t\tconst delayedOpen$ = filteredInput$.pipe(\n\t\t\tfilter((event) => event.open),\n\t\t\tdelayOrNoop(openDelay),\n\t\t);\n\t\tconst delayedClose$ = filteredInput$.pipe(\n\t\t\tfilter((event) => !event.open),\n\t\t\tdelayOrNoop(closeDelay),\n\t\t);\n\t\treturn merge(delayedOpen$, delayedClose$).pipe(\n\t\t\tfilter((event) => {\n\t\t\t\tif (event === pending) {\n\t\t\t\t\tpending = null;\n\t\t\t\t\treturn event.open !== isOpenedFn();\n\t\t\t\t}\n\t\t\t\treturn false;\n\t\t\t}),\n\t\t\tmap((event) => event.open),\n\t\t);\n\t};\n}\n\nexport function listenToTriggers(\n\trenderer: Renderer2,\n\tnativeElement: HTMLElement,\n\ttriggers: string,\n\tisOpenedFn: () => boolean,\n\topenFn: () => void,\n\tcloseFn: () => void,\n\topenDelay = 0,\n\tcloseDelay = 0,\n) {\n\tconst parsedTriggers = parseTriggers(triggers);\n\n\tif (parsedTriggers.length === 1 && parsedTriggers[0].isManual()) {\n\t\treturn () => {};\n\t}\n\n\tconst subscription = observeTriggers(renderer, nativeElement, parsedTriggers, isOpenedFn)\n\t\t.pipe(triggerDelay(openDelay, closeDelay, isOpenedFn))\n\t\t.subscribe((open) => (open ? openFn() : closeFn()));\n\n\treturn () => subscription.unsubscribe();\n}\n"]}