aboutsummaryrefslogtreecommitdiff
path: root/Src/winampa/Main.cpp
diff options
context:
space:
mode:
authorJean-Francois Mauguit <jfmauguit@mac.com>2024-09-24 09:03:25 -0400
committerGitHub <noreply@github.com>2024-09-24 09:03:25 -0400
commitbab614c421ed7ae329d26bf028c4a3b1d2450f5a (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/winampa/Main.cpp
parent4bde6044fddf053f31795b9eaccdd2a5a527d21f (diff)
parent20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (diff)
downloadwinamp-bab614c421ed7ae329d26bf028c4a3b1d2450f5a.tar.gz
Merge pull request #5 from WinampDesktop/community
Merge to main
Diffstat (limited to 'Src/winampa/Main.cpp')
-rw-r--r--Src/winampa/Main.cpp628
1 files changed, 628 insertions, 0 deletions
diff --git a/Src/winampa/Main.cpp b/Src/winampa/Main.cpp
new file mode 100644
index 00000000..137e16d1
--- /dev/null
+++ b/Src/winampa/Main.cpp
@@ -0,0 +1,628 @@
+#include "main.h"
+#include "../nu/AutoWide.h"
+#define LANG_STATIC_BUFFER_SIZE 1024
+
+extern "C" void ResolveEnvironmentVariables2(wchar_t *string, wchar_t *destString, size_t stringSize);
+static UINT WM_TASKBARCREATED;
+
+// winamp2/5
+wchar_t ini_file[MAX_PATH] = {0},
+ wa2ConfigDir[MAX_PATH] = {0},
+ winampClassName[MAX_PATH] = {0},
+ winampaLngPath[MAX_PATH] = {0},
+ icon_tmp[MAX_PATH] = {0},
+ winamp_exe_file[MAX_PATH] = {0},
+ bm_file[MAX_PATH] = {0};
+
+static HWND hwndWinamp;
+static HINSTANCE g_hInstance, winampaLng, nativeLng;
+static HICON m_icon;
+
+typedef HRESULT(WINAPI *CHANGEWINDOWMESSAGEFILTER)(UINT message, DWORD dwFlag);
+static CHANGEWINDOWMESSAGEFILTER changeWMFilter;
+
+int config_iconidx = -1, config_systray_icon = 1;
+
+static wchar_t ini_sec[] = L"WinampAgent";
+
+int ReadStr(HANDLE hFile, char *str, int len)
+{
+ while (1)
+ {
+ DWORD l = 0;
+ ReadFile(hFile, str, 1, &l, 0);
+ if (l != 1 || *str == '\r' || *str == '\n')
+ {
+ DWORD t = 0;
+ ReadFile(hFile, str, 1, &t, 0);
+ *str = 0;
+ return (l == 1);
+ }
+ str++;
+ if (--len < 1)
+ {
+ *str = 0;
+ return 1;
+ }
+ }
+}
+
+static BOOL LoadWMFilter(void){
+ if (!changeWMFilter){
+ changeWMFilter = (CHANGEWINDOWMESSAGEFILTER)GetProcAddress(GetModuleHandle(L"USER32"), "ChangeWindowMessageFilter");
+ }
+ return (!!changeWMFilter);
+}
+
+void LoadWinampaLng(void){
+ winampaLng = LoadLibraryExW(winampaLngPath, NULL, LOAD_LIBRARY_AS_DATAFILE);
+}
+
+void UnloadWinampaLng(void){
+ if(winampaLng){
+ FreeLibrary(winampaLng);
+ winampaLng = 0;
+ }
+}
+
+wchar_t* GetStringW(UINT uID)
+{
+ static wchar_t *buf;
+ if (!buf)
+ buf = (wchar_t *)GlobalAlloc(LPTR,(LANG_STATIC_BUFFER_SIZE*sizeof(buf[0])));
+
+ if (!LoadStringW(winampaLng, uID, buf, LANG_STATIC_BUFFER_SIZE))
+ {
+ if (winampaLng == nativeLng || !LoadStringW(nativeLng, uID, buf, LANG_STATIC_BUFFER_SIZE))
+ {
+ lstrcpynW(buf, L"Error loading string", LANG_STATIC_BUFFER_SIZE);
+ }
+ }
+ return buf;
+}
+
+// about the most reliable way i can find to get the Winamp window as it could
+// have been started with the /CLASS= parameter which then means it won't be
+// 'Winamp v1.x' so instead go for a fixed child window which will always be
+// there (and deals with other apps who create a 'fake' Winamp window (like AIMP)
+// and there are two versions to cope with classic or modern skins being used.
+BOOL CALLBACK EnumWindowsProc(HWND hwnd, LPARAM lParam)
+{
+ wchar_t name[24] = {0};
+ GetClassNameW(hwnd, name, 24);
+ // this check will only work for classic skins
+ if (!lstrcmpiW(name, L"Winamp PE"))
+ {
+ HWND child = GetWindow(GetWindow(hwnd, GW_CHILD), GW_CHILD);
+ GetClassNameW(child, name, 24);
+ // this check improves reliability of this check against players
+ // like KMPlayer which also create a fake playlist editor window
+ if (!lstrcmpiW(name, L"WinampVis") || lstrcmpiW(name, L"TSkinPanel"))
+ {
+ hwndWinamp = GetWindow(hwnd, GW_OWNER);
+ return FALSE;
+ }
+ }
+ // this check will only work for modern skins
+ else if (!lstrcmpiW(name, L"BaseWindow_RootWnd"))
+ {
+ HWND child = GetWindow(GetWindow(hwnd,GW_CHILD),GW_CHILD);
+ GetClassNameW(child, name, 24);
+ if (!lstrcmpiW(name, L"Winamp PE") ||
+ !lstrcmpiW(name, L"Winamp Gen"))
+ {
+ hwndWinamp = GetWindow(hwnd,GW_OWNER);
+ return FALSE;
+ }
+ }
+ // and then we just try what we can (default and
+ // taking into account where possible /CLASS use
+ else if (!lstrcmpiW(name, L"Winamp v1.x") ||
+ !lstrcmpiW(name, winampClassName))
+ {
+ HWND child = GetWindow(hwnd,GW_CHILD);
+ GetClassNameW(child, name, 24);
+ if (!lstrcmpiW(name, L"WinampVis"))
+ {
+ hwndWinamp = hwnd;
+ return FALSE;
+ }
+ }
+ return TRUE;
+}
+
+HWND GetWinampHWND(void)
+{
+ // incase things changed since last time, always re-check
+ hwndWinamp = 0;
+ EnumWindows(EnumWindowsProc, 0);
+ return hwndWinamp;
+}
+
+LRESULT CALLBACK WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
+{
+ int force_icon = 0;
+ if (WM_TASKBARCREATED && uMsg == WM_TASKBARCREATED)
+ {
+ uMsg = WM_USER + 1;
+ force_icon = 1;
+ }
+
+ switch (uMsg)
+ {
+ case WM_CREATE:
+ SendMessage(hwnd, WM_USER + 1, 0, 0);
+ return TRUE;
+
+ case WM_USER + 1:
+ {
+ int iconidx;
+ int isintray;
+
+ config_systray_icon = ini_file[0] ? GetPrivateProfileIntW(ini_sec, L"is_intray", 1, ini_file) : 0;
+ iconidx = ini_file[0] ? GetPrivateProfileIntW(L"Winamp", L"sticon", 0, ini_file) : 0;
+
+ isintray = !!systray_isintray();
+
+ if ((isintray && (force_icon || iconidx != config_iconidx)) ||
+ isintray != (config_systray_icon))
+ {
+ HICON m_oldicon = m_icon;
+ m_icon = 0;
+ if (config_systray_icon)
+ {
+ if (iconidx != 0)
+ {
+ HMODULE h = LoadLibraryExW(winamp_exe_file, NULL, LOAD_LIBRARY_AS_DATAFILE);
+ if (h)
+ {
+ int geticonid(int x); // in winampicon.cpp
+
+ int icon_to_use = geticonid(iconidx);
+ if(icon_to_use != -666)
+ {
+ m_icon = (HICON)LoadImage(h,MAKEINTRESOURCE(icon_to_use),IMAGE_ICON,16,16,0);
+ }
+ else
+ {
+ if(PathFileExistsW(icon_tmp))
+ {
+ m_icon = (HICON)LoadImageW(0,icon_tmp,IMAGE_ICON,16,16,LR_LOADFROMFILE);
+ }
+ }
+ FreeLibrary(h);
+ }
+ }
+ if (!m_icon) m_icon = (HICON)LoadImage(g_hInstance, MAKEINTRESOURCE(IDI_ICON1), IMAGE_ICON, 16, 16, 0);
+ if (isintray) systray_mod(hwnd, m_icon, 0);
+ systray_add(hwnd, m_icon, GetStringW(IDS_WINAMP_AGENT));
+ }
+ else systray_del(hwnd);
+
+ if (m_oldicon) DestroyIcon(m_oldicon);
+ }
+ config_iconidx = iconidx;
+ }
+ return 0;
+
+ case WM_CLOSE:
+ DestroyWindow(hwnd);
+ return 0;
+
+ case WM_ENDSESSION: // JF JAN01001 added
+ if (wParam)
+ {
+ ExitProcess(0);
+ return 0;
+ }
+ break;
+
+ case WM_USER + 8:
+ if (LOWORD(lParam) == WM_MOUSEMOVE)
+ {
+ static DWORD last_t;
+ if (GetTickCount() - last_t > 250)
+ {
+ last_t = GetTickCount();
+ HWND hwnd2 = GetWinampHWND();
+ if (IsWindow(hwnd2))
+ {
+ wchar_t buf[128] = {0};
+ GetWindowTextW(hwnd2, buf, 128);
+ systray_mod(hwnd, 0, buf);
+ }
+ else
+ {
+ systray_mod(hwnd, 0, GetStringW(IDS_WINAMP_AGENT));
+ }
+ }
+ }
+ else if (LOWORD(lParam) == WM_LBUTTONUP ||
+ LOWORD(lParam) == WM_LBUTTONDBLCLK)
+ {
+ if(!(GetAsyncKeyState(VK_SHIFT)&0x8000))
+ {
+ HWND hwnd2 = GetWinampHWND();
+ if (IsWindow(hwnd2))
+ {
+ if (LOWORD(lParam) == WM_LBUTTONDBLCLK)
+ {
+ ShowWindow(hwnd2, SW_RESTORE);
+ }
+ SetForegroundWindow(hwnd2);
+ SendMessage(hwnd2, WM_USER + 1, 0, WM_LBUTTONUP);
+ }
+ else
+ {
+ ShellExecuteW(NULL, L"open", winamp_exe_file, L"", L".", SW_SHOW);
+ }
+ }
+ else
+ {
+ SendMessage(hwnd, WM_CLOSE, 0, 0);
+ }
+ }
+ else if (LOWORD(lParam) == WM_RBUTTONUP)
+ {
+ HWND hwnd2 = GetWinampHWND();
+ if (IsWindow(hwnd2) && !(GetAsyncKeyState(VK_CONTROL)&0x8000))
+ {
+ SetForegroundWindow(hwnd2);
+ SendMessage(hwnd2, WM_USER + 1, 0, WM_RBUTTONUP);
+ }
+ else
+ {
+ HMENU hMenu = CreatePopupMenu();
+ MENUITEMINFOW i = {0};
+ // for bookmarks menu
+ int num_bookmarks = 0;
+ // for audio cd entries
+ wchar_t g_audiocdletter[4] = {0};
+ int g_audiocdletters = 0;
+ int drivemask = 0;
+ DWORD drives = GetLogicalDrives();
+
+ char fn[1024] = {0};
+ char ft[1024] = {0};
+ POINT p = {0};
+ GetCursorPos(&p);
+ i.cbSize = sizeof(i);
+ i.fMask = MIIM_TYPE | MIIM_DATA | MIIM_ID;
+ i.fType = MFT_STRING;
+ i.wID = 1;
+ i.dwTypeData = GetStringW(IDS_OPEN_WINAMP);
+ i.cch = lstrlenW((wchar_t*)i.dwTypeData);
+ InsertMenuItemW(hMenu, 0, TRUE, &i);
+ i.wID = 0;
+ i.fType = MFT_SEPARATOR;
+ InsertMenuItemW(hMenu, 1, TRUE, &i);
+
+ i.fMask = MIIM_TYPE | MIIM_DATA | MIIM_ID;
+ i.fType = MFT_STRING;
+ i.wID = 2;
+ i.dwTypeData = GetStringW(IDS_DISABLE_WINAMP_AGENT);
+ i.cch = lstrlenW((wchar_t*)i.dwTypeData);
+ InsertMenuItemW(hMenu, 2, TRUE, &i);
+ i.wID = 3;
+ i.dwTypeData = GetStringW(IDS_CLOSE_WINAMP_AGENT);
+ i.cch = lstrlenW((wchar_t*)i.dwTypeData);
+ InsertMenuItemW(hMenu, 3, TRUE, &i);
+
+ SetMenuDefaultItem(hMenu,!(GetAsyncKeyState(VK_SHIFT)&0x8000)?0:3,1);
+
+ i.wID = 0;
+ i.fType = MFT_SEPARATOR;
+ InsertMenuItemW(hMenu, 4, TRUE, &i);
+
+ i.wID = 10;
+ for (drivemask = 0; drivemask < 32; drivemask++)
+ {
+ if (drives&(1 << drivemask))
+ {
+ wchar_t str[256] = {0};
+ StringCchPrintfW(str, 256, L"%c:\\", 'A' + drivemask);
+ if (GetDriveTypeW(str) == DRIVE_CDROM)
+ {
+ int old_error_mode = SetErrorMode(SEM_FAILCRITICALERRORS);
+ DWORD system_flags = 0, max_file_len = 0;
+ wchar_t drives[4] = {L" :\\"}, c = L'A' + drivemask, vol_buf[40] = {0}, empty[64] = {0};
+ drives[0] = g_audiocdletter[g_audiocdletters] = c;
+
+ GetVolumeInformationW(drives,vol_buf,sizeof(vol_buf),0,&max_file_len,&system_flags,0,0);
+ SetErrorMode(old_error_mode);
+
+ lstrcpynW(empty,GetStringW(IDS_EMPTY),64);
+ StringCchPrintfW(str, 256, GetStringW(IDS_AUDIO_CD),c,(vol_buf[0]?vol_buf:empty));
+ i.fType = MFT_STRING;
+ i.dwTypeData = str;
+ i.cch = (UINT)wcslen(str);
+ InsertMenuItemW(hMenu, 5 + g_audiocdletters, TRUE, &i);
+ i.wID++;
+ g_audiocdletters++;
+ if (g_audiocdletters == 4) break;
+ }
+ }
+ }
+
+ if(g_audiocdletters)
+ {
+ i.wID = 0;
+ i.fType = MFT_SEPARATOR;
+ InsertMenuItemW(hMenu, 5 + g_audiocdletters, TRUE, &i);
+ }
+
+ i.fType = MFT_STRING;
+ i.dwTypeData = GetStringW(IDS_BOOKMARKS);
+ i.cch = lstrlenW((wchar_t*)i.dwTypeData);
+ HMENU sm = i.hSubMenu = CreatePopupMenu();
+ i.fMask |= MIIM_SUBMENU;
+ i.wID = 0;
+ InsertMenuItemW(hMenu, 6 + g_audiocdletters, TRUE, &i);
+
+ // have to keep this ansi since winamp.bm doesn't support unicode
+ HANDLE hFile = CreateFileW(bm_file, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, NULL);
+ if (hFile != INVALID_HANDLE_VALUE)
+ {
+ MENUITEMINFOW ib = {0};
+ ib.cbSize = sizeof(ib);
+ ib.fMask = MIIM_TYPE | MIIM_DATA | MIIM_ID;
+ ib.fType = MFT_STRING;
+ i.wID = ib.wID = 20;
+ while (1)
+ {
+ if (!ReadStr(hFile, fn, MAX_PATH)) break;
+ if (!ReadStr(hFile, ft, 2048)) break;
+ ib.dwTypeData = AutoWideDup(ft, CP_UTF8);
+ ib.cch = lstrlenW(ib.dwTypeData);
+ InsertMenuItemW(sm, num_bookmarks, TRUE, &ib);
+ ib.wID++;
+ i.wID++;
+ num_bookmarks++;
+ }
+ }
+
+ if(i.wID == 20 || !i.wID)
+ {
+ i.fMask = MIIM_TYPE | MIIM_DATA | MIIM_ID;
+ i.fType = MFT_STRING;
+ i.dwTypeData = GetStringW(IDS_NO_BOOKMARKS_FOUND);
+ i.cch = lstrlenW((wchar_t*)i.dwTypeData);
+ InsertMenuItemW(sm, num_bookmarks, TRUE, &i);
+ EnableMenuItem(sm, i.wID, MF_BYCOMMAND | MF_GRAYED);
+ }
+
+ SetForegroundWindow(hwnd);
+ int x = TrackPopupMenu(hMenu, TPM_LEFTALIGN | TPM_TOPALIGN | TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTBUTTON | TPM_NONOTIFY, p.x, p.y, 0, hwnd, NULL);
+ if (x == 1)
+ {
+ HWND hwnd2 = GetWinampHWND();
+ if (IsWindow(hwnd2))
+ {
+ SetForegroundWindow(hwnd2);
+ SendMessage(hwnd2, WM_USER + 1, 0, WM_LBUTTONUP);
+ }
+ else
+ {
+ ShellExecuteW(NULL, L"open", winamp_exe_file, L"", L".", SW_SHOW);
+ }
+ }
+ else if (x == 2 || x == 3)
+ {
+ if (x == 2) // disable
+ {
+ HKEY key;
+ if (RegOpenKeyW(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Run", &key) == ERROR_SUCCESS)
+ {
+ RegDeleteValueW(key, L"WinampAgent");
+ RegCloseKey(key);
+ }
+ }
+ SendMessage(hwnd, WM_CLOSE, 0, 0);
+ }
+ else if(x >= 10 && x < 10 + g_audiocdletters)
+ {
+ wchar_t ftW[1024] = {0};
+ StringCchPrintfW(ftW, 1024, L"\"cda://%c\"", g_audiocdletter[x - 10]);
+ ShellExecuteW(NULL, L"open", winamp_exe_file, ftW, L".", SW_SHOW);
+ }
+ else if (x >= 20 && x < 20 + num_bookmarks && hFile != INVALID_HANDLE_VALUE)
+ {
+ int r = 0;
+ x -= 20;
+ SetFilePointer(hFile, 0, NULL, FILE_BEGIN);
+ for (; r <= x; r ++)
+ {
+ if (!ReadStr(hFile, fn, MAX_PATH)) break;
+ if (!ReadStr(hFile, ft, 2048)) break;
+ }
+ if (r == (x + 1))
+ {
+ wchar_t ftW[1024] = {0};
+ StringCchPrintfW(ftW, 1024, L"\"%s\"", AutoWide(fn, CP_UTF8));
+ ShellExecuteW(NULL, L"open", winamp_exe_file, ftW, L".", SW_SHOW);
+ }
+ }
+ DestroyMenu(hMenu);
+ if (hFile != INVALID_HANDLE_VALUE) CloseHandle(hFile);
+ }
+ }
+ return 0;
+
+ case WM_USER + 16:
+ // do this on load/unload requests just incase something went wrong
+ UnloadWinampaLng();
+ if(!wParam) LoadWinampaLng();
+ return 0;
+
+ case WM_DESTROY:
+ if (systray_isintray()) systray_del(hwnd);
+ PostQuitMessage(0);
+ return 0;
+ }
+ return DefWindowProc(hwnd, uMsg, wParam, lParam);
+}
+
+static wchar_t szClassName[] = L"WinampAgentMain";
+static wchar_t szErrorTitle[] = L"Winamp Agent Error";
+
+void queryPath(wchar_t *out, wchar_t *in, int out_len)
+{
+ wchar_t buf[MAX_PATH] = {0};
+ HKEY key = 0;
+ if (RegOpenKeyW(HKEY_CLASSES_ROOT, in, &key) == ERROR_SUCCESS)
+ {
+ DWORD s = sizeof(buf);
+ if (RegQueryValueExW(key, NULL, 0, NULL, (LPBYTE)buf, &s) == ERROR_SUCCESS)
+ {
+ if (buf[0] == L'\"')
+ {
+ wchar_t *p = buf + 1;
+ while (p && *p != L'\"' && *p) p=CharNextW(p);
+ if (p && *p) *p = 0;
+ while (p > buf && *p != L'\\') p = CharPrevW(buf,p);
+ if (p && *p) *p = 0;
+ lstrcpynW(out, buf + 1, out_len);
+ }
+ else
+ {
+ wchar_t *p = buf;
+ while (p && *p) p=CharNextW(p);
+ while (p > buf && *p != L'\\') p = CharPrevW(buf,p);
+ if (p && *p) *p = 0;
+ lstrcpynW(out, buf, out_len);
+ }
+ }
+ RegCloseKey(key);
+ }
+}
+
+void BuildDirectories()
+{
+ // get ini_file from reg
+ wchar_t winamp2Folder[MAX_PATH] = {0};
+
+ // attempt to get the winamp folder from the play then the enqueue and then just revert to current folder (wa2/5)
+ queryPath(winamp2Folder, L"Winamp.File\\shell\\play\\command", MAX_PATH);
+ if(!winamp2Folder[0]) queryPath(winamp2Folder, L"Winamp.File\\shell\\enqueue\\command", MAX_PATH);
+ if(!winamp2Folder[0])
+ {
+ wchar_t buf[MAX_PATH] = {0}, *p = buf;
+ GetModuleFileNameW(GetModuleHandleW(NULL), buf, sizeof(buf));
+ while (p && *p) p=CharNextW(p);
+ while (p > buf && *p != L'\\') p=CharPrevW(buf,p);
+ if (p && *p) *p = 0;
+ lstrcpynW(winamp2Folder, buf, sizeof(winamp2Folder));
+ }
+
+ if (winamp2Folder[0]) // wa2/5
+ {
+ wchar_t pathsIni[MAX_PATH] = {0};
+ wchar_t iniFileName[MAX_PATH] = {0};
+ wchar_t profileString[MAX_PATH] = {0};
+ StringCchPrintfW(pathsIni, MAX_PATH, L"%s\\paths.ini", winamp2Folder);
+
+ GetPrivateProfileStringW(L"Winamp", L"inidir", L"", profileString, MAX_PATH, pathsIni);
+ if (profileString[0])
+ ResolveEnvironmentVariables2(profileString, wa2ConfigDir, MAX_PATH);
+ else
+ lstrcpynW(wa2ConfigDir, winamp2Folder, MAX_PATH);
+
+ GetPrivateProfileStringW(L"Winamp", L"class", L"", profileString, MAX_PATH, pathsIni);
+ if (profileString[0])
+ ResolveEnvironmentVariables2(profileString, winampClassName, MAX_PATH);
+
+ GetPrivateProfileStringW(L"Winamp", L"inifile", L"", profileString, MAX_PATH, pathsIni);
+ if (profileString[0])
+ ResolveEnvironmentVariables2(profileString, iniFileName, MAX_PATH);
+ else
+ lstrcpynW(iniFileName, L"winamp.ini", MAX_PATH);
+
+ StringCchPrintfW(ini_file, MAX_PATH, L"%s\\%s", wa2ConfigDir, iniFileName);
+
+ // winamp.exe should extract this out for us when a new wlz is loaded so we
+ // don't have to bloat up winampa - just have to deal with wlz changes instead
+ StringCchPrintfW(winampaLngPath, MAX_PATH, L"%s\\winampa.lng", wa2ConfigDir);
+ StringCchPrintfW(icon_tmp, MAX_PATH, L"%s\\winamp.ico", wa2ConfigDir);
+
+ StringCchPrintfW(winamp_exe_file, MAX_PATH, L"%s\\winamp.exe", winamp2Folder);
+ StringCchPrintfW(bm_file, MAX_PATH, L"%s\\winamp.bm8", wa2ConfigDir);
+ // just make sure if a winamp.bm8 doesn't exist then
+ // go make one from winamp.bm - implemented for 5.58+
+ if(!PathFileExistsW(bm_file))
+ {
+ wchar_t tmp[MAX_PATH] = {0};
+ StringCchPrintfW(tmp, MAX_PATH, L"%s\\winamp.bm", wa2ConfigDir);
+ CopyFileW(tmp,bm_file,FALSE);
+ }
+ }
+
+ if (!winampClassName[0])
+ lstrcpynW(winampClassName, L"Winamp v1.x", MAX_PATH);
+}
+
+int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
+{
+ MSG msg = {0};
+ static WNDCLASSW wc;
+
+ if (FindWindowW(szClassName, NULL))
+ {
+ ExitProcess(0);
+ }
+
+ WM_TASKBARCREATED = RegisterWindowMessageW(L"TaskbarCreated");
+
+ // add in a UIPI filter so we can get required notifications from winamp.exe such
+ // as when we're started from an elevation request e.g. via prefs dialog options.
+ if (LoadWMFilter()){
+ changeWMFilter(WM_USER+1, 1/*MSGFLT_ADD*/);
+ changeWMFilter(WM_USER+16, 1/*MSGFLT_ADD*/);
+ }
+
+ wc.lpfnWndProc = WndProc;
+ g_hInstance = wc.hInstance = GetModuleHandleW(NULL);
+ wc.lpszClassName = szClassName;
+
+ BuildDirectories();
+
+ // attempt to load winampa.lng if present (if extracted from the current wlz if there is one)
+ nativeLng = hInstance;
+ LoadWinampaLng();
+
+ if (!RegisterClassW(&wc))
+ {
+ MessageBoxW(NULL, L"Cannot register window class!", szErrorTitle, MB_OK | MB_ICONSTOP);
+ return 0;
+ }
+
+ if (!(CreateWindowExW(WS_EX_TOPMOST | WS_EX_TOOLWINDOW, szClassName, L"", 0,
+ CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
+ NULL, NULL, g_hInstance, NULL)))
+ {
+ MessageBoxW(NULL, L"Cannot create window!", szErrorTitle, MB_OK | MB_ICONSTOP);
+ return 0;
+ }
+
+ while (GetMessageW(&msg, NULL, 0, 0))
+ {
+ DispatchMessageW(&msg);
+ } // while(GetMessage...
+
+ UnloadWinampaLng();
+ ExitProcess(0);
+ return 0;
+}
+
+#ifdef DO_LOG
+void do_log_print(char *p)
+{
+ HANDLE h = CreateFile("C:\\winampa.log", GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_ALWAYS, 0, NULL);
+ if (h != INVALID_HANDLE_VALUE)
+ {
+ DWORD l = 0;
+ SetFilePointer(h, 0, NULL, FILE_END);
+ WriteFile(h, p, lstrlen(p), &l, NULL);
+ CloseHandle(h);
+ }
+}
+#endif \ No newline at end of file