LCL: Allow TToolbar to show a TMainMenu. Patch by FerDeLance See MR !393

This commit is contained in:
Martin 2024-12-15 12:21:51 +01:00
parent facd1187fc
commit 2faa5bdf05
4 changed files with 181 additions and 0 deletions

View File

@ -2263,6 +2263,26 @@ type
TToolBar = class(TToolWindow) TToolBar = class(TToolWindow)
private private
type
{ TTMPObserver }
TTMPObserver = class(TObject, IFPObserver)
private
FDoOnChange: TNotifyEvent;
FDoOnFree: TNotifyEvent;
procedure FPOObservedChanged(ASender : TObject; Operation : TFPObservedOperation; Data : Pointer);
procedure SetDoOnChange(AValue: TNotifyEvent);
procedure SetDoOnFree(AValue: TNotifyEvent);
public
property DoOnChange: TNotifyEvent read FDoOnChange write SetDoOnChange;
property DoOnFree: TNotifyEvent read FDoOnFree write SetDoOnFree;
end;
private
FMenuObserver: TTMPObserver;
FMenu: TMainMenu;
FSubMenu: TPopupMenu;
FSubMenuItems: TList;
FMenuButtons: TList;
FOnPaint: TNotifyEvent; FOnPaint: TNotifyEvent;
FOnPaintButton: TToolBarOnPaintButton; FOnPaintButton: TToolBarOnPaintButton;
FButtonHeight: Integer; FButtonHeight: Integer;
@ -2315,6 +2335,7 @@ type
procedure SetImagesWidth(const aImagesWidth: Integer); procedure SetImagesWidth(const aImagesWidth: Integer);
procedure SetIndent(const AValue: Integer); procedure SetIndent(const AValue: Integer);
procedure SetList(const AValue: Boolean); procedure SetList(const AValue: Boolean);
procedure SetMenu(AValue: TMainMenu);
procedure SetOrientation(AValue: TToolBarOrientation); procedure SetOrientation(AValue: TToolBarOrientation);
procedure SetShowCaptions(const AValue: Boolean); procedure SetShowCaptions(const AValue: Boolean);
procedure SetTransparent(const AValue: Boolean); procedure SetTransparent(const AValue: Boolean);
@ -2326,6 +2347,10 @@ type
procedure MoveSubMenuItems(SrcMenuItem, DestMenuItem: TMenuItem); procedure MoveSubMenuItems(SrcMenuItem, DestMenuItem: TMenuItem);
procedure AddButton(Button: TToolButton); procedure AddButton(Button: TToolButton);
procedure RemoveButton(Button: TToolButton); procedure RemoveButton(Button: TToolButton);
procedure UpdateMenuItem(Sender: TObject);
procedure RemoveMenu(Sender: TObject);
procedure SubMenuItemClick(Sender: TObject);
procedure MenuButtonClick(Sender: TObject);
protected const protected const
cDefButtonWidth = 23; cDefButtonWidth = 23;
cDefButtonHeight = 22; cDefButtonHeight = 22;
@ -2401,6 +2426,7 @@ type
property ImagesWidth: Integer read FImagesWidth write SetImagesWidth default 0; property ImagesWidth: Integer read FImagesWidth write SetImagesWidth default 0;
property Indent: Integer read FIndent write SetIndent default 1; property Indent: Integer read FIndent write SetIndent default 1;
property List: Boolean read FList write SetList default False; property List: Boolean read FList write SetList default False;
property Menu: TMainMenu read FMenu write SetMenu;
property ParentColor; property ParentColor;
property ParentFont; property ParentFont;
property ParentShowHint; property ParentShowHint;

View File

@ -47,6 +47,7 @@ begin
// pass CM_MENUCANGED to the form which own the menu // pass CM_MENUCANGED to the form which own the menu
if WindowHandle <> 0 then if WindowHandle <> 0 then
SendMessage(WindowHandle, CM_MENUCHANGED, 0, 0); SendMessage(WindowHandle, CM_MENUCHANGED, 0, 0);
Self.FPONotifyObservers(Sender, ooChange, nil);
inherited MenuChanged(Sender, Source, Rebuild); inherited MenuChanged(Sender, Source, Rebuild);
end; end;

View File

@ -1269,6 +1269,7 @@ begin
FCaption := AValue; FCaption := AValue;
if HandleAllocated and ((Parent <> nil) or (FMenu = nil)) then if HandleAllocated and ((Parent <> nil) or (FMenu = nil)) then
TWSMenuItemClass(WidgetSetClass).SetCaption(Self, AValue); TWSMenuItemClass(WidgetSetClass).SetCaption(Self, AValue);
MenuChanged(False);
OwnerFormDesignerModified(Self); OwnerFormDesignerModified(Self);
end; end;
@ -1620,6 +1621,7 @@ begin
if MergedParent<>nil then if MergedParent<>nil then
MergedParent.InvalidateMergedItems; MergedParent.InvalidateMergedItems;
end; end;
MenuChanged(False);
end; end;
procedure TMenuItem.UpdateImage(forced: Boolean); procedure TMenuItem.UpdateImage(forced: Boolean);

View File

@ -100,6 +100,12 @@ begin
FNewStyle := True; FNewStyle := True;
FWrapable := True; FWrapable := True;
FButtons := TList.Create; FButtons := TList.Create;
FMenuButtons := TList.Create;
FSubMenuItems := TList.Create;
FSubMenu := TPopupMenu.Create(Self);
FMenuObserver := TTMPObserver.Create;
FMenuObserver.DoOnChange := @UpdateMenuItem;
FMenuObserver.DoOnFree := @RemoveMenu;
FIndent := 1; FIndent := 1;
FList := False; FList := False;
FImageChangeLink := TChangeLink.Create; FImageChangeLink := TChangeLink.Create;
@ -122,7 +128,16 @@ begin
if TControl(FButtons[I]) is TToolButton then if TControl(FButtons[I]) is TToolButton then
TToolButton(FButtons[I]).FToolBar := nil; TToolButton(FButtons[I]).FToolBar := nil;
for I := 0 to FMenuButtons.Count - 1 do
if TControl(FMenuButtons[I]) is TSpeedButton then
TSpeedButton(FMenuButtons[I]).Free;
if Assigned(FMenu) then FMenu.FPODetachObserver(FMenuObserver);
FreeThenNil(FButtons); FreeThenNil(FButtons);
FreeThenNil(FMenuButtons);
FreeAndNil(FSubMenu);
FreeThenNil(FSubMenuItems);
FreeAndNil(FMenuObserver);
FreeThenNil(FHotImageChangeLink); FreeThenNil(FHotImageChangeLink);
FreeThenNil(FImageChangeLink); FreeThenNil(FImageChangeLink);
FreeThenNil(FDisabledImageChangeLink); FreeThenNil(FDisabledImageChangeLink);
@ -242,6 +257,104 @@ begin
FButtons.Remove(Button); FButtons.Remove(Button);
end; end;
procedure TToolBar.UpdateMenuItem(Sender: TObject);
var
i, xpos, wglyph: Integer;
tmpMenuButton: TSpeedButton;
begin
xpos := 0;
while (FMenuButtons.Count > 0) do
begin
i := FMenuButtons.Count - 1;
tmpMenuButton := TSpeedButton(FMenuButtons.Items[i]);
FMenuButtons.Delete(i);
FreeAndNil(tmpMenuButton);
end;
if not Assigned(FMenu) then Exit;
for i := 0 to FMenu.Items.Count-1 do
if FMenu.Items[i].Visible then
begin
tmpMenuButton := TSpeedButton.Create(Self);
tmpMenuButton.AutoSize := False;
tmpMenuButton.Align := alNone;
tmpMenuButton.Alignment := taCenter;
tmpMenuButton.Height := FMenu.Height;
tmpMenuButton.Left := xpos;
tmpMenuButton.Flat := True;
tmpMenuButton.ParentFont := True;
tmpMenuButton.Color := Self.Color;
tmpMenuButton.ShowCaption := True;
tmpMenuButton.Caption := FMenu.Items[i].Caption;
tmpMenuButton.Enabled := FMenu.Items[i].Enabled;
tmpMenuButton.Visible := FMenu.Items[i].Visible;
tmpMenuButton.Margin := -1;
tmpMenuButton.Spacing := 7;
tmpMenuButton.Tag := i;
wglyph := 0;
tmpMenuButton.Images := nil;
if Assigned(FMenu.Images) and (FMenu.Items[i].ImageIndex >= 0) then
if FMenu.Items[i].ImageIndex < FMenu.Images.Count then
begin
FMenu.Images.GetBitmap(FMenu.Items[i].ImageIndex, tmpMenuButton.Glyph);
wglyph := tmpMenuButton.Glyph.Width + 5;
end;
tmpMenuButton.OnClick := @MenuButtonClick;
tmpMenuButton.Parent := Self;
if FMenu.Items[i].Default then tmpMenuButton.Font.Style := [fsBold];
tmpMenuButton.Width := Self.Canvas.TextWidth(FMenu.Items[i].Caption) + wglyph + 14;
tmpMenuButton.Constraints.MinWidth := tmpMenuButton.Width;
tmpMenuButton.Constraints.MaxWidth := tmpMenuButton.Width;
tmpMenuButton.Update;
Inc(xpos, tmpMenuButton.Width);
FMenuButtons.Add(tmpMenuButton);
end;
end;
procedure TToolBar.RemoveMenu(Sender: TObject);
begin
FMenu.FPODetachObserver(FMenuObserver);
FMenu := nil;
end;
procedure TToolBar.SubMenuItemClick(Sender: TObject);
var
tmpMI: TMenuItem;
begin
tmpMI := TMenuItem(FSubMenuItems.Items[TMenuItem(Sender).Tag]);
if Assigned(tmpMI) then tmpMI.Click;
end;
procedure TToolBar.MenuButtonClick(Sender: TObject);
var
MenuButton: TSpeedButton;
SubMenuPos: TPoint;
procedure PrepSubMenuItem(SubMenuItemIn, SubMenuItemOut: TMenuItem);
var
i: Integer;
begin
for i := 0 to SubMenuItemIn.Count - 1 do
begin
FSubMenuItems.Add(SubMenuItemIn.Items[i]);
SubMenuItemOut.Items[i].Tag := FSubMenuItems.Count - 1;
SubMenuItemOut.Items[i].OnClick := @SubMenuItemClick;
if SubMenuItemIn.Items[i].Count > 0 then PrepSubMenuItem(SubMenuItemIn.Items[i], SubMenuItemOut.Items[i]);
end;
end;
begin
MenuButton := TSpeedButton(Sender);
if Assigned(FMenu.Items[MenuButton.Tag].OnClick) then FMenu.Items[MenuButton.Tag].Click;
if FMenu.Items[MenuButton.Tag].Count = 0 then Exit;
FSubMenu.Items.Assign(FMenu.Items[MenuButton.Tag]);
FSubMenuItems.Clear;
PrepSubMenuItem(FMenu.Items[MenuButton.Tag], FSubMenu.Items);
SubMenuPos := ClientToScreen(TPoint.Create(MenuButton.Left, MenuButton.Top + MenuButton.Height));
FSubMenu.PopUp(SubMenuPos.X, SubMenuPos.Y);
end;
function TToolBar.IsVertical: Boolean; function TToolBar.IsVertical: Boolean;
begin begin
case FOrientation of case FOrientation of
@ -304,6 +417,18 @@ begin
UpdateVisibleBar; UpdateVisibleBar;
end; end;
procedure TToolBar.SetMenu(AValue: TMainMenu);
begin
if FMenu = AValue then Exit;
if Assigned(FMenu) then
begin
FMenu.FPODetachObserver(FMenuObserver);
end;
FMenu := AValue;
if Assigned(FMenu) then FMenu.FPOAttachObserver(FMenuObserver);
UpdateVisibleBar;
end;
procedure TToolBar.SetOrientation(AValue: TToolBarOrientation); procedure TToolBar.SetOrientation(AValue: TToolBarOrientation);
begin begin
if FOrientation = AValue then Exit; if FOrientation = AValue then Exit;
@ -341,6 +466,7 @@ begin
inherited Notification(AComponent, Operation); inherited Notification(AComponent, Operation);
if Operation = opRemove then if Operation = opRemove then
begin begin
if AComponent = FMenu then Menu := nil;
if AComponent = FImages then Images := nil; if AComponent = FImages then Images := nil;
if AComponent = FHotImages then HotImages := nil; if AComponent = FHotImages then HotImages := nil;
if AComponent = FDisabledImages then DisabledImages := nil; if AComponent = FDisabledImages then DisabledImages := nil;
@ -434,6 +560,7 @@ begin
Include(FToolBarFlags,tbfUpdateVisibleBarNeeded); Include(FToolBarFlags,tbfUpdateVisibleBarNeeded);
Exit; Exit;
end; end;
UpdateMenuItem(Self);
for i := 0 to FButtons.Count - 1 do for i := 0 to FButtons.Count - 1 do
begin begin
TControl(FButtons[i]).InvalidatePreferredSize; TControl(FButtons[i]).InvalidatePreferredSize;
@ -472,6 +599,9 @@ begin
end; end;
procedure TToolBar.EndUpdate; procedure TToolBar.EndUpdate;
var
i: Integer;
tmpMenuItem: TMenuItem;
begin begin
inherited EndUpdate; inherited EndUpdate;
if FUpdateCount=0 then begin if FUpdateCount=0 then begin
@ -1130,3 +1260,25 @@ begin
Button.Click; Button.Click;
end; end;
{ TToolBar.TTMPObserver }
procedure TToolBar.TTMPObserver.FPOObservedChanged(ASender: TObject;
Operation: TFPObservedOperation; Data: Pointer);
begin
case Operation of
ooFree: if Assigned(DoOnFree) then DoOnFree(ASender);
ooChange: if Assigned(DoOnChange) then DoOnChange(ASender);
end;
end;
procedure TToolBar.TTMPObserver.SetDoOnChange(AValue: TNotifyEvent);
begin
if FDoOnChange = AValue then Exit;
FDoOnChange := AValue;
end;
procedure TToolBar.TTMPObserver.SetDoOnFree(AValue: TNotifyEvent);
begin
if FDoOnFree = AValue then Exit;
FDoOnFree := AValue;
end;