Source code for cleo.imaging.scope

from __future__ import annotations
from typing import Any, Callable
import warnings

from attrs import define, field
from brian2 import NeuronGroup, Unit, mm, np, Quantity, um, meter, ms
import matplotlib as mpl
from matplotlib.artist import Artist
from mpl_toolkits.mplot3d import Axes3D
from nptyping import NDArray

from cleo.base import Recorder

from cleo.imaging.sensors import Sensor
from cleo.coords import coords_from_ng
from cleo.utilities import normalize_coords, rng


[docs] def target_neurons_in_plane( ng: NeuronGroup, scope_focus_depth: Quantity, scope_img_width: Quantity, scope_location: Quantity = (0, 0, 0) * mm, scope_direction: tuple = (0, 0, 1), soma_radius: Quantity = 10 * um, sensor_location: str = "cytoplasm", ) -> tuple[NDArray[(Any,), int], NDArray[(Any,), float], NDArray[(Any, 3), float]]: """ Returns a tuple of (i_targets, noise_focus_factor, coords_on_plane) Parameters ---------- ng : NeuronGroup The neuron group to target. scope_focus_depth : Quantity The depth of the focal plane of the microscope. scope_img_width : Quantity The width of the image captured by the microscope. scope_location : Quantity, optional The location of the microscope, by default (0, 0, 0) * mm. scope_direction : tuple, optional The direction of the microscope, by default (0, 0, 1). soma_radius : Quantity, optional The radius of the soma of the neuron, by default 10 * um. Used to compute noise focus factor, since smaller ROIs will have a noisier distribution of fluorescence, averaged over fewer pixels. sensor_location : str, optional The location of the sensor, by default "cytoplasm". Returns ------- Tuple[NDArray[(Any,), int], NDArray[(Any,), float], NDArray[(Any, 3), float]] A tuple of (i_targets, noise_focus_factor, coords_on_plane) """ assert sensor_location in ("cytoplasm", "membrane") ng_coords = coords_from_ng(ng) # Compute the normal vector and the center of the plane plane_normal = scope_direction plane_center = scope_location + plane_normal * scope_focus_depth # Compute the distance of each neuron from the plane perp_distances = np.abs(np.dot(ng_coords - plane_center, plane_normal)) noise_focus_factor = np.ones(ng.N) # signal falloff with shrinking cross-section (or circumference) # ignore numpy warning with np.errstate(invalid="ignore"): r_soma_visible = np.sqrt(soma_radius**2 - perp_distances**2) r_soma_visible[np.isnan(r_soma_visible)] = 0 if sensor_location == "cytoplasm": relative_num_pixels = (r_soma_visible / soma_radius) ** 2 else: relative_num_pixels = r_soma_visible / soma_radius with np.errstate(divide="ignore"): noise_focus_factor /= np.sqrt(relative_num_pixels) # get only neurons in view coords_on_plane = ng_coords - plane_normal * perp_distances[:, np.newaxis] plane_distances = np.sqrt(np.sum((coords_on_plane - plane_center) ** 2, axis=1)) i_targets = np.flatnonzero( np.logical_and(r_soma_visible > 0, plane_distances < scope_img_width / 2) ) return i_targets, noise_focus_factor[i_targets], coords_on_plane[i_targets]
[docs] @define(eq=False) class Scope(Recorder): """Two-photon microscope. Injection kwargs ---------------- rho_rel_generator : Callable[[int], NDArray[(Any,), float]], optional A function assigning expression levels. Takes n as an arg, outputs float array. ``lambda n: np.ones(n)`` by default. focus_depth : Quantity, optional The depth of the focal plane, by default that of the scope. soma_radius : Quantity, optional The radius of the soma of the neuron, by default that of the scope. Used to compute noise focus factor, since smaller ROIs will have a noisier distribution of fluorescence, averaged over fewer pixels. """ sensor: Sensor = field() img_width: Quantity = field() """The width (diameter) of the (circular) image captured by the microscope. Specified in distance units.""" focus_depth: Quantity = None """The depth of the focal plane, with distance units""" location: Quantity = [0, 0, 0] * mm """Location of the objective lens.""" direction: NDArray[(3,), float] = field( default=(0, 0, 1), converter=normalize_coords ) """Direction in which the microscope is pointing. By default straight down (`+z` direction)""" soma_radius: Quantity = field(default=10 * um) """Assumed radius of neurons, used to compute noise focus factor. Smaller neurons have noisier signals.""" snr_cutoff: float = field(default=1) """SNR below which neurons are discarded. Applied only when not focus_depth is not None""" rand_seed: int = field(default=None, repr=False) dFF: list[NDArray[(Any,), float]] = field(factory=list, init=False, repr=False) """ΔF/F from every call to :meth:`get_state`. Shape is (n_samples, n_ROIs). Stored if :attr:`~cleo.InterfaceDevice.save_history`""" t_ms: list[float] = field(factory=list, init=False, repr=False) """Times at which sensor traces are recorded, in ms, stored if :attr:`~cleo.InterfaceDevice.save_history`""" neuron_groups: list[NeuronGroup] = field(factory=list, repr=False, init=False) """neuron groups the scope has been injected into, in order of injection""" i_targets_per_injct: list[NDArray[(Any,), int]] = field( factory=list, repr=False, init=False ) """targets of neurons selected from each injection""" sigma_per_injct: list[NDArray[(Any,), float]] = field( factory=list, repr=False, init=False ) """`sigma_noise` of neurons selected from each injection""" focus_coords_per_injct: list[NDArray[(Any,), float]] = field( factory=list, repr=False, init=False ) """coordinates on the focal plane of neurons selected from each injection""" rho_rel_per_injct: list[NDArray[(Any,), float]] = field( factory=list, repr=False, init=False ) """relative expression levels of neurons selected from each injection""" @property def n(self) -> int: """Number of imaged ROIs""" return np.sum(len(i_t) for i_t in self.i_targets_per_injct) @property def sigma_noise(self) -> NDArray[(Any,), float]: """noise std dev (in terms of ΔF/F) for all targets, in order injected.""" return np.concatenate(self.sigma_per_injct) @property def dFF_1AP(self) -> NDArray[(Any,), float]: """dFF_1AP for all targets, in order injected. Varies with expression levels.""" rho_rel = np.array([]) n_prev_targets_for_ng = {} for ng, i_targets in zip(self.neuron_groups, self.i_targets_per_injct): syn = self.sensor.synapses[ng.name] subset_start = n_prev_targets_for_ng.get(ng, 0) subset_for_injct = slice(subset_start, subset_start + len(i_targets)) rho_rel = np.concatenate([rho_rel, syn.rho_rel[subset_for_injct]]) n_prev_targets_for_ng[ng] = subset_start + len(i_targets) assert rho_rel.shape == (self.n,) return rho_rel * self.sensor.dFF_1AP def _init_saved_vars(self): if self.save_history: self.t_ms = [] self.dFF = [] def _update_saved_vars(self, t_ms, dFF): if self.save_history: self.t_ms.append(t_ms) self.dFF.append(dFF) def __attrs_post_init__(self): self._init_saved_vars()
[docs] def reset(self, **kwargs) -> None: self._init_saved_vars()
[docs] def target_neurons_in_plane( self, ng, focus_depth: Quantity = None, soma_radius: Quantity = None ) -> tuple[NDArray[(Any,), int], NDArray[(Any,), float], NDArray[(Any, 3), float]]: """calls :func:`target_neurons_in_plane` with scope parameter defaults. `focus_depth` and `soma_radius` can be overridden here. Parameters ---------- ng : NeuronGroup The neuron group to target. focus_depth : Quantity The depth of the focal plane, by default that of the microscope. soma_radius : Quantity, optional The radius of the soma of the neuron, by default that of the microscope. Used to compute noise focus factor, since smaller ROIs will have a noisier distribution of fluorescence, averaged over fewer pixels. Returns ------- Tuple[NDArray[(Any,), int], NDArray[(Any,), float], NDArray[(Any, 3), float]] A tuple of (i_targets, noise_focus_factor, coords_on_plane) """ focus_depth = focus_depth or self.focus_depth soma_radius = soma_radius or self.soma_radius return target_neurons_in_plane( ng, focus_depth, self.img_width, self.location, self.direction, soma_radius, self.sensor.location, )
[docs] def get_state(self) -> NDArray[(Any,), float]: """Returns a 1D array of ΔF/F values for all targets, in order injected. Returns ------- NDArray[(Any,), float] Fluorescence values for all targets """ signal = [] signal_per_ng = self.sensor.get_state() # sensor has just one signal for neuron group, not storing # separately for each injection, so we'll recover that here n_prev_targets_for_ng = {} for ng, i_targets in zip(self.neuron_groups, self.i_targets_per_injct): subset_start = n_prev_targets_for_ng.get(ng, 0) subset_for_injct = slice(subset_start, subset_start + len(i_targets)) signal.append(signal_per_ng[ng.name][subset_for_injct]) n_prev_targets_for_ng[ng] = subset_start + len(i_targets) signal = np.concatenate(signal) noise = rng.normal(0, self.sigma_noise, len(signal)) assert self.n == len(signal) == len(noise) == len(self.sigma_noise) state = signal + noise now_ms = self.sim.network.t / ms self._update_saved_vars(now_ms, state) return signal + noise
[docs] def connect_to_neuron_group(self, neuron_group: NeuronGroup, **kwparams) -> None: focus_depth = kwparams.get("focus_depth", self.focus_depth) soma_radius = kwparams.get("soma_radius", self.soma_radius) rho_rel_generator = kwparams.get("rho_rel_generator", lambda n: np.ones(n)) if focus_depth: if "i_targets" in kwparams or "sigma_noise" in kwparams: raise ValueError( "i_targets and sigma_noise can not be specified while focus_depth", " is not None", ) ( i_targets, noise_focus_factor, focus_coords, ) = self.target_neurons_in_plane(neuron_group, focus_depth, soma_radius) base_sigma = kwparams.get("base_sigma", self.sensor.sigma_noise) sigma_noise = noise_focus_factor * base_sigma rho_rel = rho_rel_generator(len(i_targets)) if self.sensor.dFF_1AP is not None: snr = rho_rel * self.sensor.dFF_1AP / sigma_noise i_targets = i_targets[snr > self.snr_cutoff] sigma_noise = sigma_noise[snr > self.snr_cutoff] focus_coords = focus_coords[snr > self.snr_cutoff] rho_rel = rho_rel[snr > self.snr_cutoff] else: warnings.warn( f"SNR cutoff not used, since {self.sensor.name} does not have dFF_1AP defined." ) else: i_targets = kwparams.pop("i_targets", neuron_group.i_) sigma_noise = kwparams.get("sigma_noise", self.sensor.sigma_noise) rho_rel = rho_rel_generator(len(i_targets)) _, sigma_noise, rho_rel = np.broadcast_arrays( i_targets, sigma_noise, rho_rel ) focus_coords = coords_from_ng(neuron_group)[i_targets] assert len(i_targets) == len(sigma_noise) == len(focus_coords) == len(rho_rel) self.neuron_groups.append(neuron_group) self.i_targets_per_injct.append(i_targets) self.sigma_per_injct.append(sigma_noise) self.focus_coords_per_injct.append(focus_coords) self.rho_rel_per_injct.append(rho_rel)
[docs] def i_targets_for_neuron_group(self, neuron_group): """can handle multiple injections into same ng""" i_targets_for_ng = [] for ng, i_targets in zip(self.neuron_groups, self.i_targets_per_injct): if ng is neuron_group: i_targets_for_ng.extend(i_targets) return i_targets_for_ng
[docs] def inject_sensor_for_targets(self, **kwparams) -> None: for ng in set(self.neuron_groups): i_targets_for_ng = [] rho_rel_for_ng = [] for ng_, i_targets, rho_rel in zip( self.neuron_groups, self.i_targets_per_injct, self.rho_rel_per_injct ): if ng_ is not ng: continue i_targets_for_ng.extend(i_targets) rho_rel_for_ng.extend(rho_rel) self.sim.inject( self.sensor, ng, i_targets=i_targets_for_ng, rho_rel=rho_rel_for_ng, **kwparams, )
[docs] def add_self_to_plot( self, ax: Axes3D, axis_scale_unit: Unit, **kwargs ) -> list[Artist]: color = kwargs.pop("color", "xkcd:fluorescent green") snr = np.concatenate(self.sigma_per_injct) coords = ( np.concatenate( [ coords_from_ng(ng)[i_targets] / meter for ng, i_targets in zip( self.neuron_groups, self.i_targets_per_injct ) ] ) * meter ) assert coords.shape == (len(snr), 3) assert len(snr) == len(coords) scope_marker = ax.quiver( self.location[0] / axis_scale_unit, self.location[1] / axis_scale_unit, self.location[2] / axis_scale_unit, self.direction[0], self.direction[1], self.direction[2], color=color, lw=5, label=self.name, pivot="tail", length=self.focus_depth / axis_scale_unit, normalize=True, ) # Define the center, normal vector, and radius normal = self.direction center = self.location + normal * self.focus_depth radius = self.img_width / 2 # Generate the points for the circle theta = np.linspace(0, 2 * np.pi, 100) r = np.sqrt(np.linspace(0, 1, 5)) # angle away from z axis phi = np.arccos(normal[2]) # angle relative to x axis theta0 = np.arctan2(normal[1], normal[0]) x = center[0] + radius * np.cos(phi) * np.outer(np.cos(theta), r) y = center[1] + radius * np.cos(phi) * np.outer(np.sin(theta), r) z = center[2] + radius * np.sin(phi) * np.outer(-np.cos(theta - theta0), r) plane = ax.plot_surface( x / axis_scale_unit, y / axis_scale_unit, z / axis_scale_unit, color=color, alpha=0.2, ) target_markers = ax.scatter( coords[:, 0] / axis_scale_unit, coords[:, 1] / axis_scale_unit, coords[:, 2] / axis_scale_unit, marker="^", c=color, label=f"{self.sensor.name} ROIs", **kwargs, ) color_rgba = target_markers.get_facecolor() color_rgba[:, :3] = 0.3 * color_rgba[:, :3] target_markers.set(color=color_rgba) handles = ax.get_legend().legend_handles handles.append(target_markers) patch = mpl.patches.Patch(color=color, label=self.name) handles.append(patch) ax.legend(handles=handles) return [scope_marker, target_markers, plane]