其他语言

本类阅读TOP10

·基于Solaris 开发环境的整体构思
·使用AutoMake轻松生成Makefile
·BCB数据库图像保存技术
·GNU中的Makefile
·射频芯片nRF401天线设计的分析
·iframe 的自适应高度
·BCB之Socket通信
·软件企业如何实施CMM
·入门系列--OpenGL最简单的入门
·WIN95中日志钩子(JournalRecord Hook)的使用

分类导航
VC语言Delphi
VB语言ASP
PerlJava
Script数据库
其他语言游戏开发
文件格式网站制作
软件工程.NET开发
MSN / QQ 中的CRichEditCtrl (一) —— 超链接

作者:未知 来源:月光软件站 加入时间:2005-2-28 月光软件站

首先对标题说明一下,在MSN中,聊天的窗口可能是一个自定义的类。大家用Spy ++可以看看。
对与自定义窗口,可以使用CreateWindow, SetWindowLong或者是SubclassWindow实现,不过这不
是我现在讨论的话题。

好, 先看看效果再说:

我实现的主要就是:(一)字体格式; (二)超链接;(三)背景图片;(四)动画表情


关于HyperLink的是实现,我在www.codeguru.com找到了Madhu B Nair 的 Creating the OutLookRichEdit Control
他的实现方式就是自绘,自己维护,效果也很不错

就在我觉得实现OLE的插入有难度时,就想过了直接自己绘制,自己维护,
但是觉得做法不是很正统,对不起Microsoft给我留的IRichEditOle接口,就没有使用这种解决方法。
我们大致讲一下他的实现方式:

定义一个struct描述每个hyperlink对象:
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


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}
呵呵,就这么多了。先喝杯咖啡庆祝以下,下次说背景啊。



相关文章

相关软件