1、class CPictureProcessView : public CScrollView
如果原来是继承自CView,则把程序中所有的CView替换成CScrollView。
注意,一定要重载 virtual void OnInitialUpdate(); 否则可能出错。
2、添加内存DC,内存位图成员。
CDC m_MemDC; //内存DC
CBitmap m_MemBitmap; //内存位图
3、在打开文件后,要完成内存DC,和位图的初始化,同时设置滚动条的范围。
其实,就是把图像画到内存DC中。。
if (bitmap != NULL)
{
//把 图像画到内存DC中
m_MemBitmap.DeleteObject();
CDC *pDC = this->GetDC();
m_MemBitmap.CreateCompatibleBitmap(pDC,bitmap->GetWidth(),bitmap->GetHeight());
m_MemDC.SelectObject(m_MemBitmap);
Graphics graphics(m_MemDC.GetSafeHdc());
graphics.DrawImage(bitmap,0,0);
this->ReleaseDC(pDC);
// 设置滚动条范围
CSize sizeTotal(bitmap->GetWidth(),bitmap->GetHeight() );
SetScrollSizes(MM_TEXT,sizeTotal);
}
4、处理OnDraw函数。
其实就是从内存DC复制到另一个DC中。这里要注意的是,得到滚动条的位置,可以减少复制的范围。
我在测试十几M大小的图像完全不卡。
if (p_bitmap != NULL)
{
SCROLLINFO si ;
GetScrollInfo ( SB_HORZ ,& si , SIF_ALL );
int hPos = si .nPos ; //水平滚动条 位置
GetScrollInfo ( SB_VERT ,& si , SIF_ALL );
int vPos = si . nPos ; //竖直滚动条位置
CRect tempRect ;
this->GetWindowRect(&tempRect);
pDC->BitBlt(hPos,vPos,tempRect.Width(),tempRect.Height(),&m_MemDC,hPos,vPos,SRCCOPY);
}
5、屏蔽WM_ERASEBKGND消息。
不屏蔽的话会闪烁。
BOOL CPictureProcessView::OnEraseBkgnd(CDC* pDC)
{
//return CScrollView::OnEraseBkgnd(pDC);
return TRUE;
}