committed by
GitHub
8 changed files with 293 additions and 57 deletions
@ -0,0 +1,43 @@ |
|||
<!-- |
|||
|
|||
Copyright © 2016-2024 The Thingsboard Authors |
|||
|
|||
Licensed under the Apache License, Version 2.0 (the "License"); |
|||
you may not use this file except in compliance with the License. |
|||
You may obtain a copy of the License at |
|||
|
|||
http://www.apache.org/licenses/LICENSE-2.0 |
|||
|
|||
Unless required by applicable law or agreed to in writing, software |
|||
distributed under the License is distributed on an "AS IS" BASIS, |
|||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|||
See the License for the specific language governing permissions and |
|||
limitations under the License. |
|||
|
|||
--> |
|||
<mat-form-field [formGroup]="selectRuleChainGroup" class="mat-block"> |
|||
<input matInput type="text" placeholder="{{ 'rulechain.search' | translate }}" |
|||
#ruleChainInput |
|||
formControlName="ruleChainInput" |
|||
(focusin)="onFocus()" |
|||
[matAutocomplete]="ruleChainAutocomplete"> |
|||
<mat-icon matSuffix>search</mat-icon> |
|||
<mat-autocomplete class="tb-autocomplete tb-rule-chain-search" |
|||
#ruleChainAutocomplete="matAutocomplete" |
|||
(optionSelected)="selected($event)"> |
|||
<mat-option *ngFor="let ruleChain of filteredRuleChains | async" |
|||
[class.tb-selected-option]="ruleChainId === ruleChain.id.id" |
|||
[value]="ruleChain"> |
|||
<mat-icon *ngIf="ruleChainId === ruleChain.id.id">check</mat-icon> |
|||
<span [innerHTML]="ruleChain.name | highlight:searchText:true"></span> |
|||
</mat-option> |
|||
<mat-option *ngIf="!(filteredRuleChains | async)?.length" [value]="null" class="tb-not-found"> |
|||
<div class="tb-not-found-content" (click)="$event.stopPropagation()"> |
|||
<span> |
|||
{{ 'rulechain.no-rulechains-matching' | translate : {entity: searchText} }} |
|||
</span> |
|||
</div> |
|||
</mat-option> |
|||
</mat-autocomplete> |
|||
</mat-form-field> |
|||
|
|||
@ -0,0 +1,40 @@ |
|||
/** |
|||
* Copyright © 2016-2024 The Thingsboard Authors |
|||
* |
|||
* Licensed under the Apache License, Version 2.0 (the "License"); |
|||
* you may not use this file except in compliance with the License. |
|||
* You may obtain a copy of the License at |
|||
* |
|||
* http://www.apache.org/licenses/LICENSE-2.0 |
|||
* |
|||
* Unless required by applicable law or agreed to in writing, software |
|||
* distributed under the License is distributed on an "AS IS" BASIS, |
|||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|||
* See the License for the specific language governing permissions and |
|||
* limitations under the License. |
|||
*/ |
|||
@import "../../../../scss/constants"; |
|||
|
|||
:host { |
|||
width: 100%; |
|||
} |
|||
|
|||
::ng-deep { |
|||
.tb-autocomplete.tb-rule-chain-search { |
|||
.mat-mdc-option.tb-selected-option { |
|||
display: flex; |
|||
flex-direction: row-reverse; |
|||
background-color: rgba(0, 0, 0, .04); |
|||
|
|||
mat-icon { |
|||
margin-right: 0; |
|||
margin-left: 16px; |
|||
color: $tb-primary-color; |
|||
} |
|||
|
|||
.mdc-list-item__primary-text { |
|||
color: $tb-primary-color; |
|||
} |
|||
} |
|||
} |
|||
} |
|||
@ -0,0 +1,126 @@ |
|||
///
|
|||
/// Copyright © 2016-2024 The Thingsboard Authors
|
|||
///
|
|||
/// Licensed under the Apache License, Version 2.0 (the "License");
|
|||
/// you may not use this file except in compliance with the License.
|
|||
/// You may obtain a copy of the License at
|
|||
///
|
|||
/// http://www.apache.org/licenses/LICENSE-2.0
|
|||
///
|
|||
/// Unless required by applicable law or agreed to in writing, software
|
|||
/// distributed under the License is distributed on an "AS IS" BASIS,
|
|||
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|||
/// See the License for the specific language governing permissions and
|
|||
/// limitations under the License.
|
|||
///
|
|||
|
|||
import { AfterViewInit, Component, ElementRef, Inject, InjectionToken, ViewChild } from '@angular/core'; |
|||
import { FormBuilder, FormGroup } from '@angular/forms'; |
|||
import { Observable, of } from 'rxjs'; |
|||
import { catchError, debounceTime, distinctUntilChanged, map, share, startWith, switchMap } from 'rxjs/operators'; |
|||
import { PageLink } from '@shared/models/page/page-link'; |
|||
import { Direction } from '@shared/models/page/sort-order'; |
|||
import { emptyPageData, PageData } from '@shared/models/page/page-data'; |
|||
import { OverlayRef } from '@angular/cdk/overlay'; |
|||
import { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete'; |
|||
import { RuleChain, RuleChainType } from '@shared/models/rule-chain.models'; |
|||
import { RuleChainService } from '@core/http/rule-chain.service'; |
|||
|
|||
export const RULE_CHAIN_SELECT_PANEL_DATA = new InjectionToken<any>('RuleChainSelectPanelData'); |
|||
|
|||
export interface RuleChainSelectPanelData { |
|||
ruleChainId: string | null; |
|||
ruleChainType: RuleChainType; |
|||
} |
|||
|
|||
@Component({ |
|||
selector: 'tb-rule-chain-select-panel', |
|||
templateUrl: './rule-chain-select-panel.component.html', |
|||
styleUrls: ['./rule-chain-select-panel.component.scss'] |
|||
}) |
|||
export class RuleChainSelectPanelComponent implements AfterViewInit { |
|||
|
|||
ruleChainId: string; |
|||
private readonly ruleChainType: RuleChainType; |
|||
|
|||
selectRuleChainGroup: FormGroup; |
|||
|
|||
@ViewChild('ruleChainInput', {static: true}) userInput: ElementRef; |
|||
|
|||
filteredRuleChains: Observable<Array<RuleChain>>; |
|||
|
|||
searchText = ''; |
|||
|
|||
ruleChainSelected = false; |
|||
|
|||
result?: RuleChain; |
|||
|
|||
private dirty = false; |
|||
|
|||
constructor(@Inject(RULE_CHAIN_SELECT_PANEL_DATA) public data: RuleChainSelectPanelData, |
|||
private overlayRef: OverlayRef, |
|||
private fb: FormBuilder, |
|||
private ruleChainService: RuleChainService) { |
|||
this.ruleChainId = data.ruleChainId; |
|||
this.ruleChainType = data.ruleChainType; |
|||
this.selectRuleChainGroup = this.fb.group({ |
|||
ruleChainInput: ['', {nonNullable: true}] |
|||
}); |
|||
this.filteredRuleChains = this.selectRuleChainGroup.get('ruleChainInput').valueChanges |
|||
.pipe( |
|||
debounceTime(150), |
|||
startWith(''), |
|||
distinctUntilChanged((a: string, b: string) => a.trim() === b.trim()), |
|||
switchMap(name => this.fetchRuleChains(name)), |
|||
share() |
|||
); |
|||
} |
|||
|
|||
ngAfterViewInit() { |
|||
setTimeout(() => { |
|||
this.userInput.nativeElement.focus(); |
|||
}, 0); |
|||
} |
|||
|
|||
selected(event: MatAutocompleteSelectedEvent): void { |
|||
this.clear(); |
|||
this.ruleChainSelected = true; |
|||
if (event.option.value?.id) { |
|||
this.result = event.option.value; |
|||
} |
|||
this.overlayRef.dispose(); |
|||
} |
|||
|
|||
private fetchRuleChains(searchText?: string): Observable<Array<RuleChain>> { |
|||
this.searchText = searchText; |
|||
const pageLink = new PageLink(50, 0, searchText, { |
|||
property: 'name', |
|||
direction: Direction.ASC |
|||
}); |
|||
return this.getRuleChains(pageLink) |
|||
.pipe( |
|||
catchError(() => of(emptyPageData<RuleChain>())), |
|||
map(pageData => pageData.data) |
|||
); |
|||
} |
|||
|
|||
onFocus(): void { |
|||
if (!this.dirty) { |
|||
this.selectRuleChainGroup.get('ruleChainInput').updateValueAndValidity({onlySelf: true}); |
|||
this.dirty = true; |
|||
} |
|||
} |
|||
|
|||
clear() { |
|||
this.selectRuleChainGroup.get('ruleChainInput').patchValue('', {emitEvent: true}); |
|||
setTimeout(() => { |
|||
this.userInput.nativeElement.blur(); |
|||
this.userInput.nativeElement.focus(); |
|||
}, 0); |
|||
} |
|||
|
|||
private getRuleChains(pageLink: PageLink): Observable<PageData<RuleChain>> { |
|||
return this.ruleChainService.getRuleChains(pageLink, this.ruleChainType, {ignoreLoading: true}); |
|||
} |
|||
|
|||
} |
|||
Loading…
Reference in new issue