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