Skip to content

Commit

Permalink
Merge pull request #487 from rafalkowalewski1/development
Browse files Browse the repository at this point in the history
v0.7.0 ready
  • Loading branch information
rafalkowalewski1 authored Jul 8, 2024
2 parents e9a35e0 + 600a3d7 commit 6eb3536
Show file tree
Hide file tree
Showing 9 changed files with 74 additions and 49 deletions.
2 changes: 1 addition & 1 deletion .bumpversion.cfg
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
[bumpversion]
current_version = 0.6.11
current_version = 0.7.0
commit = True
tag = False
parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)(\-(?P<release>[a-z]+)(?P<build>\d+))?
Expand Down
4 changes: 2 additions & 2 deletions distribution/picasso.iss
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,10 @@
AppName=Picasso
AppPublisher=Jungmann Lab, Max Planck Institute of Biochemistry

AppVersion=0.6.11
AppVersion=0.7.0
DefaultDirName={commonpf}\Picasso
DefaultGroupName=Picasso
OutputBaseFilename="Picasso-Windows-64bit-0.6.11"
OutputBaseFilename="Picasso-Windows-64bit-0.7.0"
ArchitecturesAllowed=x64
ArchitecturesInstallIn64BitMode=x64

Expand Down
2 changes: 1 addition & 1 deletion docs/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@
# The short X.Y version
version = ""
# The full version, including alpha/beta/rc tags
release = "0.6.11"
release = "0.7.0"

# -- General configuration ---------------------------------------------------

Expand Down
2 changes: 1 addition & 1 deletion picasso/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
import os.path as _ospath
import yaml as _yaml

__version__ = "0.6.11"
__version__ = "0.7.0"

_this_file = _ospath.abspath(__file__)
_this_dir = _ospath.dirname(_this_file)
Expand Down
2 changes: 1 addition & 1 deletion picasso/__version__.py
Original file line number Diff line number Diff line change
@@ -1 +1 @@
VERSION_NO = "0.6.11"
VERSION_NO = "0.7.0"
103 changes: 64 additions & 39 deletions picasso/aim.py
Original file line number Diff line number Diff line change
Expand Up @@ -353,7 +353,8 @@ def get_fft_peak_z(roi_cc, roi_size):

def intersection_max(
x, y, ref_x, ref_y,
frame, segmentation, intersect_d, roi_r, width, progress=None,
frame, seg_bounds, intersect_d, roi_r, width,
aim_round=1, progress=None,
):
"""Maximize intersection (undrift) for 2D localizations.
Expand All @@ -369,15 +370,21 @@ def intersection_max(
y-coordinates of the reference localizations.
frame : _np.array
Frame indices of localizations.
segmentation : int
Time interval for drift tracking, unit: frames.
seg_bounds : np.array
Frame indices of the segmentation bounds. Defines temporal
intervals used to estimate drift.
intersect_d : float
Intersect distance in camera pixels.
roi_r : float
Radius of the local search region in camera pixels. Should be
higher than the maximum expected drift within one segment.
width : int
Width of the camera image in camera pixels.
aim_round : {1, 2}
Round of AIM algorithm. The first round uses the first interval
as reference, the second round uses the entire dataset as
reference. The impact is that in the second round, the first
interval is also undrifted.
progress : picasso.lib.ProgressDialog (default=None)
Progress dialog. If None, progress is displayed with tqdm.
Expand All @@ -393,8 +400,10 @@ def intersection_max(
Drift in y-direction.
"""

assert aim_round in [1, 2], "aim_round must be 1 or 2."

# number of segments
n_segments = int(_np.ceil(frame.max() / segmentation))
n_segments = len(seg_bounds) - 1
rel_drift_x = 0 # adaptive drift (updated at each interval)
rel_drift_y = 0

Expand All @@ -411,7 +420,7 @@ def intersection_max(
for i, shift_x in enumerate(steps):
for j, shift_y in enumerate(steps):
shifts_xy[i, j] = shift_x + shift_y * width_units
shifts_xy = shifts_xy.reshape(box**2)
shifts_xy = shifts_xy.reshape(box ** 2)

# convert reference to a 1D array in units of intersect_d and find
# unique values and counts
Expand All @@ -421,19 +430,30 @@ def intersection_max(
l0_coords, l0_counts = _np.unique(l0, return_counts=True)

# initialize progress such that if GUI is used, tqdm is omitted
start_idx = 1 if aim_round == 1 else 0
if progress is not None:
iterator = range(1, n_segments)
iterator = range(start_idx, n_segments)
else:
iterator = _tqdm(n_segments, desc="Undrifting z", unit="segment")
iterator = _tqdm(
range(start_idx, n_segments),
desc=f"Undrifting ({aim_round}/2)",
unit="segment",
)

# run across each segment
for s in iterator:
# get the target localizations within the current segment
min_frame_idx = frame > s * segmentation
max_frame_idx = frame <= (s + 1) * segmentation
min_frame_idx = frame > seg_bounds[s]
max_frame_idx = frame <= seg_bounds[s+1]
x1 = x[min_frame_idx & max_frame_idx]
y1 = y[min_frame_idx & max_frame_idx]

# skip if no reference localizations
if len(x1) == 0:
drift_x[s] = drift_x[s-1]
drift_y[s] = drift_y[s-1]
continue

# undrifting from the previous round
x1 += rel_drift_x
y1 += rel_drift_y
Expand Down Expand Up @@ -462,12 +482,10 @@ def intersection_max(
iterator.update(s - iterator.n)

# interpolate the drifts (cubic spline) for all frames
n_frames = n_segments * segmentation
drift_track_points = _np.linspace(0, n_frames, n_segments+1)
t = (drift_track_points[1:] + drift_track_points[:-1]) / 2
t = (seg_bounds[1:] + seg_bounds[:-1]) / 2
drift_x_pol = _InterpolatedUnivariateSpline(t, drift_x, k=3)
drift_y_pol = _InterpolatedUnivariateSpline(t, drift_y, k=3)
t_inter = _np.arange(n_frames) + 1
t_inter = _np.arange(seg_bounds[-1]) + 1
drift_x = drift_x_pol(t_inter)
drift_y = drift_y_pol(t_inter)

Expand All @@ -480,8 +498,8 @@ def intersection_max(

def intersection_max_z(
x, y, z, ref_x, ref_y, ref_z,
frame, segmentation, intersect_d, roi_r, width, height, pixelsize,
progress=None,
frame, seg_bounds, intersect_d, roi_r, width, height, pixelsize,
aim_round=1, progress=None,
):
"""Maximize intersection (undrift) for 3D localizations. Assumes
that x and y coordinates were already undrifted. x and y are in
Expand All @@ -494,7 +512,7 @@ def intersection_max_z(
ref_z = ref_z.copy() / pixelsize #TODO: remember to convert back to nm, also for drift_z

# number of segments
n_segments = int(_np.ceil(frame.max() / segmentation))
n_segments = len(seg_bounds) - 1
rel_drift_z = 0 # adaptive drift (updated at each interval)

# drift in z
Expand All @@ -521,20 +539,30 @@ def intersection_max_z(
l0_coords, l0_counts = _np.unique(l0, return_counts=True)

# initialize progress such that if GUI is used, tqdm is omitted
start_idx = 1 if aim_round == 1 else 0
if progress is not None:
iterator = range(1, n_segments)
iterator = range(start_idx, n_segments)
else:
iterator = _tqdm(n_segments, desc="Undrifting z", unit="segment")
iterator = _tqdm(
range(start_idx, n_segments),
desc=f"Undrifting z ({aim_round}/2)",
unit="segment",
)

# run across each segment
for s in iterator:
# get the target localizations within the current segment
min_frame_idx = frame > s * segmentation
max_frame_idx = frame <= (s + 1) * segmentation
min_frame_idx = frame > seg_bounds[s]
max_frame_idx = frame <= seg_bounds[s+1]
x1 = x[min_frame_idx & max_frame_idx]
y1 = y[min_frame_idx & max_frame_idx]
z1 = z[min_frame_idx & max_frame_idx]

# skip if no reference localizations
if len(x1) == 0:
drift_z[s] = drift_z[s-1]
continue

# undrifting from the previous round
z1 += rel_drift_z

Expand All @@ -561,11 +589,9 @@ def intersection_max_z(


# interpolate the drifts (cubic spline) for all frames
n_frames = n_segments * segmentation
drift_track_points = _np.linspace(0, n_frames, n_segments+1)
t = (drift_track_points[1:] + drift_track_points[:-1]) / 2
t = (seg_bounds[1:] + seg_bounds[:-1]) / 2
drift_z_pol = _InterpolatedUnivariateSpline(t, drift_z, k=3)
t_inter = _np.arange(n_frames) + 1
t_inter = _np.arange(seg_bounds[-1]) + 1
drift_z = drift_z_pol(t_inter)

# undrift the localizations
Expand Down Expand Up @@ -616,15 +642,14 @@ def aim(
width = info[0]["Width"]
height = info[0]["Height"]
pixelsize = info[1]["Pixelsize"]
n_frames = info[0]["Frames"]

# frames should start at 1
frame = locs["frame"] + 1
n_frames = _np.max(frame)

# group frames into track intervals and correct for rounding errors
n_segments = _np.floor(n_frames / segmentation).astype(int)
n_frames = n_segments * segmentation
frame[frame > n_frames] = n_frames # cap frame number
# find the segmentation bounds (temporal intervals)
seg_bounds = _np.concatenate((
_np.arange(0, n_frames, segmentation), [n_frames]
))

# get the reference localizations (first interval)
ref_x = locs["x"][frame <= segmentation]
Expand All @@ -634,23 +659,23 @@ def aim(
# the first run is with the first interval as reference
x_pdc, y_pdc, drift_x1, drift_y1 = intersection_max(
locs.x, locs.y, ref_x, ref_y,
frame, segmentation, intersect_d, roi_r, width,
progress=progress,
frame, seg_bounds, intersect_d, roi_r, width,
aim_round=1, progress=progress,
)
# the second run is with the entire dataset as reference
if progress is not None:
progress.zero_progress(description="Undrifting by AIM (2/2)")
x_pdc, y_pdc, drift_x2, drift_y2 = intersection_max(
x_pdc, y_pdc, x_pdc, y_pdc,
frame, segmentation, intersect_d, roi_r, width,
progress=progress,
frame, seg_bounds, intersect_d, roi_r, width,
aim_round=2, progress=progress,
)

# add the drifts together from the two rounds
drift_x = drift_x1 + drift_x2
drift_y = drift_y1 + drift_y2

# shift the drifts by the mean value (like in Picasso)
# # shift the drifts by the mean value
drift_x -= _np.mean(drift_x)
drift_y -= _np.mean(drift_y)

Expand All @@ -666,15 +691,15 @@ def aim(
ref_z = locs.z[frame <= segmentation]
z_pdc, drift_z1 = intersection_max_z(
x_pdc, y_pdc, locs.z, ref_x, ref_y, ref_z,
frame, segmentation, intersect_d, roi_r, width, height, pixelsize,
progress=progress,
frame, seg_bounds, intersect_d, roi_r, width, height, pixelsize,
aim_round=1, progress=progress,
)
if progress is not None:
progress.zero_progress(description="Undrifting z (2/2)")
z_pdc, drift_z2 = intersection_max_z(
x_pdc, y_pdc, z_pdc, x_pdc, y_pdc, z_pdc,
frame, segmentation, intersect_d, roi_r, width, height, pixelsize,
progress=progress,
frame, seg_bounds, intersect_d, roi_r, width, height, pixelsize,
aim_round=2, progress=progress,
)
drift_z = drift_z1 + drift_z2
drift_z -= _np.mean(drift_z)
Expand Down
2 changes: 1 addition & 1 deletion release/one_click_windows_gui/create_installer_windows.bat
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ call conda activate picasso_installer
call python setup.py sdist bdist_wheel

call cd release/one_click_windows_gui
call pip install "../../dist/picassosr-0.6.11-py3-none-any.whl"
call pip install "../../dist/picassosr-0.7.0-py3-none-any.whl"

call pip install pyinstaller==5.7
call pyinstaller ../pyinstaller/picasso.spec -y --clean
Expand Down
4 changes: 2 additions & 2 deletions release/one_click_windows_gui/picasso_innoinstaller.iss
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
[Setup]
AppName=Picasso
AppPublisher=Jungmann Lab, Max Planck Institute of Biochemistry
AppVersion=0.6.11
AppVersion=0.7.0
DefaultDirName={commonpf}\Picasso
DefaultGroupName=Picasso
OutputBaseFilename="Picasso-Windows-64bit-0.6.11"
OutputBaseFilename="Picasso-Windows-64bit-0.7.0"
ArchitecturesAllowed=x64
ArchitecturesInstallIn64BitMode=x64

Expand Down
2 changes: 1 addition & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@

setup(
name="picassosr",
version="0.6.11",
version="0.7.0",
author="Joerg Schnitzbauer, Maximilian T. Strauss, Rafal Kowalewski",
author_email=("[email protected], [email protected], [email protected]"),
url="https://github.com/jungmannlab/picasso",
Expand Down

0 comments on commit 6eb3536

Please sign in to comment.