lazarus/lcl/interfaces/qt6/qtpagecontrol.inc

506 lines
16 KiB
PHP

{%MainUnit qtwscomctrls.pp}
{
*****************************************************************************
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.
*****************************************************************************
}
const
QTabWidgetTabPositionMap: array[TTabPosition] of QTabWidgetTabPosition =
(
{ tpTop } QTabWidgetNorth,
{ tpBottom } QTabWidgetSouth,
{ tpLeft } QTabWidgetWest,
{ tpRight } QTabWidgetEast
);
{ TQtWSCustomPage }
{------------------------------------------------------------------------------
Method: TQtWSCustomPage.CreateHandle
Params: None
Returns: Nothing
Allocates memory and resources for the control and shows it
------------------------------------------------------------------------------}
class function TQtWSCustomPage.CreateHandle(const AWinControl: TWinControl;
const AParams: TCreateParams): TLCLHandle;
var
QtPage: TQtPage;
begin
{$ifdef VerboseQt}
WriteLn('Trace:> [TQtWSCustomPage.CreateHandle]');
{$endif}
QtPage := TQtPage.Create(AWinControl, AParams);
QtPage.AttachEvents;
// Returns the Handle
Result := TLCLHandle(QtPage);
{$ifdef VerboseQt}
WriteLn('Trace:< [TQtWSCustomPage.CreateHandle] Result: ', IntToStr(Result));
{$endif}
end;
class procedure TQtWSCustomPage.DestroyHandle(const AWinControl: TWinControl);
var
B: Boolean;
begin
B := (AWinControl.Parent is TPageControl) and
AWinControl.Parent.HandleAllocated;
if B then
TQtWidget(AWinControl.Parent.Handle).BeginUpdate;
TQtWidget(AWinControl.Handle).Release;
if B then
TQtWidget(AWinControl.Parent.Handle).EndUpdate;
end;
class procedure TQtWSCustomPage.UpdateTabFontColor(APage: TCustomPage; AFont: TFont);
var
AParent: TQtWidget;
ATabWidget: TQtTabWidget;
AColor: TQColor;
begin
if not Assigned(APage.Parent) or not APage.Parent.HandleAllocated then
exit;
AParent := TQtWidget(APage.Parent.Handle);
if not (AParent is TQtTabWidget) then
exit;
ATabWidget := TQtTabWidget(AParent);
AColor.Alpha := 0;
FillChar(AColor, SizeOf(AColor), #0);
if AFont.Color = clDefault then
AColor := ATabWidget.Palette.DefaultTextColor
else
ColorRefToTQColor(ColorToRGB(AFont.Color), AColor);
with ATabWidget.TabBar do
SetTabFontColor(APage.PageIndex, AColor);
end;
class procedure TQtWSCustomPage.SetFont(const AWinControl: TWinControl;
const AFont: TFont);
begin
if not WSCheckHandleAllocated(AWinControl, 'SetFont') then
Exit;
TQtWSWinControl.SetFont(AWinControl, AFont);
UpdateTabFontColor(TCustomPage(AWinControl), AFont);
end;
class procedure TQtWSCustomPage.UpdateProperties(const ACustomPage: TCustomPage);
var
ImageList: TCustomImageList;
ImageIndex: Integer;
Bmp: TBitmap;
Icon: QIconH;
B: Boolean;
Size: TSize;
Res: TScaledImageListResolution;
begin
ImageList := TCustomTabControl(ACustomPage.Parent).Images;
B := False;
if Assigned(ImageList) then
begin
Res := ImageList.ResolutionForPPI[
TCustomTabControl(ACustomPage.Parent).ImagesWidth,
TCustomTabControl(ACustomPage.Parent).Font.PixelsPerInch,
TCustomTabControl(ACustomPage.Parent).GetCanvasScaleFactor];
ImageIndex := TCustomTabControl(ACustomPage.Parent).GetImageIndex(ACustomPage.PageIndex);
if (ImageIndex >= 0) and (ImageIndex < Res.Count) then
begin
Bmp := TBitmap.Create;
try
Res.GetBitmap(ACustomPage.ImageIndex, Bmp);
if (TQtPage(ACustomPage.Handle).ChildOfComplexWidget = ccwTabWidget) and
(TQtPage(ACustomPage.Handle).getTabWidget <> nil) then
begin
Size.cx := Res.Width;
Size.cy := Res.Height;
QTabWidget_setIconSize(TQtPage(ACustomPage.Handle).getTabWidget, @Size);
end;
Icon := TQtImage(Bmp.Handle).AsIcon;
TQtPage(ACustomPage.Handle).setIcon(Icon);
QIcon_destroy(Icon);
B := True;
finally
Bmp.Free;
end;
end;
end;
// no ImageList or invalid index.
if not B then
begin
Icon := TQtPage(ACustomPage.Handle).getIcon;
if (Icon <> nil) and not QIcon_isNull(Icon) then
begin
Icon := QIcon_create;
TQtPage(ACustomPage.Handle).setIcon(Icon);
QIcon_destroy(Icon);
end;
end;
UpdateTabFontColor(ACustomPage, ACustomPage.Font);
end;
{ TQtWSCustomNotebook }
{------------------------------------------------------------------------------
Method: TQtWSCustomTabControl.CreateHandle
Params: None
Returns: Nothing
Allocates memory and resources for the control and shows it
------------------------------------------------------------------------------}
class function TQtWSCustomTabControl.CreateHandle(const AWinControl: TWinControl; const AParams: TCreateParams): TLCLHandle;
var
QtTabWidget: TQtTabWidget;
QtTTabCtl: TQtWidget;
begin
{$ifdef VerboseQt}
WriteLn('TQtWSCustomTabControl.CreateHandle');
{$endif}
if AWinControl is TTabControl then
begin
QtTTabCtl := TQtWidget.Create(AWinControl, AParams);
QtTTabCtl.ChildOfComplexWidget := ccwTTabControl;
QtTTabCtl.AttachEvents;
Result := TLCLHandle(QtTTabCtl);
end else
begin
QtTabWidget := TQtTabWidget.Create(AWinControl, AParams);
QtTabWidget.setTabPosition(QTabWidgetTabPositionMap[TCustomTabControl(AWinControl).TabPosition]);
QtTabWidget.setTabsClosable(nboShowCloseButtons in TCustomTabControl(AWinControl).Options);
QtTabWidget.setTabSize(TCustomTabControl(AWinControl).TabHeight, TCustomTabControl(AWinControl).TabWidth);
{$IFDEF DARWIN}
QTabWidget_setElideMode(QTabWidgetH(QtTabWidget.Widget), QtElideNone);
{$ENDIF}
QtTabWidget.AttachEvents;
Result := TLCLHandle(QtTabWidget);
end;
end;
class function TQtWSCustomTabControl.GetDefaultClientRect(
const AWinControl: TWinControl; const aLeft, aTop, aWidth, aHeight: integer;
var aClientRect: TRect): boolean;
procedure setTabSizeInternal(AWidget: QWidgetH; const AWidth, AHeight: integer);
var
ANewSize: TSize;
WS: WideString;
begin
ANewSize := Default(TSize);
if AWidth > 0 then
ANewSize.cx := AWidth;
if AHeight > 0 then
ANewSize.cy := AHeight;
if (AWidth <= 0) and (AHeight <= 0) then
WS := ''
else
WS := {%H-}Format('QTabBar::tab { height: %dpx; width: %dpx; }',[ANewSize.cy, ANewSize.cx]);
QWidget_setStyleSheet(AWidget, @WS);
end;
function MeasureClientRectInternal: TRect;
var
WStr: WideString;
ANewTabW: QTabWidgetH;
ASize: TSize;
begin
Result := Rect(0, 0, 0, 0);
WStr := 'TabSheet1'; // do not translate.
ANewTabW := QTabWidget_Create(nil);
QWidget_setGeometry(ANewTabW, 0, 0, AWinControl.Width, AWinControl.Height);
QTabWidget_setTabPosition(ANewTabW, QTabWidgetTabPositionMap[TCustomTabControl(AWinControl).TabPosition]);
QTabWidget_setTabsClosable(ANewTabW, nboShowCloseButtons in TCustomTabControl(AWinControl).Options);
setTabSizeInternal(QWidgetH(ANewTabW), TCustomTabControl(AWinControl).TabHeight, TCustomTabControl(AWinControl).TabWidth);
QWidget_setVisible(QTabWidget_tabBar(ANewTabW), TCustomTabControl(AWinControl).ShowTabs);
QTabWidget_addTab(ANewTabW, QWidget_Create(nil), @WStr);
QTabWidget_setCurrentIndex(ANewTabW, 0);
QWidget_show(ANewTabW);
QWidget_contentsRect(QTabWidget_widget(ANewTabW, 0), @Result);
QTabWidget_Destroy(ANewTabW);
end;
var
dx, dy: integer;
ATabWidget: TQtTabWidget;
begin
Result := False;
if AWinControl.HandleAllocated then
begin
if TQtWidget(AWinControl.Handle).ChildOfComplexWidget <> ccwTTabControl then
begin
ATabWidget := TQtTabWidget(AWinControl.Handle);
if ATabWidget.testAttribute(QtWA_PendingResizeEvent) or not ATabWidget.testAttribute(QtWA_Mapped) then
begin
aClientRect := MeasureClientRectInternal;
Result := True;
end;
end;
end else
begin
if AWinControl is TTabControl then
begin
dx := GetPixelMetric(QStylePM_TabBarBaseHeight, nil, nil);
aClientRect := Rect(0,0, Max(0, aWidth - (dx * 2)), Max(0, aHeight - (dx * 2)));
end else
aClientRect := MeasureClientRectInternal;
Result := True;
end;
end;
class procedure TQtWSCustomTabControl.AddPage(const ATabControl: TCustomTabControl;
const AChild: TCustomPage; const AIndex: integer);
var
QtTabWidget: TQtTabWidget;
begin
{$ifdef VerboseQt}
WriteLn('TQtWSCustomTabControl.AddPage');
{$endif}
QtTabWidget := TQtTabWidget(ATabControl.Handle);
QtTabWidget.setUpdatesEnabled(False);
QtTabWidget.BeginUpdate;
try
QtTabWidget.insertTab(AIndex, TQtPage(AChild.Handle).Widget, AChild.Caption{%H-});
finally
QtTabWidget.EndUpdate;
QtTabWidget.setUpdatesEnabled(True);
end;
TQtPage(AChild.Handle).ChildOfComplexWidget := ccwTabWidget;
TQtWsCustomPage.UpdateProperties(AChild);
end;
class procedure TQtWSCustomTabControl.MovePage(const ATabControl: TCustomTabControl;
const AChild: TCustomPage; const NewIndex: integer);
var
TabWidget: TQtTabWidget;
Index: Integer;
begin
AChild.HandleNeeded; {create handle if it does not exist yet}
TabWidget := TQtTabWidget(ATabControl.Handle);
Index := AChild.PageIndex;
if Index < 0 then
Index := ATabControl.IndexOf(AChild);
TabWidget.BeginUpdate;
TabWidget.setUpdatesEnabled(false);
QTabBar_moveTab(QTabBarH(TabWidget.TabBar.Widget), Index, NewIndex);
// DebugLn('TQtWSCustomTabControl.MovePage from Index=',dbgs(Index),' to ',dbgs(NewIndex),' finished.');
TabWidget.setUpdatesEnabled(true);
TabWidget.EndUpdate;
end;
class procedure TQtWSCustomTabControl.RemovePage(const ATabControl: TCustomTabControl;
const AIndex: integer);
var
TabWidget: TQtTabWidget;
begin
{$ifdef VerboseQt}
WriteLn('TQtWSCustomTabControl.RemovePage');
{$endif}
TabWidget := TQtTabWidget(ATabControl.Handle);
TabWidget.setUpdatesEnabled(false);
TabWidget.BeginUpdate;
try
TabWidget.removeTab(AIndex);
finally
TabWidget.EndUpdate;
TabWidget.setUpdatesEnabled(true);
end;
end;
class function TQtWSCustomTabControl.GetNotebookMinTabHeight(
const AWinControl: TWinControl): integer;
var
dy: integer;
ATabWidget: TQtTabWidget;
begin
if AWinControl.HandleAllocated and
(TQtWidget(AWinControl.Handle).ChildOfComplexWidget <> ccwTTabControl)
then begin
ATabWidget := TQtTabWidget(AWinControl.Handle);
Result := 0;
dy := 0;
ATabWidget.TabBar.preferredSize(dy, Result, False);
if Result > 0 then
exit;
end;
Result:=inherited GetNotebookMinTabHeight(AWinControl);
end;
class function TQtWSCustomTabControl.GetNotebookMinTabWidth(
const AWinControl: TWinControl): integer;
var
dy: integer;
ATabWidget: TQtTabWidget;
begin
if AWinControl.HandleAllocated and
(TQtWidget(AWinControl.Handle).ChildOfComplexWidget <> ccwTTabControl)
then begin
ATabWidget := TQtTabWidget(AWinControl.Handle);
Result := 0;
dy := 0;
ATabWidget.TabBar.preferredSize(Result, dy, False);
if Result > 0 then
exit;
end;
Result:=inherited GetNotebookMinTabWidth(AWinControl);
end;
class function TQtWSCustomTabControl.GetCapabilities: TCTabControlCapabilities;
begin
Result := [nbcShowCloseButtons, nbcTabsSizeable];
end;
class function TQtWSCustomTabControl.GetDesignInteractive(
const AWinControl: TWinControl; AClientPos: TPoint): Boolean;
var
TabWidget: TQtTabWidget;
TabBar: TQtTabBar;
TabIndex: Integer;
p: TQtPoint;
begin
Result := False;
if not WSCheckHandleAllocated(AWinControl, 'GetDesignInteractive') then
Exit;
if TQtWidget(AWinControl.Handle).ChildOfComplexWidget = ccwTTabControl then
exit;
TabWidget := TQtTabWidget(AWinControl.Handle);
TabBar := TabWidget.TabBar;
p := QtPoint(AClientPos.x, AClientPos.y);
TabIndex := QTabBar_tabAt(QTabBarH(TabBar.Widget), @p);
Result := (TabIndex >= 0) and (TabWidget.getCurrentIndex <> TabIndex);
end;
class function TQtWSCustomTabControl.GetTabIndexAtPos(
const ATabControl: TCustomTabControl; const AClientPos: TPoint): integer;
var
TabWidget: TQtTabWidget;
NewPos: TPoint;
R: TRect;
TabOffset: TPoint;
begin
TabWidget := TQtTabWidget(ATabControl.Handle);
NewPos := AClientPos;
TabOffset := TabWidget.TabBar.TabBarOffset;
Dec(NewPos.X, TabOffset.X);
Dec(NewPos.Y, TabOffset.Y);
R := TabWidget.TabBar.getGeometry;
case ATabControl.TabPosition of
tpTop: if NewPos.Y < 0 then NewPos.Y := R.Bottom + NewPos.Y;
tpLeft: if NewPos.X < 0 then NewPos.X := R.Left + NewPos.X;
tpRight: NewPos.X := R.Right - NewPos.X;
tpBottom: NewPos.Y := R.Bottom - NewPos.Y;
end;
// issue #28591, return -1 if we are left of first tab with mouse
if (ATabControl.BiDiMode = bdRightToLeft) and (QGUIApplication_mouseButtons = QtLeftButton) then
begin
if not QWidget_underMouse(TabWidget.TabBar.Widget) then
exit(-1);
end;
Result := TabWidget.tabAt(NewPos);
end;
class function TQtWSCustomTabControl.GetTabRect(const ATabControl: TCustomTabControl;
const AIndex: Integer): TRect;
var
TabWidget: TQtTabWidget;
Offs: TRect;
begin
Result := Rect(-1, -1, -1, -1);
if not WSCheckHandleAllocated(ATabControl, 'GetTabRect') then
Exit;
TabWidget := TQtTabWidget(ATabControl.Handle);
Offs := TabWidget.getGeometry;
Result := TabWidget.TabBar.GetTabRect(AIndex);
case ATabControl.TabPosition of
tpTop: Types.OffsetRect(Result, 0, -Result.Bottom);
tpLeft: Types.OffsetRect(Result, -Result.Right, 0);
tpRight: Types.OffsetRect(Result, Offs.Width - Result.Right, 0);
tpBottom: Types.OffsetRect(Result, 0, Offs.Height - Result.Bottom);
end;
end;
class procedure TQtWSCustomTabControl.SetPageIndex(
const ATabControl: TCustomTabControl; const AIndex: integer);
var
TabWidget: TQtTabWidget;
begin
if ATabControl is TTabControl then
exit;
if not WSCheckHandleAllocated(ATabControl, 'SetPageIndex') then
Exit;
TabWidget := TQtTabWidget(ATabControl.Handle);
if (AIndex < 0) or (AIndex > ATabControl.PageCount - 1) then
exit;
TabWidget.BeginUpdate;
if ATabControl.Page[AIndex].HandleAllocated then
TabWidget.setCurrentWidget(TQtWidget(ATabControl.Page[AIndex].Handle), False);
TabWidget.EndUpdate;
end;
class procedure TQtWSCustomTabControl.SetTabCaption(
const ATabControl: TCustomTabControl; const AChild: TCustomPage;
const AText: string);
var
Index: Integer;
begin
Index := AChild.PageIndex;
if Index < 0 then
Index := ATabControl.IndexOf(AChild);
TQtTabWidget(ATabControl.Handle).setTabText(Index, AText{%H-});
end;
class procedure TQtWSCustomTabControl.SetTabPosition(
const ATabControl: TCustomTabControl; const ATabPosition: TTabPosition);
begin
TQtTabWidget(ATabControl.Handle).SetTabPosition(QTabWidgetTabPositionMap[ATabPosition]);
end;
class procedure TQtWSCustomTabControl.SetTabSize(
const ATabControl: TCustomTabControl; const ATabWidth, ATabHeight: integer);
begin
if not WSCheckHandleAllocated(ATabControl, 'SetTabSize') then
Exit;
TQtTabWidget(ATabControl.Handle).setTabSize(ATabHeight, ATabWidth);
end;
class procedure TQtWSCustomTabControl.ShowTabs(const ATabControl: TCustomTabControl;
AShowTabs: boolean);
var
TabWidget: TQtTabWidget;
begin
if not WSCheckHandleAllocated(ATabControl, 'ShowTabs') then
Exit;
if TQtWidget(ATabControl.Handle).ChildOfComplexWidget <> ccwTTabControl then
begin
TabWidget := TQtTabWidget(ATabControl.Handle);
if TabWidget.TabBar <> nil then
TabWidget.ShowTabs := AShowTabs;
end;
end;
class procedure TQtWSCustomTabControl.UpdateProperties(const ATabControl: TCustomTabControl);
begin
if not WSCheckHandleAllocated(ATabControl, 'UpdateProperties') then
Exit;
if TQtWidget(ATabControl.Handle).ChildOfComplexWidget <> ccwTTabControl then
begin
TQtTabWidget(ATabControl.Handle).setTabsClosable(nboShowCloseButtons in ATabControl.Options);
TQtTabWidget(ATabControl.Handle).SwitchTabsByKeyboard := nboKeyboardTabSwitch in ATabControl.Options;
end;
end;