Bezier curves

Until now, we can draw straight lines for our polygons, but not curves. However, curves are everywhere, also in the typography. So we need a way to represent any type of curves mathematically in a way that the specification is both simple and performant.

Bevor the Bezier curves, costume designers and architects had this instrument called french curve or pistolet in french. It is a plastic template to draw curves of any radius. My architect parents had in their atelier.

After the World War II, the automobile industry and the fast growing economy asked for the fast development of new models. Pierre Bezier worked in the 1960s as engineer for Renault and developed Unisurf Cad, one of the earliest CAD softwares. The Peugeot 204 was the first car developed with this software using Bezier curves.

However, Bezier was not alone. The engineer Paul de Castelijau worked for Citroën and had the same idea. He invented cubic Bezier curves already in 1959.

It happens sometimes that multiple clever people have the same idea at the same time, because the idea is ready. If you want split the credits, Castelijau developed the astonishing simple algorithm we will use, while Bezier takes credits for that curve handles you see in each layout program.

So what is a cubic Bezier curve? The curve is defined by four points P0, P1, P2 and P3. Only P0 and P3 are on the curve. The other points P2 and P3 act like gravity points and form the vectors V0 and V3 with their counterpart points P0 and P3. The longer the vectors are and the more they point fare away from the opposite point, the more likely the curve will stay at the starting point.

100 100 moveto 100 150 200 150 200 100 curveto stroke
100 100 moveto 100 150 lineto 95 145 lineto 100 150 moveto 105 145 lineto stroke
200 100 moveto 200 150 lineto 195 145 lineto 200 150 moveto 205 145 lineto stroke
90 90 moveto (P0) show
190 90 moveto (P3) show
90 155 moveto (P1) show
190 155 moveto (P2) show

300 100 moveto 300 125 450 200 400 100 curveto stroke
300 100 moveto 300 125 lineto 295 120 lineto 300 125 moveto 305 120 lineto stroke
400 100 moveto 450 200 lineto 445 195 lineto 450 200 moveto 455 195 lineto stroke
290 90 moveto (P0) show
390 90 moveto (P3) show
290 130 moveto (P1) show
450 205 moveto (P2) show


showpage

De Caseljau showed that with a simple linear interpolation, a cubic Bezier curve could be split into two Bezier curves. If you do that recursively, you finish either with a straight line at least for the resolution we are drawing. So the algorithms converts the curve in a sequence of lines we can handle.

If the original points are P0, P1, P2, P3 then you calculate P4 as center of [P0 P1], P5 as center of [P1 P2], P6 as center of [P2 P3]. Then you calculate P7 as center of [P4 P5], P8 as center of [P5 P6]. Finally you calculate P9 as center of [P7 P8]. (The triple linear interpolation represents the cubic character of this curve.)

The you have to new curves:

  • P0 P4 P7 P9
  • P9 P8 P6 P3

P1 and P2 are out of the game

200 100 moveto 200 200 450 280 400 100 curveto stroke
200 100 moveto 200 200 lineto 195 195 lineto 200 200 moveto 205 195 lineto stroke
400 100 moveto 450 280 lineto 442 278 lineto 450 280 moveto 453 273 lineto stroke
190 90 moveto (P0) show
390 90 moveto (P3) show
190 210 moveto (P1) show
450 285 moveto (P2) show

0.7 setgray
200 200 moveto 450 280 lineto stroke
200 150 moveto 325 240 lineto stroke
325 240 moveto 425 190 lineto stroke
262.50 195 moveto 375 215 lineto stroke

0 setgray
180 150 moveto (P4) show
314 245 moveto (P5) show
430 185 moveto (P6) show
245 195 moveto (P7) show
380 215 moveto (P8) show
315 210 moveto (P9) show

showpage

We create a bezier function which takes a curve path element and returns an array of lines.


Javascript

We also rewrite scanfill as independent function.

Javascript

Rewrite the fill operator and add the curveto and rcurveto operator.

Javascript

Some drawings with curves

Javascript Editor
PostScript

Some letters with and without curves. The "O" looks rough because we do not have any antialiasing at all here.

Javascript Editor
PostScript

Bezier cannot draw every curve exactly, but it can imitate it very well. With 4 points, it can imitate a circle with 99% precision. The formula for circle is with n segments distance to the control points, is (4/3)*tan(pi/(2n)).

On the left, a rough cercle with only 2 points and on the right a quite perfect circle with 4 points.

Javascript Editor
PostScript

The current codebase has now 1285 lines ps20240722.js

My Journey to PostScript