You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
232 lines
6.4 KiB
232 lines
6.4 KiB
#pragma once
|
|
|
|
#include "CELLMath.hpp"
|
|
|
|
namespace CELL
|
|
{
|
|
enum DRAWMODE
|
|
{
|
|
DM_POINTS = 0,
|
|
DM_LINES = 1,
|
|
DM_LINE_LOOP = 2,
|
|
DM_LINE_STRIP = 3,
|
|
};
|
|
|
|
class Span
|
|
{
|
|
public:
|
|
int _xStart;
|
|
int _xEnd;
|
|
Rgba _colorStart;
|
|
Rgba _colorEnd;
|
|
int _y;
|
|
|
|
public:
|
|
Span(int xStart,int xEnd,int y,Rgba colorStart,Rgba colorEnd)
|
|
{
|
|
if (xStart < xEnd)
|
|
{
|
|
_xStart = xStart;
|
|
_xEnd = xEnd;
|
|
_colorStart = colorStart;
|
|
_colorEnd = colorEnd;
|
|
_y = y;
|
|
}
|
|
else
|
|
{
|
|
_xStart = _xEnd;
|
|
_xEnd = _xStart;
|
|
|
|
_colorStart = colorEnd;
|
|
_colorEnd = colorStart;
|
|
_y = y;
|
|
}
|
|
}
|
|
};
|
|
|
|
class Ege
|
|
{
|
|
public:
|
|
int _x1;
|
|
int _y1;
|
|
Rgba _color1;
|
|
|
|
int _x2;
|
|
int _y2;
|
|
Rgba _color2;
|
|
|
|
Ege(int x1,int y1,Rgba color1,int x2,int y2,Rgba color2)
|
|
{
|
|
if (y1 < y2)
|
|
{
|
|
_x1 = x1;
|
|
_y1 = y1;
|
|
_color1 = color1;
|
|
|
|
_x2 = x2;
|
|
_y2 = y2;
|
|
_color2 = color2;
|
|
}
|
|
else
|
|
{
|
|
_x1 = x2;
|
|
_y1 = y2;
|
|
_color1 = color2;
|
|
|
|
_x2 = x1;
|
|
_y2 = y1;
|
|
_color2 = color1;
|
|
}
|
|
}
|
|
};
|
|
class Raster
|
|
{
|
|
public:
|
|
uint* _buffer;
|
|
int _width;
|
|
int _height;
|
|
Rgba _color;
|
|
public:
|
|
Raster(int w,int h,void* buffer);
|
|
~Raster(void);
|
|
|
|
void clear();
|
|
|
|
void drawPoint(int x,int y, Rgba color,int ptSize);
|
|
|
|
|
|
|
|
void drawArrays(DRAWMODE mode,const float2* points,int count);
|
|
|
|
void drawFilleRect(int startX,int startY,int w,int h);
|
|
|
|
void drawRect(const int2* points,const Rgba* colors);
|
|
|
|
public:
|
|
|
|
bool isInRect(int2 pt)
|
|
{
|
|
if (pt.x >= 0 && pt.x <= _width && pt.y >= 0 && pt.y <= _height)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
void drawTriangle(int2 p0,int2 p1,int2 p2,Rgba c0,Rgba c1,Rgba c2)
|
|
{
|
|
// if ((!isInRect(p0)) && (!isInRect(p1)) && (!isInRect(p2)))
|
|
// {
|
|
// return;
|
|
// }
|
|
Ege eges[3] =
|
|
{
|
|
Ege(p0.x,p0.y,c0, p1.x,p1.y,c1),
|
|
Ege(p1.x,p1.y,c1, p2.x,p2.y,c2),
|
|
Ege(p2.x,p2.y,c2, p0.x,p0.y,c0),
|
|
};
|
|
int iMax = 0;
|
|
int length = eges[0]._y2 - eges[0]._y1;
|
|
|
|
for (int i = 1 ;i < 3 ; ++ i)
|
|
{
|
|
int len = eges[i]._y2 - eges[i]._y1;
|
|
if (len > length)
|
|
{
|
|
length = len;
|
|
iMax = i;
|
|
}
|
|
}
|
|
int iShort1 = (iMax + 1)%3;
|
|
int iShort2 = (iMax + 2)%3;
|
|
|
|
drawEge(eges[iMax],eges[iShort1]);
|
|
drawEge(eges[iMax],eges[iShort2]);
|
|
|
|
}
|
|
|
|
void drawEge(const Ege& e1,const Ege& e2)
|
|
{
|
|
float yOffset1 = e1._y2 - e1._y1;
|
|
if (yOffset1 == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
float yOffset = e2._y2 - e2._y1;
|
|
if (yOffset == 0)
|
|
{
|
|
return;
|
|
}
|
|
float xOffset = e2._x2 - e2._x1;
|
|
float scale = 0;
|
|
float step = 1.0f/yOffset;
|
|
int startY = tmax<int>(e2._y1,0);
|
|
int endY = tmin<int>(e2._y2,_height);
|
|
scale += (startY - e2._y1)/yOffset;
|
|
|
|
|
|
float xOffset1 = e1._x2 - e1._x1;
|
|
float scale1 = (float)(e2._y1 - e1._y1)/yOffset1;
|
|
float step1 = 1.0f/yOffset1;
|
|
|
|
int startY1 = tmax<int>(e1._y1,0);
|
|
int endY1 = tmin<int>(e1._y2,_height);
|
|
scale1 += (startY1 - e1._y1)/yOffset1;
|
|
|
|
for (int y = startY ; y < endY ; ++ y)
|
|
{
|
|
int x1 = e1._x1 + (int)(scale1 * xOffset1);
|
|
int x2 = e2._x1 + (int)(scale * xOffset);
|
|
Rgba color2 = colorLerp(e2._color1,e2._color2,scale);
|
|
Rgba color1 = colorLerp(e1._color1,e1._color2,scale1);
|
|
|
|
Span span(x1,x2,y,color1,color2);
|
|
drawSpan(span);
|
|
|
|
scale += step;
|
|
scale1 += step1;
|
|
|
|
}
|
|
}
|
|
|
|
void drawSpan(const Span& span)
|
|
{
|
|
float length = span._xEnd - span._xStart;
|
|
float scale = 0;
|
|
float step = 1.0f/length;
|
|
int startX = tmax<int>(span._xStart,0);
|
|
int endX = tmin<int>(span._xEnd,_width);
|
|
scale += (startX - span._xStart)/length;
|
|
for (int x = startX ; x < endX; ++ x)
|
|
{
|
|
Rgba color = colorLerp(
|
|
span._colorStart
|
|
,span._colorEnd
|
|
,scale
|
|
);
|
|
scale += step;
|
|
|
|
setPixelEx(x,span._y,color);
|
|
}
|
|
}
|
|
void drawLine(float2 pt1,float2 pt2,Rgba color1,Rgba color2);
|
|
|
|
void drawPoints(float2 pt1,Rgba4Byte color)
|
|
{
|
|
|
|
}
|
|
inline void setPixelEx(unsigned x,unsigned y,Rgba color)
|
|
{
|
|
_buffer[y * _width + x] = color._color;
|
|
}
|
|
|
|
inline void setPixel(unsigned x,unsigned y,Rgba color)
|
|
{
|
|
if (x >= _width || y >= _height)
|
|
{
|
|
return;
|
|
}
|
|
_buffer[y * _width + x] = color._color;
|
|
}
|
|
};
|
|
}
|