当模块为“npm link”时,Vue 3 组件错误初始化

Jus*_*dei 4 javascript webpack vue.js vuejs3

以下是我的库的入口点,它生成一个带有动态标签的组件:

// muvement.js

import { defineComponent, ref, onMounted, h } from 'vue';

const createMuvement = (tag) => {
    return defineComponent({
        name: `m-${tag}`,
        setup(props, context) {
            const root = ref(null);
            onMounted(() => {
                console.log(root.value);
            });
            return () => h(tag, { ...context.attrs, ref: root }, context.slots);
        }
    });
};

const muvement = (...tags) => {
    const components = {};
    tags.map((tag) => (components[`m-${tag}`] = createMuvement(tag)));
    return components;
};

export { muvement };
Run Code Online (Sandbox Code Playgroud)

预计会像这样消耗:

// Home.vue

<template>
  <div>
    <m-div>div</m-div>
    <m-button>button</m-button>
  </div>
</template>

<script>
import { muvement } from "muvement";

export default {
  name: "Home",
  components: {
    ...muvement("div", "button")
  }
};
</script>
Run Code Online (Sandbox Code Playgroud)

当库代码包含在 Vue 应用程序文件夹中时,这将按预期工作(假设我们现在"@/components/muvement.js"从而不是导入"movement")。
那是:

-muvement-test-project (scaffolded with vue-cli)
    - src
        - views
            - Home.vue
        - components
            - muvement.js
Run Code Online (Sandbox Code Playgroud)

我还发布了一个 alpha 版本,"muvement"直接从 npm 注册表(即,npm install muvement而不是npm link muvement)安装后导入时工作正常。

问题

在开发过程中,我想要一个应用程序来测试库,该库与库的目录分开。

我曾经npm link将库链接到测试应用程序(就像我过去对许多其他项目所做的那样)。

从/路径/到/库

-muvement-test-project (scaffolded with vue-cli)
    - src
        - views
            - Home.vue
        - components
            - muvement.js
Run Code Online (Sandbox Code Playgroud)

从/path/到/test/app

$ npm link
Run Code Online (Sandbox Code Playgroud)

到目前为止,一切都很好。该模块可作为测试应用程序node_modules文件夹中的符号链接使用。所以我import { muvement } from "muvement"跑了npm run serve,然后……轰隆隆。
一切都会爆炸(参见下面的错误)。还可能值得注意的是,尝试从完整路径(即C:/dev/npm/muvment/dist/es/index.js)导入会导致相同的问题npm link,因此我认为它与符号链接没有直接关系。

这是控制台中显示的内容:

Chrome 开发工具中的警告

几乎一整天我都在试图解决这个问题。我见过几个看似相似的问题,通过将 Webpack 设置resolve.symlinks为 来解决false,但这对我的问题没有影响。我已经阅读了所有文档甚至 Vue 的源代码(对于那些好奇的人来说,这是有问题的行)。

由于警告表明该错误通常归因于async setup我认为 webpack 可能做了一些奇怪的事情,这会使我的代码异步。情况似乎并非如此,因为工作尝试和失败尝试的调用堆栈是相同的。

不同的是范围。

以下是正在运行的示例的范围:
在此输入图像描述

这是失败的: 在此输入图像描述

(请注意,该target参数是null在调用期间injectHook,这显然是提示 Vue 显示警告的原因)。

我的问题是,为什么导入模块的位置在该模块的执行过程中会产生如此大的差异?

库代码和构建设置可在此处获取:
https://github.com/justintaddei/muvement

测试应用程序可在此处获取:
https://github.com/justintaddei/muvement/tree/example

如果我遗漏了一些重要的内容,请在评论中告诉我。这是漫长的一天,所以我确信我可能错过了一些东西

谢谢。

pik*_*kax 5

问题是你的应用程序在底层使用了两个不同的 vue 依赖项 - vue 需要使用相同的依赖项来跟踪反应性、生命周期等。

当您链接库时,npm/yarn 将使用该链接的文件夹node_modules,但您app正在使用它的依赖项node_modules

当您app导入时,vue它会消失app/node_modules/vue,但是当您从链接的依赖项导入时,它会消失linked_dep/node_modules/vue

app
  node_modules
    vue
linked library
  node_modules    
    vue
Run Code Online (Sandbox Code Playgroud)

调试此问题的一种简单方法是使用 a 更改两个 vue 依赖文件,console.log并检查控制台是否正在记录这两个文件。