@pillar-ai/angular 0.1.11
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.
- package/LICENSE +21 -0
- package/README.md +357 -0
- package/dist/LICENSE +21 -0
- package/dist/README.md +357 -0
- package/dist/esm2022/lib/inject-help-panel.mjs +59 -0
- package/dist/esm2022/lib/inject-pillar-tool.mjs +111 -0
- package/dist/esm2022/lib/inject-pillar.mjs +79 -0
- package/dist/esm2022/lib/pillar-panel.component.mjs +80 -0
- package/dist/esm2022/lib/pillar.service.mjs +308 -0
- package/dist/esm2022/lib/types.mjs +5 -0
- package/dist/esm2022/pillar-ai-angular.mjs +5 -0
- package/dist/esm2022/public-api.mjs +67 -0
- package/dist/fesm2022/pillar-ai-angular.mjs +695 -0
- package/dist/fesm2022/pillar-ai-angular.mjs.map +1 -0
- package/dist/index.d.ts +5 -0
- package/dist/lib/inject-help-panel.d.ts +34 -0
- package/dist/lib/inject-pillar-tool.d.ts +89 -0
- package/dist/lib/inject-pillar.d.ts +96 -0
- package/dist/lib/pillar-panel.component.d.ts +58 -0
- package/dist/lib/pillar.service.d.ts +126 -0
- package/dist/lib/types.d.ts +119 -0
- package/dist/public-api.d.ts +77 -0
- package/package.json +73 -0
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PillarPanelComponent
|
|
3
|
+
* Renders the Pillar help panel at a custom location in the DOM
|
|
4
|
+
*/
|
|
5
|
+
import { Component, ViewChild, inject, effect, input, ChangeDetectionStrategy, } from '@angular/core';
|
|
6
|
+
import { PillarService } from './pillar.service';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
/**
|
|
9
|
+
* Renders the Pillar help panel at a custom location in the DOM.
|
|
10
|
+
* Use this when you want to control where the panel is rendered instead of
|
|
11
|
+
* having it automatically appended to document.body.
|
|
12
|
+
*
|
|
13
|
+
* **Important**: When using this component, set `panel.container: 'manual'` in your
|
|
14
|
+
* Pillar configuration to prevent automatic mounting.
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```typescript
|
|
18
|
+
* // app.config.ts
|
|
19
|
+
* function initPillar() {
|
|
20
|
+
* const pillar = inject(PillarService);
|
|
21
|
+
* return () => pillar.init({
|
|
22
|
+
* productKey: 'your-product-key',
|
|
23
|
+
* config: { panel: { container: 'manual' } }
|
|
24
|
+
* });
|
|
25
|
+
* }
|
|
26
|
+
*
|
|
27
|
+
* // layout.component.ts
|
|
28
|
+
* @Component({
|
|
29
|
+
* selector: 'app-layout',
|
|
30
|
+
* standalone: true,
|
|
31
|
+
* imports: [PillarPanelComponent],
|
|
32
|
+
* template: `
|
|
33
|
+
* <div class="layout">
|
|
34
|
+
* <app-sidebar />
|
|
35
|
+
* <pillar-panel class="help-panel-container" />
|
|
36
|
+
* <main>
|
|
37
|
+
* <router-outlet />
|
|
38
|
+
* </main>
|
|
39
|
+
* </div>
|
|
40
|
+
* `,
|
|
41
|
+
* })
|
|
42
|
+
* export class LayoutComponent {}
|
|
43
|
+
* ```
|
|
44
|
+
*/
|
|
45
|
+
export class PillarPanelComponent {
|
|
46
|
+
containerRef;
|
|
47
|
+
pillarService = inject(PillarService);
|
|
48
|
+
hasMounted = false;
|
|
49
|
+
effectRef = null;
|
|
50
|
+
/**
|
|
51
|
+
* Optional class to add to the container element.
|
|
52
|
+
* Use host binding for styling instead if possible.
|
|
53
|
+
*/
|
|
54
|
+
containerClass = input('');
|
|
55
|
+
ngAfterViewInit() {
|
|
56
|
+
// Use effect to react to isReady signal changes
|
|
57
|
+
this.effectRef = effect(() => {
|
|
58
|
+
const isReady = this.pillarService.isReady();
|
|
59
|
+
if (isReady && !this.hasMounted && this.containerRef?.nativeElement) {
|
|
60
|
+
this.pillarService.mountPanelTo(this.containerRef.nativeElement);
|
|
61
|
+
this.hasMounted = true;
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
ngOnDestroy() {
|
|
66
|
+
// Effect is automatically cleaned up when component is destroyed
|
|
67
|
+
// Panel cleanup is handled by PillarService
|
|
68
|
+
this.effectRef?.destroy();
|
|
69
|
+
}
|
|
70
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PillarPanelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
71
|
+
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "17.3.12", type: PillarPanelComponent, isStandalone: true, selector: "pillar-panel", inputs: { containerClass: { classPropertyName: "containerClass", publicName: "containerClass", isSignal: true, isRequired: false, transformFunction: null } }, viewQueries: [{ propertyName: "containerRef", first: true, predicate: ["container"], descendants: true, static: true }], ngImport: i0, template: '<div #container data-pillar-panel-container></div>', isInline: true, styles: [":host{display:block}\n"], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
72
|
+
}
|
|
73
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PillarPanelComponent, decorators: [{
|
|
74
|
+
type: Component,
|
|
75
|
+
args: [{ selector: 'pillar-panel', standalone: true, template: '<div #container data-pillar-panel-container></div>', changeDetection: ChangeDetectionStrategy.OnPush, styles: [":host{display:block}\n"] }]
|
|
76
|
+
}], propDecorators: { containerRef: [{
|
|
77
|
+
type: ViewChild,
|
|
78
|
+
args: ['container', { static: true }]
|
|
79
|
+
}] } });
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGlsbGFyLXBhbmVsLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9saWIvcGlsbGFyLXBhbmVsLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxPQUFPLEVBQ0wsU0FBUyxFQUVULFNBQVMsRUFHVCxNQUFNLEVBQ04sTUFBTSxFQUNOLEtBQUssRUFDTCx1QkFBdUIsR0FDeEIsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLGtCQUFrQixDQUFDOztBQUVqRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0NHO0FBY0gsTUFBTSxPQUFPLG9CQUFvQjtJQUV2QixZQUFZLENBQThCO0lBRWpDLGFBQWEsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7SUFDL0MsVUFBVSxHQUFHLEtBQUssQ0FBQztJQUNuQixTQUFTLEdBQXFDLElBQUksQ0FBQztJQUUzRDs7O09BR0c7SUFDTSxjQUFjLEdBQUcsS0FBSyxDQUFTLEVBQUUsQ0FBQyxDQUFDO0lBRTVDLGVBQWU7UUFDYixnREFBZ0Q7UUFDaEQsSUFBSSxDQUFDLFNBQVMsR0FBRyxNQUFNLENBQUMsR0FBRyxFQUFFO1lBQzNCLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxhQUFhLENBQUMsT0FBTyxFQUFFLENBQUM7WUFFN0MsSUFBSSxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsVUFBVSxJQUFJLElBQUksQ0FBQyxZQUFZLEVBQUUsYUFBYSxFQUFFLENBQUM7Z0JBQ3BFLElBQUksQ0FBQyxhQUFhLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsYUFBYSxDQUFDLENBQUM7Z0JBQ2pFLElBQUksQ0FBQyxVQUFVLEdBQUcsSUFBSSxDQUFDO1lBQ3pCLENBQUM7UUFDSCxDQUFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUFFRCxXQUFXO1FBQ1QsaUVBQWlFO1FBQ2pFLDRDQUE0QztRQUM1QyxJQUFJLENBQUMsU0FBUyxFQUFFLE9BQU8sRUFBRSxDQUFDO0lBQzVCLENBQUM7d0dBOUJVLG9CQUFvQjs0RkFBcEIsb0JBQW9CLGdXQVZyQixvREFBb0Q7OzRGQVVuRCxvQkFBb0I7a0JBYmhDLFNBQVM7K0JBQ0UsY0FBYyxjQUNaLElBQUksWUFDTixvREFBb0QsbUJBUTdDLHVCQUF1QixDQUFDLE1BQU07OEJBSXZDLFlBQVk7c0JBRG5CLFNBQVM7dUJBQUMsV0FBVyxFQUFFLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUGlsbGFyUGFuZWxDb21wb25lbnRcbiAqIFJlbmRlcnMgdGhlIFBpbGxhciBoZWxwIHBhbmVsIGF0IGEgY3VzdG9tIGxvY2F0aW9uIGluIHRoZSBET01cbiAqL1xuXG5pbXBvcnQge1xuICBDb21wb25lbnQsXG4gIEVsZW1lbnRSZWYsXG4gIFZpZXdDaGlsZCxcbiAgQWZ0ZXJWaWV3SW5pdCxcbiAgT25EZXN0cm95LFxuICBpbmplY3QsXG4gIGVmZmVjdCxcbiAgaW5wdXQsXG4gIENoYW5nZURldGVjdGlvblN0cmF0ZWd5LFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFBpbGxhclNlcnZpY2UgfSBmcm9tICcuL3BpbGxhci5zZXJ2aWNlJztcblxuLyoqXG4gKiBSZW5kZXJzIHRoZSBQaWxsYXIgaGVscCBwYW5lbCBhdCBhIGN1c3RvbSBsb2NhdGlvbiBpbiB0aGUgRE9NLlxuICogVXNlIHRoaXMgd2hlbiB5b3Ugd2FudCB0byBjb250cm9sIHdoZXJlIHRoZSBwYW5lbCBpcyByZW5kZXJlZCBpbnN0ZWFkIG9mXG4gKiBoYXZpbmcgaXQgYXV0b21hdGljYWxseSBhcHBlbmRlZCB0byBkb2N1bWVudC5ib2R5LlxuICpcbiAqICoqSW1wb3J0YW50Kio6IFdoZW4gdXNpbmcgdGhpcyBjb21wb25lbnQsIHNldCBgcGFuZWwuY29udGFpbmVyOiAnbWFudWFsJ2AgaW4geW91clxuICogUGlsbGFyIGNvbmZpZ3VyYXRpb24gdG8gcHJldmVudCBhdXRvbWF0aWMgbW91bnRpbmcuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIGFwcC5jb25maWcudHNcbiAqIGZ1bmN0aW9uIGluaXRQaWxsYXIoKSB7XG4gKiAgIGNvbnN0IHBpbGxhciA9IGluamVjdChQaWxsYXJTZXJ2aWNlKTtcbiAqICAgcmV0dXJuICgpID0+IHBpbGxhci5pbml0KHtcbiAqICAgICBwcm9kdWN0S2V5OiAneW91ci1wcm9kdWN0LWtleScsXG4gKiAgICAgY29uZmlnOiB7IHBhbmVsOiB7IGNvbnRhaW5lcjogJ21hbnVhbCcgfSB9XG4gKiAgIH0pO1xuICogfVxuICpcbiAqIC8vIGxheW91dC5jb21wb25lbnQudHNcbiAqIEBDb21wb25lbnQoe1xuICogICBzZWxlY3RvcjogJ2FwcC1sYXlvdXQnLFxuICogICBzdGFuZGFsb25lOiB0cnVlLFxuICogICBpbXBvcnRzOiBbUGlsbGFyUGFuZWxDb21wb25lbnRdLFxuICogICB0ZW1wbGF0ZTogYFxuICogICAgIDxkaXYgY2xhc3M9XCJsYXlvdXRcIj5cbiAqICAgICAgIDxhcHAtc2lkZWJhciAvPlxuICogICAgICAgPHBpbGxhci1wYW5lbCBjbGFzcz1cImhlbHAtcGFuZWwtY29udGFpbmVyXCIgLz5cbiAqICAgICAgIDxtYWluPlxuICogICAgICAgICA8cm91dGVyLW91dGxldCAvPlxuICogICAgICAgPC9tYWluPlxuICogICAgIDwvZGl2PlxuICogICBgLFxuICogfSlcbiAqIGV4cG9ydCBjbGFzcyBMYXlvdXRDb21wb25lbnQge31cbiAqIGBgYFxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdwaWxsYXItcGFuZWwnLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICB0ZW1wbGF0ZTogJzxkaXYgI2NvbnRhaW5lciBkYXRhLXBpbGxhci1wYW5lbC1jb250YWluZXI+PC9kaXY+JyxcbiAgc3R5bGVzOiBbXG4gICAgYFxuICAgICAgOmhvc3Qge1xuICAgICAgICBkaXNwbGF5OiBibG9jaztcbiAgICAgIH1cbiAgICBgLFxuICBdLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbn0pXG5leHBvcnQgY2xhc3MgUGlsbGFyUGFuZWxDb21wb25lbnQgaW1wbGVtZW50cyBBZnRlclZpZXdJbml0LCBPbkRlc3Ryb3kge1xuICBAVmlld0NoaWxkKCdjb250YWluZXInLCB7IHN0YXRpYzogdHJ1ZSB9KVxuICBwcml2YXRlIGNvbnRhaW5lclJlZiE6IEVsZW1lbnRSZWY8SFRNTERpdkVsZW1lbnQ+O1xuXG4gIHByaXZhdGUgcmVhZG9ubHkgcGlsbGFyU2VydmljZSA9IGluamVjdChQaWxsYXJTZXJ2aWNlKTtcbiAgcHJpdmF0ZSBoYXNNb3VudGVkID0gZmFsc2U7XG4gIHByaXZhdGUgZWZmZWN0UmVmOiBSZXR1cm5UeXBlPHR5cGVvZiBlZmZlY3Q+IHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIGNsYXNzIHRvIGFkZCB0byB0aGUgY29udGFpbmVyIGVsZW1lbnQuXG4gICAqIFVzZSBob3N0IGJpbmRpbmcgZm9yIHN0eWxpbmcgaW5zdGVhZCBpZiBwb3NzaWJsZS5cbiAgICovXG4gIHJlYWRvbmx5IGNvbnRhaW5lckNsYXNzID0gaW5wdXQ8c3RyaW5nPignJyk7XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIC8vIFVzZSBlZmZlY3QgdG8gcmVhY3QgdG8gaXNSZWFkeSBzaWduYWwgY2hhbmdlc1xuICAgIHRoaXMuZWZmZWN0UmVmID0gZWZmZWN0KCgpID0+IHtcbiAgICAgIGNvbnN0IGlzUmVhZHkgPSB0aGlzLnBpbGxhclNlcnZpY2UuaXNSZWFkeSgpO1xuXG4gICAgICBpZiAoaXNSZWFkeSAmJiAhdGhpcy5oYXNNb3VudGVkICYmIHRoaXMuY29udGFpbmVyUmVmPy5uYXRpdmVFbGVtZW50KSB7XG4gICAgICAgIHRoaXMucGlsbGFyU2VydmljZS5tb3VudFBhbmVsVG8odGhpcy5jb250YWluZXJSZWYubmF0aXZlRWxlbWVudCk7XG4gICAgICAgIHRoaXMuaGFzTW91bnRlZCA9IHRydWU7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICAvLyBFZmZlY3QgaXMgYXV0b21hdGljYWxseSBjbGVhbmVkIHVwIHdoZW4gY29tcG9uZW50IGlzIGRlc3Ryb3llZFxuICAgIC8vIFBhbmVsIGNsZWFudXAgaXMgaGFuZGxlZCBieSBQaWxsYXJTZXJ2aWNlXG4gICAgdGhpcy5lZmZlY3RSZWY/LmRlc3Ryb3koKTtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,308 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PillarService
|
|
3
|
+
* Angular service that initializes and manages the Pillar SDK
|
|
4
|
+
*/
|
|
5
|
+
import { Injectable, NgZone, signal, computed, inject, createComponent, ApplicationRef, EnvironmentInjector, } from '@angular/core';
|
|
6
|
+
import { Pillar, } from '@pillar-ai/sdk';
|
|
7
|
+
import * as i0 from "@angular/core";
|
|
8
|
+
export class PillarService {
|
|
9
|
+
ngZone = inject(NgZone);
|
|
10
|
+
appRef = inject(ApplicationRef);
|
|
11
|
+
environmentInjector = inject(EnvironmentInjector);
|
|
12
|
+
// Internal state
|
|
13
|
+
_pillar = signal(null);
|
|
14
|
+
cleanupFns = [];
|
|
15
|
+
cardRefs = new Map();
|
|
16
|
+
onTaskCallback = null;
|
|
17
|
+
registeredCards = null;
|
|
18
|
+
// Public signals
|
|
19
|
+
state = signal('uninitialized');
|
|
20
|
+
isReady = computed(() => this.state() === 'ready');
|
|
21
|
+
isPanelOpen = signal(false);
|
|
22
|
+
/**
|
|
23
|
+
* Get the Pillar SDK instance.
|
|
24
|
+
*/
|
|
25
|
+
getInstance() {
|
|
26
|
+
return this._pillar();
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Initialize the Pillar SDK.
|
|
30
|
+
* Call this in your app's initialization (e.g., APP_INITIALIZER).
|
|
31
|
+
*
|
|
32
|
+
* @param initConfig - Configuration options
|
|
33
|
+
*/
|
|
34
|
+
async init(initConfig) {
|
|
35
|
+
const { productKey, helpCenter, config, onTask, cards } = initConfig;
|
|
36
|
+
// Support both productKey (new) and helpCenter (deprecated)
|
|
37
|
+
const resolvedKey = productKey ?? helpCenter;
|
|
38
|
+
if (helpCenter && !productKey) {
|
|
39
|
+
console.warn('[Pillar Angular] "helpCenter" is deprecated. Use "productKey" instead.');
|
|
40
|
+
}
|
|
41
|
+
// Store callbacks for later use
|
|
42
|
+
this.onTaskCallback = onTask ?? null;
|
|
43
|
+
this.registeredCards = cards ?? null;
|
|
44
|
+
try {
|
|
45
|
+
// Pillar is a singleton - check if already initialized
|
|
46
|
+
const existingInstance = Pillar.getInstance();
|
|
47
|
+
if (existingInstance) {
|
|
48
|
+
// Reuse existing instance (preserves chat history, panel state, etc.)
|
|
49
|
+
this._pillar.set(existingInstance);
|
|
50
|
+
this.state.set(existingInstance.state);
|
|
51
|
+
this.subscribeToEvents(existingInstance);
|
|
52
|
+
this.registerCards(existingInstance);
|
|
53
|
+
return;
|
|
54
|
+
}
|
|
55
|
+
// Initialize new instance
|
|
56
|
+
const instance = await Pillar.init({
|
|
57
|
+
productKey: resolvedKey,
|
|
58
|
+
...config,
|
|
59
|
+
});
|
|
60
|
+
this.ngZone.run(() => {
|
|
61
|
+
this._pillar.set(instance);
|
|
62
|
+
this.state.set(instance.state);
|
|
63
|
+
});
|
|
64
|
+
this.subscribeToEvents(instance);
|
|
65
|
+
this.registerCards(instance);
|
|
66
|
+
}
|
|
67
|
+
catch (error) {
|
|
68
|
+
console.error('[Pillar Angular] Failed to initialize:', error);
|
|
69
|
+
this.ngZone.run(() => {
|
|
70
|
+
this.state.set('error');
|
|
71
|
+
});
|
|
72
|
+
throw error;
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Subscribe to SDK events and sync to Angular signals.
|
|
77
|
+
*/
|
|
78
|
+
subscribeToEvents(instance) {
|
|
79
|
+
// Panel open/close events
|
|
80
|
+
const unsubOpen = instance.on('panel:open', () => {
|
|
81
|
+
this.ngZone.run(() => {
|
|
82
|
+
this.isPanelOpen.set(true);
|
|
83
|
+
});
|
|
84
|
+
});
|
|
85
|
+
this.cleanupFns.push(unsubOpen);
|
|
86
|
+
const unsubClose = instance.on('panel:close', () => {
|
|
87
|
+
this.ngZone.run(() => {
|
|
88
|
+
this.isPanelOpen.set(false);
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
this.cleanupFns.push(unsubClose);
|
|
92
|
+
// State change events
|
|
93
|
+
const unsubReady = instance.on('ready', () => {
|
|
94
|
+
this.ngZone.run(() => {
|
|
95
|
+
this.state.set('ready');
|
|
96
|
+
});
|
|
97
|
+
});
|
|
98
|
+
this.cleanupFns.push(unsubReady);
|
|
99
|
+
const unsubError = instance.on('error', () => {
|
|
100
|
+
this.ngZone.run(() => {
|
|
101
|
+
this.state.set('error');
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
this.cleanupFns.push(unsubError);
|
|
105
|
+
// Task execution events
|
|
106
|
+
if (this.onTaskCallback) {
|
|
107
|
+
const callback = this.onTaskCallback;
|
|
108
|
+
const unsubTask = instance.on('task:execute', (task) => {
|
|
109
|
+
this.ngZone.run(() => {
|
|
110
|
+
callback(task);
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
this.cleanupFns.push(unsubTask);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Register custom card components.
|
|
118
|
+
*/
|
|
119
|
+
registerCards(instance) {
|
|
120
|
+
if (!this.registeredCards)
|
|
121
|
+
return;
|
|
122
|
+
Object.entries(this.registeredCards).forEach(([cardType, CardComponent]) => {
|
|
123
|
+
const unsubscribe = instance.registerCard(cardType, (container, data, callbacks) => {
|
|
124
|
+
// Create an Angular component dynamically
|
|
125
|
+
const componentRef = createComponent(CardComponent, {
|
|
126
|
+
environmentInjector: this.environmentInjector,
|
|
127
|
+
});
|
|
128
|
+
// Set inputs
|
|
129
|
+
componentRef.setInput('data', data);
|
|
130
|
+
componentRef.setInput('onConfirm', callbacks.onConfirm);
|
|
131
|
+
componentRef.setInput('onCancel', callbacks.onCancel);
|
|
132
|
+
componentRef.setInput('onStateChange', callbacks.onStateChange);
|
|
133
|
+
// Attach to the application
|
|
134
|
+
this.appRef.attachView(componentRef.hostView);
|
|
135
|
+
// Append to container
|
|
136
|
+
container.appendChild(componentRef.location.nativeElement);
|
|
137
|
+
this.cardRefs.set(container, componentRef);
|
|
138
|
+
// Return cleanup function
|
|
139
|
+
return () => {
|
|
140
|
+
const ref = this.cardRefs.get(container);
|
|
141
|
+
if (ref) {
|
|
142
|
+
this.appRef.detachView(ref.hostView);
|
|
143
|
+
ref.destroy();
|
|
144
|
+
this.cardRefs.delete(container);
|
|
145
|
+
}
|
|
146
|
+
};
|
|
147
|
+
});
|
|
148
|
+
this.cleanupFns.push(unsubscribe);
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Open the help panel.
|
|
153
|
+
*/
|
|
154
|
+
open(options) {
|
|
155
|
+
this._pillar()?.open(options);
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Close the help panel.
|
|
159
|
+
*/
|
|
160
|
+
close() {
|
|
161
|
+
this._pillar()?.close();
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Toggle the help panel.
|
|
165
|
+
*/
|
|
166
|
+
toggle() {
|
|
167
|
+
this._pillar()?.toggle();
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Open a specific article.
|
|
171
|
+
*/
|
|
172
|
+
openArticle(slug) {
|
|
173
|
+
this._pillar()?.open({ article: slug });
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Open a specific category.
|
|
177
|
+
*/
|
|
178
|
+
async openCategory(slug) {
|
|
179
|
+
this._pillar()?.navigate('category', { slug });
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Perform a search.
|
|
183
|
+
*/
|
|
184
|
+
search(query) {
|
|
185
|
+
this._pillar()?.open({ search: query });
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Navigate to a specific view.
|
|
189
|
+
*/
|
|
190
|
+
navigate(view, params) {
|
|
191
|
+
this._pillar()?.navigate(view, params);
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Update the panel theme at runtime.
|
|
195
|
+
*/
|
|
196
|
+
setTheme(theme) {
|
|
197
|
+
this._pillar()?.setTheme(theme);
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Enable or disable the text selection "Ask AI" popover.
|
|
201
|
+
*/
|
|
202
|
+
setTextSelectionEnabled(enabled) {
|
|
203
|
+
this._pillar()?.setTextSelectionEnabled(enabled);
|
|
204
|
+
}
|
|
205
|
+
/**
|
|
206
|
+
* Subscribe to SDK events.
|
|
207
|
+
*/
|
|
208
|
+
on(event, callback) {
|
|
209
|
+
const pillar = this._pillar();
|
|
210
|
+
if (!pillar) {
|
|
211
|
+
return () => { };
|
|
212
|
+
}
|
|
213
|
+
// Wrap callback to run in Angular zone
|
|
214
|
+
const wrappedCallback = (data) => {
|
|
215
|
+
this.ngZone.run(() => callback(data));
|
|
216
|
+
};
|
|
217
|
+
return pillar.on(event, wrappedCallback);
|
|
218
|
+
}
|
|
219
|
+
/**
|
|
220
|
+
* Register a task handler.
|
|
221
|
+
*/
|
|
222
|
+
onTask(taskName, handler) {
|
|
223
|
+
const pillar = this._pillar();
|
|
224
|
+
if (!pillar) {
|
|
225
|
+
return () => { };
|
|
226
|
+
}
|
|
227
|
+
// Wrap handler to run in Angular zone
|
|
228
|
+
const wrappedHandler = (data) => {
|
|
229
|
+
this.ngZone.run(() => handler(data));
|
|
230
|
+
};
|
|
231
|
+
return pillar.onTask(taskName, wrappedHandler);
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Define a tool with metadata and handler.
|
|
235
|
+
* Returns an unsubscribe function to remove the tool.
|
|
236
|
+
*
|
|
237
|
+
* @example
|
|
238
|
+
* ```typescript
|
|
239
|
+
* const unsubscribe = pillarService.defineTool({
|
|
240
|
+
* name: 'add_to_cart',
|
|
241
|
+
* description: 'Add a product to the shopping cart',
|
|
242
|
+
* inputSchema: {
|
|
243
|
+
* type: 'object',
|
|
244
|
+
* properties: {
|
|
245
|
+
* productId: { type: 'string' },
|
|
246
|
+
* quantity: { type: 'number' },
|
|
247
|
+
* },
|
|
248
|
+
* required: ['productId', 'quantity'],
|
|
249
|
+
* },
|
|
250
|
+
* execute: async ({ productId, quantity }) => {
|
|
251
|
+
* await this.cartService.add(productId, quantity);
|
|
252
|
+
* return { content: [{ type: 'text', text: 'Added to cart' }] };
|
|
253
|
+
* },
|
|
254
|
+
* });
|
|
255
|
+
*
|
|
256
|
+
* // Later, to unregister:
|
|
257
|
+
* unsubscribe();
|
|
258
|
+
* ```
|
|
259
|
+
*/
|
|
260
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
261
|
+
defineTool(schema) {
|
|
262
|
+
const pillar = this._pillar();
|
|
263
|
+
if (!pillar) {
|
|
264
|
+
console.warn('[Pillar Angular] Cannot define tool - SDK not initialized');
|
|
265
|
+
return () => { };
|
|
266
|
+
}
|
|
267
|
+
// Wrap execute to run in Angular zone
|
|
268
|
+
const wrappedSchema = {
|
|
269
|
+
...schema,
|
|
270
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
271
|
+
execute: async (input) => {
|
|
272
|
+
return this.ngZone.run(() => schema.execute(input));
|
|
273
|
+
},
|
|
274
|
+
};
|
|
275
|
+
return pillar.defineTool(wrappedSchema);
|
|
276
|
+
}
|
|
277
|
+
/**
|
|
278
|
+
* Mount the panel to a specific container element.
|
|
279
|
+
* Used for manual panel placement with PillarPanelComponent.
|
|
280
|
+
*/
|
|
281
|
+
mountPanelTo(container) {
|
|
282
|
+
this._pillar()?.mountPanelTo(container);
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Cleanup resources.
|
|
286
|
+
* Note: We intentionally don't call Pillar.destroy() here to preserve
|
|
287
|
+
* conversation history across route changes. Call Pillar.destroy()
|
|
288
|
+
* explicitly if you need to fully reset the SDK.
|
|
289
|
+
*/
|
|
290
|
+
ngOnDestroy() {
|
|
291
|
+
// Run all cleanup functions
|
|
292
|
+
this.cleanupFns.forEach((cleanup) => cleanup());
|
|
293
|
+
this.cleanupFns.length = 0;
|
|
294
|
+
// Destroy all card component refs
|
|
295
|
+
this.cardRefs.forEach((ref) => {
|
|
296
|
+
this.appRef.detachView(ref.hostView);
|
|
297
|
+
ref.destroy();
|
|
298
|
+
});
|
|
299
|
+
this.cardRefs.clear();
|
|
300
|
+
}
|
|
301
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PillarService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
302
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PillarService, providedIn: 'root' });
|
|
303
|
+
}
|
|
304
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: PillarService, decorators: [{
|
|
305
|
+
type: Injectable,
|
|
306
|
+
args: [{ providedIn: 'root' }]
|
|
307
|
+
}] });
|
|
308
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pillar.service.js","sourceRoot":"","sources":["../../../src/lib/pillar.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PillarService\n * Angular service that initializes and manages the Pillar SDK\n */\n\nimport {\n  Injectable,\n  NgZone,\n  signal,\n  computed,\n  inject,\n  createComponent,\n  ApplicationRef,\n  EnvironmentInjector,\n  type Type,\n} from '@angular/core';\nimport {\n  Pillar,\n  type CardCallbacks,\n  type PillarConfig,\n  type PillarEvents,\n  type PillarState,\n  type TaskExecutePayload,\n  type ThemeConfig,\n  type ToolSchema,\n} from '@pillar-ai/sdk';\nimport type { CardComponent, PillarInitConfig } from './types';\n\n@Injectable({ providedIn: 'root' })\nexport class PillarService {\n  private readonly ngZone = inject(NgZone);\n  private readonly appRef = inject(ApplicationRef);\n  private readonly environmentInjector = inject(EnvironmentInjector);\n\n  // Internal state\n  private readonly _pillar = signal<Pillar | null>(null);\n  private readonly cleanupFns: Array<() => void> = [];\n  private readonly cardRefs: Map<HTMLElement, any> = new Map();\n  private onTaskCallback: ((task: TaskExecutePayload) => void) | null = null;\n  private registeredCards: Record<string, CardComponent> | null = null;\n\n  // Public signals\n  readonly state = signal<PillarState>('uninitialized');\n  readonly isReady = computed(() => this.state() === 'ready');\n  readonly isPanelOpen = signal(false);\n\n  /**\n   * Get the Pillar SDK instance.\n   */\n  getInstance(): Pillar | null {\n    return this._pillar();\n  }\n\n  /**\n   * Initialize the Pillar SDK.\n   * Call this in your app's initialization (e.g., APP_INITIALIZER).\n   *\n   * @param initConfig - Configuration options\n   */\n  async init(initConfig: PillarInitConfig): Promise<void> {\n    const { productKey, helpCenter, config, onTask, cards } = initConfig;\n\n    // Support both productKey (new) and helpCenter (deprecated)\n    const resolvedKey = productKey ?? helpCenter;\n\n    if (helpCenter && !productKey) {\n      console.warn(\n        '[Pillar Angular] \"helpCenter\" is deprecated. Use \"productKey\" instead.'\n      );\n    }\n\n    // Store callbacks for later use\n    this.onTaskCallback = onTask ?? null;\n    this.registeredCards = cards ?? null;\n\n    try {\n      // Pillar is a singleton - check if already initialized\n      const existingInstance = Pillar.getInstance();\n      if (existingInstance) {\n        // Reuse existing instance (preserves chat history, panel state, etc.)\n        this._pillar.set(existingInstance);\n        this.state.set(existingInstance.state);\n        this.subscribeToEvents(existingInstance);\n        this.registerCards(existingInstance);\n        return;\n      }\n\n      // Initialize new instance\n      const instance = await Pillar.init({\n        productKey: resolvedKey,\n        ...config,\n      });\n\n      this.ngZone.run(() => {\n        this._pillar.set(instance);\n        this.state.set(instance.state);\n      });\n\n      this.subscribeToEvents(instance);\n      this.registerCards(instance);\n    } catch (error) {\n      console.error('[Pillar Angular] Failed to initialize:', error);\n      this.ngZone.run(() => {\n        this.state.set('error');\n      });\n      throw error;\n    }\n  }\n\n  /**\n   * Subscribe to SDK events and sync to Angular signals.\n   */\n  private subscribeToEvents(instance: Pillar): void {\n    // Panel open/close events\n    const unsubOpen = instance.on('panel:open', () => {\n      this.ngZone.run(() => {\n        this.isPanelOpen.set(true);\n      });\n    });\n    this.cleanupFns.push(unsubOpen);\n\n    const unsubClose = instance.on('panel:close', () => {\n      this.ngZone.run(() => {\n        this.isPanelOpen.set(false);\n      });\n    });\n    this.cleanupFns.push(unsubClose);\n\n    // State change events\n    const unsubReady = instance.on('ready', () => {\n      this.ngZone.run(() => {\n        this.state.set('ready');\n      });\n    });\n    this.cleanupFns.push(unsubReady);\n\n    const unsubError = instance.on('error', () => {\n      this.ngZone.run(() => {\n        this.state.set('error');\n      });\n    });\n    this.cleanupFns.push(unsubError);\n\n    // Task execution events\n    if (this.onTaskCallback) {\n      const callback = this.onTaskCallback;\n      const unsubTask = instance.on('task:execute', (task) => {\n        this.ngZone.run(() => {\n          callback(task);\n        });\n      });\n      this.cleanupFns.push(unsubTask);\n    }\n  }\n\n  /**\n   * Register custom card components.\n   */\n  private registerCards(instance: Pillar): void {\n    if (!this.registeredCards) return;\n\n    Object.entries(this.registeredCards).forEach(([cardType, CardComponent]) => {\n      const unsubscribe = instance.registerCard(\n        cardType,\n        (container, data, callbacks: CardCallbacks) => {\n          // Create an Angular component dynamically\n          const componentRef = createComponent(CardComponent as Type<any>, {\n            environmentInjector: this.environmentInjector,\n          });\n\n          // Set inputs\n          componentRef.setInput('data', data);\n          componentRef.setInput('onConfirm', callbacks.onConfirm);\n          componentRef.setInput('onCancel', callbacks.onCancel);\n          componentRef.setInput('onStateChange', callbacks.onStateChange);\n\n          // Attach to the application\n          this.appRef.attachView(componentRef.hostView);\n\n          // Append to container\n          container.appendChild(componentRef.location.nativeElement);\n          this.cardRefs.set(container, componentRef);\n\n          // Return cleanup function\n          return () => {\n            const ref = this.cardRefs.get(container);\n            if (ref) {\n              this.appRef.detachView(ref.hostView);\n              ref.destroy();\n              this.cardRefs.delete(container);\n            }\n          };\n        }\n      );\n\n      this.cleanupFns.push(unsubscribe);\n    });\n  }\n\n  /**\n   * Open the help panel.\n   */\n  open(options?: {\n    view?: string;\n    article?: string;\n    search?: string;\n    focusInput?: boolean;\n  }): void {\n    this._pillar()?.open(options);\n  }\n\n  /**\n   * Close the help panel.\n   */\n  close(): void {\n    this._pillar()?.close();\n  }\n\n  /**\n   * Toggle the help panel.\n   */\n  toggle(): void {\n    this._pillar()?.toggle();\n  }\n\n  /**\n   * Open a specific article.\n   */\n  openArticle(slug: string): void {\n    this._pillar()?.open({ article: slug });\n  }\n\n  /**\n   * Open a specific category.\n   */\n  async openCategory(slug: string): Promise<void> {\n    this._pillar()?.navigate('category', { slug });\n  }\n\n  /**\n   * Perform a search.\n   */\n  search(query: string): void {\n    this._pillar()?.open({ search: query });\n  }\n\n  /**\n   * Navigate to a specific view.\n   */\n  navigate(view: string, params?: Record<string, string>): void {\n    this._pillar()?.navigate(view, params);\n  }\n\n  /**\n   * Update the panel theme at runtime.\n   */\n  setTheme(theme: Partial<ThemeConfig>): void {\n    this._pillar()?.setTheme(theme);\n  }\n\n  /**\n   * Enable or disable the text selection \"Ask AI\" popover.\n   */\n  setTextSelectionEnabled(enabled: boolean): void {\n    this._pillar()?.setTextSelectionEnabled(enabled);\n  }\n\n  /**\n   * Subscribe to SDK events.\n   */\n  on<K extends keyof PillarEvents>(\n    event: K,\n    callback: (data: PillarEvents[K]) => void\n  ): () => void {\n    const pillar = this._pillar();\n    if (!pillar) {\n      return () => {};\n    }\n\n    // Wrap callback to run in Angular zone\n    const wrappedCallback = (data: PillarEvents[K]) => {\n      this.ngZone.run(() => callback(data));\n    };\n\n    return pillar.on(event, wrappedCallback);\n  }\n\n  /**\n   * Register a task handler.\n   */\n  onTask(\n    taskName: string,\n    handler: (data: Record<string, unknown>) => void\n  ): () => void {\n    const pillar = this._pillar();\n    if (!pillar) {\n      return () => {};\n    }\n\n    // Wrap handler to run in Angular zone\n    const wrappedHandler = (data: Record<string, unknown>) => {\n      this.ngZone.run(() => handler(data));\n    };\n\n    return pillar.onTask(taskName, wrappedHandler);\n  }\n\n  /**\n   * Define a tool with metadata and handler.\n   * Returns an unsubscribe function to remove the tool.\n   *\n   * @example\n   * ```typescript\n   * const unsubscribe = pillarService.defineTool({\n   *   name: 'add_to_cart',\n   *   description: 'Add a product to the shopping cart',\n   *   inputSchema: {\n   *     type: 'object',\n   *     properties: {\n   *       productId: { type: 'string' },\n   *       quantity: { type: 'number' },\n   *     },\n   *     required: ['productId', 'quantity'],\n   *   },\n   *   execute: async ({ productId, quantity }) => {\n   *     await this.cartService.add(productId, quantity);\n   *     return { content: [{ type: 'text', text: 'Added to cart' }] };\n   *   },\n   * });\n   *\n   * // Later, to unregister:\n   * unsubscribe();\n   * ```\n   */\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  defineTool(schema: ToolSchema<any>): () => void {\n    const pillar = this._pillar();\n    if (!pillar) {\n      console.warn('[Pillar Angular] Cannot define tool - SDK not initialized');\n      return () => {};\n    }\n\n    // Wrap execute to run in Angular zone\n    const wrappedSchema = {\n      ...schema,\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      execute: async (input: any) => {\n        return this.ngZone.run(() => schema.execute(input));\n      },\n    };\n\n    return pillar.defineTool(wrappedSchema);\n  }\n\n  /**\n   * Mount the panel to a specific container element.\n   * Used for manual panel placement with PillarPanelComponent.\n   */\n  mountPanelTo(container: HTMLElement): void {\n    this._pillar()?.mountPanelTo(container);\n  }\n\n  /**\n   * Cleanup resources.\n   * Note: We intentionally don't call Pillar.destroy() here to preserve\n   * conversation history across route changes. Call Pillar.destroy()\n   * explicitly if you need to fully reset the SDK.\n   */\n  ngOnDestroy(): void {\n    // Run all cleanup functions\n    this.cleanupFns.forEach((cleanup) => cleanup());\n    this.cleanupFns.length = 0;\n\n    // Destroy all card component refs\n    this.cardRefs.forEach((ref) => {\n      this.appRef.detachView(ref.hostView);\n      ref.destroy();\n    });\n    this.cardRefs.clear();\n  }\n}\n"]}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared types for Pillar Angular SDK
|
|
3
|
+
*/
|
|
4
|
+
export {};
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbGliL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBTaGFyZWQgdHlwZXMgZm9yIFBpbGxhciBBbmd1bGFyIFNES1xuICovXG5cbmltcG9ydCB0eXBlIHtcbiAgUGlsbGFyLFxuICBQaWxsYXJDb25maWcsXG4gIFBpbGxhckV2ZW50cyxcbiAgUGlsbGFyU3RhdGUsXG4gIFRhc2tFeGVjdXRlUGF5bG9hZCxcbiAgVGhlbWVDb25maWcsXG59IGZyb20gJ0BwaWxsYXItYWkvc2RrJztcbmltcG9ydCB0eXBlIHsgU2lnbmFsLCBXcml0YWJsZVNpZ25hbCwgVHlwZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG4vLyBDYXJkIFR5cGVzXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbi8qKlxuICogUHJvcHMgcGFzc2VkIHRvIGN1c3RvbSBjYXJkIGNvbXBvbmVudHMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ2FyZENvbXBvbmVudFByb3BzPFQgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4ge1xuICAvKiogRGF0YSBleHRyYWN0ZWQgYnkgdGhlIEFJIGZvciB0aGlzIGFjdGlvbiAqL1xuICBkYXRhOiBUO1xuICAvKiogQ2FsbGVkIHdoZW4gdXNlciBjb25maXJtcyB0aGUgYWN0aW9uICovXG4gIG9uQ29uZmlybTogKG1vZGlmaWVkRGF0YT86IFJlY29yZDxzdHJpbmcsIHVua25vd24+KSA9PiB2b2lkO1xuICAvKiogQ2FsbGVkIHdoZW4gdXNlciBjYW5jZWxzIHRoZSBhY3Rpb24gKi9cbiAgb25DYW5jZWw6ICgpID0+IHZvaWQ7XG4gIC8qKiBDYWxsZWQgdG8gcmVwb3J0IHN0YXRlIGNoYW5nZXMgKGxvYWRpbmcsIHN1Y2Nlc3MsIGVycm9yKSAqL1xuICBvblN0YXRlQ2hhbmdlPzogKFxuICAgIHN0YXRlOiAnbG9hZGluZycgfCAnc3VjY2VzcycgfCAnZXJyb3InLFxuICAgIG1lc3NhZ2U/OiBzdHJpbmdcbiAgKSA9PiB2b2lkO1xufVxuXG4vKipcbiAqIEFuIEFuZ3VsYXIgY29tcG9uZW50IHRoYXQgY2FuIGJlIHVzZWQgYXMgYSBjdXN0b20gY2FyZCByZW5kZXJlci5cbiAqIFRoZSBjb21wb25lbnQgc2hvdWxkIGFjY2VwdCBDYXJkQ29tcG9uZW50UHJvcHMgYXMgaW5wdXRzLlxuICovXG5leHBvcnQgdHlwZSBDYXJkQ29tcG9uZW50PFQgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4gPSBUeXBlPENhcmRDb21wb25lbnRQcm9wczxUPj47XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFNlcnZpY2UgVHlwZXNcbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cblxuZXhwb3J0IGludGVyZmFjZSBQaWxsYXJTZXJ2aWNlU3RhdGUge1xuICAvKiogVGhlIFBpbGxhciBTREsgaW5zdGFuY2UgKi9cbiAgcGlsbGFyOiBTaWduYWw8UGlsbGFyIHwgbnVsbD47XG5cbiAgLyoqIEN1cnJlbnQgU0RLIHN0YXRlICovXG4gIHN0YXRlOiBXcml0YWJsZVNpZ25hbDxQaWxsYXJTdGF0ZT47XG5cbiAgLyoqIFdoZXRoZXIgdGhlIFNESyBpcyByZWFkeSAqL1xuICBpc1JlYWR5OiBTaWduYWw8Ym9vbGVhbj47XG5cbiAgLyoqIFdoZXRoZXIgdGhlIHBhbmVsIGlzIGN1cnJlbnRseSBvcGVuICovXG4gIGlzUGFuZWxPcGVuOiBXcml0YWJsZVNpZ25hbDxib29sZWFuPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQaWxsYXJTZXJ2aWNlQWN0aW9ucyB7XG4gIC8qKiBPcGVuIHRoZSBoZWxwIHBhbmVsICovXG4gIG9wZW46IChvcHRpb25zPzoge1xuICAgIHZpZXc/OiBzdHJpbmc7XG4gICAgYXJ0aWNsZT86IHN0cmluZztcbiAgICBzZWFyY2g/OiBzdHJpbmc7XG4gICAgZm9jdXNJbnB1dD86IGJvb2xlYW47XG4gIH0pID0+IHZvaWQ7XG5cbiAgLyoqIENsb3NlIHRoZSBoZWxwIHBhbmVsICovXG4gIGNsb3NlOiAoKSA9PiB2b2lkO1xuXG4gIC8qKiBUb2dnbGUgdGhlIGhlbHAgcGFuZWwgKi9cbiAgdG9nZ2xlOiAoKSA9PiB2b2lkO1xuXG4gIC8qKiBPcGVuIGEgc3BlY2lmaWMgYXJ0aWNsZSAqL1xuICBvcGVuQXJ0aWNsZTogKHNsdWc6IHN0cmluZykgPT4gdm9pZDtcblxuICAvKiogT3BlbiBhIHNwZWNpZmljIGNhdGVnb3J5ICovXG4gIG9wZW5DYXRlZ29yeTogKHNsdWc6IHN0cmluZykgPT4gUHJvbWlzZTx2b2lkPjtcblxuICAvKiogUGVyZm9ybSBhIHNlYXJjaCAqL1xuICBzZWFyY2g6IChxdWVyeTogc3RyaW5nKSA9PiB2b2lkO1xuXG4gIC8qKiBOYXZpZ2F0ZSB0byBhIHNwZWNpZmljIHZpZXcgKi9cbiAgbmF2aWdhdGU6ICh2aWV3OiBzdHJpbmcsIHBhcmFtcz86IFJlY29yZDxzdHJpbmcsIHN0cmluZz4pID0+IHZvaWQ7XG5cbiAgLyoqIFVwZGF0ZSB0aGUgcGFuZWwgdGhlbWUgYXQgcnVudGltZSAqL1xuICBzZXRUaGVtZTogKHRoZW1lOiBQYXJ0aWFsPFRoZW1lQ29uZmlnPikgPT4gdm9pZDtcblxuICAvKiogRW5hYmxlIG9yIGRpc2FibGUgdGhlIHRleHQgc2VsZWN0aW9uIFwiQXNrIEFJXCIgcG9wb3ZlciAqL1xuICBzZXRUZXh0U2VsZWN0aW9uRW5hYmxlZDogKGVuYWJsZWQ6IGJvb2xlYW4pID0+IHZvaWQ7XG5cbiAgLyoqIFN1YnNjcmliZSB0byBTREsgZXZlbnRzICovXG4gIG9uOiA8SyBleHRlbmRzIGtleW9mIFBpbGxhckV2ZW50cz4oXG4gICAgZXZlbnQ6IEssXG4gICAgY2FsbGJhY2s6IChkYXRhOiBQaWxsYXJFdmVudHNbS10pID0+IHZvaWRcbiAgKSA9PiAoKSA9PiB2b2lkO1xuXG4gIC8qKiBSZWdpc3RlciBhIHRhc2sgaGFuZGxlciAqL1xuICBvblRhc2s6IChcbiAgICB0YXNrTmFtZTogc3RyaW5nLFxuICAgIGhhbmRsZXI6IChkYXRhOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT4gdm9pZFxuICApID0+ICgpID0+IHZvaWQ7XG5cbiAgLyoqIEdldCB0aGUgUGlsbGFyIFNESyBpbnN0YW5jZSAqL1xuICBnZXRJbnN0YW5jZTogKCkgPT4gUGlsbGFyIHwgbnVsbDtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQaWxsYXJDb250ZXh0VmFsdWUgZXh0ZW5kcyBQaWxsYXJTZXJ2aWNlU3RhdGUsIFBpbGxhclNlcnZpY2VBY3Rpb25zIHt9XG5cbi8vID09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT1cbi8vIFByb3ZpZGVyIFR5cGVzXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGlsbGFySW5pdENvbmZpZyB7XG4gIC8qKlxuICAgKiBZb3VyIHByb2R1Y3Qga2V5IGZyb20gdGhlIFBpbGxhciBhcHAuXG4gICAqIEdldCBpdCBhdCBhcHAudHJ5cGlsbGFyLmNvbVxuICAgKi9cbiAgcHJvZHVjdEtleT86IHN0cmluZztcblxuICAvKipcbiAgICogQGRlcHJlY2F0ZWQgVXNlIGBwcm9kdWN0S2V5YCBpbnN0ZWFkLiBXaWxsIGJlIHJlbW92ZWQgaW4gdjEuMC5cbiAgICovXG4gIGhlbHBDZW50ZXI/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIEFkZGl0aW9uYWwgU0RLIGNvbmZpZ3VyYXRpb25cbiAgICpcbiAgICogTm90YWJsZSBvcHRpb25zOlxuICAgKiAtIGBwYW5lbC51c2VTaGFkb3dET01gOiBXaGV0aGVyIHRvIGlzb2xhdGUgc3R5bGVzIGluIFNoYWRvdyBET00gKGRlZmF1bHQ6IGZhbHNlKS5cbiAgICogICBTZXQgdG8gZmFsc2UgdG8gbGV0IGN1c3RvbSBjYXJkcyBpbmhlcml0IHlvdXIgYXBwJ3MgQ1NTIChUYWlsd2luZCwgZXRjLilcbiAgICovXG4gIGNvbmZpZz86IE9taXQ8UGlsbGFyQ29uZmlnLCAncHJvZHVjdEtleScgfCAnaGVscENlbnRlcic+O1xuXG4gIC8qKlxuICAgKiBIYW5kbGVyIGNhbGxlZCB3aGVuIGEgdGFzayBhY3Rpb24gaXMgdHJpZ2dlcmVkIGZyb20gdGhlIGNoYXQuXG4gICAqIFVzZSB0aGlzIHRvIGhhbmRsZSBBSS1zdWdnZXN0ZWQgYWN0aW9ucyBsaWtlIG9wZW5pbmcgbW9kYWxzLCBuYXZpZ2F0aW5nLCBldGMuXG4gICAqL1xuICBvblRhc2s/OiAodGFzazogVGFza0V4ZWN1dGVQYXlsb2FkKSA9PiB2b2lkO1xuXG4gIC8qKlxuICAgKiBDdXN0b20gY2FyZCBjb21wb25lbnRzIHRvIHJlbmRlciBmb3IgaW5saW5lX3VpIHR5cGUgYWN0aW9ucy5cbiAgICogTWFwIGNhcmQgdHlwZSBuYW1lcyB0byBBbmd1bGFyIGNvbXBvbmVudHMgdGhhdCB3aWxsIHJlbmRlciB0aGUgaW5saW5lIFVJLlxuICAgKi9cbiAgY2FyZHM/OiBSZWNvcmQ8c3RyaW5nLCBDYXJkQ29tcG9uZW50Pjtcbn1cblxuLy8gPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gSW5qZWN0IEZ1bmN0aW9uIFR5cGVzXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09XG5cbmV4cG9ydCBpbnRlcmZhY2UgSW5qZWN0UGlsbGFyUmVzdWx0IGV4dGVuZHMgUGlsbGFyQ29udGV4dFZhbHVlIHt9XG5cbmV4cG9ydCBpbnRlcmZhY2UgSW5qZWN0SGVscFBhbmVsUmVzdWx0IHtcbiAgLyoqIFdoZXRoZXIgdGhlIHBhbmVsIGlzIGN1cnJlbnRseSBvcGVuICovXG4gIGlzT3BlbjogU2lnbmFsPGJvb2xlYW4+O1xuXG4gIC8qKiBPcGVuIHRoZSBwYW5lbCAqL1xuICBvcGVuOiAob3B0aW9ucz86IHsgdmlldz86IHN0cmluZzsgYXJ0aWNsZT86IHN0cmluZzsgc2VhcmNoPzogc3RyaW5nIH0pID0+IHZvaWQ7XG5cbiAgLyoqIENsb3NlIHRoZSBwYW5lbCAqL1xuICBjbG9zZTogKCkgPT4gdm9pZDtcblxuICAvKiogVG9nZ2xlIHRoZSBwYW5lbCAqL1xuICB0b2dnbGU6ICgpID0+IHZvaWQ7XG5cbiAgLyoqIE9wZW4gYSBzcGVjaWZpYyBhcnRpY2xlIGluIHRoZSBwYW5lbCAqL1xuICBvcGVuQXJ0aWNsZTogKHNsdWc6IHN0cmluZykgPT4gdm9pZDtcblxuICAvKiogT3BlbiBhIHNwZWNpZmljIGNhdGVnb3J5IGluIHRoZSBwYW5lbCAqL1xuICBvcGVuQ2F0ZWdvcnk6IChzbHVnOiBzdHJpbmcpID0+IFByb21pc2U8dm9pZD47XG5cbiAgLyoqIE9wZW4gc2VhcmNoIHdpdGggYSBxdWVyeSAqL1xuICBvcGVuU2VhcmNoOiAocXVlcnk/OiBzdHJpbmcpID0+IHZvaWQ7XG5cbiAgLyoqIE9wZW4gdGhlIEFJIGNoYXQgKi9cbiAgb3BlbkNoYXQ6ICgpID0+IHZvaWQ7XG59XG4iXX0=
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Generated bundle index. Do not edit.
|
|
3
|
+
*/
|
|
4
|
+
export * from './public-api';
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGlsbGFyLWFpLWFuZ3VsYXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGlsbGFyLWFpLWFuZ3VsYXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
|
|
@@ -0,0 +1,67 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @pillar-ai/angular - Angular bindings for Pillar SDK
|
|
3
|
+
*
|
|
4
|
+
* @example
|
|
5
|
+
* ```typescript
|
|
6
|
+
* // app.config.ts
|
|
7
|
+
* import { ApplicationConfig, APP_INITIALIZER, inject } from '@angular/core';
|
|
8
|
+
* import { PillarService } from '@pillar-ai/angular';
|
|
9
|
+
*
|
|
10
|
+
* function initPillar() {
|
|
11
|
+
* const pillar = inject(PillarService);
|
|
12
|
+
* return () => pillar.init({ productKey: 'your-product-key' });
|
|
13
|
+
* }
|
|
14
|
+
*
|
|
15
|
+
* export const appConfig: ApplicationConfig = {
|
|
16
|
+
* providers: [
|
|
17
|
+
* { provide: APP_INITIALIZER, useFactory: initPillar, multi: true },
|
|
18
|
+
* ],
|
|
19
|
+
* };
|
|
20
|
+
*
|
|
21
|
+
* // help-button.component.ts
|
|
22
|
+
* import { Component } from '@angular/core';
|
|
23
|
+
* import { injectPillar } from '@pillar-ai/angular';
|
|
24
|
+
*
|
|
25
|
+
* @Component({
|
|
26
|
+
* selector: 'app-help-button',
|
|
27
|
+
* standalone: true,
|
|
28
|
+
* template: `
|
|
29
|
+
* <button (click)="toggle()">
|
|
30
|
+
* {{ isPanelOpen() ? 'Close Help' : 'Get Help' }}
|
|
31
|
+
* </button>
|
|
32
|
+
* `,
|
|
33
|
+
* })
|
|
34
|
+
* export class HelpButtonComponent {
|
|
35
|
+
* private pillar = injectPillar();
|
|
36
|
+
* isPanelOpen = this.pillar.isPanelOpen;
|
|
37
|
+
* toggle = this.pillar.toggle;
|
|
38
|
+
* }
|
|
39
|
+
*
|
|
40
|
+
* // Custom panel placement example:
|
|
41
|
+
* // layout.component.ts
|
|
42
|
+
* import { Component } from '@angular/core';
|
|
43
|
+
* import { PillarPanelComponent } from '@pillar-ai/angular';
|
|
44
|
+
*
|
|
45
|
+
* @Component({
|
|
46
|
+
* selector: 'app-layout',
|
|
47
|
+
* standalone: true,
|
|
48
|
+
* imports: [PillarPanelComponent],
|
|
49
|
+
* template: `
|
|
50
|
+
* <div class="layout">
|
|
51
|
+
* <pillar-panel class="custom-panel" />
|
|
52
|
+
* <main>Your content</main>
|
|
53
|
+
* </div>
|
|
54
|
+
* `,
|
|
55
|
+
* })
|
|
56
|
+
* export class LayoutComponent {}
|
|
57
|
+
* ```
|
|
58
|
+
*/
|
|
59
|
+
// Service
|
|
60
|
+
export { PillarService } from './lib/pillar.service';
|
|
61
|
+
// Components
|
|
62
|
+
export { PillarPanelComponent } from './lib/pillar-panel.component';
|
|
63
|
+
// Inject functions
|
|
64
|
+
export { injectPillar } from './lib/inject-pillar';
|
|
65
|
+
export { injectHelpPanel } from './lib/inject-help-panel';
|
|
66
|
+
export { injectPillarTool, injectPillarAction } from './lib/inject-pillar-tool';
|
|
67
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljLWFwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9wdWJsaWMtYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F5REc7QUFFSCxVQUFVO0FBQ1YsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBRXJELGFBQWE7QUFDYixPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUVwRSxtQkFBbUI7QUFDbkIsT0FBTyxFQUFFLFlBQVksRUFBMkIsTUFBTSxxQkFBcUIsQ0FBQztBQUM1RSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDMUQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLGtCQUFrQixFQUFFLE1BQU0sMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBwaWxsYXItYWkvYW5ndWxhciAtIEFuZ3VsYXIgYmluZGluZ3MgZm9yIFBpbGxhciBTREtcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogLy8gYXBwLmNvbmZpZy50c1xuICogaW1wb3J0IHsgQXBwbGljYXRpb25Db25maWcsIEFQUF9JTklUSUFMSVpFUiwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG4gKiBpbXBvcnQgeyBQaWxsYXJTZXJ2aWNlIH0gZnJvbSAnQHBpbGxhci1haS9hbmd1bGFyJztcbiAqXG4gKiBmdW5jdGlvbiBpbml0UGlsbGFyKCkge1xuICogICBjb25zdCBwaWxsYXIgPSBpbmplY3QoUGlsbGFyU2VydmljZSk7XG4gKiAgIHJldHVybiAoKSA9PiBwaWxsYXIuaW5pdCh7IHByb2R1Y3RLZXk6ICd5b3VyLXByb2R1Y3Qta2V5JyB9KTtcbiAqIH1cbiAqXG4gKiBleHBvcnQgY29uc3QgYXBwQ29uZmlnOiBBcHBsaWNhdGlvbkNvbmZpZyA9IHtcbiAqICAgcHJvdmlkZXJzOiBbXG4gKiAgICAgeyBwcm92aWRlOiBBUFBfSU5JVElBTElaRVIsIHVzZUZhY3Rvcnk6IGluaXRQaWxsYXIsIG11bHRpOiB0cnVlIH0sXG4gKiAgIF0sXG4gKiB9O1xuICpcbiAqIC8vIGhlbHAtYnV0dG9uLmNvbXBvbmVudC50c1xuICogaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG4gKiBpbXBvcnQgeyBpbmplY3RQaWxsYXIgfSBmcm9tICdAcGlsbGFyLWFpL2FuZ3VsYXInO1xuICpcbiAqIEBDb21wb25lbnQoe1xuICogICBzZWxlY3RvcjogJ2FwcC1oZWxwLWJ1dHRvbicsXG4gKiAgIHN0YW5kYWxvbmU6IHRydWUsXG4gKiAgIHRlbXBsYXRlOiBgXG4gKiAgICAgPGJ1dHRvbiAoY2xpY2spPVwidG9nZ2xlKClcIj5cbiAqICAgICAgIHt7IGlzUGFuZWxPcGVuKCkgPyAnQ2xvc2UgSGVscCcgOiAnR2V0IEhlbHAnIH19XG4gKiAgICAgPC9idXR0b24+XG4gKiAgIGAsXG4gKiB9KVxuICogZXhwb3J0IGNsYXNzIEhlbHBCdXR0b25Db21wb25lbnQge1xuICogICBwcml2YXRlIHBpbGxhciA9IGluamVjdFBpbGxhcigpO1xuICogICBpc1BhbmVsT3BlbiA9IHRoaXMucGlsbGFyLmlzUGFuZWxPcGVuO1xuICogICB0b2dnbGUgPSB0aGlzLnBpbGxhci50b2dnbGU7XG4gKiB9XG4gKlxuICogLy8gQ3VzdG9tIHBhbmVsIHBsYWNlbWVudCBleGFtcGxlOlxuICogLy8gbGF5b3V0LmNvbXBvbmVudC50c1xuICogaW1wb3J0IHsgQ29tcG9uZW50IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG4gKiBpbXBvcnQgeyBQaWxsYXJQYW5lbENvbXBvbmVudCB9IGZyb20gJ0BwaWxsYXItYWkvYW5ndWxhcic7XG4gKlxuICogQENvbXBvbmVudCh7XG4gKiAgIHNlbGVjdG9yOiAnYXBwLWxheW91dCcsXG4gKiAgIHN0YW5kYWxvbmU6IHRydWUsXG4gKiAgIGltcG9ydHM6IFtQaWxsYXJQYW5lbENvbXBvbmVudF0sXG4gKiAgIHRlbXBsYXRlOiBgXG4gKiAgICAgPGRpdiBjbGFzcz1cImxheW91dFwiPlxuICogICAgICAgPHBpbGxhci1wYW5lbCBjbGFzcz1cImN1c3RvbS1wYW5lbFwiIC8+XG4gKiAgICAgICA8bWFpbj5Zb3VyIGNvbnRlbnQ8L21haW4+XG4gKiAgICAgPC9kaXY+XG4gKiAgIGAsXG4gKiB9KVxuICogZXhwb3J0IGNsYXNzIExheW91dENvbXBvbmVudCB7fVxuICogYGBgXG4gKi9cblxuLy8gU2VydmljZVxuZXhwb3J0IHsgUGlsbGFyU2VydmljZSB9IGZyb20gJy4vbGliL3BpbGxhci5zZXJ2aWNlJztcblxuLy8gQ29tcG9uZW50c1xuZXhwb3J0IHsgUGlsbGFyUGFuZWxDb21wb25lbnQgfSBmcm9tICcuL2xpYi9waWxsYXItcGFuZWwuY29tcG9uZW50JztcblxuLy8gSW5qZWN0IGZ1bmN0aW9uc1xuZXhwb3J0IHsgaW5qZWN0UGlsbGFyLCB0eXBlIEluamVjdFBpbGxhclJlc3VsdCB9IGZyb20gJy4vbGliL2luamVjdC1waWxsYXInO1xuZXhwb3J0IHsgaW5qZWN0SGVscFBhbmVsIH0gZnJvbSAnLi9saWIvaW5qZWN0LWhlbHAtcGFuZWwnO1xuZXhwb3J0IHsgaW5qZWN0UGlsbGFyVG9vbCwgaW5qZWN0UGlsbGFyQWN0aW9uIH0gZnJvbSAnLi9saWIvaW5qZWN0LXBpbGxhci10b29sJztcblxuLy8gVHlwZXNcbmV4cG9ydCB0eXBlIHtcbiAgQ2FyZENvbXBvbmVudCxcbiAgQ2FyZENvbXBvbmVudFByb3BzLFxuICBQaWxsYXJDb250ZXh0VmFsdWUsXG4gIFBpbGxhckluaXRDb25maWcsXG4gIFBpbGxhclNlcnZpY2VBY3Rpb25zLFxuICBQaWxsYXJTZXJ2aWNlU3RhdGUsXG4gIEluamVjdEhlbHBQYW5lbFJlc3VsdCxcbn0gZnJvbSAnLi9saWIvdHlwZXMnO1xuXG4vLyBSZS1leHBvcnQgdHlwZXMgZnJvbSBjb3JlIFNESyBmb3IgY29udmVuaWVuY2VcbmV4cG9ydCB0eXBlIHtcbiAgLy8gQ29uZmlndXJhdGlvbiB0eXBlc1xuICBFZGdlVHJpZ2dlckNvbmZpZyxcbiAgTW9iaWxlVHJpZ2dlckNvbmZpZyxcbiAgTW9iaWxlVHJpZ2dlclBvc2l0aW9uLFxuICBNb2JpbGVUcmlnZ2VySWNvbixcbiAgTW9iaWxlVHJpZ2dlclNpemUsXG4gIFBhbmVsQ29uZmlnLFxuICBQaWxsYXJDb25maWcsXG4gIFBpbGxhckV2ZW50cyxcbiAgUGlsbGFyU3RhdGUsXG4gIFJlc29sdmVkQ29uZmlnLFxuICBSZXNvbHZlZE1vYmlsZVRyaWdnZXJDb25maWcsXG4gIFJlc29sdmVkVGhlbWVDb25maWcsXG4gIFRhc2tFeGVjdXRlUGF5bG9hZCxcbiAgVGV4dFNlbGVjdGlvbkNvbmZpZyxcbiAgVGhlbWVDb2xvcnMsXG4gIFRoZW1lQ29uZmlnLFxuICBUaGVtZU1vZGUsXG4gIENhcmRDYWxsYmFja3MsXG4gIENhcmRSZW5kZXJlcixcbiAgU2lkZWJhclRhYkNvbmZpZyxcbiAgLy8gVG9vbCB0eXBlcyBmb3IgdHlwZS1zYWZlIG9uVGFzayBhbmQgaW5qZWN0UGlsbGFyVG9vbFxuICBUb29sRGVmaW5pdGlvbnMsXG4gIFN5bmNUb29sRGVmaW5pdGlvbnMsXG4gIFRvb2xEYXRhVHlwZSxcbiAgVG9vbE5hbWVzLFxuICAvLyBVbmlmaWVkIHRvb2wgc2NoZW1hIChuZXcgQVBJKVxuICBUb29sRXhlY3V0ZVJlc3VsdCxcbiAgVG9vbFNjaGVtYSxcbiAgVG9vbFR5cGUsXG4gIC8vIEJhY2t3YXJkcyBjb21wYXRpYmlsaXR5IGFsaWFzZXMgKGRlcHJlY2F0ZWQpXG4gIEFjdGlvbkRlZmluaXRpb25zLFxuICBTeW5jQWN0aW9uRGVmaW5pdGlvbnMsXG4gIEFjdGlvbkRhdGFUeXBlLFxuICBBY3Rpb25OYW1lcyxcbiAgQWN0aW9uUmVzdWx0LFxuICBBY3Rpb25TY2hlbWEsXG4gIEFjdGlvblR5cGUsXG4gIC8vIENoYXQgY29udGV4dCBmb3IgZXNjYWxhdGlvblxuICBDaGF0Q29udGV4dCxcbiAgLy8gRE9NIFNjYW5uaW5nIHR5cGVzIChkZXByZWNhdGVkIC0gZmVhdHVyZSBpcyBkaXNhYmxlZClcbiAgLyoqIEBkZXByZWNhdGVkIERPTSBzY2FubmluZyBpcyBkaXNhYmxlZCAqL1xuICBET01TY2FubmluZ0NvbmZpZyxcbiAgLyoqIEBkZXByZWNhdGVkIERPTSBzY2FubmluZyBpcyBkaXNhYmxlZCAqL1xuICBSZXNvbHZlZERPTVNjYW5uaW5nQ29uZmlnLFxuICAvKiogQGRlcHJlY2F0ZWQgRE9NIHNjYW5uaW5nIGlzIGRpc2FibGVkICovXG4gIFNjYW5PcHRpb25zLFxuICAvKiogQGRlcHJlY2F0ZWQgRE9NIHNjYW5uaW5nIGlzIGRpc2FibGVkICovXG4gIENvbXBhY3RTY2FuUmVzdWx0LFxuICAvKiogQGRlcHJlY2F0ZWQgRE9NIHNjYW5uaW5nIGlzIGRpc2FibGVkICovXG4gIEludGVyYWN0aW9uVHlwZSxcbiAgLy8gRE9NIFNjYW5uZXIgdXRpbGl0aWVzIChkZXByZWNhdGVkIC0gZmVhdHVyZSBpcyBkaXNhYmxlZClcbiAgLyoqIEBkZXByZWNhdGVkIERPTSBzY2FubmluZyBpcyBkaXNhYmxlZCAqL1xuICBzY2FuUGFnZURpcmVjdCxcbn0gZnJvbSAnQHBpbGxhci1haS9zZGsnO1xuIl19
|