王朝网络
分享
 
 
 

Internet Explorer 编程简述(十二)正确地设置和转移焦点

王朝other·作者佚名  2006-06-12
宽屏版  字体: |||超大  

关键字:焦点,Focus,加速键,Accelerator,OLEIVERB_UIACTIVATE,IHTMLWindow2,IHTMLDocument4

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应用程序都或多或少犯了一些错误:

完全没有加速键。

这在国产信息系统中尤为常见。设计较差的信息系统常常会出现一个窗口拥有数十个输入框的情况,如果为每个编辑框都提供一个加速键的话,问题就出来了。字母键只有26个,就算把数字键也用上,也难免不能满足要求,所以很多信息系统干脆就不要加速键。

摆设用的加速键。

一些应用软件甚至不懂得加速键的意义,只知道依样画葫地在输入框的旁边用标签说明加速键,但仅此而已,用户根本无法通过Alt+加速键转移焦点到输入框。

错误地(或不能)转移焦点

对于基于对话框的应用程序来说,常犯的错误是用户按下Tab键时,焦点出乎用户意料地在输入框之间乱窜。而在上图这样的例子中,常犯的错误则是不能通过Tab转移焦点,或者按Tab能转移焦点但按Shift+Tab不能朝反方向转移焦点。

对嵌入的ActiveX控件缺乏处理

对于嵌入的ActiveX控件,尤其是WebBrowser控件来说,焦点的处理就更为麻烦了(这本是基于WebBrowser的浏览器编程的难题之一)。常见的浏览器要么不处理常规窗口与WebBrowser控件之间的焦点传递(Maxthon、Gosurf只支持在输入框之间传递焦点);要么处理不完整,焦点一旦从某个输入框转移到WebBrowser控件就再也回不来(如GreenBrowser);更有的根本就不处理任何焦点的传递(如世界之窗浏览器)。

按照本系列文章的惯例,本文讨论的目的将是提供一个完整(未必完美)的解决方案——:一,焦点在嵌入ReBar的各个输入框之间传递;二,焦点在普通Windows窗口(输入框)与WebBrowser控件之间传递。

3、设定目标

下图说明了我们希望实现的正常的焦点转移行为:

从工具条上的任何一个输入框出发,按Tab将焦点转移到下一个输入框,按Shift+Tab将焦点转移到上一个输入框

如果焦点所在输入框是工具条上的最后一个输入框,按Tab将焦点转移到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,按Shif+Tab将焦点转移到工具条的最后输入框

以下图为例,“Google大全”为WebBrowser当前获得焦点的Html Element,举例如下:

例1:假设当前焦点位于地址输入框,按下Tab键不松开,焦点转移的顺序应是:“地址栏”,“搜索栏”,“Google大全”……“将Google设为首页”,“地址栏”,“搜索栏”,“个性化主页”,“搜索记录”……

例2:假设当前焦点位于地址输入框,且WebBrowser控件没有活动的获得焦点的Html Element,按下Tab键不松开,焦点转移的顺序应是:“地址栏”,“搜索栏”,“个性化主页”,“搜索记录”……“将Google设为首页”,“地址栏”,……

例3:假设当前焦点位于“搜索记录”,按下Shift+Tab键不松开,焦点转移的顺序应是:“搜索记录”,“个性化主页”,“搜索栏”,“地址栏”,“将Google设为首页”……“搜索记录”……

4、工具条输入框之间的焦点转移

为实现统一的处理,我们从CDialogBar派生一个CDialogBarEx类,由该类处理Tab/Shift Tab按键,而输入框(如EditBox,ComboBox等)则放在CDialogBarEx的派生类(如CUrlAddressBar、CSearchBar等)中,这样输入框就可以专注于其它的功能。示例代码如下:

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_wndReBar是一个CReBarEx,可从CReBar派生

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接口的应用程序来说,按键处理就非常简单了。

//在此处理将焦点从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的,如下:

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》

引用地址: Internet Explorer 编程简述(十二)正确地设置和转移焦点

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
>>返回首页<<
推荐阅读
 
 
频道精选
 
静静地坐在废墟上,四周的荒凉一望无际,忽然觉得,凄凉也很美
© 2005- 王朝网络 版权所有