在Shiny中,DT数据表格中一列的总和

7
我正在尝试使用DT::datatable来计算我的shiny应用程序中一列的总和。在这里,我指的是表格中所有元素的总和,而不仅仅是当前分页中显示的内容。根据这个示例,以下代码应该可以工作(但实际上并没有):
jsCode <- "function(row, data, start, end, display) {

                  var api = this.api(), data;

                  total = api.column(1, {page: 'all'}).data().reduce( function(a, b) { return a + b}, 0); 

                  $( api.column(1).footer() ).html('Total: ' + total);
                  }"

我所理解的是,这个代码计算当前分页中元素的总和。以下是完整代码:
library(shiny)
library(DT)

set.seed(2282018)
company <- data.frame(Company = letters[1:30], Units = round(runif(30, 
                                                             1000, 10e6), 0), 
                      Price = scales::dollar(runif(30, 200, 1230)), stringsAsFactors = F)

jsCode <- "function(row, data, start, end, display) {

                  var api = this.api(), data;

                  total = api.column(1, {page: 'all'}).data().reduce( function(a, b) { return a + b}, 0); 

                  $( api.column(1).footer() ).html('Total: ' + total);
                  }"

# UI ---- 
ui <- function(){

  fluidPage(

    sidebarLayout(

      sidebarPanel(numericInput("nums", label = "Num Input", value = 1, min = 1, max = 10)),

      mainPanel(dataTableOutput("mytable"))

    )

  )

}

# server ----
server <- function(input, output, session){

  cont <- htmltools::withTags(table(
    tableHeader(names(company)),tableFooter(names(company))
  ))

  output$mytable <- DT::renderDataTable(  {

    DT::datatable(company,
                  container = cont,
                  caption = tags$caption("Example"), 
                  filter = "none", 
                  rownames = F,
                  options = list(autoWidth = T, 
                                 pageLength = 10, 
                                 scrollCollapse = T,
                                 dom = 'lftp', 
                                 footerCallback = JS(jsCode))
                  )
  }
  )
}

runApp(list(ui = ui, server = server))

Thank you


@PorkChop 不是的。那个帖子帮助我到了这一步,但它只给了我小计(或者通过分页的总计)。我需要该列所有元素的总计。顺便说一下,感谢您回答另一个帖子... - JdeMello
我标记为重复,因为这些问题是相同的,你可能应该顶一下另一个。 - Pork Chop
好的,那么我该如何在那里得到我的问题的答案呢?我同意问题的性质相似,但第一个问题的解决方案在这种特定情况下不起作用。 - JdeMello
1
这在Shiny之外可以工作,或者在Shiny内部但需要设置选项server = FALSE。如此处所述,“当在服务器端处理模式下使用DataTables时[...]选择器只能选择当前页面上的行。”。 - Stéphane Laurent
谢谢你的参与,现在更有意义了。 - JdeMello
1个回答

8
也许你可以写一个解决方法,比如以下方式:
library(shiny)
library(DT)

set.seed(2282018)
company <- data.frame(Company = letters[1:30], Units = round(runif(30,  1000, 10e6), 0), Price = scales::dollar(runif(30, 200, 1230)), stringsAsFactors = F)
jsCode <- "function(row, data, start, end, display) {var api = this.api(), data;$( api.column(1).footer() ).html('Total: ' + MYTOTAL);}"

# Workaround
getTotal <- function(data,index){

  if(index < 1 || index > ncol(data)){
    return("")
  }
  col <- data[,index]
  col <- gsub("[$]","",col)
  col <- gsub("[£]","",col)
  col <- gsub("[,]","",col)
  col <- suppressWarnings(as.numeric(col))
  if(all(is.na(col))){
    return("")
  }
  sum(col)
}


ui <- function(){
  fluidPage(
    sidebarLayout(
      sidebarPanel(numericInput("nums", label = "Num Input", value = 1, min = 1, max = 10)),
      mainPanel(dataTableOutput("mytable"))
    )
  )
}

server <- function(input, output, session){

  Total <- reactive({
    getTotal(company,2)
  })

  cont <- htmltools::withTags(table(
    tableHeader(names(company)),tableFooter(names(company))
  ))

  output$mytable <- DT::renderDataTable(  {
    jsCode <- sub("MYTOTAL",Total(),jsCode)
    DT::datatable(company,
                  container = cont,
                  caption = tags$caption("Example"), 
                  filter = "none", 
                  rownames = F,
                  options = list(autoWidth = T, 
                                 pageLength = 10, 
                                 scrollCollapse = T,
                                 dom = 'lftp', 
                                 footerCallback = JS(jsCode))
    )
  }
  )
}

runApp(list(ui = ui, server = server))

enter image description here


谢谢。你知道为什么.column(x, {page: 'all'})不起作用吗? - JdeMello
我尝试了逐行比对页脚回调提供的代码,但它并没有起作用。https://datatables.net/examples/advanced_init/footer_callback.html 这非常奇怪。我怀疑缺少某些依赖项。我进行了“hack”操作,例如显示整个表格而不分页并计算它,然后仅显示500px,但这不是一个好的解决方案。 - Pork Chop
顺便说一句,我试图使用.column(x).data()但没有成功。 - Pork Chop
2
PorkChop,请看我在原帖下面的评论。那里解释了为什么 colum(x, {page: 'all'}) 无法工作。这是由于服务器模式处理造成的。 - Stéphane Laurent

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接