import { Component, OnInit, ViewContainerRef } from '@angular/core'; import { ActivatedRoute, Route, Router } from '@angular/router'; import { FormBuilder, FormControl, FormGroup, Validators } from '@angular/forms'; import { NzFormTooltipIcon } from 'ng-zorro-antd/form'; import { NzModalService } from 'ng-zorro-antd/modal'; import { AddcameraComponent } from './addcamera/addcamera.component'; import { NzMessageService } from 'ng-zorro-antd/message'; import { HttpClient, HttpErrorResponse, HttpResponse } from '@angular/common/http'; import { EditcameraComponent } from './editcamera/editcamera.component'; import { catchError, tap } from 'rxjs/operators'; import { ConfigFormDataService } from 'src/app/service/configFormData.service'; import { SendFileComponent } from './send-file/send-file.component'; import { ImageLabel2Component } from '../image-label2/image-label2.component'; interface Camera { name: string; user: string; password: string; uri: string; type: number; order: number; dimensionedPoints: string } @Component({ selector: 'app-host-config', templateUrl: './host-config.component.html', styleUrls: ['./host-config.component.scss'] }) export class HostConfigComponent implements OnInit { constructor(private router: Router, private route: ActivatedRoute, private fb: FormBuilder, private modal: NzModalService, private message: NzMessageService, private viewContainerRef: ViewContainerRef, private http: HttpClient, public configFormData: ConfigFormDataService,) { } hostId//主机id orId//加油站id ngOnInit(): void { this.hostId = this.route.snapshot.queryParams.hostId this.orId = this.route.snapshot.queryParams.orId this.getCamera() this.http.get(`/api/EdgeDevices/${this.hostId}`).subscribe( { next: ((data: any) => { console.log(data) // let isExist = data.configFiles.find((item, index, arr) => { // if (item.name == 'source.yaml') { // console.log("存在", index) // return item // } // }) if (data.configFiles) { this.isSourceYaml = true } else { this.isSourceYaml = false } }), error: (err) => { // this.message.create('error', '请先下发source.yaml配置'); } } ) } listOfData: Camera[] = []; goback() { history.go(-1) } //摄像头 isLoading = false getCamera() { let params = { ContainsChildren: true, EdgeDeviceId: this.hostId } this.isLoading = true this.http.get('/api/Cameras',{ params: params }).subscribe((data: any) => { data.items.forEach(element => { element.dimensionedPointsObj = JSON.parse(element.dimensionedPoints) }); this.listOfData = data.items this.isLoading = false console.log('摄像头列表', data.items) }) } addCamera() { const modal = this.modal.create({ nzTitle: '新增加油站摄像头', nzContent: AddcameraComponent, nzViewContainerRef: this.viewContainerRef, nzWidth: 388, nzComponentParams: {}, nzOnOk: async () => { if (instance.validateForm.valid) { await new Promise(resolve => { console.log('表单信息', instance.validateForm) let body = { name: instance.validateForm.value.name, user: instance.validateForm.value.user, password: instance.validateForm.value.password, uri: instance.validateForm.value.uri, type: instance.validateForm.value.type, organizationId: this.orId, edgeDeviceId: this.hostId, order: instance.validateForm.value.order, } this.http.post('/api/Cameras', body).subscribe(data => { resolve(data) this.message.create('success', '创建成功!'); this.getCamera() this.isSourceYaml = false return true }) }) } else { this.message.create('warning', '请填写完整!'); return false } } }); const instance = modal.getContentComponent(); } editCamera(data) { console.log(data) const modal = this.modal.create({ nzTitle: '编辑加油站摄像头', nzContent: EditcameraComponent, nzViewContainerRef: this.viewContainerRef, nzWidth: 388, nzComponentParams: { data: data }, nzOnOk: async () => { if (instance.validateForm.valid) { await new Promise(resolve => { console.log('表单信息', instance.validateForm) let body = { name: instance.validateForm.value.name, user: instance.validateForm.value.user, password: instance.validateForm.value.password, uri: instance.validateForm.value.uri, type: instance.validateForm.value.type, organizationId: this.orId, edgeDeviceId: this.hostId, order: instance.validateForm.value.order, } this.http.put(`/api/Cameras/${data.id}`, body).subscribe(data => { resolve(data) this.message.create('success', '编辑成功!'); this.getCamera() this.isSourceYaml = false return true }) }) } else { this.message.create('warning', '请填写完整!'); return false } } }); const instance = modal.getContentComponent(); } deleteCamera(item) { console.log(item) this.modal.confirm({ nzTitle: `确定要删除${item.name}这个摄像头吗?`, nzOkText: '确定', nzOkType: 'default', nzOnOk: () => { this.http.delete(`/api/Cameras/${item.id}`).subscribe(data => { this.message.create('success', '删除成功!'); this.getCamera() this.isSourceYaml = false }) }, nzCancelText: '取消' }); } isSourceYaml: boolean label(item) { this.http.get(`/api/EdgeDevices/${this.hostId}`).subscribe( { next: ((data: any) => { if (data.configFiles) { this.isSourceYaml = true const element = document.documentElement if(element.requestFullscreen) { //进入全屏 element.requestFullscreen(); } const modal = this.modal.create({ nzContent: ImageLabel2Component, nzViewContainerRef: this.viewContainerRef, nzWidth: 1920, nzClosable: false, nzFooter: null, nzWrapClassName: 'canvasContentBox', nzBodyStyle: { 'border-radius': '0px', 'padding': '0px', 'margin': '0px', }, nzComponentParams: { data: item.id }, nzOnOk: async () => { } }); const instance = modal.getContentComponent(); //this.router.navigate(['/system/host/camera/imageLabel'], { queryParams: { id: item.id } }) } else { this.isSourceYaml = false this.message.create('error', '请先下发source.yaml配置'); } }), error: (err) => { // this.message.create('error', '请先下发source.yaml配置'); } } ) } connect() { let isAllLabel = this.listOfData.find((item: any) => { if (item.type != 1 && !item.dimensionedPoints) { console.log('存在摄像头未标注的情况') return item } }) if (isAllLabel) { this.message.create('error', '存在摄像头未标注的情况'); return } let ids = [] this.listOfData.forEach((item: any) => { ids.push(item.id) }) this.http.get('/api/Cameras/Statuses', { params: { ids: ids } }).subscribe({ next: (data) => { console.log('连接状态', data) this.disposalData() this.router.navigate(['/system/host/camera/configForm'], { queryParams: { 'hostId': this.hostId, 'orId': this.orId } }) }, error: (err) => { console.log('连接失败', err) } }) } sourceYaml() { let config4 = `video_rate: 5 inference_buffer_second: 10 sources:` this.listOfData.forEach((item: any, index) => { config4 += ` - name: '${item.name}' user: '${item.user}' password: '${item.password}' uri: '${item.uri}' type: ${item.type} ` }) const modal = this.modal.create({ nzTitle: '下发source.yaml配置', nzContent: SendFileComponent, nzViewContainerRef: this.viewContainerRef, nzWidth: 1000, nzBodyStyle: { 'border-radius': '0px', 'padding': '7px', }, nzComponentParams: { data: config4 }, nzOnOk: async () => { await new Promise(resolve => { console.log('表单信息', instance.validateForm) let body = { configFiles: [ { name: 'source.yaml', content: instance.datacopy } ] } this.http.put(`/api/EdgeDevices/${this.hostId}`, body).subscribe({ next: (data) => { this.message.create('success', `文件保存成功`); resolve('成功了') this.isSourceYaml = true let params = { edgeDeviceId: this.hostId, fileName: 'source.yaml' } this.http.put('/api/EdgeDevices/Commands/PushFile', '', { params: params }).subscribe({ next: (data) => { this.message.create('success', `发送文件名成功`); }, error: err => { // this.message.create('error', `发送文件名失败`); // reject('失败了') } }) }, error: (err) => { // this.message.create('error', `文件保存失败`); } }) }) } }); const instance = modal.getContentComponent(); } //整理配置文件数据 disposalData() { console.log(this.listOfData) let config1 = `[property] enable=1 #Width height used for configuration to which below configs are configured config-width=1980 config-height=1080 #osd-mode 0: Dont display any lines, rois and text # 1: Display only lines, rois and static text i.e. labels # 2: Display all info from 1 plus information about counts osd-mode=2 #Set OSD font size that has to be displayed display-font-size=12 ` let config2 = '' let config3 = `test_action: true logging_interval : 600 ` let xieyouqu = [] this.listOfData.forEach((item: any, index) => { if (item.type == 2) { xieyouqu.push(item) } if (item.type == 0) { let str = '' item.dimensionedPointsObj.polygon.forEach(element => { str += element.x + ',' str += element.y + ',' }); str = str.substring(0, str.lastIndexOf(',')) console.log('进出口多边形', str) config1 += ` ## Per stream configuration [roi-filtering-stream-${item.order}] #shoushiyuan-out202 #enable or disable following feature enable=1 #ROI to filter select objects, and remove from meta data roi-RF=${str} #remove objects in the ROI inverse-roi=0 class-id=-1 ` } if ((item.type == 2 || item.type == 3) && item.dimensionedPointsObj) { let arrowArr = item.dimensionedPointsObj.arrow let str = arrowArr[0].startX + ',' + arrowArr[0].startY + ',' + arrowArr[0].endX + ',' + arrowArr[0].endY + ',' + arrowArr[1].startX + ',' + arrowArr[1].startY + ',' + arrowArr[1].endX + ',' + arrowArr[1].endY console.log('十字箭头', str) config1 += ` [line-crossing-stream-${item.order}] enable=1 #Label;direction;lc line-crossing-Entry=${str} # line-crossing-Exit=789;672;1084;900;851;773;1203;732 class-id=0 #extended when 0- only counts crossing on the configured Line # 1- assumes extended Line crossing counts all the crossing extended=0 #LC modes supported: #loose : counts all crossing without strong adherence to direction #balanced: Strict direction adherence expected compared to mode=loose #strict : Strict direction adherence expected compared to mode=balanced mode=strict ` } }) if (xieyouqu.length != 1) { this.message.create('warning', '有且只能有一个卸油区!'); return } else { let xieyouguan = '' let jingdian = '' xieyouqu[0].dimensionedPointsObj.rectangle.forEach(element => { if (element.oilUnloadingArea) { xieyouguan = element.x + ',' + element.y + ',' + element.width + ',' + element.height } else { jingdian = element.x + ',' + element.y + ',' + element.width + ',' + element.height } }); console.log('泄油管区域', xieyouguan) console.log('静电接地', jingdian) config2 = `# The all in one config file. # RTSP sources # type # 0 ViolateArea.ENTRANCE, # 1 ViolateArea.GAS_AREA, # 2 ViolateArea.FUEL_AREA, # 3 ViolateArea.MART, # don't change the key name. debug: true #when the debug is on, osd. sources: config: 'config/source.yaml' tracker: config: 'config/dstest_tracker_config.txt' analytics: config: 'config/config_nvdsanalytics.txt' peoplenet: enable: true apply_on: -1 interval: 1 batch_size: 16 topk: 5 roi-top-offset: 0 roi-bottom-offset: 0 detected-min-w: 20 detected-min-h: 200 trafficcam: enable: true apply_on: 0 interval: 1 batch_size: 16 topk: 5 roi-top-offset: 0 roi-bottom-offset: 0 detected-min-w: 100 detected-min-h: 100 actionnet: enable: false apply_on: 1 # roi: # - 'fuel_island-4': # - [200, 0, 450, 500] # - 'fuel_island-5': # - [930, 93, 940, 987] # - 'fuel_island-6': # - [1174, 151, 746, 929] # - 'fuel_island-7': # - [1450, 300, 460, 650] interval: 1 batch_size: 32 idnet: enable: true apply_on: -1 interval: 1 batch_size: 32 oilnet: enable: true apply_on: 2 interval: 1 batch_size: 16 roi-top-offset: 0 roi-bottom-offset: 0 detected-min-w: 20 detected-min-h: 20 fire_smoke_net: enable: true apply_on: -1 interval: 1 batch_size: 16 smoking_calling_net: enable: true apply_on: -1 interval: 1 batch_size: 32 connet: enable: true apply_on: 2 roi: - 'oil_tube-3': - [${xieyouguan}] - 'grounder-3': - [${jingdian}] interval: 1 batch_size: 2 #new field for rule threshold rule_threshold: object_occurence_interval_second: 15 object_disappear_interval_second: 60 on_car_parking_interval_second: 1800 on_fire_smoke_interval_second: 5 threshold_relying_sitting: 0.4 #rolling mean confidence threshold_smoking_calling: 0.5 #rolling mean confidence threshold_connecting: 0.667 #rolling mean confidence threshold_identity: 0.1 #only to filter out people net error ` } sessionStorage.setItem('config1', config1) sessionStorage.setItem('config2', config2) sessionStorage.setItem('config3', config3) } }