@gravitee/ui-particles-angular 15.10.2 → 15.10.3-apim-9967-471bf98
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/esm2022/lib/gio-el/gio-el-prompt/gio-el-prompt.component.mjs +27 -7
- package/esm2022/lib/gio-el/gio-el.service.mjs +14 -2
- package/fesm2022/gravitee-ui-particles-angular.mjs +35 -5
- package/fesm2022/gravitee-ui-particles-angular.mjs.map +1 -1
- package/lib/gio-el/gio-el-prompt/gio-el-prompt.component.d.ts +5 -0
- package/lib/gio-el/gio-el.service.d.ts +4 -0
- package/package.json +1 -1
|
@@ -20,6 +20,7 @@ import { MatInput } from '@angular/material/input';
|
|
|
20
20
|
import { MatButton } from '@angular/material/button';
|
|
21
21
|
import { MatProgressBarModule } from '@angular/material/progress-bar';
|
|
22
22
|
import { MatTooltipModule } from '@angular/material/tooltip';
|
|
23
|
+
import { MatIconButton } from '@angular/material/button';
|
|
23
24
|
import { GioElService } from '../gio-el.service';
|
|
24
25
|
import { isPromptError, isPromptSuccess } from '../models/ElAiPromptState';
|
|
25
26
|
import { GioIconsModule } from '../../gio-icons/gio-icons.module';
|
|
@@ -29,17 +30,19 @@ import * as i0 from "@angular/core";
|
|
|
29
30
|
import * as i1 from "@angular/forms";
|
|
30
31
|
import * as i2 from "@angular/material/input";
|
|
31
32
|
import * as i3 from "@angular/material/progress-bar";
|
|
32
|
-
import * as i4 from "@angular/material/
|
|
33
|
-
import * as i5 from "
|
|
34
|
-
import * as i6 from "../../gio-
|
|
33
|
+
import * as i4 from "@angular/material/tooltip";
|
|
34
|
+
import * as i5 from "@angular/material/icon";
|
|
35
|
+
import * as i6 from "../../gio-banner/gio-banner.component";
|
|
36
|
+
import * as i7 from "../../gio-clipboard/gio-clipboard-copy-wrapper.component";
|
|
35
37
|
export class GioElPromptComponent {
|
|
36
38
|
constructor() {
|
|
37
39
|
this.maxPromptSize = 256;
|
|
38
40
|
this.aiRequestFormGroup = new FormGroup({
|
|
39
|
-
prompt: new FormControl('', [Validators.required, Validators.
|
|
41
|
+
prompt: new FormControl('', [Validators.required, Validators.maxLength(this.maxPromptSize)]),
|
|
40
42
|
});
|
|
41
43
|
this.elService = inject(GioElService);
|
|
42
44
|
this.responseState = signal(null);
|
|
45
|
+
this.feedbackState = signal(null);
|
|
43
46
|
this.el = computed(() => {
|
|
44
47
|
const state = this.responseState();
|
|
45
48
|
return isPromptSuccess(state) ? state.el : null;
|
|
@@ -48,6 +51,14 @@ export class GioElPromptComponent {
|
|
|
48
51
|
const state = this.responseState();
|
|
49
52
|
return isPromptError(state) ? state.message : null;
|
|
50
53
|
});
|
|
54
|
+
this.showFeedback = computed(() => {
|
|
55
|
+
const state = this.responseState();
|
|
56
|
+
const feedback = this.feedbackState();
|
|
57
|
+
return isPromptSuccess(state) && feedback === null;
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
get prompt() {
|
|
61
|
+
return this.aiRequestFormGroup.get('prompt');
|
|
51
62
|
}
|
|
52
63
|
ngAfterViewInit() {
|
|
53
64
|
setTimeout(() => {
|
|
@@ -60,10 +71,18 @@ export class GioElPromptComponent {
|
|
|
60
71
|
return;
|
|
61
72
|
}
|
|
62
73
|
this.responseState.set('loading');
|
|
74
|
+
this.feedbackState.set(null); // Reset feedback when sending new prompt
|
|
63
75
|
this.elService.prompt(prompt).subscribe(reply => this.responseState.set(reply));
|
|
64
76
|
}
|
|
77
|
+
submitFeedback(feedback) {
|
|
78
|
+
this.feedbackState.set(feedback);
|
|
79
|
+
const expression = this.el();
|
|
80
|
+
if (expression) {
|
|
81
|
+
this.elService.submitFeedback(feedback, expression).subscribe();
|
|
82
|
+
}
|
|
83
|
+
}
|
|
65
84
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioElPromptComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
66
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.4", type: GioElPromptComponent, isStandalone: true, selector: "gio-el-prompt", inputs: { responseState: "responseState" }, viewQueries: [{ propertyName: "myInput", first: true, predicate: ["promptArea"], descendants: true }], ngImport: i0, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{
|
|
85
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.4", type: GioElPromptComponent, isStandalone: true, selector: "gio-el-prompt", inputs: { responseState: "responseState" }, viewQueries: [{ propertyName: "myInput", first: true, predicate: ["promptArea"], descendants: true }], ngImport: i0, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{ prompt?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n </mat-form-field>\n\n @if (prompt?.touched && prompt?.hasError('maxlength')) {\n <gio-banner-error> Your prompt is too long. It should be less than {{ maxPromptSize }} characters. </gio-banner-error>\n }\n\n <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n <mat-icon svgIcon=\"gio:magic-wand\" />\n Ask Newt AI\n </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n @if (responseState() === 'loading') {\n <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n }\n @if (el(); as expression) {\n <gio-banner-success>\n <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n </gio-banner-success>\n @if (showFeedback()) {\n <gio-banner-info>\n Was the answer helpful?\n <span gioBannerBody class=\"feedback-buttons\">\n <button\n mat-icon-button\n (click)=\"submitFeedback('helpful')\"\n matTooltip=\"Yes, this was helpful\"\n class=\"feedback-button feedback-button--helpful\"\n >\n <mat-icon svgIcon=\"gio:thumbs-up\"></mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"submitFeedback('not-helpful')\"\n matTooltip=\"No, this was not helpful\"\n class=\"feedback-button feedback-button--not-helpful\">\n <mat-icon svgIcon=\"gio:thumbs-down\"></mat-icon>\n </button>\n </span>\n </gio-banner-info>\n }\n }\n @if (errorMessage(); as message) {\n <gio-banner-error>\n Error\n <span gioBannerBody> {{ message }} </span>\n </gio-banner-error>\n }\n</div>\n", styles: ["@charset \"UTF-8\";.gio-top-bar-menu .gio-badge-accent{background-color:var(--gio-oem-palette--active, #ffc2ac);color:var(--gio-oem-palette--active-contrast, #1e1b1b)}:host{display:flex;width:400px;flex-direction:column;padding:12px;border-radius:4px;background-color:#fff;box-shadow:0 2px 4px -1px #0003,0 4px 5px #00000024,0 1px 10px #0000001f}.aiRequestForm{display:flex;flex-direction:column;gap:8px}.banner__wrapper__title{width:100%}.banner-success-content{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.feedback-buttons{display:flex;gap:8px;align-items:center}.feedback-button{transition:all .2s ease-in-out}.feedback-button:hover{transform:scale(1.1)}.feedback-button--helpful{color:#22b374}.feedback-button--helpful:hover{background-color:#8dffce;opacity:.5}.feedback-button--not-helpful{color:#dd1d1f}.feedback-button--not-helpful:hover{background-color:#ffcfd0;opacity:.5}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i2.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i2.MatLabel, selector: "mat-label" }, { kind: "directive", type: i2.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly", "disabledInteractive"], exportAs: ["matInput"] }, { kind: "component", type: MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i3.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i4.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: GioBannerModule }, { kind: "component", type: i6.GioBannerErrorComponent, selector: "gio-banner-error" }, { kind: "component", type: i6.GioBannerInfoComponent, selector: "gio-banner-info" }, { kind: "component", type: i6.GioBannerSuccessComponent, selector: "gio-banner-success" }, { kind: "directive", type: i6.GioBannerBodyDirective, selector: "[gioBannerBody]" }, { kind: "ngmodule", type: GioClipboardModule }, { kind: "component", type: i7.GioClipboardCopyWrapperComponent, selector: "[gioClipboardCopyWrapper]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
67
86
|
}
|
|
68
87
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioElPromptComponent, decorators: [{
|
|
69
88
|
type: Component,
|
|
@@ -74,14 +93,15 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImpor
|
|
|
74
93
|
MatButton,
|
|
75
94
|
MatProgressBarModule,
|
|
76
95
|
MatTooltipModule,
|
|
96
|
+
MatIconButton,
|
|
77
97
|
GioIconsModule,
|
|
78
98
|
GioBannerModule,
|
|
79
99
|
GioClipboardModule,
|
|
80
|
-
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{
|
|
100
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{ prompt?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n </mat-form-field>\n\n @if (prompt?.touched && prompt?.hasError('maxlength')) {\n <gio-banner-error> Your prompt is too long. It should be less than {{ maxPromptSize }} characters. </gio-banner-error>\n }\n\n <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n <mat-icon svgIcon=\"gio:magic-wand\" />\n Ask Newt AI\n </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n @if (responseState() === 'loading') {\n <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n }\n @if (el(); as expression) {\n <gio-banner-success>\n <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n </gio-banner-success>\n @if (showFeedback()) {\n <gio-banner-info>\n Was the answer helpful?\n <span gioBannerBody class=\"feedback-buttons\">\n <button\n mat-icon-button\n (click)=\"submitFeedback('helpful')\"\n matTooltip=\"Yes, this was helpful\"\n class=\"feedback-button feedback-button--helpful\"\n >\n <mat-icon svgIcon=\"gio:thumbs-up\"></mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"submitFeedback('not-helpful')\"\n matTooltip=\"No, this was not helpful\"\n class=\"feedback-button feedback-button--not-helpful\">\n <mat-icon svgIcon=\"gio:thumbs-down\"></mat-icon>\n </button>\n </span>\n </gio-banner-info>\n }\n }\n @if (errorMessage(); as message) {\n <gio-banner-error>\n Error\n <span gioBannerBody> {{ message }} </span>\n </gio-banner-error>\n }\n</div>\n", styles: ["@charset \"UTF-8\";.gio-top-bar-menu .gio-badge-accent{background-color:var(--gio-oem-palette--active, #ffc2ac);color:var(--gio-oem-palette--active-contrast, #1e1b1b)}:host{display:flex;width:400px;flex-direction:column;padding:12px;border-radius:4px;background-color:#fff;box-shadow:0 2px 4px -1px #0003,0 4px 5px #00000024,0 1px 10px #0000001f}.aiRequestForm{display:flex;flex-direction:column;gap:8px}.banner__wrapper__title{width:100%}.banner-success-content{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.feedback-buttons{display:flex;gap:8px;align-items:center}.feedback-button{transition:all .2s ease-in-out}.feedback-button:hover{transform:scale(1.1)}.feedback-button--helpful{color:#22b374}.feedback-button--helpful:hover{background-color:#8dffce;opacity:.5}.feedback-button--not-helpful{color:#dd1d1f}.feedback-button--not-helpful:hover{background-color:#ffcfd0;opacity:.5}\n"] }]
|
|
81
101
|
}], propDecorators: { responseState: [{
|
|
82
102
|
type: Input
|
|
83
103
|
}], myInput: [{
|
|
84
104
|
type: ViewChild,
|
|
85
105
|
args: ['promptArea']
|
|
86
106
|
}] } });
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gio-el-prompt.component.js","sourceRoot":"","sources":["../../../../../../projects/ui-particles-angular/src/lib/gio-el/gio-el-prompt/gio-el-prompt.component.ts","../../../../../../projects/ui-particles-angular/src/lib/gio-el/gio-el-prompt/gio-el-prompt.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *         http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  signal,\n  WritableSignal,\n  Signal,\n  computed,\n  inject,\n  ViewChild,\n  ElementRef,\n  AfterViewInit,\n} from '@angular/core';\nimport { FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInput } from '@angular/material/input';\nimport { MatButton } from '@angular/material/button';\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\nimport { MatTooltipModule } from '@angular/material/tooltip';\n\nimport { GioElService } from '../gio-el.service';\nimport { ElAiPromptState, isPromptError, isPromptSuccess } from '../models/ElAiPromptState';\nimport { GioIconsModule } from '../../gio-icons/gio-icons.module';\nimport { GioBannerModule } from '../../gio-banner/gio-banner.module';\nimport { GioClipboardModule } from '../../gio-clipboard/gio-clipboard.module';\n\ntype PromptState = 'loading' | ElAiPromptState;\n\n@Component({\n  selector: 'gio-el-prompt',\n  templateUrl: './gio-el-prompt.component.html',\n  styleUrl: './gio-el-prompt.component.scss',\n  imports: [\n    ReactiveFormsModule,\n    MatFormFieldModule,\n    MatInput,\n    MatButton,\n    MatProgressBarModule,\n    MatTooltipModule,\n    GioIconsModule,\n    GioBannerModule,\n    GioClipboardModule,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class GioElPromptComponent implements AfterViewInit {\n  public readonly maxPromptSize = 256;\n  public aiRequestFormGroup: FormGroup<{\n    prompt: FormControl<string | null>;\n  }> = new FormGroup({\n    prompt: new FormControl('', [Validators.required, Validators.max(this.maxPromptSize)]),\n  });\n  public elService: GioElService = inject(GioElService);\n\n  @Input()\n  public responseState: WritableSignal<PromptState | null> = signal(null);\n  @ViewChild('promptArea') public myInput!: ElementRef<HTMLInputElement>;\n\n  public el: Signal<string | null> = computed(() => {\n    const state = this.responseState();\n    return isPromptSuccess(state) ? state.el : null;\n  });\n  public errorMessage: Signal<string | null> = computed(() => {\n    const state = this.responseState();\n    return isPromptError(state) ? state.message : null;\n  });\n\n  public ngAfterViewInit() {\n    setTimeout(() => {\n      this.myInput.nativeElement.focus();\n    }, 1000);\n  }\n\n  public sendPromptToIA() {\n    const prompt = this.aiRequestFormGroup.get('prompt')?.value;\n    if (!prompt) {\n      return;\n    }\n    this.responseState.set('loading');\n    this.elService.prompt(prompt).subscribe(reply => this.responseState.set(reply));\n  }\n}\n","<!--\n\n    Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n    Licensed under the Apache License, Version 2.0 (the \"License\");\n    you may not use this file except in compliance with the License.\n    You may obtain a copy of the License at\n\n            http://www.apache.org/licenses/LICENSE-2.0\n\n    Unless required by applicable law or agreed to in writing, software\n    distributed under the License is distributed on an \"AS IS\" BASIS,\n    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n    See the License for the specific language governing permissions and\n    limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n  <mat-form-field class=\"example-full-width\">\n    <mat-label>Describe the EL you want to generate</mat-label>\n    <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n    <mat-hint [align]=\"'end'\">{{ aiRequestFormGroup.get('prompt')?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n  </mat-form-field>\n\n  <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n    <mat-icon svgIcon=\"gio:magic-wand\" />\n    Ask Newt AI\n  </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n  @if (responseState() === 'loading') {\n    <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n  }\n  @if (el(); as expression) {\n    <gio-banner-success>\n      <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n    </gio-banner-success>\n  }\n  @if (errorMessage(); as message) {\n    <gio-banner-error>\n      Error\n      <span gioBannerBody> {{ message }} </span>\n    </gio-banner-error>\n  }\n</div>\n"]}
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gio-el-prompt.component.js","sourceRoot":"","sources":["../../../../../../projects/ui-particles-angular/src/lib/gio-el/gio-el-prompt/gio-el-prompt.component.ts","../../../../../../projects/ui-particles-angular/src/lib/gio-el/gio-el-prompt/gio-el-prompt.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *         http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\nimport {\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  signal,\n  WritableSignal,\n  Signal,\n  computed,\n  inject,\n  ViewChild,\n  ElementRef,\n  AfterViewInit,\n} from '@angular/core';\nimport { FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInput } from '@angular/material/input';\nimport { MatButton } from '@angular/material/button';\nimport { MatProgressBarModule } from '@angular/material/progress-bar';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatIconButton } from '@angular/material/button';\n\nimport { GioElService } from '../gio-el.service';\nimport { ElAiPromptState, isPromptError, isPromptSuccess } from '../models/ElAiPromptState';\nimport { GioIconsModule } from '../../gio-icons/gio-icons.module';\nimport { GioBannerModule } from '../../gio-banner/gio-banner.module';\nimport { GioClipboardModule } from '../../gio-clipboard/gio-clipboard.module';\n\ntype PromptState = 'loading' | ElAiPromptState;\ntype FeedbackState = 'helpful' | 'not-helpful' | null;\n\n@Component({\n  selector: 'gio-el-prompt',\n  templateUrl: './gio-el-prompt.component.html',\n  styleUrl: './gio-el-prompt.component.scss',\n  imports: [\n    ReactiveFormsModule,\n    MatFormFieldModule,\n    MatInput,\n    MatButton,\n    MatProgressBarModule,\n    MatTooltipModule,\n    MatIconButton,\n    GioIconsModule,\n    GioBannerModule,\n    GioClipboardModule,\n  ],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class GioElPromptComponent implements AfterViewInit {\n  public readonly maxPromptSize = 256;\n  public aiRequestFormGroup: FormGroup<{\n    prompt: FormControl<string | null>;\n  }> = new FormGroup({\n    prompt: new FormControl('', [Validators.required, Validators.maxLength(this.maxPromptSize)]),\n  });\n  public elService: GioElService = inject(GioElService);\n\n  @Input()\n  public responseState: WritableSignal<PromptState | null> = signal(null);\n  @ViewChild('promptArea') public myInput!: ElementRef<HTMLInputElement>;\n\n  public feedbackState: WritableSignal<FeedbackState> = signal(null);\n\n  public get prompt() {\n    return this.aiRequestFormGroup.get('prompt');\n  }\n  public el: Signal<string | null> = computed(() => {\n    const state = this.responseState();\n    return isPromptSuccess(state) ? state.el : null;\n  });\n  public errorMessage: Signal<string | null> = computed(() => {\n    const state = this.responseState();\n    return isPromptError(state) ? state.message : null;\n  });\n\n  public showFeedback: Signal<boolean> = computed(() => {\n    const state = this.responseState();\n    const feedback = this.feedbackState();\n    return isPromptSuccess(state) && feedback === null;\n  });\n\n  public ngAfterViewInit() {\n    setTimeout(() => {\n      this.myInput.nativeElement.focus();\n    }, 1000);\n  }\n\n  public sendPromptToIA() {\n    const prompt = this.aiRequestFormGroup.get('prompt')?.value;\n    if (!prompt) {\n      return;\n    }\n    this.responseState.set('loading');\n    this.feedbackState.set(null); // Reset feedback when sending new prompt\n    this.elService.prompt(prompt).subscribe(reply => this.responseState.set(reply));\n  }\n\n  public submitFeedback(feedback: 'helpful' | 'not-helpful') {\n    this.feedbackState.set(feedback);\n    const expression = this.el();\n    if (expression) {\n      this.elService.submitFeedback(feedback, expression).subscribe();\n    }\n  }\n}\n","<!--\n\n    Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n    Licensed under the Apache License, Version 2.0 (the \"License\");\n    you may not use this file except in compliance with the License.\n    You may obtain a copy of the License at\n\n            http://www.apache.org/licenses/LICENSE-2.0\n\n    Unless required by applicable law or agreed to in writing, software\n    distributed under the License is distributed on an \"AS IS\" BASIS,\n    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n    See the License for the specific language governing permissions and\n    limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n  <mat-form-field class=\"example-full-width\">\n    <mat-label>Describe the EL you want to generate</mat-label>\n    <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n    <mat-hint [align]=\"'end'\">{{ prompt?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n  </mat-form-field>\n\n  @if (prompt?.touched && prompt?.hasError('maxlength')) {\n    <gio-banner-error> Your prompt is too long. It should be less than {{ maxPromptSize }} characters. </gio-banner-error>\n  }\n\n  <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n    <mat-icon svgIcon=\"gio:magic-wand\" />\n    Ask Newt AI\n  </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n  @if (responseState() === 'loading') {\n    <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n  }\n  @if (el(); as expression) {\n    <gio-banner-success>\n      <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n    </gio-banner-success>\n    @if (showFeedback()) {\n      <gio-banner-info>\n        Was the answer helpful?\n        <span gioBannerBody class=\"feedback-buttons\">\n          <button\n            mat-icon-button\n            (click)=\"submitFeedback('helpful')\"\n            matTooltip=\"Yes, this was helpful\"\n            class=\"feedback-button feedback-button--helpful\"\n          >\n            <mat-icon svgIcon=\"gio:thumbs-up\"></mat-icon>\n          </button>\n          <button\n            mat-icon-button\n            (click)=\"submitFeedback('not-helpful')\"\n            matTooltip=\"No, this was not helpful\"\n            class=\"feedback-button feedback-button--not-helpful\">\n            <mat-icon svgIcon=\"gio:thumbs-down\"></mat-icon>\n          </button>\n        </span>\n      </gio-banner-info>\n    }\n  }\n  @if (errorMessage(); as message) {\n    <gio-banner-error>\n      Error\n      <span gioBannerBody> {{ message }} </span>\n    </gio-banner-error>\n  }\n</div>\n"]}
|
|
@@ -19,18 +19,30 @@ import * as i0 from "@angular/core";
|
|
|
19
19
|
export class GioElService {
|
|
20
20
|
constructor() {
|
|
21
21
|
this._promptCallback = undefined;
|
|
22
|
+
this._feedbackCallback = undefined;
|
|
22
23
|
}
|
|
23
24
|
prompt(prompt) {
|
|
24
25
|
if (this._promptCallback === undefined) {
|
|
25
26
|
return of({
|
|
26
|
-
|
|
27
|
+
el: 'Some fancy EL prompt',
|
|
27
28
|
});
|
|
28
29
|
}
|
|
29
30
|
return this._promptCallback(prompt);
|
|
30
31
|
}
|
|
32
|
+
submitFeedback(feedback, expression) {
|
|
33
|
+
if (this._feedbackCallback === undefined) {
|
|
34
|
+
// For now, just log the feedback - in a real implementation, this would send to backend
|
|
35
|
+
console.log('Feedback submitted:', { feedback, expression });
|
|
36
|
+
return of(undefined);
|
|
37
|
+
}
|
|
38
|
+
return this._feedbackCallback(feedback, expression);
|
|
39
|
+
}
|
|
31
40
|
set promptCallback(value) {
|
|
32
41
|
this._promptCallback = value;
|
|
33
42
|
}
|
|
43
|
+
set feedbackCallback(value) {
|
|
44
|
+
this._feedbackCallback = value;
|
|
45
|
+
}
|
|
34
46
|
isEnabled() {
|
|
35
47
|
return this._promptCallback !== undefined;
|
|
36
48
|
}
|
|
@@ -43,4 +55,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImpor
|
|
|
43
55
|
providedIn: 'root',
|
|
44
56
|
}]
|
|
45
57
|
}] });
|
|
46
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
58
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2lvLWVsLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy91aS1wYXJ0aWNsZXMtYW5ndWxhci9zcmMvbGliL2dpby1lbC9naW8tZWwuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRztBQUNILE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFjLEVBQUUsRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFTdEMsTUFBTSxPQUFPLFlBQVk7SUFIekI7UUFJVSxvQkFBZSxHQUFxRCxTQUFTLENBQUM7UUFDOUUsc0JBQWlCLEdBQXNFLFNBQVMsQ0FBQztLQStCMUc7SUE3QlEsTUFBTSxDQUFDLE1BQWM7UUFDMUIsSUFBSSxJQUFJLENBQUMsZUFBZSxLQUFLLFNBQVMsRUFBRSxDQUFDO1lBQ3ZDLE9BQU8sRUFBRSxDQUFDO2dCQUNSLEVBQUUsRUFBRSxzQkFBc0I7YUFDM0IsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUN0QyxDQUFDO0lBRU0sY0FBYyxDQUFDLFFBQXNCLEVBQUUsVUFBa0I7UUFDOUQsSUFBSSxJQUFJLENBQUMsaUJBQWlCLEtBQUssU0FBUyxFQUFFLENBQUM7WUFDekMsd0ZBQXdGO1lBQ3hGLE9BQU8sQ0FBQyxHQUFHLENBQUMscUJBQXFCLEVBQUUsRUFBRSxRQUFRLEVBQUUsVUFBVSxFQUFFLENBQUMsQ0FBQztZQUM3RCxPQUFPLEVBQUUsQ0FBQyxTQUFTLENBQUMsQ0FBQztRQUN2QixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUMsaUJBQWlCLENBQUMsUUFBUSxFQUFFLFVBQVUsQ0FBQyxDQUFDO0lBQ3RELENBQUM7SUFFRCxJQUFXLGNBQWMsQ0FBQyxLQUFzRDtRQUM5RSxJQUFJLENBQUMsZUFBZSxHQUFHLEtBQUssQ0FBQztJQUMvQixDQUFDO0lBRUQsSUFBVyxnQkFBZ0IsQ0FBQyxLQUF1RTtRQUNqRyxJQUFJLENBQUMsaUJBQWlCLEdBQUcsS0FBSyxDQUFDO0lBQ2pDLENBQUM7SUFFTSxTQUFTO1FBQ2QsT0FBTyxJQUFJLENBQUMsZUFBZSxLQUFLLFNBQVMsQ0FBQztJQUM1QyxDQUFDOzhHQWhDVSxZQUFZO2tIQUFaLFlBQVksY0FGWCxNQUFNOzsyRkFFUCxZQUFZO2tCQUh4QixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKEMpIDIwMjQgVGhlIEdyYXZpdGVlIHRlYW0gKGh0dHA6Ly9ncmF2aXRlZS5pbylcbiAqXG4gKiBMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgXCJMaWNlbnNlXCIpO1xuICogeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLlxuICogWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0XG4gKlxuICogICAgICAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjBcbiAqXG4gKiBVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlXG4gKiBkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiBcIkFTIElTXCIgQkFTSVMsXG4gKiBXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC5cbiAqIFNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmRcbiAqIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLlxuICovXG5pbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBPYnNlcnZhYmxlLCBvZiB9IGZyb20gJ3J4anMnO1xuXG5pbXBvcnQgeyBFbEFpUHJvbXB0U3RhdGUgfSBmcm9tICcuL21vZGVscy9FbEFpUHJvbXB0U3RhdGUnO1xuXG5leHBvcnQgdHlwZSBGZWVkYmFja1R5cGUgPSAnaGVscGZ1bCcgfCAnbm90LWhlbHBmdWwnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgR2lvRWxTZXJ2aWNlIHtcbiAgcHJpdmF0ZSBfcHJvbXB0Q2FsbGJhY2s/OiAocHJvbXB0OiBzdHJpbmcpID0+IE9ic2VydmFibGU8RWxBaVByb21wdFN0YXRlPiA9IHVuZGVmaW5lZDtcbiAgcHJpdmF0ZSBfZmVlZGJhY2tDYWxsYmFjaz86IChmZWVkYmFjazogRmVlZGJhY2tUeXBlLCBleHByZXNzaW9uOiBzdHJpbmcpID0+IE9ic2VydmFibGU8dm9pZD4gPSB1bmRlZmluZWQ7XG5cbiAgcHVibGljIHByb21wdChwcm9tcHQ6IHN0cmluZyk6IE9ic2VydmFibGU8RWxBaVByb21wdFN0YXRlPiB7XG4gICAgaWYgKHRoaXMuX3Byb21wdENhbGxiYWNrID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHJldHVybiBvZih7XG4gICAgICAgIGVsOiAnU29tZSBmYW5jeSBFTCBwcm9tcHQnLFxuICAgICAgfSk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9wcm9tcHRDYWxsYmFjayhwcm9tcHQpO1xuICB9XG5cbiAgcHVibGljIHN1Ym1pdEZlZWRiYWNrKGZlZWRiYWNrOiBGZWVkYmFja1R5cGUsIGV4cHJlc3Npb246IHN0cmluZyk6IE9ic2VydmFibGU8dm9pZD4ge1xuICAgIGlmICh0aGlzLl9mZWVkYmFja0NhbGxiYWNrID09PSB1bmRlZmluZWQpIHtcbiAgICAgIC8vIEZvciBub3csIGp1c3QgbG9nIHRoZSBmZWVkYmFjayAtIGluIGEgcmVhbCBpbXBsZW1lbnRhdGlvbiwgdGhpcyB3b3VsZCBzZW5kIHRvIGJhY2tlbmRcbiAgICAgIGNvbnNvbGUubG9nKCdGZWVkYmFjayBzdWJtaXR0ZWQ6JywgeyBmZWVkYmFjaywgZXhwcmVzc2lvbiB9KTtcbiAgICAgIHJldHVybiBvZih1bmRlZmluZWQpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fZmVlZGJhY2tDYWxsYmFjayhmZWVkYmFjaywgZXhwcmVzc2lvbik7XG4gIH1cblxuICBwdWJsaWMgc2V0IHByb21wdENhbGxiYWNrKHZhbHVlOiAocHJvbXB0OiBzdHJpbmcpID0+IE9ic2VydmFibGU8RWxBaVByb21wdFN0YXRlPikge1xuICAgIHRoaXMuX3Byb21wdENhbGxiYWNrID0gdmFsdWU7XG4gIH1cblxuICBwdWJsaWMgc2V0IGZlZWRiYWNrQ2FsbGJhY2sodmFsdWU6IChmZWVkYmFjazogRmVlZGJhY2tUeXBlLCBleHByZXNzaW9uOiBzdHJpbmcpID0+IE9ic2VydmFibGU8dm9pZD4pIHtcbiAgICB0aGlzLl9mZWVkYmFja0NhbGxiYWNrID0gdmFsdWU7XG4gIH1cblxuICBwdWJsaWMgaXNFbmFibGVkKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl9wcm9tcHRDYWxsYmFjayAhPT0gdW5kZWZpbmVkO1xuICB9XG59XG4iXX0=
|
|
@@ -8,7 +8,7 @@ import { CommonModule } from '@angular/common';
|
|
|
8
8
|
import * as i2 from '@angular/material/card';
|
|
9
9
|
import { MatCardModule } from '@angular/material/card';
|
|
10
10
|
import * as i3 from '@angular/material/button';
|
|
11
|
-
import { MatButtonModule, MatButton } from '@angular/material/button';
|
|
11
|
+
import { MatButtonModule, MatButton, MatIconButton } from '@angular/material/button';
|
|
12
12
|
import { ComponentHarness, HarnessPredicate, parallel, TestKey } from '@angular/cdk/testing';
|
|
13
13
|
import { coerceBooleanProperty } from '@angular/cdk/coercion';
|
|
14
14
|
import { MatFormFieldControl, MatFormFieldModule, MAT_FORM_FIELD_DEFAULT_OPTIONS } from '@angular/material/form-field';
|
|
@@ -1213,18 +1213,30 @@ class GioConfirmAndValidateDialogHarness extends ComponentHarness {
|
|
|
1213
1213
|
class GioElService {
|
|
1214
1214
|
constructor() {
|
|
1215
1215
|
this._promptCallback = undefined;
|
|
1216
|
+
this._feedbackCallback = undefined;
|
|
1216
1217
|
}
|
|
1217
1218
|
prompt(prompt) {
|
|
1218
1219
|
if (this._promptCallback === undefined) {
|
|
1219
1220
|
return of({
|
|
1220
|
-
|
|
1221
|
+
el: 'Some fancy EL prompt',
|
|
1221
1222
|
});
|
|
1222
1223
|
}
|
|
1223
1224
|
return this._promptCallback(prompt);
|
|
1224
1225
|
}
|
|
1226
|
+
submitFeedback(feedback, expression) {
|
|
1227
|
+
if (this._feedbackCallback === undefined) {
|
|
1228
|
+
// For now, just log the feedback - in a real implementation, this would send to backend
|
|
1229
|
+
console.log('Feedback submitted:', { feedback, expression });
|
|
1230
|
+
return of(undefined);
|
|
1231
|
+
}
|
|
1232
|
+
return this._feedbackCallback(feedback, expression);
|
|
1233
|
+
}
|
|
1225
1234
|
set promptCallback(value) {
|
|
1226
1235
|
this._promptCallback = value;
|
|
1227
1236
|
}
|
|
1237
|
+
set feedbackCallback(value) {
|
|
1238
|
+
this._feedbackCallback = value;
|
|
1239
|
+
}
|
|
1228
1240
|
isEnabled() {
|
|
1229
1241
|
return this._promptCallback !== undefined;
|
|
1230
1242
|
}
|
|
@@ -1493,10 +1505,11 @@ class GioElPromptComponent {
|
|
|
1493
1505
|
constructor() {
|
|
1494
1506
|
this.maxPromptSize = 256;
|
|
1495
1507
|
this.aiRequestFormGroup = new FormGroup({
|
|
1496
|
-
prompt: new FormControl('', [Validators.required, Validators.
|
|
1508
|
+
prompt: new FormControl('', [Validators.required, Validators.maxLength(this.maxPromptSize)]),
|
|
1497
1509
|
});
|
|
1498
1510
|
this.elService = inject(GioElService);
|
|
1499
1511
|
this.responseState = signal(null);
|
|
1512
|
+
this.feedbackState = signal(null);
|
|
1500
1513
|
this.el = computed(() => {
|
|
1501
1514
|
const state = this.responseState();
|
|
1502
1515
|
return isPromptSuccess(state) ? state.el : null;
|
|
@@ -1505,6 +1518,14 @@ class GioElPromptComponent {
|
|
|
1505
1518
|
const state = this.responseState();
|
|
1506
1519
|
return isPromptError(state) ? state.message : null;
|
|
1507
1520
|
});
|
|
1521
|
+
this.showFeedback = computed(() => {
|
|
1522
|
+
const state = this.responseState();
|
|
1523
|
+
const feedback = this.feedbackState();
|
|
1524
|
+
return isPromptSuccess(state) && feedback === null;
|
|
1525
|
+
});
|
|
1526
|
+
}
|
|
1527
|
+
get prompt() {
|
|
1528
|
+
return this.aiRequestFormGroup.get('prompt');
|
|
1508
1529
|
}
|
|
1509
1530
|
ngAfterViewInit() {
|
|
1510
1531
|
setTimeout(() => {
|
|
@@ -1517,10 +1538,18 @@ class GioElPromptComponent {
|
|
|
1517
1538
|
return;
|
|
1518
1539
|
}
|
|
1519
1540
|
this.responseState.set('loading');
|
|
1541
|
+
this.feedbackState.set(null); // Reset feedback when sending new prompt
|
|
1520
1542
|
this.elService.prompt(prompt).subscribe(reply => this.responseState.set(reply));
|
|
1521
1543
|
}
|
|
1544
|
+
submitFeedback(feedback) {
|
|
1545
|
+
this.feedbackState.set(feedback);
|
|
1546
|
+
const expression = this.el();
|
|
1547
|
+
if (expression) {
|
|
1548
|
+
this.elService.submitFeedback(feedback, expression).subscribe();
|
|
1549
|
+
}
|
|
1550
|
+
}
|
|
1522
1551
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioElPromptComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
1523
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.4", type: GioElPromptComponent, isStandalone: true, selector: "gio-el-prompt", inputs: { responseState: "responseState" }, viewQueries: [{ propertyName: "myInput", first: true, predicate: ["promptArea"], descendants: true }], ngImport: i0, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{
|
|
1552
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "19.2.4", type: GioElPromptComponent, isStandalone: true, selector: "gio-el-prompt", inputs: { responseState: "responseState" }, viewQueries: [{ propertyName: "myInput", first: true, predicate: ["promptArea"], descendants: true }], ngImport: i0, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{ prompt?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n </mat-form-field>\n\n @if (prompt?.touched && prompt?.hasError('maxlength')) {\n <gio-banner-error> Your prompt is too long. It should be less than {{ maxPromptSize }} characters. </gio-banner-error>\n }\n\n <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n <mat-icon svgIcon=\"gio:magic-wand\" />\n Ask Newt AI\n </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n @if (responseState() === 'loading') {\n <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n }\n @if (el(); as expression) {\n <gio-banner-success>\n <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n </gio-banner-success>\n @if (showFeedback()) {\n <gio-banner-info>\n Was the answer helpful?\n <span gioBannerBody class=\"feedback-buttons\">\n <button\n mat-icon-button\n (click)=\"submitFeedback('helpful')\"\n matTooltip=\"Yes, this was helpful\"\n class=\"feedback-button feedback-button--helpful\"\n >\n <mat-icon svgIcon=\"gio:thumbs-up\"></mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"submitFeedback('not-helpful')\"\n matTooltip=\"No, this was not helpful\"\n class=\"feedback-button feedback-button--not-helpful\">\n <mat-icon svgIcon=\"gio:thumbs-down\"></mat-icon>\n </button>\n </span>\n </gio-banner-info>\n }\n }\n @if (errorMessage(); as message) {\n <gio-banner-error>\n Error\n <span gioBannerBody> {{ message }} </span>\n </gio-banner-error>\n }\n</div>\n", styles: ["@charset \"UTF-8\";.gio-top-bar-menu .gio-badge-accent{background-color:var(--gio-oem-palette--active, #ffc2ac);color:var(--gio-oem-palette--active-contrast, #1e1b1b)}:host{display:flex;width:400px;flex-direction:column;padding:12px;border-radius:4px;background-color:#fff;box-shadow:0 2px 4px -1px #0003,0 4px 5px #00000024,0 1px 10px #0000001f}.aiRequestForm{display:flex;flex-direction:column;gap:8px}.banner__wrapper__title{width:100%}.banner-success-content{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.feedback-buttons{display:flex;gap:8px;align-items:center}.feedback-button{transition:all .2s ease-in-out}.feedback-button:hover{transform:scale(1.1)}.feedback-button--helpful{color:#22b374}.feedback-button--helpful:hover{background-color:#8dffce;opacity:.5}.feedback-button--not-helpful{color:#dd1d1f}.feedback-button--not-helpful:hover{background-color:#ffcfd0;opacity:.5}\n"], dependencies: [{ kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1$2.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1$2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1$2.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1$2.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i1$2.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i5.MatLabel, selector: "mat-label" }, { kind: "directive", type: i5.MatHint, selector: "mat-hint", inputs: ["align", "id"] }, { kind: "directive", type: MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly", "disabledInteractive"], exportAs: ["matInput"] }, { kind: "component", type: MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: MatProgressBarModule }, { kind: "component", type: i3$2.MatProgressBar, selector: "mat-progress-bar", inputs: ["color", "value", "bufferValue", "mode"], outputs: ["animationEnd"], exportAs: ["matProgressBar"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i2$2.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "component", type: MatIconButton, selector: "button[mat-icon-button]", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i1$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: GioBannerModule }, { kind: "component", type: GioBannerErrorComponent, selector: "gio-banner-error" }, { kind: "component", type: GioBannerInfoComponent, selector: "gio-banner-info" }, { kind: "component", type: GioBannerSuccessComponent, selector: "gio-banner-success" }, { kind: "directive", type: GioBannerBodyDirective, selector: "[gioBannerBody]" }, { kind: "ngmodule", type: GioClipboardModule }, { kind: "component", type: GioClipboardCopyWrapperComponent, selector: "[gioClipboardCopyWrapper]" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
1524
1553
|
}
|
|
1525
1554
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioElPromptComponent, decorators: [{
|
|
1526
1555
|
type: Component,
|
|
@@ -1531,10 +1560,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImpor
|
|
|
1531
1560
|
MatButton,
|
|
1532
1561
|
MatProgressBarModule,
|
|
1533
1562
|
MatTooltipModule,
|
|
1563
|
+
MatIconButton,
|
|
1534
1564
|
GioIconsModule,
|
|
1535
1565
|
GioBannerModule,
|
|
1536
1566
|
GioClipboardModule,
|
|
1537
|
-
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{
|
|
1567
|
+
], changeDetection: ChangeDetectionStrategy.OnPush, template: "<!--\n\n Copyright (C) 2025 The Gravitee team (http://gravitee.io)\n\n Licensed under the Apache License, Version 2.0 (the \"License\");\n you may not use this file except in compliance with the License.\n You may obtain a copy of the License at\n\n http://www.apache.org/licenses/LICENSE-2.0\n\n Unless required by applicable law or agreed to in writing, software\n distributed under the License is distributed on an \"AS IS\" BASIS,\n WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n See the License for the specific language governing permissions and\n limitations under the License.\n\n-->\n<h5>EL IA Assistant</h5>\n\n<div class=\"aiRequestForm\" [formGroup]=\"aiRequestFormGroup\">\n <mat-form-field class=\"example-full-width\">\n <mat-label>Describe the EL you want to generate</mat-label>\n <textarea #promptArea matInput formControlName=\"prompt\"></textarea>\n <mat-hint [align]=\"'end'\">{{ prompt?.value?.length || 0 }}/{{ maxPromptSize }}</mat-hint>\n </mat-form-field>\n\n @if (prompt?.touched && prompt?.hasError('maxlength')) {\n <gio-banner-error> Your prompt is too long. It should be less than {{ maxPromptSize }} characters. </gio-banner-error>\n }\n\n <button mat-raised-button [disabled]=\"aiRequestFormGroup.invalid\" (click)=\"sendPromptToIA()\">\n <mat-icon svgIcon=\"gio:magic-wand\" />\n Ask Newt AI\n </button>\n</div>\n\n<div class=\"ai-card__content__response\">\n @if (responseState() === 'loading') {\n <mat-progress-bar mode=\"indeterminate\" class=\"loading\" />\n }\n @if (el(); as expression) {\n <gio-banner-success>\n <code gioClipboardCopyWrapper [contentToCopy]=\"expression\" [alwaysVisible]=\"true\">{{ expression }}</code>\n </gio-banner-success>\n @if (showFeedback()) {\n <gio-banner-info>\n Was the answer helpful?\n <span gioBannerBody class=\"feedback-buttons\">\n <button\n mat-icon-button\n (click)=\"submitFeedback('helpful')\"\n matTooltip=\"Yes, this was helpful\"\n class=\"feedback-button feedback-button--helpful\"\n >\n <mat-icon svgIcon=\"gio:thumbs-up\"></mat-icon>\n </button>\n <button\n mat-icon-button\n (click)=\"submitFeedback('not-helpful')\"\n matTooltip=\"No, this was not helpful\"\n class=\"feedback-button feedback-button--not-helpful\">\n <mat-icon svgIcon=\"gio:thumbs-down\"></mat-icon>\n </button>\n </span>\n </gio-banner-info>\n }\n }\n @if (errorMessage(); as message) {\n <gio-banner-error>\n Error\n <span gioBannerBody> {{ message }} </span>\n </gio-banner-error>\n }\n</div>\n", styles: ["@charset \"UTF-8\";.gio-top-bar-menu .gio-badge-accent{background-color:var(--gio-oem-palette--active, #ffc2ac);color:var(--gio-oem-palette--active-contrast, #1e1b1b)}:host{display:flex;width:400px;flex-direction:column;padding:12px;border-radius:4px;background-color:#fff;box-shadow:0 2px 4px -1px #0003,0 4px 5px #00000024,0 1px 10px #0000001f}.aiRequestForm{display:flex;flex-direction:column;gap:8px}.banner__wrapper__title{width:100%}.banner-success-content{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.feedback-buttons{display:flex;gap:8px;align-items:center}.feedback-button{transition:all .2s ease-in-out}.feedback-button:hover{transform:scale(1.1)}.feedback-button--helpful{color:#22b374}.feedback-button--helpful:hover{background-color:#8dffce;opacity:.5}.feedback-button--not-helpful{color:#dd1d1f}.feedback-button--not-helpful:hover{background-color:#ffcfd0;opacity:.5}\n"] }]
|
|
1538
1568
|
}], propDecorators: { responseState: [{
|
|
1539
1569
|
type: Input
|
|
1540
1570
|
}], myInput: [{
|