/******************************************************************************** * FaceTrackNoIR This program is a private project of the some enthusiastic * * gamers from Holland, who don't like to pay much for * * head-tracking. * * * * Copyright (C) 2010-2011 Wim Vriend (Developing) * * Ron Hendriks (Researching and Testing) * * * * Homepage * * * * This program is free software; you can redistribute it and/or modify it * * under the terms of the GNU General Public License as published by the * * Free Software Foundation; either version 3 of the License, or (at your * * option) any later version. * * * * This program is distributed in the hope that it will be useful, but * * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * * more details. * * * * You should have received a copy of the GNU General Public License along * * with this program; if not, see . * * * * FTNoIR_Protocol_SC FTNoIR_Protocol_SC is the Class, that communicates headpose-data * * to games, using the SimConnect.dll. * * SimConnect.dll is a so called 'side-by-side' assembly, so it * * must be treated as such... * ********************************************************************************/ /* Modifications (last one on top): 20110401 - WVR: Moved protocol to a DLL, convenient for installation etc. 20101224 - WVR: Base class is no longer inheriting QThread. sendHeadposeToGame is called from run() of Tracker.cpp */ #include "ftnoir_protocol_sc.h" importSimConnect_CameraSetRelative6DOF FTNoIR_Protocol_SC::simconnect_set6DOF; HANDLE FTNoIR_Protocol_SC::hSimConnect = 0; // Handle to SimConnect float FTNoIR_Protocol_SC::virtSCPosX = 0.0f; // Headpose float FTNoIR_Protocol_SC::virtSCPosY = 0.0f; float FTNoIR_Protocol_SC::virtSCPosZ = 0.0f; float FTNoIR_Protocol_SC::virtSCRotX = 0.0f; float FTNoIR_Protocol_SC::virtSCRotY = 0.0f; float FTNoIR_Protocol_SC::virtSCRotZ = 0.0f; float FTNoIR_Protocol_SC::prevSCPosX = 0.0f; // previous Headpose float FTNoIR_Protocol_SC::prevSCPosY = 0.0f; float FTNoIR_Protocol_SC::prevSCPosZ = 0.0f; float FTNoIR_Protocol_SC::prevSCRotX = 0.0f; float FTNoIR_Protocol_SC::prevSCRotY = 0.0f; float FTNoIR_Protocol_SC::prevSCRotZ = 0.0f; /** constructor **/ FTNoIR_Protocol_SC::FTNoIR_Protocol_SC() { ProgramName = "Microsoft FSX"; blnSimConnectActive = false; hSimConnect = 0; } /** destructor **/ FTNoIR_Protocol_SC::~FTNoIR_Protocol_SC() { qDebug() << "~FTNoIR_Protocol_SC says: inside" << FTNoIR_Protocol_SC::hSimConnect; if (hSimConnect != 0) { qDebug() << "~FTNoIR_Protocol_SC says: before simconnect_close"; if (SUCCEEDED( simconnect_close( FTNoIR_Protocol_SC::hSimConnect ) ) ) { qDebug() << "~FTNoIR_Protocol_SC says: close SUCCEEDED"; } } // SCClientLib.unload(); Generates crash when tracker is ended... } /** helper to Auto-destruct **/ void FTNoIR_Protocol_SC::Release() { delete this; } void FTNoIR_Protocol_SC::Initialize() { return; } // // Load the current Settings from the currently 'active' INI-file. // void FTNoIR_Protocol_SC::loadSettings() { // None yet... } // // Update Headpose in Game. // void FTNoIR_Protocol_SC::sendHeadposeToGame( T6DOF *headpose ) { virtSCRotX = -1.0f * headpose->position.pitch; // degrees virtSCRotY = -1.0f * headpose->position.yaw; virtSCRotZ = headpose->position.roll; virtSCPosX = headpose->position.x/100.f; // cm to meters virtSCPosY = headpose->position.y/100.f; virtSCPosZ = -1.0f * headpose->position.z/100.f; // // It's only useful to send data, if the connection was made. // if (!blnSimConnectActive) { if (SUCCEEDED(simconnect_open(&hSimConnect, "FaceTrackNoIR", NULL, 0, 0, 0))) { qDebug() << "FTNoIR_Protocol_SC::sendHeadposeToGame() says: SimConnect active!"; //set up the events we want to listen for HRESULT hr; simconnect_subscribetosystemevent(hSimConnect, EVENT_PING, "Frame"); hr = simconnect_mapclienteventtosimevent(hSimConnect, EVENT_INIT, ""); hr = simconnect_addclienteventtonotificationgroup(hSimConnect, GROUP0, EVENT_INIT, false); hr = simconnect_setnotificationgrouppriority(hSimConnect, GROUP0, SIMCONNECT_GROUP_PRIORITY_HIGHEST); ////hr = SimConnect_MapInputEventToClientEvent(hSimConnect, INPUT0, "VK_COMMA", EVENT_INIT); ////hr = SimConnect_SetInputGroupState(hSimConnect, INPUT0, SIMCONNECT_STATE_ON); blnSimConnectActive = true; } } else { // // Write the 6DOF-data to FSX // // // // Only do this when the data has changed. This way, the HAT-switch can be used when tracking is OFF. // // // if ((prevPosX != virtPosX) || (prevPosY != virtPosY) || (prevPosZ != virtPosZ) || // (prevRotX != virtRotX) || (prevRotY != virtRotY) || (prevRotZ != virtRotZ)) { //// if (S_OK == simconnect_set6DOF(hSimConnect, virtPosX, virtPosY, virtPosZ, virtRotX, virtRotZ, virtRotY)) { //// qDebug() << "FTNoIR_Protocol_SC::run() says: SimConnect data written!"; //// } // } // // prevPosX = virtPosX; // prevPosY = virtPosY; // prevPosZ = virtPosZ; // prevRotX = virtRotX; // prevRotY = virtRotY; // prevRotZ = virtRotZ; if SUCCEEDED(simconnect_calldispatch(hSimConnect, processNextSimconnectEvent, NULL)) { qDebug() << "FTNoIR_Protocol_SC::sendHeadposeToGame() says: Dispatching"; } else { qDebug() << "FTNoIR_Protocol_SC::sendHeadposeToGame() says: Error Dispatching!"; } } } // // Returns 'true' if all seems OK. // bool FTNoIR_Protocol_SC::checkServerInstallationOK( HANDLE handle ) { QString aFileName; // File Path and Name // Code to activate the context for the SimConnect DLL ACTCTX act = { 0 }; HANDLE hctx; ULONG_PTR ulCookie; qDebug() << "SCCheckClientDLL says: Starting Function"; try { act.cbSize = sizeof(act); act.dwFlags = ACTCTX_FLAG_RESOURCE_NAME_VALID; QString manifest(QCoreApplication::applicationDirPath()); // manifest += "\\FaceTrackNoIR.exe"; manifest += "\\FTNoIR_Protocol_SC.dll"; const wchar_t * encodedName = reinterpret_cast(manifest.utf16()); act.lpSource = encodedName; act.lpResourceName = MAKEINTRESOURCE(101); hctx = CreateActCtx (&act); if (hctx != INVALID_HANDLE_VALUE) { if (!ActivateActCtx(hctx, &ulCookie)) { ReleaseActCtx(hctx); qDebug() << "SCCheckClientDLL says: Error activating SimConnect manifest"; } } else { qDebug() << "SCCheckClientDLL says: Error INVALID_HANDLE: " << GetLastError(); return false; } // // Just try to load the DLL. Let Windows handle the PATH's and such trivialities... // aFileName = SC_CLIENT_FILENAME; // // Load the DLL. // SCClientLib.setFileName(aFileName); if (SCClientLib.load() != true) { qDebug() << "SCCheckClientDLL says: Error loading SimConnect DLL"; return false; } // // Deactivate the context again: the function-references should stay in-tact... // DeactivateActCtx(0, ulCookie); ReleaseActCtx(hctx); } catch(...) { qDebug() << "SCCheckClientDLL says: Error loading SimConnect DLL"; return false; } // // Get the functions from the DLL. // simconnect_open = (importSimConnect_Open) SCClientLib.resolve("SimConnect_Open"); if (simconnect_open == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_Open function not found in DLL!"; return false; } simconnect_set6DOF = (importSimConnect_CameraSetRelative6DOF) SCClientLib.resolve("SimConnect_CameraSetRelative6DOF"); if (simconnect_set6DOF == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_CameraSetRelative6DOF function not found in DLL!"; return false; } simconnect_close = (importSimConnect_Close) SCClientLib.resolve("SimConnect_Close"); if (simconnect_close == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_Close function not found in DLL!"; return false; } //return true; simconnect_calldispatch = (importSimConnect_CallDispatch) SCClientLib.resolve("SimConnect_CallDispatch"); if (simconnect_calldispatch == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_CallDispatch function not found in DLL!"; return false; } simconnect_subscribetosystemevent = (importSimConnect_SubscribeToSystemEvent) SCClientLib.resolve("SimConnect_SubscribeToSystemEvent"); if (simconnect_subscribetosystemevent == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_SubscribeToSystemEvent function not found in DLL!"; return false; } simconnect_mapclienteventtosimevent = (importSimConnect_MapClientEventToSimEvent) SCClientLib.resolve("SimConnect_MapClientEventToSimEvent"); if (simconnect_subscribetosystemevent == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_MapClientEventToSimEvent function not found in DLL!"; return false; } simconnect_addclienteventtonotificationgroup = (importSimConnect_AddClientEventToNotificationGroup) SCClientLib.resolve("SimConnect_AddClientEventToNotificationGroup"); if (simconnect_subscribetosystemevent == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_AddClientEventToNotificationGroup function not found in DLL!"; return false; } simconnect_setnotificationgrouppriority = (importSimConnect_SetNotificationGroupPriority) SCClientLib.resolve("SimConnect_SetNotificationGroupPriority"); if (simconnect_subscribetosystemevent == NULL) { qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect_SetNotificationGroupPriority function not found in DLL!"; return false; } qDebug() << "FTNoIR_Protocol_SC::checkServerInstallationOK() says: SimConnect functions resolved in DLL!"; return true; } void CALLBACK FTNoIR_Protocol_SC::processNextSimconnectEvent(SIMCONNECT_RECV* pData, DWORD cbData, void *pContext) { // HRESULT hr; switch(pData->dwID) { case SIMCONNECT_RECV_ID_EVENT: { SIMCONNECT_RECV_EVENT *evt = (SIMCONNECT_RECV_EVENT*)pData; qDebug() << "FTNoIR_Protocol_SC::processNextSimconnectEvent() says: SimConnect active!"; //switch(evt->uEventID) //{ // //case EVENT_CAMERA_RIGHT: // // cameraBank = normalize180( cameraBank + 5.0f); // // hr = SimConnect_CameraSetRelative6DOF(hSimConnect, 0.0f, 0.0f, 0.0f, // // SIMCONNECT_CAMERA_IGNORE_FIELD,SIMCONNECT_CAMERA_IGNORE_FIELD, cameraBank); // // printf("\nCamera Bank = %f", cameraBank); // // break; // //case EVENT_CAMERA_LEFT: // // // // cameraBank = normalize180( cameraBank - 5.0f); // // hr = SimConnect_CameraSetRelative6DOF(hSimConnect, 0.0f, 0.0f, 0.0f, // // SIMCONNECT_CAMERA_IGNORE_FIELD,SIMCONNECT_CAMERA_IGNORE_FIELD, cameraBank); // // // // printf("\nCamera Bank = %f", cameraBank); // // break; // //default: // // break; //} //break; } case SIMCONNECT_RECV_ID_EVENT_FRAME: { // qDebug() << "FTNoIR_Protocol_SC::processNextSimconnectEvent() says: Frame event!"; if ((prevSCPosX != virtSCPosX) || (prevSCPosY != virtSCPosY) || (prevSCPosZ != virtSCPosZ) || (prevSCRotX != virtSCRotX) || (prevSCRotY != virtSCRotY) || (prevSCRotZ != virtSCRotZ)) { if (S_OK == simconnect_set6DOF(hSimConnect, virtSCPosX, virtSCPosY, virtSCPosZ, virtSCRotX, virtSCRotZ, virtSCRotY)) { // qDebug() << "FTNoIR_Protocol_SC::run() says: SimConnect data written!"; } } prevSCPosX = virtSCPosX; prevSCPosY = virtSCPosY; prevSCPosZ = virtSCPosZ; prevSCRotX = virtSCRotX; prevSCRotY = virtSCRotY; prevSCRotZ = virtSCRotZ; } case SIMCONNECT_RECV_ID_EXCEPTION: { SIMCONNECT_RECV_EXCEPTION *except = (SIMCONNECT_RECV_EXCEPTION*)pData; switch (except->dwException) { case SIMCONNECT_EXCEPTION_ERROR: printf("\nCamera error"); break; default: printf("\nException"); break; } break; } case SIMCONNECT_RECV_ID_QUIT: { qDebug() << "FTNoIR_Protocol_SC::processNextSimconnectEvent() says: Quit event!"; // quit = 1; break; } default: break; } } // // Return a name, if present the name from the Game, that is connected... // void FTNoIR_Protocol_SC::getNameFromGame( char *dest ) { sprintf_s(dest, 99, "FSX"); return; } //////////////////////////////////////////////////////////////////////////////// // Factory function that creates instances if the Protocol object. // Export both decorated and undecorated names. // GetProtocol - Undecorated name, which can be easily used with GetProcAddress // Win32 API function. // _GetProtocol@0 - Common name decoration for __stdcall functions in C language. #pragma comment(linker, "/export:GetProtocol=_GetProtocol@0") FTNOIR_PROTOCOL_BASE_EXPORT PROTOCOLHANDLE __stdcall GetProtocol() { return new FTNoIR_Protocol_SC; } //******************************************************************************************************* // FaceTrackNoIR Client Settings-dialog. //******************************************************************************************************* // // Constructor for server-settings-dialog // SCControls::SCControls() : QWidget() { ui.setupUi( this ); // Connect Qt signals to member-functions connect(ui.btnOK, SIGNAL(clicked()), this, SLOT(doOK())); connect(ui.btnCancel, SIGNAL(clicked()), this, SLOT(doCancel())); //connect(ui.cbxSelectPPJoyNumber, SIGNAL(currentIndexChanged(int)), this, SLOT(virtualJoystickSelected( int ))); //for (int i = 1 ; i < 17; i++) { // QString cbxText = QString("Virtual Joystick %1").arg(i); // ui.cbxSelectPPJoyNumber->addItem(QIcon("images/PPJoy.ico"), cbxText); //} // Load the settings from the current .INI-file loadSettings(); } // // Destructor for server-dialog // SCControls::~SCControls() { qDebug() << "~SCControls() says: started"; } void SCControls::Release() { delete this; } // // Initialize tracker-client-dialog // void SCControls::Initialize(QWidget *parent) { QPoint offsetpos(100, 100); if (parent) { this->move(parent->pos() + offsetpos); } show(); } // // OK clicked on server-dialog // void SCControls::doOK() { save(); this->close(); } // override show event void SCControls::showEvent ( QShowEvent * event ) { loadSettings(); } // // Cancel clicked on server-dialog // void SCControls::doCancel() { // // Ask if changed Settings should be saved // if (settingsDirty) { int ret = QMessageBox::question ( this, "Settings have changed", "Do you want to save the settings?", QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel, QMessageBox::Discard ); qDebug() << "doCancel says: answer =" << ret; switch (ret) { case QMessageBox::Save: save(); this->close(); break; case QMessageBox::Discard: this->close(); break; case QMessageBox::Cancel: // Cancel was clicked break; default: // should never be reached break; } } else { this->close(); } } // // Load the current Settings from the currently 'active' INI-file. // void SCControls::loadSettings() { settingsDirty = false; } // // Save the current Settings to the currently 'active' INI-file. // void SCControls::save() { settingsDirty = false; } //////////////////////////////////////////////////////////////////////////////// // Factory function that creates instances if the Protocol-settings dialog object. // Export both decorated and undecorated names. // GetProtocolDialog - Undecorated name, which can be easily used with GetProcAddress // Win32 API function. // _GetProtocolDialog@0 - Common name decoration for __stdcall functions in C language. #pragma comment(linker, "/export:GetProtocolDialog=_GetProtocolDialog@0") FTNOIR_PROTOCOL_BASE_EXPORT PROTOCOLDIALOGHANDLE __stdcall GetProtocolDialog( ) { return new SCControls; }