NIFI-12401: Allow combo editor to reference parameters (#8068)

* NIFI-12401:
- Allow combo editor to reference parameters.

* NIFI-12401:
- Addressing review feedback.
- Handling corner cases where there is no parameter context and where there are no parameters in a bound parameter context.

* NIFI-12401:
- Fixing formatting issues.

This closes #8068
This commit is contained in:
Matt Gilman 2023-11-29 12:26:25 -05:00 committed by GitHub
parent 9d50c6dd53
commit ebfb5bc12e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
11 changed files with 498 additions and 104 deletions

View File

@ -44,10 +44,7 @@ module.exports = function (config) {
coverageReporter: { coverageReporter: {
dir: require('path').join(__dirname, './coverage/nifi'), dir: require('path').join(__dirname, './coverage/nifi'),
subdir: '.', subdir: '.',
reporters: [ reporters: [{ type: 'html' }, { type: 'text-summary' }]
{ type: 'html' },
{ type: 'text-summary' }
]
}, },
reporters: ['progress', 'kjhtml'], reporters: ['progress', 'kjhtml'],
browsers: ['ChromeHeadless'], browsers: ['ChromeHeadless'],

View File

@ -43,11 +43,11 @@
"@angular-devkit/build-angular": "^16.2.0", "@angular-devkit/build-angular": "^16.2.0",
"@angular/cli": "~16.2.0", "@angular/cli": "~16.2.0",
"@angular/compiler-cli": "^16.2.0", "@angular/compiler-cli": "^16.2.0",
"@types/codemirror": "^5.60.13",
"@types/d3": "^7.4.0", "@types/d3": "^7.4.0",
"@types/humanize-duration": "^3.27.1", "@types/humanize-duration": "^3.27.1",
"@types/jasmine": "~4.3.0", "@types/jasmine": "~4.3.0",
"@types/webfontloader": "^1.6.35", "@types/webfontloader": "^1.6.35",
"@types/codemirror": "^5.60.13",
"autoprefixer": "^10.4.15", "autoprefixer": "^10.4.15",
"eslint-config-prettier": "^9.0.0", "eslint-config-prettier": "^9.0.0",
"eslint-plugin-prettier": "^5.0.0", "eslint-plugin-prettier": "^5.0.0",

View File

@ -190,7 +190,7 @@
</a> </a>
<ng-template #resultLink> <ng-template #resultLink>
<a [routerLink]="['/process-groups', result.parentGroup.id, path, result.id]"> <a [routerLink]="['/process-groups', result.parentGroup.id, path, result.id]">
{{ result.name }} {{ result.name ? result.name : result.id }}
</a> </a>
</ng-template> </ng-template>
</li> </li>

View File

@ -171,6 +171,8 @@ export interface ParameterContextReference {
export interface AffectedComponentEntity { export interface AffectedComponentEntity {
permissions: Permissions; permissions: Permissions;
id: string;
revision: Revision;
bulletins: BulletinEntity[]; bulletins: BulletinEntity[];
component: AffectedComponent; component: AffectedComponent;
processGroup: ProcessGroupName; processGroup: ProcessGroupName;
@ -289,8 +291,8 @@ export interface PropertyDescriptor {
name: string; name: string;
displayName: string; displayName: string;
description: string; description: string;
defaultValue: string; defaultValue?: string;
allowableValues: AllowableValueEntity[]; allowableValues?: AllowableValueEntity[];
required: boolean; required: boolean;
sensitive: boolean; sensitive: boolean;
dynamic: boolean; dynamic: boolean;

View File

@ -24,7 +24,8 @@
formControlName="value" formControlName="value"
[placeholder]="getComboPlaceholder()" [placeholder]="getComboPlaceholder()"
[panelClass]="'combo-panel'" [panelClass]="'combo-panel'"
(mousedown)="preventDrag($event)"> (mousedown)="preventDrag($event)"
(selectionChange)="allowableValueChanged($event.value)">
<ng-container *ngFor="let allowableValue of allowableValues"> <ng-container *ngFor="let allowableValue of allowableValues">
<ng-container *ngIf="allowableValue.description; else noDescription"> <ng-container *ngIf="allowableValue.description; else noDescription">
<mat-option <mat-option
@ -49,6 +50,41 @@
</ng-container> </ng-container>
</mat-select> </mat-select>
</div> </div>
<div *ngIf="showParameterAllowableValues">
<div *ngIf="!parametersLoaded; else showParameters">
<ngx-skeleton-loader count="1"></ngx-skeleton-loader>
</div>
<ng-template #showParameters>
<mat-select
class="combo"
formControlName="parameterReference"
[panelClass]="'combo-panel'"
(mousedown)="preventDrag($event)">
<ng-container *ngFor="let parameterAllowableValue of parameterAllowableValues">
<ng-container *ngIf="parameterAllowableValue.description; else noDescription">
<mat-option
[value]="parameterAllowableValue.id"
(mousedown)="preventDrag($event)"
nifiTooltip
[tooltipComponentType]="TextTip"
[tooltipInputData]="getAllowableValueOptionTipData(parameterAllowableValue)"
[delayClose]="false">
<span class="option-text" [class.unset]="parameterAllowableValue.value == null">{{
parameterAllowableValue.displayName
}}</span>
</mat-option>
</ng-container>
<ng-template #noDescription>
<mat-option [value]="parameterAllowableValue.id" (mousedown)="preventDrag($event)">
<span class="option-text" [class.unset]="parameterAllowableValue.value == null">{{
parameterAllowableValue.displayName
}}</span>
</mat-option>
</ng-template>
</ng-container>
</mat-select>
</ng-template>
</div>
<div class="flex justify-end items-center gap-x-2"> <div class="flex justify-end items-center gap-x-2">
<button <button
color="accent" color="accent"

View File

@ -18,19 +18,67 @@
import { ComponentFixture, TestBed } from '@angular/core/testing'; import { ComponentFixture, TestBed } from '@angular/core/testing';
import { ComboEditor } from './combo-editor.component'; import { ComboEditor } from './combo-editor.component';
import { PropertyItem } from '../../property-table.component';
import { Parameter } from '../../../../../state/shared';
import { of } from 'rxjs';
describe('ComboEditor', () => { describe('ComboEditor', () => {
let component: ComboEditor; let component: ComboEditor;
let fixture: ComponentFixture<ComboEditor>; let fixture: ComponentFixture<ComboEditor>;
let item: PropertyItem | null = null;
let parameters: Parameter[] = [
{
name: 'one',
description: 'Description for one.',
sensitive: false,
value: 'value',
provided: false,
referencingComponents: [],
parameterContext: {
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
permissions: {
canRead: true,
canWrite: true
},
component: {
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
name: 'params 1'
}
},
inherited: false
},
{
name: 'two',
description: 'Description for two.',
sensitive: false,
value: 'value',
provided: false,
referencingComponents: [],
parameterContext: {
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
permissions: {
canRead: true,
canWrite: true
},
component: {
id: '95d4f3d2-018b-1000-b7c7-b830c49a8026',
name: 'params 1'
}
},
inherited: false
}
];
beforeEach(() => { beforeEach(() => {
TestBed.configureTestingModule({ TestBed.configureTestingModule({
imports: [ComboEditor] imports: [ComboEditor]
}); });
fixture = TestBed.createComponent(ComboEditor); fixture = TestBed.createComponent(ComboEditor);
component = fixture.componentInstance; component = fixture.componentInstance;
component.supportsParameters = false;
component.item = { // re-establish the item before each test execution
item = {
property: 'Destination', property: 'Destination',
value: 'flowfile-attribute', value: 'flowfile-attribute',
descriptor: { descriptor: {
@ -69,10 +117,114 @@ describe('ComboEditor', () => {
dirty: false, dirty: false,
type: 'required' type: 'required'
}; };
fixture.detectChanges();
}); });
it('should create', () => { it('should create', () => {
if (item) {
component.item = item;
fixture.detectChanges();
expect(component).toBeTruthy(); expect(component).toBeTruthy();
}
});
it('verify combo value', () => {
if (item) {
component.item = item;
fixture.detectChanges();
const formValue = component.comboEditorForm.get('value')?.value;
expect(component.itemLookup.get(formValue)?.value).toEqual(item.value);
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(item.value);
}
});
it('verify combo not required with null value and default', () => {
if (item) {
item.value = null;
item.descriptor.required = false;
component.item = item;
fixture.detectChanges();
const formValue = component.comboEditorForm.get('value')?.value;
expect(component.itemLookup.get(formValue)?.value).toEqual(item.descriptor.defaultValue);
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(item.descriptor.defaultValue);
}
});
it('verify combo not required with null value and no default', () => {
if (item) {
item.value = null;
item.descriptor.required = false;
item.descriptor.defaultValue = undefined;
component.item = item;
fixture.detectChanges();
const formValue = component.comboEditorForm.get('value')?.value;
expect(component.itemLookup.get(formValue)?.value).toEqual(item.value);
expect(component.comboEditorForm.get('parameterReference')).toBeNull();
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(item.value);
}
});
it('verify combo with parameter reference', () => {
if (item) {
item.value = '#{one}';
component.item = item;
component.getParameters = (sensitive: boolean) => {
return of(parameters);
};
fixture.detectChanges();
fixture.whenStable().then(() => {
const formValue = component.comboEditorForm.get('value')?.value;
expect(component.itemLookup.get(formValue)?.value).toEqual(item?.value);
expect(component.comboEditorForm.get('parameterReference')).toBeDefined();
const parameterReferenceValue = component.comboEditorForm.get('parameterReference')?.value;
expect(component.itemLookup.get(parameterReferenceValue)?.value).toEqual(item?.value);
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(item?.value);
});
}
});
it('verify combo with missing parameter reference', () => {
if (item) {
item.value = '#{three}';
component.item = item;
component.getParameters = (sensitive: boolean) => {
return of(parameters);
};
fixture.detectChanges();
fixture.whenStable().then(() => {
const formValue = component.comboEditorForm.get('value')?.value;
expect(component.itemLookup.get(formValue)?.value).toEqual('#{' + parameters[0].value + '}');
expect(component.comboEditorForm.get('parameterReference')).toBeDefined();
const parameterReferenceValue = component.comboEditorForm.get('parameterReference')?.value;
expect(component.itemLookup.get(parameterReferenceValue)?.value).toEqual(item?.value);
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith('#{' + parameters[0].value + '}');
});
}
}); });
}); });

View File

@ -25,12 +25,14 @@ import { MatButtonModule } from '@angular/material/button';
import { MatCheckboxModule } from '@angular/material/checkbox'; import { MatCheckboxModule } from '@angular/material/checkbox';
import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common'; import { NgForOf, NgIf, NgTemplateOutlet } from '@angular/common';
import { NifiTooltipDirective } from '../../../tooltips/nifi-tooltip.directive'; import { NifiTooltipDirective } from '../../../tooltips/nifi-tooltip.directive';
import { PropertyDescriptor, AllowableValue, TextTipInput } from '../../../../../state/shared'; import { AllowableValue, Parameter, PropertyDescriptor, TextTipInput } from '../../../../../state/shared';
import { MatOptionModule } from '@angular/material/core'; import { MatOptionModule } from '@angular/material/core';
import { MatSelectModule } from '@angular/material/select'; import { MatSelectModule } from '@angular/material/select';
import { MatTooltipModule } from '@angular/material/tooltip'; import { MatTooltipModule } from '@angular/material/tooltip';
import { TextTip } from '../../../tooltips/text-tip/text-tip.component'; import { TextTip } from '../../../tooltips/text-tip/text-tip.component';
import { A11yModule } from '@angular/cdk/a11y'; import { A11yModule } from '@angular/cdk/a11y';
import { Observable, take } from 'rxjs';
import { NgxSkeletonLoaderModule } from 'ngx-skeleton-loader';
export interface AllowableValueItem extends AllowableValue { export interface AllowableValueItem extends AllowableValue {
id: number; id: number;
@ -55,15 +57,68 @@ export interface AllowableValueItem extends AllowableValue {
NgForOf, NgForOf,
MatTooltipModule, MatTooltipModule,
NgIf, NgIf,
A11yModule A11yModule,
NgxSkeletonLoaderModule
], ],
styleUrls: ['./combo-editor.component.scss'] styleUrls: ['./combo-editor.component.scss']
}) })
export class ComboEditor { export class ComboEditor {
@Input() set item(item: PropertyItem) { @Input() set item(item: PropertyItem) {
this.itemLookup.clear(); if (item.value != null) {
this.configuredValue = item.value;
} else if (item.descriptor.defaultValue != null) {
this.configuredValue = item.descriptor.defaultValue;
}
this.descriptor = item.descriptor; this.descriptor = item.descriptor;
this.sensitive = item.descriptor.sensitive;
this.itemSet = true;
this.initialAllowableValues();
}
@Input() set getParameters(getParameters: (sensitive: boolean) => Observable<Parameter[]>) {
this._getParameters = getParameters;
this.supportsParameters = getParameters != null;
this.initialAllowableValues();
}
@Output() ok: EventEmitter<any> = new EventEmitter<any>();
@Output() cancel: EventEmitter<void> = new EventEmitter<void>();
protected readonly TextTip = TextTip;
itemLookup: Map<number, AllowableValueItem> = new Map<number, AllowableValueItem>();
referencesParametersId: number = -1;
configuredParameterId: number = -1;
comboEditorForm: FormGroup;
descriptor!: PropertyDescriptor;
allowableValues!: AllowableValueItem[];
showParameterAllowableValues: boolean = false;
parameterAllowableValues!: AllowableValueItem[];
sensitive: boolean = false;
supportsParameters: boolean = false;
parametersLoaded: boolean = false;
itemSet: boolean = false;
configuredValue: string | null = null;
_getParameters!: (sensitive: boolean) => Observable<Parameter[]>;
constructor(private formBuilder: FormBuilder) {
this.comboEditorForm = this.formBuilder.group({
value: new FormControl(null, Validators.required)
});
}
initialAllowableValues(): void {
if (this.itemSet) {
this.itemLookup.clear();
this.allowableValues = []; this.allowableValues = [];
this.referencesParametersId = -1;
let i: number = 0; let i: number = 0;
let selectedItem: AllowableValueItem | null = null; let selectedItem: AllowableValueItem | null = null;
@ -77,11 +132,12 @@ export class ComboEditor {
this.itemLookup.set(noValue.id, noValue); this.itemLookup.set(noValue.id, noValue);
this.allowableValues.push(noValue); this.allowableValues.push(noValue);
if (noValue.value == item.value) { if (noValue.value == this.configuredValue) {
selectedItem = noValue; selectedItem = noValue;
} }
} }
if (this.descriptor.allowableValues) {
const allowableValueItems: AllowableValueItem[] = this.descriptor.allowableValues.map( const allowableValueItems: AllowableValueItem[] = this.descriptor.allowableValues.map(
(allowableValueEntity) => { (allowableValueEntity) => {
const allowableValue: AllowableValueItem = { const allowableValue: AllowableValueItem = {
@ -90,7 +146,7 @@ export class ComboEditor {
}; };
this.itemLookup.set(allowableValue.id, allowableValue); this.itemLookup.set(allowableValue.id, allowableValue);
if (allowableValue.value == item.value) { if (allowableValue.value == this.configuredValue) {
selectedItem = allowableValue; selectedItem = allowableValue;
} }
@ -98,6 +154,70 @@ export class ComboEditor {
} }
); );
this.allowableValues.push(...allowableValueItems); this.allowableValues.push(...allowableValueItems);
}
if (this.supportsParameters) {
this.parametersLoaded = false;
// parameters are supported so add the item to support showing
// and hiding the parameter options select
const referencesParameterOption: AllowableValueItem = {
id: i++,
displayName: 'Reference Parameter...',
value: null
};
this.allowableValues.push(referencesParameterOption);
this.itemLookup.set(referencesParameterOption.id, referencesParameterOption);
// record the item of the item to more easily identify this item
this.referencesParametersId = referencesParameterOption.id;
// if the current value references a parameter auto select the
// references parameter item
if (this.referencesParameter(this.configuredValue)) {
selectedItem = referencesParameterOption;
// trigger allowable value changed to show the parameters
this.allowableValueChanged(this.referencesParametersId);
}
this._getParameters(this.sensitive)
.pipe(take(1))
.subscribe((parameters) => {
if (parameters.length > 0) {
// capture the value of i which will be the id of the first
// parameter
this.configuredParameterId = i;
// create allowable values for each parameter
parameters.forEach((parameter) => {
const parameterItem: AllowableValueItem = {
id: i++,
displayName: parameter.name,
value: '#{' + parameter.name + '}',
description: parameter.description
};
this.parameterAllowableValues.push(parameterItem);
this.itemLookup.set(parameterItem.id, parameterItem);
// if the configured parameter is still available,
// capture the id, so we can auto select it
if (parameterItem.value === this.configuredValue) {
this.configuredParameterId = parameterItem.id;
}
});
// if combo still set to reference a parameter, set the default value
if (this.comboEditorForm.get('value')?.value == this.referencesParametersId) {
this.comboEditorForm.get('parameterReference')?.setValue(this.configuredParameterId);
}
}
this.parametersLoaded = true;
});
} else {
this.parameterAllowableValues = [];
}
if (selectedItem) { if (selectedItem) {
// mat-select does not have good support for options with null value so we've // mat-select does not have good support for options with null value so we've
@ -105,29 +225,37 @@ export class ComboEditor {
this.comboEditorForm.get('value')?.setValue(selectedItem.id); this.comboEditorForm.get('value')?.setValue(selectedItem.id);
} }
} }
@Input() supportsParameters: boolean = false; }
@Output() ok: EventEmitter<any> = new EventEmitter<any>(); referencesParameter(value: string | null): boolean {
@Output() cancel: EventEmitter<void> = new EventEmitter<void>(); if (value) {
return value.startsWith('#{') && value.endsWith('}');
}
protected readonly TextTip = TextTip; return false;
itemLookup: Map<number, AllowableValueItem> = new Map<number, AllowableValueItem>();
comboEditorForm: FormGroup;
descriptor!: PropertyDescriptor;
allowableValues!: AllowableValueItem[];
constructor(private formBuilder: FormBuilder) {
this.comboEditorForm = this.formBuilder.group({
value: new FormControl(null, Validators.required)
});
} }
preventDrag(event: MouseEvent): void { preventDrag(event: MouseEvent): void {
event.stopPropagation(); event.stopPropagation();
} }
allowableValueChanged(value: number): void {
this.showParameterAllowableValues = value === this.referencesParametersId;
if (this.showParameterAllowableValues) {
if (this.configuredParameterId === -1) {
this.comboEditorForm.addControl('parameterReference', new FormControl(null, Validators.required));
} else {
this.comboEditorForm.addControl(
'parameterReference',
new FormControl(this.configuredParameterId, Validators.required)
);
}
} else {
this.comboEditorForm.removeControl('parameterReference');
}
}
getComboPlaceholder(): string { getComboPlaceholder(): string {
const valueControl: AbstractControl | null = this.comboEditorForm.get('value'); const valueControl: AbstractControl | null = this.comboEditorForm.get('value');
if (valueControl) { if (valueControl) {
@ -150,10 +278,24 @@ export class ComboEditor {
if (valueControl) { if (valueControl) {
const selectedItem: AllowableValueItem | undefined = this.itemLookup.get(valueControl.value); const selectedItem: AllowableValueItem | undefined = this.itemLookup.get(valueControl.value);
if (selectedItem) { if (selectedItem) {
// if the value currently references a parameter emit the parameter, get the parameter reference control and emit that value
if (selectedItem.id == this.referencesParametersId) {
const parameterReferenceControl: AbstractControl | null =
this.comboEditorForm.get('parameterReference');
if (parameterReferenceControl) {
const selectedParameterItem: AllowableValueItem | undefined = this.itemLookup.get(
parameterReferenceControl.value
);
if (selectedParameterItem) {
this.ok.next(selectedParameterItem.value);
}
}
} else {
this.ok.next(selectedItem.value); this.ok.next(selectedItem.value);
} }
} }
} }
}
cancelClicked(): void { cancelClicked(): void {
this.cancel.next(); this.cancel.next();

View File

@ -19,6 +19,7 @@ import { ComponentFixture, TestBed } from '@angular/core/testing';
import { NfEditor } from './nf-editor.component'; import { NfEditor } from './nf-editor.component';
import { HttpClientTestingModule } from '@angular/common/http/testing'; import { HttpClientTestingModule } from '@angular/common/http/testing';
import { PropertyItem } from '../../property-table.component';
describe('NfEditor', () => { describe('NfEditor', () => {
let component: NfEditor; let component: NfEditor;
@ -30,10 +31,84 @@ describe('NfEditor', () => {
}); });
fixture = TestBed.createComponent(NfEditor); fixture = TestBed.createComponent(NfEditor);
component = fixture.componentInstance; component = fixture.componentInstance;
fixture.detectChanges();
}); });
it('should create', () => { it('should create', () => {
fixture.detectChanges();
expect(component).toBeTruthy(); expect(component).toBeTruthy();
}); });
it('verify value set', () => {
const value: string = 'my-group-id';
const item: PropertyItem = {
property: 'group.id',
value,
descriptor: {
name: 'group.id',
displayName: 'Group ID',
description:
"A Group ID is used to identify consumers that are within the same consumer group. Corresponds to Kafka's 'group.id' property.",
required: true,
sensitive: false,
dynamic: false,
supportsEl: true,
expressionLanguageScope: 'Environment variables defined at JVM level and system properties',
dependencies: []
},
id: 3,
triggerEdit: false,
deleted: false,
added: false,
dirty: false,
type: 'required'
};
component.item = item;
fixture.detectChanges();
expect(component.nfEditorForm.get('value')?.value).toEqual(value);
expect(component.nfEditorForm.get('value')?.disabled).toBeFalse();
expect(component.nfEditorForm.get('setEmptyString')?.value).toBeFalse();
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(value);
});
it('verify empty value set', () => {
const value: string = '';
const item: PropertyItem = {
property: 'group.id',
value,
descriptor: {
name: 'group.id',
displayName: 'Group ID',
description:
"A Group ID is used to identify consumers that are within the same consumer group. Corresponds to Kafka's 'group.id' property.",
required: true,
sensitive: false,
dynamic: false,
supportsEl: true,
expressionLanguageScope: 'Environment variables defined at JVM level and system properties',
dependencies: []
},
id: 3,
triggerEdit: false,
deleted: false,
added: false,
dirty: false,
type: 'required'
};
component.item = item;
fixture.detectChanges();
expect(component.nfEditorForm.get('value')?.value).toEqual(value);
expect(component.nfEditorForm.get('value')?.disabled).toBeTruthy();
expect(component.nfEditorForm.get('setEmptyString')?.value).toBeTruthy();
spyOn(component.ok, 'next');
component.okClicked();
expect(component.ok.next).toHaveBeenCalledWith(value);
});
}); });

View File

@ -15,18 +15,7 @@
* limitations under the License. * limitations under the License.
*/ */
import { import { Component, EventEmitter, Input, OnDestroy, Output, Renderer2, ViewContainerRef } from '@angular/core';
AfterViewInit,
Component,
ElementRef,
EventEmitter,
Input,
OnDestroy,
Output,
Renderer2,
ViewChild,
ViewContainerRef
} from '@angular/core';
import { PropertyItem } from '../../property-table.component'; import { PropertyItem } from '../../property-table.component';
import { CdkDrag, CdkDragHandle } from '@angular/cdk/drag-drop'; import { CdkDrag, CdkDragHandle } from '@angular/cdk/drag-drop';
import { AbstractControl, FormBuilder, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms'; import { AbstractControl, FormBuilder, FormControl, FormGroup, ReactiveFormsModule, Validators } from '@angular/forms';

View File

@ -145,6 +145,7 @@
<combo-editor <combo-editor
*ngIf="hasAllowableValues(editorItem); else nfEditor" *ngIf="hasAllowableValues(editorItem); else nfEditor"
[item]="editorItem" [item]="editorItem"
[getParameters]="getParameters"
(ok)="savePropertyValue(editorItem, $event)" (ok)="savePropertyValue(editorItem, $event)"
(cancel)="closeEditor()"></combo-editor> (cancel)="closeEditor()"></combo-editor>
<ng-template #nfEditor> <ng-template #nfEditor>

View File

@ -347,8 +347,8 @@ export class PropertyTable implements AfterViewInit, ControlValueAccessor {
} }
resolvePropertyValue(property: Property): string | null { resolvePropertyValue(property: Property): string | null {
const allowableValues: AllowableValueEntity[] = property.descriptor.allowableValues; const allowableValues: AllowableValueEntity[] | undefined = property.descriptor.allowableValues;
if (this.nifiCommon.isEmpty(allowableValues)) { if (allowableValues == null || this.nifiCommon.isEmpty(allowableValues)) {
return property.value; return property.value;
} else { } else {
const allowableValue: AllowableValueEntity | undefined = allowableValues.find( const allowableValue: AllowableValueEntity | undefined = allowableValues.find(
@ -392,7 +392,7 @@ export class PropertyTable implements AfterViewInit, ControlValueAccessor {
// TODO - add Input() for supportsGoTo? currently only false in summary table // TODO - add Input() for supportsGoTo? currently only false in summary table
const descriptor: PropertyDescriptor = item.descriptor; const descriptor: PropertyDescriptor = item.descriptor;
if (item.value && descriptor.identifiesControllerService) { if (item.value && descriptor.identifiesControllerService && descriptor.allowableValues) {
return descriptor.allowableValues.some( return descriptor.allowableValues.some(
(entity: AllowableValueEntity) => entity.allowableValue.value == item.value (entity: AllowableValueEntity) => entity.allowableValue.value == item.value
); );