Skip to content

Commit 7f0c18e

Browse files
authoredAug 1, 2023
feat: allow html blocks inside code groups (#2719)
1 parent 0f38eb4 commit 7f0c18e

File tree

7 files changed

+70
-26
lines changed

7 files changed

+70
-26
lines changed
 

‎src/client/app/composables/codeGroups.ts

+28-10
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,42 @@
1-
import { inBrowser } from 'vitepress'
1+
import { inBrowser, onContentUpdated } from 'vitepress'
22

33
export function useCodeGroups() {
4+
if (import.meta.env.DEV) {
5+
onContentUpdated(() => {
6+
document.querySelectorAll('.vp-code-group > .blocks').forEach((el) => {
7+
Array.from(el.children).forEach((child) => {
8+
child.classList.remove('active')
9+
})
10+
el.children[0].classList.add('active')
11+
})
12+
})
13+
}
14+
415
if (inBrowser) {
516
window.addEventListener('click', (e) => {
617
const el = e.target as HTMLInputElement
718

819
if (el.matches('.vp-code-group input')) {
920
// input <- .tabs <- .vp-code-group
1021
const group = el.parentElement?.parentElement
11-
const i = Array.from(group?.querySelectorAll('input') || []).indexOf(el)
22+
if (!group) return
23+
24+
const i = Array.from(group.querySelectorAll('input')).indexOf(el)
25+
if (i < 0) return
26+
27+
const blocks = group.querySelector('.blocks')
28+
if (!blocks) return
29+
30+
const current = Array.from(blocks.children).find((child) =>
31+
child.classList.contains('active')
32+
)
33+
if (!current) return
1234

13-
const current = group?.querySelector('div[class*="language-"].active')
14-
const next = group?.querySelectorAll(
15-
'div[class*="language-"]:not(.language-id)'
16-
)?.[i]
35+
const next = blocks.children[i]
36+
if (!next || current === next) return
1737

18-
if (current && next && current !== next) {
19-
current.classList.remove('active')
20-
next.classList.add('active')
21-
}
38+
current.classList.remove('active')
39+
next.classList.add('active')
2240
}
2341
})
2442
}

‎src/client/app/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,7 @@ const VitePressApp = defineComponent({
5959
useCodeGroups()
6060

6161
if (Theme.setup) Theme.setup()
62-
return () => h(Theme.Layout)
62+
return () => h(Theme.Layout!)
6363
}
6464
})
6565

‎src/client/app/theme.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ export interface EnhanceAppContext {
99
}
1010

1111
export interface Theme {
12-
Layout: Component
12+
Layout?: Component
1313
enhanceApp?: (ctx: EnhanceAppContext) => Awaitable<void>
1414
extends?: Theme
1515

‎src/client/theme-default/styles/components/vp-code-group.css

+8-2
Original file line numberDiff line numberDiff line change
@@ -66,13 +66,19 @@
6666
background-color: var(--vp-code-tab-active-bar-color);
6767
}
6868

69-
.vp-code-group div[class*='language-'] {
69+
.vp-code-group div[class*='language-'],
70+
.vp-block {
7071
display: none;
7172
margin-top: 0 !important;
7273
border-top-left-radius: 0 !important;
7374
border-top-right-radius: 0 !important;
7475
}
7576

76-
.vp-code-group div[class*='language-'].active {
77+
.vp-code-group div[class*='language-'].active,
78+
.vp-block.active {
7779
display: block;
7880
}
81+
82+
.vp-block {
83+
padding: 20px 24px;
84+
}

‎src/client/theme-default/styles/components/vp-doc.css

+4-2
Original file line numberDiff line numberDiff line change
@@ -278,7 +278,8 @@
278278
color: var(--vp-c-brand-dark);
279279
}
280280

281-
.vp-doc div[class*='language-'] {
281+
.vp-doc div[class*='language-'],
282+
.vp-block {
282283
position: relative;
283284
margin: 16px -24px;
284285
background-color: var(--vp-code-block-bg);
@@ -287,7 +288,8 @@
287288
}
288289

289290
@media (min-width: 640px) {
290-
.vp-doc div[class*='language-'] {
291+
.vp-doc div[class*='language-'],
292+
.vp-block {
291293
border-radius: 8px;
292294
margin: 16px 0;
293295
}

‎src/node/markdown/plugins/containers.ts

+15-6
Original file line numberDiff line numberDiff line change
@@ -74,13 +74,22 @@ function createCodeGroup(options: Options): ContainerArgs {
7474
);
7575
++i
7676
) {
77-
if (tokens[i].type === 'fence' && tokens[i].tag === 'code') {
78-
const title = extractTitle(tokens[i].info)
79-
const id = nanoid(7)
80-
tabs += `<input type="radio" name="group-${name}" id="tab-${id}" ${checked}><label for="tab-${id}">${title}</label>`
77+
const isHtml = tokens[i].type === 'html_block'
8178

82-
if (checked) {
83-
tokens[i].info += ' active'
79+
if (
80+
(tokens[i].type === 'fence' && tokens[i].tag === 'code') ||
81+
isHtml
82+
) {
83+
const title = extractTitle(
84+
isHtml ? tokens[i].content : tokens[i].info,
85+
isHtml
86+
)
87+
88+
if (title) {
89+
const id = nanoid(7)
90+
tabs += `<input type="radio" name="group-${name}" id="tab-${id}" ${checked}><label for="tab-${id}">${title}</label>`
91+
92+
if (checked && !isHtml) tokens[i].info += ' active'
8493
checked = ''
8594
}
8695
}

‎src/node/markdown/plugins/preWrapper.ts

+13-4
Original file line numberDiff line numberDiff line change
@@ -9,22 +9,31 @@ export function preWrapperPlugin(md: MarkdownIt, options: Options) {
99
md.renderer.rules.fence = (...args) => {
1010
const [tokens, idx] = args
1111
const token = tokens[idx]
12+
1213
// remove title from info
1314
token.info = token.info.replace(/\[.*\]/, '')
1415

16+
const active = / active( |$)/.test(token.info) ? ' active' : ''
17+
token.info = token.info.replace(/ active$/, '').replace(/ active /, ' ')
18+
1519
const lang = extractLang(token.info)
1620
const rawCode = fence(...args)
17-
return `<div class="language-${lang}${getAdaptiveThemeMarker(options)}${
18-
/ active( |$)/.test(token.info) ? ' active' : ''
19-
}"><button title="Copy Code" class="copy"></button><span class="lang">${lang}</span>${rawCode}</div>`
21+
return `<div class="language-${lang}${getAdaptiveThemeMarker(
22+
options
23+
)}${active}"><button title="Copy Code" class="copy"></button><span class="lang">${lang}</span>${rawCode}</div>`
2024
}
2125
}
2226

2327
export function getAdaptiveThemeMarker(options: Options) {
2428
return options.hasSingleTheme ? '' : ' vp-adaptive-theme'
2529
}
2630

27-
export function extractTitle(info: string) {
31+
export function extractTitle(info: string, html = false) {
32+
if (html) {
33+
return (
34+
info.replace(/<!--[^]*?-->/g, '').match(/data-title="(.*?)"/)?.[1] || ''
35+
)
36+
}
2837
return info.match(/\[(.*)\]/)?.[1] || extractLang(info) || 'txt'
2938
}
3039

0 commit comments

Comments
 (0)
Please sign in to comment.