shiny
アプリのrhandsontable
の内部にカラータイプとしてカラーピッカーを配置したいと思います。 colourpicker
パッケージのcolourInput()
を使用して、カラーピッカーをスタンドアロン入力として追加したり、HTMLタグから作成したり、HTMLテーブルに配置することができます(下記のサンプルコードを参照)。カラーピッカー入力コントロールをrhandsontable
列に追加することはできますか?光沢のある頑丈なセルにコントロール入力とHTMLウィジェットを挿入する
最終目標は、MS Excelなどのスプレッドシートからデータをコピーして、rhandsontable
オブジェクトに貼り付けるアプリケーションです。これには、色名または16進コードを指定するテキストも含まれます。ユーザーは、テキストを上書きするか、カーソル操作でピッカーから色を選択して色を編集できます。アプリケーションは、後でそれらの入力を受け取り、計算を実行し、指定された色で結果をグラフ化する。
以下は、失敗した2回の試行を示すサンプルコードです。アドバイスをいただければ幸いです。また、私はJavaScriptについて何も知らない。 colourpickerとrhandsontableビネットは素晴らしいリソースですが、それでもわかりません。
最小限の例
library(shiny); library(rhandsontable); library(colourpicker)
hotDF <- data.frame(Value = 1:4, Status = TRUE, Name = LETTERS[1:4],
Date = seq(from = Sys.Date(), by = "days", length.out = 4),
Colour = sapply(1:4, function(i) {
paste0(
'<div class="form-group shiny-input-container"
data-shiny-input-type="colour">
<input id="myColour',i,'" type="text"
class="form-control shiny-colour-input" data-init-value="#FFFFFF"
data-show-colour="both" data-palette="square"/>
</div>'
)}), stringsAsFactors = FALSE)
testColourInput <- function(DF){
ui <- shinyUI(fluidPage(rHandsontableOutput("hot")))
server <- shinyServer(function(input, output) {
DF2 <- transform(DF, Colour = c(sapply(1:4, function(x) {
jsonlite::toJSON(list(value = "black"))
}))) #create DF2 for attempt #2
output$hot <- renderRHandsontable({
#Attempt #1 = use the HTML renderer
#Results in no handsontable AND no HTML table <-- why no HTML table too?
rhandsontable(DF) %>% hot_col(col = "Colour", renderer = "html")
#Attempt #2 = use colourWidget
#Results are the same as above.
#rhandsontable(DF2) %>%
# hot_col(col = "Colour", renderer = htmlwidgets::JS("colourWidget"))
})
}) #close shinyServer
runApp(list(ui=ui, server=server))
} #close testColorInput function
testColourInput(DF = hotDF)
のscreengrabと拡張例:
library(shiny); library(rhandsontable); library(colourpicker)
#Colour cells ideally would be a colourInput() control similar to the Date input control
hotDF <- data.frame(Value = 1:4, Status = TRUE, Name = LETTERS[1:4],
Date = seq(from = Sys.Date(), by = "days", length.out = 4),
Colour = sapply(1:4, function(i) {
paste0(
'<div class="form-group shiny-input-container"
data-shiny-input-type="colour">
<input id="myColour',i,'" type="text"
class="form-control shiny-colour-input"
data-init-value="#FFFFFF"
data-show-colour="both" data-palette="square"/>
</div>'
)}),
stringsAsFactors = FALSE)
testColourInput <- function(DF){
ui <- shinyUI(fluidPage(
sidebarLayout(
sidebarPanel(
#Standalone colour Input
colourInput("myColour", label = "Just the color control:", value = "#000000"),
br(),
HTML("Build the colour Input from HTML tags:"), br(),
HTML(paste0(
"<div class='form-group shiny-input-container'
data-shiny-input-type='colour'>
<input id='myColour", 999,"' type='text'
class='form-control shiny-colour-input'
data-init-value='#FFFFFF' data-show-colour='both'
data-palette='square'/>
</div>"
))
),
mainPanel(
HTML("Failed attempt"),
rHandsontableOutput("hot"),
br(), br(),
HTML("Success, but this is not a rhandsontable"),
uiOutput("tableWithColourInput")
)
)
))
server <- shinyServer(function(input, output) {
#create DF2 for attempt #2
DF2 <- transform(DF, Colour = c(sapply(1:4, function(x) {
jsonlite::toJSON(list(value = "black"))
})))
output$hot <- renderRHandsontable({
#Attempt #1 = use the HTML renderer
#Results in no handsontable AND no HTML table <-- why no HTML table too?
rhandsontable(DF) %>% hot_col(col = "Colour", renderer = "html")
#Attempt #2 = use colourWidget
#Results are the same as above.
#rhandsontable(DF2) %>%
# hot_col(col = "Colour", renderer = htmlwidgets::JS("colourWidget"))
#Uncomment below to see the table without html formatting
#rhandsontable(DF)
#^This line was uncommented to obtain the screengrab
})
#HTML table
myHTMLtable <- data.frame(Variable = LETTERS[1:4],
Select = NA)
output$tableWithColourInput <- renderUI({
#create table cells
rowz <- list()
#Fill out table cells [i,j] with static elements
for(i in 1:nrow(myHTMLtable)) {
rowz[[i]] <- tags$tr(lapply(myHTMLtable[i,1:ncol(myHTMLtable)],
function(x) { tags$td(HTML(as.character(x))) }
))
}
#Add colourInput() to cells in the "Select" column in myHTMLtable
for(i in 1:nrow(myHTMLtable)) {
#Note: in the list rowz:
# i = row; [3] = row information; children[1] = table cells (list of 1);
# $Select = Column 'Select'
rowz[[i]][3]$children[[1]]$Select <- tags$td(
colourInput(inputId = as.character(paste0("inputColour", i)),
label = NULL, value = "#000000")
)
}
mybody <- tags$tbody(rowz)
tags$table(
tags$style(HTML(
".shiny-html-output th,td {border: 1px solid black;}"
)),
tags$thead(
tags$tr(lapply(c("Variable!", "Colour!"), function(x) tags$th(x)))
),
mybody
) #close tags$table
}) #close renderUI
}) #close shinyServer
runApp(list(ui=ui, server=server))
} #close testColorInput function
testColourInput(DF = hotDF)
セル内のHTMLをエスケープできるはずです。投稿した例はそのまま動作しないので、問題を再現するのは難しいです。あなたの例を編集してそのまま実行することをお勧めします。 – Carl
質問をご覧いただきありがとうございます。私はそれで運がなかったか、私が試したことは何もなかった。再現性: 'output $ hot'を定義する文で' rhandsontable(DF) 'のコメントを外しましたか?最小限の例でも同じことです: 'rhandsontable(DF)'を返すことで、それ自身がテーブルを出力しますが、Attempt 1の追加の 'renderer'引数は無駄になります。 – oshun