From 712e8af2ff74e96ef88f0ab468672e0fd96ed125 Mon Sep 17 00:00:00 2001 From: chenhaoli Date: Sun, 12 Dec 2021 03:38:22 +0800 Subject: [PATCH 1/2] feat: improve typescript support for config file --- docs/.vitepress/config.ts | 6 +- docs/guide/configuration.md | 58 ++++++++++++ src/client/theme-default/config.ts | 147 +---------------------------- src/client/theme-default/index.ts | 1 + src/client/tsconfig.json | 2 +- src/node/config.ts | 34 ++++--- src/node/index.ts | 2 +- src/shared/shared.ts | 3 +- src/shared/tsconfig.json | 2 +- types/default-theme.d.ts | 146 ++++++++++++++++++++++++++++ types/index.d.ts | 2 +- types/shared.d.ts | 6 +- 12 files changed, 241 insertions(+), 168 deletions(-) create mode 100644 types/default-theme.d.ts diff --git a/docs/.vitepress/config.ts b/docs/.vitepress/config.ts index 8dff7daf2b1d..0c326ba5b132 100644 --- a/docs/.vitepress/config.ts +++ b/docs/.vitepress/config.ts @@ -1,4 +1,6 @@ -export default { +import { defineConfig } from '../../src/node' + +export default defineConfig({ lang: 'en-US', title: 'VitePress', description: 'Vite & Vue powered static site generator.', @@ -42,7 +44,7 @@ export default { '/': getGuideSidebar() } } -} +}) function getGuideSidebar() { return [ diff --git a/docs/guide/configuration.md b/docs/guide/configuration.md index 7dc8d1e00189..4ec740819b8a 100644 --- a/docs/guide/configuration.md +++ b/docs/guide/configuration.md @@ -1,5 +1,7 @@ # Configuration +## Overview + Without any configuration, the page is pretty minimal, and the user has no way to navigate around the site. To customize your site, let’s first create a `.vitepress` directory inside your docs directory. This is where all VitePress-specific files will be placed. Your project structure is probably like this: ```bash @@ -21,3 +23,59 @@ module.exports = { ``` Check out the [Config Reference](/config/basics) for a full list of options. + + +## Config Intellisense + +Since VitePress ships with TypeScript typings, you can leverage your IDE's intellisense with jsdoc type hints: + +```js +/** + * @type {import('vitepress').UserConfig} + */ +const config = { + // ... +} + +export default config +``` + +Alternatively, you can use the `defineConfig` helper at which should provide intellisense without the need for jsdoc annotations: + +```js +import { defineConfig } from 'vitepress' + +export default defineConfig({ + // ... +}) +``` + +VitePress also directly supports TS config files. You can use `.vitepress/config.ts` with the `defineConfig` helper as well. + + +## Typed Theme Config + +By default, `defineConfig` helper leverages the theme config type from default theme: + +```js +import { defineConfig } from 'vitepress' + +export default defineConfig({ + themeConfig: { + // Type is `DefaultTheme.Config` + } +}) +``` + +If you use a custom theme, you'll be able to pass the generics type for your custom theme, and you need overload it with the second parameter of `defineConfig` helper: + +```js +import { defineConfig } from 'vitepress' +import { ThemeConfig } from 'your-theme' + +export default defineConfig({ + themeConfig: { + // Type is `ThemeConfig` + } +}, true); // declare `usingCustomTheme` and discard usage of the default theme. +``` diff --git a/src/client/theme-default/config.ts b/src/client/theme-default/config.ts index d72848717e19..92869ac43f43 100644 --- a/src/client/theme-default/config.ts +++ b/src/client/theme-default/config.ts @@ -1,146 +1 @@ -export namespace DefaultTheme { - export interface Config { - logo?: string - nav?: NavItem[] | false - sidebar?: SideBarConfig | MultiSideBarConfig - - /** - * GitHub repository following the format /. - * - * @example `"vuejs/vue-next"` - */ - repo?: string - - /** - * Customize the header label. Defaults to GitHub/Gitlab/Bitbucket - * depending on the provided repo. - * - * @example `"Contribute!"` - */ - repoLabel?: string - - /** - * If your docs are in a different repository from your main project. - * - * @example `"vuejs/docs-next"` - */ - docsRepo?: string - - /** - * If your docs are not at the root of the repo. - * - * @example `"docs"` - */ - docsDir?: string - - /** - * If your docs are in a different branch. Defaults to `master`. - * - * @example `"next"` - */ - docsBranch?: string - - /** - * Enable links to edit pages at the bottom of the page. - */ - editLinks?: boolean - - /** - * Custom text for edit link. Defaults to "Edit this page". - */ - editLinkText?: string - - /** - * Show last updated time at the bottom of the page. Defaults to `false`. - * If given a string, it will be displayed as a prefix (default value: - * "Last Updated"). - */ - lastUpdated?: string | boolean - - prevLinks?: boolean - nextLinks?: boolean - - locales?: Record> - - algolia?: AlgoliaSearchOptions - - carbonAds?: { - carbon: string - custom?: string - placement: string - } - } - - // navbar -------------------------------------------------------------------- - - export type NavItem = NavItemWithLink | NavItemWithChildren - - export interface NavItemBase { - text: string - target?: string - rel?: string - ariaLabel?: string - activeMatch?: string - } - - export interface NavItemWithLink extends NavItemBase { - link: string - } - - export interface NavItemWithChildren extends NavItemBase { - items: NavItemWithLink[] - } - - // sidebar ------------------------------------------------------------------- - - export type SideBarConfig = SideBarItem[] | 'auto' | false - - export interface MultiSideBarConfig { - [path: string]: SideBarConfig - } - - export type SideBarItem = SideBarLink | SideBarGroup - - export interface SideBarLink { - text: string - link: string - } - - export interface SideBarGroup { - text: string - link?: string - - /** - * @default false - */ - collapsable?: boolean - - children: SideBarItem[] - } - - // algolia ------------------------------------------------------------------ - // partially copied from @docsearch/react/dist/esm/DocSearch.d.ts - export interface AlgoliaSearchOptions { - appId?: string - apiKey: string - indexName: string - placeholder?: string - searchParameters?: any - disableUserPersonalization?: boolean - initialQuery?: string - } - - // locales ------------------------------------------------------------------- - - export interface LocaleConfig { - /** - * Text for the language dropdown. - */ - selectText?: string - - /** - * Label for this locale in the language dropdown. - */ - label?: string - } -} +export { DefaultTheme } from '../shared' diff --git a/src/client/theme-default/index.ts b/src/client/theme-default/index.ts index b9c753dd9e33..bcd79051fc1f 100644 --- a/src/client/theme-default/index.ts +++ b/src/client/theme-default/index.ts @@ -8,6 +8,7 @@ import { Theme } from 'vitepress' import Layout from './Layout.vue' import NotFound from './NotFound.vue' +export { DefaultTheme } from './config' const theme: Theme = { Layout, NotFound diff --git a/src/client/tsconfig.json b/src/client/tsconfig.json index f7f086aa6e3b..92a8af1ce58a 100644 --- a/src/client/tsconfig.json +++ b/src/client/tsconfig.json @@ -12,5 +12,5 @@ "vitepress": ["index.ts"] } }, - "include": [".", "../../types/shared.d.ts"] + "include": ["."] } diff --git a/src/node/config.ts b/src/node/config.ts index 4fdc21171a71..3c82976cc5cf 100644 --- a/src/node/config.ts +++ b/src/node/config.ts @@ -14,7 +14,8 @@ import { SiteData, HeadConfig, LocaleConfig, - createLangDictionary + createLangDictionary, + DefaultTheme, } from './shared' import { resolveAliases, APP_PATH, DEFAULT_THEME_PATH } from './alias' import { MarkdownOptions } from './markdown/markdown' @@ -26,14 +27,16 @@ const debug = _debug('vitepress:config') export type { MarkdownOptions } -export interface UserConfig { - extends?: RawConfigExports +export type ThemeConfig = any; + +export interface UserConfig { + extends?: RawConfigExports lang?: string base?: string title?: string description?: string head?: HeadConfig[] - themeConfig?: ThemeConfig + themeConfig?: T locales?: Record markdown?: MarkdownOptions /** @@ -55,15 +58,15 @@ export interface UserConfig { mpa?: boolean } -export type RawConfigExports = - | UserConfig - | Promise - | (() => UserConfig | Promise) +export type RawConfigExports = + | UserConfig + | Promise> + | (() => UserConfig | Promise>) -export interface SiteConfig { +export interface SiteConfig { root: string srcDir: string - site: SiteData + site: SiteData configPath: string | undefined themeDir: string outDir: string @@ -82,7 +85,12 @@ const resolve = (root: string, file: string) => /** * Type config helper */ -export function defineConfig(config: RawConfigExports) { + export function defineConfig( + config: UserConfig, + usingCustomTheme: true +): void +export function defineConfig(config: UserConfig): void +export function defineConfig(config: ThemeConfig) { return config } @@ -150,7 +158,7 @@ async function resolveUserConfig( } } - const userConfig: RawConfigExports = configPath + const userConfig: RawConfigExports = configPath ? (( await loadConfigFromFile( { @@ -173,7 +181,7 @@ async function resolveUserConfig( } async function resolveConfigExtends( - config: RawConfigExports + config: RawConfigExports ): Promise { const resolved = await (typeof config === 'function' ? config() : config) if (resolved.extends) { diff --git a/src/node/index.ts b/src/node/index.ts index f671e9212802..dd6b048e52f4 100644 --- a/src/node/index.ts +++ b/src/node/index.ts @@ -4,4 +4,4 @@ export * from './serve/serve' export * from './config' export * from './markdown/markdown' -export type { SiteData, HeadConfig, LocaleConfig } from '../../types/shared' +export type { SiteData, HeadConfig, LocaleConfig, DefaultTheme } from '../../types/shared' diff --git a/src/shared/shared.ts b/src/shared/shared.ts index d3d34eb8c396..1e1e0a5a4bf2 100644 --- a/src/shared/shared.ts +++ b/src/shared/shared.ts @@ -5,7 +5,8 @@ export type { PageData, HeadConfig, LocaleConfig, - Header + Header, + DefaultTheme, } from '../../types/shared' export const EXTERNAL_URL_RE = /^https?:/i diff --git a/src/shared/tsconfig.json b/src/shared/tsconfig.json index 5dd8a6dbcc61..a8ed344e696e 100644 --- a/src/shared/tsconfig.json +++ b/src/shared/tsconfig.json @@ -3,5 +3,5 @@ "compilerOptions": { "baseUrl": "." }, - "include": [".", "../../types/shared.d.ts"] + "include": ["."] } diff --git a/types/default-theme.d.ts b/types/default-theme.d.ts new file mode 100644 index 000000000000..d72848717e19 --- /dev/null +++ b/types/default-theme.d.ts @@ -0,0 +1,146 @@ +export namespace DefaultTheme { + export interface Config { + logo?: string + nav?: NavItem[] | false + sidebar?: SideBarConfig | MultiSideBarConfig + + /** + * GitHub repository following the format /. + * + * @example `"vuejs/vue-next"` + */ + repo?: string + + /** + * Customize the header label. Defaults to GitHub/Gitlab/Bitbucket + * depending on the provided repo. + * + * @example `"Contribute!"` + */ + repoLabel?: string + + /** + * If your docs are in a different repository from your main project. + * + * @example `"vuejs/docs-next"` + */ + docsRepo?: string + + /** + * If your docs are not at the root of the repo. + * + * @example `"docs"` + */ + docsDir?: string + + /** + * If your docs are in a different branch. Defaults to `master`. + * + * @example `"next"` + */ + docsBranch?: string + + /** + * Enable links to edit pages at the bottom of the page. + */ + editLinks?: boolean + + /** + * Custom text for edit link. Defaults to "Edit this page". + */ + editLinkText?: string + + /** + * Show last updated time at the bottom of the page. Defaults to `false`. + * If given a string, it will be displayed as a prefix (default value: + * "Last Updated"). + */ + lastUpdated?: string | boolean + + prevLinks?: boolean + nextLinks?: boolean + + locales?: Record> + + algolia?: AlgoliaSearchOptions + + carbonAds?: { + carbon: string + custom?: string + placement: string + } + } + + // navbar -------------------------------------------------------------------- + + export type NavItem = NavItemWithLink | NavItemWithChildren + + export interface NavItemBase { + text: string + target?: string + rel?: string + ariaLabel?: string + activeMatch?: string + } + + export interface NavItemWithLink extends NavItemBase { + link: string + } + + export interface NavItemWithChildren extends NavItemBase { + items: NavItemWithLink[] + } + + // sidebar ------------------------------------------------------------------- + + export type SideBarConfig = SideBarItem[] | 'auto' | false + + export interface MultiSideBarConfig { + [path: string]: SideBarConfig + } + + export type SideBarItem = SideBarLink | SideBarGroup + + export interface SideBarLink { + text: string + link: string + } + + export interface SideBarGroup { + text: string + link?: string + + /** + * @default false + */ + collapsable?: boolean + + children: SideBarItem[] + } + + // algolia ------------------------------------------------------------------ + // partially copied from @docsearch/react/dist/esm/DocSearch.d.ts + export interface AlgoliaSearchOptions { + appId?: string + apiKey: string + indexName: string + placeholder?: string + searchParameters?: any + disableUserPersonalization?: boolean + initialQuery?: string + } + + // locales ------------------------------------------------------------------- + + export interface LocaleConfig { + /** + * Text for the language dropdown. + */ + selectText?: string + + /** + * Label for this locale in the language dropdown. + */ + label?: string + } +} diff --git a/types/index.d.ts b/types/index.d.ts index 10a5d0c24743..ea74a292a8c2 100644 --- a/types/index.d.ts +++ b/types/index.d.ts @@ -1,4 +1,4 @@ export * from './shared' +export * from './default-theme' export * from '../dist/node/index' export * from '../dist/client/index' -export * from '../dist/client/theme-default/config' diff --git a/types/shared.d.ts b/types/shared.d.ts index 4f6558a6276b..63fbb624b5bb 100644 --- a/types/shared.d.ts +++ b/types/shared.d.ts @@ -1,5 +1,7 @@ // types shared between server and client +export { DefaultTheme } from './default-theme' + export interface LocaleConfig { lang: string title?: string @@ -9,7 +11,7 @@ export interface LocaleConfig { selectText?: string } -export interface SiteData { +export interface SiteData { base: string /** * Language of the site as it should be set on the `html` element. @@ -19,7 +21,7 @@ export interface SiteData { title: string description: string head: HeadConfig[] - themeConfig: ThemeConfig + themeConfig: T locales: Record /** * Available locales for the site when it has defined `locales` in its From 83f408d4b2c61d04f5d7ea43c1558b2d3d770f9f Mon Sep 17 00:00:00 2001 From: Evan You Date: Thu, 6 Jan 2022 10:15:25 +0800 Subject: [PATCH 2/2] tweaks --- docs/guide/configuration.md | 18 +++++++--------- src/node/config.ts | 42 +++++++++++++++++++------------------ types/shared.d.ts | 4 ++-- 3 files changed, 32 insertions(+), 32 deletions(-) diff --git a/docs/guide/configuration.md b/docs/guide/configuration.md index 4ec740819b8a..8ffaad52fa01 100644 --- a/docs/guide/configuration.md +++ b/docs/guide/configuration.md @@ -24,7 +24,6 @@ module.exports = { Check out the [Config Reference](/config/basics) for a full list of options. - ## Config Intellisense Since VitePress ships with TypeScript typings, you can leverage your IDE's intellisense with jsdoc type hints: @@ -52,30 +51,29 @@ export default defineConfig({ VitePress also directly supports TS config files. You can use `.vitepress/config.ts` with the `defineConfig` helper as well. - ## Typed Theme Config By default, `defineConfig` helper leverages the theme config type from default theme: -```js +```ts import { defineConfig } from 'vitepress' export default defineConfig({ themeConfig: { - // Type is `DefaultTheme.Config` + // Type is `DefaultTheme.Config` } }) ``` -If you use a custom theme, you'll be able to pass the generics type for your custom theme, and you need overload it with the second parameter of `defineConfig` helper: +If you use a custom theme and want type checks for the theme config, you'll need to use `defineConfigWithTheme` instead, and pass the config type for your custom theme via a generic argument: -```js -import { defineConfig } from 'vitepress' +```ts +import { defineConfigWithTheme } from 'vitepress' import { ThemeConfig } from 'your-theme' -export default defineConfig({ +export default defineConfigWithTheme({ themeConfig: { - // Type is `ThemeConfig` + // Type is `ThemeConfig` } -}, true); // declare `usingCustomTheme` and discard usage of the default theme. +}) ``` diff --git a/src/node/config.ts b/src/node/config.ts index 3c82976cc5cf..15ceaf728675 100644 --- a/src/node/config.ts +++ b/src/node/config.ts @@ -15,7 +15,7 @@ import { HeadConfig, LocaleConfig, createLangDictionary, - DefaultTheme, + DefaultTheme } from './shared' import { resolveAliases, APP_PATH, DEFAULT_THEME_PATH } from './alias' import { MarkdownOptions } from './markdown/markdown' @@ -27,16 +27,14 @@ const debug = _debug('vitepress:config') export type { MarkdownOptions } -export type ThemeConfig = any; - -export interface UserConfig { - extends?: RawConfigExports +export interface UserConfig { + extends?: RawConfigExports lang?: string base?: string title?: string description?: string head?: HeadConfig[] - themeConfig?: T + themeConfig?: ThemeConfig locales?: Record markdown?: MarkdownOptions /** @@ -58,15 +56,15 @@ export interface UserConfig { mpa?: boolean } -export type RawConfigExports = - | UserConfig - | Promise> - | (() => UserConfig | Promise>) +export type RawConfigExports = + | UserConfig + | Promise> + | (() => UserConfig | Promise>) -export interface SiteConfig { +export interface SiteConfig { root: string srcDir: string - site: SiteData + site: SiteData configPath: string | undefined themeDir: string outDir: string @@ -85,12 +83,16 @@ const resolve = (root: string, file: string) => /** * Type config helper */ - export function defineConfig( - config: UserConfig, - usingCustomTheme: true -): void -export function defineConfig(config: UserConfig): void -export function defineConfig(config: ThemeConfig) { +export function defineConfig(config: UserConfig) { + return config +} + +/** + * Type config helper for custom theme config + */ +export function defineConfigWithTheme( + config: UserConfig +) { return config } @@ -158,7 +160,7 @@ async function resolveUserConfig( } } - const userConfig: RawConfigExports = configPath + const userConfig: RawConfigExports = configPath ? (( await loadConfigFromFile( { @@ -181,7 +183,7 @@ async function resolveUserConfig( } async function resolveConfigExtends( - config: RawConfigExports + config: RawConfigExports ): Promise { const resolved = await (typeof config === 'function' ? config() : config) if (resolved.extends) { diff --git a/types/shared.d.ts b/types/shared.d.ts index 63fbb624b5bb..2beaf53f2a81 100644 --- a/types/shared.d.ts +++ b/types/shared.d.ts @@ -11,7 +11,7 @@ export interface LocaleConfig { selectText?: string } -export interface SiteData { +export interface SiteData { base: string /** * Language of the site as it should be set on the `html` element. @@ -21,7 +21,7 @@ export interface SiteData { title: string description: string head: HeadConfig[] - themeConfig: T + themeConfig: ThemeConfig locales: Record /** * Available locales for the site when it has defined `locales` in its