Simply copy and paste the following command line in your terminal to create your first Strapi project.
npx create-strapi-app
my-project
This tutorial is outdated since it is using Strapi v3.
This month, we published two tutorials on how to create a blog with Nuxt.js and how to create a blog with Next.js with Strapi. But as the saying goes, good things come in threes!
So we decided to also create a tutorial for Angular developers on how to create a blog using Strapi and Apollo, and we hope you'll enjoy using it.
If you are familiar with our blog you must have seen that we've released a series of tutorials on how to make blogs using Strapi with a lot of frontend frameworks: Gatsby Old, Gatsby new, React, Next.js, Vue or Nuxt
Angular is a TypeScript-based open-source web application framework led by the Angular Team at Google and by a community of individuals and corporations. Angular is a complete rewrite from the same team that built AngularJS.
The goal here is to be able to create a blog website using Strapi as the backend, Angular for the frontend, and Apollo for requesting the Strapi API with GraphQL.
The source code is available on GitHub.
To follow this tutorial, you'll need to have Strapi and Angular installed on your computer, but don't worry, we are going to install these together!
This tutorial uses Strapi v3.0.0-beta.17.8.
You need to have node v.12 installed and that's all.
Create a blog-strapi folder and get inside!
mkdir blog-strapi && cd blog-strapi
So that's the easy part. Since the beta.9, we have an awesome package create strapi-app that allows you to create a Strapi project in seconds without installing Strapi globally so let's try it out.
Note: for this tutorial, we will use
yarn
as your package manager.
yarn create strapi-app backend --quickstart --no-run
.
This single command line will create all you need for your back-end. Make sure to add the --no-run
flag as it will prevent your app from automatically starting the server because SPOILER ALERT: we need to install some awesome Strapi plugins.
Now that you know that we need to install some plugins to enhance your app, let's install one of our most popular. The graphql
plugin:
yarn strapi install graphql
Once the installation is completed, you can finally start your Strapi server strapi dev
and create your first Administrator.
Don't forget that Strapi is running on http://localhost:1337.
Nice! Now that Strapi is ready, we are going to create your Angular application.
Well, the easiest part has been completed, let's get our hands dirty developing our blog!
Angular setup
Install the Angular CLI by running the following command:
npm install -g @angular/cli
Create an Angular frontend
project, and accept to add Angular routing
:
ng new frontend
Now your project is all set, you can dive in it and run the server:
1
2
cd frontend
ng serve
As you might want people to read your blog or to make it "cute & pretty" we will use a popular CSS framework for styling: UiKit and also Apollo to query Strapi with GraphQL:
Dependencies setup
Make sure you are in the frontend folder before running the following commands:
Apollo setup
Install apollo-angular
by running the following command:
ng add apollo-angular
Go in src/app/graphql.modules.ts
and modify the uri
const:
const uri = 'http://localhost:1337/graphql'
;
This way, Apollo will call this address, which is the GraphQL route of your Strapi API.
UIkit setup
UIkit is a lightweight and modular front-end framework for developing fast and powerful web interfaces.
Install UIkit by running the following command:
yarn add uikit
Open your angular.json
file and add the following code in the first scripts
array. It should look like this:
1
2
3
4
5
"scripts": [
"node_modules/jquery/dist/jquery.min.js",
"node_modules/uikit/dist/js/uikit.min.js",
"node_modules/uikit/dist/js/uikit-icons.min.js"
]
src/style.css
and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
/* You can add global styles to this file, and also import other style files */
@import "../node_modules/uikit/dist/css/uikit.min.css";
@import "../node_modules/uikit/dist/css/uikit.css";
@import "../node_modules/uikit/dist/css/uikit-core.css";
@import url("https://fonts.googleapis.com/css?family=Staatliches");
a {
text-decoration: none;
}
h1 {
font-family: Staatliches;
font-size: 120px;
}
#category {
font-family: Staatliches;
font-weight: 500;
}
#title {
letter-spacing: 0.4px;
font-size: 22px;
font-size: 1.375rem;
line-height: 1.13636;
}
#banner {
margin: 20px;
height: 800px;
}
#editor {
font-size: 16px;
font-size: 1rem;
line-height: 1.75;
}
.uk-navbar-container {
background: #fff !important;
font-family: Staatliches;
}
img:hover {
opacity: 1;
transition: opacity 0.25s cubic-bezier(0.39, 0.575, 0.565, 1);
}
Note: You're importing UIkit style and a beautiful font that you'll use for this tutorial.
Finally! We are now going to create the data structure of our article by creating an Article
content type.
Content Type Builder
link in the sidebar.Create new content-type
and call it article
.Now you'll be asked to create all the fields for your content-type:
title
with type Text (required)content
with type Rich Text (required)image
with type Media (Single image) and (required)published_at
with type date (required)Press Save! Here you go, your first content type has been created. Now you may want to create your first article, but we have one thing to do before that: Grant access to the article content type.
public
role.find
and findone
routes and save.Awesome! You should be ready to create your first article right now and fetch it on the GraphQL Playground.
Here's an example:
Great! Now you may want to reach the moment when you can actually fetch your articles through the API!
Isn't that cool! You can also play with the GraphQL Playground.
You may want to assign a category to your article (news, trends, opinion). You are going to do this by creating another content type in Strapi.
category
content type with the following fieldname
with type TextPress save!
Category has many Articles
like below:public
role. And check the category find
and findone
routes and save.Now you'll be able to select a category for your article in the right sidebox.
Now that we are good with Strapi let's work on the frontend part!
Let's create our first component and see what happens.
Create a nav
component by running the following command:
ng generate c nav --skip-import
Four files have been created, but we'll only work with the .html
and .ts
files. In fact your .html
file is the view of your component and your .ts
file defines what your component will basically do.
I'm going to explain this component in detail, but it will be the same procedures for the other ones.
First of all, you want to fetch categories you've created with Strapi. To do so, you'll create a graphql request and use it in your component.
app/apollo/queries/category
directoryapp/apollo/queries/category/categories.js
file and paste the following code inside:1
2
3
4
5
6
7
8
9
10
11
12
import gql from "graphql-tag";
const CATEGORIES_QUERY = gql`
query Categories {
categories {
id
name
}
}
`;
export default CATEGORIES_QUERY;
Here, you are simply fetching the id and the name of every category.
Now lets import this query and the necessary packages:
nav/nav.component.ts
file:1
2
3
4
import { Apollo } from "apollo-angular";
import gql from "graphql-tag";
import CATEGORIES_QUERY from "../apollo/queries/category/categories";
import { Subscription } from "rxjs";
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
export class NavComponent implements OnInit {
data: any = {};
loading = true;
errors: any;
private queryCategories: Subscription;
constructor(private apollo: Apollo) {}
ngOnInit() {
this.queryCategories = this.apollo
.watchQuery({
query: CATEGORIES_QUERY
})
.valueChanges.subscribe(result => {
this.data = result.data;
this.loading = result.loading;
this.errors = result.errors;
});
}
ngOnDestroy() {
this.queryCategories.unsubscribe();
}
}
What you're doing here is fetching all your categories with Apollo on the component initialization using the query you made. Let's write our Navbar!
nav/nav.component.html
and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<nav class="uk-navbar-container" uk-navbar>
<div class="uk-navbar-left">
<ul class="uk-navbar-nav">
<li class="uk-active"><a href="#">Strapi blog</a></li>
</ul>
</div>
<div class="uk-navbar-right">
<ul *ngIf="data" class="uk-navbar-nav">
<li *ngFor="let category of data.categories" class="uk-active">
<a
routerLink="/category/{{ category.id }}"
routerLinkActive="active"
class="uk-link-reset"
>
{{ category.name }}
</a>
</li>
</ul>
</div>
</nav>
Inside the view, we can access our categories with data.categories
. We can then iterate on it in order to display them all *ngFor="let category of data.categories"
.
Perfect! Now we need to two things!
app.modules.ts
file and import your newly created nav component by importing it and adding it to the declarations
array like this:1
2
3
4
5
6
7
8
9
...
import { NavComponent } from "./nav/nav.component";
...
declarations: [
AppComponent,
NavComponent
],
...
app.component.html
file and declare your nav component by pasting the following code:1
2
3
<app-nav></app-nav>
<router-outlet></router-outlet>
This way, your nav component will always be displayed on your application
Note The current code is not suited to display a lot of categories as you may encounter a UI issue. Since this blog post is supposed to be short, I will let you improve the code to maybe add a lazy load or something.
For now, the links are not working, you'll work on it later on the tutorial ;)
Note It will now be the same exact logic for every other component:
.ts
file.html
fileapp.module.ts
This component will display all your articles and will be displayed on the main page.
Create an articles
component by running the following command:
ng generate c articles --skip-import
Create a app/apollo/queries/article
directory.
app/apollo/queries/article/articles.js
file and paste the following code inside:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import gql from "graphql-tag";
const ARTICLES_QUERY = gql`
query Articles {
articles {
id
title
category {
id
name
}
image {
url
}
}
}
`;
export default ARTICLES_QUERY;
articles/articles.component.ts
file and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
import { Component, OnInit } from "@angular/core";
import { Apollo } from "apollo-angular";
import gql from "graphql-tag";
import ARTICLES_QUERY from "../apollo/queries/article/articles";
import { Subscription } from "rxjs";
@Component({
selector: "app-articles",
templateUrl: "./articles.component.html",
styleUrls: ["./articles.component.css"]
})
export class ArticlesComponent implements OnInit {
data: any = {};
loading = true;
errors: any;
leftArticlesCount: any;
leftArticles: any[];
rightArticles: any[];
private queryArticles: Subscription;
constructor(private apollo: Apollo) {}
ngOnInit() {
this.queryArticles = this.apollo
.watchQuery({
query: ARTICLES_QUERY
})
.valueChanges.subscribe(result => {
this.data = result.data;
this.leftArticlesCount = Math.ceil(this.data.articles.length / 5);
this.leftArticles = this.data.articles.slice(0, this.leftArticlesCount);
this.rightArticles = this.data.articles.slice(
this.leftArticlesCount,
this.data.articles.length
);
this.loading = result.loading;
this.errors = result.errors;
});
}
ngOnDestroy() {
this.queryArticles.unsubscribe();
}
}
What you are doing here is fetching your articles thanks to the query you wrote, then you separate them in two arrays. In fact, leftArticles
will contain articles that will be displayed on the left side and rightArticles
on the right side.
articles/articles.component.html
and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
<div class="uk-section">
<div class="uk-container uk-container-large">
<h1>Strapi blog</h1>
<div class="uk-child-width-1-2" uk-grid>
<div>
<a
routerLink="/article/{{ article.id }}"
routerLinkActive="active"
*ngFor="let article of leftArticles"
class="uk-link-reset"
>
<div class="uk-card uk-card-muted">
<div *ngIf="article.image" class="uk-card-media-top">
<img
src="http://localhost:1337{{ article.image.url }}"
alt=""
height="100"
/>
</div>
<div class="uk-card-body">
<p
id="category"
*ngIf="article.category"
class="uk-text-uppercase"
>
{{ article.category.name }}
</p>
<p id="title" class="uk-text-large">{{ article.title }}</p>
</div>
</div>
</a>
</div>
<div>
<div class="uk-child-width-1-2@m uk-grid-match" uk-grid>
<a
routerLink="/article/{{ article.id }}"
routerLinkActive="active"
*ngFor="let article of rightArticles"
class="uk-link-reset"
>
<div class="uk-card uk-card-muted">
<div *ngIf="article.image" class="uk-card-media-top">
<img
src="http://localhost:1337{{ article.image.url }}"
alt=""
height="100"
/>
</div>
<div class="uk-card-body">
<p
id="category"
*ngIf="article.category"
class="uk-text-uppercase"
>
{{ article.category.name }}
</p>
<p id="title" class="uk-text-large">{{ article.title }}</p>
</div>
</div>
</a>
</div>
</div>
</div>
</div>
</div>
app.modules.ts
file, import your component and add it to the declarations array like this:1
2
3
4
5
6
7
8
9
...
import { ArticlesComponent } from "./articles/articles.component";
...
declarations: [
AppComponent,
ArticlesComponent,
NavComponent,
],
It's time to use the router from angular. You want to display this ArticlesComponent
on your main page
RouterModule
, Routes
from Angular and define the following appRoutes
:1
2
3
4
5
6
import { RouterModule, Routes } from "@angular/router";
...
const appRoutes: Routes = [
{ path: "", component: ArticlesComponent }
];
imports
array:1
2
3
4
5
6
7
8
9
...
imports: [
RouterModule.forRoot(appRoutes, { enableTracing: true }),
BrowserModule,
AppRoutingModule,
GraphQLModule,
HttpClientModule
],
...
Great! You can now list every article on your main page.
Nothing happens if you click on any article, let's change that!
Create an article component by running the following command:
ng generate c article --skip-import
app/apollo/queries/article/article.js
file and paste the following code inside:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import gql from "graphql-tag";
const ARTICLE_QUERY = gql`
query Articles($id: ID!) {
article(id: $id) {
id
title
content
image {
url
}
category {
id
name
}
published_at
}
}
`;
export default ARTICLE_QUERY;
article/article.component.ts
and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
import { Component, OnInit } from "@angular/core";
import { Apollo } from "apollo-angular";
import gql from "graphql-tag";
import ARTICLE_QUERY from "../apollo/queries/article/article";
import { ActivatedRoute } from "@angular/router";
import { Subscription } from "rxjs";
@Component({
selector: "app-article",
templateUrl: "./article.component.html",
styleUrls: ["./article.component.css"]
})
export class ArticleComponent implements OnInit {
data: any = {};
loading = true;
errors: any;
private queryArticle: Subscription;
constructor(private apollo: Apollo, private route: ActivatedRoute) {}
ngOnInit() {
this.queryArticle = this.apollo
.watchQuery({
query: ARTICLE_QUERY,
variables: {
id: this.route.snapshot.paramMap.get("id")
}
})
.valueChanges.subscribe(result => {
this.data = result.data;
this.loading = result.loading;
this.errors = result.errors;
});
}
ngOnDestroy() {
this.queryArticle.unsubscribe();
}
}
The content of our articles are not in Markdown, you'll need to install a dependency to display the content of your articles in Markdown.
Install ngx-markdown
by running the following command:
yarn add ngx-markdown
Open your app.modules.ts
, import add the MarkdownModule to your imports
array:
1
2
3
4
5
6
7
8
9
10
11
12
13
...
import { MarkdownModule } from "ngx-markdown";
...
imports: [
MarkdownModule.forRoot(),
RouterModule.forRoot(appRoutes, { enableTracing: true }),
BrowserModule,
AppRoutingModule,
GraphQLModule,
HttpClientModule
],
...
Awesome! Let's move on to the next step.
article/article.component.html
and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
<div
id="banner"
class="uk-height-small uk-flex uk-flex-center uk-flex-middle uk-background-cover uk-light uk-padding"
[style.background-image]="
'url(http://localhost:1337' + data.article.image.url + ')'
"
uk-img
>
<h1>{{ data.article.title }}</h1>
</div>
<div class="uk-section">
<div class="uk-container uk-container-small">
<p>
<markdown ngPreserveWhitespaces>
{{ data.article.content }}
</markdown>
</p>
<p></p>
</div>
</div>
app.modules.ts
file, import your component, add it to the declarations array and add another route for this component:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
...
import { ArticleComponent } from "./article/article.component";
...
const appRoutes: Routes = [
{ path: "", component: ArticlesComponent },
{ path: "article/:id", component: ArticleComponent },
];
...
declarations: [
AppComponent,
NavComponent,
ArticlesComponent,
ArticleComponent
],
...
Awesome you can now visit any article!
With this component, you'll be able to fetch articles depending on the selected category you are visiting.
Create a category
component by running the following command:
ng generate c category --skip-import
Create a app/apollo/queries/category/articles.js
file containing the following:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
import gql from "graphql-tag";
const CATEGORY_ARTICLES_QUERY = gql`
query Category($id: ID!) {
category(id: $id) {
id
name
articles {
id
title
content
image {
url
}
category {
id
name
}
}
}
}
`;
export default CATEGORY_ARTICLES_QUERY;
category/category.component.ts
file and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
import { Component, OnInit } from "@angular/core";
import { Apollo } from "apollo-angular";
import gql from "graphql-tag";
import CATEGORY_ARTICLES_QUERY from "../apollo/queries/category/articles";
import { ActivatedRoute, ParamMap } from "@angular/router";
import { Subscription } from "rxjs";
@Component({
selector: "app-category",
templateUrl: "./category.component.html",
styleUrls: ["./category.component.css"]
})
export class CategoryComponent implements OnInit {
data: any = {};
category: any = {};
loading = true;
errors: any;
leftArticlesCount: any;
leftArticles: any[];
rightArticles: any[];
id: any;
private queryCategoriesArticles: Subscription;
constructor(private apollo: Apollo, private route: ActivatedRoute) {}
ngOnInit() {
this.route.paramMap.subscribe((params: ParamMap) => {
this.id = params.get("id");
this.queryCategoriesArticles = this.apollo
.watchQuery({
query: CATEGORY_ARTICLES_QUERY,
variables: {
id: this.id
}
})
.valueChanges.subscribe(result => {
this.data = result.data
this.category = this.data.category.name
console.log(this.data)
this.leftArticlesCount = Math.ceil(this.data.category.articles.length / 5);
this.leftArticles = this.data.category.articles.slice(0, this.leftArticlesCount);
this.rightArticles = this.data.category.articles.slice(
this.leftArticlesCount,
this.data.category.articles.length
);
this.loading = result.loading;
this.errors = result.errors;
});
});
}
ngOnDestroy() {
this.queryCategoriesArticles.unsubscribe();
}
}
Here you are doing the same thing that you did in your ArticlesComponent
, but you are fetching articles depending on the selected category.
category/category.component.html
file and paste the following code:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
<div class="uk-section">
<div class="uk-container uk-container-large">
<h1>{{ category }}</h1>
<div class="uk-child-width-1-2" uk-grid>
<div>
<a
routerLink="/article/{{ article.id }}"
routerLinkActive="active"
*ngFor="let article of leftArticles"
class="uk-link-reset"
>
<div class="uk-card uk-card-muted">
<div *ngIf="article.image" class="uk-card-media-top">
<img
src="http://localhost:1337{{ article.image.url }}"
alt=""
height="100"
/>
</div>
<div class="uk-card-body">
<p
id="category"
*ngIf="article.category"
class="uk-text-uppercase"
>
{{ article.category.name }}
</p>
<p id="title" class="uk-text-large">{{ article.title }}</p>
</div>
</div>
</a>
</div>
<div>
<div class="uk-child-width-1-2@m uk-grid-match" uk-grid>
<a
routerLink="/article/{{ article.id }}"
routerLinkActive="active"
*ngFor="let article of rightArticles"
class="uk-link-reset"
>
<div class="uk-card uk-card-muted">
<div *ngIf="article.image" class="uk-card-media-top">
<img
src="http://localhost:1337{{ article.image.url }}"
alt=""
height="100"
/>
</div>
<div class="uk-card-body">
<p
id="category"
*ngIf="article.category"
class="uk-text-uppercase"
>
{{ article.category.name }}
</p>
<p id="title" class="uk-text-large">{{ article.title }}</p>
</div>
</div>
</a>
</div>
</div>
</div>
</div>
</div>
app.modules.ts
file, import your component, add it to the declarations array and add another route for this component:1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
...
import { CategoryComponent } from "./category/category.component";
...
const appRoutes: Routes = [
{ path: "", component: ArticlesComponent },
{ path: "article/:id", component: ArticleComponent },
{ path: "category/:id", component: CategoryComponent }
];
...
declarations: [
AppComponent,
ArticlesComponent,
ArticleComponent,
NavComponent,
CategoryComponent
],
...
Awesome! You can now navigate through categories :)
Huge congrats, you successfully achieved this tutorial. I hope you enjoyed it!
Still hungry?
Feel free to add additional features, adapt this project to your own needs, and give your feedback in the comments section.
If you want to deploy your application, check our deployment documentation.
Contribute and collaborate on educational content for the Strapi Community.
Can't wait to see your contribution!
Please note: Since we initially published this blog post, we released new versions of Strapi and tutorials may be outdated. Sorry for the inconvenience if it's the case. Please help us by reporting it here.
Get started with Strapi by creating a project using a starter or trying our live demo. Also, consult our forum if you have any questions. We will be there to help you.
See you soon!Maxime started to code in 2015 and quickly joined the Growth team of Strapi. He particularly likes to create useful content for the awesome Strapi community. Send him a meme on Twitter to make his day: @MaxCastres