diff --git a/doc/classes/ProjectSettings.xml b/doc/classes/ProjectSettings.xml index a1a320446b3..383049fb3e4 100644 --- a/doc/classes/ProjectSettings.xml +++ b/doc/classes/ProjectSettings.xml @@ -786,9 +786,45 @@ Optional name for the 2D navigation layer 2. If left empty, the layer will display as "Layer 2". + + Optional name for the 2D navigation layer 20. If left empty, the layer will display as "Layer 20". + + + Optional name for the 2D navigation layer 21. If left empty, the layer will display as "Layer 21". + + + Optional name for the 2D navigation layer 22. If left empty, the layer will display as "Layer 22". + + + Optional name for the 2D navigation layer 23. If left empty, the layer will display as "Layer 23". + + + Optional name for the 2D navigation layer 24. If left empty, the layer will display as "Layer 24". + + + Optional name for the 2D navigation layer 25. If left empty, the layer will display as "Layer 25". + + + Optional name for the 2D navigation layer 26. If left empty, the layer will display as "Layer 26". + + + Optional name for the 2D navigation layer 27. If left empty, the layer will display as "Layer 27". + + + Optional name for the 2D navigation layer 28. If left empty, the layer will display as "Layer 28". + + + Optional name for the 2D navigation layer 29. If left empty, the layer will display as "Layer 29". + Optional name for the 2D navigation layer 3. If left empty, the layer will display as "Layer 3". + + Optional name for the 2D navigation layer 30. If left empty, the layer will display as "Layer 30". + + + Optional name for the 2D navigation layer 31. If left empty, the layer will display as "Layer 31". + Optional name for the 2D navigation layer 4. If left empty, the layer will display as "Layer 4". @@ -846,9 +882,45 @@ Optional name for the 2D physics layer 2. If left empty, the layer will display as "Layer 2". + + Optional name for the 2D physics layer 20. If left empty, the layer will display as "Layer 20". + + + Optional name for the 2D physics layer 21. If left empty, the layer will display as "Layer 21". + + + Optional name for the 2D physics layer 22. If left empty, the layer will display as "Layer 22". + + + Optional name for the 2D physics layer 23. If left empty, the layer will display as "Layer 23". + + + Optional name for the 2D physics layer 24. If left empty, the layer will display as "Layer 24". + + + Optional name for the 2D physics layer 25. If left empty, the layer will display as "Layer 25". + + + Optional name for the 2D physics layer 26. If left empty, the layer will display as "Layer 26". + + + Optional name for the 2D physics layer 27. If left empty, the layer will display as "Layer 27". + + + Optional name for the 2D physics layer 28. If left empty, the layer will display as "Layer 28". + + + Optional name for the 2D physics layer 29. If left empty, the layer will display as "Layer 29". + Optional name for the 2D physics layer 3. If left empty, the layer will display as "Layer 3". + + Optional name for the 2D physics layer 30. If left empty, the layer will display as "Layer 30". + + + Optional name for the 2D physics layer 31. If left empty, the layer will display as "Layer 31". + Optional name for the 2D physics layer 4. If left empty, the layer will display as "Layer 4". @@ -966,9 +1038,45 @@ Optional name for the 3D navigation layer 2. If left empty, the layer will display as "Layer 2". + + Optional name for the 3D navigation layer 20. If left empty, the layer will display as "Layer 20". + + + Optional name for the 3D navigation layer 21. If left empty, the layer will display as "Layer 21". + + + Optional name for the 3D navigation layer 22. If left empty, the layer will display as "Layer 22". + + + Optional name for the 3D navigation layer 23. If left empty, the layer will display as "Layer 23". + + + Optional name for the 3D navigation layer 24. If left empty, the layer will display as "Layer 24". + + + Optional name for the 3D navigation layer 25. If left empty, the layer will display as "Layer 25". + + + Optional name for the 3D navigation layer 26. If left empty, the layer will display as "Layer 26". + + + Optional name for the 3D navigation layer 27. If left empty, the layer will display as "Layer 27". + + + Optional name for the 3D navigation layer 28. If left empty, the layer will display as "Layer 28". + + + Optional name for the 3D navigation layer 29. If left empty, the layer will display as "Layer 29". + Optional name for the 3D navigation layer 3. If left empty, the layer will display as "Layer 3". + + Optional name for the 3D navigation layer 30. If left empty, the layer will display as "Layer 30". + + + Optional name for the 3D navigation layer 31. If left empty, the layer will display as "Layer 31". + Optional name for the 3D navigation layer 4. If left empty, the layer will display as "Layer 4". @@ -1026,9 +1134,45 @@ Optional name for the 3D physics layer 2. If left empty, the layer will display as "Layer 2". + + Optional name for the 3D physics layer 20. If left empty, the layer will display as "Layer 20". + + + Optional name for the 3D physics layer 21. If left empty, the layer will display as "Layer 21". + + + Optional name for the 3D physics layer 22. If left empty, the layer will display as "Layer 22". + + + Optional name for the 3D physics layer 23. If left empty, the layer will display as "Layer 23". + + + Optional name for the 3D physics layer 24. If left empty, the layer will display as "Layer 24". + + + Optional name for the 3D physics layer 25. If left empty, the layer will display as "Layer 25". + + + Optional name for the 3D physics layer 26. If left empty, the layer will display as "Layer 26". + + + Optional name for the 3D physics layer 27. If left empty, the layer will display as "Layer 27". + + + Optional name for the 3D physics layer 28. If left empty, the layer will display as "Layer 28". + + + Optional name for the 3D physics layer 29. If left empty, the layer will display as "Layer 29". + Optional name for the 3D physics layer 3. If left empty, the layer will display as "Layer 3". + + Optional name for the 3D physics layer 30. If left empty, the layer will display as "Layer 30". + + + Optional name for the 3D physics layer 31. If left empty, the layer will display as "Layer 31". + Optional name for the 3D physics layer 4. If left empty, the layer will display as "Layer 4". diff --git a/editor/editor_properties.cpp b/editor/editor_properties.cpp index df763d5bf0f..fa03d281f26 100644 --- a/editor/editor_properties.cpp +++ b/editor/editor_properties.cpp @@ -691,17 +691,39 @@ EditorPropertyFlags::EditorPropertyFlags() { class EditorPropertyLayersGrid : public Control { GDCLASS(EditorPropertyLayersGrid, Control); -public: - uint32_t value; +private: Vector flag_rects; - Vector names; - Vector tooltips; - int hovered_index; + Rect2 expand_rect; + bool expand_hovered = false; + bool expanded = false; + int expansion_rows = 0; + int hovered_index = -1; - virtual Size2 get_minimum_size() const override { + Size2 get_grid_size() const { Ref font = get_theme_font(SNAME("font"), SNAME("Label")); int font_size = get_theme_font_size(SNAME("font_size"), SNAME("Label")); - return Vector2(0, font->get_height(font_size) * 2); + return Vector2(0, font->get_height(font_size) * 3); + } + +public: + uint32_t value = 0; + int layer_group_size = 0; + int layer_count = 0; + Vector names; + Vector tooltips; + + virtual Size2 get_minimum_size() const override { + Size2 min_size = get_grid_size(); + + // Add extra rows when expanded. + if (expanded) { + const int bsize = (min_size.height * 80 / 100) / 2; + for (int i = 0; i < expansion_rows; ++i) { + min_size.y += 2 * (bsize + 1) + 3; + } + } + + return min_size; } virtual String get_tooltip(const Point2 &p_pos) const override { @@ -712,80 +734,192 @@ public: } return String(); } + void _gui_input(const Ref &p_ev) { const Ref mm = p_ev; - if (mm.is_valid()) { - for (int i = 0; i < flag_rects.size(); i++) { - if (flag_rects[i].has_point(mm->get_position())) { - // Used to highlight the hovered flag in the layers grid. - hovered_index = i; - update(); - break; + bool expand_was_hovered = expand_hovered; + expand_hovered = expand_rect.has_point(mm->get_position()); + if (expand_hovered != expand_was_hovered) { + update(); + } + + if (!expand_hovered) { + for (int i = 0; i < flag_rects.size(); i++) { + if (flag_rects[i].has_point(mm->get_position())) { + // Used to highlight the hovered flag in the layers grid. + hovered_index = i; + update(); + return; + } } } + + // Remove highlight when no square is hovered. + if (hovered_index != -1) { + hovered_index = -1; + update(); + } + + return; } const Ref mb = p_ev; + if (mb.is_valid() && mb->get_button_index() == MOUSE_BUTTON_LEFT && mb->is_pressed()) { + if (hovered_index >= 0) { + // Toggle the flag. + // We base our choice on the hovered flag, so that it always matches the hovered flag. + if (value & (1 << hovered_index)) { + value &= ~(1 << hovered_index); + } else { + value |= (1 << hovered_index); + } - if (mb.is_valid() && mb->get_button_index() == MOUSE_BUTTON_LEFT && mb->is_pressed() && hovered_index >= 0) { - // Toggle the flag. - // We base our choice on the hovered flag, so that it always matches the hovered flag. - if (value & (1 << hovered_index)) { - value &= ~(1 << hovered_index); - } else { - value |= (1 << hovered_index); + emit_signal(SNAME("flag_changed"), value); + update(); + } else if (expand_hovered) { + expanded = !expanded; + minimum_size_changed(); + update(); } - - emit_signal(SNAME("flag_changed"), value); - update(); } } void _notification(int p_what) { switch (p_what) { case NOTIFICATION_DRAW: { - Rect2 rect; - rect.size = get_size(); + Size2 grid_size = get_grid_size(); + grid_size.x = get_size().x; + flag_rects.clear(); - const int bsize = (rect.size.height * 80 / 100) / 2; + int prev_expansion_rows = expansion_rows; + expansion_rows = 0; + + const int bsize = (grid_size.height * 80 / 100) / 2; const int h = bsize * 2 + 1; - const int vofs = (rect.size.height - h) / 2; Color color = get_theme_color(SNAME("highlight_color"), SNAME("Editor")); - for (int i = 0; i < 2; i++) { - Point2 ofs(4, vofs); - if (i == 1) { + + Color text_color = get_theme_color(SNAME("font_color"), SNAME("Editor")); + text_color.a *= 0.5; + + Color text_color_on = get_theme_color(SNAME("font_hover_color"), SNAME("Editor")); + text_color_on.a *= 0.7; + + const int vofs = (grid_size.height - h) / 2; + + int layer_index = 0; + int block_index = 0; + + Point2 arrow_pos; + + Point2 block_ofs(4, vofs); + + while (true) { + Point2 ofs = block_ofs; + + for (int i = 0; i < 2; i++) { + for (int j = 0; j < layer_group_size; j++) { + const bool on = value & (1 << layer_index); + Rect2 rect2 = Rect2(ofs, Size2(bsize, bsize)); + + color.a = on ? 0.6 : 0.2; + if (layer_index == hovered_index) { + // Add visual feedback when hovering a flag. + color.a += 0.15; + } + + draw_rect(rect2, color); + flag_rects.push_back(rect2); + + Ref font = get_theme_font(SNAME("font"), SNAME("Label")); + Vector2 offset; + offset.y = rect2.size.y * 0.75; + + draw_string(font, rect2.position + offset, itos(layer_index), HALIGN_CENTER, rect2.size.x, -1, on ? text_color_on : text_color); + + ofs.x += bsize + 1; + + ++layer_index; + } + + ofs.x = block_ofs.x; ofs.y += bsize + 1; } - ofs += rect.position; - for (int j = 0; j < 10; j++) { - Point2 o = ofs + Point2(j * (bsize + 1), 0); - if (j >= 5) { - o.x += 1; + if (layer_index >= layer_count) { + if (!flag_rects.is_empty() && (expansion_rows == 0)) { + const Rect2 &last_rect = flag_rects[flag_rects.size() - 1]; + arrow_pos = last_rect.position + last_rect.size; } - - const int idx = i * 10 + j; - const bool on = value & (1 << idx); - Rect2 rect2 = Rect2(o, Size2(bsize, bsize)); - - color.a = on ? 0.6 : 0.2; - if (idx == hovered_index) { - // Add visual feedback when hovering a flag. - color.a += 0.15; - } - - draw_rect(rect2, color); - flag_rects.push_back(rect2); + break; } + + int block_size_x = layer_group_size * (bsize + 1); + block_ofs.x += block_size_x + 3; + + if (block_ofs.x + block_size_x + 12 > grid_size.width) { + // Keep last valid cell position for the expansion icon. + if (!flag_rects.is_empty() && (expansion_rows == 0)) { + const Rect2 &last_rect = flag_rects[flag_rects.size() - 1]; + arrow_pos = last_rect.position + last_rect.size; + } + ++expansion_rows; + + if (expanded) { + // Expand grid to next line. + block_ofs.x = 4; + block_ofs.y += 2 * (bsize + 1) + 3; + } else { + // Skip remaining blocks. + break; + } + } + + ++block_index; + } + + if ((expansion_rows != prev_expansion_rows) && expanded) { + minimum_size_changed(); + } + + if ((expansion_rows == 0) && (layer_index == layer_count)) { + // Whole grid was drawn, no need for expansion icon. + break; + } + + Ref arrow = get_theme_icon(SNAME("arrow"), SNAME("Tree")); + ERR_FAIL_COND(arrow.is_null()); + + Color arrow_color = get_theme_color(SNAME("highlight_color"), SNAME("Editor")); + arrow_color.a = expand_hovered ? 1.0 : 0.6; + + arrow_pos.x += 2.0; + arrow_pos.y -= arrow->get_height(); + + Rect2 arrow_draw_rect(arrow_pos, arrow->get_size()); + expand_rect = arrow_draw_rect; + if (expanded) { + arrow_draw_rect.size.y *= -1.0; // Flip arrow vertically when expanded. + } + + RID ci = get_canvas_item(); + arrow->draw_rect(ci, arrow_draw_rect, false, arrow_color); + + } break; + + case NOTIFICATION_MOUSE_EXIT: { + if (expand_hovered) { + expand_hovered = false; + update(); + } + if (hovered_index != -1) { + hovered_index = -1; + update(); } } break; - case NOTIFICATION_MOUSE_EXIT: { - hovered_index = -1; - update(); - } break; + default: break; } @@ -800,12 +934,8 @@ public: ClassDB::bind_method(D_METHOD("_gui_input"), &EditorPropertyLayersGrid::_gui_input); ADD_SIGNAL(MethodInfo("flag_changed", PropertyInfo(Variant::INT, "flag"))); } - - EditorPropertyLayersGrid() { - value = 0; - hovered_index = -1; // Nothing is hovered. - } }; + void EditorPropertyLayers::_grid_changed(uint32_t p_grid) { emit_changed(get_edited_property(), p_grid); } @@ -818,30 +948,49 @@ void EditorPropertyLayers::update_property() { void EditorPropertyLayers::setup(LayerType p_layer_type) { String basename; + int layer_group_size = 0; + int layer_count = 0; switch (p_layer_type) { - case LAYER_RENDER_2D: + case LAYER_RENDER_2D: { basename = "layer_names/2d_render"; - break; - case LAYER_PHYSICS_2D: + layer_group_size = 5; + layer_count = 20; + } break; + + case LAYER_PHYSICS_2D: { basename = "layer_names/2d_physics"; - break; - case LAYER_NAVIGATION_2D: + layer_group_size = 4; + layer_count = 32; + } break; + + case LAYER_NAVIGATION_2D: { basename = "layer_names/2d_navigation"; - break; - case LAYER_RENDER_3D: + layer_group_size = 4; + layer_count = 32; + } break; + + case LAYER_RENDER_3D: { basename = "layer_names/3d_render"; - break; - case LAYER_PHYSICS_3D: + layer_group_size = 5; + layer_count = 20; + } break; + + case LAYER_PHYSICS_3D: { basename = "layer_names/3d_physics"; - break; - case LAYER_NAVIGATION_3D: + layer_group_size = 4; + layer_count = 32; + } break; + + case LAYER_NAVIGATION_3D: { basename = "layer_names/3d_navigation"; - break; + layer_group_size = 4; + layer_count = 32; + } break; } Vector names; Vector tooltips; - for (int i = 0; i < 20; i++) { + for (int i = 0; i < layer_count; i++) { String name; if (ProjectSettings::get_singleton()->has_setting(basename + vformat("/layer_%d", i))) { @@ -858,12 +1007,17 @@ void EditorPropertyLayers::setup(LayerType p_layer_type) { grid->names = names; grid->tooltips = tooltips; + grid->layer_group_size = layer_group_size; + grid->layer_count = layer_count; } void EditorPropertyLayers::_button_pressed() { + int layer_count = grid->layer_count; + int layer_group_size = grid->layer_group_size; + layers->clear(); - for (int i = 0; i < 20; i++) { - if (i == 5 || i == 10 || i == 15) { + for (int i = 0; i < layer_count; i++) { + if ((i != 0) && ((i % layer_group_size) == 0)) { layers->add_separator(); } layers->add_check_item(grid->names[i], i); @@ -894,17 +1048,21 @@ void EditorPropertyLayers::_bind_methods() { EditorPropertyLayers::EditorPropertyLayers() { HBoxContainer *hb = memnew(HBoxContainer); + hb->set_clip_contents(true); add_child(hb); grid = memnew(EditorPropertyLayersGrid); grid->connect("flag_changed", callable_mp(this, &EditorPropertyLayers::_grid_changed)); grid->set_h_size_flags(SIZE_EXPAND_FILL); hb->add_child(grid); + button = memnew(Button); button->set_toggle_mode(true); button->set_text("..."); button->connect("pressed", callable_mp(this, &EditorPropertyLayers::_button_pressed)); hb->add_child(button); + set_bottom_editor(hb); + layers = memnew(PopupMenu); add_child(layers); layers->set_hide_on_checkable_item_selection(false); diff --git a/scene/register_scene_types.cpp b/scene/register_scene_types.cpp index 8ede8e9a0b6..4d3a4ea334e 100644 --- a/scene/register_scene_types.cpp +++ b/scene/register_scene_types.cpp @@ -1000,9 +1000,12 @@ void register_scene_types() { for (int i = 0; i < 20; i++) { GLOBAL_DEF_BASIC(vformat("layer_names/2d_render/layer_%d", i), ""); + GLOBAL_DEF_BASIC(vformat("layer_names/3d_render/layer_%d", i), ""); + } + + for (int i = 0; i < 32; i++) { GLOBAL_DEF_BASIC(vformat("layer_names/2d_physics/layer_%d", i), ""); GLOBAL_DEF_BASIC(vformat("layer_names/2d_navigation/layer_%d", i), ""); - GLOBAL_DEF_BASIC(vformat("layer_names/3d_render/layer_%d", i), ""); GLOBAL_DEF_BASIC(vformat("layer_names/3d_physics/layer_%d", i), ""); GLOBAL_DEF_BASIC(vformat("layer_names/3d_navigation/layer_%d", i), ""); }