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;
}