问题描述
我有 4 个按钮没有直接应用到我的程序中。我想隐藏或删除它们。我已经搜索过它们的名称……它们看起来像导航、左/右箭头、下一个/上一个。我似乎无法找到它们的名字。我查看了 Microsoft 网站,查看 CTabView
成员似乎并没有跳出来说“嘿,这些就是你要找的东西”....
我希望这是一项相对容易的任务。有人知道如何“关闭它们”吗?
谢谢。
更新:
通过将新函数移到 OutputWnd.h
中的现有函数之前,我能够解决提到的 C4430,并且解决了箭头现在消失的输出窗格区域中的问题。我忘了提及的是,我有另一种机制 AddView
,它创建 2 个运行时类并将它们放入选项卡式文档视图中。这一次,我一直在使用 GetTabControl()
使它变得漂亮,但这也遇到了同样的问题,现在出现了滚动箭头。
这是创建 2 个新选项卡的代码:
Above this code is the constructor/destructor/headers,nothing else.
IMPLEMENT_DYNCREATE(CTrackView,CTabView)
void CTrackView::OnInitialUpdate()
{
// add document views
AddView(RUNTIME_CLASS(CTrainView),AfxStringID(IDS_TRAIN));
AddView(RUNTIME_CLASS(CStationView),AfxStringID(IDS_STATION));
GetTabControl().EnableTabSwap(TRUE);
GetTabControl().SetLocation(CMFCBaseTabCtrl::Location::LOCATION_BottOM);
GetTabControl().ModifyTabStyle(CMFCTabCtrl::STYLE_3D);
GetTabControl().SetActiveTabBoldFont(TRUE);
CTabView::OnInitialUpdate();
}
我曾尝试注释掉 CTabView::OnInitialUpdate();
,但我知道它不会也不会影响箭头。我对控件进行了一些搜索,但没有看到任何删除箭头的示例,我假设另一个覆盖是有序的。我将按照另一个示例中显示的方法进行操作,我想看看它是否可以像显示的那样修复。
这些是否共享相同的 CMFCTabCtrl
机制还是有什么不同?
更新 2:
MFC 向导创建的 OutputWnd.cpp 有同样的问题,因为我已经修改了选项卡控件,我找不到正确的指针来解决 &tabCtrl()
是未知标识符的问题。编辑:这就是问题所在,我找到了修复程序,请参阅更新 3 以查看解决方案。
int COutputWnd::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CDockablePane::OnCreate(lpCreateStruct) == -1)
return -1;
CRect rectDummy;
rectDummy.SetRectEmpty();
// Create User Define tab style:
int UserTabStyle = AfxGetApp()->GetProfileInt(_T("Settings"),_T("UserTabStyle"),0); //Get value from registry
// If the key doesn't exist,UserTableStyle will be 0 or FALSE;
if (UserTabStyle != FALSE && UserTabStyle <= 8) { // User selected tab style type
int EnumUserTabStyle = UserTabStyle - 1; // Fix enum if key doesn't exist.
if (!m_wndTabs.Create(static_cast<CMFCTabCtrl::Style>(EnumUserTabStyle),rectDummy,this,1))
{
TRACE0("Failed to create output tab window\n");
return -1; // fail to create
}
}
else { // Default tabs style if Reg key does not exist i.e. new install/program reset
if (!m_wndTabs.Create(CMFCTabCtrl::STYLE_FLAT,1))
{
TRACE0("Failed to create output tab window\n");
return -1; // fail to create
}
}
// Nicely hack to access protected member
class CMFCTabCtrlEx : public CMFCTabCtrl
{
public:
void SetdisableScroll() { m_bScroll = FALSE; }
};
// One-Liner to disable navigation control
((CMFCTabCtrlEx*)&tabCtrl())->SetdisableScroll();
// Create output panes:
const DWORD dwStyle = LBS_NOINTEGRALHEIGHT | WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL | WS_BORDER;
if (!m_wndOutputBuild.Create(dwStyle,&m_wndTabs,2) ||
!m_wndOutputDebug.Create(dwStyle,3))
{
TRACE0("Failed to create output windows\n");
return -1; // fail to create
}
UpdateFonts();
CString strTabName;
BOOL bNameValid;
//Attach list windows to tab:
bNameValid = strTabName.LoadString(IDS_STATUS_TAB);
ASSERT(bNameValid);
m_wndTabs.AddTab(&m_wndOutputBuild,strTabName,(UINT)0);
bNameValid = strTabName.LoadString(IDS_DEBUG_TAB);
ASSERT(bNameValid);
m_wndTabs.AddTab(&m_wndOutputDebug,(UINT)1);
int EnableDebugTab = AfxGetApp()->GetProfileInt(_T("Settings"),_T("EnableDebugTab"),0); //Get value from registry
if (EnableDebugTab == FALSE)
{
OnOutputtabsDebug(); //Check to see if it should be enabled
}
return 0;
}
更新 3:
我找到了未知标识符的答案,我试图调用一个不存在的函数。像这样改革,它按预期工作:
// Nicely hack to access protected member
class CMFCTabCtrlEx : public CMFCTabCtrl
{
public:
void SetdisableScroll() { m_bScroll = FALSE; }
};
// One-Liner to disable navigation control
((CMFCTabCtrlEx*)&m_wndTabs)->SetdisableScroll();
解决方法
没有公开的、记录在案的 API 来隐藏这些导航按钮。但是,深入研究 MFC 来源发现:
-
箭头是
CMFCTabButton m_btnScrollLeft,m_btnScrollRight,m_btnScrollFirst,m_btnScrollLast;
中的CMFCTabCtrl
; -
当
CMFCTabCtrl::OnCreate
为CMFCTabCtrl::m_bScroll
时,它们在TRUE
中创建; -
CMFCTabCtrl::m_bScroll
在TRUE
中设置为CMFCTabCtrl::Create
如果任何样式STYLE_FLAT
,STYLE_FLAT_SHARED_HORZ_SCROLL
,STYLE_3D_SCROLLED
,{{ 1}}、STYLE_3D_ONENOTE
或STYLE_3D_VS2005
被使用。
有了这种洞察力,可以通过多种方式关闭导航按钮。
-
根据本书:创建
STYLE_3D_ROUNDED_SCROLL
时没有启用这些按钮的任何样式,只留下CMFCTabCtrl
可用。这确实失去了导航按钮,但选项卡控件的样式也变得不同,选项卡本身显示为小矩形而不是梯形。 -
未记录:在创建标签窗口之前覆盖
STYLE_3D
并将其设置为CMFCTabCtrl::m_bScroll
。例如,派生类FALSE
以清除CMyTabCtrl
。m_bScroll
然后将控件实例化为
class CMyTabCtrl : public CMFCTabCtrl { protected: BOOL PreCreateWindow(CREATESTRUCT &cs) override { m_bScroll = FALSE; return CMFCTabCtrl::PreCreateWindow(cs); } };
而不是CMyTabCtrl m_wndTabs;
。
我记得我过去用 One-liner 解决了这个问题。
这是直接访问 CTrackView::::OnCreate() 函数中的 m_bScroll 变量的解决方案(黑客)。正如 dxiv 解释的那样,OnInitialUpdate() 来得太晚了。
int void CTrackView::::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
if (CTabView::OnCreate(lpCreateStruct) == -1)
return -1;
// add document views
AddView(RUNTIME_CLASS(CTrainView),AfxStringID(IDS_TRAIN));
AddView(RUNTIME_CLASS(CStationView),AfxStringID(IDS_STATION));
:
:
// Nicely hack to access protected member
class CMFCTabCtrlEx : public CMFCTabCtrl
{
public:
void SetDisableScroll() { m_bScroll = FALSE; }
};
// One-Liner to Disable navigation control
((CMFCTabCtrlEx*)&GetTabControl())->SetDisableScroll();
}