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

标签 javascript vue.js webpack 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 };

预计会这样消费:

// 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 链接运动)。

问题

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

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

从/path/to/library

$ npm link

从/path/to/test/app

$ npm link muvement

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

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

warning in Chrome devtools

几乎一整天我都在尝试解决这个问题。我已经看到几个看似相似的问题,通过将 Webpack 的 resolve.symlinks 设置为 false 解决了,但这对我的问题没有影响。我已经通读了 docs甚至 Vue 的源代码 ( here is the offending line for those who are curious )。

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

不同的是范围。

这是工作示例的范围:
enter image description here

这是失败的: enter image description here

(注意在调用injectHooktarget参数为null,这显然是提示Vue显示警告的原因)。

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

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

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

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

谢谢。

最佳答案

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

当你链接一个库时,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

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

关于javascript - 当模块为 `npm link` 时,Vue 3 组件未正确初始化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65501715/

相关文章:

javascript - html5 动画的自定义控件(洗涤器播放停止)

javascript - 如何在非 vue 网络应用程序中使用 Vuejs 网络组件作为输入元素?

reactjs - 如何解决此错误您可能需要适当的加载程序来处理此文件类型

javascript - babel-minify vs terser(而不是 uglify-js)

javascript - 我可以手动将调度添加到组件的 props 而不是 connect() 吗?

javascript - AngularJS 路由页面未显示

javascript - 类型错误 : Cannot read property 'props' of undefined In a React function

javascript - 如何在 Vue.js 中将驼峰式大小写转换为正确大小写?

javascript - vue 中的通用组件

Webpack:链接加载器