editor, a document can be set always visible in a split view to the right

This commit is contained in:
Basile Burg 2015-12-16 03:20:37 +01:00
parent b0ea3e918c
commit f40804ed85
10 changed files with 163 additions and 16 deletions

BIN
icons/other/splitter.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 424 B

View File

@ -55,6 +55,21 @@
<OtherUnitFiles Value="../../../src"/> <OtherUnitFiles Value="../../../src"/>
<UnitOutputDirectory Value="lib/$(TargetCPU)-$(TargetOS)"/> <UnitOutputDirectory Value="lib/$(TargetCPU)-$(TargetOS)"/>
</SearchPaths> </SearchPaths>
<Parsing>
<SyntaxOptions>
<IncludeAssertionCode Value="True"/>
</SyntaxOptions>
</Parsing>
<CodeGeneration>
<Checks>
<RangeChecks Value="True"/>
<OverflowChecks Value="True"/>
<StackChecks Value="True"/>
</Checks>
<Optimizations>
<OptimizationLevel Value="0"/>
</Optimizations>
</CodeGeneration>
<Linking> <Linking>
<Options> <Options>
<Win32> <Win32>

View File

@ -36,6 +36,7 @@ begin
AssignPng(fPageControl.addButton.Glyph, 'document_add'); AssignPng(fPageControl.addButton.Glyph, 'document_add');
AssignPng(fPageControl.moveLeftButton.Glyph, 'document_back'); AssignPng(fPageControl.moveLeftButton.Glyph, 'document_back');
AssignPng(fPageControl.moveRightButton.Glyph, 'document_next'); AssignPng(fPageControl.moveRightButton.Glyph, 'document_next');
AssignPng(fPageControl.splitButton.Glyph, 'splitter');
end; end;
procedure TForm1.pageControlChanged(sender: TObject); procedure TForm1.pageControlChanged(sender: TObject);

View File

@ -8,11 +8,11 @@ uses
Classes, SysUtils, Forms, Controls, ComCtrls, ExtCtrls, buttons; Classes, SysUtils, Forms, Controls, ComCtrls, ExtCtrls, buttons;
type type
TCEPageControlButton = (pbClose, pbMoveLeft, pbMoveRight, pbAdd); TCEPageControlButton = (pbClose, pbMoveLeft, pbMoveRight, pbAdd, pbSplit);
TCEPageControlButtons = set of TCEPageControlButton; TCEPageControlButtons = set of TCEPageControlButton;
const const
CEPageControlDefaultButtons = [pbClose, pbMoveLeft, pbMoveRight, pbAdd]; CEPageControlDefaultButtons = [pbClose, pbMoveLeft, pbMoveRight, pbAdd, pbSplit];
type type
@ -30,8 +30,9 @@ type
* Minimalist page-control dedicated to Coedit * Minimalist page-control dedicated to Coedit
* *
* - get rid of the framed aspect of the default LCL one * - get rid of the framed aspect of the default LCL one
* - no published props, since CE has to be compilable w/o extra IDE comps * - no published props, no need for design time support
* - add/close/move left and right speed buttons * - add/close/move left and right speed buttons
* - a particular tab can be set to reside on a split view
*) *)
TCEPageControl = class(TWinControl) TCEPageControl = class(TWinControl)
private private
@ -41,17 +42,22 @@ type
fMoveLeftBtn: TSpeedButton; fMoveLeftBtn: TSpeedButton;
fMoveRightBtn: TSpeedButton; fMoveRightBtn: TSpeedButton;
fAddBtn: TSpeedButton; fAddBtn: TSpeedButton;
fSplitBtn: TSpeedButton;
fContent: TPanel; fContent: TPanel;
fPages: TFPList; fPages: TFPList;
fPageIndex: integer; fPageIndex: integer;
fSplittedPageIndex: integer;
fButtons: TCEPageControlButtons; fButtons: TCEPageControlButtons;
fOnChanged: TNotifyEvent; fOnChanged: TNotifyEvent;
fOnChanging: TTabChangingEvent; fOnChanging: TTabChangingEvent;
fSplitter: TSplitter;
fOldSplitPos: integer;
procedure btnCloseClick(sender: TObject); procedure btnCloseClick(sender: TObject);
procedure btnMoveLeftClick(sender: TObject); procedure btnMoveLeftClick(sender: TObject);
procedure btnMoveRightClick(sender: TObject); procedure btnMoveRightClick(sender: TObject);
procedure btnAddClick(sender: TObject); procedure btnAddClick(sender: TObject);
procedure btnSplitClick(sender: TObject);
procedure tabsChanging(Sender: TObject; var AllowChange: Boolean); procedure tabsChanging(Sender: TObject; var AllowChange: Boolean);
procedure tabsChanged(sender: TObject); procedure tabsChanged(sender: TObject);
@ -87,6 +93,7 @@ type
property moveLeftButton: TSpeedButton read fMoveLeftBtn; property moveLeftButton: TSpeedButton read fMoveLeftBtn;
property moveRightButton: TSpeedButton read fMoveRightBtn; property moveRightButton: TSpeedButton read fMoveRightBtn;
property addButton: TSpeedButton read fAddBtn; property addButton: TSpeedButton read fAddBtn;
property splitButton: TSpeedButton read fSplitBtn;
property onChanged: TNotifyEvent read fOnChanged write fOnChanged; property onChanged: TNotifyEvent read fOnChanged write fOnChanged;
property onChanging: TTabChangingEvent read fOnChanging write fOnChanging; property onChanging: TTabChangingEvent read fOnChanging write fOnChanging;
@ -126,6 +133,8 @@ begin
fHeader.Align := alTop; fHeader.Align := alTop;
fHeader.Height:= 32; fHeader.Height:= 32;
fSplittedPageIndex:=-1;
fTabs := TTabControl.Create(self); fTabs := TTabControl.Create(self);
fTabs.Parent:= fHeader; fTabs.Parent:= fHeader;
fTabs.Align := alClient; fTabs.Align := alClient;
@ -169,6 +178,15 @@ begin
fCloseBtn.OnClick:=@btnCloseClick; fCloseBtn.OnClick:=@btnCloseClick;
fCloseBtn.Hint:='close current page'; fCloseBtn.Hint:='close current page';
fSplitBtn := TSpeedButton.Create(self);
fSplitBtn.Parent := fHeader;
fSplitBtn.Align:= alRight;
fSplitBtn.Width:= 28;
fSplitBtn.BorderSpacing.Around:= 2;
fSplitBtn.ShowCaption:=false;
fSplitBtn.OnClick:=@btnSplitClick;
fSplitBtn.Hint:= 'pin or un-pin the page to the right';
fContent := TPanel.Create(self); fContent := TPanel.Create(self);
fContent.Parent := self; fContent.Parent := self;
fContent.Align := alClient; fContent.Align := alClient;
@ -177,6 +195,12 @@ begin
fContent.BorderStyle:=bsNone; fContent.BorderStyle:=bsNone;
fContent.BorderSpacing.Top:=3; fContent.BorderSpacing.Top:=3;
fSplitter := TSplitter.Create(self);
fSplitter.Parent := fContent;
fSplitter.Visible:= false;
fSplitter.Align := alLeft;
fSplitter.Width := 6;
fPages := TFPList.Create; fPages := TFPList.Create;
fPageIndex := -1; fPageIndex := -1;
@ -201,7 +225,8 @@ end;
procedure TCEPageControl.tabsChanged(sender: TObject); procedure TCEPageControl.tabsChanged(sender: TObject);
begin begin
setPageIndex(fTabs.TabIndex); if fTabs.TabIndex < fPages.Count then
setPageIndex(fTabs.TabIndex);
end; end;
procedure TCEPageControl.tabsChanging(Sender: TObject; var AllowChange: Boolean); procedure TCEPageControl.tabsChanging(Sender: TObject; var AllowChange: Boolean);
@ -229,6 +254,8 @@ begin
exit; exit;
pge := TCEPage(fPages.Items[index]); pge := TCEPage(fPages.Items[index]);
if (fSplittedPageIndex = -1) or (index = fSplittedPageIndex) then
pge.Align:=alClient;
pge.Visible:=true; pge.Visible:=true;
pge.Repaint; pge.Repaint;
for ctl in pge.GetEnumeratorControls do for ctl in pge.GetEnumeratorControls do
@ -236,15 +263,42 @@ begin
end; end;
procedure TCEPageControl.setPageIndex(index: integer); procedure TCEPageControl.setPageIndex(index: integer);
var
leftp, rightp: TCEPage;
begin begin
if (fPageIndex <> fSplittedPageIndex) then
fOldSplitPos := fSplitter.Left;
if (index > fPages.Count-1) then if (index > fPages.Count-1) then
index := fPages.Count-1; index := fPages.Count-1;
if (index < 0) then if (index < 0) then
exit; exit;
hidePage(fPageIndex); if (fSplittedPageIndex = -1) or (index = fSplittedPageIndex) then
fPageIndex := index; begin
showPage(fPageIndex); hidePage(fPageIndex);
fPageIndex := index;
showPage(fPageIndex);
fSplitter.Visible:= false;
end
else if (fSplittedPageIndex <> -1) then
begin
hidePage(fPageIndex);
fPageIndex := index;
fSplitter.Visible:= true;
leftp := getPage(fPageIndex);
leftp.Align := alLeft;
if fOldSplitPos = 0 then
leftp.Width:= (fContent.Width - fSplitter.Width) div 2
else
leftp.Width:= fOldSplitPos;
showPage(fPageIndex);
rightp := getPage(fSplittedPageIndex);
rightp.Align := alClient;
showPage(fSplittedPageIndex);
end;
if fTabs.TabIndex <> fPageIndex then if fTabs.TabIndex <> fPageIndex then
fTabs.TabIndex:= fPageIndex; fTabs.TabIndex:= fPageIndex;
@ -272,12 +326,18 @@ begin
if (index > fPages.Count-1) or (index < 0) then if (index > fPages.Count-1) or (index < 0) then
exit; exit;
if index = fSplittedPageIndex then
fSplittedPageIndex := -1
else if index < fSplittedPageIndex then
fSplittedPageIndex -= 1;
TCEPage(fPages.Items[index]).Free; TCEPage(fPages.Items[index]).Free;
if fPageIndex >= fPages.Count then
fPageIndex -= 1;
fPages.Delete(index); fPages.Delete(index);
fTabs.Tabs.Delete(index); fTabs.Tabs.Delete(index);
if fPageIndex >= fPages.Count then
fPageIndex -= 1;
updateButtonsState; updateButtonsState;
if fPages.Count = 0 then if fPages.Count = 0 then
exit; exit;
@ -320,6 +380,8 @@ begin
fPages.Exchange(fPageIndex, fPageIndex + 1); fPages.Exchange(fPageIndex, fPageIndex + 1);
fTabs.Tabs.Exchange(fPageIndex, fPageIndex + 1); fTabs.Tabs.Exchange(fPageIndex, fPageIndex + 1);
if fPageIndex = fSplittedPageIndex then
fSplittedPageIndex += 1;
setPageIndex(fPageIndex+1); setPageIndex(fPageIndex+1);
end; end;
@ -330,6 +392,8 @@ begin
fPages.Exchange(fPageIndex, fPageIndex - 1); fPages.Exchange(fPageIndex, fPageIndex - 1);
fTabs.Tabs.Exchange(fPageIndex, fPageIndex - 1); fTabs.Tabs.Exchange(fPageIndex, fPageIndex - 1);
if fPageIndex = fSplittedPageIndex then
fSplittedPageIndex -= 1;
setPageIndex(fPageIndex-1); setPageIndex(fPageIndex-1);
end; end;
@ -353,6 +417,19 @@ begin
addPage; addPage;
end; end;
procedure TCEPageControl.btnSplitClick(sender: TObject);
begin
if fPageIndex = fSplittedPageIndex then
fSplittedPageIndex := -1
else
begin
if (fSplittedPageIndex <> -1) then
hidePage(fSplittedPageIndex);
fSplittedPageIndex:= fPageIndex;
end;
setPageIndex(fPageIndex);
end;
procedure TCEPageControl.setButtons(value: TCEPageControlButtons); procedure TCEPageControl.setButtons(value: TCEPageControlButtons);
begin begin
if fButtons = value then if fButtons = value then
@ -370,6 +447,7 @@ begin
fMoveLeftBtn.Visible:= pbMoveLeft in fButtons; fMoveLeftBtn.Visible:= pbMoveLeft in fButtons;
fCloseBtn.Visible:= pbMoveRight in fButtons; fCloseBtn.Visible:= pbMoveRight in fButtons;
fAddBtn.Visible:= pbAdd in fButtons; fAddBtn.Visible:= pbAdd in fButtons;
fSplitBtn.Visible:= pbSplit in fButtons;
fHeader.EnableAlign; fHeader.EnableAlign;
fCloseBtn.Enabled := fPageIndex <> -1; fCloseBtn.Enabled := fPageIndex <> -1;
fMoveLeftBtn.Enabled := fPageIndex > 0; fMoveLeftBtn.Enabled := fPageIndex > 0;

View File

@ -77,6 +77,7 @@ type
function findDocument(aFilename: string): TCESynMemo; function findDocument(aFilename: string): TCESynMemo;
procedure openDocument(aFilename: string); procedure openDocument(aFilename: string);
function closeDocument(index: Integer): boolean; function closeDocument(index: Integer): boolean;
function closeDocument(doc: TCESynMemo): boolean;
public public
constructor create(aOwner: TComponent); override; constructor create(aOwner: TComponent); override;
destructor destroy; override; destructor destroy; override;
@ -102,6 +103,7 @@ begin
AssignPng(pageControl.moveRightButton, 'go_next'); AssignPng(pageControl.moveRightButton, 'go_next');
AssignPng(pageControl.addButton, 'document_add'); AssignPng(pageControl.addButton, 'document_add');
AssignPng(pageControl.closeButton, 'document_delete'); AssignPng(pageControl.closeButton, 'document_delete');
AssignPng(pageControl.splitButton, 'splitter');
fTokList := TLexTokenList.Create; fTokList := TLexTokenList.Create;
fErrList := TLexErrorList.Create; fErrList := TLexErrorList.Create;
@ -245,6 +247,16 @@ begin
doc.Free; doc.Free;
result := true; result := true;
end; end;
function TCEEditorWidget.closeDocument(doc: TCESynMemo): boolean;
var
page: TCEPage = nil;
begin
page := TCEPage(doc.Parent);
if not assigned(page) then
exit(false);
exit(closeDocument(page.index));
end;
{$ENDREGION} {$ENDREGION}
{$REGION PageControl/Editor things ---------------------------------------------} {$REGION PageControl/Editor things ---------------------------------------------}

View File

@ -1879,6 +1879,22 @@ LazarusResources.Add('script_bricks','PNG',[
+#208#201#143'>'#252#244#1#12#220#156#248'7'#240'qB'#160'$'#2#254#184#240'?' +#208#201#143'>'#252#244#1#12#220#156#248'7'#240'qB'#160'$'#2#254#184#240'?'
+#248'.'#192#0#5#181#10'l'#246#0'3S'#0#0#0#0'IEND'#174'B`'#130 +#248'.'#192#0#5#181#10'l'#246#0'3S'#0#0#0#0'IEND'#174'B`'#130
]); ]);
LazarusResources.Add('splitter','PNG',[
#137'PNG'#13#10#26#10#0#0#0#13'IHDR'#0#0#0#16#0#0#0#16#8#4#0#0#0#181#250'7'
+#234#0#0#0#4'gAMA'#0#0#177#143#11#252'a'#5#0#0#0' cHRM'#0#0'z&'#0#0#128#132#0
+#0#250#0#0#0#128#232#0#0'u0'#0#0#234'`'#0#0':'#152#0#0#23'p'#156#186'Q<'#0#0
+#0#2'bKGD'#0#255#135#143#204#191#0#0#0#158'IDAT('#207#133#145'A'#10#194'0'#16
+'E_'#146#174#4'!'#189#137#160#135#240#2#30'F'#196#149'G'#17#234#29#188#131
+#224'M*'#212#173#254'.'#146#198'&'#218':'#217#204#252#188#201#12'?F'#132'0'#0
+'\Z'#239#159#143']'#13#16'n,Y'#188#253#150#133#31'+'#5' Z'#196#12#0'U'#28'6'
+#9#216'B*'#0'C5'#13#236#5#138'/'#28#244#5#28#163#228#176#128'Iu'#2'N'#155'a'
+#136#0#165#26#197#3#172#225','#233#170#144#7'}'#188#209'mh'#249#228'?'#140
+#226#159'Q'#204#251'`'#1'W8'#155#27#213'5Ku'#153#146#127'7+'#28'/'#238'iYz'
+#241#246'-'#217#139'7'#147#144#0#0#0'%tEXtdate:create'#0'2014-03-13T04:26:36'
+'+02:00'#249'}l'#149#0#0#0'%tEXtdate:modify'#0'2013-01-08T13:07:58+02:00'#133
+#178'Km'#0#0#0#25'tEXtSoftware'#0'Adobe ImageReadyq'#201'e<'#0#0#0#0'IEND'
+#174'B`'#130
]);
LazarusResources.Add('application','PNG',[ LazarusResources.Add('application','PNG',[
#137'PNG'#13#10#26#10#0#0#0#13'IHDR'#0#0#0#16#0#0#0#16#8#6#0#0#0#31#243#255'a' #137'PNG'#13#10#26#10#0#0#0#13'IHDR'#0#0#0#16#0#0#0#16#8#6#0#0#0#31#243#255'a'
+#0#0#0#25'tEXtSoftware'#0'Adobe ImageReadyq'#201'e<'#0#0#1#175'IDATx'#218#156 +#0#0#0#25'tEXtSoftware'#0'Adobe ImageReadyq'#201'e<'#0#0#1#175'IDATx'#218#156

View File

@ -289,14 +289,16 @@ type
ICEMultiDocHandler = interface(ICESingleService) ICEMultiDocHandler = interface(ICESingleService)
// returns the count of opened document // returns the count of opened document
function documentCount: Integer; function documentCount: Integer;
// returns the index-th document // returns the nth document
function getDocument(index: Integer): TCESynMemo; function getDocument(index: Integer): TCESynMemo;
// returns true if the document matching aFielanme is already opened. // returns true if the document matching aFielanme is already opened.
function findDocument(aFilename: string): TCESynMemo; function findDocument(aFilename: string): TCESynMemo;
// open or set the focus on the document matching aFilename // open or set the focus on the document matching aFilename
procedure openDocument(aFilename: string); procedure openDocument(aFilename: string);
// close the index-th document // close the nth document
function closeDocument(index: Integer): boolean; function closeDocument(index: Integer): boolean;
// close a particular document
function closeDocument(doc: TCESynMemo): boolean;
// conveniance property // conveniance property
property document[index: integer]: TCESynMemo read getDocument; property document[index: integer]: TCESynMemo read getDocument;
end; end;

View File

@ -1,7 +1,7 @@
object CEMainForm: TCEMainForm object CEMainForm: TCEMainForm
Left = 480 Left = 485
Height = 49 Height = 49
Top = 238 Top = 239
Width = 745 Width = 745
AllowDropFiles = True AllowDropFiles = True
Caption = 'Coedit' Caption = 'Coedit'

View File

@ -1575,7 +1575,7 @@ begin
if fDoc = nil then exit; if fDoc = nil then exit;
if (fDoc.modified or(fDoc.fileName = fDoc.tempFilename)) if (fDoc.modified or(fDoc.fileName = fDoc.tempFilename))
and (dlgFileChangeClose(fDoc.fileName) = mrCancel) then exit; and (dlgFileChangeClose(fDoc.fileName) = mrCancel) then exit;
fDoc.Free; getMultiDocHandler.closeDocument(fDoc);
end; end;
procedure TCEMainForm.actFileSaveAllExecute(Sender: TObject); procedure TCEMainForm.actFileSaveAllExecute(Sender: TObject);

View File

@ -8,7 +8,7 @@ uses
Classes, SysUtils, controls,lcltype, Forms, graphics, ExtCtrls, crc, Classes, SysUtils, controls,lcltype, Forms, graphics, ExtCtrls, crc,
SynEdit, SynPluginSyncroEdit, SynCompletion, SynEditKeyCmds, LazSynEditText, SynEdit, SynPluginSyncroEdit, SynCompletion, SynEditKeyCmds, LazSynEditText,
SynHighlighterLFM, SynEditHighlighter, SynEditMouseCmds, SynEditFoldedView, SynHighlighterLFM, SynEditHighlighter, SynEditMouseCmds, SynEditFoldedView,
SynEditMarks, SynEditMarkup, SynEditTypes, SynEditMarks, SynEditMarkup, SynEditTypes, Messages, LMessages,
ce_common, ce_observer, ce_writableComponent, ce_d2syn, ce_txtsyn, ce_dialogs, ce_common, ce_observer, ce_writableComponent, ce_d2syn, ce_txtsyn, ce_dialogs,
ce_sharedres; ce_sharedres;
@ -103,6 +103,7 @@ type
fIsDSource: boolean; fIsDSource: boolean;
fIsTxtFile: boolean; fIsTxtFile: boolean;
fIsConfig: boolean; fIsConfig: boolean;
fFocusForInput: boolean;
fIdentifier: string; fIdentifier: string;
fTempFileName: string; fTempFileName: string;
fMultiDocSubject: TCECustomSubject; fMultiDocSubject: TCECustomSubject;
@ -154,6 +155,8 @@ type
procedure removeBreakPoint(line: integer); procedure removeBreakPoint(line: integer);
function findBreakPoint(line: integer): boolean; function findBreakPoint(line: integer): boolean;
protected protected
procedure DoEnter; override;
procedure DoExit; override;
procedure DoOnProcessCommand(var Command: TSynEditorCommand; var AChar: TUTF8Char; procedure DoOnProcessCommand(var Command: TSynEditorCommand; var AChar: TUTF8Char;
Data: pointer); override; Data: pointer); override;
procedure MouseLeave; override; procedure MouseLeave; override;
@ -556,6 +559,25 @@ begin
subjDocFocused(TCEMultiDocSubject(fMultiDocSubject), self); subjDocFocused(TCEMultiDocSubject(fMultiDocSubject), self);
end; end;
procedure TCESynMemo.DoEnter;
begin
inherited;
checkFileDate;
if not fFocusForInput then
subjDocFocused(TCEMultiDocSubject(fMultiDocSubject), self);
fFocusForInput := true;
end;
procedure TCESynMemo.DoExit;
begin
inherited;
fFocusForInput := false;
fDDocWin.Hide;
fCallTipWin.Hide;
if fCompletion.IsActive then
fCompletion.Deactivate;
end;
procedure TCESynMemo.SetVisible(Value: Boolean); procedure TCESynMemo.SetVisible(Value: Boolean);
begin begin
inherited; inherited;
@ -569,7 +591,7 @@ begin
else begin else begin
fDDocWin.Hide; fDDocWin.Hide;
fCallTipWin.Hide; fCallTipWin.Hide;
if fCompletion.IsActive then if fCompletion.IsActive then
fCompletion.Deactivate; fCompletion.Deactivate;
end; end;
end; end;
@ -1091,6 +1113,7 @@ end;
procedure TCESynMemo.MouseLeave; procedure TCESynMemo.MouseLeave;
begin begin
inherited;
fDDocWin.Hide; fDDocWin.Hide;
fCallTipWin.Hide; fCallTipWin.Hide;
end; end;