324 lines
7.7 KiB
C++
324 lines
7.7 KiB
C++
|
/** (c) Nullsoft, Inc. C O N F I D E N T I A L
|
||
|
** Filename:
|
||
|
** Project:
|
||
|
** Description:
|
||
|
** Author: Ben Allison benski@nullsoft.com
|
||
|
** Created:
|
||
|
**/
|
||
|
#include "main.h"
|
||
|
#include "../nu/ns_wc.h"
|
||
|
#include "SkinCOM.h"
|
||
|
#include "resource.h"
|
||
|
#include "../nu/AutoWide.h"
|
||
|
#include "../nu/AutoChar.h"
|
||
|
#include "wa_dlg.h"
|
||
|
#include "Browser.h"
|
||
|
|
||
|
void WriteEscaped(FILE *fp, const char *str);
|
||
|
|
||
|
void WriteSkinsAsXML(const wchar_t *filename, int limit)
|
||
|
{
|
||
|
FILE *fp = _wfopen(filename, L"wb");
|
||
|
fputs("<?xml version=\"1.0\" encoding=\"utf-8\"?>", fp);
|
||
|
fputs("<skins>\n", fp);
|
||
|
fputs("<skin filename=\"", fp);
|
||
|
WriteEscaped(fp,"Winamp Classic");
|
||
|
fputs("\"/>\n", fp);
|
||
|
|
||
|
HANDLE h;
|
||
|
WIN32_FIND_DATAW d;
|
||
|
|
||
|
wchar_t dirmask[MAX_PATH] = {0};
|
||
|
PathCombineW(dirmask, SKINDIR, L"*");
|
||
|
|
||
|
h = FindFirstFileW(dirmask, &d);
|
||
|
if (h != INVALID_HANDLE_VALUE )
|
||
|
{
|
||
|
int count=1;
|
||
|
do
|
||
|
{
|
||
|
if ( limit && count >= limit ) break;
|
||
|
if (d.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
{
|
||
|
if (wcscmp(d.cFileName, L".") && wcscmp(d.cFileName, L".."))
|
||
|
{
|
||
|
fputs("<skin filename=\"", fp);
|
||
|
WriteEscaped(fp, AutoChar(d.cFileName, CP_UTF8));
|
||
|
fputs("\"/>\n", fp);
|
||
|
count++;
|
||
|
}
|
||
|
}
|
||
|
else if (!_wcsicmp(extensionW(d.cFileName), L"zip") || !_wcsicmp(extensionW(d.cFileName), L"wsz") || !_wcsicmp(extensionW(d.cFileName), L"wal"))
|
||
|
{
|
||
|
fputs("<skin filename=\"", fp);
|
||
|
WriteEscaped(fp, AutoChar(d.cFileName, CP_UTF8));
|
||
|
fputs("\"/>\n", fp);
|
||
|
count++;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
while (FindNextFileW(h, &d));
|
||
|
FindClose(h);
|
||
|
|
||
|
}
|
||
|
fputs("</skins>", fp);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
|
||
|
|
||
|
enum
|
||
|
{
|
||
|
DISP_CURRENTSKIN_GETCOLOR = 777,
|
||
|
DISP_CURRENTSKIN_GETNAME,
|
||
|
DISP_CURRENTSKIN_REGISTERSKINCHANGECALLBACK,
|
||
|
DISP_CURRENTSKIN_UNREGISTERSKINCHANGECALLBACK,
|
||
|
DISP_CURRENTSKIN_GETFONTNAME,
|
||
|
DISP_CURRENTSKIN_GETFONTSIZE,
|
||
|
DISP_CURRENTSKIN_GETPLAYLISTCOLOR,
|
||
|
DISP_CURRENTSKIN_SETSKIN,
|
||
|
DISP_CURRENTSKIN_GETXMLSKINLIST,
|
||
|
};
|
||
|
|
||
|
|
||
|
#define CHECK_ID(str, id)\
|
||
|
if (CSTR_EQUAL == CompareStringW(lcid, NORM_IGNORECASE, rgszNames[i], -1, L##str, -1))\
|
||
|
{ rgdispid[i] = id; continue; }
|
||
|
|
||
|
HRESULT SkinCOM::GetIDsOfNames(REFIID riid, OLECHAR FAR* FAR* rgszNames, unsigned int cNames, LCID lcid, DISPID FAR* rgdispid)
|
||
|
{
|
||
|
UNREFERENCED_PARAMETER(riid);
|
||
|
|
||
|
bool unknowns = false;
|
||
|
for (unsigned int i = 0;i != cNames;i++)
|
||
|
{
|
||
|
CHECK_ID("GetColor", DISP_CURRENTSKIN_GETCOLOR)
|
||
|
CHECK_ID("GetName", DISP_CURRENTSKIN_GETNAME)
|
||
|
CHECK_ID("RegisterSkinChangeCallback", DISP_CURRENTSKIN_REGISTERSKINCHANGECALLBACK)
|
||
|
CHECK_ID("UnregisterSkinChangeCallback", DISP_CURRENTSKIN_UNREGISTERSKINCHANGECALLBACK)
|
||
|
CHECK_ID("GetFontName", DISP_CURRENTSKIN_GETFONTNAME)
|
||
|
CHECK_ID("GetFontSize", DISP_CURRENTSKIN_GETFONTSIZE);
|
||
|
CHECK_ID("GetPlaylistColor", DISP_CURRENTSKIN_GETPLAYLISTCOLOR);
|
||
|
CHECK_ID("SetSkin", DISP_CURRENTSKIN_SETSKIN);
|
||
|
CHECK_ID("GetXMLSkinList", DISP_CURRENTSKIN_GETXMLSKINLIST);
|
||
|
|
||
|
rgdispid[i] = DISPID_UNKNOWN;
|
||
|
unknowns = true;
|
||
|
|
||
|
}
|
||
|
if (unknowns)
|
||
|
return DISP_E_UNKNOWNNAME;
|
||
|
else
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
HRESULT SkinCOM::GetTypeInfo(unsigned int itinfo, LCID lcid, ITypeInfo FAR* FAR* pptinfo)
|
||
|
{
|
||
|
UNREFERENCED_PARAMETER(itinfo);
|
||
|
UNREFERENCED_PARAMETER(lcid);
|
||
|
UNREFERENCED_PARAMETER(pptinfo);
|
||
|
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
HRESULT SkinCOM::GetTypeInfoCount(unsigned int FAR * pctinfo)
|
||
|
{
|
||
|
UNREFERENCED_PARAMETER(pctinfo);
|
||
|
|
||
|
return E_NOTIMPL;
|
||
|
}
|
||
|
|
||
|
|
||
|
HRESULT SkinCOM::Invoke(DISPID dispid, REFIID riid, LCID lcid, WORD wFlags, DISPPARAMS FAR *pdispparams, VARIANT FAR *pvarResult, EXCEPINFO FAR * pexecinfo, unsigned int FAR *puArgErr)
|
||
|
{
|
||
|
UNREFERENCED_PARAMETER(riid);
|
||
|
UNREFERENCED_PARAMETER(lcid);
|
||
|
UNREFERENCED_PARAMETER(pexecinfo);
|
||
|
UNREFERENCED_PARAMETER(wFlags);
|
||
|
|
||
|
switch (dispid)
|
||
|
{
|
||
|
case DISP_CURRENTSKIN_SETSKIN:
|
||
|
if (pdispparams->cArgs == 1)
|
||
|
{
|
||
|
const wchar_t *newSkin = pdispparams->rgvarg[0].bstrVal;
|
||
|
if (newSkin && *newSkin)
|
||
|
{
|
||
|
if (_wcsicmp(config_skin, newSkin))
|
||
|
{
|
||
|
StringCchCopyW(config_skin, MAX_PATH, newSkin);
|
||
|
PostMessageW(hMainWindow, WM_COMMAND, WINAMP_REFRESHSKIN, 0);
|
||
|
}
|
||
|
}
|
||
|
return S_OK;
|
||
|
}
|
||
|
else
|
||
|
return DISP_E_BADPARAMCOUNT;
|
||
|
case DISP_CURRENTSKIN_REGISTERSKINCHANGECALLBACK:
|
||
|
return callbacks.RegisterFromDispParam(pdispparams, 0, puArgErr);
|
||
|
case DISP_CURRENTSKIN_UNREGISTERSKINCHANGECALLBACK:
|
||
|
return callbacks.UnregisterFromDispParam(pdispparams, 0, puArgErr);
|
||
|
case DISP_CURRENTSKIN_GETCOLOR:
|
||
|
if (pdispparams->cArgs == 1)
|
||
|
{
|
||
|
COLORREF color = WADlg_getColor(pdispparams->rgvarg[0].lVal);
|
||
|
color = ((color >> 16) & 0xff | (color & 0xff00) | ((color << 16) & 0xff0000));
|
||
|
char colorString[8] = {0};
|
||
|
StringCchPrintfA(colorString, 8, "#%06X", color);
|
||
|
AutoWide answer(colorString);
|
||
|
BSTR tag = SysAllocString(answer);
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_BSTR;
|
||
|
V_BSTR(pvarResult) = tag;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
else
|
||
|
return DISP_E_BADPARAMCOUNT;
|
||
|
|
||
|
case DISP_CURRENTSKIN_GETNAME:
|
||
|
{
|
||
|
BSTR tag = SysAllocString(config_skin);
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_BSTR;
|
||
|
V_BSTR(pvarResult) = tag;
|
||
|
return S_OK;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case DISP_CURRENTSKIN_GETFONTNAME:
|
||
|
{
|
||
|
BSTR tag = SysAllocString(GetFontNameW());
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_BSTR;
|
||
|
V_BSTR(pvarResult) = tag;
|
||
|
return S_OK;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case DISP_CURRENTSKIN_GETFONTSIZE:
|
||
|
{
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_I4;
|
||
|
V_I4(pvarResult) = GetFontSize();
|
||
|
return S_OK;
|
||
|
}
|
||
|
case DISP_CURRENTSKIN_GETPLAYLISTCOLOR:
|
||
|
if (pdispparams->cArgs == 1)
|
||
|
{
|
||
|
COLORREF color = Skin_PLColors[pdispparams->rgvarg[0].lVal];
|
||
|
color = ((color >> 16) & 0xff | (color & 0xff00) | ((color << 16) & 0xff0000));
|
||
|
char colorString[8] = {0};
|
||
|
StringCchPrintfA(colorString, 8, "#%06X", color);
|
||
|
AutoWide answer(colorString);
|
||
|
BSTR tag = SysAllocString(answer);
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_BSTR;
|
||
|
V_BSTR(pvarResult) = tag;
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
else
|
||
|
return DISP_E_BADPARAMCOUNT;
|
||
|
|
||
|
case DISP_CURRENTSKIN_GETXMLSKINLIST:
|
||
|
{
|
||
|
int max=0;
|
||
|
if (pdispparams->cArgs == 1)
|
||
|
{
|
||
|
|
||
|
max = _wtoi(pdispparams->rgvarg[0].bstrVal);
|
||
|
}
|
||
|
|
||
|
wchar_t tempPath[MAX_PATH] = {0};
|
||
|
GetTempPathW(MAX_PATH, tempPath);
|
||
|
wchar_t tempFile[MAX_PATH] = {0};
|
||
|
GetTempFileNameW(tempPath, L"slx", 0, tempFile);
|
||
|
WriteSkinsAsXML(tempFile, max);
|
||
|
|
||
|
HANDLE plFile = CreateFileW(tempFile, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, 0, 0);
|
||
|
size_t flen = SetFilePointer(plFile, 0, NULL, FILE_END);
|
||
|
SetFilePointer(plFile, 0, NULL, FILE_BEGIN);
|
||
|
|
||
|
SAFEARRAY *bufferArray=SafeArrayCreateVector(VT_UI1, 0, (ULONG)flen);
|
||
|
void *data;
|
||
|
SafeArrayAccessData(bufferArray, &data);
|
||
|
|
||
|
DWORD bytesRead = 0;
|
||
|
|
||
|
ReadFile(plFile, data, (DWORD)flen, &bytesRead, 0);
|
||
|
|
||
|
SafeArrayUnaccessData(bufferArray);
|
||
|
CloseHandle(plFile);
|
||
|
VariantInit(pvarResult);
|
||
|
V_VT(pvarResult) = VT_ARRAY|VT_UI1;
|
||
|
V_ARRAY(pvarResult) = bufferArray;
|
||
|
DeleteFileW(tempFile);
|
||
|
}
|
||
|
return S_OK;
|
||
|
|
||
|
}
|
||
|
return DISP_E_MEMBERNOTFOUND;
|
||
|
}
|
||
|
|
||
|
STDMETHODIMP SkinCOM::QueryInterface(REFIID riid, PVOID *ppvObject)
|
||
|
{
|
||
|
if (!ppvObject)
|
||
|
return E_POINTER;
|
||
|
|
||
|
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 SkinCOM::AddRef(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
ULONG SkinCOM::Release(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void SkinChangedNotifyCb(IDispatch *dispatch, void *param)
|
||
|
{
|
||
|
UNREFERENCED_PARAMETER(param);
|
||
|
|
||
|
DISPPARAMS params;
|
||
|
|
||
|
params.cArgs = 0;
|
||
|
params.cNamedArgs = 0;
|
||
|
params.rgdispidNamedArgs = 0;
|
||
|
params.rgvarg = 0;
|
||
|
unsigned int ret;
|
||
|
|
||
|
if (!(config_no_visseh&8))
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
dispatch->Invoke(0, GUID_NULL, 0, DISPATCH_METHOD, ¶ms, 0, 0, &ret);
|
||
|
}
|
||
|
catch(...)
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
dispatch->Invoke(0, GUID_NULL, 0, DISPATCH_METHOD, ¶ms, 0, 0, &ret);
|
||
|
}
|
||
|
|
||
|
void SkinCOM::SkinChanged()
|
||
|
{
|
||
|
callbacks.Notify(SkinChangedNotifyCb, NULL, NULL);
|
||
|
}
|