首先对标题说明一下,在MSN中,聊天的窗口可能是一个自定义的类。大家用Spy ++可以看看。
对与自定义窗口,可以使用CreateWindow, SetWindowLong或者是SubclassWindow实现,不过这不
是我现在讨论的话题。
好, 先看看效果再说:
我实现的主要就是:(一)字体格式; (二)超链接;(三)背景图片;(四)动画表情
关于HyperLink的是实现,我在www.codeguru.com找到了Madhu B Nair 的 Creating the OutLookRichEdit Control
他的实现方式就是自绘,自己维护,效果也很不错
就在我觉得实现OLE的插入有难度时,就想过了直接自己绘制,自己维护,
但是觉得做法不是很正统,对不起Microsoft给我留的IRichEditOle接口,就没有使用这种解决方法。
我们大致讲一下他的实现方式:
定义一个struct描述每个hyperlink对象:
function ToggleSourceCodeRegion(regionNumber)
{
var divRegion = document.getElementById('region' + regionNumber);
var divRegionBlock = document.getElementById('regionBlock' + regionNumber);
if (divRegion.style.display == 'inline')
{
divRegion.style.display = 'none';
divRegionBlock.style.display = 'inline';
}
else
{
divRegion.style.display = 'inline';
divRegionBlock.style.display = 'none';
}
}
1typedef struct _HYPERLINKINFO
2{
3 CRect rectDimension;
4 CString csLinkText;
5 UINT unLinkDlgID;
6 inline operator = (struct _HYPERLINKINFO linkInfo)
{ rectDimension = linkInfo.rectDimension;csLinkText = linkInfo.csLinkText;unLinkDlgID
= linkInfo.unLinkDlgID;}
7}HYPERLINKINFO,*LPHYPERLINKINFO;
这里有一个UINT unLinkDlgID,是来描述点击这个link后弹出的dialog的。
CRect rectDimension;用来保存这个link的rect,以检查鼠标是否在其中,以显示鼠标光标;
对于多个links,他使用了一个CList来维护(如果是我,我会使用std::list,呵呵):
1CList <HYPERLINKINFO,HYPERLINKINFO&> m_lsLinks;
其中很关键的就是判断鼠标是不是在一个link的rect内。
这种实现的确很不错,但是我么要说的是另一个更现代的实现方式,实现起来也简单得多,关键的就是有个Style => ENM_LINK
这个style在RichEdit20中才有,所以我们应该使用RichEdit20A!
我们维护这样一个类:
1class _AFX_RICHEDITEX_STATE
2{
3public:
4 _AFX_RICHEDITEX_STATE();
5 virtual ~_AFX_RICHEDITEX_STATE();
6 HINSTANCE m_hInstRichEdit20 ;
7};
实现代码就一点,就是用这个类来Load Richedit20
function ToggleSourceCodeRegion(regionNumber)
{
var divRegion = document.getElementById('region' + regionNumber);
var divRegionBlock = document.getElementById('regionBlock' + regionNumber);
if (divRegion.style.display == 'inline')
{
divRegion.style.display = 'none';
divRegionBlock.style.display = 'inline';
}
else
{
divRegion.style.display = 'inline';
divRegionBlock.style.display = 'none';
}
}
1_AFX_RICHEDITEX_STATE::_AFX_RICHEDITEX_STATE()
2{
3 m_hInstRichEdit20 = NULL ;
4}
5
6
7
8_AFX_RICHEDITEX_STATE::~_AFX_RICHEDITEX_STATE()
9{
10 if( m_hInstRichEdit20 != NULL )
11 {
12 ::FreeLibrary( m_hInstRichEdit20 ) ;
13 }
14}
15
16
17
18_AFX_RICHEDITEX_STATE _afxRichEditStateEx ;
19
20
21
22BOOL PASCAL AfxInitRichEditEx()
23{
24 if( ! ::AfxInitRichEdit() )
25 {
26 return FALSE ;
27 }
28 _AFX_RICHEDITEX_STATE* l_pState = &_afxRichEditStateEx ;
29 if( l_pState->m_hInstRichEdit20 == NULL )
30 {
31 l_pState->m_hInstRichEdit20 = LoadLibraryA("RICHED20.DLL") ;
32 }
33 return l_pState->m_hInstRichEdit20 != NULL ;
34}
然后在我们创建RichEdit的时候,就使用 RichEdit20A 作为ClassName;
1CreateEx(dwExStyle, _T( "RichEdit20A" ), NULL, dwStyle, rect, pParentWnd, nID, NULL );
这样一来,你就使用先进的RichEdit20A了, 可以简单的实现你的要的功能了。
先给你的RichEditCtrl设置EN_LINK Style, 代码量也是少得可怜:
1unsigned mask = ::SendMessage(m_hWnd, EM_GETEVENTMASK, 0, 0);
2::SendMessage
(m_hWnd, EM_SETEVENTMASK, 0, mask | ENM_LINK | ENM_MOUSEEVENTS | ENM_SCROLLEVENTS | ENM_KEYEVENTS);
3::SendMessage(m_hWnd, EM_AUTOURLDETECT, true, 0);
如果你觉得使用CRichEditCtrl的Member Function更方便的话,当然可以使用GetEventMak和SetEventMask,同样方便。
是不是这样就完了呢?呵呵,并不象你想象中的那么完美,而是比你想象中的更完美呢!为了扩展的需要,RichEdit20并没有
定义为,当你点击links的时候用浏览器打开这个页面,而是让你处理这个点击,你就可以任意的扩展,做到一切皆有可能!
ON_NOTIFY(EN_LINK, IDC_SENDMSG, OnRichEditExLink )
知道在哪加吧?呵呵。这个IDC_SENGMSG就是RichEdit的Resource ID, OnRichEditExLink就是处理这个消息的函数咯,
然后看最后的代码,今天的任务就完成了!
1void CMsgerDlg::OnRichEditExLink( NMHDR* in_pNotifyHeader, LRESULT* out_pResult )
2{
3 ENLINK* l_pENLink = ( ENLINK* )in_pNotifyHeader ;
4 *out_pResult = 0 ;
5 switch( l_pENLink->msg )
6 {
7 default:{}
8 break ;
9
10 case WM_LBUTTONDOWN:
11 {
12 CString l_URL ;
13 CHARRANGE l_CharRange ;
14 CExtRichEdit *m_TempEdit;
15 m_TempEdit = (CExtRichEdit*)CExtRichEdit::FromHandle(l_pENLink->nmhdr.hwndFrom);
16 m_TempEdit->GetSel( l_CharRange ) ;
17 m_TempEdit->SetSel( l_pENLink->chrg ) ;
18 l_URL = m_TempEdit->GetSelText() ;
19 m_TempEdit->SetSel( l_CharRange ) ;
20 CWaitCursor l_WaitCursor ;
21 ShellExecute
( this->GetSafeHwnd(), _T( "open" ), l_URL, NULL, NULL, SW_SHOWNORMAL ) ;
22 *out_pResult = 1 ;
23 }
24 break ;
25
26 case WM_LBUTTONUP:
27 {
28 *out_pResult = 1 ;
29 }
30 break ;
31 }
32}
呵呵,就这么多了。先喝杯咖啡庆祝以下,下次说背景啊。
本文地址:http://com.8s8s.com/it/it26206.htm