Element-PLUS

基于 Vue 3 的UI组件库

Posted by page on June 3, 2021

Element PLUS

开始

安装:npm i element-plus -S

引入

  • 完整引入

  • 按需引入

  • 自动导入

  • 手动导入

详见 官方文档

主题

SCSS 变量

新建 src/assets/styles/element.variable.scss,编写scss变量覆盖;详见 主题

// https://github.com/element-plus/element-plus/blob/dev/packages/theme-chalk/src/common/var.scss
@use 'element-plus/theme-chalk/src/mixins/function.scss' as *;

@forward "element-plus/theme-chalk/src/common/var.scss" with (
  $colors: (
    'primary': (
      'base': #EC1620,
    ),
  ),
  $text-color: (
    "regular": #444444,
    'primary': #666666
  ),
  $border-radius: (
    "base": 4px,
  ),
  $border-color: (
    '': #D9D9D9,
    'lighter': #EBEDF0,
  ),
  $button: (
    'text-color': getCssVar('text-color', 'primary'),
    'hover-link-text-color': getCssVar('color-primary'),
  ),
  $dropdown: (
    'menu-box-shadow': 0px 2px 8px 1px rgba(0,0,0,0.1),
    'menuItem-hover-color': getCssVar('text-color', 'primary'),
    'menuItem-hover-fill': #f4f7fa,
  ),
  $date-editor: (
    "daterange-width": 300px,
  ),
);

引入变量

// vite.config.js
    css: {
      preprocessorOptions: {
        less: {},
        scss: {
          additionalData: `@use "@/assets/styles/element.variable.scss" as *;`,
        },
      },
    },

:如果项目使用了 unplugin/vue-components 自动按需导入组件样式,可能导致组件不生效,需关闭按需引入组件样式并引入全局样式:

AutoImport({
  // .....
  resolvers: [ElementPlusResolver({ importStyle: false })],
}),
Components({
  resolvers: [
    ElementPlusResolver({ importStyle: false }),
       // ......
    ]
})

引入全部样式文件

// element-plus.scss
@use "element-plus/theme-chalk/src/index.scss" as *;

CSS变量

为元素类声明 css 变量覆盖基础值,适用于单独为某个组件实例定义样式;

.custom-class {
  --el-tag-bg-color: red;
}

ElementPlus类

编写 element-plus 样式覆盖,masn入口引入

// element-plus.scss
.el-dialog {
  --el-dialog-border-radius: var(--el-border-radius-base);
}

.el-dialog__headerbtn {
  font-size: var(--el-message-close-size, 20px);
}

.el-dialog__footer {
  text-align: initial;
}

Icon(图标)

基本使用

安装

Element Plus将Icon独立为图标集合

npm i @element-plus/icons-vue -S

引入

// icon.js
import * as ElementPlusIconsVue from '@element-plus/icons-vue'

// 引入所有 也可按需手动引入
for (const [key, component] of Object.entries(ElementPlusIconsVue)) {
  app.component(key, component)
}

使用

<!-- 直接使用Svg -->
<Edit style="width: 16px; height: 1px; margin-right: 4px" />

<!-- ElIcon控制样式 -->
<el-icon class="no-inherit" color="#409EFC" :size="16">

<!-- 组件icon值 -->
<el-button type="primary" icon="Edit">修改</el-button>

自动导入

// webpack.config.js
const Components = require("unplugin-vue-components/webpack");
const Icons = require("unplugin-icons/webpack");
const IconsResolver = require("unplugin-icons/resolver");

module.export = {
  // ......
  plugins: [
    Components({
      resolvers: [
        // 自动导入svg图标组件
        IconsResolver({
          prefix: false,  // 省略前缀 默认组件前缀 icon
          enabledCollections: ["ep"],  // icon集合,ep即elementplus图标集
          alias: { 'el-svg': "ep" },  // 别名 使用'el-svg'作为'ep'别名
        }),
        ElementPlusResolver(),
      ],
    }),
    // 自动安装图标集,无需另外安装
    Icons({ compiler: "vue3", autoInstall: true })
  ]
};

自定义Icon

// webpack.config.js
const { FileSystemIconLoader } = require('unplugin-icons/loaders')
Components({
  resolvers: [
    IconsResolver({
      prefix: false,
      alias: { "el-svg": "ep" },
      enabledCollections: ["ep"],
      customCollections: ["svg"],  // 自定义集合添加解析
    }),
    ElementPlusResolver(),
  ],
}),
Icons({
  compiler: "vue3",
  autoInstall: true,
  // 自定义icon图标集合
  customCollections: {
    // 指定导入目录为名称为svg图标集
    svg: FileSystemIconLoader(resolve("src/icons/svg")),
  },
}),

使用

<!-- 
  默认组件名 按{prefix}-{collection}-{icon}
  配置别名后 elementplus图标使用 el-svg-icon名称
  自定义图标集使用 svg-icon名称
-->
<ELSvgPlus />
<SvgLogo />

vite详见 element-plus-best-practices

自动导入插件 unplugin-icons

Layouts(布局)

<script setup>
import AppHeader from "./header/header.vue";
import AppSidebar from "./sidebar/sidebar.vue";
</script>

<template>
  <div class="h-100vh flex flex-col">
    <AppHeader></AppHeader>
    <el-container class="layout-container">
      <AppSidebar></AppSidebar>
      <el-main>
        <slot></slot>
      </el-main>
    </el-container>
  </div>
</template>

<style lang="scss" scoped>
.layout-container {
  height: 0;
  .el-main {
    padding: 0;
  }
}
</style>

分页(Pagination)

推荐使用 @update:current-page 替代 @current-page

<el-pagination
  :current-page="1"
  :page-size="10"
  :total="0"
  @update:current-page="handlePageChange"
  layout="prev, pager, next"
  background
></el-pagination>

分页Hook封装

export const useActivityManageList = (filter, pager) => {
  filter = ref(filter)
  pager = ref(pager);
  const activityList = ref([]);
  const loading = ref(false);
  const getActivityList = async () => {
    if (loading.value) return;
    const params = {
      pageIndex: pager.value.pageIndex,
      pageSize: pager.value.pageSize,
      ...toValue(filter),
    };
    loading.value = true;
    try {
      const { data } = await api.getActivityManageList(params);
      activityList.value = data.records;
      pager.value.pageIndex = data.current;
      pager.value.pageSize = data.size;
      pager.value.pageTotal = data.total;
      pager.value.pageCount = data.pages;
    } catch (error) {
      console.warn(error);
      ElMessage.error("活动数据获取失败");
    }
    loading.value = false;
  };

  return {
    activityList,
    loading,
    getActivityList,
  };
}; 

export const usePager = (pagerOptions) => {
  const pager = ref({
    pageIndex: pagerOptions.pageIndex ?? 1,
    pageSize: pagerOptions.pageSize ?? 10,
    pageTotal: pagerOptions.pageTotal ?? 0,
    pageCount: pagerOptions.pageCount ?? 0,
  });
  return pager;
};

// 组件使用 
const pager = usePager();
const { 
  activityList,
  getActivityList,
  loading
} = useActivityManageList(params, pager); 
const getActivityListWithPageIndex = (pageIndex) => {
  if (loading.value) return;
  pager.value.pageIndex = pageIndex;
  getActivityList();
};
<el-pagination
  layout="jumper"
  :current-page="pager.pageIndex"
  :page-size="pager.pageSize"
  :total="pager.pageTotal"
  background
  @update:current-page="getActivityListWithPageIndex"
/>

Table(表格)

固定列适配

表格小屏下固定列实现宽度适配:

<el-table-column label="标题" min-width="200">
<el-table-column label="人数" min-width="120">
<el-table-column label="次数" min-width="120">
<el-table-column label="操作" fixed="right" min-width="250">

通过 min-width 设置小屏下表格列的最小值(大屏会随宽度计算占比)

补充:覆盖样式实现表格宽度最小值,使大屏下表格宽度占满

.el-table__header-wrapper{
  min-width: 100%;
}
.el-table__body-wrapper {
  .el-scrollbar__view, .el-table__body {
    min-width: 100%;
  }
}

Issues

outline bug

dropdown 或 popover 组件展开菜单层,出现黑色的outline

[role="button"]:focus-visible {
  outline: none;
}