javascript - bookdown 中的代码折叠

标签 javascript html r code-folding bookdown

RMarkdown 中用于 html 文档的代码折叠选项很棒。该选项使编程方法对那些感兴趣的人来说是透明的,而不会强制观众滚动浏览数英里的代码。代码与散文和交互式图形输出的紧密放置使整个项目更容易为更广泛的受众所接受,此外它还减少了对额外文档的需求。

对于较大的项目,我使用的是 bookdown,效果很好。唯一的问题是没有代码折叠选项。 bookdown 当前未启用代码折叠。 (参见 Enable code folding in bookdown)

我知道我不需要选择来实现它。我只需要将正确的代码粘贴到正确的地方。但是什么代码和在哪里?

一个可行的替代方案是将代码块放在页面中 block 输出的下方。或者,最后,将它们作为附录。我可以用 html 做到这一点,但不能像 rbookdown 那样重现。

最佳答案

整个页面的全局隐藏/显示按钮

要将@Yihui 的提示用于折叠 html 输出中所有代码的按钮,您需要将以下代码粘贴到外部文件中(我在这里将其命名为 header.html):

编辑:我修改了函数 toggle_R这样按钮就会显示Hide GlobalShow Global单击它时。

<script type="text/javascript">

// toggle visibility of R source blocks in R Markdown output
function toggle_R() {
  var x = document.getElementsByClassName('r');
  if (x.length == 0) return;
  function toggle_vis(o) {
    var d = o.style.display;
    o.style.display = (d == 'block' || d == '') ? 'none':'block';
  }

  for (i = 0; i < x.length; i++) {
    var y = x[i];
    if (y.tagName.toLowerCase() === 'pre') toggle_vis(y);
  }

    var elem = document.getElementById("myButton1");
    if (elem.value === "Hide Global") elem.value = "Show Global";
    else elem.value = "Hide Global";
}

document.write('<input onclick="toggle_R();" type="button" value="Hide Global" id="myButton1" style="position: absolute; top: 10%; right: 2%; z-index: 200"></input>')

</script>

在此脚本中,您可以直接使用style 修改与按钮关联的位置和CSS 代码。选项或将其添加到您的 css 文件中。我必须设置 z-index以确保它出现在其他部门之上的高值(value)。
请注意,此 javascript 代码仅折叠使用 echo=TRUE 调用的 R 代码,归因于 class="r"在 html 中。这是由命令 var x = document.getElementsByClassName('r'); 定义的

然后,您在 rmarkdown 脚本的 YAML header 中调用此文件,如下例所示:

---
title: "Toggle R code"
author: "StatnMap"
date: '`r format(Sys.time(), "%d %B, %Y")`'
output:
  bookdown::html_document2:
    includes:
      in_header: header.html
  bookdown::gitbook:
    includes:
      in_header: header.html
---

Stackoverflow question
<https://stackoverflow.com/questions/45360998/code-folding-in-bookdown>

```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
```

## R Markdown

This is an R Markdown document. Markdown is a simple formatting syntax for authoring HTML, PDF, and MS Word documents. For more details on using R Markdown see <http://rmarkdown.rstudio.com>.

When you click the **Knit** button a document will be generated that includes both content as well as the output of any embedded R code chunks within the document. You can embed an R code chunk like this:

```{r cars}
summary(cars)
```

新编辑:每个 block 的本地隐藏/显示按钮

我终于找到了解决方案!
在查看正常 html 输出(无 bookdown)的代码折叠行为时,我能够将其添加到 bookdown。主要的javascript函数需要找到.sourceCode与 bookdown 一起工作的类(class)划分。然而,这也需要 bootstrap 的补充 javascript 函数。 , 但不是所有的。这适用于 gitbookhtml_document2 .
以下是步骤:

  1. 创建 js与 Rmd 文件位于同一目录中的文件夹
  2. 下载 javascript 函数 transition.jscollapse.js例如:https://github.com/twbs/bootstrap/tree/v3.3.7/js并将它们存储在您的 js 中文件夹
  3. js 中创建一个新文件名为 codefolding.js 的文件夹使用以下代码。这与 rmarkdown code_folding 选项相同,但带有 pre.sourceCode。添加以查找 R 代码块:

codefolding.js代码:

window.initializeCodeFolding = function(show) {

  // handlers for show-all and hide all
  $("#rmd-show-all-code").click(function() {
    $('div.r-code-collapse').each(function() {
      $(this).collapse('show');
    });
  });
  $("#rmd-hide-all-code").click(function() {
    $('div.r-code-collapse').each(function() {
      $(this).collapse('hide');
    });
  });

  // index for unique code element ids
  var currentIndex = 1;

  // select all R code blocks
  var rCodeBlocks = $('pre.sourceCode, pre.r, pre.python, pre.bash, pre.sql, pre.cpp, pre.stan');
  rCodeBlocks.each(function() {

    // create a collapsable div to wrap the code in
    var div = $('<div class="collapse r-code-collapse"></div>');
    if (show)
      div.addClass('in');
    var id = 'rcode-643E0F36' + currentIndex++;
    div.attr('id', id);
    $(this).before(div);
    $(this).detach().appendTo(div);

    // add a show code button right above
    var showCodeText = $('<span>' + (show ? 'Hide' : 'Code') + '</span>');
    var showCodeButton = $('<button type="button" class="btn btn-default btn-xs code-folding-btn pull-right"></button>');
    showCodeButton.append(showCodeText);
    showCodeButton
        .attr('data-toggle', 'collapse')
        .attr('data-target', '#' + id)
        .attr('aria-expanded', show)
        .attr('aria-controls', id);

    var buttonRow = $('<div class="row"></div>');
    var buttonCol = $('<div class="col-md-12"></div>');

    buttonCol.append(showCodeButton);
    buttonRow.append(buttonCol);

    div.before(buttonRow);

    // update state of button on show/hide
    div.on('hidden.bs.collapse', function () {
      showCodeText.text('Code');
    });
    div.on('show.bs.collapse', function () {
      showCodeText.text('Hide');
    });
  });

}
  1. 在下面的 rmarkdown 脚本中,所有三个函数都被读取并按原样包含在标题中,因此 js文件夹对最终文档本身没有用。看js函数的时候,我也加了show的选项默认代码块,但您可以选择使用 hide 隐藏它们.

Markdown 代码:

---
title: "Toggle R code"
author: "StatnMap"
date: '`r format(Sys.time(), "%d %B, %Y")`'
output:
  bookdown::html_document2:
    includes:
      in_header: header.html
  bookdown::gitbook:
    includes:
      in_header: header.html
---

Stackoverflow question
<https://stackoverflow.com/questions/45360998/code-folding-in-bookdown>


```{r setup, include=FALSE}
# Add a common class name for every chunks
knitr::opts_chunk$set(
  echo = TRUE)
```
```{r htmlTemp3, echo=FALSE, eval=TRUE}
codejs <- readr::read_lines("js/codefolding.js")
collapsejs <- readr::read_lines("js/collapse.js")
transitionjs <- readr::read_lines("js/transition.js")

htmlhead <- 
  paste('
<script>',
paste(transitionjs, collapse = "\n"),
'</script>
<script>',
paste(collapsejs, collapse = "\n"),
'</script>
<script>',
paste(codejs, collapse = "\n"),
'</script>
<style type="text/css">
.code-folding-btn { margin-bottom: 4px; }
.row { display: flex; }
.collapse { display: none; }
.in { display:block }
</style>
<script>
$(document).ready(function () {
  window.initializeCodeFolding("show" === "show");
});
</script>
', sep = "\n")

readr::write_lines(htmlhead, path = "header.html")
```

## R Markdown

This is an R Markdown document. Markdown is a simple formatting syntax for authoring HTML, PDF, and MS Word documents. For more details on using R Markdown see <http://rmarkdown.rstudio.com>.

When you click the **Knit** button a document will be generated that includes both content as well as the output of any embedded R code chunks within the document. You can embed an R code chunk like this:

```{r cars}
summary(cars)
```

```{r plot}
plot(cars)
```

此脚本显示 Rstudio 浏览器中的按钮,但效果不佳。但是,这对于 firefox 是可以的。
你会看到有一点 css在此代码中,但当然您可以使用更多 CSS 修改这些按钮的位置和颜色以及您想要的任何内容。

编辑:合并全局和本地按钮

编辑 2017-11-13:全局代码折叠按钮与单个 block 按钮很好地集成。 函数 toggle_R终于没有必要了,但是你需要得到函数dropdown.js在 Bootstrap 中。

调用js时直接在代码块中调用全局按钮文件:

```{r htmlTemp3, echo=FALSE, eval=TRUE}
codejs <- readr::read_lines("/mnt/Data/autoentrepreneur/js/codefolding.js")
collapsejs <- readr::read_lines("/mnt/Data/autoentrepreneur/js/collapse.js")
transitionjs <- readr::read_lines("/mnt/Data/autoentrepreneur/js/transition.js")
dropdownjs <- readr::read_lines("/mnt/Data/autoentrepreneur/js/dropdown.js")

htmlhead <- c(
  paste('
<script>',
paste(transitionjs, collapse = "\n"),
'</script>
<script>',
paste(collapsejs, collapse = "\n"),
'</script>
<script>',
paste(codejs, collapse = "\n"),
'</script>
<script>',
paste(dropdownjs, collapse = "\n"),
'</script>
<style type="text/css">
.code-folding-btn { margin-bottom: 4px; }
.row { display: flex; }
.collapse { display: none; }
.in { display:block }
.pull-right > .dropdown-menu {
    right: 0;
    left: auto;
}
.open > .dropdown-menu {
    display: block;
}
.dropdown-menu {
    position: absolute;
    top: 100%;
    left: 0;
    z-index: 1000;
    display: none;
    float: left;
    min-width: 160px;
    padding: 5px 0;
    margin: 2px 0 0;
    font-size: 14px;
    text-align: left;
    list-style: none;
    background-color: #fff;
    -webkit-background-clip: padding-box;
    background-clip: padding-box;
    border: 1px solid #ccc;
    border: 1px solid rgba(0,0,0,.15);
    border-radius: 4px;
    -webkit-box-shadow: 0 6px 12px rgba(0,0,0,.175);
    box-shadow: 0 6px 12px rgba(0,0,0,.175);
}
</style>
<script>
$(document).ready(function () {
  window.initializeCodeFolding("show" === "show");
});
</script>
', sep = "\n"),
  paste0('
<script>
document.write(\'<div class="btn-group pull-right" style="position: absolute; top: 20%; right: 2%; z-index: 200"><button type="button" class="btn btn-default btn-xs dropdown-toggle" data-toggle="dropdown" aria-haspopup="true" aria-expanded="true" data-_extension-text-contrast=""><span>Code</span> <span class="caret"></span></button><ul class="dropdown-menu" style="min-width: 50px;"><li><a id="rmd-show-all-code" href="#">Show All Code</a></li><li><a id="rmd-hide-all-code" href="#">Hide All Code</a></li></ul></div>\')
</script>
')
)

readr::write_lines(htmlhead, path = "/mnt/Data/autoentrepreneur/header.html")
```

新的全局按钮显示了一个下拉菜单,可以在“显示所有代码”或“隐藏所有代码”之间进行选择。使用 window.initializeCodeFolding("show" === "show")默认显示所有代码,而使用 window.initializeCodeFolding("show" === "hide") , 所有代码默认隐藏。

关于javascript - bookdown 中的代码折叠,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45360998/

相关文章:

javascript - 使用脚本定期执行mysql查询

javascript img src路径+变量串联

r - 字符串匹配记录以计算数据框中的所有实例

r - 将时间戳(以微秒为单位)转换为 r 中的数据和时间

r - 使用 pandoc 和 RStudio Markdown 转换为 MS Word 的语法是什么?它如何用于包含引用书目和内联引用?

javascript - WebdriverJS wait() 是否返回一个 web 元素?

javascript - 为什么这会在 Node.js 中使用 --harmony 抛出错误?

javascript - 如何禁用 DocuSign 基于浏览器的地理位置弹出对话框

css - 意外填充阻止响应式网页中的全屏

html - 是否可以在每个 li 元素上隐藏 ul 背景图像?