@quadrel-enterprise-ui/framework 18.25.4 → 18.25.6

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.
@@ -161,6 +161,12 @@ export class QdPushEventsService {
161
161
  if (!this._eventSource || this._eventSource.readyState === EventSource.CLOSED || reconnect)
162
162
  this._eventSource = new EventSourcePolyfill(url, options);
163
163
  this._eventSource.onerror = (err) => {
164
+ const status = err?.status || err?.target?.status;
165
+ if (status === 401) {
166
+ this.logError('SSE connection unauthorized (401):', err);
167
+ this._eventSource.close();
168
+ return;
169
+ }
164
170
  if (this._eventSource.readyState === EventSource.CLOSED)
165
171
  this.reconnect();
166
172
  this.logError('SSE connection error:', err);
@@ -188,4 +194,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
188
194
  type: Inject,
189
195
  args: ['QdAuthenticationService']
190
196
  }] }] });
191
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"push-events.service.js","sourceRoot":"","sources":["../../../../../../libs/qd-ui/src/lib/core/services/push-events.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, Optional } from '@angular/core';\nimport { EventSourcePolyfill } from 'event-source-polyfill';\nimport { NEVER, Observable, Subject, Subscription } from 'rxjs';\n\nexport type EventName = 'RESOURCE_CREATED' | 'RESOURCE_UPDATED' | 'RESOURCE_DELETED';\n\n/**\n * Service for handling real-time server-sent events.\n *\n * ### Benefits\n *\n * - **Real-Time Interaction**: Enables live updates without user actions or polling.\n * - **Scalable and Flexible**: The system is flexible and easily expandable to handle new events.\n * - **Consistency**: Ensures synchronization between server and client states for a consistent user experience.\n * - **Authorization**: It uses the user's authentication (QdAuth) to secure the connection against the backend out-of-the-box. This requires a proper QdAuth setup. You can disable this feature when starting the connection.\n *\n * ### Usage\n *\n * ```ts\n * // Start the connection to the events\n * pushEventsService.connect('http://service-endpint/events');\n * // or start without authentication in case you don't need:\n * pushEventsService.connect('http://service-endpint/events', { disableAuthentication: true });\n *\n * // Subscribe to the event (Side Effect)\n * const subscription = pushEventsService.observe('RESOURCE_UPDATED').subscribe(event => handleEvent(event));\n *\n * // Unsubscribe to prevent memory leaks\n * subscription.unsubscribe();\n *\n * // Disconnect the connection\n * pushEventsService.disconnect();\n * ```\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class QdPushEventsService {\n  private _eventSource!: EventSource;\n  private _eventSubscriptionSubjects = new Map<string, Subject<MessageEvent>>();\n  private _listeners: [string, (message: MessageEvent) => void][] = [];\n  private _heartbeatTimeout!: ReturnType<typeof setTimeout>;\n  private _reconnectDelayTime = 10000;\n  private _heartbeatReconnectDelayTime = 100;\n  private _accessTokenSub?: Subscription;\n  private _options!: { url: string; options?: { disableAuthentication: boolean } };\n\n  constructor(@Optional() @Inject('QdAuthenticationService') private readonly authenticationService: any) {}\n\n  /**\n   * Establishes an EventSource connection to the given URL.\n   * Automatically reconnects if heartbeat fails or is delayed.\n   * Subscribers are retained across reconnections.\n   *\n   * @param url The backend URL for the event stream.\n   * @param options Options for the connection\n   */\n  connect(url: string, options?: { disableAuthentication: boolean }): void {\n    if (this.isConnectedOrConnecting()) return;\n\n    if (!options?.disableAuthentication) {\n      if (!this.authenticationService) {\n        this.logError(\n          \"Can't connect to SSE without QdAuth as the connection has to be secured. Please install QdAuth or disable authentication when connecting.\"\n        );\n\n        return;\n      }\n\n      if (this._accessTokenSub) this._accessTokenSub.unsubscribe();\n\n      this._accessTokenSub = this.authenticationService.accessToken$.subscribe((token: string) => {\n        this._options = { url, options };\n        this.connectEventSource(\n          url,\n          {\n            headers: { Authorization: `Bearer ${token}` }\n          },\n          true\n        );\n      });\n    } else {\n      this._options = { url, options };\n      this.connectEventSource(url);\n    }\n  }\n\n  /**\n   * Closes the EventSource connection and clears all listeners.\n   * Subscribers are preserved for reconnection.\n   */\n  disconnect(): void {\n    if (!this._eventSource) {\n      this.logWarn('No active connection to disconnect.');\n\n      return;\n    }\n\n    this.removeAllEventListenersFromEventSource();\n\n    this._eventSource.close();\n  }\n\n  /**\n   * Returns an Observable for the specified event name.\n   * If not connected, returns `NEVER` and logs an error.\n   * Automatically adds a listener for the event if needed.\n   *\n   * @param eventName The event type ('RESOURCE_CREATED', 'RESOURCE_UPDATED', 'RESOURCE_DELETED').\n   * @returns Observable<MessageEvent> The event stream.\n   */\n  observe(eventName: EventName): Observable<MessageEvent> {\n    if (!this._eventSource) {\n      this.logError('Cannot observe events without a connection. Call connect() first.');\n\n      return NEVER;\n    }\n\n    if (!this._eventSubscriptionSubjects.has(eventName)) {\n      this._eventSubscriptionSubjects.set(eventName, new Subject<MessageEvent>());\n\n      this.addEventListenerForEventName(eventName);\n    }\n\n    return this._eventSubscriptionSubjects.get(eventName)!.asObservable();\n  }\n\n  /**\n   * Removes all listeners and clears all subscriptions.\n   * The EventSource connection remains open.\n   */\n  unobserveAll(): void {\n    this.removeAllEventListenersFromEventSource();\n    this._eventSubscriptionSubjects.clear();\n  }\n\n  /**\n   * Checks if the EventSource is connected or in the process of connecting.\n   */\n  isConnectedOrConnecting(): boolean {\n    return (\n      this._eventSource &&\n      (this._eventSource.readyState === EventSource.OPEN || this._eventSource.readyState === EventSource.CONNECTING)\n    );\n  }\n\n  private reconnect(): void {\n    if (this.isConnectedOrConnecting()) return;\n\n    this.disconnect();\n    this.connect(this._options.url, this._options.options);\n  }\n\n  private addEventListenerForEventName(eventName: EventName): void {\n    const callback = (messageEvent: MessageEvent): void => {\n      const subject = this._eventSubscriptionSubjects.get(eventName);\n\n      if (subject) subject.next(messageEvent);\n    };\n\n    this._eventSource.addEventListener(eventName, callback);\n    this._listeners.push([eventName, callback]);\n  }\n\n  private addEventListenersForExistingSubscriptions(): void {\n    this._eventSubscriptionSubjects.forEach((subject, eventName) =>\n      this.addEventListenerToEventSource(eventName as EventName, (messageEvent: MessageEvent) =>\n        subject.next(messageEvent)\n      )\n    );\n  }\n\n  private addEventListenerToEventSource(eventName: EventName, callback: (messageEvent: MessageEvent) => void): void {\n    this._eventSource.addEventListener(eventName, callback);\n    this._listeners.push([eventName, callback]);\n  }\n\n  private removeAllEventListenersFromEventSource(): void {\n    if (!this._eventSource) {\n      this.logWarn('Cannot remove listeners: No active connection.');\n\n      return;\n    }\n\n    this._listeners.forEach(([eventName, callback]) => this._eventSource.removeEventListener(eventName, callback));\n    this._listeners = [];\n  }\n\n  private logWarn(message: string): void {\n    console.warn(`QD-UI | QdPushEventsService - ${message}`);\n  }\n\n  private logError(message: string, err?: Event): void {\n    console.error(`QD-UI | QdPushEventsService - ${message}`, err);\n  }\n\n  private connectEventSource(url: string, options: unknown = {}, reconnect?: boolean): void {\n    if (this._eventSource && this._eventSource.readyState !== EventSource.CLOSED) this._eventSource.close();\n\n    if (!this._eventSource || this._eventSource.readyState === EventSource.CLOSED || reconnect)\n      this._eventSource = new EventSourcePolyfill(url, options);\n\n    this._eventSource.onerror = (err: Event): void => {\n      if (this._eventSource.readyState === EventSource.CLOSED) this.reconnect();\n\n      this.logError('SSE connection error:', err);\n    };\n\n    this._eventSource.addEventListener('HEARTBEAT', (message: MessageEvent) => {\n      if (this._heartbeatTimeout) clearTimeout(this._heartbeatTimeout);\n\n      const interval = JSON.parse(message.data).interval;\n      this._heartbeatTimeout = setTimeout(() => this.reconnect(), interval + this._heartbeatReconnectDelayTime);\n    });\n\n    this._heartbeatTimeout = setTimeout(() => this.reconnect(), this._reconnectDelayTime);\n\n    this.addEventListenersForExistingSubscriptions();\n  }\n}\n"]}
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"push-events.service.js","sourceRoot":"","sources":["../../../../../../libs/qd-ui/src/lib/core/services/push-events.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, Optional } from '@angular/core';\nimport { EventSourcePolyfill } from 'event-source-polyfill';\nimport { NEVER, Observable, Subject, Subscription } from 'rxjs';\n\nexport type EventName = 'RESOURCE_CREATED' | 'RESOURCE_UPDATED' | 'RESOURCE_DELETED';\n\n/**\n * Service for handling real-time server-sent events.\n *\n * ### Benefits\n *\n * - **Real-Time Interaction**: Enables live updates without user actions or polling.\n * - **Scalable and Flexible**: The system is flexible and easily expandable to handle new events.\n * - **Consistency**: Ensures synchronization between server and client states for a consistent user experience.\n * - **Authorization**: It uses the user's authentication (QdAuth) to secure the connection against the backend out-of-the-box. This requires a proper QdAuth setup. You can disable this feature when starting the connection.\n *\n * ### Usage\n *\n * ```ts\n * // Start the connection to the events\n * pushEventsService.connect('http://service-endpint/events');\n * // or start without authentication in case you don't need:\n * pushEventsService.connect('http://service-endpint/events', { disableAuthentication: true });\n *\n * // Subscribe to the event (Side Effect)\n * const subscription = pushEventsService.observe('RESOURCE_UPDATED').subscribe(event => handleEvent(event));\n *\n * // Unsubscribe to prevent memory leaks\n * subscription.unsubscribe();\n *\n * // Disconnect the connection\n * pushEventsService.disconnect();\n * ```\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class QdPushEventsService {\n  private _eventSource!: EventSource;\n  private _eventSubscriptionSubjects = new Map<string, Subject<MessageEvent>>();\n  private _listeners: [string, (message: MessageEvent) => void][] = [];\n  private _heartbeatTimeout!: ReturnType<typeof setTimeout>;\n  private _reconnectDelayTime = 10000;\n  private _heartbeatReconnectDelayTime = 100;\n  private _accessTokenSub?: Subscription;\n  private _options!: { url: string; options?: { disableAuthentication: boolean } };\n\n  constructor(@Optional() @Inject('QdAuthenticationService') private readonly authenticationService: any) {}\n\n  /**\n   * Establishes an EventSource connection to the given URL.\n   * Automatically reconnects if heartbeat fails or is delayed.\n   * Subscribers are retained across reconnections.\n   *\n   * @param url The backend URL for the event stream.\n   * @param options Options for the connection\n   */\n  connect(url: string, options?: { disableAuthentication: boolean }): void {\n    if (this.isConnectedOrConnecting()) return;\n\n    if (!options?.disableAuthentication) {\n      if (!this.authenticationService) {\n        this.logError(\n          \"Can't connect to SSE without QdAuth as the connection has to be secured. Please install QdAuth or disable authentication when connecting.\"\n        );\n\n        return;\n      }\n\n      if (this._accessTokenSub) this._accessTokenSub.unsubscribe();\n\n      this._accessTokenSub = this.authenticationService.accessToken$.subscribe((token: string) => {\n        this._options = { url, options };\n        this.connectEventSource(\n          url,\n          {\n            headers: { Authorization: `Bearer ${token}` }\n          },\n          true\n        );\n      });\n    } else {\n      this._options = { url, options };\n      this.connectEventSource(url);\n    }\n  }\n\n  /**\n   * Closes the EventSource connection and clears all listeners.\n   * Subscribers are preserved for reconnection.\n   */\n  disconnect(): void {\n    if (!this._eventSource) {\n      this.logWarn('No active connection to disconnect.');\n\n      return;\n    }\n\n    this.removeAllEventListenersFromEventSource();\n\n    this._eventSource.close();\n  }\n\n  /**\n   * Returns an Observable for the specified event name.\n   * If not connected, returns `NEVER` and logs an error.\n   * Automatically adds a listener for the event if needed.\n   *\n   * @param eventName The event type ('RESOURCE_CREATED', 'RESOURCE_UPDATED', 'RESOURCE_DELETED').\n   * @returns Observable<MessageEvent> The event stream.\n   */\n  observe(eventName: EventName): Observable<MessageEvent> {\n    if (!this._eventSource) {\n      this.logError('Cannot observe events without a connection. Call connect() first.');\n\n      return NEVER;\n    }\n\n    if (!this._eventSubscriptionSubjects.has(eventName)) {\n      this._eventSubscriptionSubjects.set(eventName, new Subject<MessageEvent>());\n\n      this.addEventListenerForEventName(eventName);\n    }\n\n    return this._eventSubscriptionSubjects.get(eventName)!.asObservable();\n  }\n\n  /**\n   * Removes all listeners and clears all subscriptions.\n   * The EventSource connection remains open.\n   */\n  unobserveAll(): void {\n    this.removeAllEventListenersFromEventSource();\n    this._eventSubscriptionSubjects.clear();\n  }\n\n  /**\n   * Checks if the EventSource is connected or in the process of connecting.\n   */\n  isConnectedOrConnecting(): boolean {\n    return (\n      this._eventSource &&\n      (this._eventSource.readyState === EventSource.OPEN || this._eventSource.readyState === EventSource.CONNECTING)\n    );\n  }\n\n  private reconnect(): void {\n    if (this.isConnectedOrConnecting()) return;\n\n    this.disconnect();\n    this.connect(this._options.url, this._options.options);\n  }\n\n  private addEventListenerForEventName(eventName: EventName): void {\n    const callback = (messageEvent: MessageEvent): void => {\n      const subject = this._eventSubscriptionSubjects.get(eventName);\n\n      if (subject) subject.next(messageEvent);\n    };\n\n    this._eventSource.addEventListener(eventName, callback);\n    this._listeners.push([eventName, callback]);\n  }\n\n  private addEventListenersForExistingSubscriptions(): void {\n    this._eventSubscriptionSubjects.forEach((subject, eventName) =>\n      this.addEventListenerToEventSource(eventName as EventName, (messageEvent: MessageEvent) =>\n        subject.next(messageEvent)\n      )\n    );\n  }\n\n  private addEventListenerToEventSource(eventName: EventName, callback: (messageEvent: MessageEvent) => void): void {\n    this._eventSource.addEventListener(eventName, callback);\n    this._listeners.push([eventName, callback]);\n  }\n\n  private removeAllEventListenersFromEventSource(): void {\n    if (!this._eventSource) {\n      this.logWarn('Cannot remove listeners: No active connection.');\n\n      return;\n    }\n\n    this._listeners.forEach(([eventName, callback]) => this._eventSource.removeEventListener(eventName, callback));\n    this._listeners = [];\n  }\n\n  private logWarn(message: string): void {\n    console.warn(`QD-UI | QdPushEventsService - ${message}`);\n  }\n\n  private logError(message: string, err?: Event): void {\n    console.error(`QD-UI | QdPushEventsService - ${message}`, err);\n  }\n\n  private connectEventSource(url: string, options: unknown = {}, reconnect?: boolean): void {\n    if (this._eventSource && this._eventSource.readyState !== EventSource.CLOSED) this._eventSource.close();\n\n    if (!this._eventSource || this._eventSource.readyState === EventSource.CLOSED || reconnect)\n      this._eventSource = new EventSourcePolyfill(url, options);\n\n    this._eventSource.onerror = (err: any): void => {\n      const status = err?.status || err?.target?.status;\n\n      if (status === 401) {\n        this.logError('SSE connection unauthorized (401):', err);\n        this._eventSource.close();\n\n        return;\n      }\n\n      if (this._eventSource.readyState === EventSource.CLOSED) this.reconnect();\n\n      this.logError('SSE connection error:', err);\n    };\n\n    this._eventSource.addEventListener('HEARTBEAT', (message: MessageEvent) => {\n      if (this._heartbeatTimeout) clearTimeout(this._heartbeatTimeout);\n\n      const interval = JSON.parse(message.data).interval;\n      this._heartbeatTimeout = setTimeout(() => this.reconnect(), interval + this._heartbeatReconnectDelayTime);\n    });\n\n    this._heartbeatTimeout = setTimeout(() => this.reconnect(), this._reconnectDelayTime);\n\n    this.addEventListenersForExistingSubscriptions();\n  }\n}\n"]}
@@ -1,4 +1,4 @@
1
- import { Component, ElementRef, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core';
1
+ import { ChangeDetectorRef, Component, ElementRef, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core';
2
2
  import { v4 as uuid } from 'uuid';
3
3
  import { QdFilterService } from '../filter/filter-service/filter.service';
4
4
  import { QdSearchService } from '../search/services/search.service';
@@ -68,6 +68,7 @@ export class QdSectionComponent {
68
68
  actionService;
69
69
  filterService;
70
70
  elementRef;
71
+ cdr;
71
72
  /**
72
73
  * The configuration of the section.
73
74
  */
@@ -104,17 +105,21 @@ export class QdSectionComponent {
104
105
  filterValueChangeOutput = new EventEmitter();
105
106
  noHorizontalPaddingMobile = false;
106
107
  _outputSubscriptions = [];
107
- constructor(searchService, actionService, filterService, elementRef) {
108
+ constructor(searchService, actionService, filterService, elementRef, cdr) {
108
109
  this.searchService = searchService;
109
110
  this.actionService = actionService;
110
111
  this.filterService = filterService;
111
112
  this.elementRef = elementRef;
113
+ this.cdr = cdr;
112
114
  }
113
115
  ngOnInit() {
114
116
  this.subscribeForOutputs();
115
117
  }
116
118
  ngAfterViewInit() {
117
- this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector('[data-qd-no-horizontal-padding-small]');
119
+ Promise.resolve().then(() => {
120
+ this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector('[data-qd-no-horizontal-padding-small]');
121
+ this.cdr.detectChanges();
122
+ });
118
123
  }
119
124
  ngOnChanges(changes) {
120
125
  if (changes['config'] && !changes['config'].firstChange) {
@@ -158,7 +163,7 @@ export class QdSectionComponent {
158
163
  this._outputSubscriptions.forEach(subscription => subscription.unsubscribe());
159
164
  this._outputSubscriptions = [];
160
165
  }
161
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, deps: [{ token: i1.QdSearchService }, { token: i2.QdSectionToolbarActionService }, { token: i3.QdFilterService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
166
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, deps: [{ token: i1.QdSearchService }, { token: i2.QdSectionToolbarActionService }, { token: i3.QdFilterService }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
162
167
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdSectionComponent, selector: "qd-section", inputs: { config: "config" }, outputs: { searchQueryStringOutput: "searchQueryStringOutput", searchPostBodyOutput: "searchPostBodyOutput", actionOutput: "actionOutput", filterQueryStringOutput: "filterQueryStringOutput", filterPostBodyOutput: "filterPostBodyOutput", filterValueChangeOutput: "filterValueChangeOutput" }, host: { properties: { "class.no-horizontal-padding-mobile": "noHorizontalPaddingMobile" }, classAttribute: "qd-section" }, providers: [QdSectionToolbarActionService, QdSearchService, QdToolbarComponentsService], usesOnChanges: true, ngImport: i0, template: "<section>\n <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n", styles: ["qd-section{display:block;padding:.9375rem 1.25rem 3.125rem;background-color:#efefef}@media (max-width: 599.98px){qd-section{padding:.9375rem .9375rem 3.125rem}qd-section.no-horizontal-padding-mobile{padding:.9375rem 0 3.125rem}qd-section.no-horizontal-padding-mobile qd-section-toolbar{margin:0 .9375rem .3125rem}}.qd-section{display:block}.qd-section:has(+.qd-section){border-bottom:rgb(213,213,213) solid .0625rem}@media (max-width: 959.98px){.qd-section{padding:.75rem}}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.QdSectionToolbarComponent, selector: "qd-section-toolbar", inputs: ["config"] }], encapsulation: i0.ViewEncapsulation.None });
163
168
  }
164
169
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, decorators: [{
@@ -167,7 +172,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
167
172
  class: 'qd-section',
168
173
  '[class.no-horizontal-padding-mobile]': 'noHorizontalPaddingMobile'
169
174
  }, encapsulation: ViewEncapsulation.None, template: "<section>\n <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n", styles: ["qd-section{display:block;padding:.9375rem 1.25rem 3.125rem;background-color:#efefef}@media (max-width: 599.98px){qd-section{padding:.9375rem .9375rem 3.125rem}qd-section.no-horizontal-padding-mobile{padding:.9375rem 0 3.125rem}qd-section.no-horizontal-padding-mobile qd-section-toolbar{margin:0 .9375rem .3125rem}}.qd-section{display:block}.qd-section:has(+.qd-section){border-bottom:rgb(213,213,213) solid .0625rem}@media (max-width: 959.98px){.qd-section{padding:.75rem}}\n"] }]
170
- }], ctorParameters: () => [{ type: i1.QdSearchService }, { type: i2.QdSectionToolbarActionService }, { type: i3.QdFilterService }, { type: i0.ElementRef }], propDecorators: { config: [{
175
+ }], ctorParameters: () => [{ type: i1.QdSearchService }, { type: i2.QdSectionToolbarActionService }, { type: i3.QdFilterService }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }], propDecorators: { config: [{
171
176
  type: Input
172
177
  }], searchQueryStringOutput: [{
173
178
  type: Output
@@ -182,4 +187,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
182
187
  }], filterValueChangeOutput: [{
183
188
  type: Output
184
189
  }] } });
185
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"section.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/section/section.component.ts","../../../../../libs/qd-ui/src/lib/section/section.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewEncapsulation\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { v4 as uuid } from 'uuid';\n\nimport { QdFilterService } from '../filter/filter-service/filter.service';\nimport { QdFilterPostBodyData } from '../filter/model/filter-post-body.interface';\nimport { QdSearchPostBodyData } from '../search/model/search-post-body';\nimport { QdSearchService } from '../search/services/search.service';\nimport { QdSectionActionType, QdSectionConfig } from './model/section-config.interface';\nimport { QdSectionToolbarActionService } from './services/section-toolbar-action.service';\nimport { QdToolbarComponentsService } from './services/toolbar-components.service';\n\n/**\n * **QdSectionComponent** is a flexible container pattern for organizing UI elements like tables or forms.\n * It is a part of the **Quadrel Layout System (QLS)** and supports responsive design and includes built-in features for actions, search, and filtering.\n *\n * #### **Features**\n *\n * - **Customizable**: Fully configurable with the `QdSectionConfig` API.\n * - **Toolbar Integration**: Add actions, search, and filters directly to the toolbar.\n * - **Action Management**: Use predefined or custom actions for user interactions.\n * - **Search and Filter**: Improve data presentation with integrated search and filtering.\n *\n * #### **Interactions**\n *\n * - **Actions**: Define actions with custom labels, icons, and handlers.\n * - **Search**: Add a field to search queries and results.\n * - **Filter**: Use filters to refine the section’s displayed data.\n *\n * #### **Toolbar for Interaction**\n *\n * The **QdSection** supports toolbar tools that interact directly with its content.\n *\n * - Tools like **QdFilter** and **QdSearch** can filter or search a **QdTable** dynamically.\n * - Use directives such as ConnectToFilter, ConnectToSearch, ConnectToContext on the target component to enable this interaction.\n * - These connections ensure smooth communication between the toolbar and the section content.\n * - And much more...\n *\n * #### **Content Usage Rules**\n *\n * Each **QdSection** should contain **only one type of content**.\n * For instance:\n * - Use a **QdTable** or **QdForm** components within a **QdSection**, but not both at the same time.\n * - For multiple content types, create separate instances of **QdSectionComponent**.\n *\n * This ensures clean design and separation of concerns in your UI.\n *\n * #### **Usage**\n *\n * ```typescript\n * @Component({\n *   selector: 'app-my-section',\n *   template: `\n *     <qd-section [config]=\"sectionConfig\">\n *       <!-- Your content here -->\n *     </qd-section>\n *   `\n * })\n * export class MySectionComponent {\n *   sectionConfig: QdSectionConfig = {\n *     // Configuration options here\n *   };\n * }\n * ```\n */\n@Component({\n  selector: 'qd-section',\n  templateUrl: './section.component.html',\n  styleUrls: ['./section.component.scss'],\n  providers: [QdSectionToolbarActionService, QdSearchService, QdToolbarComponentsService],\n  host: {\n    class: 'qd-section',\n    '[class.no-horizontal-padding-mobile]': 'noHorizontalPaddingMobile'\n  },\n  encapsulation: ViewEncapsulation.None\n})\nexport class QdSectionComponent implements OnInit, AfterViewInit, OnChanges, OnDestroy {\n  /**\n   * The configuration of the section.\n   */\n  @Input() config: QdSectionConfig = {};\n\n  /**\n   * Emits an event when the search functionality integrated in the toolbar has been triggered. <br />\n   * Return value is the query string of the search.\n   */\n  @Output() searchQueryStringOutput = new EventEmitter<string>();\n\n  /**\n   * Emits an event when the search functionality integrated in the toolbar has been triggered. <br />\n   * Return value is the result object of the search.\n   */\n  @Output() searchPostBodyOutput = new EventEmitter<QdSearchPostBodyData>();\n\n  /**\n   * @description Sends an event when the toolbar action button is clicked or tapped.\n   *\n   * The event includes the action type or `null` if no specific action type is defined.\n   */\n  @Output() actionOutput = new EventEmitter<QdSectionActionType | null>();\n\n  /**\n   * Emits event on click/tap of the filter/reset button of the filter or on deletion of an active item. <br />\n   * Return value is the query string of the filter. The filter is integrated in the toolbar.\n   */\n  @Output() filterQueryStringOutput = new EventEmitter<string>();\n\n  /**\n   * Emits event on click/tap of the filter/reset button of the filter or on deletion of an active item. <br />\n   * Return value is the result object of the filter. The filter is integrated in the toolbar.\n   */\n  @Output() filterPostBodyOutput = new EventEmitter<QdFilterPostBodyData>();\n\n  /**\n   * Emits event when the values of the filter change. Return value matches the current object of the filter.\n   */\n  @Output() filterValueChangeOutput = new EventEmitter<QdFilterPostBodyData>();\n\n  noHorizontalPaddingMobile = false;\n\n  private _outputSubscriptions: Subscription[] = [];\n\n  constructor(\n    private readonly searchService: QdSearchService,\n    private readonly actionService: QdSectionToolbarActionService,\n    private readonly filterService: QdFilterService,\n    private readonly elementRef: ElementRef\n  ) {}\n\n  ngOnInit(): void {\n    this.subscribeForOutputs();\n  }\n\n  ngAfterViewInit(): void {\n    this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector(\n      '[data-qd-no-horizontal-padding-small]'\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] && !changes['config'].firstChange) {\n      this.unsubscribeOutputSubscriptions();\n      this.subscribeForOutputs();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.unsubscribeOutputSubscriptions();\n  }\n\n  private subscribeForOutputs(): void {\n    this.subscribeForSearchOutputs();\n    this.subscribeForActionOutput();\n    this.subscribeForFilterOutputs();\n  }\n\n  private subscribeForSearchOutputs(): void {\n    if (!this.config.search) return;\n\n    this._outputSubscriptions.push(\n      this.searchService.searchQueryString$.subscribe(queryString => this.searchQueryStringOutput.emit(queryString))\n    );\n\n    this._outputSubscriptions.push(\n      this.searchService.searchPostBody$.subscribe(postBody => this.searchPostBodyOutput.emit(postBody))\n    );\n  }\n\n  private subscribeForActionOutput(): void {\n    this._outputSubscriptions.push(\n      this.actionService.action$.subscribe((type?) => this.actionOutput.emit(type || null))\n    );\n  }\n\n  private subscribeForFilterOutputs(): void {\n    if (!this.config.filter) return;\n\n    if (!this.config.filter.uid) this.config.filter.uid = uuid();\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getQueryString$(this.config.filter.uid)\n        .subscribe(queryString => this.filterQueryStringOutput.emit(queryString))\n    );\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getPostBody$(this.config.filter.uid)\n        .subscribe(postBody => this.filterPostBodyOutput.emit(postBody))\n    );\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getCurrentPostBody(this.config.filter.uid)\n        .subscribe(filterValue => this.filterValueChangeOutput.emit(filterValue))\n    );\n  }\n\n  private unsubscribeOutputSubscriptions(): void {\n    this._outputSubscriptions.forEach(subscription => subscription.unsubscribe());\n\n    this._outputSubscriptions = [];\n  }\n}\n","<section>\n  <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n  <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"section.component.js","sourceRoot":"","sources":["../../../../../libs/qd-ui/src/lib/section/section.component.ts","../../../../../libs/qd-ui/src/lib/section/section.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewEncapsulation\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { v4 as uuid } from 'uuid';\n\nimport { QdFilterService } from '../filter/filter-service/filter.service';\nimport { QdFilterPostBodyData } from '../filter/model/filter-post-body.interface';\nimport { QdSearchPostBodyData } from '../search/model/search-post-body';\nimport { QdSearchService } from '../search/services/search.service';\nimport { QdSectionActionType, QdSectionConfig } from './model/section-config.interface';\nimport { QdSectionToolbarActionService } from './services/section-toolbar-action.service';\nimport { QdToolbarComponentsService } from './services/toolbar-components.service';\n\n/**\n * **QdSectionComponent** is a flexible container pattern for organizing UI elements like tables or forms.\n * It is a part of the **Quadrel Layout System (QLS)** and supports responsive design and includes built-in features for actions, search, and filtering.\n *\n * #### **Features**\n *\n * - **Customizable**: Fully configurable with the `QdSectionConfig` API.\n * - **Toolbar Integration**: Add actions, search, and filters directly to the toolbar.\n * - **Action Management**: Use predefined or custom actions for user interactions.\n * - **Search and Filter**: Improve data presentation with integrated search and filtering.\n *\n * #### **Interactions**\n *\n * - **Actions**: Define actions with custom labels, icons, and handlers.\n * - **Search**: Add a field to search queries and results.\n * - **Filter**: Use filters to refine the section’s displayed data.\n *\n * #### **Toolbar for Interaction**\n *\n * The **QdSection** supports toolbar tools that interact directly with its content.\n *\n * - Tools like **QdFilter** and **QdSearch** can filter or search a **QdTable** dynamically.\n * - Use directives such as ConnectToFilter, ConnectToSearch, ConnectToContext on the target component to enable this interaction.\n * - These connections ensure smooth communication between the toolbar and the section content.\n * - And much more...\n *\n * #### **Content Usage Rules**\n *\n * Each **QdSection** should contain **only one type of content**.\n * For instance:\n * - Use a **QdTable** or **QdForm** components within a **QdSection**, but not both at the same time.\n * - For multiple content types, create separate instances of **QdSectionComponent**.\n *\n * This ensures clean design and separation of concerns in your UI.\n *\n * #### **Usage**\n *\n * ```typescript\n * @Component({\n *   selector: 'app-my-section',\n *   template: `\n *     <qd-section [config]=\"sectionConfig\">\n *       <!-- Your content here -->\n *     </qd-section>\n *   `\n * })\n * export class MySectionComponent {\n *   sectionConfig: QdSectionConfig = {\n *     // Configuration options here\n *   };\n * }\n * ```\n */\n@Component({\n  selector: 'qd-section',\n  templateUrl: './section.component.html',\n  styleUrls: ['./section.component.scss'],\n  providers: [QdSectionToolbarActionService, QdSearchService, QdToolbarComponentsService],\n  host: {\n    class: 'qd-section',\n    '[class.no-horizontal-padding-mobile]': 'noHorizontalPaddingMobile'\n  },\n  encapsulation: ViewEncapsulation.None\n})\nexport class QdSectionComponent implements OnInit, AfterViewInit, OnChanges, OnDestroy {\n  /**\n   * The configuration of the section.\n   */\n  @Input() config: QdSectionConfig = {};\n\n  /**\n   * Emits an event when the search functionality integrated in the toolbar has been triggered. <br />\n   * Return value is the query string of the search.\n   */\n  @Output() searchQueryStringOutput = new EventEmitter<string>();\n\n  /**\n   * Emits an event when the search functionality integrated in the toolbar has been triggered. <br />\n   * Return value is the result object of the search.\n   */\n  @Output() searchPostBodyOutput = new EventEmitter<QdSearchPostBodyData>();\n\n  /**\n   * @description Sends an event when the toolbar action button is clicked or tapped.\n   *\n   * The event includes the action type or `null` if no specific action type is defined.\n   */\n  @Output() actionOutput = new EventEmitter<QdSectionActionType | null>();\n\n  /**\n   * Emits event on click/tap of the filter/reset button of the filter or on deletion of an active item. <br />\n   * Return value is the query string of the filter. The filter is integrated in the toolbar.\n   */\n  @Output() filterQueryStringOutput = new EventEmitter<string>();\n\n  /**\n   * Emits event on click/tap of the filter/reset button of the filter or on deletion of an active item. <br />\n   * Return value is the result object of the filter. The filter is integrated in the toolbar.\n   */\n  @Output() filterPostBodyOutput = new EventEmitter<QdFilterPostBodyData>();\n\n  /**\n   * Emits event when the values of the filter change. Return value matches the current object of the filter.\n   */\n  @Output() filterValueChangeOutput = new EventEmitter<QdFilterPostBodyData>();\n\n  noHorizontalPaddingMobile = false;\n\n  private _outputSubscriptions: Subscription[] = [];\n\n  constructor(\n    private readonly searchService: QdSearchService,\n    private readonly actionService: QdSectionToolbarActionService,\n    private readonly filterService: QdFilterService,\n    private readonly elementRef: ElementRef,\n    private readonly cdr: ChangeDetectorRef\n  ) {}\n\n  ngOnInit(): void {\n    this.subscribeForOutputs();\n  }\n\n  ngAfterViewInit(): void {\n    Promise.resolve().then(() => {\n      this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector(\n        '[data-qd-no-horizontal-padding-small]'\n      );\n      this.cdr.detectChanges();\n    });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['config'] && !changes['config'].firstChange) {\n      this.unsubscribeOutputSubscriptions();\n      this.subscribeForOutputs();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.unsubscribeOutputSubscriptions();\n  }\n\n  private subscribeForOutputs(): void {\n    this.subscribeForSearchOutputs();\n    this.subscribeForActionOutput();\n    this.subscribeForFilterOutputs();\n  }\n\n  private subscribeForSearchOutputs(): void {\n    if (!this.config.search) return;\n\n    this._outputSubscriptions.push(\n      this.searchService.searchQueryString$.subscribe(queryString => this.searchQueryStringOutput.emit(queryString))\n    );\n\n    this._outputSubscriptions.push(\n      this.searchService.searchPostBody$.subscribe(postBody => this.searchPostBodyOutput.emit(postBody))\n    );\n  }\n\n  private subscribeForActionOutput(): void {\n    this._outputSubscriptions.push(\n      this.actionService.action$.subscribe((type?) => this.actionOutput.emit(type || null))\n    );\n  }\n\n  private subscribeForFilterOutputs(): void {\n    if (!this.config.filter) return;\n\n    if (!this.config.filter.uid) this.config.filter.uid = uuid();\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getQueryString$(this.config.filter.uid)\n        .subscribe(queryString => this.filterQueryStringOutput.emit(queryString))\n    );\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getPostBody$(this.config.filter.uid)\n        .subscribe(postBody => this.filterPostBodyOutput.emit(postBody))\n    );\n\n    this._outputSubscriptions.push(\n      this.filterService\n        .getCurrentPostBody(this.config.filter.uid)\n        .subscribe(filterValue => this.filterValueChangeOutput.emit(filterValue))\n    );\n  }\n\n  private unsubscribeOutputSubscriptions(): void {\n    this._outputSubscriptions.forEach(subscription => subscription.unsubscribe());\n\n    this._outputSubscriptions = [];\n  }\n}\n","<section>\n  <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n  <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n"]}
@@ -8265,6 +8265,12 @@ class QdPushEventsService {
8265
8265
  if (!this._eventSource || this._eventSource.readyState === EventSource.CLOSED || reconnect)
8266
8266
  this._eventSource = new EventSourcePolyfill(url, options);
8267
8267
  this._eventSource.onerror = (err) => {
8268
+ const status = err?.status || err?.target?.status;
8269
+ if (status === 401) {
8270
+ this.logError('SSE connection unauthorized (401):', err);
8271
+ this._eventSource.close();
8272
+ return;
8273
+ }
8268
8274
  if (this._eventSource.readyState === EventSource.CLOSED)
8269
8275
  this.reconnect();
8270
8276
  this.logError('SSE connection error:', err);
@@ -31812,6 +31818,7 @@ class QdSectionComponent {
31812
31818
  actionService;
31813
31819
  filterService;
31814
31820
  elementRef;
31821
+ cdr;
31815
31822
  /**
31816
31823
  * The configuration of the section.
31817
31824
  */
@@ -31848,17 +31855,21 @@ class QdSectionComponent {
31848
31855
  filterValueChangeOutput = new EventEmitter();
31849
31856
  noHorizontalPaddingMobile = false;
31850
31857
  _outputSubscriptions = [];
31851
- constructor(searchService, actionService, filterService, elementRef) {
31858
+ constructor(searchService, actionService, filterService, elementRef, cdr) {
31852
31859
  this.searchService = searchService;
31853
31860
  this.actionService = actionService;
31854
31861
  this.filterService = filterService;
31855
31862
  this.elementRef = elementRef;
31863
+ this.cdr = cdr;
31856
31864
  }
31857
31865
  ngOnInit() {
31858
31866
  this.subscribeForOutputs();
31859
31867
  }
31860
31868
  ngAfterViewInit() {
31861
- this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector('[data-qd-no-horizontal-padding-small]');
31869
+ Promise.resolve().then(() => {
31870
+ this.noHorizontalPaddingMobile = !!this.elementRef.nativeElement.querySelector('[data-qd-no-horizontal-padding-small]');
31871
+ this.cdr.detectChanges();
31872
+ });
31862
31873
  }
31863
31874
  ngOnChanges(changes) {
31864
31875
  if (changes['config'] && !changes['config'].firstChange) {
@@ -31902,7 +31913,7 @@ class QdSectionComponent {
31902
31913
  this._outputSubscriptions.forEach(subscription => subscription.unsubscribe());
31903
31914
  this._outputSubscriptions = [];
31904
31915
  }
31905
- static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, deps: [{ token: QdSearchService }, { token: QdSectionToolbarActionService }, { token: QdFilterService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
31916
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, deps: [{ token: QdSearchService }, { token: QdSectionToolbarActionService }, { token: QdFilterService }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
31906
31917
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: QdSectionComponent, selector: "qd-section", inputs: { config: "config" }, outputs: { searchQueryStringOutput: "searchQueryStringOutput", searchPostBodyOutput: "searchPostBodyOutput", actionOutput: "actionOutput", filterQueryStringOutput: "filterQueryStringOutput", filterPostBodyOutput: "filterPostBodyOutput", filterValueChangeOutput: "filterValueChangeOutput" }, host: { properties: { "class.no-horizontal-padding-mobile": "noHorizontalPaddingMobile" }, classAttribute: "qd-section" }, providers: [QdSectionToolbarActionService, QdSearchService, QdToolbarComponentsService], usesOnChanges: true, ngImport: i0, template: "<section>\n <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n", styles: ["qd-section{display:block;padding:.9375rem 1.25rem 3.125rem;background-color:#efefef}@media (max-width: 599.98px){qd-section{padding:.9375rem .9375rem 3.125rem}qd-section.no-horizontal-padding-mobile{padding:.9375rem 0 3.125rem}qd-section.no-horizontal-padding-mobile qd-section-toolbar{margin:0 .9375rem .3125rem}}.qd-section{display:block}.qd-section:has(+.qd-section){border-bottom:rgb(213,213,213) solid .0625rem}@media (max-width: 959.98px){.qd-section{padding:.75rem}}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: QdSectionToolbarComponent, selector: "qd-section-toolbar", inputs: ["config"] }], encapsulation: i0.ViewEncapsulation.None });
31907
31918
  }
31908
31919
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: QdSectionComponent, decorators: [{
@@ -31911,7 +31922,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
31911
31922
  class: 'qd-section',
31912
31923
  '[class.no-horizontal-padding-mobile]': 'noHorizontalPaddingMobile'
31913
31924
  }, encapsulation: ViewEncapsulation.None, template: "<section>\n <qd-section-toolbar [className]=\"'qd-section__toolbar'\" [config]=\"config\"></qd-section-toolbar>\n\n <div *ngIf=\"!config.collapse?.isContentHidden\" class=\"qd-section__content\"><ng-content></ng-content></div>\n</section>\n", styles: ["qd-section{display:block;padding:.9375rem 1.25rem 3.125rem;background-color:#efefef}@media (max-width: 599.98px){qd-section{padding:.9375rem .9375rem 3.125rem}qd-section.no-horizontal-padding-mobile{padding:.9375rem 0 3.125rem}qd-section.no-horizontal-padding-mobile qd-section-toolbar{margin:0 .9375rem .3125rem}}.qd-section{display:block}.qd-section:has(+.qd-section){border-bottom:rgb(213,213,213) solid .0625rem}@media (max-width: 959.98px){.qd-section{padding:.75rem}}\n"] }]
31914
- }], ctorParameters: () => [{ type: QdSearchService }, { type: QdSectionToolbarActionService }, { type: QdFilterService }, { type: i0.ElementRef }], propDecorators: { config: [{
31925
+ }], ctorParameters: () => [{ type: QdSearchService }, { type: QdSectionToolbarActionService }, { type: QdFilterService }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }], propDecorators: { config: [{
31915
31926
  type: Input
31916
31927
  }], searchQueryStringOutput: [{
31917
31928
  type: Output