The catchandrelease
extension is a unique experiment designed to explore the identification, storage, and relocation of text within your Quarto documents.
The catchandrelease
extension does not introduce significant enhancements to your document's content. Instead, it serves as an instructive example of how to identify, retrieve, and reposition content within your Quarto project.
To install the catchandrelease
extension, follow these steps:
-
Open your terminal.
-
Execute the following command:
quarto add coatless-quarto/catchandrelease
This command will download and install the extension under the _extensions
subdirectory of your Quarto project. If you are using version control, ensure that you include this directory in your repository.
In Pandoc, various Lua Types each have their own element filter functions that can be applied throughout a document. For example, when creating a code cell using Markdown like this:
```{webr-r}
# R code here
```
Internally, Pandoc treats this as a CodeBlock. To "capture" code, you can provide a custom filter function for this specific Lua type. For instance, you can traverse and process each CodeBlock
element using the following Lua code:
function CodeBlock(elem)
-- Display the text of the CodeBlock
quarto.log.output(elem.text)
-- Return the CodeBlock unchanged
return elem
}
With some modifications, you can extract the text from each code block and store it within a global table:
-- Define a table to store the extracted code and attributes
local codeBlocksTable = {}
function captureCode(elem)
-- Extract code text and store it in a table
local codeBlockData = {
codeValue = elem.code
}
-- Append the table to the codeBlocksTable
table.insert(codeBlocksTable, codeBlockData)
-- Return the CodeBlock unchanged
return elem
}
-- Override Element filter function for CodeBlock
return {
{ CodeBlock = captureCode }
}
The second part of the process involves releasing the captured content back into the document. There are various options for releasing the data, such as marking it up using other Inline
or Block
styles. Alternatively, you can merge and release the data, which is the approach taken here. This requires two functions: data combination and insertion.
For data combination, it is simpler to use JSON encoding of the Lua table storing the CodeBlock details, e.g., quarto.json.encode()
.
For the insertion part, you can provide a custom Pandoc
function like this:
-- Call the function to combine CodeBlocks and write the output to the end of the document
local function releaseCode(doc)
-- Convert the Lua table to JSON
local jsonCode = writeCodeBlocksToJson()
-- Create a new paragraph with the combined code
local para = pandoc.Para(jsonCode)
-- Add it to the end of the document
table.insert(doc.blocks, para)
-- Return the modified document
return doc
}
-- Override Element filter functions
return {
{ CodeBlock = captureCode },
{ Pandoc = releaseCode}
}
For more details, refer to the _extensions/
directory in the Quarto project's repository.
The order in which the filter functions are applied follows the Typewise traversal default sequence:
- Functions for Inline elements
- The Inlines filter function
- Functions for Block elements
- The Blocks filter function
- The Meta filter function
- The Pandoc filter function (last)
To enable further customization of the catching and releasing mechanism, you can define a custom Meta function, e.g., customMeta()
, and specify the order as follows:
return {
{ Meta = customMeta}, -- (1)
{ CodeBlock = captureCode }, -- (2)
{ Pandoc = releaseCode } -- (3)
}
By using these techniques, you can efficiently re-organize content inside of Quarto.