现在的位置: 首页 > 综合 > 正文

如何在VC中使用自定义的SkinMagic控件?

2012年01月15日 ⁄ 综合 ⁄ 共 3000字 ⁄ 字号 评论关闭
 在SkinMagic中自定义了一个控件皮肤,并且添加到皮肤文件(.smf);但是,如何在vc程序中使用它呢?本文以CSplitterWnd为例子,说明使用方法。

我想对分割条类CSplitterWnd设计皮肤,SkinMagic没有提供该标准控件;于是,我自定义了该控件。

控件皮肤类型名叫:"SkinSplitter",三个主要属性:

SplitterBar:IMAGERECT类型

Border:IMAGESETION类型

Gripper:IMAGESETION类型

并在皮肤文件中添加了该对象,对象名为"SkinSplitter"。

如图:

现在说说在程序中使用方法,假设我们已经将皮肤文件载入程序,我们重载CSplitterWnd的类名叫CXSplitterWnd。

1、注册控件皮肤对象

重载CSplitterWnd::OnCreate(),然后调用RegisterSkinWindow(m_hWnd,0):

int CXSplitterWnd::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
    if (CSplitterWnd::OnCreate(lpCreateStruct) == -1)
        return -1;
 
    RegisterSkinWindow(m_hWnd, 0);
 
    return 0;
}

2、控件销毁时,反注册控件皮肤对象

重载CSplitterWnd::OnDestroy(),然后调用UnregisterSkinWindow和CloseSkinData:

void CXSplitterWnd::OnDestroy()
{
    CSplitterWnd::OnDestroy();
 
    UnregisterSkinWindow(m_hWnd);
    CloseSkinData(m_hSkin);// m_hSkin是在CXSplitterWnd添加的HSKIN类型成员
}

3、添加消息WM_SKINDATACHANGED的响应函数

WM_SKINDATACHANGED是SkinMagic SDK自定义的消息,表示皮肤数据发生改变。编写消息响应函数请读者自行查找,很方便的。:)

我们定义该消息响应函数为OnSkinDataChanged;在其中载入皮肤文件中的对象SkinSplitter:

LRESULT CXSplitterWnd::OnSkinDataChanged(WPARAM wParam, LPARAM lParam)
{
 if( m_hSkin ) CloseSkinData( m_hSkin );

 m_hSkin = OpenSkinData("SkinSplitter");// 皮肤文件中的自定义皮肤对象
 if( m_hSkin )
 {
  RedrawWindow();
 }
 return 1;
}

4、调用DrawSkinImageRect、DrawSkinImageSetion等方法重绘分割条

我采取的方法是重载CSplitterWnd::OnPaint(),在其中重绘分割条:

void CXSplitterWnd::OnPaint()

{

    ......

   // rectSplitter是分割条的真实方形区域

   // 获取分割条的真实方形区域的方法请看我的另一篇文章"如何获取分割条的真实RECT? "

  DrawSkinImageRect(m_hSkin, _T("SplitterBar"), dc, &rectSplitter); // 根据皮肤对象SkinSplitter属性SplitterBar重绘
  DrawSkinImageSection(m_hSkin, _T("Border"), dc, &rectSplitter); // 根据皮肤对象SkinSplitter属性Border重绘
  DrawSkinImageSection(m_hSkin, _T("Gripper"), dc, &rectSplitter); // 根据皮肤对象SkinSplitter属性Gripper重绘

}

到此结束!

其他自定义控件的使用方法大致相同,只是在重载的函数上可能不同。

希望能帮到大家,:)

 

 

是不是想重画分割条,但是用CSplitterWnd::GetRect(...)方法获取后,发现不对呢?这里教你如何获取分割条真实的方形区域。

假如你的分割条类型是CMySplitterWnd(继承自CSplitterWnd),对象是m_Splitter,则
m_Splitter.GetRect(rect)获得的方形区域实际上是整个分割窗口的方形区域。如果想获得分割条的RECT,就得自己计算。方法如
下:

//(假设我们重载了CSplitterWnd::OnPaint())

void CMySplitterWnd::OnPaint()
{
    CPaintDC dc(this); // device context for painting

    int iColumn = 0;
    int iRow = 0;
    RECT rect1, rect2, rectSplitter;

    // 获取每一个垂直的分割条的RECT

    for (iColumn=1; iColumn<GetColumnCount(); iColumn++)
    {
        GetPane(0, iColumn-1)->GetWindowRect(&rect1);
        GetPane(GetRowCount()-1, iColumn)->GetWindowRect(&rect2);

        rectSplitter.top = rect1.top;
        rectSplitter.bottom = rect2.bottom;
        rectSplitter.right = rect2.left;
        rectSplitter.left = rect1.right;

        ScreenToClient(&rectSplitter);
       // ... do what u want to do!

    }

    // 获取每一个水平的分割条的RECT

    for (iRow=1; iRow<GetRowCount(); iRow++)
    {
        GetPane(iRow-1, 0)->GetWindowRect(&rect1);
        GetPane(iRow, GetColumnCount()-1)->GetWindowRect(&rect2);

        rectSplitter.top = rect1.bottom;
        rectSplitter.bottom = rect2.top;
        rectSplitter.right = rect2.right;
        rectSplitter.left = rect1.left;

        ScreenToClient(&rectSplitter);
        // ...do what u want to do!
     }
}

 

也有另外一种办法,就是用WTL7.1中的CSplitterWindowImpl类。它的成员函数GetSplitterBarRect(...)可以直接获得,但是WTL7.1只能支持Windows2000及XP以上的系统,对于开发商业产品,使不合适的。

抱歉!评论已关闭.