Что представляет собой аналог языка OpenGL ES 2 Shader для HYDRA (пиксельный бендер) sampleLinear?
Поэтому я смотрю на спецификации шейдеров OpenGL ES, но не вижу таких...
Например - я создал простые "щепотка к зоону" и "вращение, чтобы повернуть" и "движение к центру" фильтр пиксель-изгиба HYDRA. это может быть выполнено во вспышке. Он основан на примере закрутки пиксельного бендера по умолчанию, и это:
<languageVersion: 1.0;>
kernel zoomandrotate
< namespace : "Pixel Bender Samples";
vendor : "Kabumbus";
version : 3;
description : "rotate and zoom an image around"; >
{
// define PI for the degrees to radians calculation
const float PI = 3.14159265;
// An input parameter to specify the center of the twirl effect.
// As above, we're using metadata to indicate the minimum,
// maximum, and default values, so that the tools can set the values
// in the correctly in the UI for the filter.
parameter float2 center
<
minValue:float2(0.0, 0.0);
maxValue:float2(2048.0, 2048.0);
defaultValue:float2(256.0, 256.0);
>;
// An input parameter to specify the angle that we would like to twirl.
// For this parameter, we're using metadata to indicate the minimum,
// maximum, and default values, so that the tools can set the values
// in the correctly in the UI for the filter.
parameter float twirlAngle
<
minValue:float(0.0);
maxValue:float(360.0);
defaultValue:float(90.0);
>;
parameter float zoomAmount
<
minValue:float(0.01);
maxValue:float(10.0);
defaultValue:float(1);
>;
// An input parameter that indicates how we want to vary the twirling
// within the radius. We've added support to modulate by one of two
// functions, a gaussian or a sinc function. Since Flash does not support
// bool parameters, we instead are using this as an int with two possible
// values. Setting this parameter to be 1 will
// cause the gaussian function to be used, unchecking it will cause
// the sinc function to be used.
parameter int gaussOrSinc
<
minValue:int(0);
maxValue:int(1);
defaultValue:int(0);
>;
input image4 oImage;
output float4 outputColor;
// evaluatePixel(): The function of the filter that actually does the
// processing of the image. This function is called once
// for each pixel of the output image.
void
evaluatePixel()
{
// convert the angle to radians
float twirlAngleRadians = radians(twirlAngle);
// calculate where we are relative to the center of the twirl
float2 relativePos = outCoord() - center;
// calculate the absolute distance from the center normalized
// by the twirl radius.
float distFromCenter = length( relativePos );
distFromCenter = 1.0;
// modulate the angle based on either a gaussian or a sync.
float adjustedRadians;
// precalculate either the gaussian or the sinc weight
float sincWeight = sin( distFromCenter ) * twirlAngleRadians / ( distFromCenter );
float gaussWeight = exp( -1.0 * distFromCenter * distFromCenter ) * twirlAngleRadians;
// protect the algorithm from a 1 / 0 error
adjustedRadians = (distFromCenter == 0.0) ? twirlAngleRadians : sincWeight;
// switch between a gaussian falloff or a sinc fallof
adjustedRadians = (gaussOrSinc == 1) ? adjustedRadians : gaussWeight;
// rotate the pixel sample location.
float cosAngle = cos( adjustedRadians );
float sinAngle = sin( adjustedRadians );
float2x2 rotationMat = float2x2(
cosAngle, sinAngle,
-sinAngle, cosAngle
);
relativePos = rotationMat * relativePos;
float scale = zoomAmount;
// sample and set as the output color. since relativePos
// is related to the center location, we need to add it back in.
// We use linear sampling to smooth out some of the pixelation.
outputColor = sampleLinear( oImage, relativePos/scale + center );
}
}
Теперь я хочу перенести его в шейдер OpenGL ES. математика и параметры могут быть преобразованы в язык шейдеров OpenGL ES, но что делать с sampleLinear? что является аналогом для него в OpenGL ES шейдерах?
Обновить:
Поэтому я создал нечто похожее на мой фильтр HYDRA... совместимый с шейдерами WebGL и OpenGL ES...
#ifdef GL_ES
precision highp float;
#endif
uniform vec2 resolution;
uniform float time;
uniform sampler2D tex0;
void main(void)
{
vec2 p = -1.0 + 2.0 * gl_FragCoord.xy / resolution.xy;
// a rotozoom
vec2 cst = vec2( cos(.5*time), sin(.5*time) );
mat2 rot = 0.5*cst.x*mat2(cst.x,-cst.y,cst.y,cst.x);
vec3 col = texture2D(tex0,0.5*rot*p+sin(0.1*time)).xyz;
gl_FragColor = vec4(col,1.0);
}
Чтобы увидеть, как работает современный браузер, перейдите к шадертой и предоставьте ему одну текстуру (http://www.iquilezles.org/apps/shadertoy/presets/tex4.jpg
например), вставьте мой код в редактируемый текст Aeria и нажмите ... Повеселись. Итак... теперь у меня есть другая проблема... Я хочу иметь одно изображение и черное вокруг него, а не копии того же самого изображения... Кто-нибудь знает, как это сделать?
1 ответ
В справочнике по пиксельным блендерам Adobe, sampleLinear "Обрабатывает координаты не в центрах пикселей, выполняя билинейную интерполяцию для значений соседних пикселей".
Правильный способ добиться этого в OpenGL - это использовать texture2D, как вы уже делаете, но установить текстурную среду для линейной фильтрации через glTexParameter.
Вы можете использовать функцию шага и умножить на ее результат, чтобы получить черный цвет за пределами пикселей, или дать текстуре черную рамку в один пиксель и переключиться на фиксацию, а не повторение, также через glTexParameter
,
Если вы хотите сделать это в коде, попробуйте:
#ifdef GL_ES
precision highp float;
#endif
uniform vec2 resolution;
uniform float time;
uniform sampler2D tex0;
void main(void)
{
vec2 p = -1.0 + 2.0 * gl_FragCoord.xy / resolution.xy;
// a rotozoom
vec2 cst = vec2( cos(.5*time), sin(.5*time) );
mat2 rot = 0.5*cst.x*mat2(cst.x,-cst.y,cst.y,cst.x);
vec2 samplePos = 0.5*rot*p+sin(0.1*time);
float mask = step(samplePos.x, 0.0) * step(samplePos.y, 0.0) * (1.0 - step(samplePos.x, 1.0)) * (1.0 - step(samplePos.y, 1.0));
vec3 col = texture2D(tex0,samplePos).xyz;
gl_FragColor = vec4(col*mask,1.0);
}
Это ограничивало бы приход цветов из поля от (0, 0) до (1, 1), но похоже, что шейдер уходит в несколько значительно искаженных мест, так что я не уверен точно, что вы хотите.