元件庫開發心得
Design System、UML、開發流程
大綱
1. 前言
2. Design System
3. UI 元件分類
4. 元件庫 UML
5. 一個元件的誕生
6. 小談測試
前言
-
前公司接很多後台 ERP 的案子
每次都花很多時間從頭造輪子很浪費時間 -
兩種策略 — 「自幹」或「用現成的」
採用自幹的理由:
- 強調設計、風格明確
- 資源充足(資金、規模、人力)
市面上有哪些選擇
- Design System(純設計、品牌端)
- Polaris - Shopify
- Material Design
- Ant Design
- CSS Framework(純樣式)
- Bootstrap
- Tailwind → Utility CSS
Integrate w/ component logic → Headless UI
- UI Framework(樣式+邏輯,開箱即用包)
- Material-UI
- Ant Design
- Chakra UI
大推
Design System
Why Design System?
- 為系統提供一致的介面體驗
- 避免重複造輪子
- 加速開發流程
- 專注在商業邏輯
從 Design System 出發理解
網頁設計跟頁面架構背後的概念
能讓我們更快地知道 Framework 都在幹嘛
What if we don't have Design System?
data:image/s3,"s3://crabby-images/11062/11062da766f0efbae4d261ec65c798799bfc559f" alt=""
data:image/s3,"s3://crabby-images/77ee0/77ee01c4e7a0d4774f8160951baa5f3562fa1796" alt=""
What is Design System?
由一組共享、整合的元素及原則來定義產品整體的設計
data:image/s3,"s3://crabby-images/fd0da/fd0daab06c42e3ecf3f0df4eb2eeba3b8f80d93a" alt=""
data:image/s3,"s3://crabby-images/0b5ce/0b5ce7048a5edfee3695ef13b865642f947dd292" alt=""
Design System in Project
data:image/s3,"s3://crabby-images/aba01/aba01700cfb51e2441c5b5e95be12bfdd4e869de" alt=""
Design System in Project
Color System (Palette)
data:image/s3,"s3://crabby-images/b67a5/b67a57e3bc4e6303c922af8abff080c708569e35" alt=""
data:image/s3,"s3://crabby-images/87081/87081d07b14e43e5e7dd2d6af820a76e2db37b42" alt=""
Spacing
data:image/s3,"s3://crabby-images/28605/28605fca8ac0173e338b31008f195e4651b628f9" alt=""
data:image/s3,"s3://crabby-images/35376/353767bc92b5c36e36fe5e1ebb0a755cde4a28b0" alt=""
Typography
data:image/s3,"s3://crabby-images/38b16/38b166468163a7405b2eaa20f1022e0aabda1708" alt=""
data:image/s3,"s3://crabby-images/0fafd/0fafd0891d332d8b8449706979cf40487a5e759f" alt=""
data:image/s3,"s3://crabby-images/5511c/5511ce72a95bcd0363f55eb81535eaa383e95905" alt=""
Icons
data:image/s3,"s3://crabby-images/f887c/f887c6df826b2291d940a8372f44385c8b4d5934" alt=""
import { IconDefinition } from './typings';
export const ArrowRightIcon: IconDefinition = {
name: 'arrow-right',
definition: {
svg: {
viewBox: '0 0 24 24',
},
path: {
fill: 'currentColor',
fillRule: 'evenodd',
stroke: 'none',
strokeWidth: 1,
d: 'M13.8 6.8l5.198 5.198-5.198 5.198-1.06-1.06 3.384-3.386H5v-1.5h11.131L12.74 7.86 13.8 6.8z',
},
},
};
export interface IconDefinition {
name: string;
definition: {
svg?: {
viewBox?: string;
};
path?: {
d?: string;
fill?: string;
fillRule?: 'nonzero' | 'evenodd' | 'inherit';
stroke?: string;
strokeWidth?: string | number;
transform?: string;
};
};
}
const Icon: React.FC<IconProps> = (props) => {
const { className, color = "black", icon, spin = false, style, ...rest } = props;
const { definition } = icon;
return (
<i
{...rest}
aria-hidden
className={`
inline-block flex-shrink-0 select-none w-em h-em
${spin ? "animate-spin" : ""}
${color ? Color[color] : ""}
${className ? className : ""}
`}
data-icon-name={icon.name}
style={style}
>
<svg {...definition.svg} focusable={false}>
<path {...definition.path} />
</svg>
</i>
);
};
Atomic Design
data:image/s3,"s3://crabby-images/d545b/d545b8f6eead4fa736721aa7ef77772d7b6757ca" alt=""
data:image/s3,"s3://crabby-images/561f0/561f089c5400a58f0db6b4722a584005efe9e268" alt=""
data:image/s3,"s3://crabby-images/31863/318631fcc3e992c3042c07f101de3822b03f0e5b" alt=""
data:image/s3,"s3://crabby-images/d66de/d66de8246de046a6e6fb3844ca95fb3df2b82eef" alt=""
data:image/s3,"s3://crabby-images/ec7fc/ec7fc710da988932904e642186377a52b2bc1804" alt=""
data:image/s3,"s3://crabby-images/4d51d/4d51df4dcadc8297734920a32dfef8ddf13c1bc4" alt=""
data:image/s3,"s3://crabby-images/c8e66/c8e66708a2e73a3e3277b3dbde74f04d68feba92" alt=""
Design System
UI Component Library
Composed Component
(通常是在專案各自實作)
Layout / Template / Scaffold
Overview
因系統而異,在前公司大多是參考 Ant Design
- Layout
- Inputs
- Navigation
- Surfaces
- Feedback
- Data Display
- Utils
- Lab
- General
- Layout
- Navigation
- Data Entry
- Data Display
- Feedback
- Other
- Layout
- Forms
- Data Display
- Feedback
- Typography
- Overlay
- Disclosure
- Navigation
- Media and Icons
- Others
- Hooks
General
data:image/s3,"s3://crabby-images/1dc6a/1dc6aaa4978da47abf8e4133d35d14a444f9bc76" alt=""
Layout
data:image/s3,"s3://crabby-images/1be14/1be14520fbdba95eb9aab0614e6365622828685e" alt=""
與網站的排版佈局有關係的元件
常見的還有 Box、Container、Flex 和 Wrap 等等。
Navigation
data:image/s3,"s3://crabby-images/ab78d/ab78dca81dbe288736426c8bc61cef0b33790c69" alt=""
導覽相關的元件
提示使用者處於網站中的哪一個階段
與這些元件的互動通常會切換頁面。
常見的還有 Drawer、Links 等等。
Data Entry
data:image/s3,"s3://crabby-images/33636/336368409ad3669bd74660fa1bca7a901bd36b69" alt=""
data:image/s3,"s3://crabby-images/c836d/c836de8d1c3220d7e3e410088f4db3a4592ae803" alt=""
Inputs, Forms or Data Display
Feedback
data:image/s3,"s3://crabby-images/42605/426058e273652e84003ceb2222d81fa15f85e3e5" alt=""
使用者互動過後網頁給使用者的反應
讓使用者感受到網頁還有在運作
Utility
- Portal 系列
- Overlay
- Transition
UML
What is UML (Unified Modeling Language)?
以圖像化的方式,將系統的功能與結構畫成「模型」與藍圖
一種開放的方法,用於說明、視覺化、構建和編寫一個正在開發的、物件導向的、軟體密集系統的製品的開放方法
data:image/s3,"s3://crabby-images/745e2/745e29db3a9eff29b8a58a5ed021e12223041c3e" alt=""
Component Diagram
反映出元件之間的依賴關係
Interface
Component
共用功能的抽象化
- TextField
- FormControl
- Portal
- Notifier
- Transition
就是一個個的 UI Component
- Input
- Button
- Popper
- Notification
- Fade
data:image/s3,"s3://crabby-images/ce22a/ce22a24faa13d575d5629bb33d03b76abce862f4" alt=""
data:image/s3,"s3://crabby-images/00250/00250a238d3f73b2bdb1de4bb2b145e19efad0bb" alt=""
一個元件的誕生
WorkFlow
規格 Spec
介面 Interface
實作 Implement
Design System、UML 到 Spec 都是前期規劃
元件庫要好用很大程度決定於這些前期準備夠不夠完善
data:image/s3,"s3://crabby-images/0cb81/0cb81f7da9b53c6c11ee05d80fa54be9e346cb83" alt=""
data:image/s3,"s3://crabby-images/b737d/b737dd214d46cee4a3f3592c721a4c2d08d4da7c" alt=""
// 概念:先繼承原生的 Button Props 再來擴充需要的 Props
export interface ButtonProps extends Omit<React.ComponentPropsWithRef<'button'>, 'prefix'> {
color?: ButtonColorType;
error?: boolean;
disabled?: boolean;
loading?: boolean;
prefix?: ReactNode;
size?: Size;
suffix?: ReactNode;
variant?: ButtonVariantType;
}
Button Implement
const Button = forwardRef<HTMLButtonElement, ButtonProps>(function Button(props, ref) {
const {
children,
color = 'primary',
error = false,
disabled = false,
loading = false,
onClick,
prefix: prefixProp,
size = 'medium',
suffix: suffixProp,
variant = 'text',
style,
...rest
} = props;
let prefix: ReactNode = prefixProp;
let suffix: ReactNode = suffixProp;
// 處理情境
if (loading) {
const loadingIcon = <Icon icon={SpinnerIcon} spin />;
if (suffix && !prefix) {
suffix = loadingIcon;
} else {
prefix = loadingIcon;
}
}
const asIconBtn = children == null && !!(prefix || suffix);
// 處理樣式 (CSS)
const buttonColorClass = useMemo(() => {
if (disabled) return Color['disabled'];
if (error) return Color['error'];
if (color) return Color[color];
}, [color, disabled, error]);
const padding = useMemo(() => {
if (asIconBtn) return 'px-0';
if (size === 'small') return 'px-3';
return 'px-4';
}, [asIconBtn, size]);
const iconClass = useMemo(() => size === 'small' ? "text-lg" : "text-2xl", [size]);
const variantClass = useMemo(() => {
if (variant === 'contained') return `text-white ${ButtonVariant[variant]}-${buttonColorClass}`;
if (variant === 'outlined') return `border-2 text-${buttonColorClass} ${ButtonVariant[variant]}-${buttonColorClass}`;
return `${ButtonVariant[variant]}-${buttonColorClass}`;
}, [buttonColorClass, variant]);
const disabledClass = useMemo(() => disabled ? "opacity-40 cursor-not-allowed" : "hover:opacity-60", [disabled]);
const loadingClass = useMemo(() => loading ? "cursor-default pointer-events-none" : "", [loading]);
// HTML Structure
return (
<button
{...rest}
ref={ref}
aria-disabled={disabled}
className={`relative box-border flex-center flex-shrink-0 gap-1 m-0 rounded cursor-pointer select-none whitespace-nowrap uppercase focus:outline-none ${padding} ${iconClass} ${variantClass} ${disabledClass} ${loadingClass}`}
style={style}
disabled={disabled}
onClick={(event: MouseEvent<HTMLButtonElement>) => {
if (!disabled && !loading && onClick) {
onClick(event);
}
}}
>
{prefix}
{children && <span className={`${size ? ButtonSize[size] : ""}`}>{children}</span>}
{suffix}
</button>
);
});
export default Button;
data:image/s3,"s3://crabby-images/0e1ee/0e1eec221ec41e315eedde980f2a2601f048ffbc" alt=""
測試
各種測試
單元測試 Unit Test
整合測試 Integration Test
端對端測試 E2E Test
Unit Testing
-
測 最基本的情況 Component 該有的要有
-
測 各種 props 有沒有正常顯示?
-
style, element, attribute
-
-
測 coverage
-
每一行每一個 if-else branch 都跑
-
describe('<Switch />', () => {
// 1.
describeForwardRefToHTMLElement(
HTMLSpanElement,
(ref) => render(<Switch ref={ref} />),
);
// 2.
describeHostElementClassNameAppendable(
'foo',
(className) => render(<Switch className={className} />),
);
describe('prop: checked', () => {
// 3.
[false, true].forEach((checked) => {
const message = checked
? 'should be checked if checked=true'
: 'should be unchecked if checked=false';
it(message, () => {
const { getHostHTMLElement } = render(<Switch checked={checked} />);
const element = getHostHTMLElement();
const [inputElement] = element.getElementsByTagName('input');
testChecked(element, inputElement, checked);
});
});
};
};
個人想法
公司規模不大的情況下
若能以市面上的方案出發來架構跟設計的話應該是可以避掉很多問題
畢竟錢跟資源真的挺有差的
條件不到位的情況下,也只會開發出半吊子
又浪費很多時間跟資金
Thanks For Listening
Q & A Time
元件庫開發心得 Design System、UML、開發流程
By parkerhiphop
元件庫開發心得 Design System、UML、開發流程
- 905