From 39e209a294baf819f3faf5cdcdf1b117aaa7c96d Mon Sep 17 00:00:00 2001 From: Dovydas Vabalas Date: Mon, 21 Oct 2024 21:52:19 -0700 Subject: [PATCH 1/8] Initial metadata implementation using TiffWriter --- image_utils.py | 66 ++++++++++++++++++++++++++++++++++++++++++++++++ lumascope_api.py | 15 +++++++++-- 2 files changed, 79 insertions(+), 2 deletions(-) diff --git a/image_utils.py b/image_utils.py index 8f35294..30d9674 100644 --- a/image_utils.py +++ b/image_utils.py @@ -192,6 +192,72 @@ def write_ome_tiff( **ome_tiff_support_data['options'] ) +def write_tiff( + data, + file_loc: pathlib.Path, + metadata: dict +): + # Note: OpenCV and TIFFFILE have the Red/Blue color planes swapped, so need to swap + # them before writing out to OME tiff + use_color = image_utils.is_color_image(data) + if use_color: + data = cv2.cvtColor(data, cv2.COLOR_BGR2RGB) + + tiff_support_data = generate_tiff_data(data=data, metadata=metadata) + + with tf.TiffWriter(str(file_loc), bigTiff=False) as tif: + tif.write( + data, + resolution=tiff_support_data['resolution'], + metadata=tiff_support_data['metadata'] + ) + +def generate_tiff_data(data, metadata: dict): + use_color = image_utils.is_color_image(data) + + if use_color: + photometric = 'rgb' + + else: + photometric = 'minisblack' + + """ + To Add: + ImageNumber + LensModel + + """ + + + tiff_metadata = { + "CameraMake": metadata['camera_make'], + "ExposureTime": metadata['exposure_time_ms'], + "GainControl": metadata['gain_db'], + "DateTime": metadata['datetime'], + "Software": metadata['software'], + "XPosition": metadata['x_pos'], + "YPosition": metadata['y_pos'], + "SubjectDistance": metadata['z_pos_um'], + "SubSecTime": metadata['sub_sec_time'], + "LightSource": metadata['channel'], + "BrightnessValue": metadata['illumination_ma'] +} + + options=dict( + photometric=photometric, + tile=(128, 128), + compression='lzw', + resolutionunit='CENTIMETER', + maxworkers=2 + ) + + resolution = (1e4 / metadata['pixel_size_um'], 1e4 / metadata['pixel_size_um']) + + return { + 'metadata': tiff_metadata, + 'options': options, + 'resolution': resolution, + } def add_scale_bar( image, diff --git a/lumascope_api.py b/lumascope_api.py index f167c5b..be274af 100644 --- a/lumascope_api.py +++ b/lumascope_api.py @@ -42,7 +42,7 @@ from pyloncamera import PylonCamera # Import additional libraries -from lvp_logger import logger +from lvp_logger import logger, version import modules.common_utils as common_utils import modules.coord_transformations as coord_transformations import modules.objectives_loader as objectives_loader @@ -398,9 +398,15 @@ def _validate(): ) metadata = { + 'camera_make': 'Etaluma', + 'software': f'LumaViewPro {version}', 'channel': color, + 'datetime': datetime.datetime.now().strftime("%Y:%m:%d %H:%M:%S"), # Format for metadata + 'sub_sec_time': f"{datetime.datetime.now().microsecond // 1000:03d}", 'focal_length': self._objective['focal_length'], 'plate_pos_mm': {'x': px, 'y': py}, + 'x_pos': px, + 'y_pos': py, 'z_pos_um': z, 'exposure_time_ms': round(self.get_exposure_time(), common_utils.max_decimal_precision('exposure')), 'gain_db': round(self.get_gain(), common_utils.max_decimal_precision('gain')), @@ -484,7 +490,12 @@ def save_image( metadata=metadata, ) else: - cv2.imwrite(str(file_loc), image.astype(array.dtype)) + image_utils.write_tiff( + data=image, + file_loc=file_loc, + metadata=metadata + ) + #cv2.imwrite(str(file_loc), image.astype(array.dtype)) logger.info(f'[SCOPE API ] Saving Image to {file_loc}') except: From 66037b1a05e793d3f073885fe9d9972219670704 Mon Sep 17 00:00:00 2001 From: Dovydas Vabalas Date: Thu, 24 Oct 2024 15:20:31 -0700 Subject: [PATCH 2/8] Changed VideoFrame writing to be consistent with normal image saving; Added metadata to video frames --- image_utils.py | 12 ++++++++++++ modules/sequenced_capture_executor.py | 11 +++++++++-- 2 files changed, 21 insertions(+), 2 deletions(-) diff --git a/image_utils.py b/image_utils.py index 30d9674..7208cac 100644 --- a/image_utils.py +++ b/image_utils.py @@ -6,6 +6,7 @@ import modules.common_utils as common_utils import image_utils +import datetime from lvp_logger import logger @@ -259,6 +260,17 @@ def generate_tiff_data(data, metadata: dict): 'resolution': resolution, } +# Takes in datetime object and frame_num to generate metadata for videos saved as frames +def generate_video_frame_metadata(time_taken, frame_num): + date_time_data = time_taken.strftime("%Y:%m:%d %H:%M:%S") + sub_sec_time = f"{time_taken.microsecond // 1000:03d}" + + return { + "DateTime": date_time_data, + "SubSecTime": sub_sec_time, + "ImageNumber": frame_num + } + def add_scale_bar( image, objective: dict, diff --git a/modules/sequenced_capture_executor.py b/modules/sequenced_capture_executor.py index ee0b9df..b7df7d1 100644 --- a/modules/sequenced_capture_executor.py +++ b/modules/sequenced_capture_executor.py @@ -745,9 +745,16 @@ def _capture( output_file_loc = save_folder / f"{frame_name}.tiff" - if not cv2.imwrite(filename=str(output_file_loc), img=image): + frame_metadata = image_utils.generate_video_frame_metadata(ts, frame_num) + + try: + image_utils.write_tiff(image, str(output_file_loc), frame_metadata) + except Exception as e: + logger.error(f"Protocol-Video] Failed to write frame {frame_num}: {e}") + + """if not cv2.imwrite(filename=str(output_file_loc), img=image): logger.error(f"Protocol-Video] Failed to write frame {frame_num}") - +""" else: video_writer = VideoWriter( output_file_loc=output_file_loc, From 8928cd7387aa14fc54bac9e965fbb41a71e5b52b Mon Sep 17 00:00:00 2001 From: Jordan Date: Fri, 25 Oct 2024 07:49:39 -0700 Subject: [PATCH 3/8] Combine write methods for OME and TIFF files. --- image_utils.py | 147 +++++++++----------------- lumascope_api.py | 24 ++--- modules/sequenced_capture_executor.py | 7 +- 3 files changed, 65 insertions(+), 113 deletions(-) diff --git a/image_utils.py b/image_utils.py index 7208cac..3db2b9d 100644 --- a/image_utils.py +++ b/image_utils.py @@ -115,112 +115,37 @@ def convert_16bit_to_8bit(image): return (new_image/256).astype('uint8') -def generate_ome_tiff_support_data(data, metadata: dict): - use_color = image_utils.is_color_image(data) - - if use_color: - photometric = 'rgb' - axes = 'YXS' - else: - photometric = 'minisblack' - axes = 'YX' - - ome_metadata={ - 'axes': axes, - 'SignificantBits': data.itemsize*8, - 'PhysicalSizeX': metadata['pixel_size_um'], - 'PhysicalSizeXUnit': 'µm', - 'PhysicalSizeY': metadata['pixel_size_um'], - 'PhysicalSizeYUnit': 'µm', - 'Channel': {'Name': [metadata['channel']]}, - 'Plane': { - 'PositionX': metadata['plate_pos_mm']['x'], - 'PositionY': metadata['plate_pos_mm']['y'], - 'PositionZ': metadata['z_pos_um'], - 'PositionXUnit': 'mm', - 'PositionYUnit': 'mm', - 'PositionZUnit': 'um', - 'ExposureTime': metadata['exposure_time_ms'], - 'ExposureTimeUnit': 'ms', - 'Gain': metadata['gain_db'], - 'GainUnit': 'dB', - 'Illumination': metadata['illumination_ma'], - 'IlluminationUnit': 'mA' - } - } - - options=dict( - photometric=photometric, - tile=(128, 128), - compression='lzw', - resolutionunit='CENTIMETER', - maxworkers=2 - ) - - resolution = (1e4 / metadata['pixel_size_um'], 1e4 / metadata['pixel_size_um']) - - return { - 'metadata': ome_metadata, - 'options': options, - 'resolution': resolution, - } - - -def write_ome_tiff( - data, - file_loc: pathlib.Path, - metadata: dict, -): - # Note: OpenCV and TIFFFILE have the Red/Blue color planes swapped, so need to swap - # them before writing out to OME tiff - use_color = image_utils.is_color_image(data) - if use_color: - data = cv2.cvtColor(data, cv2.COLOR_BGR2RGB) - - # if use_color: - # photometric = 'rgb' - # axes = 'YXS' - # else: - # photometric = 'minisblack' - # axes = 'YX' - ome_tiff_support_data = generate_ome_tiff_support_data(data=data, metadata=metadata) - - with tf.TiffWriter(str(file_loc), bigtiff=False) as tif: - tif.write( - data, - resolution=ome_tiff_support_data['resolution'], - metadata=ome_tiff_support_data['metadata'], - **ome_tiff_support_data['options'] - ) - def write_tiff( data, file_loc: pathlib.Path, - metadata: dict + metadata: dict, + ome: bool, ): # Note: OpenCV and TIFFFILE have the Red/Blue color planes swapped, so need to swap - # them before writing out to OME tiff + # them before writing out to tiff use_color = image_utils.is_color_image(data) if use_color: data = cv2.cvtColor(data, cv2.COLOR_BGR2RGB) - tiff_support_data = generate_tiff_data(data=data, metadata=metadata) + support_data = generate_tiff_data(data=data, metadata=metadata, ome=ome) with tf.TiffWriter(str(file_loc), bigTiff=False) as tif: tif.write( data, - resolution=tiff_support_data['resolution'], - metadata=tiff_support_data['metadata'] + resolution=support_data['resolution'], + metadata=support_data['metadata'], + **support_data['options'], ) -def generate_tiff_data(data, metadata: dict): +def generate_tiff_data(data, metadata: dict, ome: bool): use_color = image_utils.is_color_image(data) if use_color: photometric = 'rgb' - + axes = 'YXS' else: photometric = 'minisblack' + axes = 'YX' """ To Add: @@ -228,21 +153,45 @@ def generate_tiff_data(data, metadata: dict): LensModel """ + if True == ome: + tiff_metadata={ + 'axes': axes, + 'SignificantBits': data.itemsize*8, + 'PhysicalSizeX': metadata['pixel_size_um'], + 'PhysicalSizeXUnit': 'µm', + 'PhysicalSizeY': metadata['pixel_size_um'], + 'PhysicalSizeYUnit': 'µm', + 'Channel': {'Name': [metadata['channel']]}, + 'Plane': { + 'PositionX': metadata['plate_pos_mm']['x'], + 'PositionY': metadata['plate_pos_mm']['y'], + 'PositionZ': metadata['z_pos_um'], + 'PositionXUnit': 'mm', + 'PositionYUnit': 'mm', + 'PositionZUnit': 'um', + 'ExposureTime': metadata['exposure_time_ms'], + 'ExposureTimeUnit': 'ms', + 'Gain': metadata['gain_db'], + 'GainUnit': 'dB', + 'Illumination': metadata['illumination_ma'], + 'IlluminationUnit': 'mA' + } + } - - tiff_metadata = { - "CameraMake": metadata['camera_make'], - "ExposureTime": metadata['exposure_time_ms'], - "GainControl": metadata['gain_db'], - "DateTime": metadata['datetime'], - "Software": metadata['software'], - "XPosition": metadata['x_pos'], - "YPosition": metadata['y_pos'], - "SubjectDistance": metadata['z_pos_um'], - "SubSecTime": metadata['sub_sec_time'], - "LightSource": metadata['channel'], - "BrightnessValue": metadata['illumination_ma'] -} + else: + tiff_metadata={ + "CameraMake": metadata['camera_make'], + "ExposureTime": metadata['exposure_time_ms'], + "GainControl": metadata['gain_db'], + "DateTime": metadata['datetime'], + "Software": metadata['software'], + "XPosition": metadata['x_pos'], + "YPosition": metadata['y_pos'], + "SubjectDistance": metadata['z_pos_um'], + "SubSecTime": metadata['sub_sec_time'], + "LightSource": metadata['channel'], + "BrightnessValue": metadata['illumination_ma'] + } options=dict( photometric=photometric, diff --git a/lumascope_api.py b/lumascope_api.py index be274af..19e2999 100644 --- a/lumascope_api.py +++ b/lumascope_api.py @@ -482,20 +482,18 @@ def save_image( metadata = image_data['metadata'] file_loc = metadata['file_loc'] + if output_format == 'OME-TIFF': + ome=True + else: + ome=False + try: - if output_format == 'OME-TIFF': - image_utils.write_ome_tiff( - data=image, - file_loc=file_loc, - metadata=metadata, - ) - else: - image_utils.write_tiff( - data=image, - file_loc=file_loc, - metadata=metadata - ) - #cv2.imwrite(str(file_loc), image.astype(array.dtype)) + image_utils.write_tiff( + data=image, + file_loc=file_loc, + metadata=metadata, + ome=ome, + ) logger.info(f'[SCOPE API ] Saving Image to {file_loc}') except: diff --git a/modules/sequenced_capture_executor.py b/modules/sequenced_capture_executor.py index b7df7d1..4386eb7 100644 --- a/modules/sequenced_capture_executor.py +++ b/modules/sequenced_capture_executor.py @@ -748,7 +748,12 @@ def _capture( frame_metadata = image_utils.generate_video_frame_metadata(ts, frame_num) try: - image_utils.write_tiff(image, str(output_file_loc), frame_metadata) + image_utils.write_tiff( + data=image, + file_loc=output_file_loc, + metadata=frame_metadata, + ome=False, + ) except Exception as e: logger.error(f"Protocol-Video] Failed to write frame {frame_num}: {e}") From 9d2418cfbd59189d9cd005a78021b1929b07e95b Mon Sep 17 00:00:00 2001 From: Jordan Date: Fri, 25 Oct 2024 08:21:40 -0700 Subject: [PATCH 4/8] Savepoint --- image_utils.py | 9 ++++++++- requirements.txt | 2 +- 2 files changed, 9 insertions(+), 2 deletions(-) diff --git a/image_utils.py b/image_utils.py index 3db2b9d..41f8870 100644 --- a/image_utils.py +++ b/image_utils.py @@ -129,7 +129,14 @@ def write_tiff( support_data = generate_tiff_data(data=data, metadata=metadata, ome=ome) - with tf.TiffWriter(str(file_loc), bigTiff=False) as tif: + if True == ome: + kwargs = { + 'bigtiff': False + } + else: + kwargs = {} + + with tf.TiffWriter(str(file_loc), **kwargs) as tif: tif.write( data, resolution=support_data['resolution'], diff --git a/requirements.txt b/requirements.txt index 1c3a85a..4da8044 100644 --- a/requirements.txt +++ b/requirements.txt @@ -14,5 +14,5 @@ pypylon==3.0.1 pyserial==3.5 scikit-image==0.22.0 scipy==1.12.0 -tifffile==2024.2.12 +tifffile==2024.9.20 userpaths==0.1.3 From 386f61141b04a95f5104dece79306920f77a5249 Mon Sep 17 00:00:00 2001 From: Dovydas Vabalas Date: Mon, 28 Oct 2024 16:54:36 -0700 Subject: [PATCH 5/8] Attempt to pass Tiff metadata directly as EXIF tags to fix issues --- image_utils.py | 138 ++++++++++++++++++++------ modules/sequenced_capture_executor.py | 8 +- 2 files changed, 114 insertions(+), 32 deletions(-) diff --git a/image_utils.py b/image_utils.py index 41f8870..7fed264 100644 --- a/image_utils.py +++ b/image_utils.py @@ -7,6 +7,7 @@ import modules.common_utils as common_utils import image_utils import datetime +from fractions import Fraction from lvp_logger import logger @@ -120,6 +121,8 @@ def write_tiff( file_loc: pathlib.Path, metadata: dict, ome: bool, + video_frame: bool = False, + extratags: list = [], ): # Note: OpenCV and TIFFFILE have the Red/Blue color planes swapped, so need to swap # them before writing out to tiff @@ -127,7 +130,7 @@ def write_tiff( if use_color: data = cv2.cvtColor(data, cv2.COLOR_BGR2RGB) - support_data = generate_tiff_data(data=data, metadata=metadata, ome=ome) + support_data = generate_tiff_data(data=data, metadata=metadata, ome=ome, video_frame=video_frame) if True == ome: kwargs = { @@ -137,14 +140,22 @@ def write_tiff( kwargs = {} with tf.TiffWriter(str(file_loc), **kwargs) as tif: - tif.write( - data, - resolution=support_data['resolution'], - metadata=support_data['metadata'], - **support_data['options'], - ) + if ome: + tif.write( + data, + resolution=support_data['resolution'], + metadata=support_data['metadata'], + **support_data['options'], + ) + else: + tif.write( + data, + resolution=support_data['resolution'], + extratags=support_data['extratags'], + **support_data['options'] + ) -def generate_tiff_data(data, metadata: dict, ome: bool): +def generate_tiff_data(data, metadata: dict, ome: bool, video_frame: bool): use_color = image_utils.is_color_image(data) if use_color: @@ -184,21 +195,86 @@ def generate_tiff_data(data, metadata: dict, ome: bool): 'IlluminationUnit': 'mA' } } + tiff_extratags = [] + # Metadata seems to be working properly for OME-TIFF's so let's leave it alone for now. else: - tiff_metadata={ - "CameraMake": metadata['camera_make'], - "ExposureTime": metadata['exposure_time_ms'], - "GainControl": metadata['gain_db'], - "DateTime": metadata['datetime'], - "Software": metadata['software'], - "XPosition": metadata['x_pos'], - "YPosition": metadata['y_pos'], - "SubjectDistance": metadata['z_pos_um'], - "SubSecTime": metadata['sub_sec_time'], - "LightSource": metadata['channel'], - "BrightnessValue": metadata['illumination_ma'] - } + if not video_frame: + tiff_metadata={ + "CameraMake": metadata['camera_make'], + "ExposureTime": metadata['exposure_time_ms'], + "ISOSpeed": metadata['gain_db'], + "DateTime": metadata['datetime'], + "Software": metadata['software'], + "XPosition": metadata['x_pos'], + "YPosition": metadata['y_pos'], + "SubjectDistance": metadata['z_pos_um'], + "SubSecTime": metadata['sub_sec_time'], + #"LightSource": metadata['channel'], + "BrightnessValue": metadata['illumination_ma'] + } + + # Format: (tag_number, datatype, count, value, write_ifd) + # For rational number values: (numerator, denominator) + + tiff_extratags = [ + # CameraMake: Tag ID 271, 'ascii' + (271, 'ascii', len(metadata['camera_make']) + 1, metadata['camera_make'], False), + + # ExposureTime: Tag ID 33434, 'RATIONAL' + (33434, 'RATIONAL', 1, ms_exposure_to_rational(metadata['exposure_time_ms']), False), + + # ISOSpeed: Tag ID 34867, 'double' + # Using in place of GainControl (Improper use of GainControl) + (34867, 'double', 1, metadata['gain_db'], False), + + # DateTime: Tag ID 306, 'ascii' + (306, 'ascii', len(metadata['datetime']) + 1, metadata['datetime'], False), + + # Software: Tag ID 305, 'ascii' + (305, 'ascii', len(metadata['software']) + 1, metadata['software'], False), + + # XPosition: Tag ID 65001, 'RATIONAL' + # Need to double check units + (286, 'RATIONAL', 1, (metadata['x_pos'], 1), False), + + # YPosition: Custom Tag ID 65002, 'RATIONAL' + # Need to double check units + (287, 'RATIONAL', 1, (metadata['y_pos'], 1), False), + + # SubjectDistance: Tag ID 37386, 'RATIONAL' + (37386, 'RATIONAL', 1, subject_dist_to_rational(metadata['z_pos_um']), False), + + # SubSecTime: Tag ID 37520, 'ascii' + (37520, 'ascii', len(metadata['sub_sec_time']) + 1, metadata['sub_sec_time'], False), + + # Channel: Tag ID 65001, 'ascii' **CUSTOM** + (65001, 'ascii', len(metadata['channel']) + 1, metadata['channel'], False), + + # BrightnessValue: Tag ID 37393, 'SRATIONAL' + (37393, 'SRATIONAL', 1, (metadata['illumination_ma'], 1), False) + + ] + else: + # Video Frame + # Add further parameters in the future if testing goes well + + date_time_data = metadata['timestamp'].strftime("%Y:%m:%d %H:%M:%S") + sub_sec_time = f"{metadata['timestamp'].microsecond // 1000:03d}" + + tiff_extratags = [ + # Tag 306: DateTime (ASCII) + (306, 'ascii', len(date_time_data) + 1, date_time_data, False), + + # Tag 37520: SubSecTime (ASCII) + (37520, 'ascii', len(sub_sec_time) + 1, sub_sec_time, False), + + # Tag 37393: ImageNumber (long) + (37393, 'long', 1, metadata['frame_num'], False) + + ] + + options=dict( photometric=photometric, @@ -212,20 +288,22 @@ def generate_tiff_data(data, metadata: dict, ome: bool): return { 'metadata': tiff_metadata, + 'extratags': tiff_extratags, 'options': options, 'resolution': resolution, } -# Takes in datetime object and frame_num to generate metadata for videos saved as frames -def generate_video_frame_metadata(time_taken, frame_num): - date_time_data = time_taken.strftime("%Y:%m:%d %H:%M:%S") - sub_sec_time = f"{time_taken.microsecond // 1000:03d}" +def ms_exposure_to_rational(ms_exposure): + exposure_seconds = ms_exposure / 1000 + fraction = Fraction(exposure_seconds).limit_denominator(1_000_000) + # Metadata uses rational number of seconds + return fraction.numerator, fraction.denominator + +def subject_dist_to_rational(distance): + distance_meters = distance / 1_000_000 # Convert um to m + fraction = Fraction(distance_meters).limit_denominator(1_000_000) + return fraction.numerator, fraction.denominator - return { - "DateTime": date_time_data, - "SubSecTime": sub_sec_time, - "ImageNumber": frame_num - } def add_scale_bar( image, diff --git a/modules/sequenced_capture_executor.py b/modules/sequenced_capture_executor.py index 4386eb7..1c862e7 100644 --- a/modules/sequenced_capture_executor.py +++ b/modules/sequenced_capture_executor.py @@ -745,13 +745,17 @@ def _capture( output_file_loc = save_folder / f"{frame_name}.tiff" - frame_metadata = image_utils.generate_video_frame_metadata(ts, frame_num) + metadata = { + "timestamp": ts, + "frame_num": frame_num + } try: image_utils.write_tiff( data=image, + metadata=metadata, file_loc=output_file_loc, - metadata=frame_metadata, + video_frame=True, ome=False, ) except Exception as e: From 79ad7009d23f7b59d119cf30366bbf660db5305a Mon Sep 17 00:00:00 2001 From: jmcoreymv Date: Thu, 31 Oct 2024 13:30:17 -0700 Subject: [PATCH 6/8] Add support for camera emulation (#400) --- pyloncamera.py | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/pyloncamera.py b/pyloncamera.py index 972e4e5..b685529 100644 --- a/pyloncamera.py +++ b/pyloncamera.py @@ -36,6 +36,7 @@ import contextlib import datetime +import os import numpy as np from pypylon import pylon, genicam @@ -49,6 +50,13 @@ def __init__(self, **kwargs): self.error_report_count = 0 self.array = np.array([]) self.cam_image_handler = None + + if os.getenv("PYLON_CAMEMU", None) != None: + logger.info('[CAM Class ] PylonCamera.connect() detected request to use camera emulation') + self._use_camera_emulation = True + else: + self._use_camera_emulation = False + self.connect() def __delete__(self): @@ -136,7 +144,8 @@ def init_camera_config(self): self.set_pixel_format(pixel_format='Mono8') self.auto_gain(state=False) camera.ReverseX.SetValue(True) - self.init_auto_gain_focus() + if not self._use_camera_emulation: + self.init_auto_gain_focus() self.exposure_t(t=10) self.set_frame_size(w=1900, h=1900) From 3ea5d1985703edf471417925002a5dde144f9597 Mon Sep 17 00:00:00 2001 From: Dovydas Vabalas Date: Wed, 6 Nov 2024 13:38:05 -0800 Subject: [PATCH 7/8] Simplified metadata --- image_utils.py | 143 +++++++++++++++++++------- lumaviewpro.py | 18 +++- modules/sequenced_capture_executor.py | 3 +- 3 files changed, 121 insertions(+), 43 deletions(-) diff --git a/image_utils.py b/image_utils.py index 7fed264..22ca436 100644 --- a/image_utils.py +++ b/image_utils.py @@ -9,8 +9,26 @@ import datetime from fractions import Fraction -from lvp_logger import logger - +from lvp_logger import logger, version + +# Conversion to tifffile's desired datatype references +tifffile_dtypes = { + 'BYTE': 1, + 'ASCII': 2, + 'SHORT': 3, + 'LONG': 4, + 'RATIONAL': 5, + 'SBYTE': 6, + 'UNDEFINED': 7, + 'SSHORT': 8, + 'SLONG': 9, + 'SRATIONAL': 10, + 'FLOAT': 11, + 'DOUBLE': 12, + 'SINGLE': 13, + 'QWORD': 16, + 'SQWORD': 17, +} def is_color_image(image) -> bool: if len(image.shape) == 3 and image.shape[2] == 3: @@ -124,6 +142,7 @@ def write_tiff( video_frame: bool = False, extratags: list = [], ): + # Note: OpenCV and TIFFFILE have the Red/Blue color planes swapped, so need to swap # them before writing out to tiff use_color = image_utils.is_color_image(data) @@ -140,22 +159,29 @@ def write_tiff( kwargs = {} with tf.TiffWriter(str(file_loc), **kwargs) as tif: - if ome: + if not video_frame: tif.write( data, resolution=support_data['resolution'], metadata=support_data['metadata'], + datetime=metadata['datetime'], + software=f"LumaViewPro {version}", **support_data['options'], ) else: tif.write( data, - resolution=support_data['resolution'], - extratags=support_data['extratags'], - **support_data['options'] + metadata=support_data['metadata'], + datetime=metadata['datetime'], + software=f"LumaViewPro {version}", + **support_data['options'], ) + def generate_tiff_data(data, metadata: dict, ome: bool, video_frame: bool): + + dtype = tifffile_dtypes + use_color = image_utils.is_color_image(data) if use_color: @@ -210,56 +236,83 @@ def generate_tiff_data(data, metadata: dict, ome: bool, video_frame: bool): "YPosition": metadata['y_pos'], "SubjectDistance": metadata['z_pos_um'], "SubSecTime": metadata['sub_sec_time'], - #"LightSource": metadata['channel'], + "Channel": metadata['channel'], "BrightnessValue": metadata['illumination_ma'] } - + # extratags: + # Additional tags to write. A list of tuples with 5 items: + # + # 0. code (int): Tag Id. + # + # 1. dtype (:py:class:`DATATYPE`): + # Data type of items in `value`. + # + # 2. count (int): Number of data values. + # Not used for string or bytes values. + # + # 3. value (Sequence[Any]): `count` values compatible with + # `dtype`. Bytes must contain count values of dtype packed + # as binary data. + # + # 4. writeonce (bool): If *True*, write tag to first page + # of a series only. + # + # Duplicate and select tags in TIFF.TAG_FILTERED are not written + # if the extratag is specified by integer code. + # + # Extratags cannot be used to write IFD type tags. + # # Format: (tag_number, datatype, count, value, write_ifd) # For rational number values: (numerator, denominator) - - tiff_extratags = [ - # CameraMake: Tag ID 271, 'ascii' - (271, 'ascii', len(metadata['camera_make']) + 1, metadata['camera_make'], False), + tiff_extratags = [] + """tiff_extratags = [ + # CameraMake: Tag ID 271, 'ASCII' + (271, dtype['ASCII'], len(metadata['camera_make']) + 1, metadata['camera_make'], False), + # ExposureTime: Tag ID 33434, 'RATIONAL' - (33434, 'RATIONAL', 1, ms_exposure_to_rational(metadata['exposure_time_ms']), False), + (33434, dtype['RATIONAL'], 1, ms_exposure_to_rational(metadata['exposure_time_ms']), False), + # ISOSpeed: Tag ID 34867, 'double' # Using in place of GainControl (Improper use of GainControl) - (34867, 'double', 1, metadata['gain_db'], False), + (34867, dtype['DOUBLE'], 1, metadata['gain_db'], False), + + # DateTime: Tag ID 306, 'ASCII' + (306, dtype['ASCII'], len(metadata['datetime']) + 1, metadata['datetime'], False), + + # SubjectDistance: Tag ID 37386, 'RATIONAL' + (37386, dtype['RATIONAL'], 1, subject_dist_to_rational(metadata['z_pos_um']), False), + + # SubSecTime: Tag ID 37520, 'ASCII' + (37520, dtype['ASCII'], len(metadata['sub_sec_time']) + 1, metadata['sub_sec_time'], False), - # DateTime: Tag ID 306, 'ascii' - (306, 'ascii', len(metadata['datetime']) + 1, metadata['datetime'], False), + # Channel: Tag ID 65001, 'ASCII' **CUSTOM** + (65001, dtype['ASCII'], len(metadata['channel']) + 1, metadata['channel'], False), - # Software: Tag ID 305, 'ascii' - (305, 'ascii', len(metadata['software']) + 1, metadata['software'], False), + # BrightnessValue: Tag ID 37393, 'SRATIONAL' + (37393, dtype['SRATIONAL'], 1, (metadata['illumination_ma'], 1), False)]""" + """ # XPosition: Tag ID 65001, 'RATIONAL' # Need to double check units - (286, 'RATIONAL', 1, (metadata['x_pos'], 1), False), + (286, dtype['RATIONAL'], 1, (metadata['x_pos'], 1), False),] + # YPosition: Custom Tag ID 65002, 'RATIONAL' # Need to double check units - (287, 'RATIONAL', 1, (metadata['y_pos'], 1), False), - - # SubjectDistance: Tag ID 37386, 'RATIONAL' - (37386, 'RATIONAL', 1, subject_dist_to_rational(metadata['z_pos_um']), False), - - # SubSecTime: Tag ID 37520, 'ascii' - (37520, 'ascii', len(metadata['sub_sec_time']) + 1, metadata['sub_sec_time'], False), + (287, dtype['RATIONAL'], 1, (metadata['y_pos'], 1), False), - # Channel: Tag ID 65001, 'ascii' **CUSTOM** - (65001, 'ascii', len(metadata['channel']) + 1, metadata['channel'], False), + + """ + - # BrightnessValue: Tag ID 37393, 'SRATIONAL' - (37393, 'SRATIONAL', 1, (metadata['illumination_ma'], 1), False) - ] else: # Video Frame # Add further parameters in the future if testing goes well - date_time_data = metadata['timestamp'].strftime("%Y:%m:%d %H:%M:%S") + """date_time_data = metadata['timestamp'].strftime("%Y:%m:%d %H:%M:%S") sub_sec_time = f"{metadata['timestamp'].microsecond // 1000:03d}" tiff_extratags = [ @@ -272,7 +325,9 @@ def generate_tiff_data(data, metadata: dict, ome: bool, video_frame: bool): # Tag 37393: ImageNumber (long) (37393, 'long', 1, metadata['frame_num'], False) - ] + ]""" + tiff_extratags = [] + tiff_metadata = metadata @@ -284,14 +339,22 @@ def generate_tiff_data(data, metadata: dict, ome: bool, video_frame: bool): maxworkers=2 ) - resolution = (1e4 / metadata['pixel_size_um'], 1e4 / metadata['pixel_size_um']) + if not video_frame: + resolution = (1e4 / metadata['pixel_size_um'], 1e4 / metadata['pixel_size_um']) - return { - 'metadata': tiff_metadata, - 'extratags': tiff_extratags, - 'options': options, - 'resolution': resolution, - } + return { + 'metadata': tiff_metadata, + 'extratags': tiff_extratags, + 'options': options, + 'resolution': resolution, + } + + else: + return { + 'metadata': tiff_metadata, + 'extratags': tiff_extratags, + 'options': options, + } def ms_exposure_to_rational(ms_exposure): exposure_seconds = ms_exposure / 1000 diff --git a/lumaviewpro.py b/lumaviewpro.py index e4cf508..312b0a5 100644 --- a/lumaviewpro.py +++ b/lumaviewpro.py @@ -1324,8 +1324,22 @@ def recording_complete(self, dt): output_file_loc = save_folder / f"{frame_name}.tiff" - if not cv2.imwrite(filename=str(output_file_loc), img=image): - logger.error(f"Manual-Video] Failed to write frame {frame_num}") + metadata = { + "datetime": ts.strftime("%Y:%m:%d %H:%M:%S"), + "timestamp": ts.strftime("%Y:%m:%d %H:%M:%S.%f"), + "frame_num": frame_num + } + + try: + image_utils.write_tiff( + data=image, + metadata=metadata, + file_loc=output_file_loc, + video_frame=True, + ome=False, + ) + except Exception as e: + logger.error(f"Protocol-Video] Failed to write frame {frame_num}: {e}") else: if not self.video_save_folder.exists(): diff --git a/modules/sequenced_capture_executor.py b/modules/sequenced_capture_executor.py index 1c862e7..d6106a1 100644 --- a/modules/sequenced_capture_executor.py +++ b/modules/sequenced_capture_executor.py @@ -746,7 +746,8 @@ def _capture( output_file_loc = save_folder / f"{frame_name}.tiff" metadata = { - "timestamp": ts, + "datetime": ts.strftime("%Y:%m:%d %H:%M:%S"), + "timestamp": ts.strftime("%Y:%m:%d %H:%M:%S.%f"), "frame_num": frame_num } From 2fb9a9f3772e4e6227dbacc2f2f7390267badbbb Mon Sep 17 00:00:00 2001 From: Dovydas Vabalas <139030549+dovydasv2@users.noreply.github.com> Date: Wed, 6 Nov 2024 16:11:00 -0800 Subject: [PATCH 8/8] Update version.txt --- version.txt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/version.txt b/version.txt index 7ec1d6d..a37af2c 100644 --- a/version.txt +++ b/version.txt @@ -1 +1 @@ -2.1.0 +2.2.0-beta