WebGL에서 이미지 처리는 쉽습니다. 얼마나 쉽냐고요? 아래를 읽어 보면 됩니다.
이 글은 WebGL2 기초에서 이어지는 글입니다. 만약에 아직 읽어 보지 않았다면 먼저 읽어 보는 것이 좋습니다.
WebGL에서 이미지를 그리려면 텍스처를 사용 해야 합니다. 렌더링 할 때 픽셀 좌표 대신 클립 공간 좌표를 예상하는 것과 마찬가지로 일반적으로 텍스처를 읽을 때 텍스처 좌표를 예상합니다. 텍스쳐 좌표는 텍스처의 크기에 상관없이 0.0에서 1.0의 크기를 가집니다.
WebGL2에서는 픽셀 좌표를 사용하여 텍스처를 읽을수 있는 기능이 추가 되었습니다. 어떤 방법이 더 좋은지는 사람마다 다릅니다. 하지만 픽셀 좌표 보단 텍스처 좌표를 사용하는 것이 더 일반적이라고 생각 합니다.
지금은 하나의 사각형(2개의 삼각형)을 그리기 때문에 WebGL에 직사각형의 각 점이 텍스처의 어느 위치에 해당하는지 알려주어야 합니다. 이 정보를 정점 셰이더에서 프래그먼트 셰이더로 'varying’이라 불리는 특별한 변수를 사용하여 전달합니다. varying은 값이 변하기 때문에 varying이라고 불립니다. 프래그먼트 셰이더를 사용하여 각 픽셀을 그릴때 마다 WebGL은 정점 셰이더에서 제공 한 값을 보간합니다.
이전 글의 마지막 부분에서 작성한 정점 셰이더에서부터, 텍스처 좌표로 전달할 attribute를 추가 한 다음 이를 프래그먼트 셰이더로 전달 해야합니다.
...
+in vec2 a_texCoord;
...
+out vec2 v_texCoord;
void main() {
...
+ // 프래그먼트 셰이더로 texCoord 전달
+ // GPU가 이 값들을 점 사이에서 보간 할 것입니다.
+ v_texCoord = a_texCoord;
}
그런 다음 텍스처에서 색상을 찾기 위한 프래그먼트 셰이더를 작성합니다.
#version 300 es
precision highp float;
// 사용할 텍스처
uniform sampler2D u_image;
// texCoord는 정점 셰이더에서 전달된 것입니다.
in vec2 v_texCoord;
// 프래그먼트 셰이더는 출력값을 선언해야합니다.
out vec4 outColor;
void main() {
// 텍스처에서 색상을 찾습니다.
outColor = texture(u_image, v_texCoord);
}
마지막으로 이미지를 로드하고 텍스처를 생성한 뒤, 이미지를 텍스처로 복사해야 합니다. 우리는 브라우저를 사용하고 있기 때문에 이미지가 비동기적으로 로딩되고, 이를 위해 텍스처가 로드 될 때까지를 기다리도록 코드를 약간 변경해야 합니다. 일단 로드가 완료되면 그 이후에 그릴 것입니다.
+function main() {
+ var image = new Image();
+ image.src = "https://someimage/on/our/server"; // 반드시 같은 도메인 이어야 합니다!!!
+ image.onload = function() {
+ render(image);
+ }
+}
function render(image) {
...
// 정점 데이터가 가야할 location을 찾습니다.
var positionAttributeLocation = gl.getAttribLocation(program, "a_position");
+ var texCoordAttributeLocation = gl.getAttribLocation(program, "a_texCoord");
// uniform 찾기
var resolutionLocation = gl.getUniformLocation(program, "u_resolution");
+ var imageLocation = gl.getUniformLocation(program, "u_image");
...
+ // 직사각형의 텍스처 좌표를 제공합니다.
+ var texCoordBuffer = gl.createBuffer();
+ gl.bindBuffer(gl.ARRAY_BUFFER, texCoordBuffer);
+ gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([
+ 0.0, 0.0,
+ 1.0, 0.0,
+ 0.0, 1.0,
+ 0.0, 1.0,
+ 1.0, 0.0,
+ 1.0, 1.0]), gl.STATIC_DRAW);
+ gl.enableVertexAttribArray(texCoordAttributeLocation);
+ var size = 2; // iteration마다 2개의 구성요소 사용
+ var type = gl.FLOAT; // 데이터는 32비트 부동소수점
+ var normalize = false; // 데이터를 정규화하지 않음
+ var stride = 0; // 0 = 반복할 때마다 다음 위치를 얻기 위해 size * sizeof(type)씩 앞으로 이동
+ var offset = 0; // 버퍼의 맨 앞부터 시작
+ gl.vertexAttribPointer(
+ texCoordAttributeLocation, size, type, normalize, stride, offset)
+
+ // 텍스처를 생성합니다.
+ var texture = gl.createTexture();
+
+ // 텍스처 유닛 0을 활성화합니다.
+ // (이후 호출하는 텍스처 명령들이 영향을 주게 될 텍스처 유닛)
+ gl.activeTexture(gl.TEXTURE0 + 0);
+
+ // 텍스처 유닛 0에 텍스처를 바인딩합니다.
+ gl.bindTexture(gl.TEXTURE_2D, texture);
+
+ // 매개 변수를 설정하여 우리는 밉맵이 필요 없으며
+ // 필터링 하지 않을 것이고 텍스처 반복(repeat)도 필요 없다고 알립니다.
+ gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
+ gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
+ gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST);
+ gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST);
+
+ // 텍스처로 이미지를 업로드
+ var mipLevel = 0; // 가장큰 밉맵
+ var internalFormat = gl.RGBA; // 텍스처로 사용하길 원하는 포맷
+ var srcFormat = gl.RGBA; // 우리가 제공하는 데이터의 포맷
+ var srcType = gl.UNSIGNED_BYTE // 우리가 제공하는 데이터의 타입
+ gl.texImage2D(gl.TEXTURE_2D,
+ mipLevel,
+ internalFormat,
+ srcFormat,
+ srcType,
+ image);
...
// 프로그램(셰이더 쌍)을 사용하도록 합니다.
gl.useProgram(program);
// 셰이더에서 픽셀에서 클립공간으로 변환 할수 있도록
// 캔버스 해상도를 전달합니다.
gl.uniform2f(resolutionLocation, gl.canvas.width, gl.canvas.height);
+ // 셰이더에 텍스처 유닛 0에서 텍스처를 가져오라고 알려줍니다.
+ gl.uniform1i(imageLocation, 0);
+ // position 버퍼를 바인딩하여 setRectangle에서 호출 될
+ // gl.bufferData가 position 버퍼에 데이터를 넣습니다.
+ gl.bindBuffer(gl.ARRAY_BUFFER, positionBuffer);
+
+ // 사각형의 크기를 이미지와 같은 크기로 설정합니다.
+ setRectangle(gl, 0, 0, image.width, image.height);
}
여기에 WebGL로 렌더링한 이미지가 있습니다.
아직 그렇게 신기하지는 않으니까 한번 이미지를 조작해봅시다. 빨간색이랑 파란색을 바꾸는 건 어떨까요?
...
outColor = texture(u_image, v_texCoord).bgra;
...
이제 빨간색과 파란색이 바뀌었습니다.
다른 픽셀들의 값을 사용하는 이미치 처리를 하고 싶다면 어떻게 해야 할까요?
WebGL은 0.0에서 1.0까지인 텍스처 좌표에서 텍스처들을 참조하므로 1픽셀 거리가 텍스처 좌표로 얼만큼인지를 onePixel = 1.0 / textureSize
를 사용해 계산할 수 있습니다.
아래는 각 픽셀의 왼쪽, 오른쪽 픽셀값을 사용해 평균을 내는 프래그먼트 셰이더 입니다.
#version 300 es
// 프래그먼트 셰이더는 정밀도 기본값이 없으므로 우리가 선택해야 합니다.
// highp가 대개 괜찮습니다. "높은 정밀도"를 의미합니다.
precision highp float;
// 텍스처
uniform sampler2D u_image;
// texCoord는 정점 셰이더에서 전달됩니다.
in vec2 v_texCoord;
// 프래그먼트 셰이더는 출력을 선언 해야합니다.
out vec4 outColor;
void main() {
+ vec2 onePixel = vec2(1) / vec2(textureSize(u_image, 0));
+
+ // 왼쪽, 중간, 오른쪽 픽셀의 평균을 계산합니다.
+ outColor = (
+ texture(u_image, v_texCoord) +
+ texture(u_image, v_texCoord + vec2( onePixel.x, 0.0)) +
+ texture(u_image, v_texCoord + vec2(-onePixel.x, 0.0))) / 3.0;
}
블러(blur)시키지 않은 위 예제와 비교해 보세요.
이제 다른 픽셀을 참조하는 방법을 알았으니 컨볼루션 커널(convolution kernel)을 사용하여 몇 가지 일반적인 이미지 처리를 해보겠습니다. 여기에서는 3x3 커널을 사용할 것입니다. 컨볼루션 커널은 단순히 3x3 행렬이며 여기서 행렬의 각 요소는 렌더링할 픽셀 주위에 있는 8개의 픽셀에 어떤 값을 곱할 것인지를 나타냅니다. 그 다음 그 결과를 커널 가중치(커널의 모든 값들의 합) 또는 1.0 중 큰 값으로 나눕니다. 여기 꽤 좋은 읽을거리가 있습니다. 그리고 C++로 직접 작성하려면 어떻게 해야하는지를 실제 코드를 통해 보여주는 다른 읽을거리가 있습니다.
우리의 경우 셰이더에서 이 작업을 수행할 것이므로 아래와 같이 새로운 프래그먼트 셰이더를 작성합니다.
#version 300 es
// 프래그먼트 셰이더는 정밀도 기본값이 없으므로 우리가 선택해야 합니다.
// highp가 대개 괜찮습니다. "높은 정밀도"를 의미합니다.
precision highp float;
// 텍스처
uniform sampler2D u_image;
// 컨볼루션 커널 데이터
uniform float u_kernel[9];
uniform float u_kernelWeight;
// texCoord는 정점 셰이더에서 전달됩니다.
in vec2 v_texCoord;
// 프래그먼트 셰이더는 출력을 선언 해야합니다.
out vec4 outColor;
void main() {
vec2 onePixel = vec2(1) / vec2(textureSize(u_image, 0));
vec4 colorSum =
texture(u_image, v_texCoord + onePixel * vec2(-1, -1)) * u_kernel[0] +
texture(u_image, v_texCoord + onePixel * vec2( 0, -1)) * u_kernel[1] +
texture(u_image, v_texCoord + onePixel * vec2( 1, -1)) * u_kernel[2] +
texture(u_image, v_texCoord + onePixel * vec2(-1, 0)) * u_kernel[3] +
texture(u_image, v_texCoord + onePixel * vec2( 0, 0)) * u_kernel[4] +
texture(u_image, v_texCoord + onePixel * vec2( 1, 0)) * u_kernel[5] +
texture(u_image, v_texCoord + onePixel * vec2(-1, 1)) * u_kernel[6] +
texture(u_image, v_texCoord + onePixel * vec2( 0, 1)) * u_kernel[7] +
texture(u_image, v_texCoord + onePixel * vec2( 1, 1)) * u_kernel[8] ;
outColor = vec4((colorSum / u_kernelWeight).rgb, 1);
}
자바스크립트에서 컨볼루션 커널과 커널의 가중치를 제공해야 합니다.
function computeKernelWeight(kernel) {
var weight = kernel.reduce(function(prev, curr) {
return prev + curr;
});
return weight <= 0 ? 1 : weight;
}
...
var kernelLocation = gl.getUniformLocation(program, "u_kernel[0]");
var kernelWeightLocation = gl.getUniformLocation(program, "u_kernelWeight");
...
var edgeDetectKernel = [
-1, -1, -1,
-1, 8, -1,
-1, -1, -1
];
// 커널과 커널의 가중치를 설정합니다.
gl.uniform1fv(kernelLocation, edgeDetectKernel);
gl.uniform1f(kernelWeightLocation, computeKernelWeight(edgeDetectKernel));
...
그리고 짠… 드랍 다운 리스트를 사용해서 다른 커널을 선택해 보세요.
이 글을 통해 WebGL에서의 이미지 처리가 매우 간단하다는 것을 이해하셨길 바랍니다. 다음으로 이미지에 두가지 이상의 효과를 적용하는 방법을 살펴 보겠습니다.