The TrajectoryUtils package contains low-level utilities to support trajectory inference packages. Support is the key word here: this package does not perform any inference itself, deferring that to higher-level packages like TSCAN and slingshot. In fact, most of the code in this package was factored out of those two packages after we realized their commonalities. By rolling this out into a separate package, we can cut down on redundancy, facilitate better propagation of features and simplify maintenance. If you’re a developer and you have a general utility for trajectory inference, contributions are welcome.
The construction of a cluster-based minium spanning tree is the primary motivator for this package, as it was implemented separately in TSCAN and slingshot. The idea is simple - cluster cells and create a minimum spanning tree from the cluster centroids to serve as the backbone for the trajectory reconstruction.
# Mocking up a Y-shaped trajectory.
centers <- rbind(c(0,0), c(0, -1), c(1, 1), c(-1, 1))
rownames(centers) <- seq_len(nrow(centers))
clusters <- sample(nrow(centers), 1000, replace=TRUE)
cells <- centers[clusters,]
cells <- cells + rnorm(length(cells), sd=0.5)
# Creating the MST:
library(TrajectoryUtils)
mst <- createClusterMST(cells, clusters)
plot(mst)
The implementation in createClusterMST()
combines
several useful ideas from the two aforementioned packages:
outgroup=TRUE
, users can add an outgroup to break
apart distant clusters. This ensures that the MST does not form spurious
links between unrelated parts of the dataset.dist.method="mnn"
, users can create the MST based
on the distances between mutually nearest neighbors. This avoids
penalizing the formation of edges between adjacent heterogeneous
clusters.dist.method="slingshot"
, the Mahalanobis distance
is computed from the covariance matrix for cells in each cluster. This
accounts for the shape and spread of clusters when constructing the
MST.use.median=TRUE
, the centroids are computed by
taking the median instead of the mean. This protects against clusters
with many outliers.Once the MST is constructed, we can use the
guessMSTRoots()
function to guess the possible roots. One
root is guessed for each component of the MST, though the choice of root
depends on the method
in use. For example, we can choose a
root node that maximizes the steps/distance to all terminal (degree-1)
nodes; this results in fewer branch events when defining paths through
the MST.
## [1] "2"
## [1] "1"
Once a root is guessed, we can define paths from the root to all
terminal nodes using the defineMSTPaths()
function. The
interpretation of each path depends on the appropriateness of the chosen
root; for example, in developmental contexts, each path can be treated
as a lineage if the root corresponds to some undifferentiated state.
## [[1]]
## [1] "1" "2"
##
## [[2]]
## [1] "1" "3"
##
## [[3]]
## [1] "1" "4"
## [[1]]
## [1] "2" "1" "3"
##
## [[2]]
## [1] "2" "1" "4"
Alternatively, if timing information is available (e.g., from an experimental time-course, or from computational methods like RNA velocity), we can use that to define paths through the MST based on the assumed movement of cells from local minima to the nearest local maxima in time. In the mock example below, the branch point at cluster 1 has the highest time so all paths converge from the terminal nodes to cluster 1.
## [[1]]
## [1] "2" "1"
##
## [[2]]
## [1] "3" "1"
##
## [[3]]
## [1] "4" "1"
We can also use a root-free method of defining paths through the MST, by simply defining all unbranched segments as separate paths. This decomposes the MST into simpler structures for, e.g., detection of marker genes, without explicitly defining a root or requiring external information. In fact, one might prefer this approach as it allows us to interpret each section of the MST in a more modular manner, without duplication of effort caused by the sharing of nodes across different paths from a common root.
## [[1]]
## [1] "2" "1"
##
## [[2]]
## [1] "3" "1"
##
## [[3]]
## [1] "4" "1"
PseudotimeOrdering
classWe can create a compact representation of pseudotime orderings in the
form of a matrix where rows are cells and columns are paths through the
trajectory. Each cell receives a pseudotime value along a path - or
NA
, if the cell does not lie on that path. On occasion, we
may wish to annotate this with metadata on the cells or on the paths.
This is supported by the PseudotimeOrdering
class:
# Make up a matrix of pseudotime orderings.
ncells <- 200
npaths <- 5
orderings <- matrix(rnorm(1000), ncells, npaths)
# Default constructor:
(pto <- PseudotimeOrdering(orderings))
## class: PseudotimeOrdering
## dim: 200 5
## metadata(0):
## pathStats(1): ''
## cellnames: NULL
## cellData names(0):
## pathnames: NULL
## pathData names(0):
It is then straightforward to add metadata on the cells:
## DataFrame with 200 rows and 1 column
## cluster
## <character>
## 1 B
## 2 E
## 3 E
## 4 C
## 5 E
## ... ...
## 196 C
## 197 D
## 198 C
## 199 E
## 200 E
Or on the paths:
## DataFrame with 5 rows and 1 column
## description
## <character>
## 1 PATH-1
## 2 PATH-2
## 3 PATH-3
## 4 PATH-4
## 5 PATH-5
Additional matrices can also be added if multiple statistics are available for each cell/path combination. For example, one might imagine assigning the confidence to which each cell is assigned to each path.
## [1] "" "confidence"
For convenience, we also provide the averagePseudotime()
function to compute the average pseudotime for each cell. This is
sometimes necessary in applications that only expect a single set of
pseudotime values, e.g., for visualization.
## Min. 1st Qu. Median Mean 3rd Qu. Max.
## -0.963549 -0.304059 0.009442 -0.014155 0.285123 1.226246
## R version 4.4.2 (2024-10-31)
## Platform: x86_64-pc-linux-gnu
## Running under: Ubuntu 24.04.1 LTS
##
## Matrix products: default
## BLAS: /usr/lib/x86_64-linux-gnu/openblas-pthread/libblas.so.3
## LAPACK: /usr/lib/x86_64-linux-gnu/openblas-pthread/libopenblasp-r0.3.26.so; LAPACK version 3.12.0
##
## locale:
## [1] LC_CTYPE=en_US.UTF-8 LC_NUMERIC=C
## [3] LC_TIME=en_US.UTF-8 LC_COLLATE=C
## [5] LC_MONETARY=en_US.UTF-8 LC_MESSAGES=en_US.UTF-8
## [7] LC_PAPER=en_US.UTF-8 LC_NAME=C
## [9] LC_ADDRESS=C LC_TELEPHONE=C
## [11] LC_MEASUREMENT=en_US.UTF-8 LC_IDENTIFICATION=C
##
## time zone: Etc/UTC
## tzcode source: system (glibc)
##
## attached base packages:
## [1] stats4 stats graphics grDevices utils datasets methods
## [8] base
##
## other attached packages:
## [1] TrajectoryUtils_1.15.0 SingleCellExperiment_1.29.1
## [3] SummarizedExperiment_1.37.0 Biobase_2.67.0
## [5] GenomicRanges_1.59.0 GenomeInfoDb_1.43.1
## [7] IRanges_2.41.1 S4Vectors_0.45.2
## [9] BiocGenerics_0.53.3 generics_0.1.3
## [11] MatrixGenerics_1.19.0 matrixStats_1.4.1
## [13] knitr_1.49 BiocStyle_2.35.0
##
## loaded via a namespace (and not attached):
## [1] Matrix_1.7-1 jsonlite_1.8.9 compiler_4.4.2
## [4] BiocManager_1.30.25 crayon_1.5.3 jquerylib_0.1.4
## [7] yaml_2.3.10 fastmap_1.2.0 lattice_0.22-6
## [10] R6_2.5.1 XVector_0.47.0 S4Arrays_1.7.1
## [13] igraph_2.1.1 DelayedArray_0.33.2 maketools_1.3.1
## [16] GenomeInfoDbData_1.2.13 bslib_0.8.0 rlang_1.1.4
## [19] cachem_1.1.0 xfun_0.49 sass_0.4.9
## [22] sys_3.4.3 SparseArray_1.7.2 cli_3.6.3
## [25] magrittr_2.0.3 zlibbioc_1.52.0 grid_4.4.2
## [28] digest_0.6.37 lifecycle_1.0.4 glue_1.8.0
## [31] evaluate_1.0.1 buildtools_1.0.0 abind_1.4-8
## [34] rmarkdown_2.29 httr_1.4.7 pkgconfig_2.0.3
## [37] tools_4.4.2 htmltools_0.5.8.1 UCSC.utils_1.3.0