/**
 * Test for exact point/line/polygon rasterization, or at least rasterization
 * that fits the tolerance of the OpenGL spec.
 *
 * Brian Paul
 * 9 Nov 2007
 */

/*
 * Notes:
 * - 'm' to cycle through point, hline, vline and quad drawing
 * - Use cursor keys to translate coordinates (z to reset)
 * - Resize window to check for proper rasterization
 * - Make sure your LCD is running in its native resolution
 *   
 * If translation is (0,0):
 *  a point will be drawn where x%2==0 and y%2==0,
 *  a horizontal line will be drawn where x%2==0,
 *  a vertical line will be drawn where y%2==0,
 *  for quads, pixels will be set where (x%4)!=3 and (y%4)!=3
 *
 * XXX todo: do glReadPixels and test that the results are what's expected.
 * Upon failure, iterate over sub-pixel translations to find the ideal offset.
 */


#include <stdio.h>
#include <stdlib.h>
#include <GL/glut.h>

static int Width = 400, Height = 400;
static int Win;
static float Xtrans = 0, Ytrans = 0;
static float Step = 0.125;

enum {
   POINTS,
   HLINES,
   VLINES,
   QUADS,
   NUM_MODES
};

static int Mode = POINTS;


static void
Draw(void)
{
   /* See the OpenGL Programming Guide, Appendix H, "OpenGL Correctness Tips"
    * for information about the 0.375 translation factor.
    */
   float tx = 0.375, ty = 0.375;
   int i, j;

   glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

   glPushMatrix();
   glTranslatef(tx + Xtrans, ty + Ytrans, 0);

   if (Mode == POINTS) {
      glBegin(GL_POINTS);
      for (j = 0; j < Height; j += 2) {
         for (i = 0; i < Width; i += 2) {
            glVertex2f(i, j);
         }
      }
      glEnd();
   }
   else if (Mode == HLINES) {
      glBegin(GL_LINES);
      for (i = 0; i < Height; i += 2) {
         glVertex2f(0,     i);
         glVertex2f(Width, i);
      }
      glEnd();
   }
   else if (Mode == VLINES) {
      glBegin(GL_LINES);
      for (i = 0; i < Width; i += 2) {
         glVertex2f(i, 0     );
         glVertex2f(i, Height);
      }
      glEnd();
   }
   else if (Mode == QUADS) {
      glBegin(GL_QUADS);
      for (j = 0; j < Height; j += 4) {
         for (i = 0; i < Width; i += 4) {
            glVertex2f(i,     j    );
            glVertex2f(i + 3, j    );
            glVertex2f(i + 3, j + 3);
            glVertex2f(i,     j + 3);
         }
      }
      glEnd();
   }

   glPopMatrix();

   glutSwapBuffers();
}


static void
Reshape(int width, int height)
{
   Width = width;
   Height = height;
   glViewport(0, 0, width, height);
   glMatrixMode(GL_PROJECTION);
   glLoadIdentity();
   glOrtho(0, width, 0, height, -1, 1);
   glMatrixMode(GL_MODELVIEW);
   glLoadIdentity();
}


static void
Key(unsigned char key, int x, int y)
{
   (void) x;
   (void) y;
   switch (key) {
   case 'm':
   case 'M':
      Mode = (Mode + 1) % NUM_MODES;
      break;
   case 'z':
   case 'Z':
      Xtrans = Ytrans = 0;
      printf("Translation: %f, %f\n", Xtrans, Ytrans);
      break;
   case 27:
      glutDestroyWindow(Win);
      exit(0);
      break;
   }
   glutPostRedisplay();
}


static void
SpecialKey(int key, int x, int y)
{
   (void) x;
   (void) y;
   switch (key) {
   case GLUT_KEY_UP:
      Ytrans += Step;
      break;
   case GLUT_KEY_DOWN:
      Ytrans -= Step;
      break;
   case GLUT_KEY_LEFT:
      Xtrans -= Step;
      break;
   case GLUT_KEY_RIGHT:
      Xtrans += Step;
      break;
   }
   glutPostRedisplay();
   printf("Translation: %f, %f\n", Xtrans, Ytrans);
}


static void
Init(void)
{
}


static void
Usage(void)
{
   printf("Keys:\n");
   printf("  up/down/left/right - translate by %f\n", Step);
   printf("  z - reset translation to zero\n");
   printf("  m - change rendering mode (points, hlines, vlines, quads)\n");
   printf("  Esc - exit\n");
}


int
main(int argc, char *argv[])
{
   glutInit(&argc, argv);
   glutInitWindowPosition(0, 0);
   glutInitWindowSize(Width, Height);
   glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE);
   Win = glutCreateWindow(argv[0]);
   glutReshapeFunc(Reshape);
   glutKeyboardFunc(Key);
   glutSpecialFunc(SpecialKey);
   glutDisplayFunc(Draw);
   Init();
   Usage();
   glutMainLoop();
   return 0;
}