public class HorizontalLayout extends AbstractWidget<T>
withChild(AbstractWidget)
family of methods to manage new widgets. The following example code constructs a new layout with a desired height of 100px and three widgets spaced 20px apart horizontally. From left to right: widget1, 20px spacing, widget2, 20px spacing, and widget3. The default child anchor is set to center, so all children added after this line will be centered vertically inside the 100px height, except for widget3 which specifies that it overrides this anchor and will be aligned to the top.
HorizontalLayout layout = new HorizontalLayout(100.0f, 20.0f)
.withDefaultChildAnchor(AnchorPosition.CENTER)
.withChild(widget1)
.withChild(widget2)
.withChild(widget3, AnchorPosition.CENTER_TOP)
.anchoredAt(x, y, AnchorPosition.LEFT_TOP);
NOTE: child widgets are only given enough room as their preferred widths allow but can move around vertically within the preferred height; because of this, the child anchor positions only consider the vertical position of the anchor and the horizontal information is ignored.VerticalLayout
Modifier and Type | Field and Description |
---|---|
protected java.util.ArrayList<easel.ui.layouts.LayoutItem> |
children |
protected AnchorPosition |
defaultChildAnchorPosition |
protected boolean |
shouldAutoScaleToContent |
protected float |
spacing |
protected float |
totalHeight |
protected float |
totalWidth |
hasInteractivity, hb, isHovered, leftClickStarted, onLeftClick, onMouseEnter, onMouseLeave, onRightClick, rightClickStarted
Constructor and Description |
---|
HorizontalLayout(float spacing)
Constructs a new horizontal layout with the given horizontal spacing.
|
HorizontalLayout(float desiredHeight,
float spacing)
Constructs a new horizontal layout with the given row height and horizontal spacing.
|
Modifier and Type | Method and Description |
---|---|
protected void |
anchorChildren(InterpolationSpeed withDelay) |
T |
anchoredAt(float x,
float y,
AnchorPosition anchorPosition,
InterpolationSpeed movementSpeed)
Moves the widget towards a specific spot.
|
protected void |
autoscale() |
protected void |
cancelMovementQueueForAllChildren(boolean shouldTryAndResolveOneLastTime) |
void |
clear()
Stop managing all widgets previously managed by this layout.
|
T |
forceChildAnchors(AnchorPosition forcedChildAnchorPosition)
Forcibly sets the anchor position of all current children to the given value.
|
float |
getContentHeight()
The internal content height of the widget (excludes margins).
|
float |
getContentWidth()
The internal content width of the widget (excludes margins).
|
void |
hide() |
java.util.stream.Stream<AbstractWidget> |
iterator() |
<T> java.util.stream.Stream<T> |
iteratorOfType(java.lang.Class<T> clz)
A stream of children handled by this widget who are of the specific type, for convenience purposes.
|
void |
renderTopLevel(com.badlogic.gdx.graphics.g2d.SpriteBatch sb)
Renders top level (especially tooltip) effects.
|
protected void |
renderWidget(com.badlogic.gdx.graphics.g2d.SpriteBatch sb)
Custom widgets should implement this method for rendering.
|
HorizontalLayout |
scaleToTallestChild()
Replaces the
desiredHeight set in the constructor (HorizontalLayout(float, float) ) with the height of the tallest child. |
protected void |
setChildrenDelayedMovement(float deltaX,
float deltaY,
InterpolationSpeed movementSpeed,
long startingTimeMillis)
|
void |
show() |
protected void |
updateSize(AbstractWidget newChild) |
T |
withChild(AbstractWidget child)
Let this layout manage the given child using the
defaultChildAnchorPosition as its internal anchor. |
T |
withChild(AbstractWidget child,
AnchorPosition anchor)
Let this layout manage the given child using the provided internal anchor.
|
T |
withDefaultChildAnchorPosition(AnchorPosition defaultChildAnchorPosition)
Sets the AnchorPosition for all future children added to this widget.
|
anchoredAt, anchoredAtClamped, anchoredAtClamped, anchoredCenteredOnMouse, anchoredCenteredOnMouse, anchoredCenteredOnMouseClamped, anchoredCenteredOnMouseClamped, anchoredCenteredOnScreen, anchoredCenteredOnScreen, cancelMovementQueue, delayedTranslate, getBottom, getContentBottom, getContentCenterX, getContentCenterY, getContentLeft, getContentRight, getContentTop, getHeight, getLeft, getRight, getTop, getWidth, initializeInteractivity, isMouseInBounds, isMouseInContentBounds, leftMouseClick, mouseEnter, mouseLeave, onLeftClick, onMouseEnter, onMouseLeave, onRightClick, refreshAnchor, render, resolveMovementQueue, rightMouseClick, scaleHitboxToContent, setAllDelayedMovement, toString, translate, update, updateInteractivity, updateWidget, withMargins, withMargins, withMargins
protected java.util.ArrayList<easel.ui.layouts.LayoutItem> children
protected AnchorPosition defaultChildAnchorPosition
protected float totalWidth
protected float totalHeight
protected float spacing
protected boolean shouldAutoScaleToContent
public HorizontalLayout(float desiredHeight, float spacing)
desiredHeight
will be able to float around vertically in their assigned slot using their anchor position. You can use the other constructor (HorizontalLayout(float)
) if you want to automatically scale the heights to the tallest child, but using this version means the height allowed for children will not be changed unless you manually alter it using scaleToTallestChild()
.desiredHeight
- the height of the horizontal layout (for child anchoring purposes)spacing
- the horizontal gap in between elements of the layoutHorizontalLayout(float)
public HorizontalLayout(float spacing)
scaleToTallestChild()
will be called the FIRST time that any AbstractWidget.anchoredAt(float, float, AnchorPosition)
anchoring occurs (only the first time). This is essentially a convenience method for the usual pattern where the row height set by the other constructor is ignored with a manual scaleToTallestChild()
before anchoring. This variant just does it automatically. To ensure performance, this auto-scaling only occurs the first time an anchoring occurs (which, presumably is after all children have been added), and must be done manually later if things change beyond that.spacing
- the horizontal gap in between elements of the layoutHorizontalLayout(float, float)
public float getContentWidth()
AbstractWidget
getContentWidth
in class AbstractWidget<HorizontalLayout>
AbstractWidget.getWidth()
,
AbstractWidget.getContentHeight()
public float getContentHeight()
AbstractWidget
getContentHeight
in class AbstractWidget<HorizontalLayout>
AbstractWidget.getHeight()
,
AbstractWidget.getContentWidth()
protected void updateSize(AbstractWidget newChild)
protected void autoscale()
protected void anchorChildren(InterpolationSpeed withDelay)
public void clear()
public HorizontalLayout scaleToTallestChild()
desiredHeight
set in the constructor (HorizontalLayout(float, float)
) with the height of the tallest child. Useful for dynamically scaling the height of this widget to fit the heights of its children. NOTE: should call this function AFTER adding all children and before anchoring.public T withDefaultChildAnchorPosition(AnchorPosition defaultChildAnchorPosition)
AnchorPosition.LEFT_TOP
. Only referred to when using the add child methods that do not specify a specific anchor position (if specified, this default child anchor position is ignored).
NOTE: For horizontal layouts, only the vertical position in the anchor matters. This is because children are only allowed to grow horizontally up to their preferred width. However, the total height of the layout is arbitrary and that means that the vertical anchoring can affect the vertical positioning. A similar restriction exists for vertical layouts but for left/right anchors.defaultChildAnchorPosition
- the internal anchor position of each future childpublic T forceChildAnchors(AnchorPosition forcedChildAnchorPosition)
defaultChildAnchorPosition
, so any future children added will continue to use that value if not manually set.forcedChildAnchorPosition
- the new anchor for all existing childrenpublic final T withChild(AbstractWidget child, AnchorPosition anchor)
child
- the child widget to be addedanchor
- where the child will be anchored internally to the space given (see withDefaultChildAnchorPosition(AnchorPosition)
for details about the limitations of the internal anchor)withChild(AbstractWidget)
public final T withChild(AbstractWidget child)
defaultChildAnchorPosition
as its internal anchor. This child is added to the end of the layout (right-most for a horizontal layout, bottom-most for a vertical layout). If the defaultChildAnchorPosition
hasn't been set previously (i.e. through withDefaultChildAnchorPosition(AnchorPosition)
), the default value is AnchorPosition.LEFT_TOP
.child
- the child widget to be addedwithChild(AbstractWidget, AnchorPosition)
public T anchoredAt(float x, float y, AnchorPosition anchorPosition, InterpolationSpeed movementSpeed)
AbstractWidget
Moves the widget towards a specific spot. The anchorPosition
defines the point on the widget that will be placed at the position (x, y)
.
For example, anchoredAt(100, 200, AnchorPosition.LEFT_BOTTOM, InterpolationSpeed.FAST)
will set the bottom-left corner of the widget to 100 pixels from the left side of the screen and 200 pixels from the bottom of the screen. The widget then renders up and to the right of this point since we anchored at an AnchorPosition.LEFT_BOTTOM
. As a second example, using AnchorPosition.CENTER
ensures that (x, y)
will be the center of the widget.
The interpolation speed movementSpeed
determines how quickly the widget moves to the target location. Using a speed other than InterpolationSpeed.INSTANT
makes the widget move towards the desired position over the next few frames in a smoothly animated manner. For convenience since instant moving is often the desired effect, see AbstractWidget.anchoredAt(float, float, AnchorPosition)
.
Note: you should always anchor at least once before rendering. For more dynamic widgets that move a lot (e.g. a tooltip dependent on the mouse cursor location using InputHelper.mX
and InputHelper.mY
), a general pattern is to call anchoredAt
right before rendering in your main render function, e.g.:
widget.anchoredAt(x, y, AnchorPosition.CENTER, InterpolationSpeed.INSTANT)
.render(sb);
anchoredAt
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
x
- the x position in pixels from the left edge of the screeny
- the y position in pixels from the bottom edge of the screenanchorPosition
- what piece of the widget will be moved to (x, y)
movementSpeed
- how quickly the widget will move towards the desired positionAbstractWidget.anchoredAt(float, float, AnchorPosition)
,
AbstractWidget.anchoredAtClamped(float, float, AnchorPosition, float)
,
AbstractWidget.anchoredAtClamped(float, float, AnchorPosition, InterpolationSpeed, float)
protected void setChildrenDelayedMovement(float deltaX, float deltaY, InterpolationSpeed movementSpeed, long startingTimeMillis)
AbstractWidget
AbstractWidget.delayedTranslate(float, float, InterpolationSpeed, long)
. Make sure any widget with children (e.g. containers, layouts, etc.) overrides this function and calls AbstractWidget.setAllDelayedMovement(float, float, InterpolationSpeed, long)
using the input to this function on all direct descendants. This is required if you want to be able to use a delayed anchoredAt command on custom widgets.setChildrenDelayedMovement
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
deltaX
- how much movement horizontallydeltaY
- how much movement verticallymovementSpeed
- how fast the widget will move towards the target, once startingTimeMillis is reached (i.e. System.currentTimeMillis()
is greater than or equal to this starting time)startingTimeMillis
- a time generated by an offset of System.currentTimeMillis()
, determined by the original AbstractWidget.delayedTranslate(float, float, InterpolationSpeed, long)
function that starts this chainprotected void cancelMovementQueueForAllChildren(boolean shouldTryAndResolveOneLastTime)
cancelMovementQueueForAllChildren
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
public java.util.stream.Stream<AbstractWidget> iterator()
public <T> java.util.stream.Stream<T> iteratorOfType(java.lang.Class<T> clz)
iterator()
due to checking each child against the type for safe casts, but the resulting stream will be properly typed to make it easier to work with. Children managed by this layout who are not of the given type will not be included in the stream, so be wary if using this variant of the iterator.T
- the type of child to extractclz
- the class for the typeiterator()
protected void renderWidget(com.badlogic.gdx.graphics.g2d.SpriteBatch sb)
AbstractWidget
Custom widgets should implement this method for rendering. Use the inner content positions (e.g. AbstractWidget.getContentLeft()
, AbstractWidget.getContentWidth()
, etc.) to determine any position information necessary for rendering at a specific location. If the library is used as intended, these content locations should be accurate to where the widget needs to be rendered, as they reflect the most up to date location set by an anchoredAt call (this automatically will be interpolated if the anchorAt move is set to occur over several frames).
Note: you NEED to revert any changes you make to the SpriteBatch (e.g. setting a shader, changing the perspective matrix, etc.) by the time this function returns, as the SpriteBatch will be reused for rendering other widgets which will not expect those changes. You also don't technically need to render to this particular SpriteBatch (e.g. you can render to your own batch if you know what you're doing), as long as you follow the general intent of this function to render the widget.
renderWidget
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
sb
- the SpriteBatch the widget should be rendered onAbstractWidget.renderTopLevel(SpriteBatch)
public void renderTopLevel(com.badlogic.gdx.graphics.g2d.SpriteBatch sb)
AbstractWidget
Renders top level (especially tooltip) effects. Like AbstractWidget.render(SpriteBatch)
, but deliberately delayed. If any of your widgets require custom (Widget-based, not base-game TipHelper based) tooltips or other specific rendering that needs to go on top of all other widgets in the hierarchy, you'll need to set your root node to call this after its main AbstractWidget.render(SpriteBatch)
completes.
If you are hooking into BaseMod's RenderSubscriber
for your base class, you can simply call this after your main call to rootWidget.render(spriteBatch)
. All layout managers and other container widgets will extend the calls down the tree automatically (much like how render()
works), meaning you only need to call this at the highest level of your widget hierarchy.
public void receiveRender(SpriteBatch spriteBatch) {
// Our normal render call will trickle down the hierarchy in a breadth-first manner, rendering every descendant widget.
baseWidget.render(spriteBatch);
// Call this after the above so that we can support custom tooltips (if desired); also trickles down the tree
baseWidget.renderTopLevel(spriteBatch);
}
Note that this is only required if you are using custom widgets that need to be rendered last (on top of everything). If you are sticking to base game style tooltips (e.g. using something like the TipHelper
statics), this is NOT required. However, if you do end up needing to draw on top of all widgets in your hierarchy, this is a solid way to do it as it follows the same breadth-first visit order but occurs after all the regular rendering takes place.
renderTopLevel
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
sb
- the SpriteBatch to render onAbstractWidget.render(SpriteBatch)
public void hide()
hide
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>
public void show()
show
in class AbstractWidget<T extends easel.ui.layouts.AbstractOneDimensionalLayout<T>>