inst/Shiny/server.R

# Code for SemNeT----
server <- function(input, output, session)
{
  # Keep previous environment
  prev.env <<- ls(envir = globalenv())
  
  # Check if anything exists in previous environment
  if(length(prev.env) != 0)
  {
    # Initialize textcleaner objects variable
    tc.object <<- vector(length = length(prev.env))
    
    # Check for textcleaner objects
    for(i in 1:length(prev.env))
    {tc.object[i] <- class(get(prev.env[i], envir = globalenv())) == "textcleaner"}
    
    # Set up environment objects
    if(sum(tc.object) != 0)
    {
      output$clean_ui <- renderUI({
        
        tagList(
          
          tags$style(
            ".tooltip-inner {
                 width: 350px;
               }"
          ),
          
          shinyBS::tipify(
            selectInput("clean_envir", label = "textcleaner Objects",
                        choices = c("", prev.env[tc.object]), selected = 2),
            "<code>textcleaner</code> objects that were found in your R environment. SemNeTShiny will automatically load the proper data from the object for semantic network analysis",
            placement = "right"
          )
          
        )
        
      })
    }
    
    if(exists("group"))
    {
      
      output$group_ui <- renderUI({
        
        tagList(
          
          tags$style(
            ".tooltip-inner {
                 width: 350px;
               }"
          ),
          
          shinyBS::tipify(
            radioButtons("group_envir", label = "'group' Object. Use?",
                         choices = c("Yes", "No"), inline = TRUE, selected = "Yes"),
            "A <code>group</code> object that was found in your R environment",
            placement = "right"
          )
          
        )
        
      })
    
    }
  }
  
  # semna citation
  output$SEMNA_cite <- renderUI({
    
    HTML(
      
      paste('<b>Please cite:</b><br>
      Christensen, A. P., & Kenett, Y. N. (2019). Semantic network analysis (SemNA): A tutorial on preprocessing, estimating, and analyzing semantic networks. <em>PsyArXiv</em>. <a href="https://doi.org/10.31234/osf.io/eht87">https://doi.org/10.31234/osf.io/eht87</a>
      ')
    )
    
  })
  
  #%%%%%%%%%%%%%%%%%#
  #### HIDE TABS ####
  #%%%%%%%%%%%%%%%%%#
  
  hideTab(inputId = "tabs", target = "Network Estimation")
  hideTab(inputId = "tabs", target = "Random Network Analyses")
  hideTab(inputId = "tabs", target = "Bootstrap Analyses")
  #hideTab(inputId = "tabs", target = "Permutation Analyses")
  hideTab(inputId = "tabs", target = "Random Walk Analyses")
  hideTab(inputId = "tabs", target = "Spreading Activation Analyses")
  hideTab(inputId = "tabs", target = "Save and Reset All Results")
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### HIDE SAVE BUTTONS ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  shinyjs::hide("save_data")
  shinyjs::hide("save_nets")
  shinyjs::hide("save_rand")
  shinyjs::hide("save_boot")
  shinyjs::hide("save_walk")
  shinyjs::hide("save_act")
  
  #%%%%%%%%%%%%%%%%%%%%%#
  #### DATA EXAMPLES ####
  #%%%%%%%%%%%%%%%%%%%%%#
  
  # Data
  observeEvent(input$data_example,
               {
                 output$example_data_response <- renderTable({head(SemNeT::open.clean)},
                                                             rownames = TRUE,
                                                             caption = "Response Matrix (participants by typed responses)",
                                                             caption.placement = getOption("xtable.caption.placement", "top")
                 )
                 
                 output$example_data_binary <- renderTable({head(SemNeT::open.binary)},
                                                           digits = 0,
                                                           rownames = TRUE,
                                                           caption = "Binary Matrix (participants by alphabetical responses)",
                                                           caption.placement = getOption("xtable.caption.placement", "top")
                 )
               }
  )
  
  observeEvent(input$group_example,
               {
                 output$example_group <- renderTable({
                   group_example <- as.matrix(head(SemNeT::open.group))
                   row.names(group_example) <- row.names(head(SemNeT::open.binary))
                   colnames(group_example) <- "Group"
                   return(group_example)
                 },
                 rownames = TRUE,
                 caption = "Group Vector (ordered by participant)",
                 caption.placement = getOption("xtable.caption.placement", "top"),
                 width = 250
                 )
               }
  )
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### NETWORK ESTIMATION ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  # Load Data panel
  observeEvent(input$load_data,
               {
                 # Let user know
                 showNotification("Loading data...")
                 
                 if(!is.null(input$clean_envir) || !is.null(input$group_envir))
                 {
                   
                   # Load data from R environment
                   if(!is.null(input$clean_envir)){
                     
                     if(input$clean_envir != ""){
                       dat <<- get(input$clean_envir, envir = globalenv())$responses$clean
                     }
      
                   }
                   
                   # Load group from R environment
                   if(!is.null(input$group_envir)){
                     
                     if(input$group_envir == "Yes"){
                       group <<- group
                     }
                     
                   }
                   
                 }
                 
                 # Load preprocessed data
                 if(!is.null(input$data))
                 {dat <<- suppressWarnings(SemNeT:::read.data(input$data$datapath))}
                 
                 # Load group data
                 if(!is.null(input$group))
                 {group <<- suppressWarnings(SemNeT:::read.data(input$group$datapath))}
                 
                 # Load data from SemNeT package
                 if(!exists("dat"))
                 {
                   dat <<- SemNeT::open.clean
                   group <<- SemNeT::open.group
                 }
                 
                 # Load group data from SemNeT package
                 if(!exists("group"))
                 {group <<- rep(1, nrow(dat))}
                 
                 # Organize group data
                 group <<- unlist(group)
                 
                 # Show network estimation tab
                 showTab(inputId = "tabs", target = "Network Estimation")
                 
                 # Show save data button
                 shinyjs::show("save_data")
                 
                 # Print waiting message
                 # FOR R PACKAGE AND WEB
                 shinyalert::shinyalert(title = "Data Loaded Successfully",
                                        type = "info",
                                        showConfirmButton = TRUE)
                 
                 # Print waiting message
                 # FOR R PACKAGE AND WEB
                 if(nrow(dat) > 1000){
                   shinyalert::shinyalert(title = "There are over 1000 cases in the data. Analyses may take considerable time.",
                                          type = "warning",
                                          showConfirmButton = TRUE)
                 }
                 
                 # Move on to network estimation tab
                 updateTabsetPanel(session, "tabs",
                                   selected = "Network Estimation")
                 
               }
  )
  
  # Set up UI based on selection of network estimaton
  ## Option 1
  output$network_options_1 <- renderUI({
    
    network <<- input$estimation
    
    if(network == "Community Network (CN)"){
    
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          numericInput("window", label = "Window Size", value = 2, min = 1, max = Inf),
          "Sets the distance for co-occurrence from a given response (e.g., the default is 2 responses before and after a given response)",
          placement = "right"
        )
        
      )
      
    }else if(network == "Naive Random Walk (NRW)"){
      
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          selectInput("NRW_type", label = paste("Number or Proportion of Minimum Co-Occurrences"),
                      choices = c("Number", "Proportion")),
          "Whehter the minimum number of co-occurrences should be a specific number (e.g., 3) or a proportion of co-occurrences given the sample size (e.g., .05)",
          placement = "right"
        )
        
      )
      
    }else if(network == "Triangulated Maximally Filtered Graph (TMFG)"){
      
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          selectInput("assoc", label = "Association Measure", choices = c("Angular", "Cosine",
                                                                          "Euclidean Distance",
                                                                          "Faith", "Jaccard Index",
                                                                          "Pearson's Correlation",
                                                                          "RR"), selected = "Pearson's Correlation"
          ),
          "Association measure that is used to compute an association matrix (e.g., correlation matrix). There are many options but cosine similarity and Pearson's correlation are the most commonly used. Cosine produces values between 0 and 1; Pearson's correlation produces values between -1 and 1",
          placement = "right"
        )
        
      )
      
    }
    
  })
  
  ## Option 2
  output$network_options_2 <- renderUI({
    
    network <<- input$estimation
    
    if(network == "Community Network (CN)"){
      
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          selectInput("alpha", label = paste("Significance Level"), choices = c(.05, .01, .001)),
          "Sets &alpha; to infer whether responses co-occurred by random chance using a binomial distribution",
          placement = "right"
        )
        
      )
      
    }else if(network == "Naive Random Walk (NRW)"){
      
      # Handles warning for initialization
      if(!is.null(input$NRW_type)){
        
        if(input$NRW_type == "Number"){
          
          tagList(
            
            tags$style(
              ".tooltip-inner {
                 width: 350px;
               }"
            ),
            
            shinyBS::tipify(
              
              numericInput("threshold", label = "Threshold", value = 0, min = 0, max = ncol(dat), step = 1),
              "Sets the threshold for the minimum number of co-occurrences between two responses in the random walks for there to be an edge between them in the network",
              placement = "right"
              
            )
            
          )
          
        }else if(input$NRW_type == "Proportion"){
          
          tagList(
            
            tags$style(
              ".tooltip-inner {
                 width: 350px;
               }"
            ),
            
            shinyBS::tipify(
              
              numericInput("threshold", label = "Threshold", value = 0, min = 0, max = 1, step = .01),
              "Sets the threshold for the minimum number of co-occurrences between two responses in the random walks for there to be an edge between them in the network",
              placement = "right"
              
            )
            
          )
          
        }
        
      }
      
    }else if(network == "Triangulated Maximally Filtered Graph (TMFG)"){
      
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          numericInput("minCase", label = "Minimum Number of Responses", value = 2, min = 1, max = Inf),
          "The minimum number of responses provided by at least <em>n</em> participants across the groups. Must be equal to 2 for Pearson's correlation",
          placement = "right"
        )
        
      )
      
    }
    
  })
  
  ## Option 3
  output$network_options_3 <- renderUI({
    
    network <<- input$estimation
    
    if(network == "Community Network (CN)"){
      
      tagList(
        
        tags$style(
          ".tooltip-inner {
                 width: 350px;
               }"
        ),
        
        shinyBS::tipify(
          selectInput("enrich", label = paste("Enrich Network"), choices = c(FALSE, TRUE)),
          "Whether to use a clustering approach to connect all nodes that are in the same cluster. Most commonly <em>not</em> used in the literature (i.e., FALSE), however, Go&ntilde;i et al. (2011) demonstrated greater agreement for clustering and switching with human raters when the networks were enriched (i.e., TRUE)",
          placement = "right"
        )
        
      )
    
    }
    
  })
  
  # Network Estimation panel
  observeEvent(input$run_est,
               {
                 # Let user know
                 showNotification("Estimating networks...")
                 
                 ## Identify unique groups
                 uniq <<- unique(group)
                 
                 ## Grab network
                 network <<- switch(input$estimation,
                                    "Triangulated Maximally Filtered Graph (TMFG)" = "TMFG",
                                    "Community Network (CN)" = "CN",
                                    "Naive Random Walk (NRW)" = "NRW",
                                    "Pathfinder Network (PF)" = "PF"
                 )
                 
                 ## Change responses to binary matrix
                 if(network == "TMFG")
                 {
                   if(is.character(unlist(dat)))
                   {bin_dat <<- SemNeT:::resp2bin(dat)$binary}
                   
                   ## Create new data
                   for(i in 1:length(uniq))
                   {
                     assign(paste(uniq[i]),
                            bin_dat[which(group == uniq[i]),],
                            envir = globalenv())
                   }
                   
                 }else{
                   ## Create new data
                   for(i in 1:length(uniq))
                   {
                     assign(paste(uniq[i]),
                            dat[which(group == uniq[i]),],
                            envir = globalenv())
                   }
                 }
                 
                 ## Estimate networks
                 if(network == "CN")
                 {
                   window_size <<- input$window
                   sig_alpha <<- as.numeric(input$alpha)
                   enrichment <<- as.logical(input$enrich)
                   
                   ## Estimate networks
                   nets <<- lapply(mget(paste(uniq), envir = globalenv()),
                                   function(x){CN(x, window = window_size, alpha = sig_alpha, enrich = enrichment)})
                   
                   # community network citation
                   output$net_cite <- renderUI({
                     
                     HTML(
                       
                       paste('<b>Please cite:</b><br>
                       Goni, J., Arrondo, G., Sepulcre, J., Martincorena, I., de Mendizabel, N. V., Corominas-Murtra, B., ... & Villoslada, P. (2011). The semantic organization of the animal category: Evidence from semantic verbal fluency and network theory. <em>Cognitive Processing</em>, <em>12</em>, 183&ndash;196. <a href="https://doi.org/10.1007/s10339-010-0372-x">https://doi.org/10.1007/s10339-010-0372-x</a>
                             ')
                     )
                     
                   })
                   
                 }else if(network == "NRW")
                 {
                   
                   nrw_type <<- switch(input$NRW_type,
                                       "Number" = "num",
                                       "Proportion" = "prop"
                   )
                   
                   thresh <<- input$threshold
                   
                   ## Estimate networks
                   nets <<- lapply(mget(paste(uniq), envir = globalenv()),
                                   function(x){NRW(x, type = nrw_type, threshold = thresh)})
                   
                   # naive random walk citation
                   output$net_cite <- renderUI({
                     
                     HTML(
                       
                       paste('<b>Please cite:</b><br>
                       Lerner, A. J., Ogrocki, P. K., & Thomas, P. J. (2009). Network graph analysis of category fluency testing. <em>Cognitive and Behavioral Neurology</em>, <em>22</em>, 45&ndash;52. <a href="https://doi.org/10.1097/WNN.0b013e318192ccaf">https://doi.org/10.1097/WNN.0b013e318192ccaf</a>
                             ')
                     )
                     
                   })
                   
                 }else if(network == "PF")
                 {
                   ## Estimate networks
                   nets <<- lapply(mget(paste(uniq), envir = globalenv()),
                                   function(x){PF(x)})
                   
                   # pathfinder citation
                   output$net_cite <- renderUI({
                     
                     HTML(
                       
                       paste('<b>Please cite:</b><br>
                       Paulsen, J. S., Romero, R., Chan, A., Davis, A. V., Heaton, R. K., & Jeste, D. V. (1996). Impairment of the semantic network in schizophrenia. <em>Psychiatry Research</em>, <em>63</em>, 109&ndash;121. <a href="https://doi.org/10.1016/0165-1781(96)02901-0">https://doi.org/10.1016/0165-1781(96)02901-0</a>
                       <br><br>
                       Quirin, A., Cordon, O., Guerrero-Bote, V. P., Vargas-Quesada, B., & Moya-Aneon, F. (2008). A quick MST-based algorithm to obtain Pathfinder networks (&infin;, <em>n</em> - 1). <em>Journal of the American Society for Information Science and Technology</em>, <em>59</em>, 1912&ndash;1924. <a href="https://doi.org/10.1002/asi.20904">https://doi.org/10.1002/asi.20904</a>
                       <br><br>
                       Schvaneveldt, R. W. (1990). <em>Pathfinder associative networks: Studies in knowledge organization</em>. Norwood, NJ: Ablex Publishing.
                             ')
                     )
                     
                   })
                   
                 }else if(network == "TMFG")
                 {
                   ## Store binary groups
                   minCase <<- as.numeric(input$minCase)
                   
                   for(i in 1:length(uniq))
                   {assign(paste(uniq[i]),
                           SemNeT::finalize(get(paste(uniq[i]), envir = globalenv()),
                                            minCase = minCase),
                           envir = globalenv())}
                   
                   ## Equate groups
                   if(length(uniq) > 1){
                     eq <<- SemNeT:::equateShiny(mget(paste(uniq), envir = globalenv()))
                   }else{
                     eq <<- mget(paste(uniq), envir = globalenv())
                   }
                   
                   ## Grab proper association label
                   sim <<- switch(input$assoc,
                                  "Angular" = "angular",
                                  "Cosine" = "cosine",
                                  "Euclidean Distance" = "euclid",
                                  "Faith" = "faith",
                                  "Jaccard Index" = "jaccard",
                                  "Pearson's Correlation" = "cor",
                                  "RR" = "rr",
                   )
                   
                   ## Compute associations
                   assoc <<- lapply(eq, SemNeT::similarity, method = sim)
                   
                   ## Estimate networks
                   nets <<- lapply(assoc, function(x){SemNeT::TMFG(x)})
                   
                   # triangulated maximally filtered graph citation
                   output$net_cite <- renderUI({
                     
                     HTML(
                       
                       paste('<b>Please cite:</b><br>
                       Massara, G. P., Di Matteo, T., & Aste, T. (2016). Network filtering for big data: Triangulated maximally filtered graph. <em>Journal of Complex Networks</em>, <em>5</em>, 161&ndash;178. <a href="https://doi.org/10.1093/comnet/cnw015">https://doi.org/10.1093/comnet/cnw015</a>
                             ')
                     )
                     
                   })
                 }
                 
                 ## Compute network measures
                 meas <<- lapply(nets, SemNeT::semnetmeas)
                 
                 ## Organized output
                 meas.mat <<- sapply(meas, c)
                 
                 ## Generate plot
                 plots <<- SemNeT:::compare_netShiny(nets, config = "spring", weighted = FALSE)
                 
                 ## Render semantic networks plot
                 output$viz <- renderPlot({
                   
                   ### Manipulate Shiny plot window
                   if(length(plots$datalist) == 2)
                   {layout(t(1:2))
                   }else if(length(plots$datalist) > 2)
                   {
                     #Find square root
                     len <<- floor(sqrt(length(plots$datalist)))
                     
                     #Remainder
                     remain <<- length(plots$datalist)%%len
                     
                     #Change layout accordingly
                     layout(t(matrix(1:(length(plots$datalist)+remain),ncol=len)))
                   }
                   
                   ### Generate plot
                   SemNeT:::plot.compareShiny(plots)
                 })
                 
                 ## Render network measures table
                 output$measures <- renderTable(meas.mat,
                                                rownames = TRUE,
                                                caption = "Network Measures",
                                                caption.placement = getOption("xtable.caption.placement", "top"))
                 
                 ## Change later input for bootstrap networks
                 if(network != "TMFG")
                 {shinyjs::hide("type_select"); shinyjs::hide("type")
                 }else{shinyjs::show("type_select"); shinyjs::show("type")}
                 
                 # Show analysis tabs
                 showTab(inputId = "tabs", target = "Random Network Analyses")
                 showTab(inputId = "tabs", target = "Bootstrap Analyses")
                 #showTab(inputId = "tabs", target = "Permutation Analyses")
                 showTab(inputId = "tabs", target = "Random Walk Analyses")
                 showTab(inputId = "tabs", target = "Spreading Activation Analyses")
                 # Show save and reset tab
                 showTab(inputId = "tabs", target = "Save and Reset All Results")
                 
                 # Update permutation tab
                 ## Group selection
                 #if(length(uniq) > 2)
                 #{
                #   output$group1 <- renderUI({
                #     selectInput("group1", label = "Select Group 1",
                #                 choices = uniq)
                #   })
                #   
                #   output$group2 <- renderUI({
                #     selectInput("group2", label = "Select Group 2",
                #                 choices = uniq[-which(uniq == input$group1)])
                #   })
                #   
                #   output$alter <- renderUI({
                #     
                #     group1 <<- input$group1
                #     group2 <<- input$group2
                #     
                #     perm_choices <<- c(paste(group1, "<", group2, "(one-tailed)", sep = " "),
                #                        paste(group1, ">", group2, "(one-tailed)", sep = " "),
                #                        paste(group1, "!=", group2, "(two-tailed)", sep = " "))
                #     
                #     selectInput("alter", label = "Alternative Hypothesis",
                #                 choices = perm_choices,
                #                 selected = perm_choices[3]
                #     )
                #     
                #   })
                #   
                # }else{
                #   
                #   output$alter <- renderUI({
                #     
                #     group1 <<- uniq[1]
                #     group2 <<- uniq[2]
                #     
                #     perm_choices <<- c(paste(group1, "<", group2, "(one-tailed)", sep = " "),
                #                        paste(group1, ">", group2, "(one-tailed)", sep = " "),
                #                        paste(group1, "!=", group2, "(two-tailed)", sep = " "))
                #     
                #     selectInput("alter", label = "Alternative Hypothesis",
                #                 choices = perm_choices,
                #                 selected = perm_choices[3]
                #     )
                #     
                #   })
                   
                # }
                 
                 # Show save networks button
                 shinyjs::show("save_nets")
                 
                 ## Hide clear results button
                 shinyjs::show("reset")
                 
                 #%%%%%%%%%%%%%%%%%#
                 #### CITATIONS ####
                 #%%%%%%%%%%%%%%%%%#
                 
                 
                 # random network citation
                 output$randnet_cite <- renderUI({
                   
                   HTML(
                     
                     paste('<b>Please cite:</b><br>
                       Kenett, Y. N., Wechsler-Kashi, D., Kenett, D. Y., Schwartz, R. G., Ben Jacob, E., & Faust, M. (2013). Semantic organization in children with cochlear implants: Computational analysis of verbal fluency. <em>Frontiers in Psychology</em>, <em>4</em>, 543. <a href="https://doi.org/10.3389/fpsyg.2013.00543">https://doi.org/10.3389/fpsyg.2013.00543</a>
                       <br><br>
                       Viger, F., & Latapy, M. (2016). Efficient and simple generation of random simple connected graphs with prescribed degree sequence. <em>Journal of Complex Networks</em>, <em>4</em>, 15&ndash;37. <a href="https://doi.org/10.1093/comnet/cnv013">https://doi.org/10.1093/comnet/cnv013</a>
                             ')
                   )
                   
                 })
                 
                 
                 # partial bootstrap citation
                 if(network == "TMFG"){
                   output$partial_cite <- renderUI({
                     
                     if(!is.null(input$test)){
                       
                       if(input$test == "t-test"){
                         
                         HTML(
                           
                           paste('<b>Effect sizes (<em>d</em>): small (.20), medium (.50), and large (.80)</b>
                       <br>
                       Cohen, J. (1988). <em>Statistical power analysis for the behavioural sciences</em> (2nd ed.). New York, NY: Routledge. <a href="https://doi.org/10.4324/9780203771587">https://doi.org/10.4324/9780203771587</a> 
                       <br><br>
                       <b>Please cite:</b><br>
                       Christensen, A. P., Kenett, Y. N., Cotter, K. N., Beaty, R. E., & Silvia, P. J. (2018). Remotely close associations: Openness to experience and semantic memory structure. <em>European Journal of Personality</em>, <em>32</em>, 480&ndash;492. <a href="https://doi.org/10.1002/per.2157">https://doi.org/10.1002/per.2157</a>
                       <br><br>
                       Kenett, Y. N., Wechsler-Kashi, D., Kenett, D. Y., Schwartz, R. G., Ben Jacob, E., & Faust, M. (2013). Semantic organization in children with cochlear implants: Computational analysis of verbal fluency. <em>Frontiers in Psychology</em>, <em>4</em>, 543. <a href="https://doi.org/10.3389/fpsyg.2013.00543">https://doi.org/10.3389/fpsyg.2013.00543</a>
                             ')
                         )
                         
                       }else if(input$test == "ANCOVA"){
                         
                         HTML(
                           
                           paste('<b>Effect sizes (<em>&eta;<sub>p</sub><sup>2</sup></em>): small (.01), medium (.06), and large (.14)</b>
                       <br>
                       Cohen, J. (1988). <em>Statistical power analysis for the behavioural sciences</em> (2nd ed.). New York, NY: Routledge. <a href="https://doi.org/10.4324/9780203771587">https://doi.org/10.4324/9780203771587</a> 
                       <br><br>
                       <b>Please cite:</b><br>
                       Christensen, A. P., Kenett, Y. N., Cotter, K. N., Beaty, R. E., & Silvia, P. J. (2018). Remotely close associations: Openness to experience and semantic memory structure. <em>European Journal of Personality</em>, <em>32</em>, 480&ndash;492. <a href="https://doi.org/10.1002/per.2157">https://doi.org/10.1002/per.2157</a>
                       <br><br>
                       Kenett, Y. N., Wechsler-Kashi, D., Kenett, D. Y., Schwartz, R. G., Ben Jacob, E., & Faust, M. (2013). Semantic organization in children with cochlear implants: Computational analysis of verbal fluency. <em>Frontiers in Psychology</em>, <em>4</em>, 543. <a href="https://doi.org/10.3389/fpsyg.2013.00543">https://doi.org/10.3389/fpsyg.2013.00543</a>
                             ')
                         )
                         
                       }
                       
                     }
                     
                   })
                 }else{
                   output$partial_cite <- renderUI({
                     
                     if(!is.null(input$test)){
                       
                       if(input$test == "t-test"){
                         
                         HTML(
                           
                           paste('<b>Effect sizes (<em>d</em>): small (.20), medium (.50), and large (.80)</b>
                       <br>
                       Cohen, J. (1988). <em>Statistical power analysis for the behavioural sciences</em> (2nd ed.). New York, NY: Routledge. <a href="https://doi.org/10.4324/9780203771587">https://doi.org/10.4324/9780203771587</a> 
                       ')
                         )
                         
                       }else if(input$test == "ANOVA" | input$test == "ANCOVA"){
                         
                         HTML(
                           
                           paste('<b>Effect sizes (<em>&eta;<sub>p</sub><sup>2</sup></em>): small (.01), medium (.06), and large (.14)</b>
                       <br>
                       Cohen, J. (1988). <em>Statistical power analysis for the behavioural sciences</em> (2nd ed.). New York, NY: Routledge. <a href="https://doi.org/10.4324/9780203771587">https://doi.org/10.4324/9780203771587</a> 
                       ')
                         )
                         
                       }
                       
                     }
                     
                   })
                 }
                 
               }
  )
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### RANDOM NETWORK TEST ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  # Determine the number of cores
  ## Random Networks
  output$cores_rand <- renderUI({
    
    core_rand <<- seq(1,parallel::detectCores()-1,1)
    names(core_rand) <- paste(core_rand)
    
    selectInput("cores_rand", label = "Number of Processing Cores",
                choices = core_rand,
                selected = ceiling(length(core_rand) / 2)
    )
  })
  
  # Random Networks panel
  observeEvent(input$run_rand,
               {
                 # Let user know
                 showNotification("Computing statistics...")
                 
                 # Print waiting message
                 # FOR R PACKAGE
                 shinyalert::shinyalert(title = "Running...",
                                         text = "Check R Console for the Random Network Analyses Progress",
                                         type = "info")
                
                 # Run random networks
                 rand_res <- reactive({
                   
                   randres <<- SemNeT:::randnet.testShiny(nets, iter = as.numeric(input$iters_rand), cores = as.numeric(input$cores_rand))
                   
                   # Convert into matrix
                   for(i in 1:length(randres))
                   {
                     if(i == 1)
                     {randresmat <- randres[[i]]
                     }else{randresmat <- cbind(randresmat, randres[[i]])}
                   }
                   
                   return(randresmat)
                   
                 })
                 
                 # Render random networks table
                 output$randnet <- renderTable({rand.res <<- rand_res(); rand.res}, rownames = TRUE,
                                               caption = "Random Network Results",
                                               caption.placement = getOption("xtable.caption.placement", "top")
                 )
                 
                 # Show save random network analyses button
                 shinyjs::show("save_rand")
                 
               }
  )
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### BOOTSTRAP NETWORK TEST ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  ## Bootstrap Networks
  output$cores_boot <- renderUI({
    
    core_boot <<- seq(1,parallel::detectCores()-1,1)
    names(core_boot) <- paste(core_boot)
    
    selectInput("cores_boot", label = "Number of Processing Cores",
                choices = core_boot,
                selected = ceiling(length(core_boot) / 2)
    )
  })
  
  output$type <- renderUI({
    
    if(input$type_select == "Node")
    {
      checkboxGroupInput("percent", label = "Proportion of Nodes Remaining",
                         choiceNames = sprintf("%1.2f",seq(.50,.90,.10)),
                         choiceValues = seq(.50,.90,.10), inline = TRUE,
                         selected = seq(.50,.90,.10)
      )
    }
    
  })
  
  
  output$test <- renderUI({
    
    if(length(uniq) == 2){
      
      selectInput("test", label = "Statistical Test",
                  choices = c("t-test", "ANCOVA"),
                  selected = "t-test"
      )
      
    }else{
      
      selectInput("test", label = "Statistical Test",
                  choices = c("t-test", "ANOVA", "ANCOVA"),
                  selected = "ANOVA"
      )
      
    }
    
  })
  
  ## Hide plot button
  shinyjs::hide("run_plot")
  
  # Bootstrap Networks panel
  observeEvent(input$run_boot,
               {
                 # Let user know
                 showNotification("Computing statistics...")
                 
                 # Run bootstrap networks
                 boot <- reactive({
                   
                   if(network == "TMFG")
                   {
                     if(input$type_select == "Node")
                     {
                       ## Obtain percentages
                       percents <<- as.numeric(input$percent)
                       
                       # Show progress
                       withProgress({
                         
                         ## Run partial bootstrap networks
                         for(i in 1:length(percents))
                         {
                           if(exists(paste(percents[i]), envir = globalenv()))
                           {next
                           }else{
                             
                             # Print waiting message
                             # FOR R PACKAGE
                             shinyalert::shinyalert(title = paste("Running...\n","(Proportion of nodes remaining: ",sprintf("%1.2f", percents[i]),")",sep=""),
                                                    text = "Check R Console for the Bootstrap Network Analyses Progress",
                                                    type = "info")
                             
                             # Increase progress
                             setProgress(value = i)
                             
                             assign(paste(percents[i]),
                                    SemNeT:::bootSemNeTShiny(eq,
                                                             prop = percents[i],
                                                             sim = sim,
                                                             weighted = FALSE,
                                                             iter = as.numeric(input$iters_boot),
                                                             cores = as.numeric(input$cores_boot),
                                                             type = "node",
                                                             method = network),
                                    envir = globalenv())
                             
                           }
                         }
                         
                       }, message = "Computing bootstraps...", value = 0, min = 1, max = length(percents))
                     }else{
                       # Print waiting message
                       # FOR R PACKAGE
                       shinyalert::shinyalert(title = "Running...",
                                              text = "Check R Console for the Bootstrap Network Analyses Progress",
                                              type = "info")
                       
                       ## Only one
                       percents <<- as.numeric(100)
                       
                       ## Organize method arguments
                       methodArgs <- list(minCase = minCase)
                       
                       assign(paste(percents),
                              SemNeT:::bootSemNeTShiny(mget(paste(uniq), envir = globalenv()),
                                                       weighted = FALSE,
                                                       iter = as.numeric(input$iters_boot),
                                                       cores = as.numeric(input$cores_boot),
                                                       type = "case",
                                                       method = network,
                                                       methodArgs = methodArgs),
                              envir = globalenv())
                     }
                     
                   }else{
                     
                     # Print waiting message
                     # FOR R PACKAGE
                     shinyalert::shinyalert(title = "Running...",
                                            text = "Check R Console for the Bootstrap Network Analyses Progress",
                                            type = "info")
                     
                     ## Only one
                     percents <<- as.numeric(100)
                     
                     ## Organize method arguments
                     if(network == "CN")
                     {methodArgs <- list(window = window_size, alpha = sig_alpha, enrich = enrichment)
                     }else if(network == "NRW")
                     {methodArgs <- list(type = nrw_type, threshold = thresh)
                     }else if(network == "PF")
                     {methodArgs <- list()}
                     
                     assign(paste(percents),
                            SemNeT:::bootSemNeTShiny(mget(paste(uniq), envir = globalenv()),
                                                     weighted = FALSE,
                                                     iter = as.numeric(input$iters_boot),
                                                     cores = as.numeric(input$cores_boot),
                                                     type = "case",
                                                     method = network,
                                                     methodArgs = methodArgs),
                            envir = globalenv())
                     
                   }
                   
                   return(list(mget(paste(percents), envir = globalenv())))
                 })
                 
                 # Render Tables
                 res_boot <<- boot()
                 
                 # Reset Table
                 output$tab <- renderTable({})
                 
                 observeEvent(input$test,
                 {
                   
                   if(input$test == "ANCOVA"){
                     
                     ## ANCOVA
                     if(length(percents) == 1)
                     {
                       
                       output$tab <- renderTable({
                         bootTest <<- list()
                         
                         bootTest <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                    test = input$test); bootTest$ANCOVA
                       }, rownames = TRUE,
                       caption = "Bootstrap Network Results",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }else{
                       
                       ## Reset original table
                       output$tab <- renderTable({})
                       
                       bootTest <<- list()
                       full_res <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                  test = input$test)
                       
                       ## Average Shortest Path Length
                       output$aspl <- renderTable({
                         bootTest$ASPL <<- full_res$ANCOVA$ASPL; bootTest$ASPL
                       }, rownames = TRUE,
                       caption = "Average Shortest Path Length (ASPL)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Clustering Coefficient
                       output$cc <- renderTable({
                         bootTest$CC <<- full_res$ANCOVA$CC; bootTest$CC
                       }, rownames = TRUE,
                       caption = "Clustering Coefficient (CC)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Modularity
                       output$q <- renderTable({
                         bootTest$Q <<- full_res$ANCOVA$Q; bootTest$Q
                       }, rownames = TRUE,
                       caption = "Modularity",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }
                     
                   }else if(input$test == "ANOVA"){
                     
                     ## ANCOVA
                     if(length(percents) == 1)
                     {
                       
                       output$tab <- renderTable({
                         bootTest <<- list()
                         
                         bootTest <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                    test = input$test); bootTest$ANOVA
                       }, rownames = TRUE,
                       caption = "Bootstrap Network Results",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }else{
                       
                       ## Reset original table
                       output$tab <- renderTable({})
                       
                       bootTest <<- list()
                       full_res <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                  test = input$test)
                       
                       ## Average Shortest Path Length
                       output$aspl <- renderTable({
                         bootTest$ASPL <<- full_res$ANOVA$ASPL; bootTest$ASPL
                       }, rownames = TRUE,
                       caption = "Average Shortest Path Length (ASPL)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Clustering Coefficient
                       output$cc <- renderTable({
                         bootTest$CC <<- full_res$ANOVA$CC; bootTest$CC
                       }, rownames = TRUE,
                       caption = "Clustering Coefficient (CC)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Modularity
                       output$q <- renderTable({
                         bootTest$Q <<- full_res$ANOVA$Q; bootTest$Q
                       }, rownames = TRUE,
                       caption = "Modularity",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }
                     
                   }else if(input$test == "t-test"){
                     
                     if(length(percents) == 1){
                       
                       output$tab <- renderTable({
                         bootTest <<- list()
                         
                         bootTest <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                    test = input$test); bootTest[[1]]
                       }, rownames = TRUE,
                       caption = "Bootstrap Network Results",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }else{
                       
                       ## Reset original table
                       output$tab <- renderTable({})
                       
                       bootTest <<- list()
                       full_res <<- SemNeT:::test.bootSemNeTShiny(unlist(res_boot, recursive = FALSE),
                                                                  test = input$test)
                       
                       ## Average Shortest Path Length
                       output$aspl <- renderTable({
                         bootTest$ASPL <<- full_res$ASPL; bootTest$ASPL
                       }, rownames = TRUE,
                       caption = "Average Shortest Path Length (ASPL)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Clustering Coefficient
                       output$cc <- renderTable({
                         bootTest$CC <<- full_res$CC; bootTest$CC
                       }, rownames = TRUE,
                       caption = "Clustering Coefficient (CC)",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                       ## Modularity
                       output$q <- renderTable({
                         bootTest$Q <<- full_res$Q; bootTest$Q
                       }, rownames = TRUE,
                       caption = "Modularity",
                       caption.placement = getOption("xtable.caption.placement", "top")
                       )
                       
                     }
                     
                   }
                   
                 })
                 
                 ## Show plot button
                 shinyjs::show("run_plot")
                 
                 # Show save bootstrap analyses button
                 shinyjs::show("save_boot")
                 
               }
  )
  
  # Plots panel
  observeEvent(input$run_plot,
               {
                 # Let user know
                 showNotification("Generating plots...")
                 
                 # Generate plots
                 pbplot <<- SemNeT:::plotbootSemNeTShiny(unlist(res_boot, recursive = FALSE))
                 
                 ## Average Shortest Path Length
                 output$asplPlot <- renderPlot({pbplot$aspl}, height = 400, width = 700)
                 
                 ## Clustering Coefficient
                 output$ccPlot <- renderPlot({pbplot$cc}, height = 400, width = 700)
                 
                 ## Modularity
                 output$qPlot <- renderPlot({pbplot$q}, height = 400, width = 700)
                 
               }
               
  )
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### PERMUTATION NETWORK TEST ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  # Permutation
  #output$cores_perm <- renderUI({
  #  
  #  core_perm <<- seq(1,parallel::detectCores()-1,1)
  #  names(core_perm) <- paste(core_perm)
  #  
  #  selectInput("cores_perm", label = "Number of Processing Cores",
  #              choices = core_perm,
  #              selected = ceiling(length(core_perm) / 2)
  #  )
  #})
  
  # Permutation panel
  #observeEvent(input$run_perm,
  #             {
  #               # Let user know
  #               showNotification("Computing statistics...")
  #               
  #               # Get alternative hypothesis
  #               if(input$alter == perm_choices[1])
  #               {alternative <<- "less"
  #               }else if(input$alter == perm_choices[2])
  #               {alternative <<- "greater"
  #               }else{alternative <<- "two.tailed"}
  #               
  #               measure <<- switch(input$meas_perm,
  #                                  "Average Shortest Path Length (ASPL)" = "ASPL",
  #                                  "Clustering Coefficient (CC)" = "CC",
  #                                  "Modularity (Q)" = "Q"
  #                                  
  #               )
  #               
  #               # Get samples
  #               sample1 <<- dat[which(group == group1),]
  #               sample2 <<- dat[which(group == group2),]
  #               
  #               perm <- reactive({
  #                 
  #                 # Print waiting message
  #                 # FOR R PACKAGE
  #                 shinyalert::shinyalert(title = "Running...",
  #                                       text = "Check R Console for the Permutation Analyses Progress",
  #                                       type = "info")
  #                 
  #                 # Compute permutated samples if 
  #                 if(!exists("perm.table", envir = globalenv()))
  #                 {
  #                   if(network == "TMFG")
  #                   {
  #                     
  #                     perm.res <<- permSemNeT(sample1 = sample1,
  #                                             sample2 = sample2,
  #                                             iter = as.numeric(input$iters_perm),
  #                                             method = network,
  #                                             sim = sim,
  #                                             minCase = minCase,
  #                                             weighted = FALSE,
  #                                             measure = measure,
  #                                             alternative = alternative,
  #                                             cores = as.numeric(input$cores_perm),
  #                                             prev.perm = NULL
  #                     )
  #                     
  #                   }else if(network == "CN")
  #                   {
  #                     perm.res <<- permSemNeT(sample1 = sample1,
  #                                             sample2 = sample2,
  #                                             iter = as.numeric(input$iters_perm),
  #                                             method = network,
  #                                             weighted = FALSE,
  #                                             measure = measure,
  #                                             alternative = alternative,
  #                                             cores = as.numeric(input$cores_perm),
  #                                             prev.perm = NULL,
  #                                             window = window_size,
  #                                             alpha = sig_alpha,
  #                                             enrich = enrichment,
  #                                             groups = c(group1, group2)
  #                     )
  #                     
  #                   }else if(network == "NRW")
  #                   {
  #                     perm.res <<- permSemNeT(sample1 = sample1,
  #                                             sample2 = sample2,
  #                                             iter = as.numeric(input$iters_perm),
  #                                             method = network,
  #                                             weighted = FALSE,
  #                                             measure = measure,
  #                                             alternative = alternative,
  #                                             cores = as.numeric(input$cores_perm),
  #                                             prev.perm = NULL,
  #                                             threshold = thresh,
  #                                             groups = c(group1, group2)
  #                     )
  #                     
  #                   }else if(network == "PF")
  #                   {
  #                     perm.res <<- permSemNeT(sample1 = sample1,
  #                                             sample2 = sample2,
  #                                             iter = as.numeric(input$iters_perm),
  #                                             method = network,
  #                                             weighted = FALSE,
  #                                             measure = measure,
  #                                             alternative = alternative,
  #                                             cores = as.numeric(input$cores_perm),
  #                                             prev.perm = NULL,
  #                                             groups = c(group1, group2)
  #                     )
  #                     
  #                   }
  #                   
  #                 }else{
  #                   
  #                   perm.res <<- permSemNeT(prev.perm = perm.res,
  #                                           measure = measure,
  #                                           alternative = alternative,
  #                                           cores = as.numeric(input$cores_perm),
  #                                           groups = c(group1, group2))
  #                   
  #                 }
  #                 
  #                 return(perm.res)
  #                 
  #               })
  #               
  #               # Render Tables
  #               perm_table <- perm()
  #               
  #               if(!exists("perm.table", envir = globalenv()))
  #               {perm.table <<- perm_table$result 
  #               }else if(!measure %in% row.names(perm.table))
  #               {perm.table <<- rbind(perm.table, perm_table$result)}
  #               
  #               output$perm_table <- renderTable({perm.table}, rownames = TRUE)
  #               
  #             }
  #)
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### RANDOM WALK ANALYSIS ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  # random walk citation
  output$randwalk_cite <- renderUI({
    
    HTML(
      
      paste('<b>Please cite:</b><br>
            Kenett, Y. N., & Austerweil, J. L. (2016). Examining search processes in low and high creative individiuals with random walks. In <em>Proceeding of the 38th annual meeting of the cognitive science society</em> (pp. 313&ndash;318). Austin, TX. Retrieved from <a href="https://cogsci.mindmodeling.org/2016/papers/0066/index.html">https://cogsci.mindmodeling.org/2016/papers/0066/index.html</a>')
      
    )
    
  })
  
  # Determine the number of cores
  ## Random Walk
  output$cores_walk <- renderUI({
    
    core_walk <<- seq(1,parallel::detectCores()-1,1)
    names(core_walk) <- paste(core_walk)
    
    selectInput("cores_walk", label = "Number of Processing Cores",
                choices = core_walk,
                selected = ceiling(length(core_walk) / 2)
    )
  })
  
  # Determine networks
  output$A <- renderUI({
    selectInput("A", label = "Select Network A",
                choices = names(nets))
  })
  
  output$B <- renderUI({
    selectInput("B", label = "Select Network B",
                choices = names(nets)[-which(names(nets) == input$A)])
  })
  
  # Random Walks panel
  observeEvent(input$run_walk,
               {
                 # Let user know
                 showNotification("Computing statistics...")
                 
                 # Print waiting message
                 # FOR R PACKAGE
                 shinyalert::shinyalert(title = "Running...",
                                         text = "Check R Console for the Random Walk Analyses Progress",
                                         type = "info")
                 
                 # Run random networks
                 rand_walk <- reactive({
                   
                   rw <<- SemNeT:::randwalkShiny(nets, input$A, input$B,
                                                 reps = as.numeric(input$reps),
                                                 steps = as.numeric(input$steps),
                                                 iter = as.numeric(input$iters_walk),
                                                 cores = as.numeric(input$cores_walk))
                   
                   return(rw)
                   
                 })
                 
                 # Render random networks table
                 output$walk_rand <- renderTable({rw <<- rand_walk(); rw$short}, rownames = FALSE,
                                                 caption = "Random Walk Results",
                                                 caption.placement = getOption("xtable.caption.placement", "top")
                 )
                 
                 # Show save random walk analyses button
                 shinyjs::show("save_walk")
                 
               }
  )
  
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  #### SPREADING ACTIVATION ANALYSIS ####
  #%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%#
  
  # spreadr citation
  output$spreadr_cite <- renderUI({
    
    HTML(
      
      paste('<b>Please cite:</b><br>
            Siew, C. S. Q. (2019). spreadr: An R package to simulate spreading activation in a network. <em>Behavior Research Methods</em>, <em>51</em>, 910&ndash;929. <a href="https://doi.org/10.3758/s13428-018-1186-5">https://doi.org/10.3758/s13428-018-1186-5</a>')
      
    )
    
  })
  
  ## Hide animate button
  shinyjs::hide("run_spr_act")
  
  # Determine networks
  output$network_select <- renderUI({
    selectInput("network_select", label = "Select a Network",
                choices = names(nets),
                selected = NULL)
  })
  
  # Initialize blank matrix
  blank_mat <<- TRUE
  
  observeEvent(input$set_act,
               {
                 ## Show node select
                 shinyjs::show("node_select")
                 
                 # Determine nodes
                 output$node_select <- renderUI({
                   
                   # Gets rid of NULL index bug
                   req(input$network_select)
                   
                   # Name of selected network
                   net_name <<- input$network_select
                   
                   # Nodes of the selected network
                   nodes <<- colnames(nets[[net_name]])
                   
                   # Create matrix of nodes with blank activations
                   if(blank_mat)
                   {mat <<- cbind(nodes, rep(0, length(nodes)))}
                   
                   # Create Shiny matrix of nodes and activations
                   shinyMatrix::matrixInput("node_activation",
                                            cols = list(
                                              names = TRUE,
                                              editableNames = FALSE
                                            ),
                                            value = matrix(mat, ncol = 2,
                                                           dimnames = list(NULL, c("Node", "Activation")))
                   )
                   
                 })
                 
                 # Check for node activation input
                 if(!is.null(input$node_activation))
                 {
                   mat <<- input$node_activation
                   blank_mat <<- FALSE
                 }
                 
                 ## Show set activation
                 shinyjs::show("run_spr_act")
               }
  )
  
  ## Hide animate button
  shinyjs::hide("animate")
  
  ## Hide plot size
  shinyjs::hide("animate_size")
  
  ## Hide animation slider
  shinyjs::hide("animate_slider")
  
  ## Hide reset activation
  shinyjs::hide("reset_act")
  
  # Random Walks panel
  observeEvent(input$run_spr_act,
               {
                 # Let user know
                 showNotification("Computing statistics...")
                 
                 # Set up start_run data frame
                 act_mat <<- input$node_activation
                 
                 # Convert nodes to numbers
                 act_mat[,1] <<- 1:nrow(act_mat)
                 
                 # Keep activation rows
                 keep.row <<- ifelse(is.na(act_mat[,2]) | act_mat[,2] == "", FALSE, TRUE)
                 
                 # Keep only those rows
                 act_mat <<- as.matrix(act_mat[keep.row,])
                 
                 # Make sure two columns
                 if(ncol(act_mat) != 2)
                 {act_mat <<- t(act_mat)}
                 
                 # Make sure values are numeric
                 act_mat <<- apply(act_mat, 2, as.numeric)
                 
                 # Keep rows
                 act_df <<- as.data.frame(act_mat)
                 
                 # Make sure two columns (and data frame)
                 if(ncol(act_df) != 2)
                 {act_df <<- as.data.frame(t(act_df))}
                 
                 # Make sure column names are correct
                 colnames(act_df) <<- c("node", "activation")
                 
                 # Run spreading activation
                 sa <<- spreadr::spreadr(network = nets[[net_name]],
                                         start_run = act_df,
                                         retention = input$retention,
                                         time = input$time,
                                         decay = input$decay,
                                         suppress = input$suppress)
                 
                 ## Show animate button
                 shinyjs::show("animate")
                 
                 ## Show plot size
                 shinyjs::show("animate_size")
                 
                 ## Hide matrix input
                 shinyjs::hide("node_select")
                 
                 ## Hide set activation
                 shinyjs::hide("set_act")
                 
                 ## Hide inputs
                 shinyjs::hide("network_select")
                 shinyjs::hide("run_spr_act")
                 shinyjs::hide("retention")
                 shinyjs::hide("time")
                 shinyjs::hide("decay")
                 shinyjs::hide("suppress")
                 
               }
  )
  
  observeEvent(input$animate,
               {
                 
                 # Initialize plot list
                 if(!exists("plot_list", envir = globalenv()))
                 {
                   plot_list <<- vector("list", length = length(nets))
                   names(plot_list) <<- names(nets)
                 }
                 
                 # Plot size
                 plot_size <<- switch(input$animate_size,
                                      "Small (500 x 500)" = 500,
                                      "Medium (900 x 900)" = 900,
                                      "Large (1400 x 1400)" = 1400
                 )
                 
                 # Initialize time list
                 time_list <<- vector("list", length = max(sa$time))
                 
                 # Progress through plots
                 withProgress({
                   
                   # Generate animation
                   for(i in 1:max(sa$time))
                   {
                     
                     # Increase progress
                     setProgress(value = i)
                     
                     # Plots
                     suppressWarnings(SemNeT:::spreadrShinyPlot(network = nets[[net_name]], spreadr.output = sa, time = i, size = plot_size))
                     time_list[[i]] <<- recordPlot()
                   }
                   
                 }, message = "Generating animation...", value = 0, min = 1, max = max(sa$time))
                 
                 # Set class of time list
                 class(time_list) <- "animateShiny"
                 
                 # Insert into plot list
                 plot_list[[net_name]] <<- time_list
                 
                 # Render plot
                 output$spreadr_animate <- renderPlot({
                   
                   # Gets rid of NULL index bug
                   req(input$animate_slider2)
                   
                   plot_list[[net_name]][[input$animate_slider2]]
                   
                 }, width = plot_size, height = plot_size)
                 
                 # Slider for animation
                 output$animate_slider <- renderUI({
                   sliderInput("animate_slider2", "Time Step",
                               min = 1, max = max(sa$time), value = 1, step = 1,
                               animate = TRUE)
                 })
                 
                 ## Show animation slider
                 shinyjs::show("animate_slider")
                 
                 ## Show spreadr animate
                 shinyjs::show("spreadr_animate")
                 
                 # Show save spreading activation analyses button
                 shinyjs::show("save_act")
                 
                 ## Show reset activation
                 shinyjs::show("reset_act")
                 
               }
  )
  
  observeEvent(input$reset_act,
               {
                 ## Reset blank matrix
                 blank_mat <<- TRUE
                 
                 ## Renew node selection
                 shinyMatrix::updateMatrixInput(session = session,
                                                "node_activation",
                                                value = matrix(cbind(nodes, rep(0, length(nodes))), ncol = 2,
                                                               dimnames = list(NULL, c("Node", "Activation")))
                                                
                                                
                 )
                 
                 ## Show inputs
                 shinyjs::show("network_select")
                 shinyjs::show("retention")
                 shinyjs::show("time")
                 shinyjs::show("decay")
                 shinyjs::show("suppress")
                 shinyjs::show("set_act")
                 
                 ## Hide animate button
                 shinyjs::hide("animate")
                 
                 ## Hide reset activation
                 shinyjs::hide("reset_act")
                 
                 ## Hide save spreading activation analyses button
                 shinyjs::hide("save_act")
                 
                 ## Hide plot size
                 shinyjs::hide("animate_size")
                 
                 ## Hide animate slider
                 shinyjs::hide("animate_slider")
                 
                 ## Hide animate slider
                 shinyjs::hide("spreadr_animate")
                 
               }
  )
  
  # Save events
  ## Data
  observeEvent(input$save_data,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Data",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("dat", envir = globalenv()))
                     {saveList$data <<- dat}
                     
                     if(exists("group", envir = globalenv()))
                     {saveList$group <<- group}
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Data was saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  ## Networks
  observeEvent(input$save_nets,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Networks",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Networks were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  ## Random Network Analyses
  observeEvent(input$save_rand,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Random Network Analyses",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     if(exists("rand.res", envir = globalenv()))
                     {saveList$randomTest <<- rand.res}
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Random Network Analyses were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  ## Bootstrap Analyses
  observeEvent(input$save_boot,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Bootstrap Analyses",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     if(exists("res_boot", envir = globalenv()))
                     {saveList$bootstrap <<- unlist(res_boot, recursive = FALSE)}
                     
                     if(exists("bootTest", envir = globalenv()))
                     {saveList$bootstrapTest <<- bootTest}
                     
                     if(exists("pbplot", envir = globalenv()))
                     {saveList$bootstrapPlot <<- pbplot}
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Bootstrap Analyses were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  ## Random Walk Analyses
  observeEvent(input$save_walk,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Random Walk Analyses",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     if(exists("rw", envir = globalenv()))
                     {saveList$randomWalk <<- rw}
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Random Walk Analyses were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  ## Spreading Activation Analyses
  observeEvent(input$save_act,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save Spreading Activation Analyses",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     if(exists("sa", envir = globalenv()))
                     {saveList$spreadingActivation <<- sa}
                     
                     if(exists("plot_list", envir = globalenv()))
                     {
                       if(any(!is.null(unlist(plot_list))))
                       {saveList$spreadingActivationPlot <<- plot_list}
                     }
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("Spreading Activation Analyses were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  #%%%%%%%%%%%%%%%%%%%#
  #### MASTER SAVE ####
  #%%%%%%%%%%%%%%%%%%%#

  observeEvent(input$save_master,
               {
                 
                 # Allow user to type name for object
                 shinyalert::shinyalert(
                   title = "Save All Results",
                   text = "Enter name for object:",
                   type = "input",
                   callbackR = function(value){
                     
                     # Get name for object
                     res.name <<- value
                     
                     # Add name to previous environment so it's not removed
                     prev.env <<- c(prev.env, res.name)
                     
                     # Create list
                     saveList <<- list()
                     
                     if(exists("dat", envir = globalenv()))
                     {saveList$data <<- dat}
                     
                     if(exists("group", envir = globalenv()))
                     {saveList$group <<- group}
                     
                     if(exists("eq", envir = globalenv())){
                       resultShiny$equated <<- eq
                     }
                     
                     if(exists("nets", envir = globalenv()))
                     {saveList$network <<- nets}
                     
                     if(exists("meas.mat", envir = globalenv()))
                     {saveList$measures <<- meas.mat}
                     
                     if(exists("plots", envir = globalenv()))
                     {saveList$comparePlot <<- plots}
                     
                     if(exists("rand.res", envir = globalenv()))
                     {saveList$randomTest <<- rand.res}
                     
                     if(exists("res_boot", envir = globalenv()))
                     {saveList$bootstrap <<- unlist(res_boot, recursive = FALSE)}
                     
                     if(exists("bootTest", envir = globalenv()))
                     {saveList$bootstrapTest <<- bootTest}
                     
                     if(exists("pbplot", envir = globalenv()))
                     {saveList$bootstrapPlot <<- pbplot}
                     
                     #if(exists("perm.table", envir = globalenv()))
                     #{saveList$permutationTest <<- perm.table}
                     
                     if(exists("rw", envir = globalenv()))
                     {saveList$randomWalk <<- rw}
                     
                     if(exists("sa", envir = globalenv()))
                     {saveList$spreadingActivation <<- sa}
                     
                     if(exists("plot_list", envir = globalenv()))
                     {
                       if(any(!is.null(unlist(plot_list))))
                       {saveList$spreadingActivationPlot <<- plot_list}
                     }
                     
                     # Assign save list to result name
                     assign(
                       x = res.name,
                       value = saveList,
                       envir = globalenv()
                     )
                     
                     # Let user know save was successful
                     shinyalert::shinyalert(
                       title = "Save Successful",
                       text = paste("All results were saved as '", res.name, "'", sep = ""),
                       type = "info"
                     )
                   }
                 )
                 
               }
  )
  
  #%%%%%%%%%%%%%#
  #### RESET ####
  #%%%%%%%%%%%%%#
  
  observeEvent(input$reset,
               {
                 
                 shinyalert::shinyalert(title = "Are you sure?",
                                        text = "You are about to erase your results\n(Data and saved results will not be erased)",
                                        type = "error",
                                        showConfirmButton = TRUE,
                                        showCancelButton = TRUE,
                                        callbackR = function(x)
                                        {
                                          if(x)
                                          {
                                            showNotification("Results cleared")
                                            
                                            # Refresh tables and plots
                                            output$viz <- renderPlot({})
                                            output$measures <- renderTable({})
                                            output$randnet <- renderTable({})
                                            output$aspl <- renderTable({})
                                            output$cc <- renderTable({})
                                            output$q <- renderTable({})
                                            output$tab <- renderTable({})
                                            output$asplPlot <- renderPlot({})
                                            output$ccPlot <- renderPlot({})
                                            output$qPlot <- renderPlot({})
                                            #output$perm_table <- renderTable({})
                                            output$walk_rand <- renderTable({})
                                            output$spreadr_animate <- renderPlot({})
                                            
                                            # Network Estimation tab
                                            updateSelectInput(session = session,
                                                              inputId = "estimation",
                                                              label = "Network Estimation Method",
                                                              choices = c("Community Network (CN)",
                                                                          "Naive Random Walk (NRW)",
                                                                          "Pathfinder Network (PF)",
                                                                          "Triangulated Maximally Filtered Graph (TMFG)")
                                            )
                                            
                                            # Update citation
                                            output$net_cite <- renderUI({})
                                            
                                            # Update to tabs
                                            updateNavbarPage(session = session,
                                                             inputId = "tabs",
                                                             selected = "Network Estimation"
                                            )
                                            
                                            # Hide tabs
                                            hideTab(inputId = "tabs", target = "Random Network Analyses")
                                            hideTab(inputId = "tabs", target = "Bootstrap Analyses")
                                            #hideTab(inputId = "tabs", target = "Permutation Analyses")
                                            hideTab(inputId = "tabs", target = "Random Walk Analyses")
                                            hideTab(inputId = "tabs", target = "Spreading Activation Analyses")
                                            hideTab(inputId = "tabs", target = "Save and Reset All Results")
                                            
                                            # Hide save buttons
                                            shinyjs::hide("save_nets")
                                            shinyjs::hide("save_rand")
                                            shinyjs::hide("save_boot")
                                            shinyjs::hide("save_walk")
                                            shinyjs::hide("save_act")
                                            
                                            # Random Network Analyses tab
                                            updateNumericInput(session = session,
                                                               inputId = "iters_rand",
                                                               label = "Number of Iterations",
                                                               value = 1000, min = 0, step = 100)
                                            
                                            
                                            if(exists("core_rand", envir = globalenv()))
                                            {
                                              updateSelectInput(session = session,
                                                                inputId = "cores_rand",
                                                                label = "Number of Processing Cores",
                                                                choices = core_rand,
                                                                selected = ceiling(length(core_rand) / 2)
                                              )
                                            }
                                            
                                            # Bootstrap Analyses tab
                                            updateNumericInput(session = session,
                                                               inputId = "iters_boot",
                                                               label = "Number of Iterations",
                                                               value = 1000, min = 0, step = 100)
                                            
                                            if(exists("core_boot", envir = globalenv()))
                                            {
                                              updateSelectInput(session = session,
                                                                inputId = "cores_boot",
                                                                label = "Number of Processing Cores",
                                                                choices = core_boot,
                                                                selected = ceiling(length(core_boot) / 2)
                                              )
                                            }
                                            
                                            if(network == "TMFG")
                                            {
                                              updateCheckboxGroupInput(session = session,
                                                                       "percent", label = "Proportion of Nodes Remaining",
                                                                       choiceNames = sprintf("%1.2f",seq(.50,.90,.10)),
                                                                       choiceValues = seq(.50,.90,.10), inline = TRUE,
                                                                       selected = seq(.50,.90,.10)
                                              )
                                            }
                                            
                                            ## Hide plot button
                                            shinyjs::hide("run_plot")
                                            
                                            # Permutation Analyses tab
                                            
                                            #updateSelectInput(session = session,
                                            #                  inputId = "meas_perm",
                                            #                  label = "Network Measure",
                                            #                  choices = c("Average Shortest Path Length (ASPL)",
                                            #                              "Clustering Coefficient (CC)",
                                            #                              "Modularity (Q)"
                                            #                  )
                                            #)
                                            
                                            #updateNumericInput(session = session,
                                            #                   inputId = "iters_perm",
                                            #                   label = "Number of Iterations",
                                            #                   value = 1000, min = 0, step = 100)
                                            
                                            #if(exists("core_perm", envir = globalenv()))
                                            #{
                                            #  updateSelectInput(session = session,
                                            #                    inputId = "cores_perm",
                                            #                    label = "Number of Processing Cores",
                                            #                    choices = core_perm,
                                            #                    selected = ceiling(length(core_boot) / 2)
                                            #  )
                                            #}
                                            
                                            # Random Network Analyses tab
                                            updateNumericInput(session = session,
                                                               inputId = "reps",
                                                               label = "Number of Repetitions",
                                                               value = 20, min = 0, max = Inf, step = 5)
                                            
                                            updateNumericInput(session = session,
                                                               inputId = "steps",
                                                               label = "Number of Steps",
                                                               value = 10, min = 0, max = Inf, step = 1)
                                            
                                            updateNumericInput(session = session,
                                                               inputId = "iters_walk",
                                                               label = "Number of Iterations",
                                                               value = 10000, min = 0, max = Inf, step = 1000)
                                            
                                            if(exists("core_walk", envir = globalenv()))
                                            {
                                              updateSelectInput(session = session,
                                                                inputId = "cores_walk",
                                                                label = "Number of Processing Cores",
                                                                choices = core_walk,
                                                                selected = ceiling(length(core_walk) / 2)
                                              )
                                            }
                                            
                                            # Spreading Activation Analyses tab
                                            updateNumericInput(session = session,
                                                               inputId = "retention",
                                                               label = "Retention (proportion of activation that remains in spreading node)",
                                                               value = 0.5, min = 0, max = 1, step = .05)
                                            
                                            updateNumericInput(session = session,
                                                               inputId = "time",
                                                               label = "Number of Time Steps",
                                                               value = 10, min = 0, max = Inf, step = 1)
                                            
                                            updateNumericInput(session = session,
                                                               inputId = "decay",
                                                               label = "Decay (activation lost at each time step)",
                                                               value = 0, min = 0, max = 1, step = .05)
                                            
                                            updateNumericInput(session = session,
                                                               inputId = "suppress",
                                                               label = "Suppress (activation less than value is set to zero)",
                                                               value = 0, min = 0, max = 1, step = .001)
                                            
                                            updateSelectInput(session = session,
                                                              inputId = "animate_size",
                                                              label = "Plot Size",
                                                              choices = c("Small (500 x 500)", "Medium (900 x 900)", "Large (1400 x 1400)"),
                                                              selected = "Medium (900 x 900)"
                                            )
                                            
                                            ## Show inputs
                                            shinyjs::show("network_select")
                                            shinyjs::show("retention")
                                            shinyjs::show("time")
                                            shinyjs::show("decay")
                                            shinyjs::show("suppress")
                                            shinyjs::show("set_act")
                                            
                                            ## Hide inputs
                                            shinyjs::hide("animate")
                                            shinyjs::hide("animate_size")
                                            shinyjs::hide("animate_slider")
                                            shinyjs::hide("reset_act")
                                            shinyjs::hide("reset")
                                            shinyjs::hide("node_select")
                                            shinyjs::hide("run_spr_act")
                                            
                                            if(exists("clean", envir = globalenv()))
                                            {rm(list = ls(envir = globalenv())[-suppressWarnings(na.omit(match(c("prev.env", "clean", "dat", "group", prev.env), ls(globalenv()))))], envir = globalenv())
                                            }else if(exists("group", envir = globalenv()) && exists("dat", envir = globalenv()))
                                            {rm(list = ls(envir = globalenv())[-suppressWarnings(na.omit(match(c("prev.env", "dat", "group"), prev.env), ls(globalenv())))], envir = globalenv())
                                            }else{rm(list = ls(envir = globalenv())[-suppressWarnings(na.omit(match(c("prev.env", "dat", "group"), ls(globalenv()))))], envir = globalenv())}
                                            
                                          }
                                        })
               })
  
  
  
  onStop(function(x)
  {
    # Save results into condensed list
    resultShiny <<- list()
    
    if(exists("dat", envir = globalenv()))
    {resultShiny$data <<- dat}
    
    if(exists("group", envir = globalenv()))
    {resultShiny$group <<- group}
    
    if(exists("eq", envir = globalenv())){
      resultShiny$equated <<- eq
    }
    
    if(exists("nets", envir = globalenv()))
    {resultShiny$network <<- nets}
    
    if(exists("meas.mat", envir = globalenv()))
    {resultShiny$measures <<- meas.mat}
    
    if(exists("plots", envir = globalenv()))
    {resultShiny$comparePlot <<- plots}
    
    if(exists("rand.res", envir = globalenv()))
    {resultShiny$randomTest <<- rand.res}
    
    if(exists("res_boot", envir = globalenv()))
    {resultShiny$bootstrap <<- unlist(res_boot, recursive = FALSE)}
    
    if(exists("bootTest", envir = globalenv()))
    {resultShiny$bootstrapTest <<- bootTest}
    
    if(exists("pbplot", envir = globalenv()))
    {resultShiny$bootstrapPlot <<- pbplot}
    
    #if(exists("perm.table", envir = globalenv()))
    #{resultShiny$permutationTest <<- perm.table}
    
    if(exists("rw", envir = globalenv()))
    {resultShiny$randomWalk <<- rw}
    
    if(exists("sa", envir = globalenv()))
    {resultShiny$spreadingActivation <<- sa}
    
    if(exists("plot_list", envir = globalenv()))
    {
      if(any(!is.null(unlist(plot_list))))
      {resultShiny$spreadingActivationPlot <<- plot_list}
    }
    
    # Remove all other variables from global environment
    rm(list = ls(envir = globalenv())[-match(c("resultShiny", prev.env), ls(globalenv()))], envir = globalenv())
    
    # Print message to user to let them know output saved as resultShiny
    message(
      '\n\nThe last analyses and results in `SemNeTShiny()` have been saved in an object called "resultShiny" in R\'s environment'
    )
    
    # Remove plots from user view
    if(!is.null(dev.list()))
    {dev.off()}
  }
  )
  
}
AlexChristensen/SemNeT documentation built on Aug. 21, 2023, 11:01 p.m.