• 基于对话框的简单双缓冲绘图框架


       基于文档视图结构程序的双缓冲绘图框架比较多,那么如何在对话框上绘图呢?以前通常的做法是拖一个静态文本控件或其它控件当作绘图区域或者在这个区域上创建一个视图出来。看了微软的一个示例程序DrawCli(一个绘图的单文档程序),产生了一些灵感,决心把它移植到对话框绘图上,摸索了一下,搞了一个基于对话框的简单双缓冲绘图框架。


         具体代码如下,对话框头文件代码:


        

    1. #include <vector>  
    2. //@brief 直线结构体  
    3. struct stLine  
    4. {  
    5.     stLine(CPoint &Begin,CPoint &End)  
    6.     {  
    7.         m_Begin = Begin;  
    8.         m_End = End;  
    9.     }  
    10.     //@brief 起点  
    11.     CPoint m_Begin;  
    12.     //@brief 终点  
    13.     CPoint m_End;  
    14. };  
    15. class CDoubleBufDrawDlg : public CDialog  
    16. {  
    17.     DECLARE_DYNAMIC(CDoubleBufDrawDlg)  
    18. public:  
    19.     CDoubleBufDrawDlg(CWnd* pParent = NULL);   // 标准构造函数  
    20.     virtual ~CDoubleBufDrawDlg();  
    21. // 对话框数据  
    22.     enum { IDD = IDD_DIALOG_GDIPLUS };  
    23. protected:  
    24.     virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持  
    25.     DECLARE_MESSAGE_MAP()  
    26. public:  
    27.     virtual BOOL OnInitDialog();  
    28.     afx_msg void OnBnClickedDrawPoint();  
    29.     afx_msg void OnBnClickedDrawLine();  
    30.     afx_msg void OnSize(UINT nType, int cx, int cy);  
    31.     void AdjustControls();  
    32.     afx_msg void OnPaint();  
    33. protected:  
    34.     //   
    35.     void InitDrawPara();  
    36. protected:  
    37.     //@brief 上一个点  
    38.     CPoint m_PrePt;  
    39.     //@brief 鼠标按下点  
    40.     CPoint m_DownPt;  
    41.     //@brief 直线结构体数组  
    42.     std::vector<stLine> m_Lines;  
    43. public:  
    44.     afx_msg void OnLButtonDown(UINT nFlags, CPoint point);  
    45.     afx_msg void OnMouseMove(UINT nFlags, CPoint point);  
    46.     afx_msg void OnLButtonUp(UINT nFlags, CPoint point);  
    47. };   



        Cpp文件源码(这里只列出主要函数):


       

    1. //@brief 初始化绘图参数  
    2. void CDoubleBufDrawDlg::InitDrawPara()  
    3. {  
    4.     m_PrePt = CPoint(-1,-1);  
    5.     m_DownPt = CPoint(-1,-1);  
    6. }  
    7. CDoubleBufDrawDlg::CDoubleBufDrawDlg(CWnd* pParent /*=NULL*/)  
    8. : CDialog(CDoubleBufDrawDlg::IDD, pParent)  
    9. {  
    10.     InitDrawPara();  
    11. }  
    12. void CDoubleBufDrawDlg::OnLButtonDown(UINT nFlags, CPoint point)  
    13. {  
    14.     // TODO: 在此添加消息处理程序代码和/或调用默认值  
    15.     // 记录鼠标按下点  
    16.     m_DownPt = point;  
    17.     CDialog::OnLButtonDown(nFlags, point);  
    18. }  
    19. void CDoubleBufDrawDlg::OnMouseMove(UINT nFlags, CPoint point)  
    20. {  
    21.     // TODO: 在此添加消息处理程序代码和/或调用默认值  
    22.     // 假如在移动鼠标的同时按下左键  
    23.     if (MK_LBUTTON&nFlags)  
    24.     {  
    25.         CDC *pDC = GetDC();  
    26.         pDC->SetROP2(R2_NOTXORPEN);  
    27.     // 将绘图区域限制在对话框的客户区上面的/4区域  
    28. CRect rtDraw(rtClient.top,rtClient.left,rtClient.Width(),3*(rtClient.Height()/4));  
    29.     CRgn DrawRgn;  
    30.     DrawRgn.CreateRectRgn(rtDraw.left,rtDraw.top,rtDraw.right,rtDraw.bottom);  
    31.     pDC->SelectClipRgn(&DrawRgn);  
    32.         if (m_PrePt.x!=-1)  
    33.         {  
    34.             // 擦除上一条线  
    35.             pDC->MoveTo(m_DownPt);  
    36.             pDC->LineTo(m_PrePt);  
    37.         }  
    38.         // 画线  
    39.         pDC->MoveTo(m_DownPt);  
    40.         pDC->LineTo(point);  
    41.         ReleaseDC(pDC);  
    42.         m_PrePt = point;  
    43.     }  
    44.     CDialog::OnMouseMove(nFlags, point);  
    45. }  
    46. void CDoubleBufDrawDlg::OnLButtonUp(UINT nFlags, CPoint point)  
    47. {  
    48.     // TODO: 在此添加消息处理程序代码和/或调用默认值  
    49.     // 将绘制的直线加入到直线数组  
    50.     m_Lines.push_back(stLine(m_DownPt,point));  
    51.     // 初始化绘图参数  
    52.     InitDrawPara();  
    53.     CDialog::OnLButtonUp(nFlags, point);  
    54. }  
    55. void CDoubleBufDrawDlg::OnPaint()  
    56. {  
    57.     // 最小时绘制菜单图标,因为我是在一个单文档程序中弹出该对话框的,故不需要绘制图标  
    58.     // 如果是基于对话框的程序则需要绘制图标  
    59.     if (IsIconic())  
    60.     {  
    61.         CPaintDC dc(this); // device context for painting  
    62.         SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);  
    63.         // Center icon in client rectangle  
    64.         //int cxIcon = GetSystemMetrics(SM_CXICON);  
    65.         //int cyIcon = GetSystemMetrics(SM_CYICON);  
    66.         //CRect rect;  
    67.         //GetClientRect(&rect);  
    68.         //int x = (rect.Width() - cxIcon + 1) / 2;  
    69.         //int y = (rect.Height() - cyIcon + 1) / 2;  
    70.         //// Draw the icon  
    71.         //dc.DrawIcon(x, y, m_hIcon);  
    72.     }  
    73.     else  
    74.     {  
    75.         CPaintDC dc(this);  
    76.         // 定义一个兼容DC  
    77.         CDC CompaDC;  
    78.         // 初始化绘图DC指针  
    79.         CDC* pDrawDC = &dc;  
    80.         CBitmap bitmap;  
    81.         CBitmap* pOldBitmap = 0;  
    82.         CRect rtClient;  
    83.         GetClientRect(&rtClient);  
    84.         // 将对话框的客户区上面的/4区域设为绘图区域  
    85.         CRect rtDraw(rtClient.top,rtClient.left,rtClient.Width(),3*(rtClient.Height()/4));  
    86.         // 假如不是打印机DC  
    87.         if (!dc.IsPrinting())  
    88.         {  
    89.             // 创建兼容DC,创建兼容位图,将兼容位图选进兼容DC  
    90.             if (CompaDC.CreateCompatibleDC(&dc))  
    91.             {  
    92.                 if (bitmap.CreateCompatibleBitmap(&dc,rtDraw.Width(),rtDraw.Height()))  
    93.                 {  
    94.                     pDrawDC = &CompaDC;  
    95.                     pOldBitmap = CompaDC.SelectObject(&bitmap);  
    96.                 }  
    97.             }  
    98.         }  
    99.         // 定义一个白色画刷,将背景色设为白色  
    100.         CBrush brush;  
    101.         if (!brush.CreateSolidBrush(RGB(255,255,255)))  
    102.             return;  
    103.         brush.UnrealizeObject();  
    104.         pDrawDC->FillRect(rtDraw,&brush);  
    105.         // 绘制直线  
    106.         for (size_t i = 0;i<m_Lines.size();i++)  
    107.         {  
    108.             pDrawDC->MoveTo(m_Lines[i].m_Begin);  
    109.             pDrawDC->LineTo(m_Lines[i].m_End);  
    110.         }  
    111.         // 使用GDI+绘制一个线性渐变画刷  
    112.         Gdiplus::Graphics graphics(pDrawDC->m_hDC);  
    113.         LinearGradientBrush linGrBrush(Point(100,0),Point(100,100),Color(255,255,0,0),Color(255,0,255,0));  
    114.         Color colors[] = {  
    115.             Color(255, 255, 0, 0),   // red  
    116.             Color(255, 255, 255, 0), //yellow  
    117.             Color(255, 0, 0, 255),   // blue  
    118.             Color(255, 0, 255, 0)};  // green  
    119.             REAL positions[] = {  
    120.                 0.0f,     
    121.                 0.33f,     
    122.                 0.66f,  
    123.                 1.0f};    
    124.                 linGrBrush.SetInterpolationColors(colors, positions,4);  
    125.                 // 填充指定区域矩形  
    126.                 graphics.FillRectangle(&linGrBrush,100,0,100,100);  
    127.                 graphics.ReleaseHDC(pDrawDC->m_hDC);  
    128.                 if (pDrawDC != &dc)  
    129.                 {  
    130.                     // 将绘图DC贴到真正的设备DC上  
    131.                     dc.BitBlt(rtDraw.left,rtDraw.top,rtDraw.Width(),rtDraw.Height(),&CompaDC, 0, 0, SRCCOPY);  
    132.                     CompaDC.SelectObject(pOldBitmap);  
    133.                 }  
    134.     }  
    135. }   


        效果图如下,其中上面的白色区域为绘图区域:


    Dlg Double Buf



        现在你怎么改变对话框的大小绘图区域也不会产生非双缓冲绘图那种闪烁。

  • 相关阅读:
    PHP中的call_user_func()与call_user_func_array()简单理解
    PHP实现多继承
    PHP实现多继承 trait 语法
    PHP几种常见魔术方法与魔术变量解析
    tp5 的nginx配置
    PHP 扩展 trie-tree, swoole过滤敏感词方案
    PHP Ajax跨域问题解决办法
    附加个人作业
    学完软工的感受
    团队介绍
  • 原文地址:https://www.cnblogs.com/lidabo/p/3435067.html
Copyright © 2020-2023  润新知