MFC如何在树形图边上添加动态小地图

https://www.jianshu.com/p/7b1d828bf5db (简书无法识别缩进的。。。早知道先在博客园发了)

(转载请注明出处)

作者:梦镜谷雨


萌新第一次写文章,请多多包涵。末尾附上相应代码(PS公司繁体系统所以部分注释繁体请别介意)。

第一次接触MFC时做的一个小项目上有做个树形图边上带小地图的需求。(IDE:VS2010)

大四刚实习时写的,当时网上没找到现成的,打算记录下来也算篇技术谈不上的思路吧。

快2019了打算开始试着以后多记录点东西,因为本人大学微电子专业不是软件方向想往这边发展的,代码里东西这时文章写到一半自己看着都感觉很糟糕(╯﹏╰),也算是记录个黑历史吧能实现功但糟糕就是糟糕。当时第一次学自绘没想着要写文章记录下来,参考了些网上教树形图自绘好像csdn看的但链接没记现在也忘了当时是参考哪一个了。在此感谢加抱歉。


一.思路:

Step1.自绘树形图控件(在树形图文字左边显示CImagelist里的图片)

Step2.创建个CBitmap动态保存需要的图片信息存入树形图关联的CImagelist中

二.最终效果(使用60*60符文3):

在画板上的改变(60*60符文3)能动态反映到左边树形图控件的对应项目(60*60符文3)上

三.重绘(我就多添加注释吧在注释里讲解。思路很简单):

1.创建个类CViewTree继承自CTreeCtrl

2.重载OnPaint使用双缓冲(防闪烁,双缓冲原理网上很多,后面创建小地图也是同个思路)

/*雙緩衝重繪樹形圖*/

void CViewTree::OnPaint()

{

CLED_NEWDoc* pDoc =(CLED_NEWDoc*)((CMainFrame*)AfxGetApp()->GetMainWnd())->GetActiveDocument();//以为单文档工程有些东西存在doc里

CPaintDC dc(this); // device context for painting

// TODO: 重绘树形图控件

// Do not call CTreeCtrl::OnPaint() for painting messages

GetClientRect(&m_ClientRect);

CBitmap bitmap;

CDC MemeDc;

MemeDc.CreateCompatibleDC(&dc);

bitmap.CreateCompatibleBitmap(&dc, m_ClientRect.Width(), m_ClientRect.Height());

CBitmap *pOldBitmap = MemeDc.SelectObject(&bitmap);

//繪圖部分

MemeDc.FillSolidRect(0, 0,m_ClientRect.Width(),m_ClientRect.Height(),RGB(255, 255, 255)); //填充背景

if (pDoc->m_CsFileName != _T(""))//如果有打開文件則可以畫樹形圖

DrawItem(&MemeDc);

//繪圖部分

dc.BitBlt( m_ClientRect.left, m_ClientRect.top, m_ClientRect.Width(), m_ClientRect.Height(), &MemeDc, 0, 0,SRCCOPY);

MemeDc.SelectObject(pOldBitmap);

MemeDc.DeleteDC();

}

3.画显示的项目

void CViewTree::DrawItem(CDC* pDc)

{

CLED_NEWDoc* pDoc =(CLED_NEWDoc*)((CMainFrame*)AfxGetApp()->GetMainWnd())->GetActiveDocument();

HTREEITEM currentItem;//当前的句柄

DWORD    treeStyle;// 数的类型

CRect    itemRect;//每一项的区域

int      itemState;//某项的状态

int itemImage;//圖片

int     Open_num=0;

int HScroll = GetScrollPos(SB_HORZ);

CImageList* imagelist = GetImageList(TVSIL_NORMAL);

treeStyle =:: GetWindowLong( m_hWnd, GWL_STYLE );

currentItem = GetFirstVisibleItem();//获取第一个课可见的项

//設置顯示字體

static CFont font;

font.DeleteObject();

font.CreatePointFont(100, _T("新宋体"));

pDc->SelectObject(&font);

do //beginwhile ((currentItem=GetNextVisibleItem(currentItem)) != NULL);

{

Open_num++;

//判斷是否為選擇狀態

if (Open_num + GetScrollPos(SB_VERT) == ((CMainFrame*)AfxGetApp()->GetMainWnd())->m_wndMaskView.Select)//选中是第几个数存在MaskView的Select里

pDc->SetBkColor(RGB(233,233,233));

else

pDc->SetBkColor(RGB(255,255,255));

CRect   fillRect(0,itemRect.top,m_ClientRect.right,itemRect.bottom);//填背景用的后来没删

itemState = GetItemState(currentItem,TVIF_STATE);

//每一項的位置和圖片

GetItemImage(currentItem,itemImage,itemImage);

GetItemRect(currentItem,itemRect,FALSE);

CPoint point;

point.y = itemRect.top;

point.x = itemRect.left-HScroll+(GetLevel(currentItem)-1)*42;//(以前写的时候多次用上的数字都没#define成英文。很糟糕,建议养成习惯)

if (itemRect.top>m_ClientRect.bottom)  //说明这一项已超出窗口的边界

{

break;

}

if ( GetChildItem(currentItem) != NULL )

{

if (ItemHasChildren(currentItem))//有子項則畫上對應三角形圖標

{

if (itemState & TVIS_EXPANDED )

{

imagelist->Draw(pDc,1,point,ILD_TRANSPARENT);

}

else

{

imagelist->Draw(pDc,0,point,ILD_TRANSPARENT);

}

}

}else{//無子項則畫上對應imagelist圖片

//imagelist->Draw(pDc,3,point,ILD_TRANSPARENT);

point.x = point.x + 50;

imagelist->Draw(pDc,itemImage,point,ILD_TRANSPARENT);

point.x = point.x - 50;

}

if ( GetLevel(currentItem) != 3 )//項目不為第三層就加藍色文件夾圖標

{

point.x = point.x + 42;

point.y = point.y + 5;

pDc->DrawIcon(point,AfxGetApp()->LoadIcon(IDI_ICON_PROJECT));//因為存imagelist無法有透明效果(现在这个能自己能花式解决吧。以前写的真糟糕,但是因为提供的是ICON资源载入到imagelist分辨率降了很多,看他只有一张于是就直接画了)

point.x = point.x - 42;

point.y = point.y - 5;

}

GetItemRect(currentItem,itemRect,TRUE);

if (Open_num == ((CMainFrame*)AfxGetApp()->GetMainWnd())->m_wndMaskView.Rename && pDoc->m_bFlagRenameShow == TRUE);

else

pDc->TextOut(itemRect.left ,itemRect.top+itemRect.Height()/2-6,GetItemText(currentItem));//输出树形图的文字

}while ((currentItem=GetNextVisibleItem(currentItem)) != NULL);

}//好了到此为止,以下不重要可以直接看四.

//现在忘了为什么加这个

BOOL CViewTree::OnWndMsg(UINT message, WPARAM wParam, LPARAM lParam, LRESULT* pResult)

{

// TODO: Add your specialized code here and/or call the base class

return CTreeCtrl::OnWndMsg(message, wParam, lParam, pResult);

}

//这个现在也忘记了为什么加了

BOOL CViewTree::OnNotify(WPARAM wParam, LPARAM lParam, LRESULT* pResult)

{

BOOL bRes = CTreeCtrl::OnNotify(wParam, lParam, pResult);

NMHDR* pNMHDR = (NMHDR*)lParam;

ASSERT(pNMHDR != NULL);

if (pNMHDR && pNMHDR->code == TTN_SHOW && GetToolTips() != NULL)

{

GetToolTips()->SetWindowPos(&wndTop, -1, -1, -1, -1, SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOSIZE);

}

return bRes;

}

//不要擦除背景,会闪烁

BOOL CViewTree::OnEraseBkgnd(CDC* pDC)

{

// TODO: Add your message handler code here and/or call default

return TRUE;

}

//滚动时当然要刷新

void CViewTree::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar)

{

// TODO: Add your message handler code here and/or call default

Invalidate();

CTreeCtrl::OnVScroll(nSBCode, nPos, pScrollBar);

}

//搬的砖,放在里面方便用

int CViewTree::GetLevel(HTREEITEM inputTree)

{

//獲得Item所在樹形圖層數

HTREEITEM temp;

temp = inputTree;

int level = 0;

while (temp != NULL)

{

temp = GetParentItem(temp);

++level;

}

return level;

}

四.创建CBitmap存入CImagelist(那时存class CMaskView : public CDockablePane):

创建自绘的树形图:CViewTree m_wndFileView;

创建图片列表:CImageList m_FileViewImages;

1.创建树形图

//vs创建单文档工程造着里面来就好了,树形图类相关操作msdn里面自查CTreeCtrl,添加删除重命名什么的网上很多就不写了

if (!m_wndFileView.Create(dwViewStyle, rectDummy, this, 4))

{

TRACE0("Failed to create file view\n");

return -1;      // fail to create

}//调整位置大小也是vs新建单文档工程时有悬浮窗就有看vs吧

2.创建图片并填充(在画树形图时每个项对应上Imagelist里的相应项就OK了)

void CMaskView::FillTreeImage()

{

/***********************///(和双缓冲同个思路,诶,现在看起来能写的简单整洁多的,怪当时理解不深,真糟糕)

CLED_NEWDoc* pDoc =(CLED_NEWDoc*)((CMainFrame*)AfxGetApp()->GetMainWnd())->GetActiveDocument();

CDC MemDC,MemDC2;

CDC * pDC;

CDC * pDC2;

pDC =GetDC();

pDC2=GetDC();

HTREEITEM hItem;

CBitmap bmp,bmp2;

UINT nFlags = ILC_MASK;

nFlags |= (theApp.m_bHiColorIcons) ? ILC_COLOR24 : ILC_COLOR4;

m_imagelist.DeleteImageList();//清除图片列表

m_imagelist.Create(m_iImageSize,m_iImageSize,nFlags,0,300);//创建图片列表

//m_wndFileView.SetImageList(&pDoc->m_imagelist, TVSIL_NORMAL);

MemDC.CreateCompatibleDC(pDC);

bmp.CreateCompatibleBitmap(pDC,60,60);

MemDC.SelectObject(&bmp);

MemDC2.CreateCompatibleDC(pDC2);

bmp2.CreateCompatibleBitmap(pDC2,m_iImageSize,m_iImageSize);

MemDC2.SelectObject(&bmp2);

MemDC.FillSolidRect(0,0,pDoc->SymImage.m_Column,pDoc->SymImage.m_Row,RGB(255, 255, 255));

/*/畫十字(未展開狀態圖標)

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

MemDC2.FillSolidRect(m_iImageSize/2-10,m_iImageSize/2-1,20,2,RGB(0, 0, 0));

MemDC2.FillSolidRect(m_iImageSize/2-1,m_iImageSize/2-10,2,20,RGB(0, 0, 0));

m_imagelist.Add(&bmp2,RGB(255, 255, 255));

*/

//90度直角三角形

int start = m_iImageSize/3,end = m_iImageSize*2/3;

int H = m_iImageSize/2,mid = m_iImageSize/2;

int x,y;

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

for (x = 0;x<m_iImageSize;x++)

for (y = 0;y<m_iImageSize;y++)

if(x>start && (x-start) < (y - mid + H/2) && -(x-start) > (y - mid - H/2))

MemDC2.SetPixel(x,y,RGB(0, 0, 0));

m_imagelist.Add(&bmp2,RGB(255, 255, 255));

/*/畫減號(展開狀態圖標)

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

MemDC2.FillSolidRect(m_iImageSize/2-10,m_iImageSize/2-1,20,2,RGB(0, 0, 0));

m_imagelist.Add(&bmp2,RGB(255, 255, 255));

*/

//135直角三角形

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

for (x = 0;x<m_iImageSize;x++)

for (y = 0;y<m_iImageSize;y++)

if(x < mid + 7*H/20 && y < mid + 7*H/20 && -(x - mid - 7*H/20) < y - mid + 7*H/20)

MemDC2.SetPixel(x,y,RGB(0, 0, 0));

m_imagelist.Add(&bmp2,RGB(255, 255, 255));

//涂白(未含有子顯圖標)

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

m_imagelist.Add(&bmp2,RGB(255, 255, 255));

HICON hIcon[2];

hIcon[0] = AfxGetApp()->LoadIcon(IDI_ICON_FILE_SE);

m_imagelist.Add(hIcon[0]);

//bmp.LoadBitmapW(IDB_BITMAP_F1);

hIcon[1] = AfxGetApp()->LoadIcon(IDI_ICON_PROJECT);

m_imagelist.Add(hIcon[1]);

//給樹形圖對應imagelist插入自畫的圖片并設置每個項的對應

int n = 4;

hItem = m_wndFileView.GetRootItem();

hItem = m_wndFileView.GetChildItem(hItem);

while (hItem != NULL)//几个判断判断是否是第3层,因为当时只有第三层需要小地图,扫描他们

{

if(m_wndFileView.GetChildItem(hItem) != NULL)

{

hItem = m_wndFileView.GetChildItem(hItem);

while (hItem != NULL)

{

n++;

CString name;

name = m_wndFileView.GetItemText(hItem);

CString ParentItemName = m_wndFileView.GetItemText(m_wndFileView.GetParentItem(hItem));

if (ParentItemName == _T("5*8"))

pDoc->SymImage = pDoc->m_Array_List_Sym[pDoc->Finding_Sym_limit(name,5,8)];

else if (ParentItemName == _T("16*16"))

pDoc->SymImage = pDoc->m_Array_List_Sym[pDoc->Finding_Sym_limit(name,16,16)];

else if (ParentItemName == _T("32*32"))

pDoc->SymImage = pDoc->m_Array_List_Sym[pDoc->Finding_Sym_limit(name,32,32)];

else if

(ParentItemName == _T("其他") || ParentItemName == _T("Other"))

pDoc->SymImage = pDoc->m_Array_List_Sym[pDoc->Finding_Sym_for_another(name)];

//給圖片刷新背景

MemDC.FillSolidRect(0,0,pDoc->SymImage.m_Column,pDoc->SymImage.m_Row,pDoc->m_clrLedBK);

//按點填充圖案

for (x = 0;x<pDoc->SymImage.m_Column;x++)

for (y = 0;y<pDoc->SymImage.m_Row;y++)

if (pDoc->SymImage.led_data[x][y] == TRUE)

MemDC.SetPixel(x,y,pDoc->m_clrLed);//(根据已知信息绘制图,MemDC可以存其他图片,当时对CDC这些也理解不深)

MemDC2.StretchBlt(1,1,m_iImageSize-2,m_iImageSize-2,&MemDC,0,0,pDoc->SymImage.m_Column,pDoc->SymImage.m_Row,SRCCOPY);//缩放保存到MemDC2里,StretchBlt可能有失真

m_imagelist.Add(&bmp2,RGB(255,255,255));//添加到图片列表

m_wndFileView.SetItemImage(hItem,n,n);//设置对应图片

if (m_wndFileView.GetNextSiblingItem(hItem) == NULL)

break;

else

hItem = m_wndFileView.GetNextSiblingItem(hItem);

}//endwhile (hItem != NULL)

hItem = m_wndFileView.GetParentItem(hItem);

}

hItem = m_wndFileView.GetNextSiblingItem(hItem);

}//endwhile (hItem != NULL)

MemDC2.FillSolidRect(0,0,m_iImageSize,m_iImageSize,RGB(255, 255, 255));

m_imagelist.Add(&bmp2,RGB(0, 0, 0));

m_wndFileView.SetImageList(&m_imagelist, TVSIL_NORMAL);//关联图片列表

::ReleaseDC(this->m_hWnd, MemDC);

::ReleaseDC(this->m_hWnd, MemDC2);

/***********************************/

}

//就这样,树形图子项要对应的图片存到关联的CImagelist里,小地图需要变化时替换CImagelist里对应的项然后再刷新树形图就可以实现树形图旁边带着的小地图动态变换了。

五.结语:

这时写文章回顾这个代码写的真的很糟糕(负能量代码,抱歉),只是实现了功能。在博客园还是哪个地方看了篇技术观念的文章,嗯,深耕技术(这个词很喜欢)。希望未来我们都能做喜欢干的事情吧(希望终有一天能成为谷雨大神或雨神吧,嗯,希望),这算是记录下当时的思路(黑历史)的处女作吧,见谅。

转载于:https://www.cnblogs.com/MJGY/p/10198494.html

MFC如何在树形图边上添加动态小地图相关推荐

  1. MFC List Control 控件添加单元格编辑,实现可编辑重写

    在实现随机生成四则运算的个人项目中,目前已经完成基本功能,想要把程序变成一个Windows界面的程序.原本以为学习过MFC,应该很快就能完成.但是由于以前用的都是VC6.0,这次用了VS2010,稍微 ...

  2. 【MFC】在工具栏中添加编辑框

    00. 目录 文章目录 00. 目录 01. 案例概述 02. 开发环境 03. 关键技术 04. 程序设计 05. 秘笈心法 06. 源码下载 07. 附录 01. 案例概述 在工具栏中添加编辑框可 ...

  3. ArcGIS JS先添加动态图层,再添加切片图层后不显示

    场景: 现在要在一个地图上添加动态服务图层和切片服务图层,当创建好图层之后,使用map.addLayer()方法,先把动态图层加到地图上,再把切片图层加到地图上,出现一个现象:只能看到加载的动态图层, ...

  4. Qt Creator添加动态dynamics

    Qt Creator添加动态 添加动态 制作动画 添加连接 加States 添加动态 您可以为UI组件的属性设置动画,并在它们之间创建连接以使它们能够相互通信. 制作动画 您可以在"时间轴& ...

  5. 如何html中添加动态图片,把动态图片添加到视频画面中 视频添加自定义动态图片 视频加动态logo...

    我前面也编写过关于视频添加动态图片的教程.前面所说的给是视频添加的动态图片是软件中自带的素材,虽然软件中带的动态图片种类繁多,但是不外乎有些时候软件中并没有我们要用的动态图片,这个时候我们就需要重外部 ...

  6. vue 分享微信传参_vue实现微信分享链接添加动态参数的方法

    微信分享时 分享链接携带参数可能不是固定的 需要在分享的前一刻才知道 这里就是动态设置分享链接的基本写法 代码不是那么详尽 但大致流程如下 1.安装引用jssdk npm install --save ...

  7. js 给动态li添加动态点击事件

    一.给动态li添加动态点击事件 试了网上的on,live,bind绑定,都不起作用.最后看到https://bbs.csdn.net/topics/390414057 上的代码.试了一遍,demo o ...

  8. mfc中在vector里添加控件类型的数据时出现C2248错误

    @mfc中在vector里添加控件类型的数据时出现C2248错误 这是我在.h文件中定义的2个容器 public:vector<CComboBox*> cbx;vector <CSt ...

  9. Android使用Opengl录像时添加(动态)水印

    最近需要开发一个类似行车记录仪的app,其中需要给录制的视频添加动态水印.我使用的是OpenGL开发的,刚开始实现的是静态水印,后面才实现的动态水印. 先上效果图,左下角的是静态水印,中间偏下的是时间 ...

  10. Hexo博客主题Next添加动态线条背景canvas_nest

    ​欢迎光临我的博客查看最新文章: https://river106.cn Hexo 版本:5.4.0 Next版本:8.7.1 为了美化Hexo博客,可以给博客添加动态线条背景,这里使用canvas_ ...

最新文章

  1. ios 三种颜色画笔和橡皮擦的画图板demo
  2. 由于使用zen coding导致myeclipse6.5中使用Alt+/不自动提示的解决办法
  3. 基于face++的人脸识别(上)
  4. AUTOSAR解决方案 — INTEWORK-EAS-CP
  5. 富文本编辑器的配置使用
  6. C++类中的封装-9
  7. 轻量级java开发工具_成为 Java 程序员必备的 5 个工具
  8. 2022年1月国产数据库排行榜:TiDB霸榜两年势头不减,openGauss与OceanBase分数大涨...
  9. canvas-缤纷小球
  10. 电脑硬盘:数据错误(循环冗余检查)
  11. 真实的感情---可是你没有
  12. 程序员再忙也应该看看《琅琊榜》
  13. 同轴电缆传输容易出现哪些干扰?
  14. macOS Big Sur正式版下载
  15. layuiAdmin 框架单页面文档
  16. Gradle-Eclipse插件安装
  17. Git统计代码行数;Java实现统计代码行数,忽略空行、注释行
  18. @DependsOn注解详解
  19. 致远的OA软件有什么特点?
  20. TestNG Annotations示例

热门文章

  1. imax6开发版_交叉编译
  2. 揭示大师的秘密:如何创作出杰作?
  3. MySQL篇(三)--DQL数据查询语言
  4. cad2016中选择全图字体怎么操作_PPT制作技巧,你知道图片怎么批量导出?文字怎么一键替换吗?...
  5. Python绘图实例27:正方体绘制
  6. Oracle 闪回查询
  7. DB2中Dynamic SQL Variation以及Varlock (V lock,Internal V lock)详解
  8. Qt MVD模型视图代理组件详解
  9. Docker快速部署Nexus3
  10. 2022.1.17-18 Javaweb Mybatis快速入门完成增删改查