本文转载自我2006年在csdn发布的博客
1. 概述
对于99%
有UI
的Windows
应用程序来说,键盘操作都是不可或缺而又容易被人们遗忘的一环。如果对Windows
组件作一次逐个的测试,我们会发现Microsoft
提供的任何一个Windows
组件都通过键盘实现完全的控制(“计算器”比较特殊,它是一个按钮很多且每个按钮都不能获得焦点的程序,但在帮助文档中我们仍然可以找到为每个按钮设置的快捷键),这对于一个专业的Windows
应用程序或软件来说非常重要。换句话说,就算没有鼠标用户也不应该束手无策,用户应该可以通过键盘操作完成其希望的功能。焦点的转移无疑是键盘操作的一个重要方面,在浏览器编程中尤其如此。
2. 焦点的基本概念
一般说来,在Windows
中用户通过键盘转移焦点(Focus
)有两个方法:第一,对于输入框附近有标签提示的情况,按住Alt
+某个预设的字母(Accelerator
,加速键)将焦点快速转移到输入框。如下图所示,按下“Alt+D
”,焦点应转移到地址输入框;按下“Alt+G
”,焦点应转移到搜索框(本文对此不做讨论)。第二,按住Tab
键,焦点转移到由应用程序控制的下一个可获得焦点的窗口;按下Shift+Tab
,焦点转移到上一个可获得焦点的窗口。如下图所示,如果地址输入框是当前获得焦点的窗口,则按下Tab时,焦点应转移到搜索框,再按下Shift+Tab
,焦点应回到地址输入框。
焦点的设置和转移对于用户体验(Experience
)来说是细微体贴而又重要的设计,但不幸的是不少Windows
应用程序都或多或少犯了一些错误:
1) 完全没有加速键。
这在国产信息系统中尤为常见。设计较差的信息系统常常会出现一个窗口拥有数十个输入框的情况,如果为每个编辑框都提供一个加速键的话,问题就出来了。字母键只有26
个,就算把数字键也用上,也难免不能满足要求,所以很多信息系统干脆就不要加速键。
2) 摆设用的加速键。
一些应用软件甚至不懂得加速键的意义,只知道依样画葫地在输入框的旁边用标签说明加速键,但仅此而已,用户根本无法通过Alt
+加速键转移焦点到输入框。
3) 错误地(或不能)转移焦点
对于基于对话框的应用程序来说,常犯的错误是用户按下Tab
键时,焦点出乎用户意料地在输入框之间乱窜。而在上图这样的例子中,常犯的错误则是不能通过Tab
转移焦点,或者按Tab
能转移焦点但按Shift+Tab
不能朝反方向转移焦点。
4) 对嵌入的ActiveX
控件缺乏处理
对于嵌入的ActiveX
控件,尤其是WebBrowser
控件来说,焦点的处理就更为麻烦了(这本是基于WebBrowser
的浏览器编程的难题之一)。常见的浏览器要么不处理常规窗口与WebBrowser
控件之间的焦点传递(Maxthon、Gosurf
只支持在输入框之间传递焦点);要么处理不完整,焦点一旦从某个输入框转移到WebBrowser
控件就再也回不来(如GreenBrowser);更有的根本就不处理任何焦点的传递(如世界之窗浏览器)。
按照本系列文章的惯例,本文讨论的目的将是提供一个完整(未必完美)的解决方案:
一,焦点在嵌入ReBar
的各个输入框之间传递
二,焦点在普通Windows
窗口(输入框)与WebBrowser
控件之间传递。
3. 设定目标
下图说明了我们希望实现的正常的焦点转移行为:
-
从工具条上的任何一个输入框出发,按
Tab
将焦点转移到下一个输入框,按Shift+Tab
将焦点转移到上一个输入框 -
如果焦点所在输入框是工具条上的最后一个输入框,按
Ta
b将焦点转移到WebBrowser
控件当前的活动Html Element
(上一次获得焦点的Element
) -
如果焦点所在输入框是工具条上的第一个输入框,按
Shift+Tab
将焦点转移到WebBrowser
控件当前活动Html Element
-
对于上面两种情况,若
WebBrowser
控件没有当前活动的可获得焦点Html Element
,则焦点应从输入框转移到WebBrowser
控件的第一个或最后一个可获得焦点的Html Element
-
如果焦点当前位于
WebBrowser
控件中,按Tab
将焦点转移到下一个Html Element
,按Shift+Tab
将焦点转移到上一个Html Element
-
如果焦点当前位于
WebBrowser
控件中,且当前的活动Html Element
是最后一个可获得焦点的Html Element
,按Tab将焦点转移到工具条的第一个输入框 -
如果焦点当前位于
WebBrowser
控件中,且当前的活动Html Element
是第一个可获得焦点的Html Element
,按Shift+Tab
将焦点转移到工具条的最后输入框
以下图为例,“Google
大全”为WebBrowser
当前获得焦点的Html Element
,举例如下:
-
例1:假设当前焦点位于地址输入框,按下
Tab
键不松开,焦点转移的顺序应是:“地址栏”,“搜索栏”,“Google
大全”……“将Google
设为首页”,“地址栏”,“搜索栏”,“个性化主页”,“搜索记录”…… -
例2:假设当前焦点位于地址输入框,且
WebBrowser
控件没有活动的获得焦点的Html Element
,按下Tab
键不松开,焦点转移的顺序应是:“地址栏”,“搜索栏”,“个性化主页”,“搜索记录”……“将Google
设为首页”,“地址栏”,…… -
例3:假设当前焦点位于“搜索记录”,按下
Shift+Tab
键不松开,焦点转移的顺序应是:“搜索记录”,“个性化主页”,“搜索栏”,“地址栏”,“将Google
设为首页”……“搜索记录”……
4. 工具条输入框之间的焦点转移
为实现统一的处理,我们从CDialogBar
派生一个CDialogBar
Ex类,由该类处理Tab/Shift Tab
按键,而输入框(如EditBox
,ComboBox
等)则放在CDialogBar
Ex的派生类(如CUrlAddressBar
、CSearchBar
等)中,这样输入框就可以专注于其它的功能。示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
BOOL CDialogBarEx::PreTranslateMessage(MSG* pMsg)
{
if ( ( pMsg->message==WM_KEYDOWN ) )
{
if ( (pMsg->wParam == VK_TAB) )
{
//由MainFrame处理如何转移焦点,按下Shift表示焦点应转移到上一个窗口
g_pMainFrame->SetFocusToNextControl( GetKeyState(VK_SHIFT) >= 0 );
return TRUE;
}
}
......
return CDialogBar::PreTranslateMessage(pMsg);
}
void CMainFrame::SetFocusToNextControl(bool bNext)
{
//m_wnd`ReBar`是一个CReBarEx,可从C`ReBar`派生
if ( !m_wndReBar.SetFocusToNextControl(bNext) )
{
//如果CReBarEx在其子窗口中找不到下(上)一个可以设置焦点的窗口,则把焦点转移到`WebBrowser`
CChildFrame *pChildFrame = (CChildFrame *)MDIGetActive();
if ( pChildFrame && pChildFrame->GetActiveView() )
{
pChildFrame->GetActiveView()->SetFocus();
}
}
}
bool CReBarEx::SetFocusToNextControl(bool bNext)
{
return bNext ? FocusNextControl() : FocusPrevControl();
}
bool CReBarEx::FocusNextControl()
{
REBARBANDINFO rbbi;
rbbi.cbSize = sizeof( rbbi );
rbbi.fMask = RBBIM_CHILD;
//先找到当前获得焦点的Band
UINT nBand;
for ( nBand = 0; nBand < m_rbCtrl.GetBandCount(); nBand++ )
{
VERIFY( m_rbCtrl.GetBandInfo(nBand, &rbbi) );
if ( ::IsChild(rbbi.hwndChild, ::GetFocus()) )
{
break;
}
}
//如果运行到这里,必定能够找到当前获得焦点的Band
ASSERT(nBand < m_rbCtrl.GetBandCount());
for ( nBand = nBand + 1; nBand < m_rbCtrl.GetBandCount(); nBand++ )
{
VERIFY( m_rbCtrl.GetBandInfo(nBand, &rbbi) );
::SetFocus(rbbi.hwndChild);
if ( ::IsChild(rbbi.hwndChild, ::GetFocus()) )
{
//成功找到并设置焦点到下一个窗口
return true;
}
}
//当前获得焦点的窗口已经是ReBarEx中最后一个可获得焦点的窗口
return false;
}
bool CReBarEx::FocusPrevControl()
{
//实现与FocusNextControl类似,此处略去
}
void CReBarEx::OnSetFocus(CWnd* pOldWnd)
{
//如果此时Shift为按下的状态,表示焦点可能是从`WebBrowser`的第一个活动`Html Element`转过来,
//则将焦点转移到最后一个输入框,否则转移到第一个输入框
//SetFocusToLastControl与SetFocusToFirstControl的实现相当简单,略去
return GetKeyState(VK_SHIFT) < 0 ? SetFocusToLastControl() : SetFocusToFirstControl();
}
5. 焦点从WebBrowser
转移到工具条输入框
处理浏览器的按键也曾是嵌入WebBrowser
控件的编程难题之一,Delphi
对WebBrowser
的封装对按键的支持就存在很大问题。在《Programming Internet Explorer》(原链接已失效)》中曾提到的方法是处理MainFrame
的PreTranslateMessage
,并在其中从WebBrowser
的Document
查询得到IOleInPlaceActiveObject
接口,将按键交给IOleInPlaceActiveObject
的TranslateAccelerator
成员区处理。查询MSDN
我们可以知道,IOleInPlaceActiveObject
::TranslateAccelerator
被调用时,MSHTML
引擎会调用IDocHostUIHandler
接口的TranslateAccelerator
方法,从而给开发人员一个接口来处理按键。所以对于实现了IDocHostUIHandler
接口的应用程序来说,按键处理就非常简单了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
//在此处理将焦点从WebBrowser中转移到ReBar上的输入框
HRESULT CMyView::OnTranslateAccelerator(LPMSG lpMsg,const GUID* pguidCmdGroup, DWORD nCmdID)
{
if (lpMsg && lpMsg->message == WM_KEYDOWN && lpMsg->wParam == VK_TAB)
{
LPDISPATCH lpDispatch = GetHtmlDocument();
CComQIPtr<IHTMLDocument2> pHTMLDoc = lpDispatch;
if ( pHTMLDoc )
{
CComQIPtr<IHTMLElement> pElement;
if ( SUCCEEDED(pHTMLDoc->get_activeElement(&pElement)) && !pElement )
{
//没有任何活动的Html Element,把焦点转移到ReBar
g_pMainFrame->m_wndReBar.SetFocus();
//通知`MSHTML`不要再继续处理按键
return S_OK;
}
}
}
return S_FALSE;
}
6. 使WebBrowser获得焦点
使浏览器获得焦点也颇为讲究。我的一篇老文章《TWebBrowser编程简述中》写到有好几种方法可以使WebBrowser
获得焦点:IOleObject::DoVerb(OLEIVERB_UIACTIVATE...)
,IHTMLWindow2::focus()
,IHTMLDocument4::focus()
。而实际上这几种方法是有区别的(内部实现我们并不清楚,也不关心)。
-
IOleObject::DoVerb
能够将焦点设置到WebBrowser
上一次失去焦点时获得焦点的Html Element
上。缺点在于如果WebBrowser
上次失去焦点时没有任何Html Element
获得焦点,则DoVerb并不能保证焦点会转移到WebBrowser
中。 -
IHTMLWindow2::focus
不管三七二十一,将焦点转移到WebBrowser
的开头Html Element
。这显然不是我们想要的。 -
测试的结果,
IHTMLDocument4::focus
似乎能够满足要求:能够记住WebBrowser
上次失去焦点时获得焦点的Html Element
;在WebBrowser
上次失去焦点时没有任何Html Element
获得焦点的情况下,能够焦点转移到开头的Html Element
。但事实上并不理想,假如按住Tab
键不松开,反复调用IHTMLDocument4::focus
多次之后,我们会发现焦点再也到不到WebBrowser
中了。
有没有完美解决的办法呢?答案当然是Positive
的,如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
void CMyView::OnSetFocus(CWnd* pOldWnd)
{
LPDISPATCH lpDisp = GetHtmlDocument();
CComQIPtr<IHTMLDocument2, &IID_IHTMLDocument2> pHTMLDoc(lpDisp);
if ( pHTMLDoc )
{
CComQIPtr<IHTMLElement> pElement;
if ( SUCCEEDED(pHTMLDoc->get_activeElement(&pElement)) && !pElement )
{
//没有任何活动元素,把焦点转移到WebBrowser的开头
CComQIPtr<IHTMLWindow2> pHTMLWnd;
if( SUCCEEDED(pHTMLDoc->get_parentWindow( &pHTMLWnd )) && pHTMLWnd )
{
pHTMLWnd->focus();
return;
}
}
}
//有活动的元素(上一次的焦点),直接将焦点转移过去
//CWnd::SetFocus()会调用IOleObject::DoVerb()正确地设置焦点
m_wndBrowser.SetFocus();
}
7. 总结
至此,我们就算完整地实现了焦点在普通窗口和浏览器之间的传递,任何时候,按住Tab
键不松开,焦点将会在所有可获得焦点的窗口之间循环传递;同样,按住Shift-Tab
不松开,焦点会以反方向传递。而不会出现用户无法将焦点转移到浏览器窗口的情况,或者焦点无法从浏览器窗口转移到输入框的情况。当然,还有比较重要也比较抽象的一点,增强了用户体验,呵呵。
8. 参考资料
《Programming Internet Explorer》(原链接已失效)