Files
alfresco-content-app/projects/aca-content/src/lib/components/info-drawer/library-metadata-tab/library-metadata-form.component.ts
dominikiwanekhyland 5f5c238ccd Ng19 migration (#4520)
2025-07-03 13:33:05 +02:00

231 lines
7.5 KiB
TypeScript

/*!
* Copyright © 2005-2025 Hyland Software, Inc. and its affiliates. All rights reserved.
*
* Alfresco Example Content Application
*
* This file is part of the Alfresco Example Content Application.
* If the software was purchased under a paid Alfresco license, the terms of
* the paid license agreement will prevail. Otherwise, the software is
* provided under the following open source license terms:
*
* The Alfresco Example Content Application is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* The Alfresco Example Content Application is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* from Hyland Software. If not, see <http://www.gnu.org/licenses/>.
*/
import { Component, DestroyRef, inject, Input, OnChanges, OnInit, ViewEncapsulation } from '@angular/core';
import {
FormControl,
FormGroupDirective,
FormsModule,
NgForm,
ReactiveFormsModule,
UntypedFormControl,
UntypedFormGroup,
ValidationErrors,
Validators
} from '@angular/forms';
import { QueriesApi, SiteEntry, SitePaging } from '@alfresco/js-api';
import { Store } from '@ngrx/store';
import { AppStore, isAdmin, UpdateLibraryAction } from '@alfresco/aca-shared/store';
import { AppHookService } from '@alfresco/aca-shared';
import { debounceTime, mergeMap } from 'rxjs/operators';
import { AlfrescoApiService } from '@alfresco/adf-content-services';
import { from, Observable } from 'rxjs';
import { ErrorStateMatcher, MatOptionModule } from '@angular/material/core';
import { CommonModule } from '@angular/common';
import { MatCardModule } from '@angular/material/card';
import { TranslateModule } from '@ngx-translate/core';
import { MatFormFieldModule } from '@angular/material/form-field';
import { MatSelectModule } from '@angular/material/select';
import { MatInputModule } from '@angular/material/input';
import { A11yModule } from '@angular/cdk/a11y';
import { MatButtonModule } from '@angular/material/button';
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
export class InstantErrorStateMatcher implements ErrorStateMatcher {
isErrorState(control: UntypedFormControl | null, form: FormGroupDirective | NgForm | null): boolean {
const isSubmitted = form?.submitted;
return !!(control?.invalid && (control?.dirty || control?.touched || isSubmitted));
}
}
@Component({
imports: [
CommonModule,
MatCardModule,
TranslateModule,
MatFormFieldModule,
FormsModule,
ReactiveFormsModule,
MatSelectModule,
MatOptionModule,
MatInputModule,
A11yModule,
MatButtonModule
],
selector: 'app-library-metadata-form',
templateUrl: './library-metadata-form.component.html',
styleUrls: ['./library-metadata-form.component.scss'],
encapsulation: ViewEncapsulation.None
})
export class LibraryMetadataFormComponent implements OnInit, OnChanges {
private _queriesApi: QueriesApi;
private _titleErrorTranslationKey: string;
get queriesApi(): QueriesApi {
this._queriesApi = this._queriesApi ?? new QueriesApi(this.alfrescoApiService.getInstance());
return this._queriesApi;
}
get titleErrorTranslationKey(): string {
return this._titleErrorTranslationKey;
}
@Input()
node: SiteEntry;
libraryTitleExists = false;
libraryType = [
{ value: 'PUBLIC', label: 'LIBRARY.VISIBILITY.PUBLIC' },
{ value: 'PRIVATE', label: 'LIBRARY.VISIBILITY.PRIVATE' },
{ value: 'MODERATED', label: 'LIBRARY.VISIBILITY.MODERATED' }
];
form: UntypedFormGroup = new UntypedFormGroup({
id: new UntypedFormControl({ value: '', disabled: true }),
title: new UntypedFormControl({ value: '' }, [Validators.required, Validators.maxLength(256), this.validateEmptyName]),
description: new UntypedFormControl({ value: '' }, [Validators.maxLength(512)]),
visibility: new UntypedFormControl(this.libraryType[0].value)
});
matcher = new InstantErrorStateMatcher();
canUpdateLibrary = false;
isAdmin = false;
private readonly destroyRef = inject(DestroyRef);
constructor(
private readonly alfrescoApiService: AlfrescoApiService,
protected readonly store: Store<AppStore>,
private readonly appHookService: AppHookService
) {}
toggleEdit() {
if (this.form.enabled) {
this.form.disable({
emitEvent: false
});
} else {
this.form.enable({
emitEvent: false
});
this.form.controls.id.disable();
}
}
cancel() {
this.updateForm(this.node);
this.toggleEdit();
this.form.markAsPristine();
}
ngOnInit() {
this.toggleEdit();
this.updateForm(this.node);
this.form.controls.title.statusChanges
.pipe(takeUntilDestroyed(this.destroyRef))
.subscribe(
() =>
(this._titleErrorTranslationKey = this.form.controls.title.errors?.empty
? 'LIBRARY.ERRORS.ONLY_SPACES'
: 'LIBRARY.ERRORS.TITLE_TOO_LONG_OR_MISSING')
);
this.form.controls['title'].valueChanges
.pipe(
debounceTime(300),
mergeMap((title) => this.findLibraryByTitle(title)),
takeUntilDestroyed(this.destroyRef)
)
.subscribe((result) => {
const { entries } = result.list;
if (entries.length) {
if (this.form.controls.title.value === this.node.entry.title) {
this.libraryTitleExists = false;
} else {
this.libraryTitleExists = this.form.controls.title.value === entries[0].entry.title;
}
} else {
this.libraryTitleExists = false;
}
});
this.store
.select(isAdmin)
.pipe(takeUntilDestroyed(this.destroyRef))
.subscribe((value) => {
this.isAdmin = value;
});
this.canUpdateLibrary = this.node?.entry?.role === 'SiteManager' || this.isAdmin;
this.appHookService.libraryUpdated.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => {
Object.assign(this.node.entry, this.form.value);
});
this.appHookService.libraryUpdateFailed.pipe(takeUntilDestroyed(this.destroyRef)).subscribe(() => {
this.form.markAsDirty();
});
}
ngOnChanges() {
this.updateForm(this.node);
this.canUpdateLibrary = this.node?.entry?.role === 'SiteManager' || this.isAdmin;
}
update() {
if (this.canUpdateLibrary && this.form.valid) {
this.form.markAsPristine();
this.store.dispatch(
new UpdateLibraryAction({
...this.form.value,
title: this.form.value.title.trim()
})
);
}
}
private updateForm(node: SiteEntry) {
const { entry } = node;
this.form.setValue({
id: entry.id,
title: entry.title,
description: entry.description || '',
visibility: entry.visibility
});
}
private findLibraryByTitle(libraryTitle: string): Observable<SitePaging | { list: { entries: any[] } }> {
return from(
this.queriesApi
.findSites(libraryTitle.trim(), {
maxItems: 1,
fields: ['title']
})
.catch(() => ({ list: { entries: [] } }))
);
}
private validateEmptyName(control: FormControl<string>): ValidationErrors {
return control.value.length && !control.value.trim() ? { empty: true } : null;
}
}