visibility rule changes
This commit is contained in:
parent
9b30fb60d7
commit
21fb68128d
|
@ -68,6 +68,7 @@ public class RuleBuilder extends BaseBuilder<Rule, RuleEntity> {
|
|||
List<Rule> models = new ArrayList<>();
|
||||
for (RuleEntity d : data) {
|
||||
Rule m = new Rule();
|
||||
if (fields.hasField(this.asIndexer(Rule._target))) m.setTarget(d.getTarget());
|
||||
if (fields.hasField(this.asIndexer(Rule._dateValue)) && FieldType.isDateType(fieldType)) m.setDateValue(d.getDateValue());
|
||||
if (fields.hasField(this.asIndexer(Rule._textValue)) && FieldType.isTextType(fieldType)) m.setTextValue(d.getTextValue());
|
||||
if (fields.hasField(this.asIndexer(Rule._textListValue)) && FieldType.isTextListType(fieldType)) m.setTextListValue(d.getTextListValue());
|
||||
|
|
|
@ -360,8 +360,7 @@ public class VisibilityServiceImpl implements VisibilityService {
|
|||
}
|
||||
else if (eu.eudat.commons.enums.FieldType.isReferenceType(fieldType)) {
|
||||
return rule.getTextListValue() != null &&
|
||||
new HashSet<>(field.getTextListValue()).containsAll(rule.getTextListValue()) &&
|
||||
new HashSet<>(rule.getTextListValue()).containsAll(field.getTextListValue());
|
||||
new HashSet<>(field.getTextListValue()).containsAll(rule.getTextListValue());
|
||||
}
|
||||
else if (eu.eudat.commons.enums.FieldType.isDateType(fieldType) && field.getDateValue() != null) return field.getDateValue().equals(rule.getDateValue());
|
||||
else if (eu.eudat.commons.enums.FieldType.isExternalIdentifierType(fieldType) && field.getExternalIdentifier() != null) {
|
||||
|
|
|
@ -5,6 +5,7 @@ import { DescriptionTemplateStatus } from "@app/core/common/enum/description-tem
|
|||
import { UserDescriptionTemplateRole } from "@app/core/common/enum/user-description-template-role";
|
||||
import { BaseEntityPersist } from "@common/base/base-entity.model";
|
||||
import { Guid } from "@common/types/guid";
|
||||
import { ReferencePersist } from "../reference/reference";
|
||||
|
||||
|
||||
export interface DescriptionTemplatePersist extends BaseEntityPersist {
|
||||
|
@ -82,7 +83,9 @@ export interface DescriptionTemplateFieldPersist {
|
|||
|
||||
export interface DescriptionTemplateRulePersist {
|
||||
target: string;
|
||||
value: string;
|
||||
textValue: string;
|
||||
textListValue: string[];
|
||||
dateValue: Date;
|
||||
}
|
||||
|
||||
export interface DescriptionTemplateMultiplicityPersist {
|
||||
|
|
|
@ -8,6 +8,7 @@ import { Guid } from "@common/types/guid";
|
|||
import { DescriptionTemplateType } from "../description-template-type/description-template-type";
|
||||
import { ReferenceType } from "../reference-type/reference-type";
|
||||
import { User } from "../user/user";
|
||||
import { Reference } from "../reference/reference";
|
||||
|
||||
|
||||
export interface DescriptionTemplate extends BaseEntity {
|
||||
|
@ -78,7 +79,9 @@ export interface DescriptionTemplateField {
|
|||
|
||||
export interface DescriptionTemplateRule {
|
||||
target: string;
|
||||
value: string;
|
||||
textValue: string;
|
||||
textListValue: string[];
|
||||
dateValue: Date;
|
||||
}
|
||||
|
||||
export interface DescriptionTemplateMultiplicity {
|
||||
|
|
|
@ -67,7 +67,7 @@ export class DescriptionTemplatePreviewDialogComponent extends BaseComponent imp
|
|||
|
||||
buildForm() {
|
||||
this.formGroup = this.editorModel.buildForm(null, true);
|
||||
this.visibilityRulesService.buildVisibilityRules(this.visibilityRulesService.getVisibilityRulesFromDescriptionTempalte(this.descriptionTemplate), this.formGroup);
|
||||
this.visibilityRulesService.setContext(this.descriptionTemplate.definition, this.formGroup.get('properties'));
|
||||
}
|
||||
|
||||
select(): void {
|
||||
|
|
|
@ -893,7 +893,9 @@ export class DescriptionTemplateFieldEditorModel implements DescriptionTemplateF
|
|||
|
||||
export class DescriptionTemplateRuleEditorModel implements DescriptionTemplateRulePersist {
|
||||
target: string;
|
||||
value: string;
|
||||
textValue: string;
|
||||
textListValue: string[];
|
||||
dateValue: Date;
|
||||
|
||||
protected formBuilder: UntypedFormBuilder = new UntypedFormBuilder();
|
||||
|
||||
|
@ -901,10 +903,12 @@ export class DescriptionTemplateRuleEditorModel implements DescriptionTemplateRu
|
|||
public validationErrorModel: ValidationErrorModel = new ValidationErrorModel()
|
||||
) { }
|
||||
|
||||
fromModel(item: DescriptionTemplateRule): DescriptionTemplateRuleEditorModel {
|
||||
fromModel(item: DescriptionTemplateRule | DescriptionTemplateRuleEditorModel): DescriptionTemplateRuleEditorModel {
|
||||
if (item) {
|
||||
this.target = item.target;
|
||||
this.value = item.value;
|
||||
this.textValue = item.textValue;
|
||||
this.textListValue = item.textListValue;
|
||||
this.dateValue = item.dateValue;
|
||||
}
|
||||
return this;
|
||||
}
|
||||
|
@ -924,7 +928,9 @@ export class DescriptionTemplateRuleEditorModel implements DescriptionTemplateRu
|
|||
|
||||
return this.formBuilder.group({
|
||||
target: [{ value: this.target, disabled: disabled }, context.getValidation('target').validators],
|
||||
value: [{ value: this.value, disabled: disabled }, context.getValidation('value').validators]
|
||||
textValue: [{ value: this.textValue, disabled: disabled }, context.getValidation('textValue').validators],
|
||||
textListValue: [{ value: this.textListValue, disabled: disabled }, context.getValidation('textListValue').validators],
|
||||
dateValue: [{ value: this.dateValue, disabled: disabled }, context.getValidation('dateValue').validators]
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -937,7 +943,9 @@ export class DescriptionTemplateRuleEditorModel implements DescriptionTemplateRu
|
|||
const baseContext: ValidationContext = new ValidationContext();
|
||||
const baseValidationArray: Validation[] = new Array<Validation>();
|
||||
baseValidationArray.push({ key: 'target', validators: [BackendErrorValidator(validationErrorModel, `${rootPath}target`)] });
|
||||
baseValidationArray.push({ key: 'value', validators: [Validators.required, BackendErrorValidator(validationErrorModel, `${rootPath}value`)] });
|
||||
baseValidationArray.push({ key: 'textValue', validators: [Validators.required, BackendErrorValidator(validationErrorModel, `${rootPath}textValue`)] });
|
||||
baseValidationArray.push({ key: 'textListValue', validators: [Validators.required, BackendErrorValidator(validationErrorModel, `${rootPath}textListValue`)] });
|
||||
baseValidationArray.push({ key: 'dateValue', validators: [Validators.required, BackendErrorValidator(validationErrorModel, `${rootPath}dateValue`)] });
|
||||
|
||||
baseContext.validation = baseValidationArray;
|
||||
return baseContext;
|
||||
|
|
|
@ -3,6 +3,7 @@ import { ActivatedRouteSnapshot, RouterStateSnapshot } from '@angular/router';
|
|||
import { DescriptionTemplateType } from '@app/core/model/description-template-type/description-template-type';
|
||||
import { DescriptionTemplate, DescriptionTemplateBaseFieldData, DescriptionTemplateDefinition, DescriptionTemplateExternalDatasetData, DescriptionTemplateField, DescriptionTemplateFieldSet, DescriptionTemplateLabelAndMultiplicityData, DescriptionTemplateMultiplicity, DescriptionTemplatePage, DescriptionTemplateReferenceTypeData, DescriptionTemplateRule, DescriptionTemplateSection, DescriptionTemplateSelectData, DescriptionTemplateSelectOption, DescriptionTemplateUploadData, DescriptionTemplateUploadOption, UserDescriptionTemplate } from '@app/core/model/description-template/description-template';
|
||||
import { ReferenceType } from '@app/core/model/reference-type/reference-type';
|
||||
import { Reference } from '@app/core/model/reference/reference';
|
||||
import { User } from '@app/core/model/user/user';
|
||||
import { DescriptionTemplateService } from '@app/core/services/description-template/description-template.service';
|
||||
import { BreadcrumbService } from '@app/ui/misc/breadcrumb/breadcrumb.service';
|
||||
|
@ -66,7 +67,9 @@ export class DescriptionTemplateEditorResolver extends BaseEditorResolver {
|
|||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.validations)].join('.'),
|
||||
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.target)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.value)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.textValue)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.textListValue)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.dateValue)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateBaseFieldData>(x => x.label)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateBaseFieldData>(x => x.fieldType)].join('.'),
|
||||
[nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateSelectData>(x => x.multipleSelect)].join('.'),
|
||||
|
|
|
@ -568,7 +568,7 @@ export class DescriptionEditorComponent extends BaseEditor<DescriptionEditorMode
|
|||
permissionPerSection => {
|
||||
const canedit = permissionPerSection && permissionPerSection[this.item.dmpDescriptionTemplate.sectionId.toString()] && permissionPerSection[this.item.dmpDescriptionTemplate.sectionId.toString()].some(x => x === AppPermission.EditDescription);
|
||||
this.formGroup = this.editorModel.buildForm(null, this.isDeleted || !canedit);
|
||||
//this.visibilityRulesService.buildVisibilityRules(this.visibilityRulesService.getVisibilityRulesFromDescriptionTempalte(this.item.descriptionTemplate), this.formGroup);
|
||||
this.visibilityRulesService.setContext(this.item.descriptionTemplate.definition, this.formGroup.get('properties'));
|
||||
|
||||
// this.selectedSystemFields = this.selectedSystemFieldDisabled();
|
||||
this.descriptionEditorService.setValidationErrorModel(this.editorModel.validationErrorModel);
|
||||
|
|
|
@ -108,7 +108,9 @@ export class DescriptionEditorResolver extends BaseEditorResolver {
|
|||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.defaultValue)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.validations)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.target)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.value)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.textValue)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.textListValue)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.visibilityRules), nameof<DescriptionTemplateRule>(x => x.dateValue)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateBaseFieldData>(x => x.label)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateBaseFieldData>(x => x.fieldType)].join('.'),
|
||||
(prefix ? prefix + '.' : '') + [nameof<DescriptionTemplate>(x => x.definition), nameof<DescriptionTemplateDefinition>(x => x.pages), nameof<DescriptionTemplatePage>(x => x.sections), nameof<DescriptionTemplateSection>(x => x.fieldSets), nameof<DescriptionTemplateFieldSet>(x => x.fields), nameof<DescriptionTemplateField>(x => x.data), nameof<DescriptionTemplateSelectData>(x => x.options), nameof<DescriptionTemplateSelectOption>(x => x.label)].join('.'),
|
||||
|
|
|
@ -13,10 +13,10 @@
|
|||
</div>
|
||||
<div *ngIf="!fieldSet?.multiplicity?.tableView" class="col-12">
|
||||
<div class="row" *ngFor="let fieldSetItemPropertiesControl of propertiesFormGroup?.get('items')?.controls">
|
||||
<div class="col">
|
||||
<div class="col" *ngIf="visibilityRulesService.isVisibleMap[fieldSet.id + '_' + fieldSetItemPropertiesControl.get('ordinal').value]">
|
||||
<div class="row">
|
||||
<div *ngFor="let field of fieldSet.fields; let i = index;" class="col-12 compositeField">
|
||||
<ng-container>
|
||||
<ng-container *ngIf="visibilityRulesService.isVisibleMap[field.id + '_' + fieldSetItemPropertiesControl.get('ordinal').value]">
|
||||
<div class="row">
|
||||
<h5 *ngIf="placeholderTitle" class="col-auto font-weight-bold">{{field.label}}</h5>
|
||||
</div>
|
||||
|
@ -47,17 +47,24 @@
|
|||
<th *ngFor="let field of fieldSet.fields; let i = index;" class="text-wrap">{{field.data?.label}}</th>
|
||||
<th class="actions"></th>
|
||||
</tr>
|
||||
<tr *ngFor="let fieldSetItemPropertiesControl of propertiesFormGroup?.get('items')?.controls; let j = index">
|
||||
<td *ngFor="let field of fieldSet.fields;" class="text-wrap">{{fieldSetItemPropertiesControl.get('fields').get(field.id).get('value').getRawValue()}}</td>
|
||||
<td class="actions">
|
||||
<button mat-icon-button type="button" class="deleteBtn btn-sm" (click)="editTableMultiplicityFieldInDialog(j)" [disabled]="fieldSetItemPropertiesControl.disabled">
|
||||
<mat-icon>edit</mat-icon>
|
||||
</button>
|
||||
<button *ngIf="propertiesFormGroup.get('items').length > 1" mat-icon-button type="button" class="deleteBtn" (click)="deleteMultiplicityField(j);" [disabled]="fieldSetItemPropertiesControl.disabled">
|
||||
<mat-icon>delete</mat-icon>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
<ng-container *ngFor="let fieldSetItemPropertiesControl of propertiesFormGroup?.get('items')?.controls; let j = index">
|
||||
<tr *ngIf="visibilityRulesService.isVisibleMap[fieldSet.id + '_' + fieldSetItemPropertiesControl.get('ordinal').value]">
|
||||
<td *ngFor="let field of fieldSet.fields;" class="text-wrap">
|
||||
<ng-container *ngIf="visibilityRulesService.isVisibleMap[field.id + '_' + fieldSetItemPropertiesControl.get('ordinal').value]">
|
||||
{{fieldSetItemPropertiesControl.get('fields').get(field.id).get('value').getRawValue()}}
|
||||
</ng-container>
|
||||
</td>
|
||||
<td class="actions">
|
||||
<button mat-icon-button type="button" class="deleteBtn btn-sm" (click)="editTableMultiplicityFieldInDialog(j)" [disabled]="fieldSetItemPropertiesControl.disabled">
|
||||
<mat-icon>edit</mat-icon>
|
||||
</button>
|
||||
<button *ngIf="propertiesFormGroup.get('items').length > 1" mat-icon-button type="button" class="deleteBtn" (click)="deleteMultiplicityField(j);" [disabled]="fieldSetItemPropertiesControl.disabled">
|
||||
<mat-icon>delete</mat-icon>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
</ng-container>
|
||||
|
||||
<tr *ngIf="(fieldSet.multiplicity.max - 1) > propertiesFormGroup?.get('items')?.controls.length">
|
||||
<td [colSpan]="fieldSet.fields.length + 1" class="text-center">
|
||||
<span class="d-inline-flex align-items-center" [ngClass]="propertiesFormGroup.disabled ? '' : 'pointer'" (click)="addMultiplicityField()">
|
||||
|
|
|
@ -17,7 +17,7 @@ import { VisibilityRulesService } from '@app/ui/description/editor/description-f
|
|||
import { BaseComponent } from '@common/base/base.component';
|
||||
import { TranslateService } from '@ngx-translate/core';
|
||||
import { Observable } from 'rxjs';
|
||||
import { map, takeUntil } from 'rxjs/operators';
|
||||
import { distinctUntilChanged, map, takeUntil } from 'rxjs/operators';
|
||||
|
||||
@Component({
|
||||
selector: 'app-description-form-field',
|
||||
|
@ -204,21 +204,14 @@ export class DescriptionFormFieldComponent extends BaseComponent implements OnIn
|
|||
// this.form = this.visibilityRulesService.getFormGroup(this.field.id);
|
||||
|
||||
//TODO: refactor
|
||||
// this.propertiesFormGroup.get(this.field.id).get('value').valueChanges
|
||||
// .pipe(
|
||||
// takeUntil(this._destroyed),
|
||||
// distinctUntilChanged()
|
||||
// )
|
||||
// .subscribe(item => {
|
||||
// // if (this.field?.data?.fieldType === DescriptionTemplateFieldType.ComboBox && this.form.get('data').value.type === DatasetProfileComboBoxType.Select && this.form.get('data').value.multipleSelect) {
|
||||
// // item.forEach(element => {
|
||||
// // this.visibilityRulesService.updateValueAndVisibility(this.field?.id, element);
|
||||
// // });
|
||||
|
||||
// // } else {
|
||||
// this.visibilityRulesService.updateValueAndVisibility(this.field?.id, item);
|
||||
// // }
|
||||
// });
|
||||
this.propertiesFormGroup.get(this.field.id).valueChanges
|
||||
.pipe(
|
||||
takeUntil(this._destroyed),
|
||||
distinctUntilChanged()
|
||||
)
|
||||
.subscribe(item => {
|
||||
this.visibilityRulesService.updateVisibilityForSource(this.field?.id);
|
||||
});
|
||||
}
|
||||
|
||||
// _optionRemove(event) {
|
||||
|
|
|
@ -11,7 +11,7 @@
|
|||
<h6 class='panel-desc' *ngIf="section.description" [innerHTML]="section.description"></h6>
|
||||
</mat-panel-description>
|
||||
<div *ngFor="let fieldSet of section.fieldSets; let i = index;" class="col-12" [id]="fieldSet.id" (click)="onAskedToScroll($event, fieldSet.id)">
|
||||
<div class="row">
|
||||
<div class="row" *ngIf="visibilityRulesService.isVisibleMap[fieldSet.id] ?? true">
|
||||
<div class="col-12">
|
||||
<div class="row">
|
||||
<app-description-form-field-set class="align-self-center col"
|
||||
|
@ -29,7 +29,7 @@
|
|||
|
||||
<div *ngIf="section?.sections?.length > 0" class="col-12">
|
||||
<ng-container *ngFor="let innerSection of section.sections; let j = index;">
|
||||
<div class="row">
|
||||
<div class="row" *ngIf="visibilityRulesService.isVisibleMap[innerSection.id]">
|
||||
<app-description-form-section class="col-12" [section]="innerSection" [path]="path+'.'+(j+1)" [pathName]="pathName+'.sections.'+j" (askedToScroll)="onAskedToScroll($event)" [propertiesFormGroup]="propertiesFormGroup" [descriptionId]="descriptionId" [visibilityRulesService]="visibilityRulesService" [linkToScroll]="subsectionLinkToScroll"></app-description-form-section>
|
||||
</div>
|
||||
</ng-container>
|
||||
|
|
|
@ -14,7 +14,7 @@
|
|||
</mat-panel-title>
|
||||
</mat-expansion-panel-header>
|
||||
<ng-container *ngFor="let section of page.sections; let i = index;">
|
||||
<div class="row">
|
||||
<div class="row" *ngIf="visibilityRulesService.isVisibleMap[section.id]">
|
||||
<app-description-form-section class="col-12" [section]="section" [path]="(z+1)+'.'+(i+1)" [pathName]="'pages.'+z+'.sections.'+i" [propertiesFormGroup]="propertiesFormGroup" [descriptionId]="descriptionId" [visibilityRulesService]="visibilityRulesService" (askedToScroll)="onAskedToScroll(expansionPanel, $event)" [linkToScroll]="linkToScroll" [validationErrorModel]="validationErrorModel"></app-description-form-section>
|
||||
</div>
|
||||
</ng-container>
|
||||
|
|
|
@ -4,10 +4,10 @@ import { MatExpansionPanel } from '@angular/material/expansion';
|
|||
import { DescriptionTemplate, DescriptionTemplateSection } from '@app/core/model/description-template/description-template';
|
||||
import { BaseComponent } from '@common/base/base.component';
|
||||
import { LinkToScroll } from '../table-of-contents/table-of-contents.component';
|
||||
import { Rule } from './visibility-rules/models/rule';
|
||||
import { VisibilityRulesService } from './visibility-rules/visibility-rules.service';
|
||||
import { ValidationErrorModel } from '@common/forms/validation/error-model/validation-error-model';
|
||||
import { Guid } from '@common/types/guid';
|
||||
import { RuleWithTarget } from './visibility-rules/models/rule';
|
||||
|
||||
@Component({
|
||||
selector: 'app-description-form',
|
||||
|
@ -24,7 +24,6 @@ export class DescriptionFormComponent extends BaseComponent implements OnInit, A
|
|||
|
||||
// @ViewChild('stepper', { static: false }) stepper: MatStepper;
|
||||
@Input() path: string;
|
||||
@Input() visibilityRules: Rule[] = [];
|
||||
@Input() datasetDescription: String;
|
||||
@Input() linkToScroll: LinkToScroll;
|
||||
@Output() formChanged: EventEmitter<any> = new EventEmitter();
|
||||
|
@ -34,7 +33,6 @@ export class DescriptionFormComponent extends BaseComponent implements OnInit, A
|
|||
|
||||
|
||||
@Input() TOCENTRY_ID_PREFIX = "";
|
||||
@Output() visibilityRulesInstance = new EventEmitter<VisibilityRulesService>();
|
||||
@Input() validationErrorModel: ValidationErrorModel;
|
||||
|
||||
// public hiddenEntriesIds: string[] = [];
|
||||
|
|
|
@ -1,5 +1,19 @@
|
|||
export class Rule {
|
||||
sourceField: string;
|
||||
targetField: string;
|
||||
requiredValue: any;
|
||||
import { DescriptionTemplateField, DescriptionTemplateRule } from "@app/core/model/description-template/description-template";
|
||||
|
||||
export class RuleWithTarget {
|
||||
source: string;
|
||||
target: string;
|
||||
textValue: string;
|
||||
textListValue: string[];
|
||||
dateValue: Date;
|
||||
field: DescriptionTemplateField;
|
||||
|
||||
public constructor(source: string, rule: DescriptionTemplateRule , fieldEntity: DescriptionTemplateField) {
|
||||
this.target = rule.target;
|
||||
this.source = source;
|
||||
this.field = fieldEntity;
|
||||
this.textValue = rule.textValue;
|
||||
this.textListValue = rule.textListValue;
|
||||
this.dateValue = rule.dateValue;
|
||||
}
|
||||
}
|
|
@ -1,4 +0,0 @@
|
|||
export class VisibilityRuleSource {
|
||||
public sourceControlId: string;
|
||||
public sourceControlValue: string;
|
||||
}
|
|
@ -1,6 +0,0 @@
|
|||
import { VisibilityRuleSource } from "./visibility-rule-source";
|
||||
|
||||
export class VisibilityRule {
|
||||
public targetControlId: string;
|
||||
public sourceVisibilityRules: Array<VisibilityRuleSource>;
|
||||
}
|
|
@ -1,47 +0,0 @@
|
|||
import { Rule } from "./rule";
|
||||
import { VisibilityRule } from "./visibility-rule";
|
||||
|
||||
export class VisibilityRulesContext {
|
||||
|
||||
public rules: Array<VisibilityRule> = new Array();
|
||||
|
||||
constructor() { }
|
||||
|
||||
public getRulesFromKey(id: string): VisibilityRule {
|
||||
|
||||
for (let i = 0; i < this.rules.length; i++) {
|
||||
if (id === this.rules[i].targetControlId) { return this.rules[i]; }
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
public buildVisibilityRuleContext(items: Array<Rule>) {
|
||||
items.forEach(item => {
|
||||
this.addToVisibilityRulesContext(item);
|
||||
});
|
||||
}
|
||||
|
||||
public addToVisibilityRulesContext(item: Rule): void {
|
||||
for (let i = 0; i < this.rules.length; i++) {
|
||||
if (this.rules[i].targetControlId === item.targetField) {
|
||||
|
||||
const newRule = { sourceControlId: item.sourceField, sourceControlValue: item.requiredValue };
|
||||
const ruleExists = this.rules[i].sourceVisibilityRules.find(x => {
|
||||
return Object.keys(x).reduce((exact, key) => {
|
||||
if (!exact) return false;
|
||||
return x[key] === newRule[key];
|
||||
}, true);
|
||||
})
|
||||
|
||||
if (!ruleExists) {
|
||||
this.rules[i].sourceVisibilityRules.push(newRule);
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
const newVisibilityRuleArray = [({ sourceControlId: item.sourceField, sourceControlValue: item.requiredValue })];
|
||||
this.rules.push({ targetControlId: item.targetField, sourceVisibilityRules: newVisibilityRuleArray });
|
||||
return;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,456 @@
|
|||
// import { ApplicationRef, Injectable, NgZone } from '@angular/core';
|
||||
// import { AbstractControl, UntypedFormArray, UntypedFormGroup } from '@angular/forms';
|
||||
// import { isNumeric } from '@app/utilities/enhancers/utils';
|
||||
// import { Observable, Subject } from 'rxjs';
|
||||
// import { VisibilityRule } from './models/visibility-rule';
|
||||
// import { VisibilityRuleSource } from './models/visibility-rule-source';
|
||||
// import { VisibilityRulesContext } from './models/visibility-rules-context';
|
||||
// import { DescriptionTemplate, DescriptionTemplatePage, DescriptionTemplateSection } from '@app/core/model/description-template/description-template';
|
||||
// import { Rule } from './models/rule';
|
||||
|
||||
// @Injectable()
|
||||
// export class VisibilityRulesService {
|
||||
|
||||
// private readonly VISIBILITY_RULE_LOGIC: 'OR' | 'AND' = 'OR';
|
||||
// private readonly DEFAULTVISIBILITY = false;
|
||||
|
||||
// private visibilityRuleContext: VisibilityRulesContext;
|
||||
// private form: AbstractControl;
|
||||
|
||||
// public isVisibleMap: { [key: string]: boolean } = {};
|
||||
|
||||
// private elementVisibilityMapSubject = new Subject<{ [key: string]: boolean }>();
|
||||
// private elementComputationalMap = new Map<String, Map<String, boolean>>(); /// keep saved the values of each form control validity value
|
||||
// private _changeMade$ = new Subject<void>();
|
||||
|
||||
|
||||
// // get isVisibleMap(): MapWithDefault {
|
||||
// // // console.log('isVisibleMap');
|
||||
// // return this.elementVisibilityMap;
|
||||
|
||||
// // }; /// keep saved the values of each form control validity value
|
||||
|
||||
// constructor(
|
||||
// public applicationReference: ApplicationRef,
|
||||
// public ngZone: NgZone
|
||||
// ) {
|
||||
|
||||
// }
|
||||
|
||||
// getElementVisibilityMapObservable(): Observable<{ [key: string]: boolean }> {
|
||||
// // this.isVisibleMap
|
||||
// // console.log('getElementVisibilityMapObservable: ');
|
||||
// return this.elementVisibilityMapSubject.asObservable();
|
||||
// }
|
||||
|
||||
// public checkElementVisibility(id: string): boolean {
|
||||
// //console.log('checkElementVisibility: ' + id);
|
||||
// return true;
|
||||
// // if (this.visibilityRuleContext.rules.filter(item => item.targetControlId === id).length === 0) { return true; }
|
||||
// // console.log(this.elementVisibilityMap.has(id) ? this.elementVisibilityMap.get(id) : false);
|
||||
// // return this.elementVisibilityMap.has(id) ? this.elementVisibilityMap.get(id) : false;
|
||||
// }
|
||||
|
||||
// public buildVisibilityRules(item: Array<Rule>, form: AbstractControl) {
|
||||
// this.visibilityRuleContext = new VisibilityRulesContext();
|
||||
// this.visibilityRuleContext.buildVisibilityRuleContext(item || []);
|
||||
// this.form = form;
|
||||
// this.resetVisibilityRules();
|
||||
// }
|
||||
|
||||
// public updateValueAndVisibility(id: string, value: any) {
|
||||
// //console.log('updateValueAndVisibility: ' + id + ' value: ' + value);
|
||||
// const visibilityRules = this.visibilityRuleContext.rules.filter(item => item.sourceVisibilityRules.filter(source => source.sourceControlId === id).length > 0);
|
||||
// if (visibilityRules.length > 0) {
|
||||
// visibilityRules.forEach(item => this.evaluateVisibility(item, value, id));
|
||||
// this.elementVisibilityMapSubject.next(this.isVisibleMap);
|
||||
// }
|
||||
// }
|
||||
|
||||
// private evaluateVisibility(visibilityRule: VisibilityRule, value: any, sourceId: string) {// source controlId is the same
|
||||
// //console.log('evaluateVisibility: ' + visibilityRule + ' value: ' + value + ' sourceId: ' + sourceId);
|
||||
|
||||
// const targetId = visibilityRule.targetControlId;
|
||||
// const visibilityMap = this.elementComputationalMap.get(targetId) ? this.elementComputationalMap.get(targetId) : new Map<String, boolean>();
|
||||
|
||||
|
||||
// if (value instanceof Array) {
|
||||
|
||||
// const parsedSourceControlValues = visibilityRule.sourceVisibilityRules.map(e => this.parseValue(e.sourceControlValue));
|
||||
// const parsedValues = value.map(e => this.parseValue(e));
|
||||
|
||||
// const isVisible = parsedValues.map(v => parsedSourceControlValues.includes(v)).reduce((acc, current) => acc || current, false);
|
||||
|
||||
|
||||
// // if(isVisible){
|
||||
// // this._emitChangesIfNeeded(visibilityRule.targetControlId, true);
|
||||
// // this.elementVisibilityMap.set(visibilityRule.targetControlId, true);
|
||||
// // return;
|
||||
// // }
|
||||
// visibilityMap.set(sourceId, isVisible);
|
||||
|
||||
// } else {
|
||||
// const visibilityDependencySource = visibilityRule.sourceVisibilityRules.filter(x => x.sourceControlId === sourceId);
|
||||
|
||||
// const shouldBeVisible = visibilityDependencySource.reduce((isVisible, x) => {
|
||||
|
||||
// const shouldBeHidden = value !== null && (this.parseValue(value) !== this.parseValue(x.sourceControlValue));
|
||||
// return this.VISIBILITY_RULE_LOGIC === 'OR' ? (isVisible || !shouldBeHidden) : (isVisible && !shouldBeHidden);
|
||||
// // if(value !== null && )
|
||||
// }, this.VISIBILITY_RULE_LOGIC === 'AND');
|
||||
// visibilityMap.set(sourceId, shouldBeVisible);
|
||||
// }
|
||||
|
||||
|
||||
// this.elementComputationalMap.set(targetId, visibilityMap);// unnessecary
|
||||
|
||||
|
||||
// const isVisible = this._computeVisibility(targetId);
|
||||
// this._emitChangesIfNeeded(targetId, isVisible);
|
||||
// const previousVisibility = this.isVisibleMap[targetId];
|
||||
// this.isVisibleMap[targetId] = isVisible;
|
||||
|
||||
// if (!isVisible && previousVisibility !== isVisible) {
|
||||
// this.resetControlWithId(this.form, targetId);
|
||||
// }
|
||||
|
||||
|
||||
// // for (let i = 0; i < visibilityRule.sourceVisibilityRules.length; i++) {
|
||||
// // if (value != null && (this.parseValue(value) !== this.parseValue(visibilityRule.sourceVisibilityRules[i].sourceControlValue))) {
|
||||
// // this._emitChangesIfNeeded(visibilityRule.targetControlId, false);
|
||||
// // this.elementVisibilityMap.set(visibilityRule.targetControlId, false);
|
||||
// // this.resetControlWithId(this.form, visibilityRule.targetControlId);
|
||||
// // //this.updateValueAndVisibility(visibilityRule.targetControlId, null);
|
||||
// // // this.clearValues(targetPathKey);
|
||||
// // return;
|
||||
// // }
|
||||
// // }
|
||||
// // this._emitChangesIfNeeded(visibilityRule.targetControlId, true);
|
||||
// // this.elementVisibilityMap.set(visibilityRule.targetControlId, true);
|
||||
|
||||
// // this.updateValueAndVisibility(visibilityRule.targetControlId, null);
|
||||
// }
|
||||
|
||||
|
||||
// private _computeVisibility(targetId: string): boolean {
|
||||
// //console.log('_computeVisibility: ' + targetId);
|
||||
|
||||
// const visibilityMap = this.elementComputationalMap.get(targetId);
|
||||
// const values = visibilityMap.values();
|
||||
// let currentVal = values.next();
|
||||
// let visibilityValues: boolean[] = [];
|
||||
// while (!currentVal.done) {
|
||||
// visibilityValues.push(currentVal.value);
|
||||
// currentVal = values.next();
|
||||
// }
|
||||
|
||||
|
||||
// if (visibilityValues.length) {
|
||||
// return visibilityValues.reduce((r, c) => {
|
||||
// if (this.VISIBILITY_RULE_LOGIC === 'OR') {
|
||||
// return r || c;
|
||||
// } else {
|
||||
// return r && c;
|
||||
// }
|
||||
// }, visibilityValues[0]);
|
||||
// }
|
||||
|
||||
// return this.DEFAULTVISIBILITY;
|
||||
// }
|
||||
|
||||
// private resetVisibilityRules() {
|
||||
// //console.log('resetVisibilityRules: ');
|
||||
|
||||
// this.isVisibleMap = {};
|
||||
// this.elementComputationalMap.clear();
|
||||
// this.elementComputationalMap = new Map<String, Map<String, boolean>>();
|
||||
// this._populateComputationMap(); /// !IMPORTANT FOR THE AND LOGIC
|
||||
// this._changeMade$.next();
|
||||
// }
|
||||
|
||||
// private _populateComputationMap(): void {
|
||||
// //console.log('_populateComputationMap: ');
|
||||
// // this.visibilityRuleContext.rules.forEach(rule => {
|
||||
// // const targetId = rule.targetControlId;
|
||||
// // const visibilityMap = this.elementComputationalMap.get(targetId) ? this.elementComputationalMap.get(targetId) : new Map<String, boolean>();
|
||||
// // rule.sourceVisibilityRules.forEach(vr => {
|
||||
// // visibilityMap.set(vr.sourceControlId, this.DEFAULTVISIBILITY);
|
||||
// // });
|
||||
// // this.elementComputationalMap.set(targetId, visibilityMap);
|
||||
// // });
|
||||
// }
|
||||
|
||||
// parseValue(value: any) {
|
||||
// if (typeof value === 'string') {
|
||||
// if (isNumeric(value)) { return value; }
|
||||
// else if (value === 'true') {
|
||||
// return true;
|
||||
// }
|
||||
// else if (value === 'false') {
|
||||
// return false;
|
||||
// }
|
||||
// else { return this.translate(value); }
|
||||
// } else { return value; }
|
||||
// }
|
||||
|
||||
// search(path, obj, target) {
|
||||
// for (const k in obj) {
|
||||
// if (obj.hasOwnProperty(k)) {
|
||||
// if (obj[k] === target) {
|
||||
// return path + '.' + k;
|
||||
// } else if (typeof obj[k] === 'object') {
|
||||
// const result = this.search(path + '.' + k, obj[k], target);
|
||||
// if (result) {
|
||||
// return result;
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// return false;
|
||||
// }
|
||||
|
||||
// scanIfChildsOfCompositeFieldHasVisibleItems(compositeFieldParent: UntypedFormGroup): boolean {
|
||||
// // console.log('scanIfChildsOfCompositeFieldHasVisibleItems: ' + compositeFieldParent);
|
||||
|
||||
// //TODO: implement this
|
||||
// return true;
|
||||
// // let isVisible = false;
|
||||
// // (<UntypedFormArray>(compositeFieldParent.get('fields'))).controls.forEach(element => {
|
||||
// // if (this.checkElementVisibility(element.get('id').value)) {
|
||||
// // isVisible = true;
|
||||
// // }
|
||||
// // });
|
||||
// // return isVisible;
|
||||
// }
|
||||
|
||||
// private translate(item: any) {
|
||||
// try {
|
||||
// return JSON.parse(item).value;
|
||||
// } catch (error) {
|
||||
// return item;
|
||||
// }
|
||||
// }
|
||||
|
||||
// private resetControlWithId(formControl: AbstractControl, id: string) {
|
||||
// //'resetControlWithId: ' + id);
|
||||
// //TODO: implement this
|
||||
// // if (formControl instanceof UntypedFormGroup) {
|
||||
// // if ((formControl as UntypedFormGroup).contains('id') && (formControl as UntypedFormGroup).contains('value') && (formControl as UntypedFormGroup).get('id').value === id) {
|
||||
// // this.resetFieldFormGroup(formControl);
|
||||
// // } if ((formControl as UntypedFormGroup).contains('id') && (formControl as UntypedFormGroup).contains('fields') && (formControl as UntypedFormGroup).get('id').value === id) {
|
||||
// // this.resetCompositeFieldFormGroup(formControl);
|
||||
// // } else {
|
||||
// // Object.keys(formControl.controls).forEach(item => {
|
||||
// // const control = formControl.get(item);
|
||||
// // this.resetControlWithId(control, id);
|
||||
// // });
|
||||
// // }
|
||||
// // } else if (formControl instanceof UntypedFormArray) {
|
||||
// // formControl.controls.forEach(item => {
|
||||
// // this.resetControlWithId(item, id);
|
||||
// // });
|
||||
// // }
|
||||
// }
|
||||
|
||||
// private resetFieldFormGroup(formGroup: UntypedFormGroup) {
|
||||
// //console.log('resetFieldFormGroup: ' + formGroup);
|
||||
// //TODO: implement this
|
||||
// // const renderStyle = formGroup.getRawValue().viewStyle.renderStyle;
|
||||
// // if (renderStyle === DatasetProfileFieldViewStyle.Validation || renderStyle === DatasetProfileFieldViewStyle.DatasetIdentifier) {
|
||||
// // formGroup.get('value').setValue({ identifier: '', type: '' });
|
||||
// // } else {
|
||||
// // formGroup.get('value').setValue(formGroup.get('defaultValue').value ? this.parseValue(formGroup.get('defaultValue').value.value) : undefined);
|
||||
// // }
|
||||
|
||||
// }
|
||||
|
||||
// private resetCompositeFieldFormGroup(formGroup: UntypedFormGroup) {
|
||||
// //console.log('resetCompositeFieldFormGroup: ' + formGroup);
|
||||
// //TODO: implement this
|
||||
// // (formGroup.get('fields') as UntypedFormArray).controls.forEach((element: UntypedFormGroup) => {
|
||||
// // this.resetFieldFormGroup(element);
|
||||
// // });
|
||||
// // (formGroup.get('multiplicityItems') as UntypedFormArray).controls.splice(0);
|
||||
// }
|
||||
// private _emitChangesIfNeeded(id: string, valueToBeSet: boolean) {
|
||||
// if (this.isVisibleMap[id]) {
|
||||
// if (this.isVisibleMap[id] != valueToBeSet) {
|
||||
// this._changeMade$.next();
|
||||
// }
|
||||
// } else {
|
||||
// this._changeMade$.next();
|
||||
// }
|
||||
// }
|
||||
// public get visibilityChange() {
|
||||
// return this._changeMade$.asObservable();
|
||||
// }
|
||||
// public getVisibilityDependency(targetId: string): VisibilityRuleSource[] | null {
|
||||
// //console.log('getVisibilityDependency: ' + targetId);
|
||||
// return this.visibilityRuleContext.rules.reduce((hasDependency, rule) => {
|
||||
// if (hasDependency) return hasDependency;
|
||||
|
||||
// if (rule.targetControlId === targetId) {
|
||||
// return rule.sourceVisibilityRules;
|
||||
// }
|
||||
|
||||
// return null;
|
||||
// }, null) as VisibilityRuleSource[];
|
||||
// }
|
||||
|
||||
// public getVisibilityTargets(sourceId: string): string[] {
|
||||
// console.log('getVisibilityTargets: ' + sourceId);
|
||||
// return this.visibilityRuleContext.rules.filter(x => {
|
||||
// const result = x.sourceVisibilityRules.filter(y => y.sourceControlId === sourceId);
|
||||
// return result.length;
|
||||
// }).map(x => x.targetControlId);
|
||||
// }
|
||||
|
||||
// // public appendVisibilityRule(rule: VisibilityRule): void{
|
||||
|
||||
// // const existingTargetRule = this.visibilityRuleContext.rules.find( r => r.targetControlId === rule.targetControlId);
|
||||
|
||||
// // if(existingTargetRule){
|
||||
// // rule.sourceVisibilityRules.forEach(svr =>{
|
||||
// // existingTargetRule.sourceVisibilityRules.push(svr);
|
||||
// // });
|
||||
// // }else{
|
||||
// // this.visibilityRuleContext.rules.push(rule);
|
||||
// // }
|
||||
|
||||
|
||||
// // }
|
||||
|
||||
|
||||
// //removes rule that has the specific id either as a source either as a target
|
||||
// public removeAllIdReferences(id: string): void {
|
||||
// //console.log('removeAllIdReferences: ' + id);
|
||||
|
||||
// // * Remove from visibility rues and visibility rules context
|
||||
|
||||
// //remove as a target
|
||||
// const temp = this.visibilityRuleContext.rules.map((x, i) => (x.targetControlId === id) ? i : null);
|
||||
// const indexes = temp.filter(x => x !== null);
|
||||
// indexes.reverse().forEach(index => this.visibilityRuleContext.rules.splice(index, 1));
|
||||
// this.isVisibleMap[id] = undefined;
|
||||
|
||||
|
||||
|
||||
// //remove as a source
|
||||
// const tbd = this.visibilityRuleContext.rules.reduce((to_be_deleted, rule, ruleIdx) => {
|
||||
// const idxs = rule.sourceVisibilityRules.map((x, i) => (x.sourceControlId === id) ? i : null).filter(x => x !== null);
|
||||
// idxs.reverse().forEach(index => rule.sourceVisibilityRules.splice(index, 1));
|
||||
|
||||
// if (!rule.sourceVisibilityRules.length) {
|
||||
// to_be_deleted.push(ruleIdx);
|
||||
// }
|
||||
// return to_be_deleted
|
||||
// }, []);
|
||||
|
||||
|
||||
// //clean up empty
|
||||
// tbd.reverse().forEach(index => {
|
||||
// this.visibilityRuleContext.rules.splice(index, 1);
|
||||
// });
|
||||
|
||||
|
||||
|
||||
// // * Remove from computational map
|
||||
|
||||
// // as a target
|
||||
// if (this.elementComputationalMap.get(id)) {
|
||||
// this.elementComputationalMap.delete(id);
|
||||
// }
|
||||
|
||||
|
||||
// // as a source
|
||||
// const keyIterator = this.elementComputationalMap.keys();
|
||||
// let currentKey = keyIterator.next();
|
||||
|
||||
|
||||
// while (!currentKey.done) {
|
||||
// const currentVals = this.elementComputationalMap.get(currentKey.value);
|
||||
// currentVals.delete(id);
|
||||
// currentKey = keyIterator.next();
|
||||
// }
|
||||
// }
|
||||
|
||||
|
||||
// public addNewRule(rule: Rule, currentVisibility = this.DEFAULTVISIBILITY): void {
|
||||
// //console.log('addNewRule: ' + rule + ' currentVisibility: ' + currentVisibility);
|
||||
|
||||
// const targetId = rule.targetField;
|
||||
// const sourceId = rule.sourceField;
|
||||
// this.visibilityRuleContext.addToVisibilityRulesContext(rule);
|
||||
|
||||
|
||||
// let visibilityMap = this.elementComputationalMap.get(targetId);
|
||||
|
||||
// if (!visibilityMap) {
|
||||
// visibilityMap = new Map<String, boolean>();
|
||||
// this.elementComputationalMap.set(targetId, visibilityMap);
|
||||
// }
|
||||
|
||||
// visibilityMap.set(sourceId, currentVisibility);
|
||||
// const isVisible = this._computeVisibility(targetId);
|
||||
|
||||
// this._emitChangesIfNeeded(targetId, isVisible);
|
||||
// this.isVisibleMap[targetId] = isVisible;
|
||||
// this.elementVisibilityMapSubject.next(this.isVisibleMap);
|
||||
// }
|
||||
|
||||
|
||||
// /**
|
||||
// * Check what sourceId hides or shows the target field
|
||||
// * return true if no rule found
|
||||
// */
|
||||
// public checkTargetVisibilityProvidedBySource(sourceId: string, targetId: string): boolean {
|
||||
// //console.log('checkTargetVisibilityProvidedBySource: ' + sourceId + ' targetId: ' + targetId);
|
||||
|
||||
// const computationalMap = this.elementComputationalMap.get(targetId);
|
||||
// if (computationalMap) {
|
||||
// return !!computationalMap.get(sourceId);
|
||||
// }
|
||||
|
||||
// return true;
|
||||
// }
|
||||
|
||||
// public getVisibilityRulesFromDescriptionTempalte(descriptionTemplate: DescriptionTemplate): Rule[] {
|
||||
// //console.log('getVisibilityRulesFromDescriptionTempalte: ' + descriptionTemplate);
|
||||
// const result: Rule[] = this.getVisibilityRulesFromDescriptionTempalteSections(descriptionTemplate?.definition?.pages);
|
||||
// return result;
|
||||
// }
|
||||
|
||||
// public getVisibilityRulesFromDescriptionTempalteSections(pages: DescriptionTemplatePage[]): Rule[] {
|
||||
// //console.log('getVisibilityRulesFromDescriptionTempalteSections: ' + sections);
|
||||
// const result: Rule[] = [];
|
||||
// pages.forEach(page => {
|
||||
// page?.sections?.forEach(section => {
|
||||
// if (section.sections != null) { result.push(...this.getVisibilityRulesFromDescriptionTempalteSections(section.sections)); };
|
||||
// section?.fieldSets?.forEach(fieldSet => {
|
||||
// fieldSet?.fields?.forEach(field => {
|
||||
// field.visibilityRules?.forEach(visibilityRule => {
|
||||
// result.push({
|
||||
// sourceField: field.id.toString(),
|
||||
// targetField: visibilityRule.target,
|
||||
// dateValue: visibilityRule.dateValue,
|
||||
// textValue: visibilityRule.textValue,
|
||||
// textListValue: visibilityRule.textListValue
|
||||
// })
|
||||
// });
|
||||
// });
|
||||
// });
|
||||
// });
|
||||
// });
|
||||
// return result;
|
||||
// }
|
||||
// }
|
||||
|
||||
// class MapWithDefault extends Map<string, boolean> {
|
||||
// get(key) {
|
||||
// //console.log('MapWithDefault');
|
||||
// if (!this.has(key)) {
|
||||
// this.set(key, true);
|
||||
// }
|
||||
// return super.get(key);
|
||||
// }
|
||||
// }
|
|
@ -1,454 +1,563 @@
|
|||
import { ApplicationRef, Injectable, NgZone } from '@angular/core';
|
||||
import { AbstractControl, UntypedFormArray, UntypedFormGroup } from '@angular/forms';
|
||||
import { isNumeric } from '@app/utilities/enhancers/utils';
|
||||
import { Injectable } from '@angular/core';
|
||||
import { AbstractControl } from '@angular/forms';
|
||||
import { Observable, Subject } from 'rxjs';
|
||||
import { VisibilityRule } from './models/visibility-rule';
|
||||
import { VisibilityRuleSource } from './models/visibility-rule-source';
|
||||
import { VisibilityRulesContext } from './models/visibility-rules-context';
|
||||
import { DescriptionTemplate, DescriptionTemplatePage, DescriptionTemplateSection } from '@app/core/model/description-template/description-template';
|
||||
import { Rule } from './models/rule';
|
||||
import { DescriptionTemplateDefinition, DescriptionTemplateField, DescriptionTemplateFieldSet, DescriptionTemplatePage, DescriptionTemplateSection } from '@app/core/model/description-template/description-template';
|
||||
import { RuleWithTarget } from './models/rule';
|
||||
import { FormService } from '@common/forms/form-service';
|
||||
import { DescriptionFieldPersist, DescriptionPropertyDefinitionFieldSetPersist, DescriptionPropertyDefinitionPersist } from '@app/core/model/description/description';
|
||||
import { DescriptionTemplateFieldType } from '@app/core/common/enum/description-template-field-type';
|
||||
|
||||
@Injectable()
|
||||
export class VisibilityRulesService {
|
||||
|
||||
private readonly VISIBILITY_RULE_LOGIC: 'OR' | 'AND' = 'OR';
|
||||
private readonly DEFAULTVISIBILITY = false;
|
||||
|
||||
private visibilityRuleContext: VisibilityRulesContext;
|
||||
private form: AbstractControl;
|
||||
|
||||
public isVisibleMap: { [key: string]: boolean } = {};
|
||||
|
||||
private elementVisibilityMapSubject = new Subject<{ [key: string]: boolean }>();
|
||||
private elementComputationalMap = new Map<String, Map<String, boolean>>(); /// keep saved the values of each form control validity value
|
||||
private _changeMade$ = new Subject<void>();
|
||||
|
||||
|
||||
// get isVisibleMap(): MapWithDefault {
|
||||
// // console.log('isVisibleMap');
|
||||
// return this.elementVisibilityMap;
|
||||
|
||||
// }; /// keep saved the values of each form control validity value
|
||||
private definition: DescriptionTemplateDefinition;
|
||||
private rulesBySources: Map<String, RuleWithTarget[]> ;
|
||||
public isVisibleMap: { [key: string]: boolean } = null;
|
||||
|
||||
constructor(
|
||||
public applicationReference: ApplicationRef,
|
||||
public ngZone: NgZone
|
||||
protected formService: FormService
|
||||
) {
|
||||
|
||||
}
|
||||
|
||||
getElementVisibilityMapObservable(): Observable<{ [key: string]: boolean }> {
|
||||
// this.isVisibleMap
|
||||
// console.log('getElementVisibilityMapObservable: ');
|
||||
return this.elementVisibilityMapSubject.asObservable();
|
||||
}
|
||||
|
||||
public checkElementVisibility(id: string): boolean {
|
||||
//console.log('checkElementVisibility: ' + id);
|
||||
return true;
|
||||
// if (this.visibilityRuleContext.rules.filter(item => item.targetControlId === id).length === 0) { return true; }
|
||||
// console.log(this.elementVisibilityMap.has(id) ? this.elementVisibilityMap.get(id) : false);
|
||||
// return this.elementVisibilityMap.has(id) ? this.elementVisibilityMap.get(id) : false;
|
||||
}
|
||||
|
||||
public buildVisibilityRules(item: Array<Rule>, form: AbstractControl) {
|
||||
this.visibilityRuleContext = new VisibilityRulesContext();
|
||||
this.visibilityRuleContext.buildVisibilityRuleContext(item || []);
|
||||
public setContext(definition: DescriptionTemplateDefinition, form: AbstractControl) {
|
||||
this.definition = definition;
|
||||
this.form = form;
|
||||
this.resetVisibilityRules();
|
||||
this.rulesBySources = null;
|
||||
this.isVisibleMap = null;
|
||||
this.calculateVisibility();
|
||||
}
|
||||
|
||||
public updateValueAndVisibility(id: string, value: any) {
|
||||
//console.log('updateValueAndVisibility: ' + id + ' value: ' + value);
|
||||
const visibilityRules = this.visibilityRuleContext.rules.filter(item => item.sourceVisibilityRules.filter(source => source.sourceControlId === id).length > 0);
|
||||
if (visibilityRules.length > 0) {
|
||||
visibilityRules.forEach(item => this.evaluateVisibility(item, value, id));
|
||||
this.elementVisibilityMapSubject.next(this.isVisibleMap);
|
||||
public buildVisibilityKey(id: string, ordinal: number | null): string {
|
||||
if (ordinal == null) return id;
|
||||
else return id + "_" + ordinal;
|
||||
}
|
||||
|
||||
public isVisible(id: string, ordinal: number | null): boolean {
|
||||
this.calculateVisibility();
|
||||
const fieldKey = this.buildVisibilityKey(id, ordinal);
|
||||
return this.isVisibleMap[fieldKey] ?? false;
|
||||
}
|
||||
|
||||
public getVisibilityStates(): { [key: string]: boolean } {
|
||||
this.calculateVisibility();
|
||||
return this.isVisibleMap;
|
||||
}
|
||||
|
||||
public updateVisibilityForSource(id: string) {
|
||||
const visibilityRules = this.rulesBySources.has(id) ? this.rulesBySources.get(id) : null;
|
||||
if (visibilityRules && visibilityRules.length > 0) {
|
||||
this.reloadVisibility();
|
||||
}
|
||||
}
|
||||
|
||||
private evaluateVisibility(visibilityRule: VisibilityRule, value: any, sourceId: string) {// source controlId is the same
|
||||
//console.log('evaluateVisibility: ' + visibilityRule + ' value: ' + value + ' sourceId: ' + sourceId);
|
||||
|
||||
const targetId = visibilityRule.targetControlId;
|
||||
const visibilityMap = this.elementComputationalMap.get(targetId) ? this.elementComputationalMap.get(targetId) : new Map<String, boolean>();
|
||||
|
||||
|
||||
if (value instanceof Array) {
|
||||
|
||||
const parsedSourceControlValues = visibilityRule.sourceVisibilityRules.map(e => this.parseValue(e.sourceControlValue));
|
||||
const parsedValues = value.map(e => this.parseValue(e));
|
||||
|
||||
const isVisible = parsedValues.map(v => parsedSourceControlValues.includes(v)).reduce((acc, current) => acc || current, false);
|
||||
|
||||
|
||||
// if(isVisible){
|
||||
// this._emitChangesIfNeeded(visibilityRule.targetControlId, true);
|
||||
// this.elementVisibilityMap.set(visibilityRule.targetControlId, true);
|
||||
// return;
|
||||
// }
|
||||
visibilityMap.set(sourceId, isVisible);
|
||||
|
||||
} else {
|
||||
const visibilityDependencySource = visibilityRule.sourceVisibilityRules.filter(x => x.sourceControlId === sourceId);
|
||||
|
||||
const shouldBeVisible = visibilityDependencySource.reduce((isVisible, x) => {
|
||||
|
||||
const shouldBeHidden = value !== null && (this.parseValue(value) !== this.parseValue(x.sourceControlValue));
|
||||
return this.VISIBILITY_RULE_LOGIC === 'OR' ? (isVisible || !shouldBeHidden) : (isVisible && !shouldBeHidden);
|
||||
// if(value !== null && )
|
||||
}, this.VISIBILITY_RULE_LOGIC === 'AND');
|
||||
visibilityMap.set(sourceId, shouldBeVisible);
|
||||
}
|
||||
|
||||
|
||||
this.elementComputationalMap.set(targetId, visibilityMap);// unnessecary
|
||||
|
||||
|
||||
const isVisible = this._computeVisibility(targetId);
|
||||
this._emitChangesIfNeeded(targetId, isVisible);
|
||||
const previousVisibility = this.isVisibleMap[targetId];
|
||||
this.isVisibleMap[targetId] = isVisible;
|
||||
|
||||
if (!isVisible && previousVisibility !== isVisible) {
|
||||
this.resetControlWithId(this.form, targetId);
|
||||
}
|
||||
|
||||
|
||||
// for (let i = 0; i < visibilityRule.sourceVisibilityRules.length; i++) {
|
||||
// if (value != null && (this.parseValue(value) !== this.parseValue(visibilityRule.sourceVisibilityRules[i].sourceControlValue))) {
|
||||
// this._emitChangesIfNeeded(visibilityRule.targetControlId, false);
|
||||
// this.elementVisibilityMap.set(visibilityRule.targetControlId, false);
|
||||
// this.resetControlWithId(this.form, visibilityRule.targetControlId);
|
||||
// //this.updateValueAndVisibility(visibilityRule.targetControlId, null);
|
||||
// // this.clearValues(targetPathKey);
|
||||
// return;
|
||||
// }
|
||||
// }
|
||||
// this._emitChangesIfNeeded(visibilityRule.targetControlId, true);
|
||||
// this.elementVisibilityMap.set(visibilityRule.targetControlId, true);
|
||||
|
||||
// this.updateValueAndVisibility(visibilityRule.targetControlId, null);
|
||||
public reloadVisibility() {
|
||||
this.rulesBySources = null;
|
||||
this.isVisibleMap = null;
|
||||
this.calculateVisibility();
|
||||
}
|
||||
|
||||
private calculateVisibility(){
|
||||
if (this.isVisibleMap != null) return;
|
||||
if (this.definition == null || this.form == null) return;
|
||||
|
||||
private _computeVisibility(targetId: string): boolean {
|
||||
//console.log('_computeVisibility: ' + targetId);
|
||||
this.initRules();
|
||||
this.buildTargetVisibility();
|
||||
this.expandVisibilityToChildren();
|
||||
this.setDefaultVisibilityForNotCaclucted();
|
||||
this.hideParentIfAllChildrenAreHidden();
|
||||
this.ensureFieldSetVisibility();
|
||||
}
|
||||
|
||||
const visibilityMap = this.elementComputationalMap.get(targetId);
|
||||
const values = visibilityMap.values();
|
||||
let currentVal = values.next();
|
||||
let visibilityValues: boolean[] = [];
|
||||
while (!currentVal.done) {
|
||||
visibilityValues.push(currentVal.value);
|
||||
currentVal = values.next();
|
||||
}
|
||||
private initRules(){
|
||||
if (this.definition == null || this.form == null) return;
|
||||
if (this.rulesBySources != null) return;
|
||||
this.rulesBySources = new Map();
|
||||
|
||||
|
||||
if (visibilityValues.length) {
|
||||
return visibilityValues.reduce((r, c) => {
|
||||
if (this.VISIBILITY_RULE_LOGIC === 'OR') {
|
||||
return r || c;
|
||||
} else {
|
||||
return r && c;
|
||||
const fields: DescriptionTemplateField[] = this.getAllDescriptionTemplateDefinitionFields(this.definition);
|
||||
for (let i = 0; i < fields.length; i++) {
|
||||
const fieldEntity = fields[i];
|
||||
if (fieldEntity.visibilityRules != null && fieldEntity.visibilityRules.length > 0) {
|
||||
for (let j = 0; j < fieldEntity.visibilityRules.length; j++) {
|
||||
const rule = fieldEntity.visibilityRules[j];
|
||||
if (!this.rulesBySources.has(fieldEntity.id)) this.rulesBySources.set(fieldEntity.id, []);
|
||||
const ruleWithTarget: RuleWithTarget = new RuleWithTarget(fieldEntity.id, rule, fieldEntity);
|
||||
this.rulesBySources.get(fieldEntity.id).push(ruleWithTarget);
|
||||
}
|
||||
}, visibilityValues[0]);
|
||||
}
|
||||
}
|
||||
|
||||
return this.DEFAULTVISIBILITY;
|
||||
}
|
||||
|
||||
private resetVisibilityRules() {
|
||||
//console.log('resetVisibilityRules: ');
|
||||
private getDescriptionTemplateDefinitionFieldById(definition: DescriptionTemplateDefinition, fieldId: string): DescriptionTemplateField[] {
|
||||
const fields: DescriptionTemplateField[] = this.getAllDescriptionTemplateDefinitionFields(definition);
|
||||
return fields.filter(x=> x.id == fieldId);
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplateDefinitionFields(definition: DescriptionTemplateDefinition): DescriptionTemplateField[] {
|
||||
let fields: DescriptionTemplateField[] = [];
|
||||
if (definition.pages == null) return fields;
|
||||
for (let i = 0; i < definition.pages.length; i++) {
|
||||
const item = definition.pages[i];
|
||||
fields = [...fields, ...this.getAllDescriptionTemplatePageFields(item)];
|
||||
}
|
||||
return fields;
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplatePageFields(definition: DescriptionTemplatePage): DescriptionTemplateField[] {
|
||||
let fields: DescriptionTemplateField[] = [];
|
||||
if (definition.sections == null) return fields;
|
||||
for (let i = 0; i < definition.sections.length; i++) {
|
||||
const item = definition.sections[i];
|
||||
fields = [...fields, ...this.getAllDescriptionTemplateSectionFields(item)];
|
||||
}
|
||||
return fields;
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplateSectionFields(definition: DescriptionTemplateSection): DescriptionTemplateField[] {
|
||||
let fields: DescriptionTemplateField[] = [];
|
||||
if (definition.sections != null) {
|
||||
for (let i = 0; i < definition.sections.length; i++) {
|
||||
const item = definition.sections[i];
|
||||
fields = [...fields, ...this.getAllDescriptionTemplateSectionFields(item)];
|
||||
}
|
||||
}
|
||||
if (definition.fieldSets != null) {
|
||||
for (let i = 0; i < definition.fieldSets.length; i++) {
|
||||
const item = definition.fieldSets[i];
|
||||
fields = [...fields, ...this.getAllDescriptionTemplateFieldSetFields(item)];
|
||||
}
|
||||
}
|
||||
return fields;
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplateFieldSetFields(definition: DescriptionTemplateFieldSet): DescriptionTemplateField[] {
|
||||
return definition.fields == null ? [] : definition.fields;
|
||||
}
|
||||
|
||||
private getDescriptionTemplateDefinitionFieldSetById(definition: DescriptionTemplateDefinition, fieldSetId: string): DescriptionTemplateFieldSet[] {
|
||||
const fieldSets: DescriptionTemplateFieldSet[] = this.getAllDescriptionTemplateDefinitionFieldSets(definition);
|
||||
return fieldSets.filter(x=> x.id == fieldSetId);
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplateDefinitionFieldSets(definition: DescriptionTemplateDefinition): DescriptionTemplateFieldSet[] {
|
||||
let fieldSets: DescriptionTemplateFieldSet[] = [];
|
||||
if (definition.pages == null) return fieldSets;
|
||||
for (let i = 0; i < definition.pages.length; i++) {
|
||||
const item = definition.pages[i];
|
||||
fieldSets = [...fieldSets, ...this.getAllDescriptionTemplatePageFieldSets(item)];
|
||||
}
|
||||
return fieldSets;
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplatePageFieldSets(definition: DescriptionTemplatePage): DescriptionTemplateFieldSet[] {
|
||||
let fieldSets: DescriptionTemplateFieldSet[] = [];
|
||||
if (definition.sections == null) return fieldSets;
|
||||
for (let i = 0; i < definition.sections.length; i++) {
|
||||
const item = definition.sections[i];
|
||||
fieldSets = [...fieldSets, ...this.getAllDescriptionTemplateSectionFieldSets(item)];
|
||||
}
|
||||
return fieldSets;
|
||||
}
|
||||
|
||||
private getAllDescriptionTemplateSectionFieldSets(definition: DescriptionTemplateSection): DescriptionTemplateFieldSet[] {
|
||||
let fieldSets: DescriptionTemplateFieldSet[] = [];
|
||||
if (definition.sections != null) {
|
||||
for (let i = 0; i < definition.sections.length; i++) {
|
||||
const item = definition.sections[i];
|
||||
fieldSets = [...fieldSets, ...this.getAllDescriptionTemplateSectionFieldSets(item)];
|
||||
}
|
||||
}
|
||||
if (definition.fieldSets != null) {
|
||||
for (let i = 0; i < definition.fieldSets.length; i++) {
|
||||
const item = definition.fieldSets[i];
|
||||
fieldSets = [...fieldSets, ...definition.fieldSets];
|
||||
}
|
||||
}
|
||||
return fieldSets;
|
||||
}
|
||||
|
||||
private buildTargetVisibility(){
|
||||
this.isVisibleMap = {};
|
||||
this.elementComputationalMap.clear();
|
||||
this.elementComputationalMap = new Map<String, Map<String, boolean>>();
|
||||
this._populateComputationMap(); /// !IMPORTANT FOR THE AND LOGIC
|
||||
this._changeMade$.next();
|
||||
}
|
||||
this.rulesBySources.forEach((ruleForSource: RuleWithTarget[], ruleForSourceKey: string) => {
|
||||
for (let i = 0; i < ruleForSource.length; i++) {
|
||||
const rule = ruleForSource[i];
|
||||
const propertyDefinition: DescriptionPropertyDefinitionPersist = this.formService.getValue(this.form.value) as DescriptionPropertyDefinitionPersist;
|
||||
if (propertyDefinition.fieldSets != null) {
|
||||
new Map(Object.entries(propertyDefinition.fieldSets)).forEach((propertyDefinitionFieldSet: DescriptionPropertyDefinitionFieldSetPersist, propertyDefinitionFieldSetKey: string) => {
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let j = 0; j < propertyDefinitionFieldSet.items.length; j++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[j];
|
||||
if (definitionFieldSetItem?.fields != null) {
|
||||
new Map(Object.entries(definitionFieldSetItem.fields)).forEach((field: DescriptionFieldPersist, key: string) => {
|
||||
if (rule.source == key){
|
||||
|
||||
private _populateComputationMap(): void {
|
||||
//console.log('_populateComputationMap: ');
|
||||
// this.visibilityRuleContext.rules.forEach(rule => {
|
||||
// const targetId = rule.targetControlId;
|
||||
// const visibilityMap = this.elementComputationalMap.get(targetId) ? this.elementComputationalMap.get(targetId) : new Map<String, boolean>();
|
||||
// rule.sourceVisibilityRules.forEach(vr => {
|
||||
// visibilityMap.set(vr.sourceControlId, this.DEFAULTVISIBILITY);
|
||||
// });
|
||||
// this.elementComputationalMap.set(targetId, visibilityMap);
|
||||
// });
|
||||
}
|
||||
if (new Map(Object.entries(definitionFieldSetItem.fields)).has(rule.target)){ //Rule applies only for current multiple item
|
||||
const fieldKey = this.buildVisibilityKey(rule.target, definitionFieldSetItem.ordinal);
|
||||
const currentState = this.isVisibleMap[fieldKey] ?? false;
|
||||
this.isVisibleMap[fieldKey] = currentState || this.ruleIsTrue(rule, field);
|
||||
} else if (this.getDescriptionTemplateDefinitionFieldById(this.definition, rule.target).length > 0 || this.getDescriptionTemplateDefinitionFieldSetById(this.definition, rule.target).length > 0) { //Rule applies to different fieldset, so we apply for all multiple items
|
||||
const ordinals: number[] = this.getKeyOrdinals(rule.target, propertyDefinition);
|
||||
for (let k = 0; k < ordinals.length; k++) {
|
||||
const ordinal = ordinals[j];
|
||||
const fieldKey = this.buildVisibilityKey(rule.target, ordinal);
|
||||
const currentState = this.isVisibleMap[fieldKey] ?? false;
|
||||
this.isVisibleMap[fieldKey] = currentState || this.ruleIsTrue(rule, field);
|
||||
|
||||
parseValue(value: any) {
|
||||
if (typeof value === 'string') {
|
||||
if (isNumeric(value)) { return value; }
|
||||
else if (value === 'true') {
|
||||
return true;
|
||||
}
|
||||
} else {
|
||||
const fieldKey = this.buildVisibilityKey(rule.target, null); //Ordinal is null if target not on field
|
||||
const currentState = this.isVisibleMap[fieldKey] ?? false;
|
||||
this.isVisibleMap[fieldKey] = currentState || this.ruleIsTrue(rule, field);
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
else if (value === 'false') {
|
||||
return false;
|
||||
}
|
||||
else { return this.translate(value); }
|
||||
} else { return value; }
|
||||
});
|
||||
}
|
||||
|
||||
search(path, obj, target) {
|
||||
for (const k in obj) {
|
||||
if (obj.hasOwnProperty(k)) {
|
||||
if (obj[k] === target) {
|
||||
return path + '.' + k;
|
||||
} else if (typeof obj[k] === 'object') {
|
||||
const result = this.search(path + '.' + k, obj[k], target);
|
||||
if (result) {
|
||||
return result;
|
||||
private getKeyOrdinals(key: string, propertyDefinition: DescriptionPropertyDefinitionPersist): number[]{
|
||||
let ordinals = [];
|
||||
if (propertyDefinition.fieldSets != null) {
|
||||
new Map(Object.entries(propertyDefinition.fieldSets)).forEach((propertyDefinitionFieldSet: DescriptionPropertyDefinitionFieldSetPersist, propertyDefinitionFieldSetKey: string) => {
|
||||
if (propertyDefinitionFieldSetKey == key) {
|
||||
ordinals = propertyDefinitionFieldSet.items?.map(x => x.ordinal) ?? [];
|
||||
return ordinals;
|
||||
}
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let i = 0; i < propertyDefinitionFieldSet.items.length; i++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[i];
|
||||
if (definitionFieldSetItem?.fields != null) {
|
||||
new Map(Object.entries(definitionFieldSetItem.fields)).forEach((field: DescriptionFieldPersist, fieldKey: string) => {
|
||||
if (fieldKey == key) ordinals = propertyDefinitionFieldSet.items?.map(x=> x.ordinal) ?? [];
|
||||
});
|
||||
if (ordinals != null && ordinals.length > 0) return ordinals;
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
return ordinals;
|
||||
}
|
||||
|
||||
private ruleIsTrue(rule: RuleWithTarget, field: DescriptionFieldPersist) :boolean{
|
||||
if (field != null){
|
||||
const fieldType: DescriptionTemplateFieldType = rule.field != null && rule.field.data != null ? rule.field.data.fieldType : DescriptionTemplateFieldType.FREE_TEXT;
|
||||
if ([DescriptionTemplateFieldType.FREE_TEXT, DescriptionTemplateFieldType.CHECK_BOX, DescriptionTemplateFieldType.TEXT_AREA,
|
||||
DescriptionTemplateFieldType.RICH_TEXT_AREA, DescriptionTemplateFieldType.UPLOAD, DescriptionTemplateFieldType.BOOLEAN_DECISION,
|
||||
DescriptionTemplateFieldType.RADIO_BOX, DescriptionTemplateFieldType.CURRENCY, DescriptionTemplateFieldType.SELECT].includes(fieldType) && field.textValue != null && field.textValue.length > 0) {
|
||||
if (DescriptionTemplateFieldType.UPLOAD == fieldType){
|
||||
return false; //not apply visibility logic
|
||||
} else {
|
||||
return field.textValue == rule.textValue;
|
||||
}
|
||||
}
|
||||
else if ([DescriptionTemplateFieldType.SELECT, DescriptionTemplateFieldType.TAGS, DescriptionTemplateFieldType.INTERNAL_ENTRIES_DMPS,
|
||||
DescriptionTemplateFieldType.INTERNAL_ENTRIES_DESCRIPTIONS].includes(fieldType) && field.textListValue != null && field.textListValue.length > 0) {
|
||||
return rule.textListValue != null &&
|
||||
rule.textListValue.every(x=> field.textListValue.includes(x));
|
||||
}
|
||||
else if (DescriptionTemplateFieldType.REFERENCE_TYPES == fieldType) {
|
||||
return false; //not implemented visibility logic
|
||||
}
|
||||
else if (DescriptionTemplateFieldType.DATE_PICKER == fieldType && field.dateValue != null) return field.dateValue == rule.dateValue;
|
||||
else if ([DescriptionTemplateFieldType.VALIDATION, DescriptionTemplateFieldType.DATASET_IDENTIFIER].includes(fieldType) && field.externalIdentifier != null) {
|
||||
return false; //not implemented visibility logic
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
scanIfChildsOfCompositeFieldHasVisibleItems(compositeFieldParent: UntypedFormGroup): boolean {
|
||||
// console.log('scanIfChildsOfCompositeFieldHasVisibleItems: ' + compositeFieldParent);
|
||||
|
||||
//TODO: implement this
|
||||
return true;
|
||||
// let isVisible = false;
|
||||
// (<UntypedFormArray>(compositeFieldParent.get('fields'))).controls.forEach(element => {
|
||||
// if (this.checkElementVisibility(element.get('id').value)) {
|
||||
// isVisible = true;
|
||||
// }
|
||||
// });
|
||||
// return isVisible;
|
||||
}
|
||||
|
||||
private translate(item: any) {
|
||||
try {
|
||||
return JSON.parse(item).value;
|
||||
} catch (error) {
|
||||
return item;
|
||||
private expandVisibilityToChildren(){
|
||||
if (this.definition?.pages == null) return;
|
||||
for (let i = 0; i < this.definition?.pages.length; i++) {
|
||||
const pageEntity = this.definition?.pages[i];
|
||||
const fieldKey = this.buildVisibilityKey(pageEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
this.expandPageVisibility(pageEntity, currentValue);
|
||||
}
|
||||
}
|
||||
|
||||
private resetControlWithId(formControl: AbstractControl, id: string) {
|
||||
//'resetControlWithId: ' + id);
|
||||
//TODO: implement this
|
||||
// if (formControl instanceof UntypedFormGroup) {
|
||||
// if ((formControl as UntypedFormGroup).contains('id') && (formControl as UntypedFormGroup).contains('value') && (formControl as UntypedFormGroup).get('id').value === id) {
|
||||
// this.resetFieldFormGroup(formControl);
|
||||
// } if ((formControl as UntypedFormGroup).contains('id') && (formControl as UntypedFormGroup).contains('fields') && (formControl as UntypedFormGroup).get('id').value === id) {
|
||||
// this.resetCompositeFieldFormGroup(formControl);
|
||||
// } else {
|
||||
// Object.keys(formControl.controls).forEach(item => {
|
||||
// const control = formControl.get(item);
|
||||
// this.resetControlWithId(control, id);
|
||||
// });
|
||||
// }
|
||||
// } else if (formControl instanceof UntypedFormArray) {
|
||||
// formControl.controls.forEach(item => {
|
||||
// this.resetControlWithId(item, id);
|
||||
// });
|
||||
// }
|
||||
}
|
||||
|
||||
private resetFieldFormGroup(formGroup: UntypedFormGroup) {
|
||||
//console.log('resetFieldFormGroup: ' + formGroup);
|
||||
//TODO: implement this
|
||||
// const renderStyle = formGroup.getRawValue().viewStyle.renderStyle;
|
||||
// if (renderStyle === DatasetProfileFieldViewStyle.Validation || renderStyle === DatasetProfileFieldViewStyle.DatasetIdentifier) {
|
||||
// formGroup.get('value').setValue({ identifier: '', type: '' });
|
||||
// } else {
|
||||
// formGroup.get('value').setValue(formGroup.get('defaultValue').value ? this.parseValue(formGroup.get('defaultValue').value.value) : undefined);
|
||||
// }
|
||||
|
||||
}
|
||||
|
||||
private resetCompositeFieldFormGroup(formGroup: UntypedFormGroup) {
|
||||
//console.log('resetCompositeFieldFormGroup: ' + formGroup);
|
||||
//TODO: implement this
|
||||
// (formGroup.get('fields') as UntypedFormArray).controls.forEach((element: UntypedFormGroup) => {
|
||||
// this.resetFieldFormGroup(element);
|
||||
// });
|
||||
// (formGroup.get('multiplicityItems') as UntypedFormArray).controls.splice(0);
|
||||
}
|
||||
private _emitChangesIfNeeded(id: string, valueToBeSet: boolean) {
|
||||
if (this.isVisibleMap[id]) {
|
||||
if (this.isVisibleMap[id] != valueToBeSet) {
|
||||
this._changeMade$.next();
|
||||
private expandPageVisibility(pageEntity : DescriptionTemplatePage, parentVisibility : boolean | null){
|
||||
if (pageEntity.sections == null) return;
|
||||
for (let i = 0; i < pageEntity.sections.length; i++) {
|
||||
const sectionEntity = pageEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(sectionEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue != null){
|
||||
if (parentVisibility != null && !parentVisibility) { //Parent is hidden so all childs should be hidden
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
this.expandSectionVisibility(sectionEntity, currentValue);
|
||||
} else {
|
||||
this.expandSectionVisibility(sectionEntity, currentValue);
|
||||
}
|
||||
} else {
|
||||
if (parentVisibility != null) this.isVisibleMap[fieldKey] = parentVisibility;
|
||||
this.expandSectionVisibility(sectionEntity, parentVisibility);
|
||||
}
|
||||
} else {
|
||||
this._changeMade$.next();
|
||||
}
|
||||
}
|
||||
public get visibilityChange() {
|
||||
return this._changeMade$.asObservable();
|
||||
}
|
||||
public getVisibilityDependency(targetId: string): VisibilityRuleSource[] | null {
|
||||
//console.log('getVisibilityDependency: ' + targetId);
|
||||
return this.visibilityRuleContext.rules.reduce((hasDependency, rule) => {
|
||||
if (hasDependency) return hasDependency;
|
||||
|
||||
if (rule.targetControlId === targetId) {
|
||||
return rule.sourceVisibilityRules;
|
||||
private expandSectionVisibility(sectionEntity: DescriptionTemplateSection, parentVisibility : boolean | null){
|
||||
if (sectionEntity.sections != null) {
|
||||
for (let i = 0; i < sectionEntity.sections.length; i++) {
|
||||
const subSectionEntity = sectionEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(subSectionEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue != null){
|
||||
if (parentVisibility != null && !parentVisibility) { //Parent is hidden so all childs should be hidden
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
this.expandSectionVisibility(subSectionEntity, currentValue);
|
||||
} else {
|
||||
this.expandSectionVisibility(subSectionEntity, currentValue);
|
||||
}
|
||||
} else {
|
||||
if (parentVisibility != null) this.isVisibleMap[fieldKey] = parentVisibility;
|
||||
this.expandSectionVisibility(subSectionEntity, parentVisibility);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (sectionEntity.fieldSets != null) {
|
||||
for (let i = 0; i < sectionEntity.fieldSets.length; i++) {
|
||||
const fieldSetEntity = sectionEntity.fieldSets[i];
|
||||
|
||||
return null;
|
||||
}, null) as VisibilityRuleSource[];
|
||||
}
|
||||
|
||||
public getVisibilityTargets(sourceId: string): string[] {
|
||||
console.log('getVisibilityTargets: ' + sourceId);
|
||||
return this.visibilityRuleContext.rules.filter(x => {
|
||||
const result = x.sourceVisibilityRules.filter(y => y.sourceControlId === sourceId);
|
||||
return result.length;
|
||||
}).map(x => x.targetControlId);
|
||||
}
|
||||
|
||||
// public appendVisibilityRule(rule: VisibilityRule): void{
|
||||
|
||||
// const existingTargetRule = this.visibilityRuleContext.rules.find( r => r.targetControlId === rule.targetControlId);
|
||||
|
||||
// if(existingTargetRule){
|
||||
// rule.sourceVisibilityRules.forEach(svr =>{
|
||||
// existingTargetRule.sourceVisibilityRules.push(svr);
|
||||
// });
|
||||
// }else{
|
||||
// this.visibilityRuleContext.rules.push(rule);
|
||||
// }
|
||||
|
||||
|
||||
// }
|
||||
|
||||
|
||||
//removes rule that has the specific id either as a source either as a target
|
||||
public removeAllIdReferences(id: string): void {
|
||||
//console.log('removeAllIdReferences: ' + id);
|
||||
|
||||
// * Remove from visibility rues and visibility rules context
|
||||
|
||||
//remove as a target
|
||||
const temp = this.visibilityRuleContext.rules.map((x, i) => (x.targetControlId === id) ? i : null);
|
||||
const indexes = temp.filter(x => x !== null);
|
||||
indexes.reverse().forEach(index => this.visibilityRuleContext.rules.splice(index, 1));
|
||||
this.isVisibleMap[id] = undefined;
|
||||
|
||||
|
||||
|
||||
//remove as a source
|
||||
const tbd = this.visibilityRuleContext.rules.reduce((to_be_deleted, rule, ruleIdx) => {
|
||||
const idxs = rule.sourceVisibilityRules.map((x, i) => (x.sourceControlId === id) ? i : null).filter(x => x !== null);
|
||||
idxs.reverse().forEach(index => rule.sourceVisibilityRules.splice(index, 1));
|
||||
|
||||
if (!rule.sourceVisibilityRules.length) {
|
||||
to_be_deleted.push(ruleIdx);
|
||||
const propertyDefinition = this.formService.getValue(this.form.value) as DescriptionPropertyDefinitionPersist;
|
||||
if (propertyDefinition.fieldSets != null && new Map(Object.entries(propertyDefinition.fieldSets)).has(fieldSetEntity.id)) {
|
||||
const propertyDefinitionFieldSet = new Map(Object.entries(propertyDefinition.fieldSets)).get(fieldSetEntity.id)
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let j = 0; j < propertyDefinitionFieldSet.items.length; j++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[j];
|
||||
const fieldKey = this.buildVisibilityKey(fieldSetEntity.id, definitionFieldSetItem.ordinal);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue != null){
|
||||
if (parentVisibility != null && !parentVisibility) { //Parent is hidden so all childs should be hidden
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
this.expandFieldSetVisibility(fieldSetEntity, currentValue, definitionFieldSetItem.ordinal);
|
||||
} else {
|
||||
this.expandFieldSetVisibility(fieldSetEntity, currentValue, definitionFieldSetItem.ordinal);
|
||||
}
|
||||
} else {
|
||||
if (parentVisibility != null) this.isVisibleMap[fieldKey] = parentVisibility;
|
||||
this.expandFieldSetVisibility(fieldSetEntity, parentVisibility, definitionFieldSetItem.ordinal);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return to_be_deleted
|
||||
}, []);
|
||||
|
||||
|
||||
//clean up empty
|
||||
tbd.reverse().forEach(index => {
|
||||
this.visibilityRuleContext.rules.splice(index, 1);
|
||||
});
|
||||
|
||||
|
||||
|
||||
// * Remove from computational map
|
||||
|
||||
// as a target
|
||||
if (this.elementComputationalMap.get(id)) {
|
||||
this.elementComputationalMap.delete(id);
|
||||
}
|
||||
|
||||
|
||||
// as a source
|
||||
const keyIterator = this.elementComputationalMap.keys();
|
||||
let currentKey = keyIterator.next();
|
||||
|
||||
|
||||
while (!currentKey.done) {
|
||||
const currentVals = this.elementComputationalMap.get(currentKey.value);
|
||||
currentVals.delete(id);
|
||||
currentKey = keyIterator.next();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public addNewRule(rule: Rule, currentVisibility = this.DEFAULTVISIBILITY): void {
|
||||
//console.log('addNewRule: ' + rule + ' currentVisibility: ' + currentVisibility);
|
||||
|
||||
const targetId = rule.targetField;
|
||||
const sourceId = rule.sourceField;
|
||||
this.visibilityRuleContext.addToVisibilityRulesContext(rule);
|
||||
|
||||
|
||||
let visibilityMap = this.elementComputationalMap.get(targetId);
|
||||
|
||||
if (!visibilityMap) {
|
||||
visibilityMap = new Map<String, boolean>();
|
||||
this.elementComputationalMap.set(targetId, visibilityMap);
|
||||
private expandFieldSetVisibility(fieldSetEntity: DescriptionTemplateFieldSet, parentVisibility: boolean | null, ordinal: number){
|
||||
if (fieldSetEntity.fields != null) {
|
||||
for (let i = 0; i < fieldSetEntity.fields.length; i++) {
|
||||
const fieldEntity = fieldSetEntity.fields[i];
|
||||
const fieldKey = this.buildVisibilityKey(fieldEntity.id, ordinal);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue != null){
|
||||
if (parentVisibility != null && !parentVisibility) { //Parent is hidden so all childs should be hidden
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
}
|
||||
} else if (parentVisibility != null){
|
||||
this.isVisibleMap[fieldKey] = parentVisibility;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
visibilityMap.set(sourceId, currentVisibility);
|
||||
const isVisible = this._computeVisibility(targetId);
|
||||
|
||||
this._emitChangesIfNeeded(targetId, isVisible);
|
||||
this.isVisibleMap[targetId] = isVisible;
|
||||
this.elementVisibilityMapSubject.next(this.isVisibleMap);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Check what sourceId hides or shows the target field
|
||||
* return true if no rule found
|
||||
*/
|
||||
public checkTargetVisibilityProvidedBySource(sourceId: string, targetId: string): boolean {
|
||||
//console.log('checkTargetVisibilityProvidedBySource: ' + sourceId + ' targetId: ' + targetId);
|
||||
|
||||
const computationalMap = this.elementComputationalMap.get(targetId);
|
||||
if (computationalMap) {
|
||||
return !!computationalMap.get(sourceId);
|
||||
private setDefaultVisibilityForNotCaclucted() {
|
||||
if (this.definition?.pages == null) return;
|
||||
for (let i = 0; i < this.definition?.pages.length; i++) {
|
||||
const pageEntity = this.definition?.pages[i];
|
||||
const fieldKey = this.buildVisibilityKey(pageEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue == null) this.isVisibleMap[fieldKey] = true;
|
||||
this.setDefaultPageVisibility(pageEntity);
|
||||
}
|
||||
}
|
||||
|
||||
private setDefaultPageVisibility(pageEntity: DescriptionTemplatePage) {
|
||||
if (pageEntity.sections == null) return;
|
||||
for (let i = 0; i < pageEntity.sections.length; i++) {
|
||||
const sectionEntity = pageEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(sectionEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue == null){
|
||||
this.isVisibleMap[fieldKey] = true;
|
||||
this.setDefaultSectionVisibility(sectionEntity);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private setDefaultSectionVisibility(sectionEntity: DescriptionTemplateSection) {
|
||||
if (sectionEntity.sections != null) {
|
||||
for (let i = 0; i < sectionEntity.sections.length; i++) {
|
||||
const subSectionEntity = sectionEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(subSectionEntity.id, null);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue == null){
|
||||
this.isVisibleMap[fieldKey] = true;
|
||||
this.setDefaultSectionVisibility(subSectionEntity);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (sectionEntity.fieldSets != null) {
|
||||
for (let i = 0; i < sectionEntity.fieldSets.length; i++) {
|
||||
const fieldSetEntity = sectionEntity.fieldSets[i];
|
||||
|
||||
const propertyDefinition = this.formService.getValue(this.form.value) as DescriptionPropertyDefinitionPersist;
|
||||
if (propertyDefinition.fieldSets != null && new Map(Object.entries(propertyDefinition.fieldSets)).has(fieldSetEntity.id)) {
|
||||
const propertyDefinitionFieldSet = new Map(Object.entries(propertyDefinition.fieldSets)).get(fieldSetEntity.id)
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let j = 0; j < propertyDefinitionFieldSet.items.length; j++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[j];
|
||||
const fieldKey = this.buildVisibilityKey(fieldSetEntity.id, definitionFieldSetItem.ordinal);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue == null){
|
||||
this.isVisibleMap[fieldKey] = true;
|
||||
this.setDefaultFieldSetVisibility(fieldSetEntity, definitionFieldSetItem.ordinal);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private setDefaultFieldSetVisibility(fieldSetEntity: DescriptionTemplateFieldSet, ordinal: number){
|
||||
if (fieldSetEntity.fields != null) {
|
||||
for (let i = 0; i < fieldSetEntity.fields.length; i++) {
|
||||
const fieldEntity = fieldSetEntity.fields[i];
|
||||
const fieldKey = this.buildVisibilityKey(fieldEntity.id, ordinal);
|
||||
const currentValue: boolean | null = this.isVisibleMap[fieldKey] ?? null;
|
||||
if (currentValue == null){
|
||||
this.isVisibleMap[fieldKey] = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private hideParentIfAllChildrenAreHidden() {
|
||||
if (this.definition?.pages == null) return;
|
||||
for (let i = 0; i < this.definition?.pages.length; i++) {
|
||||
const pageEntity = this.definition?.pages[i];
|
||||
const fieldKey = this.buildVisibilityKey(pageEntity.id, null);
|
||||
const isCurrentHidden = this.isHiddenPageVisibilityIfAllChildrenIsHidden(pageEntity);
|
||||
|
||||
if (isCurrentHidden && (this.isVisibleMap[fieldKey] ?? true)) {
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private isHiddenPageVisibilityIfAllChildrenIsHidden(pageEntity: DescriptionTemplatePage): boolean{
|
||||
let isHidden = true;
|
||||
if (pageEntity?.sections == null) return;
|
||||
for (let i = 0; i < pageEntity.sections.length; i++) {
|
||||
const sectionEntity = pageEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(sectionEntity.id, null);
|
||||
const isCurrentHidden = this.isHiddenSectionIfAllChildrenIsHidden(sectionEntity);
|
||||
if (isCurrentHidden && (this.isVisibleMap[fieldKey] ?? true)) {
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
}
|
||||
isHidden = isHidden && isCurrentHidden;
|
||||
}
|
||||
return isHidden;
|
||||
}
|
||||
|
||||
private isHiddenSectionIfAllChildrenIsHidden(sectionEntity: DescriptionTemplateSection): boolean{
|
||||
let isHidden = true;
|
||||
if (sectionEntity.sections != null) {
|
||||
for (let i = 0; i < sectionEntity.sections.length; i++) {
|
||||
const subSectionEntity = sectionEntity.sections[i];
|
||||
const fieldKey = this.buildVisibilityKey(subSectionEntity.id, null);
|
||||
const isCurrentHidden = this.isHiddenSectionIfAllChildrenIsHidden(subSectionEntity);
|
||||
if (isCurrentHidden && (this.isVisibleMap[fieldKey] ?? true)) {
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
}
|
||||
isHidden = isHidden && isCurrentHidden;
|
||||
}
|
||||
}
|
||||
if (sectionEntity.fieldSets != null) {
|
||||
for (let i = 0; i < sectionEntity.fieldSets.length; i++) {
|
||||
const fieldSetEntity = sectionEntity.fieldSets[i];
|
||||
const propertyDefinition = this.formService.getValue(this.form.value) as DescriptionPropertyDefinitionPersist;
|
||||
if (propertyDefinition.fieldSets != null && new Map(Object.entries(propertyDefinition.fieldSets)).has(fieldSetEntity.id)) {
|
||||
const propertyDefinitionFieldSet = new Map(Object.entries(propertyDefinition.fieldSets)).get(fieldSetEntity.id)
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let j = 0; j < propertyDefinitionFieldSet.items.length; j++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[j];
|
||||
const fieldKey = this.buildVisibilityKey(fieldSetEntity.id, definitionFieldSetItem.ordinal);
|
||||
const isCurrentHidden = this.isHiddenFieldSetIfAllChildrenIsHidden(fieldSetEntity, definitionFieldSetItem.ordinal);
|
||||
if (isCurrentHidden && (this.isVisibleMap[fieldKey] ?? true)) {
|
||||
this.isVisibleMap[fieldKey] = false;
|
||||
}
|
||||
isHidden = isHidden && isCurrentHidden;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return isHidden;
|
||||
}
|
||||
|
||||
private isHiddenFieldSetIfAllChildrenIsHidden(fieldSetEntity: DescriptionTemplateFieldSet, ordinal: number): boolean{
|
||||
let isHidden = true;
|
||||
if (fieldSetEntity.fields != null) {
|
||||
for (let i = 0; i < fieldSetEntity.fields.length; i++) {
|
||||
const fieldEntity = fieldSetEntity.fields[i];
|
||||
const fieldKey = this.buildVisibilityKey(fieldEntity.id, ordinal);
|
||||
const currentValue: boolean | null = (this.isVisibleMap[fieldKey] ?? true);
|
||||
isHidden = isHidden && !currentValue;
|
||||
}
|
||||
return isHidden;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
public getVisibilityRulesFromDescriptionTempalte(descriptionTemplate: DescriptionTemplate): Rule[] {
|
||||
//console.log('getVisibilityRulesFromDescriptionTempalte: ' + descriptionTemplate);
|
||||
const result: Rule[] = this.getVisibilityRulesFromDescriptionTempalteSections(descriptionTemplate?.definition?.pages);
|
||||
return result;
|
||||
}
|
||||
|
||||
public getVisibilityRulesFromDescriptionTempalteSections(pages: DescriptionTemplatePage[]): Rule[] {
|
||||
//console.log('getVisibilityRulesFromDescriptionTempalteSections: ' + sections);
|
||||
const result: Rule[] = [];
|
||||
pages.forEach(page => {
|
||||
page?.sections?.forEach(section => {
|
||||
if (section.sections != null) { result.push(...this.getVisibilityRulesFromDescriptionTempalteSections(section.sections)); };
|
||||
section?.fieldSets?.forEach(fieldSet => {
|
||||
fieldSet?.fields?.forEach(field => {
|
||||
field.visibilityRules?.forEach(visibilityRule => {
|
||||
result.push({
|
||||
sourceField: field.id.toString(),
|
||||
targetField: visibilityRule.target,
|
||||
requiredValue: visibilityRule.value
|
||||
})
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
return result;
|
||||
}
|
||||
}
|
||||
|
||||
class MapWithDefault extends Map<string, boolean> {
|
||||
get(key) {
|
||||
//console.log('MapWithDefault');
|
||||
if (!this.has(key)) {
|
||||
this.set(key, true);
|
||||
private ensureFieldSetVisibility() {
|
||||
if (this.definition?.pages == null) return;
|
||||
for (let i = 0; i < this.definition?.pages.length; i++) {
|
||||
const pageEntity = this.definition?.pages[i];
|
||||
this.ensurePageFieldSetVisibility(pageEntity);
|
||||
}
|
||||
}
|
||||
|
||||
private ensurePageFieldSetVisibility(pageEntity: DescriptionTemplatePage){
|
||||
if (pageEntity?.sections == null) return;
|
||||
for (let i = 0; i < pageEntity.sections.length; i++) {
|
||||
const sectionEntity = pageEntity.sections[i];
|
||||
this.ensureSectionFieldSetVisibility(sectionEntity);
|
||||
}
|
||||
}
|
||||
|
||||
private ensureSectionFieldSetVisibility(sectionEntity: DescriptionTemplateSection){
|
||||
if (sectionEntity.sections != null) {
|
||||
for (let i = 0; i < sectionEntity.sections.length; i++) {
|
||||
const subSectionEntity = sectionEntity.sections[i];
|
||||
this.ensureSectionFieldSetVisibility(subSectionEntity);
|
||||
}
|
||||
}
|
||||
if (sectionEntity.fieldSets != null) {
|
||||
for (let i = 0; i < sectionEntity.fieldSets.length; i++) {
|
||||
const fieldSetEntity = sectionEntity.fieldSets[i];
|
||||
let isHidden = true;
|
||||
const propertyDefinition = this.formService.getValue(this.form.value) as DescriptionPropertyDefinitionPersist;
|
||||
if (propertyDefinition.fieldSets != null && new Map(Object.entries(propertyDefinition.fieldSets)).has(fieldSetEntity.id)) {
|
||||
const propertyDefinitionFieldSet = new Map(Object.entries(propertyDefinition.fieldSets)).get(fieldSetEntity.id)
|
||||
if (propertyDefinitionFieldSet.items != null && propertyDefinitionFieldSet.items.length > 0) {
|
||||
for (let j = 0; j < propertyDefinitionFieldSet.items.length; j++) {
|
||||
const definitionFieldSetItem = propertyDefinitionFieldSet.items[j];
|
||||
const fieldKey = this.buildVisibilityKey(fieldSetEntity.id, definitionFieldSetItem.ordinal);
|
||||
const isCurrentHidden = !this.isVisibleMap[fieldKey] ?? false;
|
||||
isHidden = isHidden && isCurrentHidden;
|
||||
}
|
||||
}
|
||||
}
|
||||
const globalFieldSetKey = this.buildVisibilityKey(fieldSetEntity.id, null);
|
||||
this.isVisibleMap[globalFieldSetKey] = !isHidden;
|
||||
}
|
||||
}
|
||||
return super.get(key);
|
||||
}
|
||||
}
|
|
@ -147,7 +147,7 @@ export class TableOfContentsInternal implements OnInit {
|
|||
}
|
||||
if (entry.type === this.tocEntryTypeEnum.FieldSet) {
|
||||
const id = entry.id;
|
||||
if (!(this.visibilityRulesService.isVisibleMap[id.toString()] ?? true)) {
|
||||
if (!(this.visibilityRulesService.isVisibleMap[id.toString()])) {
|
||||
return false;
|
||||
}
|
||||
// const fieldsArray = entry.form.get('fields') as UntypedFormArray;
|
||||
|
|
|
@ -83,9 +83,9 @@ export class TableOfContentsComponent extends BaseComponent implements OnInit, O
|
|||
|
||||
if (this.descriptionTemplate) {
|
||||
this.tocentries = this.getTocEntries(this.descriptionTemplate);
|
||||
// if (this.visibilityRulesService) {
|
||||
// this.hiddenEntries = this._findHiddenEntries(this.tocentries);
|
||||
// }
|
||||
if (this.visibilityRulesService) {
|
||||
this.hiddenEntries = this._findHiddenEntries(this.tocentries);
|
||||
}
|
||||
|
||||
} else {
|
||||
|
||||
|
@ -157,15 +157,9 @@ export class TableOfContentsComponent extends BaseComponent implements OnInit, O
|
|||
const invisibleEntries: string[] = []
|
||||
tocentries.forEach(entry => {
|
||||
if (entry.type === ToCEntryType.FieldSet) {
|
||||
const isVisible = this.visibilityRulesService.isVisibleMap[entry.id.toString()] ?? true;
|
||||
const isVisible = this.visibilityRulesService.isVisibleMap[entry.id.toString()];
|
||||
if (!isVisible) {
|
||||
invisibleEntries.push(entry.id.toString());
|
||||
} else {
|
||||
//check field inputs
|
||||
const oneFieldAtLeastIsVisible = entry.subEntries.some(field => this.visibilityRulesService.isVisibleMap[field.id.toString()] ?? true);
|
||||
if (!oneFieldAtLeastIsVisible) {
|
||||
invisibleEntries.push(entry.id.toString());
|
||||
}
|
||||
}
|
||||
} else {
|
||||
const hiddenEntries = this._findHiddenEntries(entry.subEntries);
|
||||
|
@ -214,15 +208,15 @@ export class TableOfContentsComponent extends BaseComponent implements OnInit, O
|
|||
|
||||
if (!this.visibilityRulesService) return;
|
||||
|
||||
this._visibilityRulesSubscription = this.visibilityRulesService.visibilityChange
|
||||
.pipe(takeUntil(this._destroyed))
|
||||
.pipe(debounceTime(200))
|
||||
.subscribe(_ => {
|
||||
if (this.hasFocus) {
|
||||
this._resetObserver();
|
||||
// this.hiddenEntries = this._findHiddenEntries(this.tocentries);
|
||||
}
|
||||
});
|
||||
// this._visibilityRulesSubscription = this.visibilityRulesService.visibilityChange
|
||||
// .pipe(takeUntil(this._destroyed))
|
||||
// .pipe(debounceTime(200))
|
||||
// .subscribe(_ => {
|
||||
// if (this.hasFocus) {
|
||||
// this._resetObserver();
|
||||
// // this.hiddenEntries = this._findHiddenEntries(this.tocentries);
|
||||
// }
|
||||
// });
|
||||
// this.hiddenEntries = this._findHiddenEntries(this.tocentries);
|
||||
}
|
||||
// if (!this.isActive && this.links && this.links.length > 0) {
|
||||
|
|
|
@ -169,6 +169,7 @@ public class DescriptionTemplateXmlMigrationService {
|
|||
data.setOrdinal(persist.getOrdinal());
|
||||
data.setNumbering(persist.getNumbering());
|
||||
data.setTitle(persist.getTitle());
|
||||
data.setDescription(persist.getDescription());
|
||||
data.setExtendedDescription(persist.getExtendedDescription());
|
||||
data.setAdditionalInformation(persist.getAdditionalInformation());
|
||||
data.setHasCommentField(persist.getHasCommentField());
|
||||
|
|
Loading…
Reference in New Issue