Луч, идущий с фрагментным шейдером, кажется странным
Я пытаюсь некоторые эксперименты рендеринга 3d сферы с помощью three.js
, Целью эксперимента является
"Можем ли мы отрисовать трехмерную сферу с заданными мировыми координатами сферы?
(x,y,z)
и радиусr
используя raymarching?"
Эксперимент, казалось, работал хорошо, как показано ниже. Левый красный круг three.js
сфера и хорошо визуализируется в перспективе камеры. Правильный отрисовывается с помощью raymarching с равномерными переменными, такими как,
uniforms: {
x: { value: -testSphere.position.x }, // just for comparing left and right
y: { value: testSphere.position.y },
z: { value: testSphere.position.z },
r: { value: 100 }, //fixed but can be variable
u_resolution: {
type: "v2",
value: new THREE.Vector2(SCREEN_WIDTH, SCREEN_HEIGHT)
},
u_pixel_ratio: {
type: "float",
value: window.devicePixelRatio
},
campos: {
type: "v3",
value: camPos // in my case, it was (0, 0, 1000)
}
}
А также, чтобы правильно обработать событие изменения размера окна пользователя, прикреплен обработчик события, подобный этому.
function onWindowResize(event) {
SCREEN_WIDTH = window.innerWidth;
SCREEN_HEIGHT = window.innerHeight - 2 * MARGIN;
camera.aspect = SCREEN_WIDTH / SCREEN_HEIGHT;
camera.updateProjectionMatrix();
renderer.setSize(SCREEN_WIDTH, SCREEN_HEIGHT);
howBackPlane.material.uniforms.u_resolution.value = new THREE.Vector2(
SCREEN_WIDTH,
SCREEN_HEIGHT); // howBackPlane is plane mesh for draw fragment shader of right red sphere.
}
Однако, когда я изменяю размер окна по горизонтали или вертикали, кажется, что правильное окно не отображается так, как я ожидал.
Изменить размер по горизонтали (уменьшить) Изменить размер по вертикали (увеличить)
Что еще хуже, каждый раз, когда я обновляю свою страницу в любом размере, она возвращается к нормальной (работает правильно). Я даже не знаю, почему тот с three.js
работает отлично и почему не мое. Как я могу сделать сферу как three.js
один?
Вот мой вершинный шейдер и фрагмент кода шейдера howBackPlane
,
//vertex shader
void main(){
gl_Position = projectionMatrix * modelViewMatrix * vec4(position, 1.0);
}
//fragment shader
uniform vec2 u_resolution;
uniform float u_pixel_ratio;
uniform vec3 campos;
uniform float x;
uniform float y;
uniform float z;
uniform float r;
mat3 calcLookAtMatrix( in vec3 ro, in vec3 ta)
{
vec3 ww = normalize( ta - ro );
vec3 uu = normalize( cross(ww,vec3(0.0,1.0,0.0) ) );
vec3 vv = normalize( cross(uu,ww));
return mat3( uu, vv, ww );
}
float sdSphere( vec3 p, float s )
{
return length(p - vec3(x, y, z))-s;
}
float calcIntersection( in vec3 ro, in vec3 rd )
{
const float maxd = 20000.0;
const float precis = 0.01;
float h = precis*2.0;
float t = 0.0;
float res = -1.0;
for( int i=0; i<150; i++ )
{
if( h<precis||t>maxd ) break;
h = sdSphere( ro+rd*t , r);
t += h;
}
if( t<maxd ) res = t;
return res;
}
void main(){
vec3 camtar = vec3(0.0, 0.0, 0.0);
mat3 camMat = calcLookAtMatrix( campos, camtar );
vec2 p = -1.0 + 2.0 * (gl_FragCoord.xy/u_pixel_ratio) / (u_resolution.xy);
p.x *= u_resolution.x/u_resolution.y;
vec3 camdir = normalize(camMat * vec3(p, 1.0/tan(radians(45.0/2.0)))); // fov of camera is 45 deg in this app.
float dist = calcIntersection(campos, camdir);
if (dist != -1.0) {
vec3 col = vec3(0.0);
vec3 pos = campos + dist * camdir;
col += vec3(pos);
gl_FragColor = vec4(vec3(1.0, 0.0, 0.0), 1.0);
}else{
gl_FragColor = vec4(0.0);
}
}
Вы можете увидеть ресурс лучевого марширования, на который я ссылался здесь.