2505 字
13 分钟
富有表现力的代码示例
在这里,我们将探讨如何使用 表达性代码。 提供的示例基于官方文档,您可以参考这些文档了解更多详细信息。
富有表现力的代码
语法突出显示(高亮)
常规语法突出显示
console.log('This code is syntax highlighted!')在 Markdown 中这样写:```jsconsole.log('This code is syntax highlighted!')```
渲染 ANSI 转义序列
ANSI colors:- Regular: Red Green Yellow Blue Magenta Cyan- Bold: Red Green Yellow Blue Magenta Cyan- Dimmed: Red Green Yellow Blue Magenta Cyan
256 colors (showing colors 160-177):160 161 162 163 164 165166 167 168 169 170 171172 173 174 175 176 177
Full RGB colors:ForestGreen - RGB(34, 139, 34)
Text formatting: Bold Dimmed Italic Underline编辑器 & 终端
编辑器样式
- 可以通过加上
title="xxx.xx"做到:
console.log('Title attribute example')在 Markdown 中这样写:```js title="my-test-file.js"console.log('Title attribute example')```
- 而html文件可以直接在代码里注释文件路径,这个注释不会在代码中显示出来,但是会读取并当做编辑器标签
<div>File name comment example</div>在 Markdown 中这样写:```html<!-- src/content/index.html --><div>File name comment example</div>```
终端样式
- 在渲染出来的终端中显示代码:
echo "This terminal frame has no title"在 Markdown 中这样写:```bashecho "This terminal frame has no title"```
- 带上标题同样是使用
title="xxx"
Write-Output "This one has a title!"在 Markdown 中这样写:```powershell title="PowerShell terminal example"Write-Output "This one has a title!"```
覆盖原有框架
- 不使用渲染的终端窗口显示代码,可以用
frame="none":
echo "Look ma, no frame!"在 Markdown 中这样写:```sh frame="none"echo "Look ma, no frame!"```
- 自定义一下可以做到:
# Without overriding, this would be a terminal framefunction Watch-Tail { Get-Content -Tail 20 -Wait $args }New-Alias tail Watch-Tail在 Markdown 中这样写:```ps frame="code" title="PowerShell Profile.ps1"# Without overriding, this would be a terminal framefunction Watch-Tail { Get-Content -Tail 20 -Wait $args }New-Alias tail Watch-Tail```
文本和线条标记
标记整行 & 行范围
- 使用
{}括起需要标记的行号和行范围,之间用,隔开
// Line 1 - targeted by line number// Line 2// Line 3// Line 4 - targeted by line number// Line 5// Line 6// Line 7 - targeted by range "7-8"// Line 8 - targeted by range "7-8"在 Markdown 中这样写:```js {1, 4, 7-8}// Line 1 - targeted by line number// Line 2// Line 3// Line 4 - targeted by line number// Line 5// Line 6// Line 7 - targeted by range "7-8"// Line 8 - targeted by range "7-8"```
- 展示行号不是必须的。关闭展示行号后仍然可以用行号定位:
// Line 1 - targeted by line number// Line 2// Line 3// Line 4 - targeted by line number// Line 5// Line 6// Line 7 - targeted by range "7-8"// Line 8 - targeted by range "7-8"在 Markdown 中这样写:```js showLineNumbers=false {1, 4, 7-8}// Line 1 - targeted by line number// Line 2// Line 3// Line 4 - targeted by line number// Line 5// Line 6// Line 7 - targeted by range "7-8"// Line 8 - targeted by range "7-8"```
选择线标记类型(mark、ins、del)
- 标记的时候可以标记这行代码是新添加进来还是删除的,只需要分别为它们添加标识
ins=和del=。不添加的会保持原样:
function demo() { console.log('this line is marked as deleted') // This line and the next one are marked as inserted console.log('this is the second inserted line')
return 'this line uses the neutral default marker type'}在 Markdown 中这样写:```js title="line-markers.js" del={2} ins={3-4} {6}function demo() { console.log('this line is marked as deleted') // This line and the next one are marked as inserted console.log('this is the second inserted line') return 'this line uses the neutral default marker type'}```
向线标记添加标签
- 在
{}内写入标签名,冒号后面跟上行号(标签名太长会遮住代码):
<button role="button" {...props} value={value} className={buttonClassName} disabled={disabled} active={active}> {children && !active && (typeof children === 'string' ? <span>{children}</span> : children)}</button><button> <span>abc</span> <span>abc</span></button>在 Markdown 中这样写:```jsx {"1":5} del={"2":7-8} ins={"3":10-12} ins={"标签A":15} ins={"标签B":16}// labeled-line-markers.jsx<buttonrole="button"{...props}value={value}className={buttonClassName}disabled={disabled}active={active}>{children &&!active &&(typeof children === 'string' ? <span>{children}</span> : children)}</button><button><span>abc</span><span>abc</span></button>```
关于行号从上面这个例子也可以看出行号是从源码中开始数的。开头的
// labeled-line-markers.jsx也算进行号里面,但在展示时这行忽略并提取成了文件名,所以导致展示的行号和标记的行号不一致。
在单独的行上添加长标签
- 前面提到,标签太长会遮住代码,如果一定要写长标签,就在那一行上方留出一个空行,将两行一起标起来:
<button role="button" {...props}
value={value} className={buttonClassName}
disabled={disabled} active={active}>
{children && !active && (typeof children === 'string' ? <span>{children}</span> : children)}</button>在 Markdown 中这样写:```jsx {"1. Provide the value prop here:":5-6} del={"2. Remove the disabled and active states:":8-10} ins={"3. Add this to render the children inside the button:":12-15}// labeled-line-markers.jsx<buttonrole="button"{...props}value={value}className={buttonClassName}disabled={disabled}active={active}>{children &&!active &&(typeof children === 'string' ? <span>{children}</span> : children)}</button>```
也可以使用类似 diff 的语法
this line will be marked as insertedthis line will be marked as deletedthis is a regular line在 Markdown 中这样写:```diff+this line will be marked as inserted-this line will be marked as deletedthis is a regular line```
- 或者这样:
--- a/README.md+++ b/README.md@@ -1,3 +1,4 @@+this is an actual diff file-all contents will remain unmodified no whitespace will be removed either在 Markdown 中这样写:```diff--- a/README.md+++ b/README.md@@ -1,3 +1,4 @@+this is an actual diff file-all contents will remain unmodifiedno whitespace will be removed either```
将语法突出显示与类似 diff 的语法相结合
function thisIsJavaScript() { // This entire block gets highlighted as JavaScript, // and we can still add diff markers to it! console.log('Old code to be removed') console.log('New and shiny code!')}在 Markdown 中这样写:```diff lang="js"function thisIsJavaScript() {// This entire block gets highlighted as JavaScript,// and we can still add diff markers to it!- console.log('Old code to be removed')+ console.log('New and shiny code!')}```
在行内标记单个文本
- 这我前面一直在用。
function demo() { // Mark any given text inside lines return 'Multiple matches of the given text are supported';}在 Markdown 中这样写:```js "given text"function demo() {// Mark any given text inside linesreturn 'Multiple matches of the given text are supported';}```
注意不要他妈这么写:```js ""function demo() {// Mark any given text inside linesreturn 'Multiple matches of the given text are supported';}```这对空白引号会泄露内存,然后把你的网站搞崩掉。 这在 Expressive Code 的 内联文本标记(inline text markers) 语法中会被解析为:“标记所有空字符串”。
但由于空字符串在每一行都“匹配无数次”(比如行首、行中、行尾的零宽位置),Expressive Code 会尝试为每一个匹配位置生成标记节点,导致:
- AST 节点爆炸式增长;
- 内存占用急剧上升;
- 最终触发 JavaScript heap out of memory。
这本质上是一个 边界情况下的解析陷阱,虽然语法上合法,但语义上无意义且极具破坏性。
所以不要写空引号
""和空正则//(下面会讲)
正则表达式
console.log('The words yes and yep will be marked.')在 Markdown 中这样写:```ts /ye[sp]/console.log('The words yes and yep will be marked.')```
转义正斜杠
echo "Test" > /home/test.txt在 Markdown 中这样写:```sh /\/ho.*\//echo "Test" > /home/test.txt```
选择内联标记类型(mark、ins、del)
function demo() { console.log('These are inserted and deleted marker types'); // The return statement uses the default marker type return true;}在 Markdown 中这样写:```js "return true;" ins="inserted" del="deleted"function demo() {console.log('These are inserted and deleted marker types');// The return statement uses the default marker typereturn true;}```
自动换行
配置每个块的自动换行
// Example with wrapfunction getLongString() { return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}在 Markdown 中这样写:```js wrap// Example with wrapfunction getLongString() {return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}```
// Example with wrap=falsefunction getLongString() { return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}在 Markdown 中这样写:```js wrap=false// Example with wrap=falsefunction getLongString() {return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}```
配置换行的缩进
// Example with preserveIndent (enabled by default)function getLongString() { return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}在 Markdown 中这样写:```js wrap preserveIndent// Example with preserveIndent (enabled by default)function getLongString() {return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}```
// Example with preserveIndent=falsefunction getLongString() { return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}在 Markdown 中这样写:```js wrap preserveIndent=false// Example with preserveIndent=falsefunction getLongString() {return 'This is a very long string that will most probably not fit into the available space unless the container is extremely wide'}```
可折叠部分
- 用
collapse={}来设置哪些行号被折叠:
5 collapsed lines
// All this boilerplate setup code will be collapsedimport { someBoilerplateEngine } from '@example/some-boilerplate'import { evenMoreBoilerplate } from '@example/even-more-boilerplate'
const engine = someBoilerplateEngine(evenMoreBoilerplate())
// This part of the code will be visible by defaultengine.doSomething(1, 2, 3, calcFn)
function calcFn() { // You can have multiple collapsed sections3 collapsed lines
const a = 1 const b = 2 const c = a + b
// This will remain visible console.log(`Calculation result: ${a} + ${b} = ${c}`) return c}
4 collapsed lines
// All this code until the end of the block will be collapsed againengine.closeConnection()engine.freeMemory()engine.shutdown({ reason: 'End of example boilerplate code' })在 Markdown 中这样写:```js collapse={1-5, 12-14, 21-24}// All this boilerplate setup code will be collapsedimport { someBoilerplateEngine } from '@example/some-boilerplate'import { evenMoreBoilerplate } from '@example/even-more-boilerplate'const engine = someBoilerplateEngine(evenMoreBoilerplate())// This part of the code will be visible by defaultengine.doSomething(1, 2, 3, calcFn)function calcFn() {// You can have multiple collapsed sectionsconst a = 1const b = 2const c = a + b// This will remain visibleconsole.log(`Calculation result: ${a} + ${b} = ${c}`)return c}// All this code until the end of the block will be collapsed againengine.closeConnection()engine.freeMemory()engine.shutdown({ reason: 'End of example boilerplate code' })```
行号
显示每个块的行号
- 可以用
showLineNumbers,也可以不用(默认显示行号)
// This code block will show line numbersconsole.log('Greetings from line 2!')console.log('I am on line 3')在 Markdown 中这样写:```js showLineNumbers// This code block will show line numbersconsole.log('Greetings from line 2!')console.log('I am on line 3')```
或者不显示行号
- 需要额外设置
showLineNumbers=false:
// Line numbers are disabled for this blockconsole.log('Hello?')console.log('Sorry, do you know what line I am on?')在 Markdown 中这样写:```js showLineNumbers=false// Line numbers are disabled for this blockconsole.log('Hello?')console.log('Sorry, do you know what line I am on?')```
更改起始行号
- 设置
startLineNumber=5
console.log('Greetings from line 5!')console.log('I am on line 6')在 Markdown 中这样写:```js showLineNumbers startLineNumber=5console.log('Greetings from line 5!')console.log('I am on line 6')```
这篇文章在fuwari示例文档的基础上翻译并添加了许多内容。