Skip to content

napari_ndev.nimage #

Additional functionality for BioImage objects to be used in napari-ndev.

nImage #

Bases: BioImage

An nImage is a BioImage with additional functionality for napari-ndev.

Parameters:

  • image #

    (ImageLike) –

    Image to be loaded. Can be a path to an image file, a numpy array, or an xarray DataArray.

  • reader #

    (Reader, default: None ) –

    Reader to be used to load the image. If not provided, a reader will be determined based on the image type.

Attributes:

  • See BioImage for inherited attributes.

Methods:

Source code in src/napari_ndev/nimage.py
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
class nImage(BioImage):
    """
    An nImage is a BioImage with additional functionality for napari-ndev.

    Parameters
    ----------
    image : ImageLike
        Image to be loaded. Can be a path to an image file, a numpy array,
        or an xarray DataArray.
    reader : Reader, optional
        Reader to be used to load the image. If not provided, a reader will be
        determined based on the image type.

    Attributes
    ----------
    See BioImage for inherited attributes.

    Methods
    -------
    get_napari_image_data(in_memory=None)
        Get the image data as a xarray, optionally loading it into memory.


    """

    def __init__(
        self,
        image: ImageLike,
        reader: Reader | None = None
    ) -> None:
        """
        Initialize an nImage with an image, and optionally a reader.

        If a reader is not provided, a reader will be determined by bioio.
        However, if the image is supported by bioio-ome-tiff, the reader
        will be set to bioio_ome_tiff.Reader to override the softer decision
        made by bioio.BioImage.determine_plugin().

        Note: The issue here is that bioio.BioImage.determine_plugin() will
        sort by install time and choose the first plugin that supports the
        image. This is not always the desired behavior, because bioio-tifffile
        can take precedence over bioio-ome-tiff, even if the image was saved
        as an OME-TIFF via bioio.writers.OmeTiffWriter (which is the case
        for napari-ndev).
        """
        if reader is None:
            from bioio import plugin_feasibility_report as pfr
            fr = pfr(image)
            if 'bioio-ome-tiff' in fr and fr['bioio-ome-tiff'].supported:
                import bioio_ome_tiff
                reader = bioio_ome_tiff.Reader

        super().__init__(image, reader)
        self.napari_data = None
        self.napari_metadata = {}
        self.path = image if isinstance(image, (str, Path)) else None

    def _determine_in_memory(
        self,
        path=None,
        max_in_mem_bytes: int = 4e9,
        max_in_mem_percent: int = 0.3
    ) -> bool:
        """
        Determine whether the image should be loaded into memory or not.

        If the image is smaller than the maximum filesize or percentage of the
        available memory, this will determine to load image in memory.
        Otherwise, suggest to load as a dask array.

        Parameters
        ----------
        path : str or Path
            Path to the image file.
        max_in_mem_bytes : int
            Maximum number of bytes that can be loaded into memory.
            Default is 4 GB (4e9 bytes)
        max_in_mem_percent : float
            Maximum percentage of memory that can be loaded into memory.
            Default is 30% of available memory (0.3)

        Returns
        -------
        bool
            True if image should be loaded in memory, False otherwise.

        """
        from bioio_base.io import pathlike_to_fs
        from psutil import virtual_memory

        if path is None:
            path = self.path

        fs, path = pathlike_to_fs(path)
        filesize = fs.size(path)
        available_mem = virtual_memory().available
        return (
            filesize <= max_in_mem_bytes
            and filesize < max_in_mem_percent * available_mem
        )

    def get_napari_image_data(self, in_memory: bool | None = None) -> xr.DataArray:
        """
        Get the image data as a xarray DataArray.

        From BioImage documentation:
        If you do not want the image pre-stitched together, you can use the base reader
        by either instantiating the reader independently or using the `.reader` property.

        Parameters
        ----------
        in_memory : bool, optional
            Whether to load the image in memory or not.
            If None, will determine whether to load in memory based on the image size.

        Returns
        -------
        xr.DataArray
            Image data as a xarray DataArray.

        """
        if in_memory is None:
            in_memory = self._determine_in_memory()

        if DimensionNames.MosaicTile in self.reader.dims.order:
            try:
                if in_memory:
                    self.napari_data = self.reader.mosaic_xarray_data.squeeze()
                else:
                    self.napari_data = self.reader.mosaic_xarray_dask_data.squeeze()

            except NotImplementedError:
                logger.warning(
                    "Bioio: Mosaic tile switching not supported for this reader"
                )
                return None
        else:
            if in_memory:
                self.napari_data = self.reader.xarray_data.squeeze()
            else:
                self.napari_data = self.reader.xarray_dask_data.squeeze()

        return self.napari_data

    def get_napari_metadata(
        self,
        path: PathLike,
    ) -> dict:
        """
        Get the metadata for the image to be displayed in napari.

        Parameters
        ----------
        path : PathLike
            Path to the image file.

        Returns
        -------
        dict
            Metadata for the image to be displayed in napari.

        """
        if self.napari_data is None:
            self.get_napari_image_data() # this also sets self.path

        meta = {}
        scene = self.current_scene
        scene_idx = self.current_scene_index
        single_no_scene = len(self.scenes) == 1 and self.current_scene == "Image:0"
        channel_dim = DimensionNames.Channel

        if channel_dim in self.napari_data.dims:
            # use filename if single scene and no scene name available
            if single_no_scene:
                channels_with_scene_index = [
                    f'{C}{LABEL_DELIMITER}{Path(path).stem}'
                    for C in self.napari_data.coords[channel_dim].data.tolist()
                ]
            else:
                channels_with_scene_index = [
                    f'{C}{LABEL_DELIMITER}{scene_idx}{LABEL_DELIMITER}{scene}'
                    for C in self.napari_data.coords[channel_dim].data.tolist()
                ]
            meta['name'] = channels_with_scene_index
            meta['channel_axis'] = self.napari_data.dims.index(channel_dim)

        # not multi-chnanel, use current scene as image name
        else:
            if single_no_scene:
                meta['name'] = Path(path).stem
            else:
                meta['name'] = self.reader.current_scene

        # Handle if RGB
        if DimensionNames.Samples in self.reader.dims.order:
            meta['rgb'] = True

        # Handle scales
        scale = [
            getattr(self.physical_pixel_sizes, dim)
            for dim in self.napari_data.dims
            if dim in {DimensionNames.SpatialX, DimensionNames.SpatialY, DimensionNames.SpatialZ}
            and getattr(self.physical_pixel_sizes, dim) is not None
        ]

        if scale:
            meta['scale'] = tuple(scale)

        # get all other metadata
        img_meta = {'bioimage': self, 'raw_image_metadata': self.metadata}

        with contextlib.suppress(NotImplementedError):
            img_meta['metadata'] = self.ome_metadata

        meta['metadata'] = img_meta
        self.napari_metadata = meta
        return self.napari_metadata

__init__ #

__init__(image, reader=None)

Initialize an nImage with an image, and optionally a reader.

If a reader is not provided, a reader will be determined by bioio. However, if the image is supported by bioio-ome-tiff, the reader will be set to bioio_ome_tiff.Reader to override the softer decision made by bioio.BioImage.determine_plugin().

Note: The issue here is that bioio.BioImage.determine_plugin() will sort by install time and choose the first plugin that supports the image. This is not always the desired behavior, because bioio-tifffile can take precedence over bioio-ome-tiff, even if the image was saved as an OME-TIFF via bioio.writers.OmeTiffWriter (which is the case for napari-ndev).

Source code in src/napari_ndev/nimage.py
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
def __init__(
    self,
    image: ImageLike,
    reader: Reader | None = None
) -> None:
    """
    Initialize an nImage with an image, and optionally a reader.

    If a reader is not provided, a reader will be determined by bioio.
    However, if the image is supported by bioio-ome-tiff, the reader
    will be set to bioio_ome_tiff.Reader to override the softer decision
    made by bioio.BioImage.determine_plugin().

    Note: The issue here is that bioio.BioImage.determine_plugin() will
    sort by install time and choose the first plugin that supports the
    image. This is not always the desired behavior, because bioio-tifffile
    can take precedence over bioio-ome-tiff, even if the image was saved
    as an OME-TIFF via bioio.writers.OmeTiffWriter (which is the case
    for napari-ndev).
    """
    if reader is None:
        from bioio import plugin_feasibility_report as pfr
        fr = pfr(image)
        if 'bioio-ome-tiff' in fr and fr['bioio-ome-tiff'].supported:
            import bioio_ome_tiff
            reader = bioio_ome_tiff.Reader

    super().__init__(image, reader)
    self.napari_data = None
    self.napari_metadata = {}
    self.path = image if isinstance(image, (str, Path)) else None

get_napari_image_data #

get_napari_image_data(in_memory=None)

Get the image data as a xarray DataArray.

From BioImage documentation: If you do not want the image pre-stitched together, you can use the base reader by either instantiating the reader independently or using the .reader property.

Parameters:

  • in_memory #

    (bool, default: None ) –

    Whether to load the image in memory or not. If None, will determine whether to load in memory based on the image size.

Returns:

  • DataArray

    Image data as a xarray DataArray.

Source code in src/napari_ndev/nimage.py
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
def get_napari_image_data(self, in_memory: bool | None = None) -> xr.DataArray:
    """
    Get the image data as a xarray DataArray.

    From BioImage documentation:
    If you do not want the image pre-stitched together, you can use the base reader
    by either instantiating the reader independently or using the `.reader` property.

    Parameters
    ----------
    in_memory : bool, optional
        Whether to load the image in memory or not.
        If None, will determine whether to load in memory based on the image size.

    Returns
    -------
    xr.DataArray
        Image data as a xarray DataArray.

    """
    if in_memory is None:
        in_memory = self._determine_in_memory()

    if DimensionNames.MosaicTile in self.reader.dims.order:
        try:
            if in_memory:
                self.napari_data = self.reader.mosaic_xarray_data.squeeze()
            else:
                self.napari_data = self.reader.mosaic_xarray_dask_data.squeeze()

        except NotImplementedError:
            logger.warning(
                "Bioio: Mosaic tile switching not supported for this reader"
            )
            return None
    else:
        if in_memory:
            self.napari_data = self.reader.xarray_data.squeeze()
        else:
            self.napari_data = self.reader.xarray_dask_data.squeeze()

    return self.napari_data

get_napari_metadata #

get_napari_metadata(path)

Get the metadata for the image to be displayed in napari.

Parameters:

  • path #

    (PathLike) –

    Path to the image file.

Returns:

  • dict

    Metadata for the image to be displayed in napari.

Source code in src/napari_ndev/nimage.py
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
def get_napari_metadata(
    self,
    path: PathLike,
) -> dict:
    """
    Get the metadata for the image to be displayed in napari.

    Parameters
    ----------
    path : PathLike
        Path to the image file.

    Returns
    -------
    dict
        Metadata for the image to be displayed in napari.

    """
    if self.napari_data is None:
        self.get_napari_image_data() # this also sets self.path

    meta = {}
    scene = self.current_scene
    scene_idx = self.current_scene_index
    single_no_scene = len(self.scenes) == 1 and self.current_scene == "Image:0"
    channel_dim = DimensionNames.Channel

    if channel_dim in self.napari_data.dims:
        # use filename if single scene and no scene name available
        if single_no_scene:
            channels_with_scene_index = [
                f'{C}{LABEL_DELIMITER}{Path(path).stem}'
                for C in self.napari_data.coords[channel_dim].data.tolist()
            ]
        else:
            channels_with_scene_index = [
                f'{C}{LABEL_DELIMITER}{scene_idx}{LABEL_DELIMITER}{scene}'
                for C in self.napari_data.coords[channel_dim].data.tolist()
            ]
        meta['name'] = channels_with_scene_index
        meta['channel_axis'] = self.napari_data.dims.index(channel_dim)

    # not multi-chnanel, use current scene as image name
    else:
        if single_no_scene:
            meta['name'] = Path(path).stem
        else:
            meta['name'] = self.reader.current_scene

    # Handle if RGB
    if DimensionNames.Samples in self.reader.dims.order:
        meta['rgb'] = True

    # Handle scales
    scale = [
        getattr(self.physical_pixel_sizes, dim)
        for dim in self.napari_data.dims
        if dim in {DimensionNames.SpatialX, DimensionNames.SpatialY, DimensionNames.SpatialZ}
        and getattr(self.physical_pixel_sizes, dim) is not None
    ]

    if scale:
        meta['scale'] = tuple(scale)

    # get all other metadata
    img_meta = {'bioimage': self, 'raw_image_metadata': self.metadata}

    with contextlib.suppress(NotImplementedError):
        img_meta['metadata'] = self.ome_metadata

    meta['metadata'] = img_meta
    self.napari_metadata = meta
    return self.napari_metadata