Layouts, ListViews, and Cards
Figuring out the different ways of placing things on a page
Last updated
Figuring out the different ways of placing things on a page
Last updated
Now that we understand how pages work, it is time to add stuff to them. We will be going through a number of important layout components and elements that will be useful when designing our app.
By the end of this section you'll have a neat-looking app.
If you've ever used , , or Plasma's System Settings, you will have come across a . Quite simply, it lets you display data on a list.
That seems cryptic, but don't worry. Let's start from the beginning.
We add this component inside our from the last tutorial.
We're using , which is a that allows us to easily display cards in a list. However, ListViews are made to show data taken from a model - to automatically populate itself from a set of data that we point it to. That's where the model
property comes in: in this example, it's pointing to kountdownModel
.
Models are also useful in how they can be modified through the use of several methods. Some important ones are:
implicitWidth and implicitHeight
The first part we will take a look at is how to manage the width and height of our component:
Layouts
smallSpacing
4px
largeSpacing
8px
gridUnit
18px
Inner components
We could just create three labels within our delegate component and call it a day, but that wouldn't look particularly nice. We'll make use of a few more convenient components:
So there is our basic card!
With these steps, we have now laid the basic groundwork for adding all the functionality to our app.
We add our kountdownModel
inside the from the last tutorial.
A model defines the way that a data entry is structured. Our kountdownModel
will consist of only one element for now. By looking at our above, we can see how the data of our kountdownModel
are structured: it contains a name, a description, and a date. This isn't set in stone, and you may have different sorts of data in your model. The first two are just strings, and the third is a number we're using as a placeholder.
Since QML is built on top of JavaScript, many of this language's features are available for use in QML files. However, QML variables have to be prefixed with property
, unless it is inside a JS code block. You can read more about it in .
adds a JavaScript object (JSObject) yourobject
to the ListModel and places it after the last item in the model. For this to happen correctly, you must provide a JSObject with the correct properties and corresponding datatypes.
returns the JSObject at the index location you provide.
removes the JSObject at the provided index
location, and as many after that index location as you put in count
(1 includes only the JSObject at the provided index)
changes the item at the provided index
location with the values provided in yourobject
. Same rules as with append()
.
While our kountdownModel
contains the data that will be displayed, our kountdownDelegate
will handle how the data will be displayed in the . For that we use a designed to display card-type delegates, and those delegates will be visually represented by means of a .
Delegates automatically receive the properties of the that we have specified in our model. We can therefore just refer to their name
, description
, and date
properties as if they were conventional variables within our delegate.
The that will represent our delegate can be added inside our . We will then check what each part of our delegate component does.
Looking at our , the first properties we set are and . We have set these to the delegateLayout.implicitWidth
and delegateLayout.implicitHeight
, i.e. the implicitWidth
and implicitHeight
of the GridLayout
element.
Implicit widths and heights are properties available in any that function as hints and are set as a default, or as a fallback, if there is no explicit width or height set for these components. These values default to 0x0, so it is very important that you define those in raw Item components as done above.
Here we have set the implicitWidth
and implicitHeight
of our to that of the below to ensure it does not spill out of the card. This way, the card takes as much space is necessary for its contents.
The is inside the component we have provided for the property . This is the that contains what will be displayed in your card.
We also need to choose a for our components so that they don't just pile on top of each other. There are three main types that we can choose from:
lays out your components vertically, in a single column
lays out your components horizontally, in a single row
lays out your components in a grid with a composition of your choosing
With and , all we have to do is write our components inside the Layout component. As you can see, we went with a , which entails a bit more handiwork.
The first thing you see is our anchors
. provides a useful way of making sure your components are positioned in certain parts of a parent component. We have anchored our to the left, top, and right of the parent card, ensuring our content stretches across the whole card.
Next we specify the spacing between the rows and columns within our grid, so that our components don't bunch up. Kirigami provides a number of handy to use for this purpose:
KDE's Visual Design Group (VDG) has a lot more information about the different units defined within Plasma and Kirigami on the .
As you might remember, root
is the id of our . It provides the property, used to determine whether the current device screen is a widescreen (i.e. a computer monitor or a phone in landscape). We use a here to vary the number of columns in our grid depending on the screen we are using: if it's a widescreen, the grid will have 4 columns, else it will have 2.
Left, : uses the ListElement
's date
as a level 1 heading.
Middle, : has a that displays the task name; a , which provides the horizontal line; and a , that displays a task's optional description. The latter two components have a property, which checks if the description is empty or not and displays the components depending on the result of description.length > 0
.
Right, : a button that will do something... soon!