Skip to content

Commit

Permalink
cleaning up example code
Browse files Browse the repository at this point in the history
  • Loading branch information
IvoVellekoop committed Oct 3, 2024
1 parent 703a205 commit eaa6d4e
Show file tree
Hide file tree
Showing 5 changed files with 51 additions and 106 deletions.
60 changes: 16 additions & 44 deletions examples/micro_manager_microscope.py
Original file line number Diff line number Diff line change
@@ -1,41 +1,30 @@
""" Sample microscope
""" Micro-Manager simulated microscope
=======================
This script simulates a microscopic imaging system, generating a random noise image as a mock source and capturing it
through a microscope with adjustable magnification, numerical aperture, and wavelength. It visualizes the original and
processed images dynamically, demonstrating how changes in optical parameters affect image quality and resolution.
This script simulates a microscope with a random noise image as a mock specimen.
The numerical aperture, stage position, and other parameters can be modified through the Micro-Manager GUI.
To use this script as a device in Micro-Manager, make sure you have the PyDevice adapter installed and
select this script in the hardware configuration wizard for the PyDevice component.
This script should be opened from the μManager microscope GUI software using the PyDevice plugin.
To do so, add a PyDevice adapter to the μManager hardware configuration, and select this script as the device script.
See the 'Sample Microscope' example for a microscope simulation that runs from Python directly.
"""

import astropy.units as u
import numpy as np

from openwfs.simulation import Microscope, StaticSource

# height × width, and resolution, of the specimen image
specimen_size = (1024, 1024)
specimen_resolution = 60 * u.nm

# magnification from object plane to camera.
magnification = 40

# numerical aperture of the microscope objective
numerical_aperture = 0.85

# wavelength of the light, for computing diffraction.
wavelength = 532.8 * u.nm

# Size of the pixels on the camera
pixel_size = 6.45 * u.um

# number of pixels on the camera
camera_resolution = (256, 256)
specimen_resolution = (1024, 1024) # height × width in pixels of the specimen image
specimen_pixel_size = 60 * u.nm # resolution (pixel size) of the specimen image
magnification = 40 # magnification from object plane to camera.
numerical_aperture = 0.85 # numerical aperture of the microscope objective
wavelength = 532.8 * u.nm # wavelength of the light, for computing diffraction.
camera_resolution = (256, 256) # number of pixels on the camera
camera_pixel_size = 6.45 * u.um # Size of the pixels on the camera

# Create a random noise image with a few bright spots
src = StaticSource(
data=np.maximum(np.random.randint(-10000, 100, specimen_size, dtype=np.int16), 0),
pixel_size=specimen_resolution,
data=np.maximum(np.random.randint(-10000, 100, specimen_resolution, dtype=np.int16), 0),
pixel_size=specimen_pixel_size,
)

# Create a microscope with the given parameters
Expand All @@ -51,25 +40,8 @@
shot_noise=True,
digital_max=255,
data_shape=camera_resolution,
pixel_size=pixel_size,
pixel_size=camera_pixel_size,
)

# expose the xy-stage of the microscope
stage = mic.xy_stage

# construct dictionary of objects to expose to Micro-Manager
devices = {"camera": cam, "stage": stage}

if __name__ == "__main__":
# When running this script directly (not from μManager)
# the code below shows how to operate the stage and change the numerical aperture of the microscope
import matplotlib.pyplot as plt

for i in range(20):
stage.x = stage.x + 2 * u.um
mic.numerical_aperture -= 0.025
plt.imshow(mic.read(), cmap="gray")
if i == 0:
plt.colorbar()
plt.show(block=False)
plt.pause(0.5)
14 changes: 7 additions & 7 deletions examples/micro_manager_scanning_microscope.py
Original file line number Diff line number Diff line change
@@ -1,16 +1,16 @@
"""
Constructs a scanning microscope controller for use with Micro-Manager
The microscope object can be loaded into Micro-Manager through the PyDevice
device adapter.
""" Micro-Manager simulated scanning microscope
=======================
This script simulates a scanning microscope with a pre-set image as a mock specimen.
The scan parameters can be modified through the Micro-Manager GUI.
To use this script as a device in Micro-Manager, make sure you have the PyDevice adapter installed and
select this script in the hardware configuration wizard for the PyDevice component.
"""

import astropy.units as u
import skimage

# add 'openwfs' to the search path. This is only needed when developing openwfs
# otherwise it is just installed as a package
import set_path # noqa
import skimage
from openwfs.devices import ScanningMicroscope, Axis
from openwfs.devices.galvo_scanner import InputChannel

Expand Down
57 changes: 18 additions & 39 deletions examples/sample_microscope.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,42 +11,23 @@
import set_path # noqa - needed for setting the module search path to find openwfs
from openwfs.plot_utilities import grab_and_show, imshow
from openwfs.simulation import Microscope, StaticSource
from openwfs.utilities import set_pixel_size

# Parameters that can be altered

img_size_x = 1024
# Determines how wide the image is.

img_size_y = 1024
# Determines how high the image is.

magnification = 40
# magnification from object plane to camera.

numerical_aperture = 0.85
# numerical aperture of the microscope objective

wavelength = 532.8 * u.nm
# wavelength of the light, different wavelengths are possible, units can be adjusted accordingly.

pixel_size = 6.45 * u.um
# Size of the pixels on the camera

camera_resolution = (256, 256)
# number of pixels on the camera

p_limit = 100
# Number of iterations. Influences how quick the 'animation' is complete.

# Code
img = set_pixel_size(
np.maximum(
np.random.randint(-10000, 100, (img_size_y, img_size_x), dtype=np.int16), 0
),
60 * u.nm,
specimen_resolution = (1024, 1024) # height × width in pixels of the specimen image
specimen_pixel_size = 60 * u.nm # resolution (pixel size) of the specimen image
magnification = 40 # magnification from object plane to camera.
numerical_aperture = 0.85 # numerical aperture of the microscope objective
wavelength = 532.8 * u.nm # wavelength of the light, for computing diffraction.
camera_resolution = (256, 256) # number of pixels on the camera
camera_pixel_size = 6.45 * u.um # Size of the pixels on the camera
p_limit = 100 # Number steps in the animation

# Create a random noise image with a few bright spots
src = StaticSource(
data=np.maximum(np.random.randint(-10000, 100, specimen_resolution, dtype=np.int16), 0),
pixel_size=specimen_pixel_size,
)
src = StaticSource(img)

# Create a microscope with the given parameters
mic = Microscope(
src,
magnification=magnification,
Expand All @@ -59,15 +40,15 @@
shot_noise=True,
digital_max=255,
data_shape=camera_resolution,
pixel_size=pixel_size,
pixel_size=camera_pixel_size,
)
devices = {"camera": cam, "stage": mic.xy_stage}

if __name__ == "__main__":
import matplotlib.pyplot as plt

plt.subplot(1, 2, 1)
imshow(img)
imshow(src.data)
plt.title("Original image")
plt.subplot(1, 2, 2)
plt.title("Scanned image")
Expand All @@ -76,7 +57,5 @@
mic.xy_stage.x = p * 1 * u.um
mic.numerical_aperture = 1.0 * (p + 1) / p_limit # NA increases to 1.0
ax = grab_and_show(cam, ax)
plt.title(
f"NA: {mic.numerical_aperture}, δ: {mic.abbe_limit.to_value(u.um):2.2} μm"
)
plt.title(f"NA: {mic.numerical_aperture}, δ: {mic.abbe_limit.to_value(u.um):2.2} μm")
plt.pause(0.2)
14 changes: 5 additions & 9 deletions examples/wfs_demonstration_experimental.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,9 @@
WFS demo experiment
=====================
This script demonstrates how to perform a wavefront shaping experiment using the openwfs library.
It assumes that you have a genicam camera and an SLM connected to your computer.
Please adjust the path to the camera driver and (when needed) the monitor id in the Camera and SLM objects.
"""

import astropy.units as u
Expand All @@ -19,16 +22,12 @@

# constructs the actual slm for wavefront shaping, and a monitor window to display the current phase pattern
slm = SLM(monitor_id=2, duration=2)
monitor = slm.clone(
monitor_id=0, pos=(0, 0), shape=(slm.shape[0] // 4, slm.shape[1] // 4)
)
monitor = slm.clone(monitor_id=0, pos=(0, 0), shape=(slm.shape[0] // 4, slm.shape[1] // 4))

# we are using a setup with an SLM that produces 2pi phase shift
# at a gray value of 142
slm.lookup_table = range(142)
alg = FourierDualReference(
feedback=roi_detector, slm=slm, slm_shape=[800, 800], k_radius=7
)
alg = FourierDualReference(feedback=roi_detector, slm=slm, slm_shape=[800, 800], k_radius=7)

result = alg.execute()
print(result)
Expand All @@ -46,6 +45,3 @@
plt.pause(1.0)
slm.set_phases(0.0)
plt.pause(1.0)

# plt.show()
# input("press any key")
12 changes: 5 additions & 7 deletions openwfs/simulation/mockdevices.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@

from ..core import Detector, Processor, Actuator
from ..processors import CropProcessor
from ..utilities import ExtentType, get_pixel_size
from ..utilities import ExtentType, get_pixel_size, set_pixel_size


class StaticSource(Detector):
Expand Down Expand Up @@ -42,6 +42,8 @@ def __init__(
pixel_size = Quantity(extent) / data.shape
else:
pixel_size = get_pixel_size(data)
else:
data = set_pixel_size(data, pixel_size) # make sure the data array holds the pixel size

if (
pixel_size is not None
Expand Down Expand Up @@ -176,19 +178,15 @@ def _fetch(self, data) -> np.ndarray: # noqa
if self.analog_max == 0.0: # auto scaling
max_value = np.max(data)
if max_value > 0.0:
data = data * (
self.digital_max / max_value
) # auto-scale to maximum value
data = data * (self.digital_max / max_value) # auto-scale to maximum value
else:
data = data * (self.digital_max / self.analog_max)

if self._shot_noise:
data = self._rng.poisson(data)

if self._gaussian_noise_std > 0.0:
data = data + self._rng.normal(
scale=self._gaussian_noise_std, size=data.shape
)
data = data + self._rng.normal(scale=self._gaussian_noise_std, size=data.shape)

return np.clip(np.rint(data), 0, self.digital_max).astype("uint16")

Expand Down

0 comments on commit eaa6d4e

Please sign in to comment.