Formatting your tutorial

Get to know the tools needed to write your own tutorials.

Creating a new tutorial

You accept that your contributions are licensed under the CC-BY-SA-4.0.

Each page of the website is located in content, while documentation-specific pages are located in content/docs.

To create a new tutorial, we use what the Hugo documentation calls a branch bundle.

You will need to create a folder inside content/docs, for example, content/docs/getting-started/installation/, and inside it, create an _index.md file. This will be the index page for the tutorial, the section, which will group other pages inside.

So you would be creating the following, for example:

  • content/docs/getting-started/installation/_index.md

To add new pages to the tutorial, you can then create individual Markdown files containing the name of the tutorial page. For example:

  • content/docs/getting-started/installation/howto.md
  • content/docs/getting-started/installation/configuring.md

To add additional content like screenshots or example code for each individual page, you can create a folder with the same name as the individual Markdown file and then add screenshots and code to it, for example:

  • content/docs/getting-started/installation/howto/code.qml
  • content/docs/getting-started/installation/howto/image.png
  • content/docs/getting-started/installation/configuring/code.qml
  • content/docs/getting-started/installation/configuring/image.png

It should look like so:

content/
└── docs/
    └── getting-started/
        ├── _index.md
        └── installation/       <-- Your new tutorial
            ├── _index.md       <-- Your main introduction page
            ├── configuring.md
            ├── howto.md
            ├── configuring/
            │   ├── code.qml
            │   └── image.png
            └── howto/
                ├── code.qml
                └── image.png

This is the structure of a typical page:

---
title: Installation of the development libraries # title of the page
linkTitle: Installation # title in the sidebar, no need to add if the same as `title`
description:
  > In this tutorial you will learn.... # Short summary of the page (displayed in the google description and as subtitle)
weight: 1 # ordering of the articles in the section
authors:
  - SPDX-FileCopyrightText: 2023 Your name <email@address>
SPDX-License-Identifier: CC-BY-SA-4.0
---

## Introduction

Normal markdown content

The minimum requirements for a page are:

  • title:
  • weight:
  • SPDX-FileCopyrightText:
  • SPDX-License-Identifier:

Other available options are:

  • group:, which lists sections with the same name under the same group. It can be seen in action in the Kirigami tutorial, with the groups Introduction, Style, Components, and Advanced. The groups need to be listed in the _index.md file of the tutorial.
  • aliases:, which creates aliases that can be used to shorten links. This is useful when linking to that page from elsewhere, or to keep old links working when content is moved.

Hugo shortcodes

There are some custom shortcodes that can be used to create more complex content.

They can be identified by their characteristic {{< >}}, which have HTML tags inside <>. Certain shortcodes require closing tags.

For readability, we add spaces between the {{< >}} and the tag, e.g. {{< myshortcode parameter="" >}}.

Do not add spaces between the { and <, or between > and } when using shortcodes. In other words, don't do {{ <myshortcode parameter=""> }}.

The most important shortcodes are as follows.

readfile

Displays the contents of a file and applies syntax highlighting to it. It has five parameters, two of which are mandatory and three optional:

  • file (mandatory) is the file to be displayed. If the file is in the bundle of the content file you are writing, you can specify a relative path; otherwise, you need to specify an absolute path starting from the root of your project directory (namely /content/docs).
  • highlight (mandatory) is the language used for syntax highlighting.
  • start (optional) is the first line that should be displayed. By default this is 1, which means starting from the first line.
  • lines (optional) is how many lines should be displayed. By default this is 0, which displays all lines from start to the end of the file.
  • emphasize (optional) is a set of line numbers or line number ranges to be highlighted. Note that they are relative to the displayed lines, not to the absolute file line numbers (important in case file is not displayed from the start). For example, if start="5" and emphasize="3 6-7", lines 7, 10 and 11 of the file will be highlighted. Also note that like start, emphasize starts from 1, not 0.

For example, in any content file you can write:

{{< readfile file="/content/docs/getting-started/kirigami/introduction-getting_started/src/qml/Main.qml" highlight="qml" start=17 lines=9 emphasize="1-2 7" >}}

which will be rendered as:

17
18
19
20
21
22
23
24
25
    // and provides additional context for the translators
    title: i18nc("@title:window", "Hello World")

    // Set the first page that will be loaded when the app opens
    // This can also be set to an id of a Kirigami.Page
    pageStack.initialPage: Kirigami.Page {
        Controls.Label {
            // Center label horizontally and vertically within parent object
            anchors.centerIn: parent

Since /content/docs/getting-started/kirigami/introduction-getting_started/ is a bundle, to show the file /content/docs/getting-started/kirigami/introduction-getting_started/index.md you can also write as below and achieve the same result:

{{< readfile file="src/qml/main.qml" highlight="qml" start=17 lines=9 emphasize="1-2 7" >}}

Commonly used highlighting options are:

  • cpp
  • qml
  • cmake
  • python
  • ini

This shortcode is used once to display the file, so no closing tag is used.

snippet

Displays the contents of a remote file and applies syntax highlighting to it.

It has mandatory parameters:

  • repo: The repository as located on Invent.
  • file: The file in the repository that will be displayed.
  • part: The section of the file in the repository that will be displayed. It requires special formatting. If it is not specified, the whole file will be shown.
  • lang: The language of the snippet, for syntax highlighting.

For example:

ThreadWeaver belongs to the Frameworks group, so its repo value would be frameworks/threadweaver.

The file helloworld.cpp, which we want to display, is located in examples/HelloWorld/HelloWorld.cpp.

The part requires special formatting added to the example code in the remote repository, namely:

  • //@@snippet_begin(the-snippet-name)
  • //@@snippet_end

In the ThreadWeaver repository, we can see that the snippet is called "sample-helloworld", and that should be the value passed to the part.

We can see that the file is written in C++, so the lang value should be cpp.

The shortcode therefore is written like so:

{{< snippet repo="frameworks/threadweaver" file="examples/HelloWorld/HelloWorld.cpp" part="sample-helloworld" lang="cpp" >}}

The last requirement for this shortcode to work is to add download_file('group-name/repo-name', 'path/to/file.cpp') to the Python script in scripts/extract-plasma-applet-config-keys.py, and then following the instructions in the KDE Developer Repository.

alert

Displays a block of text with optional title and background color.

If no title or color is specified, it defaults to no title and color "info".

{{< alert title="Note" color="info" >}}
Text you want to display in the alert.
{{< /alert >}}

Note that this shortcode requires a closing tag, or else it will envelop the whole page into an alert.

The available colors are info, success, warning, and error.

Usage suggestions:

  • Use info to display general information or additional content:

  • Use success to display advice or important pieces of information, such as tips or best practices:

  • Use warning to warn the reader that a certain action can have dangerous results if misused:

  • Use error to display content that is of very high importance:

In cases where your alert is too long and might disrupt the flow of the text, you can make it collapsible using Markdown's details summary.

{{< alert title="Tip" color="success" >}}
<details>
<summary>Click here to find more about this feature!</summary>
This content was hidden!
</details>
{{< /alert >}}

Hugo provides the relref and ref shortcodes. Use them whenever you need to link to other pages or anchors.

These shortcodes allow linking directly to a certain file without needing to pass its path, for example:

[This links to the Style Guidelines]({{< ref "style.md" >}})

This would look like so:

This links to the Style Guidelines

The shortcode also works with anchors. For example, linking to an anchor in the helloworld.md file would be:

[This links to the #anchors section of the Style Guidelines]({{< ref "style.md#anchors" >}})

This would look like so:

This links to the #anchors section of the Style Guidelines

Linking to an anchor in the current file can be done this way:

[This links to the #alert section of this page]({{< #alert >}})

This links to the #alert section of this page

They also error out upon finding an invalid link, which protects you from merging broken links.

Do not use rel/relref for the content/hig/_index.md and content/docs/use/kirigami/*.md pages, because they are translated and do not deal well with these shortcodes.

These shortcodes are used once to display the link, so no closing tag is used.

You can read more about rel/relref in the Hugo Documentation.

installpackage

We have a custom shortcode that can be used to list packages or commands that are specific to certain major distributions, the {{< installpackage >}} shortcode. This can be used at the beginning of a tutorial or page to make it convenient for the user to install the prerequisites for the project.

There are two types of attributes that can be passed: a distro name, and a distro command.

Here's an example taken from the Kirigami tutorial:

{{< installpackage
  ubuntu="build-essential cmake extra-cmake-modules qtbase5-dev qtdeclarative5-dev qtquickcontrols2-5-dev kirigami2-dev libkf5i18n-dev gettext libkf5coreaddons-dev"
  arch="base-devel extra-cmake-modules cmake qt5-base qt5-declarative qt5-quickcontrols2 kirigami2 ki18n kcoreaddons breeze"
  opensuseCommand=`sudo zypper install --type pattern devel_C_C++
sudo zypper install cmake extra-cmake-modules libQt5Core-devel libqt5-qtdeclarative-devel libQt5QuickControls2-devel kirigami2-devel ki18n-devel kcoreaddons-devel qqc2-breeze-style`
  fedoraCommand=`sudo dnf groupinstall "Development Tools" "Development Libraries"
sudo dnf install cmake extra-cmake-modules qt5-qtbase-devel qt5-qtdeclarative-devel qt5-qtquickcontrols2-devel kf5-kirigami2-devel kf5-ki18n-devel kf5-kcoreaddons-devel qqc2-breeze-style` >}}

Which looks like so:

logo of Linux operating system KubuntuKubuntulogo of Linux operating system KDE neonKDE Neon
sudo apt install build-essential cmake extra-cmake-modules qtbase5-dev qtdeclarative5-dev qtquickcontrols2-5-dev kirigami2-dev libkf5i18n-dev gettext libkf5coreaddons-dev
logo of Linux operating system ManjaroManjarologo of Linux operating system Arch LinuxArch
sudo pacman -S base-devel extra-cmake-modules cmake qt5-base qt5-declarative qt5-quickcontrols2 kirigami2 ki18n kcoreaddons breeze
logo of Linux operating system openSUSEOpenSUSE
sudo zypper install --type pattern devel_C_C++
sudo zypper install cmake extra-cmake-modules libQt5Core-devel libqt5-qtdeclarative-devel libQt5QuickControls2-devel kirigami2-devel ki18n-devel kcoreaddons-devel qqc2-breeze-style
logo of Linux operating system FedoraFedora
sudo dnf groupinstall "Development Tools" "Development Libraries"
sudo dnf install cmake extra-cmake-modules qt5-qtbase-devel qt5-qtdeclarative-devel qt5-qtquickcontrols2-devel kf5-kirigami2-devel kf5-ki18n-devel kf5-kcoreaddons-devel qqc2-breeze-style

tabset/tab

The tabset and tab shortcodes can be used to show equivalent code written in different languages together with readfile. While tabset requires no arguments, tab requires a single argument, tabName, which will become the title of the tab.

Here is an usage example from the flatpak tutorial:

{{< tabset >}}

{{< tab tabName="PySide6" >}}

{{< readfile file="/content/docs/getting-started/python/pyside-app/src/simplemdviewer_app-3.py" highlight="python" emphasize="9-10 34-35" >}}

{{< /tab >}}

{{< tab tabName="PyQt6" >}}

{{< readfile file="/content/docs/getting-started/python/pyqt-app/src/simplemdviewer_app-3.py" highlight="python" emphasize="9-10 36-37" >}}

{{< /tab >}}

{{< /tabset >}}

Which looks like this:

 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
#!/usr/bin/env python3

import os
import sys
import signal
from PySide6.QtGui import QGuiApplication
from PySide6.QtCore import QUrl
from PySide6.QtQml import QQmlApplicationEngine
# from md_converter import MdConverter
from simplemdviewer.md_converter import MdConverter  # noqa: F401

def main():
    """Initializes and manages the application execution"""
    app = QGuiApplication(sys.argv)
    engine = QQmlApplicationEngine()

    """Needed to close the app with Ctrl+C"""
    signal.signal(signal.SIGINT, signal.SIG_DFL)

    """Needed to get proper KDE style outside of Plasma"""
    if not os.environ.get("QT_QUICK_CONTROLS_STYLE"):
        os.environ["QT_QUICK_CONTROLS_STYLE"] = "org.kde.desktop"

    base_path = os.path.abspath(os.path.dirname(__file__))
    url = QUrl(f"file://{base_path}/qml/main.qml")
    engine.load(url)

    if len(engine.rootObjects()) == 0:
        quit()

    app.exec()


if __name__ == "__main__":
    main()
 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
#!/usr/bin/env python3

import os
import sys
import signal
from PyQt6.QtGui import QGuiApplication
from PyQt6.QtCore import QUrl
from PyQt6.QtQml import QQmlApplicationEngine, qmlRegisterType
# from md_converter import MdConverter
from simplemdviewer.md_converter import MdConverter

def main():
    """Initializes and manages the application execution"""
    app = QGuiApplication(sys.argv)
    engine = QQmlApplicationEngine()

    """Needed to close the app with Ctrl+C"""
    signal.signal(signal.SIGINT, signal.SIG_DFL)

    """Needed to get proper KDE style outside of Plasma"""
    if not os.environ.get("QT_QUICK_CONTROLS_STYLE"):
        os.environ["QT_QUICK_CONTROLS_STYLE"] = "org.kde.desktop"

    qmlRegisterType(MdConverter, "org.kde.simplemdviewer", 1, 0, "MdConverter")

    base_path = os.path.abspath(os.path.dirname(__file__))
    url = QUrl(f"file://{base_path}/qml/main.qml")
    engine.load(url)

    if len(engine.rootObjects()) == 0:
        quit()

    app.exec()


if __name__ == "__main__":
    main()

Add a line between shortcodes

We have tooling that parses through shortcodes.

Adding a line between shortcodes (for example, two alerts) ensures that no parsing error occurs.

Also, keep a new line around both opening and closing part of shortcode:

Some normal text.

{{< alert title="First note" color="info" >}}

This is the first alert.

{{< /alert >}}

{{< alert title="Second note" color="info" >}}

This is the second alert. Notice the line between alerts.

{{< /alert >}}

As a bonus, this improves readability.

Adding images

Images should be placed in a folder that has the same name as the Markdown file without the .md extension. This way, it is possible to link to the image without needing to specify its file path.

installation/
    ├── _index.md
    ├── configuring.md  <-- Same name
    ├── howto.md
    ├── configuring/    <-- Same name
    │   └── image.png
    └── howto/
        └── image.png

You can embed images in the tutorial page using ![](image.png).

Optionally, you can add alt text inside the brackets, e.g. ![Alternative text that will be read by screen readers.](imagefile.png).

If you need captions or need to position the image in a specific way, like in the horizontal center of the page, you may also use the figure shortcode:

{{< figure class="text-center" caption="Caption of the image" alt="Alternative text" src="image.png" >}}

The available options for positioning the image using class= can be found in the Bootstrap documentation.

Be careful not to leave empty attributes inside this shortcode, as this can lead to a parsing error by some of our tooling. Example: {{< figure src="image.png" caption="" >}}.

Images in separate sections

Sometimes you'll need to display side-by-side images or images next to text for various reasons, like matching explanatory text to an example, comparing two images, for better use of space, or simply because it looks good. There are two shortcodes for this:

First, the {{< sections >}} shortcode. Use this to put one element on the left and another on the right.

It needs a closing shortcode {{< /sections >}}. Inside this shortcode, you may use {{< section-left >}} and {{< section-right >}}, both with corresponding closing versions. Here's an example:

{{< sections >}}

{{< section-left >}}

Your fancy text!

{{< /section-left >}}

{{< section-right >}}

![Your fancy image!](image.png)

{{< /section-right >}}

{{< /sections >}}

An example of the sections shortcode can be seen in Kirigami Tutorial: Controls and interactive elements.

Second, the {{< compare >}} shortcode. Use this only with images, to showcase bad and good behavior. You'll see this most commonly used in the HIG.

It needs a closing shortcode {{< /compare >}}. Inside this shortcode, you may use {{< dont src="bad-image.png" >}} and {{< do src="good-image.png" >}}, both with corresponding closing versions, with text in between. Here's an example:

{{< compare >}}

{{< dont src="bad-image.png" >}}

Don't do this!

{{< /dont >}}

{{< do src="good-image.png" >}}

Do this instead!

{{< /do >}}

{{< /compare >}}

An example of the compare shortcode can be seen in Kirigami Tutorial: Actions based components.

Code blocks

You can specify the language to be used to highlight code blocks written in Markdown by writing the language name after the three backticks, e.g.:

```cmake
cmake -B build
cmake --build build
```

The most commonly used highlighting options are:

  • c++
  • qml
  • cmake
  • python
  • ini

Code formatting

Use 4 spaces for source code files.

Do not use tabs.

Links to the KDE API Documentation and the Qt Documentation can be generated as follows:

[text](docs:component;link)
  • text is the text for the link
  • component is the component group (e.g. kirigami2, qtquickcontrols)
  • link is the item to link to (e.g. QtQuick.Controls.Label, org::kde::kirigami::BasicListItem, KMessageBox).

The component matches the name of the .json files in _data, in the root folder of the KDE Developer Repository. Inside the corresponding .json file, searching for the name of the class or function you need will give you the name of the link.

Whenever you need to link to the main page for the documentation of a KDE component, you can omit the link, as in [Kirigami](docs:kirigami2). This cannot be done for Qt components.

If the component you want to link to wasn't added to scripts/doxygen_integration.py yet, add it to the file and execute python3 scripts/doxygen_integration.py. After running the script, make sure to add the generated JSON files in the _data/ folder and commit them together with your MR.

Examples

  • [AbstractCard](docs:kirigami2;AbstractCard)
  • [Kirigami.Units.devicePixelRatio](docs:kirigami2;Kirigami::Units::devicePixelRatio)
  • [KMessageBox](docs:kwidgetsaddons;KMessageBox)
  • [Label](docs:qtquickcontrols;QtQuick.Controls.Label)

White space

In the body of the article, separate special blocks (lists, code blocks, alerts, tips, warnings etc.) from normal text with an empty line.

Avoid:

The code to say hello:
```
echo "Hello"
```

Do:

The code to say hello:

```
echo "Hello"
```