vue中的css作用域

555 阅读5分钟

最近在做vue项目的时候碰到一个问题,使用js动态新增Dom的时候部分样式不生效,绑定的事件也不生效。

静态html:

<span class="el-tag el-tag--info">
                                    <span class="el-select-item-text" id="2">土豆泥</span>
                                    <i class="el-close-tag el-icon-close" @click="deleteItem(2)"></i>
                                </span>

动态生成代码:

this.innerHTML += `<span class="el-tag el-tag--info">
                                    <span class="el-select-item-text" id="${obj.id}">${obj.label}</span>
                                    <i class="el-close-tag el-icon-close" @click="deleteItem(${obj.id})"></i>
                                </span>`

土豆泥标签是写死的,另一个标签是js新增的,可以看到大部分样式是一致的,只是删除按钮不一致。

现在把图贴上来后自己马上反应过来事件不生效的原因了,@clickVue语法,只会在打包的时候解析一次,程序运行过程中新增的vue语法肯定就无法被正确解析了。

<style type="text/css">
.page[data-v-a1796016] {
  display: -ms-flexbox;
  display: flex;
  -ms-flex-direction: column;
      flex-direction: column;
  background: #eff4f3;
  height: 100%;
  position: relative;
  font-size: 14px;
}
.page .page-title[data-v-a1796016] {
    background: url() no-repeat;
    background-size: 40px 40px;
}
.page .page-content[data-v-a1796016] {
    background: white;
    height: 100%;
    margin: 24px;
    -ms-flex: 1;
        flex: 1;
    padding-bottom: 24px;
}
.page .page-content .info-body[data-v-a1796016] {
      padding-left: 53px;
      border-bottom: 1px solid #EAF0F4;
      padding-bottom: 20px;
}
.page .page-content .info-body[data-v-a1796016]:last-child {
        border-bottom: none;
}
.page .page-content .info-body .el-row .el-col[data-v-a1796016] {
        padding: 8px 0;
        color: #333;
}
.page .page-content .info-body .el-row .col-border[data-v-a1796016] {
        color: #666;
}
.page .page-content .info-body .el-row .col-qr[data-v-a1796016] {
        padding: 0;
}
.page .page-content .info-body .el-row .col-qr .qr-img[data-v-a1796016] {
          width: 24px;
          height: 24px;
}
.page .downloadImg[data-v-a1796016] {
    display: inline-block;
    width: 120px;
    height: 34px;
    border-radius: 2px;
    line-height: 34px;
}
header[data-v-a1796016] {
  vertical-align: middle;
  height: 96px;
  padding: 12px 32px 0 32px;
  background: #fff;
}
header p[data-v-a1796016] {
  font-size: 14px;
  font-family: PingFangSC-Regular;
  font-weight: 400;
  color: rgba(0, 0, 0, 0.65);
  line-height: 22px;
}
header p[data-v-a1796016]:nth-child(2) {
  font-size: 20px;
  font-family: PingFangSC-Medium;
  font-weight: bolder;
  color: rgba(0, 0, 0, 0.85);
  line-height: 28px;
}
.select-tree input[data-v-a1796016] {
  border: 0px solid;
}
.el-close-tag[data-v-a1796016] {
  color: #909399;
  background: #c0c4cc;
  border-radius: 50%;
}
.setOtherUsersTable .el-dialog[data-v-a1796016] {
  width: 75%;
}
</style>

可以看到页面的最后一个样式标签就是写在组件里的样式(vue通过在DOM结构以及css样式上加唯一不重复的标记,以保证唯一,达到样式私有化模块化的目的。加了scope属性打包后就会有[data-v-hash标记)。但我们在代码中js自动生成的元素显然没有data-v-hash属性,所以部分本模块私有样式没有呈现出来。