This project contains the code for the public-facing part of my blog app.
The project is built on TypeScript / React / Redux. This is the frontend stack I've been working with full time for years now, and I love it, so it was an obvious choice.
Of course I want to make every hobby-project a learning opportunity. I used this project to learn about new patterns, ever "better" ways to architect & structure a React app, and experimented with new React features and libraries.
I have quite a bit of experience configuring Webpack and its plugins, and I prefer custom configurations for non-trivial projects. However, I think CRA gives you a pretty decent environment out of the box (especially since v2). And going by the "if it ain't broke don't fix it" principle, I don't think the (great) effort of setting up a custom Webpack configuration is worth it unless you start to bump into CRA's limits.
This might just be my C# background, but I have embraced the principle of trying to convert what would be runtime errors into compile-time errors as much as possible. I think it greatly helps you avoid bugs, especially in larger projects.
I'm not a type everything, whatever it takes kind of guy, however. There are places where it's just not worth the effort of trying to completely write types for everything. In those cases I'm happy to go "plain" JavaScript (as you can see at some places).
The outer structure is default CRA setup. The interesting part of the application is inside the src folder.
@types contains typings I wrote for external libraries that came without a .d.ts file. I will contribute those back to DefinitelyTyped when I'll have some spare time.
store contains Redux architecture. I like to structure those into 4 folders (3 on non-TS projects):
- state that holds the usually type-only files that define the shape of the state.
- reducers that holds the reducers.
- actions that holds both the action type definitions and the action creators themselves. For asynchronous actions I use redux-thunk.
- selectors that hold the (memoized) selectors. In this case because of the size of the project I am using a single file instead of a directory. I try to follow the pattern where only the reducers and the selectors can ever care about the shape of the store. Everything else goes through the reducers. This allows for much easier shape-refactoring, that could otherwise be a pain for bigger applications.
components holds the actual React components. Nowadays I try to go for the newer Functional Component pattern rather than using classes. For styling I am using SCSS Modules along with a classnames
utility that is good enough for the complexity I require. I tried several CSS-in-JS frameworks, but I was never convinced that the added complexity was entirely justified.
The entry point is index.js
, which is one example where I omit TypeScript. This is also where I set up my main infrastructure, including Hot Module Reloading and the service worker (which is disabled now, but I am planning on building a complete PWA.)