Autotile Enhancement
This commit is contained in:
parent
7f1265b41f
commit
e850360948
2 changed files with 246 additions and 63 deletions
|
@ -343,12 +343,13 @@ AutotileEditor::AutotileEditor(EditorNode *p_editor) {
|
||||||
split->add_child(property_editor);
|
split->add_child(property_editor);
|
||||||
|
|
||||||
helper = memnew(AutotileEditorHelper(this));
|
helper = memnew(AutotileEditorHelper(this));
|
||||||
property_editor->call_deferred("edit", helper);
|
property_editor->edit(helper);
|
||||||
|
|
||||||
// Editor
|
// Editor
|
||||||
|
|
||||||
dragging_point = -1;
|
dragging_point = -1;
|
||||||
creating_shape = false;
|
creating_shape = false;
|
||||||
|
snap_step = Vector2(32, 32);
|
||||||
|
|
||||||
set_custom_minimum_size(Size2(0, 150));
|
set_custom_minimum_size(Size2(0, 150));
|
||||||
|
|
||||||
|
@ -426,10 +427,78 @@ AutotileEditor::AutotileEditor(EditorNode *p_editor) {
|
||||||
tools[SHAPE_KEEP_INSIDE_TILE]->set_toggle_mode(true);
|
tools[SHAPE_KEEP_INSIDE_TILE]->set_toggle_mode(true);
|
||||||
tools[SHAPE_KEEP_INSIDE_TILE]->set_pressed(true);
|
tools[SHAPE_KEEP_INSIDE_TILE]->set_pressed(true);
|
||||||
tool_containers[TOOLBAR_SHAPE]->add_child(tools[SHAPE_KEEP_INSIDE_TILE]);
|
tool_containers[TOOLBAR_SHAPE]->add_child(tools[SHAPE_KEEP_INSIDE_TILE]);
|
||||||
tools[SHAPE_SNAP_TO_BITMASK_GRID] = memnew(ToolButton);
|
tools[SHAPE_GRID_SNAP] = memnew(ToolButton);
|
||||||
tools[SHAPE_SNAP_TO_BITMASK_GRID]->set_toggle_mode(true);
|
tools[SHAPE_GRID_SNAP]->set_toggle_mode(true);
|
||||||
tools[SHAPE_SNAP_TO_BITMASK_GRID]->set_pressed(true);
|
tools[SHAPE_GRID_SNAP]->connect("toggled", this, "_on_grid_snap_toggled");
|
||||||
tool_containers[TOOLBAR_SHAPE]->add_child(tools[SHAPE_SNAP_TO_BITMASK_GRID]);
|
tool_containers[TOOLBAR_SHAPE]->add_child(tools[SHAPE_GRID_SNAP]);
|
||||||
|
|
||||||
|
hb_grid = memnew(HBoxContainer);
|
||||||
|
tool_containers[TOOLBAR_SHAPE]->add_child(hb_grid);
|
||||||
|
|
||||||
|
hb_grid->add_child(memnew(VSeparator));
|
||||||
|
hb_grid->add_child(memnew(Label(TTR("Offset:"))));
|
||||||
|
|
||||||
|
sb_off_x = memnew(SpinBox);
|
||||||
|
sb_off_x->set_min(-256);
|
||||||
|
sb_off_x->set_max(256);
|
||||||
|
sb_off_x->set_step(1);
|
||||||
|
sb_off_x->set_value(snap_offset.x);
|
||||||
|
sb_off_x->set_suffix("px");
|
||||||
|
sb_off_x->connect("value_changed", this, "_set_snap_off_x");
|
||||||
|
hb_grid->add_child(sb_off_x);
|
||||||
|
|
||||||
|
sb_off_y = memnew(SpinBox);
|
||||||
|
sb_off_y->set_min(-256);
|
||||||
|
sb_off_y->set_max(256);
|
||||||
|
sb_off_y->set_step(1);
|
||||||
|
sb_off_y->set_value(snap_offset.y);
|
||||||
|
sb_off_y->set_suffix("px");
|
||||||
|
sb_off_y->connect("value_changed", this, "_set_snap_off_y");
|
||||||
|
hb_grid->add_child(sb_off_y);
|
||||||
|
|
||||||
|
hb_grid->add_child(memnew(VSeparator));
|
||||||
|
hb_grid->add_child(memnew(Label(TTR("Step:"))));
|
||||||
|
|
||||||
|
sb_step_x = memnew(SpinBox);
|
||||||
|
sb_step_x->set_min(-256);
|
||||||
|
sb_step_x->set_max(256);
|
||||||
|
sb_step_x->set_step(1);
|
||||||
|
sb_step_x->set_value(snap_step.x);
|
||||||
|
sb_step_x->set_suffix("px");
|
||||||
|
sb_step_x->connect("value_changed", this, "_set_snap_step_x");
|
||||||
|
hb_grid->add_child(sb_step_x);
|
||||||
|
|
||||||
|
sb_step_y = memnew(SpinBox);
|
||||||
|
sb_step_y->set_min(-256);
|
||||||
|
sb_step_y->set_max(256);
|
||||||
|
sb_step_y->set_step(1);
|
||||||
|
sb_step_y->set_value(snap_step.y);
|
||||||
|
sb_step_y->set_suffix("px");
|
||||||
|
sb_step_y->connect("value_changed", this, "_set_snap_step_y");
|
||||||
|
hb_grid->add_child(sb_step_y);
|
||||||
|
|
||||||
|
hb_grid->add_child(memnew(VSeparator));
|
||||||
|
hb_grid->add_child(memnew(Label(TTR("Separation:"))));
|
||||||
|
|
||||||
|
sb_sep_x = memnew(SpinBox);
|
||||||
|
sb_sep_x->set_min(0);
|
||||||
|
sb_sep_x->set_max(256);
|
||||||
|
sb_sep_x->set_step(1);
|
||||||
|
sb_sep_x->set_value(snap_separation.x);
|
||||||
|
sb_sep_x->set_suffix("px");
|
||||||
|
sb_sep_x->connect("value_changed", this, "_set_snap_sep_x");
|
||||||
|
hb_grid->add_child(sb_sep_x);
|
||||||
|
|
||||||
|
sb_sep_y = memnew(SpinBox);
|
||||||
|
sb_sep_y->set_min(0);
|
||||||
|
sb_sep_y->set_max(256);
|
||||||
|
sb_sep_y->set_step(1);
|
||||||
|
sb_sep_y->set_value(snap_separation.y);
|
||||||
|
sb_sep_y->set_suffix("px");
|
||||||
|
sb_sep_y->connect("value_changed", this, "_set_snap_sep_y");
|
||||||
|
hb_grid->add_child(sb_sep_y);
|
||||||
|
|
||||||
|
hb_grid->hide();
|
||||||
|
|
||||||
spin_priority = memnew(SpinBox);
|
spin_priority = memnew(SpinBox);
|
||||||
spin_priority->set_min(1);
|
spin_priority->set_min(1);
|
||||||
|
@ -489,6 +558,13 @@ void AutotileEditor::_bind_methods() {
|
||||||
ClassDB::bind_method("_on_workspace_input", &AutotileEditor::_on_workspace_input);
|
ClassDB::bind_method("_on_workspace_input", &AutotileEditor::_on_workspace_input);
|
||||||
ClassDB::bind_method("_on_tool_clicked", &AutotileEditor::_on_tool_clicked);
|
ClassDB::bind_method("_on_tool_clicked", &AutotileEditor::_on_tool_clicked);
|
||||||
ClassDB::bind_method("_on_priority_changed", &AutotileEditor::_on_priority_changed);
|
ClassDB::bind_method("_on_priority_changed", &AutotileEditor::_on_priority_changed);
|
||||||
|
ClassDB::bind_method("_on_grid_snap_toggled", &AutotileEditor::_on_grid_snap_toggled);
|
||||||
|
ClassDB::bind_method("_set_snap_step_x", &AutotileEditor::_set_snap_step_x);
|
||||||
|
ClassDB::bind_method("_set_snap_step_y", &AutotileEditor::_set_snap_step_y);
|
||||||
|
ClassDB::bind_method("_set_snap_off_x", &AutotileEditor::_set_snap_off_x);
|
||||||
|
ClassDB::bind_method("_set_snap_off_y", &AutotileEditor::_set_snap_off_y);
|
||||||
|
ClassDB::bind_method("_set_snap_sep_x", &AutotileEditor::_set_snap_sep_x);
|
||||||
|
ClassDB::bind_method("_set_snap_sep_y", &AutotileEditor::_set_snap_sep_y);
|
||||||
}
|
}
|
||||||
|
|
||||||
void AutotileEditor::_notification(int p_what) {
|
void AutotileEditor::_notification(int p_what) {
|
||||||
|
@ -501,7 +577,7 @@ void AutotileEditor::_notification(int p_what) {
|
||||||
tools[SHAPE_NEW_POLYGON]->set_icon(get_icon("CollisionPolygon2D", "EditorIcons"));
|
tools[SHAPE_NEW_POLYGON]->set_icon(get_icon("CollisionPolygon2D", "EditorIcons"));
|
||||||
tools[SHAPE_DELETE]->set_icon(get_icon("Remove", "EditorIcons"));
|
tools[SHAPE_DELETE]->set_icon(get_icon("Remove", "EditorIcons"));
|
||||||
tools[SHAPE_KEEP_INSIDE_TILE]->set_icon(get_icon("Snap", "EditorIcons"));
|
tools[SHAPE_KEEP_INSIDE_TILE]->set_icon(get_icon("Snap", "EditorIcons"));
|
||||||
tools[SHAPE_SNAP_TO_BITMASK_GRID]->set_icon(get_icon("SnapGrid", "EditorIcons"));
|
tools[SHAPE_GRID_SNAP]->set_icon(get_icon("SnapGrid", "EditorIcons"));
|
||||||
tools[ZOOM_OUT]->set_icon(get_icon("ZoomLess", "EditorIcons"));
|
tools[ZOOM_OUT]->set_icon(get_icon("ZoomLess", "EditorIcons"));
|
||||||
tools[ZOOM_1]->set_icon(get_icon("ZoomReset", "EditorIcons"));
|
tools[ZOOM_1]->set_icon(get_icon("ZoomReset", "EditorIcons"));
|
||||||
tools[ZOOM_IN]->set_icon(get_icon("ZoomMore", "EditorIcons"));
|
tools[ZOOM_IN]->set_icon(get_icon("ZoomMore", "EditorIcons"));
|
||||||
|
@ -632,6 +708,7 @@ void AutotileEditor::_on_workspace_draw() {
|
||||||
Vector2 coord = edited_shape_coord;
|
Vector2 coord = edited_shape_coord;
|
||||||
draw_highlight_tile(coord);
|
draw_highlight_tile(coord);
|
||||||
draw_polygon_shapes();
|
draw_polygon_shapes();
|
||||||
|
draw_grid_snap();
|
||||||
} break;
|
} break;
|
||||||
case EDITMODE_PRIORITY: {
|
case EDITMODE_PRIORITY: {
|
||||||
spin_priority->set_value(tile_set->autotile_get_subtile_priority(get_current_tile(), edited_shape_coord));
|
spin_priority->set_value(tile_set->autotile_get_subtile_priority(get_current_tile(), edited_shape_coord));
|
||||||
|
@ -880,15 +957,15 @@ void AutotileEditor::_on_workspace_input(const Ref<InputEvent> &p_ie) {
|
||||||
Vector<TileSet::ShapeData> sd = tile_set->tile_get_shapes(get_current_tile());
|
Vector<TileSet::ShapeData> sd = tile_set->tile_get_shapes(get_current_tile());
|
||||||
for (int i = 0; i < sd.size(); i++) {
|
for (int i = 0; i < sd.size(); i++) {
|
||||||
if (sd[i].autotile_coord == coord) {
|
if (sd[i].autotile_coord == coord) {
|
||||||
Ref<ConcavePolygonShape2D> shape = sd[i].shape;
|
Ref<ConvexPolygonShape2D> shape = sd[i].shape;
|
||||||
if (shape.is_valid()) {
|
if (shape.is_valid()) {
|
||||||
//FIXME: i need a way to know if the point is countained on the polygon instead of the rect
|
|
||||||
Rect2 bounding_rect;
|
Rect2 bounding_rect;
|
||||||
PoolVector2Array polygon;
|
PoolVector2Array polygon;
|
||||||
bounding_rect.position = shape->get_segments()[0];
|
bounding_rect.position = shape->get_points()[0];
|
||||||
for (int j = 0; j < shape->get_segments().size(); j += 2) {
|
for (int j = 0; j < shape->get_points().size(); j++) {
|
||||||
polygon.push_back(shape->get_segments()[j] + shape_anchor);
|
polygon.push_back(shape->get_points()[j] + shape_anchor);
|
||||||
bounding_rect.expand_to(shape->get_segments()[j] + shape_anchor);
|
bounding_rect.expand_to(shape->get_points()[j] + shape_anchor);
|
||||||
}
|
}
|
||||||
if (bounding_rect.has_point(mb->get_position())) {
|
if (bounding_rect.has_point(mb->get_position())) {
|
||||||
current_shape = polygon;
|
current_shape = polygon;
|
||||||
|
@ -905,17 +982,17 @@ void AutotileEditor::_on_workspace_input(const Ref<InputEvent> &p_ie) {
|
||||||
if (dragging_point >= 0) {
|
if (dragging_point >= 0) {
|
||||||
dragging_point = -1;
|
dragging_point = -1;
|
||||||
|
|
||||||
PoolVector<Vector2> segments;
|
Vector<Vector2> points;
|
||||||
segments.resize(current_shape.size() * 2);
|
|
||||||
PoolVector<Vector2>::Write w = segments.write();
|
|
||||||
|
|
||||||
for (int i = 0; i < current_shape.size(); i++) {
|
for (int i = 0; i < current_shape.size(); i++) {
|
||||||
w[(i << 1) + 0] = current_shape[i] - shape_anchor;
|
Vector2 p = current_shape[i];
|
||||||
w[(i << 1) + 1] = current_shape[(i + 1) % current_shape.size()] - shape_anchor;
|
if (tools[SHAPE_GRID_SNAP]->is_pressed() || tools[SHAPE_KEEP_INSIDE_TILE]->is_pressed()) {
|
||||||
|
p = snap_point(p);
|
||||||
|
}
|
||||||
|
points.push_back(p - shape_anchor);
|
||||||
}
|
}
|
||||||
|
|
||||||
w = PoolVector<Vector2>::Write();
|
edited_collision_shape->set_points(points);
|
||||||
edited_collision_shape->set_segments(segments);
|
|
||||||
|
|
||||||
workspace->update();
|
workspace->update();
|
||||||
}
|
}
|
||||||
|
@ -982,11 +1059,30 @@ void AutotileEditor::_on_workspace_input(const Ref<InputEvent> &p_ie) {
|
||||||
current_shape.push_back(pos);
|
current_shape.push_back(pos);
|
||||||
workspace->update();
|
workspace->update();
|
||||||
} else {
|
} else {
|
||||||
|
int t_id = get_current_tile();
|
||||||
|
if (t_id >= 0) {
|
||||||
|
Vector<TileSet::ShapeData> sd = tile_set->tile_get_shapes(t_id);
|
||||||
|
for (int i = 0; i < sd.size(); i++) {
|
||||||
|
if (sd[i].autotile_coord == edited_shape_coord) {
|
||||||
|
Ref<ConvexPolygonShape2D> shape = sd[i].shape;
|
||||||
|
|
||||||
|
if (!shape.is_null()) {
|
||||||
|
sd.remove(i);
|
||||||
|
tile_set->tile_set_shapes(get_current_tile(), sd);
|
||||||
|
edited_collision_shape = Ref<Shape2D>();
|
||||||
|
current_shape.resize(0);
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
creating_shape = true;
|
creating_shape = true;
|
||||||
current_shape.resize(0);
|
current_shape.resize(0);
|
||||||
current_shape.push_back(snap_point(pos));
|
current_shape.push_back(snap_point(pos));
|
||||||
}
|
}
|
||||||
} else if (mb->is_pressed() && mb->get_button_index() == BUTTON_RIGHT) {
|
} else if (mb->is_pressed() && mb->get_button_index() == BUTTON_RIGHT && current_shape.size() > 2) {
|
||||||
if (creating_shape) {
|
if (creating_shape) {
|
||||||
close_shape(shape_anchor);
|
close_shape(shape_anchor);
|
||||||
}
|
}
|
||||||
|
@ -1034,7 +1130,7 @@ void AutotileEditor::_on_tool_clicked(int p_tool) {
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
sd.remove(index);
|
sd.remove(index);
|
||||||
tile_set->tile_set_shapes(get_current_tile(), sd);
|
tile_set->tile_set_shapes(get_current_tile(), sd);
|
||||||
edited_collision_shape = Ref<ConcavePolygonShape2D>();
|
edited_collision_shape = Ref<Shape2D>();
|
||||||
current_shape.resize(0);
|
current_shape.resize(0);
|
||||||
workspace->update();
|
workspace->update();
|
||||||
}
|
}
|
||||||
|
@ -1081,6 +1177,43 @@ void AutotileEditor::_on_priority_changed(float val) {
|
||||||
workspace->update();
|
workspace->update();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_on_grid_snap_toggled(bool p_val) {
|
||||||
|
if (p_val)
|
||||||
|
hb_grid->show();
|
||||||
|
else
|
||||||
|
hb_grid->hide();
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_set_snap_step_x(float p_val) {
|
||||||
|
snap_step.x = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_set_snap_step_y(float p_val) {
|
||||||
|
snap_step.y = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_set_snap_off_x(float p_val) {
|
||||||
|
snap_offset.x = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_set_snap_off_y(float p_val) {
|
||||||
|
snap_offset.y = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
void AutotileEditor::_set_snap_sep_x(float p_val) {
|
||||||
|
snap_separation.x = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::_set_snap_sep_y(float p_val) {
|
||||||
|
snap_separation.y = p_val;
|
||||||
|
workspace->update();
|
||||||
|
}
|
||||||
|
|
||||||
void AutotileEditor::draw_highlight_tile(Vector2 coord, const Vector<Vector2> &other_highlighted) {
|
void AutotileEditor::draw_highlight_tile(Vector2 coord, const Vector<Vector2> &other_highlighted) {
|
||||||
|
|
||||||
Vector2 size = tile_set->autotile_get_size(get_current_tile());
|
Vector2 size = tile_set->autotile_get_size(get_current_tile());
|
||||||
|
@ -1103,6 +1236,49 @@ void AutotileEditor::draw_highlight_tile(Vector2 coord, const Vector<Vector2> &o
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void AutotileEditor::draw_grid_snap() {
|
||||||
|
if (tools[SHAPE_GRID_SNAP]->is_pressed()) {
|
||||||
|
Color grid_color = Color(0.39, 0, 1, 0.2f);
|
||||||
|
Size2 s = workspace->get_size();
|
||||||
|
|
||||||
|
Vector2 size = tile_set->autotile_get_size(get_current_tile());
|
||||||
|
|
||||||
|
int width_count = (int)(s.width / (snap_step.x + snap_separation.x));
|
||||||
|
int height_count = (int)(s.height / (snap_step.y + snap_separation.y));
|
||||||
|
|
||||||
|
if (snap_step.x != 0) {
|
||||||
|
int last_p = 0;
|
||||||
|
for (int i = 0; i <= width_count; i++) {
|
||||||
|
if (i == 0 && snap_offset.x != 0) {
|
||||||
|
last_p = snap_offset.x;
|
||||||
|
}
|
||||||
|
if (snap_separation.x != 0 && i != 0) {
|
||||||
|
workspace->draw_rect(Rect2(last_p, 0, snap_separation.x, s.height), grid_color);
|
||||||
|
last_p += snap_separation.x;
|
||||||
|
} else
|
||||||
|
workspace->draw_line(Point2(last_p, 0), Point2(last_p, s.height), grid_color);
|
||||||
|
|
||||||
|
last_p += snap_step.x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (snap_step.y != 0) {
|
||||||
|
int last_p = 0;
|
||||||
|
for (int i = 0; i <= height_count; i++) {
|
||||||
|
if (i == 0 && snap_offset.y != 0) {
|
||||||
|
last_p = snap_offset.y;
|
||||||
|
}
|
||||||
|
if (snap_separation.x != 0 && i != 0) {
|
||||||
|
workspace->draw_rect(Rect2(0, last_p, s.width, snap_separation.y), grid_color);
|
||||||
|
last_p += snap_separation.y;
|
||||||
|
} else
|
||||||
|
workspace->draw_line(Point2(0, last_p), Point2(s.width, last_p), grid_color);
|
||||||
|
last_p += snap_step.y;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void AutotileEditor::draw_polygon_shapes() {
|
void AutotileEditor::draw_polygon_shapes() {
|
||||||
|
|
||||||
int t_id = get_current_tile();
|
int t_id = get_current_tile();
|
||||||
|
@ -1119,7 +1295,7 @@ void AutotileEditor::draw_polygon_shapes() {
|
||||||
anchor.y += tile_set->autotile_get_spacing(t_id);
|
anchor.y += tile_set->autotile_get_spacing(t_id);
|
||||||
anchor.x *= coord.x;
|
anchor.x *= coord.x;
|
||||||
anchor.y *= coord.y;
|
anchor.y *= coord.y;
|
||||||
Ref<ConcavePolygonShape2D> shape = sd[i].shape;
|
Ref<ConvexPolygonShape2D> shape = sd[i].shape;
|
||||||
if (shape.is_valid()) {
|
if (shape.is_valid()) {
|
||||||
Color c_bg;
|
Color c_bg;
|
||||||
Color c_border;
|
Color c_border;
|
||||||
|
@ -1138,19 +1314,22 @@ void AutotileEditor::draw_polygon_shapes() {
|
||||||
colors.push_back(c_bg);
|
colors.push_back(c_bg);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
for (int j = 0; j < shape->get_segments().size(); j += 2) {
|
for (int j = 0; j < shape->get_points().size(); j++) {
|
||||||
polygon.push_back(shape->get_segments()[j] + anchor);
|
polygon.push_back(shape->get_points()[j] + anchor);
|
||||||
colors.push_back(c_bg);
|
colors.push_back(c_bg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
workspace->draw_polygon(polygon, colors);
|
if (polygon.size() > 2) {
|
||||||
|
workspace->draw_polygon(polygon, colors);
|
||||||
|
}
|
||||||
if (coord == edited_shape_coord) {
|
if (coord == edited_shape_coord) {
|
||||||
for (int j = 0; j < shape->get_segments().size(); j += 2) {
|
for (int j = 0; j < shape->get_points().size() - 1; j++) {
|
||||||
workspace->draw_line(shape->get_segments()[j] + anchor, shape->get_segments()[j + 1] + anchor, c_border, 1, true);
|
workspace->draw_line(shape->get_points()[j] + anchor, shape->get_points()[j + 1] + anchor, c_border, 1, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (shape == edited_collision_shape) {
|
if (shape == edited_collision_shape) {
|
||||||
for (int j = 0; j < current_shape.size(); j++) {
|
for (int j = 0; j < current_shape.size(); j++) {
|
||||||
workspace->draw_circle(current_shape[j], 5, Color(1, 0, 0));
|
workspace->draw_circle(current_shape[j], 8 / workspace->get_scale().x, Color(1, 0, 0, 0.7f));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1198,7 +1377,7 @@ void AutotileEditor::draw_polygon_shapes() {
|
||||||
workspace->draw_line(shape->get_polygon()[shape->get_polygon().size() - 1] + anchor, shape->get_polygon()[0] + anchor, c_border, 1, true);
|
workspace->draw_line(shape->get_polygon()[shape->get_polygon().size() - 1] + anchor, shape->get_polygon()[0] + anchor, c_border, 1, true);
|
||||||
if (shape == edited_occlusion_shape) {
|
if (shape == edited_occlusion_shape) {
|
||||||
for (int j = 0; j < current_shape.size(); j++) {
|
for (int j = 0; j < current_shape.size(); j++) {
|
||||||
workspace->draw_circle(current_shape[j], 5, Color(1, 0, 0));
|
workspace->draw_circle(current_shape[j], 8 / workspace->get_scale().x, Color(1, 0, 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1248,7 +1427,7 @@ void AutotileEditor::draw_polygon_shapes() {
|
||||||
}
|
}
|
||||||
if (shape == edited_navigation_shape) {
|
if (shape == edited_navigation_shape) {
|
||||||
for (int j = 0; j < current_shape.size(); j++) {
|
for (int j = 0; j < current_shape.size(); j++) {
|
||||||
workspace->draw_circle(current_shape[j], 5, Color(1, 0, 0));
|
workspace->draw_circle(current_shape[j], 8 / workspace->get_scale().x, Color(1, 0, 0));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1270,22 +1449,21 @@ void AutotileEditor::close_shape(const Vector2 &shape_anchor) {
|
||||||
creating_shape = false;
|
creating_shape = false;
|
||||||
|
|
||||||
if (edit_mode == EDITMODE_COLLISION) {
|
if (edit_mode == EDITMODE_COLLISION) {
|
||||||
Ref<ConcavePolygonShape2D> shape = memnew(ConcavePolygonShape2D);
|
if (current_shape.size() >= 3) {
|
||||||
|
Ref<ConvexPolygonShape2D> shape = memnew(ConvexPolygonShape2D);
|
||||||
|
|
||||||
PoolVector<Vector2> segments;
|
Vector<Vector2> segments;
|
||||||
segments.resize(current_shape.size() * 2);
|
|
||||||
PoolVector<Vector2>::Write w = segments.write();
|
|
||||||
|
|
||||||
for (int i = 0; i < current_shape.size(); i++) {
|
for (int i = 0; i < current_shape.size(); i++) {
|
||||||
w[(i << 1) + 0] = current_shape[i] - shape_anchor;
|
segments.push_back(current_shape[i] - shape_anchor);
|
||||||
w[(i << 1) + 1] = current_shape[(i + 1) % current_shape.size()] - shape_anchor;
|
}
|
||||||
|
|
||||||
|
shape->set_points(segments);
|
||||||
|
|
||||||
|
tile_set->tile_add_shape(get_current_tile(), shape, Transform2D(), false, edited_shape_coord);
|
||||||
|
edited_collision_shape = shape;
|
||||||
}
|
}
|
||||||
|
|
||||||
w = PoolVector<Vector2>::Write();
|
|
||||||
shape->set_segments(segments);
|
|
||||||
|
|
||||||
tile_set->tile_add_shape(get_current_tile(), shape, Transform2D(), false, edited_shape_coord);
|
|
||||||
edited_collision_shape = shape;
|
|
||||||
tools[TOOL_SELECT]->set_pressed(true);
|
tools[TOOL_SELECT]->set_pressed(true);
|
||||||
workspace->update();
|
workspace->update();
|
||||||
} else if (edit_mode == EDITMODE_OCCLUSION) {
|
} else if (edit_mode == EDITMODE_OCCLUSION) {
|
||||||
|
@ -1338,6 +1516,10 @@ Vector2 AutotileEditor::snap_point(const Vector2 &point) {
|
||||||
anchor.x *= (tile_size.x + spacing);
|
anchor.x *= (tile_size.x + spacing);
|
||||||
anchor.y *= (tile_size.y + spacing);
|
anchor.y *= (tile_size.y + spacing);
|
||||||
Rect2 region(anchor, tile_size);
|
Rect2 region(anchor, tile_size);
|
||||||
|
if (tools[SHAPE_GRID_SNAP]->is_pressed()) {
|
||||||
|
p.x = Math::snap_scalar_seperation(snap_offset.x, snap_step.x, p.x, snap_separation.x);
|
||||||
|
p.y = Math::snap_scalar_seperation(snap_offset.y, snap_step.y, p.y, snap_separation.y);
|
||||||
|
}
|
||||||
if (tools[SHAPE_KEEP_INSIDE_TILE]->is_pressed()) {
|
if (tools[SHAPE_KEEP_INSIDE_TILE]->is_pressed()) {
|
||||||
if (p.x < region.position.x)
|
if (p.x < region.position.x)
|
||||||
p.x = region.position.x;
|
p.x = region.position.x;
|
||||||
|
@ -1348,23 +1530,6 @@ Vector2 AutotileEditor::snap_point(const Vector2 &point) {
|
||||||
if (p.y > region.position.y + region.size.y)
|
if (p.y > region.position.y + region.size.y)
|
||||||
p.y = region.position.y + region.size.y;
|
p.y = region.position.y + region.size.y;
|
||||||
}
|
}
|
||||||
if (tools[SHAPE_SNAP_TO_BITMASK_GRID]->is_pressed()) {
|
|
||||||
Vector2 p2 = p;
|
|
||||||
if (tile_set->autotile_get_bitmask_mode(get_current_tile()) == TileSet::BITMASK_2X2) {
|
|
||||||
p2.x = Math::stepify(p2.x, tile_size.x / 2);
|
|
||||||
p2.y = Math::stepify(p2.y, tile_size.y / 2);
|
|
||||||
if ((p2 - p).length_squared() <= MAX(tile_size.y / 4, MIN_DISTANCE_SQUARED)) {
|
|
||||||
p = p2;
|
|
||||||
}
|
|
||||||
} else if (tile_set->autotile_get_bitmask_mode(get_current_tile()) == TileSet::BITMASK_3X3) {
|
|
||||||
p2.x = Math::stepify(p2.x, tile_size.x / 3);
|
|
||||||
p2.y = Math::stepify(p2.y, tile_size.y / 3);
|
|
||||||
if ((p2 - p).length_squared() <= MAX(tile_size.y / 6, MIN_DISTANCE_SQUARED)) {
|
|
||||||
p = p2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
p.floor();
|
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
#include "editor/editor_name_dialog.h"
|
#include "editor/editor_name_dialog.h"
|
||||||
#include "editor/editor_node.h"
|
#include "editor/editor_node.h"
|
||||||
#include "scene/2d/sprite.h"
|
#include "scene/2d/sprite.h"
|
||||||
#include "scene/resources/concave_polygon_shape_2d.h"
|
#include "scene/resources/convex_polygon_shape_2d.h"
|
||||||
#include "scene/resources/tile_set.h"
|
#include "scene/resources/tile_set.h"
|
||||||
|
|
||||||
class AutotileEditorHelper;
|
class AutotileEditorHelper;
|
||||||
|
@ -70,7 +70,7 @@ class AutotileEditor : public Control {
|
||||||
SHAPE_CREATE_FROM_BITMASK,
|
SHAPE_CREATE_FROM_BITMASK,
|
||||||
SHAPE_CREATE_FROM_NOT_BITMASK,
|
SHAPE_CREATE_FROM_NOT_BITMASK,
|
||||||
SHAPE_KEEP_INSIDE_TILE,
|
SHAPE_KEEP_INSIDE_TILE,
|
||||||
SHAPE_SNAP_TO_BITMASK_GRID,
|
SHAPE_GRID_SNAP,
|
||||||
ZOOM_OUT,
|
ZOOM_OUT,
|
||||||
ZOOM_1,
|
ZOOM_1,
|
||||||
ZOOM_IN,
|
ZOOM_IN,
|
||||||
|
@ -78,7 +78,7 @@ class AutotileEditor : public Control {
|
||||||
};
|
};
|
||||||
|
|
||||||
Ref<TileSet> tile_set;
|
Ref<TileSet> tile_set;
|
||||||
Ref<ConcavePolygonShape2D> edited_collision_shape;
|
Ref<ConvexPolygonShape2D> edited_collision_shape;
|
||||||
Ref<OccluderPolygon2D> edited_occlusion_shape;
|
Ref<OccluderPolygon2D> edited_occlusion_shape;
|
||||||
Ref<NavigationPolygon> edited_navigation_shape;
|
Ref<NavigationPolygon> edited_navigation_shape;
|
||||||
|
|
||||||
|
@ -91,10 +91,21 @@ class AutotileEditor : public Control {
|
||||||
Button *tool_editmode[EDITMODE_MAX];
|
Button *tool_editmode[EDITMODE_MAX];
|
||||||
HBoxContainer *tool_containers[TOOLBAR_MAX];
|
HBoxContainer *tool_containers[TOOLBAR_MAX];
|
||||||
HBoxContainer *toolbar;
|
HBoxContainer *toolbar;
|
||||||
|
HBoxContainer *hb_grid;
|
||||||
ToolButton *tools[TOOL_MAX];
|
ToolButton *tools[TOOL_MAX];
|
||||||
SpinBox *spin_priority;
|
SpinBox *spin_priority;
|
||||||
|
SpinBox *sb_step_y;
|
||||||
|
SpinBox *sb_step_x;
|
||||||
|
SpinBox *sb_off_y;
|
||||||
|
SpinBox *sb_off_x;
|
||||||
|
SpinBox *sb_sep_y;
|
||||||
|
SpinBox *sb_sep_x;
|
||||||
EditMode edit_mode;
|
EditMode edit_mode;
|
||||||
|
|
||||||
|
Vector2 snap_step;
|
||||||
|
Vector2 snap_offset;
|
||||||
|
Vector2 snap_separation;
|
||||||
|
|
||||||
bool creating_shape;
|
bool creating_shape;
|
||||||
int dragging_point;
|
int dragging_point;
|
||||||
Vector2 edited_shape_coord;
|
Vector2 edited_shape_coord;
|
||||||
|
@ -119,9 +130,16 @@ private:
|
||||||
void _on_workspace_input(const Ref<InputEvent> &p_ie);
|
void _on_workspace_input(const Ref<InputEvent> &p_ie);
|
||||||
void _on_tool_clicked(int p_tool);
|
void _on_tool_clicked(int p_tool);
|
||||||
void _on_priority_changed(float val);
|
void _on_priority_changed(float val);
|
||||||
|
void _on_grid_snap_toggled(bool p_val);
|
||||||
|
void _set_snap_step_x(float p_val);
|
||||||
|
void _set_snap_step_y(float p_val);
|
||||||
|
void _set_snap_off_x(float p_val);
|
||||||
|
void _set_snap_off_y(float p_val);
|
||||||
|
void _set_snap_sep_x(float p_val);
|
||||||
|
void _set_snap_sep_y(float p_val);
|
||||||
|
|
||||||
void draw_highlight_tile(Vector2 coord, const Vector<Vector2> &other_highlighted = Vector<Vector2>());
|
void draw_highlight_tile(Vector2 coord, const Vector<Vector2> &other_highlighted = Vector<Vector2>());
|
||||||
void draw_grid(const Vector2 &size, int spacing);
|
void draw_grid_snap();
|
||||||
void draw_polygon_shapes();
|
void draw_polygon_shapes();
|
||||||
void close_shape(const Vector2 &shape_anchor);
|
void close_shape(const Vector2 &shape_anchor);
|
||||||
Vector2 snap_point(const Vector2 &point);
|
Vector2 snap_point(const Vector2 &point);
|
||||||
|
|
Loading…
Reference in a new issue