]> git.mxchange.org Git - flightgear.git/blob - src/GUI/gui.cxx
568ae55ebb54289a3559203b1a91e7a2c31f82c2
[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/viewmgr.hxx>
72
73 #ifdef FG_NETWORK_OLK
74 #include <NetworkOLK/network.h>
75 #endif
76    
77 #if defined( WIN32 ) && !defined( __CYGWIN__ ) && !defined(__MINGW32__)
78 #  include <simgear/screen/win32-printer.h>
79 #  include <simgear/screen/GlBitmaps.h>
80 #endif
81
82 #include "gui.h"
83 #include "gui_local.hxx"
84 #include "apt_dlg.hxx"
85 #include "net_dlg.hxx"
86 #include "sgVec3Slider.hxx"
87 #include "prop_picker.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() != (string)"") {
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_OFF)
187         if( mouse_mode == MOUSE_POINTER )
188             TurnCursorOff();
189 #endif // WIN32_CURSOR_TWEAKS_OFF
190     } else {
191         // printf("Showing Menu\n");
192         mainMenuBar->reveal();
193 #ifdef WIN32
194         TurnCursorOn();
195 #endif // 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
253             = new puText( (150 - puGetDefaultLabelFont().getStringWidth( "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 if 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
318             = new puText( (150 - puGetDefaultLabelFont().getStringWidth( "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 - puGetDefaultLabelFont().getStringWidth ( 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     {
422         YNdialogFrame = new puFrame (0,0,400, 100);
423         
424         YNdialogBoxMessage  =  new puText         (len, 70);
425         YNdialogBoxMessage  -> setDefaultValue    (msg);
426         YNdialogBoxMessage  -> getDefaultValue    (&s);
427         YNdialogBoxMessage  -> setLabel           (s);
428         
429         YNdialogBoxOkButton =  new puOneShot      (100, 10, 160, 50);
430         YNdialogBoxOkButton -> setLegend          (gui_msg_OK);
431         YNdialogBoxOkButton -> makeReturnDefault  (TRUE );
432         YNdialogBoxOkButton -> setCallback        (goodBye);
433         
434         YNdialogBoxNoButton =  new puOneShot      (240, 10, 300, 50);
435         YNdialogBoxNoButton -> setLegend          (gui_msg_NO);
436         YNdialogBoxNoButton -> setCallback        (goAwayYesNoCb);
437     }
438     FG_FINALIZE_PUI_DIALOG( YNdialogBox );
439 }
440
441 void notCb (puObject *)
442 {
443     mkDialog ("This function isn't implemented yet");
444 }
445
446 void helpCb (puObject *)
447 {
448     string command;
449         
450 #if defined(FX) && !defined(WIN32)
451 #  if defined(XMESA_FX_FULLSCREEN) && defined(XMESA_FX_WINDOW)
452     if ( globals->get_fullscreen() ) {
453         globals->set_fullscreen(false);
454         XMesaSetFXmode( XMESA_FX_WINDOW );
455     }
456 #  endif
457 #endif
458         
459     SGPath path( globals->get_fg_root() );
460     path.append( "Docs/index.html" );
461         
462 #if !defined(WIN32)
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     command += " &";
471 #else // WIN32
472         command = "start ";
473         command += path.str();
474 #endif
475         
476     system( command.c_str() );
477     mkDialog ("Help started in netscape window.");
478 }
479
480 #define TR_HIRES_SNAP
481 #if defined( TR_HIRES_SNAP)
482 #include <simgear/screen/tr.h>
483 extern void trRenderFrame( void );
484 extern void fgUpdateHUD( GLfloat x_start, GLfloat y_start,
485                          GLfloat x_end, GLfloat y_end );
486
487 void fgHiResDump()
488 {
489     FILE *f;
490     string message;
491     bool show_pu_cursor = false;
492     bool show_menu = false;
493     char *filename = new char [24];
494     static int count = 1;
495
496     int freeze = globals->get_freeze();
497     if(!freeze)
498         globals->set_freeze( true );
499
500     if(gui_menu_on) {
501         show_menu = true;
502         guiToggleMenu();
503     }
504         
505     if ( !puCursorIsHidden() ) {
506         show_pu_cursor = true;
507         puHideCursor();
508     }
509
510     fgInitVisuals();
511     fgReshape( fgGetInt("/sim/startup/xsize"),
512                fgGetInt("/sim/startup/ysize") );
513
514     // we need two render frames here to clear the menu and cursor
515     // ... not sure why but doing an extra fgRenderFrame() shouldn't
516     // hurt anything
517     fgRenderFrame();
518     fgRenderFrame();
519
520     // Make sure we have SSG projection primed for current view
521     glMatrixMode(GL_MODELVIEW);
522     glLoadIdentity();
523     ssgSetCamera( (sgVec4 *)globals->get_current_view()->get_VIEW() );
524     float fov = globals->get_current_view()->get_fov();
525     ssgSetFOV(fov, fov * globals->get_current_view()->get_fov_ratio());
526     // ssgSetNearFar( 10.0f, 120000.0f );
527     ssgSetNearFar( 0.5f, 1200000.0f );
528
529         
530     // This ImageSize stuff is a temporary hack
531     // should probably use 128x128 tile size and
532     // support any image size
533
534     // This should be a requester to get multiplier from user
535     int multiplier = 3;
536     int width = fgGetInt("/sim/startup/xsize");
537     int height = fgGetInt("/sim/startup/ysize");
538         
539     /* allocate buffer large enough to store one tile */
540     GLubyte *tile = (GLubyte *)malloc(width * height * 3 * sizeof(GLubyte));
541     if (!tile) {
542         printf("Malloc of tile buffer failed!\n");
543         return;
544     }
545
546     int imageWidth  = multiplier*width;
547     int imageHeight = multiplier*height;
548
549     /* allocate buffer to hold a row of tiles */
550     GLubyte *buffer
551         = (GLubyte *)malloc(imageWidth * height * 3 * sizeof(GLubyte));
552     if (!buffer) {
553         free(tile);
554         printf("Malloc of tile row buffer failed!\n");
555         return;
556     }
557     TRcontext *tr = trNew();
558     trTileSize(tr, width, height, 0);
559     trTileBuffer(tr, GL_RGB, GL_UNSIGNED_BYTE, tile);
560     trImageSize(tr, imageWidth, imageHeight);
561     trRowOrder(tr, TR_TOP_TO_BOTTOM);
562     sgFrustum *frustum = ssgGetFrustum();
563     trFrustum(tr,
564               frustum->getLeft(), frustum->getRight(),
565               frustum->getBot(),  frustum->getTop(), 
566               frustum->getNear(), frustum->getFar());
567         
568     /* Prepare ppm output file */
569     while (count < 1000) {
570         snprintf(filename, 24, "fgfs-screen-%03d.ppm", count++);
571         if ( (f = fopen(filename, "r")) == NULL )
572             break;
573         fclose(f);
574     }
575     f = fopen(filename, "wb");
576     if (!f) {
577         printf("Couldn't open image file: %s\n", filename);
578         free(buffer);
579         free(tile);
580         return;
581     }
582     fprintf(f,"P6\n");
583     fprintf(f,"# ppm-file created by %s\n", "trdemo2");
584     fprintf(f,"%i %i\n", imageWidth, imageHeight);
585     fprintf(f,"255\n");
586
587     /* just to be safe... */
588     glPixelStorei(GL_PACK_ALIGNMENT, 1);
589
590     /* Because the HUD and Panel change the ViewPort we will
591      * need to handle some lowlevel stuff ourselves */
592     int ncols = trGet(tr, TR_COLUMNS);
593     int nrows = trGet(tr, TR_ROWS);
594
595     bool do_hud = fgGetBool("/sim/hud/visibility");
596     GLfloat hud_col_step = 640.0 / ncols;
597     GLfloat hud_row_step = 480.0 / nrows;
598         
599     bool do_panel = fgPanelVisible();
600     GLfloat panel_col_step = current_panel->getWidth() / ncols;
601     GLfloat panel_row_step = current_panel->getHeight() / nrows;
602         
603     /* Draw tiles */
604     int more = 1;
605     while (more) {
606         trBeginTile(tr);
607         int curColumn = trGet(tr, TR_CURRENT_COLUMN);
608         int curRow =  trGet(tr, TR_CURRENT_ROW);
609         trRenderFrame();
610         if ( do_hud )
611             fgUpdateHUD( curColumn*hud_col_step,      curRow*hud_row_step,
612                          (curColumn+1)*hud_col_step, (curRow+1)*hud_row_step );
613         if (do_panel)
614             current_panel->update( curColumn*panel_col_step, panel_col_step,
615                                    curRow*panel_row_step,    panel_row_step );
616         more = trEndTile(tr);
617
618         /* save tile into tile row buffer*/
619         int curTileWidth = trGet(tr, TR_CURRENT_TILE_WIDTH);
620         int bytesPerImageRow = imageWidth * 3*sizeof(GLubyte);
621         int bytesPerTileRow = (width) * 3*sizeof(GLubyte);
622         int xOffset = curColumn * bytesPerTileRow;
623         int bytesPerCurrentTileRow = (curTileWidth) * 3*sizeof(GLubyte);
624         int i;
625         for (i=0;i<height;i++) {
626             memcpy(buffer + i*bytesPerImageRow + xOffset, /* Dest */
627                    tile + i*bytesPerTileRow,              /* Src */
628                    bytesPerCurrentTileRow);               /* Byte count*/
629         }
630
631         if (curColumn == trGet(tr, TR_COLUMNS)-1) {
632             /* write this buffered row of tiles to the file */
633             int curTileHeight = trGet(tr, TR_CURRENT_TILE_HEIGHT);
634             int bytesPerImageRow = imageWidth * 3*sizeof(GLubyte);
635             int i;
636             for (i=0;i<curTileHeight;i++) {
637                 /* Remember, OpenGL images are bottom to top.  Have to reverse. */
638                 GLubyte *rowPtr = buffer + (curTileHeight-1-i) * bytesPerImageRow;
639                 fwrite(rowPtr, 1, imageWidth*3, f);
640             }
641         }
642
643     }
644
645     fgReshape( width, height );
646
647     trDelete(tr);
648
649     fclose(f);
650
651     message = "Snap shot saved to ";
652     message += filename;
653     mkDialog (message.c_str());
654
655     free(tile);
656     free(buffer);
657
658     //  message = "Snap shot saved to ";
659     //  message += filename;
660     //  mkDialog (message.c_str());
661
662     delete [] filename;
663
664     if( show_menu )
665         guiToggleMenu();
666
667     if ( show_pu_cursor ) {
668         puShowCursor();
669     }
670
671     if(!freeze)
672         globals->set_freeze( false );
673 }
674 #endif // #if defined( TR_HIRES_SNAP)
675
676
677 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
678
679 static void rotateView( double roll, double pitch, double yaw )
680 {
681         // rotate view
682 }
683
684 static GlBitmap *b1 = NULL;
685 extern FGInterface cur_view_fdm;
686 GLubyte *hiResScreenCapture( int multiplier )
687 {
688     float oldfov = fgGetDouble("/sim/field-of-view");
689     float fov = oldfov / multiplier;
690     FGViewer *v = globals->get_current_view();
691     fgSetDouble("/sim/field-of-view", fov);
692     fgInitVisuals();
693     int cur_width = fgGetInt("/sim/startup/xsize");
694     int cur_height = fgGetInt("/sim/startup/ysize");
695     if (b1) delete( b1 );
696     // New empty (mostly) bitmap
697     b1 = new GlBitmap( GL_RGB, 1, 1, (unsigned char *)"123" );
698     int x,y;
699     for ( y = 0; y < multiplier; y++ ) {
700         for ( x = 0; x < multiplier; x++ ) {
701             fgReshape( cur_width, cur_height );
702             // pan to tile
703             rotateView( 0, (y*fov)-((multiplier-1)*fov/2), (x*fov)-((multiplier-1)*fov/2) );
704             fgRenderFrame();
705             // restore view
706             GlBitmap b2;
707             b1->copyBitmap( &b2, cur_width*x, cur_height*y );
708         }
709     }
710     fgSetDouble("/sim/field-of-view", oldfov);
711     return b1->getBitmap();
712 }
713 #endif
714
715
716 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
717 // win32 print screen function
718 void printScreen ( puObject *obj ) {
719     bool show_pu_cursor = false;
720     TurnCursorOff();
721     if ( !puCursorIsHidden() ) {
722         show_pu_cursor = true;
723         puHideCursor();
724     }
725     // BusyCursor( 0 );
726     mainMenuBar->hide();
727
728     CGlPrinter p( CGlPrinter::PRINT_BITMAP );
729     int cur_width = fgGetInt("/sim/startup/xsize");
730     int cur_height = fgGetInt("/sim/startup/ysize");
731     p.Begin( "FlightGear", cur_width*3, cur_height*3 );
732         p.End( hiResScreenCapture(3) );
733
734     if( gui_menu_on ) {
735         mainMenuBar->reveal();
736     }
737     // BusyCursor(1);
738     if ( show_pu_cursor ) {
739         puShowCursor();
740     }
741     TurnCursorOn();
742 }
743 #endif // #ifdef WIN32
744
745
746 void dumpSnapShot ( puObject *obj ) {
747     fgDumpSnapShot();
748 }
749
750
751 void dumpHiResSnapShot ( puObject *obj ) {
752     fgHiResDump();
753 }
754
755
756 // do a screen snap shot
757 void fgDumpSnapShot () {
758     bool show_pu_cursor = false;
759     char *filename = new char [24];
760     string message;
761     static int count = 1;
762
763     int freeze = globals->get_freeze();
764     if(!freeze)
765         globals->set_freeze( true );
766
767     mainMenuBar->hide();
768     TurnCursorOff();
769     if ( !puCursorIsHidden() ) {
770         show_pu_cursor = true;
771         puHideCursor();
772     }
773
774     fgInitVisuals();
775     fgReshape( fgGetInt("/sim/startup/xsize"),
776                fgGetInt("/sim/startup/ysize") );
777
778     // we need two render frames here to clear the menu and cursor
779     // ... not sure why but doing an extra fgRenderFrame() shouldn't
780     // hurt anything
781     fgRenderFrame();
782     fgRenderFrame();
783
784     while (count < 1000) {
785         FILE *fp;
786         snprintf(filename, 24, "fgfs-screen-%03d.ppm", count++);
787         if ( (fp = fopen(filename, "r")) == NULL )
788             break;
789         fclose(fp);
790     }
791
792     if ( sg_glDumpWindow( filename,
793                           fgGetInt("/sim/startup/xsize"), 
794                           fgGetInt("/sim/startup/ysize")) ) {
795         message = "Snap shot saved to ";
796         message += filename;
797     } else {
798         message = "Failed to save to ";
799         message += filename;
800     }
801
802     mkDialog (message.c_str());
803
804     delete [] filename;
805
806     if ( show_pu_cursor ) {
807         puShowCursor();
808     }
809
810     TurnCursorOn();
811     if( gui_menu_on ) {
812         mainMenuBar->reveal();
813     }
814
815     if(!freeze)
816         globals->set_freeze( false );
817 }
818
819 #ifdef FG_NETWORK_OLK
820 static void net_display_toggle( puObject *cb)
821 {
822         net_hud_display = (net_hud_display) ? 0 : 1;
823         printf("Toggle net_hud_display : %d\n", net_hud_display);
824 }
825
826 static void net_register( puObject *cb)
827 {
828         fgd_send_com( "1", FGFS_host );
829         net_is_registered = 0;
830         printf("Registering to deamon\n");
831 }
832
833 static void net_unregister( puObject *cb)
834 {
835         fgd_send_com( "8", FGFS_host );
836         net_is_registered = -1;
837         printf("Unregistering from deamon\n");
838 }
839
840 extern void net_fgd_scan(puObject *cb);
841 #endif // #ifdef FG_NETWORK_OLK
842
843 /* -----------------------------------------------------------------------
844 The menu stuff 
845 ---------------------------------------------------------------------*/
846 char *fileSubmenu               [] = {
847     "Exit", /* "Close", "---------", */
848 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
849     "Print",
850 #endif
851     "Snap Shot",
852     "Hires Snap Shot",
853     "---------", 
854     "Reset", 
855     "Load flight",
856     "Save flight",
857     NULL
858 };
859 puCallback fileSubmenuCb        [] = {
860     MayBeGoodBye, /* hideMenuCb, NULL, */
861 #if defined( WIN32 ) && !defined( __CYGWIN__) && !defined(__MINGW32__)
862     printScreen, 
863 #endif
864     /* NULL, notCb, */
865     dumpSnapShot,
866     dumpHiResSnapShot,
867     NULL,
868     reInit, 
869     loadFlight,
870     saveFlight,
871     NULL
872 };
873
874 /*
875 char *editSubmenu               [] = {
876     "Edit text", NULL
877 };
878 puCallback editSubmenuCb        [] = {
879     notCb, NULL
880 };
881 */
882
883 extern void fgHUDalphaAdjust( puObject * );
884 char *viewSubmenu               [] = {
885     "Properties",
886     "HUD Alpha",
887     "Pilot Offset",
888     /* "Cockpit View > ", "View >","------------", */
889     "Toggle Panel...", NULL
890 };
891 puCallback viewSubmenuCb        [] = {
892     prop_pickerView,
893     fgHUDalphaAdjust,
894     PilotOffsetAdjust,
895     /* notCb, notCb, NULL, */
896     guiTogglePanel, NULL
897 };
898
899 //  "---------", 
900
901 char *autopilotSubmenu           [] = {
902     "Toggle HUD Format", "Adjust AP Settings",
903     "---------", 
904     "Clear Route", "Skip Current Waypoint", "Add Waypoint",
905     "---------", 
906     "Set Altitude", "Set Heading",
907     NULL
908 };
909
910 puCallback autopilotSubmenuCb    [] = {
911     fgLatLonFormatToggle, fgAPAdjust,
912     NULL,
913     ClearRoute, PopWayPoint, AddWayPoint,
914     NULL,
915     NewAltitude, NewHeading,
916     /* notCb, */ NULL
917 };
918
919 char *environmentSubmenu        [] = {
920     "Goto Airport", /* "Terrain", "Weather", */ NULL
921 };
922 puCallback environmentSubmenuCb [] = {
923     NewAirport, /* notCb, notCb, */ NULL
924 };
925
926 /*
927 char *optionsSubmenu            [] = {
928     "Preferences", "Realism & Reliablity...", NULL
929 };
930 puCallback optionsSubmenuCb     [] = {
931     notCb, notCb, NULL
932 };
933 */
934
935 #ifdef FG_NETWORK_OLK
936 char *networkSubmenu            [] = {
937     "Unregister from FGD ", /* "Send MSG to All", "Send MSG", "Show Pilots", */
938     "Register to FGD",
939     "Scan for Deamons", "Enter Callsign", /* "Display Netinfos", */
940     "Toggle Display", NULL
941 };
942 puCallback networkSubmenuCb     [] = {
943     /* notCb, notCb, notCb, notCb, */ 
944     net_unregister, 
945     net_register, 
946     net_fgd_scan, NewCallSign, 
947     net_display_toggle, NULL
948 };
949 #endif
950
951 char *helpSubmenu               [] = {
952     /* "About...", */ "Help", NULL
953 };
954 puCallback helpSubmenuCb        [] = {
955     /* notCb, */ helpCb, NULL
956 };
957
958
959 /* -------------------------------------------------------------------------
960 init the gui
961 _____________________________________________________________________*/
962
963
964 void guiInit()
965 {
966     char *mesa_win_state;
967
968     // Initialize PUI
969     puInit();
970     puSetDefaultStyle         ( PUSTYLE_SMALL_BEVELLED ); //PUSTYLE_DEFAULT
971     puSetDefaultColourScheme  (0.8, 0.8, 0.8, 0.4);
972
973     // Initialize our GLOBAL GUI STRINGS
974     gui_msg_OK     = msg_OK;     // "OK"
975     gui_msg_NO     = msg_NO;     // "NO"
976     gui_msg_YES    = msg_YES;    // "YES"
977     gui_msg_CANCEL = msg_CANCEL; // "CANCEL"
978     gui_msg_RESET  = msg_RESET;  // "RESET"
979
980     // Next check home directory
981     SGPath fntpath;
982     char* envp = ::getenv( "FG_FONTS" );
983     if ( envp != NULL ) {
984         fntpath.set( envp );
985     } else {
986         fntpath.set( globals->get_fg_root() );
987         fntpath.append( "Fonts" );
988     }
989
990     // Install our fast fonts
991     fntpath.append( "typewriter.txf" );
992     guiFntHandle = new fntTexFont ;
993     guiFntHandle -> load ( (char *)fntpath.c_str() ) ;
994     puFont GuiFont ( guiFntHandle, 15 ) ;
995     puSetDefaultFonts( GuiFont, GuiFont ) ;
996     guiFnt = puGetDefaultLabelFont();
997   
998     if (!fgHasNode("/sim/startup/mouse-pointer")) {
999         // no preference specified for mouse pointer, attempt to autodetect...
1000         // Determine if we need to render the cursor, or if the windowing
1001         // system will do it.  First test if we are rendering with glide.
1002         if ( strstr ( general.get_glRenderer(), "Glide" ) ) {
1003             // Test for the MESA_GLX_FX env variable
1004             if ( (mesa_win_state = getenv( "MESA_GLX_FX" )) != NULL) {
1005                 // test if we are fullscreen mesa/glide
1006                 if ( (mesa_win_state[0] == 'f') ||
1007                      (mesa_win_state[0] == 'F') ) {
1008                     puShowCursor ();
1009                 }
1010             }
1011         }
1012 //        mouse_active = ~mouse_active;
1013     } else if ( !fgGetBool("/sim/startup/mouse-pointer") ) {
1014         // don't show pointer
1015     } else {
1016         // force showing pointer
1017         puShowCursor();
1018 //        mouse_active = ~mouse_active;
1019     }
1020         
1021     // MOUSE_VIEW mode stuff
1022         initMouseQuat();
1023
1024     // Set up our Dialog Boxes
1025     ConfirmExitDialogInit();
1026     NewAirportInit();
1027 #ifdef FG_NETWORK_OLK
1028     NewNetIdInit();
1029     NewNetFGDInit();
1030 #endif
1031
1032         mkDialogInit();
1033     
1034     // Make the menu bar
1035     mainMenuBar = new puMenuBar ();
1036     mainMenuBar -> add_submenu ("File", fileSubmenu, fileSubmenuCb);
1037     // mainMenuBar -> add_submenu ("Edit", editSubmenu, editSubmenuCb);
1038     mainMenuBar -> add_submenu ("View", viewSubmenu, viewSubmenuCb);
1039     mainMenuBar -> add_submenu ("Environment", environmentSubmenu, environmentSubmenuCb);
1040     mainMenuBar -> add_submenu ("Autopilot", autopilotSubmenu, autopilotSubmenuCb);
1041     // mainMenuBar -> add_submenu ("Options", optionsSubmenu, optionsSubmenuCb);
1042 #ifdef FG_NETWORK_OLK
1043     if ( fgGetBool("/sim/networking/network-olk") ) {
1044         mainMenuBar -> add_submenu ("Network", networkSubmenu, networkSubmenuCb);
1045     }
1046 #endif
1047     mainMenuBar -> add_submenu ("Help", helpSubmenu, helpSubmenuCb);
1048     mainMenuBar-> close ();
1049     // Set up menu bar toggle
1050     gui_menu_on = ~0;
1051
1052     if (fgGetString("/sim/flight-model") == "ada") {
1053         guiToggleMenu(); // Menu off by default
1054     }
1055 }
1056