pouët.net

Rendering a bezier curve on the GPU

category: code [glöplog]
I found this shader on glslsandbox : http://glslsandbox.com/e#40709.0

It renders a bezier curve by calculating closest distance between a point and a curve.

I was wondering if there was a mathematical way to calculate another value (let's call it T) which will indicate how far we are from start and end point of the curve :

Example (red is curve)

That pair of values (distance + T) could be used like X/Y axis. Eg : to map a texture on the curve : something like this
added on the 2017-06-02 12:50:45 by Tigrou Tigrou
Tigrou: for quadratic there is a closed formula for "t" as well. Most shader just don't use it. Try this https://www.shadertoy.com/view/ltXSDB and you can see in sdBezier function "t" is actually calculated. Later it is additionally compared if endpoints are closer - than you just set "t" to 0 or 1. It's just not returned.

For cubic: usually some form of iterative bisection is used and you also calculate "t" along the way,like here: https://www.shadertoy.com/view/4sXyDr. Again,you should be able to easily return it as well.
added on the 2017-06-02 15:13:31 by tomkh tomkh
Thanks for your help.
I have adapted code a little bit. It's far from perfect, there is glitches near start / end points and near middle point with hairpin turns.
Note : you need a texture in iChannel0 to make it work.

Code:// Signed Distance to a Quadratic Bezier Curve // - Adam Simmons (@adamjsimmons) 2015 // // License Creative Commons Attribution-NonCommercial-ShareAlike 3.0 Unported License // // Inspired by http://www.pouet.net/topic.php?which=9119 // and various shaders by iq, T21, and demofox // // I needed the -signed- distance to a quadratic bezier // curve but couldn't find any examples online that // were both fast and precise. This is my solution. // // v1 - Initial release // v2 - Faster and more robust sign computation // // Test if point p crosses line (a, b), returns sign of result float testCross(vec2 a, vec2 b, vec2 p) { return sign((b.y-a.y) * (p.x-a.x) - (b.x-a.x) * (p.y-a.y)); } // Determine which side we're on (using barycentric parameterization) float signBezier(vec2 A, vec2 B, vec2 C, vec2 p) { vec2 a = C - A, b = B - A, c = p - A; vec2 bary = vec2(c.x*b.y-b.x*c.y,a.x*c.y-c.x*a.y) / (a.x*b.y-b.x*a.y); vec2 d = vec2(bary.y * 0.5, 0.0) + 1.0 - bary.x - bary.y; return mix(sign(d.x * d.x - d.y), mix(-1.0, 1.0, step(testCross(A, B, p) * testCross(B, C, p), 0.0)), step((d.x - d.y), 0.0)) * testCross(A, C, B); } // Solve cubic equation for roots vec3 solveCubic(float a, float b, float c) { float p = b - a*a / 3.0, p3 = p*p*p; float q = a * (2.0*a*a - 9.0*b) / 27.0 + c; float d = q*q + 4.0*p3 / 27.0; float offset = -a / 3.0; if(d >= 0.0) { float z = sqrt(d); vec2 x = (vec2(z, -z) - q) / 2.0; vec2 uv = sign(x)*pow(abs(x), vec2(1.0/3.0)); return vec3(offset + uv.x + uv.y); } float v = acos(-sqrt(-27.0 / p3) * q / 2.0) / 3.0; float m = cos(v), n = sin(v)*1.732050808; return vec3(m + m, -n - m, n - m) * sqrt(-p / 3.0) + offset; } vec2 min2(vec2 a, vec2 b) { if(a.x < b.x) return vec2(a.x, a.y); else return vec2(b.x, b.y); } // Find the signed distance from a point to a bezier curve vec2 sdBezier(vec2 A, vec2 B, vec2 C, vec2 p) { B = mix(B + vec2(1e-4), B, abs(sign(B * 2.0 - A - C))); vec2 a = B - A, b = A - B * 2.0 + C, c = a * 2.0, d = A - p; vec3 k = vec3(3.*dot(a,b),2.*dot(a,a)+dot(d,b),dot(d,a)) / dot(b,b); vec3 tc = solveCubic(k.x, k.y, k.z); vec3 t = clamp(tc, 0.0, 1.0); vec2 pos = A + (c + b*t.x)*t.x; vec2 dis = vec2(length(pos - p), tc.x); pos = A + (c + b*t.y)*t.y; dis = min2(dis, vec2(length(pos - p), tc.y)); pos = A + (c + b*t.z)*t.z; dis = min2(dis, vec2(length(pos - p), tc.z)); return vec2(dis.x * signBezier(A, B, C, p), dis.y); } void mainImage( out vec4 fragColor, in vec2 fragCoord ) { vec2 p = (2.0*fragCoord.xy-iResolution.xy)/iResolution.y; vec2 m = (2.0*iMouse.xy-iResolution.xy)/iResolution.y; // Define the control points of our curve vec2 A = vec2(0.0, -0.6), C = vec2(0.0, +0.6), B = (4.0 * m - A - C) / 2.0; // Render the control points float d = min(distance(p, A),(min(distance(p, m),distance(p, C)))); if (d < 0.04) { fragColor = vec4(1.0 - smoothstep(0.025, 0.034, d)); return; } // Get the signed distance to bezier curve vec2 uv = sdBezier(A, B, C, p); // Visualize the distance field if(abs(uv.x) < 0.3) fragColor = texture(iChannel0, uv); else fragColor = vec4(0.0); }
added on the 2017-06-02 17:09:33 by Tigrou Tigrou
I cannot check/compile right now,but shouldn't you use clamped "t" (not "tc") passed to min2 as well?
Also min2 could be just return (a.x<b.x)?a:b; but that is just minor of course.
added on the 2017-06-02 19:04:02 by tomkh tomkh
Tigrou: ok, I see the problem now (after compiling). You have to clamp "t" for endpoints like I said,but treat areas that are closer to endpoints a bit differently - you could try to extended "t" in angular space (using atan2 or something).
As for glitches in the middle: of course if the curvature is bigger than 1/width you will have discontinuities, also I see some precision issues there,but I think you cannot really do much about it :/
added on the 2017-06-02 19:33:48 by tomkh tomkh
Or even better: extend "t" by projecting to straight lines extending the curve at endpoints.
added on the 2017-06-02 19:35:58 by tomkh tomkh
This is what I mean:
Code: // Based on: // Signed Distance to a Quadratic Bezier Curve // - Adam Simmons (@adamjsimmons) 2015 // https://www.shadertoy.com/view/ltXSDB // Solve cubic equation for roots vec3 solveCubic(float a, float b, float c) { float p = b - a*a / 3.0, p3 = p*p*p; float q = a * (2.0*a*a - 9.0*b) / 27.0 + c; float d = q*q + 4.0*p3 / 27.0; float offset = -a / 3.0; if(d >= 0.0) { float z = sqrt(d); vec2 x = (vec2(z, -z) - q) / 2.0; vec2 uv = sign(x)*pow(abs(x), vec2(1.0/3.0)); return vec3(offset + uv.x + uv.y); } float v = acos(-sqrt(-27.0 / p3) * q / 2.0) / 3.0; float m = cos(v), n = sin(v)*1.732050808; return vec3(m + m, -n - m, n - m) * sqrt(-p / 3.0) + offset; } // Signed distance to cubic bezier with parametrization. // Tom'2017 // returns vec4( // unsigned distance to clamped curve, // signed distance to extended curve, // extended t ) vec3 sdBezier(vec2 A, vec2 B, vec2 C, vec2 p) { B = mix(B + vec2(1e-4), B, abs(sign(B * 2.0 - A - C))); vec2 a = B - A, b = A - B * 2.0 + C, c = a * 2.0, d = A - p; vec3 k = vec3(3.*dot(a,b),2.*dot(a,a)+dot(d,b),dot(d,a)) / dot(b,b); vec3 t = clamp(solveCubic(k.x, k.y, k.z), 0.0, 1.0); vec2 dp1 = d + (c + b*t.x)*t.x; float d1 = dot(dp1, dp1); vec2 dp2 = d + (c + b*t.y)*t.y; float d2 = dot(dp2, dp2); // note: 3rd root is actually never closest, we can just ignore it // Find closest distance and t vec2 r = (d1 < d2) ? vec2(d1, t.x) : vec2(d2, t.y); // Find on which side (t=0 or t=1) is extension vec2 e = vec2(step(0.,-r.y),step(1.,r.y)); // Calc. gradient vec2 g = 2.*b*r.y + c; // Calc. extension to t float et = (e.x*dot(-d,g) + e.y*dot(p-C,g))/dot(g,g); // Find closest point on curve with extension vec2 dp = d + (c + b*r.y)*r.y + et*g; // Sign is just cross product with gradient float s = sign(g.x*dp.y - g.y*dp.x); return vec3(sqrt(r.x), s*length(dp), r.y + et); } void mainImage( out vec4 fragColor, in vec2 fragCoord ) { vec2 p = (2.0*fragCoord.xy-iResolution.xy)/iResolution.y; vec2 m = (2.0*iMouse.xy-iResolution.xy)/iResolution.y; // Define the control points of our curve vec2 A = vec2(0.0, -0.6), C = vec2(0.0, +0.6), B = (4.0 * m - A - C) / 2.0; // Render the control points float d = min(distance(p, A),(min(distance(p, m),distance(p, C)))); if (d < 0.04) { fragColor = vec4(1.0 - smoothstep(0.025, 0.034, d)); return; } // Get the signed distance to bezier curve vec3 r = sdBezier(A, B, C, p); vec2 uv = r.yz; // Visualize the distance field fragColor = texture(iChannel0, uv)*smoothstep(.5,.4,r.x); }

(again you need to add texture of course)
added on the 2017-06-03 01:38:06 by tomkh tomkh
Thanks. Not only it works but it is also lot simpler. Good job.
added on the 2017-06-03 01:44:04 by Tigrou Tigrou
Of course you can make it even simpler version, but without calculating distance to endpoints (open-ended):
Code: // Signed distance to open-ended quadratic bezier with parametrization. // Tom'2017 // returns vec4( unsigned distance, signed distance, t ) vec3 sdBezier(vec2 A, vec2 B, vec2 C, vec2 p) { B = mix(B + vec2(1e-4), B, abs(sign(B * 2.0 - A - C))); vec2 a = B - A, b = A - B * 2.0 + C, c = a * 2.0, d = A - p; vec3 k = vec3(3.*dot(a,b),2.*dot(a,a)+dot(d,b),dot(d,a)) / dot(b,b); vec3 t = solveCubic(k.x, k.y, k.z); vec2 dp1 = d + (c + b*t.x)*t.x; float d1 = dot(dp1, dp1); vec2 dp2 = d + (c + b*t.y)*t.y; float d2 = dot(dp2, dp2); vec4 r = (d1 < d2) ? vec4(d1, t.x, dp1) : vec4(d2, t.y, dp2); // Calc. gradient vec2 g = 2.*b*r.y + c; // Sign is just cross product with gradient float s = sign(g.x*r.w - g.y*r.z); float dist = sqrt(r.x); return vec3(dist, s*dist, r.y); }


The question is the use-case. Obviously, more advanced version should support a path of connected curves, which is another story - I'm too tired to try ;)
added on the 2017-06-03 02:29:21 by tomkh tomkh
Ok, ok, as long as gradients are continuous on joints, multiple connected curves should work just fine with previous version.
You can just check which curve segment is closer (using unsigned distance), and also yeah, when two join, the second one should have t+1 to maintain continuity.
added on the 2017-06-03 03:07:32 by tomkh tomkh
This is the source code for 3D Quadratic Bezier, both approximate and exact though cubic solver.

tomkh, look a the code, it shows how to connect different segments to create an arbitrary curve: https://www.shadertoy.com/view/ldj3Wh

I copy here the relevant code from that link. It returns the closest distance, and also the parameter along the curve.

Code: // The MIT License // Copyright © 2013 Inigo Quilez vec2 sdBezier(vec3 A, vec3 B, vec3 C, vec3 pos) { vec3 a = B - A; vec3 b = A - 2.0*B + C; vec3 c = a * 2.0; vec3 d = A - pos; float kk = 1.0 / dot(b,b); float kx = kk * dot(a,b); float ky = kk * (2.0*dot(a,a)+dot(d,b)) / 3.0; float kz = kk * dot(d,a); vec2 res; float p = ky - kx*kx; float p3 = p*p*p; float q = kx*(2.0*kx*kx - 3.0*ky) + kz; float h = q*q + 4.0*p3; if(h >= 0.0) { h = sqrt(h); vec2 x = (vec2(h, -h) - q) / 2.0; vec2 uv = sign(x)*pow(abs(x), vec2(1.0/3.0)); float t = uv.x + uv.y - kx; t = clamp( t, 0.0, 1.0 ); // 1 root vec3 qos = d + (c + b*t)*t; res = vec2( length(qos),t); } else { float z = sqrt(-p); float v = acos( q/(p*z*2.0) ) / 3.0; float m = cos(v); float n = sin(v)*1.732050808; vec3 t = vec3(m + m, -n - m, n - m) * z - kx; t = clamp( t, 0.0, 1.0 ); // 3 roots vec3 qos = d + (c + b*t.x)*t.x; float dis = dot(qos,qos); res = vec2(dis,t.x); qos = d + (c + b*t.y)*t.y; dis = dot(qos,qos); if( dis<res.x ) res = vec2(dis,t.y ); qos = d + (c + b*t.z)*t.z; dis = dot(qos,qos); if( dis<res.x ) res = vec2(dis,t.z ); res.x = sqrt( res.x ); } return res; }
added on the 2017-06-04 07:01:39 by iq iq
I forgot to link to the image that gets rendered by that code and shader:
[img]
added on the 2017-06-04 07:08:54 by iq iq
Holly shit, that worked with the "preview" button, but not with the actual display. Sorry for that. Maybe an admin can delete that post and this one??
added on the 2017-06-04 07:10:50 by iq iq
That's an epic post, Iñigo. :D
added on the 2017-06-04 08:58:31 by ham ham
WHAT the HELL happened there, IQ?
added on the 2017-06-04 10:13:48 by Foebane72 Foebane72
iq: surely I know your shader for 3d quadratic bezier.

The thing is the question was about 2d and I believe sign might be useful there. While in 3d, sign doesn't make any sense.

BTW actually I'm pretty sure you don't need 3rd root, apparently it's never closer than other roots. I have no proof/reference "why", but I guess it's because there is only one critical point and consequently only two candidates for the closest point.
So, you could simplify your code a little bit to safe just a few cycles:
Code: // calculate 2 roots and just ignore the 3rd vec2 t = vec2(m + m, -n - m) * z - kx; t = clamp( t, 0.0, 1.0 ); vec3 dp1 = d + (c + b*t.x)*t.x; float d1 = dot(dp1,dp1); vec3 dp2 = d + (c + b*t.y)*t.y; float d2 = dot(dp2,dp2); res = (d1<d2) ? vec2(d1,t.x) : vec2(d2,t.y);


That's basically what I did already in a previous example (and here).

On a side note: are you obsessed by getting credit for all the code that is using 1st grade algebra or something? ;)
added on the 2017-06-04 11:36:57 by tomkh tomkh
If anyone cares: here is my derivation with a little hand-wavy explanation why 3rd root is unnecessary. The formulas obviously works for 3d as well, because it's just scalar products. It's really trivial/basic math.
added on the 2017-06-04 14:33:49 by tomkh tomkh
Quote:
Holly shit, that worked with the "preview" button, but not with the actual display.

Post size limits are the reason we have image hosting services :)
added on the 2017-06-04 18:07:23 by Gargaj Gargaj
Code:echo parse_message( substr( $_POST["message"], 0, 65535 ) );

So 64k ought to be enough for anybody? ;)
added on the 2017-06-04 20:28:21 by tomaes tomaes
an "Edit" button would be an interesting feature too...
added on the 2017-06-05 00:28:30 by Tigrou Tigrou
Except Gargaj loves to have "dirt" on anyone just in case they will run for presidency one day ;)
I would actually record keystrokes already in "post" window and whatever you type should be irreversibly recorded - posted on not!

But I understand - nobody is paid to develop this forum, so mostly kidding.
added on the 2017-06-05 00:58:00 by tomkh tomkh
tomkh, cool observation about the need for 2 roots only. Proving it will be a nice game to play some night. In practice it doesn't save that much performance, but eh, free performance is free and if it's more mathematically elegant, why not. Have you given the prove any try so far?
added on the 2017-06-06 05:29:16 by iq iq
Hey tomkh, how can I reach you out? I modified your shader a bit to show which regions of the plane has which root as closest point, and I see some interesting effects/patterns/errors_may_be. I sent you a message in twitter, let me know if there's a better way to reach you
added on the 2017-06-06 05:56:36 by iq iq
iq: for the proof, I would start simply with root finding formula and proof 3rd root is always between other two.
Just by looking at cubic formula, we can see that roots are eventually found as follows:
phi = acos( x )/3 for some "x", assuming "x" can be anywhere within -1..1 range, phi is between 0..pi/3 range,
t1 = s*cos(phi) - o, for some positive "s"
t2 = s*cos(phi + 2pi/3) - o
t3 = s*cos(phi + 4pi/3) - o
Now it is trivial (left as an excercise;)) to show that for all phi in 0..pi/3:
cos(phi) <= cos(phi + 4pi/3) <= cos(phi + 2pi/3)
Therefore t1 <= t3 <= t2.

So now we know t3 is always between t1 and t2. Now we have to show that f(t3) (point at a root) is never closer than f(t1) and f(t2). And here I don't know :)

Somehow we have to proof that for all t1,t2 and t3: f(t).f'(t)=0, where t in {t1,t2,t3} implies f(t1).f(t1) <= f(t3).f(t3) <= f(t2).f(t2). Maybe it's not that hard, but I need more time to sort it out.
added on the 2017-06-06 18:27:55 by tomkh tomkh

login