📄 admmscrollview.cpp
字号:
// need a horizontal scrollbar after all
bNeedH = TRUE;
sizeRange.cy += sizeSb.cy;
}
// if current scroll position will be past the limit, scroll to limit
if (sizeRange.cx > 0 && ptMove.x >= sizeRange.cx)
ptMove.x = sizeRange.cx;
if (sizeRange.cy > 0 && ptMove.y >= sizeRange.cy)
ptMove.y = sizeRange.cy;
// now update the bars as appropriate
needSb.cx = bNeedH;
needSb.cy = bNeedV;
// needSb, sizeRange, and ptMove area now all updated
}
void CADMMScrollView::UpdateBars()
{
// UpdateBars may cause window to be resized - ignore those resizings
if (m_bInsideUpdate)
return; // Do not allow recursive calls
// Lock out recursion
m_bInsideUpdate = TRUE;
// update the horizontal to reflect reality
// NOTE: turning on/off the scrollbars will cause 'OnSize' callbacks
ASSERT(m_totalDev.cx >= 0 && m_totalDev.cy >= 0);
CRect rectClient;
BOOL bCalcClient = TRUE;
// allow parent to do inside-out layout first
CWnd* pParentWnd = GetParent();
if (pParentWnd != NULL)
{
// if parent window responds to this message, use just
// client area for scroll bar calc -- not "true" client area
if ((BOOL)pParentWnd->SendMessage(WM_RECALCPARENT, 0,
(LPARAM)(LPCRECT)&rectClient) != 0)
{
// use rectClient instead of GetTrueClientSize for
// client size calculation.
bCalcClient = FALSE;
}
}
CSize sizeClient;
CSize sizeSb;
if (bCalcClient)
{
// get client rect
if (!GetTrueClientSize(sizeClient, sizeSb))
{
// no room for scroll bars (common for zero sized elements)
CRect rect;
GetClientRect(&rect);
if (rect.right > 0 && rect.bottom > 0)
{
// if entire client area is not invisible, assume we have
// control over our scrollbars
EnableScrollBarCtrl(SB_BOTH, FALSE);
}
m_bInsideUpdate = FALSE;
return;
}
}
else
{
// let parent window determine the "client" rect
GetScrollBarSizes(sizeSb);
sizeClient.cx = rectClient.right - rectClient.left;
sizeClient.cy = rectClient.bottom - rectClient.top;
}
// enough room to add scrollbars
CSize sizeRange;
CPoint ptMove;
CSize needSb;
// get the current scroll bar state given the true client area
GetScrollBarState(sizeClient, needSb, sizeRange, ptMove, bCalcClient);
if (needSb.cx)
sizeClient.cy -= sizeSb.cy;
if (needSb.cy)
sizeClient.cx -= sizeSb.cx;
// first scroll the window as needed
ScrollToDevicePosition(ptMove); // will set the scroll bar positions too
// this structure needed to update the scrollbar page range
SCROLLINFO info;
info.fMask = SIF_PAGE|SIF_RANGE;
info.nMin = 0;
// now update the bars as appropriate
EnableScrollBarCtrl(SB_HORZ, needSb.cx);
if (needSb.cx)
{
info.nPage = sizeClient.cx;
info.nMax = m_totalDev.cx-1;
if (!SetScrollInfo(SB_HORZ, &info, TRUE))
SetScrollRange(SB_HORZ, 0, sizeRange.cx, TRUE);
}
EnableScrollBarCtrl(SB_VERT, needSb.cy);
if (needSb.cy)
{
info.nPage = sizeClient.cy;
info.nMax = m_totalDev.cy-1;
if (!SetScrollInfo(SB_VERT, &info, TRUE))
SetScrollRange(SB_VERT, 0, sizeRange.cy, TRUE);
}
// remove recursion lockout
m_bInsideUpdate = FALSE;
}
void CADMMScrollView::CalcWindowRect(LPRECT lpClientRect, UINT nAdjustType)
{
if (nAdjustType == adjustOutside)
{
// allow for special client-edge style
::AdjustWindowRectEx(lpClientRect, 0, FALSE, GetExStyle());
if (m_nMapMode != MM_SCALETOFIT)
{
// if the view is being used in-place, add scrollbar sizes
// (scollbars should appear on the outside when in-place editing)
CSize sizeClient(
lpClientRect->right - lpClientRect->left,
lpClientRect->bottom - lpClientRect->top);
CSize sizeRange = m_totalDev - sizeClient;
// > 0 => need to scroll
// get scroll bar sizes (used to adjust the window)
CSize sizeSb;
GetScrollBarSizes(sizeSb);
// adjust the window size based on the state
if (sizeRange.cy > 0)
{ // vertical scroll bars take up horizontal space
lpClientRect->right += sizeSb.cx;
}
if (sizeRange.cx > 0)
{ // horizontal scroll bars take up vertical space
lpClientRect->bottom += sizeSb.cy;
}
}
}
else
{
// call default to handle other non-client areas
::AdjustWindowRectEx(lpClientRect, GetStyle(), FALSE,
GetExStyle() & ~(WS_EX_CLIENTEDGE));
}
}
/////////////////////////////////////////////////////////////////////////////
// CADMMScrollView scrolling
void CADMMScrollView::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
if (pScrollBar != NULL && pScrollBar->SendChildNotifyLastMsg())
return; // eat it
// ignore scroll bar msgs from other controls
if (pScrollBar != GetScrollBarCtrl(SB_HORZ))
return;
OnScroll(MAKEWORD(nSBCode, 0xff), nPos);
}
void CADMMScrollView::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
if (pScrollBar != NULL && pScrollBar->SendChildNotifyLastMsg())
return; // eat it
// ignore scroll bar msgs from other controls
if (pScrollBar != GetScrollBarCtrl(SB_VERT))
return;
OnScroll(MAKEWORD(0xff, nSBCode), nPos);
}
BOOL CADMMScrollView::OnMouseWheel(UINT fFlags, short zDelta, CPoint point)
{
// we don't handle anything but scrolling
if (fFlags & (MK_SHIFT | MK_CONTROL))
return FALSE;
// if the parent is a splitter, it will handle the message
if (GetParentSplitter(this, TRUE))
return FALSE;
// we can't get out of it--perform the scroll ourselves
return DoMouseWheel(fFlags, zDelta, point);
}
void CADMMScrollView::CheckScrollBars(BOOL& bHasHorzBar, BOOL& bHasVertBar) const
{
DWORD dwStyle = GetStyle();
CScrollBar* pBar = GetScrollBarCtrl(SB_VERT);
bHasVertBar = ((pBar != NULL) && pBar->IsWindowEnabled()) ||
(dwStyle & WS_VSCROLL);
pBar = GetScrollBarCtrl(SB_HORZ);
bHasHorzBar = ((pBar != NULL) && pBar->IsWindowEnabled()) ||
(dwStyle & WS_HSCROLL);
}
// This function isn't virtual. If you need to override it,
// you really need to override OnMouseWheel() here or in
// CSplitterWnd.
BOOL CADMMScrollView::DoMouseWheel(UINT fFlags, short zDelta, CPoint point)
{
UNUSED_ALWAYS(point);
UNUSED_ALWAYS(fFlags);
// if we have a vertical scroll bar, the wheel scrolls that
// if we have _only_ a horizontal scroll bar, the wheel scrolls that
// otherwise, don't do any work at all
BOOL bHasHorzBar, bHasVertBar;
CheckScrollBars(bHasHorzBar, bHasVertBar);
if (!bHasVertBar && !bHasHorzBar)
return FALSE;
BOOL bResult = FALSE;
UINT uWheelScrollLines = _AfxGetMouseScrollLines();
int nToScroll = ::MulDiv(-zDelta, uWheelScrollLines, WHEEL_DELTA);
int nDisplacement;
if (bHasVertBar)
{
if (uWheelScrollLines == WHEEL_PAGESCROLL)
{
nDisplacement = m_pageDev.cy;
if (zDelta > 0)
nDisplacement = -nDisplacement;
}
else
{
nDisplacement = nToScroll * m_lineDev.cy;
nDisplacement = min(nDisplacement, m_pageDev.cy);
}
bResult = OnScrollBy(CSize(0, nDisplacement), TRUE);
}
else if (bHasHorzBar)
{
if (uWheelScrollLines == WHEEL_PAGESCROLL)
{
nDisplacement = m_pageDev.cx;
if (zDelta > 0)
nDisplacement = -nDisplacement;
}
else
{
nDisplacement = nToScroll * m_lineDev.cx;
nDisplacement = min(nDisplacement, m_pageDev.cx);
}
bResult = OnScrollBy(CSize(nDisplacement, 0), TRUE);
}
if (bResult)
UpdateWindow();
return bResult;
}
BOOL CADMMScrollView::OnScroll(UINT nScrollCode, UINT nPos, BOOL bDoScroll)
{
// calc new x position
int x = GetScrollPos(SB_HORZ);
int xOrig = x;
switch (LOBYTE(nScrollCode))
{
case SB_TOP:
x = 0;
break;
case SB_BOTTOM:
x = INT_MAX;
break;
case SB_LINEUP:
x -= m_lineDev.cx;
break;
case SB_LINEDOWN:
x += m_lineDev.cx;
break;
case SB_PAGEUP:
x -= m_pageDev.cx;
break;
case SB_PAGEDOWN:
x += m_pageDev.cx;
break;
case SB_THUMBTRACK:
x = nPos;
break;
}
// calc new y position
int y = GetScrollPos(SB_VERT);
int yOrig = y;
switch (HIBYTE(nScrollCode))
{
case SB_TOP:
y = 0;
break;
case SB_BOTTOM:
y = INT_MAX;
break;
case SB_LINEUP:
y -= m_lineDev.cy;
break;
case SB_LINEDOWN:
y += m_lineDev.cy;
break;
case SB_PAGEUP:
y -= m_pageDev.cy;
break;
case SB_PAGEDOWN:
y += m_pageDev.cy;
break;
case SB_THUMBTRACK:
y = nPos;
break;
}
BOOL bResult = OnScrollBy(CSize(x - xOrig, y - yOrig), bDoScroll);
if (bResult && bDoScroll)
UpdateWindow();
return bResult;
}
BOOL CADMMScrollView::OnScrollBy(CSize sizeScroll, BOOL bDoScroll)
{
int xOrig, x;
int yOrig, y;
// don't scroll if there is no valid scroll range (ie. no scroll bar)
CScrollBar* pBar;
DWORD dwStyle = GetStyle();
pBar = GetScrollBarCtrl(SB_VERT);
if ((pBar != NULL && !pBar->IsWindowEnabled()) ||
(pBar == NULL && !(dwStyle & WS_VSCROLL)))
{
// vertical scroll bar not enabled
sizeScroll.cy = 0;
}
pBar = GetScrollBarCtrl(SB_HORZ);
if ((pBar != NULL && !pBar->IsWindowEnabled()) ||
(pBar == NULL && !(dwStyle & WS_HSCROLL)))
{
// horizontal scroll bar not enabled
sizeScroll.cx = 0;
}
// adjust current x position
xOrig = x = GetScrollPos(SB_HORZ);
int xMax = GetScrollLimit(SB_HORZ);
x += sizeScroll.cx;
if (x < 0)
x = 0;
else if (x > xMax)
x = xMax;
// adjust current y position
yOrig = y = GetScrollPos(SB_VERT);
int yMax = GetScrollLimit(SB_VERT);
y += sizeScroll.cy;
if (y < 0)
y = 0;
else if (y > yMax)
y = yMax;
// did anything change?
if (x == xOrig && y == yOrig)
return FALSE;
if (bDoScroll)
{
UpdateWindow();
// do scroll and update scroll positions
ScrollWindow(-(x-xOrig), -(y-yOrig));
if (x != xOrig)
SetScrollPos(SB_HORZ, x);
if (y != yOrig)
SetScrollPos(SB_VERT, y);
}
return TRUE;
}
/////////////////////////////////////////////////////////////////////////////
// CADMMScrollView diagnostics
#ifdef _DEBUG
void CADMMScrollView::Dump(CDumpContext& dc) const
{
CView::Dump(dc);
dc << "m_totalLog = " << m_totalLog;
dc << "\nm_totalDev = " << m_totalDev;
dc << "\nm_pageDev = " << m_pageDev;
dc << "\nm_lineDev = " << m_lineDev;
dc << "\nm_bCenter = " << m_bCenter;
dc << "\nm_bInsideUpdate = " << m_bInsideUpdate;
dc << "\nm_nMapMode = ";
switch (m_nMapMode)
{
case MM_NONE:
dc << "MM_NONE";
break;
case MM_SCALETOFIT:
dc << "MM_SCALETOFIT";
break;
case MM_TEXT:
dc << "MM_TEXT";
break;
case MM_LOMETRIC:
dc << "MM_LOMETRIC";
break;
case MM_HIMETRIC:
dc << "MM_HIMETRIC";
break;
case MM_LOENGLISH:
dc << "MM_LOENGLISH";
break;
case MM_HIENGLISH:
dc << "MM_HIENGLISH";
break;
case MM_TWIPS:
dc << "MM_TWIPS";
break;
default:
dc << "*unknown*";
break;
}
dc << "\n";
}
void CADMMScrollView::AssertValid() const
{
CView::AssertValid();
//switch (m_nMapMode)
//{
//case MM_NONE:
//case MM_SCALETOFIT:
//case MM_TEXT:
//case MM_LOMETRIC:
//case MM_HIMETRIC:
//case MM_LOENGLISH:
//case MM_HIENGLISH:
//case MM_TWIPS:
// break;
//case MM_ISOTROPIC:
//case MM_ANISOTROPIC:
// ASSERT(FALSE); // illegal mapping mode
//default:
// ASSERT(FALSE); // unknown mapping mode
//}
}
#endif //_DEBUG
#ifdef AFX_INIT_SEG
#pragma code_seg(AFX_INIT_SEG)
#endif
IMPLEMENT_DYNAMIC(CADMMScrollView, CView)
/////////////////////////////////////////////////////////////////////////////
void CADMMScrollView::SetISOMapMode(CDC* pDC)
{
//单位为0.01mm x轴向右递增 y轴向下递增 原点在左上角
pDC->SetMapMode(MM_ISOTROPIC);
pDC->SetWindowExt(m_XLogMm*100,m_YLogMm*100);
pDC->SetViewportExt(m_XLogPix, m_YLogPix);
CPoint ptVpOrg(0, 0);
//ptVpOrg = -GetDeviceScrollPosition(); //视图滚动后要重新设置坐标原点
pDC->SetViewportOrg(ptVpOrg);
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -