mirror of
https://github.com/WinampDesktop/winamp.git
synced 2025-01-28 20:48:23 +00:00
654 lines
20 KiB
C++
654 lines
20 KiB
C++
/** (c) Nullsoft, Inc. C O N F I D E N T I A L
|
|
** Filename:
|
|
** Project:
|
|
** Description:
|
|
** Author:
|
|
** Created:
|
|
**/
|
|
|
|
#include "main.h"
|
|
#include "menuv5.h"
|
|
#include "../nu/AutoWide.h"
|
|
#include "../Plugins/General/gen_ml/ml.h"
|
|
|
|
#define TREE_LOCALMEDIA 1000
|
|
#define TREE_PLAYLISTS 3001
|
|
#define TREE_DEVICES 10000
|
|
#define TREE_QUERIES 1000
|
|
|
|
extern HINSTANCE language_pack_instance;
|
|
|
|
|
|
void ensureInScreen(HMENU menu, int *x, int *y, int *flag, int width, int height);
|
|
|
|
HMENU v5_top_menu = NULL;
|
|
|
|
|
|
int MergeMenu(HMENU pMenuDestination, const HMENU pMenuAdd, int bTopLevel /*=false*/)
|
|
{
|
|
// Abstract:
|
|
// Merges two menus.
|
|
//
|
|
// Parameters:
|
|
// pMenuDestination - [in, retval] destination menu handle
|
|
// pMenuAdd - [in] menu to merge
|
|
// bTopLevel - [in] indicator for special top level behavior
|
|
//
|
|
// Return value:
|
|
// <false> in case of error.
|
|
//
|
|
// Comments:
|
|
// This function calles itself recursivley. If bTopLevel is set to true,
|
|
// we append popups at top level or we insert before <Window> or <Help>.
|
|
|
|
// get the number menu items in the menus
|
|
int iMenuAddItemCount = GetMenuItemCount(pMenuAdd);
|
|
int iMenuDestItemCount = GetMenuItemCount(pMenuDestination);
|
|
int iLoop;
|
|
|
|
// if there are no items return
|
|
if( iMenuAddItemCount == 0 )
|
|
return 1;
|
|
|
|
// if we are not at top level and the destination menu is not empty
|
|
// -> we append a seperator
|
|
if( !bTopLevel && iMenuDestItemCount > 0 )
|
|
AppendMenu(pMenuDestination, MF_SEPARATOR, 0, 0);
|
|
|
|
// iterate through the top level of
|
|
for( iLoop = 0; iLoop < iMenuAddItemCount; iLoop++ )
|
|
{
|
|
HMENU pSubMenu = 0;
|
|
// get the menu string from the add menu
|
|
wchar_t sMenuAddString[1024] = {0}; // hope it's enough
|
|
|
|
GetMenuStringW(pMenuAdd, iLoop, sMenuAddString, 1024, MF_BYPOSITION);
|
|
|
|
// try to get the submenu of the current menu item
|
|
pSubMenu =GetSubMenu(pMenuAdd, iLoop);
|
|
|
|
// check if we have a sub menu
|
|
if (!pSubMenu)
|
|
{
|
|
// normal menu item
|
|
// read the source and append at the destination
|
|
UINT nState = GetMenuState(pMenuAdd, iLoop, MF_BYPOSITION);
|
|
UINT nItemID = GetMenuItemID(pMenuAdd, iLoop);
|
|
|
|
if(AppendMenuW(pMenuDestination, nState, nItemID, sMenuAddString))
|
|
{
|
|
// menu item added, don't forget to correct the item count
|
|
iMenuDestItemCount++;
|
|
}
|
|
else
|
|
{
|
|
// MergeMenu: AppendMenu failed!
|
|
return 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
HMENU NewPopupMenu=NULL;
|
|
// create or insert a new popup menu item
|
|
|
|
// default insert pos is like ap
|
|
int iInsertPosDefault = -1;
|
|
|
|
// if we are at top level merge into existing popups rather than
|
|
// creating new ones
|
|
if( bTopLevel )
|
|
{
|
|
//ASSERT( sMenuAddString != "&?" && sMenuAddString != "?" );
|
|
//CString sAdd( sMenuAddString );
|
|
//sAdd.Remove('&'); // for comparison of menu items supress '&'
|
|
int bAdded = 0;
|
|
int iLoop1=0;
|
|
|
|
// try to find existing popup
|
|
for( iLoop1 = 0; iLoop1 < iMenuDestItemCount; iLoop1++ )
|
|
{
|
|
// get the menu string from the destination menu
|
|
wchar_t sDest[1024] = {0}; // hope it's enough
|
|
GetMenuStringW(pMenuDestination, iLoop1, sDest, 1024, MF_BYPOSITION );
|
|
//sDest.Remove( '&' ); // for a better compare (s.a.)
|
|
|
|
//if( !lstrcmp(sAdd,sDest))
|
|
{
|
|
// we got a hit -> merge the two popups
|
|
// try to get the submenu of the desired destination menu item
|
|
HMENU pSubMenuDest = GetSubMenu(pMenuDestination, iLoop1 );
|
|
|
|
if( pSubMenuDest )
|
|
{
|
|
// merge the popup recursivly and continue with outer for loop
|
|
if( !MergeMenu( pSubMenuDest, pSubMenu, 0 ))
|
|
return 0;
|
|
|
|
bAdded = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// alternativ insert before <Window> or <Help>
|
|
//if( iInsertPosDefault == -1 && ( sDest == "Window" || sDest == "?" || sDest == "Help" ))
|
|
// iInsertPosDefault = iLoop1;
|
|
|
|
}
|
|
|
|
if( bAdded )
|
|
{
|
|
// menu added, so go on with loop over pMenuAdd's top level
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// if the top level search did not find a position append the menu
|
|
if( iInsertPosDefault == -1 )
|
|
iInsertPosDefault = GetMenuItemCount(pMenuDestination);
|
|
|
|
// create a new popup and insert before <Window> or <Help>
|
|
NewPopupMenu = CreatePopupMenu();
|
|
if( !NewPopupMenu)
|
|
{
|
|
// MergeMenu: CreatePopupMenu failed!
|
|
return 0;
|
|
}
|
|
|
|
// merge the new popup recursivly
|
|
if( !MergeMenu( NewPopupMenu, pSubMenu, 0 ))
|
|
return 0;
|
|
|
|
// insert the new popup menu into the destination menu
|
|
HMENU hNewMenu = NewPopupMenu;
|
|
{
|
|
MENUITEMINFOW menuItem={sizeof(MENUITEMINFO),
|
|
MIIM_TYPE|MIIM_SUBMENU,
|
|
MFT_STRING,
|
|
MFS_ENABLED,
|
|
0, //wID
|
|
hNewMenu, // hSubMenu
|
|
NULL, // hbmpChecked
|
|
NULL, // hbmpUnchecked
|
|
0, // dwItemData
|
|
sMenuAddString, // dwTypeData
|
|
0, // cch
|
|
};
|
|
|
|
if (InsertMenuItemW(pMenuDestination, iInsertPosDefault, TRUE, &menuItem))
|
|
{
|
|
// don't forget to correct the item count
|
|
iMenuDestItemCount++;
|
|
}
|
|
else
|
|
{
|
|
// MergeMenu: InsertMenu failed!
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int getMenuItemPos(HMENU menu, UINT command)
|
|
{
|
|
int i;
|
|
for (i = 0;i < 256;i++)
|
|
{
|
|
MENUITEMINFO mii = {sizeof(mii), MIIM_ID, };
|
|
if (!GetMenuItemInfo(menu, i, TRUE, &mii)) break;
|
|
if (mii.wID == command) return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
extern int g_SkinTop, g_BookmarkTop;
|
|
|
|
int V5_File_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
int flag = TPM_LEFTALIGN;
|
|
HMENU file_menu = GetSubMenu(v5_top_menu, 0);
|
|
HMENU hMenu = GetSubMenu(file_menu, 3);
|
|
MENUITEMINFOW i = {sizeof(i), };
|
|
FILE *fp;
|
|
int a = 34768;
|
|
int offs = 3;
|
|
int count = GetMenuItemCount(hMenu) + 1;
|
|
|
|
i.fMask = MIIM_TYPE | MIIM_DATA | MIIM_ID;
|
|
i.fType = MFT_STRING;
|
|
i.wID = 34768;
|
|
|
|
// remove all of the items we might have added - do by command for certainty
|
|
while (count){
|
|
if(!RemoveMenu(hMenu, a++, MF_BYCOMMAND)) break;
|
|
count--;
|
|
}
|
|
|
|
fp = _wfopen(BOOKMARKFILE8, L"rt");
|
|
if (fp)
|
|
{
|
|
while (1)
|
|
{
|
|
char ft[FILETITLE_SIZE] = {0}, fn[FILENAME_SIZE] = {0};
|
|
fgets(fn, FILENAME_SIZE, fp);
|
|
if (feof(fp)) break;
|
|
fgets(ft, FILETITLE_SIZE, fp);
|
|
if (feof(fp)) break;
|
|
if (ft[0] && fn[0])
|
|
{
|
|
if (fn[lstrlenA(fn) - 1] == '\n') fn[lstrlenA(fn) - 1] = 0;
|
|
if (ft[lstrlenA(ft) - 1] == '\n') ft[lstrlenA(ft) - 1] = 0;
|
|
if (ft[0] && fn[0])
|
|
{
|
|
i.dwTypeData = AutoWideDup(ft, CP_UTF8);
|
|
i.cch = lstrlenW(i.dwTypeData);
|
|
RemoveMenu(hMenu, i.wID, MF_BYCOMMAND);
|
|
InsertMenuItemW(hMenu, i.wID + offs - 34768, TRUE, &i);
|
|
i.wID++;
|
|
}
|
|
}
|
|
}
|
|
fclose(fp);
|
|
}
|
|
g_BookmarkTop = i.wID;
|
|
|
|
// put in a place holder item if there were no read bookmarks
|
|
if (g_BookmarkTop == 34768)
|
|
{
|
|
i.dwTypeData = getStringW(IDS_NO_BOOKMARKS,NULL,0);
|
|
i.cch = lstrlenW(i.dwTypeData);
|
|
InsertMenuItemW(hMenu, i.wID + offs - 34768, TRUE, &i);
|
|
EnableMenuItem(hMenu, i.wID, MF_BYCOMMAND | MF_GRAYED);
|
|
}
|
|
|
|
ensureInScreen(file_menu, &x, &y, &flag, width, height);
|
|
|
|
DoTrackPopup(file_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_Play_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
HMENU play_menu = GetSubMenu(v5_top_menu, 1);
|
|
int flag = TPM_LEFTALIGN;
|
|
ensureInScreen(play_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(play_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
|
|
int V5_Options_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
int flag = TPM_LEFTALIGN;
|
|
HMENU options_menu = GetSubMenu(v5_top_menu, 2);
|
|
HMENU eqMenu = NULL;
|
|
|
|
{ // set options skin menu to the skin menu
|
|
extern HMENU g_submenus_skins1;
|
|
MENUITEMINFO mi = {sizeof(mi), MIIM_SUBMENU};
|
|
mi.hSubMenu = g_submenus_skins1;
|
|
SetMenuItemInfoW(options_menu, 0, TRUE, &mi);
|
|
}
|
|
|
|
eqMenu = GetSubMenu(options_menu, 2);
|
|
CheckMenuItem(eqMenu, EQ_ENABLE, config_use_eq ? MF_CHECKED : MF_UNCHECKED);
|
|
CheckMenuItem(options_menu, WINAMP_OPTIONS_DSIZE, config_dsize ? MF_CHECKED : MF_UNCHECKED);
|
|
CheckMenuItem(options_menu, WINAMP_OPTIONS_AOT, config_aot ? MF_CHECKED : MF_UNCHECKED);
|
|
CheckMenuItem(options_menu, WINAMP_OPTIONS_ELAPSED, config_timeleftmode ? MF_UNCHECKED : MF_CHECKED);
|
|
CheckMenuItem(options_menu, WINAMP_OPTIONS_REMAINING, config_timeleftmode ? MF_CHECKED : MF_UNCHECKED);
|
|
CheckMenuItem(options_menu, WINAMP_OPTIONS_PREFS, IsWindow(prefs_hwnd) ? MF_CHECKED : MF_UNCHECKED);
|
|
ensureInScreen(options_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(options_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_Windows_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
HMENU windows_menu = GetSubMenu(v5_top_menu, 3);
|
|
int flag = TPM_LEFTALIGN;
|
|
CheckMenuItem(windows_menu, WINAMP_OPTIONS_PLEDIT, config_pe_open ? MF_CHECKED : MF_UNCHECKED);
|
|
CheckMenuItem(windows_menu, WINAMP_VISPLUGIN, vis_running() ? MF_CHECKED : MF_UNCHECKED);
|
|
if (g_has_video_plugin) CheckMenuItem(windows_menu, WINAMP_OPTIONS_VIDEO, config_video_open ? MF_CHECKED : MF_UNCHECKED);
|
|
ensureInScreen(windows_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(windows_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int Help_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
HMENU help_menu = GetSubMenu(v5_top_menu, 4);
|
|
int flag = TPM_LEFTALIGN;
|
|
|
|
MENUITEMINFOW i = {sizeof(i), };
|
|
i.fMask = MIIM_TYPE;
|
|
i.fType = MFT_STRING;
|
|
i.dwTypeData = getStringW(IDS_WINAMP_MENUITEM, NULL, 0);
|
|
i.cch = (UINT)wcslen(i.dwTypeData);
|
|
SetMenuItemInfoW(help_menu, WINAMP_HELP_ABOUT, FALSE, &i);
|
|
|
|
ensureInScreen(help_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(help_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_Help_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
return Help_Menu(hwnd, x, y, width, height);
|
|
}
|
|
|
|
LRESULT sendMlIpc(int msg, WPARAM param);
|
|
|
|
// TODO:: need to make this only show what's needed at the time ie it'll still show even if there's no ml_playlists
|
|
// and properly show if there's no playlists
|
|
int V5_PE_File_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
int flag = TPM_LEFTALIGN;
|
|
HMENU pefile_menu = GetSubMenu(v5_top_menu, 5);
|
|
HMENU playlistsmenu = NULL;
|
|
HMENU viewmenu = NULL;
|
|
HWND mlwnd = (HWND)sendMlIpc(0, 0);
|
|
int viewmenu_added = 0;
|
|
g_open_ml_item_in_pe = 1;
|
|
if (mlwnd)
|
|
{
|
|
mlGetTreeStruct mgts = { TREE_PLAYLISTS, 45000, -1 };
|
|
playlistsmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (WPARAM) & mgts, ML_IPC_GETTREE);
|
|
if (playlistsmenu)
|
|
{
|
|
mlGetTreeStruct mgts = { TREE_QUERIES, 45000, -1 };
|
|
viewmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (WPARAM) & mgts, ML_IPC_GETTREE);
|
|
if (GetMenuItemCount(playlistsmenu) == 0) InsertMenuW(playlistsmenu, 0, MF_BYPOSITION | MF_STRING | MF_GRAYED, 0, getStringW(IDS_ML_NO_PLAYLISTS,NULL,0));
|
|
InsertMenuW(pefile_menu, 2, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT_PTR)playlistsmenu, getStringW(IDS_ML_OPEN_PLAYLIST,NULL,0));
|
|
if (viewmenu && GetMenuItemCount(viewmenu) > 0)
|
|
{
|
|
viewmenu_added = 1;
|
|
InsertMenuW(pefile_menu, 3, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT_PTR)viewmenu, getStringW(IDS_ML_OPEN_VIEW_RESULTS,NULL,0));
|
|
}
|
|
}
|
|
}
|
|
|
|
ModifyMenuW(pefile_menu, ID_PE_CLOSE, MF_BYCOMMAND | MF_STRING, ID_PE_CLOSE, config_pe_open ? getStringW(IDS_PE_CLOSE,NULL,0) : getStringW(IDS_PE_OPEN,NULL,0));
|
|
ensureInScreen(pefile_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(pefile_menu, flag, x, y, hwnd);
|
|
|
|
if (playlistsmenu)
|
|
{
|
|
if (viewmenu_added) RemoveMenu(pefile_menu, 3, MF_BYPOSITION);
|
|
if (viewmenu) DestroyMenu(viewmenu);
|
|
RemoveMenu(pefile_menu, 2, MF_BYPOSITION);
|
|
DestroyMenu(playlistsmenu);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int V5_PE_Playlist_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
HMENU peplaylist_menu = GetSubMenu(v5_top_menu, 6);
|
|
int flag = TPM_LEFTALIGN;
|
|
ensureInScreen(peplaylist_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(peplaylist_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_PE_Sort_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
HMENU pesort_menu = GetSubMenu(v5_top_menu, 7);
|
|
int flag = TPM_LEFTALIGN;
|
|
ensureInScreen(pesort_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(pesort_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_PE_Help_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
return Help_Menu(hwnd, x, y, width, height);
|
|
}
|
|
|
|
int V5_ML_File_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
int flag = TPM_LEFTALIGN;
|
|
HMENU mlfile_menu = GetSubMenu(v5_top_menu, 8);
|
|
HWND mlwnd = (HWND)sendMlIpc(0, 0);
|
|
HWND mlplwnd = (HWND)SendMessageW(mlwnd, WM_ML_IPC, 0, ML_IPC_GETPLAYLISTWND);
|
|
ModifyMenuW(mlfile_menu, 3, MF_BYPOSITION | (mlplwnd == NULL ? MF_GRAYED : 0), ID_MLFILE_SAVEPLAYLIST, getStringW(IDS_ML_EXPORT_PLAYLIST,NULL,0));
|
|
{
|
|
int visible = IsWindowVisible(mlwnd);
|
|
int p = getMenuItemPos(mlfile_menu, ID_FILE_CLOSELIBRARY);
|
|
if (p == -1) p = getMenuItemPos(mlfile_menu, ID_FILE_SHOWLIBRARY);
|
|
ModifyMenuW(mlfile_menu, p, MF_BYPOSITION | MF_STRING, visible ? ID_FILE_CLOSELIBRARY : ID_FILE_SHOWLIBRARY, visible ? getStringW(IDS_ML_CLOSE_ML,NULL,0) : getStringW(IDS_ML_OPEN_ML,NULL,0));
|
|
}
|
|
ensureInScreen(mlfile_menu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(mlfile_menu, flag, x, y, hwnd);
|
|
return 1;
|
|
}
|
|
|
|
int V5_ML_View_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
int flag = TPM_LEFTALIGN;
|
|
HMENU mlview_menu = GetSubMenu(v5_top_menu, 9), mediamenu = NULL;
|
|
HWND mlwnd = (HWND)sendMlIpc(0, 0);
|
|
if (mlwnd)
|
|
{
|
|
mlGetTreeStruct mgts = { 0, 45000, -1 };
|
|
mediamenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (WPARAM) & mgts, ML_IPC_GETTREE);
|
|
if (mediamenu)
|
|
{
|
|
MergeMenu(mediamenu, mlview_menu, 0);
|
|
//InsertMenu(mediamenu, 0, MF_BYPOSITION | MF_STRING, ID_MLVIEW_MEDIA, "All &Media");
|
|
//InsertMenu(mlview_menu, 1, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT)mediamenu, "&Local Media");
|
|
}/*
|
|
mgts.item_start = TREE_PLAYLISTS;
|
|
playlistsmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (int) & mgts, ML_IPC_GETTREE);
|
|
if (playlistsmenu)
|
|
{
|
|
if (GetMenuItemCount(playlistsmenu) == 0) InsertMenu(playlistsmenu, 0, MF_BYPOSITION | MF_STRING | MF_GRAYED, 0, "No playlists");
|
|
InsertMenu(mlview_menu, 2, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT)playlistsmenu, "&Playlists");
|
|
}
|
|
mgts.item_start = TREE_DEVICES;
|
|
devicesmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (int) & mgts, ML_IPC_GETTREE);
|
|
if (devicesmenu)
|
|
{
|
|
if (GetMenuItemCount(devicesmenu) == 0) InsertMenu(devicesmenu, 0, MF_BYPOSITION | MF_STRING | MF_GRAYED, 0, "No devices");
|
|
InsertMenu(mlview_menu, 3, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT)devicesmenu, "&Devices");
|
|
}*/
|
|
}
|
|
|
|
g_open_ml_item_in_pe = 0;
|
|
|
|
//ID_MLVIEW_PLAYLISTS
|
|
//ID_MLVIEW_DEVICES
|
|
|
|
ensureInScreen(mediamenu, &x, &y, &flag, width, height);
|
|
DoTrackPopup(mediamenu, flag, x, y, hwnd);
|
|
/*
|
|
if (devicesmenu)
|
|
{
|
|
RemoveMenu(mlview_menu, 3, MF_BYPOSITION);
|
|
DestroyMenu(devicesmenu);
|
|
}
|
|
if (playlistsmenu)
|
|
{
|
|
RemoveMenu(mlview_menu, 2, MF_BYPOSITION);
|
|
DestroyMenu(playlistsmenu);
|
|
}*/
|
|
if (mediamenu)
|
|
{
|
|
//RemoveMenu(mlview_menu, 1, MF_BYPOSITION);
|
|
DestroyMenu(mediamenu);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int V5_ML_Help_Menu(HWND hwnd, int x, int y, int width, int height)
|
|
{
|
|
return Help_Menu(hwnd, x, y, width, height);
|
|
}
|
|
|
|
int V5_PE_ListOfPlaylists_Menu(int x, int y)
|
|
{
|
|
HMENU viewmenu = NULL;
|
|
mlGetTreeStruct mgts = { TREE_PLAYLISTS, 55000, -1 };
|
|
HWND mlwnd = (HWND)sendMlIpc(0, 0);
|
|
HMENU playlistsmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (WPARAM) & mgts, ML_IPC_GETTREE);
|
|
if (playlistsmenu)
|
|
{
|
|
InsertMenuW(playlistsmenu, 0, MF_BYPOSITION | MF_STRING, ID_MANAGEPLAYLISTS, getStringW(IDS_ML_MANAGE_PLAYLISTS,NULL,0));
|
|
{
|
|
mlGetTreeStruct mgts = { TREE_QUERIES, 55000, -1 };
|
|
viewmenu = (HMENU)SendMessageW(mlwnd, WM_ML_IPC, (WPARAM) &mgts, ML_IPC_GETTREE);
|
|
}
|
|
if (viewmenu && GetMenuItemCount(viewmenu) > 0)
|
|
InsertMenuW(playlistsmenu, 1, MF_BYPOSITION | MF_ENABLED | MF_POPUP, (UINT_PTR)viewmenu, getStringW(IDS_ML_SMART_VIEW_RESULTS,NULL,0));
|
|
InsertMenu(playlistsmenu, 1, MF_BYPOSITION | MF_SEPARATOR, 0, NULL);
|
|
g_open_ml_item_in_pe = 1;
|
|
DoTrackPopup(playlistsmenu, TPM_LEFTALIGN, x, y, hMainWindow);
|
|
}
|
|
DestroyMenu(playlistsmenu);
|
|
DestroyMenu(viewmenu);
|
|
return 1;
|
|
}
|
|
|
|
void getViewportFromPoint(POINT *pt, RECT *r)
|
|
{
|
|
if (!r || !pt) return ;
|
|
{
|
|
|
|
HMONITOR hm;
|
|
hm = MonitorFromPoint(*pt, MONITOR_DEFAULTTONULL);
|
|
if (hm)
|
|
{
|
|
MONITORINFOEXW mi;
|
|
memset(&mi, 0, sizeof(mi));
|
|
mi.cbSize = sizeof(mi);
|
|
|
|
if (GetMonitorInfo(hm, &mi))
|
|
{
|
|
*r = mi.rcMonitor;
|
|
return ;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#undef GetSystemMetrics
|
|
|
|
void ensureInScreen(HMENU menu, int *x, int *y, int *flag, int width, int height)
|
|
{
|
|
POINT pt = {*x, *y};
|
|
int nitems = GetMenuItemCount(menu);
|
|
int i;
|
|
RECT mwr;
|
|
RECT monitor;
|
|
int rightdone = 0;
|
|
int bottomdone = 0;
|
|
int xedge = GetSystemMetrics(SM_CXEDGE);
|
|
int yedge = GetSystemMetrics(SM_CYEDGE);
|
|
int itemheight = GetSystemMetrics(SM_CYMENU);
|
|
int checkmarkwidth = GetSystemMetrics(SM_CXMENUCHECK);
|
|
int cury = *y + yedge + 1;
|
|
GetWindowRect(hMainWindow, &mwr);
|
|
getViewportFromPoint(&pt, &monitor);
|
|
/*if (nitems*GetSystemMetrics(SM_CYMENU)+yedge*2+*y > monitor.bottom) {
|
|
bottomdone = 1;
|
|
*y -= height;
|
|
*flag &= ~TPM_TOPALIGN;
|
|
*flag |= TPM_BOTTOMALIGN;
|
|
}*/
|
|
for (i = 0;i < nitems;i++)
|
|
{
|
|
SIZE s={0};
|
|
RECT item;
|
|
MENUITEMINFOW info = {sizeof(info), MIIM_DATA | MIIM_TYPE | MIIM_STATE | MIIM_ID, MFT_STRING, };
|
|
GetMenuItemRect(hMainWindow, menu, i, &item);
|
|
item.left -= mwr.left;
|
|
item.top -= mwr.top;
|
|
item.right -= mwr.left;
|
|
item.bottom -= mwr.top;
|
|
if (item.top == 0 && item.left == 0)
|
|
{
|
|
// item has never been shown so MS wont give us the rect, I HATE THEM ! I HATE THEM SO MUCH ARRRG !
|
|
|
|
// y
|
|
item.top = cury;
|
|
{
|
|
GetMenuItemInfoW(menu, i, TRUE, &info);
|
|
if (info.fType & MFT_SEPARATOR)
|
|
cury += (itemheight - 1) >> 1;
|
|
else
|
|
{
|
|
cury += itemheight - 2;
|
|
//info.dwTypeData = (LPTSTR) MALLOC(++info.cch + 1);
|
|
GetMenuItemInfoW(menu, i, TRUE, &info);
|
|
//info.dwTypeData[info.cch] = 0;
|
|
}
|
|
}
|
|
item.bottom = cury;
|
|
|
|
// x
|
|
if (info.dwTypeData)
|
|
{
|
|
LOGFONT m_lf;
|
|
HFONT font = nullptr;
|
|
HDC dc = nullptr;
|
|
NONCLIENTMETRICS nm = {sizeof (NONCLIENTMETRICS), };
|
|
memset((PVOID) &m_lf, 0, sizeof (LOGFONT));
|
|
SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, nm.cbSize, &nm, 0);
|
|
m_lf = nm.lfMenuFont;
|
|
font = CreateFontIndirectW(&m_lf);
|
|
dc = GetDC(hMainWindow);
|
|
GetTextExtentPoint32(dc, info.dwTypeData, lstrlen(info.dwTypeData), &s);
|
|
ReleaseDC(hMainWindow, dc);
|
|
DeleteObject(font);
|
|
//free(info.dwTypeData);
|
|
}
|
|
if (!(info.fType & MFT_SEPARATOR))
|
|
{
|
|
item.left = *x + xedge + 1;
|
|
item.right = item.left + s.cx + checkmarkwidth * 2 + xedge + 1;
|
|
}
|
|
else
|
|
{
|
|
item.left = *x + xedge + 1;
|
|
item.right = item.left + xedge + 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
item.left += *x;
|
|
item.top += *y;
|
|
item.right += *x;
|
|
item.bottom += *y;
|
|
item.right += xedge * 2 + 2; // to avoid last test
|
|
cury = item.bottom;
|
|
}
|
|
if (!bottomdone && cury > monitor.bottom)
|
|
{
|
|
bottomdone = 1;
|
|
*y -= height;
|
|
*flag &= ~TPM_TOPALIGN;
|
|
*flag |= TPM_BOTTOMALIGN;
|
|
}
|
|
if (!rightdone && item.right > monitor.right)
|
|
{
|
|
rightdone = 1;
|
|
*x += width;
|
|
*flag &= ~TPM_LEFTALIGN;
|
|
*flag |= TPM_RIGHTALIGN;
|
|
}
|
|
if (rightdone && bottomdone) return ;
|
|
}
|
|
cury += yedge + 1;
|
|
if (!bottomdone && cury > monitor.bottom)
|
|
{
|
|
//bottomdone = 1;
|
|
*y -= height;
|
|
*flag &= ~TPM_TOPALIGN;
|
|
*flag |= TPM_BOTTOMALIGN;
|
|
}
|
|
}
|
|
|
|
|