Vue + TSX 中使用 class 报错 解决方法

遇到一个问题:

问题回顾

在 Vue 3 + TSX 项目中,可能会这样写:

html 复制代码
<div class="container" style={{ height: '100px' }}>Hello</div>

TypeScript 会提示:

属性"class"在类型"DetailedHTMLProps<HTMLAttributes<HTMLDivElement>, HTMLDivElement>"上不存在。你是否指的是"className"?

这在普通 Vue 模板中完全合法,但在 .tsx 文件中,TypeScript 的类型定义默认认为:

  • JSX 中只允许使用 className,不认识 class;
  • 这是因为 JSX 的类型系统最早源自 React,它沿用了 className 语法;
  • Vue 虽然在运行时支持 class,但 TS 类型声明中有时缺少该属性,尤其在自定义 jsx 模式下。

于是,TypeScript 就会报错提示你把 class 改为 className。

**本质:**是类型安全校验阻止把一个具有 class 字段的 props 对象赋给 JSX.IntrinsicElements['div']。运行时代码不一定会失败(取决于运行时是否接受该属性),但类型层面会阻止编译或 IDE 的类型检查。

解决方法

方法一:修改代码

在 TSX 中统一使用 className(Vue TSX 也支持 className)。对代码风格友好,类型正确。

方法二:添加声明文件(侧重)
  1. src/types 目录下新增一个文件,例如:

src/types/shim-jsx-class.d.ts

javascript 复制代码
// src/types/shim-jsx-class.d.ts
import 'vue';

declare module 'vue' {
  interface HTMLAttributes {
    /**
     * 允许在 TSX/JSX 中使用 `class` 属性(兼容 string / object / array)
     * - string: "a b"
     * - Record<string, boolean>: { active: true }
     * - Array<string | Record<string, boolean>>
     */
    class?: string | Record<string, boolean> | Array<string | Record<string, boolean>>;

    /** 也允许 className(可选,但推荐保留) */
    className?: string | Record<string, boolean> | Array<string | Record<string, boolean>>;
  }
}

这段代码的意思是:扩展 Vue 内部的 HTMLAttributes 类型,为它添加一个 class 属性定义,从而让 TypeScript 在 TSX 中识别它。

  1. 修改 tsconfig.json,确保 TypeScript 能识别这个声明文件。

打开项目根目录的 tsconfig.json,在 include 中加上 src/types:

javascript 复制代码
{
  "compilerOptions": {
    "target": "ESNext",
    "module": "ESNext",
    "jsx": "preserve",
    "strict": true,
    "esModuleInterop": true,
    "skipLibCheck": true
  },
  "include": ["src", "src/types/**/*.d.ts"]
}

如果 include 已经包含 src,通常不用改。确保声明文件路径在 include 范围内。

⚠️ 注意:skipLibCheck 不是必须,但在部分场景下能避免第三方类型冲突。

  1. 重启编辑器 / TypeScript 服务

VSCode 用户可执行命令:TypeScript: Restart TS Server 或者直接重启编辑器。

此时重新打开 .tsx 文件,报错应该已经消失。

为什么这能解决问题?

Vue TSX 的类型定义(在 vue 包内)通过 HTMLAttributes 控制 HTML 元素的可用属性。默认声明没有 class,所以 TSX 编译器认为它不存在。

通过 declare module 'vue' {} 的方式"合并声明",为 HTMLAttributes 增加 class 字段。TypeScript 会自动合并这部分扩展声明,达到了"类型补丁"的效果。

这不会影响 Vue 的运行逻辑,仅扩展了静态类型检查范围。

最后:Vue 在 TSX 环境中实际上同时兼容 class 与 className,但推荐保持一致风格:在 TSX 中依然用 class,不要混用 className。

相关推荐
专注前端30年17 小时前
Vue2 中 v-if 与 v-show 深度对比及实战指南
开发语言·前端·vue
专注前端30年1 天前
【Vue2】基础知识汇总与实战指南
开发语言·前端·vue
麦麦大数据2 天前
F039 python五种算法美食推荐可视化大数据系统vue+flask前后端分离架构
python·算法·vue·推荐算法·美食·五种算法
java水泥工3 天前
课程答疑系统|基于SpringBoot和Vue的课程答疑系统(源码+数据库+文档)
spring boot·vue·计算机毕业设计·java毕业设计·大学生毕业设计·课程答疑系统
阿洛学长3 天前
高质量 AI 提示词之(从 0-1 开发 Vue 项目)
vue·ai编程·1024程序员节
IT教程资源D3 天前
[N_149]基于微信小程序网上商城系统
mysql·vue·前后端分离·springboot网上商城·网上商城小程序
ytgytg283 天前
芋道源码:VUE3部署:避坑--验证码不现显示,管理后台无法访问后端接口等,完善中。。。
java·vue
学会煎墙4 天前
使用uniapp——实现微信小程序的拖拽排序(vue3+ts)
微信小程序·uni-app·vue·ts
神膘护体小月半4 天前
bug 记录 - 路由守卫 beforeRouteLeave 与 confirm 结合,不生效问题
javascript·vue