Advanced usage of scp

About this vignette

This vignette is dedicated to advanced users and to method developers. It assumes that you are already familiar with QFeatures and scp and that you are looking for more flexibility in the analysis of your single-cell proteomics (SCP) data. In fact, scp provides wrapper functions around generic functions and metrics. However, advanced users may want to apply or develop their own features. The QFeatures class offers a flexible data container while guaranteeing data consistency.

In this vignette, you will learn how to:

  • Modify the quantitative data
  • Modify the sample annotations
  • Modify the feature annotations
  • Create a new function for scp

As a general guideline, you can add/remove/update data in a QFeatures in 4 main steps:

  1. Gather the data to change and other required data involved in the processing.
  2. Apply the transformation/computation.
  3. Insert the changes in the QFeatures object.
  4. Make sure the updated QFeatures object is still valid.

To illustrate the different topics, we will load the scp1 example data.

library(scp)
data("scp1")
scp1
#> An instance of class QFeatures containing 5 assays:
#>  [1] 190321S_LCA10_X_FP97AG: SingleCellExperiment with 166 rows and 11 columns 
#>  [2] 190222S_LCA9_X_FP94BM: SingleCellExperiment with 176 rows and 11 columns 
#>  [3] 190914S_LCB3_X_16plex_Set_21: SingleCellExperiment with 215 rows and 16 columns 
#>  [4] peptides: SingleCellExperiment with 539 rows and 38 columns 
#>  [5] proteins: SingleCellExperiment with 292 rows and 38 columns

Modify the quantitative data

To illustrate how to modify quantitative data, we will implement a normByType function that will normalize the feature (row) for each cell type separately. This function is probably not relevant for a real case analysis, but it provides a good example of a custom data processing. The process presented in this section is applicable to any custom function that takes at least a matrix-like object as input and returns a matrix-like object as output.

normByType <- function(x, type) {
    ## Check argument
    stopifnot(length(type) == ncol(x))
    ## Normalize for each type separately
    for (i in unique(type)) {
        ## Get normalization factor
        nf <- rowMedians(x[, type == i], na.rm = TRUE)
        ## Perform normalization
        x[, type == i] <- x[, type == i] / nf
    }
    ## Return normalized data
    x
}

Suppose we want to apply the function to the proteins assay, we need to first extract that assay. We here need to transfer the sample annotations from the QFeatures object to the extracted SingleCellExperiment in order to get the sample types required by the normByType function. We therefore use getWithColData.

sce <- getWithColData(scp1, "proteins")
sce
#> class: SingleCellExperiment 
#> dim: 292 38 
#> metadata(0):
#> assays(2): assay aggcounts
#> rownames(292): A1A519 A5D8V6 ... REV__CON__Q32PI4 REV__CON__Q3MHN5
#> rowData names(9): protein Match.time.difference ...
#>   Potential.contaminant .n
#> colnames(38): 190321S_LCA10_X_FP97AG_RI1 190321S_LCA10_X_FP97AG_RI2 ...
#>   190914S_LCB3_X_16plex_Set_21_RI15 190914S_LCB3_X_16plex_Set_21_RI16
#> colData names(7): Set Channel ... sortday digest
#> reducedDimNames(0):
#> mainExpName: NULL
#> altExpNames(0):

Next, we can apply the data transformation to the quantitative data. As mentioned above, our function expects a matrix-like object as an input, so we use the assay function. We then update the SingleCellExperiment object.

mnorm <- normByType(assay(sce), type = sce$SampleType)
assay(sce) <- mnorm

We are now faced with 2 possibilities: either we want to create a new assay or we want to overwrite an existing assay. In both cases we need to make sure your data is still valid after data transformation.

Create a new assay

Creating a new assay has the advantage that you don’t modify an existing assay and hence limit the risk of introducing inconsistency in the data and avoid losing intermediate steps of the data processing.

We add the transformed assay using the addAssay function, then link the parent assay to the transformed assay using addAssayLinkOneToOne. Note that if each row name in the parent assay does not match exactly one row in the child assay, you can also use addAssayLink that will require a linking variable in the rowData.

scp1 <- addAssay(scp1, sce, name = "proteinsNorm")
scp1 <- addAssayLinkOneToOne(scp1, from = "proteins", to = "proteinsNorm")
scp1
#> An instance of class QFeatures containing 6 assays:
#>  [1] 190321S_LCA10_X_FP97AG: SingleCellExperiment with 166 rows and 11 columns 
#>  [2] 190222S_LCA9_X_FP94BM: SingleCellExperiment with 176 rows and 11 columns 
#>  [3] 190914S_LCB3_X_16plex_Set_21: SingleCellExperiment with 215 rows and 16 columns 
#>  [4] peptides: SingleCellExperiment with 539 rows and 38 columns 
#>  [5] proteins: SingleCellExperiment with 292 rows and 38 columns 
#>  [6] proteinsNorm: SingleCellExperiment with 292 rows and 38 columns

Overwrite an existing assay

Overwriting an existing assay has the advantage to limit the memory consumption as compared to adding a new assay. You can overwrite an assay simply by replacing the target assay in its corresponding slot.

scp1[["proteins"]] <- sce

Check for validity

Applying custom changes to the data increases the risk for data inconsistencies. You can however verify that a QFeatures object is still valid after some processing thanks to the validObject function.

validObject(scp1)
#> [1] TRUE

If the function detects no issues in the data, it will return TRUE. Otherwise the function will throw an informative error that should guide the user to identifying the issue.

Modify the sample annotations

To illustrate how to modify the sample annotations, we will compute the average expression in each sample and include to the colData of the QFeatures object. This is typically performed when computing QC metrics for sample filtering. So, we first extract the required data, in this case the quantitative values, and compute the sample-wise average protein expression.

m <- assay(scp1, "proteins")
meanExprs <- colMeans(m, na.rm = TRUE)
meanExprs
#>        190321S_LCA10_X_FP97AG_RI1        190321S_LCA10_X_FP97AG_RI2 
#>                        45368.3628                         7566.8782 
#>        190321S_LCA10_X_FP97AG_RI3        190321S_LCA10_X_FP97AG_RI4 
#>                         3271.0402                         2158.4230 
#>        190321S_LCA10_X_FP97AG_RI5        190321S_LCA10_X_FP97AG_RI6 
#>                         1162.3936                         1835.7939 
#>        190321S_LCA10_X_FP97AG_RI7        190321S_LCA10_X_FP97AG_RI8 
#>                         1293.4738                         1415.4205 
#>        190321S_LCA10_X_FP97AG_RI9       190321S_LCA10_X_FP97AG_RI10 
#>                         1965.6531                         1264.4064 
#>       190321S_LCA10_X_FP97AG_RI11         190222S_LCA9_X_FP94BM_RI1 
#>                         1238.5979                        48901.4654 
#>         190222S_LCA9_X_FP94BM_RI2         190222S_LCA9_X_FP94BM_RI3 
#>                         5189.5912                         2802.1850 
#>         190222S_LCA9_X_FP94BM_RI4         190222S_LCA9_X_FP94BM_RI5 
#>                         1204.2350                          715.7667 
#>         190222S_LCA9_X_FP94BM_RI6         190222S_LCA9_X_FP94BM_RI7 
#>                         1032.7122                          403.8267 
#>         190222S_LCA9_X_FP94BM_RI8         190222S_LCA9_X_FP94BM_RI9 
#>                         1089.0548                         1076.2613 
#>        190222S_LCA9_X_FP94BM_RI10        190222S_LCA9_X_FP94BM_RI11 
#>                         1572.3337                         1186.4431 
#>  190914S_LCB3_X_16plex_Set_21_RI1  190914S_LCB3_X_16plex_Set_21_RI2 
#>                        96881.8350                         2731.9430 
#>  190914S_LCB3_X_16plex_Set_21_RI3  190914S_LCB3_X_16plex_Set_21_RI4 
#>                         8208.1958                          208.2857 
#>  190914S_LCB3_X_16plex_Set_21_RI5  190914S_LCB3_X_16plex_Set_21_RI6 
#>                         3209.7730                         2014.4889 
#>  190914S_LCB3_X_16plex_Set_21_RI7  190914S_LCB3_X_16plex_Set_21_RI8 
#>                         2897.1103                         3006.6741 
#>  190914S_LCB3_X_16plex_Set_21_RI9 190914S_LCB3_X_16plex_Set_21_RI10 
#>                         2760.8564                         2549.9348 
#> 190914S_LCB3_X_16plex_Set_21_RI11 190914S_LCB3_X_16plex_Set_21_RI12 
#>                         2388.5289                         2533.8006 
#> 190914S_LCB3_X_16plex_Set_21_RI13 190914S_LCB3_X_16plex_Set_21_RI14 
#>                         2134.5615                         3125.1980 
#> 190914S_LCB3_X_16plex_Set_21_RI15 190914S_LCB3_X_16plex_Set_21_RI16 
#>                         3276.5618                         3195.8827

Next, we insert the computed averages into the colData. You need to make sure to match sample names because an extracted assay may not contain all samples and may be in a different order compared to the colData.

colData(scp1)[names(meanExprs), "meanProtExprs"] <- meanExprs

The new sample variable meanProtExprs is now accessible for filtering or plotting. The $ operator makes it straightforward to access the new data.

hist(log2(scp1$meanProtExprs))

To make sure that the process did not corrupt the colData, we advise to verify the data is still valid.

validObject(scp1)
#> [1] TRUE

Modify the feature annotations

We will again illustrate how to modify the feature annotations with an example. We here demonstrate how to add the number of samples in which each feature is detected for the three first assays (PSM assays). The challenge here is that the metric needs to be computed for each assay separately and inserted in the corresponding assay.

We will take advantage of the replacement function for rowData as implemented in QFeatures. It expects a list-like object where names indicate in which assays we want to modify the rowData and each element contains a table with the replacement values.

We therefore compute the metrics for each assay separately and store the results in a named List.

## Initialize the List object that will store the computed values
res <- List()
## We compute the metric for the first 3 assays
for (i in 1:3) {
    ## We get the quantitative values for the current assay
    m <- assay(scp1[[i]])
    ## We compute the number of samples in which each features is detected
    n <- rowSums(!is.na(m) & m != 0)
    ## We store the result as a DataFrame in the List
    res[[i]] <- DataFrame(nbSamples = n)
}
names(res) <- names(scp1)[1:3]
res
#> List of length 3
#> names(3): 190321S_LCA10_X_FP97AG 190222S_LCA9_X_FP94BM 190914S_LCB3_X_16plex_Set_21
res[[1]]
#> DataFrame with 166 rows and 1 column
#>           nbSamples
#>           <numeric>
#> PSM3773          11
#> PSM9078          11
#> PSM9858          11
#> PSM11744         11
#> PSM21752          0
#> ...             ...
#> PSM732069        11
#> PSM735396        11
#> PSM744756        10
#> PSM745037        11
#> PSM745130        11

Now that we have a List of DataFrames, we can update the object.

rowData(scp1) <- res

The new feature variable nbSamples is now accessible for filtering or plotting. The rbindRowData function facilitates the access the new data.

library("ggplot2")
rd <- rbindRowData(scp1, i = 1:3)
ggplot(data.frame(rd)) +
    aes(x = nbSamples) +
    geom_histogram(bins = 16) +
    facet_wrap(~ assay)

To make sure that the process did not corrupt the rowData in any assay, we advise to verify the data is still valid.

validObject(scp1)
#> [1] TRUE

Create a new function for scp

The modifying data in a QFeatures involves a multiple-step process. Creating a wrapper function that would take care of those different steps in a single line of code is a good habit to reduce the length of analysis scripts and hence making it easier to understand and less error-prone.

We will wrap the last example in a new function that we call computeNbDetectedSamples.

computeNbDetectedSamples <- function(object, i) {
    res <- List()
    for (ii in i) {
        m <- assay(object[[ii]])
        n <- rowSums(!is.na(m) & m != 0)
        res[[ii]] <- DataFrame(nbSamples = n)
    }
    names(res) <- names(object)[i]
    rowData(object) <- res
    stopifnot(validObject(object))
    object
}

Thanks to this new function, the previous section now simply boils down to running:

scp1 <- computeNbDetectedSamples(scp1, i = 1:3)

Keep in mind a few recommendations when creating a new function for scp:

  • The function should take a QFeatures object as input (and more arguments if needed) and return a QFeatures object as output. This will make workflows much easier to understand.
  • Allow user to select assays (if required) either as numeric, character, or logical.
  • Use conventional argument names: when naming an argument, try to match the names that already exist. For instance, selecting assays is passed through the i argument, selecting rowData variables is passed through rowvars and selecting colData variables is passed through colvars.
  • Follow the rformassspectrometry coding style

What’s next?

So you developed a new metric or method and believe it might benefit the community? We would love to hear about your improvements and eventually include your new functionality into scp or associate your new package to our documentation. Please, raise an issue in our Github repository to suggest your improvements or, better, submit your code as a pull request.

Session information

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] ggplot2_3.5.1               scp_1.17.0                 
 [3] QFeatures_1.17.0            MultiAssayExperiment_1.33.1
 [5] SummarizedExperiment_1.37.0 Biobase_2.67.0             
 [7] GenomicRanges_1.59.1        GenomeInfoDb_1.43.2        
 [9] IRanges_2.41.1              S4Vectors_0.45.2           
[11] BiocGenerics_0.53.3         generics_0.1.3             
[13] MatrixGenerics_1.19.0       matrixStats_1.4.1          
[15] BiocStyle_2.35.0           

loaded via a namespace (and not attached):
 [1] tidyselect_1.2.1            farver_2.1.2               
 [3] dplyr_1.1.4                 fastmap_1.2.0              
 [5] SingleCellExperiment_1.29.1 lazyeval_0.2.2             
 [7] nipals_0.8                  digest_0.6.37              
 [9] lifecycle_1.0.4             cluster_2.1.6              
[11] ProtGenerics_1.39.0         magrittr_2.0.3             
[13] compiler_4.4.2              rlang_1.1.4                
[15] sass_0.4.9                  tools_4.4.2                
[17] igraph_2.1.1                utf8_1.2.4                 
[19] yaml_2.3.10                 knitr_1.49                 
[21] labeling_0.4.3              S4Arrays_1.7.1             
[23] DelayedArray_0.33.2         plyr_1.8.9                 
[25] RColorBrewer_1.1-3          abind_1.4-8                
[27] withr_3.0.2                 purrr_1.0.2                
[29] sys_3.4.3                   grid_4.4.2                 
[31] fansi_1.0.6                 colorspace_2.1-1           
[33] scales_1.3.0                MASS_7.3-61                
[35] cli_3.6.3                   rmarkdown_2.29             
[37] crayon_1.5.3                metapod_1.15.0             
[39] httr_1.4.7                  reshape2_1.4.4             
[41] BiocBaseUtils_1.9.0         cachem_1.1.0               
[43] stringr_1.5.1               zlibbioc_1.52.0            
[45] impute_1.81.0               AnnotationFilter_1.31.0    
[47] BiocManager_1.30.25         XVector_0.47.0             
[49] vctrs_0.6.5                 Matrix_1.7-1               
[51] jsonlite_1.8.9              slam_0.1-55                
[53] IHW_1.35.0                  ggrepel_0.9.6              
[55] clue_0.3-66                 maketools_1.3.1            
[57] tidyr_1.3.1                 jquerylib_0.1.4            
[59] glue_1.8.0                  stringi_1.8.4              
[61] gtable_0.3.6                UCSC.utils_1.3.0           
[63] munsell_0.5.1               lpsymphony_1.35.0          
[65] tibble_3.2.1                pillar_1.9.0               
[67] htmltools_0.5.8.1           GenomeInfoDbData_1.2.13    
[69] R6_2.5.1                    evaluate_1.0.1             
[71] lattice_0.22-6              bslib_0.8.0                
[73] Rcpp_1.0.13-1               fdrtool_1.2.18             
[75] SparseArray_1.7.2           xfun_0.49                  
[77] MsCoreUtils_1.19.0          buildtools_1.0.0           
[79] pkgconfig_2.0.3            

License

This vignette is distributed under a CC BY-SA license license.