The Canvas Layout Typescript library provides layout algorithms for laying out components. A component is an abstraction; it can be implemented in many ways, for example as items in a HTML5 Canvas drawing or as HTML elements. The library allows you to focus on drawing the individual components instead of on how to arrange them on your screen.
The library currently provides four (at time) layout algorithms: Border
, which lays out components in five different regions; Grid
, which lays out components in a user defined grid, FlexGrid
which offers a grid with flexible column and row sizes, and Flow
which flows components in a user defined direction. Using the Grid
and FlexGrid
algorithms you can also create horizontal and vertical layouts.
We start with the definition of a component; a component is something that has a minimum size, a preferred size, and a maximum size. It also has a method to set its size and position. A container is a component that contains other components and lays them out according to a layout algorithm, Components need to satisfy the following interface requirements in order to be used with the layout algorithms.
- bounds()
- Returns the component's position and size as a value object with properties `x`, `y`, `width` and `height`. All properties should be present in the returned object.
<dt>bounds(value)</dt>
<dd>Set the position and size of a component by using a value object with properties `x`, `y`, `width` and `height`. The individual properties are all optional, it is thus legal to supply an empty object, or any combination of properties.</dd>
<dt>preferredSize()</dt>
<dd>Returns the component's preferred size (i.e. the size it wishes to have) as an object with properties `width` and `height`. This is only a hint, there is no guarantee the component will get the size it prefers.</dd>
<dt>minimumSize()</dt>
<dd>Returns the component's minimum size (i.e. the size it should at least have) as an object with properties `width` and `height`.</dd>
<dt>maximumSize()</dt>
<dd>Returns the component's maximum size (i.e. the size it should stay below or equal to) as an object with properties `width` and `height`.</dd>
<dt>isVisible()</dt>
<dd>Returns `true` if the component is visible and should be taken into account when calculating the layout. Returns `false` otherwise.</dd>
<dt>insets()</dt>
<dd>Returns the offset between a container and its contents as an object with properties: `top`, `bottom`, `left`, and `right`.</dd>
<dt>doLayout()</dt>
<dd>Calls the `layout` method on the layout algorithm used to lay out the component (container) it is called on. If the component is not a container (and does not have a layout algorithm) this method can be left empty.</dd>
Note that the distinction between containers and components is artificial, both implement the same interface.
All algorithms are in the Layout
namespace and implement the following interface.
- preferred(container)
- Returns the preferred size of the container and its children according to the layout algorithm.
<dt>minimum(container)</dt>
<dd>Returns the minimum size the container and its children are allowed to have according to the layout algorithm.</dd>
<dt>maximum(container)</dt><dd>
Returns the maximum size the container and its children are allowed to have according to the layout algorithm.
<dt>layout(container)</dt>
<dd>Performs the layout according to the algorithm; resizing and positioning children if necessary.</dd>
The layout method will not resize the container to accommodate its children's preferred size. If a resize is desired, set the bounds of the container to its preferred size. The example below shows both ways of laying out a container; resizing the children to fit in the container, and resizing the container to fit the children.
// create a layout algorithm
const layout: AbstractLayout = (…)
// lay out without resizing the container
layout.layout(container);
// resize the container, then lay it out
container.bounds(layout.preferred());
layout.layout(container);
A layout algorithm can be created by calling either the Grid
or Border
constructor in the Layout
namespace. Both constructors take an option object containing layout specific properties. Both layouts have the following properties in common:
- hgap
- The horizontal space between the laid out components. Should be a number in a coordinate space of your choice. Defaults to 0. Optional.
<dt>vgap</dt>
<dd>The vertical space between the laid out components. Should be a number in a coordinate space of your choice. Defaults to 0. Optional.</dd>
The other properties are specific to the layout algorithm and are discussed below.
The border algorithm lays out components in five different regions. These regions are called center
, north
, south
, east
and west
. The center component will be laid out in the center of the container, north on top of it, south beneath it and west and east on the left and right side respectively. The layout can only contain one of each region, but all are optional. Below is a visualization of a layout using all five regions.
The border algorithm takes an option object as parameter which can contain the following properties:
- center, north, south east, west
- The center, north, south, east or west component. All are optional.
The following example lays out a west, center and north component with a vertical space of 5 units between each component. There may be additional space between the components and the container if the container returns non-zero insets.
var border = new Border({
west: myWestComponent,
center: myCenterComponent,
north: myNorthComponent,
vgap: 5
});
border.layout(myContainer);
If a region is not specified or the component is not visible its space will be taken up by the other components.
The grid algorithm lays out the components in a grid, and resizes each component to the same size. The number of columns and rows can be specified by the user. Below is a visualization of a grid layout with four components in a 2x2 grid.
The grid algorithm takes an option object as parameter which can contain the following properties:
- rows
- The number of rows in the grid. Optional.
<dt>columns</dt>
<dd>The number of columns in the grid. Optional.</dd>
<dt>items</dt>
<dd>An array containing the components to be laid out by the algorithm. Optional.</dd>
<dt>fill</dt>
<dd>The direction in which the grid is filled in. Valid values are `horizontal` or `vertical` for filling in the grid left to right and top to bottom, or top to bottom and left to right respectivily. Optional.</dd>
The following example lays out four components in a 2x2 grid, without any spacing between the components.
var layout = new Grid({
rows: 2,
columns: 2,
items: [myComponent1, myComponent2, myComponent3, myComponent4]
});
layout.layout(myContainer);
If the number of rows is given, the number of columns is calculated automatically by taking the number of components into account. If the number of rows is not given (or set to zero), and the number of columns is given, the number of rows will be automatically calculated using the number of components. If neither is given, the number of rows is set equal to the number of components and the number of rows is set to zero.
The flex grid algorithm lays out the components in a grid with flexible row and columns sizes. The number of columns and rows can be specified by the user. Below is a visualization of a flex grid layout with six components in a 3x2 grid.
The flex grid algorithm takes an option object as parameter which can contain the following properties:
- rows
- The number of rows in the flex grid. Optional.
<dt>columns</dt>
<dd>The number of columns in the flex grid. Optional.</dd>
<dt>items</dt>
<dd>An array containing the components to be laid out by the algorithm. Optional.</dd>
The following example lays out six components in a 3x2 grid, without any spacing between the components.
var layout = new FlexGrid({
rows: 2,
columns: 2,
items: [myComponent1, myComponent2, myComponent3,
myComponent4, myComponent5, myComponent6]
});
layout.layout(myContainer);
If the number of rows is given, the number of columns is calculated automatically by taking the number of components into account. If the number of rows is not given (or set to zero,) and the number of columns is given, the number of rows will be automatically calculated using the number of components. If neither is given, the number of rows is set equal to the number of components and the number of rows is set to zero.
The flow algorithm lays out the components on a row. When the component does not fit on the current row it is moved to the next row. The alignment within the row can be user specified. Below is an example of a flow layout using five components with the alignment for each row set to left.
The flow algorithm takes an option object as parameter which can contain the following properties:
- alignment
- A string of either `center`, `left`, or `right`. Defaults to `left`.
<dt>items</dt>
<dd>An array containing the components to be laid out by the algorithm. Optional.</dd>
The following example lays out five components using a center alignment, without any spacing between the components.
var layout = new Flow({
alignment: FlowAlignment.center,
items: [myComponent1, myComponent2, myComponent3,
myComponent4, myComponent5]
});
layout.layout(myContainer);
Horizontal and vertical layouts can be achieved by using a grid or a flexGrid layout with one row for horizontal layouts, and one column for vertical layouts. The choice of a grid layout or a flexGrid layout depends on whether or not you want the items in the grid to have uniform sizes (grid) or their natural sizes (flexGrid.) The following layout uses a flex grid so that all items are laid out in the horizontal direction while still allowing the individual items to take up their natural size (i.e. the second component is longer than the other two.)
var horizontalLayout = new FlexGrid({
rows: 1,
items: [myComponent1, myComponent2, myComponent3]
});
You can also lay out components vertically, by just changing the rows
parameter to columns
as shown in the next example.
var verticalLayout = new FlexGrid({
columns: 1,
items: [myComponent1, myComponent2, myComponent3]
});
GPL v2