X-Git-Url: http://shamusworld.gotdns.org/cgi-bin/gitweb.cgi?a=blobdiff_plain;f=src%2Fgui%2Fcontrollertab.cpp;h=98080032851a6d9168816c52ba8e5aad25411654;hb=f005034d4a3990efc8e445ce46222ce32600411b;hp=036855f7e8279bd2b9bcadf4c56e4810f81e83f1;hpb=5da604521611a960140b58a2fb0f236c65610b70;p=virtualjaguar diff --git a/src/gui/controllertab.cpp b/src/gui/controllertab.cpp index 036855f..9808003 100644 --- a/src/gui/controllertab.cpp +++ b/src/gui/controllertab.cpp @@ -5,72 +5,353 @@ // (C) 2011 Underground Software // See the README and GPLv3 files for licensing and warranty information // -// JLH = James L. Hammons +// JLH = James Hammons // // WHO WHEN WHAT // --- ---------- ------------------------------------------------------------ // JLH 06/23/2011 Created this file +// JLH 07/20/2011 Fixed a bunch of stuff +// #include "controllertab.h" +#include "controllerwidget.h" +#include "gamepad.h" +#include "joystick.h" +#include "keygrabber.h" +#include "profile.h" -ControllerTab::ControllerTab(QWidget * parent/*= 0*/): QWidget(parent) -{ - QLabel * img = new QLabel; - img->setPixmap(QPixmap(":/res/controller.png")); +ControllerTab::ControllerTab(QWidget * parent/*= 0*/): QWidget(parent), + label1(new QLabel(tr("Host Device:"))), + label2(new QLabel(tr("Map Name:"))), + label3(new QLabel(tr("Maps to:"))), + deviceList(new QComboBox(this)), + mapNameList(new QComboBox(this)), + mapToList(new QComboBox(this)), + addMapName(new QPushButton(tr("+"))), + deleteMapName(new QPushButton(tr("-"))), + redefineAll(new QPushButton(tr("Define All Inputs"))), + controllerWidget(new ControllerWidget(this)) +{ QVBoxLayout * layout = new QVBoxLayout; - layout->addWidget(img); + QHBoxLayout * top = new QHBoxLayout; + QVBoxLayout * left = new QVBoxLayout; + QVBoxLayout * right = new QVBoxLayout; + QHBoxLayout * middle = new QHBoxLayout; + top->addLayout(left, 0); + top->addLayout(right, 1); + layout->addLayout(top); + left->addWidget(label1, 0, Qt::AlignRight); + left->addWidget(label2, 0, Qt::AlignRight); + left->addWidget(label3, 0, Qt::AlignRight); + right->addWidget(deviceList); + + right->addLayout(middle); + middle->addWidget(mapNameList, 1); + middle->addWidget(addMapName, 0); + middle->addWidget(deleteMapName, 0); + + right->addWidget(mapToList); + layout->addWidget(controllerWidget); + layout->addWidget(redefineAll, 0, Qt::AlignHCenter); setLayout(layout); + // At least by doing this, it keeps the QComboBox from resizing itself too + // large and breaking the layout. :-P + setFixedWidth(sizeHint().width()); + + connect(redefineAll, SIGNAL(clicked()), this, SLOT(DefineAllKeys())); + connect(deviceList, SIGNAL(activated(int)), this, SLOT(ChangeDevice(int))); + connect(mapNameList, SIGNAL(activated(int)), this, SLOT(ChangeMapName(int))); + connect(addMapName, SIGNAL(clicked()), this, SLOT(AddMapName())); + connect(deleteMapName, SIGNAL(clicked()), this, SLOT(DeleteMapName())); + connect(controllerWidget, SIGNAL(KeyDefined(int, uint32_t)), this, SLOT(UpdateProfileKeys(int, uint32_t))); + connect(mapToList, SIGNAL(activated(int)), this, SLOT(UpdateProfileConnections(int))); + + // Set up the device combobox (Keyboard is the default, and always + // present) + deviceList->addItem(tr("Keyboard"), 0); + + for(int i=0; iaddItem(Gamepad::GetJoystickName(i), deviceNum); + } + + // Set up "Map To" combobox + mapToList->addItem(tr("None"), 0); + mapToList->addItem(tr("Controller #1"), CONTROLLER1); + mapToList->addItem(tr("Controller #2"), CONTROLLER2); + mapToList->addItem(tr("Either one that's free"), CONTROLLER1 | CONTROLLER2); } +/* +So now we come to implementation. When changing devices, could have a helper function +in profile.cpp that fills the mapNameList combobox with the appropriate names/profile +numbers. + +There needs to be some way of getting data from the ControllerWidget and the current +profile. + +Gamepad will have to have some way of knowing which profile is mapped to which +Jaguar controllers and filtering out everything else. + +Will have to have some intelligent handling of profiles when first run, to see first +what is connected and second, to assign profiles to Jaguar controllers. In this +case, keyboard is the lowest priority--if a controller is plugged in and assigned to +the same Jaguar controller as a keyboard, the controller is used. Not sure what to +do in the case of multiple controllers plugged in and assigned to the same Jaguar +controller. + +Also, need a way to load/save profiles. + +Meaning of checkboxes: None checked == profile not used. +1 checked == prefer connection to Jaguar controller X. +2 checked == no preference, use any available. + +Single mapping cannot be deleted ("-" will be disabled). Can always add, up to the max +limit of profiles (MAX_PROFILES). + +------------------------------ + +Now the main window passes in/removes the last edited profile #. From here, when starting +up, we need to pull that number from the profile store and populate all our boxes. + +------------------------------- + +Need to do AutoConnectProfiles from here, and detect any conflicts +*/ + ControllerTab::~ControllerTab() { } + +void ControllerTab::SetupLastUsedProfile(void) +{ + int deviceNumIndex = deviceList->findData(profile[profileNum].device); + int mapNumIndex = mapNameList->findText(profile[profileNum].mapName); + + if (deviceNumIndex == -1 || mapNumIndex == -1) + { + // We're doing the default, so set it up... +// mapToList->setCurrentIndex(mapToList->findData(profile[0].preferredController)); +//printf("ControllerTab::SetupLastUsedProfile: [FAILED] profileNum=%i, controllerIndex=%i, preferredController=%i\n", profileNum, controllerIndex, profile[0].preferredController); +// return; + deviceNumIndex = 0; + mapNumIndex = 0; + profileNum = 0; + } + + deviceList->setCurrentIndex(deviceNumIndex); + mapNameList->setCurrentIndex(mapNumIndex); +//no more: #warning "!!! bug in here where it doesn't save your preferred controller !!!" + + int controllerIndex = mapToList->findData(profile[profileNum].preferredController); + mapToList->setCurrentIndex(controllerIndex); +//printf("ControllerTab::SetupLastUsedProfile: profileNum=%i, controllerIndex=%i, preferredController=%i\n", profileNum, controllerIndex, profile[profileNum].preferredController); + + // We have to do this manually, since it's no longer done automagically... + ChangeDevice(deviceNumIndex); + ChangeMapName(mapNumIndex); +} + + +void ControllerTab::DefineAllKeys(void) +{ +// char jagButtonName[21][10] = { "Up", "Down", "Left", "Right", +// "*", "7", "4", "1", "0", "8", "5", "2", "#", "9", "6", "3", +// "A", "B", "C", "Option", "Pause" }; + int orderToDefine[21] = { 0, 1, 2, 3, 18, 17, 16, 20, 19, 7, 11, 15, 6, 10, 14, 5, 9, 13, 8, 4, 12 }; + KeyGrabber keyGrab(this); + + for(int i=BUTTON_FIRST; i<=BUTTON_LAST; i++) + { + keyGrab.SetKeyText(orderToDefine[i]); + keyGrab.exec(); + int key = keyGrab.key; + + if (key == Qt::Key_Escape) + break; + + // Otherwise, populate the appropriate spot in the settings & update screen... + controllerWidget->keys[orderToDefine[i]] = key; + controllerWidget->update(); + profile[profileNum].map[orderToDefine[i]] = key; + } +} + + +void ControllerTab::UpdateProfileKeys(int mapPosition, uint32_t key) +{ + profile[profileNum].map[mapPosition] = key; +} + + +void ControllerTab::UpdateProfileConnections(int selection) +{ +// profile[profileNum].preferredController = (controller1->isChecked() ? CONTROLLER1 : 0) | (controller2->isChecked() ? CONTROLLER2 : 0); + profile[profileNum].preferredController = mapToList->itemData(selection).toInt(); +//printf("Setting profile #%i 'Maps To' to %i...\n", profileNum, mapToList->itemData(selection).toInt()); +} + + +void ControllerTab::ChangeDevice(int selection) +{ +//printf("ControllerTab::ChangeDevice\n"); + int deviceNum = deviceList->itemData(selection).toInt(); + mapNameList->clear(); + int numberOfMappings = FindMappingsForDevice(deviceNum, mapNameList); + deleteMapName->setDisabled(numberOfMappings == 1 ? true : false); +//printf("Found %i mappings for device #%u...\n", numberOfMappings, deviceNum); +} + + +void ControllerTab::ChangeMapName(int selection) +{ +//printf("ControllerTab::ChangeMapName\n"); + profileNum = mapNameList->itemData(selection).toInt(); +//printf("You selected mapping: %s (profile #%u)\n", (mapNameList->itemText(selection)).toAscii().data(), profileNum); + + for(int i=BUTTON_FIRST; i<=BUTTON_LAST; i++) + controllerWidget->keys[i] = profile[profileNum].map[i]; + + controllerWidget->update(); +// controller1->setChecked(profile[profileNum].preferredController & CONTROLLER1); +// controller2->setChecked(profile[profileNum].preferredController & CONTROLLER2); + mapToList->setCurrentIndex(mapToList->findData(profile[profileNum].preferredController)); +} + + +void ControllerTab::AddMapName(void) +{ + int freeProfile = GetFreeProfile(); + + if (freeProfile == -1) + { + // Oh crap, we're out of room! Alert the media! + // (Really, tho, we should pop this up *before* asking for user input. + // Which we now do!) #if 0 - // Keybindings in order of U, D, L, R, C, B, A, Op, Pa, 0-9, #, * - vjs.p1KeyBindings[BUTTON_U] = settings.value("p1k_up", Qt::Key_Up).toInt(); - vjs.p1KeyBindings[BUTTON_D] = settings.value("p1k_down", Qt::Key_Down).toInt(); - vjs.p1KeyBindings[BUTTON_L] = settings.value("p1k_left", Qt::Key_Left).toInt(); - vjs.p1KeyBindings[BUTTON_R] = settings.value("p1k_right", Qt::Key_Right).toInt(); - vjs.p1KeyBindings[BUTTON_C] = settings.value("p1k_c", Qt::Key_Z).toInt(); - vjs.p1KeyBindings[BUTTON_B] = settings.value("p1k_b", Qt::Key_X).toInt(); - vjs.p1KeyBindings[BUTTON_A] = settings.value("p1k_a", Qt::Key_C).toInt(); - vjs.p1KeyBindings[BUTTON_OPTION] = settings.value("p1k_option", Qt::Key_Apostrophe).toInt(); - vjs.p1KeyBindings[BUTTON_PAUSE] = settings.value("p1k_pause", Qt::Key_Return).toInt(); - vjs.p1KeyBindings[BUTTON_0] = settings.value("p1k_0", Qt::Key_0).toInt(); - vjs.p1KeyBindings[BUTTON_1] = settings.value("p1k_1", Qt::Key_1).toInt(); - vjs.p1KeyBindings[BUTTON_2] = settings.value("p1k_2", Qt::Key_2).toInt(); - vjs.p1KeyBindings[BUTTON_3] = settings.value("p1k_3", Qt::Key_3).toInt(); - vjs.p1KeyBindings[BUTTON_4] = settings.value("p1k_4", Qt::Key_4).toInt(); - vjs.p1KeyBindings[BUTTON_5] = settings.value("p1k_5", Qt::Key_5).toInt(); - vjs.p1KeyBindings[BUTTON_6] = settings.value("p1k_6", Qt::Key_6).toInt(); - vjs.p1KeyBindings[BUTTON_7] = settings.value("p1k_7", Qt::Key_7).toInt(); - vjs.p1KeyBindings[BUTTON_8] = settings.value("p1k_8", Qt::Key_8).toInt(); - vjs.p1KeyBindings[BUTTON_9] = settings.value("p1k_9", Qt::Key_9).toInt(); - vjs.p1KeyBindings[BUTTON_d] = settings.value("p1k_pound", Qt::Key_Slash).toInt(); - vjs.p1KeyBindings[BUTTON_s] = settings.value("p1k_star", Qt::Key_Asterisk).toInt(); - - vjs.p2KeyBindings[BUTTON_U] = settings.value("p2k_up", Qt::Key_Up).toInt(); - vjs.p2KeyBindings[BUTTON_D] = settings.value("p2k_down", Qt::Key_Down).toInt(); - vjs.p2KeyBindings[BUTTON_L] = settings.value("p2k_left", Qt::Key_Left).toInt(); - vjs.p2KeyBindings[BUTTON_R] = settings.value("p2k_right", Qt::Key_Right).toInt(); - vjs.p2KeyBindings[BUTTON_C] = settings.value("p2k_c", Qt::Key_Z).toInt(); - vjs.p2KeyBindings[BUTTON_B] = settings.value("p2k_b", Qt::Key_X).toInt(); - vjs.p2KeyBindings[BUTTON_A] = settings.value("p2k_a", Qt::Key_C).toInt(); - vjs.p2KeyBindings[BUTTON_OPTION] = settings.value("p2k_option", Qt::Key_Apostrophe).toInt(); - vjs.p2KeyBindings[BUTTON_PAUSE] = settings.value("p2k_pause", Qt::Key_Return).toInt(); - vjs.p2KeyBindings[BUTTON_0] = settings.value("p2k_0", Qt::Key_0).toInt(); - vjs.p2KeyBindings[BUTTON_1] = settings.value("p2k_1", Qt::Key_1).toInt(); - vjs.p2KeyBindings[BUTTON_2] = settings.value("p2k_2", Qt::Key_2).toInt(); - vjs.p2KeyBindings[BUTTON_3] = settings.value("p2k_3", Qt::Key_3).toInt(); - vjs.p2KeyBindings[BUTTON_4] = settings.value("p2k_4", Qt::Key_4).toInt(); - vjs.p2KeyBindings[BUTTON_5] = settings.value("p2k_5", Qt::Key_5).toInt(); - vjs.p2KeyBindings[BUTTON_6] = settings.value("p2k_6", Qt::Key_6).toInt(); - vjs.p2KeyBindings[BUTTON_7] = settings.value("p2k_7", Qt::Key_7).toInt(); - vjs.p2KeyBindings[BUTTON_8] = settings.value("p2k_8", Qt::Key_8).toInt(); - vjs.p2KeyBindings[BUTTON_9] = settings.value("p2k_9", Qt::Key_9).toInt(); - vjs.p2KeyBindings[BUTTON_d] = settings.value("p2k_pound", Qt::Key_Slash).toInt(); - vjs.p2KeyBindings[BUTTON_s] = settings.value("p2k_star", Qt::Key_Asterisk).toInt(); + QMessageBox msg; + msg.setText(QString(tr("Can't create any more profiles!"))); + msg.setIcon(QMessageBox::Warning); + msg.exec(); +#else + QMessageBox::warning(this, tr("Houston, we have a problem..."), tr("Can't create any more profiles!")); #endif + + return; + } + + QString text = QInputDialog::getText(this, tr("Add Map Name"), tr("Map name:"), QLineEdit::Normal); + + if (text.isEmpty()) + return; + + // Add mapping... + profileNum = freeProfile; + profile[profileNum].device = deviceList->itemData(deviceList->currentIndex()).toInt(); + strncpy(profile[profileNum].mapName, text.toAscii().data(), 31); + profile[profileNum].mapName[31] = 0; + profile[profileNum].preferredController = CONTROLLER1; + + for(int i=BUTTON_FIRST; iaddItem(text, profileNum); + mapNameList->setCurrentIndex(mapNameList->count() - 1); +} + + +void ControllerTab::DeleteMapName(void) +{ +//printf("Delete current mapping (TODO)...\n"); + + // hmm, don't need to check this... Because presumably, it's already been checked for. +// if (mapNameList->count() == 1) ; + + QMessageBox::StandardButton retVal = QMessageBox::question(this, tr("Remove Mapping"), tr("Are you sure you want to remove this mapping?"), QMessageBox::No | QMessageBox::Yes, QMessageBox::No); + + if (retVal == QMessageBox::No) + return; + + int index = mapNameList->currentIndex(); + int profileToRemove = profileNum; + mapNameList->removeItem(index); + DeleteProfile(profileToRemove); +} + + +/* +The profiles need the following: + + - The name of the controller + - A unique human readable ID + - The key definitions for that controller (keyboard keys can be mixed in) + +So there can be more than one profile for each unique controller; the +relationship is many-to-one. So basically, how it works it like this: SDL +reports all connected controllers. If there are none connected, the default +controller is the keyboard (which can have multiple profiles). The UI only +presents those profiles which are usuable with the controllers that are plugged +in, all else is ignored. The user can pick the profile for the controller and +configure the keys for it; the UI automagically saves everything. + +How to handle the case of identical controllers being plugged in? How does the +UI know which is which? Each controller will have a mapping to a default +Jaguar controller (#1 or #2). Still doesn't prevent confusion though. Actually, +it can: The profile can have a field that maps it to a preferred Jaguar +controller, which can also be both (#1 AND #2--in this case we can set it to +zero which means no preference). If the UI detects two of the same controller +and each can be mapped to the same profile, it assigns them in order since it +doesn't matter, the profiles are identical. + +The default profile is always available and is the keyboard (hey, we're PC +centric here). The default profile is usually #0. + +Can there be more than one keyboard profile? Why not? You will need separate +ones for controller #1 and controller #2. + +A profile might look like this: + +Field 1: Nostomo N45 Analog +Field 2: Dad's #1 +Field 3: Jaguar controller #1 +Field 4: The button/stick mapping + +Profile # would be implicit in the order that they are stored in the internal +data structure. + +When a new controller is plugged in with no profiles attached, it defaults to +a set keyboard layout which the user can change. So every new controller will +always have at least one profile. + +Data structures: +The Gamepad class has the name of the controller (except for Keyboard) +The profile list is just a list +The controller name index + profile index makes a unique key +Probably the best way to deal with it is to stuff the name/profile indices +into the key definition structure. + +#define CONTROLLER1 0x01 +#define CONTROLLER2 0x02 + +struct Profile +{ + int device; // Host device number + char mapName[32]; // Human readable map name + int preferredController; // CONTROLLER1 and/or CONTROLLER2 + int map[21]; // Keys/buttons/axes +}; + +NOTE that device is an int, and the list is maintained elsewhere. It is +*not* the same as what you see in GetJoystickName(); the device names have +to be able to persist even when not available. + +Where to store the master profile list? It has to be accessible to this class. +vjs.profile[x] would be good, but it's not really a concern for the Jaguar core. +So it shouldn't go there. There should be a separate global setting place for +GUI stuff... +*/ +