Well, I've tried to avoid the subject for as long as possible, but I think it's not handy to delay it any more.
Small confession: creating UIs is completely not my thing :) However handling the cards is really not handy at the moment. so we have to fix that - at least a little bit. The code in this part is going to be rather simple (in it's logic), but the volume might get heavier :)
The final code for this part can be found here: https://github.com/maciekglowka/hike_deck/tree/part_08
Assets
As the UI will surely grow into a large part of the game sooner or later, it's definitely gonna need a separate module / plugin. Let's start then by creating a brand new UI module. Next we'll define a resource in it, that is going to hold some necessary assets:
For now we're going to use a single font and a single texture image for our card. (I've put those files into a separate assets/ui/ folder). The card background is a simple static PNG file (like you can see below). I'd probably prefer to use something slightly more fancy, like a 9-slice - but haven't found a good / simple way to achieve it. I now that there are some crates for that, however they're either quite early in their development or have some quirky behaviour (or I could not find the good ones - let me know :).

The texture handles are held within a HashMap so we'll be able to easily add more in the future. (without redefining the asset struct)
Let's look at loading those assets then. We are going to take the same approach as with the sprites before:
// ui/assets.rs
use *;
use HashMap;
use UiAssets;
const TEXTURES:  = ;
The single element TEXTURES array might look weird at the moment, but surely we are going to add more images later. So it's just a bit of future-proofing :)
Buttons
As the UI is usually drifting towards many nested pieces of code (like children nodes, inside of other children nodes etc.), I'd like to break it down a little - into separate helper functions.
Let's start with something that will be the basis for our in-game buttons:
// ui/helpers.rs
use *;
;
It is mostly just a plain ButtonBundle, with an extra marker component - that we are going to use as a filter for the click animation:
As you can see, we take a very simple approach here. If we detect that the button is in a Clicked state we scale it down a bit. Otherwise we return to a scale of 1. Not very pixel-perfect - but will do for now. (this fn will need to be registered as a system).
Let's also create a helper for a TextBundle- that we're gonna use as a content for the buttons:
// in ui/helpers.rs
const FONT_SIZE: f32 = 18.;
// cut
Deck Menu
Now we can move on to the actual deck menu code:
// ui/deck.rs
use *;
use crate;
use ;
const DECK_HEIGHT: f32 = 150.;
const CARD_WIDTH: f32 = 96.;
const CARD_HEIGHT: f32 = 128.;
const CARD_MARGIN: f32 = 4.;
const CARD_SELECT: f32 = 24.;
;
;
Phew, that's a lot of code. Let's break it down.
At the beginning we import some structs from the Player module - as we are going to render the UI elements based on that. Then we define constants, describing basic dimensions of our deck menu (again, it's easier to keep track of those values when they're in one place).
Next, we define two components: a marker for the entire deck menu and a CardButton. The button component has two fields. The first one is an entity of a card the button is going to represent (we'll get it from the deck resource). The bool field is going to help us determine the current button state (I'm gonna come back to this a bit later).
Then we create a rather lengthy system to build our card menu. At the beginning we make sure that no other deck is present (we do it via a separate helper func). Next, we spawn the container node, which will occupy the bottom part of our screen.
We iterate through our Deck resource to find all the cards the player is currently holding and create buttons based on that. The button size is set to match our card background sprite, so there is no sub-pixel scaling. To mark the actively selected card, we are going to shift the button up a bit in the deck layout - by increasing it's bottom margin. (you'll see how it works once we run it).
Finally, we create the button's text content, based on a result of a get_label method - that does not exist yet and will give you compiler errors :)
So let's jump for a while into the Player module and fix that:
// in player/cards.rs
We include the get_label method in our Card trait and then do some basic implementations for both our concrete structs. For now we include a line break in the MeleeCard's label, to keep thing more simple - but it's not the best long-term solution, as we mix here data with presentation (so in the future we'll hopefully come back to that :)
UI Plugin
Now that we have all the sub-elements, we can finally create our UI Plugin and register the systems (don't forget to register the plugin in the main.rs as well!):
// ui/mod.rs
use *;
use HashMap;
use crateGameState;
;
;
You might notice that I've created a new event here - the ReloadUiEvent. We are going to fire it whenever something changes within the game and the UI has to be refreshed (eg. the player will lose HP or smth). Having a single event, means that we are going to redraw always the entire UI. If we run into performance issues we can always split this into some partial updates in the future. But again, for now I would not worry about that. The solution is not as fancy as implementing some kind of reactivity - but for a simple turn-based game it should be ok-ish.
We are going to use the event as a run condition for our deck drawing system (and in the future for all the other UI builders). We are also going to create a helper system, that would fire this event every time we enter the PlayerInput phase. This way we won't have to schedule those other systems twice (once on reload, second time on the input start).
Now we are finally ready to test effects of the above code. If you do cargo run you'll hopefully see a deck drawn like this:

Card selection
You can click the cards and they should nicely animate the button press, but there will be no game effect yet. We still have to implement the actual card selection logic.
Let's start then by removing some redundant parts from our input module:
// delete this from input/mod.rs
    if keys.just_pressed 
    if keys.just_pressed 
Now, we will create a system somewhat similar to the button_click_animation (shown earlier) - as it is also going to deal with button interactions:
// in ui/deck.rs
As Bevy's interactions do not report the released state (at least for now, I believe it is actually proposed to change) - we have to do some workaround here, in order to fire an action only after the release. We could react on the button down press, but I kinda prefer it the other way. It feels more natural somehow and allows for eg. cancelling the click (if we move the mouse outside of the button area, before the release).
We are going to use the boolean parameter of the CardButton component to track whether the button is currently in the press state. This way when we enter the Hovered state we can detect the button release (with the mouse cursor still above the button). This works because we can enter the Hovered state in two ways:
- when we move the cursor over the button with no button pressed (the expected behaviour)
- when the button has been pressed and we release it with the mouse cursor still over it
If we detect Hovered state and the bool parameter is set - we will send the card-select event to let the Player module know about our intent. We are also going the fire the ReloadUiEvent as we want to show that a card has been selected (so we redraw the deck).
Lastly we have to register this system in our plugin (ideally to run only during the PlayerInput phase) and we can test run our game again. Now we should be able to switch between the player's action types from our deck!