Tilemap Editor: Line drawing and erasing
This commit is contained in:
parent
e2061e0025
commit
fd06683cb4
2 changed files with 133 additions and 9 deletions
|
@ -387,6 +387,52 @@ void TileMapEditor::_update_copydata() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline Vector<Point2i> line(int x0, int x1, int y0, int y1) {
|
||||||
|
|
||||||
|
Vector<Point2i> points;
|
||||||
|
|
||||||
|
float dx = ABS(x1 - x0);
|
||||||
|
float dy = ABS(y1 - y0);
|
||||||
|
|
||||||
|
int x = x0;
|
||||||
|
int y = y0;
|
||||||
|
|
||||||
|
int sx = x0 > x1 ? -1 : 1;
|
||||||
|
int sy = y0 > y1 ? -1 : 1;
|
||||||
|
|
||||||
|
if (dx > dy) {
|
||||||
|
float err = dx/2;
|
||||||
|
|
||||||
|
while (x != x1) {
|
||||||
|
points.push_back(Vector2(x, y));
|
||||||
|
|
||||||
|
err -= dy;
|
||||||
|
if (err < 0) {
|
||||||
|
y += sy;
|
||||||
|
err += dx;
|
||||||
|
}
|
||||||
|
x += sx;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
float err = dy/2;
|
||||||
|
|
||||||
|
while (y != y1) {
|
||||||
|
points.push_back(Vector2(x, y));
|
||||||
|
|
||||||
|
err -= dx;
|
||||||
|
if (err < 0) {
|
||||||
|
x += sx;
|
||||||
|
err += dy;
|
||||||
|
}
|
||||||
|
y += sy;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
points.push_back(Vector2(x, y));
|
||||||
|
|
||||||
|
return points;
|
||||||
|
}
|
||||||
|
|
||||||
bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
|
|
||||||
if (!node || !node->get_tileset().is_valid() || !node->is_visible())
|
if (!node || !node->get_tileset().is_valid() || !node->is_visible())
|
||||||
|
@ -409,7 +455,7 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
return false; //drag
|
return false; //drag
|
||||||
|
|
||||||
if (tool==TOOL_NONE) {
|
if (tool==TOOL_NONE) {
|
||||||
if (mb.mod.shift) {
|
if (mb.mod.shift && mb.mod.control) {
|
||||||
|
|
||||||
tool=TOOL_RECTANGLE_PAINT;
|
tool=TOOL_RECTANGLE_PAINT;
|
||||||
selection_active=false;
|
selection_active=false;
|
||||||
|
@ -417,6 +463,14 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (mb.mod.shift) {
|
||||||
|
|
||||||
|
tool=TOOL_LINE_PAINT;
|
||||||
|
selection_active=false;
|
||||||
|
rectangle_begin=node->world_to_map(xform_inv.xform(Point2(mb.x,mb.y)));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (mb.mod.control) {
|
if (mb.mod.control) {
|
||||||
|
|
||||||
tool=TOOL_PICKING;
|
tool=TOOL_PICKING;
|
||||||
|
@ -494,10 +548,24 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
undo_redo->add_do_method(node,"set_cellv",Point2(p),id,node->is_cell_x_flipped(p.x,p.y),node->is_cell_y_flipped(p.x,p.y),node->is_cell_transposed(p.x,p.y));
|
undo_redo->add_do_method(node,"set_cellv",Point2(p),id,node->is_cell_x_flipped(p.x,p.y),node->is_cell_y_flipped(p.x,p.y),node->is_cell_transposed(p.x,p.y));
|
||||||
undo_redo->add_undo_method(node,"set_cellv",Point2(p),E->get().idx,E->get().xf,E->get().yf,E->get().tr);
|
undo_redo->add_undo_method(node,"set_cellv",Point2(p),E->get().idx,E->get().xf,E->get().yf,E->get().tr);
|
||||||
}
|
}
|
||||||
|
|
||||||
undo_redo->commit_action();
|
undo_redo->commit_action();
|
||||||
paint_undo.clear();
|
paint_undo.clear();
|
||||||
}
|
}
|
||||||
|
} else if (tool==TOOL_LINE_PAINT) {
|
||||||
|
|
||||||
|
int id=get_selected_tile();
|
||||||
|
|
||||||
|
if (id!=TileMap::INVALID_CELL) {
|
||||||
|
|
||||||
|
undo_redo->create_action("Line Draw");
|
||||||
|
for(Map<Point2i,CellOp>::Element *E=paint_undo.front();E;E=E->next()) {
|
||||||
|
|
||||||
|
_set_cell(E->key(), id, flip_h, flip_v, transpose, true);
|
||||||
|
}
|
||||||
|
undo_redo->commit_action();
|
||||||
|
|
||||||
|
canvas_item_editor->update();
|
||||||
|
}
|
||||||
} else if (tool==TOOL_RECTANGLE_PAINT) {
|
} else if (tool==TOOL_RECTANGLE_PAINT) {
|
||||||
|
|
||||||
int id=get_selected_tile();
|
int id=get_selected_tile();
|
||||||
|
@ -551,7 +619,7 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
|
|
||||||
if (tool==TOOL_NONE) {
|
if (tool==TOOL_NONE) {
|
||||||
|
|
||||||
if (mb.mod.shift) {
|
if (mb.mod.shift && mb.mod.control) {
|
||||||
|
|
||||||
tool=TOOL_RECTANGLE_ERASE;
|
tool=TOOL_RECTANGLE_ERASE;
|
||||||
|
|
||||||
|
@ -559,6 +627,14 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
rectangle_begin=node->world_to_map(xform_inv.xform(Point2(mb.x,mb.y)));
|
rectangle_begin=node->world_to_map(xform_inv.xform(Point2(mb.x,mb.y)));
|
||||||
paint_undo.clear();
|
paint_undo.clear();
|
||||||
|
|
||||||
|
} else if (mb.mod.shift) {
|
||||||
|
|
||||||
|
tool=TOOL_LINE_ERASE;
|
||||||
|
|
||||||
|
selection_active=false;
|
||||||
|
rectangle_begin=node->world_to_map(xform_inv.xform(Point2(mb.x,mb.y)));
|
||||||
|
paint_undo.clear();
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
tool=TOOL_ERASING;
|
tool=TOOL_ERASING;
|
||||||
|
|
||||||
|
@ -573,7 +649,7 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
if (tool==TOOL_ERASING || tool==TOOL_RECTANGLE_ERASE) {
|
if (tool==TOOL_ERASING || tool==TOOL_RECTANGLE_ERASE || tool==TOOL_LINE_ERASE) {
|
||||||
|
|
||||||
if (paint_undo.size()) {
|
if (paint_undo.size()) {
|
||||||
undo_redo->create_action("Erase TileMap");
|
undo_redo->create_action("Erase TileMap");
|
||||||
|
@ -588,7 +664,7 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
paint_undo.clear();
|
paint_undo.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (tool==TOOL_RECTANGLE_ERASE) {
|
if (tool==TOOL_RECTANGLE_ERASE || tool==TOOL_LINE_ERASE) {
|
||||||
canvas_item_editor->update();
|
canvas_item_editor->update();
|
||||||
}
|
}
|
||||||
tool=TOOL_NONE;
|
tool=TOOL_NONE;
|
||||||
|
@ -630,6 +706,38 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
if (tool==TOOL_LINE_PAINT || tool==TOOL_LINE_ERASE) {
|
||||||
|
|
||||||
|
int id = get_selected_tile();
|
||||||
|
|
||||||
|
bool erasing = (tool==TOOL_LINE_ERASE);
|
||||||
|
|
||||||
|
if (id!=TileMap::INVALID_CELL) {
|
||||||
|
|
||||||
|
if (erasing && paint_undo.size()) {
|
||||||
|
|
||||||
|
for (Map<Point2i, CellOp>::Element *E=paint_undo.front();E;E=E->next()) {
|
||||||
|
|
||||||
|
_set_cell(E->key(), E->get().idx, E->get().xf, E->get().yf, E->get().tr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
paint_undo.clear();
|
||||||
|
|
||||||
|
Vector<Point2i> points = line(rectangle_begin.x, over_tile.x, rectangle_begin.y, over_tile.y);
|
||||||
|
for (int i=0;i<points.size();i++) {
|
||||||
|
|
||||||
|
paint_undo[points[i]]=_get_op_from_cell(points[i]);
|
||||||
|
|
||||||
|
if (erasing)
|
||||||
|
_set_cell(points[i], TileMap::INVALID_CELL);
|
||||||
|
}
|
||||||
|
|
||||||
|
canvas_item_editor->update();
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
if (tool==TOOL_RECTANGLE_PAINT || tool==TOOL_RECTANGLE_ERASE) {
|
if (tool==TOOL_RECTANGLE_PAINT || tool==TOOL_RECTANGLE_ERASE) {
|
||||||
|
|
||||||
_select(rectangle_begin, over_tile);
|
_select(rectangle_begin, over_tile);
|
||||||
|
@ -755,6 +863,7 @@ bool TileMapEditor::forward_input_event(const InputEvent& p_event) {
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void TileMapEditor::_canvas_draw() {
|
void TileMapEditor::_canvas_draw() {
|
||||||
|
|
||||||
if (!node)
|
if (!node)
|
||||||
|
@ -867,7 +976,6 @@ void TileMapEditor::_canvas_draw() {
|
||||||
canvas_item_editor->draw_colored_polygon(points, Color(0.2,0.8,1,0.4));
|
canvas_item_editor->draw_colored_polygon(points, Color(0.2,0.8,1,0.4));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (mouse_over){
|
if (mouse_over){
|
||||||
|
|
||||||
Vector2 endpoints[4]={
|
Vector2 endpoints[4]={
|
||||||
|
@ -898,6 +1006,23 @@ void TileMapEditor::_canvas_draw() {
|
||||||
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tool==TOOL_LINE_PAINT) {
|
||||||
|
|
||||||
|
if (paint_undo.empty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
int id = get_selected_tile();
|
||||||
|
|
||||||
|
if (id==TileMap::INVALID_CELL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (Map<Point2i, CellOp>::Element *E=paint_undo.front();E;E=E->next()) {
|
||||||
|
|
||||||
|
_draw_cell(id, E->key(), flip_h, flip_v, transpose, xform);
|
||||||
|
}
|
||||||
|
|
||||||
} else if (tool==TOOL_RECTANGLE_PAINT) {
|
} else if (tool==TOOL_RECTANGLE_PAINT) {
|
||||||
|
|
||||||
int id = get_selected_tile();
|
int id = get_selected_tile();
|
||||||
|
@ -954,11 +1079,8 @@ void TileMapEditor::_canvas_draw() {
|
||||||
_draw_cell(st, over_tile, flip_h, flip_v, transpose, xform);
|
_draw_cell(st, over_tile, flip_h, flip_v, transpose, xform);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void TileMapEditor::edit(Node *p_tile_map) {
|
void TileMapEditor::edit(Node *p_tile_map) {
|
||||||
|
|
||||||
search_box->set_text("");
|
search_box->set_text("");
|
||||||
|
|
|
@ -52,6 +52,8 @@ class TileMapEditor : public VBoxContainer {
|
||||||
TOOL_ERASING,
|
TOOL_ERASING,
|
||||||
TOOL_RECTANGLE_PAINT,
|
TOOL_RECTANGLE_PAINT,
|
||||||
TOOL_RECTANGLE_ERASE,
|
TOOL_RECTANGLE_ERASE,
|
||||||
|
TOOL_LINE_PAINT,
|
||||||
|
TOOL_LINE_ERASE,
|
||||||
TOOL_SELECTING,
|
TOOL_SELECTING,
|
||||||
TOOL_DUPLICATING,
|
TOOL_DUPLICATING,
|
||||||
TOOL_PICKING
|
TOOL_PICKING
|
||||||
|
|
Loading…
Reference in a new issue