winamp/Src/nu/menushortcuts.cpp
2024-09-24 14:54:57 +02:00

204 lines
5.6 KiB
C++

#include "./menushortcuts.h"
#include <strsafe.h>
// change key string to capitalised only so it's more UI consistant (otherwise messes up the localisation look)
wchar_t* CapitaliseKeyName(wchar_t* szAccelName){
if(szAccelName) {
wchar_t *p = &szAccelName[1];
int space = 0;
while(p && *p){
if(*p == L' ') space = 1;
else {
if(!space) {
*p = tolower(*p);
}
else {
space = 0;
}
}
p = CharNextW(p);
}
}
return szAccelName;
}
BOOL AppendShortcutTextEx(LPWSTR pszItemText, INT cchTextMax, WORD wID, ACCEL *pszAccel, INT cchAccel, UINT uMode)
{
BOOL bDirty = FALSE;
if (!pszItemText) return FALSE;
UINT cchLen = lstrlenW(pszItemText);
if (MSF_REPLACE == (0x0F & uMode))
{
UINT len;
for (len = 0; len < cchLen && L'\t' != pszItemText[len]; len++);
if (cchLen != len)
{
cchLen = len;
pszItemText[len] = L'\0';
bDirty = TRUE;
}
}
if (wID > 0)
{
wchar_t szAccelName[64] = {0};
BOOL bFirst = TRUE;
pszItemText += cchLen;
size_t cchText = cchTextMax - cchLen;
for(int k = 0; k < cchAccel; k++)
{
if (wID == pszAccel[k].cmd)
{
HRESULT hr = StringCchCopyExW(pszItemText, cchText, ((bFirst) ? L"\t" : L", "), &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
if (S_OK == hr && (FCONTROL & pszAccel[k].fVirt)) {
GetKeyNameTextW(MapVirtualKey(VK_CONTROL, 0)<<16, szAccelName, sizeof(szAccelName)/sizeof(szAccelName[0]));
hr = StringCchCopyExW(pszItemText, cchText, CapitaliseKeyName(szAccelName), &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
if(S_OK == hr) StringCchCatExW(pszItemText, cchText, L"+", &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
}
if (S_OK == hr && (FALT & pszAccel[k].fVirt)) {
GetKeyNameTextW(MapVirtualKey(VK_MENU, 0)<<16, szAccelName, sizeof(szAccelName)/sizeof(szAccelName[0]));
hr = StringCchCopyExW(pszItemText, cchText, CapitaliseKeyName(szAccelName), &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
if(S_OK == hr) hr = StringCchCatExW(pszItemText, cchText, L"+", &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
}
if (S_OK == hr && (FSHIFT & pszAccel[k].fVirt)) {
GetKeyNameTextW(MapVirtualKey(VK_SHIFT, 0)<<16, szAccelName, sizeof(szAccelName)/sizeof(szAccelName[0]));
hr = StringCchCopyExW(pszItemText, cchText, CapitaliseKeyName(szAccelName), &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
if(S_OK == hr) hr = StringCchCatExW(pszItemText, cchText, L"+", &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
}
if (S_OK == hr)
{
if (FVIRTKEY & pszAccel[k].fVirt)
{
szAccelName[0] = L'\0';
UINT vkey = MapVirtualKey(pszAccel[k].key, 0);
/* benski> this removes "NUM" from the descriptions of certain characters */
switch(pszAccel[k].key)
{
case VK_INSERT:
case VK_DELETE:
case VK_HOME:
case VK_END:
case VK_NEXT: // Page down
case VK_PRIOR: // Page up
case VK_LEFT:
case VK_RIGHT:
case VK_UP:
case VK_DOWN:
vkey |= 0x100; // Add extended bit
}
vkey<<=16;
if (GetKeyNameTextW(vkey, szAccelName, sizeof(szAccelName)/sizeof(szAccelName[0]))) {
hr = StringCchCopyExW(pszItemText, cchText, CapitaliseKeyName(szAccelName), &pszItemText, &cchText, STRSAFE_IGNORE_NULLS);
}
}
else if (cchText > 1)
{
pszItemText[0] = (wchar_t)pszAccel[k].key;
pszItemText[1] = L'\0';
pszItemText += 2;
cchText -= 2;
}
}
bFirst = FALSE;
bDirty = (S_OK == hr);
}
}
}
return bDirty;
}
BOOL AppendMenuShortcutsEx(HMENU hMenu, ACCEL *pszAccel, INT cchAccel, UINT uMode)
{
wchar_t szText[4096] = {0};
if (!hMenu) return FALSE;
MENUITEMINFOW mii = { sizeof(MENUITEMINFOW), };
INT c = GetMenuItemCount(hMenu);
if (0 == c || -1 == c) return TRUE;
for (int i = 0; i < c; i++)
{
mii.fMask = MIIM_ID | MIIM_STRING | MIIM_FTYPE | MIIM_SUBMENU;
mii.cch = sizeof(szText)/sizeof(szText[0]);
mii.dwTypeData = szText;
if (GetMenuItemInfoW(hMenu, i, TRUE, &mii) &&
0 == ((MFT_SEPARATOR | MFT_MENUBREAK |MFT_MENUBARBREAK) & mii.fType))
{
if (AppendShortcutTextEx(mii.dwTypeData, sizeof(szText)/sizeof(szText[0]), (WORD)mii.wID, pszAccel, cchAccel, uMode))
{
mii.fMask = MIIM_STRING;
SetMenuItemInfoW(hMenu, i, TRUE, &mii);
}
if (NULL != mii.hSubMenu && (MSF_WALKSUBMENU & uMode))
{
AppendMenuShortcutsEx(mii.hSubMenu, pszAccel, cchAccel, uMode);
}
}
}
return TRUE;
}
BOOL AppendMenuShortcuts(HMENU hMenu, HACCEL *phAccel, INT count, UINT uMode)
{
ACCEL *pszAccel = NULL;
INT c = 0;
if (!hMenu) return FALSE;
if (phAccel)
{
for (int i = 0; i < count; i++)
{
c += CopyAcceleratorTable(phAccel[i], NULL, 0);
}
if (c)
{
pszAccel = (ACCEL*)calloc(c, sizeof(ACCEL));
if (!pszAccel) return FALSE;
}
int k = 0;
for (int i = 0; i < count; i++)
{
k += CopyAcceleratorTable(phAccel[i], &pszAccel[k], c - k);
}
}
BOOL r = AppendMenuShortcutsEx(hMenu, pszAccel, c, uMode);
if (pszAccel) free(pszAccel);
return r;
}
BOOL AppendShortcutText(LPWSTR pszItemText, INT cchTextMax, WORD wID, HACCEL *phAccel, INT count, UINT uMode)
{
ACCEL *pszAccel = NULL;
INT c = 0;
if (!pszItemText) return FALSE;
if (phAccel)
{
for (int i = 0; i < count; i++)
{
c += CopyAcceleratorTable(phAccel[i], NULL, 0);
}
if (c)
{
pszAccel = (ACCEL*)calloc(c, sizeof(ACCEL));
if (!pszAccel) return FALSE;
}
int k = 0;
for (int i = 0; i < count; i++)
{
k += CopyAcceleratorTable(phAccel[i], &pszAccel[k], c - k);
}
}
BOOL r = AppendShortcutTextEx(pszItemText, cchTextMax, wID, pszAccel, c, uMode);
if (pszAccel) free(pszAccel);
return r;
}