winamp/Src/Plugins/Library/ml_online/Setup/setupServicePanel.cpp

798 lines
19 KiB
C++
Raw Normal View History

2024-09-24 12:54:57 +00:00
#include "../common.h"
#include "./setupServicePanel.h"
#include "./setupDetails.h"
#include "./setupPage.h"
#include "../resource.h"
#include "../api__ml_online.h"
#include <ifc_omservice.h>
#include <ifc_omservicedetails.h>
#include <ifc_omcachemanager.h>
#include <ifc_omcachegroup.h>
#include <ifc_omcacherecord.h>
#include <ifc_imageloader.h>
#include <ifc_omgraphics.h>
#include <ifc_omserviceeventmngr.h>
#include <ifc_omserviceeditor.h>
#include <shlwapi.h>
#include <strsafe.h>
#define GetPanel(__hwnd) ((ServicePanel*)GetPropW((__hwnd), MAKEINTATOM(DETAILS_PROP)))
#define GET_IDETAILS(__service, __details)\
(NULL != (service) && SUCCEEDED((service)->QueryInterface(IFC_OmServiceDetails, (void**)&(__details))))
ServicePanel::ServicePanel(LPCWSTR pszName, ifc_omservice *service)
: ref(1), name(NULL), service(NULL), hwnd(NULL), fontTitle(NULL), fontMeta(NULL), thumbnailCache(NULL)
{
name = Plugin_CopyString(pszName);
this->service = service;
if (NULL != service)
service->AddRef();
}
ServicePanel::~ServicePanel()
{
Plugin_FreeString(name);
if (NULL != service)
service->Release();
if (NULL != fontTitle)
DeleteObject(fontTitle);
if (NULL != fontMeta)
DeleteObject(fontMeta);
if (NULL != thumbnailCache)
thumbnailCache->Release();
}
HWND ServicePanel::CreateInstance(HWND hParent, LPCWSTR pszName, ifc_omservice *service, ServicePanel **instance)
{
ServicePanel *panel = new ServicePanel(pszName, service);
if (NULL == panel)
{
if (NULL != instance) *instance = NULL;
return NULL;
}
HWND hwnd = WASABI_API_CREATEDIALOGPARAMW(IDD_SETUP_SERVICEDETAILS, hParent, ServicePanel_DialogProc, (LPARAM)panel);
if (NULL != instance)
{
if (NULL != hwnd)
{
*instance = panel;
panel->AddRef();
}
else
*instance = NULL;
}
panel->Release();
return hwnd;
}
size_t ServicePanel::AddRef()
{
return InterlockedIncrement((LONG*)&ref);
}
size_t ServicePanel::Release()
{
if (0 == ref)
return ref;
LONG r = InterlockedDecrement((LONG*)&ref);
if (0 == r)
delete(this);
return r;
}
int ServicePanel::QueryInterface(GUID interface_guid, void **object)
{
if (NULL == object) return E_POINTER;
if (IsEqualIID(interface_guid, IFC_OmServiceEvent))
*object = static_cast<ifc_omserviceevent*>(this);
else
{
*object = NULL;
return E_NOINTERFACE;
}
if (NULL == *object)
return E_UNEXPECTED;
AddRef();
return S_OK;
}
static void CALLBACK ThreadCallback_ServiceChange(Dispatchable *instance, ULONG_PTR param1, ULONG_PTR param2)
{
ifc_omserviceevent *panel = (ifc_omserviceevent*)instance;
ifc_omservice *service = (ifc_omservice*)param1;
if (NULL != service)
{
if (NULL != panel)
panel->ServiceChange(service, (UINT)param2);
service->Release();
}
}
void ServicePanel::ServiceChange(ifc_omservice *service, unsigned int modifiedFlags)
{
DWORD currentTID = GetCurrentThreadId();
DWORD windowTID = GetWindowThreadProcessId(hwnd, NULL);
if (NULL != windowTID && currentTID != windowTID)
{
if(NULL != OMUTILITY)
{
service->AddRef();
if (FAILED(OMUTILITY->PostMainThreadCallback2(ThreadCallback_ServiceChange, (ifc_omserviceevent*)this, (ULONG_PTR)service, (ULONG_PTR)modifiedFlags)))
service->Release();
}
return;
}
if ( 0 != (ifc_omserviceeditor::modifiedName & modifiedFlags))
{
UpdateName();
HWND hPage = GetParent(hwnd);
if (NULL != hPage)
PostMessage(hPage, SPM_UPDATELIST, (WPARAM)service->GetId(), NULL);
}
if ( 0 != (ifc_omserviceeditor::modifiedDescription & modifiedFlags))
UpdateDescription();
if ( 0 != (ifc_omserviceeditor::modifiedThumbnail& modifiedFlags))
UpdateThumbnail();
if ( 0 != ((ifc_omserviceeditor::modifiedAuthorFirst |
ifc_omserviceeditor::modifiedAuthorLast |
ifc_omserviceeditor::modifiedUpdated |
ifc_omserviceeditor::modifiedPublished) & modifiedFlags))
{
UpdateMeta();
}
}
HRESULT ServicePanel::LoadLocalThumbnail(LPCWSTR pszPath)
{
HWND hThumbnail = GetDlgItem(hwnd, IDC_THUMBNAIL);
if (NULL == hThumbnail) return E_UNEXPECTED;
SendMessage(hThumbnail, WM_SETREDRAW, FALSE, 0L);
HBITMAP hBitmap = NULL;
BITMAPINFOHEADER header;
void *pixelData;
ifc_omimageloader *imageLoader;
if (SUCCEEDED(OMUTILITY->QueryImageLoader(NULL, pszPath, FALSE, &imageLoader)))
{
imageLoader->LoadBitmapEx(&hBitmap, &header, &pixelData);
imageLoader->Release();
}
if (NULL == hBitmap &&
SUCCEEDED(OMUTILITY->QueryImageLoader(WASABI_API_ORIG_HINST, MAKEINTRESOURCE(IDR_SERVICE64X64_IMAGE), FALSE, &imageLoader)))
{
imageLoader->LoadBitmapEx(&hBitmap, &header, &pixelData);
imageLoader->Release();
}
HBITMAP hTest = (HBITMAP)SendMessage(hThumbnail, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)hBitmap);
if (NULL != hTest)
DeleteObject(hTest);
if (NULL != hBitmap)
{
hTest = (HBITMAP)SendMessage(hThumbnail, STM_GETIMAGE, IMAGE_BITMAP, 0L);
if (hTest != hBitmap)
{ // this is XP and up image copy was created and alpha channel will be handled properly
DeleteObject(hBitmap);
}
else
{ // fix alpha channel
if (32 == header.biBitCount)
{
HDC hdcFixed = CreateCompatibleDC(NULL);
if (NULL != hdcFixed)
{
BITMAPINFOHEADER headerFixed;
CopyMemory(&headerFixed, &header, sizeof(BITMAPINFOHEADER));
BYTE *pixelsFixed;
INT cx = header.biWidth;
INT cy = abs(header.biHeight);
HBITMAP bitmapFixed = CreateDIBSection(NULL, (LPBITMAPINFO)&headerFixed, DIB_RGB_COLORS, (void**)&pixelsFixed, NULL, 0);
if (NULL != bitmapFixed)
{
HBITMAP bitmapOrig = (HBITMAP)SelectObject(hdcFixed, bitmapFixed);
HBRUSH hb = (HBRUSH)SendMessage(hwnd, WM_CTLCOLORDLG, (WPARAM)hdcFixed, (LPARAM)hwnd);
if (NULL == hb)
hb = GetSysColorBrush(COLOR_3DFACE);
RECT rect;
SetRect(&rect, 0, 0, cx, cy);
FillRect(hdcFixed, &rect, hb);
ifc_omgraphics *graphics;
if (SUCCEEDED(OMUTILITY->GetGraphics(&graphics)))
{
HDC hdcSrc = CreateCompatibleDC(NULL);
if (NULL != hdcSrc)
{
HBITMAP bitmapSrcOrig = (HBITMAP)SelectObject(hdcSrc, hBitmap);
BLENDFUNCTION bf;
bf.BlendOp = AC_SRC_OVER;
bf.BlendFlags = 0;
bf.SourceConstantAlpha = 0xFF;
bf.AlphaFormat = AC_SRC_ALPHA;
RECT blendRect;
SetRect(&blendRect, 0, 0, cx, cy);
graphics->Premultiply((BYTE*)pixelData, cx, cy);
graphics->AlphaBlend(hdcFixed, &blendRect, hdcSrc, &blendRect, bf);
SelectObject(hdcSrc, bitmapSrcOrig);
DeleteDC(hdcSrc);
}
graphics->Release();
}
SelectObject(hdcFixed, bitmapOrig);
SendMessage(hThumbnail, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)bitmapFixed);
DeleteObject(hBitmap);
}
DeleteDC(hdcFixed);
}
}
}
}
RECT clientRect;
if (GetClientRect(hThumbnail, &clientRect))
{
INT cx = clientRect.right - clientRect.left;
INT cy = clientRect.bottom - clientRect.top;
if (64 != cx || 64 != cy)
{
SetWindowPos(hThumbnail, NULL, 0, 0, 64, 64, SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOZORDER);
}
}
SendMessage(hThumbnail, WM_SETREDRAW, TRUE, 0L);
if (0 != ShowWindow(hThumbnail, (NULL != hBitmap) ? SW_SHOWNA : SW_HIDE))
InvalidateRect(hThumbnail, NULL, TRUE);
return S_OK;
}
static void CALLBACK ThreadCallback_PathChanged(Dispatchable *instance, ULONG_PTR param1, ULONG_PTR param2)
{
ifc_omcachecallback *panel = (ifc_omcachecallback*)instance;
ifc_omcacherecord *record = (ifc_omcacherecord*)param1;
if (NULL != record)
{
if (NULL != panel)
panel->PathChanged(record);
record->Release();
}
}
void ServicePanel::PathChanged(ifc_omcacherecord *record)
{
if (NULL == hwnd || FALSE == IsWindow(hwnd))
return;
DWORD currentTID = GetCurrentThreadId();
DWORD windowTID = GetWindowThreadProcessId(hwnd, NULL);
if (NULL != windowTID && currentTID != windowTID)
{
if(NULL != OMUTILITY)
{
record->AddRef();
if (FAILED(OMUTILITY->PostMainThreadCallback2(ThreadCallback_PathChanged, (ifc_omcachecallback*)this, (ULONG_PTR)record, 0L)))
record->Release();
}
return;
}
WCHAR szPath[2048] = {0};
if (FAILED(record->GetPath(szPath, ARRAYSIZE(szPath))))
szPath[0] = L'\0';
LoadLocalThumbnail(szPath);
}
void ServicePanel::Attach(HWND hwnd)
{
this->hwnd = hwnd;
if (NULL != hwnd &&
FALSE != SetProp(hwnd, MAKEINTATOM(DETAILS_PROP), this))
{
AddRef();
}
}
void ServicePanel::Detach()
{
RemoveProp(hwnd, MAKEINTATOM(DETAILS_PROP));
if (NULL != thumbnailCache)
{
thumbnailCache->UnregisterCallback(this);
thumbnailCache->Release();
thumbnailCache = NULL;
}
if (NULL != service)
{
ifc_omserviceeventmngr *eventManager;
if (SUCCEEDED(service->QueryInterface(IFC_OmServiceEventMngr, (void**)&eventManager)))
{
eventManager->UnregisterHandler(this);
eventManager->Release();
}
}
Release();
}
HFONT ServicePanel::PickTitleFont(LPCWSTR pszTitle, INT cchTitle, INT maxWidth)
{
HFONT dialogFont = (HFONT)SendMessage(hwnd, WM_GETFONT, 0, 0L);
LOGFONT lf;
if (0 == GetObject(dialogFont, sizeof(LOGFONT), &lf))
return NULL;
HFONT titleFont = NULL;
if (cchTitle > 0)
{
LOGFONT lf;
if (0 != GetObject(dialogFont, sizeof(LOGFONT), &lf))
{
StringCchCopy(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), L"Arial Bold");
lf.lfWidth = 0;
lf.lfWeight = FW_DONTCARE;
lf.lfQuality = 5/*ANTIALIASED_QUALITY*/;
HDC hdc = GetDCEx(hwnd, NULL, DCX_CACHE | DCX_NORESETATTRS);
if (NULL != hdc)
{
HFONT origFont = (HFONT)GetCurrentObject(hdc, OBJ_FONT);
SIZE textSize;
INT heightLimit = (lf.lfHeight < 0) ? 1 : -1;
lf.lfHeight += (lf.lfHeight < 0) ? -2 : +2;
do
{
textSize.cx = 0;
if (NULL != titleFont) DeleteObject(titleFont);
titleFont = CreateFontIndirect(&lf);
if (NULL != titleFont)
{
SelectObject(hdc, titleFont);
GetTextExtentPoint32(hdc, pszTitle, cchTitle, &textSize);
}
lf.lfHeight += (lf.lfHeight < 0) ? 1 : -1;
} while(textSize.cx > maxWidth && lf.lfHeight != heightLimit);
if (0 == textSize.cx)
{
DeleteObject(titleFont);
titleFont = NULL;
}
SelectObject(hdc, origFont);
ReleaseDC(hwnd, hdc);
}
}
}
if (NULL == titleFont &&
0 != GetObject(dialogFont, sizeof(LOGFONT), &lf))
{
titleFont = CreateFontIndirect(&lf);
}
return titleFont;
}
LPCWSTR ServicePanel::FormatDate(LPCWSTR pszDate, LPWSTR pszBuffer, INT cchBufferMax)
{
SYSTEMTIME st;
ZeroMemory(&st, sizeof(SYSTEMTIME));
LPCWSTR cursor;
cursor = pszDate;
INT index = 0;
for(;;)
{
INT iVal;
if (FALSE == StrToIntEx(cursor, STIF_DEFAULT, &iVal) || iVal < 1)
{
index = 0;
break;
}
if (0 == index)
{
if (iVal < 2000 || iVal > 2100)
break;
st.wYear = iVal;
index++;
}
else if (1 == index)
{
if (iVal < 1 || iVal > 12)
break;
st.wMonth = iVal;
index++;
}
else if (2 == index)
{
if (iVal < 1 || iVal > 31)
break;
st.wDay = iVal;
index++;
}
else
{
index = 0;
break;
}
while(L'\0' != *cursor && L'-' != *cursor) cursor++;
if (L'-' == *cursor) cursor++;
if (L'\0' == *cursor)
break;
}
if (3 == index &&
0 != GetDateFormat(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &st, NULL, pszBuffer, cchBufferMax))
{
return pszBuffer;
}
return pszDate;
}
HRESULT ServicePanel::GetFullName(LPWSTR pszBuffer, UINT cchBufferMax)
{
if (NULL == pszBuffer)
return E_POINTER;
*pszBuffer = L'\0';
if (NULL == service) return E_UNEXPECTED;
ifc_omservicedetails *details;
HRESULT hr = service->QueryInterface(IFC_OmServiceDetails, (void**)&details);
if (SUCCEEDED(hr))
{
hr = details->GetAuthorFirst(pszBuffer, cchBufferMax);
if (SUCCEEDED(hr))
{
UINT cchBuffer = lstrlen(pszBuffer);
LPWSTR cursor = pszBuffer + cchBuffer;
size_t remaining = cchBufferMax - cchBuffer;
if (cursor != pszBuffer)
{
hr = StringCchCopyEx(cursor, remaining, L" ", &cursor, &remaining, 0);
if (SUCCEEDED(hr))
{
hr = details->GetAuthorLast(cursor, (UINT)remaining);
if (FAILED(hr) || L'\0' == *cursor)
{
pszBuffer[cchBuffer] = L'\0';
}
}
}
}
}
return hr;
}
void ServicePanel::UpdateName()
{
HWND hTitle = GetDlgItem(hwnd, IDC_TITLE);
if (NULL == hTitle) return;
WCHAR szBuffer[128] = {0};
if (NULL == service ||
FAILED(service->GetName(szBuffer, ARRAYSIZE(szBuffer))))
{
szBuffer[0] = L'\0';
}
INT cchBuffer = lstrlen(szBuffer);
RECT rc;
GetClientRect(hTitle, &rc);
HFONT font = PickTitleFont(szBuffer, cchBuffer, rc.right - rc.left);
if (NULL != font)
{
if (NULL != fontTitle)
DeleteObject(fontTitle);
fontTitle = font;
SendMessage(hTitle, WM_SETFONT, (WPARAM)fontTitle, 0L);
}
SetWindowText(hTitle, szBuffer);
InvalidateRect(hTitle, NULL, TRUE);
}
void ServicePanel::UpdateDescription()
{
HWND hDescription = GetDlgItem(hwnd, IDC_DESCRIPTION);
if (NULL == hDescription) return;
WCHAR szBuffer[4096] = {0};
ifc_omservicedetails *details = 0;
if (GET_IDETAILS(service, details))
{
details->GetDescription(szBuffer, ARRAYSIZE(szBuffer));
details->Release();
}
SetupDetails_SetDescription(hDescription, szBuffer);
}
void ServicePanel::UpdateMeta()
{
HWND hMeta = GetDlgItem(hwnd, IDC_SERVICEMETA);
if (NULL == hMeta) return;
WCHAR szBuffer[512] = {0};
ifc_omservicedetails *svcdetails = 0;
if (GET_IDETAILS(service, svcdetails))
{
WCHAR szValue[256] = {0}, szPrefix[64] = {0};
HRESULT hr = S_OK;
LPWSTR cursor = szBuffer;
size_t remaining = ARRAYSIZE(szBuffer);
if (SUCCEEDED(GetFullName(szValue, ARRAYSIZE(szValue))) && L'\0' != szValue[0])
{
WASABI_API_LNGSTRINGW_BUF(IDS_SERVICE_BYAUTHOR, szPrefix, ARRAYSIZE(szPrefix));
hr = StringCchPrintfEx(cursor, remaining, &cursor, &remaining, STRSAFE_NULL_ON_FAILURE,
L"%s%s", szPrefix, szValue);
}
if (SUCCEEDED(svcdetails->GetUpdated(szValue, ARRAYSIZE(szValue))) && L'\0' != szValue[0])
{
if (cursor != szBuffer)
hr = StringCchCopyEx(cursor, remaining, L"\r\n", &cursor, &remaining, STRSAFE_NULL_ON_FAILURE);
if (SUCCEEDED(hr))
{
WCHAR szDate[128] = {0};
WASABI_API_LNGSTRINGW_BUF(IDS_SERVICE_LASTUPDATED, szPrefix, ARRAYSIZE(szPrefix));
StringCchPrintfEx(cursor, remaining, &cursor, &remaining, STRSAFE_NULL_ON_FAILURE,
L"%s%s", szPrefix, FormatDate(szValue, szDate, ARRAYSIZE(szDate)));
}
}
svcdetails->Release();
}
if (NULL == fontMeta)
{
HFONT dialogFont = (HFONT)SendMessage(hwnd, WM_GETFONT, 0, 0L);
LOGFONT lf;
if (0 != GetObject(dialogFont, sizeof(LOGFONT), &lf))
{
StringCchCopy(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), L"Tahoma");
lf.lfWidth = 0;
lf.lfHeight += (lf.lfHeight < 0) ? 1 : -1;
lf.lfQuality = ANTIALIASED_QUALITY;
fontMeta = CreateFontIndirect(&lf);
}
if (NULL != fontMeta)
{
SendMessage(hMeta, WM_SETFONT, (WPARAM)fontMeta, 0L);
}
}
SetWindowText(hMeta, szBuffer);
if (0 != ShowWindow(hMeta, (L'\0' != szBuffer[0]) ? SW_SHOWNA : SW_HIDE))
InvalidateRect(hMeta, NULL, TRUE);
}
void ServicePanel::UpdateThumbnail()
{
if (NULL != thumbnailCache)
{
thumbnailCache->UnregisterCallback(this);
thumbnailCache->Release();
thumbnailCache = NULL;
}
LoadLocalThumbnail(NULL);
ifc_omservicedetails *details = 0;
if (GET_IDETAILS(service, details))
{
WCHAR szPath[2048] = {0};
if (SUCCEEDED(details->GetThumbnail(szPath, ARRAYSIZE(szPath))) && L'\0' != szPath[0])
{
ifc_omcachemanager *cacheManager;
if (SUCCEEDED(OMUTILITY->GetCacheManager(&cacheManager)))
{
ifc_omcachegroup *cacheGroup;
if (SUCCEEDED(cacheManager->Find(L"thumbnails", TRUE, &cacheGroup, NULL)))
{
if (SUCCEEDED(cacheGroup->Find(szPath, TRUE, &thumbnailCache, FALSE)))
{
thumbnailCache->RegisterCallback(this);
if (SUCCEEDED(thumbnailCache->GetPath(szPath, ARRAYSIZE(szPath))))
LoadLocalThumbnail(szPath);
}
cacheGroup->Release();
}
cacheManager->Release();
}
}
details->Release();
}
}
INT_PTR ServicePanel::OnInitDialog(HWND hFocus, LPARAM lParam)
{
UpdateName();
UpdateDescription();
UpdateThumbnail();
UpdateMeta();
if (NULL != service)
{
ifc_omserviceeventmngr *eventManager;
if (SUCCEEDED(service->QueryInterface(IFC_OmServiceEventMngr, (void**)&eventManager)))
{
eventManager->RegisterHandler(this);
eventManager->Release();
}
}
return FALSE;
}
void ServicePanel::OnDestroy()
{
HWND hThumbnail = GetDlgItem(hwnd, IDC_THUMBNAIL);
if (NULL != hThumbnail)
{
HBITMAP hBitmap = (HBITMAP)SendMessage(hThumbnail, STM_SETIMAGE, IMAGE_BITMAP, 0L);
if (NULL != hBitmap)
DeleteObject(hBitmap);
}
Detach();
}
INT_PTR ServicePanel::OnDialogColor(HDC hdc, HWND hControl)
{
HWND hParent = GetAncestor(hwnd, GA_PARENT);
if (NULL != hParent && hParent != hwnd)
return (INT_PTR)SendMessage(hParent, WM_CTLCOLORDLG, (WPARAM)hdc, (LPARAM)hControl);
return 0;
}
INT_PTR ServicePanel::OnStaticColor(HDC hdc, HWND hControl)
{
INT_PTR result = 0;
HWND hParent = GetAncestor(hwnd, GA_PARENT);
if (NULL != hParent && hParent != hwnd)
result = (INT_PTR)SendMessage(hParent, WM_CTLCOLORSTATIC, (WPARAM)hdc, (LPARAM)hControl);
INT controlId = GetDlgCtrlID(hControl);
switch(controlId)
{
case IDC_SERVICEMETA:
{
COLORREF rgbBk = GetBkColor(hdc);
COLORREF rgbFg = GetTextColor(hdc);
ifc_omgraphics *graphics;
if (SUCCEEDED(OMUTILITY->GetGraphics(&graphics)))
{
graphics->BlendColor(rgbFg, rgbBk, 180, &rgbFg);
graphics->Release();
}
SetTextColor(hdc, rgbFg);
}
break;
}
return result;
}
INT_PTR ServicePanel::OnGetUniqueName(LPWSTR pszBuffer, UINT cchBufferMax)
{
if (NULL == pszBuffer) return FALSE;
return SUCCEEDED(StringCchCopy(pszBuffer, cchBufferMax, (NULL != name) ? name : L""));
}
INT_PTR ServicePanel::DialogProc(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch(uMsg)
{
case WM_INITDIALOG: return OnInitDialog((HWND)wParam, lParam);
case WM_DESTROY: OnDestroy(); break;
case WM_CTLCOLORDLG: return OnDialogColor((HDC)wParam, (HWND)lParam);
case WM_CTLCOLORSTATIC: return OnStaticColor((HDC)wParam, (HWND)lParam);
case NSDM_GETUNIQUENAME: MSGRESULT(hwnd, OnGetUniqueName((LPWSTR)lParam, (UINT)wParam));
}
return 0;
}
static INT_PTR WINAPI ServicePanel_DialogProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
ServicePanel *panel = GetPanel(hwnd);
if (NULL == panel)
{
if (WM_INITDIALOG == uMsg)
{
panel = (ServicePanel*)lParam;
if (NULL != panel)
panel->Attach(hwnd);
}
if (NULL == panel) return 0;
}
return panel->DialogProc(uMsg, wParam, lParam);
}
#define CBCLASS ServicePanel
START_MULTIPATCH;
START_PATCH(MPIID_SERVICEEVENT)
M_CB(MPIID_SERVICEEVENT, ifc_omserviceevent, ADDREF, AddRef);
M_CB(MPIID_SERVICEEVENT, ifc_omserviceevent, RELEASE, Release);
M_CB(MPIID_SERVICEEVENT, ifc_omserviceevent, QUERYINTERFACE, QueryInterface);
M_VCB(MPIID_SERVICEEVENT, ifc_omserviceevent, API_SERVICECHANGE, ServiceChange);
NEXT_PATCH(MPIID_CACHECALLBACK)
M_CB(MPIID_CACHECALLBACK, ifc_omcachecallback, ADDREF, AddRef);
M_CB(MPIID_CACHECALLBACK, ifc_omcachecallback, RELEASE, Release);
M_CB(MPIID_CACHECALLBACK, ifc_omcachecallback, QUERYINTERFACE, QueryInterface);
M_VCB(MPIID_CACHECALLBACK, ifc_omcachecallback, API_PATHCHANGED, PathChanged);
END_PATCH
END_MULTIPATCH;
#undef CBCLASS