]> git.mxchange.org Git - flightgear.git/blob - src/GUI/gui.cxx
d3df0a87cb9b09f55681a305f82598e005165ca5
[flightgear.git] / src / GUI / gui.cxx
1 /**************************************************************************
2  * gui.cxx
3  *
4  * Written 1998 by Durk Talsma, started Juni, 1998.  For the flight gear
5  * project.
6  *
7  * Additional mouse supported added by David Megginson, 1999.
8  *
9  * This program is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU General Public License as
11  * published by the Free Software Foundation; either version 2 of the
12  * License, or (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful, but
15  * WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program; if not, write to the Free Software
21  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22  *
23  * $Id$
24  **************************************************************************/
25
26
27 #ifdef HAVE_CONFIG_H
28 #  include <config.h>
29 #endif
30
31 #include <simgear/compiler.h>
32
33 #ifdef SG_MATH_EXCEPTION_CLASH
34 #  include <math.h>
35 #endif
36
37 #ifdef HAVE_WINDOWS_H
38 #  include <windows.h>
39 #endif
40
41 #include <GL/glut.h>
42 #include <GL/gl.h>
43
44 #if defined(FX) && defined(XMESA)
45 #  include <GL/xmesa.h>
46 #endif
47
48 #include STL_FSTREAM
49 #include STL_STRING
50
51 #include <stdlib.h>
52 #include <string.h>
53
54 #include <simgear/constants.h>
55 #include <simgear/debug/logstream.hxx>
56 #include <simgear/misc/sg_path.hxx>
57 #include <simgear/screen/screen-dump.hxx>
58
59 #include <Include/general.hxx>
60 #include <Aircraft/aircraft.hxx>
61 #include <Airports/simple.hxx>
62 #include <Autopilot/auto_gui.hxx>
63 #include <Autopilot/newauto.hxx>
64 #include <Cockpit/panel.hxx>
65 #include <Controls/controls.hxx>
66 #include <FDM/flight.hxx>
67 #include <Main/fg_init.hxx>
68 #include <Main/fg_io.hxx>
69 #include <Main/globals.hxx>
70 #include <Main/fg_props.hxx>
71 #include <Main/options.hxx>
72 #include <Main/viewmgr.hxx>
73
74 #ifdef FG_NETWORK_OLK
75 #include <NetworkOLK/network.h>
76 #endif
77    
78 #if defined( WIN32 ) && !defined( __CYGWIN__ ) && !defined(__MINGW32__)
79 #  include <simgear/screen/win32-printer.h>
80 #  include <simgear/screen/GlBitmaps.h>
81 #endif
82
83 #include "gui.h"
84 #include "gui_local.hxx"
85 #include "apt_dlg.hxx"
86 #include "net_dlg.hxx"
87 #include "sgVec3Slider.hxx"
88
89 SG_USING_STD(string);
90
91 #ifndef SG_HAVE_NATIVE_SGI_COMPILERS
92 SG_USING_STD(cout);
93 #endif
94
95 #if defined(_MSC_VER) || defined(__MINGW32__)
96 #define  snprintf    _snprintf
97 #endif   /* _MSC_VER */
98
99 // main.cxx hack, should come from an include someplace
100 extern void fgInitVisuals( void );
101 extern void fgReshape( int width, int height );
102 extern void fgRenderFrame( void );
103
104 puFont guiFnt = 0;
105 fntTexFont *guiFntHandle = 0;
106 int gui_menu_on = 0;
107 static puMenuBar    *mainMenuBar = 0;
108 //static puButton     *hideMenuButton = 0;
109
110 static puDialogBox  *dialogBox = 0;
111 static puFrame      *dialogFrame = 0;
112 static puText       *dialogBoxMessage = 0;
113 static puOneShot    *dialogBoxOkButton = 0;
114
115
116 static puDialogBox  *YNdialogBox = 0;
117 static puFrame      *YNdialogFrame = 0;
118 static puText       *YNdialogBoxMessage = 0;
119 static puOneShot    *YNdialogBoxOkButton = 0;
120 static puOneShot    *YNdialogBoxNoButton = 0;
121
122 static char msg_OK[]     = "OK";
123 static char msg_NO[]     = "NO";
124 static char msg_YES[]    = "YES";
125 static char msg_CANCEL[] = "Cancel";
126 static char msg_RESET[]  = "Reset";
127
128 char *gui_msg_OK;     // "OK"
129 char *gui_msg_NO;     // "NO"
130 char *gui_msg_YES;    // "YES"
131 char *gui_msg_CANCEL; // "CANCEL"
132 char *gui_msg_RESET;  // "RESET"
133
134 static char global_dialog_string[256];
135
136 // from cockpit.cxx
137 extern void fgLatLonFormatToggle( puObject *);
138
139
140 /* ================ General Purpose Functions ================ */
141
142 // Intercept the Escape Key
143 void ConfirmExitDialog(void)
144 {
145     FG_PUSH_PUI_DIALOG( YNdialogBox );
146 }
147
148 // General Purpose Message Box
149 void mkDialog (const char *txt)
150 {
151     strncpy(global_dialog_string, txt, 256);
152     dialogBoxMessage->setLabel(global_dialog_string);
153     FG_PUSH_PUI_DIALOG( dialogBox );
154 }
155
156 // Message Box to report an error.
157 void guiErrorMessage (const char *txt)
158 {
159     SG_LOG(SG_GENERAL, SG_ALERT, txt);
160     if (dialogBox != 0)
161       mkDialog(txt);
162 }
163
164 // Message Box to report a throwable (usually an exception).
165 void guiErrorMessage (const char *txt, const sg_throwable &throwable)
166 {
167     string msg = txt;
168     msg += '\n';
169     msg += throwable.getFormattedMessage();
170     if (throwable.getOrigin() != "") {
171       msg += "\n (reported by ";
172       msg += throwable.getOrigin();
173       msg += ')';
174     }
175     SG_LOG(SG_GENERAL, SG_ALERT, msg);
176     if (dialogBox != 0)
177       mkDialog(msg.c_str());
178 }
179
180 // Toggle the Menu and Mouse display state
181 void guiToggleMenu(void)
182 {
183     if( gui_menu_on ) {
184         // printf("Hiding Menu\n");
185         mainMenuBar->hide  ();
186 #if defined(WIN32_CURSOR_TWEAKS)
187         if( mouse_mode == MOUSE_POINTER )
188             TurnCursorOff();
189 #endif // #ifdef WIN32_CURSOR_TWEAKS
190     } else {
191         // printf("Showing Menu\n");
192         mainMenuBar->reveal();
193 #ifdef WIN32
194         TurnCursorOn();
195 #endif // #ifdef WIN32
196     }
197     gui_menu_on = ~gui_menu_on;
198 }
199     
200 /* -----------------------------------------------------------------------
201 the Gui callback functions 
202 ____________________________________________________________________*/
203
204
205 // Hier Neu :-) This is my newly added code
206 // Added by David Findlay <nedz@bigpond.com>
207 // on Sunday 3rd of December
208
209 // Start new Save Dialog Box
210 static puDialogBox     *SaveDialog = 0;
211 static puFrame         *SaveDialogFrame = 0;
212 static puText          *SaveDialogMessage = 0;
213 static puInput         *SaveDialogInput = 0;
214
215 static puOneShot       *SaveDialogOkButton = 0;
216 static puOneShot       *SaveDialogCancelButton = 0;
217 // static puOneShot       *SaveDialogResetButton = 0;
218
219 // Default save filename
220 static char saveFile[256] = "fgfs.sav";
221
222 // Cancel Button
223 void SaveDialogCancel(puObject *) {
224     FG_POP_PUI_DIALOG( SaveDialog );
225 }
226
227 // If press OK do this
228 void SaveDialogOk(puObject*) {
229
230     FG_POP_PUI_DIALOG( SaveDialog );
231
232     char *s;
233     SaveDialogInput->getValue(&s);
234
235     ofstream output(s);
236     cout << saveFile << endl;
237     if (output.good() && fgSaveFlight(output)) {
238         output.close();
239         mkDialog("Saved flight");
240         SG_LOG(SG_INPUT, SG_INFO, "Saved flight");
241     } else {
242         mkDialog("Cannot save flight");
243         SG_LOG(SG_INPUT, SG_ALERT, "Cannot save flight");
244     }
245 }
246
247 // Create Dialog
248 static void saveFlight(puObject *cv) {
249     SaveDialog = new puDialogBox (150, 50);
250     {
251         SaveDialogFrame   = new puFrame           (0,0,350, 150);
252         SaveDialogMessage = new puText            (
253                         (150 - puGetStringWidth( puGetDefaultLabelFont(), "File Name:" ) / 2), 110);
254         SaveDialogMessage ->    setLabel          ("File Name:");
255
256         SaveDialogInput   = new puInput           (50, 70, 300, 100);
257         SaveDialogInput   ->    setValue          (saveFile);
258         SaveDialogInput   ->    acceptInput();
259
260         SaveDialogOkButton     =  new puOneShot   (50, 10, 110, 50);
261         SaveDialogOkButton     ->     setLegend   (gui_msg_OK);
262         SaveDialogOkButton     ->     setCallback ( SaveDialogOk );
263         SaveDialogOkButton     ->     makeReturnDefault(TRUE);
264
265         SaveDialogCancelButton =  new puOneShot   (140, 10, 210, 50);
266         SaveDialogCancelButton ->     setLegend   (gui_msg_CANCEL);
267         SaveDialogCancelButton ->     setCallback ( SaveDialogCancel );
268     }
269     FG_FINALIZE_PUI_DIALOG( SaveDialog );
270    
271     SaveDialog -> reveal();
272 }
273
274 // Load Dialog Start
275 static puDialogBox     *LoadDialog = 0;
276 static puFrame         *LoadDialogFrame = 0;
277 static puText          *LoadDialogMessage = 0;
278 static puInput         *LoadDialogInput = 0;
279
280 static puOneShot       *LoadDialogOkButton = 0;
281 static puOneShot       *LoadDialogCancelButton = 0;
282 // static puOneShot       *LoadDialogResetButton = 0;
283
284 // Default load filename
285 static char loadFile[256] = "fgfs.sav";
286
287 // Do this if the person click okay
288 void LoadDialogOk(puObject *) {
289
290     FG_POP_PUI_DIALOG( LoadDialog );
291
292     char *l;
293     LoadDialogInput->getValue(&l);
294
295     ifstream input(l);
296     if (input.good() && fgLoadFlight(input)) {
297         input.close();
298         mkDialog("Loaded flight");
299         SG_LOG(SG_INPUT, SG_INFO, "Restored flight");
300     } else {
301         mkDialog("Failed to load flight");
302         SG_LOG(SG_INPUT, SG_ALERT, "Cannot load flight");
303     }
304 }
305
306 // Do this is the person presses cancel
307 void LoadDialogCancel(puObject *) {
308     FG_POP_PUI_DIALOG( LoadDialog );
309 }
310
311 // Create Load Dialog
312 static void loadFlight(puObject *cb)
313 {
314     LoadDialog = new puDialogBox (150, 50);
315     {
316         LoadDialogFrame   = new puFrame           (0,0,350, 150);
317         LoadDialogMessage = new puText            (
318                         (150 - puGetStringWidth( puGetDefaultLabelFont(), "File Name:" ) / 2), 110);
319         LoadDialogMessage ->    setLabel          ("File Name:");
320
321         LoadDialogInput   = new puInput           (50, 70, 300, 100);
322         LoadDialogInput   ->    setValue          (loadFile);
323         LoadDialogInput   ->    acceptInput();
324
325         LoadDialogOkButton     =  new puOneShot   (50, 10, 110, 50);
326         LoadDialogOkButton     ->     setLegend   (gui_msg_OK);
327         LoadDialogOkButton     ->     setCallback ( LoadDialogOk );
328         LoadDialogOkButton     ->     makeReturnDefault(TRUE);
329
330         LoadDialogCancelButton =  new puOneShot   (140, 10, 210, 50);
331         LoadDialogCancelButton ->     setLegend   (gui_msg_CANCEL);
332         LoadDialogCancelButton ->     setCallback ( LoadDialogCancel );
333     }
334     FG_FINALIZE_PUI_DIALOG( LoadDialog );
335    
336     LoadDialog -> reveal();
337 }
338
339 // This is the accessor function
340 void guiTogglePanel(puObject *cb)
341 {
342   if (fgGetBool("/sim/panel/visibility"))
343     fgSetBool("/sim/panel/visibility", false);
344   else
345     fgSetBool("/sim/panel/visibility", true);
346
347   fgReshape(fgGetInt("/sim/startup/xsize"),
348             fgGetInt("/sim/startup/ysize"));
349 }
350     
351 //void MenuHideMenuCb(puObject *cb)
352 void hideMenuCb (puObject *cb)
353 {
354     guiToggleMenu();
355 }
356
357 void goodBye(puObject *)
358 {
359     // SG_LOG( SG_INPUT, SG_ALERT,
360     //      "Program exiting normally at user request." );
361     cout << "Program exiting normally at user request." << endl;
362
363 #ifdef FG_NETWORK_OLK    
364     if ( fgGetBool("/sim/networking/network-olk") ) {
365         if ( net_is_registered == 0 ) fgd_send_com( "8", FGFS_host);
366     }
367 #endif
368
369     // close all external I/O connections
370     fgIOShutdownAll();
371
372     exit(0);
373 }
374
375
376 void goAwayCb (puObject *me)
377 {
378     FG_POP_PUI_DIALOG( dialogBox );
379 }
380
381 void mkDialogInit (void)
382 {
383     //  printf("mkDialogInit\n");
384     int x = (fgGetInt("/sim/startup/xsize")/2 - 400/2);
385     int y = (fgGetInt("/sim/startup/ysize")/2 - 100/2);
386     dialogBox = new puDialogBox (x, y); // 150, 50
387     {
388         dialogFrame = new puFrame (0,0,400,100);
389         dialogBoxMessage  =  new puText         (10, 70);
390         dialogBoxMessage  -> setLabel           ("");
391         dialogBoxOkButton =  new puOneShot      (180, 10, 240, 50);
392         dialogBoxOkButton -> setLegend          (gui_msg_OK);
393         dialogBoxOkButton -> makeReturnDefault  (TRUE );
394         dialogBoxOkButton -> setCallback        (goAwayCb);
395     }
396     FG_FINALIZE_PUI_DIALOG( dialogBox );
397 }
398
399 void MayBeGoodBye(puObject *)
400 {
401     ConfirmExitDialog(); 
402 }
403
404 void goAwayYesNoCb(puObject *me)
405 {
406     FG_POP_PUI_DIALOG( YNdialogBox);
407 }
408
409 void ConfirmExitDialogInit(void)
410 {
411     char msg[] = "Really Quit";
412     char *s;
413
414     //  printf("ConfirmExitDialogInit\n");
415     int len = 200 - puGetStringWidth( puGetDefaultLabelFont(), msg )/2;
416
417     int x = (fgGetInt("/sim/startup/xsize")/2 - 400/2);
418     int y = (fgGetInt("/sim/startup/ysize")/2 - 100/2);
419         
420     YNdialogBox = new puDialogBox (x, y); // 150, 50
421     //  YNdialogBox = new puDialogBox (150, 50);
422     {
423         YNdialogFrame = new puFrame (0,0,400, 100);
424         
425         YNdialogBoxMessage  =  new puText         (len, 70);
426         YNdialogBoxMessage  -> setDefaultValue    (msg);
427         YNdialogBoxMessage  -> getDefaultValue    (&s);
428         YNdialogBoxMessage  -> setLabel           (s);
429         
430         YNdialogBoxOkButton =  new puOneShot      (100, 10, 160, 50);
431         YNdialogBoxOkButton -> setLegend          (gui_msg_OK);
432         YNdialogBoxOkButton -> makeReturnDefault  (TRUE );
433         YNdialogBoxOkButton -> setCallback        (goodBye);
434         
435         YNdialogBoxNoButton =  new puOneShot      (240, 10, 300, 50);
436         YNdialogBoxNoButton -> setLegend          (gui_msg_NO);
437         YNdialogBoxNoButton -> setCallback        (goAwayYesNoCb);
438     }
439     FG_FINALIZE_PUI_DIALOG( YNdialogBox );
440 }
441
442 void notCb (puObject *)
443 {
444     mkDialog ("This function isn't implemented yet");
445 }
446
447 void helpCb (puObject *)
448 {
449     string command;
450         
451 #if defined(FX) && !defined(WIN32)
452 #  if defined(XMESA_FX_FULLSCREEN) && defined(XMESA_FX_WINDOW)
453     if ( global_fullscreen ) {
454         global_fullscreen = false;
455         XMesaSetFXmode( XMESA_FX_WINDOW );
456     }
457 #  endif
458 #endif
459         
460     SGPath path( globals->get_fg_root() );
461     path.append( "Docs/index.html" );
462         
463     string help_app = fgGetString("/sim/startup/browser-app");
464
465     if ( system("xwininfo -name Netscape > /dev/null 2>&1") == 0 ) {
466         command = help_app + " -remote \"openURL(" + path.str() + ")\"";
467     } else {
468         command = help_app + " " + path.str();
469     }
470 #if !defined(WIN32)
471     command += " &";
472 #endif
473         
474     system( command.c_str() );
475     mkDialog ("Help started in netscape window.");
476 }
477
478 #define TR_HIRES_SNAP
479 #if defined( TR_HIRES_SNAP)
480 #include <simgear/screen/tr.h>
481 extern void trRenderFrame( void );
482 extern void fgUpdateHUD( GLfloat x_start, GLfloat y_start,
483                          GLfloat x_end, GLfloat y_end );
484
485 void fgHiResDump()
486 {
487     FILE *f;
488     string message;
489     bool show_pu_cursor = false;
490     bool show_menu = false;
491     char *filename = new char [24];
492     static int count = 1;
493
494     int freeze = globals->get_freeze();
495     if(!freeze)
496         globals->set_freeze( true );
497
498     if(gui_menu_on) {
499         show_menu = true;
500         guiToggleMenu();
501     }
502         
503     if ( !puCursorIsHidden() ) {
504         show_pu_cursor = true;
505         puHideCursor();
506     }
507
508     fgInitVisuals();
509     fgReshape( fgGetInt("/sim/startup/xsize"),
510                fgGetInt("/sim/startup/ysize") );
511
512     // we need two render frames here to clear the menu and cursor
513     // ... not sure why but doing an extra fgRenderFrame() shouldn't
514     // hurt anything
515     fgRenderFrame();
516     fgRenderFrame();
517
518     // Make sure we have SSG projection primed for current view
519     glMatrixMode(GL_MODELVIEW);
520     glLoadIdentity();
521     ssgSetCamera( (sgVec4 *)globals->get_current_view()->get_VIEW() );
522     float fov = globals->get_current_view()->get_fov();
523     ssgSetFOV(fov, fov * globals->get_current_view()->get_win_ratio());
524     // ssgSetNearFar( 10.0f, 120000.0f );
525     ssgSetNearFar( 0.5f, 1200000.0f );
526
527         
528     // This ImageSize stuff is a temporary hack
529     // should probably use 128x128 tile size and
530     // support any image size
531
532     // This should be a requester to get multiplier from user
533     int multiplier = 3;
534     int width = fgGetInt("/sim/startup/xsize");
535     int height = fgGetInt("/sim/startup/ysize");
536         
537     /* allocate buffer large enough to store one tile */
538     GLubyte *tile = (GLubyte *)malloc(width * height * 3 * sizeof(GLubyte));
539     if (!tile) {
540         printf("Malloc of tile buffer failed!\n");
541         return;
542     }
543
544     int imageWidth  = multiplier*width;
545     int imageHeight = multiplier*height;
546
547     /* allocate buffer to hold a row of tiles */
548     GLubyte *buffer
549         = (GLubyte *)malloc(imageWidth * height * 3 * sizeof(GLubyte));
550     if (!buffer) {
551         free(tile);
552         printf("Malloc of tile row buffer failed!\n");
553         return;
554     }
555     TRcontext *tr = trNew();
556     trTileSize(tr, width, height, 0);
557     trTileBuffer(tr, GL_RGB, GL_UNSIGNED_BYTE, tile);
558     trImageSize(tr, imageWidth, imageHeight);
559     trRowOrder(tr, TR_TOP_TO_BOTTOM);
560     sgFrustum *frustum = ssgGetFrustum();
561     trFrustum(tr,
562               frustum->getLeft(), frustum->getRight(),
563               frustum->getBot(),  frustum->getTop(), 
564               frustum->getNear(), frustum->getFar());
565         
566     /* Prepare ppm output file */
567     while (count < 1000) {
568         snprintf(filename, 24, "fgfs-screen-%03d.ppm", count++);
569         if ( (f = fopen(filename, "r")) == NULL )
570             break;
571         fclose(f);
572     }
573     f = fopen(filename, "wb");
574     if (!f) {
575         printf("Couldn't open image file: %s\n", filename);
576         free(buffer);
577         free(tile);
578         return;
579     }
580     fprintf(f,"P6\n");
581     fprintf(f,"# ppm-file created by %s\n", "trdemo2");
582     fprintf(f,"%i %i\n", imageWidth, imageHeight);
583     fprintf(f,"255\n");
584
585     /* just to be safe... */
586     glPixelStorei(GL_PACK_ALIGNMENT, 1);
587
588     /* Because the HUD and Panel change the ViewPort we will
589      * need to handle some lowlevel stuff ourselves */
590     int ncols = trGet(tr, TR_COLUMNS);
591     int nrows = trGet(tr, TR_ROWS);
592
593     bool do_hud = fgGetBool("/sim/hud/visibility");
594     GLfloat hud_col_step = 640.0 / ncols;
595     GLfloat hud_row_step = 480.0 / nrows;
596         
597     bool do_panel = fgPanelVisible();
598     GLfloat panel_col_step = current_panel->getWidth() / ncols;
599     GLfloat panel_row_step = current_panel->getHeight() / nrows;
600         
601     /* Draw tiles */
602     int more = 1;
603     while (more) {
604         trBeginTile(tr);
605         int curColumn = trGet(tr, TR_CURRENT_COLUMN);
606         int curRow =  trGet(tr, TR_CURRENT_ROW);
607         trRenderFrame();
608         if ( do_hud )
609             fgUpdateHUD( curColumn*hud_col_step,      curRow*hud_row_step,
610                          (curColumn+1)*hud_col_step, (curRow+1)*hud_row_step );
611         if (do_panel)
612             current_panel->update( curColumn*panel_col_step, panel_col_step,
613                                    curRow*panel_row_step,    panel_row_step );
614         more = trEndTile(tr);
615
616         /* save tile into tile row buffer*/
617         int curTileWidth = trGet(tr, TR_CURRENT_TILE_WIDTH);
618         int bytesPerImageRow = imageWidth * 3*sizeof(GLubyte);
619         int bytesPerTileRow = (width) * 3*sizeof(GLubyte);
620         int xOffset = curColumn * bytesPerTileRow;
621         int bytesPerCurrentTileRow = (curTileWidth) * 3*sizeof(GLubyte);
622         int i;
623         for (i=0;i<height;i++) {
624             memcpy(buffer + i*bytesPerImageRow + xOffset, /* Dest */
625                    tile + i*bytesPerTileRow,              /* Src */
626                    bytesPerCurrentTileRow);               /* Byte count*/
627         }
628
629         if (curColumn == trGet(tr, TR_COLUMNS)-1) {
630             /* write this buffered row of tiles to the file */
631             int curTileHeight = trGet(tr, TR_CURRENT_TILE_HEIGHT);
632             int bytesPerImageRow = imageWidth * 3*sizeof(GLubyte);
633             int i;
634             for (i=0;i<curTileHeight;i++) {
635                 /* Remember, OpenGL images are bottom to top.  Have to reverse. */
636                 GLubyte *rowPtr = buffer + (curTileHeight-1-i) * bytesPerImageRow;
637                 fwrite(rowPtr, 1, imageWidth*3, f);
638             }
639         }
640
641     }
642
643     fgReshape( width, height );
644
645     trDelete(tr);
646
647     fclose(f);
648
649     message = "Snap shot saved to ";
650     message += filename;
651     mkDialog (message.c_str());
652
653     free(tile);
654     free(buffer);
655
656     //  message = "Snap shot saved to ";
657     //  message += filename;
658     //  mkDialog (message.c_str());
659
660     delete [] filename;
661
662     if( show_menu )
663         guiToggleMenu();
664
665     if ( show_pu_cursor ) {
666         puShowCursor();
667     }
668
669     if(!freeze)
670         globals->set_freeze( false );
671 }
672 #endif // #if defined( TR_HIRES_SNAP)
673
674
675 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
676
677 static void rotateView( double roll, double pitch, double yaw )
678 {
679         // rotate view
680 }
681
682 static GlBitmap *b1 = NULL;
683 extern FGInterface cur_view_fdm;
684 GLubyte *hiResScreenCapture( int multiplier )
685 {
686     float oldfov = fgGetDouble("/sim/field-of-view");
687     float fov = oldfov / multiplier;
688     FGViewer *v = globals->get_current_view();
689     fgSetDouble("/sim/field-of-view", fov);
690     fgInitVisuals();
691     int cur_width = fgGetInt("/sim/startup/xsize");
692     int cur_height = fgGetInt("/sim/startup/ysize");
693     if (b1) delete( b1 );
694     // New empty (mostly) bitmap
695     b1 = new GlBitmap( GL_RGB, 1, 1, (unsigned char *)"123" );
696     int x,y;
697     for ( y = 0; y < multiplier; y++ ) {
698         for ( x = 0; x < multiplier; x++ ) {
699             fgReshape( cur_width, cur_height );
700             // pan to tile
701             rotateView( 0, (y*fov)-((multiplier-1)*fov/2), (x*fov)-((multiplier-1)*fov/2) );
702             fgRenderFrame();
703             // restore view
704             GlBitmap b2;
705             b1->copyBitmap( &b2, cur_width*x, cur_height*y );
706         }
707     }
708     fgSetDouble("/sim/field-of-view", oldfov);
709     return b1->getBitmap();
710 }
711 #endif
712
713
714 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
715 // win32 print screen function
716 void printScreen ( puObject *obj ) {
717     bool show_pu_cursor = false;
718     TurnCursorOff();
719     if ( !puCursorIsHidden() ) {
720         show_pu_cursor = true;
721         puHideCursor();
722     }
723     BusyCursor( 0 );
724     mainMenuBar->hide();
725
726     CGlPrinter p( CGlPrinter::PRINT_BITMAP );
727     int cur_width = fgGetInt("/sim/startup/xsize");
728     int cur_height = fgGetInt("/sim/startup/ysize");
729     p.Begin( "FlightGear", cur_width*3, cur_height*3 );
730         p.End( hiResScreenCapture(3) );
731
732     if( gui_menu_on ) {
733         mainMenuBar->reveal();
734     }
735     BusyCursor(1);
736     if ( show_pu_cursor ) {
737         puShowCursor();
738     }
739     TurnCursorOn();
740 }
741 #endif // #ifdef WIN32
742
743
744 void dumpSnapShot ( puObject *obj ) {
745     fgDumpSnapShot();
746 }
747
748
749 void dumpHiResSnapShot ( puObject *obj ) {
750     fgHiResDump();
751 }
752
753
754 // do a screen snap shot
755 void fgDumpSnapShot () {
756     bool show_pu_cursor = false;
757     char *filename = new char [24];
758     string message;
759     static int count = 1;
760
761     int freeze = globals->get_freeze();
762     if(!freeze)
763         globals->set_freeze( true );
764
765     mainMenuBar->hide();
766     TurnCursorOff();
767     if ( !puCursorIsHidden() ) {
768         show_pu_cursor = true;
769         puHideCursor();
770     }
771
772     fgInitVisuals();
773     fgReshape( fgGetInt("/sim/startup/xsize"),
774                fgGetInt("/sim/startup/ysize") );
775
776     // we need two render frames here to clear the menu and cursor
777     // ... not sure why but doing an extra fgRenderFrame() shouldn't
778     // hurt anything
779     fgRenderFrame();
780     fgRenderFrame();
781
782     while (count < 1000) {
783         FILE *fp;
784         snprintf(filename, 24, "fgfs-screen-%03d.ppm", count++);
785         if ( (fp = fopen(filename, "r")) == NULL )
786             break;
787         fclose(fp);
788     }
789
790     if ( sg_glDumpWindow( filename,
791                           fgGetInt("/sim/startup/xsize"), 
792                           fgGetInt("/sim/startup/ysize")) ) {
793         message = "Snap shot saved to ";
794         message += filename;
795     } else {
796         message = "Failed to save to ";
797         message += filename;
798     }
799
800     mkDialog (message.c_str());
801
802     delete [] filename;
803
804     if ( show_pu_cursor ) {
805         puShowCursor();
806     }
807
808     TurnCursorOn();
809     if( gui_menu_on ) {
810         mainMenuBar->reveal();
811     }
812
813     if(!freeze)
814         globals->set_freeze( false );
815 }
816
817 #ifdef FG_NETWORK_OLK
818 static void net_display_toggle( puObject *cb)
819 {
820         net_hud_display = (net_hud_display) ? 0 : 1;
821         printf("Toggle net_hud_display : %d\n", net_hud_display);
822 }
823
824 static void net_register( puObject *cb)
825 {
826         fgd_send_com( "1", FGFS_host );
827         net_is_registered = 0;
828         printf("Registering to deamon\n");
829 }
830
831 static void net_unregister( puObject *cb)
832 {
833         fgd_send_com( "8", FGFS_host );
834         net_is_registered = -1;
835         printf("Unregistering from deamon\n");
836 }
837
838 extern void net_fgd_scan(puObject *cb);
839 #endif // #ifdef FG_NETWORK_OLK
840
841 /* -----------------------------------------------------------------------
842 The menu stuff 
843 ---------------------------------------------------------------------*/
844 char *fileSubmenu               [] = {
845     "Exit", /* "Close", "---------", */
846 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
847     "Print",
848 #endif
849     "Snap Shot",
850     "Hires Snap Shot",
851     "---------", 
852     "Reset", 
853     "Load flight",
854     "Save flight",
855     NULL
856 };
857 puCallback fileSubmenuCb        [] = {
858     MayBeGoodBye, /* hideMenuCb, NULL, */
859 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
860     printScreen, 
861 #endif
862     /* NULL, notCb, */
863     dumpSnapShot,
864     dumpHiResSnapShot,
865     NULL,
866     reInit, 
867     loadFlight,
868     saveFlight,
869     NULL
870 };
871
872 /*
873 char *editSubmenu               [] = {
874     "Edit text", NULL
875 };
876 puCallback editSubmenuCb        [] = {
877     notCb, NULL
878 };
879 */
880
881 extern void fgHUDalphaAdjust( puObject * );
882 char *viewSubmenu               [] = {
883     "HUD Alpha",
884     "Pilot Offset",
885     /* "Cockpit View > ", "View >","------------", */
886     "Toggle Panel...", NULL
887 };
888 puCallback viewSubmenuCb        [] = {
889     fgHUDalphaAdjust,
890     PilotOffsetAdjust,
891     /* notCb, notCb, NULL, */
892     guiTogglePanel, NULL
893 };
894
895 //  "---------", 
896
897 char *autopilotSubmenu           [] = {
898     "Toggle HUD Format", "Adjust AP Settings",
899     "---------", 
900     "Clear Route", "Skip Current Waypoint", "Add Waypoint",
901     "---------", 
902     "Set Altitude", "Set Heading",
903     NULL
904 };
905
906 puCallback autopilotSubmenuCb    [] = {
907     fgLatLonFormatToggle, fgAPAdjust,
908     NULL,
909     ClearRoute, PopWayPoint, AddWayPoint,
910     NULL,
911     NewAltitude, NewHeading,
912     /* notCb, */ NULL
913 };
914
915 char *environmentSubmenu        [] = {
916     "Goto Airport", /* "Terrain", "Weather", */ NULL
917 };
918 puCallback environmentSubmenuCb [] = {
919     NewAirport, /* notCb, notCb, */ NULL
920 };
921
922 /*
923 char *optionsSubmenu            [] = {
924     "Preferences", "Realism & Reliablity...", NULL
925 };
926 puCallback optionsSubmenuCb     [] = {
927     notCb, notCb, NULL
928 };
929 */
930
931 #ifdef FG_NETWORK_OLK
932 char *networkSubmenu            [] = {
933     "Unregister from FGD ", /* "Send MSG to All", "Send MSG", "Show Pilots", */
934     "Register to FGD",
935     "Scan for Deamons", "Enter Callsign", /* "Display Netinfos", */
936     "Toggle Display", NULL
937 };
938 puCallback networkSubmenuCb     [] = {
939     /* notCb, notCb, notCb, notCb, */ 
940     net_unregister, 
941     net_register, 
942     net_fgd_scan, NewCallSign, 
943     net_display_toggle, NULL
944 };
945 #endif
946
947 char *helpSubmenu               [] = {
948     /* "About...", */ "Help", NULL
949 };
950 puCallback helpSubmenuCb        [] = {
951     /* notCb, */ helpCb, NULL
952 };
953
954
955 /* -------------------------------------------------------------------------
956 init the gui
957 _____________________________________________________________________*/
958
959
960 void guiInit()
961 {
962     char *mesa_win_state;
963
964     // Initialize PUI
965     puInit();
966     puSetDefaultStyle         ( PUSTYLE_SMALL_BEVELLED ); //PUSTYLE_DEFAULT
967     puSetDefaultColourScheme  (0.8, 0.8, 0.8, 0.4);
968
969     // Initialize our GLOBAL GUI STRINGS
970     gui_msg_OK     = msg_OK;     // "OK"
971     gui_msg_NO     = msg_NO;     // "NO"
972     gui_msg_YES    = msg_YES;    // "YES"
973     gui_msg_CANCEL = msg_CANCEL; // "CANCEL"
974     gui_msg_RESET  = msg_RESET;  // "RESET"
975
976     // Next check home directory
977     SGPath fntpath;
978     char* envp = ::getenv( "FG_FONTS" );
979     if ( envp != NULL ) {
980         fntpath.set( envp );
981     } else {
982         fntpath.set( globals->get_fg_root() );
983         fntpath.append( "Fonts" );
984     }
985
986     // Install our fast fonts
987     fntpath.append( "typewriter.txf" );
988     guiFntHandle = new fntTexFont ;
989     guiFntHandle -> load ( (char *)fntpath.c_str() ) ;
990     puFont GuiFont ( guiFntHandle, 15 ) ;
991     puSetDefaultFonts( GuiFont, GuiFont ) ;
992     guiFnt = puGetDefaultLabelFont();
993   
994     if (!fgHasNode("/sim/startup/mouse-pointer")) {
995         // no preference specified for mouse pointer, attempt to autodetect...
996         // Determine if we need to render the cursor, or if the windowing
997         // system will do it.  First test if we are rendering with glide.
998         if ( strstr ( general.get_glRenderer(), "Glide" ) ) {
999             // Test for the MESA_GLX_FX env variable
1000             if ( (mesa_win_state = getenv( "MESA_GLX_FX" )) != NULL) {
1001                 // test if we are fullscreen mesa/glide
1002                 if ( (mesa_win_state[0] == 'f') ||
1003                      (mesa_win_state[0] == 'F') ) {
1004                     puShowCursor ();
1005                 }
1006             }
1007         }
1008 //        mouse_active = ~mouse_active;
1009     } else if ( !fgGetBool("/sim/startup/mouse-pointer") ) {
1010         // don't show pointer
1011     } else {
1012         // force showing pointer
1013         puShowCursor();
1014 //        mouse_active = ~mouse_active;
1015     }
1016         
1017     // MOUSE_VIEW mode stuff
1018         initMouseQuat();
1019
1020     // Set up our Dialog Boxes
1021     ConfirmExitDialogInit();
1022     NewAirportInit();
1023 #ifdef FG_NETWORK_OLK
1024     NewNetIdInit();
1025     NewNetFGDInit();
1026 #endif
1027
1028         mkDialogInit();
1029     
1030     // Make the menu bar
1031     mainMenuBar = new puMenuBar ();
1032     mainMenuBar -> add_submenu ("File", fileSubmenu, fileSubmenuCb);
1033     // mainMenuBar -> add_submenu ("Edit", editSubmenu, editSubmenuCb);
1034     mainMenuBar -> add_submenu ("View", viewSubmenu, viewSubmenuCb);
1035     mainMenuBar -> add_submenu ("Environment", environmentSubmenu, environmentSubmenuCb);
1036     mainMenuBar -> add_submenu ("Autopilot", autopilotSubmenu, autopilotSubmenuCb);
1037     // mainMenuBar -> add_submenu ("Options", optionsSubmenu, optionsSubmenuCb);
1038 #ifdef FG_NETWORK_OLK
1039     if ( fgGetBool("/sim/networking/network-olk") ) {
1040         mainMenuBar -> add_submenu ("Network", networkSubmenu, networkSubmenuCb);
1041     }
1042 #endif
1043     mainMenuBar -> add_submenu ("Help", helpSubmenu, helpSubmenuCb);
1044     mainMenuBar-> close ();
1045     // Set up menu bar toggle
1046     gui_menu_on = ~0;
1047
1048     if (fgGetString("/sim/flight-model") == "ada") {
1049         guiToggleMenu(); // Menu off by default
1050     }
1051 }
1052