[Back to GRAPHICS SWAG index]  [Back to Main SWAG index]  [Original]

{
From: SEAN PALMER
Subj: RIP Bezier Curve
---------------------------------------------------------------------------
 NO> Does anyone have any code for constructing a RIP Bezier curve that is
 NO> exactly the same as the one used by Telegrafix developers. I have some
 NO> code that comes close, but close isn't good enough. I need this to be
 NO> dead on accurate.
 NO> PS. I'm willing to share my code with others that are interested in
 NO> RIP.

{Public domain by Sean Palmer}
{converted from Steve Enns' original Basic subroutines by Sean Palmer}

var color:byte;
procedure plot(x,y:word);begin
 mem[$A000:y*320+x]:=color;
 end;

type
 coord=record x,y:integer; end;
 CurveDataRec=array[0..65521 div sizeof(coord)]of coord;

procedure drawBSpline(var d0:coord;nPoints,nSteps:word);
 const nsa=1/6; nsb=2/3;
 var
  i,i2,xx,yy:integer;
  t,ta,t2,t2a,t3,t3a,nc1,nc2,nc3,nc4,step:real;
  d:curveDataRec absolute d0;
begin
 step:=1/nSteps;
 for i:=0 to nPoints-4 do begin
  color:=i+32+2;
  t:=0.0;
  for i2:=pred(nSteps)downto 0 do begin
   t:=t+step;
   ta:=t*0.5; t2:=t*t; t2A:=t2*0.5; t3:=t2*t; t3A:=t3*0.5;
   nc1:=-nsa*t3+t2A-ta+nsa;
   nc2:=t3a-t2+nsb;
   nc3:=-t3a+t2a+ta+nsa;
   nc4:=nsa*t3;
   xx:=round(nc1*d[i].x+nc2*d[succ(i)].x+nc3*d[i+2].x+nc4*d[i+3].x);
   yy:=round(nc1*d[i].y+nc2*d[succ(i)].y+nc3*d[i+2].y+nc4*d[i+3].y);
   plot(xx,yy);
   end;
  end;
 end;

procedure drawBezier(var d0:coord;nPoints,nSteps:word);
 const nsa=1/6; nsb=2/3;
 var
  i,i2,i3,xx,yy:integer;
  t,tm3,t2,t2m3,t3,t3m3,nc1,nc2,nc3,nc4,step:real;
  d:curveDataRec absolute d0;
begin
 step:=1/nSteps;
 for i2:=0 to pred(nPoints) div 4 do begin
  i:=i2*4;
  t:=0.0;
  for i3:=pred(nSteps) downto 0 do begin
   t:=t+step;
   tm3:=t*3.0; t2:=t*t; t2m3:=t2*3.0; t3:=t2*t; t3m3:=t3*3.0;
   nc1:=1-tm3+t2m3-t3;
   nc2:=t3m3-2.0*t2m3+tm3;
   nc3:=t2m3-t3m3;
   nc4:=t3;

   xx:=round(nc1*d[i].x+nc2*d[succ(i)].x+nc3*d[i+2].x+nc4*d[i+3].x);
   yy:=round(nc1*d[i].y+nc2*d[succ(i)].y+nc3*d[i+2].y+nc4*d[i+3].y);
   plot(xx,yy);
   end;
  end;
 end;

const numpoints=40;

var c:array[-1..2+numPoints]of coord;
var i:integer;
begin
 asm mov ax,$13; int $10; end;  {init vga/mcga graphics}
 randomize;
 for i:=1 to numPoints do with c[i] do begin
  x:=i*(319 div numPoints);    {for precision demo}
 {x:=random(320);}             {for fun demo}
  y:=random(200);
  end;
 for i:=1 to numPoints div 2 do c[i*2+1].y:=c[i*2].y;    {fit closer}
 for i:=1 to numPoints do with c[i] do begin color:=i+32; plot(x,y); end;
 c[-1]:=c[1]; c[0]:=c[1];  {replicate end points so curves fit to input}
 c[numPoints+1]:=c[numPoints]; c[numPoints+2]:=c[numPoints];
 drawBSpline(c[-1],numPoints+4,256); {set third parm to 256 for precision, 64 f}
 readln;
 asm mov ax,3; int $10; end;  {text mode again}
 end.


[Back to GRAPHICS SWAG index]  [Back to Main SWAG index]  [Original]