doublecmd/src/ufileviewnotebook.pas

512 lines
13 KiB
ObjectPascal

unit uFileViewNotebook;
{$mode objfpc}{$H+}
interface
uses
Classes, SysUtils, Controls, ComCtrls, ExtCtrls {Lazarus < 31552},
uFileView, uFilePanelSelect, uDCVersion, LCLVersion;
const
lazRevNewTabControl = '31767';
lazRevOnPageChangedRemoved = '32622';
type
TTabLockState = (
tlsNormal, //<en Default state.
tlsPathLocked, //<en Path changes are not allowed.
tlsPathResets, //<en Path is reset when activating the tab.
tlsDirsInNewTab); //<en Path change opens a new tab.
TFileViewNotebook = class;
{ TFileViewPage }
TFileViewPage = class(TCustomPage)
private
FLockState: TTabLockState;
FLockPath: String; //<en Path on which tab is locked
{$IF DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)}
FSettingCaption: Boolean;
{$ENDIF}
FOnActivate: TNotifyEvent;
{en
Shows or removes the '*' indicator of a locked tab.
}
procedure UpdateTabLockState;
{en
Retrieves the file view on this page.
}
function GetFileView: TFileView;
{en
Frees current file view and assigns a new one.
}
procedure SetFileView(aFileView: TFileView);
{en
Retrieves notebook on which this page is.
}
function GetNotebook: TFileViewNotebook;
procedure SetLockState(NewLockState: TTabLockState);
procedure DoActivate;
{$IF (DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)) or DEFINED(MSWINDOWS)}
protected
procedure RealSetText(const AValue: TCaption); override;
{$ENDIF}
public
constructor Create(TheOwner: TComponent); override;
{$IF DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)}
function HandleObjectShouldBeVisible: boolean; override;
{$ENDIF}
function IsActive: Boolean;
procedure MakeActive;
procedure UpdateCaption(NewCaption: String);
property LockState: TTabLockState read FLockState write SetLockState;
property LockPath: String read FLockPath write FLockPath;
property FileView: TFileView read GetFileView write SetFileView;
property Notebook: TFileViewNotebook read GetNotebook;
property OnActivate: TNotifyEvent read FOnActivate write FOnActivate;
end;
{ TFileViewNotebook }
{$IF (lcl_release >= 31) and (lazRevision >= lazRevNewTabControl)}
TFileViewNotebook = class(TCustomTabControl)
{$ELSE}
TFileViewNotebook = class(TCustomNotebook)
{$ENDIF}
private
FNotebookSide: TFilePanelSelect;
FStartDrag: Boolean;
FDraggedPageIndex: Integer;
{$IF DEFINED(LCLGTK2)}
FLastMouseDownTime: TDateTime;
{$ENDIF}
function GetActivePage: TFileViewPage;
function GetActiveView: TFileView;
function GetFileViewOnPage(Index: Integer): TFileView;
function GetPage(Index: Integer): TFileViewPage;
procedure DragOverEvent(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean);
procedure DragDropEvent(Sender, Source: TObject; X, Y: Integer);
protected
procedure DoChange; override;
procedure MouseDown(Button: TMouseButton; Shift: TShiftState; X, Y: Integer); override;
procedure MouseMove(Shift: TShiftState; X, Y: Integer); override;
procedure MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y: Integer); override;
public
constructor Create(ParentControl: TWinControl;
NotebookSide: TFilePanelSelect); reintroduce;
function AddPage(aCaption: String = ''): TFileViewPage;
function InsertPage(Index: Integer; aCaption: String = ''): TFileViewPage;
procedure RemovePage(Index: Integer);
procedure RemovePage(var aPage: TFileViewPage);
procedure DestroyAllPages;
procedure ActivatePrevTab;
procedure ActivateNextTab;
property ActivePage: TFileViewPage read GetActivePage;
property ActiveView: TFileView read GetActiveView;
property Page[Index: Integer]: TFileViewPage read GetPage;
property View[Index: Integer]: TFileView read GetFileViewOnPage; default;
property Side: TFilePanelSelect read FNotebookSide;
published
property OnDblClick;
{$IF DECLARED(lcl_fullversion) and (lcl_fullversion >= 093100) and (lazRevision >= lazRevOnPageChangedRemoved)}
property OnChange;
{$ENDIF}
property OnMouseDown;
property OnMouseUp;
end;
implementation
uses
LCLProc,
uGlobs
{$IF DEFINED(LCLGTK2)}
, GTK2Globals // for DblClickTime
{$ENDIF}
{$IF DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)}
, qt4, qtwidgets
{$ENDIF}
{$IF DEFINED(MSWINDOWS)}
, win32proc
{$ENDIF}
;
// -- TFileViewPage -----------------------------------------------------------
constructor TFileViewPage.Create(TheOwner: TComponent);
begin
FLockState := tlsNormal;
{$IF DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)}
FSettingCaption := False;
{$ENDIF}
inherited Create(TheOwner);
end;
{$IF DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)}
// On QT after handle is created but before the widget is visible
// setting caption fails unless the notebook and all its parents are
// set as Visible and the current page is the one of which we set caption.
// Overriding HandleObjectShouldBeVisible is a indirect workaround for that
// (see TQtPage.getIndex.CanReturnIndex).
// QT 4.6 or higher needed for this workaround.
function TFileViewPage.HandleObjectShouldBeVisible: boolean;
var
AParent: QTabWidgetH;
begin
if not HandleAllocated then
Result := inherited
else
begin
AParent := TQtPage(Handle).getTabWidget;
Result := (FSettingCaption and ((AParent = nil) or not QWidget_isVisible(AParent))) or
inherited;
end;
end;
{$ENDIF}
{$IF (DEFINED(LCLQT) and (LCL_FULLVERSION < 093100)) or DEFINED(MSWINDOWS)}
procedure TFileViewPage.RealSetText(const AValue: TCaption);
begin
{$IF DEFINED(LCLQT)}
FSettingCaption := True;
{$ENDIF}
inherited;
{$IF DEFINED(MSWINDOWS)}
if HandleAllocated then
LCLControlSizeNeedsUpdate(Parent, True);
{$ENDIF}
{$IF DEFINED(LCLQT)}
FSettingCaption := False;
{$ENDIF}
end;
{$ENDIF}
function TFileViewPage.IsActive: Boolean;
begin
Result := Assigned(Notebook) and (Notebook.PageIndex = PageIndex);
end;
procedure TFileViewPage.MakeActive;
var
aFileView: TFileView;
begin
if Assigned(Notebook) then
begin
Notebook.PageIndex := PageIndex;
aFileView := FileView;
if Assigned(aFileView) then
aFileView.SetFocus;
end;
end;
procedure TFileViewPage.UpdateCaption(NewCaption: String);
begin
if NewCaption <> '' then
begin
if (tb_text_length_limit in gDirTabOptions) and (UTF8Length(NewCaption) > gDirTabLimit) then
Caption := UTF8Copy(NewCaption, 1, gDirTabLimit) + '...'
else
Caption := NewCaption;
UpdateTabLockState;
end;
end;
procedure TFileViewPage.UpdateTabLockState;
var
NewCaption: String;
begin
if Caption[1] = '*' then
NewCaption := Copy(Caption, 2, Length(Caption) - 1)
else
NewCaption := Caption;
if (FLockState <> tlsNormal) and (tb_show_asterisk_for_locked in gDirTabOptions) then
Caption := '*' + NewCaption
else
Caption := NewCaption;
end;
function TFileViewPage.GetFileView: TFileView;
begin
if ComponentCount > 0 then
Result := TFileView(Components[0])
else
Result := nil;
end;
procedure TFileViewPage.SetFileView(aFileView: TFileView);
var
aComponent: TComponent;
begin
while ComponentCount > 0 do
begin
aComponent := Components[0];
RemoveComponent(aComponent);
aComponent.Free;
end;
if Assigned(aFileView) then
begin
InsertComponent(aFileView);
aFileView.Parent := Self;
end;
end;
function TFileViewPage.GetNotebook: TFileViewNotebook;
begin
Result := Parent as TFileViewNotebook;
end;
procedure TFileViewPage.SetLockState(NewLockState: TTabLockState);
begin
FLockState := NewLockState;
if NewLockState = tlsPathResets then
LockPath := FileView.CurrentPath;
UpdateTabLockState;
end;
procedure TFileViewPage.DoActivate;
begin
if Assigned(FOnActivate) then
FOnActivate(Self);
end;
// -- TFileViewNotebook -------------------------------------------------------
constructor TFileViewNotebook.Create(ParentControl: TWinControl;
NotebookSide: TFilePanelSelect);
begin
PageClass := TFileViewPage;
inherited Create(ParentControl);
Parent := ParentControl;
TabStop := False;
FNotebookSide := NotebookSide;
FStartDrag := False;
OnDragOver := @DragOverEvent;
OnDragDrop := @DragDropEvent;
end;
function TFileViewNotebook.GetActivePage: TFileViewPage;
begin
if PageIndex <> -1 then
Result := GetPage(PageIndex)
else
Result := nil;
end;
function TFileViewNotebook.GetActiveView: TFileView;
var
APage: TFileViewPage;
begin
APage := GetActivePage;
if Assigned(APage) then
Result := APage.FileView
else
Result := nil;
end;
function TFileViewNotebook.GetFileViewOnPage(Index: Integer): TFileView;
var
APage: TFileViewPage;
begin
APage := GetPage(Index);
Result := APage.FileView;
end;
function TFileViewNotebook.GetPage(Index: Integer): TFileViewPage;
begin
Result := TFileViewPage(CustomPage(Index));
end;
function TFileViewNotebook.AddPage(aCaption: String): TFileViewPage;
begin
Result := InsertPage(PageCount, aCaption);
end;
function TFileViewNotebook.InsertPage(Index: Integer; aCaption: String = ''): TFileViewPage;
begin
if aCaption = '' then
aCaption := IntToStr(Index);
Pages.Insert(Index, aCaption);
Result := GetPage(Index);
ShowTabs:= ((PageCount > 1) or (tb_always_visible in gDirTabOptions)) and gDirectoryTabs;
end;
procedure TFileViewNotebook.RemovePage(Index: Integer);
begin
{$IFDEF LCLGTK2}
// If removing currently active page, switch to another page first.
// Otherwise there can be no page selected.
if (PageIndex = Index) and (PageCount > 1) then
begin
if Index = PageCount - 1 then
Page[Index - 1].MakeActive
else
Page[Index + 1].MakeActive;
end;
{$ENDIF}
Pages.Delete(Index);
ShowTabs:= ((PageCount > 1) or (tb_always_visible in gDirTabOptions)) and gDirectoryTabs;
{$IFNDEF LCLGTK2}
// Force-activate current page.
if PageIndex <> -1 then
Page[PageIndex].MakeActive;
{$ENDIF}
end;
procedure TFileViewNotebook.RemovePage(var aPage: TFileViewPage);
begin
RemovePage(aPage.PageIndex);
aPage := nil;
end;
procedure TFileViewNotebook.DestroyAllPages;
begin
while PageCount > 0 do
Page[0].Free;
end;
procedure TFileViewNotebook.ActivatePrevTab;
begin
if PageIndex = 0 then
Page[PageCount - 1].MakeActive
else
Page[PageIndex - 1].MakeActive;
end;
procedure TFileViewNotebook.ActivateNextTab;
begin
if PageIndex = PageCount - 1 then
Page[0].MakeActive
else
Page[PageIndex + 1].MakeActive;
end;
procedure TFileViewNotebook.MouseDown(Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
begin
inherited;
if Button = mbLeft then
begin
FDraggedPageIndex := TabIndexAtClientPos(Classes.Point(X, Y));
FStartDrag := (FDraggedPageIndex <> -1);
end;
{$IF DEFINED(LCLGTK2)} // emulate double click under GTK2
if (Button = mbLeft) and Assigned(OnDblClick) and (FDraggedPageIndex < 0) then
begin
if ((Now - FLastMouseDownTime) > ((1/86400)*(DblClickTime/1000))) then
FLastMouseDownTime:= Now
else
begin
OnDblClick(Self);
FLastMouseDownTime:= 0;
end;
end;
{$ENDIF}
end;
procedure TFileViewNotebook.MouseMove(Shift: TShiftState; X, Y: Integer);
begin
inherited;
if FStartDrag then
begin
FStartDrag := False;
BeginDrag(False);
end;
end;
procedure TFileViewNotebook.MouseUp(Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
begin
inherited;
FStartDrag := False;
end;
procedure TFileViewNotebook.DragOverEvent(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean);
var
ATabIndex: Integer;
begin
if (Source is TFileViewNotebook) and (Sender is TFileViewNotebook) then
begin
ATabIndex := TabIndexAtClientPos(Classes.Point(X, Y));
Accept := (ATabIndex <> -1) and
((Source <> Sender) or (ATabIndex <> FDraggedPageIndex));
end
else
Accept := False;
end;
procedure TFileViewNotebook.DragDropEvent(Sender, Source: TObject; X, Y: Integer);
var
SourceNotebook: TFileViewNotebook;
ATabIndex: Integer;
NewPage, DraggedPage: TFileViewPage;
begin
if (Source is TFileViewNotebook) and (Sender is TFileViewNotebook) then
begin
ATabIndex := TabIndexAtClientPos(Classes.Point(X, Y));
if ATabIndex = -1 then
Exit;
if Source = Sender then
begin
// Move within the same panel.
Pages.Move(FDraggedPageIndex, ATabIndex);
end
else
begin
// Move page between panels.
SourceNotebook := (Source as TFileViewNotebook);
DraggedPage := SourceNotebook.Page[SourceNotebook.FDraggedPageIndex];
// Create a clone of the page in the panel.
NewPage := InsertPage(ATabIndex, DraggedPage.Caption);
DraggedPage.FileView.Clone(NewPage);
NewPage.MakeActive;
if (ssShift in GetKeyShiftState) and (SourceNotebook.PageCount > 1) then
begin
// Remove page from source panel.
SourceNotebook.RemovePage(DraggedPage);
end;
end;
end;
end;
procedure TFileViewNotebook.DoChange;
begin
inherited DoChange;
ActivePage.DoActivate;
end;
end.