#' Create the UI for the results page. #' #' @param id ID for namespacing. #' @param options Global options for the application. #' #' @return The UI elements. #' #' @noRd results_ui <- function(id, options) { ranking_choices <- purrr::lmap(options$methods, function(method) { l <- list() l[[method[[1]]$name]] <- method[[1]]$id l }) ranking_choices <- c(ranking_choices, "Combined" = "combined") sidebarLayout( sidebarPanel( width = 3, comparison_editor_ui(NS(id, "comparison_editor"), options), methods_ui(NS(id, "methods"), options) ), mainPanel( width = 9, tabsetPanel( type = "pills", tabPanel( title = "Overview", div( style = "margin-top: 16px", plotly::plotlyOutput( NS(id, "rank_plot"), width = "100%", height = "600px" ) ) ), tabPanel( title = "Gene sets", div( style = "margin-top: 16px", htmlOutput(NS(id, "comparison_text")), div( style = "margin-top: 16px;", plotly::plotlyOutput( NS(id, "boxplot"), width = "100%", height = "600px" ) ) ) ), tabPanel( title = "Methods", div( style = "margin-top: 16px", plotly::plotlyOutput( NS(id, "rankings_plot"), width = "100%", height = "600px" ) ) ), tabPanel( title = "Method correlation", div( class = "flow-layout", style = "margin-top: 16px", selectInput( NS(id, "ranking_y"), label = NULL, choices = ranking_choices ), span( style = paste0( "display: inline-block;", "margin-right: 12px;", "padding: 0.375rem 0.75rem;" ), "~" ), selectInput( NS(id, "ranking_x"), label = NULL, choices = ranking_choices, selected = "combined" ), div( style = paste0( "display: inline-block;", "padding: 0.375rem 0.75rem;" ), checkboxInput( NS(id, "use_ranks"), "Use ranks instead of scores", value = TRUE ) ), div( style = paste0( "display: inline-block;", "padding: 0.375rem 0.75rem;" ), checkboxInput( NS(id, "use_sample"), "Take random sample of genes", value = TRUE ) ) ), htmlOutput(NS(id, "method_correlation")), plotly::plotlyOutput( NS(id, "ranking_correlation_plot"), width = "100%", height = "600px" ) ), tabPanel( title = "Scores by position", div( class = "flow-layout", style = "margin-top: 16px", selectInput( NS(id, "positions_plot_chromosome_name"), label = NULL, choices = c( list( "Chromosome overview" = "overview", "All chromosomes" = "all" ), chromosome_choices() ) ), htmlOutput( NS(id, "positions_plot"), container = \(...) div(style = "width: 100%; height: 600px", ...) ) ) ), tabPanel( title = "Ortholog locations", div( style = "margin-top: 16px", plotly::plotlyOutput( NS(id, "gene_locations_plot"), width = "100%", height = "1200px" ) ) ), tabPanel( title = "Detailed results", details_ui(NS(id, "results")) ), tabPanel( title = "g:Profiler", gsea_ui(NS(id, "gsea")) ) ) ) ) } #' Application logic for the results page. #' #' @param id ID for namespacing. #' @param options Global application options. #' @param analysis A reactive containing the analysis that gets visualized. #' #' @noRd results_server <- function(id, options, analysis) { preset <- reactive(analysis()$preset) moduleServer(id, function(input, output, session) { comparison_gene_ids <- comparison_editor_server( "comparison_editor", preset, options ) # Rank the results. ranking <- methods_server("methods", options, analysis, comparison_gene_ids) genes_with_distances <- merge( geposan::genes, geposan::distances[species == 9606, .(gene, distance)], by.x = "id", by.y = "gene" ) # Add gene information to the results. results <- reactive({ merge( ranking(), genes_with_distances, by.x = "gene", by.y = "id", sort = FALSE ) }) # Server for the detailed results panel. details_server("results", options, results) output$rank_plot <- plotly::renderPlotly({ preset <- preset() gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } geposan::plot_scores(ranking(), gene_sets = gene_sets) }) output$rankings_plot <- plotly::renderPlotly({ preset <- preset() rankings <- list() methods <- preset$methods all <- ranking() for (method in methods) { weights <- list() weights[[method$id]] <- 1.0 rankings[[method$name]] <- geposan::ranking(all, weights) } rankings[["Combined"]] <- all gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } geposan::plot_rankings(rankings, gene_sets) }) ranking_x <- reactive({ if (input$ranking_x == "combined") { ranking() } else { weights <- list() weights[[input$ranking_x]] <- 1.0 geposan::ranking(ranking(), weights) } }) ranking_y <- reactive({ if (input$ranking_y == "combined") { ranking() } else { weights <- list() weights[[input$ranking_y]] <- 1.0 geposan::ranking(ranking(), weights) } }) output$method_correlation <- renderText({ data <- merge( ranking_x()[, c("gene", "score")], ranking_y()[, c("gene", "score")], by = "gene" ) c <- stats::cor( data$score.x, data$score.y, method = "spearman" ) |> round(digits = 4) |> format(nsmall = 4) p <- stats::cor.test( data$score.x, data$score.y, method = "spearman" )$p.value |> round(digits = 4) |> format(nsmall = 4) HTML(glue::glue( "Spearman's rank correlation coefficient: ", "{c}, p = {p}" )) }) output$ranking_correlation_plot <- plotly::renderPlotly({ preset <- preset() gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } method_names <- options$methods |> purrr::lmap(function(method) { l <- list() l[[method[[1]]$id]] <- method[[1]]$name l }) method_names[["combined"]] <- "Combined" geposan::plot_rankings_correlation( ranking_x(), ranking_y(), method_names[[input$ranking_x]], method_names[[input$ranking_y]], gene_sets = gene_sets, use_ranks = input$use_ranks, use_sample = input$use_sample ) }) output$comparison_text <- renderUI({ reference <- geposan::compare( ranking(), preset()$reference_gene_ids ) comparison <- if (!is.null(comparison_gene_ids())) { geposan::compare(ranking(), comparison_gene_ids()) } num <- function(x, digits) { format( round(x, digits = digits), nsmall = digits, scientific = FALSE ) } comparison_text <- function(name, comparison) { glue::glue( "The {name} have a mean score of ", "{num(comparison$mean_score, 4)} ", "resulting in a mean rank of ", "{num(comparison$mean_rank, 1)}. ", "This corresponds to a percent rank of ", "{num(100 * comparison$mean_percentile, 2)}%. ", "A Wilcoxon rank sum test gives an estimated score difference ", "between {num(comparison$test_result$conf.int[1], 3)} and ", "{num(comparison$test_result$conf.int[2], 3)} with a 95% ", "confidence. This corresponds to a p-value of ", "{num(comparison$test_result$p.value, 4)}." ) } reference_div <- div(HTML( comparison_text("reference genes", reference) )) if (!is.null(comparison)) { div( reference_div, div( style = "margin-top: 16px;", HTML(comparison_text("comparison genes", comparison)) ) ) } else { reference_div } }) output$boxplot <- plotly::renderPlotly({ preset <- preset() gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } geposan::plot_boxplot(ranking(), gene_sets) }) output$gene_locations_plot <- plotly::renderPlotly({ preset <- preset() gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } geposan::plot_positions( preset$species_ids, gene_sets, reference_gene_ids = preset$reference_gene_ids ) }) output$positions_plot <- renderUI({ preset <- preset() if (input$positions_plot_chromosome_name == "overview") { geposan::plot_chromosomes(ranking()) } else { gene_sets <- list("Reference genes" = preset$reference_gene_ids) comparison_gene_ids <- comparison_gene_ids() if (length(comparison_gene_ids) >= 1) { gene_sets <- c( gene_sets, list("Comparison genes" = comparison_gene_ids) ) } chromosome <- if (input$positions_plot_chromosome_name == "all") { NULL } else { input$positions_plot_chromosome_name } geposan::plot_scores_by_position( ranking(), chromosome_name = chromosome, gene_sets = gene_sets ) } }) gsea_server("gsea", results) }) } #' Generate a named list for choosing chromosomes. #' @noRd chromosome_choices <- function() { choices <- purrr::lmap( unique(geposan::genes$chromosome), function(name) { choice <- list(name) names(choice) <- paste0( "Chromosome ", name ) choice } ) choices[order(suppressWarnings(sapply(choices, as.integer)))] }