Click here to Skip to main content
15,895,142 members
Articles / Desktop Programming / MFC

Professional User Interface Suite

Rate me:
Please Sign up or sign in to vote.
4.99/5 (174 votes)
13 Jan 200423 min read 1.5M   23.6K   641  
MFC extension library enabling software to be provided with a professional UI
// This is part of the Professional User Interface Suite library.
// Copyright (C) 2001-2002 FOSS Software, Inc.
// All rights reserved.
//
// http://www.fossware.com
// mailto:foss@fossware.com
//
// This source code can be used, modified and redistributed
// under the terms of the license agreement that is included
// in the Professional User Interface Suite package.
//
// Warranties and Disclaimers:
// THIS SOFTWARE IS PROVIDED "AS IS" WITHOUT WARRANTY OF ANY KIND
// INCLUDING, BUT NOT LIMITED TO, WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
// IN NO EVENT WILL FOSS SOFTWARE INC. BE LIABLE FOR ANY DIRECT,
// INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY OR CONSEQUENTIAL DAMAGES,
// INCLUDING DAMAGES FOR LOSS OF PROFITS, LOSS OR INACCURACY OF DATA,
// INCURRED BY ANY PERSON FROM SUCH PERSON'S USAGE OF THIS SOFTWARE
// EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.

#include "stdafx.h"

#if (!defined __EXT_COMBO_BOX_H)
	#include <ExtComboBox.h>
#endif

#if (!defined __EXT_PAINT_MANAGER_H)
	#include <ExtPaintManager.h>
#endif

#if (!defined __EXT_MEMORY_DC_H)
	#include <../Src/ExtMemoryDC.h>
#endif

#if (!defined __EXT_POPUP_MENU_WND_H)
	#include <ExtPopupMenuWnd.h>
#endif

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

/////////////////////////////////////////////////////////////////////////////
// CExtComboEditCtrlHook

CExtComboEditCtrlHook::CExtComboEditCtrlHook()
{
}
CExtComboEditCtrlHook::~CExtComboEditCtrlHook()
{
}

CExtComboBox * CExtComboEditCtrlHook::GetExtComboBox()
{
	ASSERT( GetSafeHwnd() != NULL );
	ASSERT( ::IsWindow(GetSafeHwnd()) );
HWND hWndParent = ::GetParent( GetSafeHwnd() );
	ASSERT( hWndParent != NULL );
	ASSERT( ::IsWindow(hWndParent) );
CExtComboBox * pCombo = 
		DYNAMIC_DOWNCAST(
			CExtComboBox,
			FromHandlePermanent(hWndParent)
			);
	ASSERT( pCombo != NULL );
	return pCombo;
}

IMPLEMENT_DYNCREATE(CExtComboEditCtrlHook, CEdit)

BEGIN_MESSAGE_MAP(CExtComboEditCtrlHook, CEdit)
	//{{AFX_MSG_MAP(CExtComboEditCtrlHook)
	//}}AFX_MSG_MAP
END_MESSAGE_MAP()

LRESULT CExtComboEditCtrlHook::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
bool bFlushAutoComplete = false;
	switch( message )
	{
	case WM_CONTEXTMENU:
		if( GetExtComboBox()->OnTrackComboContextMenu(GetSafeHwnd()) )
			return 0;
		break;
	case WM_CUT:
	case WM_COPY:
	case WM_PASTE:
	case WM_CLEAR:
	case WM_UNDO:
		bFlushAutoComplete = true;
		break;
	case WM_COMMAND:
		switch( LOWORD(wParam) )
		{
		case ID_EDIT_CLEAR:
		case ID_EDIT_CLEAR_ALL:
		case ID_EDIT_COPY:
		case ID_EDIT_CUT:
		case ID_EDIT_FIND:
		case ID_EDIT_PASTE:
		case ID_EDIT_PASTE_LINK:
		case ID_EDIT_PASTE_SPECIAL:
		case ID_EDIT_REPEAT:
		case ID_EDIT_REPLACE:
		case ID_EDIT_SELECT_ALL:
		case ID_EDIT_UNDO:
		case ID_EDIT_REDO:
			bFlushAutoComplete = true;
		break;
		} // switch( LOWORD(wParam) )
		break;
	} // switch( message )
	
	if( bFlushAutoComplete )
		GetExtComboBox()->m_bAutoComplete = false;

	return CEdit::WindowProc(message,wParam,lParam);
}

/////////////////////////////////////////////////////////////////////////////
// CExtComboBox

CExtComboBox::CExtComboBox()
	: m_bAutoComplete( true )
	, m_bEnableAutoComplete( true )
	, m_bDrawing( false )
	, m_dwUpdateTimer( 10 )
	, m_dwUpdatePeriod( 50 )
	, m_pInnerEditHook( NULL )
{
}

CExtComboBox::~CExtComboBox()
{
	if( m_pInnerEditHook != NULL )
		delete m_pInnerEditHook;
}

IMPLEMENT_DYNCREATE(CExtComboBox, CComboBox)

BEGIN_MESSAGE_MAP(CExtComboBox, CComboBox)
	//{{AFX_MSG_MAP(CExtComboBox)
	ON_WM_TIMER()
	ON_WM_PAINT()
	ON_WM_ERASEBKGND()
	ON_WM_KILLFOCUS()
	//}}AFX_MSG_MAP
	ON_CONTROL_REFLECT( CBN_EDITUPDATE, OnEditCtrlUpdate )
END_MESSAGE_MAP()

/////////////////////////////////////////////////////////////////////////////
// CExtComboBox message handlers

void CExtComboBox::OnTimer(UINT nIDEvent) 
{
	if( nIDEvent != m_dwUpdateTimer )
	{
		CComboBox::OnTimer( nIDEvent );
		return;
	} // if( nIDEvent != m_dwUpdateTimer )

POINT ptCursorPos;
	VERIFY( GetCursorPos( &ptCursorPos) );
CRect rcItem;
	GetWindowRect(&rcItem);

	//if( m_bLBtnDown )
	if( CExtPopupMenuWnd::IsKeyPressed(MK_LBUTTON) )
	{
		if( !GetDroppedState() )
			KillTimer( m_dwUpdateTimer );
		if( m_bDrawing )
		{
			_OnDrawComboImpl(
				GetDroppedState() ? true : false,
				false
				);
			m_bDrawing = false;
		} // if( CExtPopupMenuWnd::IsKeyPressed(MK_LBUTTON) )
		return;
	} // if( m_bLBtnDown )
	
	if( !rcItem.PtInRect(ptCursorPos) )
	{
		if( !GetDroppedState() )
			KillTimer( m_dwUpdateTimer );
		_OnDrawComboImpl( false, false );
		m_bDrawing = false;
		return;
	} // if( !rcItem.PtInRect(ptCursorPos) )

	if( m_bDrawing )
		return;
	m_bDrawing = true;
	_OnDrawComboImpl( false, true );
}

void CExtComboBox::_OnPaintImpl(
	bool bPressed,
	bool bHover
	)
{
	if( (GetStyle()&(CBS_OWNERDRAWFIXED|CBS_OWNERDRAWVARIABLE)) != 0 )
	{
		Default();
		CClientDC dc(this);
		_OnDrawComboImpl( bPressed, bHover, &dc );
	}
	else
	{
		CRect rcClient;
		GetClientRect( &rcClient );
		CPaintDC dcPaint( this );
		CExtPaintManager::stat_ExcludeChildAreas(
			dcPaint.GetSafeHdc(),
			GetSafeHwnd()
			);
		CExtMemoryDC dcmm( &dcPaint, &rcClient );
		DefWindowProc( WM_PAINT, (WPARAM)dcmm.GetSafeHdc(), (LPARAM)0 );
		_OnDrawComboImpl( bPressed, bHover, &dcmm );
	}
}

void CExtComboBox::OnPaint() 
{
	if(		(	GetExStyle()
				& (WS_EX_DLGMODALFRAME|WS_EX_CLIENTEDGE|WS_EX_STATICEDGE)
			) != 0
		)
		ModifyStyleEx(
			WS_EX_DLGMODALFRAME|WS_EX_CLIENTEDGE|WS_EX_STATICEDGE,
			0,
			SWP_FRAMECHANGED
			);
CPoint ptCursorPos;
	VERIFY( GetCursorPos( &ptCursorPos ) );
CRect rcItem;
	GetWindowRect( &rcItem );

bool bHover = false;
	if( rcItem.PtInRect(ptCursorPos) )
	{
		if( m_bDrawing )
			return;
		bHover = true;
	} // if( rcItem.PtInRect(ptCursorPos) )

	_OnPaintImpl( false, bHover );
}

void CExtComboBox::_OnDrawComboImpl(
	bool bPressed,
	bool bHover,
	CDC * pDC // = NULL
	)
{

CRect rectClient;
	GetClientRect( &rectClient );
bool bCallReleaseDC = false;
	if( pDC == NULL )
	{
		pDC = GetDC();
		ASSERT( pDC != NULL );
		bCallReleaseDC = true;
	}
	
bool bEnabled = IsWindowEnabled() ? true : false;
bool bPushed =
		( bPressed )
		|| GetDroppedState()
		;
	if( !bEnabled )
	{
		bPushed = false;
		bHover = false;
	}
	if( CExtPopupMenuWnd::IsMenuTracking() )
		bHover = false;
	else if( !bHover )
	{
		HWND hWndFocus = ::GetFocus();
		if(		hWndFocus == GetSafeHwnd()
			||	::IsChild( GetSafeHwnd(), hWndFocus )
			)
			bHover = true;
	}

	g_PaintManager->PaintComboFrame(
		*pDC,
		rectClient,
		bHover,
		bPushed,
		bEnabled
		);

	if( bCallReleaseDC )
		ReleaseDC( pDC );
}

BOOL CExtComboBox::PreTranslateMessage(MSG* pMsg) 
{
	if( m_bEnableAutoComplete )
	{
		if( pMsg->message == WM_SYSCHAR )
			return TRUE;
		if( pMsg->message == WM_KEYDOWN )
		{
			m_bAutoComplete = true;
			int nVirtKey = (int) pMsg->wParam;
			if(nVirtKey == VK_DELETE || nVirtKey == VK_BACK)
				m_bAutoComplete = false;
		} // if( pMsg->message == WM_KEYDOWN )
	} // if( m_bEnableAutoComplete )
	else
	{
		m_bAutoComplete = false;
	} // else from if( m_bEnableAutoComplete )

		
	return CComboBox::PreTranslateMessage(pMsg);
}

void CExtComboBox::OnEditCtrlUpdate() 
{
	if( !m_bEnableAutoComplete )
	{
		Default();
		return;
	}
	if( !m_bAutoComplete ) 
		return;
CString str;
	GetWindowText( str );
int nLength = str.GetLength();
DWORD dwCurSel = GetEditSel();
WORD dStart = LOWORD( dwCurSel );
WORD dEnd   = HIWORD( dwCurSel );
	if( SelectString( -1, str ) == CB_ERR )
	{
		SetWindowText( str );
		if( dwCurSel != CB_ERR )
			SetEditSel( dStart, dEnd );
	}
	if( dEnd < nLength && dwCurSel != CB_ERR )
		SetEditSel( dStart, dEnd );
	else
		SetEditSel( nLength, -1 );
}

BOOL CExtComboBox::OnEraseBkgnd(CDC* pDC) 
{
	pDC;
	return TRUE;
}

LRESULT CExtComboBox::WindowProc(UINT message, WPARAM wParam, LPARAM lParam) 
{
	switch( message )
	{
	case WM_CONTEXTMENU:
		if( OnTrackComboContextMenu(GetSafeHwnd()) )
			return 0;
		break;
	case WM_ENABLE:
		{
			HWND hWndEdit =
				::GetWindow( GetSafeHwnd(), GW_CHILD );
			if( hWndEdit == NULL || !::IsWindow(hWndEdit) )
				break;
			::EnableWindow( hWndEdit, TRUE );
			::SendMessage( hWndEdit, EM_SETREADONLY, !wParam, 0L );
			Invalidate();
		}
		break;
	case WM_LBUTTONUP:
		Invalidate();
		break;
	case WM_MOUSEMOVE:
		SetTimer( m_dwUpdateTimer, m_dwUpdatePeriod, NULL );
		OnTimer( m_dwUpdateTimer );
		break;
	case WM_SETFOCUS:
		{
			LRESULT lResult =
				CComboBox::WindowProc(message, wParam, lParam);;
			//_OnDrawComboImpl( false, true );
			_OnDrawComboImpl(
				GetDroppedState() ? true : false,
				true
				);
			return lResult;
		}
	case WM_KILLFOCUS:
		{
			LRESULT lResult =
				CComboBox::WindowProc(message, wParam, lParam);;
			if( GetDroppedState() )
				ShowDropDown( FALSE );
			//_OnDrawComboImpl( false, false );
			_OnDrawComboImpl(
				GetDroppedState() ? true : false,
				false
				);
			return lResult;
		}
	case WM_LBUTTONDOWN:
		{
			ASSERT( CBS_SIMPLE       == 0x0001L );
			ASSERT( CBS_DROPDOWN     == 0x0002L );
			ASSERT( CBS_DROPDOWNLIST == 0x0003L );
			DWORD dwStyle = GetStyle();
			DWORD dwType = dwStyle & 0x0003L;
			if( dwType == CBS_DROPDOWN || dwType == CBS_DROPDOWNLIST )
			{
				HWND hWndOwn = GetSafeHwnd();
				ASSERT( hWndOwn != NULL && ::IsWindow(hWndOwn) );

				CWnd * pWndParent = GetParent();
				if( pWndParent != NULL )
				{
					pWndParent->SendMessage( WM_CANCELMODE );
					if( !::IsWindow(hWndOwn) )
						return TRUE;
				}

				BOOL bDropped = GetDroppedState();
				SetFocus();
				_OnDrawComboImpl(
					(!bDropped) ? true : false,
					true
					);
				::PostMessage(hWndOwn, CB_SHOWDROPDOWN, !bDropped, 0);
				return TRUE;
			} // if( dwType == CBS_DROPDOWN || dwType == CBS_DROPDOWNLIST )
		}
		break;
	} // switch( message )
	
	return CComboBox::WindowProc(message, wParam, lParam);
}

void CExtComboBox::PreSubclassWindow() 
{
	
	CComboBox::PreSubclassWindow();

	if( m_pInnerEditHook == NULL )
	{
		HWND hWndEdit =
			::GetWindow( GetSafeHwnd(), GW_CHILD );
		if( hWndEdit != NULL && ::IsWindow(hWndEdit) )
		{
			ASSERT( m_pInnerEditHook == NULL );
			m_pInnerEditHook = new CExtComboEditCtrlHook;
			VERIFY(
				m_pInnerEditHook->SubclassWindow( hWndEdit )
				);
		} // if( hWndEdit != NULL && ::IsWindow(hWndEdit) )
	} // if( m_pInnerEditHook == NULL )
}

bool CExtComboBox::OnTrackComboContextMenu(
	HWND hWndSrc // handle of combo window or child edit control
	)
{
	ASSERT_VALID( this );
	ASSERT( GetSafeHwnd() != NULL );
	ASSERT( ::IsWindow(GetSafeHwnd()) );
	ASSERT( hWndSrc != NULL );
	ASSERT( ::IsWindow(hWndSrc) );
	hWndSrc;
	return false;

//
//
// Commented code implements Prof-UIS cool menu tracking
// (idea not finished through CCmdUI issues)
//
//
//	if( m_pInnerEditHook == NULL || hWndSrc == GetSafeHwnd()
//		|| m_pInnerEditHook->GetSafeHwnd() != hWndSrc
//		)
//		return false; // default processing
//
//	// track edit context menu
//
//	ASSERT( hWndSrc == m_pInnerEditHook->GetSafeHwnd() );
//HWND hWndMenuTrack = hWndSrc;
//
//CExtPopupMenuWnd * pPopup = new CExtPopupMenuWnd;
//	VERIFY(
//		pPopup->CreatePopupMenu( hWndMenuTrack )
//		);
//
//static struct {
//	UINT m_nCmdID;
//	LPCTSTR m_sMenuText;
//} arrCmds[] = {
//	{ ID_EDIT_UNDO, _T("&Undo") },
//	{ ID_EDIT_REDO, _T("&Redo") },
//	{ ID_SEPARATOR, NULL },
//	{ ID_EDIT_CUT, _T("Cu&t") },
//	{ ID_EDIT_COPY, _T("&Copy") },
//	{ ID_EDIT_PASTE, _T("&Paste") },
//	{ ID_EDIT_CLEAR, _T("Cl&ear") },
//	{ ID_SEPARATOR, NULL },
//	{ ID_EDIT_SELECT_ALL, _T("Select &All") },
//};
//	for( int i = 0; i < sizeof(arrCmds)/sizeof(arrCmds[0]); i++ )
//	{
//		UINT nCmdID = arrCmds[i].m_nCmdID;
//		if( nCmdID == ID_SEPARATOR )
//		{
//			VERIFY(
//				pPopup->ItemInsert( ID_SEPARATOR )
//				);
//			continue;
//		} // if( nCmdID != ID_SEPARATOR )
//		CExtCmdManager::cmd_t * p_cmd =
//			g_CmdManager->CmdGetPtr(
//				g_CmdManager->ProfileNameFromWnd(GetSafeHwnd()),
//				nCmdID
//				);
//		if( p_cmd == NULL )
//			p_cmd =
//				g_CmdManager->CmdAllocPtr(
//					g_CmdManager->ProfileNameFromWnd(GetSafeHwnd()),
//					nCmdID
//					);
//		ASSERT( p_cmd != NULL );
//		if( p_cmd->m_sMenuText.IsEmpty() )
//		{
//			p_cmd->m_sMenuText = arrCmds[i].m_sMenuText;
//			ASSERT( !p_cmd->m_sMenuText.IsEmpty() );
//		}
//		p_cmd->StateSetBasic();
//		VERIFY(
//			pPopup->ItemInsert( nCmdID )
//			);
//	}
//
//POINT point;
//	VERIFY( ::GetCursorPos(&point) );
//	pPopup->TrackPopupMenu(
//		0, point.x, point.y,
//		hWndMenuTrack
//		);
//
//	return true;
//
//
//

}

By viewing downloads associated with this article you agree to the Terms of Service and the article's licence.

If a file you wish to view isn't highlighted, and is a text file (not binary), please let us know and we'll add colourisation support for it.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here


Written By
Architect Foss Software Inc
Ukraine Ukraine
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions