gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-04 08:44:19 28 4
gpt4 key购买 nike

我有一个带有模块的大型 Shiny 应用程序,我正在将其中一些模块转换为 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)

最佳答案

你的代码有点复杂,但我敢肯定你错过了调用 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())
一些备注
  • id母模块中的子模块的名称也必须命名为 MyModule$new(NS(id)("child"))遵守名称必须仅在模块内唯一而不是整体的想法。如果我没有命名空间 child在我的示例中,child顶层的元素会搞砸。 (从技术上讲,它可以在没有命名空间的情况下工作,但是如果您碰巧使用带有子元素名称的顶级元素,您的用户会得到奇怪的结果)。
  • 仅当调用相应的服务器函数时,模块才起作用(即服务器逻辑启动)。因此,您需要添加 self$child$server()在您的母亲模块中“开启” child 的服务器逻辑。
  • 关于R Shiny 应用程序: renderUI does not work with nested modules R6 classes,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64406234/

    28 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com