- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我使用 TActionMainMenuBar 来显示基于 TActions 的菜单。我通过设置相同的 GroupIndex 对操作进行分组。因此它们可以像 RadioGroup 一样操作,但问题是绘制的是检查而不是单选按钮。
有办法改变吗?
最佳答案
这是我对 TPlatformDefaultStyleActionBars
的修复。
大多数代码只是从标准单位复制而来,除了 TFixedThemedMenuItemStyle.DoDrawMenuCheck
。
请注意,如果您想在 Vista 之前的操作系统上运行软件,还必须覆盖 TXPStyleMenuItem
。
uses
// ... add these units
StdStyleActnCtrls, XPStyleActnCtrls, XPActnCtrls, ImgList, Types, Themes,
StdActnMenus, ThemedActnCtrls, ListActns, UxTheme;
type
TFixedThemedMenuItemStyle = class(TThemedMenuItem)
private
FCheckRect: TRect;
FGutterRect: TRect;
FPaintRect: TRect;
FSubMenuGlyphRect: TRect;
FSeparatorHeight: Integer;
procedure DoDrawMenuCheck;
procedure DoDrawText(DC: HDC; const Text: string; var Rect: TRect; Flags: Longint);
protected
procedure DrawGlyph(const Location: TPoint); override;
public
procedure CalcBounds; override;
end;
TFixedPlatformDefaultStyleActionBars = class(TPlatformDefaultStyleActionBars)
public
function GetControlClass(ActionBar: TCustomActionBar;
AnItem: TActionClientItem): TCustomActionControlClass; override;
function GetStyleName: string; override;
end;
TForm1 = class(TForm)
ActionMainMenuBar1: TActionMainMenuBar;
ActionManager1: TActionManager;
procedure FormCreate(Sender: TObject);
procedure FormDestroy(Sender: TObject);
private
Style: TFixedPlatformDefaultStyleActionBars;
public
end;
implementation
procedure TForm1.FormCreate(Sender: TObject);
begin
Style := TFixedPlatformDefaultStyleActionBars.Create();
ActionManager1.Style := Style;
end;
procedure TForm1.FormDestroy(Sender: TObject);
begin
Style.Free();
end;
procedure TFixedThemedMenuItemStyle.CalcBounds;
const
CheckMarkStates: array[Boolean] of Integer =
(MC_CHECKMARKDISABLED, MC_CHECKMARKNORMAL);
SubMenuStates: array[Boolean] of Integer = (MSM_DISABLED, MSM_NORMAL);
var
DC: HDC;
LFont: HFONT;
LTheme: HTheme;
LBounds: TRect;
LImageSize: TPoint;
LHeight, LWidth, Offset: Integer;
LGlyphSize, LGutterSize, LSeparatorSize, LSubMenuGlyphSize: TSize;
LCheckMargins, LGutterMargins, LMenuItemMargins, LSeparatorMargins, LSubMenuGlyphMargins: TMargins;
begin
// Fill in parent object's private fields.
inherited;
DC := CreateCompatibleDC(0);
try
LFont := SelectObject(DC, Screen.MenuFont.Handle);
try
Font.Assign(Screen.MenuFont);
inherited;
LTheme := ThemeServices.Theme[teMenu];
LHeight := 0;
LWidth := 0;
// Check/Glyph
GetThemePartSize(LTheme, DC, MENU_POPUPCHECK,
CheckMarkStates[Enabled], nil, TS_TRUE, LGlyphSize);
GetThemeMargins(LTheme, DC, MENU_POPUPCHECK,
CheckMarkStates[Enabled], TMT_CONTENTMARGINS, nil, LCheckMargins);
// Gutter
GetThemePartSize(LTheme, DC, MENU_POPUPGUTTER, 0, nil, TS_TRUE, LGutterSize);
GetThemeMargins(LTheme, DC, MENU_POPUPGUTTER, 0, TMT_SIZINGMARGINS, nil, LGutterMargins);
// Menu item
GetThemeMargins(LTheme, DC, MENU_POPUPITEM, MPI_NORMAL, TMT_SIZINGMARGINS, nil, LMenuItemMargins);
GetThemePartSize(LTheme, DC, MENU_POPUPSUBMENU, SubMenuStates[Enabled], nil, TS_TRUE, LSubMenuGlyphSize);
GetThemeMargins(LTheme, DC, MENU_POPUPSUBMENU, SubMenuStates[Enabled], TMT_CONTENTMARGINS, nil, LSubMenuGlyphMargins);
// Calculate check/glyph size
LImageSize := GetImageSize;
if LImageSize.Y > LGlyphSize.cy then
LGlyphSize.cy := LImageSize.Y;
if LImageSize.X > LGlyphSize.cx then
LGlyphSize.cx := LImageSize.X;
Inc(LHeight, LGlyphSize.cy);
Inc(LWidth, LGlyphSize.cx);
// Add margins for check/glyph
Inc(LHeight, LCheckMargins.cyTopHeight + LCheckMargins.cyBottomHeight);
Inc(LWidth, LCheckMargins.cxLeftWidth + LCheckMargins.cxRightWidth);
FCheckRect := Rect(0, 0,
LGlyphSize.cx + LCheckMargins.cxRightWidth + LCheckMargins.cxRightWidth,
LGlyphSize.cy + LCheckMargins.cyBottomHeight + LCheckMargins.cyBottomHeight);
// Add size and margins for gutter
Inc(LWidth, LGutterMargins.cxLeftWidth);
FGutterRect.Left := LWidth;
FGutterRect.Right := FGutterRect.Left + LGutterSize.cx;
Inc(LWidth, LGutterSize.cx + LGutterMargins.cxRightWidth);
// Add margins for menu item
Inc(LWidth, LMenuItemMargins.cxLeftWidth + LMenuItemMargins.cxRightWidth);
Offset := LWidth - TextBounds.Left - LMenuItemMargins.cxRightWidth;
LBounds := TextBounds;
OffsetRect(LBounds, Offset, -1);
TextBounds := LBounds;
// Add size of potential submenu glyph
Inc(LWidth, LSubMenuGlyphSize.cx);
Inc(LWidth, LSubMenuGlyphMargins.cxLeftWidth);
Inc(LWidth, LSubMenuGlyphMargins.cxRightWidth);
// Add Width of menu item to FSubMenuGlyphRect before using
FSubMenuGlyphRect := Rect(-LSubMenuGlyphMargins.cxRightWidth - LSubMenuGlyphSize.cx,
(Height - LSubMenuGlyphSize.cy) div 2,
-LSubMenuGlyphMargins.cxRightWidth,
((Height - LSubMenuGlyphSize.cy) div 2) + LSubMenuGlyphSize.cy);
// Add margins for menu short cut
if ActionClient <> nil then
begin
LBounds := Rect(0, 0, 0, 0);
DoDrawText(DC, ActionClient.ShortCutText, LBounds, DT_CALCRECT or DT_NOCLIP);
end
else
LBounds := ShortCutBounds;
Offset := FSubMenuGlyphRect.Left - LBounds.Right -
LMenuItemMargins.cxRightWidth - LSubMenuGlyphMargins.cxLeftWidth;
OffsetRect(LBounds, Offset, 0);
// Add Width of menu item to ShortCutBounds before using
ShortCutBounds := LBounds;
Inc(LWidth, LMenuItemMargins.cxLeftWidth + LMenuItemMargins.cxRightWidth);
// Adjust size if separator
if Separator then
begin
GetThemePartSize(LTheme, DC, MENU_POPUPSEPARATOR, 0, nil, TS_TRUE, LSeparatorSize);
GetThemeMargins(LTheme, DC, MENU_POPUPSEPARATOR, 0, TMT_SIZINGMARGINS, nil, LSeparatorMargins);
LHeight := LSeparatorSize.cy + LSeparatorMargins.cyBottomHeight;
LWidth := LSeparatorSize.cx;
FSeparatorHeight := LSeparatorSize.cy;
end;
FGutterRect.Top := 0;
FGutterRect.Bottom := LHeight;
SetBounds(Left, Top,
LWidth + TextBounds.Right - TextBounds.Left + ShortCutBounds.Right - ShortCutBounds.Left,
LHeight);
finally
SelectObject(DC, LFont);
end;
finally
DeleteDC(DC);
end;
end;
// THE ONLY SERIOUS DIFFERENCE: RENDERING BULLETS INSTEAD OF CHECKMARKS FOR RADIO ITEMS
procedure TFixedThemedMenuItemStyle.DoDrawMenuCheck;
const
CheckMarkBkgs: array[Boolean] of Integer = (MCB_DISABLED, MCB_NORMAL);
CheckMarkStates: array[Boolean] of Integer = (MC_CHECKMARKDISABLED, MC_CHECKMARKNORMAL);
RadioMarkStates: array[Boolean] of Integer = (MC_BULLETDISABLED, MC_BULLETNORMAL);
begin
if IsChecked then
begin
DrawThemeBackground(ThemeServices.Theme[teMenu], Canvas.Handle,
MENU_POPUPCHECKBACKGROUND, CheckMarkBkgs[Enabled], FCheckRect, nil);
if not HasGlyph then
begin
if IsGrouped then
begin
DrawThemeBackground(ThemeServices.Theme[teMenu], Canvas.Handle,
MENU_POPUPCHECK, RadioMarkStates[Enabled], FCheckRect, nil);
end
else
begin
DrawThemeBackground(ThemeServices.Theme[teMenu], Canvas.Handle,
MENU_POPUPCHECK, CheckMarkStates[Enabled], FCheckRect, nil);
end;
end;
end;
end;
procedure TFixedThemedMenuItemStyle.DoDrawText(
DC: HDC; const Text: string; var Rect: TRect; Flags: Integer);
const
MenuStates: array[Boolean] of Integer = (MPI_DISABLED, MPI_NORMAL);
var
Options: TDTTOpts;
begin
// Setup Options
{$IF NOT DEFINED(CLR)}
FillChar(Options, SizeOf(Options), 0);
Options.dwSize := SizeOf(Options);
{$ELSE}
Options.dwSize := Marshal.SizeOf(TypeOf(Options));
{$IFEND}
Options.dwFlags := DTT_TEXTCOLOR or DTT_COMPOSITED;
if Flags and DT_CALCRECT = DT_CALCRECT then
Options.dwFlags := Options.dwFlags or DTT_CALCRECT;
// Retrieve text color
GetThemeColor(ThemeServices.Theme[teMenu], MENU_POPUPITEM,
MenuStates[Enabled or ActionBar.DesignMode], TMT_TEXTCOLOR, Options.crText);
// Draw menu item text
DrawThemeTextEx(ThemeServices.Theme[teMenu], DC, MENU_POPUPITEM,
MenuStates[Enabled or ActionBar.DesignMode], Text, Length(Text), Flags, Rect, Options);
end;
procedure TFixedThemedMenuItemStyle.DrawGlyph(const Location: TPoint);
var
LImageSize, LLocation: TPoint;
begin
if (Action is TCustomAction) and TCustomAction(Action).Checked then
DoDrawMenuCheck;
if HasGlyph then
begin
LImageSize := GetImageSize;
LLocation.X := ((FCheckRect.Right - FCheckRect.Left) - LImageSize.X) div 2;
LLocation.Y := ((FCheckRect.Bottom - FCheckRect.Top) - LImageSize.Y) div 2;
inherited DrawGlyph(LLocation);
end;
end;
type
TActionControlStyle = (csStandard, csXPStyle, csThemed);
function GetActionControlStyle: TActionControlStyle;
begin
if Win32MajorVersion >= 6 then
begin
if ThemeServices.Theme[teMenu] <> 0 then
Result := csThemed
else
Result := csXPStyle;
end
else
if CheckWin32Version(5, 1) then
Result := csXPStyle
else
Result := csStandard;
end;
function TFixedPlatformDefaultStyleActionBars.GetControlClass(ActionBar: TCustomActionBar;
AnItem: TActionClientItem): TCustomActionControlClass;
begin
if ActionBar is TCustomActionToolBar then
begin
if AnItem.HasItems then
case GetActionControlStyle of
csStandard: Result := TStandardDropDownButton;
csXPStyle: Result := TXPStyleDropDownBtn;
else
Result := TThemedDropDownButton;
end
else
if (AnItem.Action is TStaticListAction) or
(AnItem.Action is TVirtualListAction) then
Result := TCustomComboControl
else
case GetActionControlStyle of
csStandard: Result := TStandardButtonControl;
csXPStyle: Result := TXPStyleButton;
else
Result := TThemedButtonControl;
end
end
else if ActionBar is TCustomActionMainMenuBar then
case GetActionControlStyle of
csStandard: Result := TStandardMenuButton;
csXPStyle: Result := TXPStyleMenuButton;
else
Result := TThemedMenuButton;
end
else if ActionBar is TCustomizeActionToolBar then
begin
with TCustomizeActionToolbar(ActionBar) do
if not Assigned(RootMenu) or
(AnItem.ParentItem <> TCustomizeActionToolBar(RootMenu).AdditionalItem) then
case GetActionControlStyle of
csStandard: Result := TStandardMenuItem;
csXPStyle: Result := TXPStyleMenuItem;
else
Result := TFixedThemedMenuItemStyle;
end
else
case GetActionControlStyle of
csStandard: Result := TStandardAddRemoveItem;
csXPStyle: Result := TXPStyleAddRemoveItem;
else
Result := TThemedAddRemoveItem;
end
end
else if ActionBar is TCustomActionPopupMenu then
case GetActionControlStyle of
csStandard: Result := TStandardMenuItem;
csXPStyle: Result := TXPStyleMenuItem;
else
Result := TFixedThemedMenuItemStyle;
end
else
case GetActionControlStyle of
csStandard: Result := TStandardButtonControl;
csXPStyle: Result := TXPStyleButton;
else
Result := TThemedButtonControl;
end
end;
function TFixedPlatformDefaultStyleActionBars.GetStyleName: string;
begin
Result := 'My fixed platform style';
end;
关于delphi - TActionMainMenuBar 菜单中的 RadioItems,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10530887/
请在标记为重复之前阅读。 我正在创建一组依赖智能卡进行身份验证的应用程序。到目前为止,每个应用程序都单独控制智能卡读卡器。几周后,我的一些客户将同时使用多个应用程序。因此,我认为创建一个控制身份验证过
我想设置一个小程序,从数据库中检索信息,然后根据请求将该信息分发给另一个程序。例如,一个名为“Master”的程序将从数据库中检索数据并创建一个对象集合(列表、数组等,无论哪种效果最好),然后一个名为
我有两台电脑,都装有 XE2。我以为我在两者上安装了相同的安装,但在其中一个上安装第 3 方软件包时遇到问题,而另一个则正常。 无论如何,我希望两者都一样。最简单的人可能只是通过移入我的 Dropbo
有冲突吗? 最佳答案 所有新版本的 Delphi 始终可以安全地安装到旧版本的下一个版本。 每个新版本都应安装在其自己的目录中。 如果您要安装多个版本,请始终先安装最旧的版本,然后再安装最新版本。 我
快速提问:如果我从代码中删除 // 或 (* *) 中的注释,Delphi 2007 的执行时间会受到影响吗?最终结果是一个可能包含数千行注释的 EXE 文件。 最佳答案 编译器会简单地忽略注释,并且
我必须对照另一个文件检查文件的每一行。 如果第二个文件中存在第一个文件中的一行,则必须删除它。 现在,我正在使用2个列表框,并且“对于listbox1.items.count-1可以开始...” 我的
我正在尝试在访问数据库中添加一些数据。但是我有麻烦,因为这会返回错误: ADOQuery1 missing sql property 实现了对代码的几次修改,到目前为止没有任何效果。 我究竟做错了什么
我用Delphi 5编写了一个程序,在Windows 8 32位PC上可以正常运行。我发现在Windows 7 64位笔记本电脑上运行它最终会导致reallocmem错误,而该错误在32位PC上不会发
看来这是我需要的工具,用于提取XML并与TClientDataset连接。我已经在几篇文章和文档中看到了它,但是我无法在XE2组件列表中找到它-在任何地方!应该在哪里?是否在可能未安装的可选软件包中?
我正在寻找一个非常通用的TDBTree组件,我想听听一些建议。我正在特别寻找一种显示主记录和“ n”个链接表记录的记录。 (我的意思是来自各个表的记录)。例如,TDBTree将钩接到主表,明细表1,附
我需要将按钮制作成旋转三角形的形状(或者说是任何多边形)。谁能提供任何建议? 最佳答案 查看Win32 API CreatePolygonRgn()和SetWindowRgn()函数,以创建一个HRG
你好专家 我的JvPasswordForm1有一个旧的JVC组件。 似乎该组件不再存在:它替换为哪个组件? 重新获得 最佳答案 尝试查找TJvLoginDialog,TjvPassword已合并到其中
几天前,我已经设置了我的开发环境(在装有Win 7的VM和域上的用户的VM上安装了delphi 2009),并安装了我的组件(jedi's,devExpress,ADS等)。 今天,我启动机器,打开d
开始对控件进行子分类的正确位置/时间是什么? 恢复原始窗口proc的正确时间是几点? 现在我在表单创建过程中子类化: procedure TForm1.FormCreate(Sender: TObje
有人可以给我一些有关如何登录访问的网页(使用任何网络浏览器)的指示吗?我应该建立一个全球代理....钩住网络....吗?我需要记录的只是页面地址,而不是其中包含的信息。 我正在使用Delphi。 谢谢
我创建了一个像 TMyClass = class(TObject) private FList1: TObjectList; FList2: TObjectList; public end;
我有一个BPG文件,我已对其进行修改以用作我们公司的自动构建服务器的make文件。为了使其正常工作,我必须进行更改 用途*用途 'unit1.pas'中的unit1 * unit1 'unit2.pa
我将Delphi 7代码迁移到了Delphi XE4。我在Delphi XE4的LoadFromStram方法中遇到错误,但对于Delphi 7来说也可以正常工作。 错误: First chance
我正在尝试学习一些新技巧,以便更好地组织我在 Delphi 中的单元中的一些源代码。 我注意到我访问的一些函数或方法似乎是类中的类,但是我还没有成功地在类中创建一个工作类,虽然它编译得很好,但在执行代
我有一个包含许多类的大单元,现在我想通过将某些类分成新的单元来重构该单元。 我不得不承认我缺乏使用Delphi内置IDE功能的经验。利用内置功能“查找|查找对类型的本地引用”并没有多大帮助,因为类方法
我是一名优秀的程序员,十分优秀!