17 changed files with 149 additions and 33 deletions
@ -1,4 +1,10 @@ |
|||
install https://github.com/OpenIPC/python-dvr |
|||
test on python 3.11 |
|||
Features: |
|||
- 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