Ray Tracer, теневой луч производит черный круг? - PullRequest
1 голос
/ 09 марта 2020

enter image description here

Как вы можете видеть на изображении, я получаю черный круг в верхней части сфер, и изображение выглядит зернистым. Он должен быть более резким, но есть маленькие черные и белые пятна.

Это код для теневого луча

int pos = 0;

float intersect(const ray &r, vector<unique_ptr<object>> &obj)
{
    //gives closest object hit point and position;
    float closest = numeric_limits<float>::max();
    for(int j = 0; j < obj.size(); j++)
    {
        float t = obj[j]->intersect(r);
        if(t > 1e-6 && t < closest)
        {
            closest = t;
            pos = j;
        }
    }
    return closest;
}
vec color(const ray& r, vector<unique_ptr<object>> &shape,  vector<unique_ptr<Light>> &lighting, int depth)
{   
    vec background_color( .678, .847, .902);
    vec total{0.0, 0.0, 0.0};
    vec ambient{0.125, 0.125, 0.125};

    float t_near = intersect(r, shape);

    if(t_near == numeric_limits<float>::max())
            return background_color;
    else
    {
        total += ambient;
        for(int i = 0; i < lighting.size(); i++){
        total += shape[pos]->shade(lighting[i]->position(), t_near, r);//gives specular + diffuse
        vec shadow_dir = unit_vector(lighting[i]->position() - r.p_at_par(t_near));
        ray shadowRay(r.p_at_par(t_near), shadow_dir);
        float dist = shadow_dir.lenght();
        float a = intersect(shadowRay, shape);
        if(a != numeric_limits<float>::max())
                return vec(0.0, 0.0, 0.0);
        }
        return total;
    }
}

1 Ответ

0 голосов
/ 09 марта 2020

enter image description here Хорошо, понял.

Для черного круга вы должны проверить, чтобы расстояние от теневого луча было меньше расстояния между точкой и источник света. Кроме того, для расстояния shadow_dir не должен быть нормализован. И чтобы справиться с черными белыми пятнами, которые чувствительны к пересечению теней, вы должны добавить смещение N * к точке попадания, где смещение, например, 1e-4. Смещение не должно быть слишком маленьким

        vec shadow_dir = lighting[i]->position() - r.p_at_par(t_near);
        float dist = shadow_dir.lenght();
        vec N = unit_vector(shape[pos]->normal(r, t_near));
        shadow_dir = unit_vector(shadow_dir);
        ray shadowRay(r.p_at_par(t_near) + N*1e-4, shadow_dir);
        float a = intersect(shadowRay, shape);
        if(a != numeric_limits<float>::max()){
            float m = shadowRay.p_at_par(a).lenght();
            if(a < dist)
                return vec(0.0, 0.0, 0.0);
            }
        }
...