This repository has been archived on 2023-11-18. You can view files and clone it, but cannot push or open issues or pull requests.
ha_client/lib/entities/camera/widgets/camera_stream_view.dart
2020-02-20 22:42:19 +00:00

139 lines
3.9 KiB
Dart

part of '../../../main.dart';
class CameraStreamView extends StatefulWidget {
CameraStreamView({Key key}) : super(key: key);
@override
_CameraStreamViewState createState() => _CameraStreamViewState();
}
class _CameraStreamViewState extends State<CameraStreamView> {
CameraEntity _entity;
String streamUrl = "";
WebViewController webViewController;
VideoPlayerController videoPlayerController;
Timer monitorTimer;
bool started = false;
double aspectRatio = 1.33;
@override
void initState() {
super.initState();
}
void loadStreamUrl() {
Logger.d("[Camera Player] Loading stream url");
HomeAssistant().getCameraStream(_entity.entityId)
.then((data) {
Logger.d("[Camera Player] Stream url: ${ConnectionManager().httpWebHost}${data["url"]}");
if (videoPlayerController != null) {
videoPlayerController.dispose().then((_) => createPlayer(data));
} else {
createPlayer(data);
}
})
.catchError((e) => Logger.e("[Camera Player] $e"));
}
void createPlayer(data) {
videoPlayerController = VideoPlayerController.network("${ConnectionManager().httpWebHost}${data["url"]}");
videoPlayerController.initialize().then((_) {
setState((){
started = true;
aspectRatio = videoPlayerController.value.aspectRatio;
});
autoPlay();
startMonitor();
}).catchError((e) {
Logger.e("[Camera Player] Error player init. Retrying");
loadStreamUrl();
});
}
void autoPlay() {
if (!videoPlayerController.value.isPlaying) {
videoPlayerController.play();
}
}
void startMonitor() {
monitorTimer = Timer.periodic(Duration(milliseconds: 500), (timer) {
if (videoPlayerController.value.hasError) {
setState(() {
timer.cancel();
started = false;
});
}
});
}
Widget buildLoading() {
return AspectRatio(
aspectRatio: aspectRatio,
child: Center(
child: CircularProgressIndicator()
)
);
}
@override
Widget build(BuildContext context) {
_entity = EntityModel
.of(context)
.entityWrapper
.entity;
if (_entity.supportStream && !started) {
loadStreamUrl();
return buildLoading();
} else if (_entity.supportStream) {
if (videoPlayerController.value.initialized) {
return AspectRatio(
aspectRatio: aspectRatio,
child: VideoPlayer(videoPlayerController),
);
} else {
return buildLoading();
}
} else {
streamUrl = '${ConnectionManager().httpWebHost}/api/camera_proxy_stream/${_entity
.entityId}?token=${_entity.attributes['access_token']}';
return AspectRatio(
aspectRatio: aspectRatio,
child: WebView(
initialUrl: streamUrl,
initialMediaPlaybackPolicy: AutoMediaPlaybackPolicy.always_allow,
debuggingEnabled: Logger.isInDebugMode,
gestureNavigationEnabled: false,
javascriptMode: JavascriptMode.unrestricted,
javascriptChannels: {
JavascriptChannel(
name: 'HA_${_entity.entityId.replaceAll('.', '_')}',
onMessageReceived: ((message) {
setState((){
aspectRatio = double.tryParse(message.message) ?? 1.33;
});
})
)
},
onWebViewCreated: (WebViewController controller) {
webViewController = controller;
},
onPageStarted: (url) {
rootBundle.loadString('assets/js/cameraImgViewHelper.js').then((js){
webViewController.evaluateJavascript(js.replaceFirst('entity_id_placeholder', _entity.entityId.replaceAll('.', '_')));
});
},
),
);
}
}
@override
void dispose() {
monitorTimer?.cancel();
videoPlayerController?.dispose();
super.dispose();
}
}