UI Positioning
For all kinds of UI content, use the uiTransform
component to set the size, position, and other properties related to the entity’s alignment.
The uiTransform
component works in the screen’s 2d space very much like the Transform
component works in the the scene’s 3D space.
import { ReactEcsRenderer } from '@dcl/sdk/react-ecs'
ReactEcsRenderer.setUiRenderer(() => (
<UiEntity
uiTransform={{
width: '200px',
height: '100px',
justifyContent: 'center',
alignItems: 'center',
}}
uiBackground={{ color: Color4.Green() }}
/>
))
Positioning properties #
The alignment of UI entities is based on the Flexbox alignment model. This is a very powerful model for dynamically organizing nested entities inside modals that may vary in size.
đź’ˇ Tip: Decentraland’s UI implementation is based on that of Yoga . Read this article for a very approachable and in-depth coverage of the properties available in Flexbox.
Entity size #
Use width
and height
to set the size of the entity. The following kinds of values are supported:
auto
: The size adapts to fit the content inside. This is very convenient for text that may vary in length. Write the value as “auto”.- Percentage: As a percentage of the parent’s measurements. Write the value as a string that ends in “%”, for example
10 %
. - Pixels: Write the value as a number.
- Screen width or height: Use vw (view width) and vh (view height) can be used to indicate a fraction of the full size of the window running Decentraland. For example
10vw
refers to 10% of the window’s width,25vh
to 25% of the window’s height.
Note that these properties affect the default size of that item, the size of the item before any flex grow and flex shrink calculations are performed. The final size may be interpreted differently based on the size of the parent entity, and the Flexbox properties that are set.
đź“” Note: In properties that support both numbers and strings, to set the value in pixels, write a number. To set these fields as a percentage of the parent’s measurements, write the value as a string that ends in “%”, for example
10 %
. You can also set a pixel value as a string by ending the string inpx
, for example200px
.
- When values are expressed as a percentage, they’re always in relation to the parent’s container. If the entity has no parents, then the value is a percentage of the whole screen.
- If values are expressed in pixels, they are absolute, and not affected by the parent’s scale.
- If values are expressed in
vh
orvw
, they are a percentage of the full window, not affected by the parent’s scale.For the
auto
width/height to work, the following rules apply:
- The UiTransform that uses width/height as “auto” should have
alignSelf
:“center”
/“flex-start”
/“flex-end”
ORpositionType: “absolute”
- If the UiTransform of a child use
positionType: “absolute”
, the parent won’t adapt to its size/position- If the UiTransform of a child uses any position overwrite, the parent won’t adapt to its size/position
These other properties are also available to adjust size in a more advanced way:
maxWidth
andmaxHeight
: number or string (like height and width). The maximum size that the entity may have.minWidth
andminHeight
: number or string (like height and width). The minimum size that the entity may have. If the parent is too small to fit the minimum size of the entities, they will overflow from their parent.flexBasis
: This is an axis-independent way of providing the default size of an item along the main axis. Setting the flex basis of a child is similar to setting the width of that child if its parent is a container with flex direction: row or setting the height of a child if its parent is a container with flex direction: column.
import { ReactEcsRenderer } from '@dcl/sdk/react-ecs'
ReactEcsRenderer.setUiRenderer(() => (
<UiEntity
uiTransform={{
alignSelf: 'center',
alignContent: 'center',
width: '80%',
height: '30%',
minWidth: 300,
maxWidth: 2500,
margin: { left: '10%', right: '10%' },
}}
uiBackground={{ color: Color4.Green() }}
/>
))
Arranging child entities #
By default, child entities are positioned in relation to the top-left corner of its parent. You can use properties like justifyContent
and alignItems
to change this behavior.
đź’ˇ Tip: Any properties that refer to content refer to entities along the main axis (determined by flexDirection
). Any properties that refer
-
flexDirection
: Flex direction controls the direction in which children of a node are laid out. This is also referred to as the main axis. The main axis is the direction in which children are laid out. The cross axis is the axis perpendicular to the main axis, or the axis which wrapping lines are laid out in. It takes its value from theFlexDirectionType
type. The following options are available:row
(DEFAULT)row-reverse
column
column-reverse
-
justifyContent
: This property describes how to align children within the main axis of their container. For example, you can use this property to center a child horizontally within a container withflexDirection
set to row or vertically within a container withflexDirection
set to column. The value of this property must be from theAlignType
type. Possible values are:flex-start
(DEFAULT): Align children of a container to the start of the container’s main axis.flex-end
: Align children of a container to the end of the container’s main axis.center
: Align children of a container in the center of the container’s main axis.space-between
: Evenly space of children across the container’s main axis, distributing remaining space between the children.space-around
: Evenly space of children across the container’s main axis, distributing remaining space around the children. Compared to space between using space around will result in space being distributed to the beginning of the first child and end of the last child.
-
alignItems
: Describes how to align children along the cross axis of their container. Align items is very similar to justify content but instead of applying to the main axis, align items applies to the cross axis. This property requires a value from theAlignType
type. The following options are available:stretch
: (DEFAULT) Stretch children of a container to match the height of the container’s cross axis.flex-start
: Align children of a container to the start of the container’s cross axis.flex-end
: Align children of a container to the end of the container’s cross axis.center
: Align children of a container in the center of the container’s cross axis.baseline
: Align children of a container along a common baseline. Individual children can be set to be the reference baseline for their parents.
-
alignSelf
: Align self has the same options and effect asalignItems
but instead of affecting the children within a container, you can apply this property to a single child to change its alignment within its parent. align self overrides any option set by the parent with align items. It takes its value fromAlignType
, seealignItems
above for details on these options. -
alignContent
: Align content defines the distribution of lines along the cross-axis. This only has effect when items are wrapped to multiple lines usingflexWrap
. It takes its value from theAlignType
type. The following options are available:flex-start
: (DEFAULT) Align wrapped lines to the start of the container’s cross axis.flex-end
: Align wrapped lines to the end of the container’s cross axis.stretch
: Stretch wrapped lines to match the height of the container’s cross axis.center
: Align wrapped lines in the center of the container’s cross axis.space-between
: Evenly space wrapped lines across the container’s main axis, distributing remaining space between the lines.space-around
: Evenly space wrapped lines across the container’s main axis, distributing remaining space around the lines. Compared to space between using space around will result in space being distributed to the begining of the first lines and end of the last line.
-
flexGrow
: This describes how any space within a container should be distributed among its children along the main axis. After laying out its children, a container will distribute any remaining space according to the flex grow values specified by its children. Flex grow accepts any floating point value >= 0, with 0 being the default value. A container will distribute any remaining space among its children weighted by the child’s flex grow value. -
flexShrink
: Describes how to shrink children along the main axis in the case that the total size of the children overflow the size of the container on the main axis. flex shrink is very similar to flex grow and can be thought of in the same way if any overflowing size is considered to be negative remaining space. These two properties also work well together by allowing children to grow and shrink as needed. Flex shrink accepts any floating point value >= 0, with 1 being the default value. A container will shrink its children weighted by the child’s flex shrink value. -
overflow
: Determines what happens if the size of the children of an entity overflow its parent. It uses values from theOverflowType
type.hidden
: Overflowing entities are made invisible.visible
: Overflowing entities break out of the margins of the parent.
-
flexWrap
: The flex wrap property is set on containers and controls what happens when children overflow the size of the container along the main axis. By default children are forced into a single line (which can shrink entities). If wrapping is allowed items are wrapped into multiple lines along the main axis if needed. wrap reverse behaves the same, but the order of the lines is reversed. This property takes its value from theFlexWrapType
type.wrap
no-wrap
wrap-reverse
Margins and padding #
margin
: This property affects the spacing around the outside of a node. A node with margin will offset itself from the bounds of its parent but also offset the location of any siblings. The margin of a node contributes to the total size of its parent if the parent is auto sized. Set space between the entity and its parent’s margins. The expected value is an object that contains the propertiestop
,left
,bottom
, andright
.padding
: This property affects the size of the node it is applied to. Padding in Yoga acts as if box-sizing: border-box; was set. That is padding will not add to the total size of an entity if it has an explicit size set. For auto sized nodes padding will increase the size of the node as well as offset the location of any children. The expected value is an object that contains the propertiestop
,left
,bottom
, andright
.
Fine-tune position #
In Flexbox, entity positions are mostly determined by how they are parented, and what arrangement properties are set on the parent and child. You often don’t have to set the position
property at all. But if you do want to tweak that, or completely override the normal flow of Flexbox and set an absolute position, here are the relevant properties:
-
positionType
: Defines how entities are positioned. It uses a value from thePositionType
enum.relative
: (DEFAULT) By default an entity is positioned relatively. This means an entity is positioned according to the normal flow of the layout, and then offset relative to that position based on the values oftop
,right
,bottom
, andleft
. The offset does not affect the position of any sibling or parent entities.absolute
: When positioned absolutely, an entity doesn’t take part in the normal layout flow. It is instead laid out independent of its siblings. The position is determined based on thetop
,right
,bottom
, andleft
values.
-
position
: The position valuestop
,right
,bottom
, andleft
behave differently depending on thepositionType
. For a relative entity they offset the position of the entity in the direction specified. For absolute entity though these properties specify the offset of the entity’s side from the same side on the parent. The expected value is an object that contains the propertiestop
,left
,bottom
, andright
.
đź“” Note : When measuring from the top, the numbers forposition
should be negative. Example: to position a component leaving a margin of 20 pixels with respect to the parent on the top and left sides, setposition
to 20, -20.
Visibility #
display
: Determines is an entity is visible or not. To make an entity invisible, setdisplay
tonone
.
Responsive UI size #
Players with different screen sizes may see your UI layout differently. If you set the size of any UI element to a fixed number of pixels, this UI may look too small to read on retina displays, that have a much higher pixel density.
Instead of positioning and scaling UI elements in terms of screen percentages, you can also obtain the canvas dimensions and then calculate the absolute positions and sizes following your own custom logic. For example, you could chose different dialog arrangements depending on the screen size.
To obtain information about the screen’s dimension, you can check the UiCanvasInformation
, that’s added by default to the scenes’s root entity.
The UiCanvasInformation
component holds the following information:
height
: Canvas height in pixelswidth
: Canvas width in pixelsdevicePixelRatio
: The ratio of the resolution in physical pixels in the device to the pixels on the canvasinteractableArea
: Arect
object, detailing the size of the area designated for scene UI elements. This object contains the following fields:height
: Height of interactable areawidth
: Width of interactable areax
: Leftmost x position of the interactable areay
: Lowest y position of the interactable area
export function Main(){
let canvas = UiCanvasInformation.get(engine.RootEntity)
console.log("CANVAS DIMENSIONS: ", canvas.width, canvas.height)
})
The following snippet continually calculates a multiplier value based on the screen size:
import { engine, UiCanvasInformation } from "@dcl/sdk/ecs"
let timer = 0
let canvasInfoTimer = 0.5
export let scaleFactor = 1
export function UIScaleUpdate() {
engine.addSystem((dt) => {
timer += dt
if (timer <= canvasInfoTimer) return
timer = 0
const uiCanvasInfo = UiCanvasInformation.getOrNull(engine.RootEntity)
if (!uiCanvasInfo) return
const newScaleFactor = Math.min(uiCanvasInfo.width / 1920, uiCanvasInfo.height / 1080)
if (newScaleFactor !== scaleFactor) {
scaleFactor = newScaleFactor
console.log('NEW UI scaleFactor: ', scaleFactor)
}
})
}
The value of the scaleFactor
variable, that this function updates, can then be used as a multiplier on any UI element in the scene, including heigh
, width
and fontSize
values.
import { ReactEcsRenderer } from '@dcl/sdk/react-ecs'
import { scaleFactor } from './calculate-scale-factor'
ReactEcsRenderer.setUiRenderer(() => (
<UiEntity
uiTransform={{
width: 200 * scaleFactor,
height: 100 * scaleFactor,
justifyContent: 'center',
alignItems: 'center',
padding: 4 * scaleFactor
}}
uiBackground={{ color: Color4.Green() }}
>
<Label
value={description}
fontSize={18 * scaleFactor}
textAlign="middle-center"
uiTransform={{
width: "auto",
height: "auto",
alignSelf: "center",
margin: { top: 10 * scaleFactor, bottom: 10 * scaleFactor },
}}
/>
</UiEntity>
))
Some other best practices regarding UI sizes:
- If the width or height of any UI element is dynamic, it’s good to also use the
maxWidth
,minWidth
,maxHeight
, andminHeight
parameters to make sure they stay within reasonable values. - The font size of text is relative to a fixed number of pixels, you should make it dynamic so it remains readable on retina displays. See Responsive text size