/****************************************************************/
/*																*/
/*  DialogResize.cpp											*/
/*																*/
/*  Implementation of the CDialogResize class.					*/
/*	Provides support for resizing dialog controls.				*/
/*																*/
/*  Based on the WTL class CDialogResize from Microsoft			*/
/*  Modified for use with MFC by LYFZ van der Meer				*/
/*																*/
/*  Copyright LYFZ Software Solutions 2002						*/
/*	http://www.LYFZvandermeer.nl								*/
/*																*/
/*  Last updated: 10 july 2002									*/
/*																*/
/****************************************************************/


#include "stdafx.h"
#include "DialogResize.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif


CDialogResize::CDialogResize()
{
	CommonConstruct();
}


CDialogResize::CDialogResize(UINT uResource, CWnd* pParent)
	: CDialog(uResource, pParent)
{
	CommonConstruct();
}


CDialogResize::CDialogResize(LPCTSTR pszResource, CWnd* pParent)
	: CDialog(pszResource, pParent)
{
	CommonConstruct();
}


void CDialogResize::CommonConstruct()
{
	//{{AFX_DATA_INIT(CDialogResize)
		// NOTE: the ClassWizard will add member initialization here
	//}}AFX_DATA_INIT
	m_bGripper = FALSE;
	m_sizeDialog.cx = 0;
	m_sizeDialog.cy = 0;
	m_ptMinTrackSize.x = -1;
	m_ptMinTrackSize.y = -1;
}


void CDialogResize::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
	//{{AFX_DATA_MAP(CDialogResize)
		// NOTE: the ClassWizard will add DDX and DDV calls here
	//}}AFX_DATA_MAP
}


BEGIN_MESSAGE_MAP(CDialogResize, CDialog)
	//{{AFX_MSG_MAP(CDialogResize)
	ON_WM_SIZE()
	ON_WM_GETMINMAXINFO()
	//}}AFX_MSG_MAP
END_MESSAGE_MAP()


BEGIN_DLGRESIZE_MAP(CDialogResize)
END_DLGRESIZE_MAP()

// Operations
void CDialogResize::InitResizing(BOOL bAddGripper, BOOL bUseMinTrackSize, DWORD dwForceStyle)
{
	// Force specified styles (default WS_THICKFRAME | WS_CLIPCHILDREN to enable resizing border and reduce flicker)
	if((GetStyle() & dwForceStyle) != dwForceStyle)
		ModifyStyle(0, dwForceStyle);

	// Cleanup in case of multiple initialization
	// block: first check for the gripper control, destroy it if needed
	{
		CWnd *wndGripper = GetDlgItem(AFX_IDW_STATUS_BAR);
		if (wndGripper != NULL)
		{

		}
	}
	// clear out everything else
	m_arrData.RemoveAll();
	m_sizeDialog.cx = 0;
	m_sizeDialog.cy = 0;
	m_ptMinTrackSize.x = -1;
	m_ptMinTrackSize.y = -1;

	// Get initial dialog client size
	RECT rectDlg;
	GetClientRect(&rectDlg);
	m_sizeDialog.cx = rectDlg.right;
	m_sizeDialog.cy = rectDlg.bottom;

	// Create gripper if requested
	m_bGripper = false;
	if(bAddGripper)
	{
		// shouldn't exist already
		ASSERT(!::IsWindow(::GetDlgItem(m_hWnd, AFX_IDW_STATUS_BAR)));
		if(!::IsWindow(::GetDlgItem(m_hWnd, AFX_IDW_STATUS_BAR)))
		{
			m_wndGripper.Create(_T("SCROLLBAR"), "", WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | SBS_SIZEBOX | SBS_SIZEGRIP | SBS_SIZEBOXBOTTOMRIGHTALIGN, rectDlg, this, AFX_IDW_STATUS_BAR);
			if(IsWindow(m_wndGripper.m_hWnd))
			{
				m_bGripper = true;
				RECT rectCtl;
				m_wndGripper.GetWindowRect(&rectCtl);
				ScreenToClient(&rectCtl);
				AfxResizeData data = {AFX_IDW_STATUS_BAR, DLSZ_MOVE_X | DLSZ_MOVE_Y | DLSZ_REPAINT | _DLSZ_GRIPPER, { rectCtl.left, rectCtl.top, rectCtl.right, rectCtl.bottom } };
				m_arrData.Add(data);
			}
		}
	}

	// Get min track position if requested
	if(bUseMinTrackSize)
	{
		RECT rect;
		GetWindowRect(&rect);
		m_ptMinTrackSize.x = rect.right - rect.left;
		m_ptMinTrackSize.y = rect.bottom - rect.top;
	}

	// Walk the map and initialize data
	const AfxResizeMap* pMap = GetDlgResizeMap();
	ASSERT(pMap != NULL);
	int nGroupStart = -1;
	for(int nCount = 1; !(pMap->m_nCtlID == -1 && pMap->m_dwResizeFlags == 0); nCount++, pMap++)
	{
		if(pMap->m_nCtlID == -1)
		{
			switch(pMap->m_dwResizeFlags)
			{
			case _DLSZ_BEGIN_GROUP:
				ASSERT(nGroupStart == -1);
				nGroupStart = m_arrData.GetSize();
				break;
			case _DLSZ_END_GROUP:
			{
				ASSERT(nGroupStart != -1);
				int nGroupCount = m_arrData.GetSize() - nGroupStart;
				m_arrData[nGroupStart].SetGroupCount(nGroupCount);
				nGroupStart = -1;
			}
			break;

			default:
				ASSERT(FALSE && _T("Invalid DLGRESIZE Map Entry"));
				break;
			}
		}
		else
		{
			// this ID conflicts with the default gripper one
			ASSERT(m_bGripper ? (pMap->m_nCtlID != AFX_IDW_STATUS_BAR) : TRUE);

			CWnd *ctl = GetDlgItem(pMap->m_nCtlID);
			if (ctl)
			{
				RECT rectCtl;
				ctl->GetWindowRect(&rectCtl);
				ScreenToClient(&rectCtl);

				DWORD dwGroupFlag = (nGroupStart != -1 && m_arrData.GetSize() == nGroupStart) ? _DLSZ_BEGIN_GROUP : 0;
				AfxResizeData data = { pMap->m_nCtlID, pMap->m_dwResizeFlags | dwGroupFlag, { rectCtl.left, rectCtl.top, rectCtl.right, rectCtl.bottom } };
				m_arrData.Add(data);
			}
		}
	}
	ASSERT((nGroupStart == -1) && _T("No End Group Entry in the DLGRESIZE Map"));
}


void CDialogResize::UpdateLayout(int cxWidth, int cyHeight)
{
	ASSERT(::IsWindow(m_hWnd));
//	SetRedraw(FALSE);
	RECT rectGroup = { 0, 0, 0, 0 };
	for(int i = 0; i < m_arrData.GetSize(); i++)
	{
		if((m_arrData[i].m_dwResizeFlags & _DLSZ_BEGIN_GROUP) != 0)	// start of a group
		{
			int nGroupCount = m_arrData[i].GetGroupCount();
			ASSERT(nGroupCount > 0 && i + nGroupCount - 1 < m_arrData.GetSize());
			rectGroup = m_arrData[i].m_rect;
			int j;
			for(j = 1; j < nGroupCount; j++)
			{
				rectGroup.left = min(rectGroup.left, m_arrData[i + j].m_rect.left);
				rectGroup.top = min(rectGroup.top, m_arrData[i + j].m_rect.top);
				rectGroup.right = max(rectGroup.right, m_arrData[i + j].m_rect.right);
				rectGroup.bottom = max(rectGroup.bottom, m_arrData[i + j].m_rect.bottom);
			}
			RECT rcThis;
			RECT rcNext;
			for(j = 0; j < nGroupCount; j++)
			{
				int xyStartNext = -1;
				if((j < (nGroupCount - 1)) && ((m_arrData[i + j].m_dwResizeFlags & (DLSZ_SIZE_X | DLSZ_SIZE_Y)) != 0) && ((m_arrData[i + j + 1].m_dwResizeFlags & (DLSZ_SIZE_X | DLSZ_SIZE_Y)) != 0))
				{
					CWnd *ctlThis = GetDlgItem(m_arrData[i + j].m_nCtlID);
					ctlThis->GetWindowRect(&rcThis);
					ScreenToClient(&rcThis);
					CWnd *ctlNext = GetDlgItem(m_arrData[i + j + 1].m_nCtlID);
					ctlNext->GetWindowRect(&rcNext);
					ScreenToClient(&rcNext);
					if((m_arrData[i + j].m_dwResizeFlags & DLSZ_SIZE_X) == DLSZ_SIZE_X && (m_arrData[i + j + 1].m_dwResizeFlags & DLSZ_SIZE_X) == DLSZ_SIZE_X)
					{
						if(rcNext.left >= rcThis.right)
							xyStartNext = m_arrData[i + j + 1].m_rect.left;
					}
					else 
					if((m_arrData[i + j].m_dwResizeFlags & DLSZ_SIZE_Y) == DLSZ_SIZE_Y && (m_arrData[i + j + 1].m_dwResizeFlags & DLSZ_SIZE_Y) == DLSZ_SIZE_Y)
					{
						if(rcNext.top >= rcThis.bottom)
							xyStartNext = m_arrData[i + j + 1].m_rect.top;
					}
				}
				PositionControl(cxWidth, cyHeight, rectGroup, m_arrData[i + j], true, xyStartNext);
			}
			// increment to skip all group controls
			i += nGroupCount - 1;
		}
		else		// one control entry
		{
			PositionControl(cxWidth, cyHeight, rectGroup, m_arrData[i], false);
		}
	}
//	SetRedraw(TRUE);
	Invalidate();
	RedrawWindow(NULL, NULL, RDW_ERASE | RDW_INVALIDATE | RDW_UPDATENOW | RDW_ALLCHILDREN);
}

void CDialogResize::OnSize(UINT nType, int cx, int cy) 
{
	CDialog::OnSize(nType, cx, cy);
	
	if(m_bGripper)
	{
		if(nType == SIZE_MAXIMIZED)
			m_wndGripper.ShowWindow(SW_HIDE);
		else 
		if(nType == SIZE_RESTORED)
			m_wndGripper.ShowWindow(SW_SHOW);
	}
	if(nType != SIZE_MINIMIZED)
	{
		ASSERT(::IsWindow(m_hWnd));
		UpdateLayout(cx, cy);
	}
}

void CDialogResize::OnGetMinMaxInfo(MINMAXINFO FAR* lpMMI) 
{
	if(m_ptMinTrackSize.x != -1 && m_ptMinTrackSize.y != -1)
	{
		lpMMI->ptMinTrackSize =  m_ptMinTrackSize;
	}
	
	CDialog::OnGetMinMaxInfo(lpMMI);
}



// Implementation
BOOL CDialogResize::PositionControl(int cxWidth, int cyHeight, RECT& rectGroup, AfxResizeData& data, bool bGroup, int xyStartNext)
{
	ASSERT(::IsWindow(m_hWnd));
	CWnd *ctl;
	CRect rectCtl;

	ctl = GetDlgItem(data.m_nCtlID);
	
	ctl->GetWindowRect(rectCtl);

	ScreenToClient(rectCtl);

	if(bGroup)
	{
		if((data.m_dwResizeFlags & (DLSZ_SIZE_X | DLSZ_MOVE_X)) != 0)
		{
			rectCtl.left = rectGroup.left + ::MulDiv(data.m_rect.left - rectGroup.left, rectGroup.right - rectGroup.left + (cxWidth - m_sizeDialog.cx), rectGroup.right - rectGroup.left);

			if((data.m_dwResizeFlags & DLSZ_SIZE_X) != 0)
			{
				if(xyStartNext != -1)
					rectCtl.right = rectGroup.left + ::MulDiv(xyStartNext - rectGroup.left, rectGroup.right - rectGroup.left + (cxWidth - m_sizeDialog.cx), rectGroup.right - rectGroup.left) - (xyStartNext - data.m_rect.right);
				else
					rectCtl.right = rectGroup.left + ::MulDiv(data.m_rect.right - rectGroup.left, rectGroup.right - rectGroup.left + (cxWidth - m_sizeDialog.cx), rectGroup.right - rectGroup.left);
			}
			else
			{
				rectCtl.right = rectCtl.left + (data.m_rect.right - data.m_rect.left);
			}
		}

		if((data.m_dwResizeFlags & (DLSZ_SIZE_Y | DLSZ_MOVE_Y)) != 0)
		{
			rectCtl.top = rectGroup.top + ::MulDiv(data.m_rect.top - rectGroup.top, rectGroup.bottom - rectGroup.top + (cyHeight - m_sizeDialog.cy), rectGroup.bottom - rectGroup.top);

			if((data.m_dwResizeFlags & DLSZ_SIZE_Y) != 0)
			{
				if(xyStartNext != -1)
					rectCtl.bottom = rectGroup.top + ::MulDiv(xyStartNext - rectGroup.top, rectGroup.bottom - rectGroup.top + (cyHeight - m_sizeDialog.cy), rectGroup.bottom - rectGroup.top)  - (xyStartNext - data.m_rect.bottom);
				else
					rectCtl.bottom = rectGroup.top + ::MulDiv(data.m_rect.bottom - rectGroup.top, rectGroup.bottom - rectGroup.top + (cyHeight - m_sizeDialog.cy), rectGroup.bottom - rectGroup.top);
			}
			else
			{
				rectCtl.bottom = rectCtl.top + (data.m_rect.bottom - data.m_rect.top);
			}
		}
	}
	else
	{
		if((data.m_dwResizeFlags & (DLSZ_SIZE_X | DLSZ_MOVE_X)) != 0)
		{
			rectCtl.right = data.m_rect.right + (cxWidth - m_sizeDialog.cx);

			if((data.m_dwResizeFlags & DLSZ_MOVE_X) != 0)
				rectCtl.left = rectCtl.right - (data.m_rect.right - data.m_rect.left);
		}

		if((data.m_dwResizeFlags & (DLSZ_SIZE_Y | DLSZ_MOVE_Y)) != 0)
		{
			rectCtl.bottom = data.m_rect.bottom + (cyHeight - m_sizeDialog.cy);

			if((data.m_dwResizeFlags & DLSZ_MOVE_Y) != 0)
				rectCtl.top = rectCtl.bottom - (data.m_rect.bottom - data.m_rect.top);
		}
	}

	if((data.m_dwResizeFlags & DLSZ_REPAINT) != 0)
		ctl->Invalidate();

	if((data.m_dwResizeFlags & (DLSZ_SIZE_X | DLSZ_SIZE_Y | DLSZ_MOVE_X | DLSZ_MOVE_Y | DLSZ_REPAINT)) != 0)
		ctl->SetWindowPos(NULL, rectCtl.left, rectCtl.top, rectCtl.right - rectCtl.left, rectCtl.bottom - rectCtl.top, SWP_NOZORDER | SWP_NOACTIVATE);

	return TRUE;
}