UNPKG

@netgrif/components-core

Version:

Netgrif Application engine frontend core Angular library

170 lines 21.8 kB
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"]}