12 changed files with 24 additions and 261 deletions
@ -1,49 +0,0 @@ |
|||
<!-- |
|||
|
|||
Copyright © 2016-2022 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]="queueFormGroup" class="mat-block"> |
|||
<mat-label>{{ 'queue.queue-name' | translate }}</mat-label> |
|||
<input matInput type="text" placeholder="{{ 'queue.select-name' | translate }}" |
|||
#queueInput |
|||
formControlName="queue" |
|||
(focusin)="onFocus()" |
|||
[required]="required" |
|||
[matAutocomplete]="queueAutocomplete"> |
|||
<button *ngIf="queueFormGroup.get('queue').value && !disabled" |
|||
type="button" |
|||
matSuffix mat-button mat-icon-button aria-label="Clear" |
|||
(click)="clear()"> |
|||
<mat-icon class="material-icons">close</mat-icon> |
|||
</button> |
|||
<mat-autocomplete |
|||
class="tb-autocomplete" |
|||
#queueAutocomplete="matAutocomplete" |
|||
[displayWith]="displayQueueFn"> |
|||
<mat-option *ngFor="let queue of filteredQueues | async" [value]="queue"> |
|||
<span [innerHTML]="queue.queueName | highlight:searchText"></span> |
|||
</mat-option> |
|||
<mat-option *ngIf="!(filteredQueues | async)?.length" [value]="null"> |
|||
<span> |
|||
{{ translate.get('queue.no-queues-matching', {queue: searchText}) | async }} |
|||
</span> |
|||
</mat-option> |
|||
</mat-autocomplete> |
|||
<mat-error *ngIf="queueFormGroup.get('queue').hasError('required')"> |
|||
{{ 'queue.name-required' | translate }} |
|||
</mat-error> |
|||
<mat-hint class="tb-hint" translate [fxShow]="!disabled">device-profile.select-queue-hint</mat-hint> |
|||
</mat-form-field> |
|||
@ -1,199 +0,0 @@ |
|||
///
|
|||
/// Copyright © 2016-2022 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, forwardRef, Input, OnDestroy, OnInit, ViewChild } from '@angular/core'; |
|||
import { ControlValueAccessor, FormBuilder, FormGroup, NG_VALUE_ACCESSOR } from '@angular/forms'; |
|||
import { Observable, of } from 'rxjs'; |
|||
import { |
|||
catchError, |
|||
debounceTime, |
|||
distinctUntilChanged, |
|||
map, |
|||
publishReplay, |
|||
refCount, |
|||
switchMap, |
|||
tap |
|||
} from 'rxjs/operators'; |
|||
import { Store } from '@ngrx/store'; |
|||
import { AppState } from '@app/core/core.state'; |
|||
import { TranslateService } from '@ngx-translate/core'; |
|||
import { coerceBooleanProperty } from '@angular/cdk/coercion'; |
|||
import { QueueService } from '@core/http/queue.service'; |
|||
import { ServiceType } from '@shared/models/queue.models'; |
|||
|
|||
interface Queue { |
|||
queueName: string; |
|||
} |
|||
|
|||
@Component({ |
|||
selector: 'tb-queue-type-list', |
|||
templateUrl: './queue-type-list.component.html', |
|||
styleUrls: [], |
|||
providers: [{ |
|||
provide: NG_VALUE_ACCESSOR, |
|||
useExisting: forwardRef(() => QueueTypeListComponent), |
|||
multi: true |
|||
}] |
|||
}) |
|||
export class QueueTypeListComponent implements ControlValueAccessor, OnInit, AfterViewInit, OnDestroy { |
|||
|
|||
queueFormGroup: FormGroup; |
|||
|
|||
modelValue: Queue | null; |
|||
|
|||
private requiredValue: boolean; |
|||
get required(): boolean { |
|||
return this.requiredValue; |
|||
} |
|||
@Input() |
|||
set required(value: boolean) { |
|||
this.requiredValue = coerceBooleanProperty(value); |
|||
} |
|||
|
|||
@Input() |
|||
disabled: boolean; |
|||
|
|||
@Input() |
|||
queueType: ServiceType; |
|||
|
|||
@ViewChild('queueInput', {static: true}) queueInput: ElementRef<HTMLInputElement>; |
|||
|
|||
filteredQueues: Observable<Array<Queue>>; |
|||
|
|||
queues: Observable<Array<Queue>>; |
|||
|
|||
searchText = ''; |
|||
|
|||
private dirty = false; |
|||
|
|||
private propagateChange = (v: any) => { }; |
|||
|
|||
constructor(private store: Store<AppState>, |
|||
public translate: TranslateService, |
|||
private queueService: QueueService, |
|||
private fb: FormBuilder) { |
|||
this.queueFormGroup = this.fb.group({ |
|||
queue: [null] |
|||
}); |
|||
} |
|||
|
|||
registerOnChange(fn: any): void { |
|||
this.propagateChange = fn; |
|||
} |
|||
|
|||
registerOnTouched(fn: any): void { |
|||
} |
|||
|
|||
ngOnInit() { |
|||
this.filteredQueues = this.queueFormGroup.get('queue').valueChanges |
|||
.pipe( |
|||
debounceTime(150), |
|||
distinctUntilChanged(), |
|||
tap(value => { |
|||
let modelValue; |
|||
if (typeof value === 'string' || !value) { |
|||
modelValue = null; |
|||
} else { |
|||
modelValue = value; |
|||
} |
|||
this.updateView(modelValue); |
|||
}), |
|||
map(value => value ? (typeof value === 'string' ? value : value.queueName) : ''), |
|||
switchMap(queue => this.fetchQueues(queue) ) |
|||
); |
|||
} |
|||
|
|||
ngAfterViewInit(): void { |
|||
} |
|||
|
|||
ngOnDestroy(): void { |
|||
} |
|||
|
|||
setDisabledState(isDisabled: boolean): void { |
|||
this.disabled = isDisabled; |
|||
if (this.disabled) { |
|||
this.queueFormGroup.disable({emitEvent: false}); |
|||
} else { |
|||
this.queueFormGroup.enable({emitEvent: false}); |
|||
} |
|||
} |
|||
|
|||
writeValue(value: string | null): void { |
|||
this.searchText = ''; |
|||
this.modelValue = value ? { queueName: value } : null; |
|||
this.queueFormGroup.get('queue').patchValue(this.modelValue, {emitEvent: false}); |
|||
this.dirty = true; |
|||
} |
|||
|
|||
onFocus() { |
|||
if (this.dirty) { |
|||
this.queueFormGroup.get('queue').updateValueAndValidity({onlySelf: true, emitEvent: true}); |
|||
this.dirty = false; |
|||
} |
|||
} |
|||
|
|||
updateView(value: Queue | null) { |
|||
if (this.modelValue !== value) { |
|||
this.modelValue = value; |
|||
this.propagateChange(this.modelValue ? this.modelValue.queueName : null); |
|||
} |
|||
} |
|||
|
|||
displayQueueFn(queue?: Queue): string | undefined { |
|||
return queue ? queue.queueName : undefined; |
|||
} |
|||
|
|||
fetchQueues(searchText?: string): Observable<Array<Queue>> { |
|||
this.searchText = searchText; |
|||
return this.getQueues().pipe( |
|||
catchError(() => of([] as Array<Queue>)), |
|||
map(queues => { |
|||
const result = queues.filter( queue => { |
|||
return searchText ? queue.queueName.toUpperCase().startsWith(searchText.toUpperCase()) : true; |
|||
}); |
|||
if (result.length) { |
|||
result.sort((q1, q2) => q1.queueName.localeCompare(q2.queueName)); |
|||
} |
|||
return result; |
|||
}) |
|||
); |
|||
} |
|||
|
|||
getQueues(): Observable<Array<Queue>> { |
|||
if (!this.queues) { |
|||
this.queues = this.queueService. |
|||
getTenantQueuesNamesByServiceType(this.queueType, {ignoreLoading: true}).pipe( |
|||
map((queues) => { |
|||
return queues.map((queueName) => { |
|||
return { queueName }; |
|||
}); |
|||
}), |
|||
publishReplay(1), |
|||
refCount() |
|||
); |
|||
} |
|||
return this.queues; |
|||
} |
|||
|
|||
clear() { |
|||
this.queueFormGroup.get('queue').patchValue(null, {emitEvent: true}); |
|||
setTimeout(() => { |
|||
this.queueInput.nativeElement.blur(); |
|||
this.queueInput.nativeElement.focus(); |
|||
}, 0); |
|||
} |
|||
|
|||
} |
|||
Loading…
Reference in new issue