summaryrefslogtreecommitdiffhomepage
path: root/FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp
diff options
context:
space:
mode:
authorWim Vriend <facetracknoir@gmail.com>2011-04-08 11:30:49 +0000
committerWim Vriend <facetracknoir@gmail.com>2011-04-08 11:30:49 +0000
commit35d8a7e4552ecb4d8a0cef199953c8384ef6dfd5 (patch)
tree18c484d136195210f7af067c020bf47a9febf519 /FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp
parentbf57ac0073435d8e77275b36b67cf831bc02741c (diff)
SimConnect also in DLL
git-svn-id: svn+ssh://svn.code.sf.net/p/facetracknoir/code@68 19e81ba0-9b1a-49c3-bd6c-561e1906d5fb
Diffstat (limited to 'FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp')
-rw-r--r--FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp504
1 files changed, 504 insertions, 0 deletions
diff --git a/FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp b/FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp
new file mode 100644
index 00000000..014e896f
--- /dev/null
+++ b/FTNoIR_Protocol_SC/FTNoIR_Protocol_SC.cpp
@@ -0,0 +1,504 @@
+/********************************************************************************
+* 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 <http://www.gnu.org/licenses/>. *
+* *
+* 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<const wchar_t *>(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;
+ }
+}
+
+
+////////////////////////////////////////////////////////////////////////////////
+// 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;
+}