Skip to content
⚠️ This article was written in 2020. Some content may be outdated.

企業級元件庫設計思路

團隊在管理後臺專案中積累了幾十個業務元件,但散落在各個專案裡,複用靠複製貼上。決定抽離出一個內部元件庫,記錄一下設計思路。

元件分層

元件庫
├── 基礎層(primitives)
│   ├── Button、Input、Select
│   ├── 不依賴任何業務邏輯
│   └── 可被任何專案使用
├── 業務層(business)
│   ├── UserSelect(使用者選擇器)
│   ├── DepartmentTree(部門樹)
│   ├── PermissionGuard(許可權守衛)
│   └── 依賴基礎層 + 業務 API
└── 複合層(composite)
    ├── SearchForm(搜尋表單)
    ├── DataTable(資料表格 + 分頁 + 篩選)
    └── 由業務層和基礎層組合而成

元件 API 設計原則

vue
{% raw %}
<!-- 原則 1:props 驅動,slot 擴充套件 -->
<template>
  <div class="data-table">
    <!-- 具名插槽覆蓋預設渲染 -->
    <table>
      <thead>
        <tr>
          <th v-for="col in columns" :key="col.key">
            <!-- 支援自定義表頭 -->
            <slot :name="`header-${col.key}`" :column="col">
              {{ col.title }}
            </slot>
          </th>
        </tr>
      </thead>
      <tbody>
        <tr v-for="row in displayData" :key="row[rowKey]">
          <td v-for="col in columns" :key="col.key">
            <!-- 支援自定義單元格 -->
            <slot :name="`cell-${col.key}`" :row="row" :column="col">
              {{ row[col.key] }}
            </slot>
          </td>
        </tr>
      </tbody>
    </table>

    <!-- 作用域插槽:自定義空狀態 -->
    <slot name="empty" v-if="!displayData.length">
      <div class="empty">暫無資料</div>
    </slot>
  </div>
</template>

<script>
export default {
  name: 'DataTable',
  props: {
    // 必填:列配置
    columns: {
      type: Array,
      required: true,
      validator: (cols) => cols.every(c => c.key && c.title),
    },
    // 必填:資料
    data: { type: Array, default: () => [] },
    // 行唯一標識
    rowKey: { type: String, default: 'id' },
    // 分頁
    pagination: {
      type: [Object, Boolean],
      default: () => ({ page: 1, pageSize: 20, total: 0 }),
    },
    // 載入狀態
    loading: { type: Boolean, default: false },
  },
  computed: {
    displayData() {
      if (!this.pagination) return this.data;
      const { page, pageSize } = this.pagination;
      return this.data.slice((page - 1) * pageSize, page * pageSize);
    },
  },
};
</script>
{% endraw %}

原則 2:事件統一

javascript
// 統一事件命名:on + 動詞 + 名詞
// 好的命名
this.$emit('change', value);
this.$emit('select', row);
this.$emit('page-change', { page, pageSize });
this.$emit('search', queryParams);

// 不好的命名
this.$emit('input', value);        // v-model 專用
this.$emit('update', value);       // 太模糊
this.$emit('onPageChange');        // 帶 on 字首多餘

原則 3:樣式隔離 + 主題化

scss
// 使用 CSS 變數實現主題
:root {
  --dt-primary-color: #409eff;
  --dt-border-color: #e4e7ed;
  --dt-bg-header: #f5f7fa;
  --dt-font-size: 14px;
  --dt-row-height: 48px;
}

.data-table {
  width: 100%;
  border: 1px solid var(--dt-border-color);
  border-radius: 4px;
  font-size: var(--dt-font-size);

  th {
    background: var(--dt-bg-header);
    height: var(--dt-row-height);
    padding: 0 16px;
    font-weight: 500;
  }

  td {
    height: var(--dt-row-height);
    padding: 0 16px;
    border-bottom: 1px solid var(--dt-border-color);
  }
}

// 暗色主題覆蓋變數即可
[data-theme='dark'] {
  --dt-border-color: #4c4d4f;
  --dt-bg-header: #2b2b2b;
}

釋出和使用

json
// package.json
{
  "name": "@company/ui",
  "version": "0.1.0",
  "main": "lib/index.js",
  "module": "es/index.js",
  "types": "lib/index.d.ts",
  "files": ["lib", "es", "types"],
  "sideEffects": ["*.css", "*.scss"]
}
javascript
// 按需引入
import { DataTable, SearchForm } from '@company/ui';

// 全量引入
import CompanyUI from '@company/ui';
Vue.use(CompanyUI);

文件和示例

markdown
每個元件需要:
1. Props 表格(名稱、型別、預設值、說明)
2. Events 表格
3. Slots 表格
4. 至少 3 個示例(基礎用法、進階用法、邊界情況)
5. 設計說明(什麼時候用、什麼時候不用)

推薦用 vuepress 或 storybook 搭建文件站。

小結

  • 元件分三層:基礎元件、業務元件、複合元件,職責清晰
  • API 設計遵循 props 驅動 + slot 擴充套件,減少使用者的理解成本
  • CSS 變數實現主題化,不用前處理器也能覆蓋樣式
  • 每個元件都需要完整文件和示例,降低團隊學習成本
  • 內部元件庫不追求大而全,解決團隊實際痛點就好

MIT Licensed