Add font_separator to PopupMenu

This commit is contained in:
Michael Alexsander 2022-05-08 14:11:59 -03:00 committed by Relintai
parent d9d2b3a7e3
commit 9af14c8d90
3 changed files with 22 additions and 11 deletions

View File

@ -527,6 +527,9 @@
<theme_item name="font" data_type="font" type="Font"> <theme_item name="font" data_type="font" type="Font">
[Font] used for the menu items. [Font] used for the menu items.
</theme_item> </theme_item>
<theme_item name="font_separator" data_type="font" type="Font">
[Font] used for the labeled separator.
</theme_item>
<theme_item name="checked" data_type="icon" type="Texture"> <theme_item name="checked" data_type="icon" type="Texture">
[Texture] icon for the checked checkbox items. [Texture] icon for the checked checkbox items.
</theme_item> </theme_item>

View File

@ -72,15 +72,18 @@ Size2 PopupMenu::get_minimum_size() const {
Size2 minsize = get_stylebox("panel")->get_minimum_size(); Size2 minsize = get_stylebox("panel")->get_minimum_size();
Ref<Font> font = get_font("font"); Ref<Font> font = get_font("font");
Ref<Font> font_separator = get_font("font_separator");
float max_w = 0; float max_w = 0;
float icon_w = 0; float icon_w = 0;
int font_h = font->get_height();
int check_w = MAX(get_icon("checked")->get_width(), get_icon("radio_checked")->get_width()) + hseparation; int check_w = MAX(get_icon("checked")->get_width(), get_icon("radio_checked")->get_width()) + hseparation;
int accel_max_w = 0; int accel_max_w = 0;
bool has_check = false; bool has_check = false;
for (int i = 0; i < items.size(); i++) { for (int i = 0; i < items.size(); i++) {
String text = items[i].xl_text;
int font_h = items[i].separator && text != String() ? font_separator->get_height() : font->get_height();
Size2 size; Size2 size;
if (!items[i].icon.is_null()) { if (!items[i].icon.is_null()) {
Size2 icon_size = items[i].icon->get_size(); Size2 icon_size = items[i].icon->get_size();
@ -96,8 +99,7 @@ Size2 PopupMenu::get_minimum_size() const {
has_check = true; has_check = true;
} }
String text = items[i].xl_text; size.width += items[i].separator ? font_separator->get_string_size(text).width : font->get_string_size(text).width;
size.width += font->get_string_size(text).width;
size.height += vseparation; size.height += vseparation;
if (items[i].accel || (items[i].shortcut.is_valid() && items[i].shortcut->is_valid())) { if (items[i].accel || (items[i].shortcut.is_valid() && items[i].shortcut->is_valid())) {
@ -137,10 +139,13 @@ int PopupMenu::_get_mouse_over(const Point2 &p_over) const {
} }
Ref<Font> font = get_font("font"); Ref<Font> font = get_font("font");
Ref<Font> font_separator = get_font("font_separator");
int vseparation = get_constant("vseparation"); int vseparation = get_constant("vseparation");
float font_h = font->get_height();
for (int i = 0; i < items.size(); i++) { for (int i = 0; i < items.size(); i++) {
String text = items[i].xl_text;
float font_h = items[i].separator && text != String() ? font_separator->get_height() : font->get_height();
ofs.y += vseparation; ofs.y += vseparation;
float h; float h;
@ -505,6 +510,7 @@ void PopupMenu::_notification(int p_what) {
Ref<StyleBox> style = get_stylebox("panel"); Ref<StyleBox> style = get_stylebox("panel");
Ref<StyleBox> hover = get_stylebox("hover"); Ref<StyleBox> hover = get_stylebox("hover");
Ref<Font> font = get_font("font"); Ref<Font> font = get_font("font");
Ref<Font> font_separator = get_font("font_separator");
// In Item::checkable_type enum order (less the non-checkable member) // In Item::checkable_type enum order (less the non-checkable member)
Ref<Texture> check[] = { get_icon("checked"), get_icon("radio_checked") }; Ref<Texture> check[] = { get_icon("checked"), get_icon("radio_checked") };
Ref<Texture> uncheck[] = { get_icon("unchecked"), get_icon("radio_unchecked") }; Ref<Texture> uncheck[] = { get_icon("unchecked"), get_icon("radio_unchecked") };
@ -522,7 +528,6 @@ void PopupMenu::_notification(int p_what) {
Color font_color_accel = get_color("font_color_accel"); Color font_color_accel = get_color("font_color_accel");
Color font_color_hover = get_color("font_color_hover"); Color font_color_hover = get_color("font_color_hover");
Color font_color_separator = get_color("font_color_separator"); Color font_color_separator = get_color("font_color_separator");
float font_h = font->get_height();
// Add the check and the wider icon to the offset of all items. // Add the check and the wider icon to the offset of all items.
float icon_ofs = 0.0; float icon_ofs = 0.0;
@ -546,6 +551,9 @@ void PopupMenu::_notification(int p_what) {
} }
for (int i = 0; i < items.size(); i++) { for (int i = 0; i < items.size(); i++) {
String text = items[i].xl_text;
float font_h = items[i].separator && text != String() ? font_separator->get_height() : font->get_height();
if (i == 0) { if (i == 0) {
ofs.y += vseparation / 2; ofs.y += vseparation / 2;
} else { } else {
@ -566,12 +574,10 @@ void PopupMenu::_notification(int p_what) {
hover->draw(ci, Rect2(item_ofs + Point2(-hseparation, -vseparation / 2), Size2(get_size().width - style->get_minimum_size().width + hseparation * 2, h + vseparation))); hover->draw(ci, Rect2(item_ofs + Point2(-hseparation, -vseparation / 2), Size2(get_size().width - style->get_minimum_size().width + hseparation * 2, h + vseparation)));
} }
String text = items[i].xl_text;
item_ofs.x += items[i].h_ofs; item_ofs.x += items[i].h_ofs;
if (items[i].separator) { if (items[i].separator) {
if (text != String()) { if (text != String()) {
int ss = font->get_string_size(text).width / 2; int ss = font_separator->get_string_size(text).width / 2;
int center = get_size().width / 2; int center = get_size().width / 2;
int l = center - ss; int l = center - ss;
int r = center + ss; int r = center + ss;
@ -604,14 +610,15 @@ void PopupMenu::_notification(int p_what) {
submenu->draw(ci, Point2(size.width - style->get_margin(MARGIN_RIGHT) - submenu->get_width(), item_ofs.y + Math::floor(h - submenu->get_height()) / 2), icon_color); submenu->draw(ci, Point2(size.width - style->get_margin(MARGIN_RIGHT) - submenu->get_width(), item_ofs.y + Math::floor(h - submenu->get_height()) / 2), icon_color);
} }
item_ofs.y += font->get_ascent();
if (items[i].separator) { if (items[i].separator) {
if (text != String()) { if (text != String()) {
int center = (get_size().width - font->get_string_size(text).width) / 2; item_ofs.y += font_separator->get_ascent();
font->draw(ci, Point2(center, item_ofs.y + Math::floor((h - font_h) / 2.0)), text, font_color_separator); int center = (get_size().width - font_separator->get_string_size(text).width) / 2;
font_separator->draw(ci, Point2(center, item_ofs.y + Math::floor((h - font_h) / 2.0)), text, font_color_separator);
} }
} else { } else {
item_ofs.x += icon_ofs + check_ofs; item_ofs.x += icon_ofs + check_ofs;
item_ofs.y += font->get_ascent();
font->draw(ci, item_ofs + Point2(0, Math::floor((h - font_h) / 2.0)), text, items[i].disabled ? font_color_disabled : (i == mouse_over ? font_color_hover : font_color)); font->draw(ci, item_ofs + Point2(0, Math::floor((h - font_h) / 2.0)), text, items[i].disabled ? font_color_disabled : (i == mouse_over ? font_color_hover : font_color));
} }

View File

@ -596,6 +596,7 @@ void fill_default_theme(Ref<Theme> &theme, const Ref<Font> &default_font, const
theme->set_icon("submenu", "PopupMenu", make_icon(submenu_png)); theme->set_icon("submenu", "PopupMenu", make_icon(submenu_png));
theme->set_font("font", "PopupMenu", default_font); theme->set_font("font", "PopupMenu", default_font);
theme->set_font("font_separator", "PopupMenu", default_font);
theme->set_color("font_color", "PopupMenu", control_font_color); theme->set_color("font_color", "PopupMenu", control_font_color);
theme->set_color("font_color_accel", "PopupMenu", Color(0.7, 0.7, 0.7, 0.8)); theme->set_color("font_color_accel", "PopupMenu", Color(0.7, 0.7, 0.7, 0.8));