Source code for dynamo.movie.fate

import numpy as np
import warnings
from scipy.integrate import odeint
from ..vectorfield.scVectorField import vectorfield
from ..plot.topography import topography
from .utils import remove_particles


[docs]class StreamFuncAnim(): """Animating cell fate commitment prediction via reconstructed vector field function."""
[docs] def __init__(self, adata, basis='umap', dims=None, n_steps=100, cell_states=None, color='ntr', ax=None, ln=None, logspace=False, ): """Animating cell fate commitment prediction via reconstructed vector field function. This class creates necessary components to produce an animation that describes the exact speed of a set of cells at each time point, its movement in gene expression and the long range trajectory predicted by the reconstructed vector field. Thus it provides intuitive visual understanding of the RNA velocity, speed, acceleration, and cell fate commitment in action. This function is originally inspired by https://tonysyu.github.io/animating-particles-in-a-flow.html and relies on animation module from matplotlib. Note that you may need to install `imagemagick` in order to properly show or save the animation. See for example, http://louistiao.me/posts/notebooks/save-matplotlib-animations-as-gifs/ for more details. Parameters ---------- adata: :class:`~anndata.AnnData` AnnData object that already went through the fate prediction. basis: `str` or None (default: `None`) The embedding data to use for predicting cell fate. If `basis` is either `umap` or `pca`, the reconstructed trajectory will be projected back to high dimensional space via the `inverse_transform` function. space. dims: `list` or `None` (default: `None') The dimensions of low embedding space where cells will be drawn and it should corresponds to the space fate prediction take place. n_steps: `int` (default: `100`) The number of times steps (frames) fate prediction will take. cell_states: `int`, `list` or `None` (default: `None`) The number of cells state that will be randomly selected (if `int`), the indices of the cells states (if `list`) or all cell states which fate prediction executed (if `None`) ax: `matplotlib.Axis` (optional, default `None`) The matplotlib axes object that will be used as background plot of the vector field animation. ln: `tuple` or None (default: `None`) An iterable of artists (for example, `matplotlib.lines.Line2D`) used to draw a clear frame. logspace: `bool` (default: `False`) Whether or to sample time points linearly on log space. If not, the sorted unique set of all time points from all cell states' fate prediction will be used and then evenly sampled up to `n_steps` time points. Returns ------- A class that contains .fig attribute and .update, .init_background that can be used to produce an animation of the prediction of cell fate commitment. Examples 1 ---------- >>> from matplotlib import animation >>> progenitor = adata.obs_names[adata.obs.clusters == 'cluster_1'] >>> fate_progenitor = progenitor >>> info_genes = adata.var_names[adata.var.use_for_velocity] >>> dyn.pd.fate(adata, basis='umap', init_cells=fate_progenitor, interpolation_num=100, direction='forward', ... inverse_transform=False, average=False, arclen_sampling=True) >>> instance = dyn.pl.StreamFuncAnim(adata=adata, ax=None, ln=None) >>> anim = animation.FuncAnimation(instance.fig, instance.update, init_func=instance.init_background, ... frames=np.arange(100), interval=100, blit=True) >>> from IPython.core.display import display, HTML >>> HTML(anim.to_jshtml()) # embedding to jupyter notebook. >>> anim.save('fate_ani.gif',writer="imagemagick") # save as gif file. Examples 2 ---------- >>> from matplotlib import animation >>> progenitor = adata.obs_names[adata.obs.clusters == 'cluster_1'] >>> fate_progenitor = progenitor >>> info_genes = adata.var_names[adata.var.use_for_velocity] >>> dyn.pd.fate(adata, basis='umap', init_cells=fate_progenitor, interpolation_num=100, direction='forward', ... inverse_transform=False, average=False, arclen_sampling=True) >>> fig, ax = plt.subplots() >>> ln, = ax.plot([], [], 'ro') >>> ax.set_xlim(xlim) >>> ax.set_ylim(ylim) >>> instance = dyn.pl.StreamFuncAnim(adata=adata, ax=ax, ln=ln) >>> anim = animation.FuncAnimation(fig, instance.update, init_func=instance.init_background, ... frames=np.arange(100), interval=100, blit=True) >>> from IPython.core.display import display, HTML >>> HTML(anim.to_jshtml()) # embedding to jupyter notebook. >>> anim.save('fate_ani.gif',writer="imagemagick") # save as gif file. Examples 3 ---------- >>> from matplotlib import animation >>> progenitor = adata.obs_names[adata.obs.clusters == 'cluster_1'] >>> fate_progenitor = progenitor >>> info_genes = adata.var_names[adata.var.use_for_velocity] >>> dyn.pd.fate(adata, basis='umap', init_cells=fate_progenitor, interpolation_num=100, direction='forward', ... inverse_transform=False, average=False, arclen_sampling=True) >>> dyn.pl.fate_animation(adata) See also:: :func:`animate_fates` """ import matplotlib.pyplot as plt self.adata = adata self.basis = basis fate_key = 'fate_' + basis if fate_key not in adata.uns_keys(): raise Exception(f"You need to first perform fate prediction before animate the prediction, please run" f"dyn.pd.fate(adata, basis='{basis}' before running this function") self.init_states = adata.uns[fate_key]['init_states'] # self.prediction = adata.uns['fate_umap']['prediction'] self.t = adata.uns[fate_key]['t'] flat_list = np.unique([item for sublist in self.t for item in sublist]) flat_list = np.hstack((0, flat_list)) flat_list = np.sort(flat_list) self.logspace = logspace if self.logspace: self.time_vec = np.logspace(0, np.log10(max(flat_list) + 1), n_steps) - 1 else: self.time_vec = flat_list[(np.linspace(0, len(flat_list) - 1, n_steps)).astype(int)] # init_states, VecFld, t_end, _valid_genes = fetch_states( # adata, init_states, init_cells, basis, layer, False, # t_end # ) n_states = self.init_states.shape[0] if n_states > 50: warnings.warn(f'the number of cell states with fate prediction is more than 50. You may want to ' f'lower the max number of cell states to draw via cell_states argument.') if cell_states is not None: if type(cell_states) is int: self.init_states = self.init_states[np.random.choice(range(n_states), min(n_states, cell_states))] elif type(cell_states) is list: self.init_states = self.init_states[cell_states] else: self.init_states = self.init_states # vf = lambda x: vector_field_function(x=x, vf_dict=VecFld) vf = vectorfield() vf.from_adata(adata, basis=basis) # Initialize velocity field and displace *functions* self.f = lambda x, _: vf.func(x) # scale * self.displace = lambda x, dt: odeint(self.f, x, [0, dt]) # Save bounds of plot X_data = adata.obsm['X_' + basis][:, :2] if dims is None else adata.obsm['X_' + basis][:, dims] m, M = np.min(X_data, 0), np.max(X_data, 0) m = m - 0.01 * np.abs(M - m) M = M + 0.01 * np.abs(M - m) self.xlim = [m[0], M[0]] self.ylim = [m[1], M[1]] # Animation objects must create `fig` and `ax` attributes. if ax is None or ln is None: self.fig, self.ax = plt.subplots() self.ln, = self.ax.plot([], [], 'ro') else: self.ax = ax self.ln = ln self.ax.set_aspect("equal") self.color = color
def init_background(self): self.ax = topography(self.adata, basis=self.basis, color=self.color, ax=self.ax, save_show_or_return='return') return self.ln, def update(self, frame): """Update locations of "particles" in flow on each frame frame.""" print(frame) init_states = self.init_states time_vec = self.time_vec pts = [i.tolist() for i in init_states] if frame == 0: x, y = init_states.T self.ax.lines = [] (self.ln,) = self.ax.plot(x, y, "ro", zorder=20) return self.ln, # return line so that blit works properly else: pts = [ self.displace(cur_pts, time_vec[frame])[1].tolist() for cur_pts in pts ] pts = np.asarray(pts) pts = np.asarray(pts) pts = remove_particles(pts, self.xlim, self.ylim) x, y = np.asarray(pts).transpose() self.ax.lines = [] (self.ln,) = self.ax.plot(x, y, "ro", zorder=20) self.ax.set_title("current vector field time is: {:12.2f}".format(time_vec[frame] - time_vec[frame - 1])) # anim.event_source.interval = (time_vec[frame] - time_vec[frame - 1]) / 100 return self.ln, # return line so that blit works properly
[docs]def animate_fates(adata, basis='umap', dims=None, n_steps=100, cell_states=None, color='ntr', ax=None, ln=None, logspace=False, interval=100, blit=True, save_show_or_return='show', save_kwargs={}, **kwargs): """Animating cell fate commitment prediction via reconstructed vector field function. This class creates necessary components to produce an animation that describes the exact speed of a set of cells at each time point, its movement in gene expression and the long range trajectory predicted by the reconstructed vector field. Thus it provides intuitive visual understanding of the RNA velocity, speed, acceleration, and cell fate commitment in action. This function is originally inspired by https://tonysyu.github.io/animating-particles-in-a-flow.html and relies on animation module from matplotlib. Note that you may need to install `imagemagick` in order to properly show or save the animation. See for example, http://louistiao.me/posts/notebooks/save-matplotlib-animations-as-gifs/ for more details. Parameters ---------- adata: :class:`~anndata.AnnData` AnnData object that already went through the fate prediction. basis: `str` or None (default: `None`) The embedding data to use for predicting cell fate. If `basis` is either `umap` or `pca`, the reconstructed trajectory will be projected back to high dimensional space via the `inverse_transform` function. space. dims: `list` or `None` (default: `None') The dimensions of low embedding space where cells will be drawn and it should corresponds to the space fate prediction take place. n_steps: `int` (default: `100`) The number of times steps (frames) fate prediction will take. cell_states: `int`, `list` or `None` (default: `None`) The number of cells state that will be randomly selected (if `int`), the indices of the cells states (if `list`) or all cell states which fate prediction executed (if `None`) ax: `matplotlib.Axis` (optional, default `None`) The matplotlib axes object that will be used as background plot of the vector field animation. ln: `tuple` or None (default: `None`) An iterable of artists (for example, `matplotlib.lines.Line2D`) used to draw a clear frame. logspace: `bool` (default: `False`) Whether or to sample time points linearly on log space. If not, the sorted unique set of all time points from all cell states' fate prediction will be used and then evenly sampled up to `n_steps` time points. interval: `float` (default: `200`) Delay between frames in milliseconds. blit: `bool` (default: `False`) Whether blitting is used to optimize drawing. Note: when using blitting, any animated artists will be drawn according to their zorder; however, they will be drawn on top of any previous artists, regardless of their zorder. save_show_or_return: `str` {'save', 'show', 'return'} (default: `save`) Whether to save, show or return the figure. By default a gif will be used. save_kwargs: `dict` (default: `{}`) A dictionary that will passed to the anim.save. By default it is an empty dictionary and the save_fig function will use the {"filename": 'fate_ani.gif', "writer": "imagemagick"} as its parameters. Otherwise you can provide a dictionary that properly modify those keys according to your needs. see https://matplotlib.org/api/_as_gen/matplotlib.animation.Animation.save.html for more details. kwargs: Additional arguments passed to animation.FuncAnimation. Returns ------- Nothing but produce an animation that will be embedded to jupyter notebook or saved to disk. Examples 1 ---------- >>> from matplotlib import animation >>> progenitor = adata.obs_names[adata.obs.clusters == 'cluster_1'] >>> fate_progenitor = progenitor >>> info_genes = adata.var_names[adata.var.use_for_velocity] >>> dyn.pd.fate(adata, basis='umap', init_cells=fate_progenitor, interpolation_num=100, direction='forward', ... inverse_transform=False, average=False, arclen_sampling=True) >>> dyn.pl.fate_animation(adata) See also:: :func:`StreamFuncAnim` """ from matplotlib import animation instance = StreamFuncAnim(adata=adata, basis=basis, dims=dims, n_steps=n_steps, cell_states=cell_states, color=color, ax=ax, ln=ln, logspace=logspace, ) anim = animation.FuncAnimation(instance.fig, instance.update, init_func=instance.init_background, frames=np.arange(n_steps), interval=interval, blit=blit, **kwargs) if save_show_or_return == 'save': save_kwargs_ = {"filename": 'fate_ani.gif', "writer": "imagemagick"} save_kwargs_.update(save_kwargs) anim.save(**save_kwargs_) # save as gif file. elif save_show_or_return == 'show': from IPython.core.display import HTML HTML(anim.to_jshtml()) # embedding to jupyter notebook. else: anim