mirror of
https://github.com/leozide/leocad
synced 2025-01-07 05:24:12 +01:00
1853 lines
43 KiB
C++
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);
|
|
}
|