mirror of
https://github.com/WinampDesktop/winamp.git
synced 2024-09-24 15:54:12 +00:00
1259 lines
44 KiB
C++
1259 lines
44 KiB
C++
|
/*
|
||
|
LICENSE
|
||
|
-------
|
||
|
Copyright 2005-2013 Nullsoft, Inc.
|
||
|
All rights reserved.
|
||
|
|
||
|
Redistribution and use in source and binary forms, with or without modification,
|
||
|
are permitted provided that the following conditions are met:
|
||
|
|
||
|
* Redistributions of source code must retain the above copyright notice,
|
||
|
this list of conditions and the following disclaimer.
|
||
|
|
||
|
* Redistributions in binary form must reproduce the above copyright notice,
|
||
|
this list of conditions and the following disclaimer in the documentation
|
||
|
and/or other materials provided with the distribution.
|
||
|
|
||
|
* Neither the name of Nullsoft nor the names of its contributors may be used to
|
||
|
endorse or promote products derived from this software without specific prior written permission.
|
||
|
|
||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
|
||
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
||
|
FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
|
||
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
|
||
|
IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
||
|
OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*/
|
||
|
|
||
|
#include "api__vis_milk2.h"
|
||
|
#include "utility.h"
|
||
|
#include <math.h>
|
||
|
#include <locale.h>
|
||
|
#include <windows.h>
|
||
|
#ifdef _DEBUG
|
||
|
#define D3D_DEBUG_INFO // declare this before including d3d9.h
|
||
|
#endif
|
||
|
#include <d3d9.h>
|
||
|
#include "../Winamp/wa_ipc.h"
|
||
|
#include "resource.h"
|
||
|
#include <shellapi.h>
|
||
|
|
||
|
intptr_t myOpenURL(HWND hwnd, wchar_t *loc)
|
||
|
{
|
||
|
if (loc)
|
||
|
{
|
||
|
bool override=false;
|
||
|
WASABI_API_SYSCB->syscb_issueCallback(SysCallback::BROWSER, BrowserCallback::ONOPENURL, reinterpret_cast<intptr_t>(loc), reinterpret_cast<intptr_t>(&override));
|
||
|
if (!override)
|
||
|
return (intptr_t)ShellExecuteW(hwnd, L"open", loc, NULL, NULL, SW_SHOWNORMAL);
|
||
|
else
|
||
|
return 33;
|
||
|
}
|
||
|
return 33;
|
||
|
}
|
||
|
|
||
|
float PowCosineInterp(float x, float pow)
|
||
|
{
|
||
|
// input (x) & output should be in range 0..1.
|
||
|
// pow > 0: tends to push things toward 0 and 1
|
||
|
// pow < 0: tends to push things toward 0.5.
|
||
|
|
||
|
if (x<0)
|
||
|
return 0;
|
||
|
if (x>1)
|
||
|
return 1;
|
||
|
|
||
|
int bneg = (pow < 0) ? 1 : 0;
|
||
|
if (bneg)
|
||
|
pow = -pow;
|
||
|
|
||
|
if (pow>1000) pow=1000;
|
||
|
|
||
|
int its = (int)pow;
|
||
|
for (int i=0; i<its; i++)
|
||
|
{
|
||
|
if (bneg)
|
||
|
x = InvCosineInterp(x);
|
||
|
else
|
||
|
x = CosineInterp(x);
|
||
|
}
|
||
|
float x2 = (bneg) ? InvCosineInterp(x) : CosineInterp(x);
|
||
|
float dx = pow - its;
|
||
|
return ((1-dx)*x + (dx)*x2);
|
||
|
}
|
||
|
|
||
|
float AdjustRateToFPS(float per_frame_decay_rate_at_fps1, float fps1, float actual_fps)
|
||
|
{
|
||
|
// returns the equivalent per-frame decay rate at actual_fps
|
||
|
|
||
|
// basically, do all your testing at fps1 and get a good decay rate;
|
||
|
// then, in the real application, adjust that rate by the actual fps each time you use it.
|
||
|
|
||
|
float per_second_decay_rate_at_fps1 = powf(per_frame_decay_rate_at_fps1, fps1);
|
||
|
float per_frame_decay_rate_at_fps2 = powf(per_second_decay_rate_at_fps1, 1.0f/actual_fps);
|
||
|
|
||
|
return per_frame_decay_rate_at_fps2;
|
||
|
}
|
||
|
|
||
|
float GetPrivateProfileFloatW(wchar_t *szSectionName, wchar_t *szKeyName, float fDefault, wchar_t *szIniFile)
|
||
|
{
|
||
|
wchar_t string[64];
|
||
|
wchar_t szDefault[64];
|
||
|
float ret = fDefault;
|
||
|
|
||
|
_swprintf_l(szDefault, L"%f", g_use_C_locale, fDefault);
|
||
|
|
||
|
if (GetPrivateProfileStringW(szSectionName, szKeyName, szDefault, string, 64, szIniFile) > 0)
|
||
|
{
|
||
|
_swscanf_l(string, L"%f", g_use_C_locale, &ret);
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
bool WritePrivateProfileFloatW(float f, wchar_t *szKeyName, wchar_t *szIniFile, wchar_t *szSectionName)
|
||
|
{
|
||
|
wchar_t szValue[32];
|
||
|
_swprintf_l(szValue, L"%f", g_use_C_locale, f);
|
||
|
return (WritePrivateProfileStringW(szSectionName, szKeyName, szValue, szIniFile) != 0);
|
||
|
}
|
||
|
|
||
|
bool WritePrivateProfileIntW(int d, wchar_t *szKeyName, wchar_t *szIniFile, wchar_t *szSectionName)
|
||
|
{
|
||
|
wchar_t szValue[32];
|
||
|
swprintf(szValue, L"%d", d);
|
||
|
return (WritePrivateProfileStringW(szSectionName, szKeyName, szValue, szIniFile) != 0);
|
||
|
}
|
||
|
|
||
|
void SetScrollLock(int bNewState, bool bPreventHandling)
|
||
|
{
|
||
|
if(bPreventHandling) return;
|
||
|
|
||
|
if (bNewState != (GetKeyState(VK_SCROLL) & 1))
|
||
|
{
|
||
|
// Simulate a key press
|
||
|
keybd_event( VK_SCROLL,
|
||
|
0x45,
|
||
|
KEYEVENTF_EXTENDEDKEY | 0,
|
||
|
0 );
|
||
|
|
||
|
// Simulate a key release
|
||
|
keybd_event( VK_SCROLL,
|
||
|
0x45,
|
||
|
KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP,
|
||
|
0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RemoveExtension(wchar_t *str)
|
||
|
{
|
||
|
wchar_t *p = wcsrchr(str, L'.');
|
||
|
if (p) *p = 0;
|
||
|
}
|
||
|
|
||
|
static void ShiftDown(wchar_t *str)
|
||
|
{
|
||
|
while (*str)
|
||
|
{
|
||
|
str[0] = str[1];
|
||
|
str++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void RemoveSingleAmpersands(wchar_t *str)
|
||
|
{
|
||
|
while (*str)
|
||
|
{
|
||
|
if (str[0] == L'&')
|
||
|
{
|
||
|
if (str[1] == L'&') // two in a row: replace with single ampersand, move on
|
||
|
str++;
|
||
|
|
||
|
ShiftDown(str);
|
||
|
}
|
||
|
else
|
||
|
str = CharNextW(str);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void TextToGuid(char *str, GUID *pGUID)
|
||
|
{
|
||
|
if (!str) return;
|
||
|
if (!pGUID) return;
|
||
|
|
||
|
DWORD d[11];
|
||
|
|
||
|
sscanf(str, "%X %X %X %X %X %X %X %X %X %X %X",
|
||
|
&d[0], &d[1], &d[2], &d[3], &d[4], &d[5], &d[6], &d[7], &d[8], &d[9], &d[10]);
|
||
|
|
||
|
pGUID->Data1 = (DWORD)d[0];
|
||
|
pGUID->Data2 = (WORD)d[1];
|
||
|
pGUID->Data3 = (WORD)d[2];
|
||
|
pGUID->Data4[0] = (BYTE)d[3];
|
||
|
pGUID->Data4[1] = (BYTE)d[4];
|
||
|
pGUID->Data4[2] = (BYTE)d[5];
|
||
|
pGUID->Data4[3] = (BYTE)d[6];
|
||
|
pGUID->Data4[4] = (BYTE)d[7];
|
||
|
pGUID->Data4[5] = (BYTE)d[8];
|
||
|
pGUID->Data4[6] = (BYTE)d[9];
|
||
|
pGUID->Data4[7] = (BYTE)d[10];
|
||
|
}
|
||
|
|
||
|
void GuidToText(GUID *pGUID, char *str, int nStrLen)
|
||
|
{
|
||
|
// note: nStrLen should be set to sizeof(str).
|
||
|
if (!str) return;
|
||
|
if (!nStrLen) return;
|
||
|
str[0] = 0;
|
||
|
if (!pGUID) return;
|
||
|
|
||
|
DWORD d[11];
|
||
|
d[0] = (DWORD)pGUID->Data1;
|
||
|
d[1] = (DWORD)pGUID->Data2;
|
||
|
d[2] = (DWORD)pGUID->Data3;
|
||
|
d[3] = (DWORD)pGUID->Data4[0];
|
||
|
d[4] = (DWORD)pGUID->Data4[1];
|
||
|
d[5] = (DWORD)pGUID->Data4[2];
|
||
|
d[6] = (DWORD)pGUID->Data4[3];
|
||
|
d[7] = (DWORD)pGUID->Data4[4];
|
||
|
d[8] = (DWORD)pGUID->Data4[5];
|
||
|
d[9] = (DWORD)pGUID->Data4[6];
|
||
|
d[10] = (DWORD)pGUID->Data4[7];
|
||
|
|
||
|
sprintf(str, "%08X %04X %04X %02X %02X %02X %02X %02X %02X %02X %02X",
|
||
|
d[0], d[1], d[2], d[3], d[4], d[5], d[6], d[7], d[8], d[9], d[10]);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
int GetPentiumTimeRaw(unsigned __int64 *cpu_timestamp)
|
||
|
{
|
||
|
// returns 0 on failure, 1 on success
|
||
|
// warning: watch out for wraparound!
|
||
|
|
||
|
// note: it's probably better to use QueryPerformanceFrequency
|
||
|
// and QueryPerformanceCounter()!
|
||
|
|
||
|
// get high-precision time:
|
||
|
__try
|
||
|
{
|
||
|
unsigned __int64 *dest = (unsigned __int64 *)cpu_timestamp;
|
||
|
__asm
|
||
|
{
|
||
|
_emit 0xf // these two bytes form the 'rdtsc' asm instruction,
|
||
|
_emit 0x31 // available on Pentium I and later.
|
||
|
mov esi, dest
|
||
|
mov [esi ], eax // lower 32 bits of tsc
|
||
|
mov [esi+4], edx // upper 32 bits of tsc
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
double GetPentiumTimeAsDouble(unsigned __int64 frequency)
|
||
|
{
|
||
|
// returns < 0 on failure; otherwise, returns current cpu time, in seconds.
|
||
|
// warning: watch out for wraparound!
|
||
|
|
||
|
// note: it's probably better to use QueryPerformanceFrequency
|
||
|
// and QueryPerformanceCounter()!
|
||
|
|
||
|
if (frequency==0)
|
||
|
return -1.0;
|
||
|
|
||
|
// get high-precision time:
|
||
|
__try
|
||
|
{
|
||
|
unsigned __int64 high_perf_time;
|
||
|
unsigned __int64 *dest = &high_perf_time;
|
||
|
__asm
|
||
|
{
|
||
|
_emit 0xf // these two bytes form the 'rdtsc' asm instruction,
|
||
|
_emit 0x31 // available on Pentium I and later.
|
||
|
mov esi, dest
|
||
|
mov [esi ], eax // lower 32 bits of tsc
|
||
|
mov [esi+4], edx // upper 32 bits of tsc
|
||
|
}
|
||
|
__int64 time_s = (__int64)(high_perf_time / frequency); // unsigned->sign conversion should be safe here
|
||
|
__int64 time_fract = (__int64)(high_perf_time % frequency); // unsigned->sign conversion should be safe here
|
||
|
// note: here, we wrap the timer more frequently (once per week)
|
||
|
// than it otherwise would (VERY RARELY - once every 585 years on
|
||
|
// a 1 GHz), to alleviate floating-point precision errors that start
|
||
|
// to occur when you get to very high counter values.
|
||
|
double ret = (time_s % (60*60*24*7)) + (double)time_fract/(double)((__int64)frequency);
|
||
|
return ret;
|
||
|
}
|
||
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
return -1.0;
|
||
|
}
|
||
|
|
||
|
return -1.0;
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
#ifdef _DEBUG
|
||
|
void OutputDebugMessage(char *szStartText, HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)
|
||
|
{
|
||
|
// note: this function does NOT log WM_MOUSEMOVE, WM_NCHITTEST, or WM_SETCURSOR
|
||
|
// messages, since they are so frequent.
|
||
|
// note: these identifiers were pulled from winuser.h
|
||
|
|
||
|
//if (msg == WM_MOUSEMOVE || msg == WM_NCHITTEST || msg == WM_SETCURSOR)
|
||
|
// return;
|
||
|
|
||
|
#ifdef _DEBUG
|
||
|
char buf[64];
|
||
|
int matched = 1;
|
||
|
|
||
|
sprintf(buf, "WM_");
|
||
|
|
||
|
switch(msg)
|
||
|
{
|
||
|
case 0x0001: lstrcat(buf, "CREATE"); break;
|
||
|
case 0x0002: lstrcat(buf, "DESTROY"); break;
|
||
|
case 0x0003: lstrcat(buf, "MOVE"); break;
|
||
|
case 0x0005: lstrcat(buf, "SIZE"); break;
|
||
|
case 0x0006: lstrcat(buf, "ACTIVATE"); break;
|
||
|
case 0x0007: lstrcat(buf, "SETFOCUS"); break;
|
||
|
case 0x0008: lstrcat(buf, "KILLFOCUS"); break;
|
||
|
case 0x000A: lstrcat(buf, "ENABLE"); break;
|
||
|
case 0x000B: lstrcat(buf, "SETREDRAW"); break;
|
||
|
case 0x000C: lstrcat(buf, "SETTEXT"); break;
|
||
|
case 0x000D: lstrcat(buf, "GETTEXT"); break;
|
||
|
case 0x000E: lstrcat(buf, "GETTEXTLENGTH"); break;
|
||
|
case 0x000F: lstrcat(buf, "PAINT"); break;
|
||
|
case 0x0010: lstrcat(buf, "CLOSE"); break;
|
||
|
case 0x0011: lstrcat(buf, "QUERYENDSESSION"); break;
|
||
|
case 0x0012: lstrcat(buf, "QUIT"); break;
|
||
|
case 0x0013: lstrcat(buf, "QUERYOPEN"); break;
|
||
|
case 0x0014: lstrcat(buf, "ERASEBKGND"); break;
|
||
|
case 0x0015: lstrcat(buf, "SYSCOLORCHANGE"); break;
|
||
|
case 0x0016: lstrcat(buf, "ENDSESSION"); break;
|
||
|
case 0x0018: lstrcat(buf, "SHOWWINDOW"); break;
|
||
|
case 0x001A: lstrcat(buf, "WININICHANGE"); break;
|
||
|
case 0x001B: lstrcat(buf, "DEVMODECHANGE"); break;
|
||
|
case 0x001C: lstrcat(buf, "ACTIVATEAPP"); break;
|
||
|
case 0x001D: lstrcat(buf, "FONTCHANGE"); break;
|
||
|
case 0x001E: lstrcat(buf, "TIMECHANGE"); break;
|
||
|
case 0x001F: lstrcat(buf, "CANCELMODE"); break;
|
||
|
case 0x0020: lstrcat(buf, "SETCURSOR"); break;
|
||
|
case 0x0021: lstrcat(buf, "MOUSEACTIVATE"); break;
|
||
|
case 0x0022: lstrcat(buf, "CHILDACTIVATE"); break;
|
||
|
case 0x0023: lstrcat(buf, "QUEUESYNC"); break;
|
||
|
case 0x0024: lstrcat(buf, "GETMINMAXINFO"); break;
|
||
|
case 0x0026: lstrcat(buf, "PAINTICON"); break;
|
||
|
case 0x0027: lstrcat(buf, "ICONERASEBKGND"); break;
|
||
|
case 0x0028: lstrcat(buf, "NEXTDLGCTL"); break;
|
||
|
case 0x002A: lstrcat(buf, "SPOOLERSTATUS"); break;
|
||
|
case 0x002B: lstrcat(buf, "DRAWITEM"); break;
|
||
|
case 0x002C: lstrcat(buf, "MEASUREITEM"); break;
|
||
|
case 0x002D: lstrcat(buf, "DELETEITEM"); break;
|
||
|
case 0x002E: lstrcat(buf, "VKEYTOITEM"); break;
|
||
|
case 0x002F: lstrcat(buf, "CHARTOITEM"); break;
|
||
|
case 0x0030: lstrcat(buf, "SETFONT"); break;
|
||
|
case 0x0031: lstrcat(buf, "GETFONT"); break;
|
||
|
case 0x0032: lstrcat(buf, "SETHOTKEY"); break;
|
||
|
case 0x0033: lstrcat(buf, "GETHOTKEY"); break;
|
||
|
case 0x0037: lstrcat(buf, "QUERYDRAGICON"); break;
|
||
|
case 0x0039: lstrcat(buf, "COMPAREITEM"); break;
|
||
|
case 0x0041: lstrcat(buf, "COMPACTING"); break;
|
||
|
case 0x0044: lstrcat(buf, "COMMNOTIFY"); break;
|
||
|
case 0x0046: lstrcat(buf, "WINDOWPOSCHANGING"); break;
|
||
|
case 0x0047: lstrcat(buf, "WINDOWPOSCHANGED"); break;
|
||
|
case 0x0048: lstrcat(buf, "POWER"); break;
|
||
|
case 0x004A: lstrcat(buf, "COPYDATA"); break;
|
||
|
case 0x004B: lstrcat(buf, "CANCELJOURNAL"); break;
|
||
|
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x004E: lstrcat(buf, "NOTIFY"); break;
|
||
|
case 0x0050: lstrcat(buf, "INPUTLANGCHANGEREQUEST"); break;
|
||
|
case 0x0051: lstrcat(buf, "INPUTLANGCHANGE"); break;
|
||
|
case 0x0052: lstrcat(buf, "TCARD"); break;
|
||
|
case 0x0053: lstrcat(buf, "HELP"); break;
|
||
|
case 0x0054: lstrcat(buf, "USERCHANGED"); break;
|
||
|
case 0x0055: lstrcat(buf, "NOTIFYFORMAT"); break;
|
||
|
case 0x007B: lstrcat(buf, "CONTEXTMENU"); break;
|
||
|
case 0x007C: lstrcat(buf, "STYLECHANGING"); break;
|
||
|
case 0x007D: lstrcat(buf, "STYLECHANGED"); break;
|
||
|
case 0x007E: lstrcat(buf, "DISPLAYCHANGE"); break;
|
||
|
case 0x007F: lstrcat(buf, "GETICON"); break;
|
||
|
case 0x0080: lstrcat(buf, "SETICON"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0081: lstrcat(buf, "NCCREATE"); break;
|
||
|
case 0x0082: lstrcat(buf, "NCDESTROY"); break;
|
||
|
case 0x0083: lstrcat(buf, "NCCALCSIZE"); break;
|
||
|
case 0x0084: lstrcat(buf, "NCHITTEST"); break;
|
||
|
case 0x0085: lstrcat(buf, "NCPAINT"); break;
|
||
|
case 0x0086: lstrcat(buf, "NCACTIVATE"); break;
|
||
|
case 0x0087: lstrcat(buf, "GETDLGCODE"); break;
|
||
|
case 0x0088: lstrcat(buf, "SYNCPAINT"); break;
|
||
|
case 0x00A0: lstrcat(buf, "NCMOUSEMOVE"); break;
|
||
|
case 0x00A1: lstrcat(buf, "NCLBUTTONDOWN"); break;
|
||
|
case 0x00A2: lstrcat(buf, "NCLBUTTONUP"); break;
|
||
|
case 0x00A3: lstrcat(buf, "NCLBUTTONDBLCLK"); break;
|
||
|
case 0x00A4: lstrcat(buf, "NCRBUTTONDOWN"); break;
|
||
|
case 0x00A5: lstrcat(buf, "NCRBUTTONUP"); break;
|
||
|
case 0x00A6: lstrcat(buf, "NCRBUTTONDBLCLK"); break;
|
||
|
case 0x00A7: lstrcat(buf, "NCMBUTTONDOWN"); break;
|
||
|
case 0x00A8: lstrcat(buf, "NCMBUTTONUP"); break;
|
||
|
case 0x00A9: lstrcat(buf, "NCMBUTTONDBLCLK"); break;
|
||
|
case 0x0100: lstrcat(buf, "KEYDOWN"); break;
|
||
|
case 0x0101: lstrcat(buf, "KEYUP"); break;
|
||
|
case 0x0102: lstrcat(buf, "CHAR"); break;
|
||
|
case 0x0103: lstrcat(buf, "DEADCHAR"); break;
|
||
|
case 0x0104: lstrcat(buf, "SYSKEYDOWN"); break;
|
||
|
case 0x0105: lstrcat(buf, "SYSKEYUP"); break;
|
||
|
case 0x0106: lstrcat(buf, "SYSCHAR"); break;
|
||
|
case 0x0107: lstrcat(buf, "SYSDEADCHAR"); break;
|
||
|
case 0x0108: lstrcat(buf, "KEYLAST"); break;
|
||
|
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x010D: lstrcat(buf, "IME_STARTCOMPOSITION"); break;
|
||
|
case 0x010E: lstrcat(buf, "IME_ENDCOMPOSITION"); break;
|
||
|
case 0x010F: lstrcat(buf, "IME_COMPOSITION"); break;
|
||
|
//case 0x010F: lstrcat(buf, "IME_KEYLAST"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0110: lstrcat(buf, "INITDIALOG"); break;
|
||
|
case 0x0111: lstrcat(buf, "COMMAND"); break;
|
||
|
case 0x0112: lstrcat(buf, "SYSCOMMAND"); break;
|
||
|
case 0x0113: lstrcat(buf, "TIMER"); break;
|
||
|
case 0x0114: lstrcat(buf, "HSCROLL"); break;
|
||
|
case 0x0115: lstrcat(buf, "VSCROLL"); break;
|
||
|
case 0x0116: lstrcat(buf, "INITMENU"); break;
|
||
|
case 0x0117: lstrcat(buf, "INITMENUPOPUP"); break;
|
||
|
case 0x011F: lstrcat(buf, "MENUSELECT"); break;
|
||
|
case 0x0120: lstrcat(buf, "MENUCHAR"); break;
|
||
|
case 0x0121: lstrcat(buf, "ENTERIDLE"); break;
|
||
|
#if(WINVER >= 0x0500)
|
||
|
case 0x0122: lstrcat(buf, "MENURBUTTONUP"); break;
|
||
|
case 0x0123: lstrcat(buf, "MENUDRAG"); break;
|
||
|
case 0x0124: lstrcat(buf, "MENUGETOBJECT"); break;
|
||
|
case 0x0125: lstrcat(buf, "UNINITMENUPOPUP"); break;
|
||
|
case 0x0126: lstrcat(buf, "MENUCOMMAND"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0132: lstrcat(buf, "CTLCOLORMSGBOX"); break;
|
||
|
case 0x0133: lstrcat(buf, "CTLCOLOREDIT"); break;
|
||
|
case 0x0134: lstrcat(buf, "CTLCOLORLISTBOX"); break;
|
||
|
case 0x0135: lstrcat(buf, "CTLCOLORBTN"); break;
|
||
|
case 0x0136: lstrcat(buf, "CTLCOLORDLG"); break;
|
||
|
case 0x0137: lstrcat(buf, "CTLCOLORSCROLLBAR"); break;
|
||
|
case 0x0138: lstrcat(buf, "CTLCOLORSTATIC"); break;
|
||
|
|
||
|
//case 0x0200: lstrcat(buf, "MOUSEFIRST"); break;
|
||
|
case 0x0200: lstrcat(buf, "MOUSEMOVE"); break;
|
||
|
case 0x0201: lstrcat(buf, "LBUTTONDOWN"); break;
|
||
|
case 0x0202: lstrcat(buf, "LBUTTONUP"); break;
|
||
|
case 0x0203: lstrcat(buf, "LBUTTONDBLCLK"); break;
|
||
|
case 0x0204: lstrcat(buf, "RBUTTONDOWN"); break;
|
||
|
case 0x0205: lstrcat(buf, "RBUTTONUP"); break;
|
||
|
case 0x0206: lstrcat(buf, "RBUTTONDBLCLK"); break;
|
||
|
case 0x0207: lstrcat(buf, "MBUTTONDOWN"); break;
|
||
|
case 0x0208: lstrcat(buf, "MBUTTONUP"); break;
|
||
|
case 0x0209: lstrcat(buf, "MBUTTONDBLCLK"); break;
|
||
|
|
||
|
#if (_WIN32_WINNT >= 0x0400) || (_WIN32_WINDOWS > 0x0400)
|
||
|
case 0x020A: lstrcat(buf, "MOUSEWHEEL"); break;
|
||
|
case 0x020E: lstrcat(buf, "MOUSELAST"); break;
|
||
|
#else
|
||
|
//case 0x0209: lstrcat(buf, "MOUSELAST"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0210: lstrcat(buf, "PARENTNOTIFY"); break;
|
||
|
case 0x0211: lstrcat(buf, "ENTERMENULOOP"); break;
|
||
|
case 0x0212: lstrcat(buf, "EXITMENULOOP"); break;
|
||
|
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x0213: lstrcat(buf, "NEXTMENU"); break;
|
||
|
case 0x0214: lstrcat(buf, "SIZING"); break;
|
||
|
case 0x0215: lstrcat(buf, "CAPTURECHANGED"); break;
|
||
|
case 0x0216: lstrcat(buf, "MOVING"); break;
|
||
|
case 0x0218: lstrcat(buf, "POWERBROADCAST"); break;
|
||
|
case 0x0219: lstrcat(buf, "DEVICECHANGE"); break;
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
case 0x0220: lstrcat(buf, "MDICREATE"); break;
|
||
|
case 0x0221: lstrcat(buf, "MDIDESTROY"); break;
|
||
|
case 0x0222: lstrcat(buf, "MDIACTIVATE"); break;
|
||
|
case 0x0223: lstrcat(buf, "MDIRESTORE"); break;
|
||
|
case 0x0224: lstrcat(buf, "MDINEXT"); break;
|
||
|
case 0x0225: lstrcat(buf, "MDIMAXIMIZE"); break;
|
||
|
case 0x0226: lstrcat(buf, "MDITILE"); break;
|
||
|
case 0x0227: lstrcat(buf, "MDICASCADE"); break;
|
||
|
case 0x0228: lstrcat(buf, "MDIICONARRANGE"); break;
|
||
|
case 0x0229: lstrcat(buf, "MDIGETACTIVE"); break;
|
||
|
*/
|
||
|
|
||
|
case 0x0230: lstrcat(buf, "MDISETMENU"); break;
|
||
|
case 0x0231: lstrcat(buf, "ENTERSIZEMOVE"); break;
|
||
|
case 0x0232: lstrcat(buf, "EXITSIZEMOVE"); break;
|
||
|
case 0x0233: lstrcat(buf, "DROPFILES"); break;
|
||
|
case 0x0234: lstrcat(buf, "MDIREFRESHMENU"); break;
|
||
|
|
||
|
|
||
|
/*
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x0281: lstrcat(buf, "IME_SETCONTEXT"); break;
|
||
|
case 0x0282: lstrcat(buf, "IME_NOTIFY"); break;
|
||
|
case 0x0283: lstrcat(buf, "IME_CONTROL"); break;
|
||
|
case 0x0284: lstrcat(buf, "IME_COMPOSITIONFULL"); break;
|
||
|
case 0x0285: lstrcat(buf, "IME_SELECT"); break;
|
||
|
case 0x0286: lstrcat(buf, "IME_CHAR"); break;
|
||
|
#endif
|
||
|
#if(WINVER >= 0x0500)
|
||
|
case 0x0288: lstrcat(buf, "IME_REQUEST"); break;
|
||
|
#endif
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x0290: lstrcat(buf, "IME_KEYDOWN"); break;
|
||
|
case 0x0291: lstrcat(buf, "IME_KEYUP"); break;
|
||
|
#endif
|
||
|
*/
|
||
|
|
||
|
#if(_WIN32_WINNT >= 0x0400)
|
||
|
case 0x02A1: lstrcat(buf, "MOUSEHOVER"); break;
|
||
|
case 0x02A3: lstrcat(buf, "MOUSELEAVE"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0300: lstrcat(buf, "CUT"); break;
|
||
|
case 0x0301: lstrcat(buf, "COPY"); break;
|
||
|
case 0x0302: lstrcat(buf, "PASTE"); break;
|
||
|
case 0x0303: lstrcat(buf, "CLEAR"); break;
|
||
|
case 0x0304: lstrcat(buf, "UNDO"); break;
|
||
|
case 0x0305: lstrcat(buf, "RENDERFORMAT"); break;
|
||
|
case 0x0306: lstrcat(buf, "RENDERALLFORMATS"); break;
|
||
|
case 0x0307: lstrcat(buf, "DESTROYCLIPBOARD"); break;
|
||
|
case 0x0308: lstrcat(buf, "DRAWCLIPBOARD"); break;
|
||
|
case 0x0309: lstrcat(buf, "PAINTCLIPBOARD"); break;
|
||
|
case 0x030A: lstrcat(buf, "VSCROLLCLIPBOARD"); break;
|
||
|
case 0x030B: lstrcat(buf, "SIZECLIPBOARD"); break;
|
||
|
case 0x030C: lstrcat(buf, "ASKCBFORMATNAME"); break;
|
||
|
case 0x030D: lstrcat(buf, "CHANGECBCHAIN"); break;
|
||
|
case 0x030E: lstrcat(buf, "HSCROLLCLIPBOARD"); break;
|
||
|
case 0x030F: lstrcat(buf, "QUERYNEWPALETTE"); break;
|
||
|
case 0x0310: lstrcat(buf, "PALETTEISCHANGING"); break;
|
||
|
case 0x0311: lstrcat(buf, "PALETTECHANGED"); break;
|
||
|
case 0x0312: lstrcat(buf, "HOTKEY"); break;
|
||
|
|
||
|
#if(WINVER >= 0x0400)
|
||
|
case 0x0317: lstrcat(buf, "PRINT"); break;
|
||
|
case 0x0318: lstrcat(buf, "PRINTCLIENT"); break;
|
||
|
|
||
|
case 0x0358: lstrcat(buf, "HANDHELDFIRST"); break;
|
||
|
case 0x035F: lstrcat(buf, "HANDHELDLAST"); break;
|
||
|
|
||
|
case 0x0360: lstrcat(buf, "AFXFIRST"); break;
|
||
|
case 0x037F: lstrcat(buf, "AFXLAST"); break;
|
||
|
#endif
|
||
|
|
||
|
case 0x0380: lstrcat(buf, "PENWINFIRST"); break;
|
||
|
case 0x038F: lstrcat(buf, "PENWINLAST"); break;
|
||
|
|
||
|
default:
|
||
|
sprintf(buf, "unknown");
|
||
|
matched = 0;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
int n = strlen(buf);
|
||
|
int desired_len = 24;
|
||
|
int spaces_to_append = desired_len-n;
|
||
|
if (spaces_to_append>0)
|
||
|
{
|
||
|
for (int i=0; i<spaces_to_append; i++)
|
||
|
buf[n+i] = ' ';
|
||
|
buf[desired_len] = 0;
|
||
|
}
|
||
|
|
||
|
char buf2[256];
|
||
|
if (matched)
|
||
|
sprintf(buf2, "%shwnd=%08x, msg=%s, w=%08x, l=%08x\n", szStartText, hwnd, buf, wParam, lParam);
|
||
|
else
|
||
|
sprintf(buf2, "%shwnd=%08x, msg=unknown/0x%08x, w=%08x, l=%08x\n", szStartText, hwnd, msg, wParam, lParam);
|
||
|
OutputDebugString(buf2);
|
||
|
#endif
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
void DownloadDirectX(HWND hwnd)
|
||
|
{
|
||
|
wchar_t szUrl[] = L"http://www.microsoft.com/download/details.aspx?id=35";
|
||
|
intptr_t ret = myOpenURL(NULL, szUrl);
|
||
|
if (ret <= 32)
|
||
|
{
|
||
|
wchar_t buf[1024];
|
||
|
switch(ret)
|
||
|
{
|
||
|
case SE_ERR_FNF:
|
||
|
case SE_ERR_PNF:
|
||
|
swprintf(buf, WASABI_API_LNGSTRINGW(IDS_URL_COULD_NOT_OPEN), szUrl);
|
||
|
break;
|
||
|
case SE_ERR_ACCESSDENIED:
|
||
|
case SE_ERR_SHARE:
|
||
|
swprintf(buf, WASABI_API_LNGSTRINGW(IDS_ACCESS_TO_URL_WAS_DENIED), szUrl);
|
||
|
break;
|
||
|
case SE_ERR_NOASSOC:
|
||
|
swprintf(buf, WASABI_API_LNGSTRINGW(IDS_ACCESS_TO_URL_FAILED_DUE_TO_NO_ASSOC), szUrl);
|
||
|
break;
|
||
|
default:
|
||
|
swprintf(buf, WASABI_API_LNGSTRINGW(IDS_ACCESS_TO_URL_FAILED_CODE_X), szUrl, ret);
|
||
|
break;
|
||
|
}
|
||
|
MessageBoxW(hwnd, buf, WASABI_API_LNGSTRINGW(IDS_ERROR_OPENING_URL),
|
||
|
MB_OK|MB_SETFOREGROUND|MB_TOPMOST|MB_TASKMODAL);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MissingDirectX(HWND hwnd)
|
||
|
{
|
||
|
// DIRECTX MISSING OR CORRUPT -> PROMPT TO GO TO WEB.
|
||
|
wchar_t title[128];
|
||
|
int ret = MessageBoxW(hwnd,
|
||
|
#ifndef D3D_SDK_VERSION
|
||
|
--- error; you need to #include <d3d9.h> ---
|
||
|
#endif
|
||
|
#if (D3D_SDK_VERSION==120)
|
||
|
// plugin was *built* using the DirectX 9.0 sdk, therefore,
|
||
|
// the dx9.0 runtime is missing or corrupt
|
||
|
"Failed to initialize DirectX 9.0 or later.\n"
|
||
|
"Milkdrop requires d3dx9_31.dll to be installed.\n"
|
||
|
"\n"
|
||
|
"Would you like to be taken to:\n"
|
||
|
"http://www.microsoft.com/download/details.aspx?id=35,\n"
|
||
|
"where you can update DirectX 9.0?\n"
|
||
|
XXXXXXX
|
||
|
#else
|
||
|
// plugin was *built* using some other version of the DirectX9 sdk, such as
|
||
|
// 9.1b; therefore, we don't know exactly what version to tell them they need
|
||
|
// to install; so we ask them to go get the *latest* version.
|
||
|
WASABI_API_LNGSTRINGW(IDS_DIRECTX_MISSING_OR_CORRUPT_TEXT)
|
||
|
#endif
|
||
|
,
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_DIRECTX_MISSING_OR_CORRUPT, title, 128),
|
||
|
MB_YESNO|MB_SETFOREGROUND|MB_TOPMOST|MB_TASKMODAL);
|
||
|
|
||
|
if (ret==IDYES)
|
||
|
DownloadDirectX(hwnd);
|
||
|
}
|
||
|
|
||
|
bool CheckForMMX()
|
||
|
{
|
||
|
#ifdef _WIN64
|
||
|
return true; // All x64 processors support SSE
|
||
|
#else
|
||
|
DWORD bMMX = 0;
|
||
|
DWORD *pbMMX = &bMMX;
|
||
|
__try {
|
||
|
__asm {
|
||
|
mov eax, 1
|
||
|
cpuid
|
||
|
mov edi, pbMMX
|
||
|
mov dword ptr [edi], edx
|
||
|
}
|
||
|
}
|
||
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
bMMX = 0;
|
||
|
}
|
||
|
|
||
|
if (bMMX & 0x00800000) // check bit 23
|
||
|
return true;
|
||
|
|
||
|
return false;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
bool CheckForSSE()
|
||
|
{
|
||
|
#ifdef _WIN64
|
||
|
return true; // All x64 processors support SSE
|
||
|
#else
|
||
|
/*
|
||
|
The SSE instruction set was introduced with the Pentium III and features:
|
||
|
* Additional MMX instructions such as min/max
|
||
|
* Prefetch and write-through instructions for optimizing data movement
|
||
|
from and to the L2/L3 caches and main memory
|
||
|
* 8 New 128 bit XMM registers (xmm0..xmm7) and corresponding 32 bit floating point
|
||
|
(single precision) instructions
|
||
|
*/
|
||
|
|
||
|
DWORD bSSE = 0;
|
||
|
DWORD *pbSSE = &bSSE;
|
||
|
__try {
|
||
|
__asm
|
||
|
{
|
||
|
mov eax, 1
|
||
|
cpuid
|
||
|
mov edi, pbSSE
|
||
|
mov dword ptr [edi], edx
|
||
|
}
|
||
|
}
|
||
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
||
|
{
|
||
|
bSSE = 0;
|
||
|
}
|
||
|
|
||
|
if (bSSE & 0x02000000) // check bit 25
|
||
|
return true;
|
||
|
|
||
|
return false;
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void GetDesktopFolder(char *szDesktopFolder) // should be MAX_PATH len.
|
||
|
{
|
||
|
// returns the path to the desktop folder, WITHOUT a trailing backslash.
|
||
|
szDesktopFolder[0] = 0;
|
||
|
ITEMIDLIST pidl;
|
||
|
ZeroMemory(&pidl, sizeof(pidl));
|
||
|
if (!SHGetPathFromIDList(&pidl, szDesktopFolder))
|
||
|
szDesktopFolder[0] = 0;
|
||
|
}
|
||
|
|
||
|
void ExecutePidl(LPITEMIDLIST pidl, char *szPathAndFile, char *szWorkingDirectory, HWND hWnd)
|
||
|
{
|
||
|
// This function was based on code by Jeff Prosise.
|
||
|
|
||
|
// Note: for some reason, ShellExecuteEx fails when executing
|
||
|
// *shortcuts* (.lnk files) from the desktop, using their PIDLs.
|
||
|
// So, if that fails, we try again w/the plain old text filename
|
||
|
// (szPathAndFile).
|
||
|
|
||
|
char szVerb[] = "open";
|
||
|
char szFilename2[MAX_PATH];
|
||
|
|
||
|
sprintf(szFilename2, "%s.lnk", szPathAndFile);
|
||
|
|
||
|
// -without the "no-verb" pass,
|
||
|
// certain icons still don't work (like shortcuts
|
||
|
// to IE, VTune...)
|
||
|
// -without the "context menu" pass,
|
||
|
// certain others STILL don't work (Netscape...)
|
||
|
// -without the 'ntry' pass, shortcuts (to folders/files)
|
||
|
// don't work
|
||
|
for (int verb_pass=0; verb_pass<2; verb_pass++)
|
||
|
{
|
||
|
for (int ntry=0; ntry<3; ntry++)
|
||
|
{
|
||
|
for (int context_pass=0; context_pass<2; context_pass++)
|
||
|
{
|
||
|
SHELLEXECUTEINFO sei = { sizeof(sei) };
|
||
|
sei.hwnd = hWnd;
|
||
|
sei.fMask = SEE_MASK_FLAG_NO_UI;
|
||
|
if (context_pass==1)
|
||
|
sei.fMask |= SEE_MASK_INVOKEIDLIST;
|
||
|
sei.lpVerb = (verb_pass) ? NULL : szVerb;
|
||
|
sei.lpDirectory = szWorkingDirectory;
|
||
|
sei.nShow = SW_SHOWNORMAL;
|
||
|
|
||
|
if (ntry==0)
|
||
|
{
|
||
|
// this case works for most non-shortcuts
|
||
|
sei.fMask |= SEE_MASK_IDLIST;
|
||
|
sei.lpIDList = pidl;
|
||
|
}
|
||
|
else if (ntry==1)
|
||
|
{
|
||
|
// this case is required for *shortcuts to folders* to work
|
||
|
sei.lpFile = szPathAndFile;
|
||
|
}
|
||
|
else if (ntry==2)
|
||
|
{
|
||
|
// this case is required for *shortcuts to files* to work
|
||
|
sei.lpFile = szFilename2;
|
||
|
}
|
||
|
|
||
|
if (ShellExecuteEx(&sei))
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WNDPROC g_pOldWndProc;
|
||
|
LPCONTEXTMENU2 g_pIContext2or3;
|
||
|
|
||
|
LRESULT CALLBACK HookWndProc(HWND hWnd, UINT msg, WPARAM wp, LPARAM lp)
|
||
|
{
|
||
|
//UINT uItem;
|
||
|
//TCHAR szBuf[MAX_PATH];
|
||
|
|
||
|
switch (msg)
|
||
|
{
|
||
|
case WM_DRAWITEM:
|
||
|
case WM_MEASUREITEM:
|
||
|
if(wp) break; // not menu related
|
||
|
case WM_INITMENUPOPUP:
|
||
|
g_pIContext2or3->HandleMenuMsg(msg, wp, lp);
|
||
|
return (msg==WM_INITMENUPOPUP ? 0 : TRUE); // handled
|
||
|
|
||
|
/*case WM_MENUSELECT:
|
||
|
// if this is a shell item, get its descriptive text
|
||
|
uItem = (UINT) LOWORD(wp);
|
||
|
if(0 == (MF_POPUP & HIWORD(wp)) && uItem >= 1 && uItem <= 0x7fff)
|
||
|
{
|
||
|
g_pIContext2or3->GetCommandString(uItem-1, GCS_HELPTEXT,
|
||
|
NULL, szBuf, sizeof(szBuf)/sizeof(szBuf[0]) );
|
||
|
|
||
|
// set the status bar text
|
||
|
((CFrameWnd*)(AfxGetApp()->m_pMainWnd))->SetMessageText(szBuf);
|
||
|
return 0;
|
||
|
}
|
||
|
break;*/
|
||
|
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// for all untreated messages, call the original wndproc
|
||
|
return ::CallWindowProc(g_pOldWndProc, hWnd, msg, wp, lp);
|
||
|
}
|
||
|
|
||
|
BOOL DoExplorerMenu (HWND hwnd, LPITEMIDLIST pidlMain, POINT point)
|
||
|
{
|
||
|
LPMALLOC pMalloc;
|
||
|
LPSHELLFOLDER psfFolder, psfNextFolder;
|
||
|
LPITEMIDLIST pidlItem, pidlNextItem, *ppidl;
|
||
|
LPCONTEXTMENU pContextMenu;
|
||
|
CMINVOKECOMMANDINFO ici;
|
||
|
UINT nCount, nCmd;
|
||
|
BOOL bResult;
|
||
|
HMENU hMenu;
|
||
|
|
||
|
//
|
||
|
// Get pointers to the shell's IMalloc interface and the desktop's
|
||
|
// IShellFolder interface.
|
||
|
//
|
||
|
bResult = FALSE;
|
||
|
|
||
|
if (!SUCCEEDED (SHGetMalloc (&pMalloc)))
|
||
|
return bResult;
|
||
|
|
||
|
if (!SUCCEEDED (SHGetDesktopFolder (&psfFolder))) {
|
||
|
pMalloc->Release();
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
if (nCount = GetItemCount (pidlMain)) // nCount must be > 0
|
||
|
{
|
||
|
//
|
||
|
// Initialize psfFolder with a pointer to the IShellFolder
|
||
|
// interface of the folder that contains the item whose context
|
||
|
// menu we're after, and initialize pidlItem with a pointer to
|
||
|
// the item's item ID. If nCount > 1, this requires us to walk
|
||
|
// the list of item IDs stored in pidlMain and bind to each
|
||
|
// subfolder referenced in the list.
|
||
|
//
|
||
|
pidlItem = pidlMain;
|
||
|
|
||
|
while (--nCount) {
|
||
|
//
|
||
|
// Create a 1-item item ID list for the next item in pidlMain.
|
||
|
//
|
||
|
pidlNextItem = DuplicateItem (pMalloc, pidlItem);
|
||
|
if (pidlNextItem == NULL) {
|
||
|
psfFolder->Release();
|
||
|
pMalloc->Release();
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Bind to the folder specified in the new item ID list.
|
||
|
//
|
||
|
if (!SUCCEEDED (psfFolder->BindToObject(pidlNextItem, NULL, IID_IShellFolder, (void**)&psfNextFolder))) // modified by RG
|
||
|
{
|
||
|
pMalloc->Free(pidlNextItem);
|
||
|
psfFolder->Release();
|
||
|
pMalloc->Release();
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Release the IShellFolder pointer to the parent folder
|
||
|
// and set psfFolder equal to the IShellFolder pointer for
|
||
|
// the current folder.
|
||
|
//
|
||
|
psfFolder->Release();
|
||
|
psfFolder = psfNextFolder;
|
||
|
|
||
|
//
|
||
|
// Release the storage for the 1-item item ID list we created
|
||
|
// just a moment ago and initialize pidlItem so that it points
|
||
|
// to the next item in pidlMain.
|
||
|
//
|
||
|
pMalloc->Free(pidlNextItem);
|
||
|
pidlItem = GetNextItem (pidlItem);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Get a pointer to the item's IContextMenu interface and call
|
||
|
// IContextMenu::QueryContextMenu to initialize a context menu.
|
||
|
//
|
||
|
ppidl = &pidlItem;
|
||
|
if (SUCCEEDED (psfFolder->GetUIObjectOf(hwnd, 1, (LPCITEMIDLIST*)ppidl, IID_IContextMenu, NULL, (void**)&pContextMenu))) // modified by RG
|
||
|
{
|
||
|
// try to see if we can upgrade to an IContextMenu3
|
||
|
// or IContextMenu2 interface pointer:
|
||
|
int level = 1;
|
||
|
void *pCM = NULL;
|
||
|
if (pContextMenu->QueryInterface(IID_IContextMenu3, &pCM) == NOERROR)
|
||
|
{
|
||
|
pContextMenu->Release();
|
||
|
pContextMenu = (LPCONTEXTMENU)pCM;
|
||
|
level = 3;
|
||
|
}
|
||
|
else if (pContextMenu->QueryInterface(IID_IContextMenu2, &pCM) == NOERROR)
|
||
|
{
|
||
|
pContextMenu->Release();
|
||
|
pContextMenu = (LPCONTEXTMENU)pCM;
|
||
|
level = 2;
|
||
|
}
|
||
|
|
||
|
hMenu = CreatePopupMenu ();
|
||
|
if (SUCCEEDED (pContextMenu->QueryContextMenu(hMenu, 0, 1, 0x7FFF, CMF_EXPLORE)))
|
||
|
{
|
||
|
ClientToScreen (hwnd, &point);
|
||
|
|
||
|
// install the subclassing "hook", for versions 2 or 3
|
||
|
if (level >= 2)
|
||
|
{
|
||
|
g_pOldWndProc = (WNDPROC)SetWindowLongPtr(hwnd, GWLP_WNDPROC, (DWORD_PTR)HookWndProc);
|
||
|
g_pIContext2or3 = (LPCONTEXTMENU2)pContextMenu; // cast ok for ICMv3
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
g_pOldWndProc = NULL;
|
||
|
g_pIContext2or3 = NULL;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Display the context menu.
|
||
|
//
|
||
|
nCmd = TrackPopupMenu (hMenu, TPM_LEFTALIGN |
|
||
|
TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_RETURNCMD,
|
||
|
point.x, point.y, 0, hwnd, NULL);
|
||
|
|
||
|
// restore old wndProc
|
||
|
if (g_pOldWndProc)
|
||
|
{
|
||
|
SetWindowLongPtr(hwnd, GWLP_WNDPROC, (LONG_PTR)g_pOldWndProc);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// If a command was selected from the menu, execute it.
|
||
|
//
|
||
|
if (nCmd >= 1 && nCmd <= 0x7fff)
|
||
|
{
|
||
|
ZeroMemory(&ici, sizeof(ici));
|
||
|
ici.cbSize = sizeof (CMINVOKECOMMANDINFO);
|
||
|
//ici.fMask = 0;
|
||
|
ici.hwnd = hwnd;
|
||
|
ici.lpVerb = MAKEINTRESOURCE (nCmd - 1);
|
||
|
//ici.lpParameters = NULL;
|
||
|
//ici.lpDirectory = NULL;
|
||
|
ici.nShow = SW_SHOWNORMAL;
|
||
|
//ici.dwHotKey = 0;
|
||
|
//ici.hIcon = NULL;
|
||
|
|
||
|
if (SUCCEEDED ( pContextMenu->InvokeCommand (&ici)))
|
||
|
bResult = TRUE;
|
||
|
}
|
||
|
/*else if (nCmd)
|
||
|
{
|
||
|
PostMessage(hwnd, WM_COMMAND, nCmd, NULL); // our command
|
||
|
}*/
|
||
|
}
|
||
|
DestroyMenu (hMenu);
|
||
|
pContextMenu->Release();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Clean up and return.
|
||
|
//
|
||
|
psfFolder->Release();
|
||
|
pMalloc->Release();
|
||
|
|
||
|
return bResult;
|
||
|
}
|
||
|
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
//
|
||
|
// Note: a special thanks goes out to Jeff Prosise for writing & publishing
|
||
|
// the following code!
|
||
|
//
|
||
|
// FUNCTION: GetItemCount
|
||
|
//
|
||
|
// DESCRIPTION: Computes the number of item IDs in an item ID list.
|
||
|
//
|
||
|
// INPUT: pidl = Pointer to an item ID list.
|
||
|
//
|
||
|
// RETURNS: Number of item IDs in the list.
|
||
|
//
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
UINT GetItemCount (LPITEMIDLIST pidl)
|
||
|
{
|
||
|
USHORT nLen;
|
||
|
UINT nCount;
|
||
|
|
||
|
nCount = 0;
|
||
|
while ((nLen = pidl->mkid.cb) != 0) {
|
||
|
pidl = GetNextItem (pidl);
|
||
|
nCount++;
|
||
|
}
|
||
|
return nCount;
|
||
|
}
|
||
|
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
//
|
||
|
// Note: a special thanks goes out to Jeff Prosise for writing & publishing
|
||
|
// the following code!
|
||
|
//
|
||
|
// FUNCTION: GetNextItem
|
||
|
//
|
||
|
// DESCRIPTION: Finds the next item in an item ID list.
|
||
|
//
|
||
|
// INPUT: pidl = Pointer to an item ID list.
|
||
|
//
|
||
|
// RETURNS: Pointer to the next item.
|
||
|
//
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
LPITEMIDLIST GetNextItem (LPITEMIDLIST pidl)
|
||
|
{
|
||
|
USHORT nLen;
|
||
|
|
||
|
if ((nLen = pidl->mkid.cb) == 0)
|
||
|
return NULL;
|
||
|
|
||
|
return (LPITEMIDLIST) (((LPBYTE) pidl) + nLen);
|
||
|
}
|
||
|
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
//
|
||
|
// Note: a special thanks goes out to Jeff Prosise for writing & publishing
|
||
|
// the following code!
|
||
|
//
|
||
|
// FUNCTION: DuplicateItem
|
||
|
//
|
||
|
// DESCRIPTION: Makes a copy of the next item in an item ID list.
|
||
|
//
|
||
|
// INPUT: pMalloc = Pointer to an IMalloc interface.
|
||
|
// pidl = Pointer to an item ID list.
|
||
|
//
|
||
|
// RETURNS: Pointer to an ITEMIDLIST containing the copied item ID.
|
||
|
//
|
||
|
// NOTES: It is the caller's responsibility to free the memory
|
||
|
// allocated by this function when the item ID is no longer
|
||
|
// needed. Example:
|
||
|
//
|
||
|
// pidlItem = DuplicateItem (pMalloc, pidl);
|
||
|
// .
|
||
|
// .
|
||
|
// .
|
||
|
// pMalloc->lpVtbl->Free (pMalloc, pidlItem);
|
||
|
//
|
||
|
// Failure to free the ITEMIDLIST will result in memory
|
||
|
// leaks.
|
||
|
//
|
||
|
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
LPITEMIDLIST DuplicateItem (LPMALLOC pMalloc, LPITEMIDLIST pidl)
|
||
|
{
|
||
|
USHORT nLen;
|
||
|
LPITEMIDLIST pidlNew;
|
||
|
|
||
|
nLen = pidl->mkid.cb;
|
||
|
if (nLen == 0)
|
||
|
return NULL;
|
||
|
|
||
|
pidlNew = (LPITEMIDLIST) pMalloc->Alloc (
|
||
|
nLen + sizeof (USHORT));
|
||
|
if (pidlNew == NULL)
|
||
|
return NULL;
|
||
|
|
||
|
CopyMemory (pidlNew, pidl, nLen);
|
||
|
*((USHORT*) (((LPBYTE) pidlNew) + nLen)) = 0;
|
||
|
|
||
|
return pidlNew;
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------
|
||
|
// A special thanks goes out to Jeroen-bart Engelen (Yeep) for providing
|
||
|
// his source code for getting the position & label information for all
|
||
|
// the icons on the desktop, as found below. See his article at
|
||
|
// http://www.digiwar.com/scripts/renderpage.php?section=2&subsection=2
|
||
|
//----------------------------------------------------------------------
|
||
|
|
||
|
void FindDesktopWindows(HWND *desktop_progman, HWND *desktopview_wnd, HWND *listview_wnd)
|
||
|
{
|
||
|
*desktop_progman = NULL;
|
||
|
*desktopview_wnd = NULL;
|
||
|
*listview_wnd = NULL;
|
||
|
|
||
|
*desktop_progman = FindWindow(NULL, ("Program Manager"));
|
||
|
if(*desktop_progman == NULL)
|
||
|
{
|
||
|
//MessageBox(NULL, "Unable to get the handle to the Program Manager.", "Fatal error", MB_OK|MB_ICONERROR);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
*desktopview_wnd = FindWindowEx(*desktop_progman, NULL, "SHELLDLL_DefView", NULL);
|
||
|
if(*desktopview_wnd == NULL)
|
||
|
{
|
||
|
//MessageBox(NULL, "Unable to get the handle to the desktopview.", "Fatal error", MB_OK|MB_ICONERROR);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Thanks ef_ef_ef@yahoo.com for pointing out this works in NT 4 and not the way I did it originally.
|
||
|
*listview_wnd = FindWindowEx(*desktopview_wnd, NULL, "SysListView32", NULL);
|
||
|
if(*listview_wnd == NULL)
|
||
|
{
|
||
|
//MessageBox(NULL, "Unable to get the handle to the folderview.", "Fatal error", MB_OK|MB_ICONERROR);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------
|
||
|
|
||
|
int GetDesktopIconSize()
|
||
|
{
|
||
|
int ret = 32;
|
||
|
|
||
|
// reads the key: HKEY_CURRENT_USER\Control Panel, Desktop\WindowMetrics\Shell Icon Size
|
||
|
unsigned char buf[64];
|
||
|
unsigned long len = sizeof(buf);
|
||
|
DWORD type;
|
||
|
HKEY key;
|
||
|
|
||
|
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_CURRENT_USER, "Control Panel\\Desktop\\WindowMetrics", 0, KEY_READ, &key))
|
||
|
{
|
||
|
if (ERROR_SUCCESS == RegQueryValueEx(key, "Shell Icon Size", NULL, &type, (unsigned char*)buf, &len) &&
|
||
|
type == REG_SZ)
|
||
|
{
|
||
|
int x = _atoi_l((char*)buf, g_use_C_locale);
|
||
|
if (x>0 && x<=128)
|
||
|
ret = x;
|
||
|
}
|
||
|
|
||
|
RegCloseKey(key);
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
//----------------------------------------------------------------------
|
||
|
|
||
|
// handy functions for populating Combo Boxes:
|
||
|
int SelectItemByValue(HWND ctrl, DWORD value)
|
||
|
{
|
||
|
int count = SendMessage(ctrl, CB_GETCOUNT, 0, 0);
|
||
|
for (int i=0; i<count; i++)
|
||
|
{
|
||
|
DWORD value_i = SendMessage( ctrl, CB_GETITEMDATA, i, 0);
|
||
|
if (value_i == value)
|
||
|
{
|
||
|
SendMessage( ctrl, CB_SETCURSEL, i, 0);
|
||
|
return i;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
bool ReadCBValue(HWND hwnd, DWORD ctrl_id, int* pRetValue)
|
||
|
{
|
||
|
if (!pRetValue)
|
||
|
return false;
|
||
|
HWND ctrl = GetDlgItem( hwnd, ctrl_id );
|
||
|
int t = SendMessage( ctrl, CB_GETCURSEL, 0, 0);
|
||
|
if (t == CB_ERR)
|
||
|
return false;
|
||
|
*pRetValue = (int)SendMessage( ctrl, CB_GETITEMDATA, t, 0);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
D3DXCREATEFONTW pCreateFontW=0;
|
||
|
D3DXMATRIXMULTIPLY pMatrixMultiply=0;
|
||
|
D3DXMATRIXTRANSLATION pMatrixTranslation=0;
|
||
|
D3DXMATRIXSCALING pMatrixScaling=0;
|
||
|
D3DXMATRIXROTATION pMatrixRotationX=0, pMatrixRotationY=0, pMatrixRotationZ=0;
|
||
|
D3DXCREATETEXTUREFROMFILEEXW pCreateTextureFromFileExW=0;
|
||
|
D3DXMATRIXORTHOLH pMatrixOrthoLH = 0;
|
||
|
D3DXCOMPILESHADER pCompileShader=0;
|
||
|
D3DXMATRIXLOOKATLH pMatrixLookAtLH=0;
|
||
|
D3DXCREATETEXTURE pCreateTexture=0;
|
||
|
//----------------------------------------------------------------------
|
||
|
HMODULE FindD3DX9(HWND winamp)
|
||
|
{
|
||
|
HMODULE d3dx9 = (HMODULE)SendMessage(winamp,WM_WA_IPC, 0, IPC_GET_D3DX9);
|
||
|
if (!d3dx9 || d3dx9 == (HMODULE)1)
|
||
|
{
|
||
|
|
||
|
// TODO: benski> this is a quick-fix, we should call FindFirstFile() on the system directory
|
||
|
d3dx9=NULL;
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_36.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_35.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_34.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_33.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_32.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_31.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_30.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_29.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_28.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_27.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_26.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_25.dll");
|
||
|
if (!d3dx9) d3dx9 = LoadLibrary("d3dx9_24.dll");
|
||
|
}
|
||
|
|
||
|
if (d3dx9)
|
||
|
{
|
||
|
pCreateFontW = (D3DXCREATEFONTW) GetProcAddress(d3dx9,"D3DXCreateFontW");
|
||
|
pMatrixMultiply = (D3DXMATRIXMULTIPLY) GetProcAddress(d3dx9,"D3DXMatrixMultiply");
|
||
|
pMatrixTranslation = (D3DXMATRIXTRANSLATION)GetProcAddress(d3dx9,"D3DXMatrixTranslation");
|
||
|
pMatrixScaling = (D3DXMATRIXSCALING)GetProcAddress(d3dx9,"D3DXMatrixScaling");
|
||
|
pMatrixRotationX = (D3DXMATRIXROTATION)GetProcAddress(d3dx9,"D3DXMatrixRotationX");
|
||
|
pMatrixRotationY = (D3DXMATRIXROTATION)GetProcAddress(d3dx9,"D3DXMatrixRotationY");
|
||
|
pMatrixRotationZ = (D3DXMATRIXROTATION)GetProcAddress(d3dx9,"D3DXMatrixRotationZ");
|
||
|
pCreateTextureFromFileExW = (D3DXCREATETEXTUREFROMFILEEXW)GetProcAddress(d3dx9,"D3DXCreateTextureFromFileExW");
|
||
|
pMatrixOrthoLH = (D3DXMATRIXORTHOLH)GetProcAddress(d3dx9,"D3DXMatrixOrthoLH");
|
||
|
pCompileShader = (D3DXCOMPILESHADER)GetProcAddress(d3dx9,"D3DXCompileShader");
|
||
|
pMatrixLookAtLH = (D3DXMATRIXLOOKATLH)GetProcAddress(d3dx9,"D3DXMatrixLookAtLH");
|
||
|
pCreateTexture = (D3DXCREATETEXTURE)GetProcAddress(d3dx9,"D3DXCreateTexture");
|
||
|
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
return d3dx9;
|
||
|
}
|
||
|
|
||
|
LRESULT GetWinampVersion(HWND winamp)
|
||
|
{
|
||
|
static LRESULT version=0;
|
||
|
if (!version)
|
||
|
version=SendMessage(winamp,WM_WA_IPC,0,0);
|
||
|
return version;
|
||
|
}
|
||
|
|
||
|
void* GetTextResource(UINT id, int no_fallback){
|
||
|
void* data = 0;
|
||
|
HINSTANCE hinst = WASABI_API_LNG_HINST;
|
||
|
HRSRC rsrc = FindResource(hinst,MAKEINTRESOURCE(id),"TEXT");
|
||
|
if(!rsrc && !no_fallback) rsrc = FindResource((hinst = WASABI_API_ORIG_HINST),MAKEINTRESOURCE(id),"TEXT");
|
||
|
if(rsrc){
|
||
|
HGLOBAL resourceHandle = LoadResource(hinst,rsrc);
|
||
|
data = LockResource(resourceHandle);
|
||
|
}
|
||
|
return data;
|
||
|
}
|