DuiLib――第一篇UIManager

DUiLib 源码分析 ――以UiLib 1.01版为分析目标
----------------------------------------------------------------------------------
分析约定:
private o------- 私有的成员变量或方法
protect x------- 受保护的成员变量或方法
public  +------- 公开的成员变量或方法
----------------------------------------------------------------------------------

本篇分析一下源文件
UIManager.h/UIManager.cpp

CPaintManagerUI
成员变量
 
窗体句柄
 o---m_hWndPaint                要CPaintManagerUI进行Direct绘图操作的窗体句柄
 o---m_hwndTooltip              提示窗口句柄
 o---m_hInstance                当前管理的Instance实例
 o---m_hResourceInstance        当前管理的资源DLL Instance实例
 o---m_pStrResourcePath         当前使用的资源路径
 o---m_pStrResourceZip          当前使用的资源压缩包文件全称

    
 
 
 ------------------------------------------------------CPaintManagerUI使用的资源
 绘图设备
 o---m_hDcPaint                 直接绘制到窗体的DC(为窗体的整个区域包括费客户区)
 o---m_hDcOffscreen             内存缓冲区绘图DC
 o---m_hDcBackground            背景绘制(支持AlphaBackground时使用)
 位图
 o---m_hbmpOffscreen            m_hDcPaint的后台作图画布
 o---m_hbmpBackground           背景图片bmp
 
 ------------------------------------------------------CPaintManagerUI用到的信息
 o---m_ToolTip                  提示消息
        typedef struct tagTOOLINFOA {
            UINT cbSize;                //该结构体的大小 sizeof(TOOLINFO)
            UINT uFlags;                //附加标识类信息
            HWND hwnd;                  //消息接受的窗体
            UINT_PTR uId;               //控件ID
            RECT rect;                  //消息产生的区域位置
            HINSTANCE hinst;            //消息接收的实例
            LPSTR lpszText;             //提示消息
            LPARAM lParam;              //IE3.0以上的版本有该属性
            void *lpReserved;           //NT5.0以上的版本有该属性 附加信息
        } TOOLINFO
 标识类信息       
 o---m_bShowUpdateRect          是否显示更新区域
 o---m_bFirstLayout             是否是首个布局 
 o---m_bUpdateNeeded            是否需要更新界面
 o---m_bFocusNeeded             是否需要焦点
 o---m_bOffscreenPaint          是否需要开双缓存绘图
 o---m_bAlphaBackground         窗体背景是否需要支持Alpha通道(如png图片的半透明效果)
 o---m_bMouseTracking           是否需要支持鼠标追踪
 o---m_bMouseCapture            是否需要支持鼠标捕获
 
 控件信息
 o---m_pRoot                    xml根节点解析成的对象,通常为各种Window
 o---m_pFocus                   处于获得焦点状态的控件
 o---m_pEventHover              处于鼠标悬停状态的控件
 o---m_pEventClick              被鼠标点击的控件
 o---m_pEventKey                接收键盘输入的控件
 
 位置记录信息
 o---m_pLastMousePos            鼠标最新的位置
 o---m_szMinWindow              设置窗体可以调整到的最小大小
 o---m_szMaxWindow              窗体可以调整到的最大大小
 o---m_szInitWindowSize         窗体初始化是的大小
 o---m_rcSizeBox                窗体外边框区域的大小
 o---m_szRoundCorner            窗体四角的圆角弧度
 o---m_rcCaption                窗体标题栏区域大小
 o---m_uTimerID                 当前定时器ID
 
 集合类信息
 o---m_aNotifiers               能够接收通知的对象集合
 o---m_aTimers                  定时器集合
 o---m_aPreMessage              预处理消息集合
 o---m_aPreMessageFilters       预处理消息过滤器集合
 o---m_aMessageFilters          消息过滤器集合
 o---m_aPostPaintControls       发送绘制请求的控件集合
 o---m_aDelayedCleanup          延迟清理的对象集合
 o---m_aAsyncNotify             异步通知消息集合
 o---m_mNameHash                名称HashMap
 o---m_mOptionGroup             选项组Map
 
 xml对应资源
 o---m_pParentResourcePM        上级(父类)资源的PaintManagerUI绘图管理器
 o---m_dwDefaultDisabledColor   默认失效状态颜色
 o---m_dwDefaultFontColor       默认字体颜色
 o---m_dwDefaultLinkFontColor   默认超链接字体颜色
 o---m_dwDefaultLinkHoverFontColor默认超链接鼠标悬停状态的字体颜色
 o---m_dwDefaultSelectedBkColor 默认选中状态背景色
 o---m_DefaultFontInfo          默认字体信息
        TFontInfo{
            hFont               该字体的句柄
            sFontName           字体名称
            iSize               字号
            bBold               是否粗体
            bUnderline          是否有下划线
            bItalic             是否为斜体
            TEXTMETRIC tm       该字体的TEXTMETRIC信息
        }
 o---m_aCustonFonts             自定义字体资源集合
 o---m_mImageHash               图片资源HashMap
 o---m_DefaultAttrHash          DefaultAttr资源HashMap

私有方法
            将所有的控件添加到m_mNameHash哈希表中
 o---static CControlUI* CALLBACK __FindControlFromNameHash(CControlUI* pThis, LPVOID pData);
            计算控件数量
 o---static CControlUI* CALLBACK __FindControlFromCount(CControlUI* pThis, LPVOID pData);
            根据点是否在区域中,查询控件
 o---static CControlUI* CALLBACK __FindControlFromPoint(CControlUI* pThis, LPVOID pData);
            通过Tab信息查询控件
 o---static CControlUI* CALLBACK __FindControlFromTab(CControlUI* pThis, LPVOID pData);
            从快照中查询控件
 o---static CControlUI* CALLBACK __FindControlFromShortcut(CControlUI* pThis, LPVOID pData);
            查找需要更新的控件
 o---static CControlUI* CALLBACK __FindControlFromUpdate(CControlUI* pThis, LPVOID pData);
            通过名称比较查询控件
 o---static CControlUI* CALLBACK __FindControlFromName(CControlUI* pThis, LPVOID pData);

公开方法
            绘图管理器的初始化(m_hWndPaint,m_hDcPaint赋值,在预处理消息中加入管理器)
 +---void Init(HWND hWnd);
            当前需要更新界面
 +---void NeedUpdate();
            指定区域失效
 +---void Invalidate(RECT& rcItem);
            获取绘图设备DC
 +---HDC GetPaintDC() const;
            获取绘图的窗口句柄
 +---HWND GetPaintWindow() const;
            获取提示窗体句柄
 +---HWND GetTooltipWindow() const;
            获取当前鼠标的位置
 +---POINT GetMousePos() const;
            获取客户区大小
 +---SIZE GetClientSize() const;
            获取窗体初始化时的大小
 +---SIZE GetInitSize();
            设置窗体初始化大小
 +---void SetInitSize(int cx, int cy);
            获取窗体的边框区域大小
 +---RECT& GetSizeBox();
            设置窗体的边框区域大小
 +---void SetSizeBox(RECT& rcSizeBox);
            获取标题区域位置
 +---RECT& GetCaptionRect();
            设置标题区域位置
 +---void SetCaptionRect(RECT& rcCaption);
            获取窗体四角的圆角弧度
 +---SIZE GetRoundCorner() const;
            设置窗体四角的圆角弧度
 +---void SetRoundCorner(int cx, int cy);
            获取窗体可以调整到的最小大小
 +---SIZE GetMinInfo() const;
            设置窗体可以调整到的最小大小
 +---void SetMinInfo(int cx, int cy);
            获取窗体可以调整到的最大大小
 +---SIZE GetMaxInfo() const;
            设置窗体可以调整到的最大大小
 +---void SetMaxInfo(int cx, int cy);
            窗体的不透明度(0完全透明-255完全不透明)
 +---void SetTransparent(int nOpacity);
            设置绘图是否支持透明处理
 +---void SetBackgroundTransparent(bool bTrans);
            是否显示更新区域
 +---bool IsShowUpdateRect() const;
            设置是否显示更新
 +---void SetShowUpdateRect(bool show);
            获取当前管理的实例句柄
 +---static HINSTANCE GetInstance();
            获得当前运行的实例的路径
 +---static CStdString GetInstancePath();
            获得当前的工作路径
 +---static CStdString GetCurrentPath();
            获取资源DLL的实例句柄
 +---static HINSTANCE GetResourceDll();
            获取资源的路径(以""结尾)
 +---static const CStdString& GetResourcePath();
            获得Zip资源的路径
 +---static const CStdString& GetResourceZip();
            设置实例句柄
 +---static void SetInstance(HINSTANCE hInst);
            设置当前的工作路径
 +---static void SetCurrentPath(LPCTSTR pStrPath);
            设置当前的DLL资源的实例句柄
 +---static void SetResourceDll(HINSTANCE hInst);
            设置资源所在文件夹路径
 +---static void SetResourcePath(LPCTSTR pStrPath);
            设置Zip资源的路径(包括Zip文件名)
 +---static void SetResourceZip(LPCTSTR pStrZip);
            设置使用上级资源的绘图管理器
 +---bool UseParentResource(CPaintManagerUI* pm);
            获得上级资源绘图管理器
 +---CPaintManagerUI* GetParentResource() const;
            获取禁用状态的默认颜色
 +---DWORD GetDefaultDisabledColor() const;
            设置禁用状态的默认颜色
 +---void SetDefaultDisabledColor(DWORD dwColor);
            获取字体默认颜色
 +---DWORD GetDefaultFontColor() const;
            设置字体默认颜色
 +---void SetDefaultFontColor(DWORD dwColor);
            设置链接文字的默认字体颜色           
 +---DWORD GetDefaultLinkFontColor() const;
            获取链接文字的默认颜色
 +---void SetDefaultLinkFontColor(DWORD dwColor);
            获取鼠标悬停与超链上的默认字体颜色
 +---DWORD GetDefaultLinkHoverFontColor() const;
            获取鼠标悬停与超链上的默认字体颜色
 +---void SetDefaultLinkHoverFontColor(DWORD dwColor);
            获取选中状体的默认背景颜色
 +---DWORD GetDefaultSelectedBkColor() const;
            设置选中状态的默认背景颜色
 +---void SetDefaultSelectedBkColor(DWORD dwColor);
            获取默认使用的字体信息
 +---TFontInfo* GetDefaultFontInfo();
            设置默认使用的字体信息
 +---void SetDefaultFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            获取用户自定义字体的数量(一般对应xml中Font的数量)
 +---DWORD GetCustomFontCount() const;
            向字体数组列表追加字体资源
 +---HFONT AddFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            向字体数组列表中插入字体资源
 +---HFONT AddFontAt(int index, LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            获取数组中指定下标的字体对象句柄
 +---HFONT GetFont(int index);
            从字体数组中获取指定配置的字体对象句柄
 +---HFONT GetFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            字体数组集合中是否存在字体对象
 +---bool FindFont(HFONT hFont);
            字体数组集合中是否存在指定配置的字体对象
 +---bool FindFont(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            获得字体对象的
 +---int GetFontIndex(HFONT hFont);
            根据指定的配置信息查询字体索引
 +---int GetFontIndex(LPCTSTR pStrFontName, int nSize, bool bBold, bool bUnderline, bool bItalic);
            从字体数组列表中移除字体对象
 +---bool RemoveFont(HFONT hFont);
            从字体数组列表中移除指定位置的字体信息
 +---bool RemoveFontAt(int index);
            清空字体数组列表
 +---void RemoveAllFonts();
            通过字体数组索引查找字体信息
 +---TFontInfo* GetFontInfo(int index);
            通过字体对象句柄获取字体信息
 +---TFontInfo* GetFontInfo(HFONT hFont);
            根据图像路径查找图像信息
 +---const TImageInfo* GetImage(LPCTSTR bitmap);
            根据名称,类型,遮罩色 查询 图像信息
 +---const TImageInfo* GetImageEx(LPCTSTR bitmap, LPCTSTR type = NULL, DWORD mask = 0);
            添加图像
 +---const TImageInfo* AddImage(LPCTSTR bitmap, LPCTSTR type = NULL, DWORD mask = 0);
            添加图像
 +---const TImageInfo* AddImage(LPCTSTR bitmap, HBITMAP hBitmap, int iWidth, int iHeight, bool bAlpha);
            根据图像名称移除图像
 +---bool RemoveImage(LPCTSTR bitmap);
            移除全部图像
 +---void RemoveAllImages();
            添加控件的默认配置信息(如button)
 +---void AddDefaultAttributeList(LPCTSTR pStrControlName, LPCTSTR pStrControlAttrList);
            根据控件名称查询该类控件的默认配置
 +---LPCTSTR GetDefaultAttributeList(LPCTSTR pStrControlName) const;
            移除指定控件类型名称的默认配置
 +---bool RemoveDefaultAttributeList(LPCTSTR pStrControlName);
            获取默认配置信息列表
 +---const CStdStringPtrMap& GetDefaultAttribultes() const;
            清空默认配置信息列表
 +---void RemoveAllDefaultAttributeList();
            将对话框控件附加到当前的管理器中
 +---bool AttachDialog(CControlUI* pControl);
            控件初始化
 +---bool InitControls(CControlUI* pControl, CControlUI* pParent = NULL);
            控件回收
 +---void ReapObjects(CControlUI* pControl);

            添加控件到指定的选项组
 +---bool AddOptionGroup(LPCTSTR pStrGroupName, CControlUI* pControl);
            查询指定选项组名称中的全部选项
 +---CStdPtrArray* GetOptionGroup(LPCTSTR pStrGroupName);
            从指定控件中移除指定选项组名称的选项组
 +---void RemoveOptionGroup(LPCTSTR pStrGroupName, CControlUI* pControl);
            清空全部选项组列表
 +---void RemoveAllOptionGroups();

            获取焦点状态的控件
 +---CControlUI* GetFocus() const;
            设置控件为获得焦点状态
 +---void SetFocus(CControlUI* pControl);
            设置控件为需要绘制焦点
 +---void SetFocusNeeded(CControlUI* pControl);

            设置下一个获得Tab键会获得焦点的控件,Tab是否继续往下走
 +---bool SetNextTabControl(bool bForward = true);
            
            为指定控件以及其子控件设置定时器
 +---bool SetTimer(CControlUI* pControl, UINT nTimerID, UINT uElapse);
            移除指定控件上的指定编号的定时器
 +---bool KillTimer(CControlUI* pControl, UINT nTimerID);
            清空所有的定时器
 +---void RemoveAllTimers();
            
            设置窗体接受鼠标事件
 +---void SetCapture();
            释放窗体捕获鼠标事件
 +---void ReleaseCapture();
            判断窗体是否接受鼠标事件
 +---bool IsCaptured();

            添加控件到通知集合中
 +---bool AddNotifier(INotifyUI* pControl);
            将控件从通知集合中移除
 +---bool RemoveNotifier(INotifyUI* pControl);   
            发送同步/异步通知
 +---void SendNotify(TNotifyUI& Msg, bool bAsync = false);
            构建同步或异步通知并发送
 +---void SendNotify(CControlUI* pControl, LPCTSTR pstrMessage, WPARAM wParam = 0, LPARAM lParam = 0, bool bAsync = false);


            向预处理消息过滤器链中添加消息过滤器
 +---bool AddPreMessageFilter(IMessageFilterUI* pFilter);
            从预处理消息过滤器链合中移除指定的消息过滤器
 +---bool RemovePreMessageFilter(IMessageFilterUI* pFilter);

            向消息过滤器链中添加消息过滤器
 +---bool AddMessageFilter(IMessageFilterUI* pFilter);
            从消息过滤器链中移除消息过滤器
 +---bool RemoveMessageFilter(IMessageFilterUI* pFilter);
            获取发送需要绘制的控件的数量
 +---int GetPostPaintCount() const;
            向绘制请求集合中添加要绘制的控件
 +---bool AddPostPaint(CControlUI* pControl);
            从绘制请求集合中移除指定的控件
 +---bool RemovePostPaint(CControlUI* pControl);
            将绘制请求控件插入到绘制请求集合的指定位置
 +---bool SetPostPaintIndex(CControlUI* pControl, int iIndex);
            向延迟清理集合中添加需要延迟清理的对象
 +---void AddDelayedCleanup(CControlUI* pControl);
            获取根节点控件
 +---CControlUI* GetRoot() const;
            从根节点开始查找指定点所在的控件
 +---CControlUI* FindControl(POINT pt) const;
            从指定节点开始查找指定点所在的控件
 +---CControlUI* FindControl(CControlUI* pParent, POINT pt) const;
            从根节点开始,查找指定名称的控件
 +---CControlUI* FindControl(LPCTSTR pstrName);
            从指定节点开始查找指定名称的控件
 +---CControlUI* FindControl(CControlUI* pParent, LPCTSTR pstrName);

            消息循环,非游戏框架消息泵,无法利用无消息的空闲时间            
 +---static void MessageLoop();
            消息翻译,在Win32原有的消息转换基础上,将需要自己处理的消息转发给消息预处理器
 +---static bool TranslateMessage(const LPMSG pMsg);
            消息预处理器
            1.消息预处理过滤(消息预处理过滤器集合对消息进行过滤处理)
            2.检查是否按下Tab键,设置下一个获得焦点的控件
            3.处理Alt+Shortcut Key按下后的控件获得焦点和激活的设置
            4.检查是否有系统键消息,有则发送获得焦点的控件的事件
 +---bool PreMessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT& lRes);
 
 
----------------------------------------------------------------------------------------
----------------------------------------------------------------------------------------
消息处理器(核心处理器)
1.消息过滤
2.检查Custom消息并处理
3.检查是否有WM_CLOSE消息并处理
4.处理WM_ERASEBKGND(不允许进行背景擦除,防止闪烁)
5.绘制处理(核心)
    5.1做延迟绘图判断,当前是否有 窗体大小调整的操作,或者是否需要初始化窗体
    5.2设置焦点控件
    5.3如果开启双缓存绘图,采用双缓存方式绘图,否则采用标准绘图方式绘图
    5.4               
6.处理客户区的绘制WM_PRINTCLIENT
7.接到WM_GETMINMAXINFO消息后向系统提交该窗体可调整大小的最小和最大限制
8.窗体大小改变时,向焦点控件发送改变大小消息并设置窗体需要更新
9.处理定时器消息,向定时器集合中广播定时消息
10.处理鼠标悬停
    10.1向鼠标悬停的控件发送鼠标悬停消息
    10.2如果当前控件有提示消息,创建消息提示窗体
11.处理鼠标离开事件,关闭消息提示框,发送鼠标离开消息,取消鼠标的追踪
12.鼠标移动时,开始追踪鼠标
    12.1处理鼠标移动时,鼠标在控件上进入,移动,悬停和离开的消息
13.处理鼠标左键按下的消息设定活动的焦点的控件
14.鼠标双击事件处理,向需要接收鼠标双击事件的控件发送双击事件
15.鼠标左键抬起时,向上次接收到点击消息的控件发送鼠标左键抬起的消息
16.鼠标右键按下时,向需要接收鼠标右键按下的控件发送右键按下消息
17.鼠标右键快捷菜单消息,将该消息通知给上次点击过的按钮
18.滚轮消息时,象鼠标所在的控件发送滚轮消息
19.WM_CHAR 消息时,向获得焦点的控件发送该消息
20.键盘按下时,向焦点控件发送该键盘消息,并设定焦点控件为键盘消息控件
21.键盘按键抬起时,向事键盘消息控件发送该事件
22.设定鼠标光标消息时,获得光标所在控件接收该消息
23.通知消息到来时,加OCM_BASE后发送通知消息
24.命令消息到来,加OCM_BASE后发送消息
25.WM_CTLCOLOREDIT,STATIC消息到来后,加OCM_BASE后发送消息
 +---bool MessageHandler(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT& lRes);


你可能感兴趣的:(UIManager)