开发者问题收集

TypeError:无法读取 null 的属性(读取“isCE”) - 自定义组件库

2022-04-28
19891

我在使用 ViteJS 和 NPM 为 Vue 3 构建自定义组件库时遇到了麻烦。我在下面提供了我的问题的基本说明,有人可以告诉我我做错了什么或指出我正确的方向吗,我已经被这个问题困扰了 2 天 :(。

我的文件夹结构:

  • dist
  • node_modules
  • src
    • components
      • Paragraph.vue
    • paragraph.js
  • .gitignore
  • package.json
  • README.md
  • vite.config.js

package.json

{
  "name": "paragraph",
  "private": true,
  "version": "0.0.0",
  "description": "The paragraph test component.",
  "main": "./dist/paragraph.umd.js",
  "module": "./dist/paragraph.es.js",
  "exports": {
    ".": {
      "import": "./dist/paragraph.es.js",
      "require": "./dist/paragraph.umd.js"
    },
    "./dist/style.css": "./dist/style.css"
  },
  "scripts": {
    "dev": "vite",
    "build": "vite build",
    "preview": "vite preview"
  },
  "dependencies": {
    "vue": "^3.2.25"
  },
  "devDependencies": {
    "@vitejs/plugin-vue": "^2.3.1",
    "vite": "^2.9.5"
  }
}

vite.config.js

import { fileURLToPath, URL } from 'url'
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'

// https://vitejs.dev/config/
export default defineConfig({
  build: {
    lib: {
      entry: fileURLToPath(new URL('./src/paragraph.js', import.meta.url)),
      name: 'Paragraph',
      fileName: (format) => `paragraph.${format}.js`,
    },
    rollupOptions: {
      external: ['vue'],
      output: {
        globals: {
          vue: 'Vue'
        },
      },
    },
  },
  plugins: [vue()],
  resolve: {
    alias: {
      '@': fileURLToPath(new URL('./src', import.meta.url))
    },
  },
})

paragraph.js

import Paragraph from './components/Paragraph.vue';

export default {
  install: (app) => {
    app.component('Paragraph', Paragraph);
  },
};

Paragraph.vue

<script setup>
  console.log('Test');
</script>

<template>
  <p class="paragraph">
    <slot />
  </p>
</template>

<style>
  .paragraph
  {
      color: black;
  }
</style>

何时我运行 npm run build ,它成功运行并创建了正确的文件,然后我将 es 文件作为插件包含到我的测试 Vue 项目中。

import { createApp } from 'vue'
import App from './App.vue'
import Paragraph from '../../paragraph/dist/paragraph.es.js'

createApp(App).use(Paragraph).mount('#app')

这样使用时,组件不起作用。

<Paragraph>Hello World 2!</Paragraph>

控制台中报告以下错误。

TypeError: Cannot read properties of null (reading 'isCE')

我调查了这个问题,似乎很多人都遇到了同样的问题,尽管我自己找不到解决办法。

我已尝试以下链接中提到的解决方案:

https://github.com/vuejs/core/issues/4344

将自制的 Vue 3 库导入 Vue 3 项目时:“未捕获的类型错误:无法读取 null 的属性(读取‘isCE’)”

这里提到的两种解决方案都不起作用。

有人可以帮忙吗!!!

我注意到,如果我排除 <slot /> ,它可以正常工作,但插槽对组件至关重要。

我知道它将 Vue 代码捆绑到构建文件中,但我该如何阻止它这样做呢。

提前致谢。

3个回答

我也遇到过这个非常令人沮丧的问题。根据 这个答案 ,这是由于 Vue 从多个包导入而不是像您怀疑的那样只使用一个单例而导致的。

据推测,您正在使用 Vite 构建您的消费者应用程序。在这种情况下,在其 vite.confg.js 中设置 dedupe 选项应该可以解决问题。

resolve: {
  dedupe: [
    'vue'
  ]
},
Greegus
2022-05-03

请参考 Library Mode 文档。确保您的 vite.config.js 如下所示:

build: {
    lib: {
      // Could also be a dictionary or array of multiple entry points
      entry: resolve(__dirname, 'lib/main.js'),
      name: 'MyLib',
      // the proper extensions will be added
      fileName: 'my-lib',
    },
    rollupOptions: {
      // make sure to externalize deps that shouldn't be bundled
      // into your library
      external: ['vue'],
      output: {
        // Provide global variables to use in the UMD build
        // for externalized deps
        globals: {
          vue: 'Vue',
        },
      },
    },
  },
Keygun2k1
2023-04-20

问题是有多个 vue 实例加载了不同的版本。

就我遇到的错误而言,问题是模块解析时加载了依赖项,例如,Vue 在直接项目中的依赖项是 3.3.7 版本。 另一个包(例如 public-ui-fw )使用 vue 版本 3.3.6。

在这种情况下,包管理器正在安装。

package.json
node_modules/
    vue/
        package.json # this is version 3.3.7
    public-ui-fw/
        package.json
        node_modules/
            vue/
                package.json # this is version 3.3.6

因此 public-ui-fw 正在使用不同的 vue 实例。


解决方案,

我强制包管理器在所有传递依赖项中仅安装 1 个版本。

取决于您使用的包管理器,请参考:

tanapoln
2023-10-28