types: (wip) improve dts output

This commit is contained in:
Evan You 2019-11-01 11:32:53 -04:00
parent 06c5b0a861
commit 5eee1152ca
9 changed files with 53 additions and 31 deletions

View File

@ -16,7 +16,8 @@ import {
createObjectProperty, createObjectProperty,
ForCodegenNode, ForCodegenNode,
ElementCodegenNode, ElementCodegenNode,
SlotOutletCodegenNode SlotOutletCodegenNode,
SlotOutletNode
} from '../ast' } from '../ast'
import { createCompilerError, ErrorCodes } from '../errors' import { createCompilerError, ErrorCodes } from '../errors'
import { import {
@ -119,7 +120,7 @@ export const transformFor = createStructuralDirectiveTransform(
: isTemplate && : isTemplate &&
node.children.length === 1 && node.children.length === 1 &&
isSlotOutlet(node.children[0]) isSlotOutlet(node.children[0])
? node.children[0] ? (node.children[0] as SlotOutletNode) // api-extractor somehow fails to infer this
: null : null
const keyProperty = keyProp const keyProperty = keyProp
? createObjectProperty( ? createObjectProperty(

View File

@ -190,18 +190,23 @@ export function createBlockExpression(
]) ])
} }
export const isVSlot = (p: ElementNode['props'][0]): p is DirectiveNode => export function isVSlot(p: ElementNode['props'][0]): p is DirectiveNode {
p.type === NodeTypes.DIRECTIVE && p.name === 'slot' return p.type === NodeTypes.DIRECTIVE && p.name === 'slot'
}
export const isTemplateNode = ( export function isTemplateNode(
node: RootNode | TemplateChildNode node: RootNode | TemplateChildNode
): node is TemplateNode => ): node is TemplateNode {
return (
node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.TEMPLATE node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.TEMPLATE
)
}
export const isSlotOutlet = ( export function isSlotOutlet(
node: RootNode | TemplateChildNode node: RootNode | TemplateChildNode
): node is SlotOutletNode => ): node is SlotOutletNode {
node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.SLOT return node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.SLOT
}
export function injectProp( export function injectProp(
node: ElementCodegenNode | ComponentCodegenNode | SlotOutletCodegenNode, node: ElementCodegenNode | ComponentCodegenNode | SlotOutletCodegenNode,

View File

@ -1,16 +1,10 @@
import { ComponentInternalInstance, currentInstance } from './component' import { ComponentInternalInstance, currentInstance } from './component'
import { import { VNode, NormalizedChildren, normalizeVNode, VNodeChild } from './vnode'
VNode,
NormalizedChildren,
normalizeVNode,
VNodeChild,
VNodeChildren
} from './vnode'
import { isArray, isFunction } from '@vue/shared' import { isArray, isFunction } from '@vue/shared'
import { ShapeFlags } from './shapeFlags' import { ShapeFlags } from './shapeFlags'
import { warn } from './warning' import { warn } from './warning'
export type Slot = (...args: any[]) => VNodeChildren export type Slot = (...args: any[]) => VNode[]
export type InternalSlots = { export type InternalSlots = {
[name: string]: Slot [name: string]: Slot

View File

@ -30,7 +30,7 @@ export { PublicPatchFlags as PatchFlags } from '@vue/shared'
export { getCurrentInstance } from './component' export { getCurrentInstance } from './component'
// For custom renderers // For custom renderers
export { createRenderer } from './createRenderer' export { createRenderer, RootRenderFunction } from './createRenderer'
export { warn } from './warning' export { warn } from './warning'
export { export {
handleError, handleError,

View File

@ -135,7 +135,7 @@ export const KeepAlive = {
return () => { return () => {
if (!slots.default) { if (!slots.default) {
return return null
} }
const children = slots.default() const children = slots.default()

View File

@ -25,9 +25,12 @@ export const Portal = Symbol(__DEV__ ? 'Portal' : undefined)
export const Text = Symbol(__DEV__ ? 'Text' : undefined) export const Text = Symbol(__DEV__ ? 'Text' : undefined)
export const Comment = Symbol(__DEV__ ? 'Comment' : undefined) export const Comment = Symbol(__DEV__ ? 'Comment' : undefined)
const Suspense = (__FEATURE_SUSPENSE__ // Export as {} to avoid circular type dependency between `suspense.ts` and
? SuspenseImpl // `createRenderer.ts` in exported types.
: null) as typeof SuspenseImpl // A circular type dependency causes tsc to generate d.ts with dynmaic import()
// calls using realtive paths, which works for separate d.ts files, but will
// fail after d.ts rollup with API Extractor.
const Suspense = (__FEATURE_SUSPENSE__ ? SuspenseImpl : null) as {}
export { Suspense } export { Suspense }
export type VNodeTypes = export type VNodeTypes =

View File

@ -13,7 +13,8 @@
"sideEffects": false, "sideEffects": false,
"buildOptions": { "buildOptions": {
"name": "VueDOMRuntime", "name": "VueDOMRuntime",
"formats": ["esm", "cjs", "global", "esm-browser"] "formats": ["esm", "cjs", "global", "esm-browser"],
"dts": ["jsx.d.ts"]
}, },
"repository": { "repository": {
"type": "git", "type": "git",

View File

@ -1,16 +1,24 @@
import { createRenderer, warn } from '@vue/runtime-core' import {
createRenderer,
warn,
App,
RootRenderFunction
} from '@vue/runtime-core'
import { nodeOps } from './nodeOps' import { nodeOps } from './nodeOps'
import { patchProp } from './patchProp' import { patchProp } from './patchProp'
// Importing from the compiler, will be tree-shaken in prod // Importing from the compiler, will be tree-shaken in prod
import { isHTMLTag, isSVGTag } from '@vue/compiler-dom' import { isHTMLTag, isSVGTag } from '@vue/compiler-dom'
import { isFunction, isString } from '@vue/shared' import { isFunction, isString } from '@vue/shared'
const { render, createApp: baseCreateApp } = createRenderer<Node, Element>({ const { render: baseRender, createApp: baseCreateApp } = createRenderer({
patchProp, patchProp,
...nodeOps ...nodeOps
}) })
const createApp = () => { // use explicit type casts here to avoid import() calls in rolled-up d.ts
export const render = baseRender as RootRenderFunction<Node, Element>
export const createApp = (): App<Element> => {
const app = baseCreateApp() const app = baseCreateApp()
if (__DEV__) { if (__DEV__) {
@ -48,8 +56,6 @@ const createApp = () => {
return app return app
} }
export { render, createApp }
// DOM-only runtime helpers // DOM-only runtime helpers
export { export {
vModelText, vModelText,

View File

@ -27,6 +27,7 @@ const targets = args._
const formats = args.formats || args.f const formats = args.formats || args.f
const devOnly = args.devOnly || args.d const devOnly = args.devOnly || args.d
const prodOnly = !devOnly && (args.prodOnly || args.p) const prodOnly = !devOnly && (args.prodOnly || args.p)
const buildTypes = args.t || args.types
const buildAllMatching = args.all || args.a const buildAllMatching = args.all || args.a
const lean = args.lean || args.l const lean = args.lean || args.l
const commit = execa.sync('git', ['rev-parse', 'HEAD']).stdout.slice(0, 7) const commit = execa.sync('git', ['rev-parse', 'HEAD']).stdout.slice(0, 7)
@ -68,7 +69,7 @@ async function build(target) {
`NODE_ENV:${env}`, `NODE_ENV:${env}`,
`TARGET:${target}`, `TARGET:${target}`,
formats ? `FORMATS:${formats}` : ``, formats ? `FORMATS:${formats}` : ``,
args.types ? `TYPES:true` : ``, buildTypes ? `TYPES:true` : ``,
prodOnly ? `PROD_ONLY:true` : ``, prodOnly ? `PROD_ONLY:true` : ``,
lean ? `LEAN:true` : `` lean ? `LEAN:true` : ``
] ]
@ -78,7 +79,7 @@ async function build(target) {
{ stdio: 'inherit' } { stdio: 'inherit' }
) )
if (args.types && pkg.types) { if (buildTypes && pkg.types) {
console.log() console.log()
console.log( console.log(
chalk.bold(chalk.yellow(`Rolling up type definitions for ${target}...`)) chalk.bold(chalk.yellow(`Rolling up type definitions for ${target}...`))
@ -97,6 +98,17 @@ async function build(target) {
}) })
if (result.succeeded) { if (result.succeeded) {
// concat additional d.ts to rolled-up dts (mostly for JSX)
if (pkg.buildOptions.dts) {
const dtsPath = path.resolve(pkgDir, pkg.types)
const existing = await fs.readFile(dtsPath, 'utf-8')
const toAdd = await Promise.all(
pkg.buildOptions.dts.map(file => {
return fs.readFile(path.resolve(pkgDir, file), 'utf-8')
})
)
await fs.writeFile(dtsPath, existing + '\n' + toAdd.join('\n'))
}
console.log( console.log(
chalk.bold(chalk.green(`API Extractor completed successfully.`)) chalk.bold(chalk.green(`API Extractor completed successfully.`))
) )