The howles.dev
site is the homepage for Lewis Howles.
This site is built as though it were a UI project, using Vue and Tailwind CSS, and tested using Vitest and Cypress. This lets me utilise the site as information, with links to relevant projects, but also to use it to demonstrate my current best practices.
Vue is my framework of choice for Front-end projects. Vue has a strong ecosystem, with best practices for everything from testing to stores, plus you don't have to use JSX, which is always a bonus.
- Vue is a lightweight, fast framework that has a gentle learning curve. Combined with extensive, easy-to-read documentation, it makes onboarding very easy.
- It is regularly updated, and has quickly improved both performance and developer experience.
- The component-based architecture helps to organise code into easily re-usable units. These self-contained units are also more manageable and scalable, and those individual components can easily be replaced.
- Vue's reactivity model is simple to understand, but powerful enough to keep complex apps up to date.
- Vue is scalable, from small-scale projects like this, to large-scale apps that use dozens of API calls.
- Single-file components keeps related concepts together, improving maintainability.
Tailwind, while sometimes divisive, has been a game-changer for the way I work.
- The utility-first approach works perfectly with frameworks like Vue, making it easy to add and removes styles based on state. It also enables rapid prototyping, often allowing examples to be built as quickly as they can be designed.
- Utility classes allow for greater consistency, giving a defined set of options for things like font size. Pulling configuration into a library also means multiple projects can be set up in the same way, giving a level of immediate consistency between projects that is more involved to replicate with regular CSS.
- Tailwind automatically removes unused CSS, optimising the CSS bundle without any effort from the developer.
- Applying styles directly to the HTML, once you're used to it, allows you to develop much more quickly. It removes the mental effort required to name things, and reduces context switching, meaning fewer delays and distractions.
Vitest is a Jest-compatible, Vite-native testing framework. Being Vite native means it runs much more quickly than Jest, and since switching I have started to enjoy unit testing a lot more.
Using Javascript to write end-to-end testing is a Front-end developer's dream come true. Cypress is regularly updated and improved, and the tooling—particularly the testing UI—is amazing for running through the UI like a user.
It's flexible enough to allow extensive helper methods, which is a personal love of mine, and the ability to easily mock API responses makes it effortless to test edge cases.
The recent ability to test components in isolation gives a level of component testing that is just cumbersome to complete in a unit test framework and gives a level of confidence that is hard to achieve elsewhere.
- The website should be almost as accessible as possible, with clear focus styling, appropriate screen-reader features, etc.
- API data is loaded from
json
files to mimic the API workflow. In a full project, I would prefer to use an API helper class to handle any transformations and data retrieval as simply as possible. - Text is provided via
vue-i18n
translations to mimic a larger-scale app. Again, in a full app, loading and selecting the preferred language would be handled by a translation composable. - All animations should be disabled when
prefers-reduced-motion: reduce
is set.
- In a full-fledge project, I would like to have a
form-wrapper
type component. This would handle the form interaction, field models, validation, and generating a proper error summary above the form. - API interaction is very basic. In a full-fledged project, I would create an
API
class, combined with a more full-fledgeuse-api
composable, which would handle the various different types of API call, and make it possible to easily mock any individual API method during unit testing.