from typing import TYPE_CHECKING

import magicgui as mgui

from .components.viewer_model import ViewerModel
from .utils import _magicgui, config

if TYPE_CHECKING:
    # helpful for IDE support
    from ._qt.qt_main_window import Window


[docs]@mgui.register_type(bind=_magicgui.find_viewer_ancestor) class Viewer(ViewerModel): """Napari ndarray viewer. Parameters ---------- title : string, optional The title of the viewer window. by default 'napari'. ndisplay : {2, 3}, optional Number of displayed dimensions. by default 2. order : tuple of int, optional Order in which dimensions are displayed where the last two or last three dimensions correspond to row x column or plane x row x column if ndisplay is 2 or 3. by default None axis_labels : list of str, optional Dimension names. by default they are labeled with sequential numbers show : bool, optional Whether to show the viewer after instantiation. by default True. """ _window: 'Window' = None # type: ignore def __init__( self, *, title='napari', ndisplay=2, order=(), axis_labels=(), show=True, ): super().__init__( title=title, ndisplay=ndisplay, order=order, axis_labels=axis_labels, ) # having this import here makes all of Qt imported lazily, upon # instantiating the first Viewer. from .window import Window self._window = Window(self, show=show) # Expose private window publically. This is needed to keep window off pydantic model @property def window(self) -> 'Window': return self._window
[docs] def update_console(self, variables): """Update console's namespace with desired variables. Parameters ---------- variables : dict, str or list/tuple of str The variables to inject into the console's namespace. If a dict, a simple update is done. If a str, the string is assumed to have variable names separated by spaces. A list/tuple of str can also be used to give the variable names. If just the variable names are give (list/tuple/str) then the variable values looked up in the callers frame. """ if self.window.qt_viewer._console is None: return else: self.window.qt_viewer.console.push(variables)
[docs] def screenshot(self, path=None, *, canvas_only=True, flash: bool = True): """Take currently displayed screen and convert to an image array. Parameters ---------- path : str Filename for saving screenshot image. canvas_only : bool If True, screenshot shows only the image display canvas, and if False include the napari viewer frame in the screenshot, By default, True. flash : bool Flag to indicate whether flash animation should be shown after the screenshot was captured. By default, True. Returns ------- image : array Numpy array of type ubyte and shape (h, w, 4). Index [0, 0] is the upper-left corner of the rendered region. """ if canvas_only: image = self.window.qt_viewer.screenshot(path=path, flash=flash) else: image = self.window.screenshot(path=path, flash=flash) return image
[docs] def show(self, *, block=False): """Resize, show, and raise the viewer window.""" self.window.show(block=block)
[docs] def close(self): """Close the viewer window.""" # Remove all the layers from the viewer self.layers.clear() # Close the main window self.window.close() if config.async_loading: from .components.experimental.chunk import chunk_loader # TODO_ASYNC: Find a cleaner way to do this? This fixes some # tests. We are telling the ChunkLoader that this layer is # going away: # https://github.com/napari/napari/issues/1500 for layer in self.layers: chunk_loader.on_layer_deleted(layer)
[docs]def current_viewer() -> Viewer: """Return the currently active napari viewer.""" try: from napari._qt.qt_main_window import _QtMainWindow return _QtMainWindow.current_viewer() except ImportError: return None