2019-10-02 17:11:07 +00:00
|
|
|
import {
|
|
|
|
SourceLocation,
|
|
|
|
Position,
|
|
|
|
ElementNode,
|
|
|
|
NodeTypes,
|
|
|
|
CallExpression,
|
2019-10-02 21:18:11 +00:00
|
|
|
createCallExpression,
|
2019-10-03 03:10:41 +00:00
|
|
|
DirectiveNode,
|
|
|
|
ElementTypes,
|
|
|
|
TemplateChildNode,
|
2019-10-03 16:03:14 +00:00
|
|
|
RootNode,
|
|
|
|
ObjectExpression,
|
|
|
|
Property,
|
|
|
|
JSChildNode,
|
2019-10-06 02:47:20 +00:00
|
|
|
createObjectExpression,
|
|
|
|
SlotOutletNode,
|
2019-10-08 14:50:00 +00:00
|
|
|
TemplateNode,
|
2020-02-11 23:12:56 +00:00
|
|
|
RenderSlotCall,
|
2019-10-19 01:51:34 +00:00
|
|
|
ExpressionNode,
|
2020-02-03 03:28:54 +00:00
|
|
|
IfBranchNode,
|
|
|
|
TextNode,
|
2020-02-11 23:12:56 +00:00
|
|
|
InterpolationNode,
|
|
|
|
VNodeCall
|
2019-10-02 17:11:07 +00:00
|
|
|
} from './ast'
|
|
|
|
import { TransformContext } from './transform'
|
2019-11-29 17:42:04 +00:00
|
|
|
import {
|
|
|
|
MERGE_PROPS,
|
|
|
|
PORTAL,
|
|
|
|
SUSPENSE,
|
|
|
|
KEEP_ALIVE,
|
|
|
|
BASE_TRANSITION
|
|
|
|
} from './runtimeHelpers'
|
|
|
|
import { isString, isFunction, isObject, hyphenate } from '@vue/shared'
|
2020-02-27 21:53:51 +00:00
|
|
|
import { parse } from '@babel/parser'
|
|
|
|
import { Node } from '@babel/types'
|
2019-11-29 17:42:04 +00:00
|
|
|
|
|
|
|
export const isBuiltInType = (tag: string, expected: string): boolean =>
|
|
|
|
tag === expected || tag === hyphenate(expected)
|
|
|
|
|
|
|
|
export function isCoreComponent(tag: string): symbol | void {
|
|
|
|
if (isBuiltInType(tag, 'Portal')) {
|
|
|
|
return PORTAL
|
|
|
|
} else if (isBuiltInType(tag, 'Suspense')) {
|
|
|
|
return SUSPENSE
|
|
|
|
} else if (isBuiltInType(tag, 'KeepAlive')) {
|
|
|
|
return KEEP_ALIVE
|
|
|
|
} else if (isBuiltInType(tag, 'BaseTransition')) {
|
|
|
|
return BASE_TRANSITION
|
|
|
|
}
|
|
|
|
}
|
2019-09-30 17:13:32 +00:00
|
|
|
|
|
|
|
// cache node requires
|
|
|
|
// lazy require dependencies so that they don't end up in rollup's dep graph
|
|
|
|
// and thus can be tree-shaken in browser builds.
|
2019-10-02 15:05:56 +00:00
|
|
|
let _parse: typeof parse
|
2020-02-27 21:53:51 +00:00
|
|
|
let _walk: any
|
2019-09-30 17:13:32 +00:00
|
|
|
|
2019-10-04 21:43:20 +00:00
|
|
|
export function loadDep(name: string) {
|
2019-12-17 22:34:10 +00:00
|
|
|
if (!__BROWSER__ && typeof process !== 'undefined' && isFunction(require)) {
|
2019-10-04 17:08:06 +00:00
|
|
|
return require(name)
|
|
|
|
} else {
|
|
|
|
// This is only used when we are building a dev-only build of the compiler
|
|
|
|
// which runs in the browser but also uses Node deps.
|
|
|
|
return (window as any)._deps[name]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-12 23:33:48 +00:00
|
|
|
export const parseJS: typeof parse = (code, options) => {
|
2019-09-30 17:13:32 +00:00
|
|
|
assert(
|
|
|
|
!__BROWSER__,
|
|
|
|
`Expression AST analysis can only be performed in non-browser builds.`
|
|
|
|
)
|
2020-02-27 21:53:51 +00:00
|
|
|
if (!_parse) {
|
|
|
|
_parse = loadDep('@babel/parser').parse
|
|
|
|
}
|
|
|
|
return _parse(code, options)
|
|
|
|
}
|
|
|
|
|
|
|
|
interface Walker {
|
|
|
|
enter?(node: Node, parent: Node): void
|
|
|
|
leave?(node: Node): void
|
2019-09-30 17:13:32 +00:00
|
|
|
}
|
|
|
|
|
2020-02-27 21:53:51 +00:00
|
|
|
export const walkJS = (ast: Node, walker: Walker) => {
|
2019-09-30 17:13:32 +00:00
|
|
|
assert(
|
|
|
|
!__BROWSER__,
|
|
|
|
`Expression AST analysis can only be performed in non-browser builds.`
|
|
|
|
)
|
2019-10-04 17:08:06 +00:00
|
|
|
const walk = _walk || (_walk = loadDep('estree-walker').walk)
|
2019-09-30 17:13:32 +00:00
|
|
|
return walk(ast, walker)
|
|
|
|
}
|
2019-09-19 17:23:49 +00:00
|
|
|
|
2019-10-10 15:15:24 +00:00
|
|
|
const nonIdentifierRE = /^\d|[^\$\w]/
|
2019-09-25 02:39:20 +00:00
|
|
|
export const isSimpleIdentifier = (name: string): boolean =>
|
2019-10-10 15:15:24 +00:00
|
|
|
!nonIdentifierRE.test(name)
|
|
|
|
|
|
|
|
const memberExpRE = /^[A-Za-z_$][\w$]*(?:\.[A-Za-z_$][\w$]*|\[[^\]]+\])*$/
|
|
|
|
export const isMemberExpression = (path: string): boolean =>
|
|
|
|
memberExpRE.test(path)
|
2019-09-25 02:39:20 +00:00
|
|
|
|
2019-09-19 17:23:49 +00:00
|
|
|
export function getInnerRange(
|
|
|
|
loc: SourceLocation,
|
|
|
|
offset: number,
|
|
|
|
length?: number
|
|
|
|
): SourceLocation {
|
2019-11-15 22:29:08 +00:00
|
|
|
__TEST__ && assert(offset <= loc.source.length)
|
2019-09-19 17:23:49 +00:00
|
|
|
const source = loc.source.substr(offset, length)
|
|
|
|
const newLoc: SourceLocation = {
|
|
|
|
source,
|
2019-09-20 03:05:51 +00:00
|
|
|
start: advancePositionWithClone(loc.start, loc.source, offset),
|
2019-09-19 17:23:49 +00:00
|
|
|
end: loc.end
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length != null) {
|
2019-11-15 22:29:08 +00:00
|
|
|
__TEST__ && assert(offset + length <= loc.source.length)
|
2019-09-20 03:05:51 +00:00
|
|
|
newLoc.end = advancePositionWithClone(
|
|
|
|
loc.start,
|
|
|
|
loc.source,
|
|
|
|
offset + length
|
|
|
|
)
|
2019-09-19 17:23:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return newLoc
|
|
|
|
}
|
|
|
|
|
2019-09-20 03:05:51 +00:00
|
|
|
export function advancePositionWithClone(
|
2019-09-19 17:23:49 +00:00
|
|
|
pos: Position,
|
|
|
|
source: string,
|
2019-09-25 23:17:45 +00:00
|
|
|
numberOfCharacters: number = source.length
|
2019-09-19 17:23:49 +00:00
|
|
|
): Position {
|
2019-09-19 17:59:24 +00:00
|
|
|
return advancePositionWithMutation({ ...pos }, source, numberOfCharacters)
|
|
|
|
}
|
2019-09-19 17:23:49 +00:00
|
|
|
|
2019-09-19 17:59:24 +00:00
|
|
|
// advance by mutation without cloning (for performance reasons), since this
|
|
|
|
// gets called a lot in the parser
|
|
|
|
export function advancePositionWithMutation(
|
|
|
|
pos: Position,
|
|
|
|
source: string,
|
2019-09-25 23:17:45 +00:00
|
|
|
numberOfCharacters: number = source.length
|
2019-09-19 17:59:24 +00:00
|
|
|
): Position {
|
2019-09-20 01:18:18 +00:00
|
|
|
let linesCount = 0
|
|
|
|
let lastNewLinePos = -1
|
|
|
|
for (let i = 0; i < numberOfCharacters; i++) {
|
|
|
|
if (source.charCodeAt(i) === 10 /* newline char code */) {
|
|
|
|
linesCount++
|
|
|
|
lastNewLinePos = i
|
|
|
|
}
|
|
|
|
}
|
2019-09-19 17:23:49 +00:00
|
|
|
|
2019-09-19 17:59:24 +00:00
|
|
|
pos.offset += numberOfCharacters
|
2019-09-20 01:18:18 +00:00
|
|
|
pos.line += linesCount
|
2019-09-19 17:59:24 +00:00
|
|
|
pos.column =
|
2019-09-20 01:18:18 +00:00
|
|
|
lastNewLinePos === -1
|
2019-09-19 17:23:49 +00:00
|
|
|
? pos.column + numberOfCharacters
|
2019-12-30 16:25:06 +00:00
|
|
|
: numberOfCharacters - lastNewLinePos
|
2019-09-19 17:23:49 +00:00
|
|
|
|
2019-09-19 17:59:24 +00:00
|
|
|
return pos
|
2019-09-19 17:23:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function assert(condition: boolean, msg?: string) {
|
2019-09-24 20:35:01 +00:00
|
|
|
/* istanbul ignore if */
|
2019-09-19 17:23:49 +00:00
|
|
|
if (!condition) {
|
2019-09-22 20:50:57 +00:00
|
|
|
throw new Error(msg || `unexpected compiler condition`)
|
2019-09-19 17:23:49 +00:00
|
|
|
}
|
|
|
|
}
|
2019-10-02 03:19:48 +00:00
|
|
|
|
2019-10-02 22:03:42 +00:00
|
|
|
export function findDir(
|
2019-10-02 21:18:11 +00:00
|
|
|
node: ElementNode,
|
2019-10-02 22:03:42 +00:00
|
|
|
name: string | RegExp,
|
|
|
|
allowEmpty: boolean = false
|
2019-10-02 21:18:11 +00:00
|
|
|
): DirectiveNode | undefined {
|
|
|
|
for (let i = 0; i < node.props.length; i++) {
|
|
|
|
const p = node.props[i]
|
|
|
|
if (
|
|
|
|
p.type === NodeTypes.DIRECTIVE &&
|
2019-10-02 22:03:42 +00:00
|
|
|
(allowEmpty || p.exp) &&
|
2019-10-03 03:10:41 +00:00
|
|
|
(isString(name) ? p.name === name : name.test(p.name))
|
2019-10-02 21:18:11 +00:00
|
|
|
) {
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-02 03:19:48 +00:00
|
|
|
export function findProp(
|
2019-10-02 21:18:11 +00:00
|
|
|
node: ElementNode,
|
2019-10-16 16:00:55 +00:00
|
|
|
name: string,
|
|
|
|
dynamicOnly: boolean = false
|
2019-10-02 03:19:48 +00:00
|
|
|
): ElementNode['props'][0] | undefined {
|
2019-10-02 21:18:11 +00:00
|
|
|
for (let i = 0; i < node.props.length; i++) {
|
|
|
|
const p = node.props[i]
|
2019-10-02 03:19:48 +00:00
|
|
|
if (p.type === NodeTypes.ATTRIBUTE) {
|
2019-10-16 16:00:55 +00:00
|
|
|
if (dynamicOnly) continue
|
2019-10-24 19:39:31 +00:00
|
|
|
if (p.name === name && p.value) {
|
2019-10-02 03:19:48 +00:00
|
|
|
return p
|
|
|
|
}
|
2020-02-06 04:07:23 +00:00
|
|
|
} else if (p.name === 'bind' && p.exp && isBindKey(p.arg, name)) {
|
2019-10-02 03:19:48 +00:00
|
|
|
return p
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-02 17:11:07 +00:00
|
|
|
|
2020-02-06 04:07:23 +00:00
|
|
|
export function isBindKey(arg: DirectiveNode['arg'], name: string): boolean {
|
|
|
|
return !!(
|
|
|
|
arg &&
|
|
|
|
arg.type === NodeTypes.SIMPLE_EXPRESSION &&
|
|
|
|
arg.isStatic &&
|
|
|
|
arg.content === name
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-02-05 20:21:47 +00:00
|
|
|
export function hasDynamicKeyVBind(node: ElementNode): boolean {
|
|
|
|
return node.props.some(
|
|
|
|
p =>
|
|
|
|
p.type === NodeTypes.DIRECTIVE &&
|
|
|
|
p.name === 'bind' &&
|
|
|
|
(!p.arg || // v-bind="obj"
|
|
|
|
p.arg.type !== NodeTypes.SIMPLE_EXPRESSION || // v-bind:[_ctx.foo]
|
|
|
|
!p.arg.isStatic) // v-bind:[foo]
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2020-02-03 03:28:54 +00:00
|
|
|
export function isText(
|
|
|
|
node: TemplateChildNode
|
|
|
|
): node is TextNode | InterpolationNode {
|
|
|
|
return node.type === NodeTypes.INTERPOLATION || node.type === NodeTypes.TEXT
|
|
|
|
}
|
|
|
|
|
2019-11-01 15:32:53 +00:00
|
|
|
export function isVSlot(p: ElementNode['props'][0]): p is DirectiveNode {
|
|
|
|
return p.type === NodeTypes.DIRECTIVE && p.name === 'slot'
|
|
|
|
}
|
2019-10-03 03:10:41 +00:00
|
|
|
|
2019-11-01 15:32:53 +00:00
|
|
|
export function isTemplateNode(
|
2019-10-03 03:10:41 +00:00
|
|
|
node: RootNode | TemplateChildNode
|
2019-11-01 15:32:53 +00:00
|
|
|
): node is TemplateNode {
|
|
|
|
return (
|
|
|
|
node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.TEMPLATE
|
|
|
|
)
|
|
|
|
}
|
2019-10-03 16:03:14 +00:00
|
|
|
|
2019-11-01 15:32:53 +00:00
|
|
|
export function isSlotOutlet(
|
2019-10-03 16:03:14 +00:00
|
|
|
node: RootNode | TemplateChildNode
|
2019-11-01 15:32:53 +00:00
|
|
|
): node is SlotOutletNode {
|
|
|
|
return node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.SLOT
|
|
|
|
}
|
2019-10-03 16:03:14 +00:00
|
|
|
|
|
|
|
export function injectProp(
|
2020-02-11 23:12:56 +00:00
|
|
|
node: VNodeCall | RenderSlotCall,
|
2019-10-03 16:03:14 +00:00
|
|
|
prop: Property,
|
|
|
|
context: TransformContext
|
2019-10-08 14:50:00 +00:00
|
|
|
) {
|
|
|
|
let propsWithInjection: ObjectExpression | CallExpression
|
|
|
|
const props =
|
2020-02-11 23:12:56 +00:00
|
|
|
node.type === NodeTypes.VNODE_CALL ? node.props : node.arguments[2]
|
2019-10-08 14:50:00 +00:00
|
|
|
if (props == null || isString(props)) {
|
|
|
|
propsWithInjection = createObjectExpression([prop])
|
2019-10-03 16:03:14 +00:00
|
|
|
} else if (props.type === NodeTypes.JS_CALL_EXPRESSION) {
|
|
|
|
// merged props... add ours
|
|
|
|
// only inject key to object literal if it's the first argument so that
|
|
|
|
// if doesn't override user provided keys
|
|
|
|
const first = props.arguments[0] as string | JSChildNode
|
|
|
|
if (!isString(first) && first.type === NodeTypes.JS_OBJECT_EXPRESSION) {
|
|
|
|
first.properties.unshift(prop)
|
|
|
|
} else {
|
|
|
|
props.arguments.unshift(createObjectExpression([prop]))
|
|
|
|
}
|
2019-10-08 14:50:00 +00:00
|
|
|
propsWithInjection = props
|
2019-10-03 16:03:14 +00:00
|
|
|
} else if (props.type === NodeTypes.JS_OBJECT_EXPRESSION) {
|
2020-01-20 15:15:53 +00:00
|
|
|
let alreadyExists = false
|
|
|
|
// check existing key to avoid overriding user provided keys
|
|
|
|
if (prop.key.type === NodeTypes.SIMPLE_EXPRESSION) {
|
|
|
|
const propKeyName = prop.key.content
|
2020-02-03 03:28:54 +00:00
|
|
|
alreadyExists = props.properties.some(
|
|
|
|
p =>
|
|
|
|
p.key.type === NodeTypes.SIMPLE_EXPRESSION &&
|
|
|
|
p.key.content === propKeyName
|
|
|
|
)
|
2020-01-20 15:15:53 +00:00
|
|
|
}
|
|
|
|
if (!alreadyExists) {
|
|
|
|
props.properties.unshift(prop)
|
|
|
|
}
|
2019-10-08 14:50:00 +00:00
|
|
|
propsWithInjection = props
|
2019-10-03 16:03:14 +00:00
|
|
|
} else {
|
|
|
|
// single v-bind with expression, return a merged replacement
|
2019-10-08 14:50:00 +00:00
|
|
|
propsWithInjection = createCallExpression(context.helper(MERGE_PROPS), [
|
2019-10-03 16:03:14 +00:00
|
|
|
createObjectExpression([prop]),
|
|
|
|
props
|
|
|
|
])
|
|
|
|
}
|
2020-02-11 23:12:56 +00:00
|
|
|
if (node.type === NodeTypes.VNODE_CALL) {
|
|
|
|
node.props = propsWithInjection
|
2019-10-08 14:50:00 +00:00
|
|
|
} else {
|
2020-02-11 23:12:56 +00:00
|
|
|
node.arguments[2] = propsWithInjection
|
2019-10-08 14:50:00 +00:00
|
|
|
}
|
2019-10-03 16:03:14 +00:00
|
|
|
}
|
2019-10-05 21:18:25 +00:00
|
|
|
|
|
|
|
export function toValidAssetId(
|
|
|
|
name: string,
|
|
|
|
type: 'component' | 'directive'
|
|
|
|
): string {
|
2019-10-10 17:55:26 +00:00
|
|
|
return `_${type}_${name.replace(/[^\w]/g, '_')}`
|
2019-10-05 21:18:25 +00:00
|
|
|
}
|
2019-10-10 14:33:58 +00:00
|
|
|
|
2019-10-19 01:51:34 +00:00
|
|
|
// Check if a node contains expressions that reference current context scope ids
|
|
|
|
export function hasScopeRef(
|
|
|
|
node: TemplateChildNode | IfBranchNode | ExpressionNode | undefined,
|
|
|
|
ids: TransformContext['identifiers']
|
|
|
|
): boolean {
|
|
|
|
if (!node || Object.keys(ids).length === 0) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
switch (node.type) {
|
|
|
|
case NodeTypes.ELEMENT:
|
|
|
|
for (let i = 0; i < node.props.length; i++) {
|
|
|
|
const p = node.props[i]
|
|
|
|
if (
|
|
|
|
p.type === NodeTypes.DIRECTIVE &&
|
|
|
|
(hasScopeRef(p.arg, ids) || hasScopeRef(p.exp, ids))
|
|
|
|
) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.FOR:
|
|
|
|
if (hasScopeRef(node.source, ids)) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.IF:
|
|
|
|
return node.branches.some(b => hasScopeRef(b, ids))
|
|
|
|
case NodeTypes.IF_BRANCH:
|
|
|
|
if (hasScopeRef(node.condition, ids)) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return node.children.some(c => hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.SIMPLE_EXPRESSION:
|
|
|
|
return (
|
|
|
|
!node.isStatic &&
|
|
|
|
isSimpleIdentifier(node.content) &&
|
|
|
|
!!ids[node.content]
|
|
|
|
)
|
|
|
|
case NodeTypes.COMPOUND_EXPRESSION:
|
|
|
|
return node.children.some(c => isObject(c) && hasScopeRef(c, ids))
|
|
|
|
case NodeTypes.INTERPOLATION:
|
2019-10-21 19:52:29 +00:00
|
|
|
case NodeTypes.TEXT_CALL:
|
2019-10-19 01:51:34 +00:00
|
|
|
return hasScopeRef(node.content, ids)
|
2019-10-21 19:52:29 +00:00
|
|
|
case NodeTypes.TEXT:
|
|
|
|
case NodeTypes.COMMENT:
|
|
|
|
return false
|
2019-10-19 01:51:34 +00:00
|
|
|
default:
|
2019-10-21 19:52:29 +00:00
|
|
|
if (__DEV__) {
|
|
|
|
const exhaustiveCheck: never = node
|
|
|
|
exhaustiveCheck
|
|
|
|
}
|
2019-10-19 01:51:34 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|