How to generalize coordinate array into pixel-wide lines - c++

What I mean by the title is that I have an std::vector of a struct called Coord:
struct Coord
{
int x;
int y;
};
It contains the pixel coordinates that are a certain color on an image. Here they're colored blue (don't mind the red lines and text that is colored)
I would want to remove pixels from that array so that we're left with only pixel-wide lines in the middle of the previous clusters of pixels. On this picture it would look something like this (my horrible sketch):
Here is my function for finding the pixels:
std::vector<Coord> findPathIn(const std::vector< std::vector<Color> >& image, const Color& pathColor, double threshold)
{
int maxError = ceil(threshold * 255.0);
std::vector<Coord> path;
for (int y = 0; y < height; y++)
{
for (int x = 0; x < width; x++)
{
auto& current = image[x][y];
if (abs(pathColor - current) <= maxError)
path.push_back({ x, y });
}
}
return path;
}
And the Color struct:
struct Color
{
private:
friend int operator - (const Color& a, const Color& b);
public:
int red;
int green;
int blue;
};
int operator - (const Color& a, const Color& b)
{
int sum = 0;
sum += a.red - b.red;
sum += a.green - b.green;
sum += a.blue - b.blue;
int avg = int(sum / 3);
return avg;
}
And to find the path in this image i used findPathIn(image, { 0, 0, 0 }, 0.48) where image is a 2d vector of Color. I'm using the STB library to read and write images.
Thanks in advance.

Related

How could I desaturate the image from color to black and white?

I have been working with my code for changing the color of an image but I did not know what to do next in the nested for loop?
Only what I have so far is averaging the Red, Green, and Blue pixel values at each pixel location in the image.
can anyone help/suggest to me what should I do next?
#include "desaturate.h"
void Desaturate(graphics::Image &image) {
// Returns the width and height of the image.
int GetWidth();
int GetHeight();
// Create variable
int red;
int green;
int blue;
// Gets the red, green or blue pixel value at an (x, y) pixel location.
int GetRed(int x, int y);
int GetGreen(int x, int y);
int GetBlue(int x, int y);
// Sets the red, green or blue pixel value at an (x, y) pixel location.
bool SetRed(int x, int y, int red);
bool SetGreen(int x, int y, int green);
bool SetBlue(int x, int y, int blue);
// Gets the width of the image.
int width = image.GetWidth();
// Gets the height of the image.
int height = image.GetHeight();
for (int i = 0; i < width; i++) {
for (int j = 0; i < height; j++) {
int average = (red + green + blue) / 3;
// What should I do next for this part?
}
}

My barycentric triangle rasterizer draws every third pixel

It also draws multiply triangles, they are all displaced and in wrong scale.
I am trying to make my own implementation of triangle rasterizer found at:
https://www.scratchapixel.com/lessons/3d-basic-rendering/rasterization-practical-implementation/rasterization-stage
I have no idea what is wrong with my code.
#include<fstream>
#include<cmath>
class Vertice
{
public:
float x, y;
Vertice(float x, float y)
{
this->x = x;
this->y = y;
}
void fitToImage(int imageWidth, int imageHeight)
{
x = (x * (imageWidth / 2)) + (imageWidth / 2);
y = (-y * (imageHeight / 2)) + (imageHeight / 2);
}
};
class Image
{
public:
int imageWidth, imageHeight;
unsigned char* pixels;
Image(int imageWidth, int imageHeight)
{
this->imageWidth = imageWidth;
this->imageHeight = imageHeight;
pixels = new unsigned char[imageWidth * imageHeight * 3];
}
~Image()
{
delete[] pixels;
}
void setPixel(int x, int y, int red, int green, int blue)
{
int help_var = ((y * imageHeight) + x) * 3;
pixels[help_var + 0] = (char)red;
pixels[help_var + 1] = (char)green;
pixels[help_var + 2] = (char)blue;
}
void fillPixels(int red, int green, int blue)
{
int help_var = imageWidth * imageHeight * 3;
for (int i = 0; i < help_var; i += 3) {
pixels[i + 0] = (char)red;
pixels[i + 1] = (char)green;
pixels[i + 2] = (char)blue;
}
}
//-------------------BARYCENTRIC TRIANGLE RASTERISATION------------------------
float edgeFunction(const Vertice& A, const Vertice& B, const Vertice& P)
{
return ((P.x - A.x)*(B.y - A.y) + (P.y - A.y)*(B.x - A.x));
}
void fillTriangleBarycentric(const Vertice& v0, const Vertice& v1, const Vertice& v2)
{
Vertice p(0.0f, 0.0f);
for (int x = 0; x < imageWidth; x++) {
for (int y = 0; y < imageHeight; y++) {
p.x = x + 0.5f; p.y = y + 0.5f;
float w0 = edgeFunction(v1, v2, p);
float w1 = edgeFunction(v2, v0, p);
float w2 = edgeFunction(v0, v1, p);
if (w0 >= 0 && w1 >= 0 && w2 >= 0) {
setPixel(x, y, 0, 0, 255);
}
}
}
}
//-----------------------------------------------------------------------------
};
int main()
{
Image image(800, 600);
image.fillPixels(255, 255, 255);
Vertice a(0.2f, 0.5f);
Vertice b(-0.5f, 0.0f);
Vertice c(0.5f, -0.5f);
a.fitToImage(image.imageWidth, image.imageHeight);
b.fitToImage(image.imageWidth, image.imageHeight);
c.fitToImage(image.imageWidth, image.imageHeight);
image.fillTriangleBarycentric(a, b, c);
std::ofstream imageFile;
imageFile.open("./drawing_triangle_test_image.ppm");
imageFile << "P6\n" << image.imageWidth << " " << image.imageHeight << "\n255\n";
imageFile.write((char*)image.pixels, image.imageWidth * image.imageHeight * 3);
imageFile.close();
return 0;
}
Here is the image I get after running my program.
Thanks for any help!
Here is the better result (where setPixel is using imageWidth instead of imageHeight):
y * imageHeight
Is definitely the type of error your code has (might have multiple instances). You need to multiply the y position by the width. Otherwise, you'll end up interlacing the triangle at random x positions.
The fact that you get four triangles relates to 800/600 simplifying to 4/3. Had you rendered to 797 by 603, you'd probably would have gotten some random mess of horizontal lines.
In addition to #Jeffrey's correction your edge function is also not quite right. It should be
float edgeFunction(const Vertice& A, const Vertice& B, const Vertice& P)
{
return ((P.x - A.x)*(B.y - A.y) - (P.y - A.y)*(B.x - A.x));
}
i.e. there should be a negative sign between the two terms (because it is the cross product of the two position vectors AB and AP).

Performant Threaded C++ Pixel Rendering: Fastest Way?

My goal is simple: I want to create a rendering system in C++ that can draw thousands of bitmaps on screen. I have been trying to use threads to speed up the process but to no avail. In most cases, I have actually slowed down performance by using multiple threads. I am using this project as an educational exercise by not using hardware acceleration. That said, my question is this:
What is the best way to use several threads to accept a massive list of images to be drawn onto the screen and render them at break-neck speeds? I know that I won’t be able to create a system that can rival hardware accelerated graphics, but I believe that my idea is still feasible because the operation is so simple: copying pixels from one memory location to another.
My renderer design uses three core blitting operations: position, rotation, and scale of a bitmap image. I have it set up to only rotate an image when needed, and only scale an image when needed.
I have gone through several designs for this system. All of them too slow to get the job done (300 64x64 bitmaps at barely 60fps).
Here are the designs I have tried:
Immediately drawing a source bitmap on a destination bitmap for every image on screen (moderate speed).
Creating workers that accept a draw instruction and immediately begin working on it while other workers receive their instructions also (slowest).
Workers that receive packages of several instructions at a time (slower).
Saving all drawing instructions up and then parting them up in one swoop to several workers while other tasks (in theory) are being done (slowest).
Here is the bitmap class I am using to blit bitmaps onto each other:
class Bitmap
{
public:
Bitmap(int w, int h)
{
width = w;
height = h;
size = w * h;
pixels = new unsigned int[size];
}
virtual ~Bitmap()
{
if (pixels != 0)
{
delete[] pixels;
pixels = 0;
}
}
void blit(Bitmap *bmp, float x, float y, float rot, float sclx,
float scly)
{
// Position only
if (rot == 0 && sclx == 1 && scly == 1)
{
blitPos(bmp, x, y);
return;
}
// Rotate only
else if (rot != 0 && sclx == 1 && scly == 1)
{
blitRot(bmp, x, y, rot);
return;
}
// Scale only
else if (rot == 0 && (sclx != 1 || scly != 1))
{
blitScl(bmp, x, y, sclx, scly);
return;
}
/////////////////////////////////////////////////////////////////////////////
// If it is not one of those, you have to do all three... :D
/////////////////////////////////////////////////////////////////////////////
// Create a bitmap that is scaled to the new size.
Bitmap tmp((int)(bmp->width * sclx), (int)(bmp->height * scly));
// Find how much each pixel steps:
float step_x = (float)bmp->width / (float)tmp.width;
float step_y = (float)bmp->height / (float)tmp.height;
// Fill the scaled image with pixels!
float inx = 0;
int xOut = 0;
while (xOut < tmp.width)
{
float iny = 0;
int yOut = 0;
while (yOut < tmp.height)
{
unsigned int sample = bmp->pixels[
(int)(std::floor(inx) + std::floor(iny) * bmp->width)
];
tmp.drawPixel(xOut, yOut, sample);
iny += step_y;
yOut++;
}
inx += step_x;
xOut++;
}
blitRot(&tmp, x, y, rot);
}
void drawPixel(int x, int y, unsigned int color)
{
if (x > width || y > height || x < 0 || y < 0)
return;
if (color == 0x00000000)
return;
int index = x + y * width;
if (index >= 0 && index <= size)
pixels[index] = color;
}
unsigned int getPixel(int x, int y)
{
return pixels[x + y * width];
}
void clear(unsigned int color)
{
std::fill(&pixels[0], &pixels[size], color);
}
private:
void blitPos(Bitmap *bmp, float x, float y)
{
// Don't draw if coordinates are already past edges
if (x > width || y > height || y + bmp->height < 0 || x + bmp->width < 0)
return;
int from;
int to;
int destfrom;
int destto;
for (int i = 0; i < bmp->height; i++)
{
from = i * bmp->width;
to = from + bmp->width;
//////// Caps
// Bitmap is being drawn past the right edge
if (x + bmp->width > width)
{
int cap = bmp->width - ((x + bmp->width) - width);
to = from + cap;
}
// Bitmap is being drawn past the left edge
else if (x + bmp->width < bmp->width)
{
int cap = bmp->width + x;
from += (bmp->width - cap);
to = from + cap;
}
//////// Destination Maths
if (x < 0)
{
destfrom = (y + i) * width;
destto = destfrom + (bmp->width + x);
}
else
{
destfrom = x + (y + i) * width;
destto = destfrom + bmp->width;
}
// Bitmap is being drawn past either top or bottom edges
if (y + i > height - 1)
{
continue;
}
if (destfrom > size || destfrom < 0)
{
continue;
}
memcpy(&pixels[destfrom], &bmp->pixels[from], sizeof(unsigned int) * (to - from));
}
}
void blitRot(Bitmap *bmp, float x, float y, float rot)
{
float sine = std::sin(-rot);
float cosine = std::cos(-rot);
int x1 = (int)(-bmp->height * sine);
int y1 = (int)(bmp->height * cosine);
int x2 = (int)(bmp->width * cosine - bmp->height * sine);
int y2 = (int)(bmp->height * cosine + bmp->width * sine);
int x3 = (int)(bmp->width * cosine);
int y3 = (int)(bmp->width * sine);
int minx = (int)std::min(0, std::min(x1, std::min(x2, x3)));
int miny = (int)std::min(0, std::min(y1, std::min(y2, y3)));
int maxx = (int)std::max(0, std::max(x1, std::max(x2, x3)));
int maxy = (int)std::max(0, std::max(y1, std::max(y2, y3)));
int w = maxx - minx;
int h = maxy - miny;
int srcx;
int srcy;
int dest_x;
int dest_y;
unsigned int color;
for (int sy = miny; sy < maxy; sy++)
{
for (int sx = minx; sx < maxx; sx++)
{
srcx = sx * cosine + sy * sine;
srcy = sy * cosine - sx * sine;
dest_x = x + sx;
dest_y = y + sy;
if (dest_x <= width - 1 && dest_y <= height - 1
&& dest_x >= 0 && dest_y >= 0)
{
color = 0;
// Only grab a pixel if it is inside of the src image
if (srcx < bmp->width && srcy < bmp->height && srcx >= 0 &&
srcy >= 0)
color = bmp->getPixel(srcx, srcy);
// Only this pixel if it is not completely transparent:
if (color & 0xFF000000)
// Only if the pixel is somewhere between 0 and the bmp size
if (0 < srcx < bmp->width && 0 < srcy < bmp->height)
drawPixel(x + sx, y + sy, color);
}
}
}
}
void blitScl(Bitmap *bmp, float x, float y, float sclx, float scly)
{
// Create a bitmap that is scaled to the new size.
int finalwidth = (int)(bmp->width * sclx);
int finalheight = (int)(bmp->height * scly);
// Find how much each pixel steps:
float step_x = (float)bmp->width / (float)finalwidth;
float step_y = (float)bmp->height / (float)finalheight;
// Fill the scaled image with pixels!
float inx = 0;
int xOut = 0;
float iny;
int yOut;
while (xOut < finalwidth)
{
iny = 0;
yOut = 0;
while (yOut < finalheight)
{
unsigned int sample = bmp->pixels[
(int)(std::floor(inx) + std::floor(iny) * bmp->width)
];
drawPixel(xOut + x, yOut + y, sample);
iny += step_y;
yOut++;
}
inx += step_x;
xOut++;
}
}
public:
int width;
int height;
int size;
unsigned int *pixels;
};
Here is some code showing the latest method I have tried: saving up all instructions and then giving them to workers once they have all been received:
class Instruction
{
public:
Instruction() {}
Instruction(Bitmap* out, Bitmap* in, float x, float y, float rot,
float sclx, float scly)
: outbuffer(out), inbmp(in), x(x), y(y), rot(rot),
sclx(sclx), scly(scly)
{ }
~Instruction()
{
outbuffer = nullptr;
inbmp = nullptr;
}
public:
Bitmap* outbuffer;
Bitmap* inbmp;
float x, y, rot, sclx, scly;
};
Layer Class:
class Layer
{
public:
bool empty()
{
return instructions.size() > 0;
}
public:
std::vector<Instruction> instructions;
int pixel_count;
};
Worker Thread Class:
class Worker
{
public:
void start()
{
done = false;
work_thread = std::thread(&Worker::processData, this);
}
void processData()
{
while (true)
{
controller.lock();
if (done)
{
controller.unlock();
break;
}
if (!layers.empty())
{
for (int i = 0; i < layers.size(); i++)
{
for (int j = 0; j < layers[i].instructions.size(); j++)
{
Instruction* inst = &layers[i].instructions[j];
inst->outbuffer->blit(inst->inbmp, inst->x, inst->y, inst->rot, inst->sclx, inst->scly);
}
}
layers.clear();
}
controller.unlock();
}
}
void finish()
{
done = true;
}
public:
bool done;
std::thread work_thread;
std::mutex controller;
std::vector<Layer> layers;
};
Finally, the Render Manager Class:
class RenderManager
{
public:
RenderManager()
{
workers.reserve(std::thread::hardware_concurrency());
for (int i = 0; i < 1; i++)
{
workers.emplace_back();
workers.back().start();
}
}
void layer()
{
layers.push_back(current_layer);
current_layer = Layer();
}
void blit(Bitmap* out, Bitmap* in, float x, float y, float rot, float sclx, float scly)
{
current_layer.instructions.emplace_back(out, in, x, y, rot, sclx, scly);
}
void processInstructions()
{
if (layers.empty())
layer();
lockall();
int index = 0;
for (int i = 0; i < layers.size(); i++)
{
// Evenly distribute the layers in a round-robin fashion
Layer l = layers[i];
workers[index].layers.push_back(layers[i]);
index++;
if (index >= workers.size()) index = 0;
}
layers.clear();
unlockall();
}
void lockall()
{
for (int i = 0; i < workers.size(); i++)
{
workers[i].controller.lock();
}
}
void unlockall()
{
for (int i = 0; i < workers.size(); i++)
{
workers[i].controller.unlock();
}
}
void finish()
{
// Wait until every worker is done rendering
lockall();
// At this point, we know they have nothing more to draw
unlockall();
}
void endRendering()
{
for (int i = 0; i < workers.size(); i++)
{
// Send each one an exit code
workers[i].finish();
}
// Let the workers finish and then return
for (int i = 0; i < workers.size(); i++)
{
workers[i].work_thread.join();
}
}
private:
std::vector<Worker> workers;
std::vector<Layer> layers;
Layer current_layer;
};
Here is a screenshot of what the 3rd method I tried, and it's results:
Sending packages of draw instructions
What would really be helpful is that if someone could simply point me in the right direction in regards to what method I should try. I have tried these four methods and have failed, so I stand before those who have done greater things than I for help. The least intelligent person in the room is the one that does not ask questions because his pride does not permit it. Please keep in mind though, this is my first question ever on Stack Overflow.

Sorting bounded rectangles in opencv

I have a set of bounded rectangles as Rect in a vector.
vector(Rect) boundRect( contours.size() );
I want to sort these rectangles like in the image below
image http://img42.com/liVFt
I have already tried using the method below, but I am not getting the order like in the image I have posted.
stable_sort( boundRect.begin(), boundRect.end(), compareX_rect );
stable_sort( boundRect.begin(), boundRect.end(), compareY_rect );
bool compareX_rect(const Rect & a, const Rect &b) {
return a.x >= b.x;
}
bool compareY_rect(const Rect & a, const Rect &b) {
return a.y >= b.y;
}
Can someone please help me with this? Thanks in advance.
combine into a single sort where the sort will compare y value first, then next on x value:
EDIT: Fixed the sort Tested on coding ground:
bool compareFn(Rectangle* l, Rectangle* r) {
if(l->y == r->y) return l->x < r->x;
return (l->y < r->y);
}
And to reduce noise (depending how much noise is involved) you can do a floor or round function, or calculate a 'cell' that the y value is a part of. Just increase the cell size until it overcomes the noise:
float cellSize = 20.0f;
bool compareFn(Rectangle* l, Rectangle* r) {
float lCell = floorf(l->y / cellSize);
float rCell = floorf(r->y / cellSize);
if(lCell == rCell) return l->x < r->x;
return (lCell < rCell);
}
And here's the program testing it (without noise reduction):
#include <iostream>
#include <vector>
#include <algorithm> // std::sort
using namespace std;
struct Rectangle {
float x;
float y;
float width;
float height;
Rectangle(float x_, float y_, float w_, float h_)
: x(x_)
, y(y_)
, width(w_)
, height(h_)
{}
};
bool compareFn(Rectangle* l, Rectangle* r) {
if(l->y == r->y) return l->x < r->x;
return (l->y < r->y);
}
int main()
{
vector<Rectangle*> rectangles;
for(int x=0; x<10; ++x) {
for(int y=0; y<10; ++y) {
Rectangle* rect = new Rectangle((9 - x) * 50, (9-y) * 50, 50, 50);
rectangles.push_back(rect);
}
}
printf("SORTING\n");
sort(rectangles.begin(), rectangles.end(), compareFn);
printf("RESULTS\n");
for(vector<Rectangle*>::iterator it=rectangles.begin(), end=rectangles.end(); it!=end; ++it) {
Rectangle* rect = *it;
printf("[%f, %f, %f, %f]\n", rect->x, rect->y, rect->width, rect->height);
}
return 0;
}

Bug in image blitting algorihtm - 'italic' output

I've got a function which blits some region of source image to destination image. And there is a problem: I've got a bug in it but I can't find it. Probably it's very trivial, but I spent many hours on it :). My algorithm 'tilts' objects on image. In debugging i saw that it copies a little more pixels than it should (e.g 36836 instead 36481)
struct Point
{
unsigned x, y;
Point(unsigned x, unsigned y) : x(x), y(y) { }
Point() : x(0), y(0) { }
};
struct Rect
{
Point lt, rd; // left-top and right-down vertexs
};
struct Img
{
vector<unsigned char> px; // pixel data in linear form (RBGARBGARBGA...)
unsigned w, h; // width and heigth of image
};
inline bool isInRect(const Point& p, const Rect& r)
{
return (p.x >= r.lt.x && p.y >= r.lt.y && p.x <= r.rd.x && p.y <= r.rd.y);
}
unsigned blit(const Img& src, Img& dest, const Rect& reg) // <--- THIS FUNCTION
{
dest.w = reg.rd.x - reg.lt.x;
dest.h = reg.rd.y - reg.lt.y;
dest.px.clear();
unsigned n = 0;
for(int i = (reg.lt.y * src.w + reg.lt.x) * 4; i < src.px.size(); i += 4)
{
unsigned y = (i / 4) / src.w;
unsigned x = (i / 4) % src.w;
if(isInRect(Point(x, y), reg))
{
dest.px.push_back(src.px[i]);
dest.px.push_back(src.px[i + 1]);
dest.px.push_back(src.px[i + 2]);
dest.px.push_back(src.px[i + 3]);
n += 4;
}
if(y > reg.rd.y)
break;
}
return n / 4;
}
Example:
Image fragment to blit: http://www.mediafire.com/view/1rb8dyc4z8xq5rl/arial-toblit.PNG
Algorithm output: http://www.mediafire.com/view/qe9j38gq5tp299v/arial-A.png