We're all familiar with this: We are starting a new project and are looking for some beautiful UI components. While we could technically build these from scratch, we want to start building instead of reinventing the wheel. We need a solution to hit the ground running without sacrificing quality or accessibility (a11y).
So we venture into the world of Angular component libraries. While they all provide incredible variety and most of them come with solid accessibility features, it seems that most Angular UI libraries come with a strong corporate branding that often doesn't quite align with the project's needs. More importantly, they mostly lack an easy way to customize or extend components and do not allow us to let them make them our own.
Then we look at the React ecosystem and all the incredible projects built on RadixUI and shadcn. I don't know about you, but when I do that I always get a little jealous.
Why? shadcn/ui comes with all the components you could ever need for a project, and all of them come in beautiful styles by default. However, it still allows you to adjust and customize every single UI primitive as you please.
How does it do that?
The problem with shadcn for Angular developer's is that it is built on top of React...
Now imagine an accessible, open-source Angular UI library that doesn't come pre-styled, allowing you to have full creative control over its appearance. Angular' shadcn implementation so to say.
Enter spartan/ui – an innovative collection of Angular UI primitives that are un-styled and accessible by default.
To achieve our goal of a shadcn-like development experience, spartan/ui comes in two parts:
To make this as easy as possible, spartan/ui comes equipped with an Nx plugin that allows you to effortlessly integrate our components into your Nx workspace. With a single command, you can add any of its 30 spartan/ui primitives to your projects.
But that's not all – the Nx plugin's capabilities extend beyond just adding components. You can also leverage it to incorporate one of 12 custom themes into your Nx applications, letting you truly own the visual appearance of your projects.
So let's see what getting up and running with spartan/ui looks like.
If you would rather follow along to a video version of this article, check it out on YouTube.
As mentioned above, spartan/ui follows the same paradigm as shadcn, that you should own the code that allows you to style, extend, and compose your UI components.
While we are working on a standalone API, Nx provides incredible tooling for exactly this use case. Therefore, the initial version of spartan/ui's CLI is an Nx plugin.
Hence, for this tutorial, we will create a new Angular project inside an Nx workspace.
Again, Nx makes this incredibly easy. Simply run the command below.
npx create-nx-workspace@latest
When prompted:
Finally, wait for Nx to work its magic, install all necessary dependencies, and set up your Angular workspace.
I am a big proponent of having your template and styles in the same file as your Component class. Therefore, I deleted the src/app/app.component.html
and src/app/app.component.css
files created by the workspace generator. I also got rid of the src/app/nx-welcome.component.ts
and changed the contents of my src/app/app.component.ts
to the following:
import { Component } from '@angular/core';
@Component({
standalone: true,
imports: [],
selector: 'app-root',
template: `<button>Hello from {{title}}</button>`
})
export class AppComponent {
title = 'sparta';
}
One more thing before we are ready to start adding spartan/ui.
As spartan/ui is built on top of TailwindCSS, we need a working setup of it for our project.
Thankfully, Nx again makes this incredibly easy for us. Simply run the following command and select your application when prompted:
npx nx g @nx/angular:setup-tailwind
This will create a tailwind.config.ts
file and install all the necessary dependencies. Let's continue.
We are now ready to add spartan/ui to our project. To make our lives much easier, we will use the Nx plugin, which we install like so:
npm i @spartan-ng/nx
Then we add the @spartan-ng/ui-core
library.
npm i @spartan-ng/ui-core
It contains a bunch of helpers, like the hlm
function, which powers our tailwind class merging, and most importantly the @spartan-ng/ui-core/hlm-tailwind-preset
, which contains all the necessary extensions to tailwind, which make our spartan/ui/helm directives and components work.
We now have to add this spartan-specific configuration to your TailwindCSS setup. Simply add @spartan-ng/ui-core/hlm-tailwind-preset
to the presets array of your config file:
const { createGlobPatternsForDependencies } = require('@nx/angular/tailwind');
const { join } = require('path');
/** @type {import('tailwindcss').Config} */
module.exports = {
presets: [require('@spartan-ng/ui-core/hlm-tailwind-preset')],
content: [
join(__dirname, 'src/**/!(*.stories|*.spec).{ts,html}'),
...createGlobPatternsForDependencies(__dirname),
],
theme: {
extend: {},
},
plugins: [],
};
To complete your TailwindCSS setup, we need to add our spartan-specific CSS variables to your style entry point. This is most likely a styles.css
in the src
folder of your application.
Again, we are using Nx, so our plugin will take care of the heavy lifting:
npx nx g @spartan-ng/nx:ui-theme
When prompted:
Then, check out your styles.css
and see the following spartan/ui-specific variables being added:
:root {
--font-sans: ''
}
:root {
--background: 0 0% 100%;
--foreground: 240 10% 3.9%;
--card: 0 0% 100%;
--card-foreground: 240 10% 3.9%;
--popover: 0 0% 100%;;
--popover-foreground: 240 10% 3.9%;
--primary: 240 5.9% 10%;
--primary-foreground: 0 0% 98%;
--secondary: 240 4.8% 95.9%;
--secondary-foreground: 240 5.9% 10%;
--muted: 240 4.8% 95.9%;
--muted-foreground: 240 3.8% 46.1%;
--accent: 240 4.8% 95.9%;
--accent-foreground: 240 5.9% 10%;
--destructive: 0 84.2% 60.2%;
--destructive-foreground: 0 0% 98%;
--border: 240 5.9% 90%;
--input: 240 5.9% 90%;
--ring: 240 5.9% 10%;
--radius: 0.5rem;
}
.dark {
--background: 240 10% 3.9%;
--foreground: 0 0% 98%;
--card: 240 10% 3.9%;
--card-foreground: 0 0% 98%;
--popover: 240 10% 3.9%;
--popover-foreground: 0 0% 98%;
--primary: 0 0% 98%;
--primary-foreground: 240 5.9% 10%;
--secondary: 240 3.7% 15.9%;
--secondary-foreground: 0 0% 98%;
--muted: 240 3.7% 15.9%;
--muted-foreground: 240 5% 64.9%;
--accent: 240 3.7% 15.9%;
--accent-foreground: 0 0% 98%;
--destructive: 0 62.8% 30.6%;
--destructive-foreground: 0 0% 98%;
--border: 240 3.7% 15.9%;
--input: 240 3.7% 15.9%;
--ring: 240 4.9% 83.9%;
}
Awesome! We are now all set up to use spartan/ui in our project. Let's leverage our Nx plugin one more time and add the button primitive to our project:
npx nx g @spartan-ng/nx:ui button
When prompted:
Once the plugin finishes, you will see that a new buildable library was added in your libs/spartan/button-helm
folder.
It contains the source code of the HlmButtonDirective
, which comes with a bunch of different styles that are applied through a HostBinding
based on the different inputs of the directive.
import { Directive, HostBinding, Input } from '@angular/core';
import { cva, VariantProps } from 'class-variance-authority';
import { hlm } from '@spartan-ng/ui-core';
import { ClassValue } from 'clsx';
const buttonVariants = cva(
'inline-flex items-center justify-center rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:opacity-50 disabled:pointer-events-none ring-offset-background',
{
variants: {
variant: {
default: 'bg-primary text-primary-foreground hover:bg-primary/90',
destructive: 'bg-destructive text-destructive-foreground hover:bg-destructive/90',
outline: 'border border-input hover:bg-accent hover:text-accent-foreground',
secondary: 'bg-secondary text-secondary-foreground hover:bg-secondary/80',
ghost: 'hover:bg-accent hover:text-accent-foreground',
link: 'underline-offset-4 hover:underline text-primary',
},
size: {
default: 'h-10 py-2 px-4',
sm: 'h-9 px-3 rounded-md',
lg: 'h-11 px-8 rounded-md',
icon: 'h-10 w-10',
},
},
defaultVariants: {
variant: 'default',
size: 'default',
},
}
);
type ButtonVariants = VariantProps<typeof buttonVariants>;
@Directive({
selector: '[hlmBtn]',
standalone: true,
})
export class HlmButtonDirective {
private _variant: ButtonVariants['variant'] = 'default';
@Input()
get variant(): ButtonVariants['variant'] {
return this._variant;
}
set variant(value: ButtonVariants['variant']) {
this._variant = value;
this._class = this.generateClasses();
}
private _size: ButtonVariants['size'] = 'default';
@Input()
get size(): ButtonVariants['size'] {
return this._size;
}
set size(value: ButtonVariants['size']) {
this._size = value;
this._class = this.generateClasses();
}
private _inputs: ClassValue = '';
@Input()
set class(inputs: ClassValue) {
this._inputs = inputs;
this._class = this.generateClasses();
}
@HostBinding('class')
private _class = this.generateClasses();
private generateClasses() {
return hlm(buttonVariants({ variant: this._variant, size: this._size }), this._inputs);
}
}
Note: Currently the plugin adds dependencies correctly, but their peer dependencies are not installed by Nx
Simply run npm i
after the @spartan-ng/nx:ui
call to make sure everything is installed correctly.
To use our new directive we simply add the directive to our button
in our src/app/app.component/ts
:
import { Component } from '@angular/core';
import { HlmButtonDirective } from '@spartan-ng/button-helm';
@Component({
standalone: true,
imports: [HlmButtonDirective],
selector: 'app-root',
template: `<button hlmBtn variant="outline">Hello from {{title}}</button>`
})
export class AppComponent {
title = 'sparta';
}
Then we start our development server with:
npm start
and see our beautifully styled spartan/ui button:
To change the appearance to another variant we simply add a variant
input to our <button hlmBtn>
:
import { Component } from '@angular/core';
import { HlmButtonDirective } from '@spartan-ng/button-helm';
@Component({
standalone: true,
imports: [HlmButtonDirective],
selector: 'app-root',
template: `<button hlmBtn variant="outline">Hello from {{title}}</button>`
})
export class AppComponent {
title = 'sparta';
}
Our styles will be updated accordingly and we see our outlined button:
With the release of the initial alpha version, there are 30 components available:
You can add new components the same way as we did for the button.
I also plan to create more blog posts and videos, which show how to use spartan/ui to build your user interface.
spartan/ui is still in alpha, so there is a long way (a marathon some history nerds might suggest) ahead of us. However, I am super excited that this project is finally getting off the ground and you get to try it out and provide me with incredibly valuable feedback. I hope spartan/ui becomes the shadcn of the Angular ecosystem and together with incredible projects like AnalogJs can bring a similar innovation explosion to all of us.
As always, do you have any further questions or suggestions for blog posts? What do you think of spartan? Could you see yourself adding it to your project? I am curious to hear your thoughts. Please don't hesitate to leave a comment or send me a message.
Finally, if you liked this article feel free to like and share it with others. If you enjoy my content follow me on Twitter or Github.