團隊在管理後臺專案中積累了幾十個業務元件,但散落在各個專案裡,複用靠複製貼上。決定抽離出一個內部元件庫,記錄一下設計思路。
元件分層
元件庫
├── 基礎層(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 變數實現主題化,不用前處理器也能覆蓋樣式
- 每個元件都需要完整文件和示例,降低團隊學習成本
- 內部元件庫不追求大而全,解決團隊實際痛點就好