import { SourceLocation, Position, ElementNode, NodeTypes, CallExpression, createCallExpression, DirectiveNode, ElementTypes, TemplateChildNode, RootNode, ObjectExpression, Property, JSChildNode, createObjectExpression, SlotOutletNode, TemplateNode, RenderSlotCall, ExpressionNode, IfBranchNode, TextNode, InterpolationNode, VNodeCall } from './ast' import { TransformContext } from './transform' import { MERGE_PROPS, PORTAL, SUSPENSE, KEEP_ALIVE, BASE_TRANSITION } from './runtimeHelpers' import { isString, isFunction, isObject, hyphenate } from '@vue/shared' import { parse } from '@babel/parser' import { Node } from '@babel/types' 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 } } // 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. let _parse: typeof parse let _walk: any export function loadDep(name: string) { if (!__BROWSER__ && typeof process !== 'undefined' && isFunction(require)) { 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] } } export const parseJS: typeof parse = (code, options) => { assert( !__BROWSER__, `Expression AST analysis can only be performed in non-browser builds.` ) if (!_parse) { _parse = loadDep('@babel/parser').parse } return _parse(code, options) } interface Walker { enter?(node: Node, parent: Node): void leave?(node: Node): void } export const walkJS = (ast: Node, walker: Walker) => { assert( !__BROWSER__, `Expression AST analysis can only be performed in non-browser builds.` ) const walk = _walk || (_walk = loadDep('estree-walker').walk) return walk(ast, walker) } const nonIdentifierRE = /^\d|[^\$\w]/ export const isSimpleIdentifier = (name: string): boolean => !nonIdentifierRE.test(name) const memberExpRE = /^[A-Za-z_$][\w$]*(?:\.[A-Za-z_$][\w$]*|\[[^\]]+\])*$/ export const isMemberExpression = (path: string): boolean => memberExpRE.test(path) export function getInnerRange( loc: SourceLocation, offset: number, length?: number ): SourceLocation { __TEST__ && assert(offset <= loc.source.length) const source = loc.source.substr(offset, length) const newLoc: SourceLocation = { source, start: advancePositionWithClone(loc.start, loc.source, offset), end: loc.end } if (length != null) { __TEST__ && assert(offset + length <= loc.source.length) newLoc.end = advancePositionWithClone( loc.start, loc.source, offset + length ) } return newLoc } export function advancePositionWithClone( pos: Position, source: string, numberOfCharacters: number = source.length ): Position { return advancePositionWithMutation({ ...pos }, source, numberOfCharacters) } // advance by mutation without cloning (for performance reasons), since this // gets called a lot in the parser export function advancePositionWithMutation( pos: Position, source: string, numberOfCharacters: number = source.length ): Position { let linesCount = 0 let lastNewLinePos = -1 for (let i = 0; i < numberOfCharacters; i++) { if (source.charCodeAt(i) === 10 /* newline char code */) { linesCount++ lastNewLinePos = i } } pos.offset += numberOfCharacters pos.line += linesCount pos.column = lastNewLinePos === -1 ? pos.column + numberOfCharacters : numberOfCharacters - lastNewLinePos return pos } export function assert(condition: boolean, msg?: string) { /* istanbul ignore if */ if (!condition) { throw new Error(msg || `unexpected compiler condition`) } } export function findDir( node: ElementNode, name: string | RegExp, allowEmpty: boolean = false ): DirectiveNode | undefined { for (let i = 0; i < node.props.length; i++) { const p = node.props[i] if ( p.type === NodeTypes.DIRECTIVE && (allowEmpty || p.exp) && (isString(name) ? p.name === name : name.test(p.name)) ) { return p } } } export function findProp( node: ElementNode, name: string, dynamicOnly: boolean = false ): ElementNode['props'][0] | undefined { for (let i = 0; i < node.props.length; i++) { const p = node.props[i] if (p.type === NodeTypes.ATTRIBUTE) { if (dynamicOnly) continue if (p.name === name && p.value) { return p } } else if (p.name === 'bind' && p.exp && isBindKey(p.arg, name)) { return p } } } export function isBindKey(arg: DirectiveNode['arg'], name: string): boolean { return !!( arg && arg.type === NodeTypes.SIMPLE_EXPRESSION && arg.isStatic && arg.content === name ) } 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] ) } export function isText( node: TemplateChildNode ): node is TextNode | InterpolationNode { return node.type === NodeTypes.INTERPOLATION || node.type === NodeTypes.TEXT } export function isVSlot(p: ElementNode['props'][0]): p is DirectiveNode { return p.type === NodeTypes.DIRECTIVE && p.name === 'slot' } export function isTemplateNode( node: RootNode | TemplateChildNode ): node is TemplateNode { return ( node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.TEMPLATE ) } export function isSlotOutlet( node: RootNode | TemplateChildNode ): node is SlotOutletNode { return node.type === NodeTypes.ELEMENT && node.tagType === ElementTypes.SLOT } export function injectProp( node: VNodeCall | RenderSlotCall, prop: Property, context: TransformContext ) { let propsWithInjection: ObjectExpression | CallExpression const props = node.type === NodeTypes.VNODE_CALL ? node.props : node.arguments[2] if (props == null || isString(props)) { propsWithInjection = createObjectExpression([prop]) } 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])) } propsWithInjection = props } else if (props.type === NodeTypes.JS_OBJECT_EXPRESSION) { let alreadyExists = false // check existing key to avoid overriding user provided keys if (prop.key.type === NodeTypes.SIMPLE_EXPRESSION) { const propKeyName = prop.key.content alreadyExists = props.properties.some( p => p.key.type === NodeTypes.SIMPLE_EXPRESSION && p.key.content === propKeyName ) } if (!alreadyExists) { props.properties.unshift(prop) } propsWithInjection = props } else { // single v-bind with expression, return a merged replacement propsWithInjection = createCallExpression(context.helper(MERGE_PROPS), [ createObjectExpression([prop]), props ]) } if (node.type === NodeTypes.VNODE_CALL) { node.props = propsWithInjection } else { node.arguments[2] = propsWithInjection } } export function toValidAssetId( name: string, type: 'component' | 'directive' ): string { return `_${type}_${name.replace(/[^\w]/g, '_')}` } // 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: case NodeTypes.TEXT_CALL: return hasScopeRef(node.content, ids) case NodeTypes.TEXT: case NodeTypes.COMMENT: return false default: if (__DEV__) { const exhaustiveCheck: never = node exhaustiveCheck } return false } }