R Shiny 应用程序: renderUI does not work with nested modules R6 classes

标签 r shiny module r6

我有一个带有模块的大型 Shiny 应用程序,我正在 R6 类中转换其中一些。除了嵌套模块之外,所有功能都运行良好:我面临命名空间问题,并且 uiOutput 不起作用。

这是一个可重现的示例。 ClassTest 类是类层次结构中的最后一个类。它可以由应用程序直接调用,也可以通过 Temp 类调用。在后一种情况(嵌套类)中,uiOutput 中包含的元素不起作用。

library(shiny); library(R6)

ClassTest = R6Class(
  "ClassTest",
  public = list(
    # attributes
    id = NULL,

# initialize
initialize = function(id){
  self$id = id
},
# UI
ui = function(){
  ns = NS(self$id)
  tagList(
    h4('Test class'),
    uiOutput(ns('showText'))
  )
},

# server
server = function(id){
  moduleServer(id,
               function(input, output, session){
  ns = session$ns

  output$showText <- renderUI({ 
    print('In showText <- renderUI')
    tagList(
      p('I am the showText renderUI of the Test class')
      )
    })
}
)
  },

call = function(input, ouput, session){
  self$server(self$id)
}
     )
    )

#----------------------------------------------------------


Temp = R6Class(
  "Temp",
  public = list(
    # attributes
    temp = NULL,
    id = NULL,

# initialize
initialize = function(id){
  self$id = id
  self$temp <- ClassTest$new('testTiers')
},
# UI
ui = function(){
  ns = NS(self$id)
  tagList(
    uiOutput(ns('showTestClass'))
  )
},

# server
server = function(id){
  moduleServer(id,
               function(input, output, session){
                 ns = session$ns 
   output$showTestClass <- renderUI({ 
    self$temp$ui()
    })
})
  },

call = function(input, ouput, session){
  self$server(self$id)
}
  )
)

#----------------------------------------------------------


App = R6Class(
  "App",
  public = list(
    # attributes
    temp = NULL,
    classT = NULL,

# initialize
initialize = function(){
  self$temp = Temp$new('temp')
  self$classT = ClassTest$new('directTest')
  
},
# UI
ui = function(){
  tagList(
    h3('Call by another class'),
    self$temp$ui(),
    hr(),
    h3('Direct call'),
    self$classT$ui()
  )
},

# server
server = function(input, output, session){
  self$temp$call()
  self$classT$call()
    }
  )
)

app = App$new()

shiny::shinyApp(app$ui(), app$server)

最佳答案

你的代码有点复杂,但我确信你错过了在母类中调用子类的server,并且子模块也需要尊重母亲的命名空间。这是一个工作示例,它执行了预期的操作:

library(shiny)
library(R6)

MyModule <- R6Class(
   "MyModule",
   public = list(id = NULL,
                 initialize = function(id) {
                    self$id <- id
                 }, 
                 ui = function() {
                    ns <- NS(self$id)
                    tagList(h4(self$id), 
                            actionButton(ns("do"), "Calc!"), 
                            verbatimTextOutput(ns("print")))
                 },
                 
                 server = function() {
                    moduleServer(self$id, function(input, output, session) {
                       output$print <- renderPrint({
                          input$do
                          sample(100, 1)
                       })
                    })
                 }
   )
)

MyMotherModule <- R6Class(
   "MyMotherModule",
   public = list(id = NULL,
                 child = NULL,
                 initialize = function(id) {
                    self$id <- id
                    self$child <- MyModule$new(NS(id)("child"))
                 },
                 ui = function() {
                    self$child$ui()
                 },
                 server = function() {
                    self$child$server()
                 }
   )
)

App <- R6Class(
   "App",
   public = list(child1 = NULL,
                 child2 = NULL,
                 mother = NULL,
                 initialize = function() {
                    self$child1 <- MyModule$new("child1")
                    self$child2 <- MyModule$new("child2")
                    self$mother <- MyMotherModule$new("mother1")
                 },
                 ui = function() {
                    fluidPage(
                       fluidRow(
                          self$child1$ui(),
                          self$child2$ui(),
                          self$mother$ui()
                       )
                    )
                 },
                 server = function() {
                    function(input, output, session) {
                       self$child1$server()
                       self$child2$server()
                       self$mother$server()
                    }
                 }
   )
)

app <- App$new()

shinyApp(app$ui(), app$server())

一些备注

  1. 母模块中子模块的 id 也必须命名为 MyModule$new(NS(id)("child")) 以符合这个想法名称只能在模块内唯一,而不是整体唯一。如果我在示例中没有命名空间 child ,那么顶层的 child 元素就会造成困惑。 (从技术上讲,它可以在没有命名空间的情况下工作,但是如果您碰巧使用带有子元素名称的顶级元素,那么您的用户将得到奇怪的结果)。
  2. 仅当调用相应的服务器函数时,模块才会工作(即服务器逻辑启动)。因此,您需要在母模块中添加 self$child$server() 来“打开”子模块的服务器逻辑。

关于R Shiny 应用程序: renderUI does not work with nested modules R6 classes,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64406234/

相关文章:

r - 在数据框中找到最佳行

r - Shiny 的下载处理程序不工作

python - 导入错误: no module named numpy (numpy already installed)

function - 传递多个参数

excel - 按钮在 Excel 中引用了错误的行

r - 在 R 中将两组向量相乘

r - 如何在不改变轴标签对齐的情况下更改 ggplot 标题的垂直位置

r - 在 R 中存储所有可能的排列

java - RStudio 和 Shiny : messages between server and client

r - Shiny 中的条件主面板