`
greenmoon
  • 浏览: 48650 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

带滚动条的大图无刷新显示框

 
阅读更多

以前看到有人开发的带滚动条的大图无刷新显示框,不知道采用什么技术开发的,基本功能可以,但是滚动条长度有时会超出图形框,用户体验大打折扣。闲来无事时,决定自己动手写一个。技术很简单,就是从CWnd继承一个窗口类,根据图像大小自动显示滚动条,由于知道图形大小和窗口大小,滚动条的最大值就确定了,根据滚动条的位置就可以算出当前窗口影射到图像上的位置,然后创建内存dc,从图像上copy相应大小的图形到内存dc上,然后复制到窗口dc上,这样就大功告成,简单吧。代码如下:

#if !defined(AFX_BAR_H__90C0CEA8_EB58_4CC7_A33A_CEC11E8527AE__INCLUDED_)
#define AFX_BAR_H__90C0CEA8_EB58_4CC7_A33A_CEC11E8527AE__INCLUDED_

#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
// Bar.h : header file
//

/////////////////////////////////////////////////////////////////////////////
// CShowBitmap window

class CShowBitmap : public CWnd
{
// Construction
public:
CShowBitmap();

// Attributes
public:

// Operations
public:

// Overrides
// ClassWizard generated virtual function overrides
//{{AFX_VIRTUAL(CShowBitmap)
//}}AFX_VIRTUAL

// Implementation
public:
virtual ~CShowBitmap();

// Generated message map functions
protected:
//{{AFX_MSG(CShowBitmap)
afx_msg void OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar);
afx_msg void OnPaint();
afx_msg void OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar);
afx_msg BOOL OnMouseWheel(UINT nFlags, short zDelta, CPoint pt);
afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
afx_msg void OnMButtonDown(UINT nFlags, CPoint point);
afx_msg void OnRButtonDown(UINT nFlags, CPoint point);
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
public:
void SetBitmap(HBITMAP hbit);
BOOL Create( LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, CCreateContext* pContext = NULL);
protected:
void DrawBmp(CDC *pDC);
HBITMAP m_hBit;
BOOL m_bHScroll;
BOOL m_bVScroll;
};

/////////////////////////////////////////////////////////////////////////////

//{{AFX_INSERT_LOCATION}}
// Microsoft Visual C++ will insert additional declarations immediately before the previous line.

#endif // !defined(AFX_BAR_H__90C0CEA8_EB58_4CC7_A33A_CEC11E8527AE__INCLUDED_)

///cpp file

// Bar.cpp : implementation file
//

#include "stdafx.h"
#include "ShowBitmap.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif

#define INTSCROL10
/////////////////////////////////////////////////////////////////////////////
// CShowBitmap

CShowBitmap::CShowBitmap()
{
m_hBit = NULL;
m_bHScroll = FALSE;
m_bVScroll = FALSE;
}

CShowBitmap::~CShowBitmap()
{
}


BEGIN_MESSAGE_MAP(CShowBitmap, CWnd)
//{{AFX_MSG_MAP(CShowBitmap)
ON_WM_VSCROLL()
ON_WM_PAINT()
ON_WM_HSCROLL()
ON_WM_MOUSEWHEEL()
ON_WM_LBUTTONDOWN()
ON_WM_MBUTTONDOWN()
ON_WM_RBUTTONDOWN()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()


/////////////////////////////////////////////////////////////////////////////
// CShowBitmap message handlers

void CShowBitmap::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{
// TODO: Add your message handler code here and/or call default
int minpos;
int maxpos;
GetScrollRange(SB_VERT,&minpos, &maxpos);
maxpos = GetScrollLimit(SB_VERT);

// Get the current position of scroll box.
int curpos = GetScrollPos(SB_VERT);

// Determine the new position of scroll box.
switch (nSBCode)
{
case SB_TOP: // Scroll to far up.
curpos = minpos;
break;

case SB_BOTTOM: // Scroll to far right.
curpos = maxpos;
break;

case SB_ENDSCROLL: // End scroll.
break;

case SB_LINEUP: // Scroll left.
if (curpos > minpos)
{
if(curpos - minpos >= INTSCROL)
curpos -= INTSCROL;
else
curpos = minpos;
}
break;

case SB_LINEDOWN: // Scroll right.
if (curpos < maxpos)
{
if(maxpos - curpos >= INTSCROL)
curpos += INTSCROL;
else
curpos = maxpos;
}
break;

case SB_PAGEUP: // Scroll one page left.
{
// Get the page size.
SCROLLINFO info;
GetScrollInfo(SB_VERT,&info, SIF_ALL);

if (curpos > minpos)
curpos = max(minpos, curpos - (int) info.nPage);
}
break;

case SB_PAGEDOWN: // Scroll one page right.
{
// Get the page size.
SCROLLINFO info;
GetScrollInfo(SB_VERT,&info, SIF_ALL);
if (curpos < maxpos)
curpos = min(maxpos, curpos + (int) info.nPage);
}
break;

case SB_THUMBPOSITION: // Scroll to absolute position. nPos is the position
curpos = nPos; // of the scroll box at the end of the drag operation.

break;

case SB_THUMBTRACK: // Drag scroll box to specified position. nPos is the
curpos = nPos; // position that the scroll box has been dragged to.
break;
}

// Set the new position of the thumb (scroll box).
SetScrollPos(SB_VERT,curpos);

//CRect rc;
// GetClientRect(rc);
DrawBmp(GetDC());
CWnd::OnVScroll(nSBCode, nPos, pScrollBar);
// InvalidateRect(rc);
}

BOOL CShowBitmap::Create( LPCTSTR lpszWindowName, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID, CCreateContext* pContext)
{
CString strCls;
// ´´½¨±à¼­¿ò
strCls = AfxRegisterWndClass(CS_CLASSDC|CS_SAVEBITS, LoadCursor(NULL, IDC_ARROW));
BOOL bRet = CWnd::Create(strCls,lpszWindowName,dwStyle|WS_BORDER|WS_VSCROLL|WS_HSCROLL|WS_CHILD|WS_VISIBLE,rect,pParentWnd,nID,pContext);
if(bRet)
{
ModifyStyleEx(0,WS_EX_TRANSPARENT |WS_EX_CLIENTEDGE );
ShowScrollBar(SB_BOTH,FALSE);
}
return bRet;
}

void CShowBitmap::SetBitmap(HBITMAP hbit)
{
m_hBit = hbit;
ShowScrollBar(SB_BOTH,FALSE);
m_bHScroll = FALSE;
m_bVScroll = FALSE;
CRect rc;
GetClientRect(rc);

// Çå¿Õ½çÃæ
if(m_hBit == NULL)
{
DrawBmp(GetDC());
return;
}
// ÉèÖÃscroll
// µÃµ½Í¼ÐÎÐÅÏ¢
BITMAP bi;
::GetObject(m_hBit,sizeof(BITMAP),&bi);
m_bHScroll = (bi.bmWidth - rc.Width()) > 0;
m_bVScroll = (bi.bmHeight - rc.Height()) > 0;
SCROLLINFO scinfo;
scinfo.cbSize = sizeof(SCROLLINFO);
scinfo.fMask = SIF_ALL;
scinfo.nMin = 0;
scinfo.nTrackPos = 0;
scinfo.nPos = 0;
ShowScrollBar(SB_HORZ,m_bHScroll);
ShowScrollBar(SB_VERT,m_bVScroll);
GetClientRect(rc);
if(m_bHScroll)
{
scinfo.nMax = bi.bmWidth;
scinfo.nPage = rc.Width();
SetScrollInfo(SB_HORZ,&scinfo);
}
if(m_bVScroll)
{
scinfo.nMax = bi.bmHeight;
scinfo.nPage = rc.Height();
SetScrollInfo(SB_VERT,&scinfo);
}
DrawBmp(GetDC());
}

void CShowBitmap::DrawBmp(CDC *pDC)
{
CRect rc;
GetClientRect(rc);
if(m_hBit == NULL)
{
CBitmap bitTmp;
CDC memDc;
memDc.CreateCompatibleDC(NULL);
bitTmp.CreateCompatibleBitmap(pDC,rc.Width(),rc.Height());
CBitmap * pOld = (CBitmap*)memDc.SelectObject(&bitTmp);
CBrush brush;
brush.CreateSolidBrush(RGB(255,255,255));
memDc.FillRect(rc,&brush);
pDC->BitBlt(0,0,rc.Width(),rc.Height(),&memDc,0,0,SRCCOPY);
memDc.SelectObject(pOld);
return;
}
int x =0,y = 0;

BITMAP bi;
::GetObject(m_hBit,sizeof(BITMAP),&bi);

int nH,nW;
int nDx,nDy;
nDx = 0;
nDy = 0;
if(m_bVScroll)// ÊúÖ±¹ö¶¯Ìõ
{
y = GetScrollPos(SB_VERT);
nH = rc.Height();
}
else
{
nH = bi.bmHeight;
nDy = (rc.Height() - nH)/2;
}
if(m_bHScroll)// ˮƽ¹ö¶¯Ìõ
{
x = GetScrollPos(SB_HORZ);
nW = rc.Width();
}
else
{
nW = bi.bmWidth;
nDx = (rc.Width() - nW)/2;
}

CBitmap bitTmp,bit;
CDC memDc,dc;
bit.Attach(m_hBit);
memDc.CreateCompatibleDC(NULL);
CBitmap * pOldbmp = memDc.SelectObject(&bit);
dc.CreateCompatibleDC(NULL);
bitTmp.CreateCompatibleBitmap(pDC,rc.Width(),rc.Height());
CBitmap * pOld = (CBitmap*)dc.SelectObject(&bitTmp);
CBrush brush;
brush.CreateSolidBrush(RGB(255,255,255));
dc.FillRect(rc,&brush);
dc.SetStretchBltMode(COLORONCOLOR);
dc.StretchBlt(nDx,nDy,nW,nH,&memDc,x,y,nW,nH,SRCCOPY);
pDC->BitBlt(0,0,rc.Width(),rc.Height(),&dc,0,0,SRCCOPY);
dc.SelectObject(pOld);
memDc.SelectObject(pOldbmp);
bit.Detach();

/*CBitmap bitTmp;
CDC dc;


dc.CreateCompatibleDC(pDC);
bitTmp.CreateCompatibleBitmap(pDC,100,100);
CBitmap * pOld = (CBitmap*)dc.SelectObject(&bitTmp);
CBrush brush;
brush.CreateSolidBrush(RGB(255,255,255));
dc.FillRect(CRect(0,0,100,100),&brush);
dc.MoveTo(10,10);
dc.LineTo(100,10);
dc.SelectObject(pOld);

pDC->SelectObject(bitTmp);
//bitTmp.Detach();*/

}

void CShowBitmap::OnPaint()
{
CPaintDC dc(this); // device context for painting
DrawBmp(&dc);

/*CDC * pDC;
pDC = CDC::FromHandle(::GetDC(NULL));
CBitmap bitTmp;
CDC dc;
dc.CreateCompatibleDC(pDC);
bitTmp.CreateCompatibleBitmap(pDC,100,100);
CBitmap * pOld = (CBitmap*)dc.SelectObject(&bitTmp);
CBrush brush;
brush.CreateSolidBrush(RGB(255,255,255));
dc.FillRect(CRect(0,0,100,100),&brush);
dc.MoveTo(10,10);
dc.LineTo(100,10);
dc.SelectObject(pOld);
pDC->SelectObject(bitTmp);*/
}

void CShowBitmap::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)
{

int minpos;
int maxpos;
GetScrollRange(SB_VERT,&minpos, &maxpos);
maxpos = GetScrollLimit(SB_HORZ);

// Get the current position of scroll box.
int curpos = GetScrollPos(SB_HORZ);

// Determine the new position of scroll box.
switch (nSBCode)
{
case SB_LEFT: // Scroll to far left.
curpos = minpos;
break;

case SB_RIGHT: // Scroll to far right.
curpos = maxpos;
break;

case SB_ENDSCROLL: // End scroll.
break;

case SB_LINELEFT: // Scroll left.
if (curpos > minpos)
{
if(curpos - minpos >= INTSCROL)
curpos -= INTSCROL;
else
curpos = minpos;
}
break;

case SB_LINERIGHT: // Scroll right.
if (curpos < maxpos)
{
if(maxpos - curpos >= INTSCROL)
curpos += INTSCROL;
else
curpos = maxpos;
}
break;

case SB_PAGELEFT: // Scroll one page left.
{
// Get the page size.
SCROLLINFO info;
GetScrollInfo(SB_HORZ,&info, SIF_ALL);

if (curpos > minpos)
curpos = max(minpos, curpos - (int) info.nPage);
}
break;

case SB_PAGERIGHT: // Scroll one page right.
{
// Get the page size.
SCROLLINFO info;
GetScrollInfo(SB_HORZ,&info, SIF_ALL);
if (curpos < maxpos)
curpos = min(maxpos, curpos + (int) info.nPage);
}
break;

case SB_THUMBPOSITION: // Scroll to absolute position. nPos is the position
curpos = nPos; // of the scroll box at the end of the drag operation.

break;

case SB_THUMBTRACK: // Drag scroll box to specified position. nPos is the
curpos = nPos; // position that the scroll box has been dragged to.
break;
}

// Set the new position of the thumb (scroll box).
SetScrollPos(SB_HORZ,curpos);

DrawBmp(GetDC());

CWnd::OnHScroll(nSBCode, nPos, pScrollBar);
}

BOOL CShowBitmap::OnMouseWheel(UINT nFlags, short zDelta, CPoint pt)
{
// TODO: Add your message handler code here and/or call default
if(m_bVScroll)
{
if(zDelta > 0)// ÏòÉÏ
OnVScroll(SB_LINEUP,0,NULL);
else// ÏòÏÂ
OnVScroll(SB_LINEDOWN,0,NULL);
}
return CWnd::OnMouseWheel(nFlags, zDelta, pt);
}


void CShowBitmap::OnLButtonDown(UINT nFlags, CPoint point)
{
// TODO: Add your message handler code here and/or call default
this->SetFocus();
CWnd::OnLButtonDown(nFlags, point);
}

void CShowBitmap::OnMButtonDown(UINT nFlags, CPoint point)
{
// TODO: Add your message handler code here and/or call default
this->SetFocus();
CWnd::OnMButtonDown(nFlags, point);
}

void CShowBitmap::OnRButtonDown(UINT nFlags, CPoint point)
{
// TODO: Add your message handler code here and/or call default
this->SetFocus();
CWnd::OnRButtonDown(nFlags, point);
}

分享到:
评论

相关推荐

    MFC_带滚动条的大图片显示方案2

    在本文中,我们将深入探讨如何在MFC(Microsoft Foundation Classes)框架中实现一个带有滚动条的大图片显示功能,特别关注“MFC_带滚动条的大图片显示方案2”。MFC是微软提供的一种C++库,用于构建Windows应用程序...

    MFC pictrue控件显示大图,带滚动条控制

    完成以上步骤后,便能实现一个可显示大图且带有滚动条控制的MFC Picture控件,用户可以通过滚动条轻松浏览大图的各个部分。 综上所述,MFC中利用Picture控件显示大图并添加滚动条控制涉及的主要知识点有:CStatic...

    易语言给大图片添加滚动条

    图片框用于显示大图片,而水平滚动条和垂直滚动条则用于控制图片的显示位置。以下是实现这一功能的主要步骤: 1. **创建窗口程序**:在易语言中新建一个窗口程序项目,设置合适的窗口大小和布局,包括图片框和两个...

    VC使用滚动条显示图片

    在VC++(Visual C++)开发环境中,使用滚动条显示图片是一项常见的需求,尤其是在处理大尺寸图像时。本文将详细讲解如何实现这一功能,以及涉及的相关知识点。 首先,我们需要了解滚动条的基本概念。滚动条是...

    易语言滚动显示大图

    易语言滚动显示大图是一种在程序中实现动态展示大图的技术,主要应用于图像查看器、游戏界面或者其他需要连续展示大面积图像的场景。易语言作为中国本土的编程语言,以其简单的语法和面向对象的设计,使得初学者也能...

    易语言画板带滚动条加载图片

    而“带滚动条”的画板则是在画板控件的基础上添加了水平和垂直滚动条,使得当画板内的内容超过其可视范围时,用户可以通过滚动条查看全部内容。 实现这一功能的关键在于两个方面:一是如何加载图片,二是如何结合...

    vfp带滚动条的容器

    在 FoxPro(Visual FoxPro,简称VFP)编程环境中,创建一个带有滚动条的容器是非常常见且实用的需求,尤其在处理大量数据或者设计用户界面时。`vfpscrollbar` 提供了这样的功能,它是一个专门用于在VFP中集成滚动条...

    易语言-易语言利用滚动条显示大图片

    在易语言中,滚动条通常与动画框配合使用,以控制显示大图片的不同部分。当用户移动滚动条时,程序会更新动画框内的显示区域,从而展示图片的其他部分。 描述中提到的“使用动画框装载大图片”,动画框并非专为动画...

    对话框上用滚动条显示图片

    本文将详细讨论如何在对话框上实现带有滚动条的图片显示功能,并解决可能出现的闪屏问题。 首先,我们需要创建一个自定义对话框类,继承自CDialog或CDialogEx。在这个类中,我们将添加两个成员变量,分别代表水平...

    VB使用滚动条实现图形的缩放

    首先,我们需要在VB界面中添加必要的组件:一个用于显示图形的PictureBox控件,以及一个水平滚动条(HScrollBar)或垂直滚动条(VScrollBar)。滚动条的最小值通常设为1,最大值可以根据实际需求设定,这将决定缩放...

    拖动滚动条自动加载图片页面

    标题中的“拖动滚动条自动加载图片页面”指的是网页设计中的一个常见技术,通常被称为“无限滚动”或“滚动监听加载”。这种技术允许用户在滚动页面时,内容(如图片)会随着滚动条的移动而逐步加载,而不是一次性...

    PHP+AJAX上传图片带滚动条

    在这个项目中,AJAX用于实现无刷新的图片上传和进度条显示。前端JavaScript代码监听用户触发的上传事件,通过XMLHttpRequest对象与服务器进行异步通信。在图片上传过程中,AJAX可以实时获取上传进度,并更新页面上的...

    网页图片随滚动条加载

    网页图片随滚动条加载是一种优化网页性能的技术,它主要针对含有大量图片的页面,通过只在用户滚动到可视区域时加载图片,来提高网页的加载速度和用户体验。这种技术也被称为懒加载(Lazy Loading)或者延迟加载(On...

    易语言高级表格数据滚动显示例程

    在易语言中,可以通过设置表格的滚动条属性来实现数据的滚动显示。滚动条分为水平和垂直两种,分别对应数据行和列的滚动。开发者需要编写适当的事件处理代码,如“滚动条位置改变”事件,来实时更新表格显示的数据。...

    MSCHart 滚动条

    在某些情况下,当图表数据量较大时,为了更有效地展示信息并提高用户体验,我们可能需要为MSCHart添加滚动条。尽管MSCHart本身并不直接支持滚动条功能,但我们可以通过一些技巧和自定义编程来实现这一效果。 首先,...

    自绘滚动条

    另外,`效果图.png`文件很可能是自绘滚动条在运行时的样子,展示了自定义的滚动条设计和功能。通过查看这个图片,我们可以更好地理解自绘滚动条的外观和可能的交互方式。 自绘滚动条的实现需要注意性能问题,因为...

    滚动条控制图片、线条运动

    在编程领域,滚动条是一种常见的用户界面元素,用于在内容超过屏幕显示范围时提供导航。在Windows编程中,通常使用MFC(Microsoft Foundation Classes)库来处理滚动条事件。本项目"滚动条控制图片、线条运动"展示了...

    Halcon+MFC,实现(1)鼠标对图像进行缩放、拖拽(2)添加滚动条

    在本文中,我们将深入探讨如何使用Halcon与MFC(Microsoft Foundation Classes)结合,实现图像处理中的关键功能:鼠标对图像进行缩放和拖拽,以及添加滚动条以查看大图像。Halcon是一款强大的机器视觉软件,而MFC是...

    显示字符的滚动条,能设置字体颜色

    3. **实时更新**:如果滚动条需要动态显示信息,如加载进度,我们需要在程序执行的适当位置更新`ProgressCtrl`的值,并调用`Refresh`方法来刷新控件,让变化立即反映在界面上。 4. **颜色设置**:使用API函数或控件...

    MFC 重绘CListCtrl、CTreeCtrl的滚动条.rar

    标题"**MFC 重绘CListCtrl、CTreeCtrl的滚动条.rar**"提供的资源正是针对这个需求,实现了滚动条的自定义重绘,以达到与控件完美融合的效果。 滚动条控件在MFC中由CScrollBar类表示,通常会与CListCtrl和CTreeCtrl...

Global site tag (gtag.js) - Google Analytics