-
I've been messing around with interpolation and I've almost reached something that I could call usable and then bam big problem. I first experimented by just interpolating between 2 static points and now I added lines that are draw when the mouse moves. The probem is after a couple of seconds the whole thing crashes and I see that .getPosition() returns numbers with 6 or more digits.
Here's the code:
#include <SFML/Graphics.hpp>
#include <iostream>
int setMaxValue(sf::CircleShape circle, bool& y)
{
int a = 0;
if(circle.getPosition().y > circle.getPosition().x)
{
a = circle.getPosition().y;
y = true;
}
else
a = circle.getPosition().x;
return a;
}
int main()
{
sf::RenderWindow window(sf::VideoMode(1920, 1080), "Test");
sf::Event event;
std::vector<sf::CircleShape> circles;
sf::Vector2f mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window))), oldMouseCoords = mouseCoords;
bool ended = false, isY = false;
int maxValue0, maxValue1, maxValue2;
while(window.isOpen())
{
mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window)));
while(window.pollEvent(event))
{
if(event.type == sf::Event::Closed)
{
window.close();
}
if(event.type == sf::Event::MouseMoved)
{
sf::CircleShape newCircle;
newCircle.setRadius(50);
newCircle.setOrigin(newCircle.getRadius(), newCircle.getRadius());
newCircle.setPosition(mouseCoords);
newCircle.setFillColor(sf::Color::Red);
circles.push_back(newCircle);
ended = false;
}
}
if(ended == false && circles.size() > 1)
{
maxValue0 = setMaxValue(circles[circles.size() - 2], isY);
maxValue1 = setMaxValue(circles[circles.size() - 1], isY);
if(maxValue0 > maxValue1)
maxValue2 = maxValue0 - maxValue1;
else
maxValue2 = maxValue1 - maxValue0;
std::cout<<circles.size()<<std::endl;
std::cout<<circles[circles.size() - 2].getPosition().x<<" "<<circles[circles.size() - 2].getPosition().y<<std::endl;
for(int i = 0; i < maxValue2; ++i)
{
sf::CircleShape newCircle;
int x, y;
if(isY == true)
{
y = circles[circles.size() - 2].getPosition().y + i;
x = (((y - circles[circles.size() - 2].getPosition().y) * (circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x)) / (circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y)) + circles[circles.size() - 2].getPosition().x;
}
else
{
x = circles[circles.size() - 2].getPosition().x + i;
y = (((x - circles[circles.size() - 1].getPosition().x) * (circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y)) / (circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x)) + circles[circles.size() - 2].getPosition().y;
}
sf::Vector2f finalPos(x, y);
newCircle.setRadius(50);
newCircle.setOrigin(newCircle.getRadius(), newCircle.getRadius());
newCircle.setPosition(finalPos);
newCircle.setFillColor(sf::Color::Red);
circles.push_back(newCircle);
if(i == maxValue2 - 1)
{
ended = true;
}
}
}
window.clear(sf::Color::White);
for(auto i : circles)
{
window.draw(i);
}
window.display();
}
}
-
From what I can see the problem seems to arrises from "circles[circles.size() -2].getPosition" no idea why that would cause it tho. -1 seems to have no problem
-
If circles.size() is ever 0 or 1 then subtracting 2 from it is going to cause the value to wrap around (size_t is an unsigned value) and you'll index your vector with a very large number. This means you'll be effectively trying to read some random memory which is most likely why the position has an apparently odd value stored in it.
-
That's not the problem - made the if accept the vector only if it was bigger than 2 and manually logged the circles.size() and circles.size() - 2. Program still crashed with bad_alloc exception. Also the .getPosition() still returns crazy numbers.
-
By "big" numbers, do you you mean large in value or many digits? It's unclear from your original description.
An example of the sorts of "crazy" numbers you are getting might help narrow down the cause.
Are you creating circles for every co-ordinate (usually pixel) between each point?
Are you also adding a circle to the vector every time the mouse moves? That's a pretty large vector. It should also be reserved in advance to avoid it moving around when increasing inside.
-
Yes, I am creating a circle per pixel. Also I probably should reserve it.
This is the kind of crazy numbers I am talking about: https://imgur.com/a/YDWXywq
-
Okay, well, I've tested your code.
Your "control circles" work fine if you remove the "if (ended == false..." section so it's clear that the interpolation is the problem.
Also, do you realise how quickly the circle vector gets to tens of thousands when interpolating?
Quick question, why are interpolating with so many circles? Can you not draw a thick line between each pair of circles?
-
I am not adding thick lines between them as this is supposed to be a thick line. I'm not really satisfied with Selba Ward's Spline so I decided to try to make something myself. Using sf::TriangleStrip would be an option to fill in, but I have no idea how to do that.
-
A thick line is simply a rectangle (or a quad or two triangles). Whether you use Selba Ward's Line class (thick line), sf::RectangleShape or a manual vertex array (quad/2 triangles), you can just draw that between each control circle.
I presume that your intention is a thick, poly-line with curved corners and a semi-circle added to each end of the poly-line?
May I ask what with Selba Ward's Spline you weren't satisfied with? Is it just missing the curved corners on non-Bezier poly-lines?
-
Selba Ward has a great concept with Splines, but I really wished it had curved(rounded) corners, also the bigger problem was that while drawing I could see the triangle realine and move. It almost looked like little artifacts that were sticking out slightly from the line it self.
-
You should be able to control Selba Ward's Spline to do what you need by adding in extra points. Basically, for each corner, add in a couple of control points at the same position with handles to create the curve. Look up Bezier curves for more information.
That said, if you still want to do it yourself, you could make a thick line between each point and the also draw a circle over each of those points. You can use the above mentioned methods for easily creating a thick line.
I'd like to see what sort of artifacts you are describing. Can you screenshot or video capture them. Since this is slightly off-topic for this thread, you can discuss it on this one and I'll see what I can do:
https://en.sfml-dev.org/forums/index.php?topic=19496.0
Also, feel free to raise an issue on GitHub about the problem. Note, however, that curved corners has already been discussed within the issues.
https://github.com/Hapaxia/SelbaWard/issues
-
I am having trouble figuring out how to calculate how many degrees to rotate the rectangle to connect the dots
-
This should help:
https://www.mathsisfun.com/algebra/sohcahtoa.html
-
Yep. Dunno if its the best way, but sin-1 seems to work well.
-
You're probably going to want to use this:
https://en.cppreference.com/w/cpp/numeric/math/atan2
-
So it's been a long time. Not because of how hard I found this to be, but rather because of how long I haven't programmed in. Anyways I've pretty much gotten it working, using asin to calculate the angle instead of atan that you recommended me, but it doesn't matter as all trigonometric functions would lead to the same result. I have a problem tho. When drawing lines in the Y coordinate its very smooth and behaves as it should, but when on the X it gets jittery sometimes individual dots are not connected and overall has too many corners and ridges(see pictures - second one is without the dots being drawn).
Code:
#include <SFML/Graphics.hpp>
#include <cmath>
#include <iostream>
int pythagoreanTheorem(int a, int b)
{
int c = sqrt((a * a) + (b * b));
return c;
}
int main()
{
sf::RenderWindow window(sf::VideoMode(1920, 1080), "Test");
sf::Event event;
std::vector<sf::CircleShape> circles;
std::vector<sf::RectangleShape> lines;
sf::Vector2f mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window))), oldMouseCoords = mouseCoords;
while(window.isOpen())
{
mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window)));
while(window.pollEvent(event))
{
if(event.type == sf::Event::Closed)
{
window.close();
}
if(event.type == sf::Event::MouseMoved)
{
sf::CircleShape circle;
circle.setRadius(20);
circle.setPosition(mouseCoords);
circle.setOrigin(circle.getRadius(), circle.getRadius());
circle.setFillColor(sf::Color::Black);
circles.push_back(circle);
if(circles.size() > 1)
{
sf::RectangleShape line;
line.setSize(sf::Vector2f(circles[circles.size() - 2].getRadius() * 2, pythagoreanTheorem(circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x, circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y)));
line.setOrigin(line.getSize().x / 2, 0);
//calculate Sine
double idk = (circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x) / pythagoreanTheorem(circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x,
circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y);
//Sine to degrees
line.setRotation(-(asin(idk) * 180 / 3.14));
if(circles[circles.size() - 1].getPosition().y < circles[circles.size() - 2].getPosition().y)
{
line.setRotation(180 - line.getRotation());
}
line.setPosition(circles[circles.size() - 2].getPosition().x, circles[circles.size() - 2].getPosition().y);
line.setFillColor(circles[circles.size() - 2].getFillColor());
lines.push_back(line);
}
}
}
window.clear(sf::Color::White);
for(auto i : circles)
{
window.draw(i);
}
for(auto i : lines)
{
window.draw(i);
}
window.display();
}
}
-
I think pythagoreanTheorem should return a float instead of an int.
-
Seems to have fixed the rigidness, but the skipping of dots is still present.
-
Maybe it's because of std::asin, which can only give results in [-pi/2, pi/2], therefore making other angles impossible to reach. std::atan2 is the only function that works correctly for the full range of angles, since it takes both X and Y coordinates. Read their documentation carefully, it's easy to miss something important.
-
How do I use atan2? I am kinda confused how to find the angle at which to rotate.
Nvm. Figured out how to use it. Thanks for the advice! Let me know if there is anything I can improve performance wise.
Code:
#include <SFML/Graphics.hpp>
#include <cmath>
#include <iostream>
float pythagoreanTheorem(int a, int b)
{
float c = sqrt((a * a) + (b * b));
return c;
}
int main()
{
sf::RenderWindow window(sf::VideoMode(1920, 1080), "Drawing test");
sf::Event event;
sf::Color color = sf::Color::Blue;
std::vector<sf::CircleShape> circles;
std::vector<sf::RectangleShape> lines;
sf::Vector2f mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window))), oldMouseCoords = mouseCoords;
while(window.isOpen())
{
mouseCoords = window.mapPixelToCoords(sf::Vector2i(sf::Mouse::getPosition(window)));
while(window.pollEvent(event))
{
if(event.type == sf::Event::Closed)
{
window.close();
}
if(event.type == sf::Event::MouseMoved)
{
sf::CircleShape circle;
circle.setRadius(5);
circle.setPosition(mouseCoords);
circle.setOrigin(circle.getRadius(), circle.getRadius());
circle.setFillColor(color);
circles.push_back(circle);
if(circles.size() > 1)
{
sf::RectangleShape line;
line.setSize(sf::Vector2f(pythagoreanTheorem(circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x, circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y), circles[circles.size() - 2].getRadius() * 2));
line.setOrigin(sf::Vector2f(0, line.getSize().y / 2));
line.setPosition(circles[circles.size() - 2].getPosition().x, circles[circles.size() - 2].getPosition().y);
//calculate arc-tangent
double idk = atan2(circles[circles.size() - 1].getPosition().y - circles[circles.size() - 2].getPosition().y, circles[circles.size() - 1].getPosition().x - circles[circles.size() - 2].getPosition().x);
//convert radians to degrees and set them as the rotation
line.setRotation(idk * 180 / 3.14);
line.setFillColor(circles[circles.size() - 2].getFillColor());
lines.push_back(line);
}
}
if(event.type == sf::Event::KeyPressed && event.key.code == sf::Keyboard::Enter)
{
color = sf::Color::Red;
}
}
window.clear(sf::Color::White);
for(auto i : circles)
{
window.draw(i);
}
for(auto i : lines)
{
window.draw(i);
}
window.display();
}
}
-
Is it working as expected now?
Performance wise, the most obvious change would be to use a single vertex array rather than multiple circles and rectangles. But that requires some work ;)
-
Yes it is working as expected now, bust after intergrating into my painting program my performance concerns were assured. After a paint for a while, I notice that the whole program feels slower(mainly the dropdown menu speed changes). How would I go about making it into a single vertex array?
It is quite obvious that I am novice to all things programming, but I am learning so some advice would be very much appreciated.
-
Rather than asking for the complete solution, I would suggest you have a look at the sf::VertexArray tutorial (https://www.sfml-dev.org/tutorials/2.5/graphics-vertex-array.php) and the API documentation (https://www.sfml-dev.org/documentation/2.5.1/classsf_1_1VertexArray.php).
If things are still unclear after that, feel free to ask more concrete questions :)
-
Oh, I'd forgotten about this thread.
If I'm not mistaken, your intention is to have multiple thick straight lines with circular end points and circular corners?
You may want to give Selba Ward's Spline (https://github.com/Hapaxia/SelbaWard/wiki/Spline) a try. Its most recent updates added these sorts of things :) so it's a single vertex array/draw call!