import 'dart:typed_data'; import 'package:vid/us/vid_us_data_http_reader.dart'; import 'package:vid/us/vid_us_data_reader.dart'; import 'package:vid/us/vid_us_image.dart'; import 'package:vid/us/vid_us_probe.dart'; enum VidUsImageFormat { Jpeg, Png, H264, Zip, Diff, } ///Only suport read for now. ///Not suport carotid3d for now. class VidUsImageData { late VidUsDataReader _reader; late List _imagePositionList; late int _version; late int _imageCount; late VidUsProbe _probe; late VidUsImageFormat _imageFormat; late Uint8List _extendedData; final String header = "VINNO IMAGE DATA"; /// Gets the version of this image data. int get version => _version; /// Gets the image count of this image data. int get imageCount => _imageCount; /// Gets the probe information. VidUsProbe get probe => _probe; /// Gets the image format of this image data. VidUsImageFormat get imageFormat => _imageFormat; /// Gets or sets the extended data. Uint8List get extendedData => _extendedData; /// Create a VINNO Image Data. VidUsImageData(Uint8List data) { _reader = VidUsDataReader(data); var header = _reader.readString(); if (header != header) { throw Exception("The input data is not a VID data."); } _version = _reader.readInt(); //Get probe info var probeData = _reader.readBytes(); _probe = VidUsProbe.fromBytes(probeData); _imageFormat = VidUsImageFormat.values[_reader.readInt()]; _extendedData = _reader.readBytes(); _imagePositionList = []; var imagePositionListData = _reader.readBytes(); _imageCount = imagePositionListData.length ~/ 8; var imagePositionReader = VidUsDataReader(imagePositionListData); for (var i = 0; i < _imageCount; i++) { _imagePositionList.add(imagePositionReader.readInt64V2()); } } /// Get one image from the vid. VidUsImage getImage(int index) { if (index >= _imageCount || index < 0) { throw Exception("Can not find image Data"); } //Jump to image. var imageData = _reader.readBytes(_imagePositionList[index]); return VidUsImage.fromBytes(imageData); } } ///Raised when getting data from the downloaded data timeout. depends on the readHeaderTimeout and ///readImageTimeout parameters. class ReadTimeoutException implements Exception {} ///Raised when the Http operation already closed. class AlreadyClosedException implements Exception {} ///Only suport read for now. ///Not suport carotid3d for now. class HttpVidUsImageData { late VidUsDataHttpReader _reader; late List _imagePositionList; late int _version; late int _imageCount; late VidUsProbe _probe; late VidUsImageFormat _imageFormat; late Uint8List _extendedData; late int _readHeaderTimeout; late int _readImageTimeout; late bool _initialized; late bool _closed; final Duration delayDuration = const Duration(milliseconds: 10); final String header = "VINNO IMAGE DATA"; /// Gets the version of this image data. int get version => _version; /// Gets the image count of this image data. int get imageCount => _imageCount; /// Gets the probe information. VidUsProbe get probe => _probe; /// Gets the image format of this image data. VidUsImageFormat get imageFormat => _imageFormat; /// Gets the extended data. Uint8List get extendedData => _extendedData; /// Gets the initialized state. bool get initialized => _initialized; /// Create a VINNO Image Data. /// [readHeaderTimeout] is the timeout value of reading header and probe information etc..., we need give as much time as possible /// to let the reader read the header. /// [readImageTimeout] is the timeout value of reading one frame. /// [minChunkSize] The min download chunk size, set to a large value may speedup the download speed, but will cause the progress not smooth. HttpVidUsImageData(String url, {DownloadCallback? downloadCallback, int minChunkSize = 65536, int readHeaderTimeout = 3000, int readImageTimeout = 500}) { _initialized = false; _closed = false; _readHeaderTimeout = readHeaderTimeout; _readImageTimeout = readImageTimeout; _reader = VidUsDataHttpReader(url, downloadCallback: downloadCallback, minChunkSize: minChunkSize); } ///Get the downloaded data of this HttpVidUsImageData Uint8List getDownloadedData() { return _reader.toBytes(); } ///Initialize the HttpVidUsImageData, must be called firstly. Future initialize() async { if (!_initialized) { var header = await _readHeader(); if (header != header) { throw Exception("The input data is not a VID data."); } _version = await _readVersion(); //Get probe info _probe = await _readProbe(); _imageFormat = await _readImageFormat(); _extendedData = await _readExtendedData(); _imagePositionList = []; var imagePositionListData = await _readImagePositionListData(); _imageCount = imagePositionListData.length ~/ 8; var imagePositionReader = VidUsDataReader(imagePositionListData); for (var i = 0; i < _imageCount; i++) { _imagePositionList.add(imagePositionReader.readInt64V2()); } _initialized = true; } } ///Close the HttpVidUsImageData, it will force close the download operation and reading operation. void close() { _closed = true; _reader.close(); } ///Read header from http data. Future _readHeader() async { var timeout = 0; while (!_closed) { try { var header = _reader.readString(); return header; } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } ///Read the version from http data. Future _readVersion() async { var timeout = 0; while (!_closed) { try { var version = _reader.readInt(); return version; } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } ///Read the probe info from http data. Future _readProbe() async { var timeout = 0; while (!_closed) { try { var probeData = _reader.readBytes(); return VidUsProbe.fromBytes(probeData); } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } ///Read the image format from http data. Future _readImageFormat() async { var timeout = 0; while (!_closed) { try { return VidUsImageFormat.values[_reader.readInt()]; } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } ///Read extended data from http data. Future _readExtendedData() async { var timeout = 0; while (!_closed) { try { return _reader.readBytes(); } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } ///Read the image positions data from http data. Future _readImagePositionListData() async { var timeout = 0; while (!_closed) { try { return _reader.readBytes(); } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readHeaderTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } /// Get one image from the vid. Future getImage(int index) async { if (index >= _imageCount || index < 0) { throw Exception("Can not find image Data"); } //Jump to image. var timeout = 0; while (!_closed) { try { var imageData = _reader.readBytes(_imagePositionList[index]); return VidUsImage.fromBytes(imageData); } catch (ex) { if (ex is NotReadyException) { if (timeout >= _readImageTimeout) { throw ReadTimeoutException(); } await Future.delayed(delayDuration); timeout += delayDuration.inMilliseconds; } else { rethrow; } } } throw AlreadyClosedException(); } }