1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
|
#include "app.hpp"
#include "floormat/main.hpp"
#include "floormat/settings.hpp"
#include "shaders/tile.hpp"
#include "main/clickable.hpp"
#include "src/anim-atlas.hpp"
#include "draw/anim.hpp"
#include "src/camera-offset.hpp"
#include <Magnum/Math/Color.h>
#include <Magnum/Math/Vector3.h>
namespace floormat {
void app::draw_cursor()
{
constexpr float LINE_WIDTH = 2;
auto& shader = M->shader();
shader.set_tint({1, 0, 0, 1});
const auto inactive_color = 0xff00ffff_rgbaf;
if (cursor.tile && !cursor.in_imgui)
{
const auto draw = [&, pos = *cursor.tile](auto& mesh, const auto& size) {
const auto pt = pos.to_signed();
const Vector3 center{Vector3i(pt[0], pt[1], 0) * iTILE_SIZE};
mesh.draw(shader, {center, size, LINE_WIDTH});
};
if (const auto* ed = _editor.current_tile_editor(); ed)
{
if (!ed->is_anything_selected())
shader.set_tint(inactive_color);
if (ed->mode() == editor_mode::walls)
switch (ed->rotation())
{
case editor_wall_rotation::N: draw(_wireframe_wall_n, TILE_SIZE); break;
case editor_wall_rotation::W: draw(_wireframe_wall_w, TILE_SIZE); break;
}
else if (ed->mode() == editor_mode::floor)
draw(_wireframe_quad, TILE_SIZE2);
}
else if (const auto* ed = _editor.current_scenery_editor(); ed)
{
if (!ed->is_anything_selected())
shader.set_tint(inactive_color);
const auto& sel = ed->get_selected().proto;
draw(_wireframe_quad, TILE_SIZE2);
if (ed->is_anything_selected())
{
shader.set_tint({1, 1, 1, 0.75f});
auto [_f, _w, anim_mesh] = M->meshes();
const auto pos = Vector3i(cursor.tile->to_signed(), 0)*iTILE_SIZE;
anim_mesh.draw(shader, *sel.atlas, sel.frame.r, sel.frame.frame, Vector3(pos), 1);
}
}
}
}
void app::draw()
{
if (_editor.current_tile_editor() || _editor.current_scenery_editor())
draw_cursor();
draw_ui();
render_menu();
}
clickable_scenery* app::find_clickable_scenery(const Optional<Vector2i>& pixel_)
{
if (!pixel_ || _editor.mode() != editor_mode::none)
return nullptr;
const auto pixel = Vector2ui(*pixel_);
clickable_scenery* item = nullptr;
float depth = -1;
const auto array = M->clickable_scenery();
for (clickable_scenery& c : array)
if (c.depth > depth && c.dest.contains(pixel))
{
const auto pos_ = pixel - c.dest.min() + c.src.min();
const auto pos = c.atlas.group(c.item.r).mirror_from.isEmpty()
? pos_
: Vector2ui(c.src.sizeX() - 1 - pos_[0], pos_[1]);
const auto stride = c.atlas.info().pixel_size[0];
std::size_t idx = pos.y() * stride + pos.x();
fm_debug_assert(idx < c.bitmask.size());
if (c.bitmask[idx])
{
depth = c.depth;
item = &c;
}
}
if (item && item->item.can_activate(item->atlas))
return item;
else
return nullptr;
}
} // namespace floormat
|