aboutsummaryrefslogtreecommitdiff
path: root/Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp
diff options
context:
space:
mode:
authorJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
committerJef <jef@targetspot.com>2024-09-24 08:54:57 -0400
commit20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp
parent537bcbc86291b32fc04ae4133ce4d7cac8ebe9a7 (diff)
downloadwinamp-20d28e80a5c861a9d5f449ea911ab75b4f37ad0d.tar.gz
Initial community commit
Diffstat (limited to 'Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp')
-rw-r--r--Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp235
1 files changed, 235 insertions, 0 deletions
diff --git a/Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp b/Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp
new file mode 100644
index 00000000..9ab159ac
--- /dev/null
+++ b/Src/external_dependencies/openmpt-trunk/mptrack/IPCWindow.cpp
@@ -0,0 +1,235 @@
+/*
+* IPCWindow.cpp
+* -------------
+* Purpose: Hidden window to receive file open commands from another OpenMPT instance
+* Notes : (currently none)
+* Authors: OpenMPT Devs
+* The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
+*/
+
+
+#include "stdafx.h"
+#include "IPCWindow.h"
+
+#include "../common/version.h"
+#include "Mptrack.h"
+
+
+OPENMPT_NAMESPACE_BEGIN
+
+namespace IPCWindow
+{
+
+ static constexpr TCHAR ClassName[] = _T("OpenMPT_IPC_Wnd");
+ static HWND ipcWindow = nullptr;
+
+ static LRESULT CALLBACK IPCWindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
+ {
+ if(uMsg == WM_COPYDATA)
+ {
+ const auto &copyData = *reinterpret_cast<const COPYDATASTRUCT *>(lParam);
+ LRESULT result = 0;
+ switch(static_cast<Function>(copyData.dwData))
+ {
+ case Function::Open:
+ {
+ std::size_t count = copyData.cbData / sizeof(WCHAR);
+ const WCHAR* data = static_cast<const WCHAR *>(copyData.lpData);
+ const std::wstring name = std::wstring(data, data + count);
+ result = theApp.OpenDocumentFile(mpt::PathString::FromWide(name).AsNative().c_str()) ? 1 : 2;
+ }
+ break;
+ case Function::SetWindowForeground:
+ {
+ auto mainWnd = theApp.GetMainWnd();
+ if(mainWnd)
+ {
+ if(mainWnd->IsIconic())
+ {
+ mainWnd->ShowWindow(SW_RESTORE);
+ }
+ mainWnd->SetForegroundWindow();
+ result = 1;
+ } else
+ {
+ result = 0;
+ }
+ }
+ break;
+ case Function::GetVersion:
+ {
+ result = Version::Current().GetRawVersion();
+ }
+ break;
+ case Function::GetArchitecture:
+ {
+ #if MPT_OS_WINDOWS
+ result = static_cast<int32>(mpt::OS::Windows::GetProcessArchitecture());
+ #else
+ result = -1;
+ #endif
+ }
+ break;
+ case Function::HasSameBinaryPath:
+ {
+ std::size_t count = copyData.cbData / sizeof(WCHAR);
+ const WCHAR* data = static_cast<const WCHAR *>(copyData.lpData);
+ const std::wstring path = std::wstring(data, data + count);
+ result = (theApp.GetInstallBinArchPath().ToWide() == path) ? 1 : 0;
+ }
+ break;
+ case Function::HasSameSettingsPath:
+ {
+ std::size_t count = copyData.cbData / sizeof(WCHAR);
+ const WCHAR* data = static_cast<const WCHAR *>(copyData.lpData);
+ const std::wstring path = std::wstring(data, data + count);
+ result = (theApp.GetConfigPath().ToWide() == path) ? 1 : 0;
+ }
+ break;
+ default:
+ result = 0;
+ break;
+ }
+ return result;
+ }
+ return ::DefWindowProc(hwnd, uMsg, wParam, lParam);
+ }
+
+ void Open(HINSTANCE hInstance)
+ {
+ WNDCLASS ipcWindowClass =
+ {
+ 0,
+ IPCWindowProc,
+ 0,
+ 0,
+ hInstance,
+ nullptr,
+ nullptr,
+ nullptr,
+ nullptr,
+ ClassName
+ };
+ auto ipcAtom = RegisterClass(&ipcWindowClass);
+ ipcWindow = CreateWindow(MAKEINTATOM(ipcAtom), _T("OpenMPT IPC Window"), 0, CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, 0);
+ }
+
+ void Close()
+ {
+ ::DestroyWindow(ipcWindow);
+ ipcWindow = nullptr;
+ }
+
+ LRESULT SendIPC(HWND ipcWnd, Function function, mpt::const_byte_span data)
+ {
+ if(!ipcWnd)
+ {
+ return 0;
+ }
+ if(!mpt::in_range<DWORD>(data.size()))
+ {
+ return 0;
+ }
+ COPYDATASTRUCT copyData{};
+ copyData.dwData = static_cast<ULONG>(function);
+ copyData.cbData = mpt::saturate_cast<DWORD>(data.size());
+ copyData.lpData = const_cast<void*>(mpt::void_cast<const void*>(data.data()));
+ return ::SendMessage(ipcWnd, WM_COPYDATA, 0, reinterpret_cast<LPARAM>(&copyData));
+ }
+
+ HWND FindIPCWindow()
+ {
+ return ::FindWindow(ClassName, nullptr);
+ }
+
+ struct EnumWindowState
+ {
+ FlagSet<InstanceRequirements> require;
+ HWND result = nullptr;
+ };
+
+ static BOOL CALLBACK EnumWindowsProc(HWND hwnd, LPARAM lParam)
+ {
+ EnumWindowState &state = *reinterpret_cast<EnumWindowState*>(lParam);
+ if(hwnd)
+ {
+ TCHAR className[256];
+ MemsetZero(className);
+ if(::GetClassName(hwnd, className, 256) > 0)
+ {
+ if(!_tcscmp(className, IPCWindow::ClassName))
+ {
+ if(state.require[SameVersion])
+ {
+ if(Version(static_cast<uint32>(SendIPC(hwnd, Function::GetVersion))) != Version::Current())
+ {
+ return TRUE; // continue
+ }
+ }
+ if(state.require[SameArchitecture])
+ {
+ if(SendIPC(hwnd, Function::GetArchitecture) != static_cast<int>(mpt::OS::Windows::GetProcessArchitecture()))
+ {
+ return TRUE; // continue
+ }
+ }
+ if(state.require[SamePath])
+ {
+ if(SendIPC(hwnd, Function::HasSameBinaryPath, mpt::as_span(theApp.GetInstallBinArchPath().ToWide())) != 1)
+ {
+ return TRUE; // continue
+ }
+ }
+ if(state.require[SameSettings])
+ {
+ if(SendIPC(hwnd, Function::HasSameSettingsPath, mpt::as_span(theApp.GetConfigPath().ToWide())) != 1)
+ {
+ return TRUE; // continue
+ }
+ }
+ state.result = hwnd;
+ return TRUE; // continue
+ //return FALSE; // done
+ }
+ }
+ }
+ return TRUE; // continue
+ }
+
+ HWND FindIPCWindow(FlagSet<InstanceRequirements> require)
+ {
+ EnumWindowState state;
+ state.require = require;
+ if(::EnumWindows(&EnumWindowsProc, reinterpret_cast<LPARAM>(&state)) == 0)
+ {
+ return nullptr;
+ }
+ return state.result;
+ }
+
+
+
+ bool SendToIPC(const std::vector<mpt::PathString> &filenames)
+ {
+ HWND ipcWnd = FindIPCWindow();
+ if(!ipcWnd)
+ {
+ return false;
+ }
+ DWORD processID = 0;
+ GetWindowThreadProcessId(ipcWnd, &processID);
+ AllowSetForegroundWindow(processID);
+ SendIPC(ipcWnd, Function::SetWindowForeground);
+ for(const auto &filename : filenames)
+ {
+ if(SendIPC(ipcWnd, Function::Open, mpt::as_span(filename.ToWide())) == 0)
+ {
+ return false;
+ }
+ }
+ return true;
+ }
+
+}
+
+OPENMPT_NAMESPACE_END