[英]CRichEditCtrl selects all text when it gets focus
我有一个菜单和CTabCtrl的对话框。 CTabCtrl有一个选项卡,其中包含一个CDialog。 反过来,它包含一些静态文本和一个CRichEditCtrl。 窗口获得和失去焦点没有特别的问题。
我添加了第二个相同的选项卡,现在每次更改选项卡时,显然选择了CRichEditCtrl中的所有文本。 它以反转颜色方案显示,如果您按下某个键,则会替换所有文本。
标志ECO_NOHIDESEL的说明(强调我的):
否定编辑控件的默认行为。 当控件丢失输入焦点时,默认行为会隐藏选择,并在控件接收输入焦点时显示选择。 如果指定ECO_NOHIDESEL,则即使控件没有焦点,也会反转所选文本。
“显示选择”对我来说听起来像“显示这个控件最后一次选择的焦点”,这不是正在发生的事情。 通常在焦点丢失之前没有选择任何内容,但是如果我尝试留下选择,返回到另一个选项卡并返回,则选择整个文本。
可以预防这种选择吗?
void EditorDialog :: OnTabSelChange(NMHDR * phdr,LRESULT * pResult){
CTabCtrl* ptab = (CTabCtrl*) GetDlgItem( IDC_TAB );
int iPageActive = ptab->GetCurSel();
if ( iPageActive >= appage.N() ) {
AKS( AKSWarn, "got tab change to tab %d when I only have %d ppages", iPageActive, appage.N() );
return;
}
ppageActive = appage[ iPageActive ];
SetActivePagePos();
SCWinUtilSetWindowTextVA( this, "Editor: %s", ppageActive->pszFileName );
}
void EditorDialog::SetActivePagePos() {
// STEP 1: Make the proper tab page visible.
for ( int i = 0; i < appage.N(); i++ )
appage[i]->ShowWindow( SW_HIDE );
ppageActive->ShowWindow( SW_SHOW );
// STEP 2: Make the new tab page the right size and position.
CTabCtrl* ptab = (CTabCtrl*) GetDlgItem( IDC_TAB );
CRect rectTab, rectItem;
ptab->GetClientRect( &rectTab );
ptab->GetItemRect( 0, &rectItem );
int iPageX = rectItem.left + 2;
int iPageY = rectItem.bottom + 4;
int iPageW = rectTab.right - 2 - iPageX;
int iPageH = rectTab.bottom - 2 - iPageY;
ppageActive->SetWindowPos( &wndTop, iPageX, iPageY, iPageW, iPageH, SWP_SHOWWINDOW | SWP_NOZORDER );
// STEP 3: Give the window focus and let it know to redraw.
ppageActive->SetFocus();
// When the tab changes the entire content of the RichEdit is selected for some reason.
// As a workaround I manually clear the selection.
CRichEditCtrl* prich = (CRichEditCtrl*) ppageActive->GetDlgItem( IDC_PATCH );
prich->SetSel(-1,-1);
// Redrawing just the prich, or the ppageActive, or the ptab, doesn't
// cause the RichEdit to redraw correctly, but Redrawing the entire dialog does.
RedrawWindow();
}
Edit和RichEdit控件的默认行为是在获得焦点时选择整个内容。 ES_NOHIDESEL
不会修改此行为,只是指示控件始终显示其选择,即使它没有输入焦点。
要更改RichEdit控件的默认行为以保留其选择,您必须从中派生并提供自定义OnGetDlgCode
实现:
UINT RichEditSelectionPreserving::OnGetDlgCode() {
// Call the base class implementation
UINT code = CRichEditCtrl::OnGetDlgCode();
// And mask out the undesired feature
code = code & ~DLGC_HASSETSEL;
return code;
}
“编辑”和“Rich Edit”控件的默认行为是在控件没有输入焦点时使选择不可见 ,并且仅在控件具有焦点时使其可见。 但是,选择不会改变 。 ES_NOHIDESEL
样式会覆盖此默认行为,并使选择始终显示在控件中,无论它是否具有焦点。 您之前确实看到过这种行为:它是Microsoft Word和Visual Studio所做的。
因此,您对SDK文档的理解是完全正确的。 不幸的是,Rich Edit控件的行为还有另一个方面。 每当对话框中托管的“编辑”或“Rich Edit”控件收到焦点时,它都会自动选择其所有文本,从而消除过程中当前的插入符号位置。 ES_NOHIDESEL
对此行为没有任何影响; 它只是在控件未聚焦时改变选择是否可见 。 您当然可以通过执行IInspectable建议来覆盖此select-all-on-focus行为, 并对控件进行子类化以自定义其对WM_GETDLGCODE
消息的处理 。
但是有一个更简单的解决方案。 与ES_NOHIDESEL
, 您希望在创建时为控件设置ES_SAVESEL
样式 。 虽然您可以在资源编辑器中设置ES_NOHIDESEL
(“无隐藏选择”),但ES_SAVESEL
没有等效属性。 您可以手动将其添加到RC文件,但无法保证在Visual Studio重新生成该文件时不会将其删除。
或者,您可以向Rich Edit控件发送指定ECO_SAVESEL
选项的EM_SETOPTIONS
消息。 在MFC中, SetOptions
成员函数包装此消息的发送。 例如,在OnInitDialog
函数中,您可能具有以下内容:
m_myRichEditCtrl.SetOptions(ECOOP_OR, ECO_SAVESEL); // maintain selection across focus events
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.