R ggiraph 动态设置没有 Shiny 的工具提示文本
R ggiraph dynamically setting tooltip text without Shiny
What:在页面加载时将 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 页面中的迷你 html 页面,即“html 小部件”。
我试过将我的临时工具提示包装在
标签中,但由于它们被捆绑在 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>
可能有几种不同的方法可以实现这个目标,特别是我怀疑使用 htmlwidgets 的“onRender”函数可以更干净地实现这个目标,但我从来没有让它起作用。
我发现对 htmlwidget 或 htmlwidget 数据的 innerHTML 进行任何编辑都会完全破坏鼠标悬停文本,因为它会破坏为鼠标悬停文本提供支持的事件侦听器。
相反,我发现我可以通过将结束脚本包装在 $(window).load(function(){ ... }) 调用中来编辑 htmlwidget 的结果 svg。如果我找到所有 svg 元素(在本例中为圆圈!)并编辑这些 svg 属性 的标题 objects 我可以保留事件侦听器并更改标题内容(成为图像!)。
请参阅下面的完整示例,其中有 1000 张 1kb 的图像作为工具提示,但文件大小没有增加:
---
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)
library(htmlwidgets)
\```
\```{r,echo=FALSE}
lookup = list()
lookup$FullID1 = '<img src=\"\" />'
jsonData = toJSON(lookup)
\```
<script type="application/json" id="lookuptable">
`r jsonData`
</script>
\```{r,echo=FALSE}
g = ggplot()+
geom_point_interactive(aes(x=1:1000,y=1:1000,tooltip='FullID1'))
girafe(code=print(g))
\```
<script type="application/javascript">
$(window).load(function(){
var lookuptable = JSON.parse(document.getElementById('lookuptable').innerHTML);
var keys = Object.keys(lookuptable);
var circleTags = document.getElementsByTagName("circle");
for (s=0; s < circleTags.length; s++){
var item = circleTags[s];
var itemTitle = item.attributes.title.nodeValue;
console.log(itemTitle);
if (keys.includes(itemTitle)){
item.attributes.title.nodeValue=lookuptable[itemTitle];
}
console.log(item.attributes.title.nodeValue);
}
});
</script>
What:在页面加载时将 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 页面中的迷你 html 页面,即“html 小部件”。
我试过将我的临时工具提示包装在
我尝试过天真地替换所有脚本标签中的 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>
可能有几种不同的方法可以实现这个目标,特别是我怀疑使用 htmlwidgets 的“onRender”函数可以更干净地实现这个目标,但我从来没有让它起作用。
我发现对 htmlwidget 或 htmlwidget 数据的 innerHTML 进行任何编辑都会完全破坏鼠标悬停文本,因为它会破坏为鼠标悬停文本提供支持的事件侦听器。
相反,我发现我可以通过将结束脚本包装在 $(window).load(function(){ ... }) 调用中来编辑 htmlwidget 的结果 svg。如果我找到所有 svg 元素(在本例中为圆圈!)并编辑这些 svg 属性 的标题 objects 我可以保留事件侦听器并更改标题内容(成为图像!)。
请参阅下面的完整示例,其中有 1000 张 1kb 的图像作为工具提示,但文件大小没有增加:
---
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)
library(htmlwidgets)
\```
\```{r,echo=FALSE}
lookup = list()
lookup$FullID1 = '<img src=\"\" />'
jsonData = toJSON(lookup)
\```
<script type="application/json" id="lookuptable">
`r jsonData`
</script>
\```{r,echo=FALSE}
g = ggplot()+
geom_point_interactive(aes(x=1:1000,y=1:1000,tooltip='FullID1'))
girafe(code=print(g))
\```
<script type="application/javascript">
$(window).load(function(){
var lookuptable = JSON.parse(document.getElementById('lookuptable').innerHTML);
var keys = Object.keys(lookuptable);
var circleTags = document.getElementsByTagName("circle");
for (s=0; s < circleTags.length; s++){
var item = circleTags[s];
var itemTitle = item.attributes.title.nodeValue;
console.log(itemTitle);
if (keys.includes(itemTitle)){
item.attributes.title.nodeValue=lookuptable[itemTitle];
}
console.log(item.attributes.title.nodeValue);
}
});
</script>