最近用VC做了一个画图的控件。控件在使用的时候遇到点问题。在控件里画了图之后切换到其他页面,等再切换回来的时候,发现控件里画的图都不见了。这是因为VC里面,当缩小、遮挡页面后客户区域就会失效,当再次显示的时候系统就自动调用OnDraw 函数进行重绘。所以原来保存的图形都消失了。由于我做的是控件,所以不可能用一般的方法来解决比如在OnDraw 函数里绘图。经过这种查找决定使用双缓冲绘图来解决这个问题。

普通的绘图是直接将图像绘制到设备上,双缓冲绘图是将图形绘制到内存的一张图片上,等所有的绘图完成后再将整幅图片显示在设备上(个人的理解可能 不太准确)。所以这里我们可以将图片设为成员变量。各绘图函数都在该图片上绘图。并且在OnDraw 函数里显示这幅图片,这样就可以在页面切换回来的时候显示以前画的图了。

代码如下:

文件DrawShapeCtrl.h 中:

class CDrawShapeCtrl : public COleControl
{
........
private:
CBitmap m_Bitmap;
//CDC m_dcMem;//切记不能这样定义
BOOL m_bFisrtTime;//第一次加载
BOOL m_bClear; };

DrawShapeCtrl.cpp 中:

CDrawShapeCtrl::CDrawShapeCtrl()
{
InitializeIIDs(&IID_DDrawShape, &IID_DDrawShapeEvents);
// TODO: Initialize your control's instance data here.
m_bFisrtTime = TRUE;
m_bClear = FALSE; }

OnDraw 函数中:

void CDrawShapeCtrl::OnDraw(
CDC* pdc, const CRect& rcBounds, const CRect& rcInvalid)
{
if (!pdc)
return; //// TODO: Replace the following code with your own drawing code. CRect rect;
GetClientRect(&rect); CDC dcMem;
dcMem.CreateCompatibleDC(pdc); //first time
if (m_bFisrtTime)
{ m_Bitmap.CreateCompatibleBitmap(pdc,rect.Width(),rect.Height());
dcMem.SelectObject(&m_Bitmap); //填充为白色
dcMem.FillSolidRect(,,rect.Width(),rect.Height(),RGB(,,)); //画坐标轴
CPen pen;
CPen* pOldPen;
pen.CreatePen( PS_DOT, , RGB(,,));
pOldPen = dcMem.SelectObject( &pen ); dcMem.MoveTo(rect.left,rect.Height()/);
dcMem.LineTo(rect.right,rect.Height()/);
dcMem.MoveTo(rect.Width()/,rect.top);
dcMem.LineTo(rect.Width()/,rect.bottom); dcMem.SelectObject(pOldPen);
pOldPen->DeleteObject(); m_bFisrtTime = FALSE; } dcMem.SelectObject(&m_Bitmap);
if (m_bClear)
{ //填充为白色
dcMem.FillSolidRect(,,rect.Width(),rect.Height(),RGB(,,)); //画坐标轴
CPen pen;
CPen* pOldPen;
pen.CreatePen( PS_DOT, , RGB(,,));
pOldPen = dcMem.SelectObject( &pen ); dcMem.MoveTo(rect.left,rect.Height()/);
dcMem.LineTo(rect.right,rect.Height()/);
dcMem.MoveTo(rect.Width()/,rect.top);
dcMem.LineTo(rect.Width()/,rect.bottom); dcMem.SelectObject(pOldPen);
pOldPen->DeleteObject(); m_bClear = FALSE;
} //显示
pdc->BitBlt(,,rect.Width(),rect.Height(),&dcMem,,,SRCCOPY);
dcMem.DeleteDC(); }

画线的函数:

//Lstype--penstyle 线型 Lwidth --线宽度    color--颜色
void CDrawShapeCtrl::Line(LONG x1, LONG y1, LONG x2, LONG y2,LONG Lstyle,LONG Lwidth,OLE_COLOR color)
{
AFX_MANAGE_STATE(AfxGetStaticModuleState()); // TODO: Add your dispatch handler code here CDC *pdc = GetDC(); CRect rcClient;
GetClientRect(rcClient);
CDC memDC;
memDC.CreateCompatibleDC(pdc);
memDC.SelectObject(&m_Bitmap); // Change map mode, positive x right, positive y up.
int nOldMode = memDC.SaveDC();
memDC.SetMapMode(MM_ISOTROPIC);
memDC.SetViewportExt(, );
memDC.SetWindowExt(, -);
memDC.SetViewportOrg(rcClient.Width() / , rcClient.Height()/); //设置画笔线型 宽度 颜色
LOGBRUSH logBrush;
logBrush.lbStyle = BS_SOLID;
logBrush.lbColor = color;
if (Lstyle<||Lstyle>)
{
return;
}
CPen Pen(Lstyle|PS_GEOMETRIC|PS_ENDCAP_ROUND, Lwidth,&logBrush);
memDC.SelectObject(&Pen); //画线
memDC.MoveTo(x1,y1);
memDC.LineTo(x2,y2); // We must restore mapping mode before copy bitmap to client context.
memDC.RestoreDC(nOldMode); // Copy double buffer bitmap to client context.
pdc->BitBlt(, , rcClient.Width(), rcClient.Height(), &memDC, , , SRCCOPY); ReleaseDC(pdc);
memDC.DeleteDC();
}

清空控件内的图形:

void CDrawShapeCtrl::Clear(void)
{
AFX_MANAGE_STATE(AfxGetStaticModuleState()); // TODO: Add your dispatch handler code here
m_bClear = TRUE; Invalidate(TRUE);
}

在使用的过程当中发现,CDC 不能定义为全局的,必须定义为局部变量,用完随即释放。否则第一次能画出图形,第二次调用的时候什么也画不出来。原因参考

VC CDC类的使用 详解

VC 绘图,使用双缓冲技术实现

https://wenku.baidu.com/view/2c6aaf1ba8114431b90dd862.html

参考:

https://blog.csdn.net/imletterh/article/details/46372753

VC使用双缓冲制作绘图控件的更多相关文章

  1. VC使用双缓冲避免绘图闪烁的正确使用方法【转】

    使用内存DC绘图,然后实现双缓冲,避免绘图闪烁,这个小技术简单但很有效.但是仍然有很多人说使用了双缓冲,图片却仍然有闪烁,分析了几个这样的例子,发现 其实不是双缓冲的技术问题,而是使用者没有正确理解和 ...

  2. VC GDI双缓冲机制绘图防屏幕闪烁实现步骤

    在OnDraw(CDC* pDC) 中添加如下代码 CDC MemDC; //首先定义一个显示设备对象 CBitmap MemBitmap;//定义一个位图对象 //随后建立与屏幕显示兼容的内存显示设 ...

  3. 怎样在VS2013/MFC中使用TeeChart绘图控件

    TeeChart作为一款强大好用的绘图控件,通过它可以绘制出各式各样的图表,包括2D的,还有3D的,绘制的图表美观实用,这里主要讲述如何在VS2013/MFC中使用TeeChart控件,顺便说一下在V ...

  4. PaintView 绘图控件解析

    博客地址:博客园,版权所有,转载须联系作者. GitHub地址:JustWeTools 最近做了个绘图的控件,实现了一些有趣的功能. 先上效果图: PaintView画图工具: 1.可直接使用设定按钮 ...

  5. WPF--Blend制作Button控件模板

    博客园新人,WPF初学者.不涉及理论知识,直接进入操作. 记录一下使用Blend制作Button控件模板过程中,学到Blend几个知识点: 1.渐变画笔编辑器的Alpha选项可以调控件的透明度.即下图 ...

  6. qt超强精美绘图控件 - QCustomPlot一览 及 安装使用教程

    1.概述 QCustomPlot 是一个超强超小巧的qt绘图类,非常漂亮,非常易用,只需要加入一个qcustomplot.h和qcustomplot.cpp文件即可使用,远比qwt方便和漂亮,可以自己 ...

  7. VC/MFC 当鼠标移到控件上时显示提示信息

    VC/MFC 当鼠标移到控件上时显示提示信息 ToolTip是Win32中一个通用控件,MFC中为其生成了一个类CToolTipCtrl,总的说来其使用方法是较简单的,下面讲一下它的一般用法和高级用法 ...

  8. TeeChart绘图控件 - 之三 - 提高绘图的效率 .

    TeeChart是个很强大的控件,其绘图能力之强,其他控件难以比拟,但是有个问题就是他的绘图速度,其实TeeChart绘图速度还是很快的,只是大家一直都没正确运用其功能所以导致绘图速度慢的假象. 下面 ...

  9. VS2010 使用TeeChart绘图控件 - 之二 - 绘制图形(折线图,柱状图)

    1.前期准备 具体可见VS2010 使用TeeChart绘图控件 - 之一 控件和类的导入 1. 1 添加TeeChart控件,给控件添加变量m_TeeChart 添加TeeChart控件,右击控件, ...

随机推荐

  1. JS制作图片切换

    <!DOCTYPE html> <html> <head> <title>纯JS制作简单的图片切换</title> <meta cha ...

  2. Django学习笔记之视图高级-错误处理

    错误处理 在一些网站开发中.经常会需要捕获一些错误,然后将这些错误返回比较优美的界面,或者是将这个错误的请求做一些日志保存.那么我们本节就来讲讲如何实现. 常见的错误码 404:服务器没有指定的url ...

  3. 腾讯云cos对象在线显示

    问题 前端使用了一张cos的图片,但是无法显示图片,使用的是cdn的加速域名地址. 对比:使用服务器的直接域名是可以显示的. 两者地址直接访问时,一者在线显示,一者直接下载到本地. 原因: 使用默认提 ...

  4. Panel 中加载窗体

    pnlMain.Controls.Clear(); ControlAnalyzer1 CA1 = new ControlAnalyzer1(); CA1.TopLevel = false; CA1.S ...

  5. 两个Integer变量a和b,值相等,a==b等于多少?

    Integer a = Integer.valueOf(127); Integer b = Integer.valueOf(127);   Integer c = Integer.valueOf(12 ...

  6. Redis入门的简单使用

    Redis是什么? redis是一个开源的,面向键/值对的NOSQL的分布式数据库系统 NOSQL指的是非关系型的数据,简单直白地讲就是在非关系型的数据库中不存在表的概念,而是以键值对的方式, 即一个 ...

  7. Python 模块源

    1.官方:https://pypi.org/ 2.LFD UCI :https://www.lfd.uci.edu/~gohlke/pythonlibs/ 4.清华源:https://pypi.tun ...

  8. CSS控制文字显示一行,超出显示省略号

    这几天在项目需求里面遇到了很多之前没做过的需求,也慢慢更加认识到了css的强大,是真的强大.以后会把自己技术调研的东西都写出来,哪怕只是一点点或者很小的点,重在学习. “CSS控制文字显示一行,超出显 ...

  9. 工控随笔_17_西门子_WinCC的VBS脚本_06_过程和函数

    和其他语言一样,vbs提供了过程和函数机制,通过函数和过程可以优化代码结构和实现代码复用, 减少代码的编写量. 一.代码 具体不再多说,看实例代码. ' vbs的函数和过程 '1.过程式编程是一大编程 ...

  10. Ubuntu 15.10 下Scala 操作Redis Cluster

    1 前言 Redis Standalone,Redis Cluster的安装在前面介绍过,地址:http://www.cnblogs.com/liuchangchun/p/5063477.html,这 ...