Geometry, visual and layout bounds explained

A bound is a territorial limit, a boundary. In terms of a graphical element, it is a space in which the graphical element is rendered. For simplicity, bounds are rectangular areas. Bloc defines three different bounds (layout, geometry, and visual) that we will explain using the following example:

centeredOutskirts
	<gtExample>
	<label: 'Rectangle and star elements with centered stroke'>
	| aContainer |

	aContainer := self container.
	aContainer addChild: self rectangleWithSmallerGeometryCentered.
	aContainer addChild: self starWithSmallerGeometryCentered.

	^ aContainer
    

"Layout" bounds are drawn with gray dashed rectangles in the figure above. They are of this size in this particula example, because each element defines its size explicitly using size: method. Layout bounds are considered by layout algorithms to define mutual locations for all considered elements.

"Geometry" bounds are drawn with red dashed rectangles in the figure above. The area is defined by minimum and maximum values of a polygon vertices. In case of rectangle, geometry bounds are defined by the following polygon:

BlGeometryVisualAndLayoutBoundsExamples>>#rectanglePolygon
	<gtExample>
	<label: 'Rectangle polygon'>
	^ BlPolygonGeometry vertices: {
		(50@50). (150@50). (150@100). (50@100)
	}
        

"Visual" bounds are drawn with blue rectangles in the figure above. It is an exact area occupied by an element. Computing visual bounds is the most expensive computation as it takes strokes and rendering into account.

The first figure above draws all strokes centered around the defined vertices. For that reason, the geometry bounds are in the middle. The following example draws strokes outside:

BlGeometryVisualAndLayoutBoundsExamples>>#outsideOutskirts
	<gtExample>
	<label: 'Rectangle and star elements with outside stroke'>
	| aContainer |

	aContainer := self container.
	aContainer addChild: self rectangleWithSmallerGeometryOutside.
	aContainer addChild: self starWithSmallerGeometryOutside.
		
	^ aContainer
        

In case of the rectangle, you can notice that its stroke is inside of the visual (blue, outside) and the geometry (red, inside) bounds. If the strokes are drawn inside, then visual and geometry bounds are equal:

BlGeometryVisualAndLayoutBoundsExamples>>#insideOutskirts
	<gtExample>
	<label: 'Rectangle and star elements with inside stroke'>
	| aContainer |

	aContainer := self container.
	aContainer addChild: self rectangleWithSmallerGeometryInside.
	aContainer addChild: self starWithSmallerGeometryInside.
		
	^ aContainer
        

Being able to display those three bounds helps when elements are not visually aligned as expected. Consider the following example:

BlGeometryVisualAndLayoutBoundsExamples>>#misalignedIcon
	<gtExample>
	| anIconElement aLabelElement aContainerElement |
	anIconElement := self emptyIconElement
		geometry: self misalignedTrianglePolygon.
	aLabelElement := self textElementWithRunText.
	aContainerElement := self iconAndTextContainer.
	aContainerElement
		addChildren:
			{anIconElement.
			aLabelElement}.
	^ aContainerElement
        

The triangle (icon) and text are not vertically centered despide the fact that we have BlLinearLayout horizontal alignCenter in the code. To understand why it happens, aTriangleElement debug: true can be added to the code with the following result:

BlGeometryVisualAndLayoutBoundsExamples>>#debuggingMisalignedIcon
	<gtExample>
	| aContainerElement |
	aContainerElement := self misalignedIcon.
	aContainerElement children first.
	^ aContainerElement
        

As we can see the layout bounds (gray dashed rectangle) consider larger area then expected. Looking at the definition of the triangle polygon, we can spot that the polygon starts at 0@3 instead of 0@0:

BlGeometryVisualAndLayoutBoundsExamples>>#misalignedTrianglePolygon
	<gtExample>
	<label: 'Triangle polygon'>
	^ BlPolygonGeometry
		vertices:
			{(0 @ 3).
			(8 @ 7.5).
			(0 @ 12)}
        

If we fix it to:

BlGeometryVisualAndLayoutBoundsExamples>>#trianglePolygon
	<gtExample>
	<label: 'Triangle polygon'>
	^ BlPolygonGeometry
		vertices:
			{(0 @ 0).
			(8 @ 6).
			(0 @ 12)}
        

The icon and text are aligned as expected:

BlGeometryVisualAndLayoutBoundsExamples>>#debuggingAlignedIcon
	<gtExample>
	| aContainerElement |
	aContainerElement := self alignedIcon.
	^ aContainerElement
        

And we can disactivate the debug mode:

BlGeometryVisualAndLayoutBoundsExamples>>#alignedIcon
	<gtExample>
	| anIconElement aLabelElement aContainerElement |
	anIconElement := self emptyIconElement
		geometry: self trianglePolygon.
	aLabelElement := self textElementWithRunText.
	aContainerElement := self iconAndTextContainer.
	aContainerElement
		addChildren:
			{anIconElement.
			aLabelElement}.
	^ aContainerElement
        

The debugging mode is also possible to activate in the GT-Inspector's menu, next to the tabs.