Vue 3 + Element Plus 重置el-drawer样式失效

在 Vue 3 + Element Plus 开发中,若直接将 <el-drawer> 作为 <template> 的根元素(即上层是 <template>),‌Scoped CSS 样式将无法正确生效‌。

示例说明

‌样式失效的情况‌:

javascript 复制代码
<template>
  <el-drawer
    v-model="drawer"
    title="I am the title"
    :direction="direction"
    :before-close="handleClose"
  >
    <span>Hi, there!</span>
  </el-drawer>
</template>
<style scoped>
:deep(.el-drawer) { /* 这些样式不会作用于内部元素 */ }
</style>

‌样式生效的正确做法‌:

javascript 复制代码
<template>
  <div class="drawer-wrapper">
    <el-drawer
    v-model="drawer"
    title="I am the title"
    :direction="direction"
    :before-close="handleClose"
  >
    <span>Hi, there!</span>
  </el-drawer>
  </div>
</template>
<style lang="scss" scoped>
.drawer-wrapper  { 
/* 样式可正常生效 */
.el-drawer__header {
      border-bottom: 1px solid #e8e8e8 !important;
      margin-bottom: 0 !important;
    }
 }
</style>

通过外层包裹 <div>,Scoped CSS 的作用域锚定在该容器上,再通过 :deep() 深度选择器穿透到 <el-drawer> 及其内部结构,从而正确应用样式 ‌‌。

核心原因
‌Vue 的 Scoped CSS 机制‌: Vue 会为组件模板中的每个元素注入唯一的 data-v-* 属性,并在编译时将 CSS 选择器重写为包含该属性的形式,以实现样式作用域隔离。

‌Element Plus 组件的内部结构‌:<el-drawer> 是一个‌复杂组件‌,其内部会动态生成多个子元素(如 .el-drawer__header、.el-drawer__body 等),这些子元素‌不会继承父组件的 data-v-* 属性‌。
‌当 <el-drawer> 直接作为根元素时‌: Scoped CSS 的作用域仅作用于 <el-drawer> 本身,而无法穿透到其内部动态生成的子元素,导致自定义样式失效 ‌‌。

最后,附带一份重置`el-drawer`组件的CSS样式,包括z-index、位置、尺寸、背景色和字体大小,以及头部和主体部分的定制,可根据需要自行调整。

复制代码
 :deep(.el-drawer__wrapper) {
    z-index: -3000!important;

    .el-drawer {
      position: relative;
      left: 1100px;
      top: 154px;
      padding: 0;
      width: 28.125vw !important;
      height: 28.385vw;
      background: rgba(7, 18, 26, 0.8);
      font-size: 16px;
    }

    .el-drawer__header {
      padding: 0 !important;
      width: 520px !important;
      height: 30px;
      margin: 12px;
      font-size: 16px;
      color: #ffffFF;
      background: rgba(19, 32, 42, 0.5) !important;
    }

    .el-drawer__body {
      margin: 0px 10px 10px 10px;
      background: rgba(19, 32, 42, 0.5) !important;
      padding: 0;
      width: 520px !important;
      height: 340px;
      position: relative;
    }
}
相关推荐
lichenyang4533 小时前
Docker 学习笔记(一):为什么需要镜像、容器和仓库?
前端
kyriewen3 小时前
别再对着 TypeScript 报错发呆了:我把 10 个最常见的红色波浪线翻译成了人话
前端·javascript·typescript
IT_陈寒3 小时前
SpringBoot自动配置的坑,我的API突然就404了
前端·人工智能·后端
free354 小时前
从 0 实现一个 Tiny JavaScript VM:项目架构拆解
javascript
暴走的小呆4 小时前
Vue 2 中 Object 的变化侦测:从 getter/setter 到 Dep、Watcher、Observer
vue.js
奇奇怪怪的4 小时前
Embedding 模型 10+ 横向评测
前端
陈广亮4 小时前
Monorepo 从 0 到 1 实操指南 2026 版:pnpm catalogs + Turborepo 2.x + changesets 全链路
前端
子兮曰4 小时前
OpenMontage 深度解剖:你的 AI 编程助手,其实是个视频工作室
前端·后端·ai编程
敲代码的鱼4 小时前
PDF 预览与签名批注写回 支持安卓 iOS 鸿蒙 UTS插件
android·前端·ios
英勇无比的消炎药4 小时前
TinyVue v-auto-tip: 文本超长自动提示的优雅方案
vue.js