Adding common components. Card, Button, Input, List, ListItem
Using previously configured Storybook in this lesson we will create and prepare a few common components that later we’ll use in apps.
In this lesson, we'll create the most basic component we will use in later lessons.
Styles#
As I previously mentioned, we won't focus on CSS in this course. Please paste my styles into your repository to make our code look nice.
xxxxxxxxxx
/* You can add global styles to this file, and also import other style files */
@import url('https://fonts.googleapis.com/css2?family=Roboto:wght@400;700&display=swap');
:root {
--button: #38bdf8;
--button-hover: #0ea5e9;
--primary: #38bdf8;
--red: #ef4444;
--disabled: #d4d4d4;
--shadow: #bae6fd;
--border: #e5e5e5;
--border-width-sm: 1px;
--border-width: 2px;
--border-width-lg: 4px;
--radius: 5px;
--size-1: 5px;
--size-2: 10px;
--size-3: 15px;
--font-small: 12px;
--font-regular: 14px;
}
* {
font-family: 'Roboto', sans-serif;
font-size: var(--font-regular);
box-sizing: border-box;
}
html, body {
padding: 0;
margin: 0;
}
label {
font-size: var(--font-small);
font-weight: bold;
display: block;
margin-bottom: var(--size-1);
}
input {
padding: var(--size-2) var(--size-3);
border-radius: var(--radius);
border: var(--border-width) solid var(--border);
outline: none;
}
input:hover,
input:active,
input:focus {
border-color: var(--primary);
}
h1 {
font-size: 48px;
}
h2 {
font-size: 38px;
}
h3 {
font-size: 30px;
}
h4 {
font-size: 24px;
}
hr {
border-top: 1px solid var(--border);
margin: var(--size-3) 0;
}
.margin-top {
margin-top: var(--size-3);
}
shared-button button {
background: none;
border: none;
outline: none;
color: white;
padding: 0;
margin: 0;
}
shared-button {
display: inline-block;
padding: var(--size-2) var(--size-3);
background: var(--button);
transition: background-color .2s;
border: var(--border-width) solid var(--button);
border-radius: var(--radius);
cursor: pointer;
}
shared-button:hover {
background: var(--button-hover);
border: var(--border-width) solid var(--button-hover);
}
shared-card {
display: flex;
flex-direction: column;
border: var(--border-width-sm) solid var(--border);
border-radius: var(--radius);
background: white;
overflow: hidden;
}
shared-card img {
width: 100%;
flex-grow: 1;
}
shared-card .card-body {
padding: var(--size-3);
}
.card-active {
box-shadow: 0 0 var(--size-3) var(--shadow);
}
shared-list shared-list-item + shared-list-item {
border-top: var(--border-width) solid var(--border);
}
shared-list-item {
display: block;
padding: var(--size-1);
}
shared-room-background {
display: block;
width: 100%;
height: 100%;
}
shared-room-background .image {
width: 100%;
height: 100%;
background-size: cover;
}
shared-room-list {
display: grid;
grid-template-columns: repeat(5, 1fr);
gap: var(--size-3);
}
shared-room-list shared-card {
cursor: pointer;
aspect-ratio: 1 / 1;
}
.text-center {
text-align: center;
}
.add-room-button {
width: 100%;
height: 100%;
}
.add-room-button .card-body {
display: flex;
align-items: center;
justify-content: center;
height: 100%;
}
shared-room-timeline-list shared-list-item {
display: flex;
justify-content: space-between;
}
shared-room-timeline-list .live {
color: var(--red);
font-weight: bold;
}
shared-room-timeline-list .past {
text-decoration: line-through;
color: var(--disabled);
}
admin-room-list,
admin-room-edit,
room-list {
display: block;
margin: var(--size-3) auto;
width: 1200px;
}
.background-list {
display: grid;
gap: var(--size-3);
grid-template-columns: repeat(6, 1fr);
}
.background-list shared-room-background {
width: 100%;
aspect-ratio: 3 / 2;
border-radius: var(--radius);
overflow: hidden;
}
.input-row {
display: flex;
align-items: center;
justify-content: center;
gap: var(--size-3);
}
.selected-background {
border: var(--border-width-lg) solid var(--primary);
}
.room-name {
width: 50%;
}
room-details {
display: block;
width: 100vw;
height: 100vh;
position: relative;
}
room-details shared-room-background {
position: absolute;
left: 0;
top: 0;
width: 100%;
height: 100%;
}
room-details .back-button {
position: absolute;
top: var(--size-3);
left: var(--size-3);
}
room-details .bottom-row {
position: absolute;
display: flex;
gap: var(--size-3);
padding: var(--size-3);
bottom: 0;
left: 0;
width: 100%;
max-height: 80vh;
}
room-details .bottom-row shared-card {
flex-grow: 1;
}
To make it work with Storybook,
place these styles in apps/storybook/src/styles.css
and then modify the file apps/storybook/.storybook/preview.ts
to look in the code block below.
It's a simple Webpack trick to use style-loader
which is not a standard thing for Angular application
and import a CSS file as a global style.
xxxxxxxxxx
import '!style-loader!css-loader!../src/styles.css';
ButtonComponent
#
We have a few buttons in our design specification. The button component is relatively simple. In fact, it's just a wrapper for the native button element.
Let's start with a standard cleanup.
Go to libs/components/button/src/lib
and remove the automatically generated module.
We don't need that because we're going to use a new Angular feature, standalone components.
In the same directory, create a file called button.component.ts
.
Now, we can focus on component code.
First, create an empty component with an empty template. You can use a generator if you want.
Then, let's switch to the Storybook.
In apps/storybook/src/app/
, I'm creating a new directory button
.
Next,
I'm creating a new Angular component called StorybookButtonComponent
which I will use as a wrapper in my Storybook story.
xxxxxxxxxx
import {Component} from "@angular/core";
import {ButtonComponent} from "shared/components/button";
@Component({
selector: 'storybook-button',
template: `
<div style="margin-bottom: 15px;">
<shared-button>Click me!</shared-button>
</div>
<div>
<shared-button>
← <!-- I'm using the unicode symbol for arrow, but feel free to import any icon set you like -->
</shared-button>
</div>
`,
imports: [
ButtonComponent
],
standalone: true
})
export class StorybookButtonComponent {
}
In this wrapper component, I'm placing all planned usages of the component.
Of course, we need to configure a story,
so next to file storybook-button.component.ts
, create a new file button.stories.ts
.
xxxxxxxxxx
import {Meta} from '@storybook/angular';
import {StorybookButtonComponent} from "./storybook-button.component";
export default {
title: 'ButtonComponent',
component: StorybookButtonComponent,
} as Meta<StorybookButtonComponent>;
export const Component = {
render: (args: StorybookButtonComponent) => ({
props: args,
})
};
At this point, we have everything we need to create our component. We have an empty component where we can place our code and a quick preview in Storybook.This way of creating components has a huge advantage because it allows you to focus only on a smart piece of application. You won't be distracted by anything, and it's easier to create a perfect component. In addition, as an extra benefit, you automatically document your work.
After all my changes, the code looks like this.
This lesson preview is part of the Next-Level Angular Apps with NX course and can be unlocked immediately with a \newline Pro subscription or a single-time purchase. Already have access to this course? Log in here.
Get unlimited access to Next-Level Angular Apps with NX, plus 70+ \newline books, guides and courses with the \newline Pro subscription.
