@netgrif/components-core
Version:
Netgrif Application engine frontend core Angular library
170 lines • 21.8 kB
JavaScript
import { Component, Inject, Input, ViewChild } from '@angular/core';
import { NAE_SEARCH_CATEGORIES } from '../category-factory/search-categories-injection-token';
import { Query } from '../models/query/query';
import * as i0 from "@angular/core";
import * as i1 from "../../logger/services/logger.service";
import * as i2 from "../advanced-search-component-initialization-service/advanced-search-component-initialization.service";
import * as i3 from "../category-factory/category-factory";
/**
* Is responsible for the interactive creation of a single {@link ElementaryPredicate} object instance.
*/
export class AbstractSearchPredicateComponent {
_naeSearchCategories;
_logger;
_initializationService;
_categoryFactory;
/**
* Whether the contents displayed in this component can be edited by the user or not.
*
* Defaults to `true`
*/
editable = true;
predicate;
predicateId;
remove$;
/**
* Optional generator with prefilled values. Can be used to add prefilled predicate components to the search GUI.
*/
generator;
_selectedCategory;
_predicateChange;
_searchCategories;
constructor(_naeSearchCategories, _logger, _initializationService, _categoryFactory) {
this._naeSearchCategories = _naeSearchCategories;
this._logger = _logger;
this._initializationService = _initializationService;
this._categoryFactory = _categoryFactory;
}
ngOnInit() {
let found = false;
this._searchCategories = this._naeSearchCategories.map(category => {
// if the provided generator is the same class as one of the injected search categories
if (this.generator && this.generator.constructor === category) {
found = true;
this.categoryChanged(this.generator);
return this.generator;
}
return this._categoryFactory.get(category);
});
if (this.generator && !found) {
this._logger.error('Provided predicate generator is not an allowed category from the NAE_SEARCH_CATEGORIES injection token!'
+ ' Behavior in this case is undefined.');
}
this.predicate.setMetadataGenerator(() => {
if (!!this._selectedCategory) {
return this._selectedCategory.createMetadata();
}
return undefined;
});
this.predicate.setFilterTextSegmentsGenerator(() => {
if (!!this._selectedCategory) {
return this._selectedCategory.createFilterTextSegments();
}
return [];
});
}
ngOnDestroy() {
if (this._predicateChange && !this._predicateChange.closed) {
this._predicateChange.unsubscribe();
}
this._searchCategories.forEach(cat => {
if (cat !== this.generator) {
cat.destroy();
}
});
}
get searchCategories() {
return this._searchCategories;
}
set categoryInput(input) {
if (input && this._initializationService.isInitialized) {
setTimeout(() => {
input.focus();
input.open();
});
}
}
get selectedCategory() {
return this._selectedCategory;
}
set selectedCategory(newCategory) {
this.categoryChanged(newCategory);
}
/**
* Lambda that is used to preserve `this` reference in HTML binding.
*
* See [_renderSelection()]{@link AbstractSearchPredicateComponent#_renderSelection} for information about the transform function.
* @param option the {@link SearchAutocompleteOption} object that was selected in the autocomplete list.
*/
renderSelection = (option) => this._renderSelection(option);
remove() {
this.remove$.next(this.predicateId);
}
categoryChanged(newCategory) {
if (this.selectedCategory !== undefined) {
this.selectedCategory.reset();
}
this._selectedCategory = newCategory;
if (newCategory !== undefined) {
if (this._predicateChange) {
this._predicateChange.unsubscribe();
}
this._predicateChange = newCategory.generatedPredicate$.subscribe(predicate => this.processPredicateChange(predicate));
}
}
clearCategorySelection() {
if (!this.editable) {
return;
}
this.categoryChanged(undefined);
}
/**
* Transforms a {@link SearchAutocompleteOption} object into it's name.
* Used for displaying user selection in the input field, when an autocomplete option is selected.
* @param option the object we want to transform. It might not exist if user input doesn't match any autocomplete option
* @returns option name if it exists, empty string otherwise
*/
_renderSelection(option) {
return option ? option.text : '';
}
/**
* Notifies the `EditableElementaryPredicate` about changes to the predicate generated by the selected `Category`.
* @param newPredicate predicate generated by the selected `Category`
*/
processPredicateChange(newPredicate) {
if (newPredicate === undefined) {
this.predicate.query = Query.emptyQuery();
this._logger.debug('Editable query changed to empty query');
}
else {
this.predicate.query = newPredicate.query;
this._logger.debug(`Editable query changed to: ${newPredicate.query.value}`);
}
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AbstractSearchPredicateComponent, deps: [{ token: NAE_SEARCH_CATEGORIES }, { token: i1.LoggerService }, { token: i2.AdvancedSearchComponentInitializationService }, { token: i3.CategoryFactory }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: AbstractSearchPredicateComponent, selector: "ncc-abstract-search-predicate", inputs: { editable: "editable", predicate: "predicate", predicateId: "predicateId", remove$: "remove$", generator: "generator" }, viewQueries: [{ propertyName: "categoryInput", first: true, predicate: ["categoryInput"], descendants: true }], ngImport: i0, template: '', isInline: true });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AbstractSearchPredicateComponent, decorators: [{
type: Component,
args: [{
selector: 'ncc-abstract-search-predicate',
template: ''
}]
}], ctorParameters: () => [{ type: Array, decorators: [{
type: Inject,
args: [NAE_SEARCH_CATEGORIES]
}] }, { type: i1.LoggerService }, { type: i2.AdvancedSearchComponentInitializationService }, { type: i3.CategoryFactory }], propDecorators: { editable: [{
type: Input
}], predicate: [{
type: Input
}], predicateId: [{
type: Input
}], remove$: [{
type: Input
}], generator: [{
type: Input
}], categoryInput: [{
type: ViewChild,
args: ['categoryInput']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstract-search-predicate.component.js","sourceRoot":"","sources":["../../../../../../projects/netgrif-components-core/src/lib/search/search-predicate-component/abstract-search-predicate.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {Component, Inject, Input, OnDestroy, OnInit, Type, ViewChild} from '@angular/core';\nimport {Subject, Subscription} from 'rxjs';\nimport {NAE_SEARCH_CATEGORIES} from '../category-factory/search-categories-injection-token';\nimport {Category} from '../models/category/category';\nimport {SearchAutocompleteOption} from '../models/category/search-autocomplete-option';\nimport {ElementaryPredicate} from '../models/predicate/elementary-predicate';\nimport {Query} from '../models/query/query';\nimport {LoggerService} from '../../logger/services/logger.service';\nimport {MatSelect} from '@angular/material/select';\nimport {EditableElementaryPredicate} from '../models/predicate/editable-elementary-predicate';\nimport {\n    AdvancedSearchComponentInitializationService\n} from '../advanced-search-component-initialization-service/advanced-search-component-initialization.service';\nimport {CategoryFactory} from '../category-factory/category-factory';\n\n\n/**\n * Is responsible for the interactive creation of a single {@link ElementaryPredicate} object instance.\n */\n@Component({\n    selector: 'ncc-abstract-search-predicate',\n    template: ''\n})\nexport abstract class AbstractSearchPredicateComponent implements OnInit, OnDestroy {\n\n    /**\n     * Whether the contents displayed in this component can be edited by the user or not.\n     *\n     * Defaults to `true`\n     */\n    @Input() editable = true;\n    @Input() predicate: EditableElementaryPredicate;\n    @Input() predicateId: number;\n    @Input() remove$: Subject<number>;\n    /**\n     * Optional generator with prefilled values. Can be used to add prefilled predicate components to the search GUI.\n     */\n    @Input() generator: Category<any> | undefined;\n\n    protected _selectedCategory: Category<any>;\n\n    protected _predicateChange: Subscription;\n\n    protected _searchCategories: Array<Category<any>>;\n\n    protected constructor(@Inject(NAE_SEARCH_CATEGORIES) private _naeSearchCategories: Array<Type<Category<any>>>,\n                          protected _logger: LoggerService,\n                          protected _initializationService: AdvancedSearchComponentInitializationService,\n                          protected _categoryFactory: CategoryFactory) {\n    }\n\n    ngOnInit() {\n        let found = false;\n        this._searchCategories = this._naeSearchCategories.map(category => {\n            // if the provided generator is the same class as one of the injected search categories\n            if (this.generator && this.generator.constructor === category) {\n                found = true;\n                this.categoryChanged(this.generator);\n                return this.generator;\n            }\n            return this._categoryFactory.get(category);\n        });\n\n        if (this.generator && !found) {\n            this._logger.error('Provided predicate generator is not an allowed category from the NAE_SEARCH_CATEGORIES injection token!'\n                + ' Behavior in this case is undefined.');\n        }\n\n        this.predicate.setMetadataGenerator(() => {\n            if (!!this._selectedCategory) {\n                return this._selectedCategory.createMetadata();\n            }\n            return undefined;\n        });\n\n        this.predicate.setFilterTextSegmentsGenerator(() => {\n            if (!!this._selectedCategory) {\n                return this._selectedCategory.createFilterTextSegments();\n            }\n            return [];\n        });\n    }\n\n    ngOnDestroy(): void {\n        if (this._predicateChange && !this._predicateChange.closed) {\n            this._predicateChange.unsubscribe();\n        }\n        this._searchCategories.forEach(cat => {\n            if (cat !== this.generator) {\n                cat.destroy();\n            }\n        });\n    }\n\n    public get searchCategories(): Array<Category<any>> {\n        return this._searchCategories;\n    }\n\n    @ViewChild('categoryInput')\n    public set categoryInput(input: MatSelect) {\n        if (input && this._initializationService.isInitialized) {\n            setTimeout(() => {\n                input.focus();\n                input.open();\n            });\n        }\n    }\n\n    public get selectedCategory(): Category<any> {\n        return this._selectedCategory;\n    }\n\n    public set selectedCategory(newCategory: Category<any>) {\n        this.categoryChanged(newCategory);\n    }\n\n    /**\n     * Lambda that is used to preserve `this` reference in HTML binding.\n     *\n     * See [_renderSelection()]{@link AbstractSearchPredicateComponent#_renderSelection} for information about the transform function.\n     * @param option the {@link SearchAutocompleteOption} object that was selected in the autocomplete list.\n     */\n    public renderSelection = (option: SearchAutocompleteOption<unknown>) => this._renderSelection(option);\n\n    public remove(): void {\n        this.remove$.next(this.predicateId);\n    }\n\n    public categoryChanged(newCategory: Category<any>): void {\n        if (this.selectedCategory !== undefined) {\n            this.selectedCategory.reset();\n        }\n        this._selectedCategory = newCategory;\n        if (newCategory !== undefined) {\n            if (this._predicateChange) {\n                this._predicateChange.unsubscribe();\n            }\n            this._predicateChange = newCategory.generatedPredicate$.subscribe(predicate => this.processPredicateChange(predicate));\n        }\n    }\n\n    public clearCategorySelection(): void {\n        if (!this.editable) {\n            return;\n        }\n\n        this.categoryChanged(undefined);\n    }\n\n    /**\n     * Transforms a {@link SearchAutocompleteOption} object into it's name.\n     * Used for displaying user selection in the input field, when an autocomplete option is selected.\n     * @param option the object we want to transform. It might not exist if user input doesn't match any autocomplete option\n     * @returns option name if it exists, empty string otherwise\n     */\n    protected _renderSelection(option: SearchAutocompleteOption<unknown>): string {\n        return option ? option.text : '';\n    }\n\n    /**\n     * Notifies the `EditableElementaryPredicate` about changes to the predicate generated by the selected `Category`.\n     * @param newPredicate predicate generated by the selected `Category`\n     */\n    protected processPredicateChange(newPredicate: ElementaryPredicate | undefined): void {\n        if (newPredicate === undefined) {\n            this.predicate.query = Query.emptyQuery();\n            this._logger.debug('Editable query changed to empty query');\n        } else {\n            this.predicate.query = newPredicate.query;\n            this._logger.debug(`Editable query changed to: ${newPredicate.query.value}`);\n        }\n    }\n}\n"]}