leocad/win/terrctrl.cpp
2011-09-07 21:06:51 +00:00

1853 lines
43 KiB
C++

// TerrCtrl.cpp : implementation file
//
#include "stdafx.h"
#include "LeoCAD.h"
#include "TerrCtrl.h"
#include "IPEdit.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
#define IDC_INPLACE_EDIT 8 // ID of inplace edit control
#define GRIDSIZE 28
#define ControlPoint(row, col) m_pControl[row-1][(col-1)*3+2]
#define GetFixedRowHeight() GRIDSIZE
#define GetFixedColumnWidth() GRIDSIZE
#define m_nFixedRows 1
#define m_nFixedCols 1
#define GetRowHeight(a) GRIDSIZE
#define GetColumnWidth(a) GRIDSIZE
#define GetVirtualWidth() m_nCols*GRIDSIZE
#define GetVirtualHeight() m_nRows*GRIDSIZE
#define m_nMargin 1
#define GetFixedRowCount() 1
#define GetFixedColumnCount() 1
#define GetColumnCount() m_nCols
#define GetRowCount() m_nRows
/////////////////////////////////////////////////////////////////////////////
// CTerrainCtrl
CTerrainCtrl::CTerrainCtrl()
{
RegisterWindowClass();
m_nRows = 0;
m_nCols = 0;
m_nVScrollMax = 0; // Scroll position
m_nHScrollMax = 0;
m_MouseMode = MOUSE_NOTHING;
m_pControl = NULL;
/*
m_nMargin = 0; // cell padding
m_nRowsPerWheelNotch = GetMouseScrollLines(); // Get the number of lines
// per mouse wheel notch to scroll
m_bHandleTabKey = TRUE;
m_bTitleTips = TRUE; // show cell title tips
m_nTimerID = 0; // For drag-selection
m_nTimerInterval = 25; // (in milliseconds)
m_crShadow = ::GetSysColor(COLOR_3DSHADOW);
m_crGridColour = RGB(0,0,0);
SetTextColor(m_crWindowText);
SetBkColor(m_crShadow);
SetTextBkColor(RGB(0xFF, 0xFF, 0xE0));
*/
// Set the colours
m_crFixedBkColour = GetSysColor(COLOR_3DFACE);
m_crFixedTextColour = GetSysColor(COLOR_WINDOWTEXT);
m_crTextBkColour = GetSysColor(COLOR_WINDOW);
// Initially use the system message font for the GridCtrl font
NONCLIENTMETRICS ncm;
LOGFONT lf;
ncm.cbSize = sizeof(NONCLIENTMETRICS);
VERIFY(SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICS), &ncm, 0));
memcpy(&lf, &(ncm.lfMessageFont), sizeof(LOGFONT));
// lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
m_Font.CreateFontIndirect(&lf);
// Set up the initial grid size
SetRowCount(5);
SetColumnCount(5);
/*
// set initial selection range (ie. none)
m_SelectedCellMap.RemoveAll();
m_PrevSelectedCellMap.RemoveAll();
*/
}
CTerrainCtrl::~CTerrainCtrl()
{
/*
DeleteAllItems();
*/
DestroyWindow();
m_Font.DeleteObject();
}
// Register the window class if it has not already been registered.
BOOL CTerrainCtrl::RegisterWindowClass()
{
WNDCLASS wndcls;
HINSTANCE hInst = AfxGetResourceHandle();
if (!(::GetClassInfo(hInst, TERRAINCTRL_CLASSNAME, &wndcls)))
{
// otherwise we need to register a new class
wndcls.style = CS_DBLCLKS | CS_HREDRAW | CS_VREDRAW;
wndcls.lpfnWndProc = ::DefWindowProc;
wndcls.cbClsExtra = wndcls.cbWndExtra = 0;
wndcls.hInstance = hInst;
wndcls.hIcon = NULL;
wndcls.hCursor = LoadCursor(NULL,IDC_ARROW);
wndcls.hbrBackground = (HBRUSH) (COLOR_3DFACE + 1);
wndcls.lpszMenuName = NULL;
wndcls.lpszClassName = TERRAINCTRL_CLASSNAME;
if (!AfxRegisterClass(&wndcls)) {
AfxThrowResourceException();
return FALSE;
}
}
return TRUE;
}
BEGIN_MESSAGE_MAP(CTerrainCtrl, CWnd)
//{{AFX_MSG_MAP(CTerrainCtrl)
ON_WM_PAINT()
ON_WM_ERASEBKGND()
ON_WM_SIZE()
ON_WM_GETDLGCODE()
ON_WM_HSCROLL()
ON_WM_VSCROLL()
ON_WM_LBUTTONDOWN()
ON_WM_LBUTTONUP()
ON_WM_MOUSEMOVE()
ON_WM_KEYDOWN()
ON_WM_CHAR()
//}}AFX_MSG_MAP
ON_MESSAGE(WM_GETFONT, OnGetFont)
ON_MESSAGE(WM_LC_EDIT_CLOSED, OnEditClosed)
END_MESSAGE_MAP()
/////////////////////////////////////////////////////////////////////////////
// CTerrainCtrl message handlers
BOOL CTerrainCtrl::Create(const RECT& rect, CWnd* pParentWnd, UINT nID, DWORD dwStyle)
{
ASSERT(pParentWnd->GetSafeHwnd());
if (!CWnd::Create(TERRAINCTRL_CLASSNAME, NULL, dwStyle, rect, pParentWnd, nID))
return FALSE;
// Create titletips
#ifdef GRIDCONTROL_USE_TITLETIPS
if (m_bTitleTips)
m_TitleTip.Create(this);
#endif
ResetScrollBars();
return TRUE;
}
void CTerrainCtrl::PreSubclassWindow()
{
CWnd::PreSubclassWindow();
/*
HFONT hFont = ::CreateFontIndirect(&m_Logfont);
OnSetFont((LPARAM)hFont, 0);
DeleteObject(hFont);
*/
ResetScrollBars();
}
BOOL CTerrainCtrl::SubclassWindow(HWND hWnd)
{
if (!CWnd::SubclassWindow(hWnd))
return FALSE;
#ifdef GRIDCONTROL_USE_TITLETIPS
if (m_bTitleTips && !IsWindow(m_TitleTip.m_hWnd))
m_TitleTip.Create(this);
#endif
return TRUE;
}
LRESULT CTerrainCtrl::OnGetFont(WPARAM /*wParam*/, LPARAM /*lParam*/)
{
return (LRESULT) (HFONT) m_Font;
}
void CTerrainCtrl::OnPaint()
{
CPaintDC dc(this); // device context for painting
CDC MemDC;
CRect rect;
CBitmap bitmap, *pOldBitmap;
dc.GetClipBox(&rect);
MemDC.CreateCompatibleDC(&dc);
bitmap.CreateCompatibleBitmap(&dc, rect.Width(), rect.Height());
pOldBitmap = MemDC.SelectObject(&bitmap);
MemDC.SetWindowOrg(rect.left, rect.top);
OnDraw(&MemDC);
dc.BitBlt(rect.left, rect.top, rect.Width(), rect.Height(), &MemDC, rect.left, rect.top, SRCCOPY);
MemDC.SelectObject(pOldBitmap);
}
BOOL CTerrainCtrl::OnEraseBkgnd(CDC* pDC)
{
return TRUE; // Don't erase the background.
}
// Custom background erasure. This gets called from within the OnDraw function,
// since we will (most likely) be using a memory DC to stop flicker. If we just
// erase the background normally through OnEraseBkgnd, and didn't fill the memDC's
// selected bitmap with colour, then all sorts of vis problems would occur
void CTerrainCtrl::EraseBkgnd(CDC* pDC)
{
CRect VisRect, ClipRect, rect;
CBrush FixedBack(m_crFixedBkColour), TextBack(m_crTextBkColour);
if (pDC->GetClipBox(ClipRect) == ERROR)
return;
int nFixedColumnWidth = GetFixedColumnWidth();
int nFixedRowHeight = GetFixedRowHeight();
GetClientRect(VisRect);
VisRect.top = nFixedRowHeight;
VisRect.left = nFixedColumnWidth;
// Draw Fixed columns background
if (ClipRect.left < nFixedColumnWidth && ClipRect.top < VisRect.bottom)
pDC->FillRect(CRect(ClipRect.left, ClipRect.top, nFixedColumnWidth, VisRect.bottom),
&FixedBack);
// Draw Fixed rows background
if (ClipRect.top < nFixedRowHeight &&
ClipRect.right > nFixedColumnWidth && ClipRect.left < VisRect.right)
pDC->FillRect(CRect(nFixedColumnWidth-1, ClipRect.top, VisRect.right, nFixedRowHeight),
&FixedBack);
// Draw non-fixed cell background
if (rect.IntersectRect(VisRect, ClipRect))
{
CRect CellRect(max(nFixedColumnWidth, rect.left), max(nFixedRowHeight, rect.top),
rect.right, rect.bottom);
pDC->FillRect(CellRect, &TextBack);
}
}
void CTerrainCtrl::OnSize(UINT nType, int cx, int cy)
{
if (::IsWindow(GetSafeHwnd()) && GetFocus()->GetSafeHwnd() != GetSafeHwnd())
SetFocus(); // Auto-destroy any InPlaceEdit's
CWnd::OnSize(nType, cx, cy);
ResetScrollBars();
}
UINT CTerrainCtrl::OnGetDlgCode()
{
UINT nCode = DLGC_WANTARROWS | DLGC_WANTCHARS;
/*
if (m_bHandleTabKey && !IsCTRLpressed())
nCode |= DLGC_WANTTAB;
*/
return nCode;
}
// wParam = key pressed, lParam = modified
LRESULT CTerrainCtrl::OnEditClosed(WPARAM wParam, LPARAM lParam)
{
// In case OnEndInPlaceEdit called as window is being destroyed
if (!IsWindow(GetSafeHwnd()))
return TRUE;
// Only set as modified if it actually was, and ESC was not hit.
if ((wParam != VK_ESCAPE) && (lParam == TRUE))
{
CWnd* pParent = GetOwner();
if (pParent)
pParent->SendMessage(WM_LC_EDIT_CLOSED);
// SetModified(TRUE);
}
switch (wParam)
{
case VK_DOWN:
case VK_UP:
case VK_RIGHT:
case VK_LEFT:
case VK_NEXT:
case VK_PRIOR:
case VK_HOME:
case VK_END:
OnKeyDown(wParam, 0, 0);
EditCell(m_idCurrentCell.row, m_idCurrentCell.col, wParam);
}
return TRUE;
}
// Handle horz scrollbar notifications
void CTerrainCtrl::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
if (GetFocus()->GetSafeHwnd() != GetSafeHwnd())
SetFocus(); // Auto-destroy any InPlaceEdit's
#ifdef GRIDCONTROL_USE_TITLETIPS
m_TitleTip.Hide(); // hide any titletips
#endif
int scrollPos = GetScrollPos32(SB_HORZ);
CELLID idTopLeft = GetTopleftNonFixedCell();
CRect rect;
GetClientRect(rect);
switch (nSBCode)
{
case SB_LINERIGHT:
if (scrollPos < m_nHScrollMax)
{
int xScroll = GetColumnWidth(nTopLeftCol);
SetScrollPos32(SB_HORZ, scrollPos + xScroll);
if (GetScrollPos32(SB_HORZ) == scrollPos)
break; // didn't work
rect.left = GetFixedColumnWidth() + xScroll;
ScrollWindow(-xScroll, 0, rect);
rect.left = rect.right - xScroll;
InvalidateRect(rect);
} break;
case SB_LINELEFT:
if (scrollPos > 0 && idTopLeft.col > GetFixedColumnCount())
{
int xScroll = GetColumnWidth(nTopLeftCol-1);
SetScrollPos32(SB_HORZ, max(0,scrollPos - xScroll));
rect.left = GetFixedColumnWidth();
ScrollWindow(xScroll, 0, rect);
rect.right = rect.left + xScroll;
InvalidateRect(rect);
} break;
case SB_PAGERIGHT:
if (scrollPos < m_nHScrollMax)
{
rect.left = GetFixedColumnWidth();
int offset = rect.Width();
int pos = min(m_nHScrollMax, scrollPos + offset);
SetScrollPos32(SB_HORZ, pos);
rect.left = GetFixedColumnWidth();
InvalidateRect(rect);
} break;
case SB_PAGELEFT:
if (scrollPos > 0)
{
rect.left = GetFixedColumnWidth();
int offset = -rect.Width();
int pos = max(0, scrollPos + offset);
SetScrollPos32(SB_HORZ, pos);
rect.left = GetFixedColumnWidth();
InvalidateRect(rect);
} break;
case SB_THUMBPOSITION:
case SB_THUMBTRACK:
{
SetScrollPos32(SB_HORZ, GetScrollPos32(SB_HORZ, TRUE));
rect.left = GetFixedColumnWidth();
InvalidateRect(rect);
} break;
case SB_LEFT:
if (scrollPos > 0)
{
SetScrollPos32(SB_HORZ, 0);
Invalidate();
} break;
case SB_RIGHT:
if (scrollPos < m_nHScrollMax)
{
SetScrollPos32(SB_HORZ, m_nHScrollMax);
Invalidate();
} break;
}
}
// Handle vert scrollbar notifications
void CTerrainCtrl::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
if (GetFocus()->GetSafeHwnd() != GetSafeHwnd())
SetFocus(); // Auto-destroy any InPlaceEdit's
#ifdef GRIDCONTROL_USE_TITLETIPS
m_TitleTip.Hide(); // hide any titletips
#endif
// Get the scroll position ourselves to ensure we get a 32 bit value
int scrollPos = GetScrollPos32(SB_VERT);
CELLID idTopLeft = GetTopleftNonFixedCell();
CRect rect;
GetClientRect(rect);
switch (nSBCode)
{
case SB_LINEDOWN:
if (scrollPos < m_nVScrollMax)
{
int yScroll = GetRowHeight(nTopLeftRow);
SetScrollPos32(SB_VERT, scrollPos + yScroll);
if (GetScrollPos32(SB_VERT) == scrollPos)
break; // didn't work
rect.top = GetFixedRowHeight() + yScroll;
ScrollWindow( 0, -yScroll, rect);
rect.top = rect.bottom - yScroll;
InvalidateRect(rect);
} break;
case SB_LINEUP:
if (scrollPos > 0 && idTopLeft.row > GetFixedRowCount())
{
int yScroll = GetRowHeight(nTopLeftRow-1);
SetScrollPos32(SB_VERT, max(0, scrollPos - yScroll));
rect.top = GetFixedRowHeight();
ScrollWindow(0, yScroll, rect);
rect.bottom = rect.top + yScroll;
InvalidateRect(rect);
} break;
case SB_PAGEDOWN:
if (scrollPos < m_nVScrollMax)
{
rect.top = GetFixedRowHeight();
scrollPos = min(m_nVScrollMax, scrollPos + rect.Height());
SetScrollPos32(SB_VERT, scrollPos);
rect.top = GetFixedRowHeight();
InvalidateRect(rect);
} break;
case SB_PAGEUP:
if (scrollPos > 0)
{
rect.top = GetFixedRowHeight();
int offset = -rect.Height();
int pos = max(0, scrollPos + offset);
SetScrollPos32(SB_VERT, pos);
rect.top = GetFixedRowHeight();
InvalidateRect(rect);
} break;
case SB_THUMBPOSITION:
case SB_THUMBTRACK:
{
SetScrollPos32(SB_VERT, GetScrollPos32(SB_VERT, TRUE));
rect.top = GetFixedRowHeight();
InvalidateRect(rect);
} break;
case SB_TOP:
if (scrollPos > 0)
{
SetScrollPos32(SB_VERT, 0);
Invalidate();
} break;
case SB_BOTTOM:
if (scrollPos < m_nVScrollMax)
{
SetScrollPos32(SB_VERT, m_nVScrollMax);
Invalidate();
}
}
}
void CTerrainCtrl::OnDraw(CDC* pDC)
{
int row,col;
CRect clientRect;
GetClientRect(clientRect);
CRect clipRect;
if (pDC->GetClipBox(&clipRect) == ERROR)
return;
// OnEraseBkgnd does nothing, so erase bkgnd here.
// This necessary since we may be using a Memory DC.
EraseBkgnd(pDC);
int nFixedRowHeight = GetFixedRowHeight();
int nFixedColWidth = GetFixedColumnWidth();
CELLID idTopLeft = GetTopleftNonFixedCell();
/*
CRect VisRect;
CCellRange VisCellRange = GetVisibleNonFixedCellRange(VisRect);
int maxVisibleRow = VisCellRange.GetMaxRow(),
maxVisibleCol = VisCellRange.GetMaxCol();
*/
// calc bottom
int i, bottom = GetFixedRowHeight();
for (i = idTopLeft.row; i < m_nRows; i++)
{
bottom += GetRowHeight(i);
if (bottom >= clientRect.bottom)
{
bottom = clientRect.bottom;
break;
}
}
int maxVisibleRow = min(i, m_nRows - 1);
// calc right
int right = GetFixedColumnWidth();
for (i = idTopLeft.col; i < m_nCols; i++)
{
right += GetColumnWidth(i);
if (right >= clientRect.right)
{
right = clientRect.right;
break;
}
}
int maxVisibleCol = min(i, m_nCols - 1);
// draw top-left cell
CRect rect;
rect.top = 0;
rect.bottom = GetRowHeight(0)-1;
rect.left = 0;
rect.right = GetColumnWidth(0)-1;
DrawFixedCell(pDC, 0, 0, rect);
// draw fixed column cells
rect.bottom = nFixedRowHeight-1;
for (row = idTopLeft.row; row <= maxVisibleRow; row++)
{
rect.top = rect.bottom+1;
rect.bottom = rect.top + GetRowHeight(row)-1;
// rect.bottom = bottom pixel of previous row
if (rect.top > clipRect.bottom) break; // Gone past cliprect
if (rect.bottom < clipRect.top) continue; // Reached cliprect yet?
rect.left = 0;
rect.right = GetColumnWidth(0)-1;
if (rect.left > clipRect.right) break; // gone past cliprect
if (rect.right < clipRect.left) continue; // Reached cliprect yet?
DrawFixedCell(pDC, row, 0, rect);
}
// draw end of column
if (rect.bottom < clientRect.bottom)
{
rect.top = rect.bottom+1;
rect.bottom = clientRect.bottom-1;
if (rect.top < clipRect.bottom)
if (rect.bottom > clipRect.top)
if (rect.left < clipRect.right)
if (rect.right > clipRect.left)
DrawFixedCell(pDC, 0, 0, rect);
}
// draw fixed row cells 0..m_nFixedRows, m_nFixedCols..n
rect.top = 0;
rect.bottom = GetRowHeight(0)-1;
// rect.bottom = bottom pixel of previous row
if (rect.top < clipRect.bottom) // Gone past cliprect
if (rect.bottom > clipRect.top) // Reached cliprect yet?
{
rect.right = nFixedColWidth-1;
for (col = idTopLeft.col; col <= maxVisibleCol; col++)
{
rect.left = rect.right+1;
rect.right = rect.left + GetColumnWidth(col)-1;
if (rect.left > clipRect.right) break; // gone past cliprect
if (rect.right < clipRect.left) continue; // Reached cliprect yet?
DrawFixedCell(pDC, 0, col, rect);
}
}
// draw end of row
if (rect.right < clientRect.right)
{
rect.left = rect.right+1;
rect.right = clientRect.right - 1;
if (rect.top < clipRect.bottom)
if (rect.bottom > clipRect.top)
if (rect.left < clipRect.right)
if (rect.right > clipRect.left)
DrawFixedCell(pDC, 0, 0, rect);
}
// draw rest of non-fixed cells
rect.bottom = nFixedRowHeight-1;
for (row = idTopLeft.row; row <= maxVisibleRow; row++)
{
rect.top = rect.bottom+1;
rect.bottom = rect.top + GetRowHeight(row)-1;
// rect.bottom = bottom pixel of previous row
if (rect.top > clipRect.bottom) break; // Gone past cliprect
if (rect.bottom < clipRect.top) continue; // Reached cliprect yet?
rect.right = nFixedColWidth-1;
for (col = idTopLeft.col; col <= maxVisibleCol; col++)
{
rect.left = rect.right+1;
rect.right = rect.left + GetColumnWidth(col)-1;
if (rect.left > clipRect.right) break; // gone past cliprect
if (rect.right < clipRect.left) continue; // Reached cliprect yet?
DrawCell(pDC, row, col, rect);
}
}
}
// Get/Set scroll position using 32 bit functions
int CTerrainCtrl::GetScrollPos32(int nBar, BOOL bGetTrackPos /* = FALSE */)
{
SCROLLINFO si;
si.cbSize = sizeof(SCROLLINFO);
if (bGetTrackPos)
{
if (GetScrollInfo(nBar, &si, SIF_TRACKPOS))
return si.nTrackPos;
}
else
{
if (GetScrollInfo(nBar, &si, SIF_POS))
return si.nPos;
}
return 0;
}
BOOL CTerrainCtrl::SetScrollPos32(int nBar, int nPos, BOOL bRedraw /* = TRUE */)
{
SCROLLINFO si;
si.cbSize = sizeof(SCROLLINFO);
si.fMask = SIF_POS;
si.nPos = nPos;
return SetScrollInfo(nBar, &si, bRedraw);
}
void CTerrainCtrl::ResetScrollBars()
{
if (!::IsWindow(GetSafeHwnd()))
return;
CRect rect;
GetClientRect(rect);
rect.left += GetFixedColumnWidth();
rect.top += GetFixedRowHeight();
if (rect.left >= rect.right || rect.top >= rect.bottom)
return;
CRect VisibleRect(GetFixedColumnWidth(), GetFixedRowHeight(), rect.right, rect.bottom);
CRect VirtualRect(GetFixedColumnWidth(), GetFixedRowHeight(), GetVirtualWidth(), GetVirtualHeight());
/*
CCellRange visibleCells = GetUnobstructedNonFixedCellRange();
if (!IsValid(visibleCells)) return;
*/
SCROLLINFO si;
si.cbSize = sizeof(SCROLLINFO);
si.fMask = SIF_PAGE;
si.nPage = VisibleRect.Width();
SetScrollInfo(SB_HORZ, &si, FALSE);
si.nPage = VisibleRect.Height();
SetScrollInfo(SB_VERT, &si, FALSE);
if (VisibleRect.Height() < VirtualRect.Height())
m_nVScrollMax = VirtualRect.Height()-1;//-VisibleRect.Height();//+GetRowHeight(0);
else
m_nVScrollMax = 0;
if (VisibleRect.Width() < VirtualRect.Width())
m_nHScrollMax = VirtualRect.Width()-1;//-VisibleRect.Width();//+GetColumnWidth(0);
else
m_nHScrollMax = 0;
ASSERT(m_nVScrollMax < INT_MAX && m_nHScrollMax < INT_MAX); // This should be fine :)
SetScrollRange(SB_VERT, 0, m_nVScrollMax, TRUE);
SetScrollRange(SB_HORZ, 0, m_nHScrollMax, TRUE);
}
BOOL CTerrainCtrl::DrawFixedCell(CDC* pDC, int nRow, int nCol, CRect rect, BOOL bEraseBk)
{
if (bEraseBk)
{
CBrush brush(m_crFixedBkColour);
pDC->FillRect(rect, &brush);
}
pDC->SetTextColor(m_crFixedTextColour);
int nSavedDC = pDC->SaveDC();
/*
// Create the appropriate font and select into DC
LOGFONT lf, *pLF = GetItemFont(nRow, nCol);
if (pLF)
memcpy(&lf, pLF, sizeof(LOGFONT));
else
memcpy(&lf, &m_Logfont, sizeof(LOGFONT));
CCellID FocusCell = GetFocusCell();
if (FocusCell.row == nRow || FocusCell.col == nCol)
lf.lfWeight = SELECTED_CELL_FONT_WEIGHT;
CFont Font;
Font.CreateFontIndirect(&lf);
pDC->SelectObject(&Font);
if (IsValid(FocusCell) && (FocusCell.row == nRow || FocusCell.col == nCol))
{
rect.right++; rect.bottom++;
pDC->DrawEdge(rect, EDGE_RAISED, BF_RECT);
rect.DeflateRect(1,1);
}
else
*/ {
CPen lightpen(PS_SOLID, 1, ::GetSysColor(COLOR_3DHIGHLIGHT)),
darkpen(PS_SOLID, 1, ::GetSysColor(COLOR_3DDKSHADOW)),
*pOldPen = pDC->GetCurrentPen();
pDC->SelectObject(&lightpen);
pDC->MoveTo(rect.right, rect.top);
pDC->LineTo(rect.left, rect.top);
pDC->LineTo(rect.left, rect.bottom);
pDC->SelectObject(&darkpen);
pDC->MoveTo(rect.right, rect.top);
pDC->LineTo(rect.right, rect.bottom);
pDC->LineTo(rect.left, rect.bottom);
pDC->SelectObject(pOldPen);
rect.DeflateRect(1,1);
}
pDC->SetBkMode(TRANSPARENT);
rect.DeflateRect(m_nMargin, 0);
if ((nRow != 0 || nCol != 0) && nRow < m_nRows && nCol < m_nCols)
{
char szText[10];
sprintf(szText, "%d", nRow == 0 ? nCol : nRow);
DrawText(pDC->m_hDC, szText, -1, rect, DT_CENTER|DT_VCENTER|DT_SINGLELINE);
}
pDC->RestoreDC(nSavedDC);
return TRUE;
}
BOOL CTerrainCtrl::DrawCell(CDC* pDC, int nRow, int nCol, CRect rect, BOOL bEraseBk)
{
COLORREF TextClr = m_crFixedTextColour;
int nSavedDC = pDC->SaveDC();
pDC->SetBkMode(TRANSPARENT);
if (m_RowData[nRow][nCol].state & GS_FOCUSED)
{
rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
if (bEraseBk)
{
CBrush brush(m_crTextBkColour);
pDC->FillRect(rect, &brush);
}
rect.right--; rect.bottom--;
pDC->SelectStockObject(BLACK_PEN);
pDC->SelectStockObject(NULL_BRUSH);
pDC->Rectangle(rect);
pDC->SetTextColor(TextClr);
rect.DeflateRect(1,1);
}
else if (m_RowData[nRow][nCol].state & GS_SELECTED)
{
rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
pDC->FillSolidRect(rect, ::GetSysColor(COLOR_HIGHLIGHT));
rect.right--; rect.bottom--;
pDC->SetTextColor(::GetSysColor(COLOR_HIGHLIGHTTEXT));
}
else
{
rect.right++; rect.bottom++; // FillRect doesn't draw RHS or bottom
if (bEraseBk)
{
CBrush brush(m_crTextBkColour);
pDC->FillRect(rect, &brush);
}
rect.right--; rect.bottom--;
pDC->SetTextColor(TextClr);
}
/*
if (Item.state & GVIS_DROPHILITED)
{
pDC->SelectStockObject(BLACK_PEN);
pDC->SelectStockObject(NULL_BRUSH);
pDC->Rectangle(rect);
}
*/
CPen lightpen(PS_SOLID, 1, RGB(0xE0, 0xE0, 0xE0)),
darkpen(PS_SOLID, 1, RGB(0,0,0)),
*pOldPen = pDC->GetCurrentPen();
int left = nCol == 1 ? GRIDSIZE/2-1 : 0;
int top = nRow == 1 ? GRIDSIZE/2-1 : 0;
int right = nCol == m_nCols-1 ? GRIDSIZE/2-1 : -1;
int bottom = nRow == m_nRows-1 ? GRIDSIZE/2-1 : -1;
if ((nRow-1) % 3)
{
pDC->SelectObject(&lightpen);
pDC->MoveTo(rect.left + left, (rect.top + rect.bottom)/2);
pDC->LineTo(rect.right - right, (rect.top + rect.bottom)/2);
}
if ((nCol-1) % 3)
{
pDC->SelectObject(&lightpen);
pDC->MoveTo((rect.left + rect.right)/2, rect.top + top);
pDC->LineTo((rect.left + rect.right)/2, rect.bottom - bottom);
}
if ((nRow-1) % 3 == 0)
{
pDC->SelectObject(&darkpen);
pDC->MoveTo(rect.left + left, (rect.top + rect.bottom)/2);
pDC->LineTo(rect.right - right, (rect.top + rect.bottom)/2);
}
if ((nCol-1) % 3 == 0)
{
pDC->SelectObject(&darkpen);
pDC->MoveTo((rect.left + rect.right)/2, rect.top + top);
pDC->LineTo((rect.left + rect.right)/2, rect.bottom - bottom);
}
pDC->SelectObject(pOldPen);
/* // Create the appropriate font and select into DC
CFont Font;
LOGFONT *pLF = GetItemFont(nRow, nCol);
if (pLF)
Font.CreateFontIndirect(pLF);
else
Font.CreateFontIndirect(&m_Logfont);
*/
CFont *pOldFont = pDC->SelectObject(&m_Font);
rect.DeflateRect(m_nMargin, 0);
rect.OffsetRect(GRIDSIZE/2, GRIDSIZE/4+2);
char szText[10];
sprintf(szText, "%.0f", ControlPoint(nRow, nCol));
DrawText(pDC->m_hDC, szText, -1, rect, DT_LEFT|DT_VCENTER|DT_SINGLELINE);
pDC->SelectObject(pOldFont);
pDC->RestoreDC(nSavedDC);
return TRUE;
}
BOOL CTerrainCtrl::SetRowCount(int nRows)
{
ASSERT(nRows > 0);
if (nRows == m_nRows)
return TRUE;
if (m_idCurrentCell.row >= nRows)
SetFocusCell(-1,-1);
int addedRows = nRows - m_nRows;
// Change the number of rows.
m_nRows = nRows;
m_RowData.SetSize(m_nRows);
// If we have just added rows, we need to construct new elements for each cell
// and set the default row height
if (addedRows > 0)
{
// initialize row heights and data
int startRow = nRows - addedRows;
for (int row = startRow; row < m_nRows; row++)
{
m_RowData[row].SetSize(m_nCols);
for (int col = 0; col < m_nCols; col++)
m_RowData[row][col].state = 0;
}
}
if (GetSafeHwnd())
{
ResetScrollBars();
Invalidate();
}
return TRUE;
}
BOOL CTerrainCtrl::SetColumnCount(int nCols)
{
ASSERT(nCols > 0);
if (nCols == m_nCols)
return TRUE;
if (m_idCurrentCell.col >= nCols)
SetFocusCell(-1,-1);
int addedCols = nCols - m_nCols;
// Change the number of columns.
m_nCols = nCols;
// Change the number of columns in each row.
for (int i = 0; i < m_nRows; i++)
m_RowData[i].SetSize(nCols);
// If we have just added columns, we need to construct new elements for each cell
// and set the default column width
if (addedCols > 0)
{
int startCol = nCols - addedCols;
for (int row = 0; row < m_nRows; row++)
for (int col = startCol; col < m_nCols; col++)
m_RowData[row][col].state = 0;
}
if (GetSafeHwnd())
{
ResetScrollBars();
Invalidate();
}
return TRUE;
}
CELLID CTerrainCtrl::GetTopleftNonFixedCell()
{
CELLID cell;
int nVertScroll = GetScrollPos(SB_VERT), nHorzScroll = GetScrollPos(SB_HORZ);
int nColumn = m_nFixedCols, nRight = 0;
while (nRight < nHorzScroll && nColumn < (m_nCols-1))
{
nColumn++;
nRight += GetColumnWidth(nColumn);
}
int nRow = m_nFixedRows, nTop = 0;
while (nTop < nVertScroll && nRow < (m_nRows-1))
{
nRow++;
nTop += GetRowHeight(nRow);
}
cell.row = nRow;
cell.col = nColumn;
return cell;
}
void CTerrainCtrl::SetControlPoints(int uCount, int vCount, float** pControl)
{
SetRowCount(uCount+1);
SetColumnCount(vCount+1);
m_pControl = pControl;
InvalidateRect(NULL, FALSE);
}
void CTerrainCtrl::OnLButtonDown(UINT nFlags, CPoint point)
{
/*
HWND hOldFocusWnd = ::GetFocus();
*/
m_LeftClickDownPoint = point;
m_LeftClickDownCell = GetCellFromPt(point);
if (!IsValid(m_LeftClickDownCell))
return;
/*
m_SelectionStartCell = (nFlags & MK_SHIFT)? m_idCurrentCell : m_LeftClickDownCell;
*/
SetFocus(); // Auto-destroy any InPlaceEdit's
// If the user clicks on the current cell, then prepare to edit it.
// (If the user moves the mouse, then dragging occurs)
if (m_LeftClickDownCell == m_idCurrentCell)
{
m_MouseMode = MOUSE_PREPARE_EDIT;
return;
}
else
{
SetFocusCell(-1,-1);
SetFocusCell(max(m_LeftClickDownCell.row, 1),
max(m_LeftClickDownCell.col, 1));
}
/*
// If the user clicks on a selected cell, then prepare to drag it.
// (If the user moves the mouse, then dragging occurs)
if (m_bAllowDragAndDrop && hOldFocusWnd == GetSafeHwnd() &&
GetItemState(m_LeftClickDownCell.row, m_LeftClickDownCell.col) & GVNI_SELECTED)
{
m_MouseMode = MOUSE_PREPARE_DRAG;
return;
}
*/
SetCapture();
// If Ctrl pressed, save the current cell selection. This will get added
/* // to the new cell selection at the end of the cell selection process
m_PrevSelectedCellMap.RemoveAll();
if (nFlags & MK_CONTROL)
{
for (POSITION pos = m_SelectedCellMap.GetStartPosition(); pos != NULL; )
{
DWORD key;
CCellID cell;
m_SelectedCellMap.GetNextAssoc(pos, key, (CCellID&)cell);
m_PrevSelectedCellMap.SetAt(key, cell);
}
}
if (m_LeftClickDownCell.row < GetFixedRowCount())
OnFixedRowClick(m_LeftClickDownCell);
else if (m_LeftClickDownCell.col < GetFixedColumnCount())
OnFixedColumnClick(m_LeftClickDownCell);
else
{
m_MouseMode = m_bListMode? MOUSE_SELECT_ROW : MOUSE_SELECT_CELLS;
OnSelecting(m_LeftClickDownCell);
}
m_nTimerID = SetTimer(WM_LBUTTONDOWN, m_nTimerInterval, 0);
m_LastMousePoint = point;
*/
}
void CTerrainCtrl::OnLButtonUp(UINT nFlags, CPoint point)
{
CWnd::OnLButtonUp(nFlags, point);
ClipCursor(NULL);
if (GetCapture()->GetSafeHwnd() == GetSafeHwnd())
{
ReleaseCapture();
/* KillTimer(m_nTimerID);
m_nTimerID = 0;
*/ }
// m_MouseMode == MOUSE_PREPARE_EDIT only if user clicked down on current cell
// and then didn't move mouse before clicking up (releasing button)
if (m_MouseMode == MOUSE_PREPARE_EDIT)
{
EditCell(m_idCurrentCell.row, m_idCurrentCell.col, VK_LBUTTON);
}
// m_MouseMode == MOUSE_PREPARE_DRAG only if user clicked down on a selected cell
/* // and then didn't move mouse before clicking up (releasing button)
else if (m_MouseMode == MOUSE_PREPARE_DRAG)
{
ResetSelectedRange();
}
*/
m_MouseMode = MOUSE_NOTHING;
SetCursor(::LoadCursor(NULL, IDC_ARROW));
if (!IsValid(m_LeftClickDownCell))
return;
/*
CWnd *pOwner = GetOwner();
if (pOwner && IsWindow(pOwner->m_hWnd))
pOwner->PostMessage(WM_COMMAND, MAKELONG(GetDlgCtrlID(), BN_CLICKED),
(LPARAM) GetSafeHwnd());
*/
}
void CTerrainCtrl::OnMouseMove(UINT nFlags, CPoint point)
{
CRect rect;
GetClientRect(rect);
/*
// If outside client area, return (unless we are drag n dropping)
if (m_MouseMode != MOUSE_DRAGGING && !rect.PtInRect(point))
return;
// If the left mouse button is up, then test to see if row/column sizing is imminent
if (!(nFlags & MK_LBUTTON))
{
if (point.y < GetFixedRowHeight() && m_bAllowColumnResize)
{
CCellID idCurrentCell = GetCellFromPt(point);
CPoint start;
if (!GetCellOrigin(idCurrentCell, &start)) return;
int endx = start.x + GetColumnWidth(idCurrentCell.col);
if ((point.x - start.x <= m_nResizeCaptureRange && idCurrentCell.col != 0) ||
endx - point.x <= m_nResizeCaptureRange)
{
if (m_MouseMode != MOUSE_OVER_COL_DIVIDE)
SetCursor(::LoadCursor(NULL, IDC_SIZEWE));
m_MouseMode = MOUSE_OVER_COL_DIVIDE;
}
else
{
if (m_MouseMode != MOUSE_NOTHING)
SetCursor(::LoadCursor(NULL, IDC_ARROW));
m_MouseMode = MOUSE_NOTHING;
}
}
else if (point.x < GetFixedColumnWidth() && m_bAllowRowResize)
{
CCellID idCurrentCell = GetCellFromPt(point);
CPoint start;
if (!GetCellOrigin(idCurrentCell, &start)) return;
int endy = start.y + GetRowHeight(idCurrentCell.row);
if ((point.y - start.y <= m_nResizeCaptureRange && idCurrentCell.row != 0) ||
endy - point.y <= m_nResizeCaptureRange)
{
if (m_MouseMode != MOUSE_OVER_ROW_DIVIDE)
SetCursor(::LoadCursor(NULL, IDC_SIZENS));
m_MouseMode = MOUSE_OVER_ROW_DIVIDE;
}
else
{
if (m_MouseMode != MOUSE_NOTHING)
SetCursor(::LoadCursor(NULL, IDC_ARROW));
m_MouseMode = MOUSE_NOTHING;
}
}
else
{
if (m_MouseMode != MOUSE_NOTHING)
SetCursor(::LoadCursor(NULL, IDC_ARROW));
m_MouseMode = MOUSE_NOTHING;
}
#ifdef GRIDCONTROL_USE_TITLETIPS
if (m_MouseMode == MOUSE_NOTHING && m_bTitleTips)
{
CCellID idCurrentCell = GetCellFromPt(point);
CRect rect;
if (GetCellRect(idCurrentCell.row, idCurrentCell.col, rect))
m_TitleTip.Show( rect, GetItemText(idCurrentCell.row, idCurrentCell.col), 0);
}
#endif
m_LastMousePoint = point;
return;
}
if (!IsValid(m_LeftClickDownCell))
{
m_LastMousePoint = point;
return;
}
// If the left mouse button is down, the process appropriately
if (nFlags & MK_LBUTTON)
{
switch(m_MouseMode)
{
case MOUSE_SELECT_ALL:
break;
case MOUSE_SELECT_COL:
case MOUSE_SELECT_ROW:
case MOUSE_SELECT_CELLS:
{
CCellID idCurrentCell = GetCellFromPt(point);
if (!IsValid(idCurrentCell))
return;
OnSelecting(idCurrentCell);
// SetFocusCell(max(idCurrentCell.row, m_nFixedRows),
// max(idCurrentCell.col, m_nFixedCols));
if (idCurrentCell.row >= m_nFixedRows &&
idCurrentCell.col >= m_nFixedCols)
SetFocusCell(idCurrentCell);
break;
}
case MOUSE_PREPARE_DRAG:
OnBeginDrag();
break;
}
}
m_LastMousePoint = point;
*/
}
// Sets the currently selected cell
void CTerrainCtrl::SetFocusCell(CELLID cell)
{
SetFocusCell(cell.row, cell.col);
}
void CTerrainCtrl::SetFocusCell(int nRow, int nCol)
{
CELLID cell(nRow, nCol);
if (cell == m_idCurrentCell)
return;
CELLID idPrev = m_idCurrentCell;
m_idCurrentCell = cell;
if (IsValid(idPrev))
{
/* SendMessageToParent(idPrev.row, idPrev.col, GVN_SELCHANGING);
*/
m_RowData[idPrev.row][idPrev.col].state &= ~GS_FOCUSED;
RedrawCell(idPrev);
/*
if (idPrev.col != m_idCurrentCell.col)
for (int row = 0; row < m_nFixedRows; row++)
RedrawCell(row, idPrev.col);
if (idPrev.row != m_idCurrentCell.row)
for (int col = 0; col < m_nFixedCols; col++)
RedrawCell(idPrev.row, col);
*/ }
if (IsValid(m_idCurrentCell))
{
m_RowData[nRow][nCol].state |= GS_FOCUSED;
RedrawCell(m_idCurrentCell);
/*
if (idPrev.col != m_idCurrentCell.col)
for (int row = 0; row < m_nFixedRows; row++)
RedrawCell(row, m_idCurrentCell.col);
if (idPrev.row != m_idCurrentCell.row)
for (int col = 0; col < m_nFixedCols; col++)
RedrawCell(m_idCurrentCell.row, col);
SendMessageToParent(m_idCurrentCell.row, m_idCurrentCell.col, GVN_SELCHANGED);
*/ }
}
BOOL CTerrainCtrl::IsValid(int nRow, int nCol)
{
return (nRow >= 0 && nRow < m_nRows && nCol >= 0 && nCol < m_nCols);
}
BOOL CTerrainCtrl::IsValid(CELLID cell)
{
return IsValid(cell.row, cell.col);
}
// Get cell from point
CELLID CTerrainCtrl::GetCellFromPt(CPoint point, BOOL bAllowFixedCellCheck)
{
CELLID idTopLeft = GetTopleftNonFixedCell();
CELLID cellID; // return value
// calculate column index
int fixedColWidth = GetFixedColumnWidth();
if (point.x < 0 || (!bAllowFixedCellCheck && point.x < fixedColWidth)) // not in window
cellID.col = -1;
else if (point.x < fixedColWidth) // in fixed col
{
int col, xpos = 0;
for (col = 0; col < m_nFixedCols; col++)
{
xpos += GetColumnWidth(col);
if (xpos > point.x) break;
}
cellID.col = col;
}
else // in non-fixed col
{
int col, xpos = fixedColWidth;
for (col = idTopLeft.col; col < m_nCols; col++)
{
xpos += GetColumnWidth(col);
if (xpos > point.x) break;
}
if (col >= GetColumnCount())
cellID.col = -1;
else
cellID.col = col;
}
// calculate row index
int fixedRowHeight = GetFixedRowHeight();
if (point.y < 0 || (!bAllowFixedCellCheck && point.y < fixedRowHeight)) // not in window
cellID.row = -1;
else if (point.y < fixedRowHeight) // in fixed col
{
int row, ypos = 0;
for (row = 0; row < m_nFixedRows; row++)
{
ypos += GetRowHeight(row);
if (ypos > point.y) break;
}
cellID.row = row;
}
else
{
int row, ypos = fixedRowHeight;
for (row = idTopLeft.row; row < GetRowCount(); row++)
{
ypos += GetRowHeight(row);
if (ypos > point.y) break;
}
if (row >= GetRowCount())
cellID.row = -1;
else
cellID.row = row;
}
return cellID;
}
// Forces a redraw of a cell immediately using a direct
// DC construction, or the supplied DC
BOOL CTerrainCtrl::RedrawCell(CELLID cell, CDC* pDC)
{
return RedrawCell(cell.row, cell.col, pDC);
}
BOOL CTerrainCtrl::RedrawCell(int nRow, int nCol, CDC* pDC)
{
BOOL bResult = TRUE;
BOOL bMustReleaseDC = FALSE;
if (!IsCellVisible(nRow, nCol))
return FALSE;
CRect rect;
if (!GetCellRect(nRow, nCol, rect))
return FALSE;
if (!pDC)
{
pDC = GetDC();
if (pDC)
bMustReleaseDC = TRUE;
}
if (pDC)
{
// Redraw cells directly
if (nRow < m_nFixedRows || nCol < m_nFixedCols)
bResult = DrawFixedCell(pDC, nRow, nCol, rect, TRUE);
else
bResult = DrawCell(pDC, nRow, nCol, rect, TRUE);
/*
// Since we have erased the background, we will need to redraw the gridlines
CPen pen;
try {
pen.CreatePen(PS_SOLID, 0, m_crGridColour);
} catch (...) {}
CPen* pOldPen = (CPen*) pDC->SelectObject(&pen);
if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_HORZ)
{
pDC->MoveTo(rect.left, rect.bottom);
pDC->LineTo(rect.right+1, rect.bottom);
}
if (m_nGridLines == GVL_BOTH || m_nGridLines == GVL_VERT)
{
pDC->MoveTo(rect.right, rect.top);
pDC->LineTo(rect.right, rect.bottom+1);
}
pDC->SelectObject(pOldPen);
*/
} else
InvalidateRect(rect, TRUE); // Could not get a DC - invalidate it anyway
// and hope that OnPaint manages to get one
if (bMustReleaseDC)
ReleaseDC(pDC);
return bResult;
}
// returns the top left point of the cell. Returns FALSE if cell not visible.
BOOL CTerrainCtrl::GetCellOrigin(CELLID cell, LPPOINT p)
{
return GetCellOrigin(cell.row, cell.col, p);
}
BOOL CTerrainCtrl::GetCellOrigin(int nRow, int nCol, LPPOINT p)
{
int i;
if (!IsValid(nRow, nCol))
return FALSE;
CELLID idTopLeft;
if (nCol >= m_nFixedCols || nRow >= m_nFixedRows)
idTopLeft = GetTopleftNonFixedCell();
if ((nRow >= m_nFixedRows && nRow < idTopLeft.row) ||
(nCol>= m_nFixedCols && nCol < idTopLeft.col))
return FALSE;
p->x = 0;
if (nCol < m_nFixedCols) // is a fixed column
for (i = 0; i < nCol; i++)
p->x += GetColumnWidth(i);
else // is a scrollable data column
{
for (i = 0; i < m_nFixedCols; i++)
p->x += GetColumnWidth(i);
for (i = idTopLeft.col; i < nCol; i++)
p->x += GetColumnWidth(i);
}
p->y = 0;
if (nRow < m_nFixedRows) // is a fixed row
for (i = 0; i < nRow; i++)
p->y += GetRowHeight(i);
else // is a scrollable data row
{
for (i = 0; i < m_nFixedRows; i++)
p->y += GetRowHeight(i);
for (i = idTopLeft.row; i < nRow; i++)
p->y += GetRowHeight(i);
}
return TRUE;
}
// Returns the bounding box of the cell
BOOL CTerrainCtrl::GetCellRect(CELLID cell, LPRECT pRect)
{
return GetCellRect(cell.row, cell.col, pRect);
}
BOOL CTerrainCtrl::GetCellRect(int nRow, int nCol, LPRECT pRect)
{
CPoint CellOrigin;
if (!GetCellOrigin(nRow, nCol, &CellOrigin))
return FALSE;
pRect->left = CellOrigin.x;
pRect->top = CellOrigin.y;
pRect->right = CellOrigin.x + GetColumnWidth(nCol)-1;
pRect->bottom = CellOrigin.y + GetRowHeight(nRow)-1;
return TRUE;
}
BOOL CTerrainCtrl::IsCellVisible(CELLID cell)
{
return IsCellVisible(cell.row, cell.col);
}
BOOL CTerrainCtrl::IsCellVisible(int nRow, int nCol)
{
if (!IsWindow(m_hWnd))
return FALSE;
int x,y;
CELLID TopLeft;
if (nCol >= GetFixedColumnCount() || nRow >= GetFixedRowCount())
{
TopLeft = GetTopleftNonFixedCell();
if (nCol >= GetFixedColumnCount() && nCol < TopLeft.col) return FALSE;
if (nRow >= GetFixedRowCount() && nRow < TopLeft.row) return FALSE;
}
CRect rect;
GetClientRect(rect);
if (nCol < GetFixedColumnCount())
{
x = 0;
for (int i = 0; i <= nCol; i++)
{
if (x >= rect.right) return FALSE;
x += GetColumnWidth(i);
}
}
else
{
x = GetFixedColumnWidth();
for (int i = TopLeft.col; i <= nCol; i++)
{
if (x >= rect.right) return FALSE;
x += GetColumnWidth(i);
}
}
if (nRow < GetFixedRowCount())
{
y = 0;
for (int i = 0; i <= nRow; i++)
{
if (y >= rect.bottom) return FALSE;
y += GetRowHeight(i);
}
}
else
{
if (nRow < TopLeft.row) return FALSE;
y = GetFixedRowHeight();
for (int i = TopLeft.row; i <= nRow; i++)
{
if (y >= rect.bottom) return FALSE;
y += GetRowHeight(i);
}
}
return TRUE;
}
// move about with keyboard
void CTerrainCtrl::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
{
if (!IsValid(m_idCurrentCell))
{
CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
return;
}
CELLID next = m_idCurrentCell;
BOOL bChangeLine = FALSE;
/*
if (IsCTRLpressed())
{
switch (nChar)
{
case 'A': OnEditSelectAll(); break;
}
}
*/
switch (nChar)
{
/*
case VK_DELETE:
if (IsCellEditable(m_idCurrentCell.row, m_idCurrentCell.col))
{
SetItemText(m_idCurrentCell.row, m_idCurrentCell.col, _T(""));
RedrawCell(m_idCurrentCell);
SetModified(TRUE);
}
break;
case VK_TAB:
if (IsSHIFTpressed())
{
if (next.col > m_nFixedCols)
next.col--;
else if (next.col == m_nFixedCols && next.row > m_nFixedRows)
{
next.row--;
next.col = GetColumnCount() - 1;
bChangeLine = TRUE;
}
else
CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
}
else
{
if (next.col < (GetColumnCount() - 1))
next.col++;
else if (next.col == (GetColumnCount() - 1) &&
next.row < (GetRowCount() - 1) )
{
next.row++;
next.col = m_nFixedCols;
bChangeLine = TRUE;
}
else
CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
}
break;
*/
case VK_DOWN:
if (next.row < (GetRowCount() - 1))
next.row++;
break;
case VK_UP:
if (next.row > 1)
next.row--;
break;
case VK_RIGHT:
if (next.col < (GetColumnCount() - 1))
next.col++;
break;
case VK_LEFT:
if (next.col > 1)
next.col--;
break;
case VK_NEXT:
{
CELLID idOldTopLeft = GetTopleftNonFixedCell();
SendMessage(WM_VSCROLL, SB_PAGEDOWN, 0);
CELLID idNewTopLeft = GetTopleftNonFixedCell();
int increment = idNewTopLeft.row - idOldTopLeft.row;
if (increment)
{
next.row += increment;
if (next.row > (GetRowCount() - 1))
next.row = GetRowCount() - 1;
}
else
next.row = GetRowCount() - 1;
break;
}
case VK_PRIOR:
{
CELLID idOldTopLeft = GetTopleftNonFixedCell();
SendMessage(WM_VSCROLL, SB_PAGEUP, 0);
CELLID idNewTopLeft = GetTopleftNonFixedCell();
int increment = idNewTopLeft.row - idOldTopLeft.row;
if (increment)
{
next.row += increment;
if (next.row < m_nFixedRows)
next.row = m_nFixedRows;
}
else
next.row = m_nFixedRows;
break;
}
case VK_HOME:
SendMessage(WM_VSCROLL, SB_TOP, 0);
next.row = m_nFixedRows;
break;
case VK_END:
SendMessage(WM_VSCROLL, SB_BOTTOM, 0);
next.row = GetRowCount() - 1;
break;
default:
CWnd::OnKeyDown(nChar, nRepCnt, nFlags);
}
if (next != m_idCurrentCell)
{
// While moving with the Cursorkeys the current ROW/CELL will get selected
// OR Selection will get expanded when SHIFT is pressed
// Cut n paste from OnLButtonDown - Franco Bez
/* // Added check for NULL mouse mode - Chris Maunder.
if (m_MouseMode == MOUSE_NOTHING)
{
m_PrevSelectedCellMap.RemoveAll();
m_MouseMode = m_bListMode? MOUSE_SELECT_ROW : MOUSE_SELECT_CELLS;
if (!IsSHIFTpressed() || nChar == VK_TAB)
m_SelectionStartCell = next;
OnSelecting(next);
m_MouseMode = MOUSE_NOTHING;
}
*/
SetFocusCell(next);
if (!IsCellVisible(next))
{
EnsureVisible(next); // Make sure cell is visible
switch (nChar)
{
case VK_RIGHT:
SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
break;
case VK_LEFT:
SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
break;
case VK_DOWN:
SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
break;
case VK_UP:
SendMessage(WM_VSCROLL, SB_LINEUP, 0);
break;
/*
case VK_TAB:
if (IsSHIFTpressed())
{
if (bChangeLine)
{
SendMessage(WM_VSCROLL, SB_LINEUP, 0);
SetScrollPos32(SB_HORZ, m_nHScrollMax);
break;
}
else
SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
}
else
{
if (bChangeLine)
{
SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
SetScrollPos32(SB_HORZ, 0);
break;
}
else
SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
}
break;
*/
}
Invalidate();
}
}
}
void CTerrainCtrl::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
{
if (
/* !IsCTRLpressed() &&
*/
m_MouseMode == MOUSE_NOTHING)
{
/* if (!m_bHandleTabKey || (m_bHandleTabKey && nChar != VK_TAB))
*/ EditCell(m_idCurrentCell.row, m_idCurrentCell.col, nChar);
}
CWnd::OnChar(nChar, nRepCnt, nFlags);
}
// Instant editing of cells when keys are pressed
void CTerrainCtrl::EditCell(int nRow, int nCol, UINT nChar)
{
EnsureVisible(nRow, nCol);
if (!IsValid(nRow, nCol) || !IsCellVisible(nRow, nCol))
return;
CRect rect;
if (!GetCellRect(nRow, nCol, rect))
return;
/*
SendMessageToParent(nRow, nCol, GVN_BEGINLABELEDIT);
*/
char szText[20];
sprintf(szText, "%.2f", ControlPoint(nRow, nCol));
new CInPlaceEdit(this, rect, ES_LEFT, IDC_INPLACE_EDIT, &ControlPoint(nRow, nCol), szText, nChar);
}
void CTerrainCtrl::EnsureVisible(CELLID cell)
{
EnsureVisible(cell.row, cell.col);
}
void CTerrainCtrl::EnsureVisible(int nRow, int nCol)
{
// CCellRange VisibleCells = GetVisibleNonFixedCellRange();
CRect rect;
GetClientRect(rect);
CELLID idTopLeft = GetTopleftNonFixedCell();
// calc bottom
int i, bottom = GetFixedRowHeight();
for (i = idTopLeft.row; i < m_nRows; i++)
{
bottom += GetRowHeight(i);
if (bottom >= rect.bottom)
{
bottom = rect.bottom;
break;
}
}
int maxVisibleRow = min(i, m_nRows - 1);
// calc right
int right = GetFixedColumnWidth();
for (i = idTopLeft.col; i < m_nCols; i++)
{
right += GetColumnWidth(i);
if (right >= rect.right)
{
right = rect.right;
break;
}
}
int maxVisibleCol = min(i, m_nCols - 1);
right = nCol - maxVisibleCol;
int left = idTopLeft.col - nCol;
int down = nRow - maxVisibleRow;
int up = idTopLeft.row - nRow;
while (right > 0)
{
SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
right--;
}
while (left > 0)
{
SendMessage(WM_HSCROLL, SB_LINELEFT, 0);
left--;
}
while (down > 0)
{
SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
down--;
}
while (up > 0)
{
SendMessage(WM_VSCROLL, SB_LINEUP, 0);
up--;
}
// Move one more if we only see a small bit of the cell
CRect rectCell, rectWindow;
GetCellRect(nRow, nCol, rectCell);
GetClientRect(rectWindow);
if (rectCell.right > rectWindow.right)
SendMessage(WM_HSCROLL, SB_LINERIGHT, 0);
if (rectCell.bottom > rectWindow.bottom)
SendMessage(WM_VSCROLL, SB_LINEDOWN, 0);
}