Add current doc "Selection" as search scope, close #430

This commit is contained in:
Basile Burg 2019-03-15 22:18:17 +01:00
parent b47c43d38a
commit fd39a248c0
2 changed files with 81 additions and 10 deletions

View File

@ -24,7 +24,8 @@ The _Find all_ results are displayed in the [messages widget](widgets_messages),
![](img/find_all_results.png) ![](img/find_all_results.png)
The scope of _Find all_ can be set either to the current editor, to the whole project or to the opened documents by clicking the icon at the right. The scope of _Find all_ can be set either to the current editor, to the selection, to the whole project or to the opened documents by clicking the icon at the right.
The scope of _Replace all_ only works in the current document or its selection, according to the text displayed on the button mentioned above.
Notes: Notes:

View File

@ -46,7 +46,7 @@ type
procedure assignTo(target: TPersistent); override; procedure assignTo(target: TPersistent); override;
end; end;
TSearchScope = (scDoc, scProj, scOpened); TSearchScope = (scDoc, scSel, scProj, scOpened);
TSearchWidget = class(TDexedWidget, IDocumentObserver, IProjectObserver) TSearchWidget = class(TDexedWidget, IDocumentObserver, IProjectObserver)
btnAllScope: TBitBtn; btnAllScope: TBitBtn;
@ -78,6 +78,8 @@ type
procedure chkEnableRepChange(Sender: TObject); procedure chkEnableRepChange(Sender: TObject);
private private
fDoc: TDexedMemo; fDoc: TDexedMemo;
fDocSelStart: TPoint;
fDocSelStop: TPoint;
fToFind: string; fToFind: string;
fReplaceWth: string; fReplaceWth: string;
fActReplaceNext: TAction; fActReplaceNext: TAction;
@ -94,7 +96,8 @@ type
procedure actReplaceAllExecute(sender: TObject); procedure actReplaceAllExecute(sender: TObject);
procedure replaceEvent(Sender: TObject; const ASearch, AReplace: procedure replaceEvent(Sender: TObject; const ASearch, AReplace:
string; Line, Column: integer; var ReplaceAction: TSynReplaceAction); string; Line, Column: integer; var ReplaceAction: TSynReplaceAction);
procedure replaceInSelEvent(Sender: TObject; const ASearch, AReplace:
string; Line, Column: integer; var ReplaceAction: TSynReplaceAction);
procedure projNew(project: ICommonProject); procedure projNew(project: ICommonProject);
procedure projChanged(project: ICommonProject); procedure projChanged(project: ICommonProject);
procedure projClosing(project: ICommonProject); procedure projClosing(project: ICommonProject);
@ -125,7 +128,7 @@ implementation
const const
OptsFname = 'search.txt'; OptsFname = 'search.txt';
FindScopeStr: array[TSearchScope] of string = ('Document', 'Project', 'Opened docs'); FindScopeStr: array[TSearchScope] of string = ('Document', 'Selection', 'Project', 'Opened docs');
{$REGION TSearchOptions ------------------------------------------------------} {$REGION TSearchOptions ------------------------------------------------------}
constructor TSearchOptions.create(aOwner: TComponent); constructor TSearchOptions.create(aOwner: TComponent);
@ -324,9 +327,47 @@ begin
exit( MessageDlg('dexed', 'Replace this match ?', mtConfirmation, Btns, '')); exit( MessageDlg('dexed', 'Replace this match ?', mtConfirmation, Btns, ''));
end; end;
procedure TSearchWidget.replaceInSelEvent(Sender: TObject; const ASearch, AReplace:
string; Line, Column: integer; var ReplaceAction: TSynReplaceAction);
var
p: TPoint;
begin
if fFindScope = scSel then
begin
p := Point(Column, Line);
ReplaceAction := raReplace;
if p < fDocSelStart then
ReplaceAction := raSkip
else if p > fDocSelStop then
begin
ReplaceAction := raCancel;
fCancelAll := true;
end;
end;
end;
procedure TSearchWidget.replaceEvent(Sender: TObject; const ASearch, AReplace: procedure TSearchWidget.replaceEvent(Sender: TObject; const ASearch, AReplace:
string; Line, Column: integer; var ReplaceAction: TSynReplaceAction); string; Line, Column: integer; var ReplaceAction: TSynReplaceAction);
var
p: TPoint;
begin begin
if fFindScope = scSel then
begin
p := Point(Column, Line);
if p < fDocSelStart then
begin
ReplaceAction := raSkip;
exit;
end
else if p > fDocSelStop then
begin
ReplaceAction := raCancel;
fCancelAll := true;
exit;
end;
end;
case dlgReplaceAll of case dlgReplaceAll of
mrYes: ReplaceAction := raReplace; mrYes: ReplaceAction := raReplace;
mrNo: ReplaceAction := raSkip; mrNo: ReplaceAction := raSkip;
@ -355,8 +396,11 @@ begin
fSearchMru.Insert(0,fToFind); fSearchMru.Insert(0,fToFind);
cbToFind.Items.Assign(fSearchMru); cbToFind.Items.Assign(fSearchMru);
fDocSelStart := fDoc.BlockBegin;
fDocSelStop := fDoc.BlockEnd;
case fFindScope of case fFindScope of
scDoc: scDoc, scSel:
begin begin
findAll(fDoc.fileName, fDoc.Lines, true); findAll(fDoc.fileName, fDoc.Lines, true);
end; end;
@ -426,8 +470,16 @@ begin
search.Whole := ssoWholeWord in options; search.Whole := ssoWholeWord in options;
search.RegularExpressions:= ssoRegExpr in options; search.RegularExpressions:= ssoRegExpr in options;
search.Pattern:=fToFind; search.Pattern:=fToFind;
start := Point(1,1); if (fFindScope = scSel) and fDoc.SelAvail then
stop := Point(high(integer), lines.Count); begin
start := fDoc.BlockBegin;
stop := fDoc.BlockEnd;
end
else
begin
start := Point(1,1);
stop := Point(high(integer), lines.Count);
end;
while search.FindNextOne(lines, start, stop, startf, stopf) do while search.FindNextOne(lines, start, stop, startf, stopf) do
begin begin
setLength(res, length(res) + 1); setLength(res, length(res) + 1);
@ -530,6 +582,9 @@ begin
if fDoc.isNil then if fDoc.isNil then
exit; exit;
fDocSelStart := fDoc.BlockBegin;
fDocSelStop := fDoc.BlockEnd;
cbReplaceWth.Items.Assign(fReplaceMru); cbReplaceWth.Items.Assign(fReplaceMru);
opts := getOptions + [ssoReplace]; opts := getOptions + [ssoReplace];
opts -= [ssoBackwards]; opts -= [ssoBackwards];
@ -537,8 +592,14 @@ begin
fSearchMru.Insert(0, fToFind); fSearchMru.Insert(0, fToFind);
fReplaceMru.Insert(0, fReplaceWth); fReplaceMru.Insert(0, fReplaceWth);
if chkPrompt.Checked then if chkPrompt.Checked then
fDoc.OnReplaceText := @replaceEvent; fDoc.OnReplaceText := @replaceEvent
fDoc.CaretXY := Point(0,0); else if fFindScope = scSel then
begin
fDoc.OnReplaceText := @replaceInSelEvent;
// the event only called if ssoPrompt is included
opts += [ssoPrompt];
end;
fDoc.CaretXY := fDocSelStart;
while(true) do while(true) do
begin begin
if fDoc.SearchReplace(fToFind, fReplaceWth, opts) = 0 then if fDoc.SearchReplace(fToFind, fReplaceWth, opts) = 0 then
@ -550,6 +611,14 @@ begin
end; end;
end; end;
fDoc.OnReplaceText := nil; fDoc.OnReplaceText := nil;
if fFindScope = scSel then
begin
fDoc.BlockBegin := fDocSelStart;
fDoc.BlockEnd := fDocSelStop;
end;
updateImperative; updateImperative;
end; end;
{$ENDREGION} {$ENDREGION}
@ -658,7 +727,8 @@ end;
procedure TSearchWidget.btnAllScopeClick(Sender: TObject); procedure TSearchWidget.btnAllScopeClick(Sender: TObject);
begin begin
case fFindScope of case fFindScope of
scDoc: fFindScope := scProj; scDoc: fFindScope := scSel;
scSel: fFindScope := scProj;
scProj: fFindScope := scOpened; scProj: fFindScope := scOpened;
scOpened: fFindScope := scDoc; scOpened: fFindScope := scDoc;
end; end;