lazarus/lcl/interfaces/gtk/gtkproc.inc
Bad Sector f5718e9f06 LCL-GTK1: Fix various startup warnings and Gtk1 assertions
This fixes a bunch of warnings when Gtk1 applications start.  The fixes
are on keyboard initialization (make a previously statically sized array
dynamic as the old value wasn't long enough and disable an unnecessary
warning about filling the VK table as the user can't do anything about
it - nor we unless the whole thing is redesigned), module loading (this
is a side effect of an environment variable collision between Gtk1, Gtk2
and Gtk3 - all of these use the GTK_MODULES variable to load some
modules but since as of 2023 no distribution aside from Slackware comes
with Gtk1, all of these warnings are bogus, so this patch temporarily
cleans the environment variable before initializing Gtk and restores it
later so that child processes can still access it) and passing NULL
styles to gtk_style_copy (the previous code assumed the style retrieval
functions always return a valid object, which is not the case).
2023-11-30 23:13:16 +00:00

9978 lines
313 KiB
PHP

{%MainUnit gtkproc.pp}
{******************************************************************************
Misc Support Functs
******************************************************************************
used by:
GTKObject
GTKWinAPI
GTKCallback
******************************************************************************
*****************************************************************************
This file is part of the Lazarus Component Library (LCL)
See the file COPYING.modifiedLGPL.txt, included in this distribution,
for details about the license.
*****************************************************************************
}
{off $DEFINE VerboseAccelerator}
{off $DEFINE VerboseUpdateSysColorMap}
{$IFOPT C-}
// Uncomment for local trace
//{$C+}
//{$DEFINE ASSERT_IS_ON}
{$ENDIF}
function gtk_widget_get_xthickness(Style : PGTKStyle) : gint;
begin
If (Style <> nil) then begin
{$IfNDef GTK2}
If (Style^.klass = nil) then
result := 0
else
{$EndIf}
result := Style^.{$IfNDef GTK2}klass^.{$EndIF}xthickness
end else
result := 0;
end;
function gtk_widget_get_ythickness(Style : PGTKStyle) : gint;
begin
If (Style <> nil) then begin
{$IfNDef GTK2}
If (Style^.klass = nil) then
result := 0
else
{$EndIf}
result := Style^.{$IfNDef GTK2}klass^.{$EndIF}ythickness
end else
result := 0;
end;
function gtk_widget_get_xthickness(Widget : PGTKWidget) : gint; overload;
begin
result := gtk_widget_get_xthickness(gtk_widget_get_style(Widget));
end;
function gtk_widget_get_ythickness(Widget : PGTKWidget) : gint; overload;
begin
result := gtk_widget_get_ythickness(gtk_widget_get_style(Widget));
end;
function GetGtkContainerBorderWidth(Widget: PGtkContainer): gint;
begin
Result:=(Widget^.flag0 and bm_TGtkContainer_border_width)
shr bp_TGtkContainer_border_width;
end;
procedure gdk_event_key_get_string(Event : PGDKEventKey; var theString : Pointer);
begin
{$IfDef GTK2}
theString := Pointer(Event^._String);
{$Else}
theString := Pointer(Event^.TheString);
{$EndIF}
end;
procedure gdk_event_key_set_string(Event: PGDKEventKey; const NewString: PChar);
var
OldString: PChar;
begin
{$IfDef GTK2}
OldString := Pointer(Event^._String);
{$Else}
OldString := Pointer(Event^.TheString);
{$EndIF}
// MG: should we set Event^.length := 0; or is this used for mem allocation?
if (OldString<>nil) then begin
if (NewString<>nil) then
OldString[0]:=NewString[0]
else
OldString[0]:=#0;
end;
end;
function gdk_event_get_type(Event : Pointer) : TGdkEventType;
begin
{$IfDef GTK2}
result := PGdkEvent(Event)^._type;
{$Else}
result := PGdkEvent(Event)^.TheType;
{$EndIF}
end;
procedure RememberKeyEventWasHandledByLCL(Event: PGdkEventKey;
BeforeEvent: boolean);
var
HandledEvent: TLCLHandledKeyEvent;
EventList: TFPList;
begin
if KeyEventWasHandledByLCL(Event,BeforeEvent) then exit;
if BeforeEvent then begin
if LCLHandledKeyEvents=nil then
LCLHandledKeyEvents:=TFPList.Create;
EventList:=LCLHandledKeyEvents;
end else begin
if LCLHandledKeyAfterEvents=nil then
LCLHandledKeyAfterEvents:=TFPList.Create;
EventList:=LCLHandledKeyAfterEvents;
end;
HandledEvent:=TLCLHandledKeyEvent.Create(Event);
EventList.Add(HandledEvent);
while EventList.Count>10 do begin
HandledEvent:=TLCLHandledKeyEvent(EventList[0]);
HandledEvent.Free;
EventList.Delete(0);
end;
end;
function KeyEventWasHandledByLCL(Event: PGdkEventKey; BeforeEvent: boolean
): boolean;
var
i: Integer;
HandledEvent: TLCLHandledKeyEvent;
EventList: TFPList;
begin
Result:=false;
if BeforeEvent then
EventList:=LCLHandledKeyEvents
else
EventList:=LCLHandledKeyAfterEvents;
if EventList=nil then exit;
for i:=0 to EventList.Count-1 do begin
HandledEvent:=TLCLHandledKeyEvent(EventList[i]);
if HandledEvent.IsEqual(Event) then begin
Result:=true;
exit;
end;
end;
end;
{$Ifdef GTK2}
function gtk_class_get_type(aclass : Pointer) : TGtkType;
begin
If (aclass <> nil) then
result := PGtkTypeClass(aclass)^.g_Type
else
result := 0;
end;
function gtk_object_get_class(anobject : Pointer) : Pointer;
begin
If (anobject <> nil) then
result := PGtkTypeObject(anobject)^.g_Class
else
result := nil;
end;
function gtk_window_get_modal(window:PGtkWindow):gboolean;
begin
if assigned(Window) then
result := GTK2.gtk_window_get_modal(window)
else
result := False;
end;
function gdk_region_union_with_rect(region:PGdkRegion; rect:PGdkRectangle) : PGdkRegion;
begin
result := gdk_region_copy(region);
GDK2.gdk_region_union_with_rect(result, rect);
end;
function gdk_region_intersect(source1:PGdkRegion; source2:PGdkRegion) : PGdkRegion;
begin
result := gdk_region_copy(source1);
GDK2.gdk_region_intersect(result, source2);
end;
function gdk_region_union(source1:PGdkRegion; source2:PGdkRegion) : PGdkRegion;
begin
result := gdk_region_copy(source1);
GDK2.gdk_region_union(result, source2);
end;
function gdk_region_subtract(source1:PGdkRegion; source2:PGdkRegion) : PGdkRegion;
begin
result := gdk_region_copy(source1);
GDK2.gdk_region_subtract(result, source2);
end;
function gdk_region_xor(source1:PGdkRegion; source2:PGdkRegion) : PGdkRegion;
begin
result := gdk_region_copy(source1);
GDK2.gdk_region_xor(result, source2);
end;
Procedure gdk_text_extents(TheFont: TGtkIntfFont;
Str: PChar; StrLength: integer;
lbearing, rbearing, width, ascent, descent: Pgint);
var
Layout : PPangoLayout;
Extents : TPangoRectangle;
begin
//DebugLn(['gdk_text_extents Str="',Str,'" StrLength=',StrLength,' lbearing=',lbearing<>nil,' rbearing=',rbearing<>Nil,' width=',width<>nil,' ascent=',ascent<>nil,' descent=',descent<>Nil,' ',TheFont<>Nil]);
Layout:=TheFont;
pango_layout_set_single_paragraph_mode(Layout, TRUE);
pango_layout_set_width(Layout, -1);
pango_layout_set_text(Layout, Str, StrLength);
if Assigned(width) then
pango_layout_get_pixel_size(Layout, width, nil);
if Assigned(lbearing) or Assigned(rbearing)
or Assigned(ascent) or Assigned(descent) then begin
pango_layout_get_extents(Layout, nil, @Extents);
if Assigned(lbearing) then
lbearing^ := PANGO_LBEARING(extents) div PANGO_SCALE;
if Assigned(rbearing) then
rBearing^ := PANGO_RBEARING(extents) div PANGO_SCALE;
if Assigned(ascent) then
ascent^ := PANGO_ASCENT(extents) div PANGO_SCALE;
if Assigned(descent) then
descent^ := PANGO_DESCENT(extents) div PANGO_SCALE;
end;
end;
{$EndIf Gtk2}
procedure BeginGDKErrorTrap;
begin
Inc(GdkTrapCalls);
if GdkTrapIsSet then
exit;
gdk_error_trap_push; //try to prevent GDK Bad Drawable/X Windows Errors
// from killing us...
{$IfDef GDK_ERROR_TRAP_FLUSH}
gdk_flush; //only for debugging purposes DO NOT enable by default.
// slows things down intolerably for actual use, if we ever
// have a real need for it, it should be called from that
// specific function, since this gets called constantly during
// drawing.
{$EndIf}
GdkTrapIsSet:=true;
end;
procedure EndGDKErrorTrap;
var
Xerror : gint;
begin
Dec(GdkTrapCalls);
if (not GdkTrapIsSet) then
RaiseGDBException('EndGDKErrorTrap without BeginGDKErrorTrap');
if (GdkTrapCalls > 0) then
exit;
Xerror := gdk_error_trap_pop;
GdkTrapIsSet:=false;
{$IFDEF VerboseGtkToDos}{$note TODO: enable standard error_log handling}{$ENDIF}
{$IfDef REPORT_GDK_ERRORS}
If (Xerror<>0) then
RaiseGDBException('A GDK/X Error occurred, this is normally fatal. The error code was: ' + IntToStr(Xerror));
{$EndIf}
end;
function dbgGRect(const ARect: PGDKRectangle): string;
begin
if ARect=nil then begin
Result:='nil';
end else begin
Result:='x='+dbgs(ARect^.x)+',y='+dbgs(ARect^.y)
+',w='+dbgs(ARect^.width)+',h='+dbgs(ARect^.height);
end;
end;
{------------------------------------------------------------------------------
Allocates a new PChar
------------------------------------------------------------------------------
function CreatePChar(const s: string): PChar;
begin
Result:=StrAlloc(length(s) + 1);
StrPCopy(Result, s);
end;
}
function FindChar(c: char; p:PChar; Max: integer): integer;
begin
Result:=0;
while (Result<Max) do begin
if p[Result]<>c then
inc(Result)
else
exit;
end;
Result:=-1;
end;
{------------------------------------------------------------------------------
function FindLineLen(p: PChar; Max: integer): integer;
Find line end
------------------------------------------------------------------------------}
function FindLineLen(p: PChar; Max: integer): integer;
begin
Result:=0;
while (Result<Max) do begin
if not (p[Result] in [#10,#13]) then
inc(Result)
else
exit;
end;
Result:=-1;
end;
function RectFromGdkRect(const AGdkRect: TGdkRectangle): TRect;
begin
with Result do
begin
Left := AGdkRect.x;
Top := AGdkRect.y;
Right := AGdkRect.Width + AGdkRect.x;
Bottom := AGdkRect.Height + AGdkRect.y;
end;
end;
function GdkRectFromRect(const R: TRect): TGdkRectangle;
begin
with Result do
begin
x := R.Left;
y := R.Top;
width := R.Right-R.Left;
height := R.Bottom-R.Top;
end;
end;
function AlignToGtkAlign(Align: TAlignment): gfloat;
begin
case Align of
taLeftJustify : AlignToGtkAlign := 0.0;
taCenter : AlignToGtkAlign := 0.5;
taRightJustify: AlignToGtkAlign := 1.0;
end;
end;
{$ifdef gtk2}
function GtkScrollTypeToScrollCode(ScrollType: TGtkScrollType): LongWord;
begin
case ScrollType of
GTK_SCROLL_NONE : Result := SB_ENDSCROLL;
GTK_SCROLL_JUMP : Result := SB_THUMBPOSITION;
GTK_SCROLL_STEP_BACKWARD : Result := SB_LINELEFT;
GTK_SCROLL_STEP_FORWARD : Result := SB_LINERIGHT;
GTK_SCROLL_PAGE_BACKWARD : Result := SB_PAGELEFT;
GTK_SCROLL_PAGE_FORWARD : Result := SB_PAGERIGHT;
GTK_SCROLL_STEP_UP : Result := SB_LINEUP;
GTK_SCROLL_STEP_DOWN : Result := SB_LINEDOWN;
GTK_SCROLL_PAGE_UP : Result := SB_PAGEUP;
GTK_SCROLL_PAGE_DOWN : Result := SB_PAGEDOWN;
GTK_SCROLL_STEP_LEFT : Result := SB_LINELEFT;
GTK_SCROLL_STEP_RIGHT : Result := SB_LINERIGHT;
GTK_SCROLL_PAGE_LEFT : Result := SB_PAGELEFT;
GTK_SCROLL_PAGE_RIGHT : Result := SB_PAGERIGHT;
GTK_SCROLL_START : Result := SB_TOP;
GTK_SCROLL_END : Result := SB_BOTTOM;
end;
end;
{$endif}
{------------------------------------------------------------------------------
function GtkWidgetIsA(Widget: PGtkWidget; AType: TGtkType): boolean;
The GTK_IS_XXX macro functions in the fpc gtk1.x bindings are not correct.
They just test the highest level.
This function checks as the real C macros.
------------------------------------------------------------------------------}
function GtkWidgetIsA(Widget: PGtkWidget; AType: TGtkType): boolean;
begin
Result:=(Widget<>nil)
and (gtk_object_get_class(Widget)<>nil)
and gtk_type_is_a(gtk_class_get_type(gtk_object_get_class(Widget)), AType);
end;
{------------------------------------------------------------------------------
function GetWidgetClassName(Widget: PGtkWidget): string;
Returns the gtk class name of Widget.
------------------------------------------------------------------------------}
function GetWidgetClassName(Widget: PGtkWidget): string;
var
AType: TGtkType;
ClassPGChar: Pgchar;
ClassLen: Integer;
begin
Result:='';
if Widget=nil then begin
Result:='nil';
exit;
end;
if (gtk_object_get_class(Widget)=nil) then begin
Result:='<Widget without class>';
exit;
end;
AType:=gtk_class_get_type(gtk_object_get_class(Widget));
ClassPGChar:=gtk_type_name(AType);
if ClassPGChar=nil then begin
Result:='<Widget without classname>';
exit;
end;
ClassLen:=strlen(ClassPGChar);
SetLength(Result,ClassLen);
if ClassLen>0 then
Move(ClassPGChar[0],Result[1],ClassLen);
end;
function GetWidgetDebugReport(Widget: PGtkWidget): string;
var
LCLObject: TObject;
AWinControl: TWinControl;
MainWidget: PGtkWidget;
WinWidgetInfo: PWinWidgetInfo;
FixedWidget: PGTKWidget;
begin
if Widget = nil
then begin
Result := 'nil';
exit;
end;
Result := Format('%p=%s %s', [Pointer(Widget), GetWidgetClassName(Widget), WidgetFlagsToString(Widget)]);
LCLObject:=GetNearestLCLObject(Widget);
Result := Result + Format(' LCLObject=%p', [Pointer(LCLObject)]);
if LCLObject=nil then exit;
if LCLObject is TControl then
Result:=Result+'='+TControl(LCLObject).Name+':'+LCLObject.ClassName
else
Result:=Result+'='+LCLObject.ClassName;
if LCLObject is TWinControl then begin
AWinControl:=TWinControl(LCLObject);
if AWinControl.HandleAllocated then begin
MainWidget:=PGTKWidget(AWinControl.Handle);
if MainWidget=Widget
then Result:=Result+'<Is MainWidget>'
else Result:=Result+Format('<MainWidget=%p=%s>', [Pointer(MainWidget), GetWidgetClassName(MainWidget)]);
FixedWidget:=GetFixedWidget(MainWidget);
if FixedWidget=Widget then
Result:=Result+'<Is FixedWidget>';
WinWidgetInfo:=GetWidgetInfo(MainWidget,false);
if WinWidgetInfo<>nil then begin
if WinWidgetInfo^.CoreWidget = Widget then
Result:=Result+'<Is CoreWidget>';
end;
end
else begin
Result:=Result+'<NOT HandleAllocated>'
end;
end;
end;
function GetWindowDebugReport(AWindow: PGDKWindow): string;
var
p: gpointer;
Widget: PGtkWidget;
WindowType: TGdkWindowType;
Width: Integer;
Height: Integer;
{$ifdef gtk1}
Visual: PGdkVisual;
{$endif}
TypeAsStr: String;
begin
Result := DbgS(AWindow);
if AWindow = nil then Exit;
// window type
WindowType := gdk_window_get_type(AWindow);
case WindowType of
GDK_WINDOW_ROOT: TypeAsStr := 'Root';
GDK_WINDOW_TOPLEVEL: TypeAsStr := 'TopLvl';
GDK_WINDOW_CHILD: TypeAsStr := 'Child';
GDK_WINDOW_DIALOG: TypeAsStr := 'Dialog';
GDK_WINDOW_TEMP: TypeAsStr := 'Temp';
{$ifdef gtk1}
GDK_WINDOW_PIXMAP: TypeAsStr := 'Pixmap';
{$endif gtk1}
GDK_WINDOW_FOREIGN: TypeAsStr := 'Foreign';
else
TypeAsStr := 'Unknown';
end;
Result:=Result + ' Type=' + TypeAsStr;
DebugLn(Result);
// user data
if WindowType in [GDK_WINDOW_ROOT,GDK_WINDOW_TOPLEVEL,GDK_WINDOW_CHILD, GDK_WINDOW_DIALOG] then
begin
p := nil;
gdk_window_get_user_data(AWindow, @p);
if GtkWidgetIsA(PGTKWidget(p), gtk_widget_get_type) then
begin
Widget := PGTKWidget(p);
Result := Result + '<Widget[' + GetWidgetDebugReport(Widget) + ']>';
end
else
Result := Result + '<UserData=' + DbgS(p) + ']>';
end;
// size
gdk_window_get_size(AWindow, @Width, @Height);
Result := Result + ' Size=' + IntToStr(Width) + 'x' + IntToStr(Height);
{$ifdef gtk1}
// visual
Visual := gdk_window_get_visual(AWindow);
if Visual <> nil then
if WindowType in [GDK_WINDOW_PIXMAP] then
Result := Result + ' Depth=' + IntToStr(Visual^.bits_per_rgb);
{$endif gtk1}
end;
function GetStyleDebugReport(AStyle: PGTKStyle): string;
begin
Result:='[';
if AStyle=nil then
Result:=Result+'nil'
else begin
Result:=Result+'FG[N]:='+GdkColorToStr(@AStyle^.fg[GTK_STATE_NORMAL])+' ';
Result:=Result+'BG[N]:='+GdkColorToStr(@AStyle^.bg[GTK_STATE_NORMAL])+' ';
Result:=Result+'Base[N]:='+GdkColorToStr(@AStyle^.base[GTK_STATE_NORMAL])+' ';
Result:=Result+'BG_Pixmap[N]:='+DbgS(AStyle^.bg_pixmap[GTK_STATE_NORMAL])+' ';
Result:=Result+'rc_style='+GetRCStyleDebugReport(AStyle^.rc_style);
end;
Result:=Result+']';
end;
function GetRCStyleDebugReport(AStyle: PGtkRcStyle): string;
begin
Result:='[';
if AStyle=nil then
Result:=Result+'nil'
else begin
Result:=Result+'name="'+AStyle^.name+'" ';
{$IFDEF GTK1}
Result:=Result+'font_name="'+AStyle^.font_name+'" ';
Result:=Result+'fontset_name="'+AStyle^.fontset_name+'" ';
Result:=Result+'FG[N]='+GdkColorToStr(@AStyle^.fg[GTK_STATE_NORMAL])+' ';
Result:=Result+'BG[N]='+GdkColorToStr(@AStyle^.bg[GTK_STATE_NORMAL])+' ';
Result:=Result+'Base[N]='+GdkColorToStr(@AStyle^.base[GTK_STATE_NORMAL])+' ';
Result:=Result+'flagi='+intTostr(AStyle^.color_flags[GTK_STATE_NORMAL])+' ';
{$ELSE GTK2}
Result:=Result+'font_desc=['+GetPangoDescriptionReport(AStyle^.font_desc)+'] ';
{$ENDIF GTK2}
Result:=Result+'bg_pixmap_name[N]="'+AStyle^.bg_pixmap_name[GTK_STATE_NORMAL]+'" ';
{$IFDEF GTK1}
Result:=Result+'engine='+DbgS(AStyle^.engine);
{$ENDIF GTK1}
end;
Result:=Result+']';
end;
{$IFDEF Gtk2}
function GetPangoDescriptionReport(Desc: PPangoFontDescription): string;
begin
if Desc=nil then begin
Result:='nil';
end else begin
Result:='family='+pango_font_description_get_family(Desc);
Result:=Result+' size='+IntToStr(pango_font_description_get_size(Desc));
Result:=Result+' weight='+IntToStr(pango_font_description_get_weight(Desc));
Result:=Result+' variant='+IntToStr(pango_font_description_get_variant(Desc));
Result:=Result+' style='+IntToStr(pango_font_description_get_style(Desc));
Result:=Result+' stretch='+IntToStr(pango_font_description_get_stretch(Desc));
end;
end;
{$ENDIF}
function WidgetFlagsToString(Widget: PGtkWidget): string;
begin
Result:='[';
if Widget=nil then
Result:=Result+'nil'
else begin
if GTK_WIDGET_REALIZED(Widget) then
Result:=Result+'R';
if GTK_WIDGET_MAPPED(Widget) then
Result:=Result+'M';
if GTK_WIDGET_VISIBLE(Widget) then
Result:=Result+'V';
if GTK_WIDGET_DRAWABLE(Widget) then
Result:=Result+'D';
if GTK_WIDGET_CAN_FOCUS(Widget) then
Result:=Result+'F';
if GTK_WIDGET_RC_STYLE(Widget) then
Result:=Result+'St';
if GTK_WIDGET_PARENT_SENSITIVE(Widget) then
Result:=Result+'Pr';
{$IFDEF Gtk2}
if GTK_WIDGET_NO_WINDOW(Widget) then
Result:=Result+'Nw';
if GTK_WIDGET_COMPOSITE_CHILD(Widget) then
Result:=Result+'Cc';
if GTK_WIDGET_APP_PAINTABLE(Widget) then
Result:=Result+'Ap';
if GTK_WIDGET_DOUBLE_BUFFERED(Widget) then
Result:=Result+'Db';
{$ENDIF}
end;
Result:=Result+']';
end;
function GdkColorToStr(Color: PGDKColor): string;
begin
if Color=nil then
Result:='nil'
else
Result:='R'+HexStr(Color^.Red,4)+'G'+HexStr(Color^.Green,4)
+'B'+HexStr(Color^.Blue,4);
end;
function GetWidgetStyleReport(Widget: PGtkWidget): string;
var
AStyle: PGtkStyle;
ARCStyle: PGtkRcStyle;
begin
Result:='';
if Widget=nil then exit;
AStyle:=gtk_widget_get_style(Widget);
if AStyle=nil then begin
Result:='nil';
exit;
end;
Result:=Result+'attach_count='+dbgs(AStyle^.attach_count);
ARCStyle:=AStyle^.rc_style;
if ARCStyle=nil then begin
Result:=Result+' rc_style=nil';
end else begin
Result:=Result+' rc_style=[';
{$IFDEF GTK1}
Result:=Result+ARCStyle^.font_name+',';
Result:=Result+ARCStyle^.fontset_name+',';
{$ELSE GTK1}
Result:=Result+GetPangoDescriptionReport(AStyle^.font_desc);
{$ENDIF GTK1}
Result:=Result+']';
end;
end;
{------------------------------------------------------------------------------
function WidgetIsDestroyingHandle(Widget: PGtkWidget): boolean;
Tests if Destruction Mark is set.
------------------------------------------------------------------------------}
function WidgetIsDestroyingHandle(Widget: PGtkWidget): boolean;
begin
Result:=gtk_object_get_data(PGtkObject(Widget),'LCLDestroyingHandle')<>nil;
end;
{------------------------------------------------------------------------------
procedure SetWidgetIsDestroyingHandle(Widget: PGtkWidget);
Marks widget for destruction.
------------------------------------------------------------------------------}
procedure SetWidgetIsDestroyingHandle(Widget: PGtkWidget);
begin
gtk_object_set_data(PGtkObject(Widget),'LCLDestroyingHandle',Widget);
end;
{------------------------------------------------------------------------------
function ComponentIsDestroyingHandle(AWinControl: TWinControl): boolean;
Tests if Destruction Mark is set.
------------------------------------------------------------------------------}
function ComponentIsDestroyingHandle(AWinControl: TWinControl): boolean;
begin
Result:=
(AWinControl<>nil) and (AWinControl is TWinControl)
and (AWinControl.HandleAllocated)
and WidgetIsDestroyingHandle(PGtkWidget(AWinControl.Handle));
end;
{------------------------------------------------------------------------------
function LockOnChange(GtkObject: PGtkObject; LockOffset: integer): integer;
Adds LockOffset to the OnChangeLock and returns the result.
------------------------------------------------------------------------------}
function LockOnChange(GtkObject: PGtkObject; LockOffset: integer): integer;
var
Info: PWidgetInfo;
begin
Info := GetWidgetInfo(GtkObject, True);
if Info = nil
then begin
Result := 0;
Exit;
end;
Inc(Info^.ChangeLock, LockOffset);
Result := Info^.ChangeLock;
end;
procedure SetFormShowInTaskbar(AForm: TCustomForm;
const AValue: TShowInTaskbar);
var
Enable: boolean;
Widget: PGtkWidget;
begin
if (AForm.Parent <> nil) or
(AForm.ParentWindow <> 0) or
not (AForm.HandleAllocated) then Exit;
Widget := PGtkWidget(AForm.Handle);
// if widget not yet realized then exit
if Widget^.Window = nil then
Exit;
Enable := AValue <> stNever;
{if (AValue = stDefault)
and (Application<>nil) and (Application.MainForm <> nil)
and (Application.MainForm <> AForm) then
Enable := false;}
//debugln('SetGtkWindowShowInTaskbar ',DbgSName(AForm),' ',dbgs(Enable));
// The button reappears in some (still unknown) situations, but has the
//'skip-taskbar-hint' property still set to True, so invoking the function
//doesn't have an effect. Resetting the property makes it work.
{$IFNDEF GTK1}
if (not Enable) and gtk_window_get_skip_taskbar_hint(PGtkWindow(Widget)) then
gtk_window_set_skip_taskbar_hint(PGtkWindow(Widget), False);
{$ENDIF}
SetGtkWindowShowInTaskbar(PGtkWindow(Widget), Enable);
end;
procedure SetGtkWindowShowInTaskbar(AGtkWindow: PGtkWindow; Value: boolean);
begin
{$IFDEF GTK1}
if PgtkWidget(AGtkWindow)^.Window=nil then begin
// widget not yet realized
exit;
end;
GDK_WINDOW_SHOW_IN_TASKBAR(PGdkWindowPrivate(PGtkWidget(AGtkWindow)^.Window),
Value);
{$ELSE}
//DebugLn(['SetGtkWindowShowInTaskbar ',GetWidgetDebugReport(PGtkWidget(AGtkWindow)),' ',Value]);
gtk_window_set_skip_taskbar_hint(AGtkWindow, not Value);
{$ENDIF}
end;
procedure SetWindowFullScreen(AForm: TCustomForm; const AValue: Boolean);
{$IFDEF GTK1}
var
XDisplay: PDisplay;
XScreen: PScreen;
XRootWindow,
XWindow: TWindow;
XEvent: TXClientMessageEvent;
_NET_WM_STATE: Integer;
//_NET_WM_STATE_MODAL: Integer;
//_NET_WM_STATE_ABOVE: Integer;
//_NET_WM_STATE_FULLSCREEN: Integer;
_NET_WM_STATE_ATOMS: array [0..2] of Integer;
I: Integer;
{$ENDIF}
begin
{$IFDEF GTK2}
If AValue then
GTK_Window_FullScreen(PGTKWindow(AForm.Handle))
else
GTK_Window_UnFullScreen(PGTKWindow(AForm.Handle));
{$ENDIF}
{$IFDEF GTK1}
XDisplay := gdk_display;
XScreen := XDefaultScreenOfDisplay(xdisplay);
XRootWindow := XRootWindowOfScreen(xscreen);
XWindow := FormToX11Window(AForm);
_NET_WM_STATE := XInternAtom(xdisplay, '_NET_WM_STATE', false);
//_NET_WM_STATE_MODAL := XInternAtom(xdisplay, '_NET_WM_STATE_MODAL', false);
//_NET_WM_STATE_ABOVE := XInternAtom(xdisplay, '_NET_WM_STATE_ABOVE', false);
//_NET_WM_STATE_FULLSCREEN := XInternAtom(xdisplay, '_NET_WM_STATE_FULLSCREEN', false);
_NET_WM_STATE_ATOMS[0] := XInternAtom(xdisplay, '_NET_WM_STATE_MODAL', false);
_NET_WM_STATE_ATOMS[1] := XInternAtom(xdisplay, '_NET_WM_STATE_ABOVE', false);
_NET_WM_STATE_ATOMS[2] := XInternAtom(xdisplay, '_NET_WM_STATE_FULLSCREEN', false);
for I := 0 to 2 do begin
XEvent._type := ClientMessage;
XEvent.window := XWindow;
XEvent.message_type := _NET_WM_STATE;
XEvent.format := 32;
XEvent.data.l[0] := Ord(AValue);// 0=Remove 1=Add 2=Toggle
XEvent.data.l[1] := _NET_WM_STATE_ATOMS[I];
XSendEvent(XDisplay, XRootWindow, False, SubstructureNotifyMask, PXEvent(@XEvent));
end;
{$ENDIF}
end;
procedure GrabKeyBoardToForm(AForm: TCustomForm);
begin
{$IFDEF HasX}
XGrabKeyboard(gdk_display, FormToX11Window(AForm), true, GrabModeASync,
GrabModeASync, CurrentTime);
{$ENDIF}
end;
procedure ReleaseKeyBoardFromForm(AForm: TCustomForm);
begin
{$IFDEF HasX}
XUngrabKeyboard(gdk_display, CurrentTime);
{$ENDIF}
end;
procedure GrabMouseToForm(AForm: TCustomForm);
{$IFDEF HasX}
var
eventMask: LongInt;
begin
eventMask := ButtonPressMask or ButtonReleaseMask
or PointerMotionMask or PointerMotionHintMask;
XGrabPointer(gdk_display, FormToX11Window(AForm), true,
eventMask, GrabModeASync, GrabModeAsync, FormToX11Window(AForm),
None, CurrentTime);
end;
{$ELSE}
begin
end;
{$ENDIF}
procedure ReleaseMouseFromForm(AForm: TCustomForm);
begin
{$IFDEF HasX}
XUngrabPointer(gdk_display, CurrentTime);
{$ENDIF}
end;
{$IFDEF HasX}
function FormToX11Window(const AForm: TCustomForm): X.TWindow;
var
Widget: PGtkWidget;
begin
Result:=0;
if (AForm=nil) or (not AForm.HandleAllocated) then exit;
Widget:=PGtkWidget(AForm.Handle);
if Widget^.window = nil then exit;
{$ifdef gtk1}
Result := PGdkWindowPrivate(Widget^.window)^.xwindow;
{$else}
Result := gdk_window_xwindow(Widget^.window);
{$endif}
end;
{$ENDIF}
procedure SetLabelAlignment(LabelWidget: PGtkLabel;
const NewAlignment: TAlignment);
const
cLabelAlignX : array[TAlignment] of gfloat = (0.0, 1.0, 0.5);
cLabelAlignY : array[TTextLayout] of gfloat = (0.0, 0.5, 1.0);
cLabelAlign : array[TAlignment] of TGtkJustification =
(GTK_JUSTIFY_LEFT, GTK_JUSTIFY_RIGHT, GTK_JUSTIFY_CENTER);
begin
gtk_label_set_justify(LabelWidget, cLabelAlign[NewAlignment]);
gtk_misc_set_alignment(GTK_MISC(LabelWidget), cLabelAlignX[NewAlignment],
cLabelAlignY[tlTop]);
end;
{------------------------------------------------------------------------------
function GtkPaintMessageToPaintMessage(var GtkPaintMsg: TLMGtkPaint;
FreeGtkPaintMsg: boolean): TLMPaint;
Converts a LM_GTKPAINT message to a LM_PAINT message
------------------------------------------------------------------------------}
function GtkPaintMessageToPaintMessage(var GtkPaintMsg: TLMGtkPaint;
FreeGtkPaintMsg: boolean): TLMPaint;
var
PS : PPaintStruct;
Widget: PGtkWidget;
begin
FillByte(Result,SizeOf(Result),0);
Result.Msg := LM_PAINT;
New(PS);
FillChar(PS^, SizeOf(TPaintStruct), 0);
Widget := GtkPaintMsg.Data.Widget;
If GtkPaintMsg.Data.RepaintAll then
PS^.rcPaint := Rect(0, 0, Widget^.Allocation.Width, Widget^.Allocation.Height)
else
PS^.rcPaint := GtkPaintMsg.Data.Rect;
Result.DC := BeginPaint(HWND(Widget), PS^);
Result.PaintStruct := PS;
Result.Result := 0;
if FreeGtkPaintMsg then
FreeThenNil(GtkPaintMsg.Data);
end;
procedure FinalizePaintMessage(Msg: PLMessage);
var
PS: PPaintStruct;
DC: TGtkDeviceContext;
begin
if (Msg^.Msg = LM_PAINT) then
begin
if Msg^.LParam <> 0 then
begin
PS := PPaintStruct(Msg^.LParam);
if Msg^.WParam <> 0 then
DC := TGtkDeviceContext(Msg^.WParam)
else
DC := TGtkDeviceContext(PS^.hdc);
EndPaint(HWND(DC.Widget), PS^);
Dispose(PS);
Msg^.LParam:=0;
Msg^.WParam:=0;
end
else
if Msg^.WParam<>0 then
begin
ReleaseDC(0, Msg^.WParam);
Msg^.WParam := 0;
end;
end else
if Msg^.Msg = LM_GTKPAINT then
FreeThenNil(TLMGtkPaintData(Msg^.WParam));
end;
procedure FinalizePaintTagMsg(Msg: PMsg);
var
PS: PPaintStruct;
DC: TGtkDeviceContext;
begin
if (Msg^.Message = LM_PAINT) then
begin
if Msg^.LParam <> 0 then
begin
PS := PPaintStruct(Msg^.LParam);
if Msg^.WParam<>0 then
DC := TGtkDeviceContext(Msg^.WParam)
else
DC := TGtkDeviceContext(PS^.hdc);
EndPaint(HWND(DC.Widget), PS^);
Dispose(PS);
Msg^.LParam:=0;
Msg^.WParam:=0;
end else
if Msg^.WParam<>0 then
begin
ReleaseDC(0, Msg^.WParam);
Msg^.WParam:=0;
end;
end else
if Msg^.Message = LM_GTKPAINT then
FreeThenNil(TObject(Msg^.WParam));
end;
procedure SetGCRasterOperation(TheGC: PGDKGC; Rop: Cardinal);
begin
case ROP of
WHITENESS,
BLACKNESS,
SRCCOPY :
gdk_gc_set_function(TheGC, GDK_Copy);
SRCPAINT :
gdk_gc_set_function(TheGC, GDK_NOOP);
SRCAND :
gdk_gc_set_function(TheGC, GDK_Clear);
SRCINVERT :
gdk_gc_set_function(TheGC, GDK_XOR);
SRCERASE :
gdk_gc_set_function(TheGC, GDK_AND);
NOTSRCCOPY :
gdk_gc_set_function(TheGC, GDK_OR_REVERSE);
NOTSRCERASE :
gdk_gc_set_function(TheGC, GDK_AND);
MERGEPAINT :
gdk_gc_set_function(TheGC, GDK_Copy_Invert);
DSTINVERT :
gdk_gc_set_function(TheGC, GDK_INVERT);
else begin
gdk_gc_set_function(TheGC, GDK_COPY);
DebugLn('WARNING: [SetRasterOperation] Got unknown/unsupported CopyMode!!');
end;
end;
end;
procedure MergeClipping(DestinationDC: TGtkDeviceContext; DestinationGC: PGDKGC;
X,Y,Width,Height: integer; ClipMergeMask: PGdkBitmap;
ClipMergeMaskX, ClipMergeMaskY: integer;
var NewClipMask: PGdkBitmap);
// merge ClipMergeMask into the destination clipping mask at the
// destination rectangle
var
temp_gc : PGDKGC;
temp_color : TGDKColor;
RGNType : Longint;
OffsetXY: TPoint;
//ClipMergeMaskWidth, ClipMergeMaskHeight: integer;
begin
{$IFDEF VerboseStretchCopyArea}
DebugLn('MergeClipping START DestinationDC=',DbgS(DestinationDC),
' DestinationGC=',DbgS(DestinationGC),
' X='+dbgs(X),' Y='+dbgs(Y),' Width='+dbgs(Width),' Height='+dbgs(Height),
' ClipMergeMask=',DbgS(ClipMergeMask),
' ClipMergeMaskX=',dbgs(ClipMergeMaskX),' ClipMergeMaskY=',dbgs(ClipMergeMaskY));
{$ENDIF}
// activate clipping region of destination
DestinationDC.SelectRegion;
NewClipMask := nil;
if (ClipMergeMask = nil) then exit;
BeginGDKErrorTrap;
// create temporary mask with the size of the destination rectangle
NewClipMask := PGdkBitmap(gdk_pixmap_new(nil, width, height, 1));
// create temporary GC for combination mask
temp_gc := gdk_gc_new(NewClipMask);
gdk_gc_set_clip_region(temp_gc, nil); // no default clipping
gdk_gc_set_clip_rectangle(temp_gc, nil);
// clear mask
temp_color.pixel := 0;
gdk_gc_set_foreground(temp_gc, @temp_color);
gdk_draw_rectangle(NewClipMask, temp_gc, 1, 0, 0, width+1, height+1);
// copy the destination clipping mask into the temporary mask
with DestinationDC do begin
If (ClipRegion <> nil) then begin
RGNType := RegionType(ClipRegion^.GDIRegionObject);
If (RGNType <> ERROR) and (RGNType <> NULLREGION) then begin
// destination has a clipping mask
{$IFDEF VerboseStretchCopyArea}
DebugLn('MergeClipping Destination has clipping mask -> apply to temp GC');
{$ENDIF}
// -> copy the destination clipping mask to the temporary mask
// The X,Y coordinate in the destination relates to
// 0,0 in the temporary mask.
// The clip region of dest is always at 0,0 in dest
OffsetXY:=Point(-X,-Y);
// 1. Move the region
gdk_region_offset(ClipRegion^.GDIRegionObject,OffsetXY.X,OffsetXY.Y);
// 2. Apply region to temporary mask
gdk_gc_set_clip_region(temp_gc, ClipRegion^.GDIRegionObject);
// 3. Undo moving the region
gdk_region_offset(ClipRegion^.GDIRegionObject,-OffsetXY.X,-OffsetXY.Y);
end;
end;
end;
// merge the source clipping mask into the temporary mask
//gdk_window_get_size(ClipMergeMask,@ClipMergeMaskWidth,@ClipMergeMaskHeight);
//DebugLn('MergeClipping A MergeMask Size=',ClipMergeMaskWidth,',',ClipMergeMaskHeight);
gdk_draw_pixmap(NewClipMask, temp_gc,
ClipMergeMask, ClipMergeMaskX, ClipMergeMaskY, 0, 0, -1, -1);
// free the temporary GC
gdk_gc_destroy(temp_gc);
// apply the new mask to the destination GC
// The new mask has only the size of the destination rectangle, not of
// the whole destination. Apply it to destination and move it to the right
// position.
gdk_gc_set_clip_mask(DestinationGC, NewClipMask);
gdk_gc_set_clip_origin(DestinationGC, x, y);
EndGDKErrorTrap;
end;
function CreatePixbufFromImageAndMask(ASrc: PGdkDrawable; ASrcX, ASrcY, ASrcWidth,
ASrcHeight: integer; ASrcColorMap: PGdkColormap; ASrcMask: PGdkBitmap): PGdkPixbuf;
procedure Warn(const AText: String);
begin
DebugLn('[WARNING] ScalePixmapAndMask: ' + AText);
end;
procedure ApplyMask(APixels, AMask: pguchar);
type
TPixbufPixel = record
R,G,B,A: Byte;
end;
var
RGBA: ^TPixbufPixel absolute APixels;
Mask: ^TPixbufPixel absolute AMask;
n: Integer;
begin
for n := 0 to (ASrcHeight * ASrcWidth) - 1 do
begin
if (Mask^.B = 0) and (Mask^.G = 0) and (Mask^.R = 0)
then RGBA^.A := 0;
inc(RGBA);
inc(Mask);
end;
end;
var
Msk: PGdkPixbuf;
FullSrcWidth, FullSrcHeight: integer;
begin
Result := nil;
if ASrc = nil then Exit;
gdk_window_get_size(PGDKWindow(ASrc), @FullSrcWidth, @FullSrcHeight);
if ASrcX + ASrcWidth > FullSrcWidth
then begin
Warn('ASrcX+ASrcWidth>FullSrcWidth');
end;
if ASrcY + ASrcHeight > FullSrcHeight
then begin
Warn('ASrcY+ASrcHeight>FullSrcHeight');
end;
// Creating PixBuf from pixmap
Result := CreatePixbufFromDrawable(ASrc, ASrcColorMap, ASrcMask <> nil, ASrcX, ASrcY, 0, 0, ASrcWidth, ASrcHeight);
if Result = nil
then begin
Warn('Result=nil');
Exit;
end;
//DbgDumpPixbuf(Result, 'Pixbuf from Source');
// Apply mask if present
if ASrcMask <> nil
then begin
if gdk_pixbuf_get_rowstride(Result) <> ASrcWidth shl 2
then begin
Warn('rowstride <> 4*width');
gdk_pixbuf_unref(Result);
Result := nil;
Exit;
end;
Msk := CreatePixbufFromDrawable(ASrcMask, nil, True, ASrcX, ASrcY, 0, 0, ASrcWidth, ASrcHeight);
ApplyMask(gdk_pixbuf_get_pixels(Result), gdk_pixbuf_get_pixels(Msk));
gdk_pixbuf_unref(Msk);
end;
end;
function ScalePixmapAndMask(AScaleGC: PGDKGC; AScaleMethod: TGdkInterpType;
ASrc: PGdkPixmap; ASrcX, ASrcY, ASrcWidth, ASrcHeight: integer;
ASrcColorMap: PGdkColormap; ASrcMask: PGdkBitmap;
ADstWidth, ADstHeight: Integer; FlipHorz, FlipVert: Boolean;
out ADst, ADstMask: PGdkPixmap) : Boolean;
procedure Warn(const AText: String);
begin
DebugLn('[WARNING] ScalePixmapAndMask: ' + AText);
end;
var
ScaleSrc, ScaleDst: PGdkPixbuf;
begin
Result := False;
ADst:=nil;
ADstMask:=nil;
// Creating PixBuf from pixmap
ScaleSrc := CreatePixbufFromImageAndMask(ASrc, ASrcX, ASrcY, ASrcWidth, ASrcHeight,
ASrcColorMap, ASrcMask);
// Scaling PixBuf
ScaleDst := gdk_pixbuf_scale_simple(ScaleSrc, ADstWidth, ADstHeight, AScaleMethod);
gdk_pixbuf_unref(ScaleSrc);
if ScaleDst = nil
then begin
Warn('ScaleDst=nil');
exit;
end;
// flip if needed
if FlipHorz then
begin
{$IFNDEF GTK1}
ScaleSrc := ScaleDst;
ScaleDst := gdk_pixbuf_flip(ScaleSrc, True);
gdk_pixbuf_unref(ScaleSrc);
if ScaleDst = nil
then begin
Warn('ScaleDst=nil');
exit;
end;
{$ELSE}
// TODO: implement flipping for gtk1
{$ENDIF}
end;
if FlipVert then
begin
{$IFNDEF GTK1}
ScaleSrc := ScaleDst;
ScaleDst := gdk_pixbuf_flip(ScaleSrc, False);
gdk_pixbuf_unref(ScaleSrc);
if ScaleDst = nil
then begin
Warn('ScaleDst=nil');
exit;
end;
{$ELSE}
// TODO: implement flipping for gtk1
{$ENDIF}
end;
// BeginGDKErrorTrap;
// Creating pixmap from scaled pixbuf
gdk_pixbuf_render_pixmap_and_mask(ScaleDst, ADst, ADstMask, $80);
// EndGDKErrorTrap;
gdk_pixbuf_unref(ScaleDst);
Result := True;
end;
{$IFDEF VerboseGtkToDos}{$note remove when gtk native imagelist will be ready}{$ENDIF}
procedure DrawImageListIconOnWidget(ImgList: TCustomImageList;
Index: integer; AEffect: TGraphicsDrawEffect; DestWidget: PGTKWidget;
CenterHorizontally, CenterVertically: boolean;
DestLeft, DestTop: integer);
// draw icon of imagelist centered on gdkwindow
var
Bitmap: TBitmap;
ImageWidth: Integer;
ImageHeight: Integer;
WindowWidth, WindowHeight: integer;
DestDC: HDC;
Offset: TPoint;
{$ifdef gtk2}
FixedWidget: PGtkWidget;
{$ENDIF}
begin
if ImgList=nil then exit;
if (Index<0) or (Index>=ImgList.Count) then exit;
if (DestWidget=nil) then exit;
ImageWidth:=ImgList.Width;
ImageHeight:=ImgList.Height;
Bitmap := TBitmap.Create;
ImgList.GetBitmap(Index, Bitmap, AEffect);
if (ImageWidth<1) or (ImageHeight<1) then exit;
WindowWidth := DestWidget^.allocation.width;
WindowHeight := DestWidget^.allocation.height;
Offset := Point(0, 0);
{$ifdef gtk2}
// if our widget is placed on non-window fixed then we should substract its allocation here
// since in GetDC we will get this difference in offset
FixedWidget := GetFixedWidget(DestWidget);
if (FixedWidget <> nil) and GTK_WIDGET_NO_WINDOW(FixedWidget) then
Offset := Point(FixedWidget^.allocation.x, FixedWidget^.allocation.y);
{$endif}
if CenterHorizontally then
DestLeft := DestWidget^.allocation.x - Offset.x + ((WindowWidth-ImageWidth) div 2);
if CenterVertically then
DestTop := DestWidget^.allocation.y - Offset.y + ((WindowHeight-ImageHeight) div 2);
DestDC := GetDC(HDC(PtrUInt(DestWidget)));
//DebugLn('DrawImageListIconOnWidget B DestXY=',DestLeft,',',DestTop,
// ' DestWindowSize=',WindowWidth,',',WindowWidth,
// ' SrcRect=',ImageRect.Left,',',ImageRect.Top,',',ImageWidth,'x',ImageHeight);
StretchBlt(DestDC, DestLeft, DestTop, ImageWidth, ImageHeight,
Bitmap.Canvas.Handle, 0, 0, ImageWidth, ImageHeight,
SRCCOPY);
ReleaseDC(HDC(PtrUInt(DestWidget)),DestDC);
Bitmap.Free;
end;
procedure DrawImageListIconOnWidget(ImgList: TCustomImageList;
Index: integer; DestWidget: PGTKWidget);
begin
DrawImageListIconOnWidget(ImgList, Index, gdeNormal, DestWidget, true, true, 0, 0);
end;
function GetGdkImageBitsPerPixel(Image: PGdkImage): cardinal;
begin
Result:=Image^.bpp;
if Result<Image^.Depth then
Result:=Result*8;
end;
{------------------------------------------------------------------------------
Function: CreateGtkBitmapMask
Params: AImageMask: Then internal gtkBitmap for imagemask
AMask: External gtkbitmap
Returns: A GdkBitmap
This function returns a bitmap based on the internal alpha bitmap and the
maskhandle passed.
If both internal mask and the given mask is valid, then a new bitmap is created
else either internal mask or given mask (with increased reference)
------------------------------------------------------------------------------}
function CreateGdkMaskBitmap(AImageMask, AMask: PGdkBitmap): PGdkBitmap;
var
W, H: Integer;
GC: PGdkGc;
begin
Result := nil;
if (AImageMask = nil) and (AMask = nil) then Exit;
if AMask = nil
then begin
Result := AImageMask;
gdk_pixmap_ref(Result);
Exit;
end;
if AImageMask = nil
then begin
Result := AMask;
gdk_pixmap_ref(Result);
Exit;
end;
// if we are here we need a combination (=AND) of both masks
gdk_window_get_size(AImageMask, @W, @H);
Result := gdk_pixmap_new(nil, W, H, 1);
GC := gdk_gc_new(Result);
// copy image mask
gdk_draw_pixmap(Result, GC, AImageMask, 0, 0, 0, 0, -1, -1);
// and with mask
gdk_gc_set_function(GC, GDK_AND);
gdk_draw_pixmap(Result, GC, AMask, 0, 0, 0, 0, -1, -1);
gdk_gc_unref(GC);
end;
{------------------------------------------------------------------------------
Function: CreateGdkMaskBitmap
Params: AImage: Handle to the (LCL) bitmap image
AMask: Handle to the (LCL) bitmap mask
Returns: A GdkBitmap
This function returns a bitmap based on the internal alpha bitmap of the
image handle and the maskhandle passed.
If only internal mask is valid, then that one is returned (with increased reference)
Otherwise a new bitmap is created.
------------------------------------------------------------------------------}
function CreateGdkMaskBitmap(AImage, AMask: HBITMAP): PGdkBitmap;
var
GdiImage: PGdiObject absolute AImage;
GdiMask: PGdiObject absolute AMask;
W, H: Integer;
GC: PGdkGc;
begin
Result := nil;
if (AImage = 0) and (AMask = 0) then Exit;
if GdiMask = nil
then begin
if GdiImage^.GDIBitmapType = gbPixmap
then Result := GdiImage^.GDIPixmapObject.Mask;
if Result <> nil
then gdk_pixmap_ref(Result);
// DbgDumpBitmap(Result, 'CreateGdkMaskBitmap - Internal mask');
Exit;
end;
if GdiMask^.GDIBitmapType <> gbBitmap
then begin
DebugLN('[WARNING] CreateGtkBitmapMask: GDIBitmapType <> dbBitmap');
Exit;
end;
if (GdiImage = nil)
or (GdiImage^.GDIBitmapType <> gbPixmap)
or (GdiImage^.GDIPixmapObject.Mask = nil)
then begin
gdk_window_get_size(GdiMask^.GDIBitmapObject, @W, @H);
Result := gdk_pixmap_new(nil, W, H, 1);
GC := gdk_gc_new(Result);
gdk_gc_set_function(GC, {$ifdef gtk1}11{$else}GDK_COPY_INVERT{$endif});
gdk_draw_pixmap(Result, GC, GdiMask^.GDIBitmapObject, 0, 0, 0, 0, -1, -1);
gdk_gc_unref(GC);
//DbgDumpBitmap(Result, 'CreateGdkMaskBitmap - Mask');
Exit;
end;
// if we are here we need a combination (=AND) of both masks
gdk_window_get_size(GdiImage^.GDIPixmapObject.Mask, @W, @H);
Result := gdk_pixmap_new(nil, W, H, 1);
GC := gdk_gc_new(Result);
// copy image mask
gdk_draw_pixmap(Result, GC, GdiImage^.GDIPixmapObject.Mask, 0, 0, 0, 0, -1, -1);
// and with mask
gdk_gc_set_function(GC, {$ifdef gtk1}6{$else}GDK_AND_INVERT{$endif});
gdk_draw_pixmap(Result, GC, GdiMask^.GDIBitmapObject, 0, 0, 0, 0, -1, -1);
gdk_gc_unref(GC);
// DbgDumpBitmap(Result, 'CreateGdkMaskBitmap - Combi');
end;
function ExtractGdkBitmap(Bitmap: PGdkBitmap; const SrcRect: TRect): PGdkBitmap;
var
MaxRect: TRect;
SourceRect: TRect;
SrcWidth: Integer;
SrcHeight: Integer;
GC: PGdkGC;
begin
Result:=nil;
if Bitmap=nil then exit;
MaxRect:=Rect(0,0,0,0);
gdk_window_get_size(Bitmap,@MaxRect.Right,@MaxRect.Bottom);
IntersectRect(SourceRect,SrcRect,MaxRect);
SrcWidth:=SourceRect.Right-SourceRect.Left;
SrcHeight:=SourceRect.Bottom-SourceRect.Top;
DebugLn('ExtractGdkBitmap SourceRect=',dbgs(SourceRect));
if (SrcWidth<1) or (SrcHeight<1) then exit;
Result:= gdk_pixmap_new(nil, SrcWidth, SrcHeight, 1);
GC := GDK_GC_New(Result);
gdk_window_copy_area(Result,GC,0,0,Bitmap,
SourceRect.Left,SourceRect.Top,SrcWidth,SrcHeight);
GDK_GC_Unref(GC);
end;
procedure CheckGdkImageBitOrder(AImage: PGdkImage; AData: PByte; ADataCount: Integer);
var
b, count: Byte;
c: Cardinal;
{$ifdef hasx}
XImage: XLib.PXimage;
{$endif}
begin
{$ifdef hasx}
if AImage = nil then Exit;
XImage := gdk_x11_image_get_ximage(AImage);
if XImage^.bitmap_bit_order = LSBFirst then Exit;
{$endif}
// on windows or bigendian servers the bits need to be swapped
// align dataptr first
count := PtrUint(AData) and 3;
if count > ADataCount then count := ADataCount;
Dec(ADataCount, Count);
while (Count > 0) do
begin
// reduce dereferences
b := AData^;
b := ((b shr 4) and $0F) or ((b shl 4) and $F0);
b := ((b shr 2) and $33) or ((b shl 2) and $CC);
AData^ := ((b shr 1) and $55) or ((b shl 1) and $AA);
Dec(Count);
Inc(AData);
end;
// get remainder
Count := ADataCount and 3;
// now swap bits with 4 in a row
ADataCount := ADataCount shr 2;
while (ADataCount > 0) do
begin
// reduce dereferences
c := PCardinal(AData)^;
c := ((c shr 4) and $0F0F0F0F) or ((c shl 4) and $F0F0F0F0);
c := ((c shr 2) and $33333333) or ((c shl 2) and $CCCCCCCC);
PCardinal(AData)^ := ((c shr 1) and $55555555) or ((c shl 1) and $AAAAAAAA);
Dec(ADataCount);
Inc(AData, 4);
end;
// process remainder
while (Count > 0) do
begin
// reduce dereferences
b := AData^;
b := ((b shr 4) and $0F) or ((b shl 4) and $F0);
b := ((b shr 2) and $33) or ((b shl 2) and $CC);
AData^ := ((b shr 1) and $55) or ((b shl 1) and $AA);
Dec(Count);
Inc(AData);
end;
end;
{------------------------------------------------------------------------------
Function: AllocGDKColor
Params: AColor: A RGB color (TColor)
Returns: an Allocated GDKColor
Allocated a GDKColor from a winapi color
------------------------------------------------------------------------------}
function AllocGDKColor(const AColor: TColorRef): TGDKColor;
begin
with Result do
begin
Red := ((AColor shl 8) and $00FF00) or ((AColor ) and $0000FF);
Green := ((AColor ) and $00FF00) or ((AColor shr 8 ) and $0000FF);
Blue := ((AColor shr 8) and $00FF00) or ((AColor shr 16) and $0000FF);
end;
{$IFDEF DebugGDK}
BeginGDKErrorTrap;
{$ENDIF}
gdk_colormap_alloc_color(gdk_colormap_get_system, @Result, False, True);
{$IFDEF DebugGDK}
EndGDKErrorTrap;
{$ENDIF}
end;
function RegionType(RGN: PGDKRegion) : Longint;
var
aRect : TGDKRectangle;
SimpleRGN: PGdkRegion;
begin
{$IFDEF DebugGDK}
BeginGDKErrorTrap;
{$ENDIF}
If RGN = nil then
Result := ERROR
else
If gdk_region_empty(RGN) then
Result := NULLREGION
else begin
gdk_region_get_clipbox(RGN,@aRect);
SimpleRGN := gdk_region_rectangle(@aRect);
if gdk_region_equal(SimpleRGN, RGN) then
Result := SIMPLEREGION
else
Result := COMPLEXREGION;
gdk_region_destroy(SimpleRGN);
end;
{$IFDEF DebugGDK}
EndGDKErrorTrap;
{$ENDIF}
end;
function GDKRegionAsString(RGN: PGDKRegion): string;
var
aRect: TGDKRectangle;
begin
Result:=DbgS(RGN);
BeginGDKErrorTrap;
gdk_region_get_clipbox(RGN,@aRect);
EndGDKErrorTrap;
Result:=Result+'(x='+IntToStr(Integer(aRect.x))+',y='+IntToStr(Integer(aRect.y))+',w='
+IntToStr(aRect.Width)+',h='+IntToStr(aRect.Height)+' '
+'Type='+IntToStr(RegionType(RGN))+')';
end;
function CreateRectGDKRegion(const ARect: TRect): PGDKRegion;
var
GDkRect: TGDKRectangle;
begin
GDkRect.x:=ARect.Left;
GDkRect.y:=ARect.Top;
GDkRect.Width:=ARect.Right-ARect.Left;
GDkRect.Height:=ARect.Bottom-ARect.Top;
{$IFDEF DebugGDK}
BeginGDKErrorTrap;
{$ENDIF}
Result:=gdk_region_rectangle(@GDKRect);
{$IFDEF DebugGDK}
EndGDKErrorTrap;
{$ENDIF}
end;
Procedure FreeGDIColor(GDIColor: PGDIColor);
begin
if (cfColorAllocated in GDIColor^.ColorFlags) then begin
if (GDIColor^.Colormap <> nil) then begin
BeginGDKErrorTrap;
gdk_colormap_free_colors(GDIColor^.Colormap,@(GDIColor^.Color), 1);
EndGDKErrorTrap;
end;
//GDIColor.Color.Pixel := -1;
Exclude(GDIColor^.ColorFlags,cfColorAllocated);
end;
end;
procedure SetGDIColorRef(var GDIColor: TGDIColor; NewColorRef: TColorRef);
begin
if GDIColor.ColorRef=NewColorRef then exit;
FreeGDIColor(@GDIColor);
GDIColor.ColorRef:=NewColorRef;
end;
Procedure AllocGDIColor(DC: hDC; GDIColor: PGDIColor);
var
RGBColor : TColorRef;
begin
if DC=0 then ;
if not (cfColorAllocated in GDIColor^.ColorFlags) then begin
RGBColor := ColorToRGB(TColor(GDIColor^.ColorRef));
With GDIColor^.Color do begin
Red := gushort(GetRValue(RGBColor)) shl 8;
Green := gushort(GetGValue(RGBColor)) shl 8;
Blue := gushort(GetBValue(RGBColor)) shl 8;
Pixel := 0;
end;
{with TGtkDeviceContext(DC) do
If CurrentPalette <> nil then
GDIColor.Colormap := CurrentPalette^.PaletteColormap
else}
GDIColor^.Colormap := GDK_Colormap_get_system;
gdk_colormap_alloc_color(GDIColor^.Colormap, @(GDIColor^.Color),True,True);
Include(GDIColor^.ColorFlags,cfColorAllocated);
end;
end;
procedure BuildColorRefFromGDKColor(var GDIColor: TGDIColor);
begin
GDIColor.ColorRef:=TGDKColorToTColor(GDIColor.Color);
Include(GDIColor.ColorFlags,cfColorAllocated);
end;
procedure EnsureGCColor(DC: hDC; ColorType: TDevContextsColorType;
IsSolidBrush, AsBackground: Boolean);
var
GC: PGDKGC;
GDIColor: PGDIColor;
procedure WarnAllocFailed(const foreground : TGdkColor);
begin
DebugLn('NOTE: EnsureGCColor.EnsureAsGCValues gdk_colormap_alloc_color failed ',
' Foreground=',
DbgS(Foreground.red),',',
DbgS(Foreground.green),',',
DbgS(Foreground.blue),
' GDIColor^.ColorRef=',DbgS(GDIColor^.ColorRef)
);
end;
procedure EnsureAsGCValues;
var
AllocFG : Boolean;
SysGCValues: TGdkGCValues;
begin
FreeGDIColor(GDIColor);
SysGCValues:=GetSysGCValues(GDIColor^.ColorRef,
TGtkDeviceContext(DC).Widget);
{$IFDEF DebugGDK}BeginGDKErrorTrap;{$ENDIF}
with SysGCValues do
begin
AllocFG := Foreground.Pixel = 0;
if AllocFG then
if not gdk_colormap_alloc_color(GDK_Colormap_get_system, @Foreground,
True, True) then
WarnAllocFailed(Foreground);
gdk_gc_set_fill(GC, fill);
if AsBackground then
gdk_gc_set_background(GC, @foreground)
else
gdk_gc_set_foreground(GC, @foreground);
case Fill of
GDK_TILED :
if Tile <> nil then
begin
gdk_gc_set_ts_origin(GC, ts_x_origin, ts_y_origin);
gdk_gc_set_tile(GC, Tile);
end;
GDK_STIPPLED,
GDK_OPAQUE_STIPPLED:
if stipple <> nil then
begin
gdk_gc_set_background(GC, @background);
gdk_gc_set_ts_origin(GC, ts_x_origin, ts_y_origin);
gdk_gc_set_stipple(GC, stipple);
end;
end;
if AllocFG then
gdk_colormap_free_colors(GDK_Colormap_get_system, @Foreground,1);
end;
{$IFDEF DebugGDK}EndGDKErrorTrap;{$ENDIF}
end;
procedure EnsureAsColor;
begin
AllocGDIColor(DC, GDIColor);
//DebugLn('EnsureAsColor ',DbgS(GDIColor^.ColorRef),' AsBackground=',AsBackground);
{$IFDEF DebugGDK}BeginGDKErrorTrap;{$ENDIF}
if AsBackground then
gdk_gc_set_background(GC, @(GDIColor^.Color))
else
begin
gdk_gc_set_fill(GC, GDK_SOLID);
gdk_gc_set_foreground(GC, @(GDIColor^.Color));
end;
{$IFDEF DebugGDK}EndGDKErrorTrap;{$ENDIF}
end;
begin
GC:=TGtkDeviceContext(DC).GC;
GDIColor:=nil;
with TGtkDeviceContext(DC) do
begin
case ColorType of
dccCurrentBackColor: GDIColor:=@CurrentBackColor;
dccCurrentTextColor: GDIColor:=@CurrentTextColor;
dccGDIBrushColor : GDIColor:=@(GetBrush^.GDIBrushColor);
dccGDIPenColor : GDIColor:=@(GetPen^.GDIPenColor);
end;
end;
if GDIColor=nil then exit;
// FPC bug workaround:
// clScrollbar = $80000000 can't be used in case statements
if TColor(GDIColor^.ColorRef)=clScrollbar then
begin
//often have a BK Pixmap
if IsSolidBrush then
EnsureAsGCValues
else
EnsureAsColor;//GC's with Pixmaps can't work w/Hatch's (yet)
exit;
end;
case TColor(GDIColor^.ColorRef) of
//clScrollbar: see above
clInfoBk,
clMenu,
clHighlight,
clBtnFace,
clWindow,
clForm:
//often have a BK Pixmap
if IsSolidBrush then
EnsureAsGCValues
else
EnsureAsColor;//GC's with Pixmaps can't work w/Hatch's (yet)
clHighlightText,
clBtnShadow,
clBtnHighlight,
clBtnText,
clInfoText,
clWindowText,
clMenuText,
clGrayText:
//should never have a BK Pixmap
EnsureAsGCValues;
else
EnsureAsColor;
end;
end;
procedure CopyGDIColor(var SourceGDIColor, DestGDIColor: TGDIColor);
begin
SetGDIColorRef(DestGDIColor,SourceGDIColor.ColorRef);
end;
function IsBackgroundColor(Color: TColor): boolean;
begin
Result := (Color = clForm) or
(Color = clInfoBk) or
(Color = clBackground);
end;
function CompareGDIColor(const Color1, Color2: TGDIColor): boolean;
begin
Result:=Color1.ColorRef=Color2.ColorRef;
end;
function CompareGDIFill(const Fill1, Fill2: TGdkFill): boolean;
begin
Result:=Fill1=Fill2;
end;
function CompareGDIBrushes(Brush1, Brush2: PGdiObject): boolean;
begin
Result:=Brush1^.IsNullBrush=Brush2^.IsNullBrush;
if Result then begin
Result:=CompareGDIColor(Brush1^.GDIBrushColor,Brush2^.GDIBrushColor);
if Result then begin
Result:=CompareGDIFill(Brush1^.GDIBrushFill,Brush2^.GDIBrushFill);
if Result then begin
Result:=Brush1^.GDIBrushPixMap=Brush2^.GDIBrushPixMap;
end;
end;
end;
end;
//-----------------------------------------------------------------------------
{ Palette Index<->RGB Hash Functions }
type
TIndexRGB = record
Index: longint;
RGB: longint;
end;
PIndexRGB = ^TIndexRGB;
function GetIndexAsKey(p: pointer): pointer;
begin
Result:=Pointer(PIndexRGB(p)^.Index + 1);
end;
function GetRGBAsKey(p: pointer): pointer;
begin
Result:=Pointer(PIndexRGB(p)^.RGB + 1);
end;
function PaletteIndexToIndexRGB(Pal : PGDIObject; I : longint): PIndexRGB;
var
HashItem: PDynHashArrayItem;
begin
Result := nil;
HashItem:=Pal^.IndexTable.FindHashItemWithKey(Pointer(I + 1));
if HashItem<>nil then
Result:=PIndexRGB(HashItem^.Item);
end;
function PaletteRGBToIndexRGB(Pal : PGDIObject; RGB : longint): PIndexRGB;
var
HashItem: PDynHashArrayItem;
begin
Result := nil;
HashItem:=Pal^.RGBTable.FindHashItemWithKey(Pointer(RGB + 1));
if HashItem<>nil then
Result:=PIndexRGB(HashItem^.Item);
end;
{ Palette Index<->RGB lookup Functions }
function PaletteIndexExists(Pal : PGDIObject; I : longint): Boolean;
begin
Result := Pal^.IndexTable.ContainsKey(Pointer(I + 1));
end;
function PaletteRGBExists(Pal : PGDIObject; RGB : longint): Boolean;
begin
Result := Pal^.RGBTable.ContainsKey(Pointer(RGB + 1));
end;
function PaletteAddIndex(Pal : PGDIObject; I, RGB : Longint): Boolean;
var
IndexRGB: PIndexRGB;
begin
New(IndexRGB);
IndexRGB^.Index:=I;
IndexRGB^.RGB:=RGB;
Pal^.IndexTable.Add(IndexRGB);
Result := PaletteIndexExists(Pal, I);
If Not Result then
Dispose(IndexRGB)
else begin
Pal^.RGBTable.Add(IndexRGB);
Result := PaletteRGBExists(Pal, RGB);
If not Result then begin
Pal^.IndexTable.Remove(IndexRGB);
Dispose(IndexRGB);
end;
end;
end;
function PaletteDeleteIndex(Pal : PGDIObject; I : Longint): Boolean;
var
RGBIndex : PIndexRGB;
begin
RGBIndex := PaletteIndextoIndexRGB(Pal,I);
Result := RGBIndex = nil;
If not Result then begin
Pal^.IndexTable.Remove(RGBIndex);
If PaletteRGBExists(Pal, RGBIndex^.RGB) then
Pal^.RGBTable.Remove(RGBIndex);
Dispose(RGBIndex);
end;
end;
function PaletteIndexToRGB(Pal : PGDIObject; I : longint): longint;
var
RGBIndex : PIndexRGB;
begin
RGBIndex := PaletteIndextoIndexRGB(Pal,I);
if RGBIndex = nil then
Result := -1//InvalidRGB
else
Result := RGBIndex^.RGB;
end;
function PaletteRGBToIndex(Pal : PGDIObject; RGB : longint): longint;
var
RGBIndex : PIndexRGB;
begin
RGBIndex := PaletteRGBtoIndexRGB(Pal,RGB);
if RGBIndex = nil then
Result:=-1//InvalidIndex
else
Result := RGBIndex^.Index;
end;
procedure InitializePalette(const Pal: PGDIObject; const Entries: PPaletteEntry; const RGBCount: Longint);
var
I: Integer;
RGBValue: Longint;
begin
for I := 0 to RGBCount - 1 do
begin
if PaletteIndexExists(Pal, I) then
PaletteDeleteIndex(Pal, I);
with Entries[I] do
RGBValue := RGB(peRed, peGreen, peBlue) {or (peFlags shl 32)??};
if not PaletteRGBExists(Pal, RGBValue) then
PaletteAddIndex(Pal, I, RGBValue);
end;
end;
function HandleGTKKeyUpDown(AWidget: PGtkWidget; AEvent: PGdkEventKey;
AData: gPointer; ABeforeEvent, AHandleDown: Boolean;
const AEventName: PGChar) : GBoolean;
// returns CallBackDefaultReturn if event can continue in gtk's message system
{off $DEFINE VerboseKeyboard}
const
KEYUP_MAP: array[Boolean {syskey}, Boolean {before}] of Cardinal = (
(LM_KEYUP, CN_KEYUP),
(LM_SYSKEYUP, CN_SYSKEYUP)
);
KEYDOWN_MAP: array[Boolean {syskey}, Boolean {before}] of Cardinal = (
(LM_KEYDOWN, CN_KEYDOWN),
(LM_SYSKEYDOWN, CN_SYSKEYDOWN)
);
CHAR_MAP: array[Boolean {syskey}, Boolean {before}] of Cardinal = (
(LM_CHAR, CN_CHAR),
(LM_SYSCHAR, CN_SYSCHAR)
);
var
Msg: TLMKey;
EventStopped: Boolean;
EventString: PChar; // GTK1 and GTK2 workaround
// (and easy access to bytes)
KeyCode: Word;
KCInfo: TKeyCodeInfo;
VKey: Byte;
ShiftState: TShiftState;
Character: TUTF8Char;
SysKey: Boolean;
CommonKeyData: Integer;
Flags: Integer;
FocusedWidget: PGtkWidget;
LCLObject: TObject;
FocusedWinControl: TWinControl;
HandledByLCL: Boolean;
TargetWidget: PGtkWidget;
TargetObj: gPointer;
KeyPressesChar: char;
procedure StopKeyEvent;
begin
{$IFDEF VerboseKeyboard}
DebugLn('StopKeyEvent AEventName="',AEventName,'" ABeforeEvent=',dbgs(ABeforeEvent));
{$ENDIF}
if not EventStopped
then begin
g_signal_stop_emission_by_name(PGtkObject(AWidget), AEventName);
EventStopped := True;
end;
//MWE: still need to skip on win32 ?
{MWE:.$IfNDef Win32}
if EventString <> nil
then begin
gdk_event_key_set_string(AEvent, #0);
AEvent^.length := 0;
end;
{MWE:.$EndIf}
ResetDefaultIMContext;
AEvent^.KeyVal := 0;
end;
function DeliverKeyMessage(const Target: Pointer; var AMessage): boolean;
begin
Result:=DeliverMessage(Target,AMessage)=0;
if not Result then StopKeyEvent;
end;
function GetSpecialChar: Char;
begin
if (AEvent^.keyval > $FF00) and (AEvent^.keyval < $FF20) and
(AEvent^.keyval <> GDK_KEY_Tab) then
Result := Chr(AEvent^.keyval xor $FF00)
else
Result := #0;
end;
function CanSendChar: Boolean;
begin
Result := False;
if AEvent^.Length > 1 then Exit;
// to be delphi compatible we should not send a space here
if AEvent^.KeyVal = GDK_KEY_KP_SPACE then Exit;
// Check if CTRL is pressed
if ssCtrl in ShiftState
then begin
// Check if we pressed ^@
if (AEvent^.Length = 0)
and (AEvent^.KeyVal = GDK_KEY_AT)
then begin
Result := True;
Exit;
end;
// check if we send the ^Char subset
if (AEvent^.Length = 1) and (EventString <> nil)
then begin
Result := (EventString^ > #0) and (EventString^ < ' ');
end;
Exit;
end;
Result := (AEvent^.Length > 0) or (GetSpecialChar <> #0);
end;
function KeyAlreadyHandledByGtk: boolean;
begin
Result := false;
if AWidget = nil then exit;
if GtkWidgetIsA(AWidget, gtk_entry_get_type)
then begin
// the gtk_entry handles the following keys
case Aevent^.keyval of
GDK_Key_Return,
GDK_Key_Escape,
GDK_Key_Tab: Exit;
end;
Result := AEvent^.length > 0;
if Result then Exit;
case AEvent^.keyval of
GDK_Key_BackSpace,
GDK_Key_Clear,
GDK_Key_Insert,
GDK_Key_Delete,
GDK_Key_Home,
GDK_Key_End,
GDK_Key_Left,
GDK_Key_Right,
$20..$FF: Result := True;
end;
exit;
end;
if GtkWidgetIsA(AWidget, gtk_text_get_type)
then begin
// the gtk_text handles the following keys
case AEvent^.keyval of
GDK_Key_Escape: Exit;
end;
Result := AEvent^.length > 0;
if Result then Exit;
case AEvent^.keyval of
GDK_Key_Return,
GDK_Key_Tab,
GDK_Key_BackSpace,
GDK_Key_Clear,
GDK_Key_Insert,
GDK_Key_Delete,
GDK_Key_Home,
GDK_Key_End,
GDK_Key_Left,
GDK_Key_Right,
GDK_Key_Up,
GDK_Key_Down,
$20..$FF: Result := True;
end;
exit;
end;
end;
procedure CharToKeyVal(C: Char; out KeyVal: guint; out Length: gint);
begin
Length := 1;
{$ifndef gtk1}
if C in [#$01..#$1B] then
begin
KeyVal := $FF00 or Ord(C);
if KeyVal = GDK_KEY_BackSpace then
Length := 0;
end
else
{$endif}
KeyVal := Ord(C);
end;
function KeyActivatedAccelerator: boolean;
function CheckMenuChilds(AMenuItem: TMenuItem): boolean;
var
i: Integer;
Item: TMenuItem;
MenuItemWidget: PGtkWidget;
begin
Result:=false;
if (AMenuItem=nil) or (not AMenuItem.HandleAllocated) then exit;
for i:=0 to AMenuItem.Count-1 do begin
Item:=AMenuItem[i];
if not Item.HandleAllocated then continue;
if not GTK_WIDGET_SENSITIVE(PGTKWidget(Item.Handle)) then continue;
if IsAccel(Msg.CharCode,Item.Caption) then begin
// found
Result:=true;
MenuItemWidget:=PGTKWidget(Item.Handle);
if GtkWidgetIsA(MenuItemWidget,gtk_menu_item_get_type) then begin
//DebugLn(['CheckMenuChilds popup: ',dbgsName(Item)]);
// popup the submenu
gtk_signal_emit_by_name(PGtkObject(MenuItemWidget),'activate-item');
end;
exit;
end;
end;
end;
var
AComponent: TComponent;
AControl: TControl;
AForm: TCustomForm;
begin
Result:=false;
//debugln('KeyActivatedAccelerator A');
if not SysKey then exit;
// it is a system key -> try menus
if (Msg.CharCode in [VK_A..VK_Z]) then begin
if (TObject(TargetObj) is TComponent) then begin
AComponent:=TComponent(TargetObj);
//DebugLn(['KeyActivatedAccelerator ',dbgsName(AComponent)]);
if AComponent is TControl then begin
AControl:=TControl(AComponent);
repeat
AForm:=GetFirstParentForm(AControl);
if AForm<>nil then begin
if AForm.Menu<>nil then begin
Result:=CheckMenuChilds(AForm.Menu.Items);
if Result then exit;
end;
end;
AControl:=AForm.Parent;
until AControl=nil;
{$IFDEF Gtk2}
// check main menu of MainForm
if (Application.MainForm<>nil) then begin
AControl:=TControl(AComponent);
AForm:=GetParentForm(AControl);
if (AForm<>nil)
and (not (fsModal in AForm.FormState))
and (not Application.MainForm.IsParentOf(AControl))
and (Application.MainForm.Menu<>nil) then begin
Result:=CheckMenuChilds(Application.MainForm.Menu.Items);
if Result then exit;
end;
end;
{$ENDIF}
end;
end;
end;
end;
procedure EmulateEatenKeys;
begin
// some widgets eats keys, but do not do anything useful for the LCL
// emulate the keys
if not ABeforeEvent then Exit;
if EventStopped then Exit;
//DebugLn(['EmulateEatenKeys TargetWidget=',dbghex(PtrInt(TargetWidget))]);
//DebugLn(['EmulateEatenKeys ',GetWidgetDebugReport(TargetWidget),' gdk_event_get_type(AEvent)=',gdk_event_get_type(AEvent),' GDK_KEY_PRESS=',GDK_KEY_PRESS,' VKey=',VKey]);
{$IFDEF Gtk2}
// the gtk2 gtkentry handles the return key and emits an activate signal
// The LCL does not use that and needs the return key event
// => emulate it
if GtkWidgetIsA(TargetWidget, gtk_type_entry)
and (gdk_event_get_type(AEvent) = GDK_KEY_PRESS)
and (VKey=13)
then begin
//DebugLn(['EmulateKeysEatenByGtk ']);
FillChar(Msg, SizeOf(Msg), 0);
Msg.CharCode := VKey;
if SysKey then
Msg.msg := LM_SYSKEYDOWN
else
Msg.msg := LM_KEYDOWN;
Msg.KeyData := CommonKeyData or (Flags shl 16) or $0001 {TODO: repeatcount};
// send the (Sys)KeyDown message directly to the LCL
NotifyApplicationUserInput(TControl(TargetObj), Msg.Msg);
DeliverKeyMessage(TargetObj, Msg);
end;
{$ENDIF}
end;
procedure CheckDeadKey;
begin
if ABeforeEvent then begin
{$IFDEF Gtk2}
if im_context_widget<>TargetWidget then begin
//DebugLn(['CheckDeadKey init im_context ',GetWidgetDebugReport(TargetWidget)]);
ResetDefaultIMContext;
im_context_widget:=TargetWidget;
gtk_im_context_set_client_window(im_context,GetControlWindow(TargetWidget));
//DebugLn(['CheckDeadKey im_context initialized']);
end;
// Note: gtk_im_context_filter_keypress understands keypress and keyrelease
gtk_im_context_filter_keypress (im_context, AEvent);
//DebugLn(['CheckDeadKey DeadKey=',DeadKey,' str="',im_context_string,'"']);
{$ENDIF}
end;
end;
begin
Result := CallBackDefaultReturn;
EventStopped := False;
HandledByLCL := KeyEventWasHandledByLCL(AEvent, ABeforeEvent);
{$IFDEF VerboseKeyboard}
DebugLn(['[HandleGTKKeyUpDown] ',DbgSName(TControl(AData)),
' ',(AEvent^.{$IFDEF GTK1}theType{$ELSE}_Type{$ENDIF}),' Widget=',GetWidgetClassName(AWidget),
' Before=',ABeforeEvent,' Down=',AHandleDown,' HandledByLCL=',HandledByLCL]);
{$ENDIF}
// handle every key event only once
if HandledByLCL then Exit;
TargetWidget := AWidget;
TargetObj := AData;
FocusedWinControl := nil;
FocusedWidget := nil;
// The gtk sends keys first to the gtkwindow and then to the focused control.
// The LCL expects only once to the focused control.
// And some gtk widgets (combo) eats keys, so that the LCL has no chance to
// handle it. Therefore keys to the form are immediately redirected to the
// focused control without changing the normal gtk event path.
if GtkWidgetIsA(AWidget, gtk_window_get_type)
then begin
FocusedWidget := PGtkWindow(AWidget)^.focus_widget;
if FocusedWidget <> nil
then begin
LCLObject := GetNearestLCLObject(FocusedWidget);
if LCLObject is TWinControl
then begin
FocusedWinControl := TWinControl(LCLObject);
if FocusedWidget <> AWidget
then begin
{$IFDEF VerboseKeyboard}
DebugLn('[HandleGTKKeyUpDown] REDIRECTING ',
' FocusedWidget=',GetWidgetClassName(FocusedWidget),
' Control=',FocusedWinControl.Name,':',FocusedWinControl.ClassName);
{$ENDIF}
// redirect key to lcl control
TargetWidget := FocusedWidget;
TargetObj := FocusedWinControl;
end;
end;
end;
end;
// remember this event
RememberKeyEventWasHandledByLCL(AEvent, ABeforeEvent);
if TargetWidget = nil then Exit;
//DebugLn(['HandleGTKKeyUpDown TargetWidget=',GetWidgetDebugReport(TargetWidget)]);
//DebugLn(['HandleGTKKeyUpDown TargetWidget=',GetWidgetDebugReport(TargetWidget),' ',DbgStr(EventString),' state=',AEvent^.state,' keyval=',AEvent^.keyval]);
FillChar(Msg, SizeOf(Msg), 0);
gdk_event_key_get_string(AEvent, EventString);
//DebugLn(['HandleGTKKeyUpDown TargetWidget=',GetWidgetDebugReport(TargetWidget),' ',DbgStr(EventString),' state=',AEvent^.state,' keyval=',AEvent^.keyval]);
CheckDeadKey;
Flags := 0;
SysKey := False;
ShiftState := GTKEventStateToShiftState(AEvent^.state);
{$ifdef gtk1}
KeyCode := XKeysymToKeycode(gdk_display, AEvent^.keyval);
{$else}
KeyCode := AEvent^.hardware_keycode;
{$endif}
if (KeyCode = 0)
or (KeyCode > High(MKeyCodeInfo))
or (MKeyCodeInfo[KeyCode].VKey1 = 0)
then begin
// no VKey defined, maybe composed char ?
CommonKeyData := 0;
end
else begin
KCInfo := MKeyCodeInfo[KeyCode];
if (KCInfo.Flags and KCINFO_FLAG_SHIFT_XOR_NUM <> 0)
and ((ssShift in ShiftState) xor (ssNum in ShiftState))
then VKey := KCInfo.VKey2
else VKey := KCInfo.VKey1;
if (KCInfo.Flags and KCINFO_FLAG_EXT) <> 0
then Flags := KF_EXTENDED;
// ssAlt + a key pressed is always a syskey
// ssAltGr + a key is only a syskey when the key pressed has no levelshift or when ssHift is pressed to0
SysKey := (ssAlt in ShiftState);
if not SysKey
then begin
// Check ssAltGr
if (KCInfo.Flags and KCINFO_FLAG_ALTGR) = 0
then begin
// VKey has no levelshift char so AltGr is syskey
SysKey := ssAltGr in ShiftState;
end
else begin
// VKey has levelshift char so AltGr + Shift is syskey
SysKey := ShiftState * [ssShift, ssAltGr] = [ssShift, ssAltGr]
end;
end;
if SysKey
then Flags := Flags or KF_ALTDOWN;
CommonKeyData := KeyCode shl 16; // Not really scancode, but will do
if AHandleDown
then begin
{$IFDEF VerboseKeyboard}
DebugLn('[HandleGTKKeyUpDown] GDK_KEY_PRESS VKey=',dbgs(VKey),' SysKey=',dbgs(SysKey));
{$ENDIF}
Msg.CharCode := VKey;
Msg.Msg := KEYDOWN_MAP[SysKey, ABeforeEvent];
// todo repeat
// Flags := Flags or KF_REPEAT;
Msg.KeyData := CommonKeyData or (Flags shl 16) or $0001 {TODO: repeatcount};
if not KeyAlreadyHandledByGtk
then begin
// send the (Sys)KeyDown message directly to the LCL
NotifyApplicationUserInput(TControl(TargetObj), Msg.Msg);
if DeliverKeyMessage(TargetObj, Msg)
and (Msg.CharCode <> Vkey) then
StopKeyEvent;
end;
if (not EventStopped) and ABeforeEvent
then begin
if KeyActivatedAccelerator then exit;
end;
end
else begin
{$IFDEF VerboseKeyboard}
DebugLn('[HandleGTKKeyUpDown] GDK_KEY_RELEASE VKey=',dbgs(VKey));
{$ENDIF}
Msg.CharCode := VKey;
Msg.Msg := KEYUP_MAP[SysKey, ABeforeEvent];
Flags := Flags or KF_UP or KF_REPEAT;
Msg.KeyData := CommonKeyData or (Flags shl 16) or $0001 {always};
// send the message directly to the LCL
Msg.Result:=0;
NotifyApplicationUserInput(TControl(TargetObj), Msg.Msg);
if DeliverKeyMessage(TargetObj, Msg)
and (Msg.CharCode <> VKey)
then begin
// key was handled by LCL
StopKeyEvent;
end;
end;
end;
// send keypresses
if not EventStopped and AHandleDown then begin
// send the UTF8 keypress
if ABeforeEvent then begin
// try to get the UTF8 representation of the key
{$IFDEF GTK1}
Character := '';
if (AEvent^.length > 0) and (AEvent^.length <= 8) //max composed UTF8 char has lenght 8
then begin
SetLength(Character, AEvent^.length);
System.Move(AEvent^.thestring^, Character[1], length(Character));
end;
{$ELSE GTK2}
if im_context_string <> '' then
begin
Character := UTF8Copy(im_context_string,1,1);
im_context_string:='';// clear, to avoid sending again
end
else
begin
KeyPressesChar := GetSpecialChar;
if KeyPressesChar <> #0 then
Character := KeyPressesChar
else
Character := '';
end;
{$ENDIF GTK2}
{$IFDEF VerboseKeyboard}
debugln('[HandleGTKKeyUpDown] GDK_KEY_PRESS UTF8="',DbgStr(Character),'"');
{$ENDIF}
if Character <> ''
then begin
LCLObject := GetNearestLCLObject(TargetWidget);
if LCLObject is TWinControl
then begin
// send the key after navigation keys were handled
Result := TWinControl(LCLObject).IntfUTF8KeyPress(Character, 1, SysKey);
if Result or (Character = '')
then StopKeyEvent
else if (Length(Character) = 1)
{$IFDEF Gtk1}
// GTK1 only supports normal ASCII characters (Note: #127 is delete)
and (Character[1] in [#32..#126])
{$ENDIF}
then begin
CharToKeyVal(Character[1], AEvent^.KeyVal, AEvent^.length);
if AEvent^.length = 1 then
begin
EventString^ := Character[1];
EventString[1] := #0;
end
else
EventString^ := #0;
end;
end;
end;
end;
// send a normal KeyPress Event for Delphi compatibility
if not EventStopped and CanSendChar
then begin
{$IFDEF EventTrace}
EventTrace('char', data);
{$ENDIF}
KeyPressesChar := #0;
if AEvent^.Length = 1
then begin
// ASCII key was pressed
KeyPressesChar := EventString^;
end
else
KeyPressesChar := GetSpecialChar;
if KeyPressesChar <> #0
then begin
FillChar(Msg, SizeOf(Msg), 0);
Msg.KeyData := CommonKeyData;
Msg.Msg := CHAR_MAP[SysKey, ABeforeEvent];
// send the (Sys)Char message directly (not queued) to the LCL
Msg.Result:=0;
Msg.CharCode := Ord(KeyPressesChar);
if DeliverKeyMessage(TargetObj, Msg)
and (Ord(KeyPressesChar) <> Msg.CharCode)
then begin
// key was changed by lcl
if (Msg.CharCode=0) or (Msg.CharCode>=128)
then begin
// key set to invalid => just stop
StopKeyEvent;
end
else begin
// try to change the key
CharToKeyVal(chr(Msg.CharCode), AEvent^.KeyVal, AEvent^.length);
if AEvent^.length = 1 then
begin
EventString^ := Character[1];
EventString[1] := #0;
end
else
EventString^ := #0;
gdk_event_key_set_string(AEvent, EventString);
end;
end;
end;
end;
end;
EmulateEatenKeys;
{$IFDEF Gtk1}
Result:=true;
{$ELSE}
Result:=EventStopped;
{$ENDIF}
end;
{------------------------------------------------------------------------------
Procedure: InitKeyboardTables
Params: none
Returns: none
Initializes the CharToVK and CKeyToVK tables
------------------------------------------------------------------------------}
procedure InitKeyboardTables;
procedure FindVKeyInfo(const AKeySym: Cardinal; var AVKey: Byte;
var AExtended, AHasMultiVK, ASecondKey: Boolean);
var
ByteKey: Byte;
begin
AExtended := False;
AHasMultiVK := False;
AVKey := VK_UNDEFINED;
ASecondKey := False;
case AKeySym of
32..255: begin
ByteKey:=Byte(AKeySym);
case Chr(ByteKey) of // Normal ASCII chars
//only unshifted values are checked
//'A'..'Z',
'0'..'9',
' ': AVKey := ByteKey;
'a'..'z': AVKey := ByteKey - Ord('a') + Ord('A');
'+': AVKey := VK_OEM_PLUS;
',': AVKey := VK_OEM_COMMA;
'-': AVKey := VK_OEM_MINUS;
'.': AVKey := VK_OEM_PERIOD;
// try the US keycodes first
';': AVKey := VK_OEM_1;
'/': AVKey := VK_OEM_2;
'`': AVKey := VK_OEM_3;
'[': AVKey := VK_OEM_4;
'\': AVKey := VK_OEM_5;
']': AVKey := VK_OEM_6;
'''': AVKey := VK_OEM_7;
end;
end;
GDK_KEY_Tab,
GDK_KEY_ISO_Left_Tab: AVKey := VK_TAB;
GDK_KEY_RETURN: AVKey := VK_RETURN;
// GDK_KEY_LINEFEED; AVKey := $0A;
// Cursor block / keypad
GDK_KEY_INSERT:
begin
AExtended := True;
AVKey := VK_INSERT;
end;
GDK_KEY_DELETE:
begin
AExtended := True;
AVKey := VK_DELETE;
end;
GDK_KEY_HOME:
begin
AExtended := True;
AVKey := VK_HOME;
end;
GDK_KEY_LEFT:
begin
AExtended := True;
AVKey := VK_LEFT;
end;
GDK_KEY_UP:
begin
AExtended := True;
AVKey := VK_UP;
end;
GDK_KEY_RIGHT:
begin
AExtended := True;
AVKey := VK_RIGHT;
end;
GDK_KEY_DOWN:
begin
AExtended := True;
AVKey := VK_DOWN;
end;
GDK_KEY_PAGE_UP:
begin
AExtended := True;
AVKey := VK_PRIOR;
end;
GDK_KEY_PAGE_DOWN:
begin
AExtended := True;
AVKey := VK_NEXT;
end;
GDK_KEY_END:
begin
AExtended := True;
AVKey := VK_END;
end;
// Keypad
GDK_KEY_KP_ENTER:
begin
AExtended := True;
AVKey := VK_Return;
end;
GDK_KEY_KP_Space, GDK_KEY_KP_Begin:
begin
AVKey := VK_CLEAR;
AHasMultiVK := True;
end;
GDK_KEY_KP_INSERT:
begin
// Keypad key is not extended
AVKey := VK_INSERT;
AHasMultiVK := True;
end;
GDK_KEY_KP_HOME:
begin
// Keypad key is not extended
AVKey := VK_HOME;
AHasMultiVK := True;
end;
GDK_KEY_KP_LEFT:
begin
// Keypad key is not extended
AVKey := VK_LEFT;
AHasMultiVK := True;
end;
GDK_KEY_KP_UP:
begin
// Keypad key is not extended
AVKey := VK_UP;
AHasMultiVK := True;
end;
GDK_KEY_KP_RIGHT:
begin
// Keypad key is not extended
AVKey := VK_RIGHT;
AHasMultiVK := True;
end;
GDK_KEY_KP_DOWN:
begin
// Keypad key is not extended
AVKey := VK_DOWN;
AHasMultiVK := True;
end;
GDK_KEY_KP_PAGE_UP:
begin
// Keypad key is not extended
AVKey := VK_PRIOR;
AHasMultiVK := True;
end;
GDK_KEY_KP_PAGE_DOWN:
begin
// Keypad key is not extended
AVKey := VK_NEXT;
AHasMultiVK := True;
end;
GDK_KEY_KP_END:
begin
// Keypad key is not extended
AVKey := VK_END;
AHasMultiVK := True;
end;
GDK_KEY_Num_Lock:
begin
AExtended := True;
AVKey := VK_NUMLOCK;
end;
GDK_KEY_KP_F1..GDK_KEY_KP_F4:
begin
// not on "normal" keyboard so defined extended to differentiate between normal Fn
AExtended := True;
AVKey := VK_F1 + AKeySym - GDK_KEY_KP_F1;
end;
GDK_KEY_KP_TAB:
begin
// not on "normal" keyboard so defined extended to differentiate between normal TAB
AExtended := True;
AVKey := VK_TAB;
end;
GDK_KEY_KP_Multiply:
begin
AVKey := VK_MULTIPLY;
end;
GDK_KEY_KP_Add:
begin
AVKey := VK_ADD;
end;
GDK_KEY_KP_Separator:
begin
// Keypad key is not extended
AVKey := VK_SEPARATOR;
AHasMultiVK := True;
end;
GDK_KEY_KP_Subtract:
begin
AVKey := VK_SUBTRACT;
end;
GDK_KEY_KP_Decimal:
begin
// Keypad key is not extended
AVKey := VK_DECIMAL;
AHasMultiVK := True;
end;
GDK_KEY_KP_Delete:
begin
// Keypad key is not extended
AVKey := VK_DELETE;
AHasMultiVK := True;
end;
GDK_KEY_KP_Divide:
begin
AExtended := True;
AVKey := VK_DIVIDE;
end;
GDK_KEY_KP_0..GDK_KEY_KP_9:
begin
// Keypad key is not extended, it is identified by VK
AVKey := VK_NUMPAD0 + AKeySym - GDK_KEY_KP_0;
AHasMultiVK := True;
end;
GDK_KEY_BackSpace: AVKey := VK_BACK;
GDK_KEY_Clear: AVKey := VK_CLEAR;
GDK_KEY_Pause: AVKey := VK_PAUSE;
GDK_KEY_Scroll_Lock: AVKey := VK_SCROLL;
GDK_KEY_Sys_Req: AVKey := VK_SNAPSHOT;
GDK_KEY_Escape: AVKey := VK_ESCAPE;
GDK_KEY_Kanji: AVKey := VK_KANJI;
GDK_Key_Select: AVKey := VK_SELECT;
GDK_Key_Print: AVKey := VK_PRINT;
GDK_Key_Execute: AVKey := VK_EXECUTE;
GDK_Key_Cancel: AVKey := VK_CANCEL;
GDK_Key_Help: AVKey := VK_HELP;
GDK_Key_Break: AVKey := VK_CANCEL;
GDK_Key_Mode_switch: AVKey := VK_MODECHANGE;
GDK_Key_Caps_Lock: AVKey := VK_CAPITAL;
GDK_Key_Shift_L: AVKey := VK_SHIFT;
GDK_Key_Shift_R:
begin
AVKey := VK_SHIFT;
ASecondKey := True;
end;
GDK_Key_Control_L: AVKey := VK_CONTROL;
GDK_Key_Control_R:
begin
AVKey := VK_CONTROL;
ASecondKey := True;
end;
// GDK_Key_Meta_L: AVKey := VK_MENU; //shifted alt, so it is found by alt
// GDK_Key_Meta_R: AVKey := VK_MENU;
GDK_Key_Alt_L: AVKey := VK_MENU;
GDK_Key_Alt_R:
begin
AVKey := VK_MENU;
ASecondKey := True;
end;
GDK_Key_Super_L: AVKey := VK_LWIN;
GDK_Key_Super_R: begin
AVKey := VK_RWIN;
ASecondKey := True;
end;
GDK_Key_Menu: AVKey := VK_APPS;
// function keys
GDK_KEY_F1..GDK_KEY_F24: AVKey := VK_F1 + AKeySym - GDK_Key_F1;
// Extra keys on a "internet" keyboard
GDKX_KEY_Sleep:
begin
AExtended := True;
AVKey := VK_SLEEP;
end;
GDKX_KEY_AudioLowerVolume:
begin
AExtended := True;
AVKey := VK_VOLUME_DOWN;
end;
GDKX_KEY_AudioMute:
begin
AExtended := True;
AVKey := VK_VOLUME_MUTE;
end;
GDKX_KEY_AudioRaiseVolume:
begin
AExtended := True;
AVKey := VK_VOLUME_UP;
end;
GDKX_KEY_AudioPlay:
begin
AExtended := True;
AVKey := VK_MEDIA_PLAY_PAUSE;
end;
GDKX_KEY_AudioStop:
begin
AExtended := True;
AVKey := VK_MEDIA_STOP;
end;
GDKX_KEY_AudioPrev:
begin
AExtended := True;
AVKey := VK_MEDIA_PREV_TRACK;
end;
GDKX_KEY_AudioNext:
begin
AExtended := True;
AVKey := VK_MEDIA_NEXT_TRACK;
end;
GDKX_KEY_Mail:
begin
AExtended := True;
AVKey := VK_LAUNCH_MAIL;
end;
GDKX_KEY_HomePage:
begin
AExtended := True;
AVKey := VK_BROWSER_HOME;
end;
GDKX_KEY_Back:
begin
AExtended := True;
AVKey := VK_BROWSER_BACK;
end;
GDKX_KEY_Forward:
begin
AExtended := True;
AVKey := VK_BROWSER_FORWARD;
end;
GDKX_KEY_Stop:
begin
AExtended := True;
AVKey := VK_BROWSER_STOP;
end;
GDKX_KEY_Refresh:
begin
AExtended := True;
AVKey := VK_BROWSER_REFRESH;
end;
GDKX_KEY_WWW:
begin
AExtended := True;
AVKey := VK_BROWSER_HOME;
end;
GDKX_KEY_Favorites:
begin
AExtended := True;
AVKey := VK_BROWSER_FAVORITES;
end;
GDKX_KEY_AudioMedia:
begin
AExtended := True;
AVKey := VK_LAUNCH_MEDIA_SELECT;
end;
GDKX_KEY_MyComputer:
begin
AExtended := True;
AVKey := VK_LAUNCH_APP1;
end;
GDKX_KEY_Calculator:
begin
AExtended := True;
AVKey := VK_LAUNCH_APP2;
end;
// For faster cases, group by families
$400..$4FF: begin
// Katakana
end;
$500..$5FF: begin
// Arabic
case AKeySym of
GDK_KEY_arabic_hamza: AVKey := VK_X;
GDK_KEY_arabic_hamzaonwaw: AVKey := VK_C;
GDK_KEY_arabic_hamzaonyeh: AVKey := VK_Z;
GDK_KEY_arabic_alef: AVKey := VK_H;
GDK_KEY_arabic_beh: AVKey := VK_F;
GDK_KEY_arabic_tehmarbuta: AVKey := VK_M;
GDK_KEY_arabic_teh: AVKey := VK_J;
GDK_KEY_arabic_theh: AVKey := VK_E;
GDK_KEY_arabic_jeem: AVKey := VK_OEM_4;
GDK_KEY_arabic_hah: AVKey := VK_P;
GDK_KEY_arabic_khah: AVKey := VK_O;
GDK_KEY_arabic_dal: AVKey := VK_OEM_6;
GDK_KEY_arabic_thal: AVKey := VK_OEM_3;
GDK_KEY_arabic_ra: AVKey := VK_V;
GDK_KEY_arabic_zain: AVKey := VK_OEM_PERIOD;
GDK_KEY_arabic_seen: AVKey := VK_S;
GDK_KEY_arabic_sheen: AVKey := VK_A;
GDK_KEY_arabic_sad: AVKey := VK_W;
GDK_KEY_arabic_dad: AVKey := VK_Q;
GDK_KEY_arabic_tah: AVKey := VK_OEM_7;
GDK_KEY_arabic_zah: AVKey := VK_OEM_2;
GDK_KEY_arabic_ain: AVKey := VK_U;
GDK_KEY_arabic_ghain: AVKey := VK_Y;
GDK_KEY_arabic_feh: AVKey := VK_T;
GDK_KEY_arabic_qaf: AVKey := VK_R;
GDK_KEY_arabic_kaf: AVKey := VK_OEM_1;
GDK_KEY_arabic_lam: AVKey := VK_G;
GDK_KEY_arabic_meem: AVKey := VK_L;
GDK_KEY_arabic_noon: AVKey := VK_K;
GDK_KEY_arabic_heh: AVKey := VK_I;
GDK_KEY_arabic_waw: AVKey := VK_OEM_COMMA;
GDK_KEY_arabic_alefmaksura: AVKey := VK_N;
GDK_KEY_arabic_yeh: AVKey := VK_D;
end;
end;
$600..$6FF: begin
// Cyrillic
// MWE:
// These VK codes are not compatible with all cyrillic KBlayouts
// Example:
// VK_A on a russian layout generates a cyrillic_EF
// VK_A on a serbian layout generates a cyrillic_A
//
// Mapping cyrillic_A to VK_A is easier so that encoding is used.
// Maybe in future we can take the KBLayout into account
case AKeySym of
GDK_KEY_cyrillic_a..GDK_KEY_cyrillic_ze:
begin
AVKey := VK_A + AKeySym - GDK_KEY_cyrillic_a;
end;
// Capital is not needed, the lower will match
//GDK_KEY_cyrillic_A..GDK_KEY_cyrillic_ZE:
//begin
// AVKey := VK_A + AKeySym - GDK_KEY_cyrillic_A;
//end;
end;
end;
$700..$7FF: begin
// Greek
case AKeySym of
// Capital is not needed, the lower will match
GDK_KEY_greek_alpha: AVKey := VK_A;
GDK_KEY_greek_beta: AVKey := VK_B;
GDK_KEY_greek_gamma: AVKey := VK_G;
GDK_KEY_greek_delta: AVKey := VK_D;
GDK_KEY_greek_epsilon: AVKey := VK_E;
GDK_KEY_greek_zeta: AVKey := VK_Z;
GDK_KEY_greek_eta: AVKey := VK_H;
GDK_KEY_greek_theta: AVKey := VK_U;
GDK_KEY_greek_iota: AVKey := VK_I;
GDK_KEY_greek_kappa: AVKey := VK_K;
GDK_KEY_greek_lamda: AVKey := VK_L;
GDK_KEY_greek_mu: AVKey := VK_M;
GDK_KEY_greek_nu: AVKey := VK_N;
GDK_KEY_greek_xi: AVKey := VK_J;
GDK_KEY_greek_omicron: AVKey := VK_O;
GDK_KEY_greek_pi: AVKey := VK_P;
GDK_KEY_greek_rho: AVKey := VK_R;
GDK_KEY_greek_sigma: AVKey := VK_S;
GDK_KEY_greek_finalsmallsigma: AVKey := VK_W;
GDK_KEY_greek_tau: AVKey := VK_T;
GDK_KEY_greek_upsilon: AVKey := VK_Y;
GDK_KEY_greek_phi: AVKey := VK_F;
GDK_KEY_greek_chi: AVKey := VK_X;
GDK_KEY_greek_psi: AVKey := VK_C;
GDK_KEY_greek_omega: AVKey := VK_V;
end;
end;
$C00..$CFF: begin
// Hebrew
// Shifted keys will produce A..Z so the VK codes will be assigned there
end;
$D00..$DFF: begin
// Thai
// To many differences to assign VK codes through lookup
// Thai Kedmanee and Thai Pattachote are complete different layouts
end;
$E00..$EFF: begin
// Korean
end;
end;
end;
function IgnoreShifted(const AUnshiftKeySym: Cardinal): Boolean;
begin
case AUnshiftKeySym of
GDK_KEY_END,
GDK_KEY_HOME,
GDK_KEY_LEFT,
GDK_KEY_RIGHT,
GDK_KEY_UP,
GDK_KEY_DOWN,
GDK_KEY_PAGE_UP,
GDK_KEY_PAGE_DOWN: Result := True;
else
Result := False;
end;
end;
procedure NextFreeVK(var AFreeVK: Byte);
begin
case AFreeVK of
$96: AFreeVK := $E1;
$E1: AFreeVK := $E3;
$E4: AFreeVK := $E6;
$E6: AFreeVK := $E9;
$F5: begin
{$ifndef HideKeyTableWarnings}
DebugLn('[WARNING] Out of OEM specific VK codes, changing to unassigned');
{$endif}
AFreeVK := $88;
end;
$8F: AFreeVK := $97;
$9F: AFreeVK := $D8;
$DA: AFreeVK := $E5;
$E5: AFreeVK := $E8;
$E8: begin
{$ifndef HideKeyTableWarnings}
DebugLn('[WARNING] Out of unassigned VK codes, assigning $FF');
{$endif}
AFreeVK := $FF;
end;
$FF: AFreeVK := $FF; // stay there
else
Inc(AFreeVK);
end;
end;
const
KEYFLAGS: array[0..3] of Byte = (
$00,
KCINFO_FLAG_SHIFT,
KCINFO_FLAG_ALTGR,
KCINFO_FLAG_ALTGR or KCINFO_FLAG_SHIFT
);
EXTFLAG: array[Boolean] of Byte = (
$00,
KCINFO_FLAG_EXT
);
MULTIFLAG: array[Boolean] of Byte = (
$00,
KCINFO_FLAG_SHIFT_XOR_NUM
);
{$ifdef HasX}
{
Starting gdk 2.10 Alt, meta, hyper are reported by a own mask. Since we support
older versions, we need to create the modifiermap ourselves for X and we cannot
ise them
}
type
TModMap = array[Byte] of Cardinal;
procedure SetupModifiers(ADisplay: Pointer; var AModMap: TModMap);
const
MODIFIERS: array[0..7] of Cardinal = (
GDK_SHIFT_MASK,
GDK_LOCK_MASK,
GDK_CONTROL_MASK,
GDK_MOD1_MASK,
GDK_MOD2_MASK,
GDK_MOD3_MASK,
GDK_MOD4_MASK,
GDK_MOD5_MASK
);
var
Map: PXModifierKeymap;
KeyCode: PKeyCode;
Modifier, n: Integer;
begin
FillByte(AModMap, SizeOf(AModMap), 0);
Map := XGetModifierMapping(ADisplay);
KeyCode := Map^.modifiermap;
for Modifier := Low(MODIFIERS) to High(MODIFIERS) do
begin
for n := 1 to Map^.max_keypermod do
begin
if KeyCode^ <> 0
then begin
AModMap[KeyCode^] := MODIFIERS[Modifier];
{$ifdef VerboseModifiermap}
DebugLn('Mapped keycode=%u to modifier=$%2.2x', [KeyCode^, MODIFIERS[Modifier]]);
{$endif}
end;
Inc(KeyCode);
end;
end;
XFreeModifiermap(Map);
end;
procedure UpdateModifierMap(const AModMap: TModMap; AKeyCode: Byte; AKeySym: Cardinal);
var
{$ifdef VerboseModifiermap}
s: string;
{$endif}
ShiftState: TShiftStateEnum;
begin
if AModMap[AKeyCode] = 0 then Exit;
case AKeySym of
GDK_KEY_Caps_Lock,
GDK_KEY_Shift_Lock: ShiftState := ssCaps;
GDK_KEY_Num_Lock: ShiftState := ssNum;
GDK_KEY_Scroll_Lock: ShiftState := ssScroll;
GDK_Key_Shift_L,
GDK_Key_Shift_R: ShiftState := ssShift;
GDK_KEY_Control_L,
GDK_KEY_Control_R: ShiftState := ssCtrl;
{$ifndef UseOwnShiftState}
// UseOwnShiftState will track these, so we don't have to put them in the modmap
GDK_KEY_Meta_L,
GDK_KEY_Meta_R: ShiftState := ssMeta;
GDK_KEY_Alt_L,
GDK_KEY_Alt_R: ShiftState := ssAlt;
GDK_KEY_Super_L,
GDK_KEY_Super_R: ShiftState := ssSuper;
GDK_KEY_Hyper_L,
GDK_KEY_Hyper_R: ShiftState := ssHyper;
GDK_KEY_ISO_Level3_Shift{,
GDK_KEY_Mode_switch}: ShiftState := ssAltGr;
{$endif}
else
Exit;
end;
MModifiers[ShiftState].Mask := AModMap[AKeyCode];
MModifiers[ShiftState].UseValue := False;
{$ifdef VerboseModifiermap}
WriteStr(s, ShiftState);
DebugLn('Mapped keycode=%u, keysym=$%x, modifier=$%2.2x to shiftstate %s',
[AKeyCode, AKeySym, AModMap[AKeyCode], s]);
{$endif}
end;
{$ifdef UseOwnShiftState}
procedure UpdateKeyStateMap(var AIndex: integer; AKeyCode: Byte; AKeySym: Cardinal);
var
Enum: TShiftStateEnum;
begin
case AKeySym of
GDK_KEY_Alt_L, GDK_KEY_Alt_R: Enum := ssAlt;
GDK_KEY_Meta_L, GDK_KEY_Meta_R: Enum := ssMeta;
GDK_KEY_Super_L, GDK_KEY_Super_R: Enum := ssSuper;
GDK_KEY_Hyper_L, GDK_KEY_Hyper_R: Enum := ssHyper;
GDK_KEY_ISO_Level3_Shift: Enum := ssAltGr;
else
Exit;
end;
if High(MKeyStateMap) < AIndex
then SetLength(MKeyStateMap, AIndex + 8);
MKeyStateMap[AIndex].Index := AKeyCode shr 3;
MKeyStateMap[AIndex].Mask := 1 shl (AKeyCode and 7);
MKeyStateMap[AIndex].Enum := Enum;
Inc(AIndex)
end;
{$endif UseOwnShiftState}
{$endif HasX}
const
// first OEM specific VK
VK_FIRST_OEM = $92;
var
{$ifdef gtk1}
XKeyEvent: TXKeyEvent;
KeySymStart, KeySymNext: PKeySym;
UpKeySym, LoKeySym: TKeySym;
KeySyms: array of TKeySym = nil;
{$else}
KeySyms: array of guint;
KeyVals: Pguint;
KeymapKeys: PGdkKeymapKey;
UniChar: gunichar;
{$endif}
KeySymCount: Integer;
KeySymChars: array[0..16] of Char;
KeySymCharLen: Integer;
{$ifdef HasX}
XDisplay: Pointer;
ModMap: TModMap;
{$endif}
{$ifdef UseOwnShiftState}
KeyStateMapIndex: Integer;
{$endif}
KeyCode: Byte;
m: Integer;
LoKey, HiKey, I: Integer;
VKey, FreeVK: Byte;
HasMultiVK, DummyBool, Extended, SecondKey, HasKey, ComputeVK: Boolean;
begin
{$ifdef HasX}
XDisplay := gdk_display;
if XDisplay = nil then Exit;
FillByte(MKeyStateMap, SizeOF(MKeyStateMap), 0);
SetupModifiers(XDisplay, ModMap);
{$endif}
{$ifdef gtk1}
// Init dummy XEvent to retrieve the char corresponding to a key
FillChar(XKeyEvent, SizeOf(XKeyEvent), 0);
XKeyEvent._Type := GDK_KEY_PRESS;
XKeyEvent.Display := XDisplay;
XKeyEvent.Same_Screen := 1;
// Retrieve the KeyCode bounds
XDisplayKeyCodes(XDisplay, @LoKey, @HiKey);
if LoKey < 0
then begin
DebugLn('[WARNING] Low keycode (%d) negative, adjusting to 0', [LoKey]);
LoKey := 0;
end;
if HiKey > 255
then begin
DebugLn('[WARNING] High keycode (%d) larget than 255, adjusting to 255', [HiKey]);
HiKey := 255;
end;
KeySymCount := 0;
KeySymStart := XGetKeyboardMapping(XDisplay, LoKey, HiKey - LoKey + 1, @KeySymCount);
KeySymNext := KeySymStart;
if (KeySymCount = 0) or (KeySymStart = nil)
then begin
DebugLn('[WARNING] failed to retrieve keyboardmapping');
if KeySymStart <> nil
then XFree(KeySymStart);
Exit;
end;
// The code in Accelerate below assumes at least 2 items in the KeySym array
if KeySymCount > 2 then begin
for I:=Low(MVKeyInfo) to High(MVKeyInfo) do
SetLength(MVKeyInfo[I].KeySym, KeySymCount);
end else begin
for I:=Low(MVKeyInfo) to High(MVKeyInfo) do
SetLength(MVKeyInfo[I].KeySym, 2);
end;
SetLength(KeySyms, KeySymCount);
{$else gtk1}
LoKey := 0;
HiKey := 255;
{$endif}
{$ifdef UseOwnShiftState}
KeyStateMapIndex := 0;
{$endif}
FreeVK := VK_FIRST_OEM;
for KeyCode := LoKey to HiKey do
begin
{$ifdef gtk1}
Move(KeySymNext^, KeySyms[0], SizeOf(KeySyms[0]) * KeySymCount);
Inc(KeySymNext, KeySymCount);
HasKey := False;
m := 0;
while m < KeySymCount do
begin
// there might be only uppercase chars are in the map,
// so we have to add the lowercase ourselves
// when a group consists of one char(next =0)
if KeySyms[m] <> 0
then begin
HasKey := True;
if KeySyms[m+1] = 0
then begin
XConvertCase(KeySyms[m], @LoKeySym, @UpKeySym);
if LoKeySym <> UpKeySym
then begin
KeySyms[m] := LoKeySym;
KeySyms[m+1] := UpKeySym;
end;
end;
end;
Inc(m, 2);
end;
{$else}
if not gdk_keymap_get_entries_for_keycode(nil, KeyCode, KeymapKeys, KeyVals, @KeySymCount) then Continue;
SetLength(KeySyms, KeySymCount);
Move(KeyVals^, KeySyms[0], SizeOf(KeySyms[0]) * KeySymCount);
g_free(KeymapKeys); // unused but we cannot pass a nil as param
g_free(KeyVals);
HasKey := KeySyms[0] <> 0;
//DebugLn(['InitKeyboardTables ',KeyCode,' ',HasKey,' ',KeySyms[0]]);
{$endif}
{$ifdef HasX}
// Check if this keycode is in the modifiers map
// loop through all keysyms till one found.
// Some maps have a modifier with an undefined first keysym. It is checked for
// modifiers, but not for vkeys
for m := 0 to KeySymCount - 1 do
begin
if KeySyms[m] = 0 then Continue;
UpdateModifierMap(ModMap, KeyCode, KeySyms[m]);
{$ifdef UseOwnShiftState}
UpdateKeyStateMap(KeyStateMapIndex, KeyCode, KeySyms[m]);
{$endif}
Break;
end;
{$endif}
// Continue if there is no keysym found
if not HasKey then Continue;
// Start looking for a VKcode
VKey := VK_UNDEFINED;
for m := 0 to KeySymCount - 1 do
begin
if KeySyms[m] = 0 then Continue;
FindVKeyInfo(KeySyms[m], VKey, Extended, HasMultiVK, SecondKey);
{$ifdef Windows}
// on windows, the keycode is perdef the VK,
// we only enter this loop to set the correct flags
VKey := KeyCode;
Break;
{$else}
if HasMultiVK then Break; // has VK per def
if VKey = VK_UNDEFINED then Continue;
if MVKeyInfo[VKey].KeyCode[SecondKey or Extended] = 0 then Break; // found unused VK
// already in use
VKey := VK_UNDEFINED;
{$endif}
end;
ComputeVK := VKey = VK_UNDEFINED;
if ComputeVK and not HasMultiVK
then begin
VKey := FreeVK;
NextFreeVK(FreeVK);
end;
if VKey = VK_UNDEFINED
then begin
MKeyCodeInfo[KeyCode].Flags := $FF
end
else begin
MKeyCodeInfo[KeyCode].Flags := EXTFLAG[Extended] or MULTIFLAG[HasMultiVK];
MVKeyInfo[VKey].KeyCode[SecondKey] := KeyCode;
end;
MKeyCodeInfo[KeyCode].VKey1 := VKey;
for m := 0 to Min(High(MVKeyInfo[0].KeyChar), KeySymCount - 1) do
begin
if KeySyms[m] = 0 then Continue;
if (m >= 2) and (KeySyms[m] = KeySyms[m - 2]) then Continue;
if HasMultiVK
then begin
if m >= 2 then Break; // Only process shift
// The keypadkeys have 2 VK_keycodes :(
// In that case we have to FIndKeyInfo for every keysym
if m = 1
then begin
FindVKeyInfo(KeySyms[m], VKey, Extended, DummyBool, DummyBool);
MKeyCodeInfo[KeyCode].VKey2 := VKey;
end;
end;
if VKey = VK_UNDEFINED then Continue;
MKeyCodeInfo[KeyCode].Flags := MKeyCodeInfo[KeyCode].Flags or KEYFLAGS[m];
FillByte(KeySymChars, SizeOf(KeySymChars), 0);
{$ifdef gtk1}
// Retrieve the chars for this KeySym
XKeyEvent.KeyCode := KeyCode;
case m of
0: XKeyEvent.State := 0;
1: XKeyEvent.State := MModifiers[ssShift].Mask;
2: XKeyEvent.State := MModifiers[ssAltGr].Mask;
3: XKeyEvent.State := MModifiers[ssAltGr].Mask or MModifiers[ssShift].Mask;
else
// TODO: m > 3 ??
Continue;
end;
KeySymCharLen := XLookupString(@XKeyEvent, KeySymChars, SizeOf(KeySymChars), nil, nil);
if (KeySymCharLen > 0) and (KeySymChars[KeySymCharLen - 1] = #0)
then Dec(KeySymCharLen);
if (KeySymCharLen <= 0) then Continue;
{$else gtk1}
UniChar := gdk_keyval_to_unicode(KeySyms[m]);
if UniChar = 0 then Continue;
KeySymCharLen := g_unichar_to_utf8(UniChar, @KeySymChars[0]);
{$endif}
if (KeySymCharLen > SizeOf(TVKeyUTF8Char))
then DebugLn('[WARNING] InitKeyboardTables - Keysymstring for keycode=%u longer than %u bytes: %s', [KeyCode, SizeOf(TVKeyUTF8Char), KeySymChars]);
Move(KeySymChars[0], MVKeyInfo[VKey].KeyChar[m], SizeOf(TVKeyUTF8Char));
end;
end;
{$ifdef UseOwnShiftState}
SetLength(MKeyStateMap, KeyStateMapIndex);
{$endif}
{$ifdef gtk1}
XFree(KeySymStart);
{$endif}
end;
{------------------------------------------------------------------------------
Procedure: DoneKeyboardTables
Params: none
Returns: none
Frees the dynamic keyboard tables
------------------------------------------------------------------------------}
procedure DoneKeyboardTables;
var
i: Integer;
begin
if LCLHandledKeyEvents<>nil then begin
for i:=0 to LCLHandledKeyEvents.Count-1 do
TObject(LCLHandledKeyEvents[i]).Free;
LCLHandledKeyEvents.Free;
LCLHandledKeyEvents:=nil;
end;
if LCLHandledKeyAfterEvents<>nil then begin
for i:=0 to LCLHandledKeyAfterEvents.Count-1 do
TObject(LCLHandledKeyAfterEvents[i]).Free;
LCLHandledKeyAfterEvents.Free;
LCLHandledKeyAfterEvents:=nil;
end;
end;
{------------------------------------------------------------------------------
Function: GetVKeyInfo
Params: AVKey: A virtual key to get the info for
Returns: A Info record
This function is more a safety to make sure MVkeyInfo isn't accessed out of
it's bounds
------------------------------------------------------------------------------}
function GetVKeyInfo(const AVKey: Byte): TVKeyInfo;
begin
Result := MVKeyInfo[AVKey];
end;
{------------------------------------------------------------------------------
Procedure: GTKEventState2ShiftState
Params: KeyState: The gtk keystate
Returns: the TShiftState for the given KeyState
GTKEventStateToShiftState converts a GTK event state to a LCL/Delphi TShiftState
------------------------------------------------------------------------------}
function GTKEventStateToShiftState(KeyState: Word): TShiftState;
{$ifdef HasX}
function GetState: TShiftState;
var
Keys: chararr32;
n: Integer;
begin
Result := [];
keys:='';
XQueryKeyMap(gdk_display, Keys);
for n := Low(MKeyStateMap) to High(MKeyStateMap) do
begin
if Ord(Keys[MKeyStateMap[n].Index]) and MKeyStateMap[n].Mask = 0 then Continue;
Include(Result, MKeyStateMap[n].Enum);
Break;
end;
end;
{$else}
{$ifdef windows}
function GetState: TShiftState;
begin
Result := [];
if GetKeyState(VK_MENU) < 0 then Include(Result, ssAlt);
if (GetKeyState(VK_LWIN) < 0) or (GetKeyState(VK_RWIN) < 0) then Include(Result, ssMeta);
end;
{$else}
function GetState: TShiftState;
begin
Result := [];
end;
{$endif}
{$endif}
var
State: TShiftStateEnum;
begin
{$ifdef UseOwnShiftState}
Result := GetState;
{$else}
Result := [];
{$endif}
for State := Low(State) to High(State) do
begin
if MModifiers[State].Mask = 0 then Continue;
if MModifiers[State].UseValue
then begin
if KeyState and MModifiers[State].Mask = MModifiers[State].Value
then Include(Result, State);
end
else begin
if KeyState and MModifiers[State].Mask <> 0
then Include(Result, State);
end;
end;
end;
{------------------------------------------------------------------------------
Procedure: StoreCommonDialogSetup
Params: ADialog: TCommonDialog
Returns: none
Stores the size of a TCommonDialog.
------------------------------------------------------------------------------}
procedure StoreCommonDialogSetup(ADialog: TCommonDialog);
var DlgWindow: PGtkWidget;
begin
if (ADialog=nil) or (ADialog.Handle=0) then exit;
DlgWindow:=PGtkWidget(ADialog.Handle);
if DlgWindow^.Allocation.Width>0 then
ADialog.Width:=DlgWindow^.Allocation.Width;
if DlgWindow^.Allocation.Height>0 then
ADialog.Height:=DlgWindow^.Allocation.Height;
end;
{------------------------------------------------------------------------------
Procedure: DestroyCommonDialogAddOns
Params: ADialog: TCommonDialog
Returns: none
Free the memory of additional data of a TCommonDialog
------------------------------------------------------------------------------}
procedure DestroyCommonDialogAddOns(ADialog: TCommonDialog);
var
DlgWindow: PGtkWidget;
HistoryList: TFPList; // list of TFileSelHistoryListEntry
AHistoryEntry: PFileSelHistoryEntry;
i: integer;
FileSelWidget: PGtkFileSelection;
LCLHistoryMenu: PGTKWidget;
{$IFDEF Gtk1}
//AFilterEntry: TFileSelFilterEntry;
FilterList: TFPList; // list of TFileSelFilterListEntry
LCLFilterMenu: PGTKWidget;
{$ENDIF}
begin
if (ADialog=nil) or (not ADialog.HandleAllocated) then exit;
DlgWindow:=PGtkWidget(ADialog.Handle);
{$IFDEF VerboseTransient}
DebugLn('DestroyCommonDialogAddOns ',ADialog.Name,':',ADialog.ClassName);
{$ENDIF}
gtk_window_set_transient_for(PGtkWindow(DlgWindow),nil);
if ADialog is TOpenDialog then begin
{$IFDEF GTK2}
FileSelWidget:=GTK_FILE_CHOOSER(DlgWindow);
{$ELSE}
FileSelWidget:=GTK_FILE_SELECTION(DlgWindow);
FreeWidgetInfo(FileSelWidget^.selection_entry);
FreeWidgetInfo(FileSelWidget^.dir_list);
FreeWidgetInfo(FileSelWidget^.file_list);
LCLFilterMenu:=PGTKWidget(gtk_object_get_data(PGtkObject(FileSelWidget),
'LCLFilterMenu'));
if LCLFilterMenu<>nil then FreeWidgetInfo(LCLFilterMenu);
{$ENDIF}
LCLHistoryMenu:=PGTKWidget(gtk_object_get_data(PGtkObject(FileSelWidget),
'LCLHistoryMenu'));
if LCLHistoryMenu<>nil then FreeWidgetInfo(LCLHistoryMenu);
// free history
HistoryList:=TFPList(gtk_object_get_data(PGtkObject(DlgWindow),
'LCLHistoryList'));
if HistoryList<>nil then begin
for i:=0 to HistoryList.Count-1 do begin
AHistoryEntry:=PFileSelHistoryEntry(HistoryList[i]);
StrDispose(AHistoryEntry^.Filename);
AHistoryEntry^.Filename:=nil;
Dispose(AHistoryEntry);
end;
HistoryList.Free;
gtk_object_set_data(PGtkObject(DlgWindow),'LCLHistoryList',nil);
end;
{$IFDEF GTK1}
// free filter
FilterList:=TFPList(gtk_object_get_data(PGtkObject(DlgWindow),
'LCLFilterList'));
if FilterList<>nil then begin
for i:=0 to FilterList.Count-1 do
TObject(FilterList[i]).Free;
FilterList.Free;
gtk_object_set_data(PGtkObject(DlgWindow),'LCLFilterList',nil);
end;
{$ENDIF}
// free preview handle
if ADialog is TPreviewFileDialog then begin
if TPreviewFileDialog(ADialog).PreviewFileControl<>nil then
TPreviewFileDialog(ADialog).PreviewFileControl.Handle:=0;
end;
end;
end;
{------------------------------------------------------------------------------
Procedure: PopulateFileAndDirectoryLists
Params: FileSelection: PGtkFileSelection;
Mask: string (File mask, such as *.txt)
Returns: none
Populate the directory and file lists according to the given mask
------------------------------------------------------------------------------}
procedure PopulateFileAndDirectoryLists(FileSelection: PGtkFileSelection;
const Mask: string);
var
Dirs, Files: PGtkCList;
Text: array [0..1] of Pgchar;
Info: TSearchRec;
DirName: PChar;
Dir: string;
StrList: TStringList;
CurFileMask: String;
procedure Add(List: PGtkCList; const s: string);
begin
Text[0] := PChar(s);
gtk_clist_append(List, Text);
end;
procedure AddList(List: PGtkCList);
var
i: integer;
begin
StrList.Sorted := True;
//DebugLn(['AddList ',StrList.Text]);
for i:=0 to StrList.Count-1 do
Add(List, StrList[i]);
StrList.Sorted := False;
end;
begin
StrList := TStringList.Create;
dirs := PGtkCList(FileSelection^.dir_list);
files := PGtkCList(FileSelection^.file_list);
DirName := gtk_file_selection_get_filename(FileSelection);
if DirName <> nil then begin
SetString(Dir, DirName, strlen(DirName));
SetLength(Dir, LastDelimiter(PathDelim,Dir));
end else
Dir := '';
//DebugLn(['PopulateFileAndDirectoryLists ',Dir]);
Text[1] := nil;
gtk_clist_freeze(Dirs);
gtk_clist_clear(Dirs);
gtk_clist_freeze(Files);
gtk_clist_clear(Files);
{ Add all directories }
Strlist.Add('..'+PathDelim);
if FindFirstUTF8(AppendPathDelim(Dir)+GetAllFilesMask, faAnyFile and faDirectory,
Info) = 0
then begin
repeat
if ((Info.Attr and faDirectory) = faDirectory) and (Info.Name <> '.')
and (Info.Name <> '..') and (Info.Name<>'') then
StrList.Add(AppendPathDelim(Info.Name));
until FindNextUTF8(Info) <> 0;
end;
FindCloseUTF8(Info);
AddList(Dirs);
// add required files
StrList.Clear;
CurFileMask:=Mask;
if CurFileMask='' then CurFileMask:=GetAllFilesMask;
if FindFirstUTF8(AppendPathDelim(Dir)+GetAllFilesMask, faAnyFile, Info) = 0 then
begin
repeat
if ((Info.Attr and faDirectory) <> faDirectory) then begin
//debugln('PopulateFileAndDirectoryLists CurFileMask="',CurFileMask,'" Info.Name="',Info.Name,'" ',dbgs(MatchesMaskList(Info.Name,CurFileMask)));
if (CurFileMask='') or (MatchesMaskList(Info.Name,CurFileMask)) then
begin
Strlist.Add(Info.Name);
end;
end;
until FindNextUTF8(Info) <> 0;
end;
FindCloseUTF8(Info);
AddList(Files);
StrList.Free;
gtk_clist_thaw(Dirs);
gtk_clist_thaw(Files);
end;
{------------------------------------------------------------------------------
Procedure: DeliverMessage
Params: Message: the message to process
Returns: True if handled
Generic function which calls the WindowProc if defined, otherwise the
dispatcher
------------------------------------------------------------------------------}
function DeliverMessage(const Target: Pointer; var AMessage): PtrInt;
begin
if (TLMessage(AMessage).Msg = LM_PAINT) or
(TLMessage(AMessage).Msg = LM_GTKPAINT) then
CurrentSentPaintMessageTarget := TObject(Target);
Result := LCLMessageGlue.DeliverMessage(TObject(Target), AMessage);
CurrentSentPaintMessageTarget := nil;
end;
{------------------------------------------------------------------------------
Function: ObjectToGTKObject
Params: AnObject: A LCL Object
Returns: The GTKObject of the given object
Returns the GTKObject of the given object, nil if no object available
------------------------------------------------------------------------------}
function ObjectToGTKObject(const AnObject: TObject): PGtkObject;
var
handle : HWND;
begin
Handle := 0;
if not assigned(AnObject) then
begin
assert (false, 'TRACE: [ObjectToGtkObject] Object not assigned');
end
else if (AnObject is TWinControl) then
begin
if TWinControl(AnObject).HandleAllocated then
handle := TWinControl(AnObject).Handle;
end
else if (AnObject is TMenuItem) then
begin
if TMenuItem(AnObject).HandleAllocated then
handle := TMenuItem(AnObject).Handle;
end
else if (AnObject is TMenu) then
begin
if TMenu(AnObject).HandleAllocated then
handle := TMenu(AnObject).Items.Handle;
end
else if (AnObject is TCommonDialog) then
begin
{if TCommonDialog(AObject).HandleAllocated then }
handle := TCommonDialog(AnObject).Handle;
end
else begin
//DebugLn(Format('Trace: [ObjectToGtkObject] Message received with unhandled class-type <%s>', [AnObject.ClassName]));
end;
Result := PGTKObject(handle);
if handle = 0 then
Assert (false, 'Trace: [ObjectToGtkObject]****** Warning: handle = 0 *******');
end;
(***********************************************************************
Widget member functions
************************************************************************)
// ----------------------------------------------------------------------
// the main widget is the widget passed as handle to the winAPI
// main data is stored in the fixed form to get a reference to its parent
// ----------------------------------------------------------------------
function GetMainWidget(const Widget: Pointer): Pointer;
begin
if Widget = nil
then raise EInterfaceException.Create('GetMainWidget Widget=nil');
Result := gtk_object_get_data(Widget, 'Main');
if Result = nil then Result := Widget; // the widget is the main widget itself.
end;
procedure SetMainWidget(const ParentWidget, ChildWidget: Pointer);
begin
if ParentWidget = nil
then raise EInterfaceException.Create('SetMainWidget ParentWidget=nil');
if ChildWidget = nil
then raise EInterfaceException.Create('SetMainWidget ChildWidget=nil');
if ParentWidget = ChildWidget
then raise EInterfaceException.Create('SetMainWidget ParentWidget=ChildWidget');
{$IFDEF Gtk2}
if PGtkWidget(ParentWidget)^.parent=ChildWidget
then raise EInterfaceException.Create('SetMainWidget Parent^.Parent=ChildWidget');
{$ENDIF}
gtk_object_set_data(ChildWidget, 'Main', ParentWidget)
end;
{ ------------------------------------------------------------------------------
Get the fixed widget of a widget.
Every LCL control with a clientarea, has at least a main widget for the control
and a fixed widget for the client area. If the Fixed widget is not set, use
try to get it trough WinWidgetInfo
------------------------------------------------------------------------------ }
//TODO: remove when WinWidgetInfo implementation is complete
function GetFixedWidget(const Widget: Pointer): Pointer;
var
WidgetInfo: PWinWidgetInfo;
begin
if Widget = nil
then raise EInterfaceException.Create('GetFixedWidget Widget=nil');
WidgetInfo := GetWidgetInfo(Widget, False);
if WidgetInfo <> nil
then Result := WidgetInfo^.ClientWidget
else Result := nil;
if Result <> nil then Exit;
Result := gtk_object_get_data(Widget, 'Fixed');
// A last resort
if Result = nil then Result := Widget;
end;
{ ------------------------------------------------------------------------------
Set the fixed widget of a widget.
Every LCL control with a clientarea, has at least a main widget for the control
and a fixed widget for the client area.
------------------------------------------------------------------------------ }
procedure SetFixedWidget(const ParentWidget, FixedWidget: Pointer);
var
WidgetInfo: PWinWidgetInfo;
begin
if ParentWidget = nil
then raise EInterfaceException.Create('SetFixedWidget ParentWidget=nil');
WidgetInfo := GetWidgetInfo(ParentWidget, True);
WidgetInfo^.ClientWidget := FixedWidget;
//TODO: remove old compatebility
gtk_object_set_data(ParentWidget, 'Fixed', FixedWidget)
end;
{-------------------------------------------------------------------------------
Set the LCLobject which created this widget.
-------------------------------------------------------------------------------}
procedure SetLCLObject(const Widget: Pointer; const AnObject: TObject);
var
WidgetInfo: PWinWidgetInfo;
begin
if Widget = nil
then raise EInterfaceException.Create('SetLCLObject Widget=nil');
if AnObject = nil
then raise EInterfaceException.Create('SetLCLObject AnObject=nil');
WidgetInfo := GetWidgetInfo(Widget, True);
WidgetInfo^.LCLObject := AnObject;
end;
function GetLCLObject(const Widget: Pointer): TObject;
var
WidgetInfo: PWinWidgetInfo;
begin
if Widget = nil
then raise EInterfaceException.Create('GetLCLObject Widget=nil');
WidgetInfo := GetWidgetInfo(Widget);
if WidgetInfo <> nil
then Result := WidgetInfo^.LCLObject
else Result := nil;
end;
{-------------------------------------------------------------------------------
Some need the HiddenLCLobject which created a parent of this widget.
MWE: is this obsolete ?
-------------------------------------------------------------------------------}
procedure SetHiddenLCLObject(const Widget: Pointer; const AnObject: TObject);
begin
if (Widget <> nil) then
gtk_object_set_data(Widget, 'LCLHiddenClass', Pointer(AnObject));
end;
function GetHiddenLCLObject(const Widget: Pointer): TObject;
begin
Result := TObject(gtk_object_get_data(Widget, 'LCLHiddenClass'));
end;
{-------------------------------------------------------------------------------
function GetNearestLCLObject(Widget: PGtkWidget): TObject;
Retrieves the LCLObject belonging to the widget. If the widget is created as
child of a main widget, the parent is queried.
This function probably obsoletes Get/SetMainWidget
-------------------------------------------------------------------------------}
//TODO: check if Get/SetMainWidget is still required
function GetNearestLCLObject(Widget: PGtkWidget): TObject;
begin
while (Widget<>nil) do begin
Result:=GetLCLObject(Widget);
if Result<>nil then exit;
Widget:=Widget^.Parent;
end;
Result:=nil;
end;
function CreateFixedClientWidget(WithWindow: Boolean = True): PGTKWidget;
begin
Result := gtk_fixed_new();
{$IFDEF GTK2}
if WithWindow then
gtk_fixed_set_has_window(PGtkFixed(Result), true);
{$ENDIF}
end;
{------------------------------------------------------------------------------
procedure FixedMoveControl(Parent, Child : PGTKWidget; Left, Top : Longint);
Move a childwidget on a client area (fixed or layout widget).
------------------------------------------------------------------------------}
procedure FixedMoveControl(Parent, Child : PGTKWidget; Left, Top : Longint);
begin
If GTKWidgetIsA(Parent, GTK_Layout_Get_Type) then begin
// parent is layout
gtk_Layout_move(PGtkLayout(Parent), Child, Left, Top)
end else If GTKWidgetIsA(Parent, GTK_Fixed_Get_Type) then begin
// parent is fixed
gtk_fixed_move(PGtkFixed(Parent), Child, gint16(Left), gint16(Top));
end else begin
// parent is invalid
DebugLn('[FixedMoveControl] WARNING: Invalid Fixed Widget');
end;
end;
{------------------------------------------------------------------------------
procedure FixedPutControl(Parent, Child : PGTKWIdget; Left, Top : Longint);
Add a childwidget onto a client area (fixed or layout widget).
------------------------------------------------------------------------------}
procedure FixedPutControl(Parent, Child: PGTKWidget; Left, Top: Longint);
procedure RaiseInvalidFixedWidget;
begin
// this is in a separate procedure for optimisation
DebugLn('[FixedPutControl] WARNING: Invalid Fixed Widget.',
' Parent=',DbgS(Parent),
' Child=',DbgS(Child)
);
end;
begin
if GtkWidgetIsA(Parent, gtk_fixed_get_type) then
gtk_fixed_put(PGtkFixed(Parent), Child, gint16(Left), gint16(Top))
else
if GtkWidgetIsA(Parent, gtk_layout_get_type) then
gtk_layout_put(PGtkLayout(Parent), Child, Left, Top)
else
RaiseInvalidFixedWidget;
end;
function GetWinControlWidget(Child: PGtkWidget): PGtkWidget;
// return the first widget, which is associated with a TWinControl handle
var
LCLParent: TObject;
begin
Result:=nil;
LCLParent:=GetNearestLCLObject(Child);
if (LCLParent=nil) or (not (LCLParent is TWinControl))
or (not TWinControl(LCLParent).HandleAllocated)
then exit;
Result:=PGtkWidget(TWinControl(LCLParent).Handle);
end;
function GetWinControlFixedWidget(Child: PGtkWidget): PGtkWidget;
begin
Result:=GetWinControlWidget(Child);
if Result=nil then exit;
Result:=GetFixedWidget(Result);
end;
function FindFixedChildListItem(ParentFixed: PGtkFixed; Child: PGtkWidget): PGList;
begin
Result:=ParentFixed^.children;
while (Result<>nil) do begin
if (Result^.Data<>nil) and (PGtkFixedChild(Result^.Data)^.Widget=Child) then
exit;
Result:=Result^.Next;
end;
end;
function FindFixedLastChildListItem(ParentFixed: PGtkFixed): PGList;
begin
Result:=g_list_last(ParentFixed^.children);
end;
function GetFixedChildListWidget(Item: PGList): PGtkWidget;
begin
Result:=PGtkFixedChild(Item^.Data)^.Widget;
end;
{------------------------------------------------------------------------------
procedure MoveGListLinkBehind(First, Item, After: PGList);
Move the list item 'Item' behind the list item 'After'.
If After=nil then insert as first item.
------------------------------------------------------------------------------}
procedure MoveGListLinkBehind(First, Item, After: PGList);
var
Data: Pointer;
NewPos: Integer;
begin
if (Item=After) or (Item^.Next=After) then exit;
if (g_list_position(First,Item)<0) then
RaiseGDBException('MoveGListLinkBehind Item not found');
if (After<>nil) and (g_list_position(First,After)<0) then
RaiseGDBException('MoveGListLinkBehind After not found');
Data:=Item^.Data;
g_list_remove_link(First,Item);
if After<>nil then begin
NewPos:=g_list_position(First,After)+1;
end else begin
NewPos:=0;
end;
g_list_insert(First,Data,NewPos);
end;
procedure MoveGListLink(First: PGList; FromIndex, ToIndex: integer);
var
Item: PGList;
InsertAfter: PGList;
i: Integer;
begin
if (FromIndex=ToIndex) then exit;
Item:=First;
i:=0;
while (i<FromIndex) do begin
Item:=Item^.next;
inc(i);
end;
// unbind
if Item^.next<>nil then Item^.next^.prev:=Item^.prev;
if Item^.prev<>nil then Item^.prev^.next:=Item^.next;
Item^.next:=nil;
Item^.prev:=nil;
// insert
if ToIndex=0 then begin
Item^.next:=First;
First^.prev:=Item;
end else begin
i:=0;
InsertAfter:=First;
while (i<ToIndex-1) do begin
if InsertAfter^.next=nil then break;
InsertAfter:=InsertAfter^.next;
inc(i);
end;
Item^.prev:=InsertAfter;
Item^.next:=InsertAfter^.next;
InsertAfter^.next:=Item;
if Item^.next<>nil then Item^.next^.prev:=Item;
end;
end;
{------------------------------------------------------------------------------
function GetControlWindow(Widget: Pointer) : PGDKWindow;
Get the gdkwindow of a widget.
------------------------------------------------------------------------------}
function GetControlWindow(Widget: Pointer) : PGDKWindow;
begin
if Widget <> nil then
begin
If not GTKWidgetIsA(PGTKWidget(Widget), GTK_Layout_Get_Type) then
Result := PGTKWidget(Widget)^.Window
else
Result := PGtkLayout(Widget)^.bin_window;
{$IFDEF Gtk2}
if (Result=nil) and (GTK_WIDGET_NO_WINDOW(Widget)) then
Result:=gtk_widget_get_parent_window(Widget);
{$ENDIF}
end else
RaiseGDBException('GetControlWindow Widget=nil');
end;
{------------------------------------------------------------------------------
function CreateWidgetInfo(const AWidget: Pointer): PWidgetInfo;
Creates a WidgetInfo structure for the given widget
Info needed by the API of a HWND (=Widget)
This structure obsoletes all other object data, like
"core-child", "fixed", "class"
------------------------------------------------------------------------------}
function CreateWidgetInfo(const AWidget: Pointer): PWidgetInfo;
begin
if AWidget = nil then Result:= nil
else begin
New(Result);
FillChar(Result^, SizeOf(Result^), 0);
gtk_object_set_data(AWidget, 'widgetinfo', Result);
Result^.DefaultCursor := HCursor(-1);
end;
end;
function CreateWidgetInfo(const AWidget: Pointer; const AObject: TObject;
const AParams: TCreateParams): PWidgetInfo;
begin
Result := CreateWidgetInfo(AWidget);
if Result = nil then Exit;
Result^.LCLObject := AObject;
// in most cases the created widget is the core widget
// so default to it
Result^.CoreWidget := AWidget;
Result^.Style := AParams.Style;
Result^.ExStyle := AParams.ExStyle;
Result^.WndProc := PtrUInt(AParams.WindowClass.lpfnWndProc);
end;
function GetWidgetInfo(const AWidget: Pointer {; const Create: Boolean = False}): PWidgetInfo;
begin
Result := GetWidgetInfo(AWidget, False);
end;
function GetWidgetInfo(const AWidget: Pointer;
const ACreate: Boolean): PWidgetInfo;
var
MainWidget: PGtkObject;
begin
if AWidget <> nil then
begin
MainWidget := GetMainWidget(AWidget);
Result := gtk_object_get_data(MainWidget, 'widgetinfo');
if (Result = nil) and ACreate then
begin
Result := CreateWidgetInfo(MainWidget);
// use the main widget as default
Result^.CoreWidget := PGtkWidget(MainWidget);
end;
end
else Result := nil;
end;
procedure FreeWidgetInfo(AWidget: Pointer);
var
Info: PWidgetInfo;
begin
if AWidget = nil then Exit;
//DebugLn(['FreeWidgetInfo ',GetWidgetDebugReport(AWidget)]);
Info := gtk_object_get_data(AWidget, 'widgetinfo');
if Info = nil then Exit;
if Info^.DoubleBuffer <> nil then
gdk_pixmap_unref(Info^.DoubleBuffer);
if (Info^.UserData <> nil) and (Info^.DataOwner) then begin
FreeMem(Info^.UserData);
//Info^.UserData := nil; // see below the whole memory is cleared by Fillchar
end;
gtk_object_set_data(AWidget,'widgetinfo',nil);
// Set WidgetInfo memory to nil. This will expose bugs that use widgetinfo after
// it has been freed and is still referenced by something!
FillChar(Info^, SizeOf(TWidgetInfo), 0);
Dispose(Info);
//DebugLn(['FreeWidgetInfo END']);
end;
{-------------------------------------------------------------------------------
procedure DestroyWidget(Widget: PGtkWidget);
- sends LM_DESTROY
- frees the WidgetInfo
- destroys the widget in the gtk
IMPORTANT:
The above order must be kept, to avoid callbacks working with dangling
pointers.
Some widgets have a LM_DESTROY set, so if the gtk or some other code
destroys those widget, the above is done in gtkdestroyCB.
-------------------------------------------------------------------------------}
procedure DestroyWidget(Widget: PGtkWidget);
var
Info: PWidgetInfo;
AWinControl: TWinControl;
Mess: TLMessage;
begin
//DebugLn(['DestroyWidget A ',GetWidgetDebugReport(Widget)]);
{$IFDEF DebugLCLComponents}
if DebugGtkWidgets.FindInfo(Widget)=nil then
DebugLn(['DestroyWidget ',GetWidgetDebugReport(Widget)]);
{$ENDIF}
Info:=GetWidgetInfo(Widget);
if Info<>nil then begin
if (Info^.LCLObject is TWinControl) then begin
AWinControl:=TWinControl(Info^.LCLObject);
if AWinControl.HandleAllocated
and (PGtkWidget(AWinControl.Handle)=Widget) then begin
// send the LM_DESTROY message before destroying the widget
FillChar(Mess,SizeOf(Mess),0);
Mess.msg := LM_DESTROY;
DeliverMessage(Info^.LCLObject, Mess);
end;
end;
FreeWidgetInfo(Widget);
end;
{$IFDEF DebugLCLComponents}
DebugGtkWidgets.MarkDestroyed(Widget);
{$ENDIF}
gtk_widget_destroy(Widget);
//DebugLn(['DestroyWidget B']);
end;
{-------------------------------------------------------------------------------
function GetGtkNoteBookDummyPage(ANoteBookWidget: PGtkNoteBook): PGtkWidget;
Retrieves the DummyWidget associated with the ANoteBookWidget
-------------------------------------------------------------------------------}
function GetGtkNoteBookDummyPage(ANoteBookWidget: PGtkNoteBook): PGtkWidget;
begin
Result:=gtk_object_get_data(PGtkObject(ANoteBookWidget),'LCLDummyPage');
end;
{-------------------------------------------------------------------------------
procedure SetGtkNoteBookDummyPage(ANoteBookWidget: PGtkNoteBook;
DummyWidget: PGtkWidget): PGtkWidget;
Associates the DummyWidget with the ANoteBookWidget
-------------------------------------------------------------------------------}
procedure SetGtkNoteBookDummyPage(ANoteBookWidget: PGtkNoteBook;
DummyWidget: PGtkWidget);
begin
gtk_object_set_data(PGtkObject(ANoteBookWidget),'LCLDummyPage',DummyWidget);
end;
{------------------------------------------------------------------------------
UpdateNoteBookClientWidget
Params: ANoteBook: TObject
This procedure updates the 'Fixed' object data.
* obsolete *
------------------------------------------------------------------------------}
procedure UpdateNoteBookClientWidget(ANoteBook: TObject);
var
ClientWidget: PGtkWidget;
NoteBookWidget: PGtkNotebook;
begin
if not TCustomTabControl(ANoteBook).HandleAllocated then exit;
NoteBookWidget := PGtkNotebook(TCustomTabControl(ANoteBook).Handle);
ClientWidget := nil;
SetFixedWidget(NoteBookWidget, ClientWidget);
end;
{-------------------------------------------------------------------------------
function GetGtkNoteBookPageCount(ANoteBookWidget: PGtkNoteBook): integer;
Returns the number of pages in a PGtkNotebook
-------------------------------------------------------------------------------}
function GetGtkNoteBookPageCount(ANoteBookWidget: PGtkNoteBook): integer;
var
AListItem: PGList;
begin
Result:=0;
if ANoteBookWidget=nil then exit;
AListItem:=ANoteBookWidget^.children;
while AListItem<>nil do begin
inc(Result);
AListItem:=AListItem^.Next;
end;
end;
{$IFDef GTK1}
var
NoteBookCloseBtnPixmapImg: PGdkPixmap = nil;
NoteBookCloseBtnPixmapMask: PGdkPixmap = nil;
{$EndIf}
{-------------------------------------------------------------------------------
procedure RemoveDummyNoteBookPage(NoteBookWidget: PGtkNotebook);
Removes the dummy page.
See also AddDummyNoteBookPage
-------------------------------------------------------------------------------}
procedure RemoveDummyNoteBookPage(NoteBookWidget: PGtkNotebook);
var
DummyWidget: PGtkWidget;
begin
DummyWidget:=GetGtkNoteBookDummyPage(NoteBookWidget);
if DummyWidget=nil then exit;
gtk_notebook_remove_page(NoteBookWidget,
gtk_notebook_page_num(NoteBookWidget,DummyWidget));
DummyWidget:=nil;
SetGtkNoteBookDummyPage(NoteBookWidget,DummyWidget);
end;
{-------------------------------------------------------------------------------
method GetNoteBookCloseBtnImage
Params:
Result: none
Loads the image for the close button in the tabs of the TCustomTabControl(s).
-------------------------------------------------------------------------------}
{$IfDef GTK1}
procedure GetNoteBookCloseBtnImage(Window: PGdkWindow;
var Img, Mask: PGdkPixmap);
begin
if (NoteBookCloseBtnPixmapImg=nil)
and (Window<>nil) then begin
LoadXPMFromLazResource('tnotebook_close_tab',Window,
NoteBookCloseBtnPixmapImg,NoteBookCloseBtnPixmapMask);
end;
Img:=NoteBookCloseBtnPixmapImg;
Mask:=NoteBookCloseBtnPixmapMask;
end;
{$EndIF}
{-------------------------------------------------------------------------------
method UpdateNotebookPageTab
Params: ANoteBook: TCustomTabControl; APage: TCustomPage
Result: none
Updates the tab of a page of a notebook. This contains the image to the left
side, the label, the close button, the menu image and the menu label.
-------------------------------------------------------------------------------}
procedure UpdateNotebookPageTab(ANoteBook, APage: TObject);
var
TheNoteBook: TCustomTabControl;
ThePage: TCustomPage;
NoteBookWidget: PGtkWidget; // the notebook
PageWidget: PGtkWidget; // the page (content widget)
TabWidget: PGtkWidget; // the tab (hbox containing a pixmap, a label
// and a close button)
TabImageWidget: PGtkWidget; // the icon widget in the tab (a fixed widget)
TabLabelWidget: PGtkWidget; // the label in the tab
TabCloseBtnWidget: PGtkWidget;// the close button in the tab
TabCloseBtnImageWidget: PGtkWidget; // the pixmap in the close button
MenuWidget: PGtkWidget; // the popup menu (hbox containing a pixmap and
// a label)
MenuImageWidget: PGtkWidget; // the icon widget in the popup menu item (a fixed widget)
MenuLabelWidget: PGtkWidget; // the label in the popup menu item
procedure UpdateTabImage;
var
HasIcon: Boolean;
IconSize: TPoint;
ImageIndex: Integer;
begin
HasIcon:=false;
IconSize:=Point(0,0);
ImageIndex := TheNoteBook.GetImageIndex(ThePage.PageIndex);
if (TheNoteBook.Images<>nil)
and (ImageIndex >= 0)
and (ImageIndex < TheNoteBook.Images.Count) then
begin
// page has valid image
IconSize := Point(TheNoteBook.Images.Width, TheNoteBook.Images.Height);
HasIcon := (IconSize.X>0) and (IconSize.Y>0);
end;
if HasIcon then
begin
// page has an image
if TabImageWidget <> nil then
begin
// there is already an icon widget for the image in the tab
// -> resize the icon widget
gtk_widget_set_usize(TabImageWidget,IconSize.X,IconSize.Y);
end else
begin
// there is no pixmap for the image in the tab
// -> insert one ot the left side of the label
TabImageWidget := gtk_label_new(#0);
g_signal_connect(PgtkObject(TabImageWidget), 'expose_event',
TGTKSignalFunc(@PageIconWidgetExposeAfter), ThePage);
{$IFNDEF GTK2}
g_signal_connect(PgtkObject(TabImageWidget), 'draw',
TGTKSignalFunc(@PageIconWidgetDrawAfter), ThePage);
{$ENDIF}
gtk_object_set_data(PGtkObject(TabWidget), 'TabImage', TabImageWidget);
gtk_widget_set_usize(TabImageWidget, IconSize.X, IconSize.Y);
gtk_widget_show(TabImageWidget);
gtk_box_pack_start_defaults(PGtkBox(TabWidget), TabImageWidget);
gtk_box_reorder_child(PGtkBox(TabWidget), TabImageWidget, 0);
end;
if MenuImageWidget<>nil then
begin
// there is already an icon widget for the image in the menu
// -> resize the icon widget
gtk_widget_set_usize(MenuImageWidget, IconSize.X, IconSize.Y);
end else
begin
// there is no icon widget for the image in the menu
// -> insert one at the left side of the label
MenuImageWidget:=gtk_label_new(#0);
g_signal_connect_after(PgtkObject(MenuImageWidget), 'expose_event',
TGTKSignalFunc(@PageIconWidgetExposeAfter), ThePage);
{$IFNDEF GTK2}
g_signal_connect_after(PgtkObject(MenuImageWidget), 'draw',
TGTKSignalFunc(@PageIconWidgetDrawAfter), ThePage);
{$ENDIF}
gtk_widget_set_usize(MenuImageWidget,IconSize.X,IconSize.Y);
gtk_object_set_data(PGtkObject(MenuWidget),'TabImage',MenuImageWidget);
gtk_widget_show(MenuImageWidget);
gtk_box_pack_start_defaults(PGtkBox(MenuWidget),MenuImageWidget);
gtk_box_reorder_child(PGtkBox(MenuWidget),MenuImageWidget,0);
end;
end else
begin
// page does not have an image
if TabImageWidget<>nil then
begin
// there is a pixmap for an old image in the tab
// -> remove the icon widget
DestroyWidget(TabImageWidget);
gtk_object_set_data(PGtkObject(TabWidget), 'TabImage', nil);
TabImageWidget:=nil;
end;
if MenuImageWidget<>nil then
begin
// there is a pixmap for an old image in the menu
// -> remove the icon widget
DestroyWidget(MenuImageWidget);
gtk_object_set_data(PGtkObject(MenuWidget), 'TabImage', nil);
MenuImageWidget:=nil;
end;
end;
end;
procedure UpdateTabLabel;
var
ACaption: String;
begin
ACaption := ThePage.Caption;
GTKWidgetSet.SetLabelCaption(PGtkLabel(TabLabelWidget), ACaption);
if MenuLabelWidget <> nil then
GTKWidgetSet.SetLabelCaption(PGtkLabel(MenuLabelWidget), ACaption);
end;
procedure UpdateTabCloseBtn;
var
{$IfDef GTK1}
Img: PGdkPixmap;
Mask: PGdkBitmap;
{$Else}
style: PGtkRcStyle;
{$EndIf}
begin
{$IfDef GTK1}
//debugln('UpdateTabCloseBtn ',GetWidgetDebugReport(NoteBookWidget));
Img:=nil;
Mask:=nil;
GetNoteBookCloseBtnImage(GetControlWindow(NoteBookWidget), Img, Mask);
{$EndIf}
//debugln('UpdateTabCloseBtn ',dbgs(nboShowCloseButtons in TheNotebook.Options),' ',dbgs(Img<>nil));
if (nboShowCloseButtons in TheNotebook.Options)
{$ifdef GTK1}and (Img <> nil){$ENDIF} then
begin
// close buttons enabled
if TabCloseBtnWidget = nil then
begin
// there is no close button yet
// -> add one to the right side of the label in the tab
TabCloseBtnWidget := gtk_button_new;
gtk_button_set_relief(PGtkButton(TabCloseBtnWidget), GTK_RELIEF_NONE);
{$ifdef gtk2}
gtk_button_set_focus_on_click(PGtkButton(TabCloseBtnWidget), False);
style := gtk_widget_get_modifier_style(TabCloseBtnWidget);
style^.xthickness := 0;
style^.ythickness := 0;
gtk_widget_modify_style(TabCloseBtnWidget, style);
{$endif}
gtk_object_set_data(PGtkObject(TabWidget), 'TabCloseBtn',
TabCloseBtnWidget);
// put a pixmap into the button
{$IfDef GTK1}
TabCloseBtnImageWidget:=gtk_pixmap_new(Img,Mask);
{$Else}
TabCloseBtnImageWidget:=gtk_image_new_from_stock(GTK_STOCK_CLOSE, GTK_ICON_SIZE_MENU);
{$EndIf}
gtk_object_set_data(PGtkObject(TabCloseBtnWidget),'TabCloseBtnImage',
TabCloseBtnImageWidget);
gtk_widget_show(TabCloseBtnImageWidget);
gtk_container_add(PGtkContainer(TabCloseBtnWidget),
TabCloseBtnImageWidget);
gtk_widget_show(TabCloseBtnWidget);
g_signal_connect(PGtkObject(TabCloseBtnWidget), 'clicked',
TGTKSignalFunc(@gtkNoteBookCloseBtnClicked), APage);
gtk_box_pack_start(PGtkBox(TabWidget), TabCloseBtnWidget, False, False, 0);
end;
end else begin
// close buttons disabled
if TabCloseBtnWidget<>nil then begin
// there is a close button
// -> remove it
gtk_object_set_data(PGtkObject(TabWidget), 'TabCloseBtn',
nil);
DestroyWidget(TabCloseBtnWidget);
TabCloseBtnWidget:=nil;
end;
end;
end;
begin
ThePage := TCustomPage(APage);
TheNoteBook := TCustomTabControl(ANoteBook);
if (APage=nil) or (not ThePage.HandleAllocated) then exit;
if TheNoteBook=nil then begin
TheNoteBook:=TCustomTabControl(ThePage.Parent);
if TheNoteBook=nil then exit;
end;
NoteBookWidget:=PGtkWidget(TWinControl(TheNoteBook).Handle);
PageWidget:=PGtkWidget(TWinControl(ThePage).Handle);
// get the tab container and the tab components: pixmap, label and closebtn
TabWidget:=gtk_notebook_get_tab_label(PGtkNoteBook(NotebookWidget),
PageWidget);
if TabWidget=nil then exit;
TabImageWidget:=gtk_object_get_data(PGtkObject(TabWidget), 'TabImage');
TabLabelWidget:=gtk_object_get_data(PGtkObject(TabWidget), 'TabLabel');
TabCloseBtnWidget:=gtk_object_get_data(PGtkObject(TabWidget),'TabCloseBtn');
// get the menu container and its components: pixmap and label
MenuWidget:=gtk_notebook_get_menu_label(PGtkNoteBook(NotebookWidget),
PageWidget);
if MenuWidget<>nil then begin
MenuImageWidget:=gtk_object_get_data(PGtkObject(MenuWidget), 'TabImage');
MenuLabelWidget:=gtk_object_get_data(PGtkObject(MenuWidget), 'TabLabel');
end else begin
MenuImageWidget:=nil;
MenuLabelWidget:=nil;
end;
UpdateTabImage;
UpdateTabLabel;
UpdateTabCloseBtn;
end;
{-------------------------------------------------------------------------------
GetWidgetScreenPos
Returns the absolute left top position of a widget on the screen.
-------------------------------------------------------------------------------}
function GetWidgetOrigin(TheWidget: PGtkWidget): TPoint;
var
TheWindow: PGdkWindow;
{$IFDEF RaiseExceptionOnNilPointers}
LCLObject: TObject;
{$ENDIF}
begin
TheWindow:=GetControlWindow(TheWidget);
if TheWindow<>nil then begin
BeginGDKErrorTrap;
gdk_window_get_origin(TheWindow,@Result.X,@Result.Y);
EndGDKErrorTrap;
end else begin
{$IFDEF RaiseExceptionOnNilPointers}
LCLobject:=GetLCLObject(TheWidget);
DbgOut('GetWidgetOrigin ');
if LCLObject=nil then
DbgOut(' LCLObject=nil')
else if LCLObject is TControl then
DbgOut(' LCLObject=',TControl(LCLObject).Name,':',TControl(LCLObject).ClassName)
else
DbgOut(' LCLObject=',TControl(LCLObject).ClassName);
DebugLn('');
RaiseException('GetWidgetOrigin Window=nil');
{$ENDIF}
Result.X:=0;
Result.Y:=0;
end;
// check if the gdkwindow is the clientwindow of the parent
if (TheWidget^.Parent<>nil) and (gtk_widget_get_parent_window(TheWidget)=TheWindow) then begin
// the widget is using its parent window
// -> adjust the coordinates
inc(Result.X,TheWidget^.Allocation.X);
inc(Result.Y,TheWidget^.Allocation.Y);
end;
end;
{-------------------------------------------------------------------------------
GetWidgetClientScreenPos
Returns the absolute left top position of a widget's client area
on the screen.
-------------------------------------------------------------------------------}
function GetWidgetClientOrigin(TheWidget: PGtkWidget): TPoint;
{$IFDEF Gtk2}
procedure GetNoteBookClientOrigin(NBWidget: PGtkNotebook);
var
PageIndex: LongInt;
PageWidget: PGtkWidget;
ClientWidget: PGTKWidget;
FrameBorders: TRect;
begin
// get current page
PageIndex:=gtk_notebook_get_current_page(NBWidget);
if PageIndex>=0 then
PageWidget:=gtk_notebook_get_nth_page(NBWidget,PageIndex)
else
PageWidget:=nil;
// get client widget of page
if (PageWidget<>nil) then
ClientWidget:=GetFixedWidget(PageWidget)
else
ClientWidget:=nil;
// Be careful while using ClientWidget here, it may be nil
if (ClientWidget<>nil) and (ClientWidget^.window<>nil) then
begin
// get the position of the current page
gdk_window_get_origin(ClientWidget^.window,@Result.X,@Result.Y);
if GTK_WIDGET_NO_WINDOW(ClientWidget)
then begin
Inc(Result.X, ClientWidget^.Allocation.X);
Inc(Result.Y, ClientWidget^.Allocation.Y);
end;
end
else
begin
// use defaults
Result:=GetWidgetOrigin(TheWidget);
FrameBorders:=GetStyleNotebookFrameBorders;
GetWidgetClientOrigin.x:=Result.x+FrameBorders.Left;
GetWidgetClientOrigin.y:=Result.y+FrameBorders.Top;
end;
end;
{$ENDIF}
var
ClientWidget: PGtkWidget;
ClientWindow: PGdkWindow;
begin
ClientWidget := GetFixedWidget(TheWidget);
if ClientWidget <> TheWidget then
begin
ClientWindow := GetControlWindow(ClientWidget);
if ClientWindow <> nil then
begin
{$IFDEF DebugGDK}
BeginGDKErrorTrap;
{$ENDIF}
gdk_window_get_origin(ClientWindow, @Result.X, @Result.Y);
{$Ifdef GTK2}
if GTK_WIDGET_NO_WINDOW(ClientWidget) then
begin
Inc(Result.X, ClientWidget^.Allocation.X);
Inc(Result.Y, ClientWidget^.Allocation.Y);
end;
{$EndIf}
{$IFDEF DebugGDK}
EndGDKErrorTrap;
{$ENDIF}
exit;
end;
{$IFDEF Gtk2}
end
else
if GtkWidgetIsA(TheWidget,GTK_TYPE_NOTEBOOK) then
begin
GetNoteBookClientOrigin(PGtkNoteBook(TheWidget));
Exit;
{$ENDIF}
end;
Result := GetWidgetOrigin(TheWidget);
end;
{-------------------------------------------------------------------------------
TranslateGdkPointToClientArea
Translates SourcePos relative to SourceWindow to a coordinate relative to the
client area of the LCL WinControl.
-------------------------------------------------------------------------------}
function TranslateGdkPointToClientArea(SourceWindow: PGdkWindow;
SourcePos: TPoint; DestinationWidget: PGtkWidget): TPoint;
var
SrcWindowOrigin: TPoint;
ClientAreaWindowOrigin: TPoint;
Src2ClientAreaVector: TPoint;
begin
if SourceWindow = nil then
begin
{$IFDEF RaiseExceptionOnNilPointers}
RaiseException('TranslateGdkPointToClientArea Window=nil');
{$ENDIF}
DebugLn('WARNING: TranslateGdkPointToClientArea SourceWindow=nil');
end;
gdk_window_get_origin(SourceWindow, @SrcWindowOrigin.X, @SrcWindowOrigin.Y);
ClientAreaWindowOrigin := GetWidgetClientOrigin(DestinationWidget);
Src2ClientAreaVector.X := ClientAreaWindowOrigin.X - SrcWindowOrigin.X;
Src2ClientAreaVector.Y := ClientAreaWindowOrigin.Y - SrcWindowOrigin.Y;
Result.X := SourcePos.X - Src2ClientAreaVector.X;
Result.Y := SourcePos.Y - Src2ClientAreaVector.Y;
end;
function SubtractScoll(AWidget: PGtkWidget; APosition: TPoint): TPoint;
begin
Result := APosition;
AWidget := gtk_object_get_data(PGTKObject(AWidget), odnScrollArea);
if GTK_IS_SCROLLED_WINDOW(AWidget) then
begin
with gtk_scrolled_window_get_hadjustment(PGtkScrolledWindow(AWidget))^ do
dec(Result.x, Trunc(value - lower));
with gtk_scrolled_window_get_vadjustment(PGtkScrolledWindow(AWidget))^ do
dec(Result.y, Trunc(value - lower));
end;
end;
{------------------------------------------------------------------------------
Function: UpdateMouseCaptureControl
Params: none
Returns: none
Sets MouseCaptureWidget to the current capturing widget.
------------------------------------------------------------------------------}
procedure UpdateMouseCaptureControl;
var
OldMouseCaptureWidget,
CurMouseCaptureWidget: PGtkWidget;
begin
OldMouseCaptureWidget:=MouseCaptureWidget;
CurMouseCaptureWidget:=gtk_grab_get_current;
if OldMouseCaptureWidget<>CurMouseCaptureWidget then begin
// the mouse grab changed
// -> this means the gtk itself has changed the mouse grab
{$IFDEF VerboseMouseCapture}
DebugLn('UpdateMouseCaptureControl Capture changed from ',
'[',GetWidgetDebugReport(OldMouseCaptureWidget),' type=',MouseCaptureTypeNames[MouseCaptureType],']',
' to [',GetWidgetDebugReport(CurMouseCaptureWidget),' type=GTK]');
if CurMouseCaptureWidget<>nil then
DebugLn('parent ', GetWidgetDebugReport(CurMouseCaptureWidget^.Parent));
{$ENDIF}
// notify the new capture control
MouseCaptureWidget:=CurMouseCaptureWidget;
MouseCaptureType:=mctGTK;
if MouseCaptureWidget<>nil then begin
// the MouseCaptureWidget is probably not a main widget
SendMessage(HWnd(PtrUInt(MouseCaptureWidget)), LM_CAPTURECHANGED, 0,
HWnd(PtrUInt(OldMouseCaptureWidget)));
end;
end;
end;
procedure IncreaseMouseCaptureIndex;
begin
if MouseCaptureIndex<$ffffffff then
inc(MouseCaptureIndex)
else
MouseCaptureIndex:=0;
end;
procedure CaptureMouseForWidget(Widget: PGtkWidget; Owner: TMouseCaptureType);
var
CaptureWidget: PGtkWidget;
NowIndex: Cardinal;
begin
{$IFDEF VerboseMouseCapture}
DebugLn('CaptureMouseForWidget START ',GetWidgetDebugReport(Widget));
{$ENDIF}
if not (Owner in [mctGTKIntf,mctLCL]) then exit;
// not every widget can capture the mouse
CaptureWidget:=GetDefaultMouseCaptureWidget(Widget);
if CaptureWidget=nil then exit;
UpdateMouseCaptureControl;
if (MouseCaptureType<>mctGTK) then begin
// we are capturing
if (MouseCaptureWidget=CaptureWidget) then begin
// we are already capturing this widget
exit;
end;
// release old capture
ReleaseMouseCapture;
end;
{$IFDEF VerboseMouseCapture}
DebugLn('CaptureMouseForWidget Start Capturing for ',GetWidgetDebugReport(CaptureWidget));
{$ENDIF}
IncreaseMouseCaptureIndex;
NowIndex:=MouseCaptureIndex;
if not gtk_widget_has_focus(CaptureWidget) then
gtk_widget_grab_focus(CaptureWidget);
if NowIndex=MouseCaptureIndex then begin
{$IFDEF VerboseMouseCapture}
DebugLn('CaptureMouseForWidget Commit Capturing for ',GetWidgetDebugReport(CaptureWidget));
{$ENDIF}
MouseCaptureWidget:=CaptureWidget;
MouseCaptureType:=Owner;
gtk_grab_add(CaptureWidget);
end;
end;
function GetDefaultMouseCaptureWidget(Widget: PGtkWidget
): PGtkWidget;
var
WidgetInfo: PWinWidgetInfo;
LCLObject: TObject;
begin
Result:=nil;
if Widget=nil then exit;
if GtkWidgetIsA(Widget,GTKAPIWidget_Type) then begin
WidgetInfo:=GetWidgetInfo(Widget,false);
if WidgetInfo<>nil then
Result:=WidgetInfo^.CoreWidget;
exit;
end;
LCLObject:=GetNearestLCLObject(Widget);
if LCLObject=nil then exit;
if (TWinControl(LCLObject) is TCustomSplitter) and (TWinControl(LCLObject).HandleAllocated)
then begin
WidgetInfo:=GetWidgetInfo(PGtkWidget(TWinControl(LCLObject).Handle),false);
if WidgetInfo<>nil then
Result:=WidgetInfo^.CoreWidget;
end;
end;
{------------------------------------------------------------------------------
procedure ReleaseMouseCapture;
If the current mouse capture was captured by the LCL or the gtk intf, release
the capture. Don't release mouse captures of the gtk, because captures must
be balanced and this is already done by the gtk.
------------------------------------------------------------------------------}
procedure ReleaseMouseCapture;
var
OldMouseCaptureWidget: PGtkWidget;
Info: PWidgetInfo;
begin
{$IFDEF VerboseMouseCapture}
DebugLn('ReleaseMouseCapture ',dbgs(ord(MouseCaptureType)),' MouseCaptureWidget=[',GetWidgetDebugReport(MouseCaptureWidget),']');
{$ENDIF}
if MouseCaptureType=mctGTK then
begin
Info := GetWidgetInfo(gtk_grab_get_current, false);
if (Info <> nil) and (Info^.CoreWidget <> nil) then
begin
if GtkWidgetIsA(Info^.CoreWidget, gtk_list_get_type) then
begin
// Paul Ishenin:
// listbox grabs pointer and other control for itself, when we click on listbox item
// also it changes its state to drag_selection
// this is not expected in LCL and as result cause bugs, such as 7892
// so we need end drag selection manually
OldMouseCaptureWidget := Info^.CoreWidget;
gtk_list_end_drag_selection(PGtkList(OldMouseCaptureWidget));
end;
end;
exit;
end;
OldMouseCaptureWidget:=MouseCaptureWidget;
MouseCaptureWidget:=nil;
MouseCaptureType:=mctGTK;
if OldMouseCaptureWidget<>nil then
gtk_grab_remove(OldMouseCaptureWidget);
// tell the LCL
SetCaptureControl(nil);
end;
procedure ReleaseCaptureWidget(Widget : PGtkWidget);
begin
if (Widget=nil)
or ((MouseCaptureWidget<>Widget) and (MouseCaptureWidget<>Widget^.parent))
then
exit;
DebugLn('ReleaseCaptureWidget ',GetWidgetDebugReport(Widget));
ReleaseMouseCapture;
end;
{-------------------------------------------------------------------------------
procedure: SignalConnect
Params: AWidget: PGTKWidget
ASignal: PChar
AProc: Pointer
AInfo: PWidgetInfo
Returns: Nothing
Connects a gtk signal handler.
This is a wrapper to get around gtk casting
-------------------------------------------------------------------------------}
procedure SignalConnect(const AWidget: PGTKWidget; const ASignal: PChar;
const AProc: Pointer; const AInfo: PWidgetInfo);
begin
g_signal_connect(PGtkObject(AWidget), ASignal, TGTKSignalFunc(AProc), AInfo);
end;
{-------------------------------------------------------------------------------
procedure: SignalConnectAfter
Params: AWidget: PGTKWidget
ASignal: PChar
AProc: Pointer
AInfo: PGtkWSWidgetInfo
Returns: Nothing
Connects a gtk signal after handler.
This is a wrapper to get around gtk casting
-------------------------------------------------------------------------------}
procedure SignalConnectAfter(const AWidget:PGTKWidget; const ASignal: PChar;
const AProc: Pointer; const AInfo: PWidgetInfo);
begin
g_signal_connect_after(PGTKObject(AWidget), ASignal, TGTKSignalFunc(AProc), AInfo);
end;
{-------------------------------------------------------------------------------
procedure ConnectSignal(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ReqSignalMask: TGdkEventMask;
Flags: TConnectSignalFlags);
Connects a gtk signal handler.
-------------------------------------------------------------------------------}
procedure InitDesignSignalMasks;
var
SignalType: TDesignSignalType;
begin
DesignSignalMasks[dstUnknown]:=0;
for SignalType:=Low(TDesignSignalType) to High(TDesignSignalType) do
DesignSignalMasks[SignalType]:=1 shl ord(SignalType);
end;
function DesignSignalNameToType(Name: PChar; After: boolean): TDesignSignalType;
begin
for Result:=Low(TDesignSignalType) to High(TDesignSignalType) do
if SamePChar(DesignSignalNames[Result],Name)
and (DesignSignalAfter[Result]=After) then exit;
Result:=dstUnknown;
end;
function GetDesignSignalMask(Widget: PGtkWidget): TDesignSignalMask;
begin
Result:=TDesignSignalMask(PtrUInt(gtk_object_get_data(PGtkObject(Widget),
'LCLDesignMask')));
end;
procedure SetDesignSignalMask(Widget: PGtkWidget; NewMask: TDesignSignalMask);
begin
gtk_object_set_data(PGtkObject(Widget),'LCLDesignMask',Pointer(PtrInt(NewMask)));
end;
function GetDesignOnlySignalFlag(Widget: PGtkWidget;
DesignSignalType: TDesignSignalType): boolean;
begin
Result:=(GetDesignSignalMask(Widget)
and DesignSignalMasks[DesignSignalType])<>0;
end;
function SignalConnected(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject;
const ASFlags: TConnectSignalFlags): boolean;
{$IFDEF Gtk1}
var
Handler: PGTKHandler;
SignalID: guint;
begin
Handler := gtk_object_get_data_by_id (AnObject, gtk_handler_quark);
SignalID := g_signal_lookup(ASignal, GTK_OBJECT_TYPE(AnObject));
if SignalID>$ffffff then
RaiseGDBException('SignalConnected');
while (Handler <> nil) do begin
with Handler^ do
begin
// check if signal is already connected
//debugln('ConnectSignal Id=',dbgs(Id));
if (Id > 0)
and (Signal_ID = SignalID)
and (Func = TGTKSignalFunc(ACallBackProc))
and (func_data = Pointer(ALCLObject))
and (((flags and bmSignalAfter)<>0)=(csfAfter in ASFlags))
then begin
// signal is already connected
Result:=true;
Exit;
end;
Handler := Next;
end;
end;
Result:=false;
end;
{$ELSE}
begin
Result:=g_signal_handler_find(AnObject,
G_SIGNAL_MATCH_FUNC or G_SIGNAL_MATCH_DATA,
0,0,nil,ACallBackProc,ALCLObject)<>0;
end;
{$ENDIF}
procedure ConnectSignal(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject;
const AReqSignalMask: TGdkEventMask; const ASFlags: TConnectSignalFlags);
var
WinWidgetInfo: PWinWidgetInfo;
MainWidget: PGtkWidget;
OldDesignMask, NewDesignMask: TDesignSignalMask;
DesignSignalType: TDesignSignalType;
RealizeConnected: Boolean;
HasRealizeSignal: Boolean;
begin
if ACallBackProc = nil then
RaiseGDBException('ConnectSignal');
// first loop through the handlers to:
// - check if a handler already exists
// - Find the realize handler to change data
DesignSignalType:=DesignSignalNameToType(ASignal,csfAfter in ASFlags);
if SignalConnected(AnObject,ASignal,ACallBackProc,ALCLObject,ASFlags) then
begin
// signal is already connected
// update the DesignSignalMask
if (DesignSignalType <> dstUnknown)
and (not (csfDesignOnly in ASFlags))
then begin
OldDesignMask := GetDesignSignalMask(PGtkWidget(AnObject));
NewDesignMask :=OldDesignMask and not DesignSignalMasks[DesignSignalType];
if OldDesignMask <> NewDesignMask
then SetDesignSignalMask(PGtkWidget(AnObject),NewDesignMask);
end;
Exit;
end;
// if we are here, then no handler was defined yet
// -> register handler
//if (Msg=LM_LBUTTONUP) then DebugLn('CONNECT ',ReqSignalMask,' Widget=',DbgS(AnObject));
//debugln('ConnectSignal ',DbgSName(ALCLObject),' ',ASignal,' After=',dbgs(csfAfter in ASFlags));
if csfAfter in ASFlags then
g_signal_connect_after(AnObject, ASignal,
TGTKSignalFunc(ACallBackProc), ALCLObject)
else
g_signal_connect (AnObject, ASignal,
TGTKSignalFunc(ACallBackProc), ALCLObject);
// update signal mask which will be set in the realize handler
if (csfUpdateSignalMask in ASFlags) and (AReqSignalMask <> 0)
then begin
MainWidget := GetMainWidget(PGtkWidget(AnObject));
if MainWidget=nil
then MainWidget := PGtkWidget(AnObject);
WinWidgetInfo := GetWidgetInfo(MainWidget,true);
WinWidgetInfo^.EventMask := WinWidgetInfo^.EventMask or AReqSignalMask;
end;
// -> register realize handler
if (csfConnectRealize in ASFlags) then begin
HasRealizeSignal:=g_signal_lookup('realize', GTK_OBJECT_TYPE(AnObject))>0;
if HasRealizeSignal then begin
RealizeConnected:=SignalConnected(AnObject,'realize',@GTKRealizeCB,
ALCLObject,[]);
if not RealizeConnected then begin
g_signal_connect(AnObject, 'realize',
TGTKSignalFunc(@GTKRealizeCB), ALCLObject);
g_signal_connect_after(AnObject, 'realize',
TGTKSignalFunc(@GTKRealizeAfterCB), ALCLObject);
end;
end;
end;
// update the DesignSignalMask
if (DesignSignalType <> dstUnknown)
then begin
OldDesignMask:=GetDesignSignalMask(PGtkWidget(AnObject));
if csfDesignOnly in ASFlags then
NewDesignMask:=OldDesignMask or DesignSignalMasks[DesignSignalType]
else
NewDesignMask:=OldDesignMask and not DesignSignalMasks[DesignSignalType];
if OldDesignMask<>NewDesignMask then
SetDesignSignalMask(PGtkWidget(AnObject),NewDesignMask);
end;
end;
procedure ConnectSignal(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject;
const AReqSignalMask: TGdkEventMask);
begin
ConnectSignal(AnObject,ASignal,ACallBackProc, ALCLObject, AReqSignalMask,
[csfConnectRealize,csfUpdateSignalMask]);
end;
procedure ConnectSignalAfter(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject;
const AReqSignalMask: TGdkEventMask);
begin
ConnectSignal(AnObject,ASignal,ACallBackProc, ALCLObject, AReqSignalMask,
[csfConnectRealize,csfUpdateSignalMask,csfAfter]);
end;
procedure ConnectSignal(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject);
begin
ConnectSignal(AnObject,ASignal,ACallBackProc, ALCLObject, 0);
end;
procedure ConnectSignalAfter(const AnObject:PGTKObject; const ASignal: PChar;
const ACallBackProc: Pointer; const ALCLObject: TObject);
begin
ConnectSignalAfter(AnObject,ASignal,ACallBackProc, ALCLObject, 0);
end;
{------------------------------------------------------------------------------
procedure: ConnectInternalWidgetsSignals
Params: AWidget: PGtkWidget; AWinControl: TWinControl
Returns: Nothing
Connects hidden child widgets signals.
Many gtk widgets create internally child widgets (e.g. scrollbars). In
Design mode these widgets should not auto react themselves, but instead send
messages to the lcl. Therefore these widgets are connected also to our
signal handlers.
This procedure is called by the realize-after handler of all LCL widgets
and each time the design mode of a LCL control changes.
------------------------------------------------------------------------------}
procedure ConnectInternalWidgetsSignals(AWidget: PGtkWidget;
AWinControl: TWinControl);
function WidgetIsInternal(TheWidget: PGtkWidget): boolean;
begin
Result:=(TheWidget<>nil)
and (PGtkWidget(AWinControl.Handle)<>TheWidget)
and (GetMainWidget(TheWidget)=nil);
end;
procedure ConnectSignals(TheWidget: PGtkWidget); forward;
procedure ConnectChilds(TheWidget: PGtkWidget);
var
ScrolledWindow: PGtkScrolledWindow;
BinWidget: PGtkBin;
{$IFDEF Gtk2}
ChildEntry2: PGList;
{$ELSE}
ChildEntry: PGSList;
{$ENDIF}
ChildWidget: PGtkWidget;
begin
//if AWinControl is TListView then DebugLn('ConnectChilds A ',DbgS(TheWidget));
if GtkWidgetIsA(TheWidget,GTK_TYPE_CONTAINER) then begin
//if AWinControl is TListView then DebugLn('ConnectChilds B ');
// this is a container widget -> connect all children
{$IFDEF Gtk2}
ChildEntry2:=gtk_container_get_children(PGtkContainer(TheWidget));
while ChildEntry2<>nil do begin
ChildWidget:=PGtkWidget(ChildEntry2^.Data);
if ChildWidget<>TheWidget then
ConnectSignals(ChildWidget);
ChildEntry2:=ChildEntry2^.Next;
end;
{$ELSE}
ChildEntry:=PGtkContainer(TheWidget)^.resize_widgets;
while ChildEntry<>nil do begin
ChildWidget:=PGtkWidget(ChildEntry^.Data);
ConnectSignals(ChildWidget);
ChildEntry:=ChildEntry^.Next;
end;
{$endif}
end;
if GtkWidgetIsA(TheWidget,GTK_TYPE_BIN) then begin
//if AWinControl is TListView then DebugLn('ConnectChilds C ');
BinWidget:=PGtkBin(TheWidget);
ConnectSignals(BinWidget^.child);
end;
if GtkWidgetIsA(TheWidget,GTK_TYPE_SCROLLED_WINDOW) then begin
//if AWinControl is TListView then DebugLn('ConnectChilds D ');
ScrolledWindow:=PGtkScrolledWindow(TheWidget);
ConnectSignals(ScrolledWindow^.hscrollbar);
ConnectSignals(ScrolledWindow^.vscrollbar);
end;
if GtkWidgetIsA(TheWidget,GTK_TYPE_COMBO) then begin
//if AWinControl is TListView then DebugLn('ConnectChilds E ');
ConnectSignals(PGtkCombo(TheWidget)^.entry);
ConnectSignals(PGtkCombo(TheWidget)^.button);
end;
end;
procedure ConnectSignals(TheWidget: PGtkWidget);
var
LCLObject, HiddenLCLObject: TObject;
DesignSignalType: TDesignSignalType;
DesignFlags: TConnectSignalFlags;
begin
//if AWinControl is TListView then DebugLn('ConnectSignals A ',DbgS(TheWidget));
if TheWidget=nil then exit;
// check if TheWidget belongs to another LCL object
LCLObject:=GetLCLObject(TheWidget);
HiddenLCLObject:=GetHiddenLCLObject(TheWidget);
if (LCLObject<>nil) and (LCLObject<>AWinControl) then begin
exit;
end;
if (HiddenLCLObject<>nil) and (HiddenLCLObject<>AWinControl) then begin
exit;
end;
//if AWinControl is TListView then DebugLn('ConnectSignals B ',DbgS(TheWidget));
// connect signals needed for design mode:
for DesignSignalType:=Low(TDesignSignalType) to High(TDesignSignalType) do
begin
if DesignSignalType=dstUnknown then continue;
if (not DesignSignalBefore[DesignSignalType])
and (not DesignSignalAfter[DesignSignalType]) then
continue;
DesignFlags:=[csfDesignOnly];
if DesignSignalAfter[DesignSignalType] then
Include(DesignFlags,csfAfter);
ConnectSignal(PGtkObject(TheWidget),DesignSignalNames[DesignSignalType],
DesignSignalFuncs[DesignSignalType],AWinControl,0,
DesignFlags);
end;
if WidgetIsInternal(TheWidget) then
// mark widget as 'hidden' connected
SetHiddenLCLObject(TheWidget,AWinControl);
// connect recursively ...
ConnectChilds(TheWidget);
end;
begin
if (AWinControl=nil) or (AWidget=nil)
or (not (csDesigning in AWinControl.ComponentState)) then exit;
ConnectSignals(AWidget);
end;
// ----------------------------------------------------------------------
// The Accelgroup and AccelKey is needed by menus
// ----------------------------------------------------------------------
function GetAccelGroup(const Widget: PGtkWidget;
CreateIfNotExists: boolean): PGTKAccelGroup;
begin
Result := PGTKAccelGroup(gtk_object_get_data(PGtkObject(Widget),'AccelGroup'));
if (Result=nil) and CreateIfNotExists then begin
{$IFDEF VerboseAccelerator}
DebugLn('GetAccelGroup CREATING Widget=',DbgS(Widget),' CreateIfNotExists=',dbgs(CreateIfNotExists));
{$ENDIF}
Result:=gtk_accel_group_new;
SetAccelGroup(Widget,Result);
if GtkWidgetIsA(Widget,GTK_TYPE_WINDOW) then
ShareWindowAccelGroups(Widget);
end;
end;
procedure SetAccelGroup(const Widget: PGtkWidget;
const AnAccelGroup: PGTKAccelGroup);
begin
if (Widget = nil) then exit;
gtk_object_set_data(PGtkObject(Widget), 'AccelGroup', AnAccelGroup);
if AnAccelGroup<>nil then begin
// attach group to widget
{$IFDEF VerboseAccelerator}
DebugLn(['SetAccelGroup AnAccelGroup=',DbgS(AnAccelGroup),' IsMenu=',GtkWidgetIsA(Widget,GTK_TYPE_MENU)]);
{$ENDIF}
if GtkWidgetIsA(Widget,GTK_TYPE_MENU) then
gtk_menu_set_accel_group(PGtkMenu(Widget), AnAccelGroup)
else begin
{$IfDef GTK2}
Assert(GtkWidgetIsA(Widget,GTK_TYPE_WINDOW));
gtk_window_add_accel_group(GTK_WINDOW(widget), AnAccelGroup);
{$else}
gtk_accel_group_attach(AnAccelGroup, PGtkObject(Widget));
{$endif}
end;
end;
end;
procedure FreeAccelGroup(const Widget: PGtkWidget);
var
AccelGroup: PGTKAccelGroup;
begin
AccelGroup:=GetAccelGroup(Widget,false);
if AccelGroup<>nil then begin
{$IFDEF VerboseAccelerator}
DebugLn('FreeAccelGroup AccelGroup=',DbgS(AccelGroup));
{$ENDIF}
gtk_accel_group_unref(AccelGroup);
SetAccelGroup(Widget,nil);
end;
end;
procedure ShareWindowAccelGroups(AWindow: PGtkWidget);
procedure AttachUnique(TheWindow: PGtkWidget; TheAccelGroup: PGTKAccelGroup);
begin
{$IfDef GTK2}
if (TheWindow=nil) or (TheAccelGroup=nil)
or (TheAccelGroup^.acceleratables=nil)
or (g_slist_find(TheAccelGroup^.acceleratables, TheWindow)=nil)
then
exit;
gtk_window_add_accel_group(GTK_WINDOW(TheWindow), TheAccelGroup);
{$else}
if (TheAccelGroup=nil)
or ((TheAccelGroup^.attach_objects<>nil)
and (g_slist_find(TheAccelGroup^.attach_objects, TheWindow)<>nil))
then
exit;
gtk_accel_group_attach(TheAccelGroup, PGtkObject(TheWindow));
{$endif}
end;
var
TheForm, CurForm: TCustomForm;
i: integer;
TheAccelGroup, CurAccelGroup: PGTKAccelGroup;
CurWindow: PGtkWidget;
begin
TheForm:=TCustomForm(GetLCLObject(AWindow));
// check if visible TCustomForm (not frame)
if (TheForm=nil) or (not (TheForm is TCustomForm))
or (not TheForm.Visible) or (TheForm.Parent<>nil)
or (csDesigning in TheForm.ComponentState)
then
exit;
// check if modal form
if fsModal in TheForm.FormState then begin
// a modal form does not share accelerators
exit;
end;
// check if there is an accelerator group
TheAccelGroup:=GetAccelGroup(AWindow,false);
// this is a normal form
// -> share accelerators with all other visible normal forms
for i:=0 to Screen.FormCount-1 do begin
CurForm:=Screen.Forms[i];
if (CurForm=TheForm)
or (not CurForm.HandleAllocated)
or (not CurForm.Visible)
or (fsModal in CurForm.FormState)
or (CurForm.Parent<>nil)
or (csDesigning in CurForm.ComponentState)
then continue;
CurWindow:=PGtkWidget(CurForm.Handle);
CurAccelGroup:=GetAccelGroup(CurWindow,false);
{$IFDEF VerboseAccelerator}
DebugLn('ShareWindowAccelGroups ',TheForm.Name,':',TheForm.ClassName,
' <-> ',CurForm.Name,':',CurForm.ClassName);
{$ENDIF}
// cross connect
AttachUnique(CurWindow,TheAccelGroup);
AttachUnique(AWindow,CurAccelGroup);
end;
end;
procedure UnshareWindowAccelGroups(AWindow: PGtkWidget);
procedure Detach(TheWindow: PGtkWidget; TheAccelGroup: PGTKAccelGroup);
begin
{$IfDef GTK2}
if (TheWindow=nil) or (TheAccelGroup=nil)
or (TheAccelGroup^.acceleratables=nil)
or (g_slist_find(TheAccelGroup^.acceleratables, TheWindow)=nil)
then
exit;
gtk_window_remove_accel_group(GTK_WINDOW(TheWindow), TheAccelGroup);
{$else}
if (TheAccelGroup=nil)
or (TheAccelGroup^.attach_objects=nil)
or (g_slist_find(TheAccelGroup^.attach_objects, TheWindow)=nil)
then
exit;
gtk_accel_group_detach(TheAccelGroup, PGtkObject(TheWindow));
{$endif}
end;
var
TheForm, CurForm: TCustomForm;
i: integer;
TheAccelGroup, CurAccelGroup: PGTKAccelGroup;
CurWindow: PGtkWidget;
begin
TheForm:=TCustomForm(GetLCLObject(AWindow));
// check if TCustomForm
if (TheForm=nil) or (not (TheForm is TCustomForm))
then exit;
TheAccelGroup:=GetAccelGroup(AWindow,false);
// -> unshare accelerators with all other forms
for i:=0 to Screen.FormCount-1 do begin
CurForm:=Screen.Forms[i];
if (CurForm=TheForm)
or (not CurForm.HandleAllocated)
then continue;
CurWindow:=PGtkWidget(CurForm.Handle);
CurAccelGroup:=GetAccelGroup(CurWindow,false);
{$IFDEF VerboseAccelerator}
DebugLn('UnshareWindowAccelGroups ',TheForm.Name,':',TheForm.ClassName,
' <-> ',CurForm.Name,':',CurForm.ClassName);
{$ENDIF}
// unlink
Detach(CurWindow,TheAccelGroup);
Detach(AWindow,CurAccelGroup);
end;
end;
function GetAccelGroupForComponent(Component: TComponent;
CreateIfNotExists: boolean): PGTKAccelGroup;
var
Control: TControl;
MenuItem: TMenuItem;
Form: TCustomForm;
Menu: TMenu;
begin
Result:=nil;
if Component=nil then exit;
if Component is TMenuItem then begin
MenuItem:=TMenuItem(Component);
Menu:=MenuItem.GetParentMenu;
if (Menu=nil) or (Menu.Parent=nil) then exit;
{$IFDEF VerboseAccelerator}
DebugLn('GetAccelGroupForComponent A ',Component.Name,':',Component.ClassName);
{$ENDIF}
Result:=GetAccelGroupForComponent(Menu.Parent,CreateIfNotExists);
end else if Component is TControl then begin
Control:=TControl(Component);
while Control.Parent<>nil do Control:=Control.Parent;
if Control is TCustomForm then begin
Form:=TCustomForm(Control);
if Form.HandleAllocated then begin
Result:=GetAccelGroup(PGtkWidget(Form.Handle),CreateIfNotExists);
{$IFDEF VerboseAccelerator}
DebugLn('GetAccelGroupForComponent C ',Component.Name,':',Component.ClassName);
{$ENDIF}
end;
end;
end;
{$IFDEF VerboseAccelerator}
DebugLn('GetAccelGroupForComponent END ',Component.Name,':',Component.ClassName,' Result=',DbgS(Result));
{$ENDIF}
end;
function GetAccelKey(Widget: PGtkWidget): PAcceleratorKey;
begin
Result := PAcceleratorKey(gtk_object_get_data(PGtkObject(Widget),'AccelKey'));
end;
function SetAccelKey(const Widget: PGtkWidget;
Key: guint; Mods: TGdkModifierType; const Signal: string): PAcceleratorKey;
begin
if (Widget = nil) then exit(nil);
Result:=GetAccelKey(Widget);
if Result=nil then begin
if Key>0 then begin
New(Result);
FillChar(Result^,SizeOf(Result),0);
end;
end else begin
if Key=0 then begin
Dispose(Result);
Result:=nil;
end;
end;
if (Result<>nil) then begin
Result^.Key:=Key;
Result^.Mods:=Mods;
Result^.Signal:=Signal;
Result^.Realized:=false;
end;
{$IFDEF VerboseAccelerator}
DebugLn('SetAccelKey Widget=',DbgS(Widget),
' Key=',dbgs(Key),' Mods=',DbgS(Mods),
' Signal="',Signal,'" Result=',DbgS(Result));
{$ENDIF}
gtk_object_set_data(PGtkObject(Widget), 'AccelKey', Result);
end;
procedure ClearAccelKey(Widget: PGtkWidget);
begin
SetAccelKey(Widget,0,0,'');
end;
procedure RealizeAccelerator(Component: TComponent; Widget : PGtkWidget);
var
AccelKey: PAcceleratorKey;
AccelGroup: PGTKAccelGroup;
begin
if (Component=nil) or (Widget=nil) then
RaiseGDBException('RealizeAccelerate: invalid input');
// Set the accelerator
AccelKey:=GetAccelKey(Widget);
if (AccelKey=nil) or (AccelKey^.Realized) then exit;
if AccelKey^.Key>0 then begin
AccelGroup:=GetAccelGroupForComponent(Component,true);
if AccelGroup<>nil then begin
{$IFDEF VerboseAccelerator}
DebugLn('RealizeAccelerator Add Accelerator ',
Component.Name,':',Component.ClassName,
' Widget=',DbgS(Widget),
' Signal=',AccelKey^.Signal,
' Key=',dbgs(AccelKey^.Key),' Mods=',dbgs(AccelKey^.Mods),
'');
{$ENDIF}
gtk_widget_add_accelerator(Widget, PChar(AccelKey^.Signal),
AccelGroup, AccelKey^.Key, AccelKey^.Mods, GTK_ACCEL_VISIBLE);
AccelKey^.Realized:=true;
end else begin
AccelKey^.Realized:=false;
end;
end else begin
AccelKey^.Realized:=true;
end;
end;
procedure UnrealizeAccelerator(Widget : PGtkWidget);
var
AccelKey: PAcceleratorKey;
begin
if (Widget=nil) then
RaiseGDBException('UnrealizeAccelerate: invalid input');
AccelKey:=GetAccelKey(Widget);
if (AccelKey=nil) or (not AccelKey^.Realized) then exit;
if AccelKey^.Signal<>'' then begin
{$IFDEF VerboseAccelerator}
DebugLn('UnrealizeAccelerator ',
' Widget=',DbgS(Widget),
' Signal=',AccelKey^.Signal,
' Key=',dbgs(AccelKey^.Key),' Mods=',dbgs(AccelKey^.Mods),
'');
{$ENDIF}
{$Ifdef GTK2}
DebugLn('ToDo: gtkproc.inc UnrealizeAccelerator');
{$else}
gtk_widget_remove_accelerators(Widget, PChar(AccelKey^.Signal), false);
{$EndIf}
end;
AccelKey^.Realized:=false;
end;
procedure RegroupAccelerator(Widget: PGtkWidget);
begin
UnrealizeAccelerator(Widget);
RealizeAccelerator(TComponent(GetLCLObject(Widget)),Widget);
end;
procedure Accelerate(Component: TComponent; const Widget : PGtkWidget;
const Key: guint; Mods: TGdkModifierType; const Signal : string);
var
OldAccelKey: PAcceleratorKey;
begin
if (Component=nil) or (Widget=nil) or (Signal='') then
RaiseGDBException('Accelerate: invalid input');
{$IFDEF VerboseAccelerator}
DebugLn('Accelerate ',DbgSName(Component),' Key=',dbgs(Key),' Mods=',DbgS(Mods),' Signal=',Signal);
{$ENDIF}
// delete old accelerator key
OldAccelKey:=GetAccelKey(Widget);
if (OldAccelKey <> nil) then begin
if (OldAccelKey^.Key=Key) and (OldAccelKey^.Mods=Mods)
and (OldAccelKey^.Signal=Signal)
then begin
// no change
exit;
end;
UnrealizeAccelerator(Widget);
end;
// Set the accelerator
SetAccelKey(Widget,Key,Mods,Signal);
if (Key>0) and (not (csDesigning in Component.ComponentState))
then
RealizeAccelerator(Component,Widget);
end;
procedure Accelerate(Component: TComponent; const Widget : PGtkWidget;
const NewShortCut: TShortCut; const Signal : string);
var
GDKModifier: TGdkModifierType;
GDKKey: guint;
NewKey: word;
NewModifier: TShiftState;
Shift: TShiftStateEnum;
begin
{ Map the shift states }
GDKModifier := 0;
ShortCutToKey(NewShortCut, NewKey, NewModifier);
for Shift := Low(Shift) to High(Shift) do
begin
if Shift in NewModifier
then GDKModifier := GDKModifier or MModifiers[Shift].Mask;
end;
// Send the unmodified keysym ?
if (ssShift in NewModifier)
and ((NewKey < VK_F1) or (NewKey > VK_F24))
then GDKKey := GetVKeyInfo(NewKey).KeySym[1]
else GDKKey := GetVKeyInfo(NewKey).KeySym[0];
Accelerate(Component,Widget,GDKKey,GDKModifier,Signal);
end;
{-------------------------------------------------------------------------------
method TGtkWidgetSet LoadPixbufFromLazResource
Params: const ResourceName: string;
var Pixbuf: PGdkPixbuf
Result: none
Loads a pixbuf from a lazarus resource. The resource must be a XPM file.
-------------------------------------------------------------------------------}
procedure LoadPixbufFromLazResource(const ResourceName: string;
var Pixbuf: PGdkPixbuf);
var
ImgData: PPChar;
begin
Pixbuf:=nil;
try
ImgData:=LazResourceXPMToPPChar(ResourceName);
except
on e: Exception do
DebugLn('WARNING: TGtkWidgetSet.LoadXPMFromLazResource: '+e.Message);
end;
{$IFDEF DebugGDK}BeginGDKErrorTrap;{$ENDIF}
{$IFDEF VerboseGdkPixbuf}
debugln('LoadPixbufFromLazResource A1');
{$ENDIF}
pixbuf:=gdk_pixbuf_new_from_xpm_data(ImgData);
{$IFDEF VerboseGdkPixbuf}
debugln('LoadPixbufFromLazResource A2');
{$ENDIF}
{$IFDEF DebugGDK}EndGDKErrorTrap;{$ENDIF}
FreeMem(ImgData);
end;
{-------------------------------------------------------------------------------
method CreatePixbufFromDrawable
Params: ASource: The source drawable
AColorMap: The colormap to use, when nil a matching colormap is passed
AIncludeAplha: If set, the resulting pixmap has an alpha channel
ASrcX, ASrcY: Offset within the source
ADstX, ADstY: Offset within destination
AWidth, AHeight: Size of the new image
Result: New Pixbuf with refcount = 1
Replaces the gdk_pixbuf_get_from_drawable function which is buggy on big endian
X servers when an alpha channel is requested.
-------------------------------------------------------------------------------}
function CreatePixbufFromDrawable(ASource: PGdkDrawable; AColorMap:PGdkColormap; AIncludeAplha: Boolean; ASrcX, ASrcY, ADstX, ADstY, AWidth, AHeight: longint): PGdkPixbuf;
{$ifndef HasX}
const
CanRequestAlpha: Boolean = True;
var
{$else}
var
CanRequestAlpha: Boolean;
{$endif}
PixBuf: PGdkPixBuf;
{$ifdef Windows}
Image: PGdkImage;
{$endif}
begin
{$ifdef HasX}
CanRequestAlpha := BitmapBitOrder(gdk_display) = LSBFirst;
{$endif}
// If Source is GdkBitmap then gdk_pixbuf_get_from_drawable will get
// pixbuf with 2 colors: transparent and white, but we need only Black and White.
// If we all alpha at the end then problem is gone.
CanRequestAlpha := CanRequestAlpha and (gdk_drawable_get_depth(ASource) > 1);
if CanRequestAlpha and AIncludeAplha
then Pixbuf := gdk_pixbuf_new(GDK_COLORSPACE_RGB, True, 8, AWidth, AHeight)
else Pixbuf := nil;
// gtk1 requires always a colormap and fails when none passed
// gtk2 fails when the colormap depth is different than the drawable depth.
// It wil use the correct system map when none passed.
// Bitmaps (depth = 1) don't need a colormap
{$ifdef gtk1}
if AColormap = nil
then AColorMap := gdk_colormap_get_system;
{$else}
if (AColorMap = nil)
and (gdk_drawable_get_depth(ASource) > 1)
and (gdk_drawable_get_colormap(ASource) = nil)
then AColorMap := gdk_colormap_get_system;
{$endif}
{$ifdef Windows}
if gdk_drawable_get_depth(ASource) = 1 then
begin
// Fix gdk error in converter. For 1 bit Byte order is not significant
Image := gdk_drawable_get_image(ASource, ASrcX, ASrcY, AWidth, AHeight);
Image^.byte_order := GDK_MSB_FIRST;
Result := gdk_pixbuf_get_from_image(Pixbuf, Image, nil, ASrcX, ASrcY, ADstX, ADstY, AWidth, AHeight);
gdk_image_unref(Image);
end
else
{$endif}
Result := gdk_pixbuf_get_from_drawable(Pixbuf, ASource, AColorMap, ASrcX, ASrcY, ADstX, ADstY, AWidth, AHeight);
//DbgDumpPixbuf(Result, '');
if CanRequestAlpha then Exit; // we're done
if not AIncludeAplha then Exit;
pixbuf := gdk_pixbuf_add_alpha(Result, false, guchar(0),guchar(0),guchar(0));
gdk_pixbuf_unref(Result);
Result := pixbuf;
end;
{-------------------------------------------------------------------------------
method LoadXPMFromLazResource
Params: const ResourceName: string;
Window: PGdkWindow;
var PixmapImg, PixmapMask: PGdkPixmap
Result: none
Loads a pixmap from a lazarus resource. The resource must be a XPM file.
-------------------------------------------------------------------------------}
procedure LoadXPMFromLazResource(const ResourceName: string;
Window: PGdkWindow; var PixmapImg, PixmapMask: PGdkPixmap);
var
ImgData: PPGChar;
begin
PixmapImg:=nil;
PixmapMask:=nil;
try
ImgData:=PPGChar(LazResourceXPMToPPChar(ResourceName));
except
on e: Exception do
DebugLn('WARNING: TGtkWidgetSet.LoadXPMFromLazResource: '+e.Message);
end;
{$IFDEF DebugGDK}BeginGDKErrorTrap;{$ENDIF}
PixmapImg:=gdk_pixmap_create_from_xpm_d(Window,PixmapMask,nil,ImgData);
{$IFDEF DebugGDK}EndGDKErrorTrap;{$ENDIF}
FreeMem(ImgData);
end;
{------------------------------------------------------------------------------
function MENU_ITEM_CLASS(widget: PGtkWidget): PGtkMenuItemClass;
Returns the gtk klass of a menuitem widget.
------------------------------------------------------------------------------}
function MENU_ITEM_CLASS(widget: PGtkWidget): PGtkMenuItemClass;
begin
Result:=GTK_MENU_ITEM_CLASS(gtk_object_get_class(widget));
end;
{------------------------------------------------------------------------------
function CHECK_MENU_ITEM_CLASS(widget: PGtkWidget): PGtkCheckMenuItemClass;
Returns the gtk klass of a checkmenuitem widget.
------------------------------------------------------------------------------}
function CHECK_MENU_ITEM_CLASS(widget: PGtkWidget): PGtkCheckMenuItemClass;
begin
Result:=GTK_CHECK_MENU_ITEM_CLASS(gtk_object_get_class(widget));
end;
{------------------------------------------------------------------------------
procedure LockRadioGroupOnChange(RadioGroup: PGSList; const ADelta: Integer);
Calls LockOnChange for all groupmembers
------------------------------------------------------------------------------}
procedure LockRadioGroupOnChange(RadioGroup: PGSList; const ADelta: Integer);
begin
while RadioGroup <> nil do
begin
if RadioGroup^.Data <> nil
then LockOnChange(PgtkObject(RadioGroup^.Data), ADelta);
RadioGroup := RadioGroup^.Next;
end;
end;
{------------------------------------------------------------------------------
procedure UpdateRadioGroupChecks(RadioGroup: PGSList);
Set 'checked' for all menuitems in the group
------------------------------------------------------------------------------}
procedure UpdateRadioGroupChecks(RadioGroup: PGSList);
var
CurListItem: PGSList;
MenuItem: PGtkCheckMenuItem;
LCLMenuItem: TMenuItem;
begin
// Check if it is a single entry
if (RadioGroup = nil) or (RadioGroup^.Next = nil)
then Exit;
// Lock whole group for update
LockRadioGroupOnChange(RadioGroup, +1);
CurListItem := RadioGroup;
try
// set active radiomenuitem
while CurListItem <> nil do
begin
MenuItem := PGtkCheckMenuItem(CurListItem^.Data);
if MenuItem<>nil
then begin
LCLMenuItem := TMenuItem(GetLCLObject(MenuItem));
if (LCLMenuItem <> nil)
and (gtk_check_menu_item_get_active(MenuItem) <> LCLMenuItem.Checked)
then gtk_check_menu_item_set_active(MenuItem, LCLMenuItem.Checked);
end;
CurListItem := CurListItem^.Next;
end;
finally
// Unlock whole group for update
LockRadioGroupOnChange(RadioGroup, -1);
end;
end;
{------------------------------------------------------------------------------
procedure DrawMenuItemIcon(MenuItem: PGtkCheckMenuItem;
area: PGdkRectangle); cdecl;
Handler for drawing the icon of a menuitem.
------------------------------------------------------------------------------}
procedure DrawMenuItemIcon(MenuItem: PGtkCheckMenuItem;
Area: PGdkRectangle); cdecl;
var
Widget: PGtkWidget;
Container: PgtkContainer;
ALeft, ATop, BorderWidth: gint;
LCLMenuItem: TMenuItem;
AWindow: PGdkWindow;
IconWidth, IconHeight: integer;
IconSize: TPoint;
{$IFDEF Gtk2}
HorizPadding, ToggleSpacing: Integer;
{$ENDIF}
AEffect: TGraphicsDrawEffect;
AImageList: TCustomImageList;
FreeImageList: Boolean;
AImageIndex: Integer;
ItemBmp: TBitmap;
begin
if (MenuItem=nil) then
exit;
if not (GTK_WIDGET_DRAWABLE (PGtkWidget(MenuItem))) then
exit;
// get icon
LCLMenuItem:=TMenuItem(GetLCLObject(MenuItem));
if LCLMenuItem=nil then begin // needed for gtk2 dialog
if GtkWidgetIsA(PGtkWidget(MenuItem), gtk_check_menu_item_get_type) then
OldCheckMenuItemDrawProc(MenuItem, Area);
Exit;
end;
if not LCLMenuItem.HasIcon then
begin
// call default draw function
if LCLMenuItem.IsCheckItem then
OldCheckMenuItemDrawProc(MenuItem,Area);
exit;
end;
IconSize:=LCLMenuItem.GetIconSize(0);
IconWidth:=IconSize.X;
IconHeight:=IconSize.Y;
// calculate left and top
Widget := PGtkWidget(MenuItem);
AWindow:=GetControlWindow(Widget);
if AWindow = nil then
exit;
Container := GTK_CONTAINER (MenuItem);
BorderWidth := Container^.flag0 and bm_TGtkContainer_border_width;
{$IFDEF Gtk2}
gtk_widget_style_get(PGtkWidget(MenuItem),
'horizontal-padding', @HorizPadding,
'toggle-spacing', @ToggleSpacing,
nil);
ALeft := BorderWidth +
gtk_widget_get_xthickness(gtk_widget_get_style(Widget)) +
HorizPadding +
((PGtkMenuItem(MenuItem)^.toggle_size-ToggleSpacing-IconWidth) div 2);
if gtk_widget_get_direction(Widget) = GTK_TEXT_DIR_RTL then
ALeft := Widget^.Allocation.width - IconWidth - ALeft; //not sure it is the correct Width
{$ELSE}
ALeft := (BorderWidth + gtk_widget_get_xthickness(gtk_widget_get_style(Widget)) + 2)
+((PGtkMenuItem(MenuItem)^.toggle_size-IconWidth) div 2);
{$ENDIF}
ATop := (Widget^.Allocation.Height - IconHeight) div 2;
// draw icon
AImageList := LCLMenuItem.GetImageList;
if AImageList = nil then
begin
AImageList := TImageList.Create(nil);
// prevent multiple calls to GetBitmap;
ItemBmp := LCLMenuItem.Bitmap;
AImageList.Width := ItemBmp.Width; // maybe height to prevent too wide bitmaps?
AImageList.Height := ItemBmp.Height;
if ItemBmp.Masked
then AImageIndex := AImageList.AddMasked(ItemBmp, ItemBmp.TransparentColor)
else AImageIndex := AImageList.Add(ItemBmp, nil);
FreeImageList := True;
end
else
begin
FreeImageList := False;
AImageIndex := LCLMenuItem.ImageIndex;
end;
if not LCLMenuItem.Enabled then
AEffect := gdeDisabled
else
AEffect := gdeNormal;
if AImageIndex < AImageList.Count then
{$IFDEF VerboseGtkToDos}{$note reimplement}{$ENDIF}
DrawImageListIconOnWidget(AImageList, AImageIndex, AEffect,
Widget, false, false, ALeft, ATop);
if FreeImageList then
AImageList.Free;
end;
{------------------------------------------------------------------------------
procedure MenuSizeRequest(widget:PGtkWidget;
requisition:PGtkRequisition); cdecl;
SizeAllocate Handler for check menuitem widgets.
------------------------------------------------------------------------------}
procedure MenuSizeRequest(widget:PGtkWidget; requisition:PGtkRequisition); cdecl;
var
CurToggleSize, MaxToggleSize: integer;
MenuShell: PGtkMenuShell;
ListItem: PGList;
MenuItem: PGtkMenuItem;
CheckMenuItem: PGtkMenuItem;
LCLMenuItem: TMenuItem;
IconSize: TPoint;
begin
MaxToggleSize:=0;
MenuShell:=GTK_MENU_SHELL(widget);
ListItem:=MenuShell^.Children;
CheckMenuItem:=nil;
while ListItem<>nil do begin
MenuItem:=PGtkMenuItem(ListItem^.Data);
if GTK_IS_CHECK_MENU_ITEM(PGtkWidget(MenuItem)) then begin
CheckMenuItem:=MenuItem;
CurToggleSize:=1;
LCLMenuItem:=TMenuItem(GetLCLObject(MenuItem));
if LCLMenuItem<>nil then begin
IconSize:=LCLMenuItem.GetIconSize(0);
{if IconSize.X>100 then
debugln('MenuSizeRequest LCLMenuItem=',LCLMenuItem.Name,' ',LCLMenuItem.Caption,
' ');}
if CurToggleSize<IconSize.X then
CurToggleSize:=IconSize.X;
end;
if MaxToggleSize<CurToggleSize then
MaxToggleSize:=CurToggleSize;
end;
ListItem:=ListItem^.Next;
end;
//DebugLn('MenuSizeRequest A MaxToggleSize=',MaxToggleSize);
{$IFDEF Gtk2}
// Gtk2ToDo
if CheckMenuItem<>nil then begin
GTK_MENU_ITEM(CheckMenuItem)^.toggle_size := 0;
gtk_menu_item_toggle_size_allocate(GTK_MENU_ITEM(CheckMenuItem),MaxToggleSize);
GTK_MENU_ITEM(CheckMenuItem)^.toggle_size := MaxToggleSize;
end;
{$ELSE}
if CheckMenuItem<>nil then
MENU_ITEM_CLASS(PGtkWidget(CheckMenuItem))^.toggle_size:=MaxToggleSize;
{$ENDIF}
//DebugLn('MenuSizeRequest B ',GetWidgetDebugReport(widget),' ',requisition^.width,',',requisition^.height);
OldMenuSizeRequestProc(Widget,requisition);
//DebugLn('MenuSizeRequest C ',GetWidgetDebugReport(widget),' ',requisition^.width,',',requisition^.height);
end;
procedure UpdateInnerMenuItem(LCLMenuItem: TMenuItem; MenuItemWidget: PGtkWidget);
begin
UpdateInnerMenuItem(LCLMenuItem, MenuItemWidget, LCLMenuItem.ShortCut, LCLMenuItem.ShortCutKey2);
end;
{------------------------------------------------------------------------------
Update the inner widgets of a menuitem widget.
------------------------------------------------------------------------------}
procedure UpdateInnerMenuItem(LCLMenuItem: TMenuItem; MenuItemWidget: PGtkWidget;
NewShortCut, ShortCutKey2: TShortCut);
{$ifdef GTK2}
const
WidgetDirection : array[boolean] of longint = (GTK_TEXT_DIR_LTR, GTK_TEXT_DIR_RTL);
{$endif}
function UseRTL: Boolean;
begin
Result := LCLMenuItem.GetIsRightToLeft;
end;
var
HBoxWidget: PGtkWidget;
procedure SetMenuItemLabelText(LCLMenuItem: TMenuItem;
MenuItemWidget: PGtkWidget);
var
LabelWidget: PGtkLabel;
begin
if (MenuItemWidget = nil) or (LCLMenuItem = nil) then
Exit;
LabelWidget := gtk_object_get_data(PGtkObject(MenuItemWidget), 'LCLLabel');
GtkWidgetset.SetLabelCaption(LabelWidget, LCLMenuItem.Caption);
{$ifdef GTK2}
gtk_widget_set_direction(PGtkWidget(LabelWidget), WidgetDirection[UseRTL]);
{$endif}
end;
procedure UpdateShortCutLabel;
var
LabelWidget: PGtkLabel;
NeedShortCut: Boolean;
Key, Key2: Word;
Shift, Shift2: TShiftState;
s: String;
begin
//DebugLn(['UpdateShortCutLabel ',dbgsName(LCLMenuItem),' ',ShortCutToText(NewShortCut)]);
ShortCutToKey(NewShortCut, Key, Shift);
ShortCutToKey(ShortCutKey2, Key2, Shift2);
// Check if shortcut is needed. No shortcut captions for items in menubar
NeedShortCut := (Key <> 0) and
not ( (LCLMenuItem.Parent <> nil) and LCLMenuItem.Parent.HandleAllocated and
GtkWidgetIsA(PGtkWidget(LCLMenuItem.Parent.Handle), GTK_TYPE_MENU_BAR) );
LabelWidget := PGtkLabel(gtk_object_get_data(PGtkObject(MenuItemWidget),'LCLShortCutLabel'));
if NeedShortCut then
begin
s := GetAcceleratorString(Key, Shift);
if Key2 <> 0 then
s := s + ', ' + GetAcceleratorString(Key2, Shift2);
// ShortCutToText(NewShortCut);
if LabelWidget = nil then
begin
// create a label to act as padding
LabelWidget := PGtkLabel(gtk_label_new(''));
gtk_misc_set_padding(PGtkMisc(LabelWidget), 10, 0);
gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget));
gtk_widget_show(PGtkWidget(LabelWidget));
// create a label for the ShortCut
LabelWidget := PGtkLabel(gtk_label_new(PChar(Pointer(s))));
gtk_object_set_data(PGtkObject(MenuItemWidget), 'LCLShortCutLabel', LabelWidget);
gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget));
gtk_widget_show(PGtkWidget(LabelWidget));
end
else
begin
gtk_label_set_text(LabelWidget, PChar(Pointer(s)));
end;
{$ifdef GTK2}
gtk_widget_set_direction(PGtkWidget(LabelWidget), GTK_TEXT_DIR_LTR); //Shortcut always LTR
{$endif}
if UseRTL then
gtk_misc_set_alignment(GTK_MISC(LabelWidget), 0.0, 0.5)
else
gtk_misc_set_alignment(GTK_MISC (LabelWidget), 1.0, 0.5);
end else
if LabelWidget <> nil then
gtk_widget_destroy(PGtkWidget(LabelWidget));
end;
procedure CreateIcon;
var
{$IFNDEF Gtk2}
IconWidth, IconHeight: integer;
IconSize: TPoint;
{$ENDIF}
MinHeightWidget: PGtkWidget;
begin
// the icon will be painted instead of the toggle
// of a normal gtkcheckmenuitem
if LCLMenuItem.HasIcon then
begin
{$IFNDEF Gtk2}
IconSize := LCLMenuItem.GetIconSize(0);
IconWidth := IconSize.X;
IconHeight := IconSize.Y;
// set the toggle width
GTK_MENU_ITEM(MenuItemWidget)^.toggle_size := guint16(IconWidth);
{$ENDIF}
GTK_MENU_ITEM(MenuItemWidget)^.flag0:=
PGtkMenuItem(MenuItemWidget)^.flag0 or
{$IFDEF Gtk2}
bm_TGtkCheckMenuItem_always_show_toggle;
{$ELSE}
bm_show_toggle_indicator;
{$ENDIF}
{$IFNDEF Gtk2}
// add a dummy widget for the icon height
MinHeightWidget := gtk_label_new('');
gtk_widget_show(MinHeightWidget);
gtk_widget_set_usize(MinHeightWidget, 1, IconHeight);
gtk_box_pack_start(GTK_BOX(HBoxWidget), MinHeightWidget, False, False, 0);
{$ENDIF}
end
else
MinHeightWidget := nil;
gtk_object_set_data(PGtkObject(MenuItemWidget),
'LCLMinHeight', MinHeightWidget);
end;
procedure CreateLabel;
var
LabelWidget: PGtkLabel;
begin
// create a label for the Caption
LabelWidget := PGtkLabel(gtk_label_new(''));
gtk_misc_set_alignment(GTK_MISC (LabelWidget), 0.0, 0.5);
gtk_object_set_data(PGtkObject(MenuItemWidget), 'LCLLabel', LabelWidget);
gtk_container_add(GTK_CONTAINER(HBoxWidget), PGtkWidget(LabelWidget));
SetMenuItemLabelText(LCLMenuItem, MenuItemWidget);
//gtk_accel_label_set_accel_widget(GTK_ACCEL_LABEL(LabelWidget), MenuItemWidget);
gtk_widget_show(PGtkWidget(LabelWidget));
end;
begin
HBoxWidget := gtk_object_get_data(PGtkObject(MenuItemWidget), 'LCLHBox');
if HBoxWidget = nil then
begin
// create inner widgets
if LCLMenuItem.Caption = cLineCaption then
begin
// a separator is an empty gtkmenuitem
exit;
end;
HBoxWidget := gtk_hbox_new(false, 0);
{$ifdef GTK2}
gtk_widget_set_direction(PGtkWidget(HBoxWidget), WidgetDirection[UseRTL]);
{$endif}
gtk_object_set_data(PGtkObject(MenuItemWidget), 'LCLHBox', HBoxWidget);
CreateIcon;
CreateLabel;
UpdateShortCutLabel;
gtk_container_add(GTK_CONTAINER(MenuItemWidget), HBoxWidget);
gtk_widget_show(HBoxWidget);
end else
begin
// there are already inner widgets
if LCLMenuItem.Caption = cLineCaption then
begin
// a separator is an empty gtkmenuitem -> delete the inner widgets
DestroyWidget(HBoxWidget);
gtk_object_set_data(PGtkObject(MenuItemWidget), 'LCLHBox', nil);
end else
begin
// just update the content
{$ifdef GTK2}
gtk_widget_set_direction(PGtkWidget(HBoxWidget), WidgetDirection[UseRTL]);
{$endif}
SetMenuItemLabelText(LCLMenuItem, MenuItemWidget);
UpdateShortCutLabel;
end;
end;
end;
function CreateStatusBarPanel(StatusBar: TObject; Index: integer): PGtkWidget;
begin
Result := gtk_statusbar_new;
gtk_widget_show(Result);
// other properties are set in UpdateStatusBarPanels
end;
procedure UpdateStatusBarPanels(StatusBar: TObject; StatusBarWidget: PGtkWidget);
var
AStatusBar: TStatusBar;
HBox: PGtkWidget;
CurPanelCount: integer;
NewPanelCount: Integer;
CurStatusPanelWidget: PGtkWidget;
ListItem: PGList;
i: Integer;
ExpandItem: boolean;
{$IFNDEF GTK1}
ShowSizeGrip: Boolean;
{$ENDIF}
begin
AStatusBar := StatusBar as TStatusBar;
HBox := PGtkWidget(StatusBarWidget);
if (not GtkWidgetIsA(StatusBarWidget, GTK_HBOX_GET_TYPE)) then
RaiseGDBException('');
// create needed panels
CurPanelCount := integer(g_list_length(PGtkBox(HBox)^.children));
if AStatusBar.SimplePanel or (AStatusBar.Panels.Count < 1) then
NewPanelCount := 1
else
NewPanelCount := AStatusBar.Panels.Count;
while CurPanelCount < NewPanelCount do
begin
CurStatusPanelWidget := CreateStatusBarPanel(StatusBar, CurPanelCount);
ExpandItem := (CurPanelCount = NewPanelCount - 1);
gtk_box_pack_start(PGtkBox(HBox), CurStatusPanelWidget,
ExpandItem, ExpandItem, 0);
inc(CurPanelCount);
end;
// remove unneeded panels
while CurPanelCount > NewPanelCount do
begin
CurStatusPanelWidget := PGtkBoxChild(
g_list_nth_data(PGtkBox(HBox)^.children, CurPanelCount - 1))^.Widget;
{$IFDEF GTK2}
gtk_object_remove_data(PGtkObject(CurStatusPanelWidget),'lcl_statusbar_id');
{$ENDIF}
DestroyConnectedWidgetCB(CurStatusPanelWidget, True);
dec(CurPanelCount);
end;
// check new panel count
CurPanelCount := integer(g_list_length(PGtkBox(HBox)^.children));
//DebugLn('TGtkWidgetSet.UpdateStatusBarPanels B ',Dbgs(StatusBar),' NewPanelCount=',dbgs(NewPanelCount),' CurPanelCount=',dbgs(CurPanelCount));
if CurPanelCount <> NewPanelCount then
RaiseGDBException('');
// set panel properties
{$IFNDEF GTK1}
ShowSizeGrip := AStatusBar.SizeGrip and AStatusBar.SizeGripEnabled;
{$ENDIF}
ListItem := PGTKBox(HBox)^.children;
i := 0;
while ListItem <> nil do
begin
CurStatusPanelWidget := PGtkBoxChild(PGTKWidget(ListItem^.data))^.widget;
ExpandItem := (ListItem^.next = nil);
gtk_box_set_child_packing(PGtkBox(HBox), CurStatusPanelWidget,
ExpandItem, ExpandItem, 0, GTK_PACK_START);
UpdateStatusBarPanel(StatusBar, i, CurStatusPanelWidget);
inc(i);
ListItem := ListItem^.next;
{$IFNDEF GTK1}
gtk_statusbar_set_has_resize_grip(PGtkStatusBar(CurStatusPanelWidget),
(ListItem = nil) and ShowSizeGrip);
{$ENDIF}
end;
end;
{$IFDEF GTK2}
function gtk2PaintStatusBarWidget(Widget: PGtkWidget; Event : PGDKEventExpose;
Data: gPointer): GBoolean; cdecl;
var
Msg: TLMDrawItems;
PS : TPaintStruct;
ItemStruct: PDrawItemStruct;
ItemID: Integer;
begin
Result := CallBackDefaultReturn;
if (Event^.Count > 0) then exit;
if (csDesigning in TComponent(Data).ComponentState) then
exit;
if TStatusBar(Data).SimplePanel then
exit;
ItemId := PtrInt(gtk_object_get_data(PGtkObject(Widget), 'lcl_statusbar_id')^);
if not ((ItemId >= 0) and (ItemId < TStatusBar(Data).Panels.Count)) then
exit;
if TStatusBar(Data).Panels[ItemId].Style <> psOwnerDraw then
exit;
FillChar(Msg, SizeOf(Msg), #0);
FillChar(PS, SizeOf(PS), #0);
FillChar(ItemStruct, SizeOf(ItemStruct), #0);
New(ItemStruct);
// we must fill up complete area otherwise gtk2 will do
// strange paints when item is not fully exposed.
ItemStruct^.rcItem := Rect(Widget^.allocation.x,
Widget^.allocation.y,
Widget^.allocation.width + Widget^.allocation.x,
Widget^.allocation.height + Widget^.allocation.y);
OffsetRect(ItemStruct^.rcItem, -ItemStruct^.rcItem.Left, -ItemStruct^.rcItem.Top);
// take frame borders into account
with ItemStruct^.rcItem do
begin
Left := Left + Widget^.style^.xthickness;
Top := Top + Widget^.style^.ythickness;
Right := Right - Widget^.style^.xthickness;
Bottom := Bottom - Widget^.style^.ythickness;
end;
ItemStruct^.itemID := ItemID;
PS.rcPaint := ItemStruct^.rcItem;
ItemStruct^._hDC := BeginPaint(HWND(Widget), PS);
Msg.Ctl := TStatusBar(Data).Handle;
Msg.DrawItemStruct := ItemStruct;
Msg.Msg := LM_DRAWITEM;
try
DeliverMessage(TStatusBar(Data), Msg);
Result := not CallBackDefaultReturn;
finally
PS.hdc := ItemStruct^._hDC;
EndPaint(HWND(TGtkDeviceContext(PS.hdc).Widget), PS);
Dispose(ItemStruct);
end;
end;
{$ENDIF}
procedure UpdateStatusBarPanel(StatusBar: TObject; Index: integer;
StatusPanelWidget: PGtkWidget);
var
AStatusBar: TStatusBar;
CurPanel: TStatusPanel;
FrameWidget: PGtkWidget;
LabelWidget: PGtkLabel;
PanelText: String;
ContextID: LongWord;
NewShadowType: TGtkShadowType;
NewJustification: TGtkJustification;
{$ifndef gtk1}
xalign, yalign: gfloat;
{$endif}
begin
//DebugLn('UpdateStatusBarPanel ',DbgS(StatusBar),' Index=',dbgs(Index));
AStatusBar := StatusBar as TStatusBar;
CurPanel := nil;
if (not AStatusBar.SimplePanel) and (AStatusBar.Panels.Count > Index) then
CurPanel := AStatusBar.Panels[Index];
//DebugLn('Panel ',Index,' ',GetWidgetClassName(StatusPanelWidget),
// ' frame=',GetWidgetClassName(PGTKStatusBar(StatusPanelWidget)^.frame),
// ' thelabel=',GetWidgetClassName(PGTKStatusBar(StatusPanelWidget)^.thelabel),
// '');
FrameWidget := PGTKStatusBar(StatusPanelWidget)^.frame;
LabelWidget := PGtkLabel(
{$ifndef gtk1}
PGTKStatusBar(StatusPanelWidget)^._label
{$else}
PGTKStatusBar(StatusPanelWidget)^.thelabel
{$endif});
// Text
if AStatusBar.SimplePanel then
PanelText := AStatusBar.SimpleText
else
if CurPanel <> nil then
PanelText := CurPanel.Text
else
PanelText := '';
ContextID := gtk_statusbar_get_context_id(PGTKStatusBar(StatusPanelWidget),
'state');
//DebugLn(' PanelText="',PanelText,'"');
if PanelText <> '' then
gtk_statusbar_push(PGTKStatusBar(StatusPanelWidget), ContextID, PGChar(PanelText))
else
gtk_statusbar_push(PGTKStatusBar(StatusPanelWidget), ContextID, '');
if CurPanel <> nil then
begin
//DebugLn(' Alignment="',ord(CurPanel.Alignment),'"');
// Alignment
NewJustification := aGtkJustification[CurPanel.Alignment];
if GTK_IS_LABEL(LabelWidget) then
begin
{$ifndef gtk1}
if GTK_IS_MISC(LabelWidget) then
begin
{gtk_label_set_justify() has no effect on labels containing
only a single line !}
gtk_misc_get_alignment(GTK_MISC(LabelWidget), @xalign, @yalign);
xalign := AlignToGtkAlign(CurPanel.Alignment);
gtk_misc_set_alignment(GTK_MISC(LabelWidget), xalign, yalign);
end else
gtk_label_set_justify(LabelWidget, NewJustification);
{$else}
gtk_label_set_justify(LabelWidget, NewJustification);
{$endif}
end;
// Bevel
// Paul: this call will not modify frame on gtk2. GtkStatusBar resets frame
// shadow on every size request. I have tried to modify rcStyle and tried to
// hook property change event. Both ways are 1) not valid 2) does not give me
// any result.
// As a possible solution we can subclass PGtkStatusBar but if gtk developers
// decided that stausbar should work so whether we need to override that?
NewShadowType := aGtkShadowFromBevel[CurPanel.Bevel];
if GTK_IS_FRAME(FrameWidget) then
gtk_frame_set_shadow_type(PGtkFrame(FrameWidget), NewShadowType);
// Width
//DebugLn(' CurPanel.Width="',CurPanel.Width,'"');
gtk_widget_set_usize(StatusPanelWidget, CurPanel.Width,
StatusPanelWidget^.allocation.height);
{$IFDEF GTK2}
gtk_object_set_data(PGtkObject(StatusPanelWidget),'lcl_statusbar_id',
@AStatusBar.Panels[Index].ID);
g_signal_connect_after(StatusPanelWidget, 'expose-event',
TGtkSignalFunc(@gtk2PaintStatusBarWidget), AStatusBar);
{$ENDIF}
end;
end;
function gtkListGetSelectionMode(list: PGtkList): TGtkSelectionMode; cdecl;
begin
Result:=TGtkSelectionMode(
(list^.flag0 and bm_TGtkList_selection_mode) shr bp_TGtkList_selection_mode);
end;
{------------------------------------------------------------------------------
SaveSizeNotification
Params: Widget: PGtkWidget A widget that is the handle of a lcl control.
When the gtk sends a size signal, it is not send directly to the LCL. All gtk
size/move messages are collected and only the last one for each widget is sent
to the LCL.
This is neccessary, because the gtk sends size messages several times and
it replays resizes. Since the LCL reacts to every size notification and
resizes child controls, this results in a perpetuum mobile.
------------------------------------------------------------------------------}
procedure SaveSizeNotification(Widget: PGtkWidget);
{$IFDEF VerboseSizeMsg}
var
LCLControl: TWinControl;
{$ENDIF}
begin
{$IFDEF VerboseSizeMsg}
DbgOut('SaveSizeNotification Widget=',DbgS(Widget));
LCLControl:=TWinControl(GetLCLObject(Widget));
if (LCLControl<>nil) then begin
if LCLControl is TWinControl then
DebugLn(' ',LCLControl.Name,':',LCLControl.ClassName)
else
DebugLn(' ERROR: ',LCLControl.ClassName);
end else begin
DebugLn(' ERROR: LCLControl=nil');
end;
{$ENDIF}
if not FWidgetsResized.Contains(Widget) then
FWidgetsResized.Add(Widget);
end;
{------------------------------------------------------------------------------
SaveClientSizeNotification
Params: FixWidget: PGtkWidget A widget that is the fixed widget
of a lcl control.
When the gtk sends a size signal, it is not sent directly to the LCL. All gtk
size/move messages are collected and only the last one for each widget is sent
to the LCL.
This is neccessary, because the gtk sends size messages several times and
it replays resizes. Since the LCL reacts to every size notification and
resizes child controls, this results in a perpetuum mobile.
------------------------------------------------------------------------------}
procedure SaveClientSizeNotification(FixWidget: PGtkWidget);
{$IFDEF VerboseSizeMsg}
var
LCLControl: TWinControl;
MainWidget: PGtkWidget;
{$ENDIF}
begin
{$IFDEF VerboseSizeMsg}
MainWidget:=GetMainWidget(FixWidget);
//write('SaveClientSizeNotification',
// ' FixWidget=',DbgS(FixWidget),
// ' MainWIdget=',DbgS(MainWidget));
LCLControl:=TWinControl(GetLCLObject(MainWidget));
if (LCLControl<>nil) then begin
if LCLControl is TWinControl then begin
//DebugLn('SaveClientSizeNotification ',LCLControl.Name,':',LCLControl.ClassName,
// ' FixWidget=',DbgS(FixWidget),
// ' MainWidget=',DbgS(MainWidget));
end else begin
DbgOut('ERROR: SaveClientSizeNotification ',
' LCLControl=',LCLControl.ClassName,
' FixWidget=',DbgS(FixWidget),
' MainWidget=',DbgS(MainWidget));
RaiseGDBException('SaveClientSizeNotification');
end;
end else begin
DbgOut('ERROR: SaveClientSizeNotification LCLControl=nil',
' FixWidget=',DbgS(FixWidget),
' MainWIdget=',DbgS(MainWidget));
RaiseGDBException('SaveClientSizeNotification');
end;
{$ENDIF}
if not FFixWidgetsResized.Contains(FixWidget) then
FFixWidgetsResized.Add(FixWidget);
end;
{-------------------------------------------------------------------------------
CreateTopologicalSortedWidgets
Params: HashArray: TDynHashArray of PGtkWidget
Creates a topologically sorted TFPList of PGtkWidget.
-------------------------------------------------------------------------------}
function CreateTopologicalSortedWidgets(HashArray: TDynHashArray): TFPList;
type
PTopologicalEntry = ^TTopologicalEntry;
TTopologicalEntry = record
Widget: PGtkWidget;
ParentLevel: integer;
end;
function GetParentLevel(AControl: TControl): integer;
// nil has lvl -1
// a control without parent has lvl 0
begin
Result:=-1;
while AControl<>nil do begin
inc(Result);
AControl:=AControl.Parent;
end;
end;
var
TopologicalList: PTopologicalEntry;
HashItem: PDynHashArrayItem;
i, Lvl, MaxLevel: integer;
LCLControl: TControl;
LevelCounts: PInteger;
begin
Result:=TFPList.Create;
if HashArray.Count=0 then exit;
// put all widgets into an array and calculate their parent levels
GetMem(TopologicalList,SizeOf(TTopologicalEntry)*HashArray.Count);
HashItem:=HashArray.FirstHashItem;
i:=0;
MaxLevel:=0;
//DebugLn('CreateTopologicalSortedWidgets HashArray.Count=',HashArray.Count);
while HashItem<>nil do begin
TopologicalList[i].Widget:=HashItem^.Item;
//DebugLn('CreateTopologicalSortedWidgets i=',i,' Widget=',DbgS(TopologicalList[i].Widget));
LCLControl:=TControl(GetLCLObject(TopologicalList[i].Widget));
if (LCLControl=nil) or (not (LCLControl is TControl)) then
RaiseGDBException('CreateTopologicalSortedWidgets: '
+'Widget without LCL control');
Lvl:=GetParentLevel(LCLControl);
TopologicalList[i].ParentLevel:=Lvl;
if MaxLevel<Lvl then
MaxLevel:=Lvl;
//DebugLn('CreateTopologicalSortedWidgets i=',i,' Lvl=',Lvl,' MaxLvl=',MaxLevel,' LCLControl=',LCLControl.Name,':',LCLControl.ClassName);
inc(i);
HashItem:=HashItem^.Next;
end;
inc(MaxLevel);
// bucket sort the widgets
// count each number of levels (= bucketsizes)
GetMem(LevelCounts,SizeOf(Integer)*MaxLevel);
FillChar(LevelCounts^,SizeOf(Integer)*MaxLevel,0);
for i:=0 to HashArray.Count-1 do
inc(LevelCounts[TopologicalList[i].ParentLevel]);
// calculate bucketends
for i:=1 to MaxLevel-1 do
inc(LevelCounts[i],LevelCounts[i-1]);
// bucket sort the widgets in Result
Result.Count:=HashArray.Count;
for i:=0 to HashArray.Count-1 do
Result[i]:=nil;
for i:=0 to HashArray.Count-1 do begin
Lvl:=TopologicalList[i].ParentLevel;
dec(LevelCounts[Lvl]);
//DebugLn('CreateTopologicalSortedWidgets bucket sort i=',i,' Lvl=',Lvl,' LevelCounts[Lvl]=',LevelCounts[Lvl],
// ' Widget=',DbgS(TopologicalList[i].Widget));
Result[LevelCounts[Lvl]]:=TopologicalList[i].Widget;
end;
FreeMem(LevelCounts);
FreeMem(TopologicalList);
end;
procedure GetGTKDefaultWidgetSize(AWinControl: TWinControl;
var PreferredWidth, PreferredHeight: integer; WithThemeSpace: Boolean);
var
Widget: PGtkWidget;
Requisition: TGtkRequisition;
begin
Widget := PGtkWidget(AWinControl.Handle);
// set size to default
//DebugLn(['GetGTKDefaultWidgetSize ',GetWidgetDebugReport(Widget)]);
{$IFDEF GTK1}
gtk_widget_set_usize(Widget, -1, -1); // deprecated in gtk2
{$ELSE}
gtk_widget_set_size_request(Widget, -1, -1);
{$ENDIF}
// ask default size
gtk_widget_size_request(Widget,@Requisition);
PreferredWidth:=Requisition.width;
PreferredHeight:=Requisition.height;
if WithThemeSpace then begin
{$IFDEF Gtk1}
//DebugLn(['GetGTKDefaultWidgetSize WithThemeSpace ',DbgSName(AWinControl),' ',GtkWidgetIsA(Widget,GTK_BUTTON_TYPE),' ',GetWidgetDebugReport(Widget),' ',2*gtk_widget_get_ythickness(Widget)]);
if gtk_class_get_type(gtk_object_get_class(Widget))=GTK_BUTTON_TYPE then
inc(PreferredHeight,2*gtk_widget_get_ythickness(Widget))
else if not GtkWidgetIsA(Widget,GTK_ENTRY_TYPE) then
dec(PreferredHeight,2*gtk_widget_get_ythickness(Widget));
{$ENDIF}
end else begin
//debugLn('GetGTKDefaultWidgetSize ',DbgSName(AWinControl),' ',dbgs(gtk_widget_get_xthickness(Widget)),' ythickness=',dbgs(gtk_widget_get_ythickness(Widget)));
//debugLn(['GetGTKDefaultWidgetSize ',GetWidgetDebugReport(Widget)]);
//dec(PreferredWidth,gtk_widget_get_xthickness(Widget));
{$IFDEF Gtk1}
//if not GtkWidgetIsA(Widget,GTK_ENTRY_TYPE) then
// dec(PreferredHeight,2*gtk_widget_get_ythickness(Widget));
{$ELSE}
//if gtk_class_get_type(gtk_object_get_class(Widget))=GTK_TYPE_BUTTON then
// dec(PreferredHeight,2*gtk_widget_get_ythickness(Widget));
{$ENDIF}
end;
{DebugLn(['GetGTKDefaultWidgetSize Allocation=',Widget^.allocation.x,',',Widget^.allocation.y,',',Widget^.allocation.width,',',Widget^.allocation.height,
' requisition=',Widget^.requisition.width,',',Widget^.requisition.height,
' PreferredWidth=',PreferredWidth,' PreferredHeight=',PreferredHeight,
' WithThemeSpace=',WithThemeSpace]);}
// set new size
{$IFDEF GTK1}
gtk_widget_set_usize(Widget, AWinControl.Width, AWinControl.Height);
{$ELSE}
gtk_widget_set_size_request(Widget, AWinControl.Width, AWinControl.Height);
{$ENDIF}
//debugln('GetGTKDefaultSize PreferredWidth=',dbgs(PreferredWidth),' PreferredHeight=',dbgs(PreferredHeight));
end;
procedure SendSizeNotificationToLCL(aWidget: PGtkWidget);
var
LCLControl: TWinControl;
LCLLeft, LCLTop, LCLWidth, LCLHeight: integer;
GtkLeft, GtkTop, GtkWidth, GtkHeight: integer;
TopLeftChanged, WidthHeightChanged, IsTopLevelWidget: boolean;
MessageDelivered: boolean;
SizeMsg: TLMSize;
MoveMsg: TLMMove;
PosMsg : TLMWindowPosChanged;
MainWidget: PGtkWidget;
FixedWidget: PGtkWidget;
procedure UpdateLCLPos;
begin
LCLLeft:=LCLControl.Left;
LCLTop:=LCLControl.Top;
TopLeftChanged:=(LCLLeft<>GtkLeft) or (LCLTop<>GtkTop);
end;
procedure UpdateLCLSize;
begin
LCLWidth:=LCLControl.Width;
LCLHeight:=LCLControl.Height;
WidthHeightChanged:=(LCLWidth<>GtkWidth) or (LCLHeight<>GtkHeight);
if LCLControl.ClientRectNeedsInterfaceUpdate then begin
WidthHeightChanged:=true;
//DebugLn(['UpdateLCLSize InvalidateClientRectCache ',DbgSName(LCLControl)]);
LCLControl.InvalidateClientRectCache(false);
end;
end;
begin
LCLControl:=TWinControl(GetLCLObject(aWidget));
if LCLControl=nil then exit;
{$IFDEF VerboseSizeMsg}
DebugLn('SendSizeNotificationToLCL checking ... ',DbgSName(LCLControl),' Widget=',WidgetFlagsToString(aWidget));
{$ENDIF}
MainWidget:=PGtkWidget(LCLControl.Handle);
FixedWidget:=PGtkWidget(GetFixedWidget(MainWidget));
FWidgetsResized.Remove(MainWidget);
FFixWidgetsResized.Remove(FixedWidget);
{$IF defined(Gtk1)}
if not GTK_WIDGET_REALIZED(aWidget) then begin
// the widget is not yet realized, so this GTK resize was not a user change.
// => ignore
{$IFDEF VerboseSizeMsg}
LCLControl:=TWinControl(GetLCLObject(aWidget));
DebugLn('SendSizeNotificationToLCL ',DbgSName(LCLControl),' aWidget=',WidgetFlagsToString(aWidget),' Ignored, because not realized ');
{$ENDIF}
exit;
end;
{$ENDIF}
GetWidgetRelativePosition(MainWidget,GtkLeft,GtkTop);
{$ifdef gtk2}
gtk_widget_get_size_request(MainWidget, @GtkWidth, @GtkHeight);
if GtkWidth < 0 then
GtkWidth:=MainWidget^.Allocation.Width
else
MainWidget^.Allocation.Width:=GtkWidth;
if GtkHeight < 0 then
GtkHeight:=MainWidget^.Allocation.Height
else
MainWidget^.Allocation.Height:=GtkHeight;
//DebugLn(['SendSizeNotificationToLCL ',DbgSName(LCLControl),' gtk=',GtkLeft,',',GtkTop,',',GtkWidth,'x',GtkHeight,' Allocation=',MainWidget^.Allocation.Width,'x',MainWidget^.Allocation.Height]);
{$else}
GtkWidth:=MainWidget^.Allocation.Width;
GtkHeight:=MainWidget^.Allocation.Height;
{$endif}
if GtkWidth<0 then GtkWidth:=0;
if GtkHeight<0 then GtkHeight:=0;
IsTopLevelWidget:=(LCLControl is TCustomForm) and (LCLControl.Parent=nil);
if IsTopLevelWidget then begin
if not GTK_WIDGET_VISIBLE(MainWidget) then begin
// size/move messages of invisible windows are not reliable
// -> ignore
exit;
end;
if (GtkWidth=1) and (GtkHeight=1) then begin
// this is default size of the gtk. Ignore.
exit;
end;
//DebugLn(['SendSizeNotificationToLCL FORM ',GetWidgetDebugReport(MainWidget)]);
{$IFDEF VerboseFormPositioning}
DebugLn(['VFP SendSizeNotificationToLCL ',DbgSName(LCLControl),' ',
GtkLeft,',',GtkTop,',',GtkWidth,'x',GtkHeight,' ',GetWidgetDebugReport(MainWidget)]);
{$ENDIF}
end;
UpdateLCLPos;
UpdateLCLSize;
// first send a LM_WINDOWPOSCHANGED message
if TopLeftChanged or WidthHeightChanged then begin
{$IFDEF VerboseSizeMsg}
DebugLn('SendSizeNotificationToLCL ',DbgSName(LCLControl),
' GTK=',dbgs(GtkLeft)+','+dbgs(GtkTop)+','+dbgs(GtkWidth)+'x'+dbgs(GtkHeight),
' LCL=',dbgs(LCLLeft)+','+dbgs(LCLTop)+','+dbgs(LCLWidth)+'x'+dbgs(LCLHeight)
);
{$ENDIF}
PosMsg.Msg := LM_WINDOWPOSCHANGED; //LM_SIZEALLOCATE;
PosMsg.Result := 0;
New(PosMsg.WindowPos);
try
with PosMsg.WindowPos^ do begin
hWndInsertAfter := 0;
x := GtkLeft;
y := GtkTop;
cx := GtkWidth;
cy := GtkHeight;
flags:=0;
// flags := SWP_SourceIsInterface;
end;
MessageDelivered := DeliverMessage(LCLControl, PosMsg) = 0;
finally
Dispose(PosMsg.WindowPos);
end;
if (not MessageDelivered) then exit;
if FWidgetsWithResizeRequest.Contains(MainWidget) then exit;
UpdateLCLPos;
UpdateLCLSize;
end;
// then send a LM_SIZE message
if WidthHeightChanged then begin
{$IFDEF VerboseSizeMsg}
DebugLn('Send LM_SIZE To LCL ',LCLControl.Name,':',LCLControl.ClassName);
{$ENDIF}
with SizeMsg do
begin
Result := 0;
Msg := LM_SIZE;
{$IFDEF GTK1}
if GDK_WINDOW_GET_MAXIMIZED(PGdkWindowPrivate(MainWidget^.window)) then
SizeType := SIZEFULLSCREEN
else
SizeType := SIZENORMAL;
{$ELSE}
if LCLControl is TCustomForm then begin
// if the LCL gets an event without a State it resets it to SIZENORMAL
// so we send it the state it already is
case TCustomForm(LCLControl).WindowState of
wsNormal: SizeType := SIZENORMAL;
wsMinimized: SizeType := SIZEICONIC;
wsMaximized: SizeType := SIZEFULLSCREEN;
end;
end
else
SizeType := 0;
{$ENDIF}
SizeType := SizeType or Size_SourceIsInterface;
Width := SmallInt(GtkWidth);
Height := SmallInt(GtkHeight);
end;
MessageDelivered := (DeliverMessage(LCLControl, SizeMsg) = 0);
if not MessageDelivered then exit;
if FWidgetsWithResizeRequest.Contains(MainWidget) then exit;
UpdateLCLPos;
end;
// then send a LM_MOVE message
if TopLeftChanged then begin
{$IFDEF VerboseSizeMsg}
DebugLn('Send LM_MOVE To LCL ',LCLControl.Name,':',LCLControl.ClassName);
{$ENDIF}
with MoveMsg do
begin
Result := 0;
Msg := LM_MOVE;
MoveType := Move_SourceIsInterface;
XPos := SmallInt(GtkLeft);
YPos := SmallInt(GtkTop);
end;
MessageDelivered := (DeliverMessage(LCLControl, MoveMsg) = 0);
if not MessageDelivered then exit;
end;
{$ifndef gtk1}
if GtkWidgetIsA(aWidget, GTKAPIWidget_Type) and
not (wwiNoEraseBkgnd in GetWidgetInfo(aWidget)^.Flags) then
gtk_widget_queue_draw(aWidget);
{$endif}
end;
procedure SendCachedGtkResizeNotifications;
{ This proc sends all cached size messages from the gtk to lcl but in an
optimized order.
When sending the LCL a size/move/windowposchanged messages the LCL will
automatically realign all child controls. This realigning is based on the
clientrect.
Therefore, before a size message is sent to the lcl, all clientrect must be
updated.
If a size message results in resizing a widget that was also resized, then
the message for the dependent widget is not sent to the lcl, because the lcl
resize was after the gtk resize.
}
var
FixWidget, MainWidget: PGtkWidget;
LCLControl: TWinControl;
List: TFPList;
i: integer;
procedure RaiseInvalidLCLControl;
begin
RaiseGDBException(Format('SendCachedGtkResizeNotifications FixWidget=%p MainWidget=%p LCLControl=%p',
[FixWidget, MainWidget, Pointer(LCLControl)]));
end;
begin
if (FWidgetsResized.Count=0) and (FFixWidgetsResized.Count=0) then exit;
List:=TFPList.Create;
{ if any fixed widget was resized then a client area of a LCL control was
resized
-> invalidate client rectangles
}
{$IFDEF VerboseSizeMsg}
DebugLn('HHH1 SendCachedGtkClientResizeNotifications Invalidating ClientRects ... '
,' FixSizeMsgCount=',dbgs(FFixWidgetsResized.Count));
{$ENDIF}
FFixWidgetsResized.AssignTo(List);
for i:=0 to List.Count-1 do begin
FixWidget:=List[i];
MainWidget:=GetMainWidget(FixWidget);
LCLControl:=TWinControl(GetLCLObject(MainWidget));
if (LCLControl=nil) or (not (LCLControl is TWinControl)) then
RaiseInvalidLCLControl;
LCLControl.InvalidateClientRectCache(false);
end;
{ if any main widget (= not fixed widget) was resized
then a LCL control was resized
-> send WMSize, WMMove, and WMWindowPosChanged messages
}
{$IFDEF VerboseSizeMsg}
if FWidgetsResized.First<>nil then
DebugLn('HHH2 SendCachedGtkClientResizeNotifications SizeMsgCount=',dbgs(FWidgetsResized.Count));
{$ENDIF}
repeat
MainWidget:=FWidgetsResized.First;
if MainWidget<>nil then begin
FWidgetsResized.Remove(MainWidget);
if not FWidgetsWithResizeRequest.Contains(MainWidget) then begin
SendSizeNotificationToLCL(MainWidget);
end;
end else break;
until Application.Terminated;
{ if any client area was resized, which MainWidget Size was already in sync
with the LCL, no message was sent. So, tell each changed client area to
check its size.
}
{$IFDEF VerboseSizeMsg}
if FFixWidgetsResized.First<>nil then
DebugLn('HHH3 SendCachedGtkClientResizeNotifications Updating ClientRects ...');
{$ENDIF}
repeat
FixWidget:=FFixWidgetsResized.First;
if FixWidget<>nil then begin
FFixWidgetsResized.Remove(FixWidget);
MainWidget:=GetMainWidget(FixWidget);
LCLControl:=TWinControl(GetLCLObject(MainWidget));
LCLControl.DoAdjustClientRectChange(False);
end else begin
break;
end;
until Application.Terminated;
List.Free;
{$IFDEF VerboseSizeMsg}
DebugLn('HHH4 SendCachedGtkClientResizeNotifications completed.');
{$ENDIF}
end;
procedure ResizeHandle(LCLControl: TWinControl);
var
Widget: PGtkWidget;
Later: Boolean;
{$IFDEF Gtk2}
IsTopLevelWidget: Boolean;
{$ENDIF}
begin
Widget := PGtkWidget(LCLControl.Handle);
if not WidgetSizeIsEditable(Widget) then
Exit;
Later := true;
{$IFDEF Gtk2}
// add resize request immediately
IsTopLevelWidget:= (LCLControl is TCustomForm) and
(LCLControl.Parent = nil) and
(LCLControl.ParentWindow = 0);
if not IsTopLevelWidget then
begin
SetWidgetSizeAndPosition(LCLControl);
Later := false;
end;
{$ENDIF}
if Later then
SetResizeRequest(Widget);
end;
procedure SetWidgetSizeAndPosition(LCLControl: TWinControl);
var
Requisition: TGtkRequisition;
FixedWidget: PGtkWidget;
{$IFDEF Gtk2}
allocation: TGtkAllocation;
{$ENDIF}
LCLLeft: LongInt;
LCLTop: LongInt;
LCLWidth: LongInt;
LCLHeight: LongInt;
Widget: PGtkWidget;
ParentWidget: PGtkWidget;
ParentFixed: PGtkWidget;
WinWidgetInfo: PWidgetInfo;
{$IFDEF VerboseSizeMsg}
LCLObject: TObject;
{$ENDIF}
procedure WriteBigWarning;
begin
DebugLn('WARNING: SetWidgetSizeAndPosition: resizing BIG ',
' Control=',LCLControl.Name,':',LCLControl.ClassName,
' NewSize=',dbgs(LCLWidth),',',dbgs(LCLHeight));
//RaiseException('');
end;
procedure WriteWarningParentWidgetNotFound;
begin
DebugLn('WARNING: SetWidgetSizeAndPosition - '
,'Parent''s Fixed Widget not found');
DebugLn(' Control=',LCLControl.Name,':',LCLControl.ClassName,
' Parent=',LCLControl.Parent.Name,':',LCLControl.Parent.ClassName,
' ParentWidget=',DbgS(ParentWidget),
'');
end;
begin
{$IFDEF VerboseSizeMsg}
DebugLn(['SetWidgetSizeAndPosition ',DbgSName(LCLControl)]);
{$ENDIF}
Widget:=PGtkWidget(LCLControl.Handle);
LCLLeft := LCLControl.Left;
LCLTop := LCLControl.Top;
// move widget on the fixed widget of parent control
if ((LCLControl.Parent <> nil) and (LCLControl.Parent.HandleAllocated)) or
((LCLControl.Parent = nil) and (LCLControl.ParentWindow <> 0)) then
begin
if LCLControl.Parent <> nil then
ParentWidget := PGtkWidget(LCLControl.Parent.Handle)
else
ParentWidget := PGtkWidget(LCLControl.ParentWindow);
ParentFixed := GetFixedWidget(ParentWidget);
if GtkWidgetIsA(ParentFixed,GTK_FIXED_GET_TYPE) or
GtkWidgetIsA(ParentFixed,GTK_LAYOUT_GET_TYPE) then
begin
//DebugLn(['SetWidgetSizeAndPosition ',DbgSName(LCLControl),' Widget=[',GetWidgetDebugReport(Widget),'] ParentFixed=[',GetWidgetDebugReport(ParentFixed),']']);
FixedMoveControl(ParentFixed, Widget, LCLLeft, LCLTop);
end
else
begin
WinWidgetInfo := GetWidgetInfo(Widget, False);
if (WinWidgetInfo = nil) or (not (wwiNotOnParentsClientArea in WinWidgetInfo^.Flags)) then
WriteWarningParentWidgetNotFound;
end;
end;
// resize widget
LCLWidth := LCLControl.Width;
if LCLWidth <= 0 then
LCLWidth := 1;
LCLHeight := LCLControl.Height;
if LCLHeight <= 0 then
LCLHeight := 1;
if (LCLWidth > 10000) or (LCLHeight > 10000) then
begin
WriteBigWarning;
if LCLWidth > 10000 then
LCLWidth := 10000;
if LCLHeight > 10000 then
LCLHeight := 10000;
end;
{$IFDEF VerboseSizeMsg}
LCLObject:=GetNearestLCLObject(Widget);
DbgOut('TGtkWidgetSet.SetWidgetSizeAndPosition Widget='+DbgS(Widget)+WidgetFlagsToString(Widget)+
' New='+dbgs(LCLWidth)+','+dbgs(LCLHeight));
if (LCLObject<>nil) and (LCLObject is TControl) then begin
with TControl(LCLObject) do
DebugLn(' LCL=',Name,':',ClassName,' ',dbgs(Left),',',dbgs(Top),',',dbgs(Width),',',dbgs(Height));
end else begin
DebugLn(' LCL=',DbgS(LCLObject));
end;
{$ENDIF}
if GtkWidgetIsA(Widget,GTK_TYPE_SCROLLBAR) then
begin
// the width of a scrollbar is fixed and depends only on the theme
gtk_widget_size_request(widget, @Requisition);
if GtkWidgetIsA(Widget, GTK_TYPE_HSCROLLBAR) then
begin
LCLHeight:=Requisition.height;
end else begin
LCLWidth:=Requisition.width;
end;
//DebugLn('TGtkWidgetSet.SetWidgetSizeAndPosition A ',LCLwidth,',',LCLheight);
end;
gtk_widget_set_usize(Widget, LCLWidth, LCLHeight);
//DebugLn(['TGtkWidgetSet.SetWidgetSizeAndPosition ',GetWidgetDebugReport(Widget),' LCLWidth=',LCLWidth,' LCLHeight=',LCLHeight]);
{$IFDEF Gtk1}
if GtkWidgetIsA(Widget, GTK_TYPE_COMBO) then
begin
// the combobox has an entry, which height is not resized
// automatically. Do it manually.
gtk_widget_set_usize(PGtkCombo(Widget)^.entry,
PGtkCombo(Widget)^.entry^.allocation.width, LCLHeight);
end;
{$ENDIF}
if GtkWidgetIsA(Widget,gtk_toolbar_get_type) then begin
FixedWidget:=GetFixedWidget(Widget);
if (FixedWidget<>nil) and (FixedWidget<>Widget) then begin
//DebugLn('WARNING: ToDo TGtkWidgetSet.SetWidgetSizeAndPosition for TToolBar ',LCLWidth,',',LCLHeight);
gtk_widget_set_usize(FixedWidget,LCLWidth,LCLHeight);
end;
end;
{$IFDEF Gtk2}
if (Widget^.parent<>nil)
and GtkWidgetIsA(Widget^.parent,GTK_TYPE_FIXED)
and GTK_WIDGET_NO_WINDOW(Widget^.parent)
then begin
inc(LCLLeft, Widget^.parent^.allocation.x);
inc(LCLTop, Widget^.parent^.allocation.y);
end;
// commit size and position
allocation:=Widget^.allocation;
allocation.x:=LCLLeft;
allocation.y:=LCLTop;
allocation.width:=LCLWidth;
allocation.height:=LCLHeight;
//DebugLn(['SetWidgetSizeAndPosition ',DbgSName(LCLControl),' LCL=',dbgs(LCLControl.BoundsRect),' allocation=',dbgs(allocation),' ',GetWidgetDebugReport(Widget)]);
gtk_widget_size_allocate(Widget,@allocation);// Beware: this triggers callbacks
{$ENDIF}
end;
{------------------------------------------------------------------------------
Method: SetWindowSizeAndPosition
Params: Widget: PGtkWidget; AWinControl: TWinControl
Returns: Nothing
Set the size and position of a top level window.
------------------------------------------------------------------------------}
procedure SetWindowSizeAndPosition(Window: PGtkWindow;
AWinControl: TWinControl);
var
Width, Height: integer;
{$IFDEF Gtk2}
allocation: TGtkAllocation;
{$ENDIF}
//Info: PGtkWindowGeometryInfo;
begin
Width:=AWinControl.Width;
// 0 and negative values have a special meaning, so don't use them
if Width<=0 then Width:=1;
Height:=AWinControl.Height;
if Height<=0 then Height:=1;
{$IFDEF VerboseSizeMsg}
DebugLn(['TGtkWidgetSet.SetWindowSizeAndPosition START ',DbgSName(AWinControl),' ',AWinControl.Visible,' Old=',PGtkWidget(Window)^.allocation.Width,',',PGtkWidget(Window)^.allocation.Width,' New=',Width,',',Height]);
{$ENDIF}
// set geometry default size
//Info:=gtk_window_get_geometry_info(Window, TRUE);
//if (Info^.default_width<>Width) or (Info^.default_height<>Height) then
gtk_window_set_default_size(Window, Width, Height);
{$IFDEF Gtk2}
// resize
gtk_window_resize(Window, Width, Height);
// reposition
gtk_window_move(Window, AWinControl.Left, AWinControl.Top);
// force early resize
allocation := PGtkWidget(Window)^.allocation;
allocation.width := Width;
allocation.height := Height;
//DebugLn(['SetWindowSizeAndPosition ',DbgSName(AWinControl),' ',dbgs(allocation)]);
gtk_widget_size_allocate(PGtkWidget(Window), @allocation);// Beware: this triggers callbacks
if (PGtkWidget(Window)^.Window <> nil) then
begin
// resize gdkwindow directly (sometimes the gtk forgets this)
gdk_window_move_resize(PGtkWidget(Window)^.Window,AWinControl.Left,
AWinControl.Top,Width,Height)
end;
{$ELSE}
// resize
if assigned(PGtkWidget(Window)^.Window) then
// widget is realized, resize gdkwindow directly
gdk_window_move_resize(PGtkWidget(Window)^.Window,AWinControl.Left,
AWinControl.Top,Width,Height)
else begin
// widget is not yet realized, force resize needed for shrinking under gtk1
gtk_widget_set_usize(PGtkWidget(Window), -1,-1);
end;
// reposition
gtk_widget_set_usize(PGtkWidget(Window),Width,Height);
gtk_widget_set_uposition(PGtkWidget(Window),AWinControl.Left,AWinControl.Top);
{$ENDIF}
{$IFDEF VerboseSizeMsg}
DebugLn(['SetWindowSizeAndPosition B ',DbgSName(AWinControl),
' Visible=',AWinControl.Visible,
' Cur=',PGtkWidget(Window)^.allocation.X,',',PGtkWidget(Window)^.allocation.Y,
' New=',AWinControl.Left,',',AWinControl.Top,',',Width,'x',Height]);
{$ENDIF}
end;
{-------------------------------------------------------------------------------
GetWidgetRelativePosition
Returns the Left, Top, relative to the client origin of its parent
-------------------------------------------------------------------------------}
procedure GetWidgetRelativePosition(aWidget: PGtkWidget; var Left, Top: integer);
var
GdkWindow: PGdkWindow;
LCLControl: TWinControl;
GtkLeft, GtkTop: GInt;
begin
Left:=aWidget^.allocation.X;
Top:=aWidget^.allocation.Y;
{$IFDEF Gtk2}
if (aWidget^.parent<>nil)
and (not GtkWidgetIsA(aWidget^.parent,GTK_TYPE_FIXED))
and (not GtkWidgetIsA(aWidget^.parent,GTK_TYPE_LAYOUT))
then begin
// widget is not on a normal client area. e.g. TPage
Left:=0;
Top:=0;
end
else
if (aWidget^.parent<>nil)
and GtkWidgetIsA(aWidget^.parent,GTK_TYPE_FIXED)
and GTK_WIDGET_NO_WINDOW(aWidget^.parent)
then begin
// widget on a fixed, but fixed w/o window
Dec(Left, PGtkWidget(aWidget^.parent)^.allocation.x);
Dec(Top, PGtkWidget(aWidget^.parent)^.allocation.y);
end;
{$ENDIF}
if GtkWidgetIsA(aWidget,GTK_TYPE_WINDOW) then begin
GdkWindow:=GetControlWindow(aWidget);
if (GdkWindow<>nil) and (GTK_WIDGET_MAPPED(aWidget)) then begin
// window is mapped = window manager has put the window somewhere
gdk_window_get_root_origin(GdkWindow, @GtkLeft, @GtkTop);
Left := GtkLeft;
Top := GtkTop;
end else begin
// the gtk has not yet put the window to the final position
// => the gtk/gdk position is not reliable
// => use the LCL coords
LCLControl:=GetLCLObject(aWidget) as TWinControl;
Left:=LCLControl.Left;
Top:=LCLControl.Top;
end;
//DebugLn(['TGtkWidgetSet.GetWindowRelativePosition ',GetWidgetDebugReport(aWidget),' Left=',Left,' Top=',Top,' GdkWindow=',GdkWindow<>nil]);
end;
//DebugLn(['TGtkWidgetSet.GetWindowRelativePosition ',GetWidgetDebugReport(aWidget),' Left=',Left,' Top=',Top]);
end;
{------------------------------------------------------------------------------
UnsetResizeRequest
Params: Widget: PGtkWidget
Unset the mark for the Widget to send a ResizeRequest to the gtk.
LCL size requests for a widget are cached and only the last one is sent. Some
widgets like forms send a resize request immediately. To avoid sending resize
requests multiple times they can unset the mark with this procedure.
------------------------------------------------------------------------------}
procedure UnsetResizeRequest(Widget: PGtkWidget);
begin
{$IFDEF VerboseSizeMsg}
if FWidgetsWithResizeRequest.Contains(Widget) then begin
DebugLn(['UnsetResizeRequest ',GetWidgetDebugReport(Widget)]);
end;
{$ENDIF}
FWidgetsWithResizeRequest.Remove(Widget);
end;
{------------------------------------------------------------------------------
TGtkWidgetSet SetResizeRequest
Params: Widget: PGtkWidget
Marks the widget to send a ResizeRequest to the gtk.
When the LCL resizes a control the new bounds will not be set directly, but
cached. This is needed, because it is common behaviour to set the bounds step
by step. For example: Left:=10; Top:=10; Width:=100; Height:=50; results in
SetBounds(10,0,0,0);
SetBounds(10,10,0,0);
SetBounds(10,10,100,0);
SetBounds(10,10,100,50);
Because the gtk puts all size requests into a queue, it will process the
requests not immediately, but _after_ all requests. This results in changing
the widget size four times and everytime the LCL gets a message. If the
control has children, this will resize the children four times.
Therefore LCL size requests for a widget are cached and only the final one is
sent in: TGtkWidgetSet.SendCachedLCLMessages.
------------------------------------------------------------------------------}
procedure SetResizeRequest(Widget: PGtkWidget);
{$IFDEF VerboseSizeMsg}
var
LCLControl: TWinControl;
{$ENDIF}
begin
{$IFDEF Gtk2}
if not WidgetSizeIsEditable(Widget) then exit;
{$ENDIF}
{$IFDEF VerboseSizeMsg}
LCLControl:=TWinControl(GetLCLObject(Widget));
DbgOut('SetResizeRequest Widget=',DbgS(Widget));
if LCLControl is TWinControl then
DebugLn(' ',DbgSName(LCLControl),' LCLBounds=',dbgs(LCLControl.BoundsRect))
else
DebugLn(' ERROR: ',DbgSName(LCLControl));
{$ENDIF}
if not FWidgetsWithResizeRequest.Contains(Widget) then
FWidgetsWithResizeRequest.Add(Widget);
end;
{------------------------------------------------------------------------------
function WidgetSizeIsEditable(Widget: PGtkWidget): boolean;
True if the widget can be resized.
False if the size is under complete control of the gtk.
------------------------------------------------------------------------------}
function WidgetSizeIsEditable(Widget: PGtkWidget): boolean;
begin
if Widget=nil then exit(false);
if (GtkWidgetIsA(Widget,GTK_TYPE_WINDOW))
or (GtkWidgetIsA(Widget^.Parent,gtk_fixed_get_type))
or (GtkWidgetIsA(Widget^.Parent,gtk_layout_get_type))
then
Result:=true
else
Result:=false;
end;
procedure ReportNotObsolete(const Texts : String);
Begin
DebugLn('*********************************************');
DebugLn('*********************************************');
DebugLn('*************Non-Obsolete report*************');
DebugLn('*********************************************');
DebugLn('*************'+Texts+'*is being used yet.****');
DebugLn('*******Please remove this function from******');
DebugLn('*******the obsolete section in gtkproc.inc***');
DebugLn('*********************************************');
DebugLn('*********************************************');
DebugLn('*********************************************');
DebugLn('*********************************************');
end;
function TGDKColorToTColor(const value : TGDKColor) : TColor;
begin
Result := ((Value.Blue shr 8) shl 16) + ((Value.Green shr 8) shl 8)
+ (Value.Red shr 8);
end;
function TColortoTGDKColor(const value : TColor) : TGDKColor;
var
newColor : TGDKColor;
begin
if Value<0 then begin
FillChar(Result,SizeOf(Result),0);
exit;
end;
newColor.pixel := 0;
newColor.red := (value and $ff) * 257;
newColor.green := ((value shr 8) and $ff) * 257;
newColor.blue := ((value shr 16) and $ff) * 257;
Result := newColor;
end;
{------------------------------------------------------------------------------
Function: UpdateSysColorMap
Params: none
Returns: none
Reads the system colors.
------------------------------------------------------------------------------}
procedure UpdateSysColorMap(Widget: PGtkWidget; Lgs: TLazGtkStyle);
{$IFDEF VerboseUpdateSysColorMap}
function GdkColorAsString(c: TgdkColor): string;
begin
Result:='LCL='+DbgS(TGDKColorToTColor(c))
+' Pixel='+DbgS(c.Pixel)
+' Red='+DbgS(c.Red)
+' Green='+DbgS(c.Green)
+' Blue='+DbgS(c.Blue)
;
end;
{$ENDIF}
var
MainStyle: PGtkStyle;
begin
if Widget=nil then exit;
if not (Lgs in [lgsButton, lgsWindow, lgsMenuBar, lgsMenuitem,
lgsVerticalScrollbar, lgsHorizontalScrollbar, lgsTooltip]) then exit;
{$IFDEF NoStyle}
exit;
{$ENDIF}
//debugln('UpdateSysColorMap ',GetWidgetDebugReport(Widget));
gtk_widget_set_rc_style(Widget);
MainStyle := gtk_widget_get_style(Widget);
if MainStyle = nil then exit;
with MainStyle^ do
begin
{$IFDEF VerboseUpdateSysColorMap}
if rc_style<>nil then
begin
with rc_style^ do
begin
DebugLn('rc_style:');
DebugLn(' FG GTK_STATE_NORMAL ',GdkColorAsString(fg[GTK_STATE_NORMAL]));
DebugLn(' FG GTK_STATE_ACTIVE ',GdkColorAsString(fg[GTK_STATE_ACTIVE]));
DebugLn(' FG GTK_STATE_PRELIGHT ',GdkColorAsString(fg[GTK_STATE_PRELIGHT]));
DebugLn(' FG GTK_STATE_SELECTED ',GdkColorAsString(fg[GTK_STATE_SELECTED]));
DebugLn(' FG GTK_STATE_INSENSITIVE ',GdkColorAsString(fg[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' BG GTK_STATE_NORMAL ',GdkColorAsString(bg[GTK_STATE_NORMAL]));
DebugLn(' BG GTK_STATE_ACTIVE ',GdkColorAsString(bg[GTK_STATE_ACTIVE]));
DebugLn(' BG GTK_STATE_PRELIGHT ',GdkColorAsString(bg[GTK_STATE_PRELIGHT]));
DebugLn(' BG GTK_STATE_SELECTED ',GdkColorAsString(bg[GTK_STATE_SELECTED]));
DebugLn(' BG GTK_STATE_INSENSITIVE ',GdkColorAsString(bg[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' TEXT GTK_STATE_NORMAL ',GdkColorAsString(text[GTK_STATE_NORMAL]));
DebugLn(' TEXT GTK_STATE_ACTIVE ',GdkColorAsString(text[GTK_STATE_ACTIVE]));
DebugLn(' TEXT GTK_STATE_PRELIGHT ',GdkColorAsString(text[GTK_STATE_PRELIGHT]));
DebugLn(' TEXT GTK_STATE_SELECTED ',GdkColorAsString(text[GTK_STATE_SELECTED]));
DebugLn(' TEXT GTK_STATE_INSENSITIVE ',GdkColorAsString(text[GTK_STATE_INSENSITIVE]));
DebugLn('');
end;
end;
DebugLn('MainStyle:');
DebugLn(' FG GTK_STATE_NORMAL ',GdkColorAsString(fg[GTK_STATE_NORMAL]));
DebugLn(' FG GTK_STATE_ACTIVE ',GdkColorAsString(fg[GTK_STATE_ACTIVE]));
DebugLn(' FG GTK_STATE_PRELIGHT ',GdkColorAsString(fg[GTK_STATE_PRELIGHT]));
DebugLn(' FG GTK_STATE_SELECTED ',GdkColorAsString(fg[GTK_STATE_SELECTED]));
DebugLn(' FG GTK_STATE_INSENSITIVE ',GdkColorAsString(fg[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' BG GTK_STATE_NORMAL ',GdkColorAsString(bg[GTK_STATE_NORMAL]));
DebugLn(' BG GTK_STATE_ACTIVE ',GdkColorAsString(bg[GTK_STATE_ACTIVE]));
DebugLn(' BG GTK_STATE_PRELIGHT ',GdkColorAsString(bg[GTK_STATE_PRELIGHT]));
DebugLn(' BG GTK_STATE_SELECTED ',GdkColorAsString(bg[GTK_STATE_SELECTED]));
DebugLn(' BG GTK_STATE_INSENSITIVE ',GdkColorAsString(bg[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' TEXT GTK_STATE_NORMAL ',GdkColorAsString(text[GTK_STATE_NORMAL]));
DebugLn(' TEXT GTK_STATE_ACTIVE ',GdkColorAsString(text[GTK_STATE_ACTIVE]));
DebugLn(' TEXT GTK_STATE_PRELIGHT ',GdkColorAsString(text[GTK_STATE_PRELIGHT]));
DebugLn(' TEXT GTK_STATE_SELECTED ',GdkColorAsString(text[GTK_STATE_SELECTED]));
DebugLn(' TEXT GTK_STATE_INSENSITIVE ',GdkColorAsString(text[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' LIGHT GTK_STATE_NORMAL ',GdkColorAsString(light[GTK_STATE_NORMAL]));
DebugLn(' LIGHT GTK_STATE_ACTIVE ',GdkColorAsString(light[GTK_STATE_ACTIVE]));
DebugLn(' LIGHT GTK_STATE_PRELIGHT ',GdkColorAsString(light[GTK_STATE_PRELIGHT]));
DebugLn(' LIGHT GTK_STATE_SELECTED ',GdkColorAsString(light[GTK_STATE_SELECTED]));
DebugLn(' LIGHT GTK_STATE_INSENSITIVE ',GdkColorAsString(light[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' DARK GTK_STATE_NORMAL ',GdkColorAsString(dark[GTK_STATE_NORMAL]));
DebugLn(' DARK GTK_STATE_ACTIVE ',GdkColorAsString(dark[GTK_STATE_ACTIVE]));
DebugLn(' DARK GTK_STATE_PRELIGHT ',GdkColorAsString(dark[GTK_STATE_PRELIGHT]));
DebugLn(' DARK GTK_STATE_SELECTED ',GdkColorAsString(dark[GTK_STATE_SELECTED]));
DebugLn(' DARK GTK_STATE_INSENSITIVE ',GdkColorAsString(dark[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' MID GTK_STATE_NORMAL ',GdkColorAsString(mid[GTK_STATE_NORMAL]));
DebugLn(' MID GTK_STATE_ACTIVE ',GdkColorAsString(mid[GTK_STATE_ACTIVE]));
DebugLn(' MID GTK_STATE_PRELIGHT ',GdkColorAsString(mid[GTK_STATE_PRELIGHT]));
DebugLn(' MID GTK_STATE_SELECTED ',GdkColorAsString(mid[GTK_STATE_SELECTED]));
DebugLn(' MID GTK_STATE_INSENSITIVE ',GdkColorAsString(mid[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' BASE GTK_STATE_NORMAL ',GdkColorAsString(base[GTK_STATE_NORMAL]));
DebugLn(' BASE GTK_STATE_ACTIVE ',GdkColorAsString(base[GTK_STATE_ACTIVE]));
DebugLn(' BASE GTK_STATE_PRELIGHT ',GdkColorAsString(base[GTK_STATE_PRELIGHT]));
DebugLn(' BASE GTK_STATE_SELECTED ',GdkColorAsString(base[GTK_STATE_SELECTED]));
DebugLn(' BASE GTK_STATE_INSENSITIVE ',GdkColorAsString(base[GTK_STATE_INSENSITIVE]));
DebugLn('');
DebugLn(' BLACK ',GdkColorAsString(black));
DebugLn(' WHITE ',GdkColorAsString(white));
{$ENDIF}
{$IFNDEF DisableGtkSysColors}
// this map is taken from this research:
// http://www.endolith.com/wordpress/2008/08/03/wine-colors/
case Lgs of
lgsButton:
begin
SysColorMap[COLOR_ACTIVEBORDER] := TGDKColorToTColor(bg[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_INACTIVEBORDER] := TGDKColorToTColor(bg[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_WINDOWFRAME] := TGDKColorToTColor(mid[GTK_STATE_SELECTED]);
SysColorMap[COLOR_BTNFACE] := TGDKColorToTColor(bg[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_BTNSHADOW] := TGDKColorToTColor(dark[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_BTNTEXT] := TGDKColorToTColor(fg[GTK_STATE_NORMAL]);
SysColorMap[COLOR_BTNHIGHLIGHT] := TGDKColorToTColor(light[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_3DDKSHADOW] := TGDKColorToTColor(black);
SysColorMap[COLOR_3DLIGHT] := TGDKColorToTColor(bg[GTK_STATE_INSENSITIVE]);
end;
lgsWindow:
begin
// colors which can be only retrieved from the window manager (metacity)
SysColorMap[COLOR_ACTIVECAPTION] := TGDKColorToTColor(dark[GTK_STATE_SELECTED]);
SysColorMap[COLOR_INACTIVECAPTION] := TGDKColorToTColor(dark[GTK_STATE_NORMAL]);
SysColorMap[COLOR_GRADIENTACTIVECAPTION] := TGDKColorToTColor(light[GTK_STATE_SELECTED]);
SysColorMap[COLOR_GRADIENTINACTIVECAPTION] := TGDKColorToTColor(base[GTK_STATE_NORMAL]);
SysColorMap[COLOR_CAPTIONTEXT] := TGDKColorToTColor(white);
SysColorMap[COLOR_INACTIVECAPTIONTEXT] := TGDKColorToTColor(white);
// others
SysColorMap[COLOR_APPWORKSPACE] := TGDKColorToTColor(base[GTK_STATE_NORMAL]);
SysColorMap[COLOR_GRAYTEXT] := TGDKColorToTColor(fg[GTK_STATE_INSENSITIVE]);
SysColorMap[COLOR_HIGHLIGHT] := TGDKColorToTColor(base[GTK_STATE_SELECTED]);
SysColorMap[COLOR_HIGHLIGHTTEXT] := TGDKColorToTColor(fg[GTK_STATE_SELECTED]);
SysColorMap[COLOR_WINDOW] := TGDKColorToTColor(base[GTK_STATE_NORMAL]);
SysColorMap[COLOR_WINDOWTEXT] := TGDKColorToTColor(text[GTK_STATE_NORMAL]);
SysColorMap[COLOR_HOTLIGHT] := TGDKColorToTColor(light[GTK_STATE_NORMAL]);
SysColorMap[COLOR_BACKGROUND] := TGDKColorToTColor(bg[GTK_STATE_PRELIGHT]);
SysColorMap[COLOR_FORM] := TGDKColorToTColor(bg[GTK_STATE_NORMAL]);
end;
lgsMenuBar:
begin
SysColorMap[COLOR_MENUBAR] := TGDKColorToTColor(bg[GTK_STATE_NORMAL]);
end;
lgsMenuitem:
begin
SysColorMap[COLOR_MENU] := TGDKColorToTColor(light[GTK_STATE_ACTIVE]);
SysColorMap[COLOR_MENUTEXT] := TGDKColorToTColor(fg[GTK_STATE_NORMAL]);
SysColorMap[COLOR_MENUHILIGHT] := TGDKColorToTColor(bg[GTK_STATE_PRELIGHT]);
end;
lgsVerticalScrollbar,
lgsHorizontalScrollbar:
begin
SysColorMap[COLOR_SCROLLBAR] := TGDKColorToTColor(bg[GTK_STATE_ACTIVE]);
end;
lgsTooltip:
begin
SysColorMap[COLOR_INFOTEXT] := TGDKColorToTColor(fg[GTK_STATE_NORMAL]);
SysColorMap[COLOR_INFOBK] := TGDKColorToTColor(bg[GTK_STATE_NORMAL]);
end;
end;
{$ENDIF}
end;
end;
{------------------------------------------------------------------------------
Function: WaitForClipbrdAnswerDummyTimer
this is a helper function for WaitForClipboardAnswer
------------------------------------------------------------------------------}
function WaitForClipbrdAnswerDummyTimer(Client: Pointer): {$IFDEF Gtk2}gboolean{$ELSE}gint{$ENDIF}; cdecl;
begin
if CLient=nil then ;
Result:=GdkTrue; // go on, make sure getting a message at least every second
end;
function GetScreenWidthMM(GdkValue: boolean): integer;
begin
Result:=gdk_screen_width_mm;
if (Result<=0) and not GdkValue then
Result:=300; // some TV-out screens don't know there size
end;
function GetScreenHeightMM(GdkValue: boolean): integer;
begin
Result:=gdk_screen_height_mm;
if (Result<=0) and not GdkValue then
Result:=300; // some TV-out screens don't know there size
end;
{------------------------------------------------------------------------------
Function: WaitForClipboardAnswer
Params: none
Returns: true, if clipboard data arrived
waits til clipboard/selection answer arrived (max 1 second)
! While waiting the messagequeue will be processed !
------------------------------------------------------------------------------}
function WaitForClipboardAnswer(c: PClipboardEventData): boolean;
var
StartTime, CurTime: TSystemTime;
Timer: cardinal;
function ValidDateSelection : boolean;
begin
result := c^.Data.Selection<>0;
end;
begin
Result:=false;
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] A');
{$ENDIF}
if (ValidDateSelection) or (c^.Waiting) or (c^.Stopping) then begin
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] ValidDateSelection=',dbgs(ValidDateSelection),' Waiting=',dbgs(c^.Waiting),' Stopping=',dbgs(c^.Stopping));
{$ENDIF}
Result:=(ValidDateSelection);
exit;
end;
c^.Waiting:=true;
DateTimeToSystemTime(Time,StartTime);
//DebugLn('[WaitForClipboardAnswer] C');
Application.ProcessMessages;
//DebugLn('[WaitForClipboardAnswer] D');
if (ValidDateSelection) or (c^.Stopping) then begin
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] E Yeah, Response received after processing messages');
{$ENDIF}
Result:=(ValidDateSelection);
exit;
end;
//DebugLn('[WaitForClipboardAnswer] F');
// start a timer to make sure not waiting forever
Timer := gtk_timeout_add(500, @WaitForClipbrdAnswerDummyTimer, nil);
try
repeat
// just wait ...
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] G');
{$ENDIF}
Application.ProcessMessages;
if (ValidDateSelection) or (c^.Stopping) then begin
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] H Yeah, Response received after waiting with timer');
{$ENDIF}
Result:=(ValidDateSelection);
exit;
end;
DateTimeToSystemTime(Time,CurTime);
until (CurTime.Second*1000+CurTime.MilliSecond
-StartTime.Second*1000-StartTime.MilliSecond
>1000);
finally
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] H');
{$ENDIF}
// stop the timer
gtk_timeout_remove(Timer);
//DebugLn('[WaitForClipboardAnswer] END');
end;
{ $IFDEF DEBUG_CLIPBOARD}
DebugLn('[WaitForClipboardAnswer] WARNING: no answer received in time');
{ $ENDIF}
end;
{------------------------------------------------------------------------------
Function: RequestSelectionData
Params: ClipboardWidget - widget with connected signals 'selection_get'
and 'selection_clear_event'
ClipboardType
FormatID - the selection target format wanted
Returns: the TGtkSelectionData record
requests the format FormatID of clipboard of type ClipboardType and
waits til clipboard/selection answer arrived (max 1 second)
! While waiting the messagequeue will be processed !
------------------------------------------------------------------------------}
function RequestSelectionData(ClipboardWidget: PGtkWidget;
ClipboardType: TClipboardType; FormatID: PtrUInt): TGtkSelectionData;
function TimeIDExists(TimeID: guint32): boolean;
var
i: Integer;
begin
i:=ClipboardSelectionData.Count-1;
while (i>=0) do begin
if (PClipboardEventData(ClipboardSelectionData[i])^.TimeID=TimeID) then
exit(true);
dec(i);
end;
Result:=false;
end;
var
TimeID: cardinal;
c: PClipboardEventData;
sanity: Integer = 0;
begin
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[RequestSelectionData] FormatID=',dbgs(FormatID));
{$ENDIF}
FillChar(Result,SizeOf(TGtkSelectionData),0);
if (ClipboardWidget=nil) or (FormatID=0)
or (ClipboardTypeAtoms[ClipboardType]=0) then exit;
TimeID:= gdk_event_get_time(gtk_get_current_event);
// IMPORTANT: To retrieve data from xterm or kde applications
// the time id must be 0 or event^.time
repeat
while TimeIDExists(TimeID) do begin
inc(TimeID);
if TimeID>1010 then exit;
end;
New(c);
FillChar(c^,SizeOf(TClipboardEventData),0);
c^.TimeID:=TimeID;
ClipboardSelectionData.Add(c);
try
{$IFDEF DEBUG_CLIPBOARD}
DebugLn('[RequestSelectionData] TimeID=',dbgs(TimeID),' Type=',GdkAtomToStr(ClipboardTypeAtoms[ClipboardType]),' FormatID=',GdkAtomToStr(FormatID), ' Sanity=', IntToStr(Sanity));
{$ENDIF}
if gtk_selection_convert(ClipboardWidget, ClipboardTypeAtoms[ClipboardType],
FormatID, TimeID)<>GdkFalse
then begin
if not WaitForClipboardAnswer(c) then exit;
Result:=c^.Data;
break;
end;
finally
ClipboardSelectionData.Remove(c);
Dispose(c);
end;
Inc(sanity);
sleep(100);
until false or (sanity > 10);
end;
{------------------------------------------------------------------------------
Function: FreeClipboardTargetEntries
Params: ClipboardType
Returns: -
frees the memory of a ClipboardTargetEntries list
------------------------------------------------------------------------------}
procedure FreeClipboardTargetEntries(ClipboardType: TClipboardType);
var i: integer;
begin
if ClipboardTargetEntries[ClipboardType]<>nil then begin
for i:=0 to ClipboardTargetEntryCnt[ClipboardType]-1 do
StrDispose(ClipboardTargetEntries[ClipboardType][i].Target);
FreeMem(ClipboardTargetEntries[ClipboardType]);
end;
end;
{------------------------------------------------------------------------------
function GdkAtomToStr(const Atom: TGdkAtom): string;
Returns the associated string
------------------------------------------------------------------------------}
function GdkAtomToStr(const Atom: TGdkAtom): string;
var
p: Pgchar;
begin
p:=gdk_atom_name(Atom);
Result:=p;
if p<>nil then g_free(p);
end;
{-------------------------------------------------------------------------------
function CreateFormContents(AForm: TCustomForm;
var FormWidget: Pointer): Pointer;
Creates the contents for the form (normally a hbox plus a client area.
The hbox is needed for the menu.) The FormWidget is the main widget, for which
the client area is associated. If FormWidget=nil then the hbox will be used
as main widget.
-------------------------------------------------------------------------------}
function CreateFormContents(AForm: TCustomForm;
var FormWidget: Pointer; AWidgetInfo: PWidgetInfo = nil): Pointer;
var
ScrolledWidget, ClientAreaWidget: PGtkWidget;
WindowStyle: PGtkStyle;
Adjustment: PGtkAdjustment;
begin
// Create the VBox. We need that to place controls outside
// the client area (like menu)
Result := gtk_vbox_new(False, 0);
if FormWidget = nil then
FormWidget := Result;
// Create the form client area (a scrolled window with a gtklayout
// with the style of a window)
ScrolledWidget := gtk_scrolled_window_new(nil, nil);
gtk_box_pack_end(Result, ScrolledWidget, True, True, 0);
gtk_widget_show(ScrolledWidget);
ClientAreaWidget := gtk_layout_new(nil, nil);
WindowStyle := GetStyle(lgsWindow);
gtk_widget_set_style(ClientAreaWidget, WindowStyle);
//debugln('CreateFormContents Style=',GetStyleDebugReport(WindowStyle));
gtk_container_add(PGtkContainer(ScrolledWidget), ClientAreaWidget);
gtk_object_set_data(FormWidget, odnScrollArea, ScrolledWidget);
gtk_widget_show(ClientAreaWidget);
SetFixedWidget(FormWidget, ClientAreaWidget);
SetMainWidget(FormWidget, ClientAreaWidget);
if ScrolledWidget <> nil then
begin
GTK_WIDGET_UNSET_FLAGS(PGtkScrolledWindow(ScrolledWidget)^.hscrollbar,
GTK_CAN_FOCUS);
GTK_WIDGET_UNSET_FLAGS(PGtkScrolledWindow(ScrolledWidget)^.vscrollbar,
GTK_CAN_FOCUS);
gtk_scrolled_window_set_policy(PGtkScrolledWindow(ScrolledWidget),
GTK_POLICY_NEVER,GTK_POLICY_NEVER);
Adjustment := gtk_scrolled_window_get_vadjustment(PGTKScrolledWindow(ScrolledWidget));
if Adjustment <> nil then
gtk_object_set_data(PGTKObject(Adjustment), odnScrollBar,
PGTKScrolledWindow(ScrolledWidget)^.vscrollbar);
Adjustment := gtk_scrolled_window_get_hadjustment(PGTKScrolledWindow(ScrolledWidget));
if Adjustment <> nil then
gtk_object_set_data(PGTKObject(Adjustment), odnScrollBar,
PGTKScrolledWindow(ScrolledWidget)^.hscrollbar);
{$ifdef gtk2}
if (AWidgetInfo <> nil) and
(gtk_major_version >= 2) and (gtk_minor_version > 8) then
begin
g_signal_connect(PGTKScrolledWindow(ScrolledWidget)^.hscrollbar, 'change-value',
TGCallback(@Gtk2RangeScrollCB), AWidgetInfo);
g_signal_connect(PGTKScrolledWindow(ScrolledWidget)^.vscrollbar, 'change-value',
TGCallback(@Gtk2RangeScrollCB), AWidgetInfo);
end;
{$endif}
end;
end;
function IndexOfStyle(aStyle: TLazGtkStyle): integer;
begin
Result:=IndexOfStyleWithName(LazGtkStyleNames[aStyle]);
end;
{------------------------------------------------------------------------------
Function: IndexOfWithNameStyle
Params: WName
Returns: Index of Style
Returns the Index within the Styles property of WNAME
------------------------------------------------------------------------------}
function IndexOfStyleWithName(const WName : String): integer;
begin
if Styles<>nil then begin
for Result:=0 to Styles.Count-1 do
if CompareText(WName,Styles[Result])=0 then exit;
end;
Result:=-1;
end;
{------------------------------------------------------------------------------
Function: ReleaseStyle
Params: WName
Returns: nothing
Tries to release a Style corresponding to the Widget Name passed, aka 'button',
'default', checkbox', etc. This should only be called on theme change or on
application terminate.
------------------------------------------------------------------------------}
function NewStyleObject : PStyleObject;
begin
New(Result);
FillChar(Result^, SizeOf(TStyleObject), 0);
end;
procedure FreeStyleObject(var StyleObject : PStyleObject);
// internal function to dispose a styleobject
// it does *not* remove it from the style lists
begin
if StyleObject <> nil then
begin
if StyleObject^.Obj <> nil then
gtk_object_destroy(StyleObject^.Obj);
if StyleObject^.Widget <> nil then
begin
// first unref
gtk_widget_unref(StyleObject^.Widget);
// then destroy
gtk_widget_destroy(StyleObject^.Widget);
end;
if StyleObject^.Style <> nil then
if StyleObject^.Style^.{$IFDEF Gtk2}attach_count{$ELSE}Ref_Count{$ENDIF} > 0 then
gtk_style_unref(StyleObject^.Style);
Dispose(StyleObject);
StyleObject := nil;
end;
end;
procedure ReleaseAllStyles;
var
StyleObject: PStyleObject;
lgs: TLazGtkStyle;
i: Integer;
begin
if Styles=nil then exit;
{$IFDEF Gtk2}
if DefaultPangoLayout<>nil then begin
g_object_unref(DefaultPangoLayout);
DefaultPangoLayout:=nil;
end;
{$ENDIF}
for i:=Styles.Count-1 downto 0 do begin
StyleObject:=PStyleObject(Styles.Objects[i]);
FreeStyleObject(StyleObject);
end;
Styles.Clear;
for lgs:=Low(TLazGtkStyle) to High(TLazGtkStyle) do
StandardStyles[lgs]:=nil;
end;
procedure ReleaseStyle(aStyle: TLazGtkStyle);
var
StyleObject: PStyleObject;
l: Integer;
begin
if Styles=nil then exit;
if aStyle in [lgsUserDefined] then
RaiseGDBException('');// user styles are defined by name
StyleObject:=StandardStyles[aStyle];
if StyleObject<>nil then begin
l:=IndexOfStyle(aStyle);
Styles.Delete(l);
StandardStyles[aStyle]:=nil;
FreeStyleObject(StyleObject);
end;
end;
procedure ReleaseStyleWithName(const WName : String);
var
l : Longint;
s : PStyleObject;
begin
if Styles=nil then exit;
l := IndexOfStyleWithName(WName);
If l >= 0 then begin
If Styles.Objects[l] <> nil then
Try
s := PStyleObject(Styles.Objects[l]);
FreeStyleObject(S);
Except
DebugLn('[ReleaseStyle] : Unable To Unreference Style');
end;
Styles.Delete(l);
end;
end;
function GetStyle(aStyle: TLazGtkStyle): PGTKStyle;
begin
if Styles = nil then Exit(nil);
if aStyle in [lgsUserDefined] then
RaiseGDBException(''); // user styles are defined by name
if StandardStyles[aStyle] <> nil then // already created
Result := StandardStyles[aStyle]^.Style
else // create it
Result := GetStyleWithName(LazGtkStyleNames[aStyle]);
end;
procedure tooltip_window_style_set(Widget: PGtkWidget; PreviousStyle: PGtkStyle;
StyleObject: PStyleObject); cdecl;
begin
StyleObject^.Style := gtk_widget_get_style(Widget);
UpdateSysColorMap(Widget, lgsToolTip);
end;
{------------------------------------------------------------------------------
Function: GetStyleWithName
Params: none
Returns: Returns a Corresponding Style
Tries to get the Style corresponding to the Widget Name passed, aka 'button',
'default', checkbox', etc. for use within such routines as DrawFrameControl
to attempt to supply theme dependent drawing. Styles are stored in a TStrings
list which is only updated on theme change, to ensure fast efficient retrieval
of Styles.
------------------------------------------------------------------------------}
function GetStyleWithName(const WName: String) : PGTKStyle;
var
StyleObject : PStyleObject;
function CreateStyleNotebook: PGTKWidget;
var
NoteBookWidget: PGtkNotebook;
//NoteBookPageWidget: PGtkWidget;
NoteBookPageClientAreaWidget: PGtkWidget;
NoteBookTabLabel: PGtkWidget;
NoteBookTabMenuLabel: PGtkWidget;
begin
Result:=gtk_notebook_new;
NoteBookWidget := PGtkNoteBook(Result);
//NoteBookPageWidget := gtk_hbox_new(false, 0);
NoteBookPageClientAreaWidget := CreateFixedClientWidget{$IFNDEF GtkFixedWithWindow}(false){$ENDIF};
gtk_widget_show(NoteBookPageClientAreaWidget);
//gtk_container_add(GTK_CONTAINER(NoteBookPageWidget),
// NoteBookPageClientAreaWidget);
//gtk_widget_show(NoteBookPageWidget);
NoteBookTabLabel:=gtk_label_new('Lazarus');
gtk_widget_show(NoteBookTabLabel);
NoteBookTabMenuLabel:=gtk_label_new('Lazarus');
gtk_widget_show(NoteBookTabMenuLabel);
gtk_notebook_append_page_menu(NoteBookWidget,NoteBookPageClientAreaWidget,
NoteBookTabLabel,NoteBookTabMenuLabel);
gtk_widget_set_usize(Result,400,400);
end;
procedure ResizeWidget(CurWidget: PGTKWidget; NewWidth, NewHeight: integer);
{$IFDEF Gtk1}
begin
gtk_widget_set_usize(StyleObject^.Widget,NewWidth,NewHeight);
end;
{$ELSE}
var
allocation: TGtkAllocation;
begin
allocation.x:=0;
allocation.y:=0;
allocation.width:=NewWidth;
allocation.height:=NewHeight;
//gtk_widget_set_usize(StyleObject^.Widget,NewWidth,NewHeight);
gtk_widget_size_allocate(CurWidget,@allocation);
StyleObject^.FrameBordersValid:=false;
end;
{$ENDIF}
var
Tp : Pointer;
l : Longint;
NoName: PGChar;
lgs: TLazGtkStyle;
WidgetName: String;
//VBox: PGtkWidget;
AddToStyleWindow: Boolean;
StyleWindowWidget: PGtkWidget;
Requisition: TGtkRequisition;
WindowFixedWidget: PGtkWidget;
VBox: PGtkWidget;
begin
Result := nil;
if Styles=nil then exit;
{$IFDEF NoStyle}
exit;
{$ENDIF}
if (WName='') then exit;
l:=IndexOfStyleWithName(WName);
//DebugLn(['GetStyleWithName START ',WName,' ',l]);
if l >= 0 then
begin
StyleObject:=PStyleObject(Styles.Objects[l]);
Result := StyleObject^.Style;
end else
begin
// create a new style object
StyleObject := NewStyleObject;
lgs := lgsUserDefined;
Tp := nil;
AddToStyleWindow := True;
WidgetName := 'LazStyle' + WName;
// create a style widget
If CompareText(WName,LazGtkStyleNames[lgsButton])=0 then begin
StyleObject^.Widget := GTK_BUTTON_NEW;
lgs:=lgsButton;
end
else
If CompareText(WName,LazGtkStyleNames[lgsLabel])=0 then begin
StyleObject^.Widget := GTK_LABEL_NEW('StyleLabel');
lgs:=lgsLabel;
end
else
If CompareText(WName,LazGtkStyleNames[lgsDefault])=0 then begin
lgs:=lgsDefault;
AddToStyleWindow:=false;
NoName:=nil;
StyleObject^.Widget :=
// GTK2 does not allow to instantiate the abstract base Widget
// so we use the "invisible" widget, which should never be defined
// by the theme
GTK_WIDGET_NEW(
{$IFDEF Gtk2}GTK_TYPE_INVISIBLE{$ELSE}GTK_WIDGET_TYPE{$ENDIF},
NoName,[]);
end
else
If CompareText(WName,LazGtkStyleNames[lgsWindow])=0 then begin
lgs:=lgsWindow;
StyleObject^.Widget := GTK_WINDOW_NEW(GTK_WINDOW_TOPLEVEL);
AddToStyleWindow:=false;
gtk_widget_hide(StyleObject^.Widget);
// create the fixed widget
// (where to put all style widgets, that need a parent for realize)
VBox:=gtk_vbox_new(false,0);// vbox is needed for menu above and fixed widget below
gtk_widget_show(VBox);
gtk_container_add(PGtkContainer(StyleObject^.Widget), VBox);
gtk_object_set_data(PGtkObject(StyleObject^.Widget),'vbox',VBox);
WindowFixedWidget:=CreateFixedClientWidget;
gtk_widget_show(WindowFixedWidget);
gtk_container_add(PGtkContainer(VBox), WindowFixedWidget);
gtk_object_set_data(PGtkObject(StyleObject^.Widget),'fixedwidget',WindowFixedWidget);
gtk_widget_realize(StyleObject^.Widget);
end
else
If CompareText(WName,LazGtkStyleNames[lgsCheckbox])=0 then begin
lgs:=lgsCheckbox;
StyleObject^.Widget := GTK_CHECK_BUTTON_NEW;
end
else
If CompareText(WName,LazGtkStyleNames[lgsRadiobutton])=0 then begin
lgs:=lgsRadiobutton;
StyleObject^.Widget := GTK_RADIO_BUTTON_NEW(nil);
end
else
If CompareText(WName,LazGtkStyleNames[lgsMenu])=0 then begin
lgs:=lgsMenu;
{$IFDEF Gtk1}
AddToStyleWindow:=false;
{$ENDIF}
StyleObject^.Widget := gtk_menu_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsMenuBar])=0 then begin
lgs:=lgsMenuBar;
{$IFDEF Gtk1}
AddToStyleWindow:=false;
{$ENDIF}
StyleObject^.Widget := gtk_menu_bar_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsMenuitem])=0 then begin
lgs:=lgsMenuitem;
{$IFDEF Gtk1}
AddToStyleWindow:=false;
StyleObject^.Widget := gtk_menu_item_new;
{$ELSE}
// image menu item is needed to correctly return theme options
StyleObject^.Widget := gtk_image_menu_item_new;
{$ENDIF}
end
else
If CompareText(WName,LazGtkStyleNames[lgsStatusBar])=0 then begin
lgs:=lgsStatusBar;
AddToStyleWindow:=true;
StyleObject^.Widget := gtk_statusbar_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsCalendar])=0 then begin
lgs:=lgsCalendar;
AddToStyleWindow:=true;
StyleObject^.Widget := gtk_calendar_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsList])=0 then begin
lgs:=lgsList;
StyleObject^.Widget := gtk_list_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsVerticalScrollbar])=0 then begin
lgs:=lgsVerticalScrollbar;
StyleObject^.Widget := gtk_vscrollbar_new(nil);
end
else
If CompareText(WName,LazGtkStyleNames[lgsHorizontalScrollbar])=0 then begin
lgs:=lgsHorizontalScrollbar;
StyleObject^.Widget := gtk_hscrollbar_new(nil);
end
else
If CompareText(WName,LazGtkStyleNames[lgsVerticalPaned])=0 then begin
lgs:=lgsVerticalPaned;
StyleObject^.Widget := gtk_vpaned_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsHorizontalPaned])=0 then begin
lgs:=lgsHorizontalPaned;
StyleObject^.Widget := gtk_hpaned_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsNotebook])=0 then begin
lgs:=lgsNotebook;
StyleObject^.Widget := CreateStyleNotebook;
end
else
if CompareText(WName,LazGtkStyleNames[lgsTooltip])=0 then
begin
lgs := lgsTooltip;
Tp := gtk_tooltips_new;
gtk_tooltips_force_window(Tp);
StyleObject^.Widget := PGTKTooltips(Tp)^.Tip_Window;
gtk_widget_ref(StyleObject^.Widget);// MG: why is this needed?
{$IFNDEF GTK1}
g_signal_connect(StyleObject^.Widget, 'style-set',
TGCallback(@tooltip_window_style_set), StyleObject);
{$ENDIF}
WidgetName := 'gtk-tooltip-lcl';
StyleObject^.Obj := Tp;
Tp := nil;
{$IFDEF GTK1}
AddToStyleWindow := False;
{$ENDIF}
end
else
If CompareText(WName,LazGtkStyleNames[lgsHScale])=0 then begin
lgs:=lgsHScale;
TP := PGtkWidget( gtk_adjustment_new (0, 0, 100, 1, 10, 0));
StyleObject^.Widget := gtk_hscale_new (PGTKADJUSTMENT (TP));
end
else
If CompareText(WName,LazGtkStyleNames[lgsVScale])=0 then begin
lgs:=lgsVScale;
TP := PGtkWidget( gtk_adjustment_new (0, 0, 100, 1, 10, 0));
StyleObject^.Widget := gtk_vscale_new (PGTKADJUSTMENT (TP));
end
else
If CompareText(WName,LazGtkStyleNames[lgsGroupBox])=0 then begin
lgs:=lgsGroupBox;
StyleObject^.Widget := gtk_frame_new('GroupBox');
WindowFixedWidget:=CreateFixedClientWidget{$IFNDEF GtkFixedWithWindow}(false){$ENDIF};
gtk_widget_show(WindowFixedWidget);
gtk_container_add(PGtkContainer(StyleObject^.Widget), WindowFixedWidget);
gtk_object_set_data(PGtkObject(StyleObject^.Widget),'fixedwidget',WindowFixedWidget);
end
{$ifdef gtk2}
else
If CompareText(WName,LazGtkStyleNames[lgsTreeView])=0 then begin
lgs:=lgsTreeView;
StyleObject^.Widget := gtk_tree_view_new;
gtk_tree_view_append_column(PGtkTreeView(StyleObject^.Widget), gtk_tree_view_column_new);
end
{$endif}
else
If CompareText(WName,LazGtkStyleNames[lgsToolBar])=0 then begin
lgs:=lgsToolBar;
StyleObject^.Widget := gtk_toolbar_new;
end
else
If CompareText(WName,LazGtkStyleNames[lgsToolButton])=0 then begin
lgs:=lgsToolButton;
StyleObject^.Widget := gtk_toolbar_append_item(PGtkToolBar(GetStyleWidget(lgsToolBar)), 'B', nil, nil, nil, nil, nil);
end
else
if CompareText(WName,LazGtkStyleNames[lgsScrolledWindow])=0 then begin
lgs:=lgsScrolledWindow;
StyleObject^.Widget := gtk_scrolled_window_new(nil, nil);
end
else
If CompareText(WName,LazGtkStyleNames[lgsGTK_Default])=0 then begin
lgs:=lgsGTK_Default;
AddToStyleWindow:=false;
StyleObject^.Widget := nil;
StyleObject^.Style := gtk_style_new;
end
else begin
// unknown style name -> bug
FreeStyleObject(StyleObject);
AddToStyleWindow:=false;
RaiseGDBException('');
end;
if (lgs<>lgsUserDefined) and (StandardStyles[lgs]<>nil) then begin
// consistency error
RaiseGDBException('');
end;
// ensure style of the widget
If (StyleObject^.Widget <> nil) then begin
gtk_widget_ref(StyleObject^.Widget);
// put style widget on style window, so that it can be realized
if AddToStyleWindow then
begin
gtk_widget_show_all(StyleObject^.Widget);
if GtkWidgetIsA(StyleObject^.Widget,GTK_TYPE_MENU) then
begin
// attach menu to window
gtk_menu_attach_to_widget(PGtkMenu(StyleObject^.Widget),
GetStyleWidget(lgsWindow), nil);
end
else
if GtkWidgetIsA(StyleObject^.Widget,GTK_TYPE_MENU_BAR) then
begin
StyleWindowWidget:=GetStyleWidget(lgsWindow);
// add menu above the forms client area (fixed widget)
VBox:=PGTKWidget(
gtk_object_get_data(PGtkObject(StyleWindowWidget),'vbox'));
gtk_box_pack_start(PGTKBox(VBox), StyleObject^.Widget, False, False, 0);
end
else
if GtkWidgetIsA(StyleObject^.Widget,GTK_TYPE_MENU_ITEM) then
begin
gtk_menu_bar_append({$IFDEF Gtk1}
PGtkMenuBar(GetStyleWidget(lgsMenuBar)),
{$ELSE}
GetStyleWidget(lgsMenuBar),
{$ENDIF}
StyleObject^.Widget);
end
else
{$ifdef gtk2}
if GtkWidgetIsA(StyleObject^.Widget, GTK_TYPE_TOOL_BUTTON) then
begin
//gtk_toolbar_insert();
gtk_toolbar_append_widget(GTK_TOOLBAR(GetStyleWidget(lgsToolBar)),
StyleObject^.Widget, nil, nil);
end
else
{$endif}
if (lgs = lgsToolButton) or
(lgs = lgsTooltip) then
begin
// already on a parent => nothing to do
end
else
begin
StyleWindowWidget:=GetStyleWidget(lgsWindow);
// add widget on client area of form
WindowFixedWidget:=PGTKWidget(
gtk_object_get_data(PGtkObject(StyleWindowWidget),'fixedwidget'));
//DebugLn('GetStyleWithName adding on hidden stylewindow ',WName,' ',GetWidgetDebugReport(StyleObject^.Widget));
if WindowFixedWidget <> nil then
gtk_fixed_put(PGtkFixed(WindowFixedWidget),StyleObject^.Widget,10,10);
end;
end;
gtk_widget_set_name(StyleObject^.Widget,PChar(WidgetName));
gtk_widget_ensure_style(StyleObject^.Widget);
// request default sizing
FillChar(Requisition,SizeOf(Requisition),0);
gtk_widget_size_request(StyleObject^.Widget, @Requisition);
StyleObject^.Style:=gtk_widget_get_style(StyleObject^.Widget);
// ToDo: find out, why sometimes the style is not initialized.
// for example: why the following occurs:
if CompareText(WName,'button')=0 then begin
if StyleObject^.Style^.light_gc[GTK_STATE_NORMAL]=nil then begin
//DebugLn('GetStyleWithName ',WName);
end;
end;
if AddToStyleWindow then begin
if not GtkWidgetIsA(StyleObject^.Widget,GTK_WINDOW_GET_TYPE) then begin
//DebugLn(['GetStyleWithName realizing ...']);
gtk_widget_realize(StyleObject^.Widget);
//DebugLn('AddToStyleWindow realized: ',WName,' ',GetWidgetDebugReport(StyleObject^.Widget));
end;
ResizeWidget(StyleObject^.Widget,200,200);
end;
end;
// increase refcount of style
if StyleObject^.Style <> nil then
if CompareText(WName,LazGtkStyleNames[lgsGTK_Default])<>0 then
StyleObject^.Style := GTK_Style_Ref(StyleObject^.Style);
// if successful add to style objects list
if StyleObject^.Style <> nil then
begin
Styles.AddObject(WName, TObject(StyleObject));
if lgs <> lgsUserDefined then
StandardStyles[lgs] := StyleObject;
Result := StyleObject^.Style;
UpdateSysColorMap(StyleObject^.Widget, lgs);
// ToDo: create all gc of the style
//gtk_widget_set_rc_style(StyleObject^.Widget);
if lgs = lgsTooltip then
gtk_widget_hide_all(StyleObject^.Widget);
end
else begin
// no success, clean up
FreeStyleObject(StyleObject);
DebugLn('WARNING: GetStyleWithName ',WName,' failed');
end;
// clean up
if Tp <> nil then
gtk_object_destroy(Tp);
end;
end;
function GetStyleWidget(aStyle: TLazGtkStyle) : PGTKWidget;
begin
if aStyle in [lgsUserDefined] then
RaiseGDBException('');// user styles are defined by name
if StandardStyles[aStyle]<>nil then
// already created
Result:=StandardStyles[aStyle]^.Widget
else
// create it
Result:=GetStyleWidgetWithName(LazGtkStyleNames[aStyle]);
end;
function GetStyleWidgetWithName(const WName : String) : PGTKWidget;
var
l : Longint;
begin
Result := nil;
// init style
GetStyleWithName(WName);
// return widget
l:=IndexOfStyleWithName(WName);
if l>=0 then
Result := PStyleObject(Styles.Objects[l])^.Widget;
end;
{------------------------------------------------------------------------------
Function: LoadDefaultFont(Desc)
Params: none
Returns: Returns the default Font
For Text/Font Routines: if the Font is invalid, this can be used instead, or
if the DT_internal flag is used(aka use system font) this is used. This is
also the font returned by GetStockObject(SYSTEM_FONT).
It attempts to get the font from the default Style, or if none is available,
a new style(aka try and get GTK builtin values), if that fails tries to get
a generic fixed font, if THAT fails, it gets whatever font is available.
If the result is not nil it MUST be GDK_FONT_UNREF'd when done.
------------------------------------------------------------------------------}
function LoadDefaultFont: TGtkIntfFont;
{$IFDEF Gtk1}
var
Style : PGTKStyle;
{$ENDIF}
begin
{$IFDEF Gtk2}
Result:=gtk_widget_create_pango_layout(GetStyleWidget(lgsdefault), nil);
{$ELSE Gtk1}
Result := nil;
Style := GetStyle(lgsDefault);
if Style = nil then
Style := GetStyle(lgsGTK_Default);
if Style <> nil then begin
Result := Style^.Font;
if Result = nil then
{$IFNDEF NoStyle}
if (Style^.RC_Style <> nil) then begin
if (Style^.RC_Style^.font_name <> nil) then
Result := gdk_font_load(Style^.RC_Style^.font_name);
end;
{$ENDIF}
end;
If Result = nil then
Result := gdk_fontset_load('-*-fixed-*-*-*-*-*-120-*-*-*-*-*-*');
if Result = nil then
Result := gdk_fontset_load('-*-*-*-*-*-*-*-*-*-*-*-*-*-*');
{$ENDIF}
If Result <> nil then
ReferenceGtkIntfFont(Result);
end;
{$Ifdef GTK2}
function LoadDefaultFontDesc: PPangoFontDescription;
var
Style : PGTKStyle;
begin
Result := nil;
{$IFDEF VerboseGtkToDos}{$WARNING ToDo LoadDefaultFontDesc: get a working default pango font description}{$ENDIF}
Result := pango_font_description_from_string('sans 12');
exit;
Style := GetStyle(lgsLabel);
if Style = nil then
Style := GetStyle(lgsDefault);
if Style = nil then
Style := GetStyle(lgsGTK_Default);
If (Style <> nil) then begin
Result := pango_font_description_copy(Style^.font_desc);
end;
If Result = nil then
Result := pango_font_description_from_string('sans 12');
if Result = nil then
Result := pango_font_description_from_string('12');
end;
{$ENDIF}
function GetDefaultFontName: string;
var
Style: PGtkStyle;
{$IFDEF GTK2}
PangoFontDesc: PPangoFontDescription;
{$ELSE}
p,t: pchar;
AFont: PGdkFont;
{$ENDIF}
begin
Result:='';
Style := GetStyle(lgsDefault);
if Style = nil then
Style := GetStyle(lgsGTK_Default);
If Style <> nil then begin
{$IFDEF GTK1}
{$IFNDEF NoStyle}
if (Style^.RC_Style <> nil) then
with style^.RC_Style^ do begin
if (font_name <> nil) then
Result := font_name;
if (Result='') and (fontset_name<>nil) then
begin
// fontset_name it's usually a comma separated list of font names
// try to get the first valid font.
p := fontset_name;
while p<>nil do begin
t := strscan(p, ',');
if t=nil then
result := p
else begin
result := copy(p, 1, t-p);
while (t<>nil) and (t^ in [',',' ',#9,#10,#13]) do
inc(t);
end;
AFont := gdk_font_load(pchar(result));
if AFont<>nil then begin
gdk_font_unref(AFont);
{$IFDEF VerboseFonts}
debugln('DefaultFont found in fontset: ',result);
{$ENDIF}
break;
end;
p := t;
end;
end;
end;
{$ENDIF}
{$ENDIF}
{$IFDEF GTK2}
If (Style <> nil) then begin
PangoFontDesc := Style^.font_desc;
if PangoFontDesc<>nil then begin
Result:=pango_font_description_get_family(PangoFontDesc);
end;
end;
{$ENDIF}
end;
{$IFDEF VerboseFonts}
DebugLn('GetDefaultFontName: DefaultFont=',result);
{$ENDIF}
end;
procedure RealizeGDKColor(ColorMap: PGdkColormap; Color: PGDKColor);
var
AllocResult: gboolean;
begin
if ColorMap=nil then ColorMap:=gdk_colormap_get_system;
if (Color^.pixel = 0)
and ((Color^.red<>0) or (Color^.blue<>0) or (Color^.green<>0)) then
gdk_colormap_alloc_colors(ColorMap, Color, 1, false, true, @AllocResult)
else
gdk_colormap_query_color(ColorMap, Color^.pixel, Color);
end;
procedure RealizeGtkStyleColor(Style: PGTKStyle; Color: PGDKColor);
begin
if (Style<>nil) then
RealizeGDKColor(Style^.ColorMap,Color)
else
RealizeGDKColor(nil,Color);
end;
function GetSysGCValues(Color: TColorRef;
ThemeWidget: PGtkWidget): TGDKGCValues;
// ThemeWidget can be nil
function GetWidgetWithBackgroundWindow(Widget: PGtkWidget): PGtkWidget;
// returns the gtk widget which has the background gdk window
var
WindowOwnerWidget: PGtkWidget;
begin
Result:=Widget;
if Result=nil then exit;
if Result^.window=nil then exit;
gdk_window_get_user_data(Result^.window,PGPointer(@WindowOwnerWidget));
Result:=WindowOwnerWidget;
if Result=nil then exit;
end;
var
Style: PGTKStyle;
GC: PGDKGC;
Pixmap: PGDKPixmap;
SysColor: TColorRef;
BaseColor: TColorRef;
Red, Green, Blue: byte;
begin
// Set defaults in case something goes wrong
FillChar(Result, SizeOf(Result), 0);
Style := nil;
GC := nil;
Pixmap := nil;
SysColor := ColorToRGB(TColor(Color));
Result.Fill := GDK_Solid;
RedGreenBlue(TColor(SysColor), Red, Green, Blue);
Result.foreground.Red:=gushort(Red) shl 8 + Red;
Result.foreground.Green:=gushort(Green) shl 8 + Green;
Result.foreground.Blue:=gushort(Blue) shl 8 + Blue;
{$IfDef Disable_GC_SysColors}
exit;
{$EndIf}
BaseColor := Color and $FF;
case BaseColor of
{These are WM/X defined, but might be possible to get
COLOR_CAPTIONTEXT
COLOR_INACTIVECAPTIONTEXT}
{These Are incompatible or WM defined
COLOR_ACTIVECAPTION
COLOR_INACTIVECAPTION
COLOR_GRADIENTACTIVECAPTION
COLOR_GRADIENTINACTIVECAPTION
COLOR_WINDOWFRAME
COLOR_ACTIVEBORDER
COLOR_INACTIVEBORDER}
COLOR_BACKGROUND:
begin
Style := GetStyle(lgsDefault);
if Style = nil then
Style := GetStyle(lgsWindow);
if Style = nil then
exit;
Pixmap := Style^.bg_pixmap[GTK_STATE_NORMAL];
if Pixmap <> nil then
begin
Result.Fill := GDK_Tiled;
Result.Tile := Pixmap;
end
else
begin
GC := Style^.bg_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.bg[GTK_STATE_PRELIGHT];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
end;
COLOR_INFOBK :
begin
Style := GetStyle(lgsTooltip);
if Style = nil then
Style := GetStyle(lgsWindow);
if Style = nil then
exit;
Pixmap := Style^.bg_pixmap[GTK_STATE_NORMAL];
if Pixmap <> nil then
begin
Result.Fill := GDK_Tiled;
Result.Tile := Pixmap;
end
else
begin
GC := Style^.bg_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
{$IFDEF Gtk1}
Result.foreground := Style^.bg[GTK_STATE_PRELIGHT];
{$ELSE}
Result.foreground := Style^.bg[GTK_STATE_NORMAL];
{$ENDIF}
end
else
GDK_GC_Get_Values(GC, @Result);
end;
end;
COLOR_INFOTEXT :
begin
Style := GetStyle(lgsTooltip);
if Style = nil then
Style := GetStyle(lgsWindow);
if Style = nil then
exit;
GC := Style^.fg_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.fg[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_FORM,
COLOR_MENU,
COLOR_SCROLLBAR,
COLOR_BTNFACE :
begin
case BaseColor of
COLOR_FORM: Style := GetStyle(lgsWindow);
COLOR_BTNFACE: Style := GetStyle(lgsButton);
COLOR_MENU: Style := GetStyle(lgsMenu);
COLOR_SCROLLBAR: Style := GetStyle(lgsHorizontalScrollbar);
end;
if Style = nil then
exit;
Pixmap := Style^.bg_pixmap[GTK_STATE_NORMAL];
if Pixmap <> nil then
begin
Result.Fill := GDK_Tiled;
Result.Tile := Pixmap;
end else
begin
GC := Style^.bg_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.bg[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
end;
COLOR_3DDKSHADOW,
COLOR_BTNSHADOW :
begin
Style := GetStyle(lgsButton);
if Style = nil then
exit;
GC := Style^.dark_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.dark[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_GRAYTEXT :
begin
Style := GetStyle(lgsDefault);
if Style = nil then
exit;
GC := Style^.text_gc[GTK_STATE_INSENSITIVE];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.text[GTK_STATE_NORMAL];
end else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_MENUTEXT,
COLOR_BTNTEXT :
begin
case BaseColor of
COLOR_BTNTEXT : Style := GetStyle(lgsButton);
COLOR_MENUTEXT : Style := GetStyle(lgsMenuitem);
end;
if Style = nil then
exit;
GC := Style^.fg_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.fg[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_WINDOWTEXT:
begin
Style := GetStyle(lgsDefault);
if Style = nil then
exit;
GC := Style^.text_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.text[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_3DLIGHT,
COLOR_BTNHIGHLIGHT :
begin
Style := GetStyle(lgsButton);
if Style = nil then
exit;
GC := Style^.light_gc[GTK_STATE_NORMAL];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.light[GTK_STATE_NORMAL];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_WINDOW :
begin
Style := GetStyle(lgsList);
if Style = nil then
exit;
GC := Style^.base_gc[GTK_STATE_NORMAL];
if (GC = nil) then
begin
Result.Fill := GDK_Solid;
if Style^.base[GTK_STATE_NORMAL].Pixel<>0 then
begin
Result.foreground := Style^.base[GTK_STATE_NORMAL];
Result.background := Style^.base[GTK_STATE_NORMAL];
end;
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_HIGHLIGHT :
begin
Style := GetStyle(lgsDefault);
if Style = nil then
exit;
GC := Style^.bg_gc[GTK_STATE_SELECTED];
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.bg[GTK_STATE_SELECTED];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
COLOR_HIGHLIGHTTEXT :
begin
Style := GetStyle(lgsDefault);
if Style = nil then
exit;
{$IFDEF Gtk1}
GC := Style^.bg_gc[GTK_STATE_PRELIGHT];
{$ELSE}
GC := Style^.text_gc[GTK_STATE_SELECTED];
{$ENDIF}
if GC = nil then
begin
Result.Fill := GDK_Solid;
Result.foreground := Style^.bg[GTK_STATE_PRELIGHT];
end
else
GDK_GC_Get_Values(GC, @Result);
end;
{?????????????
COLOR_HOTLIGHT :
begin
end;
?????????????}
{?????????????????
COLOR_APPWORKSPACE :
begin
end;
?????????????????}
end;
RealizeGtkStyleColor(Style, @Result.foreground);
end;
function StyleForegroundColor(Color: TColorRef;
DefaultColor: PGDKColor): PGDKColor;
var
style : PGTKStyle;
begin
style := nil;
Result := DefaultColor;
Case TColor(Color) of
clINFOTEXT :
begin
Style := GetStyle(lgsTooltip);
If Style = nil then
exit;
Result := @Style^.fg[GTK_STATE_NORMAL];
end;
cl3DDKSHADOW,
clBTNSHADOW :
begin
Style := GetStyle(lgsButton);
If Style = nil then
exit;
Result := @Style^.dark[GTK_STATE_NORMAL];
end;
clGRAYTEXT :
begin
Style := GetStyle(lgsDefault);
If Style = nil then
exit;
Result := @Style^.text[GTK_STATE_INSENSITIVE];
end;
clMENUTEXT,
clBTNTEXT :
begin
Case TColor(Color) of
clBTNTEXT : Style := GetStyle(lgsButton);
clMENUTEXT : Style := GetStyle(lgsMenuitem);
end;
If Style = nil then
exit;
Result := @Style^.fg[GTK_STATE_NORMAL];
end;
clWINDOWTEXT:
begin
Style := GetStyle(lgsDefault);
If Style = nil then
exit;
Result := @Style^.text[GTK_STATE_NORMAL];
end;
cl3DLIGHT,
clBTNHIGHLIGHT :
begin
Style := GetStyle(lgsButton);
If Style = nil then
exit;
Result := @Style^.light[GTK_STATE_NORMAL];
end;
clHIGHLIGHTTEXT :
begin
DebugLn(['StyleForegroundColor clHIGHLIGHTTEXT']);
Style := GetStyle(lgsDefault);
If Style = nil then
exit;
Result := @Style^.text[GTK_STATE_PRELIGHT];
DebugLn(['StyleForegroundColor clHIGHLIGHTTEXT 2 ',Result<>nil]);
end;
end;
If Result = nil then
Result := DefaultColor;
if (Result <> nil) and (Result <> DefaultColor) then
RealizeGtkStyleColor(Style,Result);
end;
function GetStyleGroupboxFrameBorders: TRect;
const s = 200;
var
StyleObject: PStyleObject;
allocation: TGtkAllocation;
FrameWidget: PGtkFrame;
f: TRect;
begin
GetStyleWidget(lgsGroupBox);
StyleObject:=StandardStyles[lgsGroupBox];
if not StyleObject^.FrameBordersValid then begin
allocation.x:=0;
allocation.y:=0;
allocation.width:=s;
allocation.height:=s;
gtk_widget_size_allocate(StyleObject^.Widget,@allocation);
FrameWidget:=pGtkFrame(StyleObject^.Widget);
{$IFDEF Gtk1}
allocation:=FrameWidget^.bin.child^.allocation;
{$ELSE}
GTK_FRAME_GET_CLASS(FrameWidget)^.compute_child_allocation(
FrameWidget,@allocation);
{$ENDIF}
//DebugLn(['GetStyleGroupboxFrame BBB2 ',dbgs(allocation)]);
f.Left:=Min(s,Max(0,allocation.x));
f.Top:=Min(s,Max(0,allocation.y));
f.Right:=Max(0,Min(s-f.Left,s-allocation.x-allocation.width));
f.Bottom:=Max(0,Min(s-f.Top,s-allocation.x-allocation.width));
StyleObject^.FrameBorders:=f;
//DebugLn(['GetStyleGroupboxFrame FrameBorders=',dbgs(StyleObject^.FrameBorders)]);
StyleObject^.FrameBordersValid:=true;
end;
Result:=StyleObject^.FrameBorders;
end;
function GetStyleNotebookFrameBorders: TRect;
const s = 400;
var
StyleObject: PStyleObject;
allocation: TGtkAllocation;
f: TRect;
PageWidget: PGtkWidget;
begin
GetStyleWidget(lgsNotebook);
StyleObject:=StandardStyles[lgsNotebook];
if not StyleObject^.FrameBordersValid then begin
allocation.x:=0;
allocation.y:=0;
allocation.width:=s;
allocation.height:=s;
gtk_widget_size_allocate(StyleObject^.Widget,@allocation);
PageWidget:=gtk_notebook_get_nth_page(PGtkNoteBook(StyleObject^.Widget),0);
//DebugLn(['GetStyleNotebookFrameBorders BBB2 ',dbgs(allocation)]);
allocation:=PageWidget^.allocation;
f.Left:=Min(s,Max(0,allocation.x));
f.Top:=Min(s,Max(0,allocation.y));
f.Right:=Max(0,Min(s-f.Left,s-allocation.x-allocation.width));
f.Bottom:=Max(0,Min(s-f.Top,s-allocation.x-allocation.width));
StyleObject^.FrameBorders:=f;
//DebugLn(['GetStyleNotebookFrameBorders FrameBorders=',dbgs(StyleObject^.FrameBorders)]);
StyleObject^.FrameBordersValid:=true;
end;
Result:=StyleObject^.FrameBorders;
end;
{$IFDEF Gtk2}
function GetStyleFormFrameBorders(WithMenu: boolean): TRect;
const s = 400;
var
StyleObject: PStyleObject;
allocation: TGtkAllocation;
f: TRect;
InnerWidget: PGtkWidget;
Outer: TGdkRectangle;
Inner: TGdkRectangle;
begin
GetStyleWidget(lgsMenu);
StyleObject:=StandardStyles[lgsWindow];
if not StyleObject^.FrameBordersValid then begin
allocation.x:=0;
allocation.y:=0;
allocation.width:=s;
allocation.height:=s;
gtk_widget_size_allocate(StyleObject^.Widget,@allocation);
InnerWidget:=PGTKWidget(
gtk_object_get_data(PGtkObject(StyleObject^.Widget),'fixedwidget'));
allocation:=InnerWidget^.allocation;
//DebugLn(['GetStyleFormFrameBorders BBB2 ',dbgs(allocation),' WithMenu=',WithMenu,' ClientWidget=',GetWidgetDebugReport(InnerWidget)]);
f.Left:=Min(s,Max(0,allocation.x));
f.Top:=Min(s,Max(0,allocation.y));
f.Right:=Max(0,Min(s-f.Left,s-allocation.x-allocation.width));
f.Bottom:=Max(0,Min(s-f.Top,s-allocation.x-allocation.width));
StyleObject^.FrameBorders:=f;
//DebugLn(['GetStyleFormFrameBorders FrameBorders=',dbgs(StyleObject^.FrameBorders)]);
StyleObject^.FrameBordersValid:=true;
end;
if WithMenu then begin
InnerWidget:=PGTKWidget(
gtk_object_get_data(PGtkObject(StyleObject^.Widget),'vbox'));
end else begin
InnerWidget:=PGTKWidget(
gtk_object_get_data(PGtkObject(StyleObject^.Widget),'fixedwidget'));
end;
Outer:=StyleObject^.Widget^.allocation;
Inner:=InnerWidget^.allocation;
Result.Left:=Min(Outer.width,Max(0,Inner.x));
Result.Top:=Min(Outer.height,Max(0,Inner.y));
Result.Right:=Max(0,Min(Outer.width-f.Left,Outer.width-Inner.x-Inner.width));
Result.Bottom:=Max(0,Min(Outer.height-f.Top,Outer.height-Inner.x-Inner.width));
//DebugLn(['GetStyleFormFrameBorders BBB3 Inner=',dbgs(Inner),' Outer=',dbgs(Outer),' WithMenu=',WithMenu,' InnerWidget=',GetWidgetDebugReport(InnerWidget),' Result=',dbgs(Result)]);
end;
{$ENDIF}
procedure StyleFillRectangle(drawable : PGDKDrawable; GC : PGDKGC;
Color : TColorRef; x, y, width, height : gint);
var
style: PGTKStyle;
widget: PGTKWidget;
state: TGTKStateType;
shadow: TGtkShadowType;
detail: pgchar;
begin
style := nil;
shadow := GTK_SHADOW_NONE;
state := GTK_STATE_NORMAL;
case TColor(Color) of
{ clMenu:
begin
Style := GetStyle('menuitem');
widget := GetStyleWidget('menuitem');
detail := 'menuitem';
end;
clBtnFace :
begin
Style := GetStyle('button');
widget := GetStyleWidget('button');
detail := 'button';
end;
clWindow :
begin
Style := GetStyle('default');
widget := GetStyleWidget('default');
detail := 'list';
end; }
clBackground:
begin
Style := GetStyle(lgsWindow);
widget := GetStyleWidget(lgsWindow);
detail := 'window';
end;
clInfoBk :
begin
Style := GetStyle(lgsToolTip);
Widget := GetStyleWidget(lgsToolTip);
shadow := GTK_SHADOW_OUT;
detail := 'tooltip';
end;
clForm :
begin
Style := GetStyle(lgsWindow);
widget := GetStyleWidget(lgsWindow);
detail := 'window';
end;
end;
if Assigned(Style) then
gtk_paint_flat_box(style, drawable, state, shadow, nil, widget,
detail, x, y, width, height)
else
gdk_draw_rectangle(drawable, GC, 1, x, y, width, height);
end;
procedure UpdateWidgetStyleOfControl(AWinControl: TWinControl);
var
RCStyle : PGtkRCStyle;
Widget, FixWidget : PGTKWidget;
MainWidget: PGtkWidget;
FreeFontName: boolean;
FreeFontSetName: boolean;
procedure CreateRCStyle;
begin
if RCStyle=nil then
RCStyle:=gtk_rc_style_new;
end;
procedure SetRCFont(FontGdiObject: PGdiObject);
{$IFDEF GTK1}
var
FontDesc: TGtkFontCacheDescriptor;
{$ENDIF}
begin
{$IFDEF GTK1}
CreateRCStyle;
FontDesc:=FontCache.FindADescriptor(FontGdiObject^.GDIFontObject);
if (FontDesc<>nil) and (FontDesc.xlfd<>'') then begin
RCStyle:=gtk_rc_style_new;
g_free(RCStyle^.font_name);
RCStyle^.font_name:=g_strdup(PChar(FontDesc.xlfd));
g_free(RCStyle^.fontset_name);
RCStyle^.fontset_name:=g_strdup(PChar(FontDesc.xlfd));
FreeFontName:=true;
//DebugLn('UpdateWidgetStyleOfControl.SetRCFont ',DbgSName(AWinControl),' Widget=',GetWidgetDebugReport(Widget),' Style=',GetWidgetStyleReport(Widget));
end;
{$ENDIF}
end;
begin
{$IFDEF NoStyle}
exit;
{$ENDIF}
if not AWinControl.HandleAllocated then exit;
MainWidget:=PGtkWidget(AWinControl.Handle);
FixWidget:=GetFixedWidget(MainWidget);
if (FixWidget <> nil) and (FixWidget <> MainWidget) then
Widget := FixWidget
else
Widget := MainWidget;
RCStyle:=nil;
FreeFontName:=false;
FreeFontSetName:=false;
try
// set default background
if (AWinControl.Color=clNone) then
begin
// clNone => remove default background
if (FixWidget<>nil) and (FixWidget^.Window<>nil) then
begin
gdk_window_set_back_pixmap(FixWidget^.Window, nil, GdkFalse);
end;
end
else
if not IsColorDefault(AWinControl) and ((AWinControl.Color and SYS_COLOR_BASE)=0) then
begin
// set background to user defined color
// don't set background for custom controls, which paint themselves
// (this prevents flickering)
if (csOpaque in AWinControl.ControlStyle)
and GtkWidgetIsA(MainWidget,GTKAPIWidget_Type) then exit;
{for i:=0 to 4 do begin
RCStyle^.bg[i]:=NewColor;
// Indicate which colors the GtkRcStyle will affect;
// unflagged colors will follow the theme
RCStyle^.color_flags[i]:=
RCStyle^.color_flags[i] or GTK_RC_BG;
end;}
//DebugLn('UpdateWidgetStyleOfControl ',DbgSName(AWinControl),' Color=',DbgS(AWinControl.Color));
end;
{if (AWinControl is TCustomForm) then begin
gdk_window_set_back_pixmap(FixWidget^.Window,nil,GdkFalse);
NewColor:=TColorToTGDKColor(clRed);
CreateRCStyle;
for i:=0 to 4 do begin
debugln('UpdateWidgetStyleOfControl i=',dbgs(i),' ',RCStyle^.bg_pixmap_name[i],' ',RCStyle^.Name);
RCStyle^.bg[i]:=NewColor;
// Indicate which colors the GtkRcStyle will affect;
// unflagged colors will follow the theme
RCStyle^.color_flags[i]:=
RCStyle^.color_flags[i] or GTK_RC_BG;
end;
end;}
// set font color
// set font (currently only TCustomLabel)
if (GtkWidgetIsA(Widget,gtk_label_get_type)
or GtkWidgetIsA(Widget,gtk_editable_get_type)
or GtkWidgetIsA(Widget,gtk_check_button_get_type))
and (not AWinControl.Font.IsDefault)
then begin
// allocate font (just read it)
if AWinControl.Font.Reference.Handle=0 then ;
end;
finally
if RCStyle<>nil then begin
//DebugLn('UpdateWidgetStyleOfControl Apply Modifications ',AWinControl.Name,' ',GetWidgetClassName(Widget));
gtk_widget_modify_style(Widget,RCStyle);
if FreeFontName then begin
{$ifdef gtk1}
g_free(RCStyle^.font_name);
RCStyle^.font_name:=nil;
{$else}
pango_font_description_free(RCStyle^.font_desc);
RCStyle^.font_desc:=nil;
{$endif}
end;
if FreeFontSetName then begin
{$ifdef gtk1}
g_free(RCStyle^.fontset_name);
RCStyle^.fontset_name:=nil;
{$endif}
end;
//DebugLn('UpdateWidgetStyleOfControl END ',DbgSName(AWinControl),' Widget=',GetWidgetDebugReport(Widget),' Style=',GetWidgetStyleReport(Widget));
gtk_rc_style_unref(RCStyle);
end;
end;
end;
{-------------------------------------------------------------------------------
Creates a new PChar. Deletes escaping ampersands, replaces the first single
ampersand with an underscore and deletes all other single ampersands.
-------------------------------------------------------------------------------}
function Ampersands2Underscore(Src: PChar) : PChar;
var
s: String;
begin
s := StrPas(Src);
s := Ampersands2Underscore(s);
Result := StrAlloc(Length(s)+1); // +1 for #0 char at end
strcopy(Result, PChar(s));
end;
{-------------------------------------------------------------------------------
Deletes escaping ampersands, replaces the first single
ampersand with an underscore and deletes all other single ampersands.
-------------------------------------------------------------------------------}
function Ampersands2Underscore(const ASource: String): String;
var
n: Integer;
FirstFound: Boolean;
begin
//TODO: escape underscores
FirstFound := False;
Result := ASource;
n := 1;
while n <= Length(Result) do
begin
if Result[n] = '&' then
begin
if FirstFound
or ( (n < Length(Result)) and (Result[n+1] = '&') ) // got &&
then begin
Delete(Result, n, 1);
if not FirstFound then
Inc(n); // Skip the second & of &&
end
else begin
FirstFound := True;
Result[n] := '_';
end;
end;
Inc(n);
end;
end;
{-------------------------------------------------------------------------------
function RemoveAmpersands(Src: PChar; LineLength : Longint) : PChar;
Creates a new PChar removing all escaping ampersands.
-------------------------------------------------------------------------------}
function RemoveAmpersands(Src: PChar; LineLength : Longint) : PChar;
var
i, j: Longint;
ShortenChars, NewLength, SrcLength: integer;
begin
// count ampersands and find first ampersand
ShortenChars:= 0; // chars to delete
SrcLength:= LineLength;
{ Look for amperands. If found, check if it is an escaped ampersand.
If it is, don't count it in. }
i:=0;
while i<SrcLength do begin
if Src[i] = '&' then begin
if (i < SrcLength - 1) and (Src[i+1] = '&') then begin
// escaping ampersand found
inc(ShortenChars);
inc(i,2);
Continue;
end
else
inc(ShortenChars);
end;
inc(i);
end;
// create new PChar
NewLength:= SrcLength - ShortenChars;
Result:=StrAlloc(NewLength+1); // +1 for #0 char at end
// copy string without ampersands
i:=0;
j:=0;
while (j < NewLength) do begin
if Src[i] <> '&' then begin
// copy normal char
Result[j]:= Src[i];
end else begin
// ampersand
if (i < (SrcLength - 1)) and (Src[i+1] = '&') then begin
// escaping ampersand found
inc(i);
Result[j]:='&';
end else
// delete single ampersand
dec(j);
end;
Inc(i);
Inc(j);
end;
Result[NewLength]:=#0;
end;
{-------------------------------------------------------------------------------
function RemoveAmpersands(const ASource: String): String;
Removing all escaping ampersands.
-------------------------------------------------------------------------------}
function RemoveAmpersands(const ASource: String): String;
var
n: Integer;
begin
Result := ASource;
n := 1;
while n <= Length(Result) do
begin
if Result[n] = '&'
then begin
if (n < Length(Result))
and (Result[n + 1] = '&')
then begin
// we got a &&, remove the first
Delete(Result, n, 1);
Inc(n);
Continue;
end;
// simply remove it
Delete(Result, n, 1);
Continue;
end;
Inc(n);
end;
end;
{-------------------------------------------------------------------------------
procedure LabelFromAmpersands(var AText, APattern: String; var AAccelChar: Char)
Removes all escaping ampersands &&, creates an underscore pattern and returns
the first ampersand char as accelerator char
-------------------------------------------------------------------------------}
procedure LabelFromAmpersands(var AText, APattern: String; var AAccelChar: Char);
var
n: Integer;
FirstFound: Boolean;
begin
FirstFound := False;
APattern := StringOfChar(' ', Length(AText));
AAccelChar := #0;
n := 1;
while n <= Length(AText) do
begin
case AText[n] of
'&': begin
if (n < Length(AText))
and (AText[n + 1] = '&')
then begin
// we got a &&, remove the first
Delete(AText, n, 1);
Delete(APattern, n, 1);
Inc(n);
end else begin
Delete(AText, n, 1);
Delete(APattern, n, 1);
if FirstFound
then Continue; // simply remove it
// if we are here it's our first
FirstFound := True;
AAccelChar := System.lowerCase(AText[n]);
// is there a next char we can underline ?
if n <= Length(APattern)
then APattern[n] := '_';
end;
end;
'_': begin
AText[n] := ' ';
APattern[n] := '_';
end;
end;
Inc(n);
end;
end;
{-------------------------------------------------------------------------------
function GetTextExtentIgnoringAmpersands(TheFont: PGDKFont;
Str : PChar; StrLength: integer;
MaxWidth: Longint; lbearing, rbearing, width, ascent, descent : Pgint);
Gets text extent of a string, ignoring escaped Ampersands.
That means, ampersands are not counted.
Negative MaxWidth means no limit.
-------------------------------------------------------------------------------}
procedure GetTextExtentIgnoringAmpersands(TheFont: TGtkIntfFont;
Str : PChar; StrLength: integer;
lbearing, rbearing, width, ascent, descent : Pgint);
var
NewStr : PChar;
i: integer;
begin
NewStr:=Str;
// first check if Str contains an ampersand:
if (Str<>nil) then begin
i:=0;
while (Str[i]<>'&') and (i<StrLength) do inc(i);
if i<StrLength then begin
NewStr := RemoveAmpersands(Str, StrLength);
StrLength:=StrLen(NewStr);
end;
end;
gdk_text_extents(TheFont, NewStr, StrLength,
lbearing, rBearing, width, ascent, descent);
if NewStr<>Str then
StrDispose(NewStr);
end;
{------------------------------------------------------------------------------
function FontIsDoubleByteCharsFont(TheFont: TGtkIntfFont): boolean;
This is only a heuristic
------------------------------------------------------------------------------}
function FontIsDoubleByteCharsFont(TheFont: TGtkIntfFont): boolean;
var
SingleCharLen, DoubleCharLen: integer;
begin
{$IFDEF Gtk1}
SingleCharLen:=gdk_text_width(TheFont, 'A', 1);
DoubleCharLen:=gdk_text_width(TheFont, #0'A', 2);
{$ELSE}
pango_layout_set_single_paragraph_mode(TheFont, TRUE);
pango_layout_set_width(TheFont, -1);
pango_layout_set_text(TheFont, 'A', 1);
pango_layout_get_pixel_size(TheFont, @SingleCharLen, nil);
pango_layout_set_text(TheFont, #0'A', 2);
pango_layout_get_pixel_size(TheFont, @DoubleCharLen, nil);
{$ENDIF}
Result:=(SingleCharLen=0) and (DoubleCharLen>0);
end;
{------------------------------------------------------------------------------
function FontIsMonoSpaceFont(TheFont: TGtkIntfFont): boolean;
This is only a heuristic
------------------------------------------------------------------------------}
function FontIsMonoSpaceFont(TheFont: TGtkIntfFont): boolean;
var
{$IFDEF Gtk1}
SingleCharLen: LongInt;
{$ENDIF}
MWidth: LongInt;
IWidth: LongInt;
begin
{$IFDEF Gtk1}
SingleCharLen:=gdk_text_width(TheFont, 'A', 1);
if SingleCharLen=0 then begin
// assume a double byte character font
MWidth:=gdk_text_width(TheFont, '#0m', 2);
IWidth:=gdk_text_width(TheFont, '#0i', 2);
end else begin
// assume a single byte character font
MWidth:=gdk_text_width(TheFont, 'm', 1);
IWidth:=gdk_text_width(TheFont, 'i', 1);
end;
{$ELSE}
pango_layout_set_single_paragraph_mode(TheFont, TRUE);
pango_layout_set_width(TheFont, -1);
pango_layout_set_text(TheFont, 'm', 1);
pango_layout_get_pixel_size(TheFont, @MWidth, nil);
pango_layout_set_text(TheFont, 'i', 1);
pango_layout_get_pixel_size(TheFont, @IWidth, nil);
{$ENDIF}
Result:=MWidth=IWidth;
end;
{------------------------------------------------------------------------------
Method: GDKPixel2GDIRGB
Params:
Pixel - a GDK Pixel, refers to Index in Colormap/Visual
Visual - a GDK Visual, if nil, the System Default is used
Colormap - a GDK Colormap, if nil, the System Default is used
Returns: TGDIRGB
A convenience function for use with GDK Image's. It takes a pixel value
retrieved from gdk_image_get_pixel, and uses the passed Visual and Colormap
to try and look up actual RGB values.
------------------------------------------------------------------------------}
function GDKPixel2GDIRGB(Pixel: Longint; Visual: PGDKVisual;
Colormap: PGDKColormap) : TGDIRGB;
var
Color: TGDKColor;
begin
FillChar(Result, SizeOf(TGDIRGB),0);
If (Visual = nil) or (Colormap = nil) then begin
Visual := GDK_Visual_Get_System;
Colormap := GDK_Colormap_Get_System;
end;
gdk_colormap_query_color(colormap, pixel, @color);
Result.Red := Color.Red shr 8;
Result.Green := Color.Green shr 8;
Result.Blue := Color.Blue shr 8;
end;
{------------------------------------------------------------------------------
function GetWindowDecorations(AForm : TCustomForm) : Longint;
------------------------------------------------------------------------------}
function GetWindowDecorations(AForm : TCustomForm) : Longint;
var
ABorderStyle: TFormBorderStyle;
begin
Result := 0;
if not (csDesigning in AForm.ComponentState) then
ABorderStyle:=AForm.BorderStyle
else
ABorderStyle:=bsSizeable;
{$IFDEF Gtk2}
case ABorderStyle of
bsNone: Result := 0;
bsSingle: Result := GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE or
GDK_DECOR_MAXIMIZE;
bsSizeable: Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE or GDK_DECOR_MAXIMIZE
or GDK_DECOR_RESIZEH;
bsDialog: Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE;
bsToolWindow: Result := GDK_DECOR_TITLE or GDK_DECOR_MENU;
bsSizeToolWin: Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_RESIZEH;
end;
if not (csDesigning in AForm.ComponentState) then
begin
if not (biMinimize in AForm.BorderIcons) then
Result := Result and not GDK_DECOR_MINIMIZE;
if not (biMaximize in AForm.BorderIcons) then
Result := Result and not GDK_DECOR_MAXIMIZE;
if not (biSystemMenu in AForm.BorderIcons) then
Result := Result and not GDK_DECOR_MENU;
end;
{$ELSE}
case ABorderStyle of
bsNone : Result := 0;
bsSingle : Result := GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE or
GDK_DECOR_MAXIMIZE;
bsSizeable : Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE or GDK_DECOR_MAXIMIZE
or GDK_DECOR_RESIZEH;
bsDialog : Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_MINIMIZE;
bsToolWindow : Result := GDK_DECOR_TITLE or GDK_DECOR_MENU;
bsSizeToolWin :Result := GDK_DECOR_BORDER or GDK_DECOR_TITLE or
GDK_DECOR_MENU or GDK_DECOR_RESIZEH;
end;
{$ENDIF}
//DebugLn('GetWindowDecorations ',DbgSName(AForm),' ',dbgs(ord(ABorderStyle)),' ',binStr(Result,8));
end;
{------------------------------------------------------------------------------
function GetWindowFunction(AForm : TCustomForm) : Longint;
------------------------------------------------------------------------------}
function GetWindowFunction(AForm : TCustomForm) : Longint;
var
ABorderStyle: TFormBorderStyle;
begin
Result:=0;
if not (csDesigning in AForm.ComponentState) then
ABorderStyle:=AForm.BorderStyle
else
ABorderStyle:=bsSizeable;
{$IFDEF Gtk2}
case ABorderStyle of
bsNone : Result := GDK_FUNC_RESIZE or GDK_FUNC_CLOSE {$ifndef windows}or GDK_FUNC_MOVE{$endif};
bsSingle : Result := GDK_FUNC_MOVE or GDK_FUNC_MINIMIZE or GDK_FUNC_CLOSE;
bsSizeable : Result := GDK_FUNC_RESIZE or GDK_FUNC_MOVE or
GDK_FUNC_MINIMIZE or GDK_FUNC_CLOSE or GDK_FUNC_MAXIMIZE;
bsDialog : Result := GDK_FUNC_CLOSE or GDK_FUNC_MINIMIZE
or GDK_FUNC_MOVE;
bsToolWindow : Result := GDK_FUNC_MOVE or GDK_FUNC_CLOSE;
bsSizeToolWin : Result := GDK_FUNC_RESIZE or GDK_FUNC_MOVE or GDK_FUNC_CLOSE;
end;
// X warns if marking a fixed size window resizeable:
if ((AForm.Constraints.MinWidth>0)
and (AForm.Constraints.MinWidth=AForm.Constraints.MaxWidth))
or ((AForm.Constraints.MinHeight>0)
and (AForm.Constraints.MinHeight=AForm.Constraints.MaxHeight)) then
Result:=Result-GDK_FUNC_RESIZE;
if (not (csDesigning in AForm.ComponentState)) then
begin
if not (biMinimize in AForm.BorderIcons) then
Result:=Result and not GDK_FUNC_MINIMIZE;
if not (biMaximize in AForm.BorderIcons) then
Result:=Result and not GDK_FUNC_MAXIMIZE;
end;
{$ELSE}
case ABorderStyle of
bsNone : Result := GDK_FUNC_RESIZE or GDK_FUNC_CLOSE;
bsSingle : Result := GDK_FUNC_MOVE or GDK_FUNC_MINIMIZE or GDK_FUNC_CLOSE;
bsSizeable : Result := GDK_FUNC_RESIZE or GDK_FUNC_MOVE or
GDK_FUNC_MINIMIZE or GDK_FUNC_CLOSE or GDK_FUNC_MAXIMIZE;
bsDialog : Result := GDK_FUNC_CLOSE or GDK_FUNC_MINIMIZE
or GDK_FUNC_MOVE;
bsToolWindow : Result := GDK_FUNC_MOVE or GDK_FUNC_CLOSE;
bsSizeToolWin : Result := GDK_FUNC_RESIZE or GDK_FUNC_MOVE or GDK_FUNC_CLOSE;
end;
// X warns if marking a fixed size window resizeable:
if ((AForm.Constraints.MinWidth>0)
and (AForm.Constraints.MinWidth=AForm.Constraints.MaxWidth))
or ((AForm.Constraints.MinHeight>0)
and (AForm.Constraints.MinHeight=AForm.Constraints.MaxHeight)) then
Result:=Result-GDK_FUNC_RESIZE;
{$ENDIF}
//DebugLn('GetWindowFunction ',DbgSName(AForm),' ',dbgs(ord(ABorderStyle)),' ',binStr(Result,8));
end;
procedure FillScreenFonts(ScreenFonts : TStrings);
var
{$ifdef gtk1}
theFonts : PPChar;
{$else}
Widget : PGTKWidget;
Context : PPangoContext;
families : PPPangoFontFamily;
{$endif}
Tmp: AnsiString;
I, N: Integer;
begin
ScreenFonts.Clear;
{$ifdef gtk1}
theFonts := XListFonts(gdk_display,PChar('-*-*-*-*-*-*-*-*-*-*-*-*-*-*'), 10000, @N);
debugln('FillScreenFonts N=',dbgs(N));
for I := 0 to N - 1 do
if theFonts[I] <> nil then begin
Tmp := ExtractFamilyFromXLFDName(theFonts[I]);
if Tmp <> '' then
if ScreenFonts.IndexOf(Tmp) < 0 then
ScreenFonts.Append(Tmp);
end;
XFreeFontNames(theFonts);
{$else}
Widget := GetStyleWidget(lgsDefault);
if Widget = nil then begin
exit;//raise an error here I guess
end;
Context := gtk_widget_get_pango_context(Widget);
if Context = nil then begin
exit;//raise an error here I guess
end;
families := nil;
pango_context_list_families(Context, @families, @n);
for I := 0 to N - 1 do
if families[I] <> nil then begin
Tmp := StrPas(pango_font_family_get_name(families[I]));
if Tmp <> '' then
if ScreenFonts.IndexOf(Tmp) < 0 then
ScreenFonts.Append(Tmp);
end;
if (families <> nil) then
g_free(families);
{$endif gtk2}
end;
function GetTextHeight(DCTextMetric: TDevContextTextMetric): integer;
// IMPORTANT: Before this call: UpdateDCTextMetric(TGtkDeviceContext(DC));
begin
{$IfDef Win32}
Result := DCTextMetric.TextMetric.tmHeight div 2;
{$Else}
Result := DCTextMetric.TextMetric.tmAscent;
{$EndIf}
end;
{$IFDEF GTK1}
{ Compile with UseXinerama defined to use the Xinerama extension to avoid dialog
boxes straddling two monitors. This is only required for GTK1, as it is built
into GTK2. The Xinerama library is not always available, so the libraries will
be dynamically loaded. (A single monitor is assumed if the load fails.) On
some systems only a static Xinerama library is available, so define
StaticXinerama also. MAC OSX is in this latter category, but it crashed the
X server when I tried it on a real two monitor display.
}
{$IFDEF UseXinerama}
{$IFDEF StaticXinerama}
{$LINKLIB Xinerama}
{$ENDIF}
var
FirstScreenCalled: Boolean = False;
FirstScreenResult: Boolean = False;
{ Copy record definition from Xinerama unit.
Can't use the unit itself, as it forces the executable to
refer to the libraray }
type
TXineramaScreenInfo = record
screen_number : cint;
x_org : cshort;
y_org : cshort;
width : cshort;
height : cshort;
end;
PXineramaScreenInfo = ^TXineramaScreenInfo;
function GetFirstScreen: Boolean;
var
nMonitors: cint;
XineramaScreenInfo: PXineramaScreenInfo;
opcode, firstevent, firsterror: cint;
XineramaLib: TLibHandle;
pXineramaIsActive: function (dpy: PDisplay):TBool;cdecl;
pXineramaQueryScreens: function (dpy: PDisplay;
number: Pcint): PXineramaScreenInfo;cdecl;
begin
if not FirstScreenCalled then begin
if XQueryExtension(gdk_display, 'XINERAMA', @opcode, @firstevent,
@firsterror)
then begin
XineramaLib := {$IFDEF StaticXinerama} 1 {Flag present} {$ELSE} LoadLibrary('libXinerama.so') {$ENDIF};
if XineramaLib <> 0 then begin
{$IFDEF StaticXinerama}
Pointer(pXineramaIsActive) := @XineramaIsActive;
Pointer(pXineramaQueryScreens) := @XineramaQueryScreens;
{$ELSE}
Pointer(pXineramaIsActive) :=
GetProcAddress(XineramaLib, 'XineramaIsActive');
Pointer(pXineramaQueryScreens) :=
GetProcAddress(XineramaLib, 'XineramaQueryScreens');
{$ENDIF}
if (pXineramaIsActive <> nil) and (pXineramaQueryScreens <> nil) and
pXineramaIsActive(gdk_display)
then begin
XineramaScreenInfo := pXineramaQueryScreens(gdk_display, @nMonitors);
if XineramaScreenInfo <> nil then begin
if (nMonitors > 0) and (nMonitors < 10) then begin
FirstScreen.x := XineramaScreenInfo^.width;
FirstScreen.y := XineramaScreenInfo^.height;
FirstScreenResult := True;
end;
XFree(XineramaScreenInfo);
end;
end;
// Do not FreeLibrary(XineramaLib) because it causes the X11 library to
// crash on exit
end;
end;
FirstScreenCalled := True;
end;
Result := FirstScreenResult;
end;
{$ENDIF UseXinerama}
{$ENDIF Gtk1}
{$IFDEF HasX}
function XGetWorkarea(var ax,ay,awidth,aheight:gint): gint;
var
XDisplay: PDisplay;
XScreen: PScreen;
XWindow: TWindow;
AtomType: x.TAtom;
Format: gint;
nitems: gulong;
bytes_after: gulong;
current_desktop: pguint;
res : Integer;
begin
Result := -1;
xdisplay := gdk_display;
xscreen := XDefaultScreenOfDisplay(xdisplay);
xwindow := XRootWindowOfScreen(xscreen);
res:=XGetWindowProperty (xdisplay, xwindow,
XInternAtom(xdisplay, '_NET_WORKAREA', false),
0, MaxInt, False, XA_CARDINAL, @atomtype, @format, @nitems,
@bytes_after, gpointer(@current_desktop));
if (atomtype = XA_CARDINAL) and (format = 32) and (nitems > 0) then begin
result:=res;
ax:=current_desktop[0];
ay:=current_desktop[1];
awidth:=current_desktop[2];
aheight:=current_desktop[3];
// if an invalid size was given, assume garbage from the window manager
if (awidth < 1) or (aheight < 1) then result := -1;
end;
if current_desktop <> nil then
XFree (current_desktop);
end;
{$ENDIF}
function FindFocusWidget(AWidget: PGtkWidget): PGtkWidget;
var
WinWidgetInfo: PWinWidgetInfo;
ImplWidget: PGtkWidget;
GList: PGlist;
LastFocusWidget: PGtkWidget;
begin
// Default to the widget, try to find other
Result := AWidget;
// Combo
if GtkWidgetIsA(AWidget, gtk_combo_get_type)
then begin
// handle is a gtk combo
{$IfDef VerboseFocus}
DebugLn(' D taking gtkcombo entry');
{$EndIf}
Result := PgtkWidget(PGtkCombo(AWidget)^.entry);
Exit;
end;
// check if widget has a WinWidgetInfo record
WinWidgetInfo := GetWidgetInfo(AWidget, false);
if WinWidgetInfo = nil then Exit;
ImplWidget:= WinWidgetInfo^.CoreWidget;
if ImplWidget = nil then Exit;
// set default to the implementation widget
Result := ImplWidget;
// handle has an ImplementationWidget
if GtkWidgetIsA(ImplWidget, gtk_list_get_type)
then begin
{$IfDef VerboseFocus}
DebugLn(' E using list');
{$EndIf}
// Try the last added selected
if not (selection_mode(PGtkList(ImplWidget)^) in [GTK_SELECTION_SINGLE, GTK_SELECTION_BROWSE])
and (PGtkList(ImplWidget)^.last_focus_child <> nil)
then begin
LastFocusWidget:=PGtkList(ImplWidget)^.last_focus_child;
if g_list_find(PGtkList(ImplWidget)^.selection,LastFocusWidget)<>nil
then begin
Result := PGtkList(ImplWidget)^.last_focus_child;
{$IfDef VerboseFocus}
DebugLn(' E.1 using last_focus_child');
{$EndIf}
Exit;
end;
end;
// If there is a selection, try the first
GList := PGtkList(ImplWidget)^.selection;
if (GList <> nil) and (GList^.data <> nil)
then begin
Result := GList^.data;
{$IfDef VerboseFocus}
DebugLn(' E.2 using 1st selection');
{$EndIf}
Exit;
end;
// If not in browse mode, set focus to the first child
// in browsemode, the focused item cannot be selected by mouse
// if selection_mode(PGtkList(ImplWidget)^) = GTK_SELECTION_BROWSE
// then begin
// {$IfDef VerboseFocus}
// DebugLn(' E.3 Browse mode -> using ImplWidget');
// {$EndIf}
// Exit;
// end;
GList := PGtkList(ImplWidget)^.children;
if GList = nil then Exit;
if GList^.Data = nil then Exit;
Result := GList^.Data;
{$IfDef VerboseFocus}
DebugLn(' E.4 using 1st child');
{$EndIf}
Exit;
end;
{$IfDef VerboseFocus}
DebugLn(' E taking ImplementationWidget');
{$EndIf}
end;
{$IFDEF ASSERT_IS_ON}
{$UNDEF ASSERT_IS_ON}
{$C-}
{$ENDIF}
// included by gtkproc.pp