Перейти до вмісту

Модель відбиття Бліна-Фонга

Матеріал з Вікіпедії — вільної енциклопедії.

Модель відбиття Блінна–Фонга, яка також називається модифікованою моделлю відбиття Фонга, є модифікацією моделі відбиття Фонга, що була розроблена Джимом Блінном. [1]

Затінення Блінна-Фонга — це модель затінення, яка використовується в фіксованому конвеєрі OpenGL і Direct3D (до Direct3D 10 і OpenGL 3.1). Значення освітлення обчислюється на кожній вершині, коли вона проходить по графічному конвеєру; значення пікселів між вершинами інтерполюються за затіненням Гуро за замовчуванням, а не більш дорогим затіненням Фонга.[2]

Вектори для розрахунку затінення Фонга та Блінна–Фонга

У затіненні Фонга потрібно постійно перераховувати скалярний добуток між напрямом погляду спостерігача (V) і променем від джерела світла (L), відбитим (R) від поверхні.

Якщо натомість обчислити половинний вектор між вектором спостерігача та вектором джерела світла,

можна замінити на , де нормована нормаль поверхні. У наведеному вище рівнянні і є нормованими векторами і є розв’язком рівняння де — це матриця Хаусхолдера, яка відображає точку на гіперплощині, яка містить початок координат і має нормаль

Цей скалярний добуток рівний косинусу кута, який становить половину кута, представленого скалярним добутком Фонга, якщо , , і лежать в одній площині. Це співвідношення між кутами залишається приблизно вірним, коли вектори не лежать в одній площині, особливо коли кути малі. Тому кут між і іноді називають половинним кутом.

Враховуючи, що кут між половинним вектором і нормаллю поверхні, ймовірно, буде меншим, ніж кут між і , який використовується в моделі Фонга (якщо поверхня не розглядається під дуже крутим кутом, для якого він, імовірно, буде більшим), і, оскільки, в моделі Фонга використовується експоненту можна встановити таку, що ближче до попереднього виразу.

Для поверхонь, які засвічені в бік глядача (дзеркальне відображення на поверхнях, звернених до глядача), призведе до дзеркальних відблисків, які сильно схожі на відповідні відблиски в відображенні Фонга. Однак, у той час як відблиски Фонга завжди мають круглу форму для плоскої поверхні, відблиски Блінна–Фонга стають еліптичними, коли на поверхню дивляться під крутим кутом. Це схоже на випадки, коли сонце відбивається в морі близько до горизонту, або коли віддалений вуличний ліхтар відбивається у мокрому тротуарі, де відображення завжди буде набагато більш розширеним по вертикалі, ніж по горизонталі.[3]

Visual comparison: Blinn–Phong highlights are larger than Phong with the same exponent, but by lowering the exponent, they can become nearly equivalent.

Крім того, хоча цю модель можна розглядати як наближення до моделі Фонга, вона створює більш точні моделі емпірично визначених двопроменевих функцій відбивної здатності, ніж модель Фонга для багатьох типів поверхонь.[4]

Ефективність

[ред. | ред. код]

Модель Блінна-Фонга буде швидшою за модель Фонга у випадку, коли спостерігач і світло вважаються дуже віддаленими від об'єкту спостереження, наприклад, на відстані, що наближається до нескінченності, або рівній нескінченності. Це працює в випадку спрямованого світла та ортографічної/ізометричної камери. У цьому випадку половинний вектор не залежить від позиції та кривизни поверхні тому, що половинний вектор залежить від напрямку до позиції спостерігача та напрямку до позиції світла, які окремо сходяться на віддаленій відстані, отже, половинний вектор можна вважати константним у цьому випадку. отже, може бути обчислено один раз для кожного джерела світла, а потім використано для всього кадру, або тоді, коли світло та точка огляду залишаються в тому самому відносному положенні. Те саме не стосується методу Фонга з використанням вектора відбиття, який залежить від кривизни поверхні та повинен бути перерахований для кожного пікселя зображення (або для кожної вершини моделі у випадку вершинного освітлення). У 3D-сценах із камерами, що мають перспективу, ця оптимізація неможлива.

Зразки коду

[ред. | ред. код]

Приклад коду мовою HLSL

[ред. | ред. код]

Цей зразок на мові HLSL є методом обчислення розсіяного та дзеркального світла від точкового джерела освітлення. Вхідними даними є структура даних для джерела світла, положення поверхні в просторі, вектор напрямку до спостерігача і нормаль поверхні. Структура даних з освітленням поверхні повертається;

В коді нижче, також, потрібно обмежити значення скалярного добутку до нуля у випадку негативних значень. Без цього, світло, що спрямовується від камери, розглядається так само, як світло, що спрямовується до неї. При розрахунку дзеркального відображення неправильний «ореол» світла, що відбивається від країв об’єкта та від камери, може виглядати таким же яскравим, як і світло, яке безпосередньо відбивається в бік на камеру.

struct Lighting
{
    float3 Diffuse;
    float3 Specular;
};

struct PointLight
{
	float3 position;
	float3 diffuseColor;
	float  diffusePower;
	float3 specularColor;
	float  specularPower;
};

Lighting GetPointLight(PointLight light, float3 pos3D, float3 viewDir, float3 normal)
{
	Lighting OUT;
	if (light.diffusePower > 0)
	{
		float3 lightDir = light.position - pos3D; //3D position in space of the surface
		float distance = length(lightDir);
		lightDir = lightDir / distance; // = normalize(lightDir);
		distance = distance * distance;

		//Інтенсивність розсіяного світла. Функція saturate для того, щоб значення було в рамках 0 та 1.
		float NdotL = dot(normal, lightDir);
		float diffuseIntensity = saturate(NdotL);

		// Розрахунок розсіяного світла враховуючи колір світла, ступінь розсіяного світла та затухання
		OUT.Diffuse = diffuseIntensity * light.diffuseColor * light.diffusePower / distance;

		//Розрахунок половинного вектору між напрямом світла та напрямом до спостерігача
		float3 H = normalize(lightDir + viewDir);

		//Інтенсивність відблисків
		float NdotH = dot(normal, H);
		float specularIntensity = pow(saturate(NdotH), specularHardness);

		//Обчислення відблиску враховуючи всі параметри
		OUT.Specular = specularIntensity * light.specularColor * light.specularPower / distance; 
	}
	return OUT;
}

Приклад коду мовою GLSL

[ред. | ред. код]

Цей зразок на мові GLSL складається з двох файлів коду, або шейдерів . Перший є так званим вершинним шейдером і реалізує модель затінення Фонга, яка використовується для інтерполяції нормалі поверхні між вершинами. Другий шейдер є так званим фрагментним шейдером і реалізує модель затінення Блінна–Фонга для визначення розсіяного та дзеркального світла від точкового джерела світла.

Вершиний шейдер

[ред. | ред. код]

Цей вершинний шейдер реалізує затінення Фонга :

attribute vec3 inputPosition;
attribute vec3 inputNormal;

uniform mat4 projection, modelview, normalMat;

varying vec3 normalInterp;
varying vec3 vertPos;

void main() {
    gl_Position = projection * modelview * vec4(inputPosition, 1.0);
    vec4 vertPos4 = modelview * vec4(inputPosition, 1.0);
    vertPos = vec3(vertPos4) / vertPos4.w;
    normalInterp = vec3(normalMat * vec4(inputNormal, 0.0));
}

Фрагментний шейдер

[ред. | ред. код]

Цей фрагментний шейдер реалізує модель затінення Блінна–Фонга [5] та гамма-корекцію :

precision mediump float;

in vec3 normalInterp;
in vec3 vertPos;

uniform int mode;

const vec3 lightPos = vec3(1.0, 1.0, 1.0);
const vec3 lightColor = vec3(1.0, 1.0, 1.0);
const float lightPower = 40.0;
const vec3 ambientColor = vec3(0.1, 0.0, 0.0);
const vec3 diffuseColor = vec3(0.5, 0.0, 0.0);
const vec3 specColor = vec3(1.0, 1.0, 1.0);
const float shininess = 16.0;
const float screenGamma = 2.2; // Припустімо, що монітор відкалібровано відповідно до колірного простору sRGB

void main() {

  vec3 normal = normalize(normalInterp);
  vec3 lightDir = lightPos - vertPos;
  float distance = dot(lightDir, lightDir);
  lightDir = normalize(lightDir);

  float lambertian = max(dot(lightDir, normal), 0.0);
  float specular = 0.0;

  if (lambertian > 0.0) {

    vec3 viewDir = normalize(-vertPos);

    // це модель Блінна-Фонга
    vec3 halfDir = normalize(lightDir + viewDir);
    float specAngle = max(dot(halfDir, normal), 0.0);
    specular = pow(specAngle, shininess);
       
    // Це модель Фонга (для порівняння)
    if (mode == 2) {
      vec3 reflectDir = reflect(-lightDir, normal);
      specAngle = max(dot(reflectDir, viewDir), 0.0);
      // note that the exponent is different here
      specular = pow(specAngle, shininess/4.0);
    }
  }
  vec3 colorLinear = ambientColor +
                     diffuseColor * lambertian * lightColor * lightPower / distance +
                     specColor * specular * lightColor * lightPower / distance;
  // застосовуємо гамма-корекцію (припускаємо, що ambientColor, diffuseColor та specColor
  // знаходяться в лінійному просторі, тобто не мають гамма-корекції)
  vec3 colorGammaCorrected = pow(colorLinear, vec3(1.0 / screenGamma));
  // повертаємо колір після гамма-корекції
  gl_FragColor = vec4(colorGammaCorrected, 1.0);
}

На кольори ambientColor ,diffuseColor і specColor гамма-корекція не повинна застосовуватися . Якщо це кольори, отримані з файлів зображень з гамма-корекцією (JPEG, PNG тощо), їх потрібно лінеаризувати перед роботою з ними, що робиться шляхом масштабування значень каналів кольору до діапазону [0, 1] і піднесення їх до степеня, рівному значенню гамми зображення, яке для зображень у просторі кольорів sRGB можна вважати рівним приблизно 2.2 (хоча для цього конкретного простору кольорів просте співвідношення степенів є лише наближенням фактичного перетворення). Сучасні графічні API мають можливість автоматично виконувати цю гамма-корекцію під час читання пікселя з текстури або запису в буфер кадру.[6]

Див. також

[ред. | ред. код]

Список літератури

[ред. | ред. код]
  1. James F. Blinn (1977). Models of light reflection for computer synthesized pictures. Proceedings of the 4th annual conference on Computer graphics and interactive techniques. с. 192—198. CiteSeerX 10.1.1.131.7741. doi:10.1145/563858.563893. ISBN 9781450373555. S2CID 8043767.
  2. Shreiner, Dave; The Khronos OpenGL ARB Working Group (2010). The Mathematics of Lighting. OpenGL programming guide : the official guide to learning OpenGL, version 3.0 and 3.1 (вид. 7th). Pearson Education, Inc. с. 240–245. ISBN 978-0-321-55262-4.
  3. Krus, Kristofer (2014), Wave Model and Watercraft Model for Simulation of Sea State, Linköping University, с. 97
  4. Ngan, Addy; Durand, Frédo; Matusik, Wojciech (2004). Experimental validation of analytical BRDF models. ACM SIGGRAPH 2004 Sketches on - SIGGRAPH '04. ACM Press. с. 90. doi:10.1145/1186223.1186336. ISBN 1-59593-836-2. S2CID 9259363. Процитовано 23 квітня 2019.
  5. WebGL Example: Phong / Blinn Phong Shading. www.mathematik.uni-marburg.de. Процитовано 13 вересня 2019.
  6. Framebuffer khronos.org