@bimeister/pupakit.kit
Version:
PupaKit is an open source collection of Angular components based on an atomic approach to building interfaces, which guarantees better performance and greater development flexibility.
104 lines • 20.4 kB
JavaScript
import { filterNotNil, isEmpty, isNil, resizeObservable, shareReplayWithRefCount, VOID, } from '@bimeister/utilities';
import { asyncScheduler, BehaviorSubject, combineLatest } from 'rxjs';
import { distinctUntilChanged, filter, map, observeOn, subscribeOn, switchMap, take } from 'rxjs/operators';
import '../../../components/scrollable/components/scrollable/scrollable.component';
export class TabsServiceBase {
constructor() {
this.tabNameToHtmlElementMap = new Map();
this.activeTabNameState$ = new BehaviorSubject(null);
this.activeTabName$ = this.activeTabNameState$.asObservable();
this.hostElement$ = new BehaviorSubject(null);
this.scrollable$ = new BehaviorSubject(null);
this.tabsHtmlElement$ = new BehaviorSubject(null);
this.activeHtmlElement$ = this.activeTabName$.pipe(filterNotNil(), map((activeTabName) => this.tabNameToHtmlElementMap.get(JSON.stringify(activeTabName))), shareReplayWithRefCount());
this.tabsContainerResize$ = this.tabsHtmlElement$.pipe(filterNotNil(), switchMap((tabsHtmlElement) => resizeObservable(tabsHtmlElement)));
this.refresh$ = new BehaviorSubject(VOID);
this.railHighlighterOffsetLeftPx$ = combineLatest([
this.activeHtmlElement$.pipe(filterNotNil()),
this.tabsHtmlElement$.pipe(filterNotNil()),
this.tabsContainerResize$,
this.refresh$,
]).pipe(observeOn(asyncScheduler), map(([activeHtmlElement, tabsHtmlElement]) => {
const activeClientRect = activeHtmlElement.getBoundingClientRect();
const tabsClientRect = tabsHtmlElement.getBoundingClientRect();
return activeClientRect.left - tabsClientRect.left;
}));
this.railHighlighterWidthPx$ = combineLatest([
this.activeHtmlElement$,
this.tabsContainerResize$,
this.refresh$,
]).pipe(observeOn(asyncScheduler), map(([activeHtmlElement]) => activeHtmlElement.clientWidth));
this.tabNames = [];
this.isContentDraggingState$ = new BehaviorSubject(false);
this.isContentDragging$ = this.isContentDraggingState$.pipe(distinctUntilChanged());
}
registerTab(tabName) {
this.tabNames.push(tabName);
this.refresh$.next();
}
unregisterTab(tabName) {
this.tabNames = this.tabNames.filter((tab) => tab !== tabName);
this.resetActiveTabIfUnregisteredTabIsActive(tabName);
this.refresh$.next();
}
setInitialTab() {
this.activeTabName$
.pipe(take(1), filter((activeTab) => isNil(activeTab) && !isEmpty(this.tabNames)), subscribeOn(asyncScheduler))
.subscribe(() => {
this.setActiveTab(this.tabNames[0]);
});
}
setActiveTab(tabName) {
this.activeTabNameState$.next(tabName);
this.correctScrollLeftByTargetTab(tabName);
}
registerTabsHtmlElement(htmlElement) {
this.tabsHtmlElement$.next(htmlElement);
}
registerHostHtmlElement(htmlElement) {
this.hostElement$.next(htmlElement);
}
registerScrollable(scrollable) {
this.scrollable$.next(scrollable);
}
registerTabHtmlElement(tabName, htmlElement) {
this.tabNameToHtmlElementMap.set(JSON.stringify(tabName), htmlElement);
}
setContentDraggingStateState(isContentDragging) {
this.isContentDraggingState$.next(isContentDragging);
}
correctScrollLeftByTargetTab(tabName) {
const targetElement = this.tabNameToHtmlElementMap.get(JSON.stringify(tabName));
if (isNil(targetElement)) {
return;
}
combineLatest([this.hostElement$.pipe(filterNotNil()), this.scrollable$.pipe(filterNotNil())])
.pipe(take(1), map(([hostElement, scrollable]) => {
const hostClientRect = hostElement.getBoundingClientRect();
const targetClientRect = targetElement.getBoundingClientRect();
const leftOffsetPx = targetClientRect.left - hostClientRect.left;
const rightOffsetPx = hostClientRect.right - targetClientRect.right;
const centerLeftDeltaPx = (hostClientRect.width - targetClientRect.width) / 2;
return [leftOffsetPx, rightOffsetPx, scrollable, centerLeftDeltaPx];
}))
.subscribe(([leftOffsetPx, rightOffsetPx, scrollable, centerLeftDeltaPx]) => {
const isNeedScrollToLeft = leftOffsetPx < rightOffsetPx;
const isNeedScrollToRight = rightOffsetPx < leftOffsetPx;
const isSmoothScroll = true;
if (isNeedScrollToLeft) {
scrollable.setScrollLeftByDelta(Math.ceil(-centerLeftDeltaPx + leftOffsetPx), isSmoothScroll);
return;
}
if (isNeedScrollToRight) {
scrollable.setScrollLeftByDelta(Math.ceil(-rightOffsetPx + centerLeftDeltaPx), isSmoothScroll);
return;
}
});
}
resetActiveTabIfUnregisteredTabIsActive(removedTabName) {
this.activeTabName$
.pipe(take(1), filter((activeTabName) => activeTabName === removedTabName))
.subscribe(() => this.setActiveTab(null));
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs-service-base.abstract.js","sourceRoot":"","sources":["../../../../../src/declarations/classes/abstract/tabs-service-base.abstract.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  filterNotNil,\n  isEmpty,\n  isNil,\n  Nullable,\n  resizeObservable,\n  shareReplayWithRefCount,\n  VOID,\n} from '@bimeister/utilities';\nimport { asyncScheduler, BehaviorSubject, combineLatest, Observable } from 'rxjs';\nimport { distinctUntilChanged, filter, map, observeOn, subscribeOn, switchMap, take } from 'rxjs/operators';\nimport { ScrollableComponent } from '../../../components/scrollable/components/scrollable/scrollable.component';\n\nexport abstract class TabsServiceBase<T> {\n  protected readonly tabNameToHtmlElementMap: Map<string, HTMLElement> = new Map<string, HTMLElement>();\n  private readonly activeTabNameState$: BehaviorSubject<Nullable<T>> = new BehaviorSubject<Nullable<T>>(null);\n  public readonly activeTabName$: Observable<Nullable<T>> = this.activeTabNameState$.asObservable();\n\n  protected readonly hostElement$: BehaviorSubject<Nullable<HTMLElement>> = new BehaviorSubject<Nullable<HTMLElement>>(\n    null\n  );\n  protected readonly scrollable$: BehaviorSubject<Nullable<ScrollableComponent>> = new BehaviorSubject<\n    Nullable<ScrollableComponent>\n  >(null);\n  protected readonly tabsHtmlElement$: BehaviorSubject<Nullable<HTMLElement>> = new BehaviorSubject<\n    Nullable<HTMLElement>\n  >(null);\n\n  private readonly activeHtmlElement$: Observable<HTMLElement> = this.activeTabName$.pipe(\n    filterNotNil(),\n    map((activeTabName: T) => this.tabNameToHtmlElementMap.get(JSON.stringify(activeTabName))),\n    shareReplayWithRefCount()\n  );\n\n  private readonly tabsContainerResize$: Observable<ResizeObserverEntry[]> = this.tabsHtmlElement$.pipe(\n    filterNotNil(),\n    switchMap((tabsHtmlElement: HTMLElement) => resizeObservable(tabsHtmlElement))\n  );\n\n  private readonly refresh$: BehaviorSubject<void> = new BehaviorSubject<void>(VOID);\n\n  public readonly railHighlighterOffsetLeftPx$: Observable<number> = combineLatest([\n    this.activeHtmlElement$.pipe(filterNotNil()),\n    this.tabsHtmlElement$.pipe(filterNotNil()),\n    this.tabsContainerResize$,\n    this.refresh$,\n  ]).pipe(\n    observeOn(asyncScheduler),\n    map(([activeHtmlElement, tabsHtmlElement]: [HTMLElement, HTMLElement, ResizeObserverEntry[], void]) => {\n      const activeClientRect: ClientRect = activeHtmlElement.getBoundingClientRect();\n      const tabsClientRect: ClientRect = tabsHtmlElement.getBoundingClientRect();\n      return activeClientRect.left - tabsClientRect.left;\n    })\n  );\n  public readonly railHighlighterWidthPx$: Observable<number> = combineLatest([\n    this.activeHtmlElement$,\n    this.tabsContainerResize$,\n    this.refresh$,\n  ]).pipe(\n    observeOn(asyncScheduler),\n    map(([activeHtmlElement]: [HTMLElement, ResizeObserverEntry[], void]) => activeHtmlElement.clientWidth)\n  );\n\n  private tabNames: T[] = [];\n\n  private readonly isContentDraggingState$: BehaviorSubject<boolean> = new BehaviorSubject<boolean>(false);\n  public readonly isContentDragging$: Observable<boolean> = this.isContentDraggingState$.pipe(distinctUntilChanged());\n\n  public registerTab(tabName: T): void {\n    this.tabNames.push(tabName);\n\n    this.refresh$.next();\n  }\n\n  public unregisterTab(tabName: T): void {\n    this.tabNames = this.tabNames.filter((tab: T) => tab !== tabName);\n\n    this.resetActiveTabIfUnregisteredTabIsActive(tabName);\n\n    this.refresh$.next();\n  }\n\n  public setInitialTab(): void {\n    this.activeTabName$\n      .pipe(\n        take(1),\n        filter((activeTab: Nullable<T>) => isNil(activeTab) && !isEmpty(this.tabNames)),\n        subscribeOn(asyncScheduler)\n      )\n      .subscribe(() => {\n        this.setActiveTab(this.tabNames[0]);\n      });\n  }\n\n  public setActiveTab(tabName: T): void {\n    this.activeTabNameState$.next(tabName);\n    this.correctScrollLeftByTargetTab(tabName);\n  }\n\n  public registerTabsHtmlElement(htmlElement: HTMLElement): void {\n    this.tabsHtmlElement$.next(htmlElement);\n  }\n\n  public registerHostHtmlElement(htmlElement: HTMLElement): void {\n    this.hostElement$.next(htmlElement);\n  }\n\n  public registerScrollable(scrollable: ScrollableComponent): void {\n    this.scrollable$.next(scrollable);\n  }\n\n  public registerTabHtmlElement(tabName: T, htmlElement: HTMLElement): void {\n    this.tabNameToHtmlElementMap.set(JSON.stringify(tabName), htmlElement);\n  }\n\n  public setContentDraggingStateState(isContentDragging: boolean): void {\n    this.isContentDraggingState$.next(isContentDragging);\n  }\n\n  private correctScrollLeftByTargetTab(tabName: T): void {\n    const targetElement: HTMLElement = this.tabNameToHtmlElementMap.get(JSON.stringify(tabName));\n\n    if (isNil(targetElement)) {\n      return;\n    }\n\n    combineLatest([this.hostElement$.pipe(filterNotNil()), this.scrollable$.pipe(filterNotNil())])\n      .pipe(\n        take(1),\n        map(([hostElement, scrollable]: [HTMLElement, ScrollableComponent]) => {\n          const hostClientRect: ClientRect = hostElement.getBoundingClientRect();\n          const targetClientRect: ClientRect = targetElement.getBoundingClientRect();\n\n          const leftOffsetPx: number = targetClientRect.left - hostClientRect.left;\n          const rightOffsetPx: number = hostClientRect.right - targetClientRect.right;\n          const centerLeftDeltaPx: number = (hostClientRect.width - targetClientRect.width) / 2;\n          return [leftOffsetPx, rightOffsetPx, scrollable, centerLeftDeltaPx];\n        })\n      )\n      .subscribe(\n        ([leftOffsetPx, rightOffsetPx, scrollable, centerLeftDeltaPx]: [\n          number,\n          number,\n          ScrollableComponent,\n          number\n        ]) => {\n          const isNeedScrollToLeft: boolean = leftOffsetPx < rightOffsetPx;\n          const isNeedScrollToRight: boolean = rightOffsetPx < leftOffsetPx;\n\n          const isSmoothScroll: boolean = true;\n\n          if (isNeedScrollToLeft) {\n            scrollable.setScrollLeftByDelta(Math.ceil(-centerLeftDeltaPx + leftOffsetPx), isSmoothScroll);\n            return;\n          }\n\n          if (isNeedScrollToRight) {\n            scrollable.setScrollLeftByDelta(Math.ceil(-rightOffsetPx + centerLeftDeltaPx), isSmoothScroll);\n            return;\n          }\n        }\n      );\n  }\n\n  private resetActiveTabIfUnregisteredTabIsActive(removedTabName: T): void {\n    this.activeTabName$\n      .pipe(\n        take(1),\n        filter((activeTabName: T) => activeTabName === removedTabName)\n      )\n      .subscribe(() => this.setActiveTab(null));\n  }\n}\n"]}