@gravitee/ui-policy-studio-angular 15.4.1 → 15.5.0

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.
@@ -80,6 +80,8 @@ export class GioPolicyStudioComponent {
80
80
  * @returns Observable of the policy documentation
81
81
  */
82
82
  this.policyDocumentationFetcher = () => EMPTY;
83
+ this.selectedFlowIndexes = { planIndex: 0, flowIndex: 0 };
84
+ this.selectedFlowChanged = new EventEmitter();
83
85
  /**
84
86
  * Return what is needed to save.
85
87
  **/
@@ -101,12 +103,11 @@ export class GioPolicyStudioComponent {
101
103
  .map(e => e.name)
102
104
  .join(', ')}`;
103
105
  }
104
- if (changes.commonFlows || changes.plans) {
106
+ if (changes.commonFlows || changes.plans || changes.selectedFlowIndexes) {
105
107
  this.disableSaveButton = true;
106
108
  this.flowsGroups = getFlowsGroups(this.apiType, this.commonFlows, this.plans);
107
109
  this.initialFlowsGroups = cloneDeep(this.flowsGroups);
108
- // Select first flow by default on first load
109
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows))[0];
110
+ this.selectedFlow = this.flowsGroups[this.selectedFlowIndexes?.planIndex ?? 0]?.flows[this.selectedFlowIndexes?.flowIndex ?? 0];
110
111
  // Reset saving state when flowsGroups are updated
111
112
  this.saving = false;
112
113
  this.unSavingButtonSubscription?.unsubscribe();
@@ -147,18 +148,30 @@ export class GioPolicyStudioComponent {
147
148
  this.disableSaveButton = false;
148
149
  }
149
150
  onSelectFlow(flowId) {
150
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === flowId);
151
+ let newPlanIndex = 0;
152
+ let newFlowIndex = 0;
153
+ for (let currentPlanIndex = 0; currentPlanIndex < this.flowsGroups.length; currentPlanIndex++) {
154
+ const indexWithinPlan = this.flowsGroups[currentPlanIndex].flows.findIndex(flow => flow._id === flowId);
155
+ if (indexWithinPlan !== -1) {
156
+ newPlanIndex = currentPlanIndex;
157
+ newFlowIndex = indexWithinPlan;
158
+ break;
159
+ }
160
+ }
161
+ this.selectedFlowIndexes = { planIndex: newPlanIndex, flowIndex: newFlowIndex };
162
+ this.selectedFlow = this.flowsGroups[newPlanIndex].flows[newFlowIndex];
163
+ this.selectedFlowChanged.emit(this.selectedFlowIndexes);
151
164
  }
152
165
  onDeleteSelectedFlow(flow) {
153
166
  // Define next selected flow and remove flow from flowsGroups
154
167
  const allFlowsId = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).map(f => f._id);
155
168
  const flowToDeleteIndex = allFlowsId.indexOf(flow._id);
156
- const nextSelectedFlow = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];
169
+ const nextSelectedFlowId = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];
157
170
  this.flowsGroups = this.flowsGroups.map(flowGroup => ({
158
171
  ...flowGroup,
159
172
  flows: flowGroup.flows.filter(f => f._id !== flow._id),
160
173
  }));
161
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === nextSelectedFlow);
174
+ this.onSelectFlow(nextSelectedFlowId);
162
175
  this.disableSaveButton = false;
163
176
  }
164
177
  onSave() {
@@ -183,7 +196,7 @@ export class GioPolicyStudioComponent {
183
196
  this.genericPolicies = toGenericPolicies(this.policies ?? [], this.sharedPolicyGroupPolicies ?? []);
184
197
  }
185
198
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioPolicyStudioComponent, deps: [{ token: i1.GioPolicyStudioService }], target: i0.ɵɵFactoryTarget.Component }); }
186
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "19.2.4", type: GioPolicyStudioComponent, isStandalone: true, selector: "gio-policy-studio", inputs: { readOnly: "readOnly", apiType: "apiType", flowExecution: "flowExecution", entrypointsInfo: "entrypointsInfo", endpointsInfo: "endpointsInfo", commonFlows: "commonFlows", plans: "plans", policies: ["policies", "policies", (policies) => policies ?? []], sharedPolicyGroupPolicies: "sharedPolicyGroupPolicies", trialUrl: "trialUrl", loading: "loading", policySchemaFetcher: "policySchemaFetcher", policyDocumentationFetcher: "policyDocumentationFetcher" }, outputs: { save: "save" }, usesOnChanges: true, ngImport: i0, template: "<!--\n\n Copyright (C) 2015 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\n<div class=\"header\">\n <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n <span class=\"gio-badge-primary\">\n {{ apiType | titlecase }}\n </span>\n <span\n class=\"gio-badge-neutral\"\n [matTooltip]=\"connectorsTooltip\"\n matTooltipPosition=\"right\"\n matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n >\n <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n </span>\n </div>\n <ng-template #loadingApiInfo>\n <div class=\"header__apiInfo\">\n <span class=\"gio-badge-primary\">&nbsp;</span>\n <span class=\"gio-badge-neutral\">&nbsp;</span>\n </div>\n </ng-template>\n\n <div class=\"header__btn\">\n <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n </div>\n</div>\n\n<div class=\"wrapper\">\n <div class=\"wrapper__flowsMenu\">\n <gio-ps-flows-menu\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flowExecution]=\"flowExecution\"\n [flowsGroups]=\"flowsGroups\"\n [entrypoints]=\"entrypointsInfo\"\n [entrypointsInfo]=\"entrypointsInfo\"\n (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n [selectedFlowId]=\"selectedFlow?._id\"\n (selectedFlowIdChange)=\"onSelectFlow($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flows-menu>\n </div>\n\n <div class=\"wrapper__flowDetails\">\n <gio-ps-flow-details\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flow]=\"selectedFlow\"\n [entrypointsInfo]=\"entrypointsInfo\"\n [endpointsInfo]=\"endpointsInfo\"\n [policies]=\"policies\"\n [genericPolicies]=\"genericPolicies\"\n [trialUrl]=\"trialUrl\"\n (flowChange)=\"onSelectedFlowChange($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flow-details>\n </div>\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;height:100%;flex-direction:column;padding:16px;background-color:#fff}::ng-deep .gio-policy-studio__tooltip-line-break{white-space:pre-line}.header{display:flex;align-items:center;padding-bottom:16px}.header__apiInfo{flex:1 1 auto}.wrapper{display:flex;min-height:0;flex:1 1 auto;flex-direction:row;gap:16px}.wrapper__flowsMenu{overflow:hidden;min-width:230px;max-width:400px;flex:1 1 33%;border:1px solid #d2cecd;border-radius:8px}.wrapper__flowDetails{overflow:hidden;flex:1 1 66%;border:1px solid #d2cecd;border-radius:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.TitleCasePipe, name: "titlecase" }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i3.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: GioPolicyStudioFlowsMenuComponent, selector: "gio-ps-flows-menu", inputs: ["readOnly", "loading", "apiType", "flowExecution", "flowsGroups", "selectedFlowId", "entrypoints", "entrypointsInfo"], outputs: ["selectedFlowIdChange", "flowsGroupsChange", "flowExecutionChange", "deleteFlow"] }, { kind: "component", type: GioPolicyStudioDetailsComponent, selector: "gio-ps-flow-details", inputs: ["readOnly", "loading", "apiType", "flow", "entrypointsInfo", "endpointsInfo", "policies", "genericPolicies", "trialUrl"], outputs: ["flowChange", "deleteFlow"] }] }); }
199
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "19.2.4", type: GioPolicyStudioComponent, isStandalone: true, selector: "gio-policy-studio", inputs: { readOnly: "readOnly", apiType: "apiType", flowExecution: "flowExecution", entrypointsInfo: "entrypointsInfo", endpointsInfo: "endpointsInfo", commonFlows: "commonFlows", plans: "plans", policies: ["policies", "policies", (policies) => policies ?? []], sharedPolicyGroupPolicies: "sharedPolicyGroupPolicies", trialUrl: "trialUrl", loading: "loading", policySchemaFetcher: "policySchemaFetcher", policyDocumentationFetcher: "policyDocumentationFetcher", selectedFlowIndexes: "selectedFlowIndexes" }, outputs: { selectedFlowChanged: "selectedFlowChanged", save: "save" }, usesOnChanges: true, ngImport: i0, template: "<!--\n\n Copyright (C) 2015 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\n<div class=\"header\">\n <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n <span class=\"gio-badge-primary\">\n {{ apiType | titlecase }}\n </span>\n <span\n class=\"gio-badge-neutral\"\n [matTooltip]=\"connectorsTooltip\"\n matTooltipPosition=\"right\"\n matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n >\n <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n </span>\n </div>\n <ng-template #loadingApiInfo>\n <div class=\"header__apiInfo\">\n <span class=\"gio-badge-primary\">&nbsp;</span>\n <span class=\"gio-badge-neutral\">&nbsp;</span>\n </div>\n </ng-template>\n\n <div class=\"header__btn\">\n <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n </div>\n</div>\n\n<div class=\"wrapper\">\n <div class=\"wrapper__flowsMenu\">\n <gio-ps-flows-menu\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flowExecution]=\"flowExecution\"\n [flowsGroups]=\"flowsGroups\"\n [entrypoints]=\"entrypointsInfo\"\n [entrypointsInfo]=\"entrypointsInfo\"\n (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n [selectedFlowId]=\"selectedFlow?._id\"\n (selectedFlowIdChange)=\"onSelectFlow($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flows-menu>\n </div>\n\n <div class=\"wrapper__flowDetails\">\n <gio-ps-flow-details\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flow]=\"selectedFlow\"\n [entrypointsInfo]=\"entrypointsInfo\"\n [endpointsInfo]=\"endpointsInfo\"\n [policies]=\"policies\"\n [genericPolicies]=\"genericPolicies\"\n [trialUrl]=\"trialUrl\"\n (flowChange)=\"onSelectedFlowChange($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flow-details>\n </div>\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;height:100%;flex-direction:column;padding:16px;background-color:#fff}::ng-deep .gio-policy-studio__tooltip-line-break{white-space:pre-line}.header{display:flex;align-items:center;padding-bottom:16px}.header__apiInfo{flex:1 1 auto}.wrapper{display:flex;min-height:0;flex:1 1 auto;flex-direction:row;gap:16px}.wrapper__flowsMenu{overflow:hidden;min-width:230px;max-width:400px;flex:1 1 33%;border:1px solid #d2cecd;border-radius:8px}.wrapper__flowDetails{overflow:hidden;flex:1 1 66%;border:1px solid #d2cecd;border-radius:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.TitleCasePipe, name: "titlecase" }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i3.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: GioPolicyStudioFlowsMenuComponent, selector: "gio-ps-flows-menu", inputs: ["readOnly", "loading", "apiType", "flowExecution", "flowsGroups", "selectedFlowId", "entrypoints", "entrypointsInfo"], outputs: ["selectedFlowIdChange", "flowsGroupsChange", "flowExecutionChange", "deleteFlow"] }, { kind: "component", type: GioPolicyStudioDetailsComponent, selector: "gio-ps-flow-details", inputs: ["readOnly", "loading", "apiType", "flow", "entrypointsInfo", "endpointsInfo", "policies", "genericPolicies", "trialUrl"], outputs: ["flowChange", "deleteFlow"] }] }); }
187
200
  }
188
201
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioPolicyStudioComponent, decorators: [{
189
202
  type: Component,
@@ -224,6 +237,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImpor
224
237
  type: Input
225
238
  }], policyDocumentationFetcher: [{
226
239
  type: Input
240
+ }], selectedFlowIndexes: [{
241
+ type: Input
242
+ }], selectedFlowChanged: [{
243
+ type: Output
227
244
  }], save: [{
228
245
  type: Output
229
246
  }] } });
@@ -280,4 +297,4 @@ const getPlansChangeOutput = (flowsGroups, initialFlowsGroups) => {
280
297
  }));
281
298
  return plansWithChangedFlowsOutput.length > 0 ? plansWithChangedFlowsOutput : null;
282
299
  };
283
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gio-policy-studio.component.js","sourceRoot":"","sources":["../../../../../projects/ui-policy-studio-angular/src/lib/policy-studio/gio-policy-studio.component.ts","../../../../../projects/ui-policy-studio-angular/src/lib/policy-studio/gio-policy-studio.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (C) 2022 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 { Component, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\nimport { cloneDeep, differenceBy, flatten, isEqual, omit, unionBy, uniqueId } from 'lodash';\nimport { EMPTY, Subscription, timer } from 'rxjs';\nimport { CommonModule } from '@angular/common';\nimport { GioIconsModule } from '@gravitee/ui-particles-angular';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatButtonModule } from '@angular/material/button';\n\nimport {\n  ConnectorInfo,\n  Flow,\n  FlowExecution,\n  Plan,\n  Policy,\n  PolicyDocumentationFetcher,\n  PolicySchemaFetcher,\n  SaveOutput,\n  SharedPolicyGroupPolicy,\n  toGenericPolicies,\n  GenericPolicy,\n  PolicyInput,\n  fromPolicyInput,\n  ApiType,\n} from '../models';\nimport { GioPolicyStudioFlowsMenuComponent } from '../components/flows-menu/gio-ps-flows-menu.component';\nimport { GioPolicyStudioDetailsComponent } from '../components/flow-details/gio-ps-flow-details.component';\n\nimport { FlowGroupVM, FlowVM } from './gio-policy-studio.model';\nimport { GioPolicyStudioService } from './gio-policy-studio.service';\n\n@Component({\n  imports: [\n    CommonModule,\n    MatTooltipModule,\n    MatButtonModule,\n    GioIconsModule,\n    GioPolicyStudioFlowsMenuComponent,\n    GioPolicyStudioDetailsComponent,\n  ],\n  selector: 'gio-policy-studio',\n  templateUrl: './gio-policy-studio.component.html',\n  styleUrls: ['./gio-policy-studio.component.scss'],\n})\nexport class GioPolicyStudioComponent implements OnChanges, OnDestroy {\n  /**\n   * May be set to `true` if the API is not manageable from the UI (e.g. kubernetes operator APIs)\n   */\n  @Input()\n  public readOnly = false;\n\n  /**\n   * API type (required)\n   */\n  @Input()\n  public apiType!: ApiType;\n\n  /**\n   * Flow execution config (required)\n   */\n  @Input()\n  public flowExecution!: FlowExecution;\n\n  /**\n   * List of entrypoints to display\n   */\n  @Input()\n  public entrypointsInfo: ConnectorInfo[] = [];\n\n  /**\n   * List of endpoints to display\n   */\n  @Input()\n  public endpointsInfo: ConnectorInfo[] = [];\n\n  /**\n   * List of common flows to add to common flows group\n   */\n  @Input()\n  public commonFlows: Flow[] = [];\n\n  /**\n   * List of plans with their flows\n   */\n  @Input()\n  public plans: Plan[] = [];\n\n  /**\n   * List of policies usable in the policy studio\n   */\n  @Input({\n    transform: (policies: unknown) => policies ?? [],\n  })\n  public set policies(value: (PolicyInput | Policy)[]) {\n    this._policies = value.map(fromPolicyInput);\n  }\n  public get policies(): Policy[] {\n    return this._policies;\n  }\n  private _policies: Policy[] = [];\n\n  /**\n   * List of SharedPolicyGroups usable in the studio\n   */\n  @Input()\n  public sharedPolicyGroupPolicies: SharedPolicyGroupPolicy[] = [];\n\n  /**\n   * When a policy is not available with the current license,\n   * this URL is used to redirect the user to the trial page.\n   */\n  @Input()\n  public trialUrl?: string;\n\n  /**\n   * Loading state\n   */\n  @Input()\n  public loading = false;\n\n  /**\n   * Called when Policy Studio needs to fetch the policy schema\n   * @returns Observable of the policy schema\n   */\n  @Input()\n  public policySchemaFetcher: PolicySchemaFetcher = () => EMPTY;\n\n  /**\n   * Called when Policy Studio needs to fetch the policy documentation\n   * @returns Observable of the policy documentation\n   */\n  @Input()\n  public policyDocumentationFetcher: PolicyDocumentationFetcher = () => EMPTY;\n\n  /**\n   * Return what is needed to save.\n   **/\n  @Output()\n  public save = new EventEmitter<SaveOutput>();\n\n  public connectorsTooltip = '';\n\n  public selectedFlow?: FlowVM = undefined;\n\n  public flowsGroups: FlowGroupVM[] = [];\n\n  public disableSaveButton = true;\n\n  public saving = false;\n\n  // Used to keep track of initial flows groups to know if there are deleted flows\n  private initialFlowsGroups: FlowGroupVM[] = [];\n\n  private hasFlowExecutionChanged = false;\n\n  private unSavingButtonSubscription?: Subscription;\n  public enableSavingTimer = true;\n\n  protected genericPolicies: GenericPolicy[] = [];\n\n  constructor(private readonly policyStudioService: GioPolicyStudioService) {}\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (changes.entrypointsInfo || changes.endpointsInfo) {\n      this.connectorsTooltip = `Entrypoints: ${(this.entrypointsInfo ?? []).map(e => e.name).join(', ')}\\nEndpoints: ${(\n        this.endpointsInfo ?? []\n      )\n        .map(e => e.name)\n        .join(', ')}`;\n    }\n\n    if (changes.commonFlows || changes.plans) {\n      this.disableSaveButton = true;\n      this.flowsGroups = getFlowsGroups(this.apiType, this.commonFlows, this.plans);\n      this.initialFlowsGroups = cloneDeep(this.flowsGroups);\n\n      // Select first flow by default on first load\n      this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows))[0];\n\n      // Reset saving state when flowsGroups are updated\n      this.saving = false;\n      this.unSavingButtonSubscription?.unsubscribe();\n    }\n\n    if (changes.policySchemaFetcher) {\n      this.policyStudioService.setPolicySchemaFetcher(this.policySchemaFetcher);\n    }\n\n    if (changes.policyDocumentationFetcher) {\n      this.policyStudioService.setPolicyDocumentationFetcher(this.policyDocumentationFetcher);\n    }\n\n    if (changes.policies || changes.sharedPolicyGroupPolicies) {\n      this.initGenericPolicies();\n    }\n  }\n\n  public ngOnDestroy() {\n    this.unSavingButtonSubscription?.unsubscribe();\n  }\n\n  public onFlowExecutionChange(flowExecution: FlowExecution): void {\n    this.hasFlowExecutionChanged = !isEqual(this.flowExecution, flowExecution);\n    if (this.hasFlowExecutionChanged) {\n      this.disableSaveButton = false;\n      this.flowExecution = flowExecution;\n    }\n  }\n\n  public onFlowsGroupsChange(flowsGroups: FlowGroupVM[]): void {\n    this.flowsGroups = cloneDeep(flowsGroups);\n    this.disableSaveButton = false;\n    if (this.selectedFlow) {\n      this.onSelectFlow(this.selectedFlow._id);\n    }\n  }\n\n  public onSelectedFlowChange(flow: FlowVM): void {\n    this.flowsGroups = this.flowsGroups.map(flowGroup => ({\n      ...flowGroup,\n      flows: flowGroup.flows.map(f => (f._id === flow._id ? flow : f)),\n    }));\n    this.selectedFlow = flow;\n    this.disableSaveButton = false;\n  }\n\n  public onSelectFlow(flowId: string): void {\n    this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === flowId);\n  }\n\n  public onDeleteSelectedFlow(flow: FlowVM): void {\n    // Define next selected flow and remove flow from flowsGroups\n    const allFlowsId = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).map(f => f._id);\n    const flowToDeleteIndex = allFlowsId.indexOf(flow._id);\n    const nextSelectedFlow = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];\n    this.flowsGroups = this.flowsGroups.map(flowGroup => ({\n      ...flowGroup,\n      flows: flowGroup.flows.filter(f => f._id !== flow._id),\n    }));\n\n    this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === nextSelectedFlow);\n\n    this.disableSaveButton = false;\n  }\n\n  public onSave(): void {\n    // Emit common flows only if they have been updated\n    const commonFlows = getCommonFlowsOutput(this.flowsGroups, this.initialFlowsGroups);\n\n    // Emit plans only if they have been updated\n    const plansToUpdate = getPlansChangeOutput(this.flowsGroups, this.initialFlowsGroups);\n    this.save.emit({\n      ...(commonFlows ? { commonFlows } : {}),\n      ...(plansToUpdate ? { plansToUpdate } : {}),\n      ...(this.hasFlowExecutionChanged ? { flowExecution: this.flowExecution } : {}),\n    });\n    this.saving = true;\n\n    this.unSavingButtonSubscription?.unsubscribe();\n    if (this.enableSavingTimer) {\n      this.unSavingButtonSubscription = timer(5000).subscribe(() => {\n        this.saving = false;\n      });\n    }\n  }\n\n  private initGenericPolicies(): void {\n    this.genericPolicies = toGenericPolicies(this.policies ?? [], this.sharedPolicyGroupPolicies ?? []);\n  }\n}\n\nconst getFlowsGroups = (apiType: ApiType, commonFlows: Flow[] = [], plans: Plan[] = []): FlowGroupVM[] => {\n  const commFlowsGroup: FlowGroupVM = {\n    _id: 'flowsGroup_commonFlow',\n    _isPlan: false,\n    name: 'All plans',\n    flows: commonFlows.map(flow => ({ ...flow, _id: uniqueId('flow_'), _hasChanged: false, _parentFlowGroupName: 'Common' })),\n  };\n\n  return [\n    ...plans.map(plan => ({\n      _id: uniqueId('flowsGroup_'),\n      _planId: plan.id,\n      _isPlan: true,\n      name: plan.name,\n      flows: plan.flows.map(flow => ({ ...flow, _id: uniqueId('flow_'), _hasChanged: false, _parentFlowGroupName: plan.name })),\n    })),\n    commFlowsGroup,\n  ];\n};\n\nconst getCommonFlowsOutput = (flowsGroups: FlowGroupVM[], initialFlowsGroups: FlowGroupVM[]): Flow[] | null => {\n  const commonFlowsGroup = flowsGroups.find(flowGroup => flowGroup._id === 'flowsGroup_commonFlow');\n  const initialCommonFlowsGroup = initialFlowsGroups.find(flowGroup => flowGroup._id === 'flowsGroup_commonFlow');\n  // Check if common flows have been updated\n  const hasChanged = commonFlowsGroup?.flows.some(flow => flow._hasChanged);\n\n  // Check if common flows have been deleted\n  const hasDeletedFlow = differenceBy(initialCommonFlowsGroup?.flows ?? [], commonFlowsGroup?.flows ?? [], '_id').length > 0;\n\n  return hasChanged || hasDeletedFlow\n    ? (commonFlowsGroup?.flows ?? []).map(flow => omit(flow, '_id', '_hasChanged', '_parentFlowGroupName'))\n    : null;\n};\n\nconst getPlansChangeOutput = (flowsGroups: FlowGroupVM[], initialFlowsGroups: FlowGroupVM[]): Plan[] | null => {\n  const plansGroups = flowsGroups.filter(flowGroup => flowGroup._id !== 'flowsGroup_commonFlow');\n  const initialPlansGroups = initialFlowsGroups.filter(flowGroup => flowGroup._id !== 'flowsGroup_commonFlow');\n\n  // Get plans with changed flows\n  const plansGroupsWithChangedFlows = plansGroups.filter(plan => plan.flows.some(flow => flow._hasChanged));\n\n  // Get plans with deleted flows\n  const plansGroupsWithDeletedFlows: FlowGroupVM[] = [];\n\n  // Check if there are deleted flows in initial initialPlansGroups\n  initialPlansGroups.forEach(plan => {\n    const planGroupToCompare = plansGroups.find(p => p._planId === plan._planId);\n    if (planGroupToCompare && differenceBy(plan.flows, planGroupToCompare?.flows ?? [], '_id').length > 0) {\n      // If there are deleted flows, we need to return the complete plan with the new flows\n      plansGroupsWithDeletedFlows.push(planGroupToCompare);\n    }\n  });\n\n  // Merge plans with changed flows and plans with deleted flows\n  const plansWithChangedFlowsOutput = unionBy([...plansGroupsWithChangedFlows, ...plansGroupsWithDeletedFlows], '_planId').map(plan => ({\n    ...omit(plan, '_id', '_isPlan', '_planId'),\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- planId is always defined\n    id: plan._planId!,\n    flows: plan.flows.map(flow => omit(flow, '_id', '_hasChanged', '_parentFlowGroupName')),\n  }));\n\n  return plansWithChangedFlowsOutput.length > 0 ? plansWithChangedFlowsOutput : null;\n};\n","<!--\n\n    Copyright (C) 2015 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\n<div class=\"header\">\n  <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n    <span class=\"gio-badge-primary\">\n      {{ apiType | titlecase }}\n    </span>\n    <span\n      class=\"gio-badge-neutral\"\n      [matTooltip]=\"connectorsTooltip\"\n      matTooltipPosition=\"right\"\n      matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n    >\n      <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n      <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n    </span>\n  </div>\n  <ng-template #loadingApiInfo>\n    <div class=\"header__apiInfo\">\n      <span class=\"gio-badge-primary\">&nbsp;</span>\n      <span class=\"gio-badge-neutral\">&nbsp;</span>\n    </div>\n  </ng-template>\n\n  <div class=\"header__btn\">\n    <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n    <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n  </div>\n</div>\n\n<div class=\"wrapper\">\n  <div class=\"wrapper__flowsMenu\">\n    <gio-ps-flows-menu\n      [readOnly]=\"readOnly\"\n      [loading]=\"loading\"\n      [apiType]=\"apiType\"\n      [flowExecution]=\"flowExecution\"\n      [flowsGroups]=\"flowsGroups\"\n      [entrypoints]=\"entrypointsInfo\"\n      [entrypointsInfo]=\"entrypointsInfo\"\n      (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n      (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n      [selectedFlowId]=\"selectedFlow?._id\"\n      (selectedFlowIdChange)=\"onSelectFlow($event)\"\n      (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n    ></gio-ps-flows-menu>\n  </div>\n\n  <div class=\"wrapper__flowDetails\">\n    <gio-ps-flow-details\n      [readOnly]=\"readOnly\"\n      [loading]=\"loading\"\n      [apiType]=\"apiType\"\n      [flow]=\"selectedFlow\"\n      [entrypointsInfo]=\"entrypointsInfo\"\n      [endpointsInfo]=\"endpointsInfo\"\n      [policies]=\"policies\"\n      [genericPolicies]=\"genericPolicies\"\n      [trialUrl]=\"trialUrl\"\n      (flowChange)=\"onSelectedFlowChange($event)\"\n      (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n    ></gio-ps-flow-details>\n  </div>\n</div>\n"]}
300
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"gio-policy-studio.component.js","sourceRoot":"","sources":["../../../../../projects/ui-policy-studio-angular/src/lib/policy-studio/gio-policy-studio.component.ts","../../../../../projects/ui-policy-studio-angular/src/lib/policy-studio/gio-policy-studio.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (C) 2022 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 { Component, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\nimport { cloneDeep, differenceBy, flatten, isEqual, omit, unionBy, uniqueId } from 'lodash';\nimport { EMPTY, Subscription, timer } from 'rxjs';\nimport { CommonModule } from '@angular/common';\nimport { GioIconsModule } from '@gravitee/ui-particles-angular';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatButtonModule } from '@angular/material/button';\n\nimport {\n  ConnectorInfo,\n  Flow,\n  FlowExecution,\n  Plan,\n  Policy,\n  PolicyDocumentationFetcher,\n  PolicySchemaFetcher,\n  SaveOutput,\n  SharedPolicyGroupPolicy,\n  toGenericPolicies,\n  GenericPolicy,\n  PolicyInput,\n  fromPolicyInput,\n  ApiType,\n} from '../models';\nimport { GioPolicyStudioFlowsMenuComponent } from '../components/flows-menu/gio-ps-flows-menu.component';\nimport { GioPolicyStudioDetailsComponent } from '../components/flow-details/gio-ps-flow-details.component';\n\nimport { FlowGroupVM, FlowVM } from './gio-policy-studio.model';\nimport { GioPolicyStudioService } from './gio-policy-studio.service';\n\nexport type FlowSelection = { planIndex: number; flowIndex: number };\n\n@Component({\n  imports: [\n    CommonModule,\n    MatTooltipModule,\n    MatButtonModule,\n    GioIconsModule,\n    GioPolicyStudioFlowsMenuComponent,\n    GioPolicyStudioDetailsComponent,\n  ],\n  selector: 'gio-policy-studio',\n  templateUrl: './gio-policy-studio.component.html',\n  styleUrls: ['./gio-policy-studio.component.scss'],\n})\nexport class GioPolicyStudioComponent implements OnChanges, OnDestroy {\n  /**\n   * May be set to `true` if the API is not manageable from the UI (e.g. kubernetes operator APIs)\n   */\n  @Input()\n  public readOnly = false;\n\n  /**\n   * API type (required)\n   */\n  @Input()\n  public apiType!: ApiType;\n\n  /**\n   * Flow execution config (required)\n   */\n  @Input()\n  public flowExecution!: FlowExecution;\n\n  /**\n   * List of entrypoints to display\n   */\n  @Input()\n  public entrypointsInfo: ConnectorInfo[] = [];\n\n  /**\n   * List of endpoints to display\n   */\n  @Input()\n  public endpointsInfo: ConnectorInfo[] = [];\n\n  /**\n   * List of common flows to add to common flows group\n   */\n  @Input()\n  public commonFlows: Flow[] = [];\n\n  /**\n   * List of plans with their flows\n   */\n  @Input()\n  public plans: Plan[] = [];\n\n  /**\n   * List of policies usable in the policy studio\n   */\n  @Input({\n    transform: (policies: unknown) => policies ?? [],\n  })\n  public set policies(value: (PolicyInput | Policy)[]) {\n    this._policies = value.map(fromPolicyInput);\n  }\n  public get policies(): Policy[] {\n    return this._policies;\n  }\n  private _policies: Policy[] = [];\n\n  /**\n   * List of SharedPolicyGroups usable in the studio\n   */\n  @Input()\n  public sharedPolicyGroupPolicies: SharedPolicyGroupPolicy[] = [];\n\n  /**\n   * When a policy is not available with the current license,\n   * this URL is used to redirect the user to the trial page.\n   */\n  @Input()\n  public trialUrl?: string;\n\n  /**\n   * Loading state\n   */\n  @Input()\n  public loading = false;\n\n  /**\n   * Called when Policy Studio needs to fetch the policy schema\n   * @returns Observable of the policy schema\n   */\n  @Input()\n  public policySchemaFetcher: PolicySchemaFetcher = () => EMPTY;\n\n  /**\n   * Called when Policy Studio needs to fetch the policy documentation\n   * @returns Observable of the policy documentation\n   */\n  @Input()\n  public policyDocumentationFetcher: PolicyDocumentationFetcher = () => EMPTY;\n\n  @Input()\n  public selectedFlowIndexes: FlowSelection = { planIndex: 0, flowIndex: 0 };\n\n  @Output()\n  public selectedFlowChanged = new EventEmitter<FlowSelection>();\n\n  /**\n   * Return what is needed to save.\n   **/\n  @Output()\n  public save = new EventEmitter<SaveOutput>();\n\n  public connectorsTooltip = '';\n\n  public selectedFlow?: FlowVM = undefined;\n\n  public flowsGroups: FlowGroupVM[] = [];\n\n  public disableSaveButton = true;\n\n  public saving = false;\n\n  // Used to keep track of initial flows groups to know if there are deleted flows\n  private initialFlowsGroups: FlowGroupVM[] = [];\n\n  private hasFlowExecutionChanged = false;\n\n  private unSavingButtonSubscription?: Subscription;\n  public enableSavingTimer = true;\n\n  protected genericPolicies: GenericPolicy[] = [];\n\n  constructor(private readonly policyStudioService: GioPolicyStudioService) {}\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (changes.entrypointsInfo || changes.endpointsInfo) {\n      this.connectorsTooltip = `Entrypoints: ${(this.entrypointsInfo ?? []).map(e => e.name).join(', ')}\\nEndpoints: ${(\n        this.endpointsInfo ?? []\n      )\n        .map(e => e.name)\n        .join(', ')}`;\n    }\n\n    if (changes.commonFlows || changes.plans || changes.selectedFlowIndexes) {\n      this.disableSaveButton = true;\n      this.flowsGroups = getFlowsGroups(this.apiType, this.commonFlows, this.plans);\n      this.initialFlowsGroups = cloneDeep(this.flowsGroups);\n      this.selectedFlow = this.flowsGroups[this.selectedFlowIndexes?.planIndex ?? 0]?.flows[this.selectedFlowIndexes?.flowIndex ?? 0];\n\n      // Reset saving state when flowsGroups are updated\n      this.saving = false;\n      this.unSavingButtonSubscription?.unsubscribe();\n    }\n\n    if (changes.policySchemaFetcher) {\n      this.policyStudioService.setPolicySchemaFetcher(this.policySchemaFetcher);\n    }\n\n    if (changes.policyDocumentationFetcher) {\n      this.policyStudioService.setPolicyDocumentationFetcher(this.policyDocumentationFetcher);\n    }\n\n    if (changes.policies || changes.sharedPolicyGroupPolicies) {\n      this.initGenericPolicies();\n    }\n  }\n\n  public ngOnDestroy() {\n    this.unSavingButtonSubscription?.unsubscribe();\n  }\n\n  public onFlowExecutionChange(flowExecution: FlowExecution): void {\n    this.hasFlowExecutionChanged = !isEqual(this.flowExecution, flowExecution);\n    if (this.hasFlowExecutionChanged) {\n      this.disableSaveButton = false;\n      this.flowExecution = flowExecution;\n    }\n  }\n\n  public onFlowsGroupsChange(flowsGroups: FlowGroupVM[]): void {\n    this.flowsGroups = cloneDeep(flowsGroups);\n    this.disableSaveButton = false;\n    if (this.selectedFlow) {\n      this.onSelectFlow(this.selectedFlow._id);\n    }\n  }\n\n  public onSelectedFlowChange(flow: FlowVM): void {\n    this.flowsGroups = this.flowsGroups.map(flowGroup => ({\n      ...flowGroup,\n      flows: flowGroup.flows.map(f => (f._id === flow._id ? flow : f)),\n    }));\n    this.selectedFlow = flow;\n    this.disableSaveButton = false;\n  }\n\n  public onSelectFlow(flowId: string): void {\n    let newPlanIndex = 0;\n    let newFlowIndex = 0;\n    for (let currentPlanIndex = 0; currentPlanIndex < this.flowsGroups.length; currentPlanIndex++) {\n      const indexWithinPlan = this.flowsGroups[currentPlanIndex].flows.findIndex(flow => flow._id === flowId);\n      if (indexWithinPlan !== -1) {\n        newPlanIndex = currentPlanIndex;\n        newFlowIndex = indexWithinPlan;\n        break;\n      }\n    }\n\n    this.selectedFlowIndexes = { planIndex: newPlanIndex, flowIndex: newFlowIndex };\n\n    this.selectedFlow = this.flowsGroups[newPlanIndex].flows[newFlowIndex];\n\n    this.selectedFlowChanged.emit(this.selectedFlowIndexes);\n  }\n\n  public onDeleteSelectedFlow(flow: FlowVM): void {\n    // Define next selected flow and remove flow from flowsGroups\n    const allFlowsId = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).map(f => f._id);\n    const flowToDeleteIndex = allFlowsId.indexOf(flow._id);\n    const nextSelectedFlowId = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];\n    this.flowsGroups = this.flowsGroups.map(flowGroup => ({\n      ...flowGroup,\n      flows: flowGroup.flows.filter(f => f._id !== flow._id),\n    }));\n\n    this.onSelectFlow(nextSelectedFlowId);\n\n    this.disableSaveButton = false;\n  }\n\n  public onSave(): void {\n    // Emit common flows only if they have been updated\n    const commonFlows = getCommonFlowsOutput(this.flowsGroups, this.initialFlowsGroups);\n\n    // Emit plans only if they have been updated\n    const plansToUpdate = getPlansChangeOutput(this.flowsGroups, this.initialFlowsGroups);\n    this.save.emit({\n      ...(commonFlows ? { commonFlows } : {}),\n      ...(plansToUpdate ? { plansToUpdate } : {}),\n      ...(this.hasFlowExecutionChanged ? { flowExecution: this.flowExecution } : {}),\n    });\n    this.saving = true;\n\n    this.unSavingButtonSubscription?.unsubscribe();\n    if (this.enableSavingTimer) {\n      this.unSavingButtonSubscription = timer(5000).subscribe(() => {\n        this.saving = false;\n      });\n    }\n  }\n\n  private initGenericPolicies(): void {\n    this.genericPolicies = toGenericPolicies(this.policies ?? [], this.sharedPolicyGroupPolicies ?? []);\n  }\n}\n\nconst getFlowsGroups = (apiType: ApiType, commonFlows: Flow[] = [], plans: Plan[] = []): FlowGroupVM[] => {\n  const commFlowsGroup: FlowGroupVM = {\n    _id: 'flowsGroup_commonFlow',\n    _isPlan: false,\n    name: 'All plans',\n    flows: commonFlows.map(flow => ({ ...flow, _id: uniqueId('flow_'), _hasChanged: false, _parentFlowGroupName: 'Common' })),\n  };\n\n  return [\n    ...plans.map(plan => ({\n      _id: uniqueId('flowsGroup_'),\n      _planId: plan.id,\n      _isPlan: true,\n      name: plan.name,\n      flows: plan.flows.map(flow => ({ ...flow, _id: uniqueId('flow_'), _hasChanged: false, _parentFlowGroupName: plan.name })),\n    })),\n    commFlowsGroup,\n  ];\n};\n\nconst getCommonFlowsOutput = (flowsGroups: FlowGroupVM[], initialFlowsGroups: FlowGroupVM[]): Flow[] | null => {\n  const commonFlowsGroup = flowsGroups.find(flowGroup => flowGroup._id === 'flowsGroup_commonFlow');\n  const initialCommonFlowsGroup = initialFlowsGroups.find(flowGroup => flowGroup._id === 'flowsGroup_commonFlow');\n  // Check if common flows have been updated\n  const hasChanged = commonFlowsGroup?.flows.some(flow => flow._hasChanged);\n\n  // Check if common flows have been deleted\n  const hasDeletedFlow = differenceBy(initialCommonFlowsGroup?.flows ?? [], commonFlowsGroup?.flows ?? [], '_id').length > 0;\n\n  return hasChanged || hasDeletedFlow\n    ? (commonFlowsGroup?.flows ?? []).map(flow => omit(flow, '_id', '_hasChanged', '_parentFlowGroupName'))\n    : null;\n};\n\nconst getPlansChangeOutput = (flowsGroups: FlowGroupVM[], initialFlowsGroups: FlowGroupVM[]): Plan[] | null => {\n  const plansGroups = flowsGroups.filter(flowGroup => flowGroup._id !== 'flowsGroup_commonFlow');\n  const initialPlansGroups = initialFlowsGroups.filter(flowGroup => flowGroup._id !== 'flowsGroup_commonFlow');\n\n  // Get plans with changed flows\n  const plansGroupsWithChangedFlows = plansGroups.filter(plan => plan.flows.some(flow => flow._hasChanged));\n\n  // Get plans with deleted flows\n  const plansGroupsWithDeletedFlows: FlowGroupVM[] = [];\n\n  // Check if there are deleted flows in initial initialPlansGroups\n  initialPlansGroups.forEach(plan => {\n    const planGroupToCompare = plansGroups.find(p => p._planId === plan._planId);\n    if (planGroupToCompare && differenceBy(plan.flows, planGroupToCompare?.flows ?? [], '_id').length > 0) {\n      // If there are deleted flows, we need to return the complete plan with the new flows\n      plansGroupsWithDeletedFlows.push(planGroupToCompare);\n    }\n  });\n\n  // Merge plans with changed flows and plans with deleted flows\n  const plansWithChangedFlowsOutput = unionBy([...plansGroupsWithChangedFlows, ...plansGroupsWithDeletedFlows], '_planId').map(plan => ({\n    ...omit(plan, '_id', '_isPlan', '_planId'),\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion -- planId is always defined\n    id: plan._planId!,\n    flows: plan.flows.map(flow => omit(flow, '_id', '_hasChanged', '_parentFlowGroupName')),\n  }));\n\n  return plansWithChangedFlowsOutput.length > 0 ? plansWithChangedFlowsOutput : null;\n};\n","<!--\n\n    Copyright (C) 2015 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\n<div class=\"header\">\n  <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n    <span class=\"gio-badge-primary\">\n      {{ apiType | titlecase }}\n    </span>\n    <span\n      class=\"gio-badge-neutral\"\n      [matTooltip]=\"connectorsTooltip\"\n      matTooltipPosition=\"right\"\n      matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n    >\n      <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n      <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n    </span>\n  </div>\n  <ng-template #loadingApiInfo>\n    <div class=\"header__apiInfo\">\n      <span class=\"gio-badge-primary\">&nbsp;</span>\n      <span class=\"gio-badge-neutral\">&nbsp;</span>\n    </div>\n  </ng-template>\n\n  <div class=\"header__btn\">\n    <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n    <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n  </div>\n</div>\n\n<div class=\"wrapper\">\n  <div class=\"wrapper__flowsMenu\">\n    <gio-ps-flows-menu\n      [readOnly]=\"readOnly\"\n      [loading]=\"loading\"\n      [apiType]=\"apiType\"\n      [flowExecution]=\"flowExecution\"\n      [flowsGroups]=\"flowsGroups\"\n      [entrypoints]=\"entrypointsInfo\"\n      [entrypointsInfo]=\"entrypointsInfo\"\n      (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n      (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n      [selectedFlowId]=\"selectedFlow?._id\"\n      (selectedFlowIdChange)=\"onSelectFlow($event)\"\n      (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n    ></gio-ps-flows-menu>\n  </div>\n\n  <div class=\"wrapper__flowDetails\">\n    <gio-ps-flow-details\n      [readOnly]=\"readOnly\"\n      [loading]=\"loading\"\n      [apiType]=\"apiType\"\n      [flow]=\"selectedFlow\"\n      [entrypointsInfo]=\"entrypointsInfo\"\n      [endpointsInfo]=\"endpointsInfo\"\n      [policies]=\"policies\"\n      [genericPolicies]=\"genericPolicies\"\n      [trialUrl]=\"trialUrl\"\n      (flowChange)=\"onSelectedFlowChange($event)\"\n      (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n    ></gio-ps-flow-details>\n  </div>\n</div>\n"]}
@@ -2094,6 +2094,8 @@ class GioPolicyStudioComponent {
2094
2094
  * @returns Observable of the policy documentation
2095
2095
  */
2096
2096
  this.policyDocumentationFetcher = () => EMPTY;
2097
+ this.selectedFlowIndexes = { planIndex: 0, flowIndex: 0 };
2098
+ this.selectedFlowChanged = new EventEmitter();
2097
2099
  /**
2098
2100
  * Return what is needed to save.
2099
2101
  **/
@@ -2115,12 +2117,11 @@ class GioPolicyStudioComponent {
2115
2117
  .map(e => e.name)
2116
2118
  .join(', ')}`;
2117
2119
  }
2118
- if (changes.commonFlows || changes.plans) {
2120
+ if (changes.commonFlows || changes.plans || changes.selectedFlowIndexes) {
2119
2121
  this.disableSaveButton = true;
2120
2122
  this.flowsGroups = getFlowsGroups(this.apiType, this.commonFlows, this.plans);
2121
2123
  this.initialFlowsGroups = cloneDeep(this.flowsGroups);
2122
- // Select first flow by default on first load
2123
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows))[0];
2124
+ this.selectedFlow = this.flowsGroups[this.selectedFlowIndexes?.planIndex ?? 0]?.flows[this.selectedFlowIndexes?.flowIndex ?? 0];
2124
2125
  // Reset saving state when flowsGroups are updated
2125
2126
  this.saving = false;
2126
2127
  this.unSavingButtonSubscription?.unsubscribe();
@@ -2161,18 +2162,30 @@ class GioPolicyStudioComponent {
2161
2162
  this.disableSaveButton = false;
2162
2163
  }
2163
2164
  onSelectFlow(flowId) {
2164
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === flowId);
2165
+ let newPlanIndex = 0;
2166
+ let newFlowIndex = 0;
2167
+ for (let currentPlanIndex = 0; currentPlanIndex < this.flowsGroups.length; currentPlanIndex++) {
2168
+ const indexWithinPlan = this.flowsGroups[currentPlanIndex].flows.findIndex(flow => flow._id === flowId);
2169
+ if (indexWithinPlan !== -1) {
2170
+ newPlanIndex = currentPlanIndex;
2171
+ newFlowIndex = indexWithinPlan;
2172
+ break;
2173
+ }
2174
+ }
2175
+ this.selectedFlowIndexes = { planIndex: newPlanIndex, flowIndex: newFlowIndex };
2176
+ this.selectedFlow = this.flowsGroups[newPlanIndex].flows[newFlowIndex];
2177
+ this.selectedFlowChanged.emit(this.selectedFlowIndexes);
2165
2178
  }
2166
2179
  onDeleteSelectedFlow(flow) {
2167
2180
  // Define next selected flow and remove flow from flowsGroups
2168
2181
  const allFlowsId = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).map(f => f._id);
2169
2182
  const flowToDeleteIndex = allFlowsId.indexOf(flow._id);
2170
- const nextSelectedFlow = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];
2183
+ const nextSelectedFlowId = allFlowsId[flowToDeleteIndex + 1] ?? allFlowsId[flowToDeleteIndex - 1];
2171
2184
  this.flowsGroups = this.flowsGroups.map(flowGroup => ({
2172
2185
  ...flowGroup,
2173
2186
  flows: flowGroup.flows.filter(f => f._id !== flow._id),
2174
2187
  }));
2175
- this.selectedFlow = flatten(this.flowsGroups.map(flowGroup => flowGroup.flows)).find(f => f._id === nextSelectedFlow);
2188
+ this.onSelectFlow(nextSelectedFlowId);
2176
2189
  this.disableSaveButton = false;
2177
2190
  }
2178
2191
  onSave() {
@@ -2197,7 +2210,7 @@ class GioPolicyStudioComponent {
2197
2210
  this.genericPolicies = toGenericPolicies(this.policies ?? [], this.sharedPolicyGroupPolicies ?? []);
2198
2211
  }
2199
2212
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioPolicyStudioComponent, deps: [{ token: GioPolicyStudioService }], target: i0.ɵɵFactoryTarget.Component }); }
2200
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "19.2.4", type: GioPolicyStudioComponent, isStandalone: true, selector: "gio-policy-studio", inputs: { readOnly: "readOnly", apiType: "apiType", flowExecution: "flowExecution", entrypointsInfo: "entrypointsInfo", endpointsInfo: "endpointsInfo", commonFlows: "commonFlows", plans: "plans", policies: ["policies", "policies", (policies) => policies ?? []], sharedPolicyGroupPolicies: "sharedPolicyGroupPolicies", trialUrl: "trialUrl", loading: "loading", policySchemaFetcher: "policySchemaFetcher", policyDocumentationFetcher: "policyDocumentationFetcher" }, outputs: { save: "save" }, usesOnChanges: true, ngImport: i0, template: "<!--\n\n Copyright (C) 2015 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\n<div class=\"header\">\n <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n <span class=\"gio-badge-primary\">\n {{ apiType | titlecase }}\n </span>\n <span\n class=\"gio-badge-neutral\"\n [matTooltip]=\"connectorsTooltip\"\n matTooltipPosition=\"right\"\n matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n >\n <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n </span>\n </div>\n <ng-template #loadingApiInfo>\n <div class=\"header__apiInfo\">\n <span class=\"gio-badge-primary\">&nbsp;</span>\n <span class=\"gio-badge-neutral\">&nbsp;</span>\n </div>\n </ng-template>\n\n <div class=\"header__btn\">\n <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n </div>\n</div>\n\n<div class=\"wrapper\">\n <div class=\"wrapper__flowsMenu\">\n <gio-ps-flows-menu\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flowExecution]=\"flowExecution\"\n [flowsGroups]=\"flowsGroups\"\n [entrypoints]=\"entrypointsInfo\"\n [entrypointsInfo]=\"entrypointsInfo\"\n (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n [selectedFlowId]=\"selectedFlow?._id\"\n (selectedFlowIdChange)=\"onSelectFlow($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flows-menu>\n </div>\n\n <div class=\"wrapper__flowDetails\">\n <gio-ps-flow-details\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flow]=\"selectedFlow\"\n [entrypointsInfo]=\"entrypointsInfo\"\n [endpointsInfo]=\"endpointsInfo\"\n [policies]=\"policies\"\n [genericPolicies]=\"genericPolicies\"\n [trialUrl]=\"trialUrl\"\n (flowChange)=\"onSelectedFlowChange($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flow-details>\n </div>\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;height:100%;flex-direction:column;padding:16px;background-color:#fff}::ng-deep .gio-policy-studio__tooltip-line-break{white-space:pre-line}.header{display:flex;align-items:center;padding-bottom:16px}.header__apiInfo{flex:1 1 auto}.wrapper{display:flex;min-height:0;flex:1 1 auto;flex-direction:row;gap:16px}.wrapper__flowsMenu{overflow:hidden;min-width:230px;max-width:400px;flex:1 1 33%;border:1px solid #d2cecd;border-radius:8px}.wrapper__flowDetails{overflow:hidden;flex:1 1 66%;border:1px solid #d2cecd;border-radius:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.TitleCasePipe, name: "titlecase" }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i3$1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i5$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: GioPolicyStudioFlowsMenuComponent, selector: "gio-ps-flows-menu", inputs: ["readOnly", "loading", "apiType", "flowExecution", "flowsGroups", "selectedFlowId", "entrypoints", "entrypointsInfo"], outputs: ["selectedFlowIdChange", "flowsGroupsChange", "flowExecutionChange", "deleteFlow"] }, { kind: "component", type: GioPolicyStudioDetailsComponent, selector: "gio-ps-flow-details", inputs: ["readOnly", "loading", "apiType", "flow", "entrypointsInfo", "endpointsInfo", "policies", "genericPolicies", "trialUrl"], outputs: ["flowChange", "deleteFlow"] }] }); }
2213
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "16.1.0", version: "19.2.4", type: GioPolicyStudioComponent, isStandalone: true, selector: "gio-policy-studio", inputs: { readOnly: "readOnly", apiType: "apiType", flowExecution: "flowExecution", entrypointsInfo: "entrypointsInfo", endpointsInfo: "endpointsInfo", commonFlows: "commonFlows", plans: "plans", policies: ["policies", "policies", (policies) => policies ?? []], sharedPolicyGroupPolicies: "sharedPolicyGroupPolicies", trialUrl: "trialUrl", loading: "loading", policySchemaFetcher: "policySchemaFetcher", policyDocumentationFetcher: "policyDocumentationFetcher", selectedFlowIndexes: "selectedFlowIndexes" }, outputs: { selectedFlowChanged: "selectedFlowChanged", save: "save" }, usesOnChanges: true, ngImport: i0, template: "<!--\n\n Copyright (C) 2015 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\n<div class=\"header\">\n <div class=\"header__apiInfo\" *ngIf=\"!loading; else loadingApiInfo\">\n <span class=\"gio-badge-primary\">\n {{ apiType | titlecase }}\n </span>\n <span\n class=\"gio-badge-neutral\"\n [matTooltip]=\"connectorsTooltip\"\n matTooltipPosition=\"right\"\n matTooltipClass=\"gio-policy-studio__tooltip-line-break\"\n >\n <mat-icon *ngFor=\"let entrypoint of entrypointsInfo\" class=\"gio-left\" [svgIcon]=\"entrypoint.icon\"></mat-icon>\n <mat-icon *ngFor=\"let endpoint of endpointsInfo\" class=\"gio-left\" [svgIcon]=\"endpoint.icon\"></mat-icon>\n </span>\n </div>\n <ng-template #loadingApiInfo>\n <div class=\"header__apiInfo\">\n <span class=\"gio-badge-primary\">&nbsp;</span>\n <span class=\"gio-badge-neutral\">&nbsp;</span>\n </div>\n </ng-template>\n\n <div class=\"header__btn\">\n <button *ngIf=\"!saving\" mat-flat-button color=\"primary\" [disabled]=\"loading || disableSaveButton\" (click)=\"onSave()\">Save</button>\n <button *ngIf=\"saving\" mat-flat-button color=\"primary\" disabled>Saving...</button>\n </div>\n</div>\n\n<div class=\"wrapper\">\n <div class=\"wrapper__flowsMenu\">\n <gio-ps-flows-menu\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flowExecution]=\"flowExecution\"\n [flowsGroups]=\"flowsGroups\"\n [entrypoints]=\"entrypointsInfo\"\n [entrypointsInfo]=\"entrypointsInfo\"\n (flowsGroupsChange)=\"onFlowsGroupsChange($event)\"\n (flowExecutionChange)=\"onFlowExecutionChange($event)\"\n [selectedFlowId]=\"selectedFlow?._id\"\n (selectedFlowIdChange)=\"onSelectFlow($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flows-menu>\n </div>\n\n <div class=\"wrapper__flowDetails\">\n <gio-ps-flow-details\n [readOnly]=\"readOnly\"\n [loading]=\"loading\"\n [apiType]=\"apiType\"\n [flow]=\"selectedFlow\"\n [entrypointsInfo]=\"entrypointsInfo\"\n [endpointsInfo]=\"endpointsInfo\"\n [policies]=\"policies\"\n [genericPolicies]=\"genericPolicies\"\n [trialUrl]=\"trialUrl\"\n (flowChange)=\"onSelectedFlowChange($event)\"\n (deleteFlow)=\"onDeleteSelectedFlow($event)\"\n ></gio-ps-flow-details>\n </div>\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;height:100%;flex-direction:column;padding:16px;background-color:#fff}::ng-deep .gio-policy-studio__tooltip-line-break{white-space:pre-line}.header{display:flex;align-items:center;padding-bottom:16px}.header__apiInfo{flex:1 1 auto}.wrapper{display:flex;min-height:0;flex:1 1 auto;flex-direction:row;gap:16px}.wrapper__flowsMenu{overflow:hidden;min-width:230px;max-width:400px;flex:1 1 33%;border:1px solid #d2cecd;border-radius:8px}.wrapper__flowDetails{overflow:hidden;flex:1 1 66%;border:1px solid #d2cecd;border-radius:8px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "pipe", type: i2.TitleCasePipe, name: "titlecase" }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i3$1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "component", type: i4.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "ngmodule", type: GioIconsModule }, { kind: "component", type: i5$1.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: GioPolicyStudioFlowsMenuComponent, selector: "gio-ps-flows-menu", inputs: ["readOnly", "loading", "apiType", "flowExecution", "flowsGroups", "selectedFlowId", "entrypoints", "entrypointsInfo"], outputs: ["selectedFlowIdChange", "flowsGroupsChange", "flowExecutionChange", "deleteFlow"] }, { kind: "component", type: GioPolicyStudioDetailsComponent, selector: "gio-ps-flow-details", inputs: ["readOnly", "loading", "apiType", "flow", "entrypointsInfo", "endpointsInfo", "policies", "genericPolicies", "trialUrl"], outputs: ["flowChange", "deleteFlow"] }] }); }
2201
2214
  }
2202
2215
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImport: i0, type: GioPolicyStudioComponent, decorators: [{
2203
2216
  type: Component,
@@ -2238,6 +2251,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.4", ngImpor
2238
2251
  type: Input
2239
2252
  }], policyDocumentationFetcher: [{
2240
2253
  type: Input
2254
+ }], selectedFlowIndexes: [{
2255
+ type: Input
2256
+ }], selectedFlowChanged: [{
2257
+ type: Output
2241
2258
  }], save: [{
2242
2259
  type: Output
2243
2260
  }] } });