自绘Tab控件
自绘tab按钮效果图如下:
使用例子:
MyTabControl *tabControl = NULL;
tabControl = new MyTabControl();
tabControl->Create(this, CRect(0,0,125,27), L"花生", ID_BTN_MYTAB);
tabControl->SetTabIndex(index);
tabControl->SetShowText(L"花生");
tabControl->SetTabState(TAB_STATE_DOWN);
tabControl->SetToolTipText(L"花生");
头文件:
- #pragma once
- #include "LhsButton.h"
- // MyTabControl
- #define ID_BTN_MYTAB_CLOSE 2100 //关闭按钮的id
- #define MYWM_BTN_TAB_CLOSE WM_USER+2001 //关闭按钮单击响应
- #define MYWM_BTN_TAB_CLICK WM_USER+2002 //tab按钮单击
- //tab按钮的状态
- enum TabState
- {
- TAB_STATE_NOR = 0,
- TAB_STATE_DOWN = 1,
- };
- class MyTabControl : public CWnd
- {
- DECLARE_DYNAMIC(MyTabControl)
- public:
- MyTabControl();
- virtual ~MyTabControl();
- bool Create(CWnd* pParent,CRect rc,CString text,DWORD id = 0,DWORD style = WS_VISIBLE|WS_CHILD);
- DECLARE_MESSAGE_MAP()
- public:
- void SetShowText(CString strShowText);
- protected:
- CString szClassName;
- bool m_isMouseHover; //鼠标是否悬浮
- bool m_isMouseClicked; //鼠标是否单击
- CString m_strShowText; //要显示的文字
- CString m_strTabText; //绘制在tab按钮下的文字
- Image* m_pImgNor; //正常时的图片
- Image* m_pImgHot; //鼠标悬浮时的图片
- Image* m_pImgDown; //单击按下时的图片
- void PostClickEvent();
- afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);
- afx_msg void OnMouseMove(UINT nFlags, CPoint point);
- afx_msg void OnMouseHover(UINT nFlags, CPoint point);
- afx_msg void OnMouseLeave();
- afx_msg void OnLButtonDown(UINT nFlags, CPoint point);
- afx_msg void OnLButtonUp(UINT nFlags, CPoint point);
- afx_msg BOOL OnEraseBkgnd(CDC* pDC);
- afx_msg void OnPaint();
- afx_msg void OnSize(UINT nType, int cx, int cy);
- afx_msg LRESULT OnBtnClose(WPARAM wParam, LPARAM lParam);
- virtual BOOL PreTranslateMessage(MSG* pMsg);
- public:
- void SetTabState(TabState state){m_tabState = state; Invalidate();} //设置tab状态
- TabState GetTabState(){return m_tabState;}
- void SetToolTipText(CString spText, BOOL bActivate = TRUE);
- void SetTabIndex(int idx){m_nIndex = idx;}
- int GetTabIndex(){return m_nIndex;}
- private:
- DWORD GetComfortSize(HDC hdc, DWORD dwWidth, CString &strText);
- private:
- TabState m_tabState; //tab的状态
- CLhsButton m_btnClose; //关闭按钮
- CToolTipCtrl* m_pToolTip;
- CString m_tooltext;
- bool m_bCenterAlign; //是否居中对齐
- int m_nIndex; //tab按钮的索引
- public:
- afx_msg HBRUSH OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor);
- };
- // MyTabControl.cpp : 实现文件
- //
- #include "stdafx.h"
- #include "../Lander_mini.h"
- #include "MyTabControl.h"
- #include "memdc.h"
- #include "../Utility.h"
- // MyTabControl
- IMPLEMENT_DYNAMIC(MyTabControl, CWnd)
- MyTabControl::MyTabControl()
- {
- m_isMouseHover = false;
- m_isMouseClicked = false;
- // 注册控件类
- szClassName = AfxRegisterWndClass(0);
- m_pImgNor = NULL;
- m_pImgHot = NULL;
- m_pImgDown = NULL;
- m_tabState = TAB_STATE_NOR;
- m_pToolTip = NULL;
- m_bCenterAlign = false;
- m_nIndex = 0;
- }
- MyTabControl::~MyTabControl()
- {
- SAFE_RELEASE(m_pToolTip);
- SAFE_RELEASE(m_pImgNor);
- SAFE_RELEASE(m_pImgHot);
- SAFE_RELEASE(m_pImgDown);
- }
- BEGIN_MESSAGE_MAP(MyTabControl, CWnd)
- ON_WM_MOUSEMOVE()
- ON_WM_MOUSEHOVER() // 此消息系统并不会给我们发送
- ON_WM_MOUSELEAVE()
- ON_WM_LBUTTONDOWN()
- ON_WM_LBUTTONUP()
- ON_WM_PAINT()
- ON_WM_SIZE()
- ON_WM_ERASEBKGND()
- ON_WM_CREATE()
- ON_MESSAGE(MYWM_BTN_CLICK, &MyTabControl::OnBtnClose)
- ON_WM_CTLCOLOR()
- END_MESSAGE_MAP()
- // MyTabControl 消息处理程序
- bool MyTabControl::Create(CWnd* pParent,CRect rc,CString text,DWORD id /* = 0 */,DWORD style /* = WS_VISIBLE|WS_CHILD */)
- {
- // 动态创建控件
- BOOL ret = CWnd::CreateEx(0, szClassName, text, style, rc, pParent, id);
- return ret ? true : false;
- }
- int MyTabControl::OnCreate(LPCREATESTRUCT lpCreateStruct)
- {
- //::SetWindowLong(m_hWnd, GWL_EXSTYLE, GetWindowLong(m_hWnd, GWL_EXSTYLE) ^ WS_EX_LAYERED);
- m_pImgNor = CQYUtility::LoadImgFromRes(_T("PNG"), IDB_PNG_TAB_NOR);
- m_pImgHot = CQYUtility::LoadImgFromRes(_T("PNG"), IDB_PNG_TAB_HOT);
- m_pImgDown = CQYUtility::LoadImgFromRes(_T("PNG"), IDB_PNG_TAB_DOWN);
- RECT rc = {0, 0, 20, 20};
- m_btnClose.LoadBtnImg(_T("PNG"),IDB_PNG_TAB_CLOSE_NOR, IDB_PNG_TAB_CLOSE_HOT,IDB_PNG_TAB_CLOSE_HOT);
- m_btnClose.Create(this, rc, L"",ID_BTN_MYTAB_CLOSE);
- m_btnClose.SetToolTipText(_T("关闭"));
- EnableToolTips(TRUE);
- return 0;
- }
- void MyTabControl::PostClickEvent()
- {
- if (m_tabState == TAB_STATE_DOWN)
- return; //选中的tab不响应单击了
- // 该函数用来向父窗口发送 单击 消息
- CWnd* parent = GetParent();
- if(parent)
- parent->SendMessage(MYWM_BTN_TAB_CLICK,m_nIndex,0);
- }
- void MyTabControl::OnMouseHover(UINT nFlags, CPoint point)
- {
- // 鼠标进入
- Invalidate();
- }
- void MyTabControl::OnMouseMove(UINT nFlags, CPoint point)
- {
- // 只处理鼠标第一次进入时的情况
- if(!m_isMouseHover)
- {
- m_isMouseHover = true;
- TRACKMOUSEEVENT evt = { sizeof(evt), TME_LEAVE|TME_HOVER, m_hWnd, 0 };
- TrackMouseEvent(&evt);
- OnMouseHover(0,CPoint());
- }
- }
- void MyTabControl::OnMouseLeave()
- {
- // 鼠标离开
- m_isMouseHover = false;
- m_isMouseClicked = false;
- Invalidate();
- }
- void MyTabControl::OnLButtonDown(UINT nFlags, CPoint point)
- {
- // 鼠标按下
- m_isMouseClicked = true;
- Invalidate();
- }
- void MyTabControl::OnLButtonUp(UINT nFlags, CPoint point)
- {
- // 鼠标松开
- if(m_isMouseClicked)
- {
- m_isMouseClicked = false;
- Invalidate();
- PostClickEvent();
- }
- }
- BOOL MyTabControl::OnEraseBkgnd(CDC* pDC)
- {
- // return CWnd::OnEraseBkgnd(pDC);
- return TRUE; // 阻止擦除背景,防止闪烁
- }
- void MyTabControl::OnPaint()
- {
- CPaintDC dc(this);
- CRect rc;
- GetClientRect(&rc);
- // 采用双缓存,防止闪烁
- CMemDC memdc(&dc,&rc,TRUE);
- Graphics graphic(memdc);
- if (!m_pImgNor || !m_pImgHot || !m_pImgDown)
- {//没有提供按钮图片就刷下背景
- // 刷背景
- COLORREF bkgnd = RGB(100,0,0);
- if(m_isMouseHover)
- {
- if(m_isMouseClicked)
- bkgnd = RGB(250,0,0);
- else
- bkgnd = RGB(180,0,0);
- }
- memdc.FillSolidRect(&rc,bkgnd);
- }
- if (m_isMouseClicked || m_tabState == TAB_STATE_DOWN)
- {//单击一定画单击状态
- graphic.DrawImage(m_pImgDown, 0, 0, m_pImgDown->GetWidth(), m_pImgDown->GetHeight());
- }
- else if (m_isMouseHover && !m_isMouseClicked)
- {
- //悬浮,但是没单击
- graphic.DrawImage(m_pImgHot, 0, 0, m_pImgHot->GetWidth(), m_pImgHot->GetHeight());
- }
- else
- {
- graphic.DrawImage(m_pImgNor, 0, 0, m_pImgNor->GetWidth(), m_pImgNor->GetHeight());
- }
- if (!m_strTabText.IsEmpty())
- {
- // 设置文字字体
- CFont font;
- font.CreatePointFont(100,L"宋体"); // 11号字体,该参数与实际字体号有10倍的关系
- CFont* poldFont = memdc.SelectObject(&font);
- // 设置文字属性
- memdc.SetBkMode(TRANSPARENT);
- if (m_tabState == TAB_STATE_DOWN)
- memdc.SetTextColor(RGB(136,76,25));
- else
- memdc.SetTextColor(RGB(255,245,190));
- // 绘制文本
- DWORD style = DT_SINGLELINE | DT_VCENTER ; // 文本格式:单行+水平居中+垂直居中
- //if (m_bCenterAlign)
- // style |= DT_CENTER;
- CRect fontRect(rc);
- fontRect.left += 13;
- fontRect.right -= 22;
- memdc.DrawText(m_strTabText, -1, &fontRect, style); // 更多文本显示格式可参考百度百科DrawText说明
- memdc->SelectObject(poldFont);
- }
- // 使绘制生效
- graphic.ReleaseHDC(memdc);
- }
- BOOL MyTabControl::PreTranslateMessage(MSG* pMsg)
- {
- if (m_pToolTip)
- {
- if (::IsWindow(m_pToolTip->m_hWnd))
- {
- m_pToolTip->RelayEvent(pMsg);
- }
- }
- return CWnd::PreTranslateMessage(pMsg);
- }
- void MyTabControl::OnSize(UINT nType, int cx, int cy)
- {
- CRect rect;
- GetClientRect(&rect);
- m_btnClose.SetWindowPos(NULL, rect.right - 22, rect.top + 2, 20, 20, SWP_NOZORDER);
- }
- LRESULT MyTabControl::OnBtnClose(WPARAM wParam, LPARAM lParam)
- {
- // 该函数用来向父窗口发送 单击 消息
- CWnd* parent = GetParent();
- if(parent != NULL)
- {
- parent->SendMessage(MYWM_BTN_TAB_CLOSE, m_nIndex, 0);
- }
- return 0;
- }
- void MyTabControl::SetToolTipText(CString spText, BOOL bActivate)
- {
- if (m_pToolTip == NULL)
- {
- m_pToolTip = new CToolTipCtrl;
- // Create ToolTip control
- m_pToolTip->Create(this);
- m_pToolTip->Activate(TRUE);
- }
- m_tooltext = spText;
- // If there is no tooltip defined then add it
- if (m_pToolTip->GetToolCount() == 0)
- {
- CRect rectBtn;
- GetClientRect(rectBtn);
- m_pToolTip->AddTool(this, m_tooltext, rectBtn, 1);
- }
- // Set text for tooltip
- m_pToolTip->UpdateTipText(m_tooltext, this, 1);
- m_pToolTip->SetDelayTime(2000);
- m_pToolTip->Activate(bActivate);
- }
- DWORD MyTabControl::GetComfortSize(HDC hdc,DWORD dwWidth,CString &strText)
- {
- //二分法查找
- DWORD dwComfortSize = 0;
- DWORD dwBeginSize = 0;
- DWORD dwEndSize = strText.GetLength();
- while(TRUE)
- {
- DWORD dwMiddleSize = (dwEndSize + dwBeginSize) / 2;
- if(dwMiddleSize == dwBeginSize || dwMiddleSize == dwEndSize)
- {
- //两个点之间已经没有数值可以检测,退出循环
- dwComfortSize = dwBeginSize;
- break;
- }
- SIZE sizeChk = {0};
- ::GetTextExtentPoint(hdc, strText.GetBuffer(), dwMiddleSize, &sizeChk);
- if(sizeChk.cx == dwWidth)
- {
- //数值刚好合适,跳出循环
- dwComfortSize = dwMiddleSize;
- break;
- }
- else if(static_cast<DWORD>(sizeChk.cx) > dwWidth)
- {
- //重新设置边界
- dwEndSize = dwMiddleSize;
- }
- else
- {
- //重新设置边界
- dwBeginSize = dwMiddleSize;
- }
- }
- return dwComfortSize;
- }
- void MyTabControl::SetShowText(CString strShowText)
- {
- m_strShowText = strShowText;
- CRect rect;
- GetClientRect(&rect);
- SIZE sizeEllipsis = {0};
- ::GetTextExtentPoint(GetDC()->GetSafeHdc(), strShowText.GetBuffer(), lstrlen(strShowText.GetBuffer()), &sizeEllipsis);
- if (sizeEllipsis.cx < rect.Width() - 40)
- {//显示区域够宽
- m_bCenterAlign = true;
- m_strTabText = strShowText;
- return;
- }
- memset(&sizeEllipsis,0, sizeof(sizeEllipsis));
- CString ellipsStr = _T("...");
- ::GetTextExtentPoint(GetDC()->GetSafeHdc(), ellipsStr.GetBuffer(), lstrlen(ellipsStr.GetBuffer()), &sizeEllipsis);
- DWORD dwMaxDisp = GetComfortSize(GetDC()->GetSafeHdc(), rect.Width() - sizeEllipsis.cx - 20, strShowText);
- if (dwMaxDisp >= strShowText.GetLength())
- {
- m_strTabText = strShowText;
- }
- else
- {
- m_strTabText = m_strShowText.Left(dwMaxDisp);
- m_strTabText += ellipsStr;
- }
- }
- HBRUSH MyTabControl::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
- {
- HBRUSH hbr = CWnd::OnCtlColor(pDC, pWnd, nCtlColor);
- // TODO: 在此更改 DC 的任何属性
- // m_btnClose.SetBkGnd(pDC);
- // TODO: 如果默认的不是所需画笔,则返回另一个画笔
- return hbr;
- }
//内存DC类和按钮类见链接:点击打开链接
http://blog.csdn.net/huasonl88/article/details/43227291
自绘Tab控件的更多相关文章
- 扩展easyUI tab控件,添加加载遮罩效果
项目里要用HighChart显示图表,如果返回的数量量太多,生成图表是一个很耗时的过程.tab控件又没有显示遮罩的设置(至少本菜是没有找到), Google了一下,根据另一个兄台写的方法,拿来改造了一 ...
- NVelocity+Bootstrap tab控件 异常之
异常信息:Encountered "tings" at line 54, column 55.Was expecting one of: "(" ... ...
- Android Tab控件简介
在Android中,Tab控件是一种很常用的控件:Tab控件即标签页,可以在一页中切换显示N页内容: Tab控件具有两种实现过程,一是在同一个Activity中切换显示不同的标签页,这种主要是通过修改 ...
- Web端的Tab控件在切换Tab时Load数据出错的处理
我们在应用Web端的Tab控件时,不管是Jquery easyui的还是Ext的Tab控件都会遇到一个问题,在Tab1正在加载数据的时候我们切换到Tab2,再切换回来,Load数据的控件就会出错,出错 ...
- 开源框架之TAB控件
我的开源框架之TAB控件 需求 (1)支持iframe.html.json格式的tab内容远程请求 (2)支持动态添加tab (3)支持远程加载完成监听,支持tab激活事件监听 (4)支持relo ...
- 终于知道如何使Tab控件的不出现白边的方法了
如下图,在棋盘右侧添加了Tab控件,做成属性页的样子,但出现了白边,很不美观: 后来发现,需要把Tab空间的Owner Draw Fixed 设置为TRUE.但问题又来了,属性页上的标题文字不显示了, ...
- WPF下可编辑Header的Tab控件实现
介绍 有这样一个需求,当用户双击Tab控件Header区域时, 希望可以直接编辑.对于WPF控件,提供一个ControlTemplate在加上一些Trigger就可以实现.效果如下: 代码 首先,我们 ...
- 网页Tab控件
网页Tab控件 找到:http://www.open-open.com/ajax/2_Tabs.htm 页面,查看了若干Tab控件, 找到了:http://www.open-open.com/ajax ...
- 使用jQuery的tab控件
以前写winform程序的时候tab控件是非常容易使用的,写网页时确费了我不少劲,主要原因是jQuery的混乱. 有很多前端控件都提供tab,像bootstrap等,这里只说jQuery的. 下载相应 ...
随机推荐
- Do Palapala (this)
Description 伟大的中国人民有宝箱容量为S(0<=S<=20000),有m个物品(0<m<=30,每个物品有一个体积(正整数).任取若干个装入箱内,使箱子的剩余空间为 ...
- Ecside基于数据库的过滤、分页、排序
首先ecside展现列表.排序.过滤(该三种操作以下简称为 RSF )的实现原理完全和原版EC一样, 如果您对原版EC的retrieveRowsCallback.sortRowsCallback.fi ...
- Android 开源控件系列_1
第一部分 个性化控件(View) 主要介绍那些不错个性化的View,包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Pro ...
- DataGridview 填写数字
private DataGridViewTextBoxEditingControl CellEdit = null; // 声明 一个 CellEdit private void dgv ...
- Redis深入学习(1)前言&Redis简介
前言 最近工作上使用到Redis,当然以前也使用过redis,win,linux上都使用过,不系统,不深入,仅是头痛医头,脚痛医脚,这里整理一下自己的笔记,一来方便自己记忆,二来对同行提供借鉴,不足错 ...
- 在mysql 中两种锁定问题
mysql 中15.2.10.5 中描述了两个问题,且分别给出了解决办法. 1.向子表中写入数据,但写入之前需确保父表中存在其相应信息. 可能出现,在已经读取父表中的数据,但另一请求将其删除. 办法: ...
- css中element element和element>element选择器的区别
就是这样的选择器: 比如html中有这样一段布局: <div> <p>我是一个段落</p> </div> 这时你用div p{background:ye ...
- Dynamics CRM 2013 初体验(5):Business Rule
新系统中的Business Rule是个不错的功能,相信它的出现能减少大量的开发工作.在日常开发中,我们需要对记录做大量的业务控制.比如:某字段是否要隐藏,某字段的值是否符合要求以及现实提醒信息等.在 ...
- 用ATL和MFC来创建ActiveX控件
摘要:目前MFC和ATL代表了两种框架,分别面向不同类型的基于Windows的开发.MFC代表了创建独立的Windows应用的一种简单.一致的方法:ATL提供了一种框架来实现创建COM客户机和服务器所 ...
- 在OpenCV中利用鼠标绘制矩形和截取图像的矩形区域
这是两个相关的程序,前者是后者的基础.实际上前一个程序也是在前面博文的基础上做的修改,请参考<在OpenCV中利用鼠标绘制直线> .下面贴出代码. 程序之一,在OpenCV中利用鼠标绘制矩 ...