rendereps.c
上传用户:xk288cn
上传日期:2007-05-28
资源大小:4876k
文件大小:20k
源码类别:

GIS编程

开发平台:

Visual C++

  1. /* Copyright (c) Mark J. Kilgard, 1997. */
  2. /* This program is freely distributable without licensing fees 
  3.    and is provided without guarantee or warrantee expressed or 
  4.    implied. This program is -not- in the public domain. */
  5. /* Example showing how to use OpenGL's feedback mode to capture
  6.    transformed vertices and output them as Encapsulated PostScript.
  7.    Handles limited hidden surface removal by sorting and does
  8.    smooth shading (albeit limited due to PostScript). */
  9. /* Compile: cc -o rendereps rendereps.c -lglut -lGLU -lGL -lXmu -lXext -lX11 -lm */
  10. #include <assert.h>
  11. #include <math.h>
  12. #include <stdio.h>
  13. #include <stdlib.h>
  14. #include <GL/glut.h>
  15. /* OpenGL's GL_3D_COLOR feedback vertex format. */
  16. typedef struct _Feedback3Dcolor {
  17.   GLfloat x;
  18.   GLfloat y;
  19.   GLfloat z;
  20.   GLfloat red;
  21.   GLfloat green;
  22.   GLfloat blue;
  23.   GLfloat alpha;
  24. } Feedback3Dcolor;
  25. int blackBackground = 0;  /* Initially use a white background. */
  26. int lighting = 0;       /* Initially disable lighting. */
  27. int polygonMode = 1;    /* Initially show wireframe. */
  28. int object = 1;         /* Initially show the torus. */
  29. GLfloat angle = 0.0;    /* Angle of rotation for object. */
  30. int moving, begin;      /* For interactive object rotation. */
  31. int size = 1;           /* Size of lines and points. */
  32. /* How many feedback buffer GLfloats each of the three objects need. */
  33. int objectComplexity[3] =
  34. {6000, 14000, 380000};  /* Teapot requires ~1.5 megabytes for
  35.                            its feedback results! */
  36. /* render gets called both by "display" (in OpenGL render mode)
  37.    and by "outputEPS" (in OpenGL feedback mode). */
  38. void
  39. render(void)
  40. {
  41.   glPushMatrix();
  42.   glRotatef(angle, 0.0, 1.0, 0.0);
  43.   switch (object) {
  44.   case 0:
  45.     glutSolidSphere(1.0, 10, 10);
  46.     break;
  47.   case 1:
  48.     glutSolidTorus(0.5, 1.0, 15, 15);
  49.     break;
  50.   case 2:
  51.     glutSolidTeapot(1.0);
  52.     break;
  53.   }
  54.   glPopMatrix();
  55. }
  56. void
  57. display(void)
  58. {
  59.   glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
  60.   render();
  61.   glutSwapBuffers();
  62. }
  63. void
  64. updateBackground(void)
  65. {
  66.   if (blackBackground) {
  67.     /* Clear to black. */
  68.     glClearColor(0.0, 0.0, 0.0, 1.0);
  69.   } else {
  70.     /* Clear to white. */
  71.     glClearColor(1.0, 1.0, 1.0, 1.0);
  72.   }
  73. }
  74. void
  75. updateLighting(void)
  76. {
  77.   if (lighting) {
  78.     glEnable(GL_LIGHTING);
  79.   } else {
  80.     glDisable(GL_LIGHTING);
  81.   }
  82. }
  83. void
  84. updatePolygonMode(void)
  85. {
  86.   switch (polygonMode) {
  87.   case 0:
  88.     glPolygonMode(GL_FRONT_AND_BACK, GL_POINT);
  89.     break;
  90.   case 1:
  91.     glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
  92.     break;
  93.   case 2:
  94.     glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
  95.     break;
  96.   }
  97. }
  98. /* Write contents of one vertex to stdout. */
  99. void
  100. print3DcolorVertex(GLint size, GLint * count,
  101.   GLfloat * buffer)
  102. {
  103.   int i;
  104.   printf("  ");
  105.   for (i = 0; i < 7; i++) {
  106.     printf("%4.2f ", buffer[size - (*count)]);
  107.     *count = *count - 1;
  108.   }
  109.   printf("n");
  110. }
  111. void
  112. printBuffer(GLint size, GLfloat * buffer)
  113. {
  114.   GLint count;
  115.   int token, nvertices;
  116.   count = size;
  117.   while (count) {
  118.     token = buffer[size - count];
  119.     count--;
  120.     switch (token) {
  121.     case GL_PASS_THROUGH_TOKEN:
  122.       printf("GL_PASS_THROUGH_TOKENn");
  123.       printf("  %4.2fn", buffer[size - count]);
  124.       count--;
  125.       break;
  126.     case GL_POINT_TOKEN:
  127.       printf("GL_POINT_TOKENn");
  128.       print3DcolorVertex(size, &count, buffer);
  129.       break;
  130.     case GL_LINE_TOKEN:
  131.       printf("GL_LINE_TOKENn");
  132.       print3DcolorVertex(size, &count, buffer);
  133.       print3DcolorVertex(size, &count, buffer);
  134.       break;
  135.     case GL_LINE_RESET_TOKEN:
  136.       printf("GL_LINE_RESET_TOKENn");
  137.       print3DcolorVertex(size, &count, buffer);
  138.       print3DcolorVertex(size, &count, buffer);
  139.       break;
  140.     case GL_POLYGON_TOKEN:
  141.       printf("GL_POLYGON_TOKENn");
  142.       nvertices = buffer[size - count];
  143.       count--;
  144.       for (; nvertices > 0; nvertices--) {
  145.         print3DcolorVertex(size, &count, buffer);
  146.       }
  147.     }
  148.   }
  149. }
  150. GLfloat pointSize;
  151. static char *gouraudtriangleEPS[] =
  152. {
  153.   "/bd{bind def}bind def /triangle { aload pop   setrgbcolor  aload pop 5 3",
  154.   "roll 4 2 roll 3 2 roll exch moveto lineto lineto closepath fill } bd",
  155.   "/computediff1 { 2 copy sub abs threshold ge {pop pop pop true} { exch 2",
  156.   "index sub abs threshold ge { pop pop true} { sub abs threshold ge } ifelse",
  157.   "} ifelse } bd /computediff3 { 3 copy 0 get 3 1 roll 0 get 3 1 roll 0 get",
  158.   "computediff1 {true} { 3 copy 1 get 3 1 roll 1 get 3 1 roll 1 get",
  159.   "computediff1 {true} { 3 copy 2 get 3 1 roll  2 get 3 1 roll 2 get",
  160.   "computediff1 } ifelse } ifelse } bd /middlecolor { aload pop 4 -1 roll",
  161.   "aload pop 4 -1 roll add 2 div 5 1 roll 3 -1 roll add 2 div 3 1 roll add 2",
  162.   "div 3 1 roll exch 3 array astore } bd /gouraudtriangle { computediff3 { 4",
  163.   "-1 roll aload 7 1 roll 6 -1 roll pop 3 -1 roll pop add 2 div 3 1 roll add",
  164.   "2 div exch 3 -1 roll aload 7 1 roll exch pop 4 -1 roll pop add 2 div 3 1",
  165.   "roll add 2 div exch 3 -1 roll aload 7 1 roll pop 3 -1 roll pop add 2 div 3",
  166.   "1 roll add 2 div exch 7 3 roll 10 -3 roll dup 3 index middlecolor 4 1 roll",
  167.   "2 copy middlecolor 4 1 roll 3 copy pop middlecolor 4 1 roll 13 -1 roll",
  168.   "aload pop 17 index 6 index 15 index 19 index 6 index 17 index 6 array",
  169.   "astore 10 index 10 index 14 index gouraudtriangle 17 index 5 index 17",
  170.   "index 19 index 5 index 19 index 6 array astore 10 index 9 index 13 index",
  171.   "gouraudtriangle 13 index 16 index 5 index 15 index 18 index 5 index 6",
  172.   "array astore 12 index 12 index 9 index gouraudtriangle 17 index 16 index",
  173.   "15 index 19 index 18 index 17 index 6 array astore 10 index 12 index 14",
  174.   "index gouraudtriangle 18 {pop} repeat } { aload pop 5 3 roll aload pop 7 3",
  175.   "roll aload pop 9 3 roll 4 index 6 index 4 index add add 3 div 10 1 roll 7",
  176.   "index 5 index 3 index add add 3 div 10 1 roll 6 index 4 index 2 index add",
  177.   "add 3 div 10 1 roll 9 {pop} repeat 3 array astore triangle } ifelse } bd",
  178.   NULL
  179. };
  180. GLfloat *
  181. spewPrimitiveEPS(FILE * file, GLfloat * loc)
  182. {
  183.   int token;
  184.   int nvertices, i;
  185.   GLfloat red, green, blue;
  186.   int smooth;
  187.   GLfloat dx, dy, dr, dg, db, absR, absG, absB, colormax;
  188.   int steps;
  189.   Feedback3Dcolor *vertex;
  190.   GLfloat xstep, ystep, rstep, gstep, bstep;
  191.   GLfloat xnext, ynext, rnext, gnext, bnext, distance;
  192.   token = *loc;
  193.   loc++;
  194.   switch (token) {
  195.   case GL_LINE_RESET_TOKEN:
  196.   case GL_LINE_TOKEN:
  197.     vertex = (Feedback3Dcolor *) loc;
  198.     dr = vertex[1].red - vertex[0].red;
  199.     dg = vertex[1].green - vertex[0].green;
  200.     db = vertex[1].blue - vertex[0].blue;
  201.     if (dr != 0 || dg != 0 || db != 0) {
  202.       /* Smooth shaded line. */
  203.       dx = vertex[1].x - vertex[0].x;
  204.       dy = vertex[1].y - vertex[0].y;
  205.       distance = sqrt(dx * dx + dy * dy);
  206.       absR = fabs(dr);
  207.       absG = fabs(dg);
  208.       absB = fabs(db);
  209. #define Max(a,b) (((a)>(b))?(a):(b))
  210. #define EPS_SMOOTH_LINE_FACTOR 0.06  /* Lower for better smooth 
  211.                                         lines. */
  212.       colormax = Max(absR, Max(absG, absB));
  213.       steps = Max(1.0, colormax * distance * EPS_SMOOTH_LINE_FACTOR);
  214.       xstep = dx / steps;
  215.       ystep = dy / steps;
  216.       rstep = dr / steps;
  217.       gstep = dg / steps;
  218.       bstep = db / steps;
  219.       xnext = vertex[0].x;
  220.       ynext = vertex[0].y;
  221.       rnext = vertex[0].red;
  222.       gnext = vertex[0].green;
  223.       bnext = vertex[0].blue;
  224.       /* Back up half a step; we want the end points to be
  225.          exactly the their endpoint colors. */
  226.       xnext -= xstep / 2.0;
  227.       ynext -= ystep / 2.0;
  228.       rnext -= rstep / 2.0;
  229.       gnext -= gstep / 2.0;
  230.       bnext -= bstep / 2.0;
  231.     } else {
  232.       /* Single color line. */
  233.       steps = 0;
  234.     }
  235.     fprintf(file, "%g %g %g setrgbcolorn",
  236.       vertex[0].red, vertex[0].green, vertex[0].blue);
  237.     fprintf(file, "%g %g moveton", vertex[0].x, vertex[0].y);
  238.     for (i = 0; i < steps; i++) {
  239.       xnext += xstep;
  240.       ynext += ystep;
  241.       rnext += rstep;
  242.       gnext += gstep;
  243.       bnext += bstep;
  244.       fprintf(file, "%g %g lineto stroken", xnext, ynext);
  245.       fprintf(file, "%g %g %g setrgbcolorn", rnext, gnext, bnext);
  246.       fprintf(file, "%g %g moveton", xnext, ynext);
  247.     }
  248.     fprintf(file, "%g %g lineto stroken", vertex[1].x, vertex[1].y);
  249.     loc += 14;          /* Each vertex element in the feedback
  250.                            buffer is 7 GLfloats. */
  251.     break;
  252.   case GL_POLYGON_TOKEN:
  253.     nvertices = *loc;
  254.     loc++;
  255.     vertex = (Feedback3Dcolor *) loc;
  256.     if (nvertices > 0) {
  257.       red = vertex[0].red;
  258.       green = vertex[0].green;
  259.       blue = vertex[0].blue;
  260.       smooth = 0;
  261.       for (i = 1; i < nvertices; i++) {
  262.         if (red != vertex[i].red || green != vertex[i].green || blue != vertex[i].blue) {
  263.           smooth = 1;
  264.           break;
  265.         }
  266.       }
  267.       if (smooth) {
  268.         /* Smooth shaded polygon; varying colors at vetices. */
  269.         /* Break polygon into "nvertices-2" triangle fans. */
  270.         for (i = 0; i < nvertices - 2; i++) {
  271.           fprintf(file, "[%g %g %g %g %g %g]",
  272.             vertex[0].x, vertex[i + 1].x, vertex[i + 2].x,
  273.             vertex[0].y, vertex[i + 1].y, vertex[i + 2].y);
  274.           fprintf(file, " [%g %g %g] [%g %g %g] [%g %g %g] gouraudtrianglen",
  275.             vertex[0].red, vertex[0].green, vertex[0].blue,
  276.             vertex[i + 1].red, vertex[i + 1].green, vertex[i + 1].blue,
  277.             vertex[i + 2].red, vertex[i + 2].green, vertex[i + 2].blue);
  278.         }
  279.       } else {
  280.         /* Flat shaded polygon; all vertex colors the same. */
  281.         fprintf(file, "newpathn");
  282.         fprintf(file, "%g %g %g setrgbcolorn", red, green, blue);
  283.         /* Draw a filled triangle. */
  284.         fprintf(file, "%g %g moveton", vertex[0].x, vertex[0].y);
  285.         for (i = 1; i < nvertices; i++) {
  286.           fprintf(file, "%g %g lineton", vertex[i].x, vertex[i].y);
  287.         }
  288.         fprintf(file, "closepath fillnn");
  289.       }
  290.     }
  291.     loc += nvertices * 7;  /* Each vertex element in the
  292.                               feedback buffer is 7 GLfloats. */
  293.     break;
  294.   case GL_POINT_TOKEN:
  295.     vertex = (Feedback3Dcolor *) loc;
  296.     fprintf(file, "%g %g %g setrgbcolorn", vertex[0].red, vertex[0].green, vertex[0].blue);
  297.     fprintf(file, "%g %g %g 0 360 arc fillnn", vertex[0].x, vertex[0].y, pointSize / 2.0);
  298.     loc += 7;           /* Each vertex element in the feedback
  299.                            buffer is 7 GLfloats. */
  300.     break;
  301.   default:
  302.     /* XXX Left as an excersie to the reader. */
  303.     printf("Incomplete implementation.  Unexpected token (%d).n", token);
  304.     exit(1);
  305.   }
  306.   return loc;
  307. }
  308. void
  309. spewUnsortedFeedback(FILE * file, GLint size, GLfloat * buffer)
  310. {
  311.   GLfloat *loc, *end;
  312.   loc = buffer;
  313.   end = buffer + size;
  314.   while (loc < end) {
  315.     loc = spewPrimitiveEPS(file, loc);
  316.   }
  317. }
  318. typedef struct _DepthIndex {
  319.   GLfloat *ptr;
  320.   GLfloat depth;
  321. } DepthIndex;
  322. static int
  323. compare(const void *a, const void *b)
  324. {
  325.   DepthIndex *p1 = (DepthIndex *) a;
  326.   DepthIndex *p2 = (DepthIndex *) b;
  327.   GLfloat diff = p2->depth - p1->depth;
  328.   if (diff > 0.0) {
  329.     return 1;
  330.   } else if (diff < 0.0) {
  331.     return -1;
  332.   } else {
  333.     return 0;
  334.   }
  335. }
  336. void
  337. spewSortedFeedback(FILE * file, GLint size, GLfloat * buffer)
  338. {
  339.   int token;
  340.   GLfloat *loc, *end;
  341.   Feedback3Dcolor *vertex;
  342.   GLfloat depthSum;
  343.   int nprimitives, item;
  344.   DepthIndex *prims;
  345.   int nvertices, i;
  346.   end = buffer + size;
  347.   /* Count how many primitives there are. */
  348.   nprimitives = 0;
  349.   loc = buffer;
  350.   while (loc < end) {
  351.     token = *loc;
  352.     loc++;
  353.     switch (token) {
  354.     case GL_LINE_TOKEN:
  355.     case GL_LINE_RESET_TOKEN:
  356.       loc += 14;
  357.       nprimitives++;
  358.       break;
  359.     case GL_POLYGON_TOKEN:
  360.       nvertices = *loc;
  361.       loc++;
  362.       loc += (7 * nvertices);
  363.       nprimitives++;
  364.       break;
  365.     case GL_POINT_TOKEN:
  366.       loc += 7;
  367.       nprimitives++;
  368.       break;
  369.     default:
  370.       /* XXX Left as an excersie to the reader. */
  371.       printf("Incomplete implementation.  Unexpected token (%d).n",
  372.         token);
  373.       exit(1);
  374.     }
  375.   }
  376.   /* Allocate an array of pointers that will point back at
  377.      primitives in the feedback buffer.  There will be one
  378.      entry per primitive.  This array is also where we keep the
  379.      primitive's average depth.  There is one entry per
  380.      primitive  in the feedback buffer. */
  381.   prims = (DepthIndex *) malloc(sizeof(DepthIndex) * nprimitives);
  382.   item = 0;
  383.   loc = buffer;
  384.   while (loc < end) {
  385.     prims[item].ptr = loc;  /* Save this primitive's location. */
  386.     token = *loc;
  387.     loc++;
  388.     switch (token) {
  389.     case GL_LINE_TOKEN:
  390.     case GL_LINE_RESET_TOKEN:
  391.       vertex = (Feedback3Dcolor *) loc;
  392.       depthSum = vertex[0].z + vertex[1].z;
  393.       prims[item].depth = depthSum / 2.0;
  394.       loc += 14;
  395.       break;
  396.     case GL_POLYGON_TOKEN:
  397.       nvertices = *loc;
  398.       loc++;
  399.       vertex = (Feedback3Dcolor *) loc;
  400.       depthSum = vertex[0].z;
  401.       for (i = 1; i < nvertices; i++) {
  402.         depthSum += vertex[i].z;
  403.       }
  404.       prims[item].depth = depthSum / nvertices;
  405.       loc += (7 * nvertices);
  406.       break;
  407.     case GL_POINT_TOKEN:
  408.       vertex = (Feedback3Dcolor *) loc;
  409.       prims[item].depth = vertex[0].z;
  410.       loc += 7;
  411.       break;
  412.     default:
  413.       /* XXX Left as an excersie to the reader. */
  414.       assert(1);
  415.     }
  416.     item++;
  417.   }
  418.   assert(item == nprimitives);
  419.   /* Sort the primitives back to front. */
  420.   qsort(prims, nprimitives, sizeof(DepthIndex), compare);
  421.   /* XXX Understand that sorting by a primitives average depth
  422.      doesn't allow us to disambiguate some cases like self
  423.      intersecting polygons.  Handling these cases would require
  424.      breaking up the primitives.  That's too involved for this
  425.      example.  Sorting by depth is good enough for lots of
  426.      applications. */
  427.   /* Emit the Encapsulated PostScript for the primitives in
  428.      back to front order. */
  429.   for (item = 0; item < nprimitives; item++) {
  430.     (void) spewPrimitiveEPS(file, prims[item].ptr);
  431.   }
  432.   free(prims);
  433. }
  434. #define EPS_GOURAUD_THRESHOLD 0.1  /* Lower for better (slower) 
  435.                                       smooth shading. */
  436. void
  437. spewWireFrameEPS(FILE * file, int doSort, GLint size, GLfloat * buffer, char *creator)
  438. {
  439.   GLfloat clearColor[4], viewport[4];
  440.   GLfloat lineWidth;
  441.   int i;
  442.   /* Read back a bunch of OpenGL state to help make the EPS
  443.      consistent with the OpenGL clear color, line width, point
  444.      size, and viewport. */
  445.   glGetFloatv(GL_VIEWPORT, viewport);
  446.   glGetFloatv(GL_COLOR_CLEAR_VALUE, clearColor);
  447.   glGetFloatv(GL_LINE_WIDTH, &lineWidth);
  448.   glGetFloatv(GL_POINT_SIZE, &pointSize);
  449.   /* Emit EPS header. */
  450.   fputs("%!PS-Adobe-2.0 EPSF-2.0n", file);
  451.   /* Notice %% for a single % in the fprintf calls. */
  452.   fprintf(file, "%%%%Creator: %s (using OpenGL feedback)n", creator);
  453.   fprintf(file, "%%%%BoundingBox: %g %g %g %gn",
  454.     viewport[0], viewport[1], viewport[2], viewport[3]);
  455.   fputs("%%EndCommentsn", file);
  456.   fputs("n", file);
  457.   fputs("gsaven", file);
  458.   fputs("n", file);
  459.   /* Output Frederic Delhoume's "gouraudtriangle" PostScript
  460.      fragment. */
  461.   fputs("% the gouraudtriangle PostScript fragement below is freen", file);
  462.   fputs("% written by Frederic Delhoume (delhoume@ilog.fr)n", file);
  463.   fprintf(file, "/threshold %g defn", EPS_GOURAUD_THRESHOLD);
  464.   for (i = 0; gouraudtriangleEPS[i]; i++) {
  465.     fprintf(file, "%sn", gouraudtriangleEPS[i]);
  466.   }
  467.   fprintf(file, "n%g setlinewidthn", lineWidth);
  468.   /* Clear the background like OpenGL had it. */
  469.   fprintf(file, "%g %g %g setrgbcolorn",
  470.     clearColor[0], clearColor[1], clearColor[2]);
  471.   fprintf(file, "%g %g %g %g rectfillnn",
  472.     viewport[0], viewport[1], viewport[2], viewport[3]);
  473.   if (doSort) {
  474.     spewSortedFeedback(file, size, buffer);
  475.   } else {
  476.     spewUnsortedFeedback(file, size, buffer);
  477.   }
  478.   /* Emit EPS trailer. */
  479.   fputs("grestorenn", file);
  480.   fputs("%Add `showpage' to the end of this file to be able to print to a printer.n",
  481.     file);
  482.   fclose(file);
  483. }
  484. void
  485. outputEPS(int size, int doSort, char *filename)
  486. {
  487.   GLfloat *feedbackBuffer;
  488.   GLint returned;
  489.   FILE *file;
  490.   feedbackBuffer = calloc(size, sizeof(GLfloat));
  491.   glFeedbackBuffer(size, GL_3D_COLOR, feedbackBuffer);
  492.   (void) glRenderMode(GL_FEEDBACK);
  493.   render();
  494.   returned = glRenderMode(GL_RENDER);
  495.   if (filename) {
  496.     file = fopen(filename, "w");
  497.     if (file) {
  498.       spewWireFrameEPS(file, doSort, returned, feedbackBuffer, "rendereps");
  499.     } else {
  500.       printf("Could not open %sn", filename);
  501.     }
  502.   } else {
  503.     /* Helps debugging to be able to see the decode feedback
  504.        buffer as text. */
  505.     printBuffer(returned, feedbackBuffer);
  506.   }
  507.   free(feedbackBuffer);
  508. }
  509. void
  510. choice(int value)
  511. {
  512.   switch (value) {
  513.   case 0:
  514.     glutSetCursor(GLUT_CURSOR_WAIT);
  515.     outputEPS(objectComplexity[object], 1, "render.eps");
  516.     glutSetCursor(GLUT_CURSOR_INHERIT);
  517.     break;
  518.   case 1:
  519.     glutSetCursor(GLUT_CURSOR_WAIT);
  520.     outputEPS(objectComplexity[object], 0, "render.eps");
  521.     glutSetCursor(GLUT_CURSOR_INHERIT);
  522.     break;
  523.   case 2:
  524.     /* Try to start GNU "ghostview" to preview the EPS. */
  525.     system("ghostview render.eps &");
  526.     break;
  527.   case 3:
  528.     glutSetCursor(GLUT_CURSOR_WAIT);
  529.     outputEPS(objectComplexity[object], 0, NULL);
  530.     glutSetCursor(GLUT_CURSOR_INHERIT);
  531.     break;
  532.   case 4:
  533.     blackBackground = 1 - blackBackground;
  534.     updateBackground();
  535.     glutPostRedisplay();
  536.     break;
  537.   case 5:
  538.     lighting = 1 - lighting;
  539.     updateLighting();
  540.     glutPostRedisplay();
  541.     break;
  542.   case 6:
  543.     polygonMode = (polygonMode + 1) % 3;
  544.     updatePolygonMode();
  545.     glutPostRedisplay();
  546.     break;
  547.   case 7:
  548.     size = (size % 5) + 1;
  549.     glLineWidth(size);
  550.     glPointSize(size);
  551.     glutPostRedisplay();
  552.     break;
  553.   case 8:
  554.     object = (object + 1) % 3;
  555.     glutPostRedisplay();
  556.     break;
  557.   case 666:
  558.     exit(0);
  559.     break;
  560.   }
  561. }
  562. /* ARGSUSED2 */
  563. void
  564. mouse(int button, int state, int x, int y)
  565. {
  566.   if (button == GLUT_LEFT_BUTTON && state == GLUT_DOWN) {
  567.     moving = 1;
  568.     begin = x;
  569.   }
  570.   if (button == GLUT_LEFT_BUTTON && state == GLUT_UP) {
  571.     moving = 0;
  572.   }
  573. }
  574. /* ARGSUSED1 */
  575. void
  576. motion(int x, int y)
  577. {
  578.   if (moving) {
  579.     angle = angle + (x - begin);
  580.     begin = x;
  581.     glutPostRedisplay();
  582.   }
  583. }
  584. GLfloat light_diffuse[] =
  585. {0.0, 1.0, 0.0, 1.0};   /* Green light. */
  586. GLfloat light_position[] =
  587. {1.0, 1.0, 1.0, 0.0};
  588. int
  589. main(int argc, char **argv)
  590. {
  591.   glutInit(&argc, argv);
  592.   glutInitDisplayMode(GLUT_DEPTH | GLUT_DOUBLE | GLUT_RGB);
  593.   glutCreateWindow("rendereps");
  594.   glutDisplayFunc(display);
  595.   glutMouseFunc(mouse);
  596.   glutMotionFunc(motion);
  597.   glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse);
  598.   glLightfv(GL_LIGHT0, GL_POSITION, light_position);
  599.   glEnable(GL_LIGHT0);
  600.   glMatrixMode(GL_PROJECTION);
  601.   gluPerspective( /* field of view in degree */ 22.0,
  602.   /* aspect ratio */ 1.0,
  603.     /* Z near */ 5.0, /* Z far */ 10.0);
  604.   glMatrixMode(GL_MODELVIEW);
  605.   gluLookAt(0.0, 0.0, 5.0,  /* eye is at (0,0,5) */
  606.     0.0, 0.0, 0.0,      /* center is at (0,0,0) */
  607.     0.0, 1.0, 0.);      /* up is in postivie Y direction */
  608.   glTranslatef(0.0, 0.0, -3.0);
  609.   /* Give the object an "interesting" orientation. */
  610.   glRotatef(25, 1.0, 0.0, 0.0);
  611.   glutCreateMenu(choice);
  612.   glutAddMenuEntry("Write out Encapsulated PS (sorted)", 0);
  613.   glutAddMenuEntry("Write out Encapsulated PS (UNsorted)", 1);
  614.   glutAddMenuEntry("Spawn ghostview to view EPS", 2);
  615.   glutAddMenuEntry("Display feedback buffer", 3);
  616.   glutAddMenuEntry("Toggle black/white background", 4);
  617.   glutAddMenuEntry("Toggle lighting", 5);
  618.   glutAddMenuEntry("Switch fill mode (line, poly, point)", 6);
  619.   glutAddMenuEntry("Switch line/point size", 7);
  620.   glutAddMenuEntry("Switch object", 8);
  621.   glutAddMenuEntry("Quit", 666);
  622.   glutAttachMenu(GLUT_RIGHT_BUTTON);
  623.   updateBackground();
  624.   updateLighting();
  625.   updatePolygonMode();
  626.   glEnable(GL_DEPTH_TEST);
  627.   glColor3f(1.0, 0.0, 0.0);  /* Geometry should appear red. */
  628.   glutMainLoop();
  629.   return 0;             /* ANSI C requires main to return int. */
  630. }