Research assignment for Project II, talking about: Data driven UI and Animations.
This project is maintained by Guillemsc
As we have less things on our projects being configured from code, the better. That applies to console configurations, player starting stats, positions, animations, etc, and also, to UI management and creation. That’s when data driven UI begins. This part of the research is going to talk about the creation and implementation of a UI data driven manager on our code, that will interpretate XML commands, to create full UI systems.
To begin with, we have to determine the type of UI base system that we are going to use, because that is going to change completly the internal organization of the data manager. Since it’s impossibe to create a general data manager without knowing the type of UI that is going to use, this one is going to be based on the UI that i’ve created on the past. The general structure of this UI is the following:
The way in which this works, is by creating a Window, and from there, we will create all the different UI elements that will be childs of that window.
Once we have defined our UI system, we need to know the way that our UI data driven manager is going to work. It should have the following abilities:
From that, we have to know which XML commands will the code understand, for examle:
<Element type = "Image" name ="image" position_x ="700" position_y ="200" rect_x ="0" rect_y =" 0" .../>
<Element type ="Text" name ="text" position_x ="775" position_y = "368" .../>
First of all, since we have to have different scene recongition, we will start with scene definition:
<scene name = "menu_scene"> // Name of the scene should match with the name set on code.
</scene>
Everything that we will create in a scene should be placed inside the scene brackets. Next, we need the windows:
<Window name = "main_window" position_x = "700" position_y = "200" size_w = "225" size_h = "219" blit_layer = "1" dinamic = "false" is_ui ="true">
</Window>
With that, evey UI elements that pertains to this window should be created inside the window brackets. Then we need the actual elements. They also have name to then be able to refeer to them in some actions. We can add the interpretation of any the UI elements that we have on the UI, for example:
<Element type ="Text" name ="text" position_x ="754" position_y = "265" font ="2" spacing ="0" color_r ="168" color_g ="134" color_b ="0">
As we can see, we are creating a text here, but where do we define the actual text that is goint to print? We need to be able to recieve other parameters!
<Element type ="Text" name ="text" position_x ="754" position_y = "265" font ="2" spacing ="0" color_r ="168" color_g ="134" color_b ="0">
<var text = "Start Game"/>
<var enabled ="false"/> // In this case the element won't start enabled
</Element>
We can create any type of element, and in the case that we detect that the scene is changed, we unload and we load the new UI.
As it can be very usefull when creating data driven ui actions, it can be interesting the creation of data driven dynamic variables to expand the uses that we can give to our UI. This is not mandatory, but can be usefull:
<new_var start_value ="0" name ="percentage" max_value ="100" min_value ="0"/>
Here is when the thing starts to become a little bit messy. It’s important that we can define basic UI behaviour to really explore what data driven UI can do. From this point, the possibilities of actions and interactions are infinite, to the level of even represent full c++ code, so let’s try not get very complex. Actions also should go inside the scene bracket.
<Scene>
<Action>
</Action>
</Scene>
Next, thanks to the ability to reference to other UI elements thanks to it’s names, we can start to build basic behaviours, for examle: An easy one:
<Action>
<act name ="button_start" if_left_pressed ="true"> // If button left pressed
<act load_scene ="second_scene"/> // Load scene
</act>
</Action>
A more complex one:
<act name ="button_options" if_left_pressed ="true"> // If button left pressed
<act name ="options_window" if_enabled ="true"> // If options window is enabled
// Do an animation
<act name = "options_window" animation = "movement" movement_type = "smooth" destination_x = "-500" destination_y = "130" time = "0.5">
// When animation is finished, disable element
<act name ="options_window" set_enabled ="false"/>
</act>
</act>
<act name ="options_window" if_enabled ="false"> // If options window is disabled
<act name ="options_window" set_enabled ="1"/> // Enable it
// Do an animation
<act name = "options_window" animation = "movement" movement_type = "smooth" destination_x = "400" destination_y = "130" time = "0.5"/>
</act>
</act>
With that we can archive a decent/easy UI XML creation.
## 2. UI Animations ### 2.1 Introduction Apart from easy to use and functional, and UI has to be pleasant and visual to the user. With that in mind, UI animations thake a hiuge part on the UI creation process and thoguht. We can find a lot of examples on mobile games, where everyghing has to be soft, bouncy and pleasant to the use.
### 2.2 Initial thoughts To begin with, we have a huge amount of different animations that we could implement, but in this research we are just going to focus on UI movement animations, moving from one point to another, in a fixed amount of time.
Going from there, we can have an element moving form point A to point B in a defined amount of time with a fixed speed. But having a constant speed is quite boring and not pleasant at all, so we can start looking at curves of movement, to have different effects, for example:
### 2.3 Bézier We can archive this type of movements with Bézier curves, which are very used on software development:
You can find them, for example, in photoshop/illustrator/etc tools, to define paths, etc.. The Bézier curves are defined by this recursive function:
But for our case with the cubic function is enough:
Depending on the points that we give to the ecuation, we end up with a path, that we can then interpretate into code to have smooth movements. We can see examples of this on this web pages:
easings.net | cubic-bezier.com |
---|---|
### 2.4 Implementation The implementation it’s very similar to in the other cases, and they will be used as actions inside the Action brackets:
<act name = "options_window" animation = "movement" movement_type = "smooth" destination_x = "-500" destination_y = "130" time = "0.5">
<Something here> // This will be executed when the animation finishes.
</act>
Depending on the type of movement that we use, we add the prefered points to the bézier equation.
## 3. Conclusion
The high-level summary is that it is possible to automatically generate good quality user interfaces, but only in certain conditions. The most important trade-off to consider when deciding whether or not to try automatic generation is the amount of information that must be communicated to the automatic system in order for it to produce a good interface design.
The code of the solution and the exercice that you will find it’s just only a gideline and it’s hard to adapt to another project/type of UI. Use it just as a gideline for the strucuture.
In this exercice you will be creating a basic UI with it’s behaviours, using only the XML, as an experience in how fast, easy, difficult, etc.. compared with in-code UI creation.
Download exercice and solution: http://www.mediafire.com/file/xwy95p4us45ffw1/DataDrivenUI.zip
Create all and only the UI elements using the blueprints that you can find on the XML.
Create the basic actions that you can observe on the solution:
Create a variable and increase the value of the number when pressing the buttons ‘+’ and ‘-‘.
Button options: position_x = “717” position_y = “310” size_w = “194” size_h = “37”