| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283 | #include "main.h"#include "Metadata.h"#include "api__in_mp3.h"#include "../nu/AutoWide.h"#include "AlbumArt.h"#include "Stopper.h"#include <shlwapi.h>#include <strsafe.h>bool IsMyExtension(const wchar_t *filename){	// check if it's the current stream and is playing and is SHOUTcast2	if (PathIsURLW(filename))	{		if (g_playing_file)		{			EnterCriticalSection(&streamInfoLock);			if (g_playing_file &&				(g_playing_file->uvox_artwork.uvox_stream_artwork || g_playing_file->uvox_artwork.uvox_playing_artwork) &&				!lstrcmpW(lastfn, filename)) // check again now that we've acquired the lock			{				LeaveCriticalSection(&streamInfoLock);				return true;			}			LeaveCriticalSection(&streamInfoLock);		}	}	// otherwise handle as normal embedded 	else	{		const wchar_t *extension = PathFindExtension(filename);		if (extension && *extension)		{			AutoWide wideList(config_extlist); // TODO: build a copy of this at config load time so we don't have to run this every time			extension++;			wchar_t *b = wideList;			wchar_t *c = 0;			do			{				wchar_t d[20] = {0};				StringCchCopyW(d, 15, b);				if ((c = wcschr(b, L';')))				{					if ((c-b)<15)						d[c - b] = 0;				}				if (!lstrcmpiW(extension, d))					return true;				b = c + 1;			}			while (c);		}	}	return false;}bool ID3v2_AlbumArtProvider::IsMine(const wchar_t *filename){	return IsMyExtension(filename);}int ID3v2_AlbumArtProvider::ProviderType(){	return ALBUMARTPROVIDER_TYPE_EMBEDDED;}int ID3v2_AlbumArtProvider::GetAlbumArtData(const wchar_t *filename, const wchar_t *type, void **bits, size_t *len, wchar_t **mimeType){	if (g_playing_file)	{		EnterCriticalSection(&streamInfoLock);		if (g_playing_file && !lstrcmpW(lastfn, filename)) // check again now that we've acquired the lock		{			wchar_t* mimeType[] = {					L"image/jpeg",					L"image/png",					L"image/bmp",					L"image/gif"			};			if (!g_playing_file->uvox_artwork.uvox_stream_artwork)			{				int ret = g_playing_file->info.GetAlbumArt(type, bits, len, mimeType);				LeaveCriticalSection(&streamInfoLock);				return ret;			}			else			{				// will handle "playing" and "cover" - cover is the stream branding				// with "playing" used to provide song specific stream artwork				if (!_wcsicmp(type, L"playing"))				{					if (g_playing_file->uvox_artwork.uvox_playing_artwork_len > 0)					{						*len = g_playing_file->uvox_artwork.uvox_playing_artwork_len;						int type = g_playing_file->uvox_artwork.uvox_playing_artwork_type;						if(type >= 0 && type <= 3)						{							*mimeType = (wchar_t *)WASABI_API_MEMMGR->sysMalloc(12 * sizeof(wchar_t));							wcsncpy(*mimeType, mimeType[type], 12);						}						else						{							*mimeType = 0;						}						*bits = WASABI_API_MEMMGR->sysMalloc(*len);						memcpy(*bits, g_playing_file->uvox_artwork.uvox_playing_artwork, *len);						LeaveCriticalSection(&streamInfoLock);						return ALBUMARTPROVIDER_SUCCESS;					}					else					{						LeaveCriticalSection(&streamInfoLock);						return ALBUMARTPROVIDER_FAILURE;					}				}				else				{					if (g_playing_file->uvox_artwork.uvox_stream_artwork_len > 0)					{						*len = g_playing_file->uvox_artwork.uvox_stream_artwork_len;												int type = g_playing_file->uvox_artwork.uvox_stream_artwork_type;						if(type >= 0 && type <= 3)						{							*mimeType = (wchar_t *)WASABI_API_MEMMGR->sysMalloc(12 * sizeof(wchar_t));							wcsncpy(*mimeType, mimeType[type], 12);						}						else						{							*mimeType = 0;						}						*bits = WASABI_API_MEMMGR->sysMalloc(*len);						memcpy(*bits, g_playing_file->uvox_artwork.uvox_stream_artwork, *len);						LeaveCriticalSection(&streamInfoLock);						return ALBUMARTPROVIDER_SUCCESS;					}					else					{						LeaveCriticalSection(&streamInfoLock);						return ALBUMARTPROVIDER_FAILURE;					}				}			}		}		LeaveCriticalSection(&streamInfoLock);	}	Metadata metadata;	if (metadata.Open(filename) == METADATA_SUCCESS)	{		return metadata.id3v2.GetAlbumArt(type, bits, len, mimeType);	}	return ALBUMARTPROVIDER_FAILURE;}extern Metadata *m_ext_get_mp3info;int ID3v2_AlbumArtProvider::SetAlbumArtData(const wchar_t *filename, const wchar_t *type, void *bits, size_t len, const wchar_t *mimeType){	Metadata metadata;	if (metadata.Open(filename) == METADATA_SUCCESS)	{		int ret = metadata.id3v2.SetAlbumArt(type, bits, len, mimeType);		if (ret == METADATA_SUCCESS)		{			// flush our read cache too :)			if (m_ext_get_mp3info) m_ext_get_mp3info->Release();			m_ext_get_mp3info = NULL;			Stopper stopper;			if (metadata.IsMe(lastfn))				stopper.Stop();			metadata.Save();			stopper.Play();		}		return ret;	}	return ALBUMARTPROVIDER_FAILURE;}int ID3v2_AlbumArtProvider::DeleteAlbumArt(const wchar_t *filename, const wchar_t *type){	Metadata metadata;	if (metadata.Open(filename) == METADATA_SUCCESS)	{		int ret = metadata.id3v2.DeleteAlbumArt(type);		if (ret == METADATA_SUCCESS)		{			// flush our read cache too :)			if (m_ext_get_mp3info) m_ext_get_mp3info->Release();			m_ext_get_mp3info = NULL;			Stopper stopper;			if (metadata.IsMe(lastfn))				stopper.Stop();			metadata.Save();			stopper.Play();		}		return ret;	}	return ALBUMARTPROVIDER_FAILURE;}#define CBCLASS ID3v2_AlbumArtProviderSTART_DISPATCH;CB(SVC_ALBUMARTPROVIDER_PROVIDERTYPE, ProviderType);CB(SVC_ALBUMARTPROVIDER_GETALBUMARTDATA, GetAlbumArtData);CB(SVC_ALBUMARTPROVIDER_ISMINE, IsMine);CB(SVC_ALBUMARTPROVIDER_SETALBUMARTDATA, SetAlbumArtData);CB(SVC_ALBUMARTPROVIDER_DELETEALBUMART, DeleteAlbumArt);END_DISPATCH;#undef CBCLASSstatic ID3v2_AlbumArtProvider albumArtProvider;// {C8222317-8F0D-4e79-9222-447381C46E07}static const GUID id3v2_albumartproviderGUID =  { 0xc8222317, 0x8f0d, 0x4e79, { 0x92, 0x22, 0x44, 0x73, 0x81, 0xc4, 0x6e, 0x7 } };FOURCC AlbumArtFactory::GetServiceType(){	return svc_albumArtProvider::SERVICETYPE;}const char *AlbumArtFactory::GetServiceName(){	return "ID3v2 Album Art Provider";}GUID AlbumArtFactory::GetGUID(){	return id3v2_albumartproviderGUID;}void *AlbumArtFactory::GetInterface(int global_lock){	return &albumArtProvider;}int AlbumArtFactory::SupportNonLockingInterface(){	return 1;}int AlbumArtFactory::ReleaseInterface(void *ifc){	//plugin.service->service_unlock(ifc);	return 1;}const char *AlbumArtFactory::GetTestString(){	return 0;}int AlbumArtFactory::ServiceNotify(int msg, int param1, int param2){	return 1;}#ifdef CBCLASS#undef CBCLASS#endif#define CBCLASS AlbumArtFactorySTART_DISPATCH;CB(WASERVICEFACTORY_GETSERVICETYPE, GetServiceType)CB(WASERVICEFACTORY_GETSERVICENAME, GetServiceName)CB(WASERVICEFACTORY_GETGUID, GetGUID)CB(WASERVICEFACTORY_GETINTERFACE, GetInterface)CB(WASERVICEFACTORY_SUPPORTNONLOCKINGGETINTERFACE, SupportNonLockingInterface)CB(WASERVICEFACTORY_RELEASEINTERFACE, ReleaseInterface)CB(WASERVICEFACTORY_GETTESTSTRING, GetTestString)CB(WASERVICEFACTORY_SERVICENOTIFY, ServiceNotify)END_DISPATCH;
 |