内容:在页面加载时将rmarkdown中的 ggiraph 工具提示内容动态设置为 html。
为什么:使用嵌入式 png 的工具提示可以制作成图形,这对于文本不足的某些生物结构很有价值。这是我目前正在做的一个最小示例:
encodedImage = ""
g = ggplot()+
geom_point_interactive(aes(x=1,y=1,tooltip=sprintf('<img src=\"%s\" />',encodedImage)))
girafe(code=print(g))
这样做的缺点是,对于使用工具提示图形的每个绘图,都会重复编码图像,从而导致文件太大而无法存储许多文件。
如何:为了减轻文件大小随着工具提示使用的增加而增加的问题,我希望将嵌入的图像文本分配到 json 对象中,然后使用 javascript 将所有工具提示动态更新为嵌入图像。
我已经完成的事情:我可以通过简单地包含脚本标签并在这些标签中输出 json 文本来获取存储在 json 中的嵌入图像。为了测试这里的硬编码示例,带有一个简单的替换文本:
<script type="application/json" id="lookuptable">
{"ID1":"ReplaceText"}
</script>
我不能做的是替换工具提示的文本。本质上,我计划将工具提示设置为某种类型的 ID,并使用它来将嵌入的图像与相应的点相匹配。工具提示文本由 ggiraph 存储在 json 中,如下所示:
<script type="application/json" data-for="htmlwidget-b8ceca7828d4dd46f692"> {x:{"html":.....}}</script>
在这个 json 中,所有 html 组件(<、、> 等)都被“转义”了,我相信这个“htmlwidget”数据被传递给了嵌入在 html 页面中的 mini-html 页面,即“htmlwidget”。
我尝试将我的临时工具提示包装在 <div> 标记中,但由于它们被绑定在 JSON 中,因此在 DOM 中看不到它们。
如果任何脚本标签中的 ID 存在,我已经尝试过天真地替换所有实例:
<script type="application/javascript">
console.log(document.getElementById('lookuptable').innerHTML)
var lookuptable = JSON.parse(document.getElementById('lookuptable').innerHTML);
var scriptTags = document.getElementsByTagName("script");
for (s=0; s < scriptTags.length; s++){
var item = scriptTags[s];
for(var k in lookuptable){
console.log(item.innerHTML);
item.innerHTML.replace(k,lookuptable[k])
}
}
</script>
然而,当这个脚本运行时,json 中似乎不再有工具提示文本(尽管它在 html 源代码中)。
这是我目前卡住的地方。如果我取得更多进展,我会更新或回答这个问题。
我也很清楚这对于 Shiny 来说是微不足道的,不幸的是,这不是一个选项,因为 html 页面需要完全独立,因为我正在构建的真正 rmarkdown 是一个独立的报告。
最后,这是一个完整的、可重复的示例。最终的针织项目应导致工具提示为“ReplaceText”(通过删除 \ 来修复 R 代码块):
---
title: "Demo"
author: "Zachary Klamer"
date: "12/9/2021"
output: html_document
---
\```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
library(ggplot2)
library(ggiraph)
library(rjson)
\```
\```{r,echo=FALSE}
lookup = list()
lookup$FullID1 = "ReplaceText"
jsonData = toJSON(lookup)
\```
<script type="application/json" id="lookuptable">
`r jsonData`
</script>
\```{r,echo=FALSE}
g = ggplot()+
geom_point_interactive(aes(x=1,y=1,tooltip='FullID1'))
girafe(code=print(g))
\```
<script type="application/javascript">
console.log(document.getElementById('lookuptable').innerHTML)
var lookuptable = JSON.parse(document.getElementById('lookuptable').innerHTML);
var scriptTags = document.getElementsByTagName("script");
for (s=0; s < scriptTags.length; s++){
var item = scriptTags[s];
for(var k in lookuptable){
console.log(item.innerHTML);
item.innerHTML.replace(k,lookuptable[k])
}
}
</script>