Games101 作业2 Triangles and Z-buffering(源码+常见问题)
void rst::rasterizer::rasterize_triangle(const Triangle& t) {
auto v = t.toVector4();
//Find out the bounding box of current triangle.
int left = static_cast<int>(std::floor(fminf(t.v[0].x(), fminf(t.v[1].x(), t.v[2].x()))));
int right = static_cast<int>(std::ceil(fmaxf(t.v[0].x(), fmaxf(t.v[1].x(), t.v[2].x()))));
int top = static_cast<int>(std::ceil(fmaxf(t.v[0].y(), fmaxf(t.v[1].y(), t.v[2].y()))));
int down = static_cast<int>(std::floor(fminf(t.v[0].y(), fminf(t.v[1].y(), t.v[2].y()))));
bool usemsaa = true;
// iterate through the pixel and find if the current pixel is inside the triangle
if (usemsaa)
{
vector<Vector2f> pos{ //采样点
{0.25,0.25}, //左下
{0.75,0.25}, //右下
{0.25,0.75}, //左上
{0.75,0.75} //右上
};
for (int j = top; j >= down; --j)
{
for (int i = left; i <= right; ++i)
{
float mindepth = std::numeric_limits<float>::infinity();
int count = 0;
for (int samplintpoint = 0; samplintpoint < 4; ++samplintpoint)
{
if (insideTriangle(static_cast<float>(i + pos[samplintpoint][0]), static_cast<float>(j + pos[samplintpoint][1]), t.v)) //以每个像素中心位置为单元
{
auto barycentric = computeBarycentric2D(static_cast<float>(i + pos[samplintpoint][0]), static_cast<float>(j + pos[samplintpoint][1]), t.v);
float w_reciprocal = 1.0 / (get<0>(barycentric) / v[0].w() + get<1>(barycentric) / v[1].w() + get<2>(barycentric) / v[2].w());
float z_interpolated = get<0>(barycentric) * v[0].z() / v[0].w() + get<1>(barycentric) * v[1].z() / v[1].w() + get<2>(barycentric) * v[2].z() / v[2].w();
z_interpolated *= w_reciprocal;
mindepth = fminf(z_interpolated, mindepth);
++count;
}
}
if (depth_buf[get_index(i, j)] > mindepth)
{
depth_buf[get_index(i, j)] = mindepth;
Eigen::Vector3f color = t.getColor() * (count / 4.0);
Eigen::Vector3f point;
point << static_cast<float>(i), static_cast<float>(j), mindepth;
set_pixel(point, color);
}
}
}
}
else
{
for (int j = top; j >= down; --j)
{
for (int i = left; i <= right; ++i)
{
if (insideTriangle(static_cast<float>(i + 0.5), static_cast<float>(j + 0.5), t.v)) //以每个像素中心位置为单元
{
auto barycentric = computeBarycentric2D(static_cast<float>(i + 0.5), static_cast<float>(j + 0.5), t.v);
float w_reciprocal = 1.0 / (get<0>(barycentric) / v[0].w() + get<1>(barycentric) / v[1].w() + get<2>(barycentric) / v[2].w());
float z_interpolated = get<0>(barycentric) * v[0].z() / v[0].w() + get<1>(barycentric) * v[1].z() / v[1].w() + get<2>(barycentric) * v[2].z() / v[2].w();
z_interpolated *= w_reciprocal;
if (depth_buf[get_index(i, j)] > z_interpolated)
{
depth_buf[get_index(i, j)] = z_interpolated;
Eigen::Vector3f color = t.getColor();
Eigen::Vector3f point;
point << static_cast<float>(i), static_cast<float>(j), z_interpolated;
set_pixel(point, color);
}
}
}
}
}
}