/*==LICENSE==*

CyanWorlds.com Engine - MMOG client, server and tools
Copyright (C) 2011  Cyan Worlds, Inc.

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.

Additional permissions under GNU GPL version 3 section 7

If you modify this Program, or any covered work, by linking or
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK,
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK
(or a modified version of those libraries),
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA,
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the
licensors of this Program grant you additional
permission to convey the resulting work. Corresponding Source for a
non-source form of such a combination shall include the source code for
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered
work.

You can contact Cyan Worlds, Inc. by email legal@cyan.com
 or by snail mail at:
      Cyan Worlds, Inc.
      14617 N Newport Hwy
      Mead, WA   99021

*==LICENSE==*/

#include "HeadSpin.h"
#include "hsTemplates.h"
#include "hsWindows.h"

#include <max.h>
#include <iparamm2.h>

#include "MaxMain/MaxCompat.h"
#include "MaxMain/plMaxNodeBase.h"
#include "resource.h"
#pragma hdrstop

#include "pfGUISkinComp.h"
#include "plGUICompClassIDs.h"


pfGUISkinEditProc   *pfGUISkinEditProc::fInstance = nil;
int                 pfGUISkinEditProc::fZoom = 3;       // So re-opening the dialog will keep the same zoom level

extern HINSTANCE hInstance;

pfGUISkinEditProc::pfGUISkinEditProc( plGUISkinComp *comp ) 
{
    fInstance = this; 
    fComp = comp;

    fDblDC = nil;
    fDblBitmap = nil;

    fXOffset = fYOffset = 0;

    fCurrPBRefSet = plGUISkinComp::kRefUpLeftCorner;

    fDefPen = CreatePen( PS_SOLID, 3, RGB( 200, 0, 0 ) );
    fOtherPen = CreatePen( PS_DOT, 1, RGB( 200, 0, 0 ) );

    fDragging = false;
    fDragTimer = 0;

    // Back up the old settings
    IParamBlock2 *pb = fComp->GetParamBlockByID( plComponent::kBlkComp );
    for( int i = 0; i < pfGUISkin::kNumElements; i++ )
    {
        int id = i * 4 + plGUISkinComp::kRefUpLeftCorner;
        fBackups[ i ].fX = pb->GetInt( id + 0 ); 
        fBackups[ i ].fY = pb->GetInt( id + 1 ); 
        fBackups[ i ].fWidth = pb->GetInt( id + 2 ); 
        fBackups[ i ].fHeight = pb->GetInt( id + 3 ); 
    }
}

pfGUISkinEditProc::~pfGUISkinEditProc() 
{
    fInstance = nil;
    DeleteObject( fDefPen );
    IKillDblBuffer();
}

void    pfGUISkinEditProc::IJustDrawOneRect( int whichElement, IParamBlock2 *pb, HDC hDC, HPEN whichPen, int refToIgnore )
{
    whichElement = ( whichElement * 4 ) + plGUISkinComp::kRefUpLeftCorner;

    if( whichElement == refToIgnore )
        return;

    RECT    r;
    SetRect( &r, pb->GetInt( whichElement + 0 ),
                 pb->GetInt( whichElement + 1 ),
                 pb->GetInt( whichElement + 0 ) + pb->GetInt( whichElement + 2 ),
                 pb->GetInt( whichElement + 1 ) + pb->GetInt( whichElement + 3 ) );
    
    r.left *= fZoom; r.right *= fZoom; r.top *= fZoom; r.bottom *= fZoom;

    SelectObject( hDC, whichPen );
    int rop2 = SetROP2( hDC, R2_NOTXORPEN );

    MoveToEx( hDC, r.left, r.top, nil );
    LineTo( hDC, r.right, r.top );
    LineTo( hDC, r.right, r.bottom );
    LineTo( hDC, r.left, r.bottom );
    LineTo( hDC, r.left, r.top );

    SetROP2( hDC, rop2 );
}

void    pfGUISkinEditProc::IRefreshDblBuffer( void )
{
    // Image buffer is where we keep our resized image. Dbl buffer is where we draw our bounds
    if( fDblDC == nil )
        IInitDblBuffer();
    else
    {
        // Copy the zoomed image as our backdrop
        BitBlt( fDblDC, 0, 0, fDblWidth, fDblHeight, fImageDC, 0, 0, SRCCOPY );

        RECT    r;
        IParamBlock2    *pb = fComp->GetParamBlockByID( plComponent::kBlkComp );
        if( pb != nil )
        {
            // Draw all the other elements other than our current one
            for( int i = 0; i < pfGUISkin::kNumElements; i++ )
                IJustDrawOneRect( i, pb, fDblDC, fOtherPen, fCurrPBRefSet );

            // Now draw the bounds of our current element
            SetRect( &r, pb->GetInt( fCurrPBRefSet + 0 ),
                         pb->GetInt( fCurrPBRefSet + 1 ),
                         pb->GetInt( fCurrPBRefSet + 0 ) + pb->GetInt( fCurrPBRefSet + 2 ),
                         pb->GetInt( fCurrPBRefSet + 1 ) + pb->GetInt( fCurrPBRefSet + 3 ) );
            
            // While we have it here, go ahead and update our status text for this element
            char str[ 256 ];
            sprintf( str, "Left: %d\nTop: %d\nWidth: %d\nHeight: %d\n", r.left, r.top, r.right - r.left, r.bottom - r.top );
            SetDlgItemText( fHWnd, IDC_GUI_INFO, str );
                            
            r.left *= fZoom; r.right *= fZoom; r.top *= fZoom; r.bottom *= fZoom;

            SelectObject( fDblDC, fDefPen );
            int rop2 = SetROP2( fDblDC, R2_NOTXORPEN );

            MoveToEx( fDblDC, r.left, r.top, nil );
            LineTo( fDblDC, r.right, r.top );
            LineTo( fDblDC, r.right, r.bottom );
            LineTo( fDblDC, r.left, r.bottom );
            LineTo( fDblDC, r.left, r.top );

            SetROP2( fDblDC, rop2 );

            fCurrElemRect = r;
            MapWindowPoints( GetDlgItem( fHWnd, IDC_GUI_PREVIEW ), fHWnd, (POINT *)&fCurrElemRect, 2 );
            OffsetRect( &fCurrElemRect, -fXOffset, -fYOffset );
        }
    }
}

void    pfGUISkinEditProc::IRefreshImageBuffer( void )
{
    IInitDblBuffer();

    plLayerTex *layer = fComp->GetSkinBitmap();
    PBBitmap *pbBMap = layer->GetPBBitmap();

    if( pbBMap->bm == nil )
        pbBMap->Load();
    if( pbBMap->bm != nil )
    {
        // Copy into a new temp bitmap that is the right format for us to read
        Bitmap *newBM;
        BitmapInfo bi;
        bi.SetName( _T("y879873b") );
        bi.SetType( BMM_TRUE_32 );
        bi.SetFlags( MAP_HAS_ALPHA );
        bi.SetWidth( fDblWidth );
        bi.SetHeight( fDblHeight );
        newBM = TheManager->Create( &bi );

        BMM_Color_64 foo = BMMCOLOR(0, 0, 0, 0);
        newBM->CopyImage( pbBMap->bm, COPY_IMAGE_RESIZE_LO_QUALITY, foo, nil );

        // Now copy from our newly created bitmap into our DC....way slow :(
        BITMAPINFO *bitInfo = newBM->ToDib( 24, nil, false );
        if( bitInfo != nil )
        {
            SetDIBitsToDevice( fImageDC, 0, 0, fDblWidth, fDblHeight,
                                0, 0, 0, fDblHeight,
                                ( (uint8_t *)bitInfo ) + bitInfo->bmiHeader.biSize,
                                bitInfo,
                                DIB_RGB_COLORS );
        }

        newBM->DeleteThis();
    }

    IRefreshDblBuffer();
}

void    pfGUISkinEditProc::IInitDblBuffer( void )
{
    if( fDblDC == NULL )
    {
        int     width, height;
        HDC     desk = GetDC( NULL );

        plLayerTex *layer = fComp->GetSkinBitmap();
        PBBitmap *pbBMap = layer->GetPBBitmap();
        if( pbBMap == nil )
            return;
        width = pbBMap->bi.Width() * fZoom;
        height = pbBMap->bi.Height() * fZoom;
//              GetClientRect( fHWnd, &r );
//              width = r.right - r.left;
//              height = r.bottom - r.top;

        //  Note: For some strange reason, grabbing the HDC of the window doesn't do
        //  any good, 'cause it's black-and-white (right, THAT makes sense). Grabbing
        //  the desktop DC works, however.

        fDblDC = CreateCompatibleDC( desk );
        fDblBitmap = CreateCompatibleBitmap( desk/*fDblDC*/, width, height );
        SelectObject( fDblDC, fDblBitmap );

        fImageDC = CreateCompatibleDC( desk );
        fImageBitmap = CreateCompatibleBitmap( desk/*fDblDC*/, width, height );
        SelectObject( fImageDC, fImageBitmap );

        ReleaseDC( NULL, desk );

        fDblWidth = width;
        fDblHeight = height;

        ISetScrollRanges();
        IRefreshImageBuffer();
    }
}

void    pfGUISkinEditProc::IKillDblBuffer( void )
{
    if( fDblDC != NULL )
    {
        SelectObject( fDblDC, (HBITMAP)NULL );
        DeleteObject( fDblBitmap );
        DeleteDC( fDblDC );
    }

    if( fImageDC != NULL )
    {
        SelectObject( fImageDC, (HBITMAP)NULL );
        DeleteObject( fImageBitmap );
        DeleteDC( fImageDC );
    }

    fDblDC = fImageDC = nil;
    fDblBitmap = fImageBitmap = nil;
}

void    pfGUISkinEditProc::ISetScrollRanges( void )
{
    SCROLLINFO  info;


    int visW = fPreviewRect.right - fPreviewRect.left;
    int visH = fPreviewRect.bottom - fPreviewRect.top;

    if( visW < fDblWidth )
    {
        if( fXOffset > fDblWidth - visW )
            fXOffset = fDblWidth - visW;
        else if( fXOffset < 0 )
            fXOffset = 0;

        info.fMask = SIF_PAGE | SIF_RANGE | SIF_POS;
        info.nMin = 0;
        info.nMax = fDblWidth;// - visW;
        info.nPage = visW;
        info.nPos = fXOffset;
        info.cbSize = sizeof( info );

        SetScrollInfo( GetDlgItem( fHWnd, IDC_GUI_HORZSCROLL ), SB_CTL, &info, true );
        ShowWindow( GetDlgItem( fHWnd, IDC_GUI_HORZSCROLL ), true );
    }
    else
    {
        ShowWindow( GetDlgItem( fHWnd, IDC_GUI_HORZSCROLL ), false );
        fXOffset = 0;
    }

    if( visH < fDblHeight )
    {
        if( fYOffset > fDblHeight - visH )
            fYOffset = fDblHeight - visH;
        else if( fYOffset < 0 )
            fYOffset = 0;

        info.fMask = SIF_PAGE | SIF_RANGE | SIF_POS;
        info.nMin = 0;
        info.nMax = fDblHeight;// - visH;
        info.nPage = visH;
        info.nPos = fYOffset;
        info.cbSize = sizeof( info );

        SetScrollInfo( GetDlgItem( fHWnd, IDC_GUI_VERTSCROLL ), SB_CTL, &info, true );
        ShowWindow( GetDlgItem( fHWnd, IDC_GUI_VERTSCROLL ), true );
    }
    else
    {
        ShowWindow( GetDlgItem( fHWnd, IDC_GUI_VERTSCROLL ), false );
        fYOffset = 0;
    }
}

bool    pfGUISkinEditProc::IPointWithinRange( int x, int y, int ptX, int ptY )
{
    if( x > ptX - kRangeSlop && x < ptX + kRangeSlop &&
        y > ptY - kRangeSlop && y < ptY + kRangeSlop )
        return true;
    return false;
}

bool    pfGUISkinEditProc::IPointWithinVertRange( int x, int y, int ptX, int ptY1, int ptY2 )
{
    if( x > ptX - kRangeSlop && x < ptX + kRangeSlop &&
        y > ptY1 - kRangeSlop && y < ptY2 + kRangeSlop )
        return true;
    return false;
}

bool    pfGUISkinEditProc::IPointWithinHorzRange( int x, int y, int ptX1, int ptX2, int ptY )
{
    if( x > ptX1 - kRangeSlop && x < ptX2 + kRangeSlop &&
        y > ptY - kRangeSlop && y < ptY + kRangeSlop )
        return true;
    return false;
}

uint8_t   pfGUISkinEditProc::IGetDragTypeFlags( int x, int y )
{
    // Corners
    if( IPointWithinRange( x, y, fCurrElemRect.left, fCurrElemRect.top ) )
        return kLeft | kTop;
    if( IPointWithinRange( x, y, fCurrElemRect.right, fCurrElemRect.top ) )
        return kRight | kTop;
    if( IPointWithinRange( x, y, fCurrElemRect.right, fCurrElemRect.bottom ) )
        return kRight | kBottom;
    if( IPointWithinRange( x, y, fCurrElemRect.left, fCurrElemRect.bottom ) )
        return kLeft | kBottom;

    // Edges
    if( IPointWithinVertRange( x, y, fCurrElemRect.left, fCurrElemRect.top, fCurrElemRect.bottom ) )
        return kLeft;
    if( IPointWithinVertRange( x, y, fCurrElemRect.right, fCurrElemRect.top, fCurrElemRect.bottom ) )
        return kRight;

    if( IPointWithinHorzRange( x, y, fCurrElemRect.left, fCurrElemRect.right, fCurrElemRect.top ) )
        return kTop;
    if( IPointWithinHorzRange( x, y, fCurrElemRect.left, fCurrElemRect.right, fCurrElemRect.bottom ) )
        return kBottom;

    // The middle
    if( x >= fCurrElemRect.left && x <= fCurrElemRect.right && y >= fCurrElemRect.top && y <= fCurrElemRect.bottom )
        return kDragAll;

    return 0;
}

INT_PTR CALLBACK    pfGUISkinEditProc::DlgProc( HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam )
{
    return fInstance->DialogProc( hDlg, msg, wParam, lParam );
}

INT_PTR CALLBACK    pfGUISkinEditProc::DialogProc( HWND hDlg, UINT msg, WPARAM wParam, LPARAM lParam )
{
    PAINTSTRUCT     pInfo;
    RECT            r;
    HDC             hDC;
    int             maxDim, i, j, x, y;
    bool            timerActive = false;


    static struct plElemPair
    {
        pfGUISkin::Elements el;
        const char          *name;
    } sElemPairs[] = { { pfGUISkin::kUpLeftCorner,          "Upper-left Corner" },
                       { pfGUISkin::kTopSpan,               "Top Span" },
                       { pfGUISkin::kUpRightCorner,         "Upper-right Corner" },
                       { pfGUISkin::kRightSpan,             "Right Span" },
                       { pfGUISkin::kLowerRightCorner,      "Lower-right Corner" },
                       { pfGUISkin::kBottomSpan,            "Bottom Span" },
                       { pfGUISkin::kLowerLeftCorner,       "Lower-left Corner" },
                       { pfGUISkin::kLeftSpan,              "Left Span" },
                       { pfGUISkin::kMiddleFill,            "Middle Fill" },
                       { pfGUISkin::kSelectedFill,          "Selected Middle Fill" },
                       { pfGUISkin::kSubMenuArrow,          "Sub-Menu Arrow" },
                       { pfGUISkin::kSelectedSubMenuArrow,  "Selected Sub-Menu Arrow" },
                       { pfGUISkin::kTreeButtonClosed,      "Tree-view Button, Closed" },
                       { pfGUISkin::kTreeButtonOpen,        "Tree-view Button, Open" },
                       { pfGUISkin::kNumElements, nil } };


    fHWnd = hDlg;

    switch( msg )
    {
        case WM_INITDIALOG:

            // Get preview rect
            GetClientRect( GetDlgItem( hDlg, IDC_GUI_PREVIEW ), &fPreviewRect );
            MapWindowPoints( GetDlgItem( hDlg, IDC_GUI_PREVIEW ), hDlg, (POINT *)&fPreviewRect, 2 );

            SendDlgItemMessage( hDlg, IDC_GUI_ZIN, BM_SETIMAGE, (WPARAM)IMAGE_ICON, 
                                        (LPARAM)LoadImage( hInstance, MAKEINTRESOURCE( IDI_ZOOMIN ), IMAGE_ICON, 0, 0, LR_DEFAULTCOLOR ) );
            SendDlgItemMessage( hDlg, IDC_GUI_ZOUT, BM_SETIMAGE, (WPARAM)IMAGE_ICON, 
                                        (LPARAM)LoadImage( hInstance, MAKEINTRESOURCE( IDI_ZOOMOUT ), IMAGE_ICON, 0, 0, LR_DEFAULTCOLOR ) );

            // Fill element list
            SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_RESETCONTENT, 0, 0 );
            for( i = 0; sElemPairs[ i ].el != pfGUISkin::kNumElements; i++ )
            {
                int idx = SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_ADDSTRING, 0, (LPARAM)sElemPairs[ i ].name );
                SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_SETITEMDATA, (WPARAM)idx, (LPARAM)sElemPairs[ i ].el );
                if( sElemPairs[ i ].el == pfGUISkin::kUpLeftCorner )
                    j = idx;
            }
            SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_SETCURSEL, j, 0 );

            fOrigCursor = LoadCursor( nil, IDC_ARROW );//GetCursor();

            break;

        case WM_COMMAND:
            if( LOWORD( wParam ) == IDCANCEL )
            {
                // Since we've been editing the PB directly, we have to now restore them
                // to their original values
                IParamBlock2 *pb = fComp->GetParamBlockByID( plComponent::kBlkComp );
                for( int i = 0; i < pfGUISkin::kNumElements; i++ )
                {
                    int id = i * 4 + plGUISkinComp::kRefUpLeftCorner;
                    pb->SetValue( id + 0, 0, (int)fBackups[ i ].fX );
                    pb->SetValue( id + 1, 0, (int)fBackups[ i ].fY ); 
                    pb->SetValue( id + 2, 0, (int)fBackups[ i ].fWidth );
                    pb->SetValue( id + 3, 0, (int)fBackups[ i ].fHeight );
                }
                EndDialog( hDlg, 1 );
            }
            else if( LOWORD( wParam ) == IDOK )
                EndDialog( hDlg, 0 );
            else if( LOWORD( wParam ) == IDC_GUI_ZIN )
            {
                fXOffset /= fZoom;  fYOffset /= fZoom;
                fZoom++;
                fXOffset *= fZoom;  fYOffset *= fZoom;

                IKillDblBuffer();
                IRefreshImageBuffer();
                InvalidateRect( hDlg, &fPreviewRect, false );
            }
            else if( LOWORD( wParam ) == IDC_GUI_ZOUT )
            {
                if( fZoom > 1 )
                {
                    fXOffset /= fZoom;  fYOffset /= fZoom;
                    fZoom--;
                    fXOffset *= fZoom;  fYOffset *= fZoom;

                    IKillDblBuffer();
                    IRefreshImageBuffer();
                    InvalidateRect( hDlg, &fPreviewRect, false );
                }
            }
            else if( LOWORD( wParam ) == IDC_GUI_ELEMENTS )
            {
                int idx = SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_GETCURSEL, 0, 0 );
                fCurrPBRefSet = SendDlgItemMessage( hDlg, IDC_GUI_ELEMENTS, LB_GETITEMDATA, (WPARAM)idx, 0 ) * 4 + plGUISkinComp::kRefUpLeftCorner;

                IRefreshDblBuffer();
                InvalidateRect( hDlg, &fPreviewRect, false );
            }
            return true;

        case WM_CLOSE:
            EndDialog( hDlg, 0 );
            return true;

        case WM_HSCROLL:
            OffsetRect( &fCurrElemRect, fXOffset, fYOffset );
            switch( LOWORD( wParam ) )
            {
                case SB_PAGEUP:     fXOffset -= 300; break;
                case SB_PAGEDOWN:   fXOffset += 300; break;
                case SB_LINEUP:     fXOffset -= 16; break;
                case SB_LINEDOWN:   fXOffset += 16; break;
                case SB_THUMBPOSITION:
                case SB_THUMBTRACK:
                    fXOffset = HIWORD( wParam );
                    break;
            }
            maxDim = fDblWidth - ( fPreviewRect.right - fPreviewRect.left );
            if( fXOffset < 0 )
                fXOffset = 0;
            else if( fXOffset > maxDim )
                fXOffset = maxDim;
            SetScrollPos( GetDlgItem( hDlg, IDC_GUI_HORZSCROLL ), SB_CTL, fXOffset, true );

            OffsetRect( &fCurrElemRect, -fXOffset, -fYOffset );
            InvalidateRect( hDlg, &fPreviewRect, false );
            break;

        case WM_VSCROLL:
            OffsetRect( &fCurrElemRect, fXOffset, fYOffset );
            switch( LOWORD( wParam ) )
            {
                case SB_PAGEUP:     fYOffset -= 300; break;
                case SB_PAGEDOWN:   fYOffset += 300; break;
                case SB_LINEUP:     fYOffset -= 16; break;
                case SB_LINEDOWN:   fYOffset += 16; break;
                case SB_THUMBPOSITION:
                case SB_THUMBTRACK:
                    fYOffset = HIWORD( wParam );
                    break;
            }
            maxDim = fDblHeight - ( fPreviewRect.bottom - fPreviewRect.top );
            if( fYOffset < 0 )
                fYOffset = 0;
            else if( fYOffset > maxDim )
                fYOffset = maxDim;
            SetScrollPos( GetDlgItem( hDlg, IDC_GUI_VERTSCROLL ), SB_CTL, fYOffset, true );

            OffsetRect( &fCurrElemRect, -fXOffset, -fYOffset );
            InvalidateRect( hDlg, &fPreviewRect, false );
            break;

        case WM_PAINT:
            {
                BeginPaint( hDlg, &pInfo );
                hDC = (HDC)pInfo.hdc;

                if( fDblDC == NULL )
                    IInitDblBuffer();

                int width = fDblWidth;
                int height = fDblHeight;
                if( width > fPreviewRect.right - fPreviewRect.left )
                    width = fPreviewRect.right - fPreviewRect.left;
                if( height > fPreviewRect.bottom - fPreviewRect.top )
                    height = fPreviewRect.bottom - fPreviewRect.top;

                BitBlt( hDC, fPreviewRect.left, fPreviewRect.top, width, height, fDblDC, fXOffset, fYOffset, SRCCOPY );

                r = fPreviewRect;
                r.left += width;
                FillRect( hDC, &r, ColorMan()->GetBrush( kBackground ) );

                r = fPreviewRect;
                r.top += height;
                FillRect( hDC, &r, ColorMan()->GetBrush( kBackground ) );

                EndPaint( hDlg, &pInfo );
            }
            break;

        case WM_LBUTTONDOWN:
            SetCapture( hDlg );
            fDragging = true;
            fDragType = IGetDragTypeFlags( GET_X_LPARAM( lParam ), GET_Y_LPARAM( lParam ) );
            if( fDragType == kDragAll )
            {
                fDragOffsetX = fCurrElemRect.left - GET_X_LPARAM( lParam );
                fDragOffsetY = fCurrElemRect.top - GET_Y_LPARAM( lParam );
            }
            else if( fDragType == 0 )
            {
                fDragOffsetX = GET_X_LPARAM( lParam ) + fXOffset;
                fDragOffsetY = GET_Y_LPARAM( lParam ) + fYOffset;
            }
            else
                fDragOffsetX = fDragOffsetY = 0;

            break;

        case WM_LBUTTONUP:
            ReleaseCapture();
            fDragging = false;
            break;

        case WM_TIMER:
            // We do the same processing as MOUSEMOVE, but we need to make sure we have the right
            // mouse position first
            {
                POINT pt;
                GetCursorPos( &pt );
                MapWindowPoints( nil, hDlg, &pt, 1 );
                lParam = MAKELPARAM( pt.x, pt.y );
            }
            // Fall thru...

        case WM_MOUSEMOVE:
            x = GET_X_LPARAM( lParam );
            y = GET_Y_LPARAM( lParam );

            if( fDragging )
            {
                if( fDragType == 0 )
                {
                    fXOffset = fDragOffsetX - x;
                    fYOffset = fDragOffsetY - y;
                    ISetScrollRanges(); // Will clamp offset \for us
                }
                else
                {
                    // Translate x and y into bitmap space
                    POINT pt;
                    pt.x = x;
                    pt.y = y;

                    if( PtInRect( &fPreviewRect, pt ) )
                    {
                        MapWindowPoints( hDlg, GetDlgItem( hDlg, IDC_GUI_PREVIEW ), &pt, 1 );

                        pt.x += fDragOffsetX;
                        pt.y += fDragOffsetY;

                        // Note the + 1/2 zoom so it's the closest pixel by center, not by area
                        x = ( pt.x + fXOffset + ( fZoom >> 1 ) ) / fZoom;
                        y = ( pt.y + fYOffset + ( fZoom >> 1 ) ) / fZoom;

                        // Set depending on our current drag flags
                        // Note the logic here: if we drag left, we want width and left changing,
                        // if we drag both, just left, if we drag right, just width
                        IParamBlock2 *pb = fComp->GetParamBlockByID( plComponent::kBlkComp );
                        if( fDragType & kLeft )
                        {
                            if( fDragType & kRight )
                                pb->SetValue( fCurrPBRefSet + 0, 0, (int)x );
                            else 
                            {
                                int old = pb->GetInt( fCurrPBRefSet + 0 ) + pb->GetInt( fCurrPBRefSet + 2 );

                                pb->SetValue( fCurrPBRefSet + 0, 0, (int)x );
                                pb->SetValue( fCurrPBRefSet + 2, 0, (int)old - x );
                            }
                        }
                        else if( fDragType & kRight )
                            pb->SetValue( fCurrPBRefSet + 2, 0, (int)x - pb->GetInt( fCurrPBRefSet + 0 ) );

                        if( fDragType & kTop )
                        {
                            if( fDragType & kBottom )
                                pb->SetValue( fCurrPBRefSet + 1, 0, (int)y );
                            else 
                            {
                                int old = pb->GetInt( fCurrPBRefSet + 1 ) + pb->GetInt( fCurrPBRefSet + 3 );

                                pb->SetValue( fCurrPBRefSet + 1, 0, (int)y );
                                pb->SetValue( fCurrPBRefSet + 3, 0, (int)old - y );
                            }
                        }
                        else if( fDragType & kBottom )
                            pb->SetValue( fCurrPBRefSet + 3, 0, (int)y - pb->GetInt( fCurrPBRefSet + 1 ) );

                        // Clamp width and height
                        if( pb->GetInt( fCurrPBRefSet + 2 ) < 0 )
                            pb->SetValue( fCurrPBRefSet + 2, 0, (int)0 );
                        if( pb->GetInt( fCurrPBRefSet + 3 ) < 0 )
                            pb->SetValue( fCurrPBRefSet + 3, 0, (int)0 );

                        // Clamp X and Y
                        if( pb->GetInt( fCurrPBRefSet + 0 ) < 0 )
                            pb->SetValue( fCurrPBRefSet + 0, 0, (int)0 );
                        else if( pb->GetInt( fCurrPBRefSet + 0 ) + pb->GetInt( fCurrPBRefSet + 2 ) > fDblWidth / fZoom )
                            pb->SetValue( fCurrPBRefSet + 0, 0, (int)fDblWidth / fZoom - pb->GetInt( fCurrPBRefSet + 2 ) );

                        if( pb->GetInt( fCurrPBRefSet + 1 ) < 0 )
                            pb->SetValue( fCurrPBRefSet + 1, 0, (int)0 );
                        else if( pb->GetInt( fCurrPBRefSet + 1 ) + pb->GetInt( fCurrPBRefSet + 3 ) > fDblHeight / fZoom )
                            pb->SetValue( fCurrPBRefSet + 1, 0, (int)fDblHeight / fZoom - pb->GetInt( fCurrPBRefSet + 3 ) );
                    }
                    else
                    {
                        // Mouse is outside our preview, so scroll if possible
                        int dX = ( x < fPreviewRect.left ) ? x - fPreviewRect.left : ( x > fPreviewRect.right ) ? x - fPreviewRect.right : 0;
                        int dY = ( y < fPreviewRect.top ) ? y - fPreviewRect.top : ( y > fPreviewRect.bottom ) ? y - fPreviewRect.bottom : 0;

                        fXOffset += dX;
                        fYOffset += dY;         
                        OffsetRect( &fCurrElemRect, -dX, -dY );

                        ISetScrollRanges(); // Will clamp origin for us

                        // Don't actually drag our bounds, 'cause we're scrolling

                        // Note: since we only get MOUSEMOVE when, gee, the mouse moves, if we've scrolled over, it'll only
                        // do it once and then wait for the mouse to nudge again. We'd rather it keep going until the user
                        // moves the mouse again, so we create a timer that calls us back in n somethingths so we can check again
                        if( fDragTimer == 0 )
                            fDragTimer = SetTimer( hDlg, 0, 200, nil );
                        timerActive = true;     // So we don't kill it at the end here...
                    }
                }

                IRefreshDblBuffer();
                InvalidateRect( hDlg, &fPreviewRect, false );
            }
            else
            {
                uint8_t dragType = IGetDragTypeFlags( x, y );
                HCURSOR cursor;
                switch( dragType )
                {
                    case kLeft | kTop:      
                    case kRight | kBottom:
                        cursor = LoadCursor( nil, IDC_SIZENWSE );
                        break;
                    case kLeft | kBottom:       
                    case kRight | kTop:
                        cursor = LoadCursor( nil, IDC_SIZENESW );
                        break;
                    case kLeft:     
                    case kRight:
                        cursor = LoadCursor( nil, IDC_SIZEWE );
                        break;
                    case kTop:      
                    case kBottom:
                        cursor = LoadCursor( nil, IDC_SIZENS );
                        break;
                    case kLeft | kTop | kRight | kBottom:       
                        cursor = LoadCursor( nil, IDC_SIZEALL );
                        break;
                    default:
                        {
                            POINT pt;
                            pt.x = x;
                            pt.y = y;
                            if( PtInRect( &fPreviewRect, pt ) )
                                cursor = LoadCursor( nil, IDC_HAND );
                            else
                                cursor = fOrigCursor;
                        }
                        break;
                }
                
                SetCursor( cursor );

                if( !timerActive )
                {
                    // No longer need our trick timer, so kill it
                    KillTimer( hDlg, fDragTimer );
                    fDragTimer = 0;
                }
            }
            break;
    }

    return false;
}

plGUISkinComp   *plGUISkinComp::GetGUIComp( INode *node )
{
    if( node == nil )
        return nil;

    plComponentBase *base = ( ( plMaxNodeBase *)node )->ConvertToComponent();
    if( base == nil )
        return nil;

    if( base->ClassID() == GUI_SKIN_CLASSID )
        return (plGUISkinComp *)base;

    return nil;
}