Compare commits

...

3 Commits

Author SHA1 Message Date
Massimo Magnano
a87b5c814f Merge branch 'main' into 'main'
Solved GTK2 TMenuItem.Default property does not works #41528

See merge request freepascal.org/lazarus/lazarus!449
2025-04-03 22:21:19 +02:00
zeljan1
91fbaca370 Qt5,Qt6: Fixed QLineEdit behaviour and automatic selectAll() by Qt when control is focused.Related issue #10155 and issue #41562 2025-04-03 21:57:50 +02:00
Massimo Magnano
37ce318225 Solved GTK2 TMenuItem.Default property does not works #41528 2025-03-12 17:44:26 +01:00
9 changed files with 114 additions and 50 deletions

View File

@ -1309,9 +1309,12 @@ begin
// Only one item in a menu or submenu can be default. // Only one item in a menu or submenu can be default.
for i:=0 to FParent.Count-1 do for i:=0 to FParent.Count-1 do
if FParent[i].Default then if FParent[i].Default then
FParent[i].FDefault := False; FParent[i].Default := False;
FDefault:= AValue; FDefault:= AValue;
MenuChanged(True); if HandleAllocated and not (csReading in ComponentState) and ((FParent <> nil) or (FMenu = nil)) then
TWSMenuItemClass(WidgetSetClass).SetDefault(Self, AValue);
MenuChanged(False);
OwnerFormDesignerModified(Self);
end; end;
{------------------------------------------------------------------------------ {------------------------------------------------------------------------------

View File

@ -302,7 +302,9 @@ type
procedure _SetCallbackEx(const AMsg: LongInt; const AGTKObject: PGTKObject; const ALCLObject: TObject; Direct: Boolean); procedure _SetCallbackEx(const AMsg: LongInt; const AGTKObject: PGTKObject; const ALCLObject: TObject; Direct: Boolean);
procedure SetCallbackEx(const AMsg: LongInt; const AGTKObject: PGTKObject; const ALCLObject: TObject; Direct: Boolean); procedure SetCallbackEx(const AMsg: LongInt; const AGTKObject: PGTKObject; const ALCLObject: TObject; Direct: Boolean);
procedure SetCommonCallbacks(const AGTKObject: PGTKObject; const ALCLObject: TObject); procedure SetCommonCallbacks(const AGTKObject: PGTKObject; const ALCLObject: TObject);
procedure SetLabelCaption(const ALabel: PGtkLabel; const ACaption: String); function SetLabelCaption(const ALabel: PGtkLabel; const ACaption: String): String;
function SetLabelCaptionMarkup(const ALabel: PGtkLabel; const ACaption: String;
AmpersandsEscape: Boolean=True; MarkupsEscape: Boolean=True): String;
procedure SetSelectionMode(Sender: TObject; Widget: PGtkWidget; procedure SetSelectionMode(Sender: TObject; Widget: PGtkWidget;
MultiSelect, {%H-}ExtendedSelect: Boolean); MultiSelect, {%H-}ExtendedSelect: Boolean);
function ForceLineBreaks(DC : hDC; Src: PChar; MaxWidthInPixels : Longint; function ForceLineBreaks(DC : hDC; Src: PChar; MaxWidthInPixels : Longint;

View File

@ -6023,11 +6023,17 @@ var
MenuItemWidget: PGtkWidget); MenuItemWidget: PGtkWidget);
var var
LabelWidget: PGtkLabel; LabelWidget: PGtkLabel;
s: String;
begin begin
if (MenuItemWidget = nil) or (LCLMenuItem = nil) then if (MenuItemWidget = nil) or (LCLMenuItem = nil) then
Exit; Exit;
LabelWidget := g_object_get_data(PGObject(MenuItemWidget), 'LCLLabel'); LabelWidget := g_object_get_data(PGObject(MenuItemWidget), 'LCLLabel');
Gtk2Widgetset.SetLabelCaption(LabelWidget, LCLMenuItem.Caption); if LCLMenuItem.Default then
Gtk2Widgetset.SetLabelCaptionMarkup(LabelWidget,
'<b>'+EscapeMarkups(Ampersands2Underscore(LCLMenuItem.Caption))+'</b>',
False, False)
else Gtk2Widgetset.SetLabelCaption(LabelWidget, LCLMenuItem.Caption);
gtk_widget_set_direction(PGtkWidget(LabelWidget), WidgetDirection[UseRTL]); gtk_widget_set_direction(PGtkWidget(LabelWidget), WidgetDirection[UseRTL]);
end; end;
@ -6057,15 +6063,24 @@ var
if LabelWidget = nil then if LabelWidget = nil then
begin begin
// create a label for the ShortCut // create a label for the ShortCut
LabelWidget := PGtkLabel(gtk_label_new(PChar(Pointer(s)))); LabelWidget := PGtkLabel(gtk_label_new(''));
g_object_set_data(PGObject(MenuItemWidget), 'LCLShortCutLabel', LabelWidget); g_object_set_data(PGObject(MenuItemWidget), 'LCLShortCutLabel', LabelWidget);
gtk_label_set_use_markup(LabelWidget, True);
gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget)); gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget));
if LCLMenuItem.Default then
gtk_label_set_markup(LabelWidget, PChar('<b>'+EscapeMarkups(s)+'</b>'))
else gtk_label_set_text(LabelWidget, PChar(s));
gtk_widget_show(PGtkWidget(LabelWidget)); gtk_widget_show(PGtkWidget(LabelWidget));
end end
else else
begin begin
gtk_label_set_text(LabelWidget, PChar(Pointer(s))); if LCLMenuItem.Default then
gtk_label_set_markup(LabelWidget, PChar('<b>'+EscapeMarkups(s)+'</b>'))
else gtk_label_set_text(LabelWidget, PChar(s));
end; end;
gtk_widget_set_direction(PGtkWidget(LabelWidget), GTK_TEXT_DIR_LTR); //Shortcut always LTR gtk_widget_set_direction(PGtkWidget(LabelWidget), GTK_TEXT_DIR_LTR); //Shortcut always LTR
if UseRTL then if UseRTL then
gtk_misc_set_alignment(GTK_MISC(LabelWidget), 0.0, 0.5) gtk_misc_set_alignment(GTK_MISC(LabelWidget), 0.0, 0.5)
@ -6114,6 +6129,8 @@ var
gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget)); gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget));
SetMenuItemLabelText(LCLMenuItem, MenuItemWidget); SetMenuItemLabelText(LCLMenuItem, MenuItemWidget);
//gtk_accel_label_set_accel_widget(GTK_ACCEL_LABEL(LabelWidget), MenuItemWidget); //gtk_accel_label_set_accel_widget(GTK_ACCEL_LABEL(LabelWidget), MenuItemWidget);
gtk_label_set_use_underline(LabelWidget, True);
gtk_label_set_use_markup(LabelWidget, True);
gtk_widget_show(PGtkWidget(LabelWidget)); gtk_widget_show(PGtkWidget(LabelWidget));
end; end;
@ -9162,6 +9179,19 @@ begin
end; end;
end; end;
function EscapeMarkups(const Str: String): String;
var
s_escaped: Pgchar;
begin
s_escaped:= g_markup_escape_text(PChar(Str), Length(Str));
//MaxM: maybe #95 but even with the escape it is seen as _ and treated as mnemonic
Result:= StringReplace(s_escaped, '__', '&#32;', [rfReplaceAll]);
g_free(s_escaped);
end;
{------------------------------------------------------------------------------- {-------------------------------------------------------------------------------
Creates a new PChar. Deletes escaping ampersands, replaces the first single Creates a new PChar. Deletes escaping ampersands, replaces the first single
ampersand with an underscore and deletes all other single ampersands. ampersand with an underscore and deletes all other single ampersands.
@ -9182,30 +9212,32 @@ end;
-------------------------------------------------------------------------------} -------------------------------------------------------------------------------}
function Ampersands2Underscore(const ASource: String): String; function Ampersands2Underscore(const ASource: String): String;
var var
n: Integer; n, len: Integer;
FirstFound: Boolean; FirstFound: Boolean;
s: String;
begin begin
//TODO: escape underscores //TODO: escape underscores
FirstFound := False; FirstFound := False;
Result := ASource;
n := 1; n := 1;
while n <= Length(Result) do Result:= '';
len:= Length(ASource);
while n <= len do
begin begin
if Result[n] = '&' then case ASource[n] of
begin '_': Result := Result + '__';
if FirstFound '&': if (n < len) and (ASource[n+1] = '&') then // got &&
or ( (n < Length(Result)) and (Result[n+1] = '&') ) // got && begin
then begin Result:= Result + '&';
Delete(Result, n, 1); Inc(n); // Skip the second & of &&
if not FirstFound then end
Inc(n); // Skip the second & of && else if not(FirstFound) then
end begin
else begin Result:= Result+ '_';
FirstFound := True; FirstFound:= True;
Result[n] := '_'; end;
end; else Result:= Result+ ASource[n];
end; end;
Inc(n); inc(n);
end; end;
end; end;

View File

@ -591,6 +591,9 @@ procedure ConnectInternalWidgetsSignals(AWidget: PGtkWidget;
//-- //--
// markups
function EscapeMarkups(const Str: String): String;
// accelerators // accelerators
function Ampersands2Underscore(Src: PChar): PChar; function Ampersands2Underscore(Src: PChar): PChar;
function Ampersands2Underscore(const ASource: String): String; function Ampersands2Underscore(const ASource: String): String;

View File

@ -950,32 +950,22 @@ begin
ConnectSignal(AGTKObject,'grab-notify',@gtk2GrabNotify, ALCLObject); ConnectSignal(AGTKObject,'grab-notify',@gtk2GrabNotify, ALCLObject);
end; end;
procedure TGtk2WidgetSet.SetLabelCaption(const ALabel: PGtkLabel; function TGtk2WidgetSet.SetLabelCaption(const ALabel: PGtkLabel;
const ACaption: String); const ACaption: String): String;
var
s: String;
i: Integer;
begin begin
s := ''; Result:= Ampersands2Underscore(ACaption);
i := 1; gtk_label_set_text_with_mnemonic(ALabel, PChar(Result));
while i <= Length(ACaption) do end;
begin
case ACaption[i] of function TGtk2WidgetSet.SetLabelCaptionMarkup(const ALabel: PGtkLabel;
'_': s := s + '__'; const ACaption: String; AmpersandsEscape: Boolean; MarkupsEscape: Boolean): String;
'&': begin
if (i < Length(ACaption)) and (ACaption[i + 1] = '&') then Result:= ACaption;
begin if AmpersandsEscape then Result:= Ampersands2Underscore(Result);
s := s + '&'; if MarkupsEscape then Result:= EscapeMarkups(Result);
inc(i);
end gtk_label_set_use_markup(ALabel, True);
else gtk_label_set_markup_with_mnemonic(ALabel, PChar(Result));
s := s + '_';
else
s := s + ACaption[i];
end;
inc(i);
end;
gtk_label_set_text_with_mnemonic(ALabel, PChar(s));
end; end;
{------------------------------------------------------------------------------ {------------------------------------------------------------------------------

View File

@ -44,6 +44,7 @@ type
class procedure SetShortCut(const AMenuItem: TMenuItem; const ShortCutK1, ShortCutK2: TShortCut); override; class procedure SetShortCut(const AMenuItem: TMenuItem; const ShortCutK1, ShortCutK2: TShortCut); override;
class procedure SetVisible(const AMenuItem: TMenuItem; const Visible: boolean); override; class procedure SetVisible(const AMenuItem: TMenuItem; const Visible: boolean); override;
class function SetCheck(const AMenuItem: TMenuItem; const Checked: boolean): boolean; override; class function SetCheck(const AMenuItem: TMenuItem; const Checked: boolean): boolean; override;
class function SetDefault(const AMenuItem: TMenuItem; const ADefault: boolean): boolean; override;
class function SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean; override; class function SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean; override;
class function SetRadioItem(const AMenuItem: TMenuItem; const {%H-}RadioItem: boolean): boolean; override; class function SetRadioItem(const AMenuItem: TMenuItem; const {%H-}RadioItem: boolean): boolean; override;
class function SetRightJustify(const AMenuItem: TMenuItem; const Justified: boolean): boolean; override; class function SetRightJustify(const AMenuItem: TMenuItem; const Justified: boolean): boolean; override;
@ -452,6 +453,23 @@ begin
end; end;
end; end;
class function TGtk2WSMenuItem.SetDefault(const AMenuItem: TMenuItem; const ADefault: boolean): boolean;
var
MenuItemWidget: PGtkWidget;
begin
Result:= False;
if not WSCheckMenuItem(AMenuItem, 'SetDefault') then
Exit;
MenuItemWidget:={%H-}PGtkWidget(AMenuItem.Handle);
UpdateInnerMenuItem(AMenuItem,MenuItemWidget);
// gtk_widget_set_sensitive({%H-}PGtkWidget(AMenuItem.Handle), AMenuItem.Enabled);
Result:= True;
end;
class function TGtk2WSMenuItem.SetEnable(const AMenuItem: TMenuItem; class function TGtk2WSMenuItem.SetEnable(const AMenuItem: TMenuItem;
const Enabled: boolean): boolean; const Enabled: boolean): boolean;
begin begin

View File

@ -107,6 +107,7 @@ type
TQtWidget = class(TQtObject, IUnknown) TQtWidget = class(TQtObject, IUnknown)
private private
FDefaultFocusReason: QtFocusReason;
FInResizeEvent: boolean; FInResizeEvent: boolean;
FWidgetState: TQtWidgetStates; FWidgetState: TQtWidgetStates;
FWidgetDefaultFont: TQtFont; FWidgetDefaultFont: TQtFont;
@ -319,6 +320,7 @@ type
nil): QPixmapH; nil): QPixmapH;
property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget; property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget;
property Context: HDC read GetContext; property Context: HDC read GetContext;
property DefaultFocusReason: QtFocusReason read FDefaultFocusReason write FDefaultFocusReason;
property HasCaret: Boolean read FHasCaret write SetHasCaret; property HasCaret: Boolean read FHasCaret write SetHasCaret;
property HasPaint: Boolean read FHasPaint write FHasPaint; property HasPaint: Boolean read FHasPaint write FHasPaint;
property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent; property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent;
@ -2161,6 +2163,7 @@ end;
procedure TQtWidget.InitializeWidget; procedure TQtWidget.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FInResizeEvent := False; FInResizeEvent := False;
// default states // default states
FWidgetState := []; FWidgetState := [];
@ -5164,7 +5167,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155} QWidget_setFocus(Widget, FDefaultFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;
@ -9769,6 +9772,7 @@ begin
FCachedSelectionLen := -1; FCachedSelectionLen := -1;
FIntValidator := nil; FIntValidator := nil;
FNumbersOnly := False; FNumbersOnly := False;
FDefaultFocusReason := QtOtherFocusReason;
if AParams.WndParent <> 0 then if AParams.WndParent <> 0 then
Parent := TQtWidget(AParams.WndParent).GetContainerWidget Parent := TQtWidget(AParams.WndParent).GetContainerWidget
else else
@ -16412,6 +16416,7 @@ end;
procedure TQtMenu.InitializeWidget; procedure TQtMenu.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FWidgetState := []; FWidgetState := [];
ChildOfComplexWidget := ccwNone; ChildOfComplexWidget := ccwNone;
WidgetColorRole := QPaletteWindow; WidgetColorRole := QPaletteWindow;

View File

@ -107,6 +107,7 @@ type
TQtWidget = class(TQtObject, IUnknown) TQtWidget = class(TQtObject, IUnknown)
private private
FDefaultFocusReason: QtFocusReason;
FInResizeEvent: boolean; FInResizeEvent: boolean;
FWidgetState: TQtWidgetStates; FWidgetState: TQtWidgetStates;
FWidgetDefaultFont: TQtFont; FWidgetDefaultFont: TQtFont;
@ -316,6 +317,7 @@ type
nil): QPixmapH; nil): QPixmapH;
property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget; property ChildOfComplexWidget: TChildOfComplexWidget read FChildOfComplexWidget write FChildOfComplexWidget;
property Context: HDC read GetContext; property Context: HDC read GetContext;
property DefaultFocusReason: QtFocusReason read FDefaultFocusReason write FDefaultFocusReason;
property HasCaret: Boolean read FHasCaret write SetHasCaret; property HasCaret: Boolean read FHasCaret write SetHasCaret;
property HasPaint: Boolean read FHasPaint write FHasPaint; property HasPaint: Boolean read FHasPaint write FHasPaint;
property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent; property InResizeEvent: boolean read FInResizeEvent write FInResizeEvent;
@ -2158,6 +2160,7 @@ end;
procedure TQtWidget.InitializeWidget; procedure TQtWidget.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FInResizeEvent := False; FInResizeEvent := False;
// default states // default states
FWidgetState := []; FWidgetState := [];
@ -5168,7 +5171,7 @@ end;
procedure TQtWidget.setFocus; procedure TQtWidget.setFocus;
begin begin
if getFocusPolicy <> QtNoFocus then if getFocusPolicy <> QtNoFocus then
QWidget_setFocus(Widget, QtTabFocusReason) {issue #10155} QWidget_setFocus(Widget, FDefaultFocusReason) {issue #10155}
else else
QWidget_setFocus(Widget); QWidget_setFocus(Widget);
end; end;
@ -9725,6 +9728,7 @@ begin
FCachedSelectionLen := -1; FCachedSelectionLen := -1;
FIntValidator := nil; FIntValidator := nil;
FNumbersOnly := False; FNumbersOnly := False;
FDefaultFocusReason := QtOtherFocusReason;
if AParams.WndParent <> 0 then if AParams.WndParent <> 0 then
Parent := TQtWidget(AParams.WndParent).GetContainerWidget Parent := TQtWidget(AParams.WndParent).GetContainerWidget
else else
@ -16322,6 +16326,7 @@ end;
procedure TQtMenu.InitializeWidget; procedure TQtMenu.InitializeWidget;
begin begin
FDefaultFocusReason := QtTabFocusReason;
FWidgetState := []; FWidgetState := [];
ChildOfComplexWidget := ccwNone; ChildOfComplexWidget := ccwNone;
WidgetColorRole := QPaletteWindow; WidgetColorRole := QPaletteWindow;

View File

@ -59,6 +59,7 @@ type
class procedure SetShortCut(const AMenuItem: TMenuItem; const ShortCutK1, ShortCutK2: TShortCut); virtual; class procedure SetShortCut(const AMenuItem: TMenuItem; const ShortCutK1, ShortCutK2: TShortCut); virtual;
class procedure SetVisible(const AMenuItem: TMenuItem; const Visible: boolean); virtual; class procedure SetVisible(const AMenuItem: TMenuItem; const Visible: boolean); virtual;
class function SetCheck(const AMenuItem: TMenuItem; const Checked: boolean): boolean; virtual; class function SetCheck(const AMenuItem: TMenuItem; const Checked: boolean): boolean; virtual;
class function SetDefault(const AMenuItem: TMenuItem; const ADefault: boolean): boolean; virtual;
class function SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean; virtual; class function SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean; virtual;
class function SetRadioItem(const AMenuItem: TMenuItem; const RadioItem: boolean): boolean; virtual; class function SetRadioItem(const AMenuItem: TMenuItem; const RadioItem: boolean): boolean; virtual;
class function SetRightJustify(const AMenuItem: TMenuItem; const Justified: boolean): boolean; virtual; class function SetRightJustify(const AMenuItem: TMenuItem; const Justified: boolean): boolean; virtual;
@ -157,6 +158,11 @@ begin
Result := false; Result := false;
end; end;
class function TWSMenuItem.SetDefault(const AMenuItem: TMenuItem; const ADefault: boolean): boolean;
begin
Result := false;
end;
class function TWSMenuItem.SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean; class function TWSMenuItem.SetEnable(const AMenuItem: TMenuItem; const Enabled: boolean): boolean;
begin begin
Result := false; Result := false;