0

内容:在页面加载时将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>
4

1 回答 1

0

可能有几种不同的方法可以实现这个目标,特别是我怀疑使用 htmlwidgets 的“onRender”函数可能能够更干净地实现这个目标,但我从来没有让它工作。

我发现,对 htmlwidget 或 htmlwidget 数据的 innerHTML 的任何编辑都会完全破坏鼠标悬停文本,因为它会破坏为鼠标悬停文本提供动力的事件侦听器。

相反,我发现我可以通过将结束脚本包装在 $(window).load(function(){ ... }) 调用中来编辑 htmlwidget 的结果 svg。如果我找到所有 svg 元素(在本例中为圆圈!)并编辑这些 svg 对象的 title 属性,我可以保留事件侦听器并更改标题内容(成为图像!)。

请参阅下面的完整示例,其中有 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>
于 2021-12-10T16:19:20.700 回答