【问题标题】:R shinyapp: renderUI does not work with nested modules R6 classesR shinyapp:renderUI 不适用于嵌套模块 R6 类
【发布时间】:2020-10-17 19:00:48
【问题描述】:

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

这是一个可复制的示例。 ClassTest 类是类层次结构的最后一个类。它可以由 App 直接调用,也可以通过 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)

【问题讨论】:

    标签: r shiny module r6


    【解决方案1】:

    你的代码有点复杂,但我敢肯定你错过了在母类中调用子类的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() 以“开启”子模块的服务器逻辑。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2017-11-15
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多