2022-09-16 05:03:07 +03:00
|
|
|
$input v_position, v_texcoord0
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2022 Preetish Kakkar. All rights reserved.
|
|
|
|
* License: https://github.com/bkaradzic/bgfx/blob/master/LICENSE
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
Most of the code is inspired/ported from https://github.com/mmikk/hextile-demo/blob/main/hextile-demo/shader_lighting.hlsl
|
|
|
|
|
|
|
|
The basic idea behind the algorithm is to use tiling & blending schema but instead of regular linear blending, the algorithm uses blending operator that prevents visual artifacts caused by linear blending
|
|
|
|
|
|
|
|
We partition the uv-space on a triangle grid and compute the local triangle and the barycentric coordinates inside the triangle. We use a hash function to associate a random offset with each vertex of the triangle
|
|
|
|
grid and use this random offset to fetch the example texture.
|
|
|
|
|
|
|
|
Finally, we blend the result using the barycentric coordinates as blending weights.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "../common/common.sh"
|
|
|
|
|
|
|
|
#define M_PI 3.1415926535897932384626433832795
|
|
|
|
|
|
|
|
SAMPLER2D(s_trx_d, 0);
|
|
|
|
|
|
|
|
uniform vec4 u_params;
|
|
|
|
|
|
|
|
#ifndef fmod
|
|
|
|
#define fmod(x, y) (x - y * trunc(x / y))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define moduleOper(a, b) a - (float(b) * floor(a/float(b)))
|
|
|
|
|
|
|
|
#define u_showWeights u_params.x
|
|
|
|
#define u_tileRate u_params.y
|
|
|
|
#define u_tileRotStrength u_params.z
|
|
|
|
#define u_useRegularTiling u_params.w
|
|
|
|
|
|
|
|
vec3 Gain3(vec3 x, float r)
|
|
|
|
{
|
|
|
|
// increase contrast when r>0.5 and
|
|
|
|
// reduce contrast if less
|
|
|
|
float k = log(1.0 - r) / log(0.5);
|
|
|
|
|
|
|
|
vec3 s = 2.0 * step(0.5, x);
|
|
|
|
vec3 m = 2.0 * (1.0 - s);
|
|
|
|
|
|
|
|
vec3 res = 0.5 * s + 0.25 * m * pow(max(vec3_splat(0.0), s + x * m), vec3_splat(k));
|
|
|
|
|
|
|
|
return res.xyz / (res.x + res.y + res.z);
|
|
|
|
}
|
|
|
|
|
|
|
|
mat2 LoadRot2x2(vec2 idx, float rotStrength)
|
|
|
|
{
|
|
|
|
float angle = abs(idx.x * idx.y) + abs(idx.x + idx.y) + M_PI;
|
|
|
|
|
|
|
|
// remap to +/-pi
|
|
|
|
//angle = fmod(angle, 2.0*M_PI);
|
|
|
|
if (angle < 0.0) angle += 2.0 * M_PI;
|
|
|
|
if (angle > M_PI) angle -= 2.0 * M_PI;
|
|
|
|
|
|
|
|
angle *= rotStrength;
|
|
|
|
|
|
|
|
float cs = cos(angle);
|
|
|
|
float si = sin(angle);
|
|
|
|
|
|
|
|
return mat2(cs, -si, si, cs);
|
|
|
|
}
|
|
|
|
|
|
|
|
vec2 MakeCenST(vec2 Vertex)
|
|
|
|
{
|
|
|
|
mat2 invSkewMat = mat2(1.0, 0.5, 0.0, 1.0 / 1.15470054);
|
|
|
|
|
|
|
|
return mul(invSkewMat, Vertex) / (2.0 * sqrt(3.0));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
vec3 ProduceHexWeights(vec3 W, vec2 vertex1, vec2 vertex2, vec2 vertex3)
|
|
|
|
{
|
|
|
|
vec3 res = vec3_splat(0.0);
|
|
|
|
|
|
|
|
float v1 = moduleOper(((vertex1.x - vertex1.y)), 3.0);
|
|
|
|
if (v1 < 0.0) v1 += 3.0;
|
|
|
|
|
|
|
|
float vh = v1 < 2.0 ? (v1 + 1.0) : 0.0;
|
|
|
|
float vl = v1 > 0.0 ? (v1 - 1.0) : 2.0;
|
|
|
|
float v2 = vertex1.x < vertex3.x ? vl : vh;
|
|
|
|
float v3 = vertex1.x < vertex3.x ? vh : vl;
|
|
|
|
|
|
|
|
res.x = v3 == 0.0 ? W.z : (v2 == 0.0 ? W.y : W.x);
|
|
|
|
res.y = v3 == 1.0 ? W.z : (v2 == 1.0 ? W.y : W.x);
|
|
|
|
res.z = v3 == 2.0 ? W.z : (v2 == 2.0 ? W.y : W.x);
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
vec2 hash(vec2 p)
|
|
|
|
{
|
|
|
|
vec2 r = mul(mat2(127.1, 311.7, 269.5, 183.3), p);
|
|
|
|
|
|
|
|
return fract(sin(r) * 43758.5453);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Given a point in UV, compute local triangle barycentric coordinates and vertex IDs
|
|
|
|
void TriangleGrid(out float w1, out float w2, out float w3,
|
|
|
|
out vec2 vertex1, out vec2 vertex2, out vec2 vertex3,
|
|
|
|
vec2 uv)
|
|
|
|
{
|
|
|
|
// Scaling of the input
|
|
|
|
uv *= 2.0 * sqrt(3.0); // controls the size of the input with respect to the size of the tiles.
|
|
|
|
|
|
|
|
// Skew input space into simplex triangle grid
|
|
|
|
const mat2 gridToSkewedGrid =
|
|
|
|
mat2(1.0, -0.57735027, 0.0, 1.15470054);
|
|
|
|
vec2 skewedCoord = mul(gridToSkewedGrid, uv);
|
|
|
|
|
|
|
|
vec2 baseId = floor(skewedCoord);
|
|
|
|
vec3 temp = vec3(fract(skewedCoord), 0.0);
|
|
|
|
temp.z = 1.0 - temp.x - temp.y;
|
|
|
|
|
|
|
|
float s = step(0.0, -temp.z);
|
|
|
|
float s2 = 2.0 * s - 1.0;
|
|
|
|
|
|
|
|
w1 = -temp.z * s2;
|
|
|
|
w2 = s - temp.y * s2;
|
|
|
|
w3 = s - temp.x * s2;
|
|
|
|
|
|
|
|
vertex1 = baseId + vec2(s, s);
|
|
|
|
vertex2 = baseId + vec2(s, 1.0 - s);
|
|
|
|
vertex3 = baseId + vec2(1.0 - s, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
void hex2colTex(out vec4 color, out vec3 weights, vec2 uv,
|
|
|
|
float rotStrength, float r)
|
|
|
|
{
|
|
|
|
// compute uv derivatives
|
|
|
|
vec2 dSTdx = dFdx(uv), dSTdy = dFdy(uv);
|
|
|
|
|
|
|
|
// Get triangle info
|
|
|
|
float w1, w2, w3;
|
|
|
|
vec2 vertex1, vertex2, vertex3;
|
|
|
|
TriangleGrid(w1, w2, w3, vertex1, vertex2, vertex3, uv);
|
|
|
|
|
|
|
|
mat2 rot1 = LoadRot2x2(vertex1, rotStrength);
|
|
|
|
mat2 rot2 = LoadRot2x2(vertex2, rotStrength);
|
|
|
|
mat2 rot3 = LoadRot2x2(vertex3, rotStrength);
|
|
|
|
|
|
|
|
vec2 cen1 = MakeCenST(vertex1);
|
|
|
|
vec2 cen2 = MakeCenST(vertex2);
|
|
|
|
vec2 cen3 = MakeCenST(vertex3);
|
|
|
|
|
|
|
|
// assign random offset to each triangle vertex
|
|
|
|
// this is used later to fetch from texture
|
|
|
|
vec2 uv1 = mul(uv - cen1, rot1) + cen1 + hash(vertex1);
|
|
|
|
vec2 uv2 = mul(uv - cen2, rot2) + cen2 + hash(vertex2);
|
|
|
|
vec2 uv3 = mul(uv - cen3, rot3) + cen3 + hash(vertex3);
|
|
|
|
|
|
|
|
// Fetch input
|
|
|
|
// We could simply use texture2D function, however, the sreen space derivatives could be broken
|
|
|
|
// since we are using random offsets, we use texture2DGrad to make sure that we pass correct derivatives explicitly.
|
|
|
|
vec4 c1 = texture2DGrad(s_trx_d, uv1,
|
|
|
|
mul(dSTdx, rot1), mul(dSTdy, rot1));
|
|
|
|
vec4 c2 = texture2DGrad(s_trx_d, uv2,
|
|
|
|
mul(dSTdx, rot2), mul(dSTdy, rot2));
|
|
|
|
vec4 c3 = texture2DGrad(s_trx_d, uv3,
|
|
|
|
mul(dSTdx, rot3), mul(dSTdy, rot3));
|
|
|
|
|
|
|
|
// use luminance as weight
|
|
|
|
vec3 Lw = vec3(0.299, 0.587, 0.114);
|
|
|
|
vec3 Dw = vec3(dot(c1.xyz, Lw), dot(c2.xyz, Lw), dot(c3.xyz, Lw));
|
|
|
|
|
|
|
|
Dw = mix(vec3_splat(1.0), Dw, 0.6); // 0.6 is fall off constant
|
|
|
|
vec3 W = Dw * pow(vec3(w1, w2, w3), vec3_splat(7.0)); // 7 is g_exp
|
|
|
|
W /= (W.x + W.y + W.z);
|
|
|
|
if (r != 0.5) W = Gain3(W, r);
|
|
|
|
|
|
|
|
// blend weights with color linearly
|
|
|
|
// histogram preserving blending will be better but requires precompution step to create histogram texture
|
|
|
|
color = W.x * c1 + W.y * c2 + W.z * c3;
|
|
|
|
weights = ProduceHexWeights(W.xyz, vertex1, vertex2, vertex3);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float GetTileRate()
|
|
|
|
{
|
|
|
|
return 0.05 * u_tileRate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FetchColorAndWeight(out vec3 color, out vec3 weights, vec2 uv)
|
|
|
|
{
|
|
|
|
vec4 col4;
|
|
|
|
hex2colTex(col4, weights, uv, u_tileRotStrength, 0.7);
|
|
|
|
color = col4.xyz;
|
|
|
|
}
|
|
|
|
|
|
|
|
void main()
|
|
|
|
{
|
2022-09-16 23:48:19 +03:00
|
|
|
// actual world space position
|
|
|
|
vec3 surfPosInWorld = v_position.xyz;
|
|
|
|
|
|
|
|
vec3 sp = GetTileRate() * surfPosInWorld;
|
|
|
|
|
|
|
|
vec2 uv0 = vec2(sp.x, sp.z);
|
|
|
|
|
2022-09-16 05:03:07 +03:00
|
|
|
if(u_useRegularTiling > 0.0)
|
|
|
|
{
|
2022-09-16 23:48:19 +03:00
|
|
|
gl_FragColor = vec4(texture2D(s_trx_d, uv0.xy));
|
2022-09-16 05:03:07 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
vec3 color, weights;
|
|
|
|
FetchColorAndWeight(color, weights, uv0);
|
|
|
|
|
|
|
|
if (u_showWeights > 0.0)
|
|
|
|
{
|
|
|
|
gl_FragColor = vec4(weights, 1.0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
gl_FragColor = vec4(color, 1.0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|