636 lines
20 KiB
C++
636 lines
20 KiB
C++
|
#include "playlist.h"
|
||
|
#include <shlwapi.h>
|
||
|
#include "main.h"
|
||
|
#include "api__ml_plg.h"
|
||
|
#include "../nu/MediaLibraryInterface.h"
|
||
|
#include "impl_playlist.h"
|
||
|
//#import "../gracenote/CDDBControlWinamp.dll" no_namespace, named_guids, raw_interfaces_only
|
||
|
#include "../gracenote/cddbcontrolwinamp.tlh"
|
||
|
#include "../winamp/ipc_pe.h"
|
||
|
#include "resource.h"
|
||
|
|
||
|
#include <strsafe.h>
|
||
|
|
||
|
//extern Playlist currentPlaylist;
|
||
|
ICddbPlaylist25Mgr *playlistMgr;
|
||
|
ICddbMLDBManager *mldbMgr;
|
||
|
|
||
|
Playlist currentPlaylist;
|
||
|
Playlist seedPlaylist;
|
||
|
|
||
|
class PlaylistEventHandler : public DPlaylist2Events
|
||
|
{
|
||
|
STDMETHODIMP STDMETHODCALLTYPE QueryInterface(REFIID riid, PVOID *ppvObject)
|
||
|
{
|
||
|
if (!ppvObject)
|
||
|
return E_POINTER;
|
||
|
|
||
|
else if (IsEqualIID(riid, __uuidof(DPlaylist2Events)))
|
||
|
*ppvObject = (DPlaylist2Events *)this;
|
||
|
else if (IsEqualIID(riid, IID_IDispatch))
|
||
|
*ppvObject = (IDispatch *)this;
|
||
|
else if (IsEqualIID(riid, IID_IUnknown))
|
||
|
*ppvObject = this;
|
||
|
else
|
||
|
{
|
||
|
*ppvObject = NULL;
|
||
|
return E_NOINTERFACE;
|
||
|
}
|
||
|
|
||
|
AddRef();
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
ULONG STDMETHODCALLTYPE AddRef(void)
|
||
|
{
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
ULONG STDMETHODCALLTYPE Release(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE Invoke(DISPID dispid, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS FAR *pdispparams, VARIANT FAR *pvarResult, EXCEPINFO FAR * pexecinfo, unsigned int FAR *puArgErr)
|
||
|
{
|
||
|
switch (dispid)
|
||
|
{
|
||
|
case 1:
|
||
|
break;
|
||
|
case 2:
|
||
|
break;
|
||
|
case 3:
|
||
|
break;
|
||
|
case 4:
|
||
|
break;
|
||
|
case 5:
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
return DISP_E_MEMBERNOTFOUND;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE GetIDsOfNames(REFIID riid, OLECHAR FAR* FAR* rgszNames, unsigned int cNames, LCID lcid, DISPID FAR* rgdispid)
|
||
|
{
|
||
|
*rgdispid = DISPID_UNKNOWN;
|
||
|
return DISP_E_UNKNOWNNAME;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE GetTypeInfo(unsigned int itinfo, LCID lcid, ITypeInfo FAR* FAR* pptinfo)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
HRESULT STDMETHODCALLTYPE GetTypeInfoCount(unsigned int FAR * pctinfo)
|
||
|
{
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
};
|
||
|
|
||
|
static IConnectionPoint *GetConnectionPoint(IUnknown *punk, REFIID riid)
|
||
|
{
|
||
|
if (!punk)
|
||
|
return 0;
|
||
|
|
||
|
IConnectionPointContainer *pcpc;
|
||
|
IConnectionPoint *pcp = 0;
|
||
|
|
||
|
HRESULT hr = punk->QueryInterface(IID_IConnectionPointContainer, (void **) & pcpc);
|
||
|
if (SUCCEEDED(hr))
|
||
|
{
|
||
|
pcpc->FindConnectionPoint(riid, &pcp);
|
||
|
pcpc->Release();
|
||
|
}
|
||
|
return pcp;
|
||
|
}
|
||
|
|
||
|
|
||
|
static PlaylistEventHandler events;
|
||
|
static DWORD m_dwCookie = 0;
|
||
|
//static DWORD m_dwCookie_MldbMgr = 0;
|
||
|
bool SetupPlaylistSDK()
|
||
|
{
|
||
|
if (!playlistMgr)
|
||
|
{
|
||
|
//playlistMgr = AGAVE_API_GRACENOTE->GetPlaylistManager();
|
||
|
//AGAVE_API_GRACENOTE->GetPlaylistManagerWithMLDBManager(&playlistMgr, &mldbMgr);
|
||
|
AGAVE_API_GRACENOTE->GetPlaylistManager(&playlistMgr, &mldbMgr);
|
||
|
|
||
|
IConnectionPoint *icp = GetConnectionPoint(playlistMgr, DIID_DPlaylist2Events);
|
||
|
if (icp)
|
||
|
{
|
||
|
icp->Advise(static_cast<IDispatch *>(&events), &m_dwCookie);
|
||
|
icp->Release();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return !!playlistMgr;
|
||
|
}
|
||
|
|
||
|
void ShutdownPlaylistSDK()
|
||
|
{
|
||
|
if (playlistMgr)
|
||
|
{
|
||
|
if (mldbMgr)
|
||
|
{
|
||
|
mldbMgr->Detach(playlistMgr); // Detach the mldb manager from the playlist manager if it is not null
|
||
|
}
|
||
|
|
||
|
IConnectionPoint *icp = GetConnectionPoint(playlistMgr, DIID_DPlaylist2Events);
|
||
|
if (icp)
|
||
|
{
|
||
|
icp->Unadvise(m_dwCookie);
|
||
|
icp->Release();
|
||
|
}
|
||
|
|
||
|
if (mldbMgr)
|
||
|
mldbMgr->Release(); // Release the mldb manager if its not null
|
||
|
|
||
|
playlistMgr->Shutdown();
|
||
|
playlistMgr->Release();
|
||
|
}
|
||
|
mldbMgr=0;
|
||
|
playlistMgr=0;
|
||
|
}
|
||
|
|
||
|
// Caller must cleanup the BSTR
|
||
|
BSTR SetAndCreatePath(/*wchar_t *path_to_create,*/ const wchar_t *node)
|
||
|
{
|
||
|
wchar_t path_to_create[MAX_PATH] = {0};
|
||
|
|
||
|
BSTR bPath = 0;
|
||
|
|
||
|
PathCombineW(path_to_create, WASABI_API_APP->path_getUserSettingsPath(), L"Plugins");
|
||
|
CreateDirectoryW(path_to_create, 0);
|
||
|
PathAppendW(path_to_create, node);
|
||
|
CreateDirectoryW(path_to_create, 0);
|
||
|
|
||
|
bPath = SysAllocString(path_to_create);
|
||
|
return bPath;
|
||
|
// modified path as return value
|
||
|
}
|
||
|
|
||
|
// IMPORTANT: Make sure to call this on the gracenote dedicated thread.
|
||
|
int RestoreGracenoteMLDB(void)
|
||
|
{
|
||
|
long restoreFlags = PL_MLDB_RESTORE_BASE | PL_MLDB_RESTORE_INDEX;
|
||
|
//wchar_t backupPath[MAX_PATH] = {0};
|
||
|
BSTR bDataPath = SetAndCreatePath(GRACENOTE_DB_BASE_PATH);
|
||
|
BSTR bBackupPath = SetAndCreatePath(GRACENOTE_DB_BACKUP_PATH);
|
||
|
|
||
|
// Restore the db files.
|
||
|
mldbMgr->RestoreDBFiles(restoreFlags, bDataPath, bBackupPath);
|
||
|
|
||
|
SysFreeString(bDataPath);
|
||
|
SysFreeString(bBackupPath);
|
||
|
return NErr_Success;
|
||
|
}
|
||
|
|
||
|
// Backs up the gracenote MLDB so that it can be restored on corruption
|
||
|
// IMPORTANT: Make sure to call this on the gracenote dedicated thread.
|
||
|
int BackupGracenoteMLDB(void)
|
||
|
{
|
||
|
long backupFlags = PL_MLDB_BACKUP_BASE | PL_MLDB_BACKUP_INDEX;
|
||
|
//wchar_t backupPath[MAX_PATH] = {0};
|
||
|
BSTR bDataPath = SetAndCreatePath(GRACENOTE_DB_BASE_PATH);
|
||
|
BSTR bBackupPath = SetAndCreatePath(GRACENOTE_DB_BACKUP_PATH);
|
||
|
|
||
|
// Backup the db files.
|
||
|
mldbMgr->BackupDBFiles(backupFlags, bDataPath, bBackupPath);
|
||
|
|
||
|
SysFreeString(bDataPath);
|
||
|
SysFreeString(bBackupPath);
|
||
|
return NErr_Success;
|
||
|
}
|
||
|
|
||
|
/*BOOL DeleteGracenoteFile(char *filename)
|
||
|
{
|
||
|
BOOL result;
|
||
|
char path[MAX_PATH] = {0};
|
||
|
//PathCombineA(path,mediaLibrary.GetIniDirectory(),"Plugins\\Gracenote");
|
||
|
PathCombineA(path,"C:\\Users\\bigg\\AppData\\Roaming\\Winamp\\","Plugins\\Gracenote");
|
||
|
PathAppendA(path, filename);
|
||
|
result = DeleteFileA(path);
|
||
|
return result;
|
||
|
}*/
|
||
|
|
||
|
void CheckForResetError(HRESULT error)
|
||
|
{
|
||
|
if (error != S_OK)
|
||
|
{
|
||
|
MessageBoxW(0, WASABI_API_LNGSTRINGW(IDS_ERROR_RESET), (LPWSTR)plugin.description, MB_OK| MB_ICONERROR);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void CheckForShutdownError(HRESULT error)
|
||
|
{
|
||
|
if (error != S_OK)
|
||
|
{
|
||
|
MessageBoxW(plugin.hwndWinampParent, WASABI_API_LNGSTRINGW(IDS_CANNOT_SHUT_DOWN), (LPWSTR)plugin.description, MB_OK| MB_ICONERROR);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Deprecated: Currently not used, remove at some point
|
||
|
/*
|
||
|
INT_PTR CALLBACK ResettingProcedure(HWND hwndDlg, UINT msg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
switch (msg)
|
||
|
{
|
||
|
case WM_INITDIALOG:
|
||
|
//ShowWindow(hwndDlg,SW_HIDE);
|
||
|
ShowWindow(hwndDlg, SW_SHOW);
|
||
|
return 0;
|
||
|
break;
|
||
|
case WM_QUIT:
|
||
|
EndDialog(hwndDlg,0);
|
||
|
return 0;
|
||
|
break;
|
||
|
}
|
||
|
return 0;
|
||
|
}*/
|
||
|
|
||
|
// IMPORTANT: Make sure to call this on the gracenote dedicated thread.
|
||
|
int DeleteGracenoteMLDB(bool silent)
|
||
|
{
|
||
|
long deleteFlags = PL_MLDB_DELETE_INDEX | PL_MLDB_DELETE_BASE | PL_MLDB_DELETE_OTHER | PL_MLDB_DELETE_BACKUPS;
|
||
|
//long deleteFlags = PL_MLDB_DELETE_INDEX | PL_MLDB_DELETE_BASE | PL_MLDB_DELETE_OTHER; // | PL_MLDB_DELETE_BACKUPS;
|
||
|
BSTR bDataPath = SetAndCreatePath(GRACENOTE_DB_BASE_PATH);
|
||
|
|
||
|
HRESULT error = 0;
|
||
|
|
||
|
if (playlistMgr)
|
||
|
error = playlistMgr->Shutdown();
|
||
|
|
||
|
if (!silent)
|
||
|
CheckForShutdownError(error);
|
||
|
|
||
|
// Spawn the working window
|
||
|
//HWND hwndResetWorking = WASABI_API_CREATEDIALOG(IDD_NAG, plugin.hwndWinampParent, ResettingProcedure);
|
||
|
if (mldbMgr)
|
||
|
error = mldbMgr->DeleteDBFiles(deleteFlags, bDataPath);
|
||
|
//SendMessage(hwndResetWorking, WM_QUIT, 0, 0);
|
||
|
|
||
|
if (!silent)
|
||
|
CheckForResetError(error);
|
||
|
|
||
|
SysFreeString(bDataPath);
|
||
|
return NErr_Success; // NOT the HRESULT so that non zero values return as false
|
||
|
}
|
||
|
|
||
|
int InitializeMLDBManager(void)
|
||
|
{
|
||
|
// Other initializations for the MLDB manager can go here
|
||
|
///BackupGracenoteMLDB();
|
||
|
|
||
|
return NErr_Success;
|
||
|
}
|
||
|
|
||
|
static void ConfigureGeneratorPrefs(ICddbPLMoreLikeThisCfg *config)
|
||
|
{
|
||
|
// ToDo: (BigG) Consider using some of the different algorithms
|
||
|
// GNPL_MORELIKETHIS_ALG_20 (Playlist SDK 2.0)
|
||
|
// GNPL_MORELIKETHIS_ALG_25 (Playlist SDK 2.5)
|
||
|
// GNPL_MORELIKETHIS_ALG_DSP_1 (Playlist SDK 2.6)
|
||
|
// GNPL_MORELIKETHIS_ALG_DSP_25 (Playlist SDK 2.6)
|
||
|
|
||
|
config->put_Algorithm(GNPL_MORELIKETHIS_ALG_DEFAULT);
|
||
|
if(!multipleAlbums) config->put_MaxPerAlbum(1);
|
||
|
if(!multipleArtists) config->put_MaxPerArtist(1);
|
||
|
//config->put_TrackLimit(plLength);
|
||
|
config->put_TrackLimit(0); // Dont put a limit on gracenote (return all tracks)
|
||
|
}
|
||
|
|
||
|
//void playPlaylist(Playlist &pl, bool enqueue=false, int startplaybackat=0/*-1 for don't start playback*/, const wchar_t *seedfn=NULL, int useSeed=FALSE)
|
||
|
void playPlaylist(Playlist &pl, bool enqueue=false, int startplaybackat=0/*-1 for don't start playback*/, int useSeed=FALSE)
|
||
|
{
|
||
|
extern winampMediaLibraryPlugin plugin;
|
||
|
|
||
|
const size_t number_of_seeds = seedPlaylist.GetNumItems();
|
||
|
wchar_t seedFilename[MAX_PATH] = {0};
|
||
|
seedFilename[0] = 0;
|
||
|
|
||
|
if(!enqueue)
|
||
|
mediaLibrary.ClearPlaylist();
|
||
|
|
||
|
// Enqueue the seed tracks first
|
||
|
if(useSeed)
|
||
|
{
|
||
|
for (size_t i = 0; i < number_of_seeds; i++)
|
||
|
{
|
||
|
seedPlaylist.GetItem(i, seedFilename, MAX_PATH); // Get the playlist filename
|
||
|
|
||
|
enqueueFileWithMetaStructW s={seedFilename,NULL,PathFindExtensionW(seedFilename),-1};
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&s, IPC_PLAYFILEW);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
size_t listLength = pl.GetNumItems();
|
||
|
for(size_t i=0; i<listLength; i++)
|
||
|
{
|
||
|
wchar_t filename[MAX_PATH] = {0};
|
||
|
pl.GetItem(i,filename,MAX_PATH);
|
||
|
|
||
|
//if(seedfn && !_wcsicmp(seedfn,filename)) // Not really sure this is necessary... just making sure that the same file doesnt get added twice
|
||
|
// continue;
|
||
|
enqueueFileWithMetaStructW s={filename,NULL,PathFindExtensionW(filename),-1};
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&s, IPC_PLAYFILEW);
|
||
|
}
|
||
|
|
||
|
if(!enqueue && startplaybackat != -1)
|
||
|
{ //play item startplaybackat
|
||
|
SendMessage(plugin.hwndWinampParent,WM_WA_IPC,startplaybackat,IPC_SETPLAYLISTPOS);
|
||
|
SendMessage(plugin.hwndWinampParent,WM_COMMAND,40047,0); //stop
|
||
|
SendMessage(plugin.hwndWinampParent,WM_COMMAND,40045,0); //play
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// Tests wether an item can be added to a playlist and still stay under the current target
|
||
|
bool PlaylistIsFull(Playlist *playlist, /*uint64_t*/ unsigned int currentItemLength, /*uint64_t*/ unsigned int currentItemSize)
|
||
|
{
|
||
|
#define POWER_2_TO_20 1048576 // 1024 * 1024
|
||
|
uint64_t measurement = 0;
|
||
|
|
||
|
// See what type we care about items, size, or length
|
||
|
switch (plLengthType)
|
||
|
{
|
||
|
case PL_ITEMS: // Check for number of items
|
||
|
{
|
||
|
measurement = currentPlaylist.GetNumItems() + ( (useSeed) ? seedPlaylist.GetNumItems() : 0 ); // Add the data from the seed track if we are using it
|
||
|
|
||
|
if ( (measurement + 1) > plItems ) // See if an extra item will put us over.
|
||
|
return true;
|
||
|
else
|
||
|
return false;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case PL_MINUTES: // Check for minutes used
|
||
|
{
|
||
|
measurement = currentPlaylist.GetPlaylistLengthMilliseconds() + ( (useSeed) ? seedPlaylist.GetPlaylistLengthMilliseconds() : 0 );
|
||
|
|
||
|
if ( (measurement + (currentItemLength) ) > (plMinutes * 60 * 1000) )
|
||
|
return true;
|
||
|
else
|
||
|
return false;
|
||
|
}
|
||
|
break;
|
||
|
case PL_MEGABYTES: // Check for megabytes used
|
||
|
{
|
||
|
measurement = currentPlaylist.GetPlaylistSizeBytes() + ( (useSeed) ? seedPlaylist.GetPlaylistSizeBytes() : 0 );
|
||
|
|
||
|
if ( (measurement + (uint64_t)currentItemSize) > ((uint64_t)plMegabytes * POWER_2_TO_20) )
|
||
|
return true;
|
||
|
else
|
||
|
return false;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool MatchesQuery(const wchar_t *filename, const wchar_t *user_query)
|
||
|
{
|
||
|
// Get an item that mathces both the filename and the query
|
||
|
itemRecordW *result = AGAVE_API_MLDB->GetFileIf(filename, user_query);
|
||
|
|
||
|
if (result)
|
||
|
{
|
||
|
AGAVE_API_MLDB->FreeRecord(result); // Clean up the records list since we are done using it
|
||
|
return true;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
AGAVE_API_MLDB->FreeRecord(result); // Clean up the records list since we are done using it
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
// Callback for getting a tag for gracenote library items
|
||
|
static wchar_t * TitleTagFuncGracenote(const wchar_t * tag, void * p)
|
||
|
{ //return 0 if not found, -1 for empty tag
|
||
|
//tagItem * s = (tagItem *)p;
|
||
|
//wchar_t *filename = (wchar_t *)p;
|
||
|
ICddbPL2Result *gracenoteResult = (ICddbPL2Result *)p;
|
||
|
|
||
|
BSTR tag_data;
|
||
|
|
||
|
if (!_wcsicmp(tag, L"artist")) gracenoteResult->GetArtist(&tag_data)/*wsprintf(buf,L"%s",L"artist")*/;
|
||
|
else if (!_wcsicmp(tag, L"album")) gracenoteResult->GetAlbum(&tag_data);
|
||
|
else if (!_wcsicmp(tag, L"title")) gracenoteResult->GetTitle(&tag_data);
|
||
|
else if (!_wcsicmp(tag, L"filename")) gracenoteResult->GetFilename(&tag_data);
|
||
|
else
|
||
|
return 0;
|
||
|
|
||
|
//else if (!_wcsicmp(tag, L"genre")) -;
|
||
|
//else if (!_wcsicmp(tag, L"year")) -;
|
||
|
//else if (!_wcsicmp(tag, L"tracknumber")) -;
|
||
|
//else if (!_wcsicmp(tag, L"discnumber")) -;
|
||
|
//else if (!_wcsicmp(tag, L"bitrate")) -;
|
||
|
|
||
|
//else if (!_wcsicmp(tag, L"albumartist")) -;
|
||
|
//else if (!_wcsicmp(tag, L"composer")) -;
|
||
|
//else if (!_wcsicmp(tag, L"publisher")) -;
|
||
|
|
||
|
return tag_data;
|
||
|
}
|
||
|
|
||
|
// Callback for getting a tag for media library items
|
||
|
static wchar_t * TitleTagFuncML(const wchar_t * tag, void * p)
|
||
|
{ //return 0 if not found, -1 for empty tag
|
||
|
itemRecordW *mlResult = (itemRecordW *)p;
|
||
|
|
||
|
if (!mlResult)
|
||
|
return 0; // Return 0 because we dont have this ml object
|
||
|
|
||
|
wchar_t buf[128] = {0};
|
||
|
|
||
|
wchar_t *tag_data = 0;
|
||
|
|
||
|
if (!_wcsicmp(tag, L"artist")) tag_data = mlResult->artist;
|
||
|
else if (!_wcsicmp(tag, L"album")) tag_data = mlResult->album;
|
||
|
else if (!_wcsicmp(tag, L"title")) tag_data = mlResult->title;
|
||
|
else if (!_wcsicmp(tag, L"filename")) tag_data = mlResult->filename;
|
||
|
else if (!_wcsicmp(tag, L"genre")) tag_data = mlResult->genre;
|
||
|
else if (!_wcsicmp(tag, L"year")) if (mlResult->year > 0) { StringCchPrintfW(buf, 128, L"%04d", mlResult->year); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"tracknumber") || !_wcsicmp(tag, L"track")) if (mlResult->track > 0) { StringCchPrintfW(buf, 128, L"%04d", mlResult->track); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"discnumber")) if (mlResult->disc > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->disc); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"bitrate")) if (mlResult->bitrate > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->bitrate); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"albumartist")) tag_data = mlResult->albumartist;
|
||
|
else if (!_wcsicmp(tag, L"composer")) tag_data = mlResult->composer;
|
||
|
else if (!_wcsicmp(tag, L"publisher")) tag_data = mlResult->publisher;
|
||
|
else if (!_wcsicmp(tag, L"bpm")) if (mlResult->bpm > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->bpm); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"comment")) tag_data = mlResult->comment;
|
||
|
else if (!_wcsicmp(tag, L"discs")) if (mlResult->discs > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->discs); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"filesize")) if (mlResult->filesize > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->filesize); tag_data = buf; }
|
||
|
//else if (!_wcsicmp(tag, L"filetime")) tag_data = mlResult->filetime;
|
||
|
else if (!_wcsicmp(tag, L"length")) if (mlResult->length > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->length); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"playcount")) if (mlResult->playcount > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->playcount); tag_data = buf; }
|
||
|
else if (!_wcsicmp(tag, L"rating")) if (mlResult->rating > 0) { StringCchPrintfW(buf, 128, L"%d", mlResult->rating); tag_data = buf; }
|
||
|
else
|
||
|
return 0;
|
||
|
|
||
|
return _wcsdup(tag_data);
|
||
|
}
|
||
|
|
||
|
// Callback to free a tag in gracenote library
|
||
|
static void TitleTagFreeFuncGracenote(wchar_t *tag_data, void *p)
|
||
|
{
|
||
|
if(tag_data)
|
||
|
SysFreeString(tag_data);
|
||
|
}
|
||
|
|
||
|
// Callback to free a tag in media library
|
||
|
static void TitleTagFreeFuncML(wchar_t *tag_data, void *p)
|
||
|
{
|
||
|
if(tag_data)
|
||
|
free(tag_data);
|
||
|
}
|
||
|
|
||
|
// Retreive the title formatting for gracenote library
|
||
|
void GetTitleFormattingGracenote(const wchar_t *filename, ICddbPL2Result * gracenoteResult, wchar_t * buf, int len)
|
||
|
{
|
||
|
waFormatTitleExtended fmt={ filename, 1, NULL, (void *)gracenoteResult, buf, len, TitleTagFuncGracenote, TitleTagFreeFuncGracenote };
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&fmt, IPC_FORMAT_TITLE_EXTENDED);
|
||
|
}
|
||
|
|
||
|
// Retreive the title formatting for media library
|
||
|
void GetTitleFormattingML(const wchar_t *filename, itemRecordW *mlResult, wchar_t * buf, int len)
|
||
|
{
|
||
|
waFormatTitleExtended fmt={ filename, 1, NULL, (void *)mlResult, buf, len, TitleTagFuncML, TitleTagFreeFuncML };
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&fmt, IPC_FORMAT_TITLE_EXTENDED);
|
||
|
}
|
||
|
|
||
|
static int MoreLikeTheseSongsFunc(HANDLE handle, void *user_data, intptr_t id)
|
||
|
{
|
||
|
Playlist *pl = (Playlist *)user_data;
|
||
|
const int number_of_files = (int)pl->GetNumItems();
|
||
|
|
||
|
isGenerating = true;
|
||
|
|
||
|
//Sleep(501); // Wait for the update timer to finish its cycles, This is EVIL, keep it removed...
|
||
|
SetMarqueeProgress(true); // Turn the marquee off because we are actually generating the tracks
|
||
|
SetDlgItemText(hwndDlgCurrent, IDC_STATIC_PROGRESS_STATE, WASABI_API_LNGSTRINGW(IDS_GENERATING));
|
||
|
//EnableWindow (GetDlgItem(hwndDlgCurrent, IDC_BUTTON_REGENERATE), FALSE );
|
||
|
SetButtonsEnabledState(false); // Disable the buttons once we start generating here
|
||
|
|
||
|
if (SetupPlaylistSDK())
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
|
||
|
ICddbPLMoreLikeThisCfgPtr cfg;
|
||
|
cfg.CreateInstance(CLSID_CddbPLMoreLikeThisCfg);
|
||
|
ConfigureGeneratorPrefs(cfg);
|
||
|
|
||
|
ICddbPL2ResultListPtr results;
|
||
|
wchar_t plFilename[MAX_PATH] = {0};
|
||
|
|
||
|
if (number_of_files == 1) // Call the regular morelikethis function if there is only the standard seed track
|
||
|
{
|
||
|
pl->GetItem(0, plFilename, MAX_PATH);
|
||
|
BSTR i_dunno = SysAllocString(plFilename);
|
||
|
hr=playlistMgr->MoreLikeThisSong(i_dunno, cfg, &results);
|
||
|
SysFreeString(i_dunno);
|
||
|
}
|
||
|
else if (number_of_files > 1) // We have more than 1 seed track
|
||
|
{
|
||
|
// Create the variant of an array of filenames for MoreLikeTheseSongs
|
||
|
VARIANT fileList;
|
||
|
VariantInit((VARIANTARG *)&fileList);
|
||
|
SAFEARRAY *psa = SafeArrayCreateVector (VT_BSTR, 0, number_of_files);
|
||
|
BSTR *data;
|
||
|
SafeArrayAccessData(psa, (void **)&data);
|
||
|
for (size_t i=0;i!=number_of_files;i++)
|
||
|
{
|
||
|
pl->GetItem(i, plFilename, MAX_PATH);
|
||
|
data[i] = SysAllocString(plFilename);
|
||
|
}
|
||
|
SafeArrayUnaccessData(psa);
|
||
|
V_VT(&fileList) = VT_ARRAY|VT_BSTR;
|
||
|
V_ARRAY(&fileList) = psa;
|
||
|
|
||
|
|
||
|
hr=playlistMgr->MoreLikeTheseSongs(fileList, cfg, &results);
|
||
|
}
|
||
|
else // We dont have any seed tracks (this should not happen because we shouldnt get this far.
|
||
|
{
|
||
|
return 1; // Failure
|
||
|
}
|
||
|
|
||
|
long count=-1;
|
||
|
if (results && SUCCEEDED(hr=results->get_Count(&count)) && count)
|
||
|
{
|
||
|
currentPlaylist.Clear();
|
||
|
for (long i=0;i<count;i++)
|
||
|
{
|
||
|
ICddbPL2Result *result;
|
||
|
if (SUCCEEDED(hr=results->GetResult(i+1, &result)))
|
||
|
{
|
||
|
BSTR filename = 0;
|
||
|
BSTR title = 0;
|
||
|
|
||
|
unsigned int length = -1;
|
||
|
unsigned int size = 0;
|
||
|
|
||
|
result->GetFilename(&filename);
|
||
|
result->GetTitle(&title);
|
||
|
result->GetTracklength(&length); // Gracenote is returning seconds here
|
||
|
result->GetFilesize(&size); // Gracenote took the size as kilobytes but it is returning it to us as bytes
|
||
|
|
||
|
length *= 1000; // Multiply length by 1000 to turn it into milliseconds from seconds
|
||
|
|
||
|
// Get the winamp user formatted title.
|
||
|
wchar_t winamp_title[512] = {0};
|
||
|
GetTitleFormattingGracenote(filename, result, winamp_title, 512);
|
||
|
|
||
|
// Only check for the query if the user wants to apply one
|
||
|
if ( useMLQuery == TRUE && MatchesQuery(filename, mlQuery ) == false )
|
||
|
{
|
||
|
SysFreeString(filename);
|
||
|
SysFreeString(title);
|
||
|
result->Release();
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Lets check for the playlist limit to see if we should add this track
|
||
|
if ( !PlaylistIsFull(¤tPlaylist, length, size) )
|
||
|
currentPlaylist.AppendWithInfo(filename, winamp_title, length, size);
|
||
|
// DONT break here if we are full, keep going as we may find something that will fit into the playlist eventually
|
||
|
|
||
|
SysFreeString(filename);
|
||
|
SysFreeString(title);
|
||
|
result->Release();
|
||
|
}
|
||
|
}
|
||
|
/*char cfg[1024 + 32] = {0};
|
||
|
char *dir = (char*)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_GETINIDIRECTORY);
|
||
|
PathCombineA(cfg, dir, "Plugins\\gen_ml.ini");
|
||
|
int en = GetPrivateProfileIntA("gen_ml_config","enqueuedef",0,cfg);*/
|
||
|
|
||
|
PopulateResults(¤tPlaylist);
|
||
|
}
|
||
|
else /*if (count == 0)*/
|
||
|
{
|
||
|
PopulateResults(0); // Call populate with an empty playlist
|
||
|
CantPopulateResults(); // Display warning about not being able to generate any tracks
|
||
|
}
|
||
|
}
|
||
|
|
||
|
isGenerating = false;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void MoreLikeTheseSongs(Playlist *pl)
|
||
|
{
|
||
|
// Capture the stats, we dont care if its successful or not, we only care about the try
|
||
|
if (AGAVE_API_STATS)
|
||
|
AGAVE_API_STATS->IncrementStat(api_stats::PLG_COUNT);
|
||
|
|
||
|
// Call the the mor like this fuction on the gracenote reserved thread
|
||
|
WASABI_API_THREADPOOL->RunFunction(plg_thread, MoreLikeTheseSongsFunc, pl, 0, api_threadpool::FLAG_REQUIRE_COM_STA);
|
||
|
}
|