17 changed files with 149 additions and 33 deletions
@ -1,4 +1,10 @@ |
|||||
install https://github.com/OpenIPC/python-dvr |
Features: |
||||
test on python 3.11 |
- python on backend server | full async support |
||||
|
- https://github.com/OpenIPC/python-dvr on main module |
||||
|
- Embed h264x to mp4 encoder https://git.pblr-nyk.pro/gsd/MiskaRisa264 |
||||
|
- Full windows / linux / docker support |
||||
|
- Embed go2rtc to stream |
||||
|
- Angular material on front |
||||
|
|
||||
npm add @angular/[email protected] @angular/cdk --force |
Use docker image to full component support |
||||
|
p.s replace registry to public or delete him from Dockerfile |
@ -0,0 +1,9 @@ |
|||||
|
.snapshot { |
||||
|
width: 100%; |
||||
|
} |
||||
|
|
||||
|
.center { |
||||
|
display: block; |
||||
|
margin-left: auto; |
||||
|
margin-right: auto; |
||||
|
} |
@ -1 +1,5 @@ |
|||||
<img style="width: 100%" [src]="'/api/dvrip/snapshot/' + recorder_index + '/' + channel_index"> |
<div class="center snapshot"> |
||||
|
<p style="text-align: center">Обновлено: {{updated.toLocaleString()}}</p> |
||||
|
<img class="snapshot" style="cursor: pointer" [src]="img" (click)="getSnapshot()"> |
||||
|
</div> |
||||
|
|
||||
|
@ -0,0 +1,13 @@ |
|||||
|
<div style="text-align: center"> |
||||
|
<mat-form-field> |
||||
|
<mat-label>Выбранный поток</mat-label> |
||||
|
<mat-select [(value)]="selected_stream" (valueChange)="getStreamUrl()"> |
||||
|
<mat-option [value]="0">Основной</mat-option> |
||||
|
<mat-option [value]="1">Дополнительный</mat-option> |
||||
|
</mat-select> |
||||
|
</mat-form-field> |
||||
|
<button mat-button (click)="getOpenNewWindow()">Открыть в новом окне</button> |
||||
|
</div> |
||||
|
<ng-container> |
||||
|
<iframe width="100%" height="100%" [src]="url"></iframe> |
||||
|
</ng-container> |
@ -0,0 +1,23 @@ |
|||||
|
import { ComponentFixture, TestBed } from '@angular/core/testing'; |
||||
|
|
||||
|
import { StreamComponent } from './stream.component'; |
||||
|
|
||||
|
describe('StreamComponent', () => { |
||||
|
let component: StreamComponent; |
||||
|
let fixture: ComponentFixture<StreamComponent>; |
||||
|
|
||||
|
beforeEach(async () => { |
||||
|
await TestBed.configureTestingModule({ |
||||
|
declarations: [ StreamComponent ] |
||||
|
}) |
||||
|
.compileComponents(); |
||||
|
|
||||
|
fixture = TestBed.createComponent(StreamComponent); |
||||
|
component = fixture.componentInstance; |
||||
|
fixture.detectChanges(); |
||||
|
}); |
||||
|
|
||||
|
it('should create', () => { |
||||
|
expect(component).toBeTruthy(); |
||||
|
}); |
||||
|
}); |
@ -0,0 +1,48 @@ |
|||||
|
import {Component, OnInit, Sanitizer} from '@angular/core'; |
||||
|
import {ActivatedRoute} from "@angular/router"; |
||||
|
import {HttpClient} from "@angular/common/http"; |
||||
|
import {DomSanitizer, SafeUrl} from "@angular/platform-browser"; |
||||
|
|
||||
|
interface StreamUrlResponse { |
||||
|
port:number, |
||||
|
route:string |
||||
|
} |
||||
|
|
||||
|
@Component({ |
||||
|
selector: 'app-stream', |
||||
|
templateUrl: './stream.component.html', |
||||
|
styleUrls: ['./stream.component.css'] |
||||
|
}) |
||||
|
export class StreamComponent implements OnInit { |
||||
|
|
||||
|
selected_stream:number = 1 |
||||
|
recorder_index:number = 0; |
||||
|
channel_index:number = 0; |
||||
|
url:SafeUrl|null = null; |
||||
|
unsafe_url:string = ""; |
||||
|
|
||||
|
constructor(private route:ActivatedRoute, |
||||
|
private http: HttpClient, |
||||
|
private sanitazer:DomSanitizer) {} |
||||
|
|
||||
|
ngOnInit(): void { |
||||
|
this.recorder_index = Number.parseInt(<string>this.route.snapshot.paramMap.get('recorderId')); |
||||
|
this.channel_index = Number.parseInt(<string>this.route.snapshot.paramMap.get('channelId')); |
||||
|
this.getStreamUrl(); |
||||
|
} |
||||
|
|
||||
|
getStreamUrl() { |
||||
|
this.http.get(`api/dvrip/stream/${this.recorder_index}/${this.channel_index}/${this.selected_stream}`).subscribe( |
||||
|
(r:StreamUrlResponse|any) => { |
||||
|
const url = `${location.protocol}//${location.hostname}:${r.port == 0?+location.port:r.port}/${r.route}`; |
||||
|
this.url = this.sanitazer.bypassSecurityTrustResourceUrl(url); |
||||
|
this.unsafe_url = url; |
||||
|
} |
||||
|
) |
||||
|
} |
||||
|
|
||||
|
getOpenNewWindow() { |
||||
|
window.open(this.unsafe_url); |
||||
|
} |
||||
|
|
||||
|
} |
Loading…
Reference in new issue