/* -----------------------------------------------------------------------
* Από: migf1 Primitive Text Graphics Plotter v1.5
*
* Πολύ απλοϊκό παράδειγμα προγράμματος κονσόλας που παράγει στο 1ο καρτεσιανό
* τεταρτημόριο τα εξής:
* α) το γράφημα μιας συνάρτησης: f(x) = a*x + b
* β) μια γραμμή μεταξύ δυο σημείων: (x1,y1) και (x2,y2)
* γ) έναν κύκλο με κέντρο στο σημείο: (x,y) και ακτίνα radius
* δ) μια έλλειψη που οριοθετείται από παραλληλόγραμμο με αντικριστές κορυφές
* τα σημεία: (x1,y1) πάνω αριστερά και (x2,y2) κάτω δεξιά
*
* Μπορείτε να αλλάξετε τιμές σε οποιοδήποτε από τα x, pt2.x, x2, pt2.y, y2 & radius,
* αλλά σημειώστε πως το πρόγραμμα απεικονίζει μονάχα όσα σημεία παράγονται στην
* περιοχή: (0,0) ... (74,22)
*
* ΣΗΜΕΙΩΣΕΙΣ ΥΛΟΠΟΙΗΣΗΣ:
* Οι αλγόριθμοι για τον σχεδιασμό της γραμμής του κύκλου και της έλλειψης
* είναι ΕΤΟΙΜΕΣ υλοποιήσεις των αλγορίθμων Bresenham, που χρησιμοποιήσα
* ατόφιους (με μόνη αλλαγή την προσαρμογή τους στις δομές δεδομέννων Screen
* και Point του παρόντος προγράμματος).
*
* Δείτε επίσης:
* http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
* http://e...content-available-to-author-only...a.org/wiki/Midpoint_circle_algorithm
*
* Tο πρόγραμμα δεν κάνει ούτε scaling ούτε κανονικό clipping και απεικονίζει
* ΜΟΝΑΧΑ το 1ο τεταρτημόριο του καρτεσιανού επιπέδου, με το σημείο (0,0) να
* βρίσκεται στην κάτω αριστερή μεριά της οθόνης.
* Ο κύριος σκοπός για τον οποίον γράφτηκε το πρόγραμμα ήταν η παρουσίαση μιας
* απλοϊκής υλοποίησης buffered σχεδιασμού (οι συναρτήσεις αναβοσβήνουν τα
* κατάλληλα pixels μέσα στο buffer, το οποίο κατόπιν τυπώνεται στην οθόνη
* σειρά προς σειρά, μεσω της συνάρτησης: draw_screen() ).
* -----------------------------------------------------------------------
*/
#include <stdio.h>
#include <stdlib.h> // for exit()
#include <string.h> // for memset()
#include <ctype.h> // for tolower()
#define MAXCOLS 75 // μέγιστο πλάτος της οθόνης μας
#define MAXROWS 23 // μέγιστο ύψος της οθόνης μας
#define PIXEL_ON '#' // χαρακτήρας απεικόνισης "αναμμένων" pixels
#define PIXEL_OFF '.' // χαρακτήρας απεικόνισης "σβησμένων" pixels
// έλεγχος για pixels εκτός ορίων
#define VALIDPIXEL(x,y, w,h) ( (y) > -1 && (y) < (h) && (x) > -1 && (x) < (w) )
#define REVAXIS(y,h) (h) - (y) - 1 // προσαρμόζει την συντεταγμένη y ώστε το σημείο
// που της αντιστοιχεί να απεικονιστεί στην
// αντεστραμμένη φορά του άξονά (χρήσιμο για τον
// άξονα Υ που τον θέλουμε να μετράει από κάτω
// προς τα πάνω)
#define ABS(x) ( (x) < 0 ? -(x) : (x) )
#define PAUSE() do{\
char myinbuf[256]; \
printf("press ENTER to continue..."); \
fgets(myinbuf, 256, stdin); \
}while(0)
typedef struct point { // δομή καρτεσιανού σημείου
int x; // καρτεσιανή συντεταγμένη x του σημείου
int y; // καρτεσιανή συντεταγμένη y του σημείου
} Point;
typedef struct screen { // δομή για το buffer της οθόνης μας
int nrows; // τρέχον πλάτος της οθόνης μας
int ncols; // τρέχον ύψος της οθόνης μας
char pixel[MAXROWS][MAXCOLS]; // το buffer της οθόνης μας
} Screen;
// ------------------------------------------------------------------------------------
// set_pixel:
//
// "Ανάβει" το pixel x,y στο buffer της οθόνης. Θα εμφανιστεί αναμένο όταν αργότερα
// τυπωθεί το buffer μέσω της συνάρτησης: draw_screen
//
void set_pixel( Screen *screen, int x, int y )
{
// απόρριψη pixels που βγαίνουν εκτός οθόνης
if ( !VALIDPIXEL(x,y, screen->ncols, screen->nrows ) )
return;
// προσαρμογή του y ώστε το μέτρημα για την
y = REVAXIS(y, screen->nrows); // απεικόνιση να ξεκινά από κάτω προς τα επανω
screen->pixel[y][x] = PIXEL_ON;
return;
}
// ------------------------------------------------------------------------------------
// Η συνάρτηση που παράγει το 1ο γράφημα, δοκιμάστε και με άλλες τιμές
//
int f( int x, float a, float b )
{
return a*x + b;
}
// ------------------------------------------------------------------------------------
// plot_f():
//
// "Ανάβει" τα pixels που αντιστοιχούν στο γράφημα της συνάρτησης f(), στο buffer της
// οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω της συνάρτησης:
// draw_screen() )
//
void plot_f( Screen *screen, float a, float b )
{
register int x;
for (x=0; x < screen->ncols; x++)
set_pixel( screen, x, f(x, a, b) );
return;
}
// ------------------------------------------------------------------------------------
// plot_line():
// αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" τα pixels της γραμμής που οριοθετείται από τα σημεία pt1 και pt2, στο
// buffer της οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω
// της συνάρτησης: draw_screen() )
//
void plot_line( Screen *screen, Point pt1, Point pt2)
{
int dx = ABS( pt2.x - pt1.x );
int dy = -ABS( pt2.y - pt1.y );
int sx = (pt1.x < pt2.x) ? 1 : -1;
int sy = (pt1.y < pt2.y) ? 1 : -1;
int err = dx + dy, e2; // error value e_xy
for (;;)
{
set_pixel(screen, pt1.x, pt1.y);
if ( pt1.x == pt2.x && pt1.y == pt2.y )
break;
e2 = 2 * err;
if (e2 >= dy) { // e_xy + e_x > 0
err += dy;
pt1.x += sx;
}
if (e2 <= dx) { // e_xy + e_y < 0 */
err += dx;
pt1.y += sy;
}
}
return;
}
// ------------------------------------------------------------------------------------
// plot_circle():
// αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" τα pixels της περιφέρειας του κύκλου με κέντρο το σημείο: center και ακτίνα
// radius, στο buffer της οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το
// buffer μέσω της συνάρτησης: draw_screen() )
//
void plot_circle(Screen *screen, Point center, int radius)
{
int x = -radius, y = 0, err = 2-2*radius; // II. Quadrant
do {
set_pixel(screen, center.x-x, center.y+y); // I. Quadrant
set_pixel(screen, center.x-y, center.y-x); // II. Quadrant
set_pixel(screen, center.x+x, center.y-y); // III. Quadrant
set_pixel(screen, center.x+y, center.y+x); // IV. Quadrant
radius = err;
if (radius > x) // e_xy + e_x > 0
err += ++x * 2 + 1;
if (radius <= y)
err += ++y * 2 + 1; // e_xy + e_y < 0
} while (x < 0);
return;
}
// ------------------------------------------------------------------------------------
// plot_ellipse():
// αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" στο buffer της οθόνης τα pixels της περιφέρειας της έλλειψης που
// οριοθετείται από ένα παραλληλόγραμμο με την πάνω-αριστερή γωνία του στο
// σημείο: rsta και την κάτω-δεξιά γωνία στου στο σημείο: rend (τα pixels θσ
// εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω της συνάρτησης:
// draw_screen() )
//
void plot_ellipse( Screen *screen, Point rsta, Point rend )
{
// values of diameter
int a = ABS( rend.x - rsta.x );
int b = ABS( rend.y - rsta.y );
int b1 = b & 1;
// error increment
long dx = 4 * (1-a) * b * b;
long dy = 4 * (b1+1) *a * a;
// error of 1.step
long err = dx + dy + b1 * a * a;
long e2;
if (rsta.x > rend.x) { // if called with swapped points
rsta.x = rend.x;
rend.x += a;
}
if (rsta.y > rend.y) // ... exchange them
rsta.y = rend.y;
// starting pixel
rsta.y += (b+1)/2;
rend.y = rsta.y-b1;
a *= 8 * a;
b1 = 8 * b * b;
do {
set_pixel(screen, rend.x, rsta.y); // I. Quadrant
set_pixel(screen, rsta.x, rsta.y); // II. Quadrant
set_pixel(screen, rsta.x, rend.y); // III. Quadrant
set_pixel(screen, rend.x, rend.y); // IV. Quadrant
e2 = 2 * err;
if (e2 >= dx) { // x step
rsta.x++;
rend.x--;
err += dx += b1;
}
if (e2 <= dy) { // y step
rsta.y++;
rend.y--;
err += dy += a;
}
} while ( rsta.x <= rend.x );
while ( rsta.y - rend.y < b ) { // too early stop of flat ellipses a=1
set_pixel(screen, rsta.x - 1, rsta.y);// -> finish tip of ellipse
set_pixel(screen, rend.x + 1, rsta.y++);
set_pixel(screen, rsta.x - 1, rend.y);
set_pixel(screen, rend.x + 1, rend.y--);
}
return;
}
// ------------------------------------------------------------------------------------
// "Σβήνει" όλα τα pixels στο buffer της οθόνης (α εμφανιστούν σβησμένα όταν αργότερα
// τυπωθεί η οθόνη μέσω της συνάρτησης: draw_screen() )
//
void clear_screen( Screen *screen )
{
// γρήγορος κώδικας
memset( &screen
->pixel
, PIXEL_OFF
, screen
->nrows
* screen
->ncols
* sizeof(char));
/*** DISABLED: αργός εναλλακτικός κώδικας, αλλά ενδεχομένως πιο συμβατός
register int x,y;
for (y=0; y < screen->nrows; y++)
for (x=0; x < screen->ncols; x++)
screen->pixel[y][x] = PIXEL_OFF;
***/
return;
}
// ------------------------------------------------------------------------------------
// draw_screen():
//
// Τυπώνει όλα τα pixels του buffer (αναμμένα και σβηστά) καθώς και την αρίθμηση
// των αξόνων Χ και Υ
//
void draw_screen( Screen screen )
{
register int x, y;
for (y=0; y < screen.nrows; y++)
{
// τύπωμα ετικετών αριστερά από τον άξονα Υ
int ylab = y;
printf("%3d ", REVAXIS
(ylab
, screen.
nrows) );
// τύπωμα των pixels της y-οστής σειράς του buffer
for (x=0; x < screen.ncols; x++)
}
// τύπωμα ετικετών κάτω από τον άξονα Χ
for (x=0; x < screen.ncols; x++)
return;
}
// ------------------------------------------------------------------------------------
void show_help( void )
{
puts("\n\tHELP:\n\n\tA sample program in C for plotting primitive 2D text graphics."); puts("\tIt shows buffered plotting of a Line, a Circle, an Ellipse"); puts("\tand a function of the form: f(x) = a * x + b");
puts("\n\tIt also shows how to read input lines as strings, brake 'em"); puts("\tup to tokens, and convert those tokens to desirable numbers"); puts("\t(using atoi() and atof() for keeping things simple).");
puts("\n\tFor each plot you will be first asked for appropriate input:"); puts("\t-\ta and b, for the function"); puts("\t-\tstarting and ending points, for the line"); puts("\t-\tcenter point and radius, for the circle"); puts("\t-\ttop-left and right-bottom points for the ellipse"); puts("\t\t( they define the rectangle the ellipse \"lives in\" )");
puts("\n\tComas, dashes, parenthesis, spaces and tabs are ignored"); puts("\tduring the input. Any other non-numerical input is either"); puts("\tignored or cause the conversion to produce 0 as the result"); puts("\tof the converted numbers.");
puts("\n\tEXCEPTION: in the main menu, only the first input char is checked");
return;
}
// ------------------------------------------------------------------------------------
void show_menu( void )
{
puts("\n\n\n\n\n\n\n\n\n\n\n\n\n"); puts("\n\tPrimitive Text Graphics Plotter\n\t-------------------------------"); puts("\tf\t: plot F(x) = a*x + b"); puts("\tl\t: plot a Line"); puts("\tc\t: plot a Circle"); puts("\te\t: plot an Ellipse");
return;
}
// ------------------------------------------------------------------------------------
int s_tokenize(char *s, char *tokens[], int maxtokens, const char *delimiters)
{
register int i=0;
tokens
[0] = strtok(s
, delimiters
); if (tokens[0] == NULL)
return 0;
for (i
=1; i
< maxtokens
&& (tokens
[i
]=strtok(NULL
, delimiters
)) != NULL
; i
++);
return i;
}
// ------------------------------------------------------------------------------------
void exec_selected( Screen *screen, char choice )
{
Point ptsta, ptend;
float a, b; int radius, ntoks;
char input[255+1] = ""; char *tokens[4]; const char *delims = "\n\t (),-";
switch( choice )
{
case 'h': // HELP
show_help();
PAUSE();
break;
case 'f': // FUNCTION: y = a*x + b
puts("\n\tPlotting: y = a * x + b ...");
// διάβασμα του a
do {
fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 1, delims);
} while ( ntoks < 1);
// διάβασμα του b
do {
fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 1, delims);
} while ( ntoks < 1);
// σχεδιασμός του γραφήματος της συναρτησης
clear_screen( screen );
plot_f( screen, a, b);
draw_screen( *screen );
printf("\tFunction: y = %.2f * x + %.2f ... ", a
, b
); PAUSE();
break;
case 'l': // LINE
puts("\n\tPlotting a line...");
// διάβασμα αρχικού σημείου: ptsta
do {
printf("\tenter 1st point (x,y): "); fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 2, delims);
} while( ntoks < 2);
ptsta.
x = atoi( tokens
[0] ); ptsta.
y = atoi( tokens
[1] );
// διάβασμα τελικού σημείου: ptend
do {
printf("\tenter 2nd point (x,y): "); fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 2, delims);
} while( ntoks < 2);
ptend.
x = atoi( tokens
[0] ); ptend.
y = atoi( tokens
[1] );
// σχεδιασμός της γραμμής
clear_screen( screen );
plot_line( screen, ptsta, ptend );
draw_screen( *screen );
printf( "\tLine from (%d,%d) to (%d,%d) ... ", ptsta.x, ptsta.y, ptend.x, ptend.y );
PAUSE();
break;
case 'c': // CIRCLE
puts("\n\tPlotting a circle...");
// διάβασμα κέντρου: ptsta
do {
printf("\tenter center point (x,y): "); fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 2, delims);
} while( ntoks < 2);
ptsta.
x = atoi( tokens
[0] ); ptsta.
y = atoi( tokens
[1] );
// διάβασμα της ακτίνας: radius
do {
fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 1, delims);
} while ( ntoks < 1);
// σχεδιασμός της κύκλου
clear_screen( screen );
plot_circle( screen, ptsta, radius );
draw_screen( *screen );
printf( "\tCircle with center (%d,%d) & radius %d ... ", ptsta.x, ptsta.y, radius );
PAUSE();
break;
case 'e': // ELLIPSE
puts("\n\tPlotting an ellipse...");
// διάβασμα πάνω-αριστερής γωνίας: ptsta
do {
printf("\tenter top-left point (x,y): "); fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 2, delims);
} while( ntoks < 2);
ptsta.
x = atoi( tokens
[0] ); ptsta.
y = atoi( tokens
[1] );
// διάβασμα κάτω-δεξιάς γωνίας: ptend
do {
printf("\tenter bottom-right point (x,y): "); fgets(input
, 255+1, stdin
); ntoks = s_tokenize(input, tokens, 2, delims);
} while( ntoks < 2);
ptend.
x = atoi( tokens
[0] ); ptend.
y = atoi( tokens
[1] );
// σχεδιασμός της έλλειψης
clear_screen( screen );
plot_ellipse( screen, ptsta, ptend );
draw_screen( *screen );
printf( "\tEllipse from (%d,%d) to (%d,%d) ... ", ptsta.x, ptsta.y, ptend.x, ptend.y );
PAUSE();
break;
case '\n': // empty selection
break;
default:
puts("\t*** error: unrecognised selection, try again"); break;
}
return;
}
// ------------------------------------------------------------------------------------
int main( void )
{
char input[255+1] = "";
Screen screen = { .ncols = MAXCOLS, .nrows = MAXROWS };
do {
show_menu();
fgets(input
, 255+1, stdin
); exec_selected( &screen, *input );
} while ( *input != 'x' );
}
/* -----------------------------------------------------------------------
 * Από: migf1					Primitive Text Graphics Plotter v1.5
 *
 * Πολύ απλοϊκό παράδειγμα προγράμματος κονσόλας που παράγει στο 1ο καρτεσιανό
 * τεταρτημόριο τα εξής:
 * α)	το γράφημα μιας συνάρτησης: f(x) = a*x + b
 * β)	μια γραμμή μεταξύ δυο σημείων: (x1,y1) και (x2,y2)
 * γ)	έναν κύκλο με κέντρο στο σημείο: (x,y) και ακτίνα radius
 * δ)	μια έλλειψη που οριοθετείται από παραλληλόγραμμο με αντικριστές κορυφές
 *	τα σημεία: (x1,y1) πάνω αριστερά και (x2,y2) κάτω δεξιά
 *
 * Μπορείτε να αλλάξετε τιμές σε οποιοδήποτε από τα x, pt2.x, x2, pt2.y, y2 & radius,
 * αλλά σημειώστε πως το πρόγραμμα απεικονίζει μονάχα όσα σημεία παράγονται στην
 * περιοχή: (0,0) ... (74,22)
 *
 * ΣΗΜΕΙΩΣΕΙΣ ΥΛΟΠΟΙΗΣΗΣ:
 *	Οι αλγόριθμοι για τον σχεδιασμό της γραμμής του κύκλου και της έλλειψης
 *	είναι ΕΤΟΙΜΕΣ υλοποιήσεις των αλγορίθμων Bresenham, που χρησιμοποιήσα
 *	ατόφιους (με μόνη αλλαγή την προσαρμογή τους στις δομές δεδομέννων Screen
 *	και Point του παρόντος προγράμματος).
 *
 * 	Δείτε επίσης:
 *	http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
 *	http://e...content-available-to-author-only...a.org/wiki/Midpoint_circle_algorithm
 *
 *	Tο πρόγραμμα δεν κάνει ούτε scaling ούτε κανονικό clipping και απεικονίζει
 *	ΜΟΝΑΧΑ το 1ο τεταρτημόριο του καρτεσιανού επιπέδου, με το σημείο (0,0) να
 *	βρίσκεται στην κάτω αριστερή μεριά της οθόνης.
 *	Ο κύριος σκοπός για τον οποίον γράφτηκε το πρόγραμμα ήταν η παρουσίαση μιας
 *	απλοϊκής υλοποίησης buffered σχεδιασμού (οι συναρτήσεις αναβοσβήνουν τα
 *	κατάλληλα pixels μέσα στο buffer, το οποίο κατόπιν τυπώνεται στην οθόνη
 *	σειρά προς σειρά, μεσω της συνάρτησης: draw_screen() ).
 * -----------------------------------------------------------------------
 */

#include <stdio.h>
#include <stdlib.h>			// for exit()
#include <string.h>			// for memset()
#include <ctype.h>			// for tolower()

#define	MAXCOLS		75     		// μέγιστο πλάτος της οθόνης μας
#define MAXROWS		23		// μέγιστο ύψος της οθόνης μας

#define PIXEL_ON	'#'		// χαρακτήρας απεικόνισης "αναμμένων" pixels
#define PIXEL_OFF	'.' 		// χαρακτήρας απεικόνισης "σβησμένων" pixels

					// έλεγχος για pixels εκτός ορίων
#define	VALIDPIXEL(x,y, w,h)	( (y) > -1 && (y) < (h) && (x) > -1 && (x) < (w) )

#define	REVAXIS(y,h)	(h) - (y) - 1   // προσαρμόζει την συντεταγμένη y ώστε το σημείο
					// που της αντιστοιχεί να απεικονιστεί στην
					// αντεστραμμένη φορά του άξονά (χρήσιμο για τον
					// άξονα Υ που τον θέλουμε να μετράει από κάτω
					// προς τα πάνω)

#define ABS(x)		( (x) < 0 ? -(x) : (x) )

#define PAUSE() do{\
	char myinbuf[256]; \
	printf("press ENTER to continue...");	\
	fgets(myinbuf, 256, stdin); \
}while(0)


typedef struct point {			// δομή καρτεσιανού σημείου
	int x;					// καρτεσιανή συντεταγμένη x του σημείου
	int y;					// καρτεσιανή συντεταγμένη y του σημείου
} Point;

typedef struct screen {			// δομή για το buffer της οθόνης μας
	int nrows;				// τρέχον πλάτος της οθόνης μας
	int ncols;				// τρέχον ύψος της οθόνης μας
	char pixel[MAXROWS][MAXCOLS];		// το buffer της οθόνης μας
} Screen;

// ------------------------------------------------------------------------------------
// set_pixel:
//
// "Ανάβει" το pixel x,y στο buffer της οθόνης. Θα εμφανιστεί αναμένο όταν αργότερα
// τυπωθεί το buffer μέσω της συνάρτησης: draw_screen
//
void set_pixel( Screen *screen, int x, int y )
{
					// απόρριψη pixels που βγαίνουν εκτός οθόνης
	if ( !VALIDPIXEL(x,y, screen->ncols, screen->nrows ) )
		return;
					// προσαρμογή του y ώστε το μέτρημα για την
	y = REVAXIS(y, screen->nrows);	// απεικόνιση να ξεκινά από κάτω προς τα επανω
	screen->pixel[y][x] = PIXEL_ON;

	return;
}

// ------------------------------------------------------------------------------------
// Η συνάρτηση που παράγει το 1ο γράφημα, δοκιμάστε και με άλλες τιμές
//
int f( int x, float a, float b )
{
	return a*x + b;
}

// ------------------------------------------------------------------------------------
// plot_f():
//
// "Ανάβει" τα pixels που αντιστοιχούν στο γράφημα της συνάρτησης f(), στο buffer της
// οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω της συνάρτησης:
// draw_screen() )
//
void plot_f( Screen *screen, float a, float b )
{
	register int x;
	for (x=0; x < screen->ncols; x++)
		set_pixel( screen, x, f(x, a, b) );

	return;
}

// ------------------------------------------------------------------------------------
// plot_line():
//	αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" τα pixels της γραμμής που οριοθετείται από τα σημεία pt1 και pt2, στο
// buffer της οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω
// της συνάρτησης: draw_screen() )
//
void plot_line( Screen *screen, Point pt1, Point pt2)
{
	int dx =  ABS( pt2.x - pt1.x );
	int dy = -ABS( pt2.y - pt1.y );
	int sx = (pt1.x < pt2.x) ? 1 : -1;
	int sy = (pt1.y < pt2.y) ? 1 : -1; 
	int err = dx + dy, e2;				// error value e_xy

	for (;;)
	{
		set_pixel(screen, pt1.x, pt1.y);
		if ( pt1.x == pt2.x && pt1.y == pt2.y )
			break;
		e2 = 2 * err;
		if (e2 >= dy) {				// e_xy + e_x > 0
			err += dy;
			pt1.x += sx;
		} 
		if (e2 <= dx) {				// e_xy + e_y < 0 */
			err += dx;
			pt1.y += sy;
		}
	}

	return;
}

// ------------------------------------------------------------------------------------
// plot_circle():
//	αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" τα pixels της περιφέρειας του κύκλου με κέντρο το σημείο: center και ακτίνα
// radius, στο buffer της οθόνης (θα εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το
// buffer μέσω της συνάρτησης: draw_screen() )
//
void plot_circle(Screen *screen, Point center, int radius)
{
	int x = -radius, y = 0, err = 2-2*radius;		// II. Quadrant
	do {
		set_pixel(screen, center.x-x, center.y+y);	//   I. Quadrant
		set_pixel(screen, center.x-y, center.y-x);	//  II. Quadrant
		set_pixel(screen, center.x+x, center.y-y);	// III. Quadrant
		set_pixel(screen, center.x+y, center.y+x);	//  IV. Quadrant
		radius = err;
		if (radius >  x)				// e_xy + e_x > 0
			err += ++x * 2 + 1;
		if (radius <= y)
			err += ++y * 2 + 1;			// e_xy + e_y < 0
	} while (x < 0);

	return;
}

// ------------------------------------------------------------------------------------
// plot_ellipse():
//	αλγόριθμος: http://f...content-available-to-author-only...s.at/easyfilter/bresenham.html
//
// "Ανάβει" στο buffer της οθόνης τα pixels της περιφέρειας της έλλειψης που
// οριοθετείται από ένα παραλληλόγραμμο με την πάνω-αριστερή γωνία του στο
// σημείο: rsta και την κάτω-δεξιά γωνία στου στο σημείο: rend (τα pixels θσ
// εμφανιστούν αναμμένα όταν αργότερα τυπωθεί το buffer μέσω της συνάρτησης:
// draw_screen() )
//
void plot_ellipse( Screen *screen, Point rsta, Point rend )
{
	// values of diameter
	int a = ABS( rend.x - rsta.x );
	int b = ABS( rend.y - rsta.y );
	int b1 = b & 1;	

	// error increment
	long dx = 4 * (1-a) * b * b;
	long dy = 4 * (b1+1) *a * a;

	// error of 1.step
	long err = dx + dy + b1 * a * a;
	long e2;

	if (rsta.x > rend.x) {			// if called with swapped points
		rsta.x = rend.x;
		rend.x += a;
	}
	if (rsta.y > rend.y)				 // ... exchange them
		rsta.y = rend.y;

	// starting pixel
	rsta.y += (b+1)/2;
	rend.y = rsta.y-b1;

	a *= 8 * a;
	b1 = 8 * b * b;

	do {
		set_pixel(screen, rend.x, rsta.y);	//   I. Quadrant
		set_pixel(screen, rsta.x, rsta.y);	//  II. Quadrant
		set_pixel(screen, rsta.x, rend.y);	// III. Quadrant
		set_pixel(screen, rend.x, rend.y);	//  IV. Quadrant
		e2 = 2 * err;
		if (e2 >= dx) {				// x step
			rsta.x++;
			rend.x--;
			err += dx += b1;
		}
		if (e2 <= dy) {				// y step
			rsta.y++;
			rend.y--;
			err += dy += a;
		}
	} while ( rsta.x <= rend.x );

	while ( rsta.y - rend.y < b ) {	// too early stop of flat ellipses a=1
		set_pixel(screen, rsta.x - 1, rsta.y);// -> finish tip of ellipse
		set_pixel(screen, rend.x + 1, rsta.y++); 
		set_pixel(screen, rsta.x - 1, rend.y);
		set_pixel(screen, rend.x + 1, rend.y--); 
	}

	return;
}

// ------------------------------------------------------------------------------------
// "Σβήνει" όλα τα pixels στο buffer της οθόνης (α εμφανιστούν σβησμένα όταν αργότερα
// τυπωθεί η οθόνη μέσω της συνάρτησης: draw_screen() )
//
void clear_screen( Screen *screen )
{
	// γρήγορος κώδικας
	memset( &screen->pixel, PIXEL_OFF, screen->nrows * screen->ncols * sizeof(char));

/*** DISABLED: αργός εναλλακτικός κώδικας, αλλά ενδεχομένως πιο συμβατός
	register int x,y;

	for (y=0; y < screen->nrows; y++)
		for (x=0; x < screen->ncols; x++)
			screen->pixel[y][x] = PIXEL_OFF;
***/

	return;
}

// ------------------------------------------------------------------------------------
// draw_screen():
//
// Τυπώνει όλα τα pixels του buffer (αναμμένα και σβηστά) καθώς και την αρίθμηση
// των αξόνων Χ και Υ
//
void draw_screen( Screen screen )
{
	register int x, y;

	for (y=0; y < screen.nrows; y++)
	{
		// τύπωμα ετικετών αριστερά από τον άξονα Υ
		int ylab = y;
		printf("%3d ", REVAXIS(ylab, screen.nrows) );

		// τύπωμα των pixels της y-οστής σειράς του buffer
		for (x=0; x < screen.ncols; x++)
			putchar( screen.pixel[y][x] );
		putchar('\n');
	}

	// τύπωμα ετικετών κάτω από τον άξονα Χ
	printf("Y/X ");
	for (x=0; x < screen.ncols; x++)
		printf("%d", x%10);
	putchar('\n');

	return;
}

// ------------------------------------------------------------------------------------
void show_help( void )
{
	puts("\n\tHELP:\n\n\tA sample program in C for plotting primitive 2D text graphics.");
	puts("\tIt shows buffered plotting of a Line, a Circle, an Ellipse");
	puts("\tand a function of the form: f(x) = a * x + b");

	puts("\n\tIt also shows how to read input lines as strings, brake 'em");
	puts("\tup to tokens, and convert those tokens to desirable numbers");
	puts("\t(using atoi() and atof() for keeping things simple).");

	puts("\n\tFor each plot you will be first asked for appropriate input:");
	puts("\t-\ta and b, for the function");
	puts("\t-\tstarting and ending points, for the line");
	puts("\t-\tcenter point and radius, for the circle");
	puts("\t-\ttop-left and right-bottom points for the ellipse");
	puts("\t\t( they define the rectangle the ellipse \"lives in\" )");

	puts("\n\tComas, dashes, parenthesis, spaces and tabs are ignored");
	puts("\tduring the input. Any other non-numerical input is either");
	puts("\tignored or cause the conversion to produce 0 as the result");
	puts("\tof the converted numbers.");

	puts("\n\tEXCEPTION: in the main menu, only the first input char is checked");

	printf("\n\t");

	return;
}
// ------------------------------------------------------------------------------------
void show_menu( void )
{
	puts("\n\n\n\n\n\n\n\n\n\n\n\n\n");
	printf("%38s\n", "by migf1");
	puts("\n\tPrimitive Text Graphics Plotter\n\t-------------------------------");
	puts("\th\t: Help");
	puts("\tf\t: plot F(x) = a*x + b");
	puts("\tl\t: plot a Line");
	puts("\tc\t: plot a Circle");
	puts("\te\t: plot an Ellipse");
	puts("\tx\t: eXit");
	printf("\tselect: ");

	return;
}

// ------------------------------------------------------------------------------------
int s_tokenize(char *s, char *tokens[], int maxtokens, const char *delimiters)
{
	register int i=0;

	tokens[0] = strtok(s, delimiters);
	if (tokens[0] == NULL)
		return 0;
	for (i=1; i < maxtokens && (tokens[i]=strtok(NULL, delimiters)) != NULL; i++); 

	return i;
}

// ------------------------------------------------------------------------------------
void exec_selected( Screen *screen, char choice )
{
	Point ptsta, ptend;
	float a, b;	int radius, ntoks;
	char input[255+1] = "";  char *tokens[4];  const char *delims = "\n\t (),-";

	switch( choice )
	{
		case 'h':	// HELP
			show_help();
			PAUSE();
		break;

		case 'f':	// FUNCTION: y = a*x + b
			puts("\n\tPlotting: y = a * x + b ...");

			// διάβασμα του a
			do {
				printf("\tenter a: ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 1, delims);
			} while ( ntoks < 1);
			a = atof(input);

			// διάβασμα του b
			do {
				printf("\tenter b: ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 1, delims);
			} while ( ntoks < 1);
			b = atof(input);

			// σχεδιασμός του γραφήματος της συναρτησης
			clear_screen( screen );
			plot_f( screen, a, b);
			draw_screen( *screen );

			printf("\tFunction: y = %.2f * x + %.2f ... ", a, b);
			PAUSE();
		break;

		case 'l':	// LINE
			puts("\n\tPlotting a line...");

			// διάβασμα αρχικού σημείου: ptsta
			do {
				printf("\tenter 1st point (x,y): ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 2, delims);
			} while( ntoks < 2);
			ptsta.x = atoi( tokens[0] );
			ptsta.y = atoi( tokens[1] );

			// διάβασμα τελικού σημείου: ptend
			do {
			printf("\tenter 2nd point (x,y): ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 2, delims);
			} while( ntoks < 2);
			ptend.x = atoi( tokens[0] );
			ptend.y = atoi( tokens[1] );

			// σχεδιασμός της γραμμής
			clear_screen( screen );
			plot_line( screen, ptsta, ptend );
			draw_screen( *screen );

			printf(	"\tLine from (%d,%d) to (%d,%d) ... ",
				ptsta.x, ptsta.y, ptend.x, ptend.y );
			PAUSE();
		break;

		case 'c':	// CIRCLE
			puts("\n\tPlotting a circle...");

			// διάβασμα κέντρου: ptsta
			do {
				printf("\tenter center point (x,y): ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 2, delims);
			} while( ntoks < 2);
			ptsta.x = atoi( tokens[0] );
			ptsta.y = atoi( tokens[1] );

			// διάβασμα της ακτίνας: radius
			do {
				printf("\tenter radius: ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 1, delims);
			} while ( ntoks < 1);
			radius = atoi( input );

			// σχεδιασμός της κύκλου
			clear_screen( screen );
			plot_circle( screen, ptsta, radius );
			draw_screen( *screen );

			printf(	"\tCircle with center (%d,%d) & radius %d ... ",
				ptsta.x, ptsta.y, radius );
			PAUSE();

		break;

		case 'e':	// ELLIPSE
			puts("\n\tPlotting an ellipse...");

			// διάβασμα πάνω-αριστερής γωνίας: ptsta
			do {
				printf("\tenter top-left     point (x,y): ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 2, delims);
			} while( ntoks < 2);
			ptsta.x = atoi( tokens[0] );
			ptsta.y = atoi( tokens[1] );

			// διάβασμα κάτω-δεξιάς γωνίας: ptend
			do {
				printf("\tenter bottom-right point (x,y): ");
				fgets(input, 255+1, stdin);
				ntoks = s_tokenize(input, tokens, 2, delims);
			} while( ntoks < 2);
			ptend.x = atoi( tokens[0] );
			ptend.y = atoi( tokens[1] );

			// σχεδιασμός της έλλειψης
			clear_screen( screen );
			plot_ellipse( screen, ptsta, ptend );
			draw_screen( *screen );

			printf(	"\tEllipse from (%d,%d) to (%d,%d) ... ",
				ptsta.x, ptsta.y, ptend.x, ptend.y );
			PAUSE();
		break;

		case '\n': // empty selection
		break;

		default:
			puts("\t*** error: unrecognised selection, try again");
		break;
	}

	return;
}
// ------------------------------------------------------------------------------------
int main( void )
{
	char input[255+1] = "";
	Screen screen = { .ncols = MAXCOLS, .nrows = MAXROWS };

	do {
		show_menu();
		fgets(input, 255+1, stdin);
		if ( tolower(*input) != 'x' )
			exec_selected( &screen, *input );
	} while ( *input != 'x' );

	exit( EXIT_SUCCESS );
}
