Friday, August 17, 2007

Mid-point Circle Scaling


/* WAP to generate a circle.
Write a function to scale the circle by sx amount in x-direction and sy amount in y-direction, where sx and sy are integers provided by the user.
Use matrix method. - Subhranath Chunder */

#include<stdio.h>
#include<conio.h>
#include<graphics.h>
#include<math.h>

void circleMidPoint(int,int,int,int,int);
int equ(float,float,int);
void setOthers(int,int,int,int,int,int);
void scalePoint(int,int,int,int,int*,int*);
void setPixel(int,int);

void main()
{
int x,y,radius,driver=DETECT,mode,sx,sy;
clrscr();

/* Mid-point and radius of the circle are accepted */
printf("Enter the co-ordinate of the centre: ");
scanf("%d %d",&x,&y);
printf("Enter the radius of the circle: ");
scanf("%d",&radius);
printf("<Press any key to continue>");

/* Graphics mode initialized */
initgraph(&driver,&mode,"F:\\tc\\bgi");

/* Circle drawn with no scaling */
circleMidPoint(x,y,radius,1,1);

/* Amount of scaling */
printf("Enter the amount of scaling in x-direction: ");
scanf("%d",&sx);
printf("Enter the amount of scaling in y-direction: ");
scanf("%d",&sy);

/* Translated circle is redrawn */
circleMidPoint(x,y,radius,sx,sy);
getch();
closegraph();
}

void circleMidPoint(int xCenter,int yCenter,int radius,int sx,int sy)
{
int x=0,y=radius;

/* Sets the points for all octants */
setOthers(x,y,xCenter,yCenter,sx,sy);

/* First octant is drawn */
while(x<y)
{
++x;

/* Checking whether y or y-1 is closer to the circle, by taking their mid-point */
if( equ(x,y-(float)1/2,radius) >=0 )
--y;

setOthers(x,y,xCenter,yCenter,sx,sy);
}

}

/* A function to find whether a given point is inside, outside, or on the circle */
int equ(float x,float y,int r)
{
int res;
if( pow(x,2)+pow(y,2)-pow(r,2) == 0)
res=0;
else if( pow(x,2)+pow(y,2)-pow(r,2) < 0)
res=-1;
else if( pow(x,2)+pow(y,2)-pow(r,2) > 0)
res=1;
return res;
}

/* A function to set all other points symmetric to the point (x,y) in all the octants */
void setOthers(int x,int y,int xCenter,int yCenter,int sx,int sy)
{
int xd,yd;

/* If sx==1 and sy==1 then no scaling is required. Otherwise, scale each point. */
if(sx==1 && sy==1)
{
setPixel(xCenter+x,yCenter+y);
setPixel(xCenter-x,yCenter+y);
setPixel(xCenter+x,yCenter-y);
setPixel(xCenter-x,yCenter-y);
setPixel(xCenter+y,yCenter+x);
setPixel(xCenter-y,yCenter+x);
setPixel(xCenter+y,yCenter-x);
setPixel(xCenter-y,yCenter-x);
}
else
{
scalePoint(xCenter+x,yCenter+y,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter-x,yCenter+y,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter+x,yCenter-y,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter-x,yCenter-y,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter+y,yCenter+x,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter-y,yCenter+x,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter+y,yCenter-x,sx,sy,&xd,&yd);
setPixel(xd,yd);
scalePoint(xCenter-y,yCenter-x,sx,sy,&xd,&yd);
setPixel(xd,yd);
}
}

/* Scales the point (x,y) by sx and sy amount, and stores the new point in (xd,yd) */
void scalePoint(int x,int y,int sx,int sy,int *xd,int *yd)
{
int i,j,a[3][3],b[3],c[3];

a[0][0]=sx; a[0][1]=0; a[0][2]=0;
a[1][0]=0; a[1][1]=sy; a[1][2]=0;
a[2][0]=0; a[2][1]=0; a[2][2]=1;

b[0]=x;
b[1]=y;
b[2]=1;

for(i=0;i<3;++i)
{
c[i]=0;
for(j=0;j<3;++j)
{
c[i]+=a[i][j]*b[j];
}
}

/* Scaled point */
*xd=c[0]/c[2];
*yd=c[1]/c[2];
}

void setPixel(int x,int y)
{
putpixel(x,y,2);
}





No comments: