Single cell and subcellular analysis on Seqscope#

Date:

2022-11-15

Dataset explanation#

Seqscope is a illumina sequencing based spatial sequencing platform developed by Jun Hee Lee Lab. The basic strategy is to use illumina sequencing by synthesis to generate the spatial barcodes and use the barcodes to capture mRNAs in tissue.

Example Raw Data needed for seqscope:#

1st-seq data (single-ended, for generating spatial barcodes)

  • SeqScope_1st.fastq.gz

2nd-seq data (pair-ended, for generating count matrix)

  • SeqScope_2nd_R1.fastq.gz

  • SeqScope_2nd_R2.fastq.gz

Image (for seqmentation)

  • Tile_No_Segmented.png

Preprocessing to generate count matrix#

Seqscope has its own pipeline to generate the count matrix(Gene * barcode). Please refer to their github page or use customized methods to proceed.

#Note you need to install seqtk, STARsolo and clone their github page first. All bash scripts are stored in script directory of their github.

## First generate the whitelist of spatial barcodes:
bash extractCoord.sh [SeqScope_1st.fastq.gz] [SeqScope_2nd_R1.fastq.gz] [HDMI_length]
## Second, generating the count matrix via STAR.Solo:
bash align.sh [SeqScope_2nd_R1.fastq.gz] [abc_SeqScope_2nd_R2.fastq.gz] [HDMI_length] [whitelists.txt] [outprefix] [starpath] [seqtkpath] [geneIndex]

You should now have the count matrix for Giotto Object. Note the Spatial Barcodes are typically at 0.6 um resolution. One way to analyze is to use the getSimpleGrid.R or getSlidingGrid.R provided by Lee Lab to bin the counts and simply follow the Giotto analysis steps of Visium and Integration of Visium

Start Giotto#

# Ensure Giotto Suite is installed.
if(!"Giotto" %in% installed.packages()) {
  devtools::install_github("drieslab/Giotto@suite")
}

library(Giotto)
# Ensure the Python environment for Giotto has been installed.
genv_exists = checkGiottoEnvironment()
if(!genv_exists){
  # The following command need only be run once to install the Giotto environment.
  installGiottoEnvironment()
}

Set up Giotto Environment#

# 1. set working directory
results_folder = 'path/to/result'

# 2. set giotto python path
# set python path to your preferred python version path
# set python path to conda env/bin/ directory if manually installed Giotto python dependencies by conda
# python_path = '/path_to_conda/.conda/envs/giotto/bin/python'
# set python path to NULL if you want to automatically install (only the 1st time) and use the giotto miniconda environment
# python_path = NULL
#if(is.null(python_path)) {
#  installGiottoEnvironment()
#}

# 3. create giotto instructions
instrs = createGiottoInstructions(save_dir = results_folder,
                                  save_plot = TRUE,
                                  show_plot = FALSE)

1 Create Giotto Subcellular Object#

In order to do single cell and subcellular analysis with Giotto, we suggest using the GiottoSubcellular Object. Giotto Subcellular objects take cell polygons and a Giotto Point file (a data.table contains Gene Name, Expression, x, y, etc). Therefore, first we need to do some transformation of the count matrix.

Giotto Point file formatting

Unlike normal cell by gene matrix, a Giotto Point file is usually a data.table contains Gene Name, Expression, x, y, where each row represent a subcellular point. For Seqscope data, one HDMI typically have more than one subcellular point.

Example Giotto Point file for Seqscope#

HDMI

Feat_ID

Count

sdimX

sdimY

HDMI1

GeneA

HDMI1

GeneB

HDMI2

GeneA

HDMI3

GeneC

1.1 Process Giotto Point file per tile#

##expression matrix
countDir = "/path/to/Solo.out/GeneFull/raw"
expr_matrix = Giotto::get10Xmatrix(path_to_data = countDir, gene_column_index = 2)

##Spatial coordinates
spatial_coords_Dir = "/path/to/extractCoord.sh/results/spatialcoordinates.txt"
spatial_coords = fread("spatial_coords_Dir")
colnames(spatial_coords)<-c("HDMI","Lane","Tile","X","Y")

##Prepare Giotto Point
# Subset expression and spatial info by tile
spatial_coords_tile = spatial_coords[Tile == '2104']
expr_matrix_tile = expr_matrix[, as.character(colnames(expr_matrix)) %in% spatial_coords_tile$HDMI]

# convert expression matrix to minimal data.table object
matrix_tile_dt = as.data.table(Matrix::summary(expr_matrix_tile))
genes = expr_matrix_tile@Dimnames[[1]]
samples = expr_matrix_tile@Dimnames[[2]]
matrix_tile_dt[, gene := genes[i]]
matrix_tile_dt[, hdmi := samples[j]]

# merge data.table matrix and spatial coordinates to create input for Giotto Polygons
gpoints = merge.data.table(matrix_tile_dt, spatial_coords_tile, by.x = 'hdmi', by.y = 'HDMI')
gpoints = gpoints[,.(hdmi, X, Y, gene, x)]
colnames(gpoints) = c('hdmi', 'x', 'y', 'gene', 'counts')

1.2 Prepare the polygon mask file#

Giotto can read in a variety of different mask files provided by common segmentation tools. But first we need to check if we need to filp the x and y axis.

# check total counts per hdmi
gpoints_aggr = gpoints[, sum(counts), by = .(hdmi, x, y)]
colnames(gpoints_aggr) = c("hdmi","x","y","total_counts")
setorder(gpoints_aggr, -total_counts)

pl = ggplot()
pl = pl + geom_point(data = gpoints_aggr[total_counts < 1000 & total_counts > 8], aes(x = x, y = y, color = total_counts), size = 0.05)
pl = pl + scale_color_gradient2(midpoint = 15, low = 'blue', mid = 'yellow', high = 'red')
pl
../../_images/HDMI_scatter.png

Read polygon mask file

segmentation_mask = "/path/to/segmentation.tif"
final_polygons = createGiottoPolygonsFromMask(segmentation_mask,
                               flip_vertical = FALSE,
                               flip_horizontal = FALSE)
plot(final_polygons)
../../_images/polygon.png

Polygon mask file manual formatting

However, sometimes Giotto does not read in the provided mask file and we will need a manual processing for the mask file and we will do that using terra and createGiottoPolygonsFromDfr.

raster = terra::rast(x = segmentation_mask)
terra_polygon = terra::as.polygons(x = raster, value = T)

# convert polygon to data.table and remove unwantedly detected polygons (e.g. canvas etc)
dt = Giotto:::spatVector_to_dt(terra_polygon)
npolygons = length(levels(factor(dt$part))) - 1
filter_dt = dt[geom == 1 & hole == 0 & part %in% c(0:npolygons), .(x,y,part)]
filter_dt[, part := as.factor(part)]

# create new polygons from filtered data.table
final_polygons = createGiottoPolygonsFromDfr(segmdfr = filter_dt)

# create giotto points first to get the extent of the points (hdmi)
original_points = createGiottoPoints(x = gpoints[,.(x, y, gene, hdmi, counts)])
original_feat_ext = ext(original_points@spatVector)



# convert polygon to spatRaster to change extent to that of original points
final_spatraster = Giotto:::polygon_to_raster(polygon = final_polygons@spatVector)
ext(final_spatraster$raster) = original_feat_ext
final_polygons@spatVector = as.polygons(final_spatraster$raster)
final_polygons@spatVector$poly_ID = final_spatraster$ID_vector[final_polygons@spatVector$poly_i]

# flip and shift, if needed
#final_polygons@spatVector = flip(final_polygons@spatVector)
#yshift = ymin(original_feat_ext) - ymax(original_feat_ext)
#final_polygons@spatVector = terra::shift(final_polygons@spatVector, dy = -yshift)

plot(final_polygons)

1.3 Create Giotto Object#

Add a random jitter to the HDMI location to make a pseudo-in situ transcript file.

# add giotto points class
gpoints_subset = gpoints[hdmi %in% gpoints_aggr[total_counts > 5]$hdmi]

# multiply rows with multiple counts and add jitter
gpoints_extra = gpoints_subset[counts > 1]
gpoints_extra = gpoints_extra[,rep(counts, counts), by = .(hdmi, gene, x, y)]
gpoints_extra = rbind(gpoints_extra[,.(hdmi, gene, x, y)], gpoints_subset[counts == 1 ,.(hdmi, gene, x, y)])
jitter_x = sample(1:3, size = nrow(gpoints_extra), replace = T)
jitter_y = sample(1:3, size = nrow(gpoints_extra), replace = T)
gpoints_extra[, x := x + jitter_x]
gpoints_extra[, y := y + jitter_y]
# add subcellular information
seqscope = createGiottoObjectSubcellular(gpoints = list(gpoints_extra[,.(x, y, gene, hdmi)]),
                                         gpolygons = list(final_polygons),
                                         instructions = instrs)

# add centroids
seqscope = addSpatialCentroidLocations(seqscope,
                                       poly_info = 'cell')

#Overlap to Polygon information
seqscope = calculateOverlapRaster(seqscope)
seqscope = overlapToMatrix(seqscope)

# Visualize top 200 expressed genes in situ
spatInSituPlotPoints(seqscope, show_legend = F,
                     show_image = FALSE,
                     feats = list('rna' = seqscope@feat_info$rna@spatVector$feat_ID[1:200]),
                     spat_unit = 'cell',
                     point_size = 1,
                     show_polygon = TRUE,
                     use_overlap = F,
                     polygon_feat_type = 'cell',
                     polygon_color = 'red',
                     polygon_bg_color = 'white',
                     polygon_line_size = 0.2,
                     coord_fix_ratio = TRUE,
                     background_color = 'white')
../../_images/top200.png

2 Process Giotto and Quality Control#

# filter
seqscope <- filterGiotto(gobject = seqscope,
                         expression_threshold = 1,
                         feat_det_in_min_cells = 5,
                         min_det_feats_per_cell = 5)

#normalize
seqscope <- normalizeGiotto(gobject = seqscope, scalefactor = 5000, verbose = T)
# add statistics
seqscope <- addStatistics(gobject = seqscope)

# View cellular data
# pDataDT(seqscope)
# View rna data
# fDataDT(seqscope)


spatPlot2D(gobject = seqscope,
           cell_color = 'total_expr', color_as_factor = F,
           show_image = F,
           point_size = 2.5, point_alpha = 0.75, coord_fix_ratio = T)
../../_images/total_expr.png
cellmeta = pDataDT(seqscope, feat_type = 'rna')
hist(cellmeta$nr_feats, 100)
../../_images/feature_distribution.png

3 Dimention Reduction#

# cluster cells
seqscope <- calculateHVF(gobject = seqscope, HVFname = 'hvg_orig')

seqscope <- runPCA(gobject = seqscope,
                   expression_values = 'normalized',
                   scale_unit = T, center = T)

seqscope <- runUMAP(seqscope, dimensions_to_use = 1:100)

4 Cluster#

seqscope <- createNearestNetwork(gobject = seqscope, dimensions_to_use = 1:100, k = 5)
seqscope <- doLeidenCluster(gobject = seqscope, resolution = 0.9, n_iterations = 1000)

# visualize UMAP cluster results


plotUMAP(gobject = seqscope, cell_color = 'leiden_clus',
         show_NN_network = F, point_size = 3.5)
../../_images/leiden_umap.png
spatInSituPlotPoints(seqscope,
                     show_polygon = TRUE,
                     polygon_color = 'white',
                     polygon_line_size = 0.1,
                     polygon_fill = 'leiden_clus',
                     polygon_fill_as_factor = T,
                     coord_fix_ratio = T)
../../_images/leiden_spat.png

5 find spatial genes#

seqscope<-createSpatialNetwork(gobject = seqscope, minimum_k = 2, maximum_distance_delaunay = 100)

km_spatialgenes = binSpect(seqscope, subset_feats = seqscope@feat_ID$rna)
spatFeatPlot2D(seqscope, expression_values = 'scaled',
               feats = km_spatialgenes$feats[1:2],
               cell_color_gradient = c('blue', 'white', 'red'),
               point_shape = 'border', point_border_stroke = 0.01,
               show_network = T, network_color = 'lightgrey', point_size = 1.2,
               cow_n_col = 1)
../../_images/spatgenes.png