当模块为 npm link 时,Vue 3 组件未正确初始化

Vue 3 component incorrectly initialized when module is `npm link`ed

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

// 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 };

预计会这样消费:

// 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>

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

-muvement-test-project (scaffolded with vue-cli)
    - src
        - views
            - Home.vue
        - components
            - muvement.js

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

问题

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

我已经使用 npm link 到 link 测试应用程序的库(正如我过去对许多其他项目所做的那样)。

来自/path/to/library

$ npm link

来自/path/to/test/app

$ npm link muvement

到目前为止一切顺利。该模块在测试应用程序的 node_modules 文件夹中作为 symlink 提供。所以我 import { muvement } from "muvement", 运行 npm run serve, 和... BOOM.
一切都爆炸了(见下面的错误)。还可能值得注意的是,尝试从完整路径(即 C:/dev/npm/muvment/dist/es/index.js)导入会导致与 npm link 相同的问题,因此我认为它与 symlink直接。

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

几乎一整天我都在努力解决这个问题。我已经看到几个看似相似的问题,通过将 Webpack 的 resolve.symlinks 设置为 false 解决了,但这对我的问题没有影响。我已经通读了 docs and even Vue's source code (here is the offending line for those who are curious).

由于警告表明该错误通常归因于 async setup,我想也许 webpack 正在做一些奇怪的事情,这会使我的代码异步。这似乎不是这种情况,因为工作尝试和失败尝试的调用堆栈是相同的。

不同的是范围。

这是工作示例的范围:

这是失败的:

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

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

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

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

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

谢谢。

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

当您 link 库 npm/yarn 将使用 linked 文件夹 node_modules,但您的 app 正在使用它的依赖项 node_modules.

当您的 app 导入 vue 时,它将变为 app/node_modules/vue,但是当您从 linked 依赖项中导入时,它将变为 linked_dep/node_modules/vue

app
  node_modules
    vue
linked library
  node_modules    
    vue

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