title: Categories
---
summary: A simple pattern for implementing categories.
---
body:

Because Lektor is based on the idea of mirroring the tree structure from the
file system to the final website it sometimes might not be quite clear how
one could achieve any form of filtering (for instance categories).  However
this is actually very easily achieved through the concept of “child
replacement”.  This will give you a quick introduction to how this feature
can be used to implement categories.

## Basic Setup

In this case we assume you have a bunch of projects that can exist in
different categories and it should be possible to see which projects belong
to which category.

## The Models

So we will end up with four models: `projects.ini`, `project.ini`,
`project-categories.ini` and `project-category.ini`.  Here is how they look:

### `projects.ini`

```ini
[model]
name = Projects
label = Projects
hidden = yes
protected = yes

[children]
model = project
order_by = -date, name
```

### `project.ini`

```ini
[model]
name = Project
label = {{ this.name }}
hidden = yes

[fields.name]
label = Name
type = string

[fields.date]
label = Date
type = date

[fields.description]
label = Description
type = markdown

[fields.categories]
label = Categories
type = checkboxes
source = site.query('/project-categories')
```

The above models should be mostly clear.  What is probably not entirely clear
is the `source` parameter for the categories.  Essentially we instruct the
admin panel to fill the selection for the checkboxes from the child pages
below the `project-categories` folder.  This is where we will maintain the
categories.

### `project-categories.ini`

```ini
[model]
name = Project Categories
label = Project Categories
hidden = yes
protected = yes

[children]
model = project-category
order_by = name
```

### `project-category.ini`

```ini
[model]
name = Project Category
label = {{ this.name }}
hidden = yes

[children]
replaced_with = site.query('/projects').filter(F.categories.contains(this))

[fields.name]
label = Name
type = string
```

So this is where the magic lives.  the `replaced_with` key in the `[children]`
section tells Lektor to ignore the child pages that would normally exist below
the path but to substitute them with another query.  In this case we replace
them with all the projects where the category (`this`) is in the checked
set of categories (`F.categories`).

## Initializing The Folders

Now that we have the models, we need to initialize the folders.  This is
necessary because most models are hidden which means they cannot be selected
from the admin interface.  We need the following things:

### `projects/contents.lr`

```
_model: projects
```

### `project-categories/contents.lr`

```
_model: project-categories
----
_slug: projects/categories
```

Here we also tell the system that we want to move the page from
`project-categories/` to `projects/categories/` which looks nicer.  This
means a category named `Foo` will then end up as `projects/categories/foo/`.

## Creating Categories

Now we can head to the admin panel to create some categories.  Each category
that is created will become available for new projects in the category
selection.

## The Templates

To render out all the pages we probably want to use some macros to reuse
markup that appears on multiple pages.

### `projects.html`

```html+jinja
{% extends "layout.html" %}
{% from "macros/projects.html" import
  render_project_list, render_category_nav %}
{% block title %}Projects{% endblock %}
{% block body %}
  <h1>Projects</h1>
  {{ render_category_nav(active=none) }}
  {{ render_project_list(this.children) }}
{% endblock %}
```

### `project.html`

```html+jinja
{% extends "layout.html" %}
{% block title %}{{ this.name }}{% endblock %}
{% block body %}
  <h1>{{ this.name }}</h1>
  <div class="description">{{ this.description }}</div>
{% endblock %}
```

### `project-categories.html`

```html+jinja
{% extends "layout.html" %}
{% from "macros/projects.html" import render_category_nav %}
{% block title %}Project Categories{% endblock %}
{% block body %}
  <h1>Project Categories</h1>
  {{ render_category_nav(active=none) }}
{% endblock %}
```

### `project-category.html`

```html+jinja
{% extends "layout.html" %}
{% from "macros/projects.html" import render_category_nav,
   render_project_list %}
{% block title %}Project Category {{ this.name }}{% endblock %}
{% block body %}
  <h1>Project Category {{ this.name }}</h1>
  {{ render_category_nav(active=this._id) }}
  {{ render_project_list(this.children) }}
{% endblock %}
```

### `macros/projects.html`

```html+jinja
{% macro render_category_nav(active=none) %}
  <ul>
  {% for category in site.query('/project-categories') %}
    <li{% if category._id == active %} class="active"{% endif
      %}><a href="{{ category|url }}">{{ category.name }}</a></li>
  {% endfor %}
  </ul>
{% endmacro %}

{% macro render_project_list(projects) %}
  <ul>
  {% for project in projects %}
    <li><a href="{{ project|url }}">{{ project.name }}</a></li>
  {% endfor %}
  </ul>
{% endmacro %}
```