mirror of
https://github.com/WinampDesktop/winamp.git
synced 2024-11-24 13:00:09 +01:00
215 lines
4.4 KiB
C++
215 lines
4.4 KiB
C++
#include "main.h"
|
|
#include "./enumIniFile.h"
|
|
#include "./service.h"
|
|
#include "./ifc_omservicehost.h"
|
|
#include "./ifc_omstorage.h"
|
|
#include "./ifc_omstoragehandlerenum.h"
|
|
#include "./ifc_omstorageext.h"
|
|
#include "./ifc_omfilestorage.h"
|
|
|
|
#include <shlwapi.h>
|
|
#include <strsafe.h>
|
|
|
|
#define OMS_GROUP "OnlineService"
|
|
|
|
#define OMS_ID "id"
|
|
#define OMS_NAME "name"
|
|
#define OMS_URL "url"
|
|
#define OMS_ICON "icon"
|
|
#define OMS_FLAGS "flags"
|
|
#define OMS_RATING "rating"
|
|
#define OMS_VERSION "version"
|
|
#define OMS_DESCRIPTION "description"
|
|
#define OMS_AUTHORFIRST "authorFirst"
|
|
#define OMS_AUTHORLAST "authorLast"
|
|
#define OMS_PUBLISHED "publishedDate"
|
|
#define OMS_UPDATED "updatedDate"
|
|
#define OMS_THUMBNAIL "thumbnail"
|
|
#define OMS_SCREENSHOT "screenshot"
|
|
#define OMS_GENERATION "generation"
|
|
|
|
EnumIniFile::EnumIniFile(LPCWSTR pszAddress, ifc_omservicehost *serviceHost)
|
|
: ref(1), address(NULL), host(serviceHost), hFind(NULL)
|
|
{
|
|
address = Plugin_CopyString(pszAddress);
|
|
|
|
if (NULL != host)
|
|
{
|
|
host->AddRef();
|
|
|
|
ifc_omstorageext *storageExt = NULL;
|
|
if (SUCCEEDED(host->QueryInterface(IFC_OmStorageExt, (void**)&storageExt)) && storageExt != NULL)
|
|
{
|
|
ifc_omstoragehandlerenum *handlerEnum = NULL;
|
|
if (SUCCEEDED(storageExt->Enumerate(&SUID_OmStorageIni, &handlerEnum)) && handlerEnum != NULL)
|
|
{
|
|
reader.RegisterHandlers(handlerEnum);
|
|
handlerEnum->Release();
|
|
}
|
|
storageExt->Release();
|
|
}
|
|
}
|
|
}
|
|
|
|
EnumIniFile::~EnumIniFile()
|
|
{
|
|
Plugin_FreeString(address);
|
|
|
|
if (NULL != host)
|
|
host->Release();
|
|
|
|
if (NULL != hFind)
|
|
FindClose(hFind);
|
|
}
|
|
|
|
HRESULT EnumIniFile::CreateInstance(LPCWSTR pszAddress, ifc_omservicehost *host, EnumIniFile **instance)
|
|
{
|
|
if (NULL == instance)
|
|
return E_POINTER;
|
|
|
|
WCHAR szBuffer[MAX_PATH * 2] = {0};
|
|
HRESULT hr = Plugin_ResolveRelativePath(pszAddress, host, szBuffer, ARRAYSIZE(szBuffer));
|
|
if (FAILED(hr)) return hr;
|
|
|
|
// test if we can load this one
|
|
if (FALSE == PathIsDirectory(szBuffer) && PathFileExists(szBuffer))
|
|
{
|
|
UINT id = GetPrivateProfileInt(TEXT(OMS_GROUP), WTEXT(OMS_ID), 0, pszAddress);
|
|
if (0 == id) return OMSTORAGE_E_UNKNOWN_FORMAT;
|
|
}
|
|
|
|
*instance = new EnumIniFile(szBuffer, host);
|
|
if (NULL == *instance) return E_OUTOFMEMORY;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
size_t EnumIniFile::AddRef()
|
|
{
|
|
return InterlockedIncrement((LONG*)&ref);
|
|
}
|
|
|
|
size_t EnumIniFile::Release()
|
|
{
|
|
if (0 == ref)
|
|
return ref;
|
|
|
|
LONG r = InterlockedDecrement((LONG*)&ref);
|
|
if (0 == r)
|
|
delete(this);
|
|
|
|
return r;
|
|
}
|
|
|
|
int EnumIniFile::QueryInterface(GUID interface_guid, void **object)
|
|
{
|
|
if (NULL == object) return E_POINTER;
|
|
|
|
if (IsEqualIID(interface_guid, IFC_OmServiceEnum))
|
|
*object = static_cast<ifc_omserviceenum*>(this);
|
|
else
|
|
{
|
|
*object = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
if (NULL == *object)
|
|
return E_UNEXPECTED;
|
|
|
|
AddRef();
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT EnumIniFile::Next(ULONG listSize, ifc_omservice **elementList, ULONG *elementCount)
|
|
{
|
|
if(NULL != elementCount)
|
|
*elementCount = 0;
|
|
|
|
if (0 == listSize || NULL == elementList)
|
|
return E_INVALIDARG;
|
|
|
|
ULONG counter = 0;
|
|
BOOL bFoundNext = FALSE;
|
|
HRESULT hr = S_OK;
|
|
|
|
if (NULL == hFind)
|
|
{
|
|
hFind = FindFirstFile(address, &fData);
|
|
if (INVALID_HANDLE_VALUE == hFind)
|
|
{
|
|
DWORD error = GetLastError();
|
|
return HRESULT_FROM_WIN32(error);
|
|
}
|
|
bFoundNext = TRUE;
|
|
}
|
|
else
|
|
{
|
|
bFoundNext = FindNextFile(hFind, &fData);
|
|
}
|
|
|
|
if (bFoundNext)
|
|
{
|
|
do
|
|
{
|
|
LPCWSTR p = address;
|
|
while(p && L'.' == *p && L'\0' != *p) p++;
|
|
if (p && L'\0' != *p)
|
|
{
|
|
WCHAR base[MAX_PATH] = {0};
|
|
StringCchCopy(base, MAX_PATH, address);
|
|
PathRemoveFileSpec(base);
|
|
|
|
int baseLen = lstrlen(base);
|
|
base[baseLen] = L'\0';
|
|
|
|
if (!PathAppend(base, fData.cFileName))
|
|
{
|
|
base[baseLen] = L'\0';
|
|
PathAppend(base, fData.cAlternateFileName);
|
|
}
|
|
|
|
hr = reader.Load(base, host, &elementList[counter]);
|
|
if (S_OK == hr)
|
|
{
|
|
listSize--;
|
|
counter++;
|
|
if (0 == listSize) break;
|
|
}
|
|
}
|
|
bFoundNext = FindNextFile(hFind, &fData);
|
|
|
|
} while(bFoundNext);
|
|
}
|
|
|
|
if(NULL != elementCount)
|
|
*elementCount = counter;
|
|
|
|
return (counter > 0) ? S_OK : S_FALSE;
|
|
}
|
|
|
|
HRESULT EnumIniFile::Reset(void)
|
|
{
|
|
if (NULL != hFind)
|
|
{
|
|
FindClose(hFind);
|
|
hFind = NULL;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT EnumIniFile::Skip(ULONG elementCount)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
#define CBCLASS EnumIniFile
|
|
START_DISPATCH;
|
|
CB(ADDREF, AddRef)
|
|
CB(RELEASE, Release)
|
|
CB(QUERYINTERFACE, QueryInterface)
|
|
CB(API_NEXT, Next)
|
|
CB(API_RESET, Reset)
|
|
CB(API_SKIP, Skip)
|
|
END_DISPATCH;
|
|
#undef CBCLASS |