valtech-components 2.0.604 → 2.0.606

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -41,33 +41,47 @@ import * as i0 from "@angular/core";
41
41
  export class DocsPageComponent {
42
42
  constructor() {
43
43
  this.elementRef = inject(ElementRef);
44
- this.props = { title: '' };
44
+ // Use signal internally so computed properties react to changes
45
+ this._props = signal({ title: '' });
45
46
  this.tocItems = signal([]);
46
47
  this.observer = null;
47
- this.tocProps = computed(() => ({
48
- title: this.props.toc?.title ?? 'On this page',
49
- items: this.tocItems(),
50
- }));
51
- this.navLinksProps = computed(() => ({
52
- previous: this.props.previousPage
53
- ? {
54
- label: this.props.navLabels?.previous ?? 'Previous',
55
- title: this.props.previousPage.title,
56
- route: this.props.previousPage.route,
57
- }
58
- : undefined,
59
- next: this.props.nextPage
60
- ? {
61
- label: this.props.navLabels?.next ?? 'Next',
62
- title: this.props.nextPage.title,
63
- route: this.props.nextPage.route,
64
- }
65
- : undefined,
66
- }));
48
+ this.tocProps = computed(() => {
49
+ const props = this._props();
50
+ return {
51
+ title: props.toc?.title ?? 'On this page',
52
+ items: this.tocItems(),
53
+ };
54
+ });
55
+ this.navLinksProps = computed(() => {
56
+ const props = this._props();
57
+ return {
58
+ previous: props.previousPage
59
+ ? {
60
+ // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels
61
+ title: props.previousPage.title,
62
+ route: props.previousPage.route,
63
+ }
64
+ : undefined,
65
+ next: props.nextPage
66
+ ? {
67
+ // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels
68
+ title: props.nextPage.title,
69
+ route: props.nextPage.route,
70
+ }
71
+ : undefined,
72
+ };
73
+ });
67
74
  this.showNavLinks = computed(() => {
68
- return !!this.props.previousPage || !!this.props.nextPage;
75
+ const props = this._props();
76
+ return !!props.previousPage || !!props.nextPage;
69
77
  });
70
78
  }
79
+ set props(value) {
80
+ this._props.set(value);
81
+ }
82
+ get props() {
83
+ return this._props();
84
+ }
71
85
  ngAfterViewInit() {
72
86
  // Initial scan
73
87
  this.scanForSections();
@@ -185,4 +199,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
185
199
  }], propDecorators: { props: [{
186
200
  type: Input
187
201
  }] } });
188
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-page.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/docs-page/docs-page.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, computed, signal, AfterViewInit, OnDestroy, ElementRef, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { DocsNavLinksComponent } from '../../molecules/docs-nav-links/docs-nav-links.component';\nimport { DocsNavLinksMetadata } from '../../molecules/docs-nav-links/types';\nimport { DocsTocComponent } from '../../organisms/docs-toc/docs-toc.component';\nimport { DocsTocMetadata } from '../../organisms/docs-toc/types';\nimport { DocsPageMetadata } from './types';\n\n/**\n * val-docs-page\n *\n * A complete documentation page template that eliminates boilerplate.\n * Provides automatic TOC generation, navigation links, and consistent styling.\n *\n * @example Basic usage\n * ```html\n * <val-docs-page [props]=\"{\n *   title: 'Button',\n *   lead: 'A clickable element for user interactions.',\n *   previousPage: { title: 'Quick Start', route: ['/docs', 'quick-start'] },\n *   nextPage: { title: 'Card', route: ['/docs', 'components', 'card'] }\n * }\">\n *   <val-docs-section [props]=\"{ id: 'basic-usage', title: 'Basic Usage' }\">\n *     <p>Content here...</p>\n *   </val-docs-section>\n *\n *   <val-docs-section [props]=\"{ id: 'variants', title: 'Variants' }\">\n *     <p>More content...</p>\n *   </val-docs-section>\n * </val-docs-page>\n * ```\n *\n * @example With badge\n * ```html\n * <val-docs-page [props]=\"{\n *   title: 'New Component',\n *   badge: 'New',\n *   badgeColor: 'success'\n * }\">\n *   ...\n * </val-docs-page>\n * ```\n */\n@Component({\n  selector: 'val-docs-page',\n  standalone: true,\n  imports: [CommonModule, DocsNavLinksComponent, DocsTocComponent],\n  template: `\n    <div class=\"docs-page\" [class]=\"props.cssClass\">\n      <div class=\"docs-page__content\" #content>\n        <header class=\"docs-page__header\">\n          <div class=\"docs-page__title-row\">\n            <h1 class=\"docs-page__title\">{{ props.title }}</h1>\n            @if (props.badge) {\n              <span\n                class=\"docs-page__badge\"\n                [class.docs-page__badge--success]=\"props.badgeColor === 'success'\"\n                [class.docs-page__badge--warning]=\"props.badgeColor === 'warning'\"\n                [class.docs-page__badge--danger]=\"props.badgeColor === 'danger'\"\n              >\n                {{ props.badge }}\n              </span>\n            }\n          </div>\n          @if (props.lead) {\n            <p class=\"docs-page__lead\">{{ props.lead }}</p>\n          }\n        </header>\n\n        <div class=\"docs-page__sections\">\n          <ng-content></ng-content>\n        </div>\n\n        @if (showNavLinks()) {\n          <val-docs-nav-links [props]=\"navLinksProps()\"></val-docs-nav-links>\n        }\n      </div>\n\n      @if (!props.toc?.hide) {\n        <aside class=\"docs-page__toc\">\n          <val-docs-toc [props]=\"tocProps()\"></val-docs-toc>\n        </aside>\n      }\n    </div>\n  `,\n  styles: [`\n    /* Main layout */\n    .docs-page {\n      display: grid;\n      grid-template-columns: 1fr;\n      gap: 2rem;\n      max-width: 1400px;\n      margin: 0 auto;\n      padding: 2rem 1.5rem;\n    }\n\n    @media (min-width: 1200px) {\n      .docs-page {\n        grid-template-columns: 1fr 220px;\n        padding: 2rem;\n      }\n    }\n\n    .docs-page__content {\n      min-width: 0;\n      max-width: 900px;\n    }\n\n    .docs-page__header {\n      margin-bottom: 2rem;\n    }\n\n    .docs-page__title-row {\n      display: flex;\n      align-items: center;\n      gap: 0.75rem;\n      flex-wrap: wrap;\n    }\n\n    .docs-page__title {\n      margin: 0;\n      font-size: 2rem;\n      font-weight: 700;\n      color: var(--ion-text-color, #1a1a1a);\n      line-height: 1.2;\n    }\n\n    @media (min-width: 768px) {\n      .docs-page__title {\n        font-size: 2.5rem;\n      }\n    }\n\n    .docs-page__badge {\n      display: inline-flex;\n      align-items: center;\n      padding: 0.25rem 0.5rem;\n      font-size: 0.6875rem;\n      font-weight: 500;\n      text-transform: uppercase;\n      letter-spacing: 0.03em;\n      border-radius: 4px;\n      background: rgba(0, 0, 0, 0.04);\n      color: #888;\n    }\n\n    .docs-page__badge--success {\n      background: var(--ion-color-success-tint, #e8f5e9);\n      color: var(--ion-color-success-shade, #2e7d32);\n    }\n\n    .docs-page__badge--warning {\n      background: var(--ion-color-warning-tint, #fff3e0);\n      color: var(--ion-color-warning-shade, #e65100);\n    }\n\n    .docs-page__badge--danger {\n      background: var(--ion-color-danger-tint, #ffebee);\n      color: var(--ion-color-danger-shade, #c62828);\n    }\n\n    .docs-page__lead {\n      margin: 1rem 0 0;\n      font-size: 1.125rem;\n      line-height: 1.7;\n      color: var(--ion-color-medium, #666);\n    }\n\n    .docs-page__sections > *:last-child {\n      margin-bottom: 0;\n    }\n\n    /* TOC - Hidden on mobile, visible on desktop */\n    .docs-page__toc {\n      display: none;\n    }\n\n    @media (min-width: 1200px) {\n      .docs-page__toc {\n        display: block;\n        position: sticky;\n        top: 2rem;\n        height: fit-content;\n        max-height: calc(100vh - 4rem);\n        overflow-y: auto;\n      }\n    }\n\n    /* Content typography styles */\n    .docs-page__sections h2 {\n      font-size: 1.5rem;\n      font-weight: 600;\n      margin: 0 0 1rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n      scroll-margin-top: 2rem;\n    }\n\n    .docs-page__sections h3 {\n      font-size: 1.125rem;\n      font-weight: 600;\n      margin: 1.5rem 0 1rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections h4 {\n      font-size: 1rem;\n      font-weight: 600;\n      margin: 1.25rem 0 0.75rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections p {\n      line-height: 1.7;\n      color: var(--ion-text-color, #1a1a1a);\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections ul,\n    .docs-page__sections ol {\n      padding-left: 1.5rem;\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections li {\n      margin-bottom: 0.5rem;\n      line-height: 1.6;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections a {\n      color: var(--ion-color-primary, #3880ff);\n      text-decoration: none;\n    }\n\n    .docs-page__sections a:hover {\n      text-decoration: underline;\n    }\n\n    .docs-page__sections code:not([class*='language-']) {\n      background: rgba(0, 0, 0, 0.06);\n      padding: 0.125rem 0.375rem;\n      border-radius: 4px;\n      font-family: 'SF Mono', 'Fira Code', Consolas, monospace;\n      font-size: 0.875em;\n    }\n\n    .docs-page__sections pre:not([class*='language-']) {\n      background: rgba(0, 0, 0, 0.04);\n      padding: 1rem;\n      border-radius: 8px;\n      overflow-x: auto;\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections pre:not([class*='language-']) code {\n      background: none;\n      padding: 0;\n    }\n\n    .docs-page__sections table {\n      width: 100%;\n      border-collapse: collapse;\n      margin: 0 0 1rem 0;\n      font-size: 0.875rem;\n    }\n\n    .docs-page__sections th,\n    .docs-page__sections td {\n      padding: 0.75rem;\n      text-align: left;\n      border-bottom: 1px solid rgba(0, 0, 0, 0.1);\n    }\n\n    .docs-page__sections th {\n      font-weight: 600;\n      background: rgba(0, 0, 0, 0.02);\n    }\n\n    .docs-page__sections blockquote {\n      margin: 0 0 1rem 0;\n      padding: 1rem 1.5rem;\n      border-left: 4px solid var(--ion-color-primary, #3880ff);\n      background: rgba(0, 0, 0, 0.02);\n      border-radius: 0 8px 8px 0;\n    }\n\n    .docs-page__sections blockquote p:last-child {\n      margin-bottom: 0;\n    }\n\n    .docs-page__sections img {\n      max-width: 100%;\n      height: auto;\n      border-radius: 8px;\n    }\n\n    .docs-page__sections hr {\n      border: none;\n      border-top: 1px solid rgba(0, 0, 0, 0.1);\n      margin: 2rem 0;\n    }\n\n    .docs-page__sections strong {\n      font-weight: 600;\n    }\n\n    .docs-page__sections section {\n      margin-bottom: 2rem;\n    }\n\n    /* Dark mode */\n    :host-context(.dark) .docs-page__badge,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge {\n      background: rgba(255, 255, 255, 0.06);\n      color: #999;\n    }\n\n    :host-context(.dark) .docs-page__badge--success,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--success {\n      background: rgba(46, 125, 50, 0.2);\n      color: #81c784;\n    }\n\n    :host-context(.dark) .docs-page__badge--warning,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--warning {\n      background: rgba(230, 81, 0, 0.2);\n      color: #ffb74d;\n    }\n\n    :host-context(.dark) .docs-page__badge--danger,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--danger {\n      background: rgba(198, 40, 40, 0.2);\n      color: #e57373;\n    }\n\n    :host-context(.dark) .docs-page__sections code:not([class*='language-']),\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections code:not([class*='language-']) {\n      background: rgba(255, 255, 255, 0.1);\n    }\n\n    :host-context(.dark) .docs-page__sections pre:not([class*='language-']),\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections pre:not([class*='language-']) {\n      background: rgba(255, 255, 255, 0.06);\n    }\n\n    :host-context(.dark) .docs-page__sections th,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections th {\n      background: rgba(255, 255, 255, 0.04);\n    }\n\n    :host-context(.dark) .docs-page__sections th,\n    :host-context(.dark) .docs-page__sections td,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections th,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections td {\n      border-color: rgba(255, 255, 255, 0.1);\n    }\n\n    :host-context(.dark) .docs-page__sections blockquote,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections blockquote {\n      background: rgba(255, 255, 255, 0.04);\n    }\n\n    :host-context(.dark) .docs-page__sections hr,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections hr {\n      border-color: rgba(255, 255, 255, 0.1);\n    }\n  `],\n})\nexport class DocsPageComponent implements AfterViewInit, OnDestroy {\n  private elementRef = inject(ElementRef);\n\n  @Input() props: DocsPageMetadata = { title: '' };\n\n  private tocItems = signal<{ id: string; label: string; level: number }[]>([]);\n  private observer: MutationObserver | null = null;\n\n  ngAfterViewInit(): void {\n    // Initial scan\n    this.scanForSections();\n\n    // Watch for dynamic content changes\n    this.observer = new MutationObserver(() => {\n      this.scanForSections();\n    });\n\n    const contentEl = this.elementRef.nativeElement.querySelector('.docs-page__sections');\n    if (contentEl) {\n      this.observer.observe(contentEl, { childList: true, subtree: true });\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.observer?.disconnect();\n  }\n\n  private scanForSections(): void {\n    const contentEl = this.elementRef.nativeElement.querySelector('.docs-page__sections');\n    if (!contentEl) return;\n\n    const headings = contentEl.querySelectorAll('h2[id], h3[id], section[id] > h2, section[id] > h3');\n    const items: { id: string; label: string; level: number }[] = [];\n\n    headings.forEach((heading: Element) => {\n      // Get ID from heading or parent section\n      let id = heading.id;\n      if (!id && heading.parentElement?.tagName === 'SECTION') {\n        id = heading.parentElement.id;\n      }\n\n      if (id) {\n        const level = heading.tagName === 'H2' ? 2 : 3;\n        items.push({ id, label: heading.textContent?.trim() || '', level });\n      }\n    });\n\n    this.tocItems.set(items);\n  }\n\n  tocProps = computed<DocsTocMetadata>(() => ({\n    title: this.props.toc?.title ?? 'On this page',\n    items: this.tocItems(),\n  }));\n\n  navLinksProps = computed<DocsNavLinksMetadata>(() => ({\n    previous: this.props.previousPage\n      ? {\n          label: this.props.navLabels?.previous ?? 'Previous',\n          title: this.props.previousPage.title,\n          route: this.props.previousPage.route,\n        }\n      : undefined,\n    next: this.props.nextPage\n      ? {\n          label: this.props.navLabels?.next ?? 'Next',\n          title: this.props.nextPage.title,\n          route: this.props.nextPage.route,\n        }\n      : undefined,\n  }));\n\n  showNavLinks = computed(() => {\n    return !!this.props.previousPage || !!this.props.nextPage;\n  });\n}\n"]}
202
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"docs-page.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/templates/docs-page/docs-page.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, computed, signal, AfterViewInit, OnDestroy, ElementRef, inject } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport { DocsNavLinksComponent } from '../../molecules/docs-nav-links/docs-nav-links.component';\nimport { DocsNavLinksMetadata } from '../../molecules/docs-nav-links/types';\nimport { DocsTocComponent } from '../../organisms/docs-toc/docs-toc.component';\nimport { DocsTocMetadata } from '../../organisms/docs-toc/types';\nimport { DocsPageMetadata } from './types';\n\n/**\n * val-docs-page\n *\n * A complete documentation page template that eliminates boilerplate.\n * Provides automatic TOC generation, navigation links, and consistent styling.\n *\n * @example Basic usage\n * ```html\n * <val-docs-page [props]=\"{\n *   title: 'Button',\n *   lead: 'A clickable element for user interactions.',\n *   previousPage: { title: 'Quick Start', route: ['/docs', 'quick-start'] },\n *   nextPage: { title: 'Card', route: ['/docs', 'components', 'card'] }\n * }\">\n *   <val-docs-section [props]=\"{ id: 'basic-usage', title: 'Basic Usage' }\">\n *     <p>Content here...</p>\n *   </val-docs-section>\n *\n *   <val-docs-section [props]=\"{ id: 'variants', title: 'Variants' }\">\n *     <p>More content...</p>\n *   </val-docs-section>\n * </val-docs-page>\n * ```\n *\n * @example With badge\n * ```html\n * <val-docs-page [props]=\"{\n *   title: 'New Component',\n *   badge: 'New',\n *   badgeColor: 'success'\n * }\">\n *   ...\n * </val-docs-page>\n * ```\n */\n@Component({\n  selector: 'val-docs-page',\n  standalone: true,\n  imports: [CommonModule, DocsNavLinksComponent, DocsTocComponent],\n  template: `\n    <div class=\"docs-page\" [class]=\"props.cssClass\">\n      <div class=\"docs-page__content\" #content>\n        <header class=\"docs-page__header\">\n          <div class=\"docs-page__title-row\">\n            <h1 class=\"docs-page__title\">{{ props.title }}</h1>\n            @if (props.badge) {\n              <span\n                class=\"docs-page__badge\"\n                [class.docs-page__badge--success]=\"props.badgeColor === 'success'\"\n                [class.docs-page__badge--warning]=\"props.badgeColor === 'warning'\"\n                [class.docs-page__badge--danger]=\"props.badgeColor === 'danger'\"\n              >\n                {{ props.badge }}\n              </span>\n            }\n          </div>\n          @if (props.lead) {\n            <p class=\"docs-page__lead\">{{ props.lead }}</p>\n          }\n        </header>\n\n        <div class=\"docs-page__sections\">\n          <ng-content></ng-content>\n        </div>\n\n        @if (showNavLinks()) {\n          <val-docs-nav-links [props]=\"navLinksProps()\"></val-docs-nav-links>\n        }\n      </div>\n\n      @if (!props.toc?.hide) {\n        <aside class=\"docs-page__toc\">\n          <val-docs-toc [props]=\"tocProps()\"></val-docs-toc>\n        </aside>\n      }\n    </div>\n  `,\n  styles: [`\n    /* Main layout */\n    .docs-page {\n      display: grid;\n      grid-template-columns: 1fr;\n      gap: 2rem;\n      max-width: 1400px;\n      margin: 0 auto;\n      padding: 2rem 1.5rem;\n    }\n\n    @media (min-width: 1200px) {\n      .docs-page {\n        grid-template-columns: 1fr 220px;\n        padding: 2rem;\n      }\n    }\n\n    .docs-page__content {\n      min-width: 0;\n      max-width: 900px;\n    }\n\n    .docs-page__header {\n      margin-bottom: 2rem;\n    }\n\n    .docs-page__title-row {\n      display: flex;\n      align-items: center;\n      gap: 0.75rem;\n      flex-wrap: wrap;\n    }\n\n    .docs-page__title {\n      margin: 0;\n      font-size: 2rem;\n      font-weight: 700;\n      color: var(--ion-text-color, #1a1a1a);\n      line-height: 1.2;\n    }\n\n    @media (min-width: 768px) {\n      .docs-page__title {\n        font-size: 2.5rem;\n      }\n    }\n\n    .docs-page__badge {\n      display: inline-flex;\n      align-items: center;\n      padding: 0.25rem 0.5rem;\n      font-size: 0.6875rem;\n      font-weight: 500;\n      text-transform: uppercase;\n      letter-spacing: 0.03em;\n      border-radius: 4px;\n      background: rgba(0, 0, 0, 0.04);\n      color: #888;\n    }\n\n    .docs-page__badge--success {\n      background: var(--ion-color-success-tint, #e8f5e9);\n      color: var(--ion-color-success-shade, #2e7d32);\n    }\n\n    .docs-page__badge--warning {\n      background: var(--ion-color-warning-tint, #fff3e0);\n      color: var(--ion-color-warning-shade, #e65100);\n    }\n\n    .docs-page__badge--danger {\n      background: var(--ion-color-danger-tint, #ffebee);\n      color: var(--ion-color-danger-shade, #c62828);\n    }\n\n    .docs-page__lead {\n      margin: 1rem 0 0;\n      font-size: 1.125rem;\n      line-height: 1.7;\n      color: var(--ion-color-medium, #666);\n    }\n\n    .docs-page__sections > *:last-child {\n      margin-bottom: 0;\n    }\n\n    /* TOC - Hidden on mobile, visible on desktop */\n    .docs-page__toc {\n      display: none;\n    }\n\n    @media (min-width: 1200px) {\n      .docs-page__toc {\n        display: block;\n        position: sticky;\n        top: 2rem;\n        height: fit-content;\n        max-height: calc(100vh - 4rem);\n        overflow-y: auto;\n      }\n    }\n\n    /* Content typography styles */\n    .docs-page__sections h2 {\n      font-size: 1.5rem;\n      font-weight: 600;\n      margin: 0 0 1rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n      scroll-margin-top: 2rem;\n    }\n\n    .docs-page__sections h3 {\n      font-size: 1.125rem;\n      font-weight: 600;\n      margin: 1.5rem 0 1rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections h4 {\n      font-size: 1rem;\n      font-weight: 600;\n      margin: 1.25rem 0 0.75rem 0;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections p {\n      line-height: 1.7;\n      color: var(--ion-text-color, #1a1a1a);\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections ul,\n    .docs-page__sections ol {\n      padding-left: 1.5rem;\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections li {\n      margin-bottom: 0.5rem;\n      line-height: 1.6;\n      color: var(--ion-text-color, #1a1a1a);\n    }\n\n    .docs-page__sections a {\n      color: var(--ion-color-primary, #3880ff);\n      text-decoration: none;\n    }\n\n    .docs-page__sections a:hover {\n      text-decoration: underline;\n    }\n\n    .docs-page__sections code:not([class*='language-']) {\n      background: rgba(0, 0, 0, 0.06);\n      padding: 0.125rem 0.375rem;\n      border-radius: 4px;\n      font-family: 'SF Mono', 'Fira Code', Consolas, monospace;\n      font-size: 0.875em;\n    }\n\n    .docs-page__sections pre:not([class*='language-']) {\n      background: rgba(0, 0, 0, 0.04);\n      padding: 1rem;\n      border-radius: 8px;\n      overflow-x: auto;\n      margin: 0 0 1rem 0;\n    }\n\n    .docs-page__sections pre:not([class*='language-']) code {\n      background: none;\n      padding: 0;\n    }\n\n    .docs-page__sections table {\n      width: 100%;\n      border-collapse: collapse;\n      margin: 0 0 1rem 0;\n      font-size: 0.875rem;\n    }\n\n    .docs-page__sections th,\n    .docs-page__sections td {\n      padding: 0.75rem;\n      text-align: left;\n      border-bottom: 1px solid rgba(0, 0, 0, 0.1);\n    }\n\n    .docs-page__sections th {\n      font-weight: 600;\n      background: rgba(0, 0, 0, 0.02);\n    }\n\n    .docs-page__sections blockquote {\n      margin: 0 0 1rem 0;\n      padding: 1rem 1.5rem;\n      border-left: 4px solid var(--ion-color-primary, #3880ff);\n      background: rgba(0, 0, 0, 0.02);\n      border-radius: 0 8px 8px 0;\n    }\n\n    .docs-page__sections blockquote p:last-child {\n      margin-bottom: 0;\n    }\n\n    .docs-page__sections img {\n      max-width: 100%;\n      height: auto;\n      border-radius: 8px;\n    }\n\n    .docs-page__sections hr {\n      border: none;\n      border-top: 1px solid rgba(0, 0, 0, 0.1);\n      margin: 2rem 0;\n    }\n\n    .docs-page__sections strong {\n      font-weight: 600;\n    }\n\n    .docs-page__sections section {\n      margin-bottom: 2rem;\n    }\n\n    /* Dark mode */\n    :host-context(.dark) .docs-page__badge,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge {\n      background: rgba(255, 255, 255, 0.06);\n      color: #999;\n    }\n\n    :host-context(.dark) .docs-page__badge--success,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--success {\n      background: rgba(46, 125, 50, 0.2);\n      color: #81c784;\n    }\n\n    :host-context(.dark) .docs-page__badge--warning,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--warning {\n      background: rgba(230, 81, 0, 0.2);\n      color: #ffb74d;\n    }\n\n    :host-context(.dark) .docs-page__badge--danger,\n    :host-context([color-scheme=\"dark\"]) .docs-page__badge--danger {\n      background: rgba(198, 40, 40, 0.2);\n      color: #e57373;\n    }\n\n    :host-context(.dark) .docs-page__sections code:not([class*='language-']),\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections code:not([class*='language-']) {\n      background: rgba(255, 255, 255, 0.1);\n    }\n\n    :host-context(.dark) .docs-page__sections pre:not([class*='language-']),\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections pre:not([class*='language-']) {\n      background: rgba(255, 255, 255, 0.06);\n    }\n\n    :host-context(.dark) .docs-page__sections th,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections th {\n      background: rgba(255, 255, 255, 0.04);\n    }\n\n    :host-context(.dark) .docs-page__sections th,\n    :host-context(.dark) .docs-page__sections td,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections th,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections td {\n      border-color: rgba(255, 255, 255, 0.1);\n    }\n\n    :host-context(.dark) .docs-page__sections blockquote,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections blockquote {\n      background: rgba(255, 255, 255, 0.04);\n    }\n\n    :host-context(.dark) .docs-page__sections hr,\n    :host-context([color-scheme=\"dark\"]) .docs-page__sections hr {\n      border-color: rgba(255, 255, 255, 0.1);\n    }\n  `],\n})\nexport class DocsPageComponent implements AfterViewInit, OnDestroy {\n  private elementRef = inject(ElementRef);\n\n  // Use signal internally so computed properties react to changes\n  private _props = signal<DocsPageMetadata>({ title: '' });\n\n  @Input()\n  set props(value: DocsPageMetadata) {\n    this._props.set(value);\n  }\n  get props(): DocsPageMetadata {\n    return this._props();\n  }\n\n  private tocItems = signal<{ id: string; label: string; level: number }[]>([]);\n  private observer: MutationObserver | null = null;\n\n  ngAfterViewInit(): void {\n    // Initial scan\n    this.scanForSections();\n\n    // Watch for dynamic content changes\n    this.observer = new MutationObserver(() => {\n      this.scanForSections();\n    });\n\n    const contentEl = this.elementRef.nativeElement.querySelector('.docs-page__sections');\n    if (contentEl) {\n      this.observer.observe(contentEl, { childList: true, subtree: true });\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.observer?.disconnect();\n  }\n\n  private scanForSections(): void {\n    const contentEl = this.elementRef.nativeElement.querySelector('.docs-page__sections');\n    if (!contentEl) return;\n\n    const headings = contentEl.querySelectorAll('h2[id], h3[id], section[id] > h2, section[id] > h3');\n    const items: { id: string; label: string; level: number }[] = [];\n\n    headings.forEach((heading: Element) => {\n      // Get ID from heading or parent section\n      let id = heading.id;\n      if (!id && heading.parentElement?.tagName === 'SECTION') {\n        id = heading.parentElement.id;\n      }\n\n      if (id) {\n        const level = heading.tagName === 'H2' ? 2 : 3;\n        items.push({ id, label: heading.textContent?.trim() || '', level });\n      }\n    });\n\n    this.tocItems.set(items);\n  }\n\n  tocProps = computed<DocsTocMetadata>(() => {\n    const props = this._props();\n    return {\n      title: props.toc?.title ?? 'On this page',\n      items: this.tocItems(),\n    };\n  });\n\n  navLinksProps = computed<DocsNavLinksMetadata>(() => {\n    const props = this._props();\n    return {\n      previous: props.previousPage\n        ? {\n            // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels\n            title: props.previousPage.title,\n            route: props.previousPage.route,\n          }\n        : undefined,\n      next: props.nextPage\n        ? {\n            // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels\n            title: props.nextPage.title,\n            route: props.nextPage.route,\n          }\n        : undefined,\n    };\n  });\n\n  showNavLinks = computed(() => {\n    const props = this._props();\n    return !!props.previousPage || !!props.nextPage;\n  });\n}\n"]}
@@ -37875,33 +37875,47 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
37875
37875
  class DocsPageComponent {
37876
37876
  constructor() {
37877
37877
  this.elementRef = inject(ElementRef);
37878
- this.props = { title: '' };
37878
+ // Use signal internally so computed properties react to changes
37879
+ this._props = signal({ title: '' });
37879
37880
  this.tocItems = signal([]);
37880
37881
  this.observer = null;
37881
- this.tocProps = computed(() => ({
37882
- title: this.props.toc?.title ?? 'On this page',
37883
- items: this.tocItems(),
37884
- }));
37885
- this.navLinksProps = computed(() => ({
37886
- previous: this.props.previousPage
37887
- ? {
37888
- label: this.props.navLabels?.previous ?? 'Previous',
37889
- title: this.props.previousPage.title,
37890
- route: this.props.previousPage.route,
37891
- }
37892
- : undefined,
37893
- next: this.props.nextPage
37894
- ? {
37895
- label: this.props.navLabels?.next ?? 'Next',
37896
- title: this.props.nextPage.title,
37897
- route: this.props.nextPage.route,
37898
- }
37899
- : undefined,
37900
- }));
37882
+ this.tocProps = computed(() => {
37883
+ const props = this._props();
37884
+ return {
37885
+ title: props.toc?.title ?? 'On this page',
37886
+ items: this.tocItems(),
37887
+ };
37888
+ });
37889
+ this.navLinksProps = computed(() => {
37890
+ const props = this._props();
37891
+ return {
37892
+ previous: props.previousPage
37893
+ ? {
37894
+ // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels
37895
+ title: props.previousPage.title,
37896
+ route: props.previousPage.route,
37897
+ }
37898
+ : undefined,
37899
+ next: props.nextPage
37900
+ ? {
37901
+ // Don't pass label - let DocsNavLinksComponent use its internal i18n-reactive labels
37902
+ title: props.nextPage.title,
37903
+ route: props.nextPage.route,
37904
+ }
37905
+ : undefined,
37906
+ };
37907
+ });
37901
37908
  this.showNavLinks = computed(() => {
37902
- return !!this.props.previousPage || !!this.props.nextPage;
37909
+ const props = this._props();
37910
+ return !!props.previousPage || !!props.nextPage;
37903
37911
  });
37904
37912
  }
37913
+ set props(value) {
37914
+ this._props.set(value);
37915
+ }
37916
+ get props() {
37917
+ return this._props();
37918
+ }
37905
37919
  ngAfterViewInit() {
37906
37920
  // Initial scan
37907
37921
  this.scanForSections();