inst/shiny/steplist_creator/server.R

#' Server of Steplist Creator `shiny` app
#'
#' The UI of Steplist Creator `shiny` app including server modules.
#'
#' @noRd
server <- function(input, output, session) {
  # Initialize steplist
  steplist <- reactiveVal(empty_steplist())

  # Upload
  observeEvent(input$upload, {
    ## Actual upload
    check_if_rds(input$upload)
    upload_content <- readRDS(input$upload$datapath)
    check_if_steplist(upload_content)
    steplist(upload_content)

    ## Update first line of IF / IFNOT / OUTC
    n <- n_if()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#if_row", i))
      }
    }
    n_if(1)
    updateSelectInput(session, "if_select_input1", label = NULL,
                      c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))))

    n <- n_ifnot()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#ifnot_row", i))
      }
    }
    n_ifnot(1)
    updateSelectInput(session, "ifnot_select_input1", label = NULL,
                      c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))))

    n <- n_outc()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#outc_row", i))
      }
    }
    n_outc(1)
    updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                      choices = c(Choose = "", steplist() %>% get_options_outc()))
  })

  # Download
  output$download <- downloadHandler(
    filename = function() {
      "steplist.rds"
    },
    content = function(file) {
      steplist_temp <- steplist()
      check_if_steplist(steplist_temp)
      steplist_temp %<>% epicmodel::uncheck_steplist()
      saveRDS(steplist_temp, file)
    }
  )

  # WHAT
  whatServer("what_tab", steplist)

  # DOES
  doesServer("does_tab", steplist)

  # WHERE
  whereServer("where_tab", steplist)

  # Module
  moduleTabServer("module_tab", steplist)

  # Incompatible Component Causes (ICC)
  iccServer("icc_tab", steplist)

  # STEP

  # Dropdown Menus STEP -----------------------------------------------------
  output$select_subject <- renderUI(selectInput("select_subject", "Subject (WHAT)", c(Choose = "", steplist()[["what"]]$key_what %>% magrittr::extract(order(.))), selectize = T))
  output$select_does <- renderUI(selectInput("select_does", "DOES", c(Choose = "", steplist()[["does"]]$key_does %>% magrittr::extract(order(.))), selectize = T))
  output$select_where <- renderUI(selectInput("select_where", "WHERE", c(Choose = "", steplist()[["where"]]$key_where %>% magrittr::extract(order(.))), selectize = T))
  output$step_input_module <- renderUI(selectInput("step_input_module", label = "Module", c(Choose = "", steplist()[["module"]]$key_module %>% magrittr::extract(order(.))), selectize = T))

  # Dropdown Menus THEN -----------------------------------------------------
  output$then_select_subject <- renderUI(selectInput("then_select_subject", "Subject (WHAT)", c(Choose = "", steplist()[["what"]]$key_what %>% magrittr::extract(order(.))), selectize = T))
  output$then_select_does <- renderUI(selectInput("then_select_does", "DOES", c(Choose = "", steplist()[["does"]]$key_does %>% magrittr::extract(order(.))), selectize = T))
  output$then_select_where <- renderUI(selectInput("then_select_where", "WHERE", c(Choose = "", steplist()[["where"]]$key_where %>% magrittr::extract(order(.))), selectize = T))

  # Object Input STEP -------------------------------------------------------
  ## Define check_object, which indicates if for the current DOES segment we need a WHAT or THEN object
  check_object <- reactive(min(length(steplist()[["does"]]$then_object_does[steplist()[["does"]]$key_does == input$select_does]),
                               steplist()[["does"]]$then_object_does[steplist()[["does"]]$key_does == input$select_does]))

  ## Define object_what & object_then
  object_what <- reactiveVal()
  object_then <- reactiveVal()

  ## object_what remembers the current value of select_object_what, even if select_object_then is currently rendered
  observeEvent(input$select_object_what, {
    object_what(input$select_object_what)
  })

  ## object_then remembers the current value of select_object_then, even if select_object_what is currently rendered
  observeEvent(input$select_object_then, {
    object_then(input$select_object_then)
  })

  ## object_key holds the current object value (WHAT or THEN depending on check_object)
  object_key <- reactive({
    if (check_object() == "1") {
      object_what("")
      object_key <- input$select_object_then
    } else {
      object_then("")
      object_key <- input$select_object_what
    }
  })

  ## Render the corresponding selectInput depending on check_object
  output$select_object <- renderUI({
    if (check_object() == "1") {
      selectInput("select_object_then", "Object (THEN)", selected = object_then(), c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))), selectize = T)
    } else {
      selectInput("select_object_what", "Object (WHAT)", selected = object_what(), c(Choose = "", steplist()[["what"]]$key_what %>% magrittr::extract(order(.))), selectize = T)
    }
  })

  # Object Input THEN -------------------------------------------------------
  ## Define check_object, which indicates if for the current DOES segment we need a WHAT or THEN object
  then_check_object <- reactive(min(length(steplist()[["does"]]$then_object_does[steplist()[["does"]]$key_does == input$then_select_does]),
                               steplist()[["does"]]$then_object_does[steplist()[["does"]]$key_does == input$then_select_does]))

  ## Define object_what & object_then
  then_object_what <- reactiveVal()
  then_object_then <- reactiveVal()

  ## object_what remembers the current value of select_object_what, even if select_object_then is currently rendered
  observeEvent(input$then_select_object_what, {
    then_object_what(input$then_select_object_what)
  })

  ## object_then remembers the current value of select_object_then, even if select_object_what is currently rendered
  observeEvent(input$then_select_object_then, {
    then_object_then(input$then_select_object_then)
  })

  ## object_key holds the current object value (WHAT or THEN depending on check_object)
  then_object_key <- reactive({
    if (then_check_object() == "1") {
      then_object_what("")
      then_object_key <- input$then_select_object_then
    } else {
      then_object_then("")
      then_object_key <- input$then_select_object_what
    }
  })

  ## Render the corresponding selectInput depending on check_object
  output$then_select_object <- renderUI({
    if (then_check_object() == "1") {
      selectInput("then_select_object_then", "Object (THEN)", selected = then_object_then(), c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))), selectize = T)
    } else {
      selectInput("then_select_object_what", "Object (WHAT)", selected = then_object_what(), c(Choose = "", steplist()[["what"]]$key_what %>% magrittr::extract(order(.))), selectize = T)
    }
  })

  # IF Input ----------------------------------------------------------------
  n_if <- reactiveVal(1)

  ## more_lines_if
  observeEvent(input$more_lines_if ,{
    n <- n_if() + 1
    n_if(n)

    insertUI(
      selector = "#if_placeholder",
      where = "beforeEnd",
      ui = div(
        id = paste0("if_row", n),
        class = "input-row",
        fluidRow(
          column(2, numericInput(paste0("if_numeric_input", n), label = NULL, value = 1, min = 1)),
          column(10, selectInput(paste0("if_select_input", n), label = NULL,
                                 choices = c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.)))))
        )))
  })

  ## less_lines_if
  observeEvent(input$less_lines_if ,{
    n <- n_if()

    if (n > 1) {
      removeUI(
        selector = paste0("#if_row", n)
      )
      n_if(n - 1)
    } else {
      updateSelectInput(session, "if_select_input1", label = NULL, selected = "")
      updateNumericInput(session, "if_numeric_input1", label = NULL, value = 1)
    }
  })

  # IFNOT Input -------------------------------------------------------------
  n_ifnot <- reactiveVal(1)

  ## more_lines_ifnot
  observeEvent(input$more_lines_ifnot ,{
    n <- n_ifnot() + 1
    n_ifnot(n)

    insertUI(
      selector = "#ifnot_placeholder",
      where = "beforeEnd",
      ui = div(
        id = paste0("ifnot_row", n),
        class = "input-row",
        fluidRow(
          column(2, numericInput(paste0("ifnot_numeric_input", n), label = NULL, value = 1, min = 1)),
          column(10, selectInput(paste0("ifnot_select_input", n), label = NULL,
                                 choices = c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.)))))
        )))
  })

  ## less_lines_ifnot
  observeEvent(input$less_lines_ifnot ,{
    n <- n_ifnot()

    if (n > 1) {
      removeUI(
        selector = paste0("#ifnot_row", n)
      )
      n_ifnot(n - 1)
    } else {
      updateSelectInput(session, "ifnot_select_input1", label = NULL, selected = "")
      updateNumericInput(session, "ifnot_numeric_input1", label = NULL, value = 1)
    }
  })

  # IF Preview --------------------------------------------------------------
  ## step_if_id
  output$step_if_id <- renderText({
    val_select <- vector(mode = "character", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_select_input",i)] %||% ""
    }
    val_numeric <- vector(mode = "numeric", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_numeric_input",i)]
    }

    val <- fun_create_step_if_id(input_select = val_select, input_numeric = val_numeric, then_data = steplist()[["then"]])

    val[[1]]
  })

  ## step_if_desc
  output$step_if_desc <- renderText({
    val_select <- vector(mode = "character", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_select_input",i)] %||% ""
    }
    val_numeric <- vector(mode = "numeric", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_numeric_input",i)]
    }

    fun_create_step_if_desc(input_select = val_select, input_numeric = val_numeric)
  })

  # IFNOT Preview -----------------------------------------------------------
  ## step_ifnot_id
  output$step_ifnot_id <- renderText({
    val_select <- vector(mode = "character", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_select_input",i)] %||% ""
    }
    val_numeric <- vector(mode = "numeric", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_numeric_input",i)]
    }

    val <- fun_create_step_ifnot_id(input_select = val_select, input_numeric = val_numeric, then_data = steplist()[["then"]])

    val[[1]]
  })

  ## step_ifnot_desc
  output$step_ifnot_desc <- renderText({
    val_select <- vector(mode = "character", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_select_input",i)] %||% ""
    }
    val_numeric <- vector(mode = "numeric", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_numeric_input",i)]
    }

    fun_create_step_ifnot_desc(input_select = val_select, input_numeric = val_numeric)
  })

  # THEN Preview ------------------------------------------------------------
  ## step_then_id
  output$step_then_id <- renderText({
    fun_create_then_step_id(subject_key = input$then_select_subject, does_key = input$then_select_does, object_key = (then_object_key() %||% ""),
                            where_key = input$then_select_where, check_object = (then_check_object() %||% ""),
                            what_data = steplist()[["what"]], does_data = steplist()[["does"]],
                            where_data = steplist()[["where"]], then_data = steplist()[["then"]])
  })

  ## step_then_desc
  output$step_then_desc <- renderText({
    fun_create_then_step_desc(subject_key = input$then_select_subject, does_key = input$then_select_does, object_key = (then_object_key() %||% ""),
                              where_key = input$then_select_where, check_object = (then_check_object() %||% ""),
                              what_data = steplist()[["what"]], does_data = steplist()[["does"]],
                              where_data = steplist()[["where"]], then_data = steplist()[["then"]])
  })


  # THEN Table --------------------------------------------------------------
  output$then_tbl <- DT::renderDataTable(steplist()[["then"]], server = T, rownames = F, selection = "none",
                                         options = list(scrollY = '500px', scrollCollapse = TRUE,
                                                        paging = FALSE, order = list(0,'asc'),
                                                        searchHighlight = TRUE))

  # THEN Add ----------------------------------------------------------------
  observeEvent(input$then_add, {
    steplist_temp <- steplist()

    id_then_temp <- fun_create_then_step_id(subject_key = input$then_select_subject, does_key = input$then_select_does,
                                            object_key = (then_object_key() %||% ""), where_key = input$then_select_where,
                                            check_object = (then_check_object() %||% ""), what_data = steplist_temp[["what"]],
                                            does_data = steplist_temp[["does"]], where_data = steplist_temp[["where"]],
                                            then_data = steplist_temp[["then"]])
    desc_then_temp <- fun_create_then_step_desc(subject_key = input$then_select_subject, does_key = input$then_select_does,
                                                object_key = (then_object_key() %||% ""), where_key = input$then_select_where,
                                                check_object = (then_check_object() %||% ""), what_data = steplist_temp[["what"]],
                                                does_data = steplist_temp[["does"]], where_data = steplist_temp[["where"]],
                                                then_data = steplist_temp[["then"]])
    if (desc_then_temp == "") {
      shinyalert::shinyalert(title = "Warning!", text = "The THEN statement is empty.", type = "warning")
    }
    if (id_then_temp %>% magrittr::is_in(steplist_temp[["then"]]$id_then)) {
      shinyalert::shinyalert(title = "Warning!", text = "This THEN statement already exists.", type = "warning")
      updateSelectInput(session, inputId = "then_select_subject", "Subject (WHAT)", selected = "")
      updateSelectInput(session, inputId = "then_select_does", "DOES", selected = "")
      updateSelectInput(session, inputId = "then_select_where", "WHERE", selected = "")
      then_object_what("")
      then_object_then("")
    }
    if ((id_then_temp %>% magrittr::is_in(steplist_temp[["then"]]$id_then) %>% magrittr::not()) & desc_then_temp != "") {
      to_add <- data.frame(id_then = id_then_temp,
                           desc_then = desc_then_temp)

      steplist_temp[["then"]] <- rbind(steplist_temp[["then"]],to_add)
      steplist(steplist_temp)

      ## Clear input fields
      updateSelectInput(session, inputId = "then_select_subject", "Subject (WHAT)", selected = "")
      updateSelectInput(session, inputId = "then_select_does", "DOES", selected = "")
      updateSelectInput(session, inputId = "then_select_where", "WHERE", selected = "")
      then_object_what("")
      then_object_then("")

      ## Clear and update if
      n <- n_if()
      if (n > 1) {
        for (i in 2:n) {
          removeUI(selector = paste0("#if_row", i))
        }
      }
      n_if(1)
      updateSelectInput(session, "if_select_input1", label = NULL, selected = "",
                        choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
      updateNumericInput(session, "if_numeric_input1", label = NULL, value = 1)

      ## Clear and update ifnot
      n <- n_ifnot()
      if (n > 1) {
        for (i in 2:n) {
          removeUI(selector = paste0("#ifnot_row", i))
        }
      }
      n_ifnot(1)
      updateSelectInput(session, "ifnot_select_input1", label = NULL, selected = "",
                        choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
      updateNumericInput(session, "ifnot_numeric_input1", label = NULL, value = 1)
    }
  })


  # THEN Delete -------------------------------------------------------------
  observeEvent(input$then_delete, {
    shinyalert::shinyalert(title = "Delete row", type = "input", inputType = "text", inputPlaceholder = "Provide id_then",
                           callbackR = function(then_delete_id) {
                             steplist_temp <- steplist()
                             steplist_temp[["then"]] %<>% dplyr::filter(id_then != then_delete_id)
                             steplist(steplist_temp)

                             ## Clear and update if
                             n <- n_if()
                             if (n > 1) {
                               for (i in 2:n) {
                                 removeUI(selector = paste0("#if_row", i))
                               }
                             }
                             n_if(1)
                             updateSelectInput(session, "if_select_input1", label = NULL, selected = "",
                                               choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
                             updateNumericInput(session, "if_numeric_input1", label = NULL, value = 1)

                             ## Clear and update ifnot
                             n <- n_ifnot()
                             if (n > 1) {
                               for (i in 2:n) {
                                 removeUI(selector = paste0("#ifnot_row", i))
                               }
                             }
                             n_ifnot(1)
                             updateSelectInput(session, "ifnot_select_input1", label = NULL, selected = "",
                                               choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
                             updateNumericInput(session, "ifnot_numeric_input1", label = NULL, value = 1)
                             }
    )
  })

  # STEP Table --------------------------------------------------------------
  output$step_tbl <- DT::renderDataTable(steplist()[["step"]], server = T, rownames = F, selection = "none",
                                         editable = list(target = "cell", disable = list(columns = c(0,1))),
                                         options = list(scrollY = '520px', scrollCollapse = TRUE,
                                                        paging = FALSE, order = list(0,'asc'),
                                                        searchHighlight = TRUE))

  observeEvent(input$step_tbl_cell_edit, {
    steplist_temp <- steplist()
    steplist_temp[["step"]] <- DT::editData(steplist_temp[["step"]], input$step_tbl_cell_edit, 'step_tbl', rownames = F)
    steplist(steplist_temp)

    ## Update outc
    n <- n_outc()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#outc_row", i))
      }
    }
    n_outc(1)
    updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                      choices = c(Choose = "", steplist_temp %>% get_options_outc()))
  })


  # STEP Add ----------------------------------------------------------------
  observeEvent(input$step_add, {
    steplist_temp <- steplist()

    ## Extract IFNOT inputs
    val_ifnot_select <- vector(mode = "character", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_ifnot_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_select_input",i)] %||% ""
    }
    val_ifnot_numeric <- vector(mode = "numeric", length = isolate(n_ifnot()))
    for (i in 1:isolate(n_ifnot())) {
      val_ifnot_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("ifnot_numeric_input",i)]
    }

    val_id_ifnot <- fun_create_step_ifnot_id(input_select = val_ifnot_select, input_numeric = val_ifnot_numeric, then_data = steplist()[["then"]])
    val_desc_ifnot <- fun_create_step_ifnot_desc(input_select = val_ifnot_select, input_numeric = val_ifnot_numeric)

    ## Make IFNOT checks
    if (val_id_ifnot[[2]]) {
      shinyalert::shinyalert(title = "Warning!", text = "Some scenario of the IFNOT condition contains duplicate statements.", type = "warning")
    }
    if (val_id_ifnot[[3]]) {
      shinyalert::shinyalert(title = "Warning!", text = "Some scenario of the IFNOT condition are identical.", type = "warning")
    }

    ## Extract IF values
    val_if_select <- vector(mode = "character", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_if_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_select_input",i)] %||% ""
    }
    val_if_numeric <- vector(mode = "numeric", length = isolate(n_if()))
    for (i in 1:isolate(n_if())) {
      val_if_numeric[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("if_numeric_input",i)]
    }

    val_id_if <- fun_create_step_if_id(input_select = val_if_select, input_numeric = val_if_numeric, then_data = steplist()[["then"]])
    val_desc_if <- fun_create_step_if_desc(input_select = val_if_select, input_numeric = val_if_numeric)

    ## Make IF checks
    if (val_id_if[[2]]) {
      shinyalert::shinyalert(title = "Warning!", text = "The IF condition contains duplicate statement.", type = "warning")
    }
    if (val_id_if[[3]]) {
      shinyalert::shinyalert(title = "Warning!", text = "Some scenario of the IF condition are identical.", type = "warning")
    }

    ## Addition process begins, if no warnings
    if ((!val_id_if[[2]]) & (!val_id_if[[3]]) & (!val_id_ifnot[[2]]) & (!val_id_ifnot[[3]])) {

      ### Get THEN ID and desc
      id_then_temp <- fun_create_then_step_id(subject_key = input$select_subject, does_key = input$select_does,
                                              object_key = (object_key() %||% ""), where_key = input$select_where,
                                              check_object = (check_object() %||% ""), what_data = steplist()[["what"]],
                                              does_data = steplist()[["does"]], where_data = steplist()[["where"]],
                                              then_data = steplist()[["then"]])
      desc_then_temp <- fun_create_then_step_desc(subject_key = input$select_subject, does_key = input$select_does,
                                                  object_key = (object_key() %||% ""), where_key = input$select_where,
                                                  check_object = (check_object() %||% ""), what_data = steplist()[["what"]],
                                                  does_data = steplist()[["does"]], where_data = steplist()[["where"]],
                                                  then_data = steplist()[["then"]])

      ### Add THEN if not already there
      if ((id_then_temp %>% magrittr::is_in(steplist_temp[["then"]]$id_then) %>% magrittr::not()) & desc_then_temp != "") {
        to_add_then <- data.frame(id_then = id_then_temp,
                                  desc_then = desc_then_temp)

        steplist_temp[["then"]] <- rbind(steplist_temp[["then"]],to_add_then)
      }

      ### STEP addition

      to_add_desc_step <- fun_create_step_desc(input_if = val_desc_if, input_ifnot = val_desc_ifnot, input_then = desc_then_temp,
                                               input_end_step = as.character(as.numeric(input$step_input_end)))
      to_add_id_step <- fun_create_step_id(input_if = val_id_if[[1]], input_ifnot = val_id_ifnot[[1]], input_then = id_then_temp)

      if (desc_then_temp == "") {
        shinyalert::shinyalert(title = "Warning!", text = "The THEN statement is empty.", type = "warning")
      }
      if (to_add_id_step %>% magrittr::is_in(steplist_temp[["step"]]$id_step)) {
        shinyalert::shinyalert(title = "Warning!", text = "This STEP already exists.", type = "warning")
      }
      if (to_add_id_step %>% magrittr::is_in(steplist_temp[["step"]]$id_step) %>% magrittr::not() & desc_then_temp != "") {
        to_add_step <- data.frame(id_step = to_add_id_step,
                                  desc_step = to_add_desc_step,
                                  end_step = as.character(as.numeric(input$step_input_end)),
                                  module_step = fun_get_module_id(steplist_temp[["module"]],input$step_input_module),
                                  note_step = input$step_input_note %>% stringr::str_trim("both"),
                                  ref_step = input$step_input_ref %>% stringr::str_trim("both"))

        steplist_temp[["step"]] <- rbind(steplist_temp[["step"]],to_add_step)
        steplist(steplist_temp)

        updateSelectInput(session, inputId = "select_subject", "Subject (WHAT)", selected = "")
        updateSelectInput(session, inputId = "select_does", "DOES", selected = "")
        updateSelectInput(session, inputId = "select_where", "WHERE", selected = "")
        updateTextInput(session, inputId = "step_input_note", label = "Notes", value = "")
        updateTextInput(session, inputId = "step_input_ref", label = "References", value = "")
        object_what("")
        object_then("")

        ## Clear if
        n <- n_if()
        if (n > 1) {
          for (i in 2:n) {
            removeUI(selector = paste0("#if_row", i))
          }
        }
        n_if(1)
        updateSelectInput(session, "if_select_input1", label = NULL, selected = "",
                          choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
        updateNumericInput(session, "if_numeric_input1", label = NULL, value = 1)

        ## Clear ifnot
        n <- n_ifnot()
        if (n > 1) {
          for (i in 2:n) {
            removeUI(selector = paste0("#ifnot_row", i))
          }
        }
        n_ifnot(1)
        updateSelectInput(session, "ifnot_select_input1", label = NULL, selected = "",
                          choices = c(Choose = "", steplist_temp[["then"]]$desc_then %>% magrittr::extract(order(.))))
        updateNumericInput(session, "ifnot_numeric_input1", label = NULL, value = 1)

        ## Update outc
        n <- n_outc()
        if (n > 1) {
          for (i in 2:n) {
            removeUI(selector = paste0("#outc_row", i))
          }
        }
        n_outc(1)
        updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                          choices = c(Choose = "", steplist_temp %>% get_options_outc()))
      }
    }
  })


  # STEP Delete -------------------------------------------------------------
  observeEvent(input$step_delete, {
    shinyalert::shinyalert(title = "Delete row", type = "input", inputType = "text", inputPlaceholder = "Provide id_step",
                           callbackR = function(step_delete_id) {
                             steplist_temp <- steplist()
                             steplist_temp[["step"]] %<>% dplyr::filter(id_step != step_delete_id)
                             steplist(steplist_temp)

                             ## Update outc
                             n <- n_outc()
                             if (n > 1) {
                               for (i in 2:n) {
                                 removeUI(selector = paste0("#outc_row", i))
                               }
                             }
                             n_outc(1)
                             updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                                               choices = c(Choose = "", steplist_temp %>% get_options_outc()))
                             }
    )
  })

  # STEP Clear --------------------------------------------------------------
  observeEvent(input$step_clear, {
    updateSelectInput(session, inputId = "select_subject", "Subject (WHAT)", selected = "")
    updateSelectInput(session, inputId = "select_does", "DOES", selected = "")
    updateSelectInput(session, inputId = "select_where", "WHERE", selected = "")
    updateTextInput(session, inputId = "step_input_note", label = "Notes", value = "")
    updateTextInput(session, inputId = "step_input_ref", label = "References", value = "")
    updateSelectInput(session, inputId = "step_input_module", label = "Module", selected = "")
    updateCheckboxInput(session ,inputId = "step_input_end", label = "End step?", value = F)
    object_what("")
    object_then("")

    ## Clear if
    n <- n_if()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#if_row", i))
      }
    }
    n_if(1)
    updateSelectInput(session, "if_select_input1", label = NULL, selected = "",
                      choices = c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))))
    updateNumericInput(session, "if_numeric_input1", label = NULL, value = 1)

    ## Clear ifnot
    n <- n_ifnot()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#ifnot_row", i))
      }
    }
    n_ifnot(1)
    updateSelectInput(session, "ifnot_select_input1", label = NULL, selected = "",
                      choices = c(Choose = "", steplist()[["then"]]$desc_then %>% magrittr::extract(order(.))))
    updateNumericInput(session, "ifnot_numeric_input1", label = NULL, value = 1)
  })


  # OUTCOME Input -----------------------------------------------------------
  n_outc <- reactiveVal(1)

  ## more_lines_outc
  observeEvent(input$more_lines_outc ,{
    n <- n_outc() + 1
    n_outc(n)

    insertUI(
      selector = "#outc_placeholder",
      where = "beforeEnd",
      ui = div(
        id = paste0("outc_row", n),
        class = "input-row",
        selectInput(paste0("outc_select_input", n), label = NULL,
                                 choices = c(Choose = "", steplist() %>% get_options_outc()))
        ))
  })

  ## less_lines_outc
  observeEvent(input$less_lines_outc ,{
    n <- n_outc()

    if (n > 1) {
      removeUI(
        selector = paste0("#outc_row", n)
      )
      n_outc(n - 1)
    } else {
      updateSelectInput(session, "outc_select_input1", label = NULL, selected = "")
    }
  })

  # OUTCOME Table -----------------------------------------------------------
  output$outc_tbl <- DT::renderDataTable(steplist()[["outc"]], server = T, rownames = F, selection = "none",
                                         options = list(scrollY = '750px', scrollCollapse = TRUE,
                                                        paging = FALSE, order = list(0,'asc'),
                                                        searchHighlight = TRUE))

  # OUTCOME Add -------------------------------------------------------------
  observeEvent(input$outc_add, {
    steplist_temp <- steplist()

    ## Extract inputs
    val_outc_select <- vector(mode = "character", length = isolate(n_outc()))
    for (i in 1:isolate(n_outc())) {
      val_outc_select[i] <- lapply(reactiveValuesToList(input), unclass)[paste0("outc_select_input",i)] %||% ""
    }

    to_add_desc_outc <- val_outc_select %>% create_outc_desc(coll = F)

    if (to_add_desc_outc %>% create_outc_desc(coll = T) == "") {
      shinyalert::shinyalert(title = "Warning!", text = "All inputs are empty.", type = "warning")
    } else {

      val_outc_id <- vector(mode = "character", length = length(to_add_desc_outc))
      for (i in 1:length(val_outc_id)) {
        val_outc_id[i] <- to_add_desc_outc[i] %>% get_id_outc(steplist_temp[["step"]])
      }

      if (val_outc_id %>% unique() %>% length() %>% magrittr::equals(length(val_outc_id)) %>% magrittr::not()) {
        shinyalert::shinyalert(title = "Warning!", text = "The outcome definition contains duplicate statements.", type = "warning")
      } else {
        to_add_id_outc <- val_outc_id %>% stringr::str_c(collapse = "+")

        if (check_outc_duplicates(to_add_id_outc, steplist_temp[["outc"]])) {
          shinyalert::shinyalert(title = "Warning!", text = "The outcome definition already exists.", type = "warning")
        } else {
          to_add_outc <- data.frame(id_outc = to_add_id_outc,
                                    desc_outc = to_add_desc_outc %>% create_outc_desc(coll = T))

          steplist_temp[["outc"]] <- rbind(steplist_temp[["outc"]],to_add_outc)
          steplist(steplist_temp)

          n <- n_outc()
          if (n > 1) {
            for (i in 2:n) {
              removeUI(selector = paste0("#outc_row", i))
            }
          }
          n_outc(1)
          updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                            choices = c(Choose = "", steplist() %>% get_options_outc()))
        }
      }
    }
  })

  # OUTCOME Delete ----------------------------------------------------------
  observeEvent(input$outc_delete, {
    shinyalert::shinyalert(title = "Delete row", type = "input", inputType = "text", inputPlaceholder = "Provide id_outc",
                           callbackR = function(outc_delete_id) {
                             steplist_temp <- steplist()
                             steplist_temp[["outc"]] %<>% dplyr::filter(id_outc != outc_delete_id)
                             steplist(steplist_temp)
                           }
    )
  })

  # OUTCOME Clear -----------------------------------------------------------
  observeEvent(input$outc_clear, {
    n <- n_outc()
    if (n > 1) {
      for (i in 2:n) {
        removeUI(selector = paste0("#outc_row", i))
      }
    }
    n_outc(1)
    updateSelectInput(session, "outc_select_input1", label = NULL, selected = "",
                      choices = c(Choose = "", steplist() %>% get_options_outc()))
  })

}

Try the epicmodel package in your browser

Any scripts or data that you put into this service are public.

epicmodel documentation built on April 12, 2025, 1:59 a.m.