]> Shamusworld >> Repos - virtualjaguar/blob - src/gui/mainwin.cpp
3cebc8ac1541904a707e847f91a01e71fe6cb934
[virtualjaguar] / src / gui / mainwin.cpp
1 //
2 // mainwin.cpp - Qt-based GUI for Virtual Jaguar: Main Application Window
3 // by James L. Hammons
4 // (C) 2009 Underground Software
5 //
6 // JLH = James L. Hammons <jlhamm@acm.org>
7 //
8 // Who  When        What
9 // ---  ----------  -------------------------------------------------------------
10 // JLH  12/23/2009  Created this file
11 // JLH  12/20/2010  Added settings, menus & toolbars
12 // JLH  07/05/2011  Added CD BIOS functionality to GUI
13 //
14
15 // FIXED:
16 //
17 // - Add dbl click/enter to select in cart list, ESC to dimiss [DONE]
18 // - Autoscan/autoload all available BIOS from 'software' folder [DONE]
19 // - Add 1 key jumping in cartridge list (press 'R', jumps to carts starting with 'R', etc) [DONE]
20 //
21 // STILL TO BE DONE:
22 //
23 // - Controller configuration
24 // - Remove SDL dependencies (sound, mainly) from Jaguar core lib
25 // - Fix inconsistency with trailing slashes in paths (eeproms needs one, software doesn't)
26 //
27
28 // Uncomment this for debugging...
29 //#define DEBUG
30 //#define DEBUGFOO                      // Various tool debugging...
31 //#define DEBUGTP                               // Toolpalette debugging...
32
33 #include "mainwin.h"
34
35 #include "SDL.h"
36 #include "glwidget.h"
37 #include "about.h"
38 #include "settings.h"
39 #include "filepicker.h"
40 #include "configdialog.h"
41 #include "generaltab.h"
42 #include "version.h"
43
44 #include "dac.h"
45 #include "jaguar.h"
46 #include "tom.h"
47 #include "log.h"
48 #include "file.h"
49 #include "jagbios.h"
50 #include "jagcdbios.h"
51 #include "jagstub2bios.h"
52 #include "joystick.h"
53
54 #ifdef __GCCWIN32__
55 // Apparently on win32, usleep() is not pulled in by the usual suspects.
56 #include <unistd.h>
57 #endif
58
59 // Uncomment this to use built-in BIOS/CD-ROM BIOS
60 // You'll need a copy of jagboot.h & jagcd.h for this to work...!
61 // Creating those is left as an exercise for the reader. ;-)
62 //#define USE_BUILT_IN_BIOS
63
64 //#ifdef USE_BUILT_IN_BIOS
65 //#include "jagboot.h"
66 //#include "jagcd.h"
67 //#endif
68
69 // The way BSNES controls things is by setting a timer with a zero
70 // timeout, sleeping if not emulating anything. Seems there has to be a
71 // better way.
72
73 // It has a novel approach to plugging-in/using different video/audio/input
74 // methods, can we do something similar or should we just use the built-in
75 // QOpenGL?
76
77 // We're going to try to use the built-in OpenGL support and see how it goes.
78 // We'll make the VJ core modular so that it doesn't matter what GUI is in
79 // use, we can drop it in anywhere and use it as-is.
80
81 MainWin::MainWin(): running(false), powerButtonOn(false), showUntunedTankCircuit(true),
82         cartridgeLoaded(false), CDActive(false),//, alpineLoadSuccessful(false),
83         pauseForFileSelector(false)
84 {
85         videoWidget = new GLWidget(this);
86         setCentralWidget(videoWidget);
87         setWindowIcon(QIcon(":/res/vj-icon.png"));
88 //      setWindowTitle("Virtual Jaguar v2.0.0");
89
90         QString title = QString(tr("Virtual Jaguar " VJ_RELEASE_VERSION ));
91
92         if (vjs.hardwareTypeAlpine)
93                 title += QString(tr(" - Alpine Mode"));
94
95         setWindowTitle(title);
96
97         aboutWin = new AboutWindow(this);
98         filePickWin = new FilePickerWindow(this);
99
100     videoWidget->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
101     setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
102
103         setUnifiedTitleAndToolBarOnMac(true);
104
105         // Create actions
106
107         quitAppAct = new QAction(tr("E&xit"), this);
108         quitAppAct->setShortcuts(QKeySequence::Quit);
109         quitAppAct->setStatusTip(tr("Quit Virtual Jaguar"));
110         connect(quitAppAct, SIGNAL(triggered()), this, SLOT(close()));
111
112         powerAct = new QAction(QIcon(":/res/power.png"), tr("&Power"), this);
113         powerAct->setStatusTip(tr("Powers Jaguar on/off"));
114         powerAct->setCheckable(true);
115         powerAct->setChecked(false);
116         powerAct->setDisabled(true);
117         connect(powerAct, SIGNAL(triggered()), this, SLOT(TogglePowerState()));
118
119         pauseAct = new QAction(QIcon(":/res/pause.png"), tr("Pause"), this);
120         pauseAct->setStatusTip(tr("Toggles the running state"));
121         pauseAct->setCheckable(true);
122         pauseAct->setDisabled(true);
123         pauseAct->setShortcut(QKeySequence(tr("Esc")));
124         connect(pauseAct, SIGNAL(triggered()), this, SLOT(ToggleRunState()));
125
126         zoomActs = new QActionGroup(this);
127
128         x1Act = new QAction(QIcon(":/res/zoom100.png"), tr("Zoom 100%"), zoomActs);
129         x1Act->setStatusTip(tr("Set window zoom to 100%"));
130         x1Act->setCheckable(true);
131         connect(x1Act, SIGNAL(triggered()), this, SLOT(SetZoom100()));
132
133         x2Act = new QAction(QIcon(":/res/zoom200.png"), tr("Zoom 200%"), zoomActs);
134         x2Act->setStatusTip(tr("Set window zoom to 200%"));
135         x2Act->setCheckable(true);
136         connect(x2Act, SIGNAL(triggered()), this, SLOT(SetZoom200()));
137
138         x3Act = new QAction(QIcon(":/res/zoom300.png"), tr("Zoom 300%"), zoomActs);
139         x3Act->setStatusTip(tr("Set window zoom to 300%"));
140         x3Act->setCheckable(true);
141         connect(x3Act, SIGNAL(triggered()), this, SLOT(SetZoom300()));
142
143         tvTypeActs = new QActionGroup(this);
144
145         ntscAct = new QAction(QIcon(":/res/ntsc.png"), tr("NTSC"), tvTypeActs);
146         ntscAct->setStatusTip(tr("Sets Jaguar to NTSC mode"));
147         ntscAct->setCheckable(true);
148         connect(ntscAct, SIGNAL(triggered()), this, SLOT(SetNTSC()));
149
150         palAct = new QAction(QIcon(":/res/pal.png"), tr("PAL"), tvTypeActs);
151         palAct->setStatusTip(tr("Sets Jaguar to PAL mode"));
152         palAct->setCheckable(true);
153         connect(palAct, SIGNAL(triggered()), this, SLOT(SetPAL()));
154
155         blurAct = new QAction(QIcon(":/res/generic.png"), tr("Blur"), this);
156         blurAct->setStatusTip(tr("Sets OpenGL rendering to GL_NEAREST"));
157         blurAct->setCheckable(true);
158         connect(blurAct, SIGNAL(triggered()), this, SLOT(ToggleBlur()));
159
160         aboutAct = new QAction(QIcon(":/res/vj-icon.png"), tr("&About..."), this);
161         aboutAct->setStatusTip(tr("Blatant self-promotion"));
162         connect(aboutAct, SIGNAL(triggered()), this, SLOT(ShowAboutWin()));
163
164         filePickAct = new QAction(QIcon(":/res/software.png"), tr("&Insert Cartridge..."), this);
165         filePickAct->setStatusTip(tr("Insert a cartridge into Virtual Jaguar"));
166         filePickAct->setShortcut(QKeySequence(tr("Ctrl+i")));
167         connect(filePickAct, SIGNAL(triggered()), this, SLOT(InsertCart()));
168
169         configAct = new QAction(QIcon(":/res/generic.png"), tr("&Configure"), this);
170         configAct->setStatusTip(tr("Configure options for Virtual Jaguar"));
171         configAct->setShortcut(QKeySequence(tr("Ctrl+c")));
172         connect(configAct, SIGNAL(triggered()), this, SLOT(Configure()));
173
174         useCDAct = new QAction(QIcon(":/res/compact-disc.png"), tr("&Use CD Unit"), this);
175         useCDAct->setStatusTip(tr("Use Jaguar Virtual CD unit"));
176 //      useCDAct->setShortcut(QKeySequence(tr("Ctrl+c")));
177         useCDAct->setCheckable(true);
178         connect(useCDAct, SIGNAL(triggered()), this, SLOT(ToggleCDUsage()));
179
180         // Misc. connections...
181         connect(filePickWin, SIGNAL(RequestLoad(QString)), this, SLOT(LoadSoftware(QString)));
182         connect(filePickWin, SIGNAL(FilePickerHiding()), this, SLOT(Unpause()));
183
184         // Create menus & toolbars
185
186         fileMenu = menuBar()->addMenu(tr("&File"));
187         fileMenu->addAction(filePickAct);
188         fileMenu->addAction(useCDAct);
189         fileMenu->addAction(powerAct);
190         fileMenu->addAction(pauseAct);
191         fileMenu->addAction(configAct);
192         fileMenu->addAction(quitAppAct);
193
194         helpMenu = menuBar()->addMenu(tr("&Help"));
195         helpMenu->addAction(aboutAct);
196
197         toolbar = addToolBar(tr("Stuff"));
198         toolbar->addAction(powerAct);
199         toolbar->addAction(pauseAct);
200         toolbar->addAction(filePickAct);
201         toolbar->addAction(useCDAct);
202         toolbar->addSeparator();
203         toolbar->addAction(x1Act);
204         toolbar->addAction(x2Act);
205         toolbar->addAction(x3Act);
206         toolbar->addSeparator();
207         toolbar->addAction(ntscAct);
208         toolbar->addAction(palAct);
209         toolbar->addSeparator();
210         toolbar->addAction(blurAct);
211
212         //      Create status bar
213         statusBar()->showMessage(tr("Ready"));
214
215         ReadSettings();
216
217         // Set toolbar buttons/menus based on settings read in (sync the UI)...
218         blurAct->setChecked(vjs.glFilter);
219         x1Act->setChecked(zoomLevel == 1);
220         x2Act->setChecked(zoomLevel == 2);
221         x3Act->setChecked(zoomLevel == 3);
222         running = powerAct->isChecked();
223         ntscAct->setChecked(vjs.hardwareTypeNTSC);
224         palAct->setChecked(!vjs.hardwareTypeNTSC);
225
226         // Do this in case original size isn't correct (mostly for the first-run case)
227         ResizeMainWindow();
228
229         // Set up timer based loop for animation...
230         timer = new QTimer(this);
231         connect(timer, SIGNAL(timeout()), this, SLOT(Timer()));
232         timer->start(20);
233
234         WriteLog("Virtual Jaguar %s (Last full build was on %s %s)\n", VJ_RELEASE_VERSION, __DATE__, __TIME__);
235         WriteLog("VJ: Initializing jaguar subsystem...\n");
236         JaguarInit();
237
238         // Get the BIOS ROM
239 #ifdef USE_BUILT_IN_BIOS
240 //      WriteLog("VJ: Using built in BIOS/CD BIOS...\n");
241 //      memcpy(jaguarBootROM, jagBootROM, 0x20000);
242 //      memcpy(jaguarCDBootROM, jagCDROM, 0x40000);
243 ////    BIOSLoaded = CDBIOSLoaded = true;
244 //      biosAvailable |= (BIOS_NORMAL | BIOS_CD);
245 #else
246 // What would be nice here would be a way to check if the BIOS was loaded so that we
247 // could disable the pushbutton on the Misc Options menu... !!! FIX !!! [DONE here, but needs to be fixed in GUI as well!]
248 //      WriteLog("VJ: About to attempt to load BIOSes...\n");
249 //This is short-circuiting the file finding thread... ??? WHY ???
250 //Not anymore. Was related to a QImage object creation/corruption bug elsewhere.
251 //      BIOSLoaded = (JaguarLoadROM(jaguarBootROM, vjs.jagBootPath) == 0x20000 ? true : false);
252 //      WriteLog("VJ: BIOS is %savailable...\n", (BIOSLoaded ? "" : "not "));
253 //      CDBIOSLoaded = (JaguarLoadROM(jaguarCDBootROM, vjs.CDBootPath) == 0x40000 ? true : false);
254 //      WriteLog("VJ: CD BIOS is %savailable...\n", (CDBIOSLoaded ? "" : "not "));
255 #endif
256
257         filePickWin->ScanSoftwareFolder(allowUnknownSoftware);
258
259         // Load up the default ROM if in Alpine mode:
260         if (vjs.hardwareTypeAlpine)
261         {
262                 bool romLoaded = JaguarLoadFile(vjs.alpineROMPath);
263
264                 // If regular load failed, try just a straight file load
265                 // (Dev only! I don't want people to start getting lazy with their releases again! :-P)
266                 if (!romLoaded)
267                         romLoaded = AlpineLoadFile(vjs.alpineROMPath);
268
269                 if (romLoaded)
270                         WriteLog("Alpine Mode: Successfully loaded file \"%s\".\n", vjs.alpineROMPath);
271                 else
272                         WriteLog("Alpine Mode: Unable to load file \"%s\"!\n", vjs.alpineROMPath);
273
274                 // Attempt to load/run the ABS file...
275                 LoadSoftware(vjs.absROMPath);
276         }
277 }
278
279 void MainWin::closeEvent(QCloseEvent * event)
280 {
281         JaguarDone();
282         WriteSettings();
283         event->accept(); // ignore() if can't close for some reason
284 }
285
286 void MainWin::keyPressEvent(QKeyEvent * e)
287 {
288         HandleKeys(e, true);
289 }
290
291 void MainWin::keyReleaseEvent(QKeyEvent * e)
292 {
293         HandleKeys(e, false);
294 }
295
296 void MainWin::HandleKeys(QKeyEvent * e, bool state)
297 {
298         // We kill bad key combos here, before they can get to the emulator...
299         // This also kills the illegal instruction problem that cropped up in Rayman!
300         // May want to do this by killing the old one instead of ignoring the new one...
301         // Seems to work better that way...
302 #if 0
303         if ((e->key() == vjs.p1KeyBindings[BUTTON_L] && joypad_0_buttons[BUTTON_R])
304                 || (e->key() == vjs.p1KeyBindings[BUTTON_R] && joypad_0_buttons[BUTTON_L])
305                 || (e->key() == vjs.p1KeyBindings[BUTTON_U] && joypad_0_buttons[BUTTON_D])
306                 || (e->key() == vjs.p1KeyBindings[BUTTON_D] && joypad_0_buttons[BUTTON_U]))
307                 return;
308 #else
309         if (e->key() == (int)vjs.p1KeyBindings[BUTTON_L] && joypad_0_buttons[BUTTON_R])
310                 joypad_0_buttons[BUTTON_R] = 0;
311         if (e->key() == (int)vjs.p1KeyBindings[BUTTON_R] && joypad_0_buttons[BUTTON_L])
312                 joypad_0_buttons[BUTTON_L] = 0;
313         if (e->key() == (int)vjs.p1KeyBindings[BUTTON_U] && joypad_0_buttons[BUTTON_D])
314                 joypad_0_buttons[BUTTON_D] = 0;
315         if (e->key() == (int)vjs.p1KeyBindings[BUTTON_D] && joypad_0_buttons[BUTTON_U])
316                 joypad_0_buttons[BUTTON_U] = 0;
317 #endif
318
319         // No bad combos exist, let's stuff the emulator key buffers...!
320         for(int i=BUTTON_FIRST; i<=BUTTON_LAST; i++)
321         {
322                 if (e->key() == (int)vjs.p1KeyBindings[i])
323                         joypad_0_buttons[i] = (uint8)state;
324         }
325 }
326
327 void MainWin::Open(void)
328 {
329 }
330
331 void MainWin::Configure(void)
332 {
333         // Call the configuration dialog and update settings
334         ConfigDialog dlg(this);
335         //ick.
336         dlg.generalTab->useUnknownSoftware->setChecked(allowUnknownSoftware);
337
338         if (dlg.exec() == false)
339                 return;
340
341         QString before = vjs.ROMPath;
342         QString alpineBefore = vjs.alpineROMPath;
343         QString absBefore = vjs.absROMPath;
344         bool audioBefore = vjs.audioEnabled;
345         dlg.UpdateVJSettings();
346         QString after = vjs.ROMPath;
347         QString alpineAfter = vjs.alpineROMPath;
348         QString absAfter = vjs.absROMPath;
349         bool audioAfter = vjs.audioEnabled;
350
351         bool allowOld = allowUnknownSoftware;
352         //ick.
353         allowUnknownSoftware = dlg.generalTab->useUnknownSoftware->isChecked();
354
355         // We rescan the "software" folder if the user either changed the path or
356         // checked/unchecked the "Allow unknown files" option in the config dialog.
357         if ((before != after) || (allowOld != allowUnknownSoftware))
358                 filePickWin->ScanSoftwareFolder(allowUnknownSoftware);
359
360         // If the "Alpine" ROM is changed, then let's load it...
361         if (alpineBefore != alpineAfter)
362         {
363                 if (!JaguarLoadFile(vjs.alpineROMPath) && !AlpineLoadFile(vjs.alpineROMPath))
364                 {
365                         // Oh crap, we couldn't get the file! Alert the media!
366                         QMessageBox msg;
367                         msg.setText(QString(tr("Could not load file \"%1\"!")).arg(vjs.alpineROMPath));
368                         msg.setIcon(QMessageBox::Warning);
369                         msg.exec();
370                 }
371         }
372
373         // If the "ABS" ROM is changed, then let's load it...
374         if (absBefore != absAfter)
375         {
376                 if (!JaguarLoadFile(vjs.absROMPath))
377                 {
378                         // Oh crap, we couldn't get the file! Alert the media!
379                         QMessageBox msg;
380                         msg.setText(QString(tr("Could not load file \"%1\"!")).arg(vjs.absROMPath));
381                         msg.setIcon(QMessageBox::Warning);
382                         msg.exec();
383                 }
384         }
385
386         // If the "Enable audio" checkbox changed, then we have to re-init the DAC...
387         if (audioBefore != audioAfter)
388         {
389                 DACDone();
390                 DACInit();
391         }
392
393         // Just in case we crash before a clean exit...
394         WriteSettings();
395 }
396
397 //
398 // Here's the main emulator loop
399 //
400 void MainWin::Timer(void)
401 {
402         if (!running)
403                 return;
404
405         if (showUntunedTankCircuit)
406         {
407                 // Random hash & trash
408                 // We try to simulate an untuned tank circuit here... :-)
409                 for(uint32_t x=0; x<videoWidget->rasterWidth; x++)
410                 {
411                         for(uint32_t y=0; y<videoWidget->rasterHeight; y++)
412                         {
413                                 videoWidget->buffer[(y * videoWidget->textureWidth) + x] = (rand() & 0xFF) << 8 | (rand() & 0xFF) << 16 | (rand() & 0xFF) << 24;// | (rand() & 0xFF);//0x000000FF;
414         //                      buffer[(y * textureWidth) + x] = x*y;
415                         }
416                 }
417         }
418         else
419         {
420                 // Otherwise, run the Jaguar simulation
421                 JaguarExecuteNew();
422 //              memcpy(videoWidget->buffer, backbuffer, videoWidget->rasterHeight * videoWidget->rasterWidth);
423                 memcpy(videoWidget->buffer, backbuffer, videoWidget->rasterHeight * videoWidget->textureWidth * sizeof(uint32_t));
424 //              memcpy(surface->pixels, backbuffer, TOMGetVideoModeWidth() * TOMGetVideoModeHeight() * 4);
425         }
426
427         videoWidget->updateGL();
428 }
429
430 void MainWin::TogglePowerState(void)
431 {
432         powerButtonOn = !powerButtonOn;
433
434         // With the power off, we simulate white noise on the screen. :-)
435         if (!powerButtonOn)
436         {
437                 pauseAct->setChecked(false);
438                 pauseAct->setDisabled(true);
439                 showUntunedTankCircuit = true;
440                 running = true;
441                 // This is just in case the ROM we were playing was in a narrow or wide field mode,
442                 // so the untuned tank sim doesn't look wrong. :-)
443                 TOMReset();
444         }
445         else
446         {
447 //NOTE: Low hanging fruit: We can simplify this a lot...
448                 // Otherwise, we prepare for running regular software...
449                 if (!CDActive)
450                 {
451                         showUntunedTankCircuit = false;//(cartridgeLoaded ? false : true);
452                         pauseAct->setChecked(false);
453                         pauseAct->setDisabled(false);//!cartridgeLoaded);
454                 }
455                 // Or, set up for the Jaguar CD
456                 else
457                 {
458 // Should check for cartridgeLoaded here as well...!
459 // We can clear it when toggling CDActive on, so that when we power cycle it does the
460 // expected thing. Otherwise, if we use the file picker to insert a cart, we expect
461 // to run the cart! Maybe have a RemoveCart function that only works if the CD unit
462 // is active?
463                         showUntunedTankCircuit = false;
464                         pauseAct->setChecked(false);
465                         pauseAct->setDisabled(false);
466                         memcpy(jagMemSpace + 0x800000, jaguarCDBootROM, 0x40000);
467                         setWindowTitle(QString("Virtual Jaguar " VJ_RELEASE_VERSION
468                                 " - Now playing: Jaguar CD"));
469                 }
470
471                 WriteLog("GUI: Resetting Jaguar...\n");
472                 JaguarReset();
473                 running = true;
474         }
475 }
476
477 void MainWin::ToggleRunState(void)
478 {
479         running = !running;
480
481         if (!running)
482         {
483                 for(uint32_t i=0; i<(uint32_t)(videoWidget->textureWidth * 256); i++)
484                 {
485                         uint32_t pixel = backbuffer[i];
486                         uint8_t r = (pixel >> 24) & 0xFF, g = (pixel >> 16) & 0xFF, b = (pixel >> 8) & 0xFF;
487                         pixel = ((r + g + b) / 3) & 0x00FF;
488                         backbuffer[i] = 0x000000FF | (pixel << 16) | (pixel << 8);
489                 }
490
491                 memcpy(videoWidget->buffer, backbuffer, videoWidget->rasterHeight * videoWidget->textureWidth * sizeof(uint32_t));
492
493                 videoWidget->updateGL();
494         }
495 }
496
497 void MainWin::SetZoom100(void)
498 {
499         zoomLevel = 1;
500         ResizeMainWindow();
501 }
502
503 void MainWin::SetZoom200(void)
504 {
505         zoomLevel = 2;
506         ResizeMainWindow();
507 }
508
509 void MainWin::SetZoom300(void)
510 {
511         zoomLevel = 3;
512         ResizeMainWindow();
513 }
514
515 void MainWin::SetNTSC(void)
516 {
517         vjs.hardwareTypeNTSC = true;
518         ResizeMainWindow();
519 }
520
521 void MainWin::SetPAL(void)
522 {
523         vjs.hardwareTypeNTSC = false;
524         ResizeMainWindow();
525 }
526
527 void MainWin::ToggleBlur(void)
528 {
529         vjs.glFilter = !vjs.glFilter;
530 }
531
532 void MainWin::ShowAboutWin(void)
533 {
534         aboutWin->show();
535 }
536
537 void MainWin::InsertCart(void)
538 {
539         if (running)
540         {
541                 ToggleRunState();
542                 pauseForFileSelector = true;
543         }
544
545         filePickWin->show();
546 }
547
548 void MainWin::Unpause(void)
549 {
550         // Here we unpause the emulator if it was paused when we went into the file selector
551         if (pauseForFileSelector)
552         {
553                 pauseForFileSelector = false;
554                 ToggleRunState();
555         }
556 }
557
558 void MainWin::LoadSoftware(QString file)
559 {
560         running = false;                                                        // Prevent bad things(TM) from happening...
561         pauseForFileSelector = false;                           // Reset the file selector pause flag
562         SET32(jaguarMainRAM, 0, 0x00200000);            // Set top of stack...
563         cartridgeLoaded = (JaguarLoadFile(file.toAscii().data()) ? true : false);
564
565         char * biosPointer = jaguarBootROM;
566
567         if (vjs.hardwareTypeAlpine)
568                 biosPointer = jaguarDevBootROM2;
569
570         memcpy(jagMemSpace + 0xE00000, biosPointer, 0x20000);
571
572         powerAct->setDisabled(false);
573         powerAct->setChecked(true);
574         powerButtonOn = false;
575         TogglePowerState();
576
577         if (!vjs.hardwareTypeAlpine)
578         {
579                 QString newTitle = QString("Virtual Jaguar " VJ_RELEASE_VERSION " - Now playing: %1")
580                         .arg(filePickWin->GetSelectedPrettyName());
581                 setWindowTitle(newTitle);
582         }
583 }
584
585 void MainWin::ToggleCDUsage(void)
586 {
587         CDActive = !CDActive;
588
589         if (CDActive)
590         {
591                 powerAct->setDisabled(false);
592         }
593         else
594         {
595                 powerAct->setDisabled(true);
596         }
597 }
598
599 void MainWin::ResizeMainWindow(void)
600 {
601         videoWidget->setFixedSize(zoomLevel * 320, zoomLevel * (vjs.hardwareTypeNTSC ? 240 : 256));
602         show();
603
604         for(int i=0; i<2; i++)
605         {
606                 resize(0, 0);
607                 usleep(2000);
608                 QApplication::processEvents();
609         }
610 }
611
612 void MainWin::ReadSettings(void)
613 {
614         QSettings settings("Underground Software", "Virtual Jaguar");
615         QPoint pos = settings.value("pos", QPoint(200, 200)).toPoint();
616         QSize size = settings.value("size", QSize(400, 400)).toSize();
617         resize(size);
618         move(pos);
619         pos = settings.value("cartLoadPos", QPoint(200, 200)).toPoint();
620         filePickWin->move(pos);
621
622         zoomLevel = settings.value("zoom", 1).toInt();
623         allowUnknownSoftware = settings.value("showUnknownSoftware", false).toBool();
624
625         vjs.useJoystick      = settings.value("useJoystick", false).toBool();
626         vjs.joyport          = settings.value("joyport", 0).toInt();
627         vjs.hardwareTypeNTSC = settings.value("hardwareTypeNTSC", true).toBool();
628         vjs.frameSkip        = settings.value("frameSkip", 0).toInt();
629         vjs.useJaguarBIOS    = settings.value("useJaguarBIOS", false).toBool();
630         vjs.DSPEnabled       = settings.value("DSPEnabled", false).toBool();
631         vjs.audioEnabled     = settings.value("audioEnabled", true).toBool();
632         vjs.usePipelinedDSP  = settings.value("usePipelinedDSP", false).toBool();
633         vjs.fullscreen       = settings.value("fullscreen", false).toBool();
634         vjs.useOpenGL        = settings.value("useOpenGL", true).toBool();
635         vjs.glFilter         = settings.value("glFilterType", 0).toInt();
636         vjs.renderType       = settings.value("renderType", 0).toInt();
637         vjs.allowWritesToROM = settings.value("writeROM", false).toBool();
638 //      strcpy(vjs.jagBootPath, settings.value("JagBootROM", "./bios/[BIOS] Atari Jaguar (USA, Europe).zip").toString().toAscii().data());
639 //      strcpy(vjs.CDBootPath, settings.value("CDBootROM", "./bios/jagcd.rom").toString().toAscii().data());
640         strcpy(vjs.EEPROMPath, settings.value("EEPROMs", "./eeproms/").toString().toAscii().data());
641         strcpy(vjs.ROMPath, settings.value("ROMs", "./software/").toString().toAscii().data());
642         strcpy(vjs.alpineROMPath, settings.value("DefaultROM", "").toString().toAscii().data());
643         strcpy(vjs.absROMPath, settings.value("DefaultABS", "").toString().toAscii().data());
644 WriteLog("MainWin: Paths\n");
645 //WriteLog("    jagBootPath = \"%s\"\n", vjs.jagBootPath);
646 //WriteLog("    CDBootPath  = \"%s\"\n", vjs.CDBootPath);
647 WriteLog("   EEPROMPath = \"%s\"\n", vjs.EEPROMPath);
648 WriteLog("      ROMPath = \"%s\"\n", vjs.ROMPath);
649 WriteLog("AlpineROMPath = \"%s\"\n", vjs.alpineROMPath);
650 WriteLog("   absROMPath = \"%s\"\n", vjs.absROMPath);
651
652         // Keybindings in order of U, D, L, R, C, B, A, Op, Pa, 0-9, #, *
653         vjs.p1KeyBindings[BUTTON_U] = settings.value("p1k_up", Qt::Key_Up).toInt();
654         vjs.p1KeyBindings[BUTTON_D] = settings.value("p1k_down", Qt::Key_Down).toInt();
655         vjs.p1KeyBindings[BUTTON_L] = settings.value("p1k_left", Qt::Key_Left).toInt();
656         vjs.p1KeyBindings[BUTTON_R] = settings.value("p1k_right", Qt::Key_Right).toInt();
657         vjs.p1KeyBindings[BUTTON_C] = settings.value("p1k_c", Qt::Key_Z).toInt();
658         vjs.p1KeyBindings[BUTTON_B] = settings.value("p1k_b", Qt::Key_X).toInt();
659         vjs.p1KeyBindings[BUTTON_A] = settings.value("p1k_a", Qt::Key_C).toInt();
660         vjs.p1KeyBindings[BUTTON_OPTION] = settings.value("p1k_option", Qt::Key_Apostrophe).toInt();
661         vjs.p1KeyBindings[BUTTON_PAUSE] = settings.value("p1k_pause", Qt::Key_Return).toInt();
662         vjs.p1KeyBindings[BUTTON_0] = settings.value("p1k_0", Qt::Key_0).toInt();
663         vjs.p1KeyBindings[BUTTON_1] = settings.value("p1k_1", Qt::Key_1).toInt();
664         vjs.p1KeyBindings[BUTTON_2] = settings.value("p1k_2", Qt::Key_2).toInt();
665         vjs.p1KeyBindings[BUTTON_3] = settings.value("p1k_3", Qt::Key_3).toInt();
666         vjs.p1KeyBindings[BUTTON_4] = settings.value("p1k_4", Qt::Key_4).toInt();
667         vjs.p1KeyBindings[BUTTON_5] = settings.value("p1k_5", Qt::Key_5).toInt();
668         vjs.p1KeyBindings[BUTTON_6] = settings.value("p1k_6", Qt::Key_6).toInt();
669         vjs.p1KeyBindings[BUTTON_7] = settings.value("p1k_7", Qt::Key_7).toInt();
670         vjs.p1KeyBindings[BUTTON_8] = settings.value("p1k_8", Qt::Key_8).toInt();
671         vjs.p1KeyBindings[BUTTON_9] = settings.value("p1k_9", Qt::Key_9).toInt();
672         vjs.p1KeyBindings[BUTTON_d] = settings.value("p1k_pound", Qt::Key_Slash).toInt();
673         vjs.p1KeyBindings[BUTTON_s] = settings.value("p1k_star", Qt::Key_Asterisk).toInt();
674
675         vjs.p2KeyBindings[BUTTON_U] = settings.value("p2k_up", Qt::Key_Up).toInt();
676         vjs.p2KeyBindings[BUTTON_D] = settings.value("p2k_down", Qt::Key_Down).toInt();
677         vjs.p2KeyBindings[BUTTON_L] = settings.value("p2k_left", Qt::Key_Left).toInt();
678         vjs.p2KeyBindings[BUTTON_R] = settings.value("p2k_right", Qt::Key_Right).toInt();
679         vjs.p2KeyBindings[BUTTON_C] = settings.value("p2k_c", Qt::Key_Z).toInt();
680         vjs.p2KeyBindings[BUTTON_B] = settings.value("p2k_b", Qt::Key_X).toInt();
681         vjs.p2KeyBindings[BUTTON_A] = settings.value("p2k_a", Qt::Key_C).toInt();
682         vjs.p2KeyBindings[BUTTON_OPTION] = settings.value("p2k_option", Qt::Key_Apostrophe).toInt();
683         vjs.p2KeyBindings[BUTTON_PAUSE] = settings.value("p2k_pause", Qt::Key_Return).toInt();
684         vjs.p2KeyBindings[BUTTON_0] = settings.value("p2k_0", Qt::Key_0).toInt();
685         vjs.p2KeyBindings[BUTTON_1] = settings.value("p2k_1", Qt::Key_1).toInt();
686         vjs.p2KeyBindings[BUTTON_2] = settings.value("p2k_2", Qt::Key_2).toInt();
687         vjs.p2KeyBindings[BUTTON_3] = settings.value("p2k_3", Qt::Key_3).toInt();
688         vjs.p2KeyBindings[BUTTON_4] = settings.value("p2k_4", Qt::Key_4).toInt();
689         vjs.p2KeyBindings[BUTTON_5] = settings.value("p2k_5", Qt::Key_5).toInt();
690         vjs.p2KeyBindings[BUTTON_6] = settings.value("p2k_6", Qt::Key_6).toInt();
691         vjs.p2KeyBindings[BUTTON_7] = settings.value("p2k_7", Qt::Key_7).toInt();
692         vjs.p2KeyBindings[BUTTON_8] = settings.value("p2k_8", Qt::Key_8).toInt();
693         vjs.p2KeyBindings[BUTTON_9] = settings.value("p2k_9", Qt::Key_9).toInt();
694         vjs.p2KeyBindings[BUTTON_d] = settings.value("p2k_pound", Qt::Key_Slash).toInt();
695         vjs.p2KeyBindings[BUTTON_s] = settings.value("p2k_star", Qt::Key_Asterisk).toInt();
696 }
697
698 void MainWin::WriteSettings(void)
699 {
700         QSettings settings("Underground Software", "Virtual Jaguar");
701         settings.setValue("pos", pos());
702         settings.setValue("size", size());
703         settings.setValue("cartLoadPos", filePickWin->pos());
704
705         settings.setValue("zoom", zoomLevel);
706         settings.setValue("showUnknownSoftware", allowUnknownSoftware);
707
708         settings.setValue("useJoystick", vjs.useJoystick);
709         settings.setValue("joyport", vjs.joyport);
710         settings.setValue("hardwareTypeNTSC", vjs.hardwareTypeNTSC);
711         settings.setValue("frameSkip", vjs.frameSkip);
712         settings.setValue("useJaguarBIOS", vjs.useJaguarBIOS);
713         settings.setValue("DSPEnabled", vjs.DSPEnabled);
714         settings.setValue("audioEnabled", vjs.audioEnabled);
715         settings.setValue("usePipelinedDSP", vjs.usePipelinedDSP);
716         settings.setValue("fullscreen", vjs.fullscreen);
717         settings.setValue("useOpenGL", vjs.useOpenGL);
718         settings.setValue("glFilterType", vjs.glFilter);
719         settings.setValue("renderType", vjs.renderType);
720         settings.setValue("writeROM", vjs.allowWritesToROM);
721         settings.setValue("JagBootROM", vjs.jagBootPath);
722         settings.setValue("CDBootROM", vjs.CDBootPath);
723         settings.setValue("EEPROMs", vjs.EEPROMPath);
724         settings.setValue("ROMs", vjs.ROMPath);
725         settings.setValue("DefaultROM", vjs.alpineROMPath);
726         settings.setValue("DefaultABS", vjs.absROMPath);
727
728         settings.setValue("p1k_up", vjs.p1KeyBindings[BUTTON_U]);
729         settings.setValue("p1k_down", vjs.p1KeyBindings[BUTTON_D]);
730         settings.setValue("p1k_left", vjs.p1KeyBindings[BUTTON_L]);
731         settings.setValue("p1k_right", vjs.p1KeyBindings[BUTTON_R]);
732         settings.setValue("p1k_c", vjs.p1KeyBindings[BUTTON_C]);
733         settings.setValue("p1k_b", vjs.p1KeyBindings[BUTTON_B]);
734         settings.setValue("p1k_a", vjs.p1KeyBindings[BUTTON_A]);
735         settings.setValue("p1k_option", vjs.p1KeyBindings[BUTTON_OPTION]);
736         settings.setValue("p1k_pause", vjs.p1KeyBindings[BUTTON_PAUSE]);
737         settings.setValue("p1k_0", vjs.p1KeyBindings[BUTTON_0]);
738         settings.setValue("p1k_1", vjs.p1KeyBindings[BUTTON_1]);
739         settings.setValue("p1k_2", vjs.p1KeyBindings[BUTTON_2]);
740         settings.setValue("p1k_3", vjs.p1KeyBindings[BUTTON_3]);
741         settings.setValue("p1k_4", vjs.p1KeyBindings[BUTTON_4]);
742         settings.setValue("p1k_5", vjs.p1KeyBindings[BUTTON_5]);
743         settings.setValue("p1k_6", vjs.p1KeyBindings[BUTTON_6]);
744         settings.setValue("p1k_7", vjs.p1KeyBindings[BUTTON_7]);
745         settings.setValue("p1k_8", vjs.p1KeyBindings[BUTTON_8]);
746         settings.setValue("p1k_9", vjs.p1KeyBindings[BUTTON_9]);
747         settings.setValue("p1k_pound", vjs.p1KeyBindings[BUTTON_d]);
748         settings.setValue("p1k_star", vjs.p1KeyBindings[BUTTON_s]);
749
750         settings.setValue("p2k_up", vjs.p2KeyBindings[BUTTON_U]);
751         settings.setValue("p2k_down", vjs.p2KeyBindings[BUTTON_D]);
752         settings.setValue("p2k_left", vjs.p2KeyBindings[BUTTON_L]);
753         settings.setValue("p2k_right", vjs.p2KeyBindings[BUTTON_R]);
754         settings.setValue("p2k_c", vjs.p2KeyBindings[BUTTON_C]);
755         settings.setValue("p2k_b", vjs.p2KeyBindings[BUTTON_B]);
756         settings.setValue("p2k_a", vjs.p2KeyBindings[BUTTON_A]);
757         settings.setValue("p2k_option", vjs.p2KeyBindings[BUTTON_OPTION]);
758         settings.setValue("p2k_pause", vjs.p2KeyBindings[BUTTON_PAUSE]);
759         settings.setValue("p2k_0", vjs.p2KeyBindings[BUTTON_0]);
760         settings.setValue("p2k_1", vjs.p2KeyBindings[BUTTON_1]);
761         settings.setValue("p2k_2", vjs.p2KeyBindings[BUTTON_2]);
762         settings.setValue("p2k_3", vjs.p2KeyBindings[BUTTON_3]);
763         settings.setValue("p2k_4", vjs.p2KeyBindings[BUTTON_4]);
764         settings.setValue("p2k_5", vjs.p2KeyBindings[BUTTON_5]);
765         settings.setValue("p2k_6", vjs.p2KeyBindings[BUTTON_6]);
766         settings.setValue("p2k_7", vjs.p2KeyBindings[BUTTON_7]);
767         settings.setValue("p2k_8", vjs.p2KeyBindings[BUTTON_8]);
768         settings.setValue("p2k_9", vjs.p2KeyBindings[BUTTON_9]);
769         settings.setValue("p2k_pound", vjs.p2KeyBindings[BUTTON_d]);
770         settings.setValue("p2k_star", vjs.p2KeyBindings[BUTTON_s]);
771 }
772
773 // Here's how Byuu does it...
774 // I think I have it working now... :-)
775 #if 0
776 void Utility::resizeMainWindow()
777 {
778   unsigned region = config().video.context->region;
779   unsigned multiplier = config().video.context->multiplier;
780   unsigned width = 256 * multiplier;
781   unsigned height = (region == 0 ? 224 : 239) * multiplier;
782
783   if(config().video.context->correctAspectRatio)
784   {
785     if(region == 0)
786         {
787       width = (double)width * config().video.ntscAspectRatio + 0.5;  //NTSC adjust
788     }
789         else
790         {
791       width = (double)width * config().video.palAspectRatio  + 0.5;  //PAL adjust
792     }
793   }
794
795   if(config().video.isFullscreen == false)
796   {
797     //get effective desktop work area region (ignore Windows taskbar, OS X dock, etc.)
798     QRect deskRect = QApplication::desktop()->availableGeometry(mainWindow);
799
800     //ensure window size will not be larger than viewable desktop area
801     constrainSize(height, width, deskRect.height()); //- frameHeight);
802     constrainSize(width, height, deskRect.width());  //- frameWidth );
803
804     mainWindow->canvas->setFixedSize(width, height);
805     mainWindow->show();
806   }
807   else
808   {
809     for(unsigned i = 0; i < 2; i++)
810         {
811       unsigned iWidth = width, iHeight = height;
812
813       constrainSize(iHeight, iWidth, mainWindow->canvasContainer->size().height());
814       constrainSize(iWidth, iHeight, mainWindow->canvasContainer->size().width());
815
816       //center canvas onscreen; ensure it is not larger than viewable area
817       mainWindow->canvas->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
818       mainWindow->canvas->setFixedSize(iWidth, iHeight);
819       mainWindow->canvas->setMinimumSize(0, 0);
820
821       usleep(2000);
822       QApplication::processEvents();
823     }
824   }
825
826   //workaround for Qt/Xlib bug:
827   //if window resize occurs with cursor over it, Qt shows Qt::Size*DiagCursor;
828   //so force it to show Qt::ArrowCursor, as expected
829   mainWindow->setCursor(Qt::ArrowCursor);
830   mainWindow->canvasContainer->setCursor(Qt::ArrowCursor);
831   mainWindow->canvas->setCursor(Qt::ArrowCursor);
832
833   //workaround for DirectSound(?) bug:
834   //window resizing sometimes breaks audio sync, this call re-initializes it
835   updateAvSync();
836 }
837
838 void Utility::setScale(unsigned scale)
839 {
840   config().video.context->multiplier = scale;
841   resizeMainWindow();
842   mainWindow->shrink();
843   mainWindow->syncUi();
844 }
845
846 void QbWindow::shrink()
847 {
848   if(config().video.isFullscreen == false)
849   {
850     for(unsigned i = 0; i < 2; i++)
851         {
852       resize(0, 0);
853       usleep(2000);
854       QApplication::processEvents();
855     }
856   }
857 }
858 #endif