From 7be9aa55b737e65f51d0571410c239732156a67f Mon Sep 17 00:00:00 2001 From: Divyansh Singh <40380293+brc-dd@users.noreply.github.com> Date: Mon, 31 Oct 2022 00:59:53 +0530 Subject: [PATCH 1/5] feat: add code-groups component Co-authored-by: "Jinjing.Zhou" --- .gitignore | 1 + docs/guide/markdown.md | 76 +++++++++++++++++-- package.json | 5 +- pnpm-lock.yaml | 2 + src/client/app/composables/codeGroups.ts | 23 ++++++ src/client/app/composables/copyCode.ts | 2 +- src/client/app/index.ts | 3 + src/client/theme-default/index.ts | 1 + .../styles/components/vp-code-group.css | 73 ++++++++++++++++++ src/client/theme-default/styles/vars.css | 7 +- src/node/markdown/plugins/containers.ts | 42 ++++++++++ src/node/markdown/plugins/highlight.ts | 3 +- src/node/markdown/plugins/preWrapper.ts | 32 ++++---- 13 files changed, 242 insertions(+), 28 deletions(-) create mode 100644 src/client/app/composables/codeGroups.ts create mode 100644 src/client/theme-default/styles/components/vp-code-group.css diff --git a/.gitignore b/.gitignore index b0bd43cf04f5..e15ebac8c727 100644 --- a/.gitignore +++ b/.gitignore @@ -13,3 +13,4 @@ TODOs.md .temp *.tgz examples-temp +TODO diff --git a/docs/guide/markdown.md b/docs/guide/markdown.md index 6c25b1a1b432..0e23304b28ff 100644 --- a/docs/guide/markdown.md +++ b/docs/guide/markdown.md @@ -371,7 +371,7 @@ export default { ```js export default { - data () { + data() { return { msg: 'Highlighted!' // [!code hl] } @@ -381,7 +381,7 @@ export default { ## Focus in Code Blocks -Adding the `// [!code focus]` comment on a line will focus it and blur the other parts of the code. +Adding the `// [!code focus]` comment on a line will focus it and blur the other parts of the code. Additionally, you can define a number of lines to focus using `// [!code focus:]`. @@ -403,7 +403,7 @@ export default { ```js export default { - data () { + data() { return { msg: 'Focused!' // [!code focus] } @@ -411,9 +411,9 @@ export default { } ``` -## Colored diffs in Code Blocks +## Colored Diffs in Code Blocks -Adding the `// [!code --]` or `// [!code ++]` comments on a line will create a diff of that line, while keeping the colors of the codeblock. +Adding the `// [!code --]` or `// [!code ++]` comments on a line will create a diff of that line, while keeping the colors of the codeblock. **Input** @@ -443,7 +443,7 @@ export default { } ``` -## Errors and warnings +## Errors and Warnings in Code Blocks Adding the `// [!code warning]` or `// [!code error]` comments on a line will color it accordingly. @@ -466,7 +466,7 @@ export default { ```js export default { - data () { + data() { return { msg: 'Error', // [!code error] msg: 'Warning' // [!code warning] @@ -475,7 +475,6 @@ export default { } ``` - ## Line Numbers You can enable line numbers for each code blocks via config: @@ -544,11 +543,72 @@ You can also specify the language inside the braces (`{}`) like this: <<< @/snippets/snippet.cs{c#} + <<< @/snippets/snippet.cs{1,2,4-6 c#} ``` This is helpful if source language cannot be inferred from your file extension. +## Code Groups + +You can group multiple code blocks like this: + +**Input** + +````md +::: code-group + +```js [config.js] +/** + * @type {import('vitepress').UserConfig} + */ +const config = { + // ... +} + +export default config +``` + +```ts [config.ts] +import type { UserConfig } from 'vitepress' + +const config: UserConfig = { + // ... +} + +export default config +``` + +::: +```` + +**Output** + +::: code-group + +```js [config.js] +/** + * @type {import('vitepress').UserConfig} + */ +const config = { + // ... +} + +export default config +``` + +```ts [config.ts] +import type { UserConfig } from 'vitepress' + +const config: UserConfig = { + // ... +} + +export default config +``` + +::: + ## Markdown File Inclusion You can include a markdown file in another markdown file like this: diff --git a/package.json b/package.json index c9a3fe0764ab..1591db2b1bf6 100644 --- a/package.json +++ b/package.json @@ -73,7 +73,7 @@ "changelog": "conventional-changelog -p angular -i CHANGELOG.md -s", "release": "node scripts/release.js", "docs": "run-p dev docs-dev", - "docs-dev": "wait-on -d 1000 dist/node/cli.js && node ./bin/vitepress dev docs", + "docs-dev": "wait-on -d 100 dist/node/cli.js && node ./bin/vitepress dev docs", "docs-debug": "node --inspect-brk ./bin/vitepress dev docs", "docs-build": "run-s build docs-build-only", "docs-build-only": "node ./bin/vitepress build docs", @@ -140,6 +140,7 @@ "markdown-it-emoji": "^2.0.2", "micromatch": "^4.0.5", "minimist": "^1.2.7", + "nanoid": "3.3.4", "npm-run-all": "^4.1.5", "ora": "^5.4.1", "picocolors": "^1.0.0", @@ -154,9 +155,9 @@ "rollup-plugin-dts": "^4.2.3", "rollup-plugin-esbuild": "^4.10.1", "semver": "^7.3.8", + "shiki-processor": "^0.1.1", "simple-git-hooks": "^2.8.1", "sirv": "^2.0.2", - "shiki-processor": "^0.1.1", "supports-color": "^9.2.3", "typescript": "~4.8.4", "vitest": "^0.24.3", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index e6333d2b11a7..d2042c152d03 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -58,6 +58,7 @@ importers: markdown-it-emoji: ^2.0.2 micromatch: ^4.0.5 minimist: ^1.2.7 + nanoid: 3.3.4 npm-run-all: ^4.1.5 ora: ^5.4.1 picocolors: ^1.0.0 @@ -142,6 +143,7 @@ importers: markdown-it-emoji: 2.0.2 micromatch: 4.0.5 minimist: 1.2.7 + nanoid: 3.3.4 npm-run-all: 4.1.5 ora: 5.4.1 picocolors: 1.0.0 diff --git a/src/client/app/composables/codeGroups.ts b/src/client/app/composables/codeGroups.ts new file mode 100644 index 000000000000..2c5c2cb0d64e --- /dev/null +++ b/src/client/app/composables/codeGroups.ts @@ -0,0 +1,23 @@ +import { inBrowser } from 'vitepress' + +export function useCodeGroups() { + if (inBrowser) { + window.addEventListener('click', (e) => { + const el = e.target as HTMLInputElement + + if (el.matches('.vp-code-group input')) { + // input <- .tabs <- .vp-code-group + const group = el.parentElement?.parentElement + const i = Array.from(group?.querySelectorAll('input') || []).indexOf(el) + + const current = group?.querySelector('div[class*="language-"].active') + const next = group?.querySelectorAll('div[class*="language-"]')?.[i] + + if (current && next && current !== next) { + current.classList.remove('active') + next.classList.add('active') + } + } + }) + } +} diff --git a/src/client/app/composables/copyCode.ts b/src/client/app/composables/copyCode.ts index b861823be1a0..6c9248aa1266 100644 --- a/src/client/app/composables/copyCode.ts +++ b/src/client/app/composables/copyCode.ts @@ -1,4 +1,4 @@ -import { inBrowser } from '../utils.js' +import { inBrowser } from 'vitepress' export function useCopyCode() { if (inBrowser) { diff --git a/src/client/app/index.ts b/src/client/app/index.ts index 84bd1c23da8b..9deba800666c 100644 --- a/src/client/app/index.ts +++ b/src/client/app/index.ts @@ -17,6 +17,7 @@ import { dataSymbol, initData } from './data.js' import { Content } from './components/Content.js' import { ClientOnly } from './components/ClientOnly.js' import { useCopyCode } from './composables/copyCode.js' +import { useCodeGroups } from './composables/codeGroups.js' const NotFound = Theme.NotFound || (() => '404 Not Found') @@ -43,6 +44,8 @@ const VitePressApp = defineComponent({ // setup global copy code handler useCopyCode() + // setup global code groups handler + useCodeGroups() if (Theme.setup) Theme.setup() return () => h(Theme.Layout) diff --git a/src/client/theme-default/index.ts b/src/client/theme-default/index.ts index 7b7921283586..d9c4ae4af9e0 100644 --- a/src/client/theme-default/index.ts +++ b/src/client/theme-default/index.ts @@ -4,6 +4,7 @@ import './styles/base.css' import './styles/utils.css' import './styles/components/custom-block.css' import './styles/components/vp-code.css' +import './styles/components/vp-code-group.css' import './styles/components/vp-doc.css' import './styles/components/vp-sponsor.css' diff --git a/src/client/theme-default/styles/components/vp-code-group.css b/src/client/theme-default/styles/components/vp-code-group.css new file mode 100644 index 000000000000..1e8935b6b947 --- /dev/null +++ b/src/client/theme-default/styles/components/vp-code-group.css @@ -0,0 +1,73 @@ +.vp-code-group { + display: flex; + flex-direction: column; + margin-top: 16px; +} + +.vp-code-group .tabs { + display: flex; + margin: 0 -24px; + padding-bottom: 2px; + overflow: auto; +} + +.vp-code-group div[class*='language-'] { + display: none; + margin-top: 0 !important; + border-top-left-radius: 0 !important; +} + +.vp-code-group div[class*='language-'].active { + display: block; +} + +.vp-code-group input { + position: absolute; + opacity: 0; + pointer-events: none; +} + +.vp-code-group label { + transition: background-color 300ms; + border: 2px solid transparent; + border-bottom-width: 0; + box-shadow: 0 2px var(--vp-c-text-dark-2); + background: var(--vp-code-tab); + cursor: pointer; + padding: 4px 20px 5px; + color: #fff; +} + +.vp-code-group label:hover { + background: var(--vp-code-tab-hover); +} + +.vp-code-group input:focus-visible + label { + border-color: var(--vp-c-text-dark-2); +} + +.vp-code-group input:checked + label { + box-shadow: 0 2px var(--vp-c-brand); + background: var(--vp-code-block-bg); +} + +@media (min-width: 640px) { + .vp-code-group .tabs { + margin: 0; + } + + .vp-code-group label:first-of-type { + border-top-left-radius: 8px; + } + + .vp-code-group label:last-of-type { + border-top-right-radius: 8px; + } +} + +@media (max-width: 639px) { + li .vp-code-group .tabs, + li .vp-code-group label:first-of-type { + border-top-left-radius: 8px; + } +} diff --git a/src/client/theme-default/styles/vars.css b/src/client/theme-default/styles/vars.css index f736d93f08a4..c1f035216668 100644 --- a/src/client/theme-default/styles/vars.css +++ b/src/client/theme-default/styles/vars.css @@ -220,10 +220,15 @@ --vp-code-copy-code-hover-bg: rgba(255, 255, 255, 0.05); --vp-code-copy-code-active-text: var(--vp-c-text-dark-2); + + --vp-code-tab: #393f57; + --vp-code-tab-hover: #49516f; } .dark { - --vp-code-block-bg: var(--vp-c-bg-alt); + --vp-code-block-bg: var(--vp-c-black); + --vp-code-tab: var(--vp-c-black-mute); + --vp-code-tab-hover: var(--vp-c-gray-dark-4); } /** diff --git a/src/node/markdown/plugins/containers.ts b/src/node/markdown/plugins/containers.ts index 40480a10ce41..f393b180e50e 100644 --- a/src/node/markdown/plugins/containers.ts +++ b/src/node/markdown/plugins/containers.ts @@ -2,6 +2,8 @@ import MarkdownIt from 'markdown-it' import { RenderRule } from 'markdown-it/lib/renderer' import Token from 'markdown-it/lib/token' import container from 'markdown-it-container' +import { nanoid } from 'nanoid' +import { extractTitle } from './preWrapper' export const containerPlugin = (md: MarkdownIt) => { md.use(...createContainer('tip', 'TIP', md)) @@ -18,6 +20,7 @@ export const containerPlugin = (md: MarkdownIt) => { render: (tokens: Token[], idx: number) => tokens[idx].nesting === 1 ? `
\n` : `
\n` }) + .use(...createCodeGroup()) } type ContainerArgs = [typeof container, string, { render: RenderRule }] @@ -47,3 +50,42 @@ function createContainer( } ] } + +function createCodeGroup(): ContainerArgs { + return [ + container, + 'code-group', + { + render(tokens, idx) { + if (tokens[idx].nesting === 1) { + const name = nanoid(5) + let tabs = '' + let checked = 'checked="checked"' + + for ( + let i = idx + 1; + !( + tokens[i].nesting === -1 && + tokens[i].type === 'container_code-group_close' + ); + ++i + ) { + if (tokens[i].type === 'fence' && tokens[i].tag === 'code') { + const title = extractTitle(tokens[i].info) + const id = nanoid(7) + tabs += `` + + if (checked) { + tokens[i].info += ' active' + checked = '' + } + } + } + + return `
${tabs}
\n` + } + return `
\n` + } + } + ] +} diff --git a/src/node/markdown/plugins/highlight.ts b/src/node/markdown/plugins/highlight.ts index 46e3f92ab24b..d6db23b098b0 100644 --- a/src/node/markdown/plugins/highlight.ts +++ b/src/node/markdown/plugins/highlight.ts @@ -20,8 +20,9 @@ import type { ThemeOptions } from '../markdown' * [{ line: number, classes: string[] }] */ const attrsToLines = (attrs: string): HtmlRendererOptions['lineOptions'] => { + attrs = attrs.replace(/.*?([\d,-]+).*/, '$1').trim() const result: number[] = [] - if (!attrs.trim()) { + if (!attrs) { return [] } attrs diff --git a/src/node/markdown/plugins/preWrapper.ts b/src/node/markdown/plugins/preWrapper.ts index 7a6b22e7eca0..547b91e47448 100644 --- a/src/node/markdown/plugins/preWrapper.ts +++ b/src/node/markdown/plugins/preWrapper.ts @@ -1,22 +1,24 @@ -// markdown-it plugin for wrapping
 ... 
. -// -// If your plugin was chained before preWrapper, you can add additional element directly. -// If your plugin was chained after preWrapper, you can use these slots: -// 1. -// 2. -// 3. -// 4. - import MarkdownIt from 'markdown-it' -export const preWrapperPlugin = (md: MarkdownIt) => { +export function preWrapperPlugin(md: MarkdownIt) { const fence = md.renderer.rules.fence! md.renderer.rules.fence = (...args) => { - const [tokens, idx] = args - const lang = tokens[idx].info.trim().replace(/-vue$/, '') + const { info } = args[0][args[1]] + const lang = extractLang(info) const rawCode = fence(...args) - return `
${ - lang === 'vue-html' ? 'template' : lang - }${rawCode}
` + return `
${lang}${rawCode}
` } } + +export function extractTitle(info: string) { + return info.match(/\[(.*)\]/)?.[1] || extractLang(info) || 'txt' +} + +const extractLang = (info: string) => { + return info + .trim() + .replace(/(-vue|{| ).*$/, '') + .replace(/^vue-html$/, 'template') +} From dfb453413ad0f7580f436139bf213ecc43993b64 Mon Sep 17 00:00:00 2001 From: Divyansh Singh <40380293+brc-dd@users.noreply.github.com> Date: Mon, 31 Oct 2022 00:59:57 +0530 Subject: [PATCH 2/5] test all possible variations --- docs/test.md | 84 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 84 insertions(+) create mode 100644 docs/test.md diff --git a/docs/test.md b/docs/test.md new file mode 100644 index 000000000000..1e9f9cbe65c0 --- /dev/null +++ b/docs/test.md @@ -0,0 +1,84 @@ +# Code Groups + +::: code-group + +```txt-vue{1} +{{ 1 + 1 }} +``` + +```js [app.vue] + +``` + + + +```vue-html{3,4} [layouts/custom.vue] + +``` + +```js{1-3,5} [layouts/default.vue] +export default { + name: 'MyComponent' + // ... +} + +``` + +::: + +- in list + +- ::: code-group + + ```js + printf('111') + ``` + + ```python + import torch as th + print("Hello world") + ``` + + ``` + import torch as th + print("Hello world") + ``` + + ::: + +``` +. +├─ index.md +├─ foo +│ ├─ index.md +│ ├─ one.md +│ └─ two.md +└─ bar + ├─ index.md + ├─ three.md + └─ four.md +``` + +- ```md{1-3,5} + [Home](/) + [foo](/foo/) + [foo heading](./#heading) + [bar - three](../bar/three) + [bar - three](../bar/three.md) + [bar - four](../bar/four.html) + ``` From c831b94aefb906b9c8cf4b502387df34ab1e999f Mon Sep 17 00:00:00 2001 From: Kia King Ishii Date: Fri, 16 Dec 2022 00:12:17 +0900 Subject: [PATCH 3/5] adjust stylings --- package.json | 1 + .../styles/components/vp-code-group.css | 96 +++++++++++-------- src/client/theme-default/styles/vars.css | 13 ++- 3 files changed, 65 insertions(+), 45 deletions(-) diff --git a/package.json b/package.json index 74d8a44108e0..6e875dae73db 100644 --- a/package.json +++ b/package.json @@ -138,6 +138,7 @@ "markdown-it-emoji": "^2.0.2", "micromatch": "^4.0.5", "minimist": "^1.2.7", + "nanoid": "3.3.4", "npm-run-all": "^4.1.5", "ora": "^5.4.1", "picocolors": "^1.0.0", diff --git a/src/client/theme-default/styles/components/vp-code-group.css b/src/client/theme-default/styles/components/vp-code-group.css index 1e8935b6b947..0384bae34933 100644 --- a/src/client/theme-default/styles/components/vp-code-group.css +++ b/src/client/theme-default/styles/components/vp-code-group.css @@ -1,73 +1,87 @@ .vp-code-group { - display: flex; - flex-direction: column; margin-top: 16px; } .vp-code-group .tabs { + position: relative; display: flex; - margin: 0 -24px; - padding-bottom: 2px; + margin-right: -24px; + margin-left: -24px; + padding: 0 12px; + background-color: var(--vp-code-tab-bg); overflow: auto; } -.vp-code-group div[class*='language-'] { - display: none; - margin-top: 0 !important; - border-top-left-radius: 0 !important; +.vp-code-group .tabs::after { + position: absolute; + right: 0; + bottom: 0; + left: 0; + height: 1px; + background-color: var(--vp-code-tab-divider); + content: ''; } -.vp-code-group div[class*='language-'].active { - display: block; +@media (min-width: 640px) { + .vp-code-group .tabs { + margin-right: 0; + margin-left: 0; + border-radius: 8px 8px 0 0; + } } -.vp-code-group input { +.vp-code-group .tabs input { position: absolute; opacity: 0; pointer-events: none; } -.vp-code-group label { - transition: background-color 300ms; - border: 2px solid transparent; - border-bottom-width: 0; - box-shadow: 0 2px var(--vp-c-text-dark-2); - background: var(--vp-code-tab); +.vp-code-group .tabs label { + position: relative; + display: inline-block; + border-bottom: 1px solid transparent; + padding: 0 12px; + line-height: 48px; + font-size: 14px; + font-weight: 500; + color: var(--vp-code-tab-text-color); + background-color: var(--vp-code-tab-bg); + white-space: nowrap; cursor: pointer; - padding: 4px 20px 5px; - color: #fff; + transition: color 0.25s; } -.vp-code-group label:hover { - background: var(--vp-code-tab-hover); +.vp-code-group .tabs label::after { + position: absolute; + right: 8px; + bottom: -1px; + left: 8px; + z-index: 10; + height: 1px; + content: ''; + background-color: transparent; + transition: background-color 0.25s; } -.vp-code-group input:focus-visible + label { - border-color: var(--vp-c-text-dark-2); +.vp-code-group label:hover { + color: var(--vp-code-tab-hover-text-color); } .vp-code-group input:checked + label { - box-shadow: 0 2px var(--vp-c-brand); - background: var(--vp-code-block-bg); + color: var(--vp-code-tab-active-text-color); } -@media (min-width: 640px) { - .vp-code-group .tabs { - margin: 0; - } - - .vp-code-group label:first-of-type { - border-top-left-radius: 8px; - } +.vp-code-group input:checked + label::after { + background-color: var(--vp-code-tab-active-bar-color); +} - .vp-code-group label:last-of-type { - border-top-right-radius: 8px; - } +.vp-code-group div[class*='language-'] { + display: none; + margin-top: 0 !important; + border-top-left-radius: 0 !important; + border-top-right-radius: 0 !important; } -@media (max-width: 639px) { - li .vp-code-group .tabs, - li .vp-code-group label:first-of-type { - border-top-left-radius: 8px; - } +.vp-code-group div[class*='language-'].active { + display: block; } diff --git a/src/client/theme-default/styles/vars.css b/src/client/theme-default/styles/vars.css index f4d6414974c1..87ae88de97ad 100644 --- a/src/client/theme-default/styles/vars.css +++ b/src/client/theme-default/styles/vars.css @@ -229,14 +229,19 @@ --vp-code-copy-code-hover-bg: rgba(255, 255, 255, 0.05); --vp-code-copy-code-active-text: var(--vp-c-text-dark-2); - --vp-code-tab: #393f57; - --vp-code-tab-hover: #49516f; + --vp-code-tab-text-color: var(--vp-c-text-dark-2); + --vp-code-tab-bg: var(--vp-code-block-bg); + --vp-code-tab-divider: var(--vp-c-divider-dark-2); + --vp-code-tab-hover-text-color: var(--vp-c-text-dark-1); + --vp-code-tab-active-text-color: var(--vp-c-text-dark-1); + --vp-code-tab-active-bar-color: var(--vp-c-brand); } .dark { --vp-code-block-bg: var(--vp-c-black); - --vp-code-tab: var(--vp-c-black-mute); - --vp-code-tab-hover: var(--vp-c-gray-dark-4); + + /* --vp-code-tab: var(--vp-c-black-mute);*/ + /* --vp-code-tab-hover: var(--vp-c-gray-dark-4);*/ } /** From f2a0f9f06577cf77b8f71b530768c8c45bdaccc6 Mon Sep 17 00:00:00 2001 From: Kia King Ishii Date: Fri, 16 Dec 2022 00:17:04 +0900 Subject: [PATCH 4/5] ignore cache --- .gitignore | 1 + 1 file changed, 1 insertion(+) diff --git a/.gitignore b/.gitignore index 3a95953c8459..7bf9a27daa31 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ +/__tests__/e2e/.vitepress/cache /coverage /src/client/shared.ts /src/node/shared.ts From da7c9967e07b4bedae101e927c005d1a27848d84 Mon Sep 17 00:00:00 2001 From: Kia Ishii Date: Fri, 16 Dec 2022 09:49:18 +0900 Subject: [PATCH 5/5] move nanoid to production dependencies --- package.json | 2 +- pnpm-lock.yaml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 6e875dae73db..7de34198a239 100644 --- a/package.json +++ b/package.json @@ -84,6 +84,7 @@ "@vue/devtools-api": "^6.4.5", "@vueuse/core": "^9.6.0", "body-scroll-lock": "4.0.0-beta.0", + "nanoid": "3.3.4", "shiki": "^0.11.1", "vite": "^4.0.0", "vue": "^3.2.45" @@ -138,7 +139,6 @@ "markdown-it-emoji": "^2.0.2", "micromatch": "^4.0.5", "minimist": "^1.2.7", - "nanoid": "3.3.4", "npm-run-all": "^4.1.5", "ora": "^5.4.1", "picocolors": "^1.0.0", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 671f7ccc26a5..aedac2a73a85 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -92,6 +92,7 @@ importers: '@vue/devtools-api': 6.4.5 '@vueuse/core': 9.6.0_vue@3.2.45 body-scroll-lock: 4.0.0-beta.0 + nanoid: 3.3.4 shiki: 0.11.1 vite: 4.0.0_@types+node@18.11.13 vue: 3.2.45 @@ -145,7 +146,6 @@ importers: markdown-it-emoji: 2.0.2 micromatch: 4.0.5 minimist: 1.2.7 - nanoid: 3.3.4 npm-run-all: 4.1.5 ora: 5.4.1 picocolors: 1.0.0