MSN / QQ 中的CRichEditCtrl (一) —— 超链接

类别:编程语言 点击:0 评论:0 推荐:

首先对标题说明一下,在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