Compare commits
1 commit
852a099930
...
3dc170b6e1
Author | SHA1 | Date | |
---|---|---|---|
3dc170b6e1 |
2 changed files with 222 additions and 51 deletions
232
src/state.cpp
232
src/state.cpp
|
@ -15,20 +15,19 @@
|
||||||
# include <cstdio>
|
# include <cstdio>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Tri::State::State(int argc, char **argv) :
|
Tri::State::State(int /*argc*/, char ** /*argv*/) :
|
||||||
width(800),
|
width(800),
|
||||||
height(600),
|
height(600),
|
||||||
dt(sf::microseconds(16666)),
|
dt(sf::microseconds(16666)),
|
||||||
notification_alpha(1.0f),
|
notification_alpha(1.0f),
|
||||||
window(sf::VideoMode(800, 600), "Triangles", sf::Style::Titlebar | sf::Style::Close),
|
window(sf::VideoMode(800, 600), "Triangles", sf::Style::Titlebar | sf::Style::Close),
|
||||||
trisIndex(0),
|
|
||||||
currentTri_state(CurrentState::NONE),
|
currentTri_state(CurrentState::NONE),
|
||||||
currentTri_maxState(CurrentState::NONE),
|
|
||||||
colorPickerColor{1.0f, 1.0f, 1.0f, 1.0f},
|
colorPickerColor{1.0f, 1.0f, 1.0f, 1.0f},
|
||||||
bgColorPickerColor{0.0f, 0.0f, 0.0f},
|
bgColorPickerColor{0.0f, 0.0f, 0.0f},
|
||||||
bgColor(sf::Color::Black),
|
bgColor(sf::Color::Black),
|
||||||
inputWidthHeight{800, 600},
|
inputWidthHeight{800, 600},
|
||||||
pi(std::acos(-1.0f))
|
pi(std::acos(-1.0f)),
|
||||||
|
history_idx(0)
|
||||||
{
|
{
|
||||||
flags.set(F_IS_RUNNING); // is running
|
flags.set(F_IS_RUNNING); // is running
|
||||||
ImGui::SFML::Init(window);
|
ImGui::SFML::Init(window);
|
||||||
|
@ -56,6 +55,50 @@ pi(std::acos(-1.0f))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Tri::State::Action::Action() :
|
||||||
|
type(Tri::State::Action::AT_NONE),
|
||||||
|
idx(0),
|
||||||
|
color(sf::Color::Black),
|
||||||
|
data{{0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f}}
|
||||||
|
{}
|
||||||
|
|
||||||
|
Tri::State::Action::Action(const Type& type, IndexT idx, const sf::Color& color, float *data) :
|
||||||
|
type(type),
|
||||||
|
idx(idx),
|
||||||
|
color(color)
|
||||||
|
{
|
||||||
|
init(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tri::State::Action::Action(Type&& type, IndexT idx, sf::Color&& color, float *data) :
|
||||||
|
type(type),
|
||||||
|
idx(idx),
|
||||||
|
color(color)
|
||||||
|
{
|
||||||
|
init(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Tri::State::Action::init(float *data) {
|
||||||
|
switch(type) {
|
||||||
|
case AT_TRI:
|
||||||
|
case AT_TRI_DEL:
|
||||||
|
for(unsigned int i = 0; i < 6; ++i) {
|
||||||
|
this->data.tri[i] = data[i];
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case AT_POINT:
|
||||||
|
this->data.point[0] = data[0];
|
||||||
|
this->data.point[1] = data[1];
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
type = AT_NONE;
|
||||||
|
idx = 0;
|
||||||
|
color = sf::Color::Black;
|
||||||
|
this->data = {{0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 0.0f}};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Tri::State::~State() {
|
Tri::State::~State() {
|
||||||
window.close();
|
window.close();
|
||||||
ImGui::SFML::Shutdown();
|
ImGui::SFML::Shutdown();
|
||||||
|
@ -74,50 +117,88 @@ void Tri::State::handle_events() {
|
||||||
flags.flip(F_DISPLAY_HELP);
|
flags.flip(F_DISPLAY_HELP);
|
||||||
} else if(event.key.code == sf::Keyboard::U) {
|
} else if(event.key.code == sf::Keyboard::U) {
|
||||||
flags.set(F_DRAW_CACHE_DIRTY);
|
flags.set(F_DRAW_CACHE_DIRTY);
|
||||||
if(currentTri_state > 0) {
|
if(history_idx > 0) {
|
||||||
switch(currentTri_state) {
|
switch(history[history_idx-1].type) {
|
||||||
case FIRST:
|
case Action::AT_TRI:
|
||||||
|
tris.erase(
|
||||||
|
tris.cbegin() + history[history_idx-1].idx);
|
||||||
|
restore_points_on_tri_del(history_idx - 1);
|
||||||
|
break;
|
||||||
|
case Action::AT_TRI_DEL:
|
||||||
|
{
|
||||||
|
tris.emplace(
|
||||||
|
tris.cbegin() + history[history_idx-1].idx,
|
||||||
|
sf::ConvexShape(3));
|
||||||
|
tris[history[history_idx-1].idx].setPoint(0, sf::Vector2f(
|
||||||
|
history[history_idx-1].data.tri[0],
|
||||||
|
history[history_idx-1].data.tri[1]));
|
||||||
|
tris[history[history_idx-1].idx].setPoint(1, sf::Vector2f(
|
||||||
|
history[history_idx-1].data.tri[2],
|
||||||
|
history[history_idx-1].data.tri[3]));
|
||||||
|
tris[history[history_idx-1].idx].setPoint(2, sf::Vector2f(
|
||||||
|
history[history_idx-1].data.tri[4],
|
||||||
|
history[history_idx-1].data.tri[5]));
|
||||||
|
tris[history[history_idx-1].idx].setFillColor(history[history_idx-1].color);
|
||||||
currentTri_state = CurrentState::NONE;
|
currentTri_state = CurrentState::NONE;
|
||||||
break;
|
break;
|
||||||
case SECOND:
|
}
|
||||||
currentTri_state = CurrentState::FIRST;
|
case Action::AT_POINT:
|
||||||
|
assert(history[history_idx-1].idx + 1 == currentTri_state
|
||||||
|
&& "Point in history must match point index");
|
||||||
|
assert(currentTri_state > 0
|
||||||
|
&& "There must be a point to undo a point");
|
||||||
|
currentTri_state = static_cast<CurrentState>(currentTri_state - 1);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
assert(!"Unreachable code");
|
assert(!"Unreachable code");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else if(trisIndex > 0) {
|
--history_idx;
|
||||||
--trisIndex;
|
|
||||||
}
|
}
|
||||||
} else if(event.key.code == sf::Keyboard::R) {
|
} else if(event.key.code == sf::Keyboard::R) {
|
||||||
flags.set(F_DRAW_CACHE_DIRTY);
|
flags.set(F_DRAW_CACHE_DIRTY);
|
||||||
if(currentTri_state != CurrentState::NONE
|
if(history_idx < history.size()) {
|
||||||
&& currentTri_state < currentTri_maxState) {
|
switch(history[history_idx].type) {
|
||||||
switch(currentTri_state) {
|
case Action::AT_TRI:
|
||||||
case NONE:
|
{
|
||||||
currentTri_state = CurrentState::FIRST;
|
tris.emplace(
|
||||||
|
tris.cbegin() + history[history_idx].idx,
|
||||||
|
sf::ConvexShape(3));
|
||||||
|
tris[history[history_idx].idx].setPoint(0, sf::Vector2f(
|
||||||
|
history[history_idx].data.tri[0],
|
||||||
|
history[history_idx].data.tri[1]));
|
||||||
|
tris[history[history_idx].idx].setPoint(1, sf::Vector2f(
|
||||||
|
history[history_idx].data.tri[2],
|
||||||
|
history[history_idx].data.tri[3]));
|
||||||
|
tris[history[history_idx].idx].setPoint(2, sf::Vector2f(
|
||||||
|
history[history_idx].data.tri[4],
|
||||||
|
history[history_idx].data.tri[5]));
|
||||||
|
tris[history[history_idx].idx].setFillColor(history[history_idx].color);
|
||||||
|
currentTri_state = CurrentState::NONE;
|
||||||
break;
|
break;
|
||||||
case FIRST:
|
}
|
||||||
currentTri_state = CurrentState::SECOND;
|
break;
|
||||||
break;
|
case Action::AT_TRI_DEL:
|
||||||
default:
|
tris.erase(
|
||||||
assert(!"Unreachable code");
|
tris.cbegin() + history[history_idx].idx);
|
||||||
break;
|
restore_points_on_tri_del(history_idx);
|
||||||
}
|
break;
|
||||||
} else if(tris.size() > trisIndex) {
|
case Action::AT_POINT:
|
||||||
++trisIndex;
|
assert(history[history_idx].idx == currentTri_state
|
||||||
} else if(currentTri_state < currentTri_maxState) {
|
&& "Point in history must match point index");
|
||||||
switch(currentTri_state) {
|
assert(currentTri_state < CurrentState::SECOND
|
||||||
case NONE:
|
&& "Current point state must be 0 or 1");
|
||||||
currentTri_state = CurrentState::FIRST;
|
currentTri[currentTri_state].x = history[history_idx].data.point[0];
|
||||||
break;
|
currentTri[currentTri_state].y = history[history_idx].data.point[1];
|
||||||
case FIRST:
|
currentTri_state = static_cast<CurrentState>(
|
||||||
currentTri_state = CurrentState::SECOND;
|
currentTri_state + 1);
|
||||||
|
pointCircle.setFillColor(history[history_idx].color);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
assert(!"Unreachable code");
|
assert(!"Unreachable code");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
++history_idx;
|
||||||
}
|
}
|
||||||
} else if(event.key.code == sf::Keyboard::C) {
|
} else if(event.key.code == sf::Keyboard::C) {
|
||||||
if(flags.test(F_DISPLAY_COLOR_P)) {
|
if(flags.test(F_DISPLAY_COLOR_P)) {
|
||||||
|
@ -166,37 +247,64 @@ void Tri::State::handle_events() {
|
||||||
if(can_draw()) {
|
if(can_draw()) {
|
||||||
switch(currentTri_state) {
|
switch(currentTri_state) {
|
||||||
case CurrentState::NONE:
|
case CurrentState::NONE:
|
||||||
|
{
|
||||||
currentTri[0] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
currentTri[0] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
||||||
if(trisIndex < tris.size()) {
|
|
||||||
tris.resize(trisIndex);
|
|
||||||
}
|
|
||||||
currentTri_state = CurrentState::FIRST;
|
currentTri_state = CurrentState::FIRST;
|
||||||
currentTri_maxState = CurrentState::FIRST;
|
|
||||||
|
if(history_idx < history.size()) {
|
||||||
|
history.resize(history_idx);
|
||||||
|
}
|
||||||
|
float points[2] = {currentTri[0].x, currentTri[0].y};
|
||||||
|
history.push_back(Action(Action::AT_POINT,
|
||||||
|
0,
|
||||||
|
pointCircle.getFillColor(),
|
||||||
|
points));
|
||||||
|
++history_idx;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
case CurrentState::FIRST:
|
case CurrentState::FIRST:
|
||||||
|
{
|
||||||
currentTri[1] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
currentTri[1] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
||||||
if(trisIndex < tris.size()) {
|
|
||||||
tris.resize(trisIndex);
|
|
||||||
}
|
|
||||||
currentTri_state = CurrentState::SECOND;
|
currentTri_state = CurrentState::SECOND;
|
||||||
currentTri_maxState = CurrentState::SECOND;
|
|
||||||
break;
|
if(history_idx < history.size()) {
|
||||||
case CurrentState::SECOND:
|
history.resize(history_idx);
|
||||||
currentTri[2] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
|
||||||
if(trisIndex < tris.size()) {
|
|
||||||
tris.resize(trisIndex);
|
|
||||||
}
|
}
|
||||||
++trisIndex;
|
float points[2] = {currentTri[1].x, currentTri[1].y};
|
||||||
|
history.push_back(Action(Action::AT_POINT,
|
||||||
|
1,
|
||||||
|
pointCircle.getFillColor(),
|
||||||
|
points));
|
||||||
|
++history_idx;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case CurrentState::SECOND:
|
||||||
|
{
|
||||||
|
currentTri[2] = sf::Vector2f(event.mouseButton.x, event.mouseButton.y);
|
||||||
tris.emplace_back(sf::ConvexShape(3));
|
tris.emplace_back(sf::ConvexShape(3));
|
||||||
tris.back().setPoint(0, currentTri[0]);
|
tris.back().setPoint(0, currentTri[0]);
|
||||||
tris.back().setPoint(1, currentTri[1]);
|
tris.back().setPoint(1, currentTri[1]);
|
||||||
tris.back().setPoint(2, currentTri[2]);
|
tris.back().setPoint(2, currentTri[2]);
|
||||||
tris.back().setFillColor(pointCircle.getFillColor());
|
tris.back().setFillColor(pointCircle.getFillColor());
|
||||||
currentTri_state = CurrentState::NONE;
|
currentTri_state = CurrentState::NONE;
|
||||||
currentTri_maxState = CurrentState::NONE;
|
|
||||||
flags.set(F_DRAW_CACHE_DIRTY);
|
flags.set(F_DRAW_CACHE_DIRTY);
|
||||||
|
|
||||||
|
if(history_idx < history.size()) {
|
||||||
|
history.resize(history_idx);
|
||||||
|
}
|
||||||
|
float points[6] = {
|
||||||
|
currentTri[0].x, currentTri[0].y,
|
||||||
|
currentTri[1].x, currentTri[1].y,
|
||||||
|
currentTri[2].x, currentTri[2].y,
|
||||||
|
};
|
||||||
|
history.push_back(Action(Action::AT_TRI,
|
||||||
|
tris.size()-1,
|
||||||
|
pointCircle.getFillColor(),
|
||||||
|
points));
|
||||||
|
++history_idx;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
} else if(flags.test(F_COPY_COLOR_MODE)) {
|
} else if(flags.test(F_COPY_COLOR_MODE)) {
|
||||||
auto color = drawCache.getTexture().copyToImage()
|
auto color = drawCache.getTexture().copyToImage()
|
||||||
.getPixel(event.mouseButton.x, event.mouseButton.y);
|
.getPixel(event.mouseButton.x, event.mouseButton.y);
|
||||||
|
@ -210,7 +318,7 @@ void Tri::State::handle_events() {
|
||||||
} else if(flags.test(F_SELECT_TRI_MODE)) {
|
} else if(flags.test(F_SELECT_TRI_MODE)) {
|
||||||
sf::Vector2f mouseXY = window.mapPixelToCoords(
|
sf::Vector2f mouseXY = window.mapPixelToCoords(
|
||||||
{event.mouseButton.x, event.mouseButton.y});
|
{event.mouseButton.x, event.mouseButton.y});
|
||||||
for(unsigned int i = trisIndex; i-- > 0; ) {
|
for(unsigned int i = tris.size(); i-- > 0; ) {
|
||||||
if(is_within_shape(tris.at(i), mouseXY)) {
|
if(is_within_shape(tris.at(i), mouseXY)) {
|
||||||
selectedTri = i;
|
selectedTri = i;
|
||||||
tris[i].setOutlineColor(invert_color(tris[i].getFillColor()));
|
tris[i].setOutlineColor(invert_color(tris[i].getFillColor()));
|
||||||
|
@ -316,7 +424,7 @@ void Tri::State::draw_to_target(sf::RenderTarget *target) {
|
||||||
target->clear(bgColor);
|
target->clear(bgColor);
|
||||||
|
|
||||||
// draw tris
|
// draw tris
|
||||||
for(unsigned int i = 0; i < trisIndex; ++i) {
|
for(unsigned int i = 0; i < tris.size(); ++i) {
|
||||||
target->draw(tris[i]);
|
target->draw(tris[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -508,3 +616,29 @@ void Tri::State::close_selected_tri_mode() {
|
||||||
flags.set(F_DRAW_CACHE_DIRTY);
|
flags.set(F_DRAW_CACHE_DIRTY);
|
||||||
reset_modes();
|
reset_modes();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Tri::State::restore_points_on_tri_del(Action::IndexT end) {
|
||||||
|
assert(end < history.size()
|
||||||
|
&& "Index on history must be in range");
|
||||||
|
currentTri[2].x = history[end].data.tri[4];
|
||||||
|
currentTri[2].y = history[end].data.tri[5];
|
||||||
|
pointCircle.setFillColor(history[end].color);
|
||||||
|
unsigned int currentTriIdx = 1;
|
||||||
|
while(end-- > 0) {
|
||||||
|
if(history[end].type == Action::AT_POINT) {
|
||||||
|
assert(history[end].idx == currentTriIdx
|
||||||
|
&& "Last point must be second point");
|
||||||
|
currentTri[currentTriIdx].x = history[end].data.point[0];
|
||||||
|
currentTri[currentTriIdx].y = history[end].data.point[1];
|
||||||
|
if(currentTriIdx > 0) {
|
||||||
|
--currentTriIdx;
|
||||||
|
} else {
|
||||||
|
currentTri_state = CurrentState::SECOND;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(!"Unreachable code");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
|
@ -35,6 +35,39 @@ namespace Tri {
|
||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
struct Action {
|
||||||
|
public:
|
||||||
|
typedef std::vector<Action>::size_type IndexT;
|
||||||
|
|
||||||
|
enum Type {
|
||||||
|
AT_TRI,
|
||||||
|
AT_TRI_DEL,
|
||||||
|
AT_POINT,
|
||||||
|
AT_NONE,
|
||||||
|
};
|
||||||
|
|
||||||
|
Action();
|
||||||
|
Action(const Type& type,
|
||||||
|
IndexT idx,
|
||||||
|
const sf::Color& color,
|
||||||
|
float *data);
|
||||||
|
Action(Type&& type,
|
||||||
|
IndexT idx,
|
||||||
|
sf::Color&& color,
|
||||||
|
float *data);
|
||||||
|
|
||||||
|
Type type;
|
||||||
|
IndexT idx;
|
||||||
|
sf::Color color;
|
||||||
|
union Data {
|
||||||
|
float tri[6];
|
||||||
|
float point[2];
|
||||||
|
} data;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void init(float *data);
|
||||||
|
};
|
||||||
|
|
||||||
// use enum FlagName
|
// use enum FlagName
|
||||||
typedef std::bitset<64> BitsetType;
|
typedef std::bitset<64> BitsetType;
|
||||||
BitsetType flags;
|
BitsetType flags;
|
||||||
|
@ -47,10 +80,8 @@ namespace Tri {
|
||||||
|
|
||||||
sf::RenderWindow window;
|
sf::RenderWindow window;
|
||||||
std::vector<sf::ConvexShape> tris;
|
std::vector<sf::ConvexShape> tris;
|
||||||
unsigned int trisIndex;
|
|
||||||
sf::Vector2f currentTri[3];
|
sf::Vector2f currentTri[3];
|
||||||
CurrentState currentTri_state;
|
CurrentState currentTri_state;
|
||||||
CurrentState currentTri_maxState;
|
|
||||||
sf::CircleShape pointCircle;
|
sf::CircleShape pointCircle;
|
||||||
|
|
||||||
sf::Event event;
|
sf::Event event;
|
||||||
|
@ -74,6 +105,9 @@ namespace Tri {
|
||||||
float selectedTriColor[4];
|
float selectedTriColor[4];
|
||||||
float selectedTriBlinkTimer;
|
float selectedTriBlinkTimer;
|
||||||
|
|
||||||
|
std::vector<Action> history;
|
||||||
|
Action::IndexT history_idx;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void handle_events();
|
void handle_events();
|
||||||
void update();
|
void update();
|
||||||
|
@ -121,6 +155,9 @@ namespace Tri {
|
||||||
float* get_selected_tri_color();
|
float* get_selected_tri_color();
|
||||||
void close_selected_tri_mode();
|
void close_selected_tri_mode();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void restore_points_on_tri_del(Action::IndexT end);
|
||||||
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue