Vue3源码 第六篇-JavaScript AST transform函数

news2025/1/10 9:00:32

系列文章目录

Vue3源码 第一篇-总览
Vue3源码 第二篇-Reactive API
Vue3源码 第三篇-Vue3是如何实现响应性
Vue3源码 第四篇-Vue3 setup
Vue3源码 第五篇-Vue3 模版compile AST生成篇


文章目录

  • 系列文章目录
  • 前言
  • 一、transform 转换
  • 二、traverseNode 遍历节点,traverseChildren 遍历子节点
  • 三、ignoreSideEffectTags 忽视副作用的标签
  • 四、transformStyle 转换样式
  • 五、warnTransitionChildren 检查过度动画的节点
  • 六、transformOnce once指令转换
  • 七、transformIf 、processIf if转换
  • 八、transformFor、processFor
  • 九、transformExpression 转换表达式
  • 十、transformSlotOutlet 转换slot
  • 十一、transformElement 转换元素
  • 十二、trackSlotScopes 追踪slot的scopes
  • 十三、transformText 转换文本
  • 总结


前言

上一篇中了解了AST的生成过程,但生成的AST并非最终的AST,Vue3在生成渲染函数之前,对AST进行了进一步的转换和整理,也就是这篇文章的主要内容JavaScriptAST,为什么需要JavaScript AST呢?不是已经生成了AST么?其实可以看到,上一篇我们生成的AST只是对HTML节点的描述信息,但节点如何转变成渲染函数还无法表达,例如文字、元素、IF语句、FOR语句,ON事件等,AST对这些节点不具备直接转换成渲染函数的能力,所以需要进一步通过转换函数,将各类不同的节点进行更加具体的转换。转换函数有两大类型,一种为节点转换类型专门处理节点的转换,一种为指令转换类型来处理节点的属性,如有不对地方希望大家多多指正。下文主要先介绍节点转换函数,不多说了,进入正题,开冲!!!!
在这里插入图片描述


一、transform 转换

转换的入口函数,上一篇讲到的是const ast = isString(template) ? baseParse(template, options) : template,下一步就是将ast转换成JavaScript AST的代码transform函数

function baseCompile(template, options = {}) {
	// 错误处理及判断
   	// ......
   	// 生成语法树
    const ast = isString(template) ? baseParse(template, options) : template;
    // 获取节点转换器和指令转换器,text,element节点,for if等指令转换器,交给转换函数执行转换
    const [nodeTransforms, directiveTransforms] = getBaseTransformPreset();
    // 将ast转换成JavaScript AST
    transform(ast, extend({}, options, {
        prefixIdentifiers,
        nodeTransforms: [
            ...nodeTransforms,
            ...(options.nodeTransforms || []) // user transforms
        ],
        directiveTransforms: extend({}, directiveTransforms, options.directiveTransforms || {} // user transforms
        )
    }));
    // 通过语法树生成
    return generate(ast, extend({}, options, {
        prefixIdentifiers
    }));
}
// 转换入口
function transform(root, options) {
	// 获取转换所需的上下文,里面包括helpers等属性和各类帮助函数,记录转换过程中的信息和转换中所需的各种辅助函数
    const context = createTransformContext(root, options);
    // 对整个节点从根节点开始进行递归,对每一个节点进行转换
    traverseNode(root, context);
    ...
    // 根节点最后需要在进行一次处理
    createRootCodegen(root, context);
    
    ...
}

二、traverseNode 遍历节点,traverseChildren 遍历子节点

通过递归该方法遍历所有的节点,对不同的节点类型进行处理,如果节点中包含子节点就会进入遍历子节点的方法,遍历子节点时会再次调用该方法进行递归调用,之后当前节点不需要继续处理子节点时,执行退出函数对当前节点进行转换,执行完退出函数后返回上一层递归。

function traverseNode(node, context) {
 	// 获取当前节点
    context.currentNode = node;
    // 获取节点转换的函数
    const { nodeTransforms } = context;
    // 退出函数,存放转换的回调函数,只有父节点下的子节点都执行完了,父节点才开始转换
    const exitFns = [];
    // 遍历所有的节点转换函数,下面会介绍这些转换函数
    // 这些转换函数并非直接执行转换,还是对节点进行判断,如果符合转换的要求就返回一个退出函数
    // 等待之后执行转换,if和for的转换还会对节点进行加工等操作
	// 1.ignoreSideEffectTags
	// 2.transformStyle
	// 3.warnTransitionChildren
	// 4.transformOnce
	// 5.transformIf
	// 6.transformFor
	// 7.transformExpression
	// 8.transformSlotOutlet
	// 9.transformElement
	// 10.trackSlotScopes
	// 11.transformText
    for (let i = 0; i < nodeTransforms.length; i++) {
    	// 对当前节点调用每一个转换函数,如果符合转换类型就会返回对应转换的转换函数(退出函数)
    	// 这里并非执行转换的地方,只是将转换函数(退出函数)返回回来,在最后执行,这样保证了执行顺序是先子节点后本节点
        const onExit = nodeTransforms[i](node, context);
        // 如果存在匹配的转换函数就存放起来,等待后续子节点遍历结束后调用
        if (onExit) {
            if (isArray(onExit)) {
                exitFns.push(...onExit);
            }
            else {
                exitFns.push(onExit);
            }
        }
        // 在转换过程中会存在节点改变的情况,比如for/if会替换当前节点。所以这里会对节点进行处理。
        if (!context.currentNode) {
            // node was removed
            return;
        }
        else {
            // node may have been replaced
            node = context.currentNode;
        }
    }
    // 当前节点遍历完转换函数后,根据当前节点的类型,执行不同的分支,这里去掉了注释的分支
    switch (node.type) {
       	...
       	// 插值 {{name}}
        case 5 /* INTERPOLATION */: 
            if (!context.ssr) {
                context.helper(TO_DISPLAY_STRING);
            }
            break;
        // if语句,父执行上边的转换后节点会有改变,这里会对分支节点进行进一步处理
        case 9 /* IF */:
            for (let i = 0; i < node.branches.length; i++) {
                traverseNode(node.branches[i], context);
            }
            break;
        // if的分支节点,for循环节点,元素节点,根节点都会遍历其子节点,对其子节点进一步转换
        case 10 /* IF_BRANCH */:
        case 11 /* FOR */:
        case 1 /* ELEMENT */:
        case 0 /* ROOT */:
            traverseChildren(node, context);
            break;
    }
    // 这里才开始执行真正的转换函数
    // 当前节点无需再遍历子节点时执行本节点前面放入的退出函数
    // 放在最后执行转换函数保证了子节点都转换完毕了再执行父节点的转换
    context.currentNode = node;
    let i = exitFns.length;
    while (i--) {
        exitFns[i]();
    }
}

这里就是对每一个子节点进行traverseNode调用

function traverseChildren(parent, context) {
    let i = 0;
    const nodeRemoved = () => {
        i--;
    };
    for (; i < parent.children.length; i++) {
        const child = parent.children[i];
        if (isString(child))
            continue;
        context.parent = parent;
        // 记录索引部分处理中对节点进行删除等操作时使用到
        context.childIndex = i;
        // 在处理中例如if分支处理时会使用该函数移除节点
        context.onNodeRemoved = nodeRemoved;
        // 调用遍历节点
        traverseNode(child, context);
    }
}

三、ignoreSideEffectTags 忽视副作用的标签

忽略script和style标签

const ignoreSideEffectTags = (node, context) => {
    if (node.type === 1 /* ELEMENT */ &&
        node.tagType === 0 /* ELEMENT */ &&
        (node.tag === 'script' || node.tag === 'style')) {
        context.onError(createDOMCompilerError(59 /* X_IGNORED_SIDE_EFFECT_TAG */, node.loc));
        context.removeNode();
    }
};

四、transformStyle 转换样式

const transformStyle = node => {
	// 如果是元素节点
    if (node.type === 1 /* ELEMENT */) {
    	// 遍历元素节点上的属性值
        node.props.forEach((p, i) => {
        	// 如果存在样式属性,并且有值
            if (p.type === 6 /* ATTRIBUTE */ && p.name === 'style' && p.value) {
                // 转换为表达式描述,将参数和值进行表达式转换
                node.props[i] = {
                    type: 7 /* DIRECTIVE */,
                    name: `bind`,
                    arg: createSimpleExpression(`style`, true, p.loc),
                    exp: parseInlineCSS(p.value.content, p.loc),
                    modifiers: [],
                    loc: p.loc
                };
            }
        });
    }
};

五、warnTransitionChildren 检查过度动画的节点

对过渡动画的子节点进行检查,只允许一个子节点,对for、if节点有单独的判断

const warnTransitionChildren = (node, context) => {
    if (node.type === 1 /* ELEMENT */ &&
        node.tagType === 1 /* COMPONENT */) {
        // 是否是内置节点
        const component = context.isBuiltInComponent(node.tag);
        // 如果节点是transition
        if (component === TRANSITION$1) {
            return () => {
            	// 检查是否有多个子节点
                if (node.children.length && hasMultipleChildren(node)) {
                    context.onError(createDOMCompilerError(58 /* X_TRANSITION_INVALID_CHILDREN */, {
                        start: node.children[0].loc.start,
                        end: node.children[node.children.length - 1].loc.end,
                        source: ''
                    }));
                }
            };
        }
    }
};

六、transformOnce once指令转换

const seen = new WeakSet();
const transformOnce = (node, context) => {
	// 如果是元素节点,就寻找once指令
    if (node.type === 1 /* ELEMENT */ && findDir(node, 'once', true)) {
    	// 如果已经有了就直接返回
        if (seen.has(node)) {
            return;
        }
        seen.add(node);
        // 添加helper
        context.helper(SET_BLOCK_TRACKING);
        // 创建once表达式
        return () => {
            const cur = context.currentNode;
            if (cur.codegenNode) {
                cur.codegenNode = context.cache(cur.codegenNode, true /* isVNode */);
            }
        };
    }
};

七、transformIf 、processIf if转换

if的转换分为2步,第一步是执行processIf函数对if节点进行加工,加工完之后再返回一个退出函数等待子节点执行完毕后进行codegen

const transformIf = createStructuralDirectiveTransform(/^(if|else|else-if)$/, (node, dir, context) => {
    return processIf(node, dir, context, (ifNode, branch, isRoot) => {
    	// 获取兄弟节点
        const siblings = context.parent.children;
        // 获取if节点的位置
        let i = siblings.indexOf(ifNode);
        let key = 0;
        while (i-- >= 0) {
            const sibling = siblings[i];
            if (sibling && sibling.type === 9 /* IF */) {
                key += sibling.branches.length;
            }
        }
        // 返回结束子元素转换后的该节点的处理函数,Codegen会在下一篇中具体讲解
        return () => {
            if (isRoot) {
                ifNode.codegenNode = createCodegenNodeForBranch(branch, key, context);
            }
            else {
                // attach this branch's codegen node to the v-if root.
                const parentCondition = getParentCondition(ifNode.codegenNode);
                parentCondition.alternate = createCodegenNodeForBranch(branch, key + ifNode.branches.length - 1, context);
            }
        };
    });
});
function processIf(node, dir, context, processCodegen) {
	// 如果指令不是else,并且不存在表达式或者表达式内容为空时需要报错,重新设置表达式内容为true
    if (dir.name !== 'else' &&
        (!dir.exp || !dir.exp.content.trim())) {
        const loc = dir.exp ? dir.exp.loc : node.loc;
        context.onError(createCompilerError(27 /* X_V_IF_NO_EXPRESSION */, dir.loc));
        dir.exp = createSimpleExpression(`true`, false, loc);
    }
    // 检查表达式是否符合规范有无关键字等
    if (dir.exp) {
        validateBrowserExpression(dir.exp, context);
    }
    // 如果指令是if
    if (dir.name === 'if') {
    	// 构建if的表达式
        const branch = createIfBranch(node, dir);
        // 这里重新构建if节点,将刚才创建的if分支的表达式放入到分支数组中去,
        // 其他的else-if和else分支也会在之后添加到branches中去
        const ifNode = {
            type: 9 /* IF */,
            loc: node.loc,
            branches: [branch]
        };
        // 对节点进行替换
        // 需要把上下文if节点的父节点素的子节点替换成新的if节点,同时也要替换掉当前的if节点
        // context.parent.children[context.childIndex] = context.currentNode = node;
        context.replaceNode(ifNode);
        // 返回代码生成函数作为结束函数,等待子元素都转换完毕后调用
        if (processCodegen) {
            return processCodegen(ifNode, branch, true);
        }
    }
    // 非if分支
    else {
        // 获取父节点下所有的子节点
        const siblings = context.parent.children;
        const comments = [];
        // 获取当前节点位置
        let i = siblings.indexOf(node);
        while (i-- >= -1) {
        	// 获取兄弟元素 找到相邻的 V-if,如果识别到v-else 和 v-else-if 需要向前寻找v-if
            const sibling = siblings[i];
            // 如果是注释就从父节点中移除,并添加到注释集合中
            if (sibling && sibling.type === 3 /* COMMENT */) {
                context.removeNode(sibling);
                comments.unshift(sibling);
                continue;
            }
            // 空文本也移除出去
            if (sibling &&
                sibling.type === 2 /* TEXT */ &&
                !sibling.content.trim().length) {
                context.removeNode(sibling);
                continue;
            }
            // 如果是if分支就需要移除该节点,并且创建if分支表达式
            if (sibling && sibling.type === 9 /* IF */) {
            	// 删除当前正在traverseNode的节点
                context.removeNode();
                // 创建if分支的对象表达式
                const branch = createIfBranch(node, dir);
                if (comments.length) {
                    branch.children = [...comments, ...branch.children];
                }
                // 检查用户是否在不同的分支上强制使用相同的键
                {
                    const key = branch.userKey;
                    if (key) {
                        sibling.branches.forEach(({ userKey }) => {
                            if (isSameKey(userKey, key)) {
                                context.onError(createCompilerError(28 /* X_V_IF_SAME_KEY */, branch.userKey.loc));
                            }
                        });
                    }
                }
                // 将自身分支添加到if主分支上,因为在上边已经重建构建了if分支节点,这里将else-if 和 else的分支
                // 表达式写入if节点的branches中去
                sibling.branches.push(branch);
                const onExit = processCodegen && processCodegen(sibling, branch, false);
                // 因为节点被移除,需要手动遍历节点,现在的节点类型为10,是IF_BRANCH类型
                traverseNode(branch, context);
                // 因为再次执行traverseNode时,因为node.type已经改为10了,所以无法执行if分支的结束函数,
                // 所以需要这里手动调用
                if (onExit)
                    onExit();
                // 执行完后重置当前节点
                context.currentNode = null;
            }
            // 兄弟节点中不存在v-if就需要报错,每次处理分支都会删除if分支节点,所以一旦出现找不到的情况
            // 就说明没有if节点
            else {
                context.onError(createCompilerError(29 /* X_V_ELSE_NO_ADJACENT_IF */, node.loc));
            }
            break;
        }
    }
}

八、transformFor、processFor

跟if类似都需要先执行加工函数processFor对节点进行加工,之后再返回退出函数。

const transformFor = createStructuralDirectiveTransform('for', (node, dir, context) => {
    const { helper, removeHelper } = context;
    // forNode方法是在执行processFor函数时触发
    return processFor(node, dir, context, forNode => {
        // 创建循环渲染函数表达式,并添加遍历完所有子项后退出时的迭代器
        const renderExp = createCallExpression(helper(RENDER_LIST), [
            forNode.source
        ]);
        // 获取key属性
        const keyProp = findProp(node, `key`);
        const keyProperty = keyProp
            ? createObjectProperty(`key`, keyProp.type === 6 /* ATTRIBUTE */
                ? createSimpleExpression(keyProp.value.content, true)
                : keyProp.exp)
            : null;
        // 是否是稳定的片段    
        const isStableFragment = forNode.source.type === 4 /* SIMPLE_EXPRESSION */ &&
            forNode.source.constType > 0 /* NOT_CONSTANT */;   
        const fragmentFlag = isStableFragment
            ? 64 /* STABLE_FRAGMENT */
            : keyProp
                ? 128 /* KEYED_FRAGMENT */
                : 256 /* UNKEYED_FRAGMENT */;
         // 获取codegenNode,具体的之后的文章讲解
        forNode.codegenNode = createVNodeCall(context, helper(FRAGMENT), undefined, renderExp, fragmentFlag +
            (` /* ${PatchFlagNames[fragmentFlag]} */` ), undefined, undefined, true /* isBlock */, !isStableFragment /* disableTracking */, node.loc);
        return () => {
            // 已经遍历了所有子代,完成代码生成
            let childBlock;
            // 是否是template节点
            const isTemplate = isTemplateNode(node);
            // 获取子节点
            const { children } = forNode;
            // 检查是否把key写在了子节点上
            // <template v-for> key should be placed on the <template> tag
            if (isTemplate) {
                node.children.some(c => {
                    if (c.type === 1 /* ELEMENT */) {
                        const key = findProp(c, 'key');
                        if (key) {
                            context.onError(createCompilerError(32 /* X_V_FOR_TEMPLATE_KEY_PLACEMENT */, key.loc));
                            return true;
                        }
                    }
                });
            }
            // 如果有多个子节点或第一个子节点的类型不是元素(类似于仅包裹了text)
            const needFragmentWrapper = children.length !== 1 || children[0].type !== 1 /* ELEMENT */;
            const slotOutlet = isSlotOutlet(node)
                ? node
                : isTemplate &&
                node.children.length === 1 &&
                isSlotOutlet(node.children[0])
                    ? node.children[0] // api-extractor somehow fails to infer this
                    : null;
            if (slotOutlet) {
                // <slot v-for="..."> or <template v-for="..."><slot/></template>
                childBlock = slotOutlet.codegenNode;
                if (isTemplate && keyProperty) {
                    // <template v-for="..." :key="..."><slot/></template>
                    // we need to inject the key to the renderSlot() call.
                    // the props for renderSlot is passed as the 3rd argument.
                    injectProp(childBlock, keyProperty, context);
                }
            }
            else if (needFragmentWrapper) {
                // <template v-for="..."> 具有文本或多元素
                // 为每个循环生成一个片段块
                childBlock = createVNodeCall(context, helper(FRAGMENT), keyProperty ? createObjectExpression([keyProperty]) : undefined, node.children, 64 /* STABLE_FRAGMENT */ +
                    (` /* ${PatchFlagNames[64 /* STABLE_FRAGMENT */]} */`
                    ), undefined, undefined, true);
            }
            else {
                // 普通的v-for,直接使用子的codegenNode
                // but mark it as a block.
                childBlock = children[0]
                    .codegenNode;
                if (isTemplate && keyProperty) {
                    injectProp(childBlock, keyProperty, context);
                }
                if (childBlock.isBlock !== !isStableFragment) {
                    if (childBlock.isBlock) {
                        // switch from block to vnode
                        removeHelper(OPEN_BLOCK);
                        removeHelper(CREATE_BLOCK);
                    }
                    else {
                        // switch from vnode to block
                        removeHelper(CREATE_VNODE);
                    }
                }
                childBlock.isBlock = !isStableFragment;
                if (childBlock.isBlock) {
                    helper(OPEN_BLOCK);
                    helper(CREATE_BLOCK);
                }
                else {
                    helper(CREATE_VNODE);
                }
            }
            renderExp.arguments.push(createFunctionExpression(createForLoopParams(forNode.parseResult), childBlock, true /* force newline */));
        };
    });
});
function processFor(node, dir, context, processCodegen) {
	// 如果不存在表达式
    if (!dir.exp) {
        context.onError(createCompilerError(30 /* X_V_FOR_NO_EXPRESSION */, dir.loc));
        return;
    }
    // 只能是简单的表达式,因为在表达式转换之前应用了 vFor 转换。
    const parseResult = parseForExpression( 
        dir.exp, context);
    if (!parseResult) {
        context.onError(createCompilerError(31 /* X_V_FOR_MALFORMED_EXPRESSION */, dir.loc));
        return;
    }
    const { addIdentifiers, removeIdentifiers, scopes } = context;
    // 获取源,值得别名,键的别名,index的别名
    const { source, value, key, index } = parseResult;
    const forNode = {
        type: 11 /* FOR */,
        loc: dir.loc,
        source,
        valueAlias: value,
        keyAlias: key,
        objectIndexAlias: index,
        parseResult,
        children: isTemplateNode(node) ? node.children : [node]
    };
    // 和if一样也要替换当前节点
    context.replaceNode(forNode);
    // 记录vFor
    scopes.vFor++;
    // 执行获取codegen代码,作为退出函数
    const onExit = processCodegen && processCodegen(forNode);
    // 返回for节点的退出函数
    return () => {
        scopes.vFor--;
        if (onExit)
            onExit();
    };
}

九、transformExpression 转换表达式

对表达式进行检查,确保表达式里不包含关键字,无论是插值还是指令中是否包含非法的关键字,比如{{let a}} {{for}}

const transformExpression = (node, context) => {
	// 如果是插值的话
    if (node.type === 5 /* INTERPOLATION */) {
    	// 验证表达式是否没问题,是否含有关键字等
        node.content = processExpression(node.content, context);
    }
    // 如果是元素节点
    else if (node.type === 1 /* ELEMENT */) {
        // handle directives on element
        // 获取节点行的属性
        for (let i = 0; i < node.props.length; i++) {
        	// 获取属性
            const dir = node.props[i];
            // do not process for v-on & v-for since they are special handled
            // 7是指令类型,这里过滤了for指令,for指令有单独的处理器
            if (dir.type === 7 /* DIRECTIVE */ && dir.name !== 'for') {
            	// 获取表达式
                const exp = dir.exp;
                // 获取参数
                const arg = dir.arg;
                // do not process exp if this is v-on:arg - we need special handling
                // for wrapping inline statements.
                if (exp &&
                    exp.type === 4 /* SIMPLE_EXPRESSION */ &&
                    !(dir.name === 'on' && arg)) {
                    dir.exp = processExpression(exp, context,
                        // slot args must be processed as function params
                        dir.name === 'slot');
                }
                if (arg && arg.type === 4 /* SIMPLE_EXPRESSION */ && !arg.isStatic) {
                    dir.arg = processExpression(arg, context);
                }
            }
        }
    }
};

十、transformSlotOutlet 转换slot

const transformSlotOutlet = (node, context) => {
	// 判断是否是slot节点
    if (isSlotOutlet(node)) {
    	
        const { children, loc } = node;
        const { slotName, slotProps } = processSlotOutlet(node, context);
        const slotArgs = [
            context.prefixIdentifiers ? `_ctx.$slots` : `$slots`,
            slotName
        ];
        if (slotProps) {
            slotArgs.push(slotProps);
        }
        if (children.length) {
            if (!slotProps) {
                slotArgs.push(`{}`);
            }
            slotArgs.push(createFunctionExpression([], children, false, false, loc));
        }
        if (context.scopeId && !context.slotted) {
            if (!slotProps) {
                slotArgs.push(`{}`);
            }
            if (!children.length) {
                slotArgs.push(`undefined`);
            }
            slotArgs.push(`true`);
        }
        node.codegenNode = createCallExpression(context.helper(RENDER_SLOT), slotArgs, loc);
    }
};

十一、transformElement 转换元素

对所有元素标签进行转换,包括html标签以及组件,下面省略的部分属性和特殊处理代码,重点看普通html标签的处理过程

const transformElement = (node, context) => {

    return function postTransformElement() {
    	// 如果不是元素或组件就直接退出,比如文字会直接退出,if,for等节点转换后type都不为1,所以会跳过
		node = context.currentNode;
        if (!(node.type === 1 /* ELEMENT */ &&
            (node.tagType === 0 /* ELEMENT */ ||
                node.tagType === 1 /* COMPONENT */))) {
            return;
        }
    
        // tag 为 html元素的标签,props为属性值比如id,on(事件)等
        const { tag, props } = node;
       
       	...
       	
        let vnodePatchFlag;
        let patchFlag = 0;
        
     	...
       
        // props
        // 构建属性值
        if (props.length > 0) {
        	// 构建属性,并且根据属性计算patchFlag
            const propsBuildResult = buildProps(node, context);
            // 获取属性值
            vnodeProps = propsBuildResult.props;
            // 获取修补flag
            patchFlag = propsBuildResult.patchFlag;
            // 获取动态节点属性
            dynamicPropNames = propsBuildResult.dynamicPropNames;
            const directives = propsBuildResult.directives;
            vnodeDirectives =
                directives && directives.length
                    ? createArrayExpression(directives.map(dir => buildDirectiveArgs(dir, context)))
                    : undefined;
        }
        // children
        if (node.children.length > 0) {
            ...
            // 这里省略了keep-alive的处理
            ...
            ...
            // 省略组件相关的处理
            ...
            // 如果只有一个子节点,并且不是TELEPORT
            else if (node.children.length === 1 && vnodeTag !== TELEPORT) {
                // 获取子节点
                const child = node.children[0];
                const type = child.type;
                // 子节点是否是动态文字,看类型是否是插值,或者是插值和静态的组合
                // {{name}} 这种是type为5的插值类型, hello {{name}} 会在text转换中转换为 type为8的组合表达式
                // 这两种均为动态的子节点
                const hasDynamicTextChild = type === 5 /* INTERPOLATION */ ||
                    type === 8 /* COMPOUND_EXPRESSION */;
                // 
                if (hasDynamicTextChild &&
                    // 获取常量类型,更新patchFlag
                    getConstantType(child, context) === 0 /* NOT_CONSTANT */) {
                    // patchFlag = patchFlag | 1 patchFlag 与 0001 按照位或运算,为之后patch过程做优化
                    patchFlag |= 1 /* TEXT */;
                }
                // pass directly if the only child is a text node
                // (plain / interpolation / expression)
                if (hasDynamicTextChild || type === 2 /* TEXT */) {
                    vnodeChildren = child;
                }
                else {
                    vnodeChildren = node.children;
                }
            }
            else {
                vnodeChildren = node.children;
            }
        }
        // 动态节点
        if (patchFlag !== 0) {
            {
                if (patchFlag < 0) {
                    // special flags (negative and mutually exclusive)
                    vnodePatchFlag = patchFlag + ` /* ${PatchFlagNames[patchFlag]} */`;
                }
                else {
                    // bitwise flags 按位获取pathch flag 
                    // PatchFlagNames 为以下值
                    // [1 /* TEXT */]: `TEXT`,
        			// [2 /* CLASS */]: `CLASS`,
        			// [4 /* STYLE */]: `STYLE`,
			        // [8 /* PROPS */]: `PROPS`,
			        // [16 /* FULL_PROPS */]: `FULL_PROPS`,
			        // [32 /* HYDRATE_EVENTS */]: `HYDRATE_EVENTS`,
			        // [64 /* STABLE_FRAGMENT */]: `STABLE_FRAGMENT`,
			        // [128 /* KEYED_FRAGMENT */]: `KEYED_FRAGMENT`,
			        // [256 /* UNKEYED_FRAGMENT */]: `UNKEYED_FRAGMENT`,
			        // [512 /* NEED_PATCH */]: `NEED_PATCH`,
			        // [1024 /* DYNAMIC_SLOTS */]: `DYNAMIC_SLOTS`,
			        // [2048 /* DEV_ROOT_FRAGMENT */]: `DEV_ROOT_FRAGMENT`,
			        // [-1 /* HOISTED */]: `HOISTED`,
			        // [-2 /* BAIL */]: `BAIL`
			        // 这里其实就是获取到底该元素有多少种patchFlag,上面合并patchFlag用的是|运算,这里解的时候
			        // 用&运算,就知道上面到底合并了多少种
			        // 如果patchFlag为9 从上表中我们能知道是1+8也就是text和props的组合 1000 | 0001 = 1001 (9)
			        // 我们拆解时做 & 运算 1001 & 0001 = 0001 所以得到 这里包含 text 
			        // 1001 & 1000 = 1000 如果和class做运算 1001 & 0010 = 0000 这里filter条件就为false
			        // 所以就通过这种巧妙地方式进行合并和拆解。
                    const flagNames = Object.keys(PatchFlagNames)
                        .map(Number)
                        .filter(n => n > 0 && patchFlag & n)
                        .map(n => PatchFlagNames[n])
                        .join(`, `);
                    vnodePatchFlag = patchFlag + ` /* ${flagNames} */`;
                }
            }
            if (dynamicPropNames && dynamicPropNames.length) {
                vnodeDynamicProps = stringifyDynamicPropNames(dynamicPropNames);
            }
        }
        // 创建生成node节点的对象
        // 在这里添加type,patchFlag,isBlock等属性
        node.codegenNode = createVNodeCall(context, vnodeTag, vnodeProps, vnodeChildren, vnodePatchFlag, vnodeDynamicProps, vnodeDirectives, !!shouldUseBlock, false /* disableTracking */, node.loc);
    };
};

十二、trackSlotScopes 追踪slot的scopes

这个暂时还不知道什么情况会使用,有了解的欢迎解答

const trackSlotScopes = (node, context) => {
    if (node.type === 1 /* ELEMENT */ &&
        (node.tagType === 1 /* COMPONENT */ ||
            node.tagType === 3 /* TEMPLATE */)) {
        // We are only checking non-empty v-slot here
        // since we only care about slots that introduce scope variables.
        const vSlot = findDir(node, 'slot');
        if (vSlot) {
            vSlot.exp;
            context.scopes.vSlot++;
            return () => {
                context.scopes.vSlot--;
            };
        }
    }
};

十三、transformText 转换文本

主要处理含有文本节点的父节点,需要对子文本节点进行组合,比如hello {{name}} 在AST中生成的节点是分开的2个节点,但JavaScript AST会将2个节点进行合并。

const transformText = (node, context) => {
    if (node.type === 0 /* ROOT */ ||
        node.type === 1 /* ELEMENT */ ||
        node.type === 11 /* FOR */ ||
        node.type === 10 /* IF_BRANCH */) {
        // perform the transform on node exit so that all expressions have already
        // been processed.
        return () => {
            // 获取ast该节点下的子节点
            const children = node.children;
            let currentContainer = undefined;
            let hasText = false;
            // 遍历子节点
            for (let i = 0; i < children.length; i++) {
                // 存储子节点,后面需要用
                const child = children[i];
                // 判断node.type为5或者为2就是text类型,5是插值,2是文本
                if (isText(child)) {
                    hasText = true;
                    // 循环拼接
                    for (let j = i + 1; j < children.length; j++) {
                        // 先获取下一个子节点
                        const next = children[j];
                        // 如果下一个子节点是文本
                        if (isText(next)) {
                            // 不存在当前容器,第一次循环为undefined
                            if (!currentContainer) {
                                // 构建当前容器,重新组装该节点,将type改为复合表达式(8),将自身放入新节点的子节点中
                                currentContainer = children[i] = {
                                    type: 8 /* COMPOUND_EXPRESSION */,
                                    loc: child.loc,
                                    children: [child]
                                };
                            }
                            // merge adjacent text node into current
                            // 将下一节点加入到容器的子节点中
                            currentContainer.children.push(` + `, next);
                            // 把下一个节点删除
                            children.splice(j, 1);
                            // 减一,在删除了开始节点后面的节点后,splice已改变了数组长度,这样需要j--才可以获取到对的下一个节点
                            j--;
                        }
                        else {
                            // 如果存在节点不是text就清空当前容器,并且退出
                            currentContainer = undefined;
                            break;
                        }
                    }
                }
            }
            // 如果内部不存在纯文本或插值,或者已经合并好了并且为根节点或html标签节点
            if (!hasText ||
                // if this is a plain element with a single text child, leave it
                // as-is since the runtime has dedicated fast path for this by directly
                // setting textContent of the element.
                // for component root it's always normalized anyway.
                (children.length === 1 &&
                    (node.type === 0 /* ROOT */ ||
                        (node.type === 1 /* ELEMENT */ &&
                            node.tagType === 0 /* ELEMENT */)))) {
                return;
            }
            // pre-convert text nodes into createTextVNode(text) calls to avoid
            // runtime normalization.
            // 当前的children已经是合并和删减完的children了,里面包含文本和插值的合并,以及其他无法合并的节点
            for (let i = 0; i < children.length; i++) {
                // 获取第一个子节点
                const child = children[i];
                // 子节点是文字或插槽,或者是已经合并好的复合表达式
                if (isText(child) || child.type === 8 /* COMPOUND_EXPRESSION */) {
                    const callArgs = [];
                    // createTextVNode defaults to single whitespace, so if it is a
                    // single space the code could be an empty call to save bytes.
                    // 如果节点不是纯文本或者节点内容不为空格
                    if (child.type !== 2 /* TEXT */ || child.content !== ' ') {
                        callArgs.push(child);
                    }
                    // mark dynamic text with flag so it gets patched inside a block
                    // 用标记标记动态文本,以便它在块中得到修补
                    if (!context.ssr &&
                        getConstantType(child, context) === 0 /* NOT_CONSTANT */) {
                        callArgs.push(1 /* TEXT */ +
                            (` /* ${PatchFlagNames[1 /* TEXT */]} */` ));
                    }
                    children[i] = {
                        type: 12 /* TEXT_CALL */,
                        content: child,
                        loc: child.loc,
                        codegenNode: createCallExpression(context.helper(CREATE_TEXT), callArgs)
                    };
                }
            }
        };
    }
};

总结

本篇文章简单介绍了transform的过程中的11个转换函数,以traverseNode为入口,通过递归调用遍历每一个节点,对每一个节点调用这11个转换函数,对符合要求的节点返回退出函数,在每一个节点退出时调用退出函数对节点进行codegen,本篇没有具体介绍codegen相关的代码,我们会在下一个篇章详细介绍有关codegen的代码。在最后如果本文有不正确的地方欢迎指正,希望大家多多点赞支持。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/475941.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

B/S结构系统的会话机制(session)

B/S结构系统的会话机制(session) 文章目录 B/S结构系统的会话机制(session)每博一文案1. session 会话机制的概述2. 什么是 session 的会话3. session 的作用4. session 的实现原理解释5. 补充&#xff1a; Cookie禁用了&#xff0c;session还能找到吗 &#xff1f;6. 总结一下…

PCA学习

前置知识 统计 假设数据集 X ∈ R n m \mathbf{X}\in\mathbb{R}^{n\times m} X∈Rnm,其中 n n n表示样本数量&#xff0c; m m m表示特征个数 均值 X ˉ 1 n e T X 1 n ∑ i 1 n X i \bar{\mathbf{X}} \frac{1}{n}\mathbf{e}^T\mathbf{X} \frac{1}{n} \sum_{i1}^{n}\mat…

[架构之路-178]-《软考-系统分析师》- 分区操作系统(Partition Operating System)概述

目录&#xff1a; 本文概述&#xff1a; 1.1 什么是分区操作系统 1.2 分区操作系统出现背景 1. 前后台系统(Foreground/Background System) 2. 实时操作系统(RTOS) 本文概述&#xff1a; 随着嵌入式系统日趋复杂化以及对安全性要求的不断提高&#xff0c;采用空间隔离、时…

java学习之枚举二

目录 一、enum关键字实现枚举 二、注意事项 一、对Season2进行反编译&#xff08;javap&#xff09; ​编辑 三、练习题 第一题 第二题 一、enum关键字实现枚举 package enum_;public class Enumeration03 {public static void main(String[] args) {System.out.println…

el-upload组件的文件回显功能和添加文件到elupload组件

省流&#xff1a; 先获取这个文件对象&#xff0c;使用handleStart方法添加到组件。 this.$refs.uploadRefName.handleStart(rawfile); 在开发的时候遇到表单里需要上传图片的功能。看了下el-upload组件的使用方法&#xff0c;在修改表单的时候&#xff0c;el-upload组件的回显…

flutter学习之旅(二)

如果不知道怎么安装编写可以查看这篇 创建项目 另一个创建方法 flutter create 项目名热部署 vscode 热部署 vscode很简单&#xff1a;可以通过Debug进行调试 使用flutter查看设备 flutter devices如图所见我现在用的是windows所以&#xff0c;我们检测不到ios因为 我们看…

【Fluent】边界类型总结,什么时候用壁面对(wall-shadow pair)、什么时候用interface?

一、fluent自动生成边界类型的规律 Enclosure是包裹在外面的气体&#xff08;流体&#xff09;&#xff0c;mold是模具&#xff08;固体&#xff09;&#xff0c;sheet是模具上的薄板件&#xff08;固体&#xff09;。 1.1 正向思维 不管是流体域还是固体域&#xff0c;每一…

高度可定制可用于商用目的全流程供应链系统(全部源码)

一、开源项目简介 高度可定制零售供应链中台基础系统&#xff0c;集成零售管理, 电子商务, 供应链管理, 财务管理, 车队管理, 仓库管理, 人员管理, 产品管理, 订单管理, 会员管理, 连锁店管理, 加盟管理, 前端React/Ant Design, 后端Java Spring自有开源框架&#xff0c;全面支…

软件测试之黑盒测试的具体方法详解

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一.基于需求的设计方法二.等价类三.边界值四.判定表4.1 **关系**4.2 如何设计测试用例4.3 实际案例第一步第二步第三步第四步 五.正交排列5.1 什么是正交表5.2 …

shell脚本的判断式

文章目录 shell脚本的判断式利用test命令的测试功能关于某个文件名的【文件类型】判断关于文件的权限检测两个文件之间的比较关于两个整数之间的比较判定字符串的数据多重条件判定例题 利用判断符号[ ]例题 shell脚本的默认变量($0、$1...)例题shift&#xff1a;造成参数变量号…

Linux安装Mysql操作步骤详解

目录 1. 检测当前系统中是否安装了MySql数据库 2. 使用FinalShell自带的上传工具将jdk的二进制发布包上传到Linux 3. 解压并解包到/usr/local/mysql&#xff08;便于区分&#xff09; 第一步&#xff1a;将包先移动到该目录下 第二步&#xff1a;解压解包 第三步&#xff1a…

LeetCode 105.106. 从前序|后序与中序遍历序列构造二叉树 | C++语言版

LeetCode 105. 从前序与中序遍历序列构造二叉树 | C语言版 LeetCode 105. 从前序与中序遍历序列构造二叉树题目描述解题思路思路一&#xff1a;使用递归代码实现运行结果参考文章&#xff1a; 思路二&#xff1a;减少遍历节点数代码实现运行结果参考文章&#xff1a; LeetCode …

C语言复习笔记1

1.不同数据类型所占字节数。 bit 01二进制的比特位 byte 字节 8 bit 比特 之后的单位都是以1024为倍数 #include<stdio.h> #include<unistd.h>int main() {printf("sizeof(char)%d\n",sizeof(char));printf("sizeof(short)%d\n",sizeof(sh…

JavaScript 笔记

1 简介 JavaScript 诞生于1995年&#xff0c;是由网景公司发明&#xff0c;起初命名为LiveScript&#xff0c;后来由于SUN公司的介入&#xff0c;更名为 JavaScript。1996年微软在其最新的IE3浏览器中引入了自己对JavaScript的实现JScript&#xff0c;于是市面上就存在两个版本…

Mybatis 框架 ( 三 ) Mybatis-Plus

4.Mybatis-plus 官网 : https://www.baomidou.com/ MyBatis-Plus 是一个 MyBatis 的增强工具&#xff0c;在 MyBatis 的基础上封装了大量常规操作&#xff0c;减少了SQL的编写量。 4.1.Maven依赖 使用时通常通过Springboot框架整合使用 并且使用Lombok框架简化实体类 <…

软件测试——基础篇(软件测试的生命周期和BUG的概念)

目录 一、软件测试生命周期 1. 软件生命周期 2. 软件测试生命周期 二、BUG 1. 如何描述一个BUG 2. BUG的级别 3. BUG的生命周期 一、软件测试生命周期 1. 软件生命周期 软件生命周期&#xff1a;需求分析&#xff0c;计划&#xff0c;设计&#xff0c;编码&#xff0c;…

20 printf 的调试

前言 在最开始的 cmd 编程中, 我们会使用到的最常见的输出, 包括一些时候调试的时候 我们最常使用到的函数 那肯定是 printf 了 我们这里来调试一下 这个 printf 还有一个原因是 之前在调试 malloc 的时候, malloc 虚拟内存分配的调试(1) 可以发现, 不仅仅是在 malloc 的时…

Codeforces Round 865 (Div. 2)

6 problems. ABC过, DE没想出来, F没看. https://codeforces.com/contest/1816 A. Ian Visits Mary 分析 - AC 每次跳跃&#xff0c;横纵互质。 限于数据量&#xff0c;不能枚举。 1与任何数互质。考虑从(0,0)跳到(1,y)&#xff0c;这一步一定合法&#xff1b;再从(1,y)跳到…

Netty小白入门教程

一、概述 1.1 概念 Netty是一个异步的基于事件驱动(即多路复用技术)的网络应用框架&#xff0c;用于快速开发可维护、高性能的网络服务器和客户端。 1.2 地位 Netty在Java网络应用框架中的地位就好比&#xff0c;Spring框架在JavaEE开发中的地位。 以下的框架都使用了Nett…

C++017-C++文件读写应用

文章目录 C017-C文件读写应用C文件读写应用CSP-J目标1. 文件的基本概念、文本文件的基本操作关闭文件 文件操作-写入文本文件2.文件读写操作基本案例seekg() 和 tellg() 函数来读取文件中的数据 2. 文本文件类型与二进制文件类型3. 文件重定向、文件读写等操作 在练习&#xff…