2023-08-03 14:13:27 -04:00
|
|
|
#include "mxdisplaysurface.h"
|
2023-10-07 11:30:04 -04:00
|
|
|
|
2023-08-28 05:49:15 -04:00
|
|
|
#include "mxomni.h"
|
2023-10-07 11:30:04 -04:00
|
|
|
#include "mxvideomanager.h"
|
2023-08-03 14:13:27 -04:00
|
|
|
|
|
|
|
DECOMP_SIZE_ASSERT(MxDisplaySurface, 0xac);
|
|
|
|
|
2023-12-26 17:20:20 -05:00
|
|
|
MxU32 g_unk0x1010215c = 0;
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba500
|
2023-08-03 14:13:27 -04:00
|
|
|
MxDisplaySurface::MxDisplaySurface()
|
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
this->Reset();
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba5a0
|
2023-08-03 14:13:27 -04:00
|
|
|
MxDisplaySurface::~MxDisplaySurface()
|
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
this->Clear();
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba610
|
2023-08-03 14:13:27 -04:00
|
|
|
void MxDisplaySurface::Reset()
|
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
this->m_ddSurface1 = NULL;
|
|
|
|
this->m_ddSurface2 = NULL;
|
|
|
|
this->m_ddClipper = NULL;
|
|
|
|
this->m_16bitPal = NULL;
|
|
|
|
this->m_initialized = FALSE;
|
|
|
|
memset(&this->m_surfaceDesc, 0, sizeof(this->m_surfaceDesc));
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-21 10:44:54 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba640
|
2023-11-21 03:44:45 -05:00
|
|
|
void MxDisplaySurface::FUN_100ba640()
|
|
|
|
{
|
2023-12-21 10:44:54 -05:00
|
|
|
MxS32 backBuffers;
|
|
|
|
DDSURFACEDESC desc;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
if (!m_videoParam.Flags().GetFlipSurfaces()) {
|
|
|
|
backBuffers = 1;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
backBuffers = m_videoParam.GetBackBuffers() + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (MxS32 i = 0; i < backBuffers; i++) {
|
|
|
|
memset(&desc, 0, sizeof(DDSURFACEDESC));
|
|
|
|
|
|
|
|
desc.dwSize = sizeof(DDSURFACEDESC);
|
|
|
|
hr = m_ddSurface2->Lock(NULL, &desc, DDLOCK_WAIT, NULL);
|
|
|
|
if (hr == DDERR_SURFACELOST) {
|
|
|
|
m_ddSurface2->Restore();
|
|
|
|
hr = m_ddSurface2->Lock(NULL, &desc, DDLOCK_WAIT, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hr != S_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
MxU8* surface = (MxU8*) desc.lpSurface;
|
|
|
|
MxS32 height = m_videoParam.GetRect().GetHeight();
|
|
|
|
|
|
|
|
while (height--) {
|
|
|
|
memset(surface, 0, m_videoParam.GetRect().GetWidth() * desc.ddpfPixelFormat.dwRGBBitCount / 8);
|
|
|
|
surface += desc.lPitch;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_ddSurface2->Unlock(desc.lpSurface);
|
|
|
|
if (m_videoParam.Flags().GetFlipSurfaces()) {
|
|
|
|
m_ddSurface1->Flip(NULL, 1);
|
|
|
|
}
|
|
|
|
}
|
2023-11-21 03:44:45 -05:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba790
|
2023-10-24 19:38:27 -04:00
|
|
|
MxResult MxDisplaySurface::Init(
|
|
|
|
MxVideoParam& p_videoParam,
|
|
|
|
LPDIRECTDRAWSURFACE p_ddSurface1,
|
|
|
|
LPDIRECTDRAWSURFACE p_ddSurface2,
|
|
|
|
LPDIRECTDRAWCLIPPER p_ddClipper
|
|
|
|
)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
MxResult result = SUCCESS;
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
this->m_videoParam = p_videoParam;
|
|
|
|
this->m_ddSurface1 = p_ddSurface1;
|
|
|
|
this->m_ddSurface2 = p_ddSurface2;
|
|
|
|
this->m_ddClipper = p_ddClipper;
|
|
|
|
this->m_initialized = FALSE;
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
memset(&this->m_surfaceDesc, 0, sizeof(this->m_surfaceDesc));
|
|
|
|
this->m_surfaceDesc.dwSize = sizeof(this->m_surfaceDesc);
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_ddSurface2->GetSurfaceDesc(&this->m_surfaceDesc))
|
|
|
|
result = FAILURE;
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
return result;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100ba7f0
|
2023-10-24 19:38:27 -04:00
|
|
|
MxResult MxDisplaySurface::Create(MxVideoParam& p_videoParam)
|
|
|
|
{
|
|
|
|
DDSURFACEDESC ddsd;
|
|
|
|
MxResult result = FAILURE;
|
|
|
|
LPDIRECTDRAW lpDirectDraw = MVideoManager()->GetDirectDraw();
|
|
|
|
HWND hWnd = MxOmni::GetInstance()->GetWindowHandle();
|
|
|
|
|
|
|
|
this->m_initialized = TRUE;
|
|
|
|
this->m_videoParam = p_videoParam;
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
if (!this->m_videoParam.Flags().GetFullScreen())
|
|
|
|
this->m_videoParam.Flags().SetFlipSurfaces(FALSE);
|
2023-10-24 19:38:27 -04:00
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
if (!this->m_videoParam.Flags().GetFlipSurfaces()) {
|
2023-10-24 19:38:27 -04:00
|
|
|
this->m_videoParam.SetBackBuffers(1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
MxU32 backBuffers = this->m_videoParam.GetBackBuffers();
|
|
|
|
|
|
|
|
if (backBuffers < 1)
|
|
|
|
this->m_videoParam.SetBackBuffers(1);
|
|
|
|
else if (backBuffers > 2)
|
|
|
|
this->m_videoParam.SetBackBuffers(2);
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
this->m_videoParam.Flags().SetBackBuffers(TRUE);
|
2023-10-24 19:38:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
if (this->m_videoParam.Flags().GetFullScreen()) {
|
2023-11-06 18:12:09 -05:00
|
|
|
MxS32 width = this->m_videoParam.GetRect().GetWidth();
|
|
|
|
MxS32 height = this->m_videoParam.GetRect().GetHeight();
|
2023-10-24 19:38:27 -04:00
|
|
|
|
|
|
|
if (lpDirectDraw->SetCooperativeLevel(hWnd, DDSCL_EXCLUSIVE | DDSCL_FULLSCREEN))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
|
|
|
|
if (lpDirectDraw->GetDisplayMode(&ddsd))
|
|
|
|
goto done;
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
MxS32 bitdepth = !this->m_videoParam.Flags().Get16Bit() ? 8 : 16;
|
2023-10-24 19:38:27 -04:00
|
|
|
|
|
|
|
if (ddsd.dwWidth != width || ddsd.dwHeight != height || ddsd.ddpfPixelFormat.dwRGBBitCount != bitdepth) {
|
|
|
|
if (lpDirectDraw->SetDisplayMode(width, height, bitdepth))
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
if (this->m_videoParam.Flags().GetFlipSurfaces()) {
|
2023-10-24 19:38:27 -04:00
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
ddsd.dwBackBufferCount = this->m_videoParam.GetBackBuffers();
|
|
|
|
ddsd.dwFlags = DDSD_CAPS | DDSD_BACKBUFFERCOUNT;
|
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_3DDEVICE | DDSCAPS_PRIMARYSURFACE | DDSCAPS_FLIP | DDSCAPS_COMPLEX;
|
|
|
|
|
|
|
|
if (lpDirectDraw->CreateSurface(&ddsd, &this->m_ddSurface1, NULL))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_BACKBUFFER;
|
|
|
|
|
|
|
|
if (this->m_ddSurface1->GetAttachedSurface(&ddsd.ddsCaps, &this->m_ddSurface2))
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
ddsd.dwFlags = DDSD_CAPS;
|
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
|
|
|
|
|
|
|
|
if (lpDirectDraw->CreateSurface(&ddsd, &this->m_ddSurface1, NULL))
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
ddsd.dwFlags = DDSD_HEIGHT | DDSD_WIDTH | DDSD_CAPS;
|
2023-11-06 18:12:09 -05:00
|
|
|
ddsd.dwWidth = this->m_videoParam.GetRect().GetWidth();
|
|
|
|
ddsd.dwHeight = this->m_videoParam.GetRect().GetHeight();
|
2023-10-24 19:38:27 -04:00
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_VIDEOMEMORY | DDSCAPS_3DDEVICE | DDSCAPS_OFFSCREENPLAIN;
|
|
|
|
|
2023-12-13 05:48:14 -05:00
|
|
|
if (!this->m_videoParam.Flags().GetBackBuffers())
|
2023-10-24 19:38:27 -04:00
|
|
|
ddsd.ddsCaps.dwCaps = DDSCAPS_3DDEVICE | DDSCAPS_SYSTEMMEMORY | DDSCAPS_OFFSCREENPLAIN;
|
|
|
|
|
|
|
|
if (lpDirectDraw->CreateSurface(&ddsd, &this->m_ddSurface2, NULL))
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(&this->m_surfaceDesc, 0, sizeof(this->m_surfaceDesc));
|
|
|
|
this->m_surfaceDesc.dwSize = sizeof(this->m_surfaceDesc);
|
|
|
|
|
|
|
|
if (!this->m_ddSurface2->GetSurfaceDesc(&this->m_surfaceDesc)) {
|
|
|
|
if (!lpDirectDraw->CreateClipper(0, &this->m_ddClipper, NULL) && !this->m_ddClipper->SetHWnd(0, hWnd) &&
|
|
|
|
!this->m_ddSurface1->SetClipper(this->m_ddClipper))
|
|
|
|
result = SUCCESS;
|
|
|
|
}
|
2023-08-28 05:49:15 -04:00
|
|
|
|
|
|
|
done:
|
2023-10-24 19:38:27 -04:00
|
|
|
return result;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100baa90
|
2023-08-03 14:13:27 -04:00
|
|
|
void MxDisplaySurface::Clear()
|
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_initialized) {
|
|
|
|
if (this->m_ddSurface2)
|
|
|
|
this->m_ddSurface2->Release();
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_ddSurface1)
|
|
|
|
this->m_ddSurface1->Release();
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_ddClipper)
|
|
|
|
this->m_ddClipper->Release();
|
|
|
|
}
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_16bitPal)
|
|
|
|
delete this->m_16bitPal;
|
2023-08-03 14:13:27 -04:00
|
|
|
|
2023-10-24 19:38:27 -04:00
|
|
|
this->Reset();
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100baae0
|
2023-10-24 19:38:27 -04:00
|
|
|
void MxDisplaySurface::SetPalette(MxPalette* p_palette)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bacc0
|
2023-12-29 16:30:17 -05:00
|
|
|
MxBool MxDisplaySurface::VTable0x28(
|
|
|
|
MxBitmap* p_bitmap,
|
|
|
|
MxS32 p_left,
|
|
|
|
MxS32 p_top,
|
|
|
|
MxS32 p_right,
|
|
|
|
MxS32 p_bottom,
|
|
|
|
MxS32 p_width,
|
|
|
|
MxS32 p_height
|
|
|
|
)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
return 0;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bb1d0
|
2023-12-13 05:48:14 -05:00
|
|
|
MxBool MxDisplaySurface::VTable0x30(
|
2023-12-29 16:30:17 -05:00
|
|
|
MxBitmap* p_bitmap,
|
|
|
|
MxS32 p_left,
|
|
|
|
MxS32 p_top,
|
|
|
|
MxS32 p_right,
|
|
|
|
MxS32 p_bottom,
|
|
|
|
MxS32 p_width,
|
|
|
|
MxS32 p_height,
|
2023-10-24 19:38:27 -04:00
|
|
|
MxBool
|
|
|
|
)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
return 0;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bb850
|
2023-12-13 05:48:14 -05:00
|
|
|
undefined4 MxDisplaySurface::VTable0x34(undefined4, undefined4, undefined4, undefined4, undefined4, undefined4)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
return 0;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-26 17:20:20 -05:00
|
|
|
// FUNCTION: LEGO1 0x100bba50
|
|
|
|
void MxDisplaySurface::Display(MxS32 p_left, MxS32 p_top, MxS32 p_left2, MxS32 p_top2, MxS32 p_width, MxS32 p_height)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-12-26 17:20:20 -05:00
|
|
|
if (m_videoParam.Flags().GetF2bit1()) {
|
|
|
|
if (m_videoParam.Flags().GetFlipSurfaces()) {
|
|
|
|
if (g_unk0x1010215c < 2) {
|
|
|
|
g_unk0x1010215c++;
|
|
|
|
|
|
|
|
DDSURFACEDESC ddsd;
|
|
|
|
memset(&ddsd, 0, sizeof(ddsd));
|
|
|
|
ddsd.dwSize = sizeof(ddsd);
|
|
|
|
if (m_ddSurface2->Lock(NULL, &ddsd, 1, NULL) == S_OK) {
|
|
|
|
MxU8* surface = (MxU8*) ddsd.lpSurface;
|
|
|
|
MxS32 height = m_videoParam.GetRect().GetHeight();
|
|
|
|
|
|
|
|
for (MxU32 i = 0; i < ddsd.dwHeight; i++) {
|
|
|
|
memset(surface, 0, ddsd.dwWidth * ddsd.ddpfPixelFormat.dwRGBBitCount / 8);
|
|
|
|
surface += ddsd.lPitch;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_ddSurface2->Unlock(ddsd.lpSurface);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
OutputDebugString("MxDisplaySurface::Display error\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_ddSurface1->Flip(NULL, 1);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
POINT point = {0, 0};
|
|
|
|
ClientToScreen(MxOmni::GetInstance()->GetWindowHandle(), &point);
|
|
|
|
|
|
|
|
// TODO: Match
|
|
|
|
RECT rect1, rect2;
|
|
|
|
rect1.left = p_left2 + m_videoParam.GetRect().GetLeft() + point.x;
|
|
|
|
rect2.left = p_left;
|
|
|
|
rect1.top = p_top2 + m_videoParam.GetRect().GetTop() + point.y;
|
|
|
|
rect2.right = p_left + p_width;
|
|
|
|
rect2.top = p_top;
|
|
|
|
rect2.bottom = p_top + p_height;
|
|
|
|
rect1.right = rect1.left + p_width;
|
|
|
|
rect1.bottom = rect1.top + p_height;
|
|
|
|
|
|
|
|
DDBLTFX data;
|
|
|
|
memset(&data, 0, sizeof(data));
|
|
|
|
data.dwSize = sizeof(data);
|
|
|
|
data.dwDDFX = 8;
|
|
|
|
|
|
|
|
if (m_ddSurface1->Blt(&rect1, m_ddSurface2, &rect2, 0, &data) == DDERR_SURFACELOST) {
|
|
|
|
m_ddSurface1->Restore();
|
|
|
|
m_ddSurface1->Blt(&rect1, m_ddSurface2, &rect2, 0, &data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100bbc10
|
2023-10-24 19:38:27 -04:00
|
|
|
void MxDisplaySurface::GetDC(HDC* p_hdc)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_ddSurface2 && !this->m_ddSurface2->GetDC(p_hdc))
|
|
|
|
return;
|
|
|
|
|
|
|
|
*p_hdc = NULL;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// FUNCTION: LEGO1 0x100bbc40
|
2023-08-03 14:13:27 -04:00
|
|
|
void MxDisplaySurface::ReleaseDC(HDC p_hdc)
|
|
|
|
{
|
2023-10-24 19:38:27 -04:00
|
|
|
if (this->m_ddSurface2 && p_hdc)
|
|
|
|
this->m_ddSurface2->ReleaseDC(p_hdc);
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bbc60
|
2023-12-13 05:48:14 -05:00
|
|
|
LPDIRECTDRAWSURFACE MxDisplaySurface::VTable0x44(MxBitmap*, undefined4*, undefined4, undefined4)
|
2023-08-03 14:13:27 -04:00
|
|
|
{
|
2023-12-01 05:59:32 -05:00
|
|
|
return NULL;
|
2023-08-03 14:13:27 -04:00
|
|
|
}
|
2023-10-05 18:51:34 -04:00
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bc200
|
2023-12-13 05:48:14 -05:00
|
|
|
void MxDisplaySurface::VTable0x24(
|
2023-11-21 03:44:45 -05:00
|
|
|
LPDDSURFACEDESC,
|
|
|
|
MxBitmap*,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4
|
|
|
|
)
|
2023-10-05 18:51:34 -04:00
|
|
|
{
|
2023-11-21 03:44:45 -05:00
|
|
|
}
|
|
|
|
|
2023-12-06 07:10:45 -05:00
|
|
|
// STUB: LEGO1 0x100bc630
|
2023-12-13 05:48:14 -05:00
|
|
|
MxBool MxDisplaySurface::VTable0x2c(
|
|
|
|
LPDDSURFACEDESC,
|
|
|
|
MxBitmap*,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
undefined4,
|
|
|
|
MxBool
|
|
|
|
)
|
2023-11-21 03:44:45 -05:00
|
|
|
{
|
|
|
|
return 0;
|
2023-10-05 18:51:34 -04:00
|
|
|
}
|