search

Home  >  Q&A  >  body text

In R Shiny, how do I make a checkbox table that also has separate rows/columns of "select all" checkboxes?

Using R Shiny, I want to display a table (e.g. via the "DT" package) where each cell contains a checkbox. Next to each row and column header, I want to display a "Select All"/"Master Checkbox" that, when selected, will select all the checkboxes in the corresponding row or column. As an added feature, the checkbox in the upper left cell will select all checkboxes in the table. one example:

Tried js

I found an example of this functionality here where one column has a main checkbox (using some JavaScript) but can't figure out how to extend it to my requirements.

Example of what I tried

library(shiny)
library(DT)

ui <- fluidPage(
    # Sidebar panel
    sidebarPanel(),
    
    # Main panel with the table
    mainPanel(
        DTOutput("myTable")
    )
)

server <- function(input, output){
    dat <- data.frame(
        vapply(1:6, function(i){
            as.character(
                checkboxInput(paste0("col1-", i), label = NULL, width = "150px")
            )
        }, character(1)),
        vapply(1:6, function(i){
            as.character(
                checkboxInput(paste0("col2-", i), label = NULL, width = "150px")
            )
        }, character(1)),
        vapply(1:6, function(i){
            as.character(
                checkboxInput(paste0("col3-", i), label = NULL, width = "150px")
            )
        }, character(1))
    )
    
    names(dat) <- c(
        as.character(checkboxInput("col1", label = "1", width = "150px")),
        as.character(checkboxInput("col2", label = "2", width = "150px")),
        as.character(checkboxInput("col3", label = "3", width = "150px"))
    )
    
    row_names <- LETTERS[1:6]
    rownames(dat) <- row_names
    
    output$myTable <- renderDT({
        datatable(
            dat, 
            escape = FALSE,
            options = list(
                columnDefs = list(
                    list(targets = c(1, 2, 3), orderable = FALSE, className = "dt-center")
                )
            ),
            callback = JS(
                "$('#col1').on('click', function(){",
                "  var cboxes = $('[id^=col1-]');",
                "  var checked = $('#col1').is(':checked');",
                "  cboxes.each(function(i, cbox) {",
                "    $(cbox).prop('checked', checked);",
                "  });",
                "});",
                "$('#col2').on('click', function(){",
                "  var cboxes = $('[id^=col2-]');",
                "  var checked = $('#col2').is(':checked');",
                "  cboxes.each(function(i, cbox) {",
                "    $(cbox).prop('checked', checked);",
                "  });",
                "});",
                "$('#col3').on('click', function(){",
                "  var cboxes = $('[id^=col3-]');",
                "  var checked = $('#col3').is(':checked');",
                "  cboxes.each(function(i, cbox) {",
                "    $(cbox).prop('checked', checked);",
                "  });",
                "});"
            )
        )
    })
}

shinyApp(ui, server)

This is a start, but I can't figure out how to get the main checkbox next to the row, nor can I find a main checkbox in the upper left corner of all the boxes. Also, the whole thing is a bit big - it would be nice if it was more compact.

P粉884667022P粉884667022473 days ago519

reply all(1)I'll reply

  • P粉757640504

    P粉7576405042023-09-09 14:55:57

    library(shiny)
    library(DT)
    
    rowName <- function(L) {
      as.character(
        checkboxInput(paste0("row", L), label = L, width = "150px")
      )
    }
    rowNames <- vapply(LETTERS[1:6], rowName, character(1))
    
    
    ui <- fluidPage(
      # Sidebar panel
      sidebarPanel(),
      
      # Main panel with the table
      mainPanel(
        DTOutput("myTable")
      )
    )
    
    server <- function(input, output){
      dat <- data.frame(
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col1-", i), label = NULL, width = "150px")
          )
        }, character(1)),
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col2-", i), label = NULL, width = "150px")
          )
        }, character(1)),
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col3-", i), label = NULL, width = "150px")
          )
        }, character(1))
      )
      
      names(dat) <- c(
        as.character(checkboxInput("col1", label = "1", width = "150px")),
        as.character(checkboxInput("col2", label = "2", width = "150px")),
        as.character(checkboxInput("col3", label = "3", width = "150px"))
      )
      
      rownames(dat) <- rowNames
      
      output$myTable <- renderDT({
        datatable(
          dat, 
          escape = FALSE,
          select = "none",
          options = list(
            columnDefs = list(
              list(targets = c(1, 2, 3), orderable = FALSE, className = "dt-center")
            ),
            initComplete = JS(
              "function() {",
              "  $('#col1').on('click', function(){",
              "    var cboxes = $('[id^=col1-]');",
              "    var checked = $('#col1').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#col2').on('click', function(){",
              "    var cboxes = $('[id^=col2-]');",
              "    var checked = $('#col2').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#col3').on('click', function(){",
              "    var cboxes = $('[id^=col3-]');",
              "    var checked = $('#col3').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowA').on('click', function(){",
              "    var cboxes = $('[id$=-A]');",
              "    var checked = $('#rowA').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowB').on('click', function(){",
              "    var cboxes = $('[id$=-B]');",
              "    var checked = $('#rowB').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowC').on('click', function(){",
              "    var cboxes = $('[id$=-C]');",
              "    var checked = $('#rowC').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowD').on('click', function(){",
              "    var cboxes = $('[id$=-D]');",
              "    var checked = $('#rowD').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowE').on('click', function(){",
              "    var cboxes = $('[id$=-E]');",
              "    var checked = $('#rowE').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowF').on('click', function(){",
              "    var cboxes = $('[id$=-F]');",
              "    var checked = $('#rowF').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "}"
            ),
            preDrawCallback = 
              JS('function() { Shiny.unbindAll(this.api().table().node()); }'),
            drawCallback = 
              JS('function() { Shiny.bindAll(this.api().table().node()); } ')
          )
        )
      })
    }
    
    shinyApp(ui, server)
    

    Edit: "Select All" checkbox

    library(shiny)
    library(DT)
    library(htmltools)
    
    rowName <- function(L) {
      as.character(
        checkboxInput(paste0("row", L), label = L, width = "150px")
      )
    }
    rowNames <- vapply(LETTERS[1:6], rowName, character(1))
    
    
    sketch <- htmltools::withTags(
      table(
        class = "display",
        thead(
          tr(
            th(HTML(as.character(checkboxInput("allboxes", label = "ALL", width = "150px")))), 
            th(HTML(as.character(checkboxInput("col1", label = "1", width = "150px")))),
            th(HTML(as.character(checkboxInput("col2", label = "2", width = "150px")))),
            th(HTML(as.character(checkboxInput("col3", label = "3", width = "150px"))))
          )
        )
      )
    )
    
    
    
    ui <- fluidPage(
      br(),
      # Sidebar panel
      sidebarPanel(),
      
      # Main panel with the table
      mainPanel(
        DTOutput("myTable")
      )
    )
    
    server <- function(input, output){
      dat <- data.frame(
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col1-", i), label = NULL, width = "150px")
          )
        }, character(1)),
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col2-", i), label = NULL, width = "150px")
          )
        }, character(1)),
        vapply(LETTERS[1:6], function(i){
          as.character(
            checkboxInput(paste0("col3-", i), label = NULL, width = "150px")
          )
        }, character(1))
      )
      
      rownames(dat) <- rowNames
      
      output$myTable <- renderDT({
        datatable(
          dat, container = sketch,
          escape = FALSE,
          select = "none",
          options = list(
            columnDefs = list(
              list(targets = c(1, 2, 3), orderable = FALSE, className = "dt-center")
            ),
            initComplete = JS(
              "function() {",
              "  $('#allboxes').on('click', function(){",
              "    var cboxes = $('input[type=checkbox]');",
              "    var checked = $('#allboxes').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#col1').on('click', function(){",
              "    var cboxes = $('[id^=col1-]');",
              "    var checked = $('#col1').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#col2').on('click', function(){",
              "    var cboxes = $('[id^=col2-]');",
              "    var checked = $('#col2').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#col3').on('click', function(){",
              "    var cboxes = $('[id^=col3-]');",
              "    var checked = $('#col3').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowA').on('click', function(){",
              "    var cboxes = $('[id$=-A]');",
              "    var checked = $('#rowA').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowB').on('click', function(){",
              "    var cboxes = $('[id$=-B]');",
              "    var checked = $('#rowB').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowC').on('click', function(){",
              "    var cboxes = $('[id$=-C]');",
              "    var checked = $('#rowC').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowD').on('click', function(){",
              "    var cboxes = $('[id$=-D]');",
              "    var checked = $('#rowD').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowE').on('click', function(){",
              "    var cboxes = $('[id$=-E]');",
              "    var checked = $('#rowE').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "  $('#rowF').on('click', function(){",
              "    var cboxes = $('[id$=-F]');",
              "    var checked = $('#rowF').is(':checked');",
              "    cboxes.each(function(i, cbox) {",
              "      $(cbox).prop('checked', checked);",
              "    });",
              "  });",
              "}"
            ),
            preDrawCallback = 
              JS('function() { Shiny.unbindAll(this.api().table().node()); }'),
            drawCallback = 
              JS('function() { Shiny.bindAll(this.api().table().node()); } ')
          )
        )
      })
    }
    
    shinyApp(ui, server)
    

    reply
    0
  • Cancelreply