Games101 作业2 Triangles and Z-buffering(源码+常见问题)

时间:2025-04-01 14:16:11
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); } } } } } }