UNPKG

@coreui/angular

Version:

CoreUI Components Library for Angular

107 lines 16.8 kB
import { afterRender, Component, computed, contentChild, ElementRef, inject, input, signal } from '@angular/core'; import { DOCUMENT, NgClass, NgTemplateOutlet } from '@angular/common'; import { BreakpointObserver } from '@angular/cdk/layout'; import { CollapseDirective } from '../collapse'; import { ThemeDirective } from '../shared'; import * as i0 from "@angular/core"; import * as i1 from "../shared/theme.directive"; // todo: fix container prop issue not rendering children // todo: workaround - use <c-container> component directly in template export class NavbarComponent { constructor() { this.#breakpointObserver = inject(BreakpointObserver); this.#document = inject(DOCUMENT); this.#hostElement = inject(ElementRef); /** * Sets the color context of the component to one of CoreUI’s themed colors. * @type Colors */ this.color = input(); /** * Defines optional container wrapping children elements. */ this.container = input(); /** * Defines the responsive breakpoint to determine when content collapses. */ this.expand = input(); /** * Place component in non-static positions. */ this.placement = input(); this.role = input('navigation'); this.collapse = contentChild(CollapseDirective); this.hostClasses = computed(() => { const color = this.color(); const expand = this.expand(); const expandClassSuffix = expand === true ? '' : `-${expand}`; const placement = this.placement(); return { navbar: true, [`navbar-expand${expandClassSuffix}`]: !!expand, [`bg-${color}`]: !!color, [`${placement}`]: !!placement }; }); this.containerClass = computed(() => { const container = this.container(); return `container${container !== true ? '-' + container : ''}`; }); this.computedStyle = signal(''); this.afterNextRenderFn = afterRender({ read: () => { const expand = this.expand(); if (typeof expand === 'string') { const computedStyle = this.#document.defaultView ?.getComputedStyle(this.#hostElement.nativeElement) ?.getPropertyValue(`--cui-breakpoint-${expand}`) ?? false; computedStyle && this.computedStyle.set(computedStyle); } } }); this.breakpoint = computed(() => { const expand = this.expand(); if (typeof expand === 'string') { return this.computedStyle(); } return false; }); } #breakpointObserver; #document; #hostElement; #observer; ngAfterContentInit() { const breakpoint = this.breakpoint(); if (breakpoint) { const onBreakpoint = `(min-width: ${breakpoint})`; this.#observer = this.#breakpointObserver .observe([onBreakpoint]) .pipe() .subscribe((result) => { const collapse = this.collapse(); if (collapse) { const animate = collapse.animate(); collapse.animate.set(false); collapse.toggle(false); setTimeout(() => { collapse.toggle(result.matches); setTimeout(() => { collapse.animate.set(animate); }); }); } }); } } ngOnDestroy() { this.#observer?.unsubscribe(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: NavbarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.12", type: NavbarComponent, isStandalone: true, selector: "c-navbar", inputs: { color: { classPropertyName: "color", publicName: "color", isSignal: true, isRequired: false, transformFunction: null }, container: { classPropertyName: "container", publicName: "container", isSignal: true, isRequired: false, transformFunction: null }, expand: { classPropertyName: "expand", publicName: "expand", isSignal: true, isRequired: false, transformFunction: null }, placement: { classPropertyName: "placement", publicName: "placement", isSignal: true, isRequired: false, transformFunction: null }, role: { classPropertyName: "role", publicName: "role", isSignal: true, isRequired: false, transformFunction: null } }, host: { properties: { "class": "hostClasses()", "attr.role": "role()" } }, queries: [{ propertyName: "collapse", first: true, predicate: CollapseDirective, descendants: true, isSignal: true }], hostDirectives: [{ directive: i1.ThemeDirective, inputs: ["colorScheme", "colorScheme"] }], ngImport: i0, template: "<ng-container *ngTemplateOutlet=\"container() ? withContainerTemplate : noContainerTemplate\" />\n\n<ng-template #withContainerTemplate>\n <div [ngClass]=\"containerClass()\">\n <ng-content />\n </div>\n</ng-template>\n\n<ng-template #noContainerTemplate>\n <ng-content />\n</ng-template>\n", dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: NavbarComponent, decorators: [{ type: Component, args: [{ selector: 'c-navbar', standalone: true, imports: [NgClass, NgTemplateOutlet], hostDirectives: [{ directive: ThemeDirective, inputs: ['colorScheme'] }], host: { '[class]': 'hostClasses()', '[attr.role]': 'role()' }, template: "<ng-container *ngTemplateOutlet=\"container() ? withContainerTemplate : noContainerTemplate\" />\n\n<ng-template #withContainerTemplate>\n <div [ngClass]=\"containerClass()\">\n <ng-content />\n </div>\n</ng-template>\n\n<ng-template #noContainerTemplate>\n <ng-content />\n</ng-template>\n" }] }] }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"navbar.component.js","sourceRoot":"","sources":["../../../../../projects/coreui-angular/src/lib/navbar/navbar.component.ts","../../../../../projects/coreui-angular/src/lib/navbar/navbar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterContentInit,\n  afterRender,\n  Component,\n  computed,\n  contentChild,\n  ElementRef,\n  inject,\n  input,\n  OnDestroy,\n  signal\n} from '@angular/core';\nimport { DOCUMENT, NgClass, NgTemplateOutlet } from '@angular/common';\nimport { BreakpointObserver } from '@angular/cdk/layout';\n\nimport { CollapseDirective } from '../collapse';\nimport { Colors } from '../coreui.types';\nimport { ThemeDirective } from '../shared';\nimport { Subscription } from 'rxjs';\n\n// todo: fix container prop issue not rendering children\n// todo: workaround -  use <c-container> component directly in template\n\n@Component({\n  selector: 'c-navbar',\n  templateUrl: './navbar.component.html',\n  standalone: true,\n  imports: [NgClass, NgTemplateOutlet],\n  hostDirectives: [{ directive: ThemeDirective, inputs: ['colorScheme'] }],\n  host: { '[class]': 'hostClasses()', '[attr.role]': 'role()' }\n})\nexport class NavbarComponent implements AfterContentInit, OnDestroy {\n  readonly #breakpointObserver = inject(BreakpointObserver);\n  readonly #document = inject(DOCUMENT);\n  readonly #hostElement = inject(ElementRef);\n\n  /**\n   * Sets the color context of the component to one of CoreUI’s themed colors.\n   * @type Colors\n   */\n  readonly color = input<Colors>();\n\n  /**\n   * Defines optional container wrapping children elements.\n   */\n  readonly container = input<boolean | 'sm' | 'md' | 'lg' | 'xl' | 'xxl' | 'fluid'>();\n\n  /**\n   * Defines the responsive breakpoint to determine when content collapses.\n   */\n  readonly expand = input<boolean | 'sm' | 'md' | 'lg' | 'xl' | 'xxl'>();\n\n  /**\n   * Place component in non-static positions.\n   */\n  readonly placement = input<'fixed-top' | 'fixed-bottom' | 'sticky-top'>();\n\n  readonly role = input('navigation');\n\n  readonly collapse = contentChild(CollapseDirective);\n\n  readonly hostClasses = computed(() => {\n    const color = this.color();\n    const expand = this.expand();\n    const expandClassSuffix: string = expand === true ? '' : `-${expand}`;\n    const placement = this.placement();\n    return {\n      navbar: true,\n      [`navbar-expand${expandClassSuffix}`]: !!expand,\n      [`bg-${color}`]: !!color,\n      [`${placement}`]: !!placement\n    } as Record<string, boolean>;\n  });\n\n  readonly containerClass = computed(() => {\n    const container = this.container();\n    return `container${container !== true ? '-' + container : ''}`;\n  });\n\n  readonly computedStyle = signal<string>('');\n\n  readonly afterNextRenderFn = afterRender({\n    read: () => {\n      const expand = this.expand();\n      if (typeof expand === 'string') {\n        const computedStyle =\n          this.#document.defaultView\n            ?.getComputedStyle(this.#hostElement.nativeElement)\n            ?.getPropertyValue(`--cui-breakpoint-${expand}`) ?? false;\n        computedStyle && this.computedStyle.set(computedStyle);\n      }\n    }\n  });\n\n  readonly breakpoint = computed(() => {\n    const expand = this.expand();\n    if (typeof expand === 'string') {\n      return this.computedStyle();\n    }\n    return false;\n  });\n\n  #observer!: Subscription;\n\n  ngAfterContentInit(): void {\n    const breakpoint = this.breakpoint();\n    if (breakpoint) {\n      const onBreakpoint = `(min-width: ${breakpoint})`;\n      this.#observer = this.#breakpointObserver\n        .observe([onBreakpoint])\n        .pipe()\n        .subscribe((result) => {\n          const collapse = this.collapse();\n          if (collapse) {\n            const animate = collapse.animate();\n            collapse.animate.set(false);\n            collapse.toggle(false);\n            setTimeout(() => {\n              collapse.toggle(result.matches);\n              setTimeout(() => {\n                collapse.animate.set(animate);\n              });\n            });\n          }\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.#observer?.unsubscribe();\n  }\n}\n","<ng-container *ngTemplateOutlet=\"container() ? withContainerTemplate : noContainerTemplate\" />\n\n<ng-template #withContainerTemplate>\n  <div [ngClass]=\"containerClass()\">\n    <ng-content />\n  </div>\n</ng-template>\n\n<ng-template #noContainerTemplate>\n  <ng-content />\n</ng-template>\n"]}