A story of a shared components library
data:image/s3,"s3://crabby-images/50896/508962b489b9eb5acd970101957ecf13960940c2" alt=""
Andrey Semin
Senior Software Engineer
at MicroMech
data:image/s3,"s3://crabby-images/734ac/734acc93118c460d9025c0ec908421d55957c5ac" alt=""
https://twitter.com/ifedyukin
data:image/s3,"s3://crabby-images/a8b09/a8b098d17159d8ae5009997b331f323ed186e82a" alt=""
https://www.instagram.com/semin_andrey
data:image/s3,"s3://crabby-images/3a55b/3a55b4b3f1848380b56f5bb60d45dbfec7580caa" alt=""
Hello Andrey!
Has anyone tried to build your own components library?
Why we decided to build this library?
- Build more than 1 app
- Attract more customers
- Provide same look & feel
Starting point
- App built with MaterialUI 🤮
- Sometimes components from MUI were wrapped with custom components
- Sometime they were not
- No documentation
- MUI-styled components API
- Weird or missing tests
Requirements
- Simple and handy component API
- Framework-independent API
- Written in TypeScript
- Well documented
- Well tested
- Include components showcase
- Built on fresh MUI version
data:image/s3,"s3://crabby-images/6145d/6145d71b68fa0b32e0de438f5f27f67de6e4179c" alt=""
How exciting it is!
- Extract basic components to separate package
- Drop lots of dependencies from the app
- Remove weird tests
- Make everything "right"
Cotton
data:image/s3,"s3://crabby-images/cd801/cd801e062661bf686596a53910e95f65f74a7d79" alt=""
Component folder
- index.ts
- ComponentName.tsx
- ComponentName.spec.tsx
- ComponentName.stories.tsx
- ComponentName.README.md
- components/
Component migration
- Search for all usages of a component
- List all used props and their values
- Use prepared list to design component API
- Migrate component
How to bundle?
Webpack!
data:image/s3,"s3://crabby-images/2dee4/2dee4414a42f2a0156702efebfe185f14fce3838" alt=""
Time to test!
- Write tests from scratch
- Jest + @testing-library/react ⚡
- External component API
- Separate snapshot for each render path
Time for Storybook!
data:image/s3,"s3://crabby-images/f4367/f4367aef62b9c90bfc108ae0b9b696bcad4b4b62" alt=""
How MaterialUI uses styles
const typographyStyles = (): StyleRules => createStyles({...})
const TypographyComponent: React.FC<TypographyProps> = () => {...}
export const Typography = withStyles(typographyStyles)(TypographyComponent)
css-in-js 👋
*.md for the rescue!
A component to add an empty space with controllable height based on default theme gutter value
| Name | Required | type | Default Value | Description |
| ---------------- | -------- | ------ | ------------- | ----------------------------------- |
| gutterMultiplier | - | number | 1 | Multiplier to default theme gutter |
| data-test-id | - | string | - | Custom data-test-id attribute value |
### Example:
```js
import { Spacer } from '@motokaptia/cotton';
<>
<SomeBlock />
<Spacer />
<SomeOtherBlock />
</>;
```
*.md for the rescue!
data:image/s3,"s3://crabby-images/d2d68/d2d68e2b726ab10e8f36de4cb4bb10038befd36f" alt=""
data:image/s3,"s3://crabby-images/6eece/6eece20503a4bd087c64181986092b1f30946134" alt=""
data:image/s3,"s3://crabby-images/0b388/0b388d46cd01ecdb73aa3c338e7434e807a6a167" alt=""
We have to ship parts of MUI itself
- ThemeProvider component
- createGenerateClassName function
- withStyles, withWidth HOCs
- StyleRules, Theme types
- isWidthDown function
css-in-js 👋
OK, let's try again
New error 😞
- Quick googling - no results
- Not-so-quick googling - no results
- 2 days of investigation
Root cause of the issue is...
React hook call wrapped with if statement
data:image/s3,"s3://crabby-images/5fcab/5fcab13d14e5fc41ec57431bd42e698003f587cc" alt=""
And the solution is
- Downgrade MaterialUI to latest v3 release
- Replace webpack with regular TS transpilation
Now it should work!
But it doesn't
data:image/s3,"s3://crabby-images/54c4c/54c4cdcd01ffe7d1dc5552d750fe7ee0c9229662" alt=""
Why?
css-in-js 👋
+
Server Side Rendering
function createStyleContext() {
return {
sheetsManager: new Map(),
sheetsRegistry: new SheetsRegistry(),
generateClassName: createGenerateClassName()
};
}
export function getStyleContext() {
if (!process.browser) {
return createStyleContext();
}
if (!global.__INIT_MATERIAL_UI__) {
global.__INIT_MATERIAL_UI__ = createStyleContext();
}
return global.__INIT_MATERIAL_UI__;
}
Same error appeared in tests
jest.mock('@material-ui/core/styles/createGenerateClassName');
createGenerateClassName.mockImplementation(() => (rule, styleSheet) =>
`${styleSheet.options.classNamePrefix}-${rule.key}`
);
One more testing issue
data:image/s3,"s3://crabby-images/c2043/c20431d2f6a5f54f041650dcf0b1c0188655fef8" alt=""
Select elements by data-test-id attribute
Naïve approach
- Specify data-test-id value in components library
- Use it in library consumers
Such approach won't work in long-term perspective
- Update value in library
- Have to update e2e tests in all apps
Better way to handle data-test-id
- Add data-test-id prop to components' API
- For complex components, generate data-test-id value
Better way to handle data-test-id
data-test-id + postfix
data-test-id="select"
data-test-id="select-root"
data-test-id="select-option"
Better way to handle data-test-id
data-test-id is omitted from production build
babel-plugin-jsx-remove-data-test-id
Finally!
- App is running
- SSR works
- Styles are not broken
- Component is on it's place
data:image/s3,"s3://crabby-images/67454/674544cfaff42a9681c7eb7061fed309f57be59e" alt=""
Conclusion
- css-in-js - 💩
- MaterialUI - 💩
data:image/s3,"s3://crabby-images/c006a/c006a4af6060f405c180f43fb85ec07ac64b0a38" alt=""
Now seriously
- Easy to work on brand redesign
- Focus on components and their state
- Reduced time to interact with design guy
- Great DX
Library is ready & in use
Benefits we got:
What went wrong
- Outdated MaterialUI version
- Quality of tests dropped over time
- Failed to incapsulate MUI within the library
- No automatic releases
What's next?
- Finish redesign
- Implement more components
- Update MaterialUI
- Configure releases
data:image/s3,"s3://crabby-images/a8b09/a8b098d17159d8ae5009997b331f323ed186e82a" alt=""
https://www.instagram.com/semin_andrey
Shared Components Library
By Andrey Semin
Shared Components Library
- 430