2022-03-18 17:46:08 +01:00
|
|
|
.. _doc_custom_gui_controls:
|
|
|
|
|
|
|
|
Custom GUI controls
|
|
|
|
===================
|
|
|
|
|
|
|
|
So many controls...
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
Yet there are never enough. Creating your own custom controls that act
|
|
|
|
just the way you want them to is an obsession of almost every GUI
|
|
|
|
programmer. Godot provides plenty of them, but they may not work exactly
|
|
|
|
the way you want. Before contacting the developers with a pull-request
|
|
|
|
to support diagonal scrollbars, at least it will be good to know how to
|
|
|
|
create these controls easily from script.
|
|
|
|
|
|
|
|
Drawing
|
|
|
|
-------
|
|
|
|
|
2023-01-12 19:29:11 +01:00
|
|
|
For drawing, it is recommended to check the `doc_custom_drawing_in_2d` tutorial.
|
2022-03-18 17:46:08 +01:00
|
|
|
The same applies. Some functions are worth mentioning due to their
|
|
|
|
usefulness when drawing, so they will be detailed next:
|
|
|
|
|
|
|
|
Checking control size
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Unlike 2D nodes, "size" is important with controls, as it helps to
|
|
|
|
organize them in proper layouts. For this, the
|
2023-01-12 19:30:47 +01:00
|
|
|
`Control.rect_size`
|
2022-03-18 17:46:08 +01:00
|
|
|
property is provided. Checking it during ``_draw()`` is vital to ensure
|
|
|
|
everything is kept in-bounds.
|
|
|
|
|
|
|
|
Checking focus
|
|
|
|
~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Some controls (such as buttons or text editors) might provide input
|
|
|
|
focus for keyboard or joypad input. Examples of this are entering text
|
|
|
|
or pressing a button. This is controlled with the
|
2023-01-12 19:30:47 +01:00
|
|
|
`Control.focus_mode`
|
2022-03-18 17:46:08 +01:00
|
|
|
property. When drawing, and if the control supports input focus, it is
|
|
|
|
always desired to show some sort of indicator (highlight, box, etc.) to
|
|
|
|
indicate that this is the currently focused control. To check for this
|
2023-01-12 19:30:47 +01:00
|
|
|
status, the `Control.has_focus()` method
|
2022-03-18 17:46:08 +01:00
|
|
|
exists. Example
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
func _draw():
|
|
|
|
if has_focus():
|
|
|
|
draw_selected()
|
|
|
|
else:
|
|
|
|
draw_normal()
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
Sizing
|
|
|
|
------
|
|
|
|
|
|
|
|
As mentioned before, size is important to controls. This allows
|
|
|
|
them to lay out properly, when set into grids, containers, or anchored.
|
|
|
|
Controls, most of the time, provide a *minimum size* to help properly
|
|
|
|
lay them out. For example, if controls are placed vertically on top of
|
2023-01-12 19:30:47 +01:00
|
|
|
each other using a `VBoxContainer`,
|
2022-03-18 17:46:08 +01:00
|
|
|
the minimum size will make sure your custom control is not squished by
|
|
|
|
the other controls in the container.
|
|
|
|
|
|
|
|
To provide this callback, just override
|
2023-01-12 19:30:47 +01:00
|
|
|
`Control.get_minimum_size()`,
|
2022-03-18 17:46:08 +01:00
|
|
|
for example:
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
func get_minimum_size():
|
|
|
|
return Vector2(30, 30)
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
Alternatively, set it using a function:
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
func _ready():
|
|
|
|
set_custom_minimum_size(Vector2(30, 30))
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
Input
|
|
|
|
-----
|
|
|
|
|
|
|
|
Controls provide a few helpers to make managing input events much easier
|
|
|
|
than regular nodes.
|
|
|
|
|
|
|
|
Input events
|
|
|
|
~~~~~~~~~~~~
|
|
|
|
|
|
|
|
There are a few tutorials about input before this one, but it's worth
|
|
|
|
mentioning that controls have a special input method that only works
|
|
|
|
when:
|
|
|
|
|
|
|
|
- The mouse pointer is over the control.
|
|
|
|
- The button was pressed over this control (control always
|
|
|
|
captures input until button is released)
|
|
|
|
- Control provides keyboard/joypad focus via
|
2023-01-12 19:30:47 +01:00
|
|
|
`Control.focus_mode`.
|
2022-03-18 17:46:08 +01:00
|
|
|
|
|
|
|
This function is
|
2023-01-12 19:30:47 +01:00
|
|
|
`Control._gui_input()`.
|
2022-03-18 17:46:08 +01:00
|
|
|
Simply override it in your control. No processing needs to be set.
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
extends Control
|
|
|
|
|
|
|
|
func _gui_input(event):
|
|
|
|
if event is InputEventMouseButton and event.button_index == BUTTON_LEFT and event.pressed:
|
|
|
|
print("Left mouse button was pressed!")
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 19:29:11 +01:00
|
|
|
For more information about events themselves, check the `doc_inputevent`
|
2022-03-18 17:46:08 +01:00
|
|
|
tutorial.
|
|
|
|
|
|
|
|
Notifications
|
|
|
|
~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Controls also have many useful notifications for which no dedicated callback
|
|
|
|
exists, but which can be checked with the _notification callback:
|
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
gdscript GDScript
|
2022-03-18 17:46:08 +01:00
|
|
|
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|
2022-03-18 17:46:08 +01:00
|
|
|
func _notification(what):
|
|
|
|
match what:
|
|
|
|
NOTIFICATION_MOUSE_ENTER:
|
|
|
|
pass # Mouse entered the area of this control.
|
|
|
|
NOTIFICATION_MOUSE_EXIT:
|
|
|
|
pass # Mouse exited the area of this control.
|
|
|
|
NOTIFICATION_FOCUS_ENTER:
|
|
|
|
pass # Control gained focus.
|
|
|
|
NOTIFICATION_FOCUS_EXIT:
|
|
|
|
pass # Control lost focus.
|
|
|
|
NOTIFICATION_THEME_CHANGED:
|
|
|
|
pass # Theme used to draw the control changed;
|
|
|
|
# update and redraw is recommended if using a theme.
|
|
|
|
NOTIFICATION_VISIBILITY_CHANGED:
|
|
|
|
pass # Control became visible/invisible;
|
|
|
|
# check new status with is_visible().
|
|
|
|
NOTIFICATION_RESIZED:
|
|
|
|
pass # Control changed size; check new size
|
|
|
|
# with get_size().
|
|
|
|
NOTIFICATION_MODAL_CLOSE:
|
|
|
|
pass # For modal pop-ups, notification
|
|
|
|
# that the pop-up was closed.
|
2023-01-12 18:31:02 +01:00
|
|
|
```
|