UNPKG

@coreui/angular

Version:

CoreUI Components Library for Angular

145 lines 21.6 kB
import { booleanAttribute, computed, Directive, effect, ElementRef, inject, input, output, Renderer2, signal } from '@angular/core'; import { AnimationBuilder, useAnimation } from '@angular/animations'; import { collapseAnimation, collapseHorizontalAnimation, expandAnimation, expandHorizontalAnimation } from './collapse.animations'; import * as i0 from "@angular/core"; export class CollapseDirective { constructor() { this.#hostElement = inject(ElementRef); this.#renderer = inject(Renderer2); this.#animationBuilder = inject(AnimationBuilder); this.#player = undefined; /** * @ignore */ this.animateInput = input(true, { transform: booleanAttribute, alias: 'animate' }); this.animate = signal(true); this.animateInputEffect = effect(() => { this.animate.set(this.animateInput()); }, { allowSignalWrites: true }); /** * Set horizontal collapsing to transition the width instead of height. * @type boolean * @default false */ this.horizontal = input(false, { transform: booleanAttribute }); /** * Toggle the visibility of collapsible element. * @type boolean * @default false */ this.visibleInput = input(false, { transform: booleanAttribute, alias: 'visible' }); this.visibleChange = output(); this.visibleInputEffect = effect(() => { this.visible.set(this.visibleInput()); }, { allowSignalWrites: true }); this.visible = signal(false); this.#init = false; this.visibleEffect = effect(() => { const visible = this.visible(); (this.#init || visible) && this.createPlayer(visible); this.#init = true; }, { allowSignalWrites: true }); /** * Add `navbar` prop for grouping and hiding navbar contents by a parent breakpoint. * @type boolean * @default false */ this.navbar = input(false, { transform: booleanAttribute }); /** * @ignore */ this.duration = input('350ms'); /** * @ignore */ this.transition = input('ease'); /** * Event emitted on visibility change. [docs] * @type string */ this.collapseChange = output(); this.hostClasses = computed(() => { return { 'navbar-collapse': this.navbar(), 'collapse-horizontal': this.horizontal() }; }); } #hostElement; #renderer; #animationBuilder; #player; #init; ngOnDestroy() { this.destroyPlayer(); } toggle(visible = !this.visible()) { this.visible.set(visible); } destroyPlayer() { this.#player?.destroy(); this.#player = undefined; } createPlayer(visible = this.visible()) { if (this.#player?.hasStarted()) { this.destroyPlayer(); } const host = this.#hostElement.nativeElement; if (visible) { this.#renderer.removeStyle(host, 'display'); } const duration = this.animate() ? this.duration() : '0ms'; const expand = this.horizontal() ? expandHorizontalAnimation : expandAnimation; const collapse = this.horizontal() ? collapseHorizontalAnimation : collapseAnimation; const dimension = this.horizontal() ? 'width' : 'height'; const capitalizedDimension = dimension[0].toUpperCase() + dimension.slice(1); const scrollSize = `scroll${capitalizedDimension}`; const animationFactory = this.#animationBuilder?.build(useAnimation(visible ? expand : collapse, { params: { time: duration, easing: this.transition() } })); this.#player = animationFactory.create(host); !visible && host.offsetHeight && host.style[dimension] && host.scrollHeight; this.#renderer.setStyle(host, dimension, visible ? 0 : `${host.getBoundingClientRect()[dimension]}px`); this.#player.onStart(() => { this.setMaxSize(); this.#renderer.removeClass(host, 'collapse'); this.#renderer.addClass(host, 'collapsing'); this.#renderer.removeClass(host, 'show'); this.#renderer.setStyle(host, dimension, visible ? `${host[scrollSize]}px` : ''); this.collapseChange?.emit(visible ? 'opening' : 'collapsing'); }); this.#player.onDone(() => { this.#renderer.removeClass(host, 'collapsing'); this.#renderer.addClass(host, 'collapse'); if (visible) { this.#renderer.addClass(host, 'show'); this.#renderer.setStyle(host, dimension, ''); } else { this.#renderer.removeClass(host, 'show'); } this.collapseChange?.emit(visible ? 'open' : 'collapsed'); this.destroyPlayer(); this.visibleChange.emit(visible); }); this.#player?.play(); } setMaxSize() { const host = this.#hostElement.nativeElement; if (this.horizontal()) { host.scrollWidth > 0 && this.#renderer.setStyle(host, 'maxWidth', `${host.scrollWidth}px`); // } else { // host.scrollHeight > 0 && this.#renderer.setStyle(host, 'maxHeight', `${host.scrollHeight}px`); } } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: CollapseDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.12", type: CollapseDirective, isStandalone: true, selector: "[cCollapse]", inputs: { animateInput: { classPropertyName: "animateInput", publicName: "animate", isSignal: true, isRequired: false, transformFunction: null }, horizontal: { classPropertyName: "horizontal", publicName: "horizontal", isSignal: true, isRequired: false, transformFunction: null }, visibleInput: { classPropertyName: "visibleInput", publicName: "visible", isSignal: true, isRequired: false, transformFunction: null }, navbar: { classPropertyName: "navbar", publicName: "navbar", isSignal: true, isRequired: false, transformFunction: null }, duration: { classPropertyName: "duration", publicName: "duration", isSignal: true, isRequired: false, transformFunction: null }, transition: { classPropertyName: "transition", publicName: "transition", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { visibleChange: "visibleChange", collapseChange: "collapseChange" }, host: { properties: { "class": "hostClasses()", "style": "{display: \"none\"}" } }, exportAs: ["cCollapse"], ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: CollapseDirective, decorators: [{ type: Directive, args: [{ selector: '[cCollapse]', exportAs: 'cCollapse', standalone: true, host: { '[class]': 'hostClasses()', '[style]': '{display: "none"}' } }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"collapse.directive.js","sourceRoot":"","sources":["../../../../../projects/coreui-angular/src/lib/collapse/collapse.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  booleanAttribute,\n  computed,\n  Directive,\n  effect,\n  ElementRef,\n  inject,\n  input,\n  OnDestroy,\n  output,\n  Renderer2,\n  signal\n} from '@angular/core';\nimport { AnimationBuilder, AnimationPlayer, useAnimation } from '@angular/animations';\n\nimport {\n  collapseAnimation,\n  collapseHorizontalAnimation,\n  expandAnimation,\n  expandHorizontalAnimation\n} from './collapse.animations';\n\n@Directive({\n  selector: '[cCollapse]',\n  exportAs: 'cCollapse',\n  standalone: true,\n  host: { '[class]': 'hostClasses()', '[style]': '{display: \"none\"}' }\n})\nexport class CollapseDirective implements OnDestroy {\n  readonly #hostElement = inject(ElementRef);\n  readonly #renderer = inject(Renderer2);\n  readonly #animationBuilder = inject(AnimationBuilder);\n  #player: AnimationPlayer | undefined = undefined;\n\n  /**\n   * @ignore\n   */\n  readonly animateInput = input(true, { transform: booleanAttribute, alias: 'animate' });\n\n  readonly animate = signal(true);\n\n  readonly animateInputEffect = effect(\n    () => {\n      this.animate.set(this.animateInput());\n    },\n    { allowSignalWrites: true }\n  );\n\n  /**\n   * Set horizontal collapsing to transition the width instead of height.\n   * @type boolean\n   * @default false\n   */\n  readonly horizontal = input(false, { transform: booleanAttribute });\n\n  /**\n   * Toggle the visibility of collapsible element.\n   * @type boolean\n   * @default false\n   */\n  readonly visibleInput = input(false, { transform: booleanAttribute, alias: 'visible' });\n\n  readonly visibleChange = output<boolean>();\n\n  readonly visibleInputEffect = effect(\n    () => {\n      this.visible.set(this.visibleInput());\n    },\n    { allowSignalWrites: true }\n  );\n\n  readonly visible = signal<boolean>(false);\n\n  #init = false;\n\n  readonly visibleEffect = effect(\n    () => {\n      const visible = this.visible();\n\n      (this.#init || visible) && this.createPlayer(visible);\n      this.#init = true;\n    },\n    { allowSignalWrites: true }\n  );\n\n  /**\n   * Add `navbar` prop for grouping and hiding navbar contents by a parent breakpoint.\n   * @type boolean\n   * @default false\n   */\n  readonly navbar = input(false, { transform: booleanAttribute });\n\n  /**\n   * @ignore\n   */\n  readonly duration = input('350ms');\n\n  /**\n   * @ignore\n   */\n  readonly transition = input('ease');\n\n  /**\n   * Event emitted on visibility change. [docs]\n   * @type string\n   */\n  readonly collapseChange = output<string>();\n\n  readonly hostClasses = computed(() => {\n    return {\n      'navbar-collapse': this.navbar(),\n      'collapse-horizontal': this.horizontal()\n    } as Record<string, boolean>;\n  });\n\n  ngOnDestroy(): void {\n    this.destroyPlayer();\n  }\n\n  toggle(visible = !this.visible()): void {\n    this.visible.set(visible);\n  }\n\n  destroyPlayer(): void {\n    this.#player?.destroy();\n    this.#player = undefined;\n  }\n\n  createPlayer(visible: boolean = this.visible()): void {\n    if (this.#player?.hasStarted()) {\n      this.destroyPlayer();\n    }\n\n    const host: HTMLElement = this.#hostElement.nativeElement;\n\n    if (visible) {\n      this.#renderer.removeStyle(host, 'display');\n    }\n\n    const duration = this.animate() ? this.duration() : '0ms';\n\n    const expand = this.horizontal() ? expandHorizontalAnimation : expandAnimation;\n    const collapse = this.horizontal() ? collapseHorizontalAnimation : collapseAnimation;\n\n    const dimension = this.horizontal() ? 'width' : 'height';\n    const capitalizedDimension = dimension[0].toUpperCase() + dimension.slice(1);\n    const scrollSize = `scroll${capitalizedDimension}`;\n\n    const animationFactory = this.#animationBuilder?.build(\n      useAnimation(visible ? expand : collapse, { params: { time: duration, easing: this.transition() } })\n    );\n\n    this.#player = animationFactory.create(host);\n\n    !visible && host.offsetHeight && host.style[dimension] && host.scrollHeight;\n\n    this.#renderer.setStyle(host, dimension, visible ? 0 : `${host.getBoundingClientRect()[dimension]}px`);\n\n    this.#player.onStart(() => {\n      this.setMaxSize();\n      this.#renderer.removeClass(host, 'collapse');\n      this.#renderer.addClass(host, 'collapsing');\n      this.#renderer.removeClass(host, 'show');\n      this.#renderer.setStyle(host, dimension, visible ? `${(host as any)[scrollSize]}px` : '');\n      this.collapseChange?.emit(visible ? 'opening' : 'collapsing');\n    });\n\n    this.#player.onDone(() => {\n      this.#renderer.removeClass(host, 'collapsing');\n      this.#renderer.addClass(host, 'collapse');\n      if (visible) {\n        this.#renderer.addClass(host, 'show');\n        this.#renderer.setStyle(host, dimension, '');\n      } else {\n        this.#renderer.removeClass(host, 'show');\n      }\n      this.collapseChange?.emit(visible ? 'open' : 'collapsed');\n      this.destroyPlayer();\n      this.visibleChange.emit(visible);\n    });\n\n    this.#player?.play();\n  }\n\n  setMaxSize() {\n    const host = this.#hostElement.nativeElement;\n    if (this.horizontal()) {\n      host.scrollWidth > 0 && this.#renderer.setStyle(host, 'maxWidth', `${host.scrollWidth}px`);\n      // } else {\n      //   host.scrollHeight > 0 && this.#renderer.setStyle(host, 'maxHeight', `${host.scrollHeight}px`);\n    }\n  }\n}\n"]}