Initial commit

This commit is contained in:
2025-12-17 16:47:48 +00:00
commit 13813f3363
4964 changed files with 1079753 additions and 0 deletions

View File

@@ -0,0 +1,228 @@
#include <algorithm>
#include <cstdint>
#include <new>
#include <shlwapi.h>
#include <string>
#include <thumbcache.h> /* for #IThumbnailProvider */
#include <vector>
#include "Wincodec.h"
#pragma comment(lib, "shlwapi.lib")
struct Thumbnail
{
std::vector<uint8_t> data;
int width;
int height;
};
/**
* This thumbnail provider implements #IInitializeWithStream to enable being
* hosted in an isolated process for robustness.
*/
class CAyaThumb
: public IInitializeWithStream
, public IThumbnailProvider
{
public:
CAyaThumb()
: _cRef(1)
, _pStream(nullptr)
{
}
virtual ~CAyaThumb()
{
if (_pStream)
{
_pStream->Release();
}
}
IFACEMETHODIMP QueryInterface(REFIID riid, void** ppv)
{
static const QITAB qit[] = {
QITABENT(CAyaThumb, IInitializeWithStream),
QITABENT(CAyaThumb, IThumbnailProvider),
{0},
};
return QISearch(this, qit, riid, ppv);
}
IFACEMETHODIMP_(ULONG) AddRef()
{
return InterlockedIncrement(&_cRef);
}
IFACEMETHODIMP_(ULONG) Release()
{
ULONG cRef = InterlockedDecrement(&_cRef);
if (!cRef)
{
delete this;
}
return cRef;
}
/** IInitializeWithStream */
IFACEMETHODIMP Initialize(IStream* pStream, DWORD grfMode);
/** IThumbnailProvider */
IFACEMETHODIMP GetThumbnail(UINT cx, HBITMAP* phbmp, WTS_ALPHATYPE* pdwAlpha);
private:
long _cRef;
IStream* _pStream; /* provided in Initialize(). */
};
HRESULT CAyaThumb_CreateInstance(REFIID riid, void** ppv)
{
CAyaThumb* pNew = new (std::nothrow) CAyaThumb();
HRESULT hr = pNew ? S_OK : E_OUTOFMEMORY;
if (SUCCEEDED(hr))
{
hr = pNew->QueryInterface(riid, ppv);
pNew->Release();
}
return hr;
}
IFACEMETHODIMP CAyaThumb::Initialize(IStream* pStream, DWORD)
{
if (_pStream != nullptr)
{
/* Can only be initialized once. */
return E_UNEXPECTED;
}
/* Take a reference to the stream. */
return pStream->QueryInterface(&_pStream);
}
IFACEMETHODIMP CAyaThumb::GetThumbnail(UINT cx, HBITMAP* phbmp, WTS_ALPHATYPE* pdwAlpha)
{
HRESULT hr = S_FALSE;
std::vector<char> buffer;
ULONG bytesRead;
char chunk[4096];
do
{
hr = _pStream->Read(chunk, sizeof(chunk), &bytesRead);
if (SUCCEEDED(hr) && bytesRead > 0)
{
buffer.insert(buffer.end(), chunk, chunk + bytesRead);
}
} while (SUCCEEDED(hr) && bytesRead > 0);
if (FAILED(hr))
{
return hr;
}
// Find the </roblox> closing tag
const std::string endTag = "</roblox>";
auto it = std::search(buffer.begin(), buffer.end(), endTag.begin(), endTag.end());
if (it == buffer.end())
{
return E_FAIL; // Closing tag not found
}
// Move iterator past the closing tag and the null byte
std::advance(it, endTag.length() + 1);
if (it >= buffer.end())
{
return E_FAIL; // No data after the closing tag
}
// The rest is JPEG data
std::vector<unsigned char> jpegData(it, buffer.end());
// Create a WIC factory
IWICImagingFactory* pFactory = nullptr;
hr = CoCreateInstance(CLSID_WICImagingFactory, nullptr, CLSCTX_INPROC_SERVER, IID_PPV_ARGS(&pFactory));
if (FAILED(hr))
{
return hr;
}
// Create a stream from the JPEG data
IWICStream* pStream = nullptr;
hr = pFactory->CreateStream(&pStream);
if (FAILED(hr))
{
pFactory->Release();
return hr;
}
hr = pStream->InitializeFromMemory(jpegData.data(), jpegData.size());
if (FAILED(hr))
{
pStream->Release();
pFactory->Release();
return hr;
}
// Create a decoder
IWICBitmapDecoder* pDecoder = nullptr;
hr = pFactory->CreateDecoderFromStream(pStream, nullptr, WICDecodeMetadataCacheOnDemand, &pDecoder);
if (FAILED(hr))
{
pStream->Release();
pFactory->Release();
return hr;
}
// Get the first frame of the image from the decoder
IWICBitmapFrameDecode* pFrame = nullptr;
hr = pDecoder->GetFrame(0, &pFrame);
if (FAILED(hr))
{
pDecoder->Release();
pStream->Release();
pFactory->Release();
return hr;
}
// Get the size of the image
UINT width, height;
hr = pFrame->GetSize(&width, &height);
if (FAILED(hr))
{
pFrame->Release();
pDecoder->Release();
pStream->Release();
pFactory->Release();
return hr;
}
UINT stride = (width * 3 + 3) & ~3;
// Create a bitmap and copy the pixels
Thumbnail thumb;
thumb.width = width;
thumb.height = height;
thumb.data.resize(height * stride);
hr = pFrame->CopyPixels(nullptr, stride, thumb.data.size(), thumb.data.data());
pFrame->Release();
pDecoder->Release();
pStream->Release();
pFactory->Release();
if (FAILED(hr))
{
return hr;
}
*phbmp = CreateBitmap(thumb.width, thumb.height, 1, 24, thumb.data.data());
if (!*phbmp)
{
return E_FAIL;
}
*pdwAlpha = WTSAT_RGB;
hr = S_OK;
return hr;
}

View File

@@ -0,0 +1,5 @@
EXPORTS
DllGetClassObject PRIVATE
DllCanUnloadNow PRIVATE
DllRegisterServer PRIVATE
DllUnregisterServer PRIVATE

View File

@@ -0,0 +1,262 @@
#include <new>
#include <objbase.h>
#include <shlobj.h> /* For #SHChangeNotify */
#include <shlwapi.h>
#include <thumbcache.h> /* For IThumbnailProvider */
extern HRESULT CAyaThumb_CreateInstance(REFIID riid, void** ppv);
#define SZ_CLSID_AYATHUMBHANDLER L"{8ABA9ABD-829D-4E87-AC2C-4A628AB78236}"
#define SZ_AYATHUMBHANDLER L"Aya Thumbnail Handler"
const CLSID CLSID_AyaThumbHandler = {0x8ABA9ABD, 0x829D, 0x4E87, {0xAC, 0x2C, 0x4A, 0x62, 0x8A, 0xB7, 0x82, 0x36}};
typedef HRESULT (*PFNCREATEINSTANCE)(REFIID riid, void** ppvObject);
struct CLASS_OBJECT_INIT
{
const CLSID* pClsid;
PFNCREATEINSTANCE pfnCreate;
};
/* Add classes supported by this module here. */
const CLASS_OBJECT_INIT c_rgClassObjectInit[] = {{&CLSID_AyaThumbHandler, CAyaThumb_CreateInstance}};
long g_cRefModule = 0;
/** Handle the DLL's module */
HINSTANCE g_hInst = nullptr;
/** Standard DLL functions. */
STDAPI_(BOOL) DllMain(HINSTANCE hInstance, DWORD dwReason, void*)
{
if (dwReason == DLL_PROCESS_ATTACH)
{
g_hInst = hInstance;
DisableThreadLibraryCalls(hInstance);
}
return TRUE;
}
STDAPI DllCanUnloadNow()
{
/* Only allow the DLL to be unloaded after all outstanding references have
* been released. */
return (g_cRefModule == 0) ? S_OK : S_FALSE;
}
void DllAddRef()
{
InterlockedIncrement(&g_cRefModule);
}
void DllRelease()
{
InterlockedDecrement(&g_cRefModule);
}
class CClassFactory : public IClassFactory
{
public:
static HRESULT CreateInstance(REFCLSID clsid, const CLASS_OBJECT_INIT* pClassObjectInits, size_t cClassObjectInits, REFIID riid, void** ppv)
{
*ppv = nullptr;
HRESULT hr = CLASS_E_CLASSNOTAVAILABLE;
for (size_t i = 0; i < cClassObjectInits; i++)
{
if (clsid == *pClassObjectInits[i].pClsid)
{
IClassFactory* pClassFactory = new (std::nothrow) CClassFactory(pClassObjectInits[i].pfnCreate);
hr = pClassFactory ? S_OK : E_OUTOFMEMORY;
if (SUCCEEDED(hr))
{
hr = pClassFactory->QueryInterface(riid, ppv);
pClassFactory->Release();
}
/* Match found. */
break;
}
}
return hr;
}
CClassFactory(PFNCREATEINSTANCE pfnCreate)
: _cRef(1)
, _pfnCreate(pfnCreate)
{
DllAddRef();
}
/** #IUnknown */
IFACEMETHODIMP QueryInterface(REFIID riid, void** ppv)
{
static const QITAB qit[] = {QITABENT(CClassFactory, IClassFactory), {0}};
return QISearch(this, qit, riid, ppv);
}
IFACEMETHODIMP_(ULONG) AddRef()
{
return InterlockedIncrement(&_cRef);
}
IFACEMETHODIMP_(ULONG) Release()
{
long cRef = InterlockedDecrement(&_cRef);
if (cRef == 0)
{
delete this;
}
return cRef;
}
/** #IClassFactory */
IFACEMETHODIMP CreateInstance(IUnknown* punkOuter, REFIID riid, void** ppv)
{
return punkOuter ? CLASS_E_NOAGGREGATION : _pfnCreate(riid, ppv);
}
IFACEMETHODIMP LockServer(BOOL fLock)
{
if (fLock)
{
DllAddRef();
}
else
{
DllRelease();
}
return S_OK;
}
private:
~CClassFactory()
{
DllRelease();
}
long _cRef;
PFNCREATEINSTANCE _pfnCreate;
};
STDAPI DllGetClassObject(REFCLSID clsid, REFIID riid, void** ppv)
{
return CClassFactory::CreateInstance(clsid, c_rgClassObjectInit, ARRAYSIZE(c_rgClassObjectInit), riid, ppv);
}
/**
* A struct to hold the information required for a registry entry.
*/
struct REGISTRY_ENTRY
{
HKEY hkeyRoot;
PCWSTR pszKeyName;
PCWSTR pszValueName;
DWORD dwValueType;
/** These two fields could/should have been a union, but C++ */
PCWSTR pszData;
/** Only lets you initialize the first field in a union. */
DWORD dwData;
};
/**
* Creates a registry key (if needed) and sets the default value of the key.
*/
HRESULT CreateRegKeyAndSetValue(const REGISTRY_ENTRY* pRegistryEntry)
{
HKEY hKey;
HRESULT hr = HRESULT_FROM_WIN32(RegCreateKeyExW(pRegistryEntry->hkeyRoot, pRegistryEntry->pszKeyName, 0, nullptr, REG_OPTION_NON_VOLATILE, KEY_SET_VALUE, nullptr, &hKey, nullptr));
if (SUCCEEDED(hr))
{
/* All this just to support #REG_DWORD. */
DWORD size;
DWORD data;
BYTE* lpData = (LPBYTE)pRegistryEntry->pszData;
switch (pRegistryEntry->dwValueType)
{
case REG_SZ:
size = ((DWORD)wcslen(pRegistryEntry->pszData) + 1) * sizeof(WCHAR);
break;
case REG_DWORD:
size = sizeof(DWORD);
data = pRegistryEntry->dwData;
lpData = (BYTE*)&data;
break;
default:
return E_INVALIDARG;
}
hr = HRESULT_FROM_WIN32(RegSetValueExW(hKey, pRegistryEntry->pszValueName, 0, pRegistryEntry->dwValueType, lpData, size));
RegCloseKey(hKey);
}
return hr;
}
/**
* Registers this COM server.
*/
STDAPI DllRegisterServer()
{
HRESULT hr;
WCHAR szModuleName[MAX_PATH];
if (!GetModuleFileNameW(g_hInst, szModuleName, ARRAYSIZE(szModuleName)))
{
hr = HRESULT_FROM_WIN32(GetLastError());
}
else
{
const REGISTRY_ENTRY rgRegistryEntries[] = {
/* `RootKey KeyName ValueName ValueType Data` */
{HKEY_CURRENT_USER, L"Software\\Classes\\CLSID\\" SZ_CLSID_AYATHUMBHANDLER, nullptr, REG_SZ, SZ_AYATHUMBHANDLER},
{HKEY_CURRENT_USER,
L"Software\\Classes\\CLSID"
L"\\" SZ_CLSID_AYATHUMBHANDLER L"\\InProcServer32",
nullptr, REG_SZ, szModuleName},
{HKEY_CURRENT_USER,
L"Software\\Classes\\CLSID"
L"\\" SZ_CLSID_AYATHUMBHANDLER L"\\InProcServer32",
L"ThreadingModel", REG_SZ, L"Apartment"},
{HKEY_CURRENT_USER, L"Software\\Classes\\.ayal\\", L"Treatment", REG_DWORD, 0, 0}, /* This doesn't appear to do anything. */
{HKEY_CURRENT_USER,
L"Software\\Classes\\.ayal\\ShellEx\\{e357fccd-a995-4576-b01f-"
L"234630154e96}",
nullptr, REG_SZ, SZ_CLSID_AYATHUMBHANDLER},
};
hr = S_OK;
for (int i = 0; i < ARRAYSIZE(rgRegistryEntries) && SUCCEEDED(hr); i++)
{
hr = CreateRegKeyAndSetValue(&rgRegistryEntries[i]);
}
}
if (SUCCEEDED(hr))
{
/* This tells the shell to invalidate the thumbnail cache.
* This is important because any `.ayal` files viewed before registering
* this handler would otherwise show cached blank thumbnails. */
SHChangeNotify(SHCNE_ASSOCCHANGED, SHCNF_IDLIST, nullptr, nullptr);
}
return hr;
}
/**
* Unregisters this COM server
*/
STDAPI DllUnregisterServer()
{
HRESULT hr = S_OK;
const PCWSTR rgpszKeys[] = {L"Software\\Classes\\CLSID\\" SZ_CLSID_AYATHUMBHANDLER, L"Software\\Classes\\.ayal\\ShellEx\\{e357fccd-a995-4576-b01f-"
L"234630154e96}"};
/* Delete the registry entries. */
for (int i = 0; i < ARRAYSIZE(rgpszKeys) && SUCCEEDED(hr); i++)
{
hr = HRESULT_FROM_WIN32(RegDeleteTreeW(HKEY_CURRENT_USER, rgpszKeys[i]));
if (hr == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND))
{
/* If the registry entry has already been deleted, say S_OK. */
hr = S_OK;
}
}
return hr;
}