响应AutoComplete下拉列表的选择事件

 

关键字:SHAutoComplete

 

1SHAutoComplete简介

Shlwapi.dll是微软提供的一个轻量级外壳工具函数库(Shell Lightweight Utility Functions),它提供了一些比较常用的函数,用以处理调色板、路径(如《Secrets in ShlWapi.Dll中提到的PathCompactPath函数)、注册表、字符串等。从5.0版本(随Internet Explorer 5推出)开始,shlwapi.dll还提供了一个函数SHAutoComplete,它使得编辑框控件(如EditComboBox)具有被称为“自动完成”的功能,即当用户在编辑框中输入的时候自动弹出一个非激活的窗口,为用户输入提供建议。如我们在Internet Explorer的地址栏输入“google”,如果系统记录了以前输入过“www.google.com”,则地址栏下方会显示出建议的网址。

 

AutoComplete

 

这样贴心的功能自然为IE 5赢得市场提供了帮助,而对开发人员来说,如果能够毫不费力地为自己的程序添加这样的功能则是再好不过。SHAutoComplete就是最直接而简单的选择。下面的代码演示了如何调用SHAutoComplete函数为某个Edit控件添加自动完成的功能:

 

typedef HRESULT (CALLBACK* LPFNDLLFUNC)(HWND ,DWORD);

HINSTANCE hIns = LoadLibrary(_T("shlwapi.dll"));

if( hIns != NULL )

{

LPFNDLLFUNC lpfnDllFunc = (LPFNDLLFUNC)GetProcAddress(hIns, "SHAutoComplete");

if( lpfnDllFunc != NULL )

{

lpfnDllFunc(m_wndAddressBar.m_hWnd, SHACF_AUTOAPPEND_FORCE_ON |

SHACF_AUTOSUGGEST_FORCE_ON | SHACF_URLALL);

}

FreeLibrary(hIns);

}

 

从微软的习惯来说,这种对用户来说极为有用的功能不会只提供这样一个简单的函数就完事。事实上,SHAutoComplete只完成系统默认实现的功能,开发人员如果需要自定义以提供更强功能的话则可通过实现IAutoComplete接口以及IAutoComplete2接口来完成(在Windows XP中还提供了IAutoCompleteDropDown接口用以控制上图中那个下拉列表窗口的状态)。

 

2、问题的提出

我们知道,当用户在ComboBox控件的下拉框中用鼠标点击某个列表项或在列表项上按回车键时,ComboBox的父窗口会通过WM_COMMAND消息接收到一个CBN_SELENDOK通知,从而可以知道用户选择了那一个列表项并进行处理,在VC++中类似这样:

 

BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd)

ON_CBN_SELENDOK(ID_ADDRESSBOX, &CMainFrame::OnSelAddress)

......

END_MESSAGE_MAP()

 

void CMainFrame::OnSelAddress()

{

CString str = m_wndAddressBar.GetLBAddress();

...... //处理用户的选择

}

 

自然而然地,对于SHAutoComplete提供的下拉框,我们也希望有这样的通知,但MSDN中似乎并没有相关的文档。

 

3SPY++

我们马上想到用Spy++来跟踪消息。以IE的地址栏为例,当我们在SHAutoComplete的下拉框中点击(按回车键有同样的效果)“http://www.google.com 这个列表项时,地址栏中的Edit的消息踪迹如下所示:

 

<00589> 001B0BF4 S WM_SETTEXT lpsz:0013D074 ("http://www.google.com")

......

<00606> 001B0BF4 R EM_SETSEL

<00607> 001B0BF4 S message:0xC2B6 [Registered:"AC_ItemActivate"] wParam:00000000 lParam:0013D494

<00608> 001B0BF4 R message:0xC2B6 [Registered:"AC_ItemActivate"] lResult:00000000

<00609> 001B0BF4 S WM_KEYDOWN nVirtKey:VK_RETURN cRepeat:0 ScanCode:00 fExtended:0 fAltDown:0 fRepeat:0 fUp:0

......

 

包含Registered:"AC_ItemActivate"的这一行立刻引起了我们的注意。这是一个用RegisterWindowMessage函数在运行时向系统注册的消息(参见Windows通知栏图标高级编程概述具有自动恢复功能的通知栏图标控件),从其字面意思来看正是我们想要的。我们注意到Edit在接收到该消息之前还接收到了WM_SETTEXT消息,即此刻Edit中的文字已经被SHAutoComplete的后台工作设置为我们所选择的列表项了,因此对于该消息的wParamlParam的意义我们也可以不去深究。

 

4、问题解决

下面是一个简单的解决方案:

 

// CACEditSubclassWnd

 

class CACEditSubclassWnd : public CWnd

{

DECLARE_DYNAMIC(CACEditSubclassWnd)

static const UINT m_nAcItemActivateMsg;

//用来保存向系统注册的消息

public:

CACEditSubclassWnd(){};

virtual ~CACEditSubclassWnd(){};

 

protected:

LRESULT OnAcItemActivate(WPARAM wParam, LPARAM lParam);

DECLARE_MESSAGE_MAP()

};

 

// CACEditSubclassWnd

// 向系统注册我们需要的消息

const UINT CACEditSubclassWnd::m_nAcItemActivateMsg = ::RegisterWindowMessage(_T("AC_ItemActivate"));

 

IMPLEMENT_DYNAMIC(CACEditSubclassWnd, CWnd)

 

BEGIN_MESSAGE_MAP(CACEditSubclassWnd, CWnd)

ON_REGISTERED_MESSAGE(CACEditSubclassWnd::m_nAcItemActivateMsg, &CACEditSubclassWnd::OnAcItemActivate)

END_MESSAGE_MAP()

 

// CACEditSubclassWnd message handlers

LRESULT CACEditSubclassWnd::OnAcItemActivate(WPARAM wParam, LPARAM lParam)

{

AfxGetMainWnd()->SendMessage(WM_COMMAND, MAKEWPARAM(LOWORD(ACCBN_SELENDOK), 0x0), 0);

//ACCBN_SELENDOK是我们自定义的通知

return 0L;

}

 

假设CUrlAddressCombo是一个地址栏类,则为其声明一个CACEditSubclassWnd类型的成员,并在适当的位置(如Init成员函数中)子类化Edit控件。

 

class CUrlAddressCombo : public CComboBoxEx

{

private:

CACEditSubclassWnd m_ACEditSubclassWnd;

......

}

void CUrlAddressCombo::Init(void)

{

m_ACEditSubclassWnd.SubclassWindow( GetEditCtrl()->m_hWnd );

}

而在CMainFrame中可以这样实现:

 

BEGIN_MESSAGE_MAP(CMainFrame, CMDIFrameWnd)

ON_COMMAND(ACCBN_SELENDOK, &CMainFrame::OnACSelEndOk)

......

END_MESSAGE_MAP()

 

void CMainFrame::OnACSelEndOk()

{

CString strAddr;

m_wndAddressBar.GetEditCtrl()->GetWindowText(strAddr);

...... //处理用户的选择

}

 

非常简单,不是吗?

5、参考文献

MSDN: SHAutoComplete Function

引用地址:《响应AutoComplete下拉列表的选择事件

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值