lazarus/components/todolist/todolist.pas

550 lines
16 KiB
ObjectPascal

(***************************************************************************
todolist.pp
--------------------
***************************************************************************/
***************************************************************************
* *
* This source is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This code is distributed in the hope that it will be useful, but *
* WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
* General Public License for more details. *
* *
* A copy of the GNU General Public License is available on the World *
* Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also *
* obtain it by writing to the Free Software Foundation, *
* Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA. *
* *
***************************************************************************
Author:
Olivier GUILBAUD <golivier@free.fr>,
Gerard Visent <gerardusmercator@gmail.com>
Mattias Gaertner
Alexander du Plessis
Silvio Clecio
Kevin Jesshope
Abstract:
List all to do comments of current project and the file
projectname.todo.
{TODO Priority -oOwnerName -cCategoryName: Todo_text}
{DONE Priority -oOwnerName -cCategoryName: Todo_text}
{NOTE Priority -oOwnerName -cCategoryName: Note_text}
{#todo Priority -oOwnerName -cCategoryName: Todo_text}
{#done Priority -oOwnerName -cCategoryName: Todo_text}
{#note Priority -oOwnerName -cCategoryName: Note_text}
the Priority, -o and -c tags are optional.
Quoted OwnerName and CategoryName are permitted.
-o'Lazarus Dev Team' etc
The colon before the text is optional. Anything to the right of the colon
will be interpretted as the description text
Sub comments in nested comments are ignored.
*)
unit ToDoList;
{$mode objfpc}{$H+}
interface
uses
// FCL, RTL
Classes, SysUtils, Math, AVL_Tree,
// LCL
LCLType, LclIntf, Forms, Controls, StdCtrls, Dialogs, ComCtrls,
ActnList, XMLPropStorage, ExtCtrls,
// LazUtils
LazFileUtils, LazFileCache, LazLoggerBase, LazTracer, AvgLvlTree,
// Codetools
CodeToolManager, FileProcs,
// IDEIntf
LazIDEIntf, IDEImagesIntf, PackageIntf, ProjectIntf,
// ToDoList
ToDoListCore, ToDoListStrConsts;
Const
ToDoWindowName = 'IDETodoWindow';
type
TOnOpenFile = procedure(Sender: TObject; const Filename: string;
const LineNumber: integer) of object;
{ TIDETodoWindow }
TIDETodoWindow = class(TForm)
acGoto: TAction;
acRefresh: TAction;
acExport: TAction;
acHelp: TAction;
ActionList: TActionList;
chkListed: TCheckBox;
chkUsed: TCheckBox;
chkPackages: TCheckBox;
chkSourceEditor: TCheckBox;
cboShowWhat: TComboBox;
lblOptions: TLabel;
lblShowWhat: TLabel;
lvTodo: TListView;
pnlOptions: TPanel;
pnlShowWhat: TPanel;
SaveDialog: TSaveDialog;
ToolBar: TToolBar;
tbGoto: TToolButton;
tbRefresh: TToolButton;
tbExport: TToolButton;
N1: TToolButton;
N2: TToolButton;
N3: TToolButton;
tbHelp: TToolButton;
XMLPropStorage: TXMLPropStorage;
procedure acExportExecute(Sender: TObject);
procedure acGotoExecute(Sender: TObject);
procedure acHelpExecute(Sender: TObject);
procedure FormCloseQuery(Sender: TObject; var {%H-}CanClose: boolean);
procedure FormCreate(Sender: TObject);
procedure FormKeyDown(Sender: TObject; var Key: Word; {%H-}Shift:TShiftState);
procedure DoUpdateToDos(Sender: TObject);
procedure lvTodoClick(Sender: TObject);
procedure lvTodoCompare(Sender : TObject; Item1, Item2 : TListItem;
{%H-}Data : Integer; var Compare : Integer);
procedure SaveDialogShow(Sender: TObject);
procedure XMLPropStorageRestoreProperties(Sender: TObject);
procedure XMLPropStorageRestoringProperties(Sender: TObject);
private
FBaseDirectory: string;
FUpdating, FUpdateNeeded: Boolean;
FIDEItem: string;
FIdleConnected: boolean;
FLoadingOptions: boolean;
FStartFilename: String;
FOwnerProjPack: TObject; // Project or package owning the FStartFilename.
FOnOpenFile : TOnOpenFile;
FScannedFiles: TAvlTree;// tree of TTLScannedFile
FScannedIncFiles: TStringMap;
procedure SetIDEItem(AValue: string);
procedure SetIdleConnected(const AValue: boolean);
function ProjectOpened(Sender: TObject; AProject: TLazProject): TModalResult;
procedure UpdateStartFilename;
procedure ResolveIDEItem(out CurOwner: TObject; out CurProject: TLazProject;
out CurPkg: TIDEPackage);
procedure AddListItem(aTodoItem: TTodoItem);
procedure ScanFile(aFileName : string);
procedure OnIdle(Sender: TObject; var {%H-}Done: Boolean);
public
constructor Create(AOwner: TComponent); override;
destructor Destroy; override;
procedure UpdateTodos(Immediately: boolean = false);
property IDEItem: string read FIDEItem write SetIDEItem; // package name or empty for active project
property BaseDirectory: string read FBaseDirectory;
property OnOpenFile: TOnOpenFile read FOnOpenFile write FOnOpenFile;
property IdleConnected: boolean read FIdleConnected write SetIdleConnected;
end;
var
IDETodoWindow: TIDETodoWindow;
implementation
{$R *.lfm}
const
DefaultTodoListCfgFile = 'todolistoptions.xml';
{ TIDETodoWindow }
constructor TIDETodoWindow.Create(AOwner: TComponent);
begin
inherited Create(AOwner);
if Name<>ToDoWindowName then
RaiseGDBException('');
ToolBar.Images := IDEImages.Images_16;
acGoto.ImageIndex := IDEImages.LoadImage('menu_goto_line');
acRefresh.ImageIndex := IDEImages.LoadImage('laz_refresh');
acExport.ImageIndex := IDEImages.LoadImage('menu_saveas');
acHelp.ImageIndex := IDEImages.LoadImage('btn_help');
SaveDialog.Filter:= dlgFilterCsv+'|*.csv';
LazarusIDE.AddHandlerOnProjectOpened(@ProjectOpened);
end;
destructor TIDETodoWindow.Destroy;
begin
FScannedFiles.FreeAndClear;
FreeAndNil(FScannedFiles);
FreeAndNil(FScannedIncFiles);
inherited Destroy;
end;
procedure TIDETodoWindow.UpdateTodos(Immediately: boolean);
var
i: integer;
St : String;
Node: TAvlTreeNode;
CurFile: TTLScannedFile;
Units: TStrings;
CurProject: TLazProject;
CurPkg: TIDEPackage;
Flags: TFindUnitsOfOwnerFlags;
begin
if FLoadingOptions then
exit;
if not Immediately then
begin
FUpdateNeeded:=true;
IdleConnected:=true;
exit;
end;
FUpdateNeeded:=false;
if FUpdating or (FOwnerProjPack=nil) then
Exit;
LazarusIDE.SaveSourceEditorChangesToCodeCache(nil);
Screen.BeginWaitCursor;
lvTodo.BeginUpdate;
Units:=nil;
try
FUpdating:=True;
CodeToolBoss.ActivateWriteLock;
FScannedFiles.FreeAndClear;
FScannedIncFiles.Clear;
lvTodo.Items.Clear;
if FStartFilename<>'' then begin
// Find a '.todo' file of the main source
St:=ChangeFileExt(FStartFilename,'.todo');
if FileExistsCached(St) then
ScanFile(St);
// Scan main source file
if FilenameIsPascalUnit(FStartFilename) then
ScanFile(FStartFilename);
end;
Flags:=[];
if chkListed.Checked then
Include(Flags, fuooListed);
if chkUsed.Checked then
Include(Flags, fuooUsed);
if chkPackages.Checked then
Include(Flags, fuooPackages);
if chkSourceEditor.Checked then
Include(Flags, fuooSourceEditor);
Units:=LazarusIDE.FindUnitsOfOwner(FOwnerProjPack,Flags);
for i:=0 to Units.Count-1 do
ScanFile(Units[i]);
Node:=FScannedFiles.FindLowest;
while Node<>nil do
begin
CurFile:=TTLScannedFile(Node.Data);
for i:=0 to CurFile.Count-1 do
AddListItem(CurFile[i]);
Node:=FScannedFiles.FindSuccessor(Node);
end;
finally
Units.Free;
CodeToolBoss.DeactivateWriteLock;
lvTodo.EndUpdate;
Screen.EndWaitCursor;
FUpdating:=False;
end;
end;
procedure TIDETodoWindow.FormKeyDown(Sender: TObject; var Key: Word; Shift: TShiftState);
begin
if (Key=VK_ESCAPE) then
ModalResult:=mrCancel;
end;
procedure TIDETodoWindow.DoUpdateToDos(Sender: TObject);
begin
UpdateTodos;
end;
procedure TIDETodoWindow.lvTodoClick(Sender: TObject);
begin
acGoto.Execute;
end;
procedure TIDETodoWindow.lvTodoCompare(Sender : TObject;
Item1, Item2 : TListItem; Data : Integer; var Compare : Integer);
var
Str1: String;
Str2: String;
Int1: Integer;
Int2: Integer;
begin
Case lvTodo.SortColumn of
0, 1, 3, 5, 6 :
begin
if lvTodo.SortColumn = 0 then
begin
Str1 := TListItem(Item1).Caption;
Str2 := TListItem(Item2).Caption;
end else
begin
// Checks against Subitems.Count necessary??
if lvTodo.SortColumn <= Item1.SubItems.Count then
Str1 := Item1.SubItems.Strings[lvTodo.SortColumn-1]
else Str1 := '';
if lvTodo.SortColumn <= Item2.SubItems.Count then
Str2 := Item2.SubItems.Strings[lvTodo.SortColumn-1]
else Str2 := '';
end;
Compare := AnsiCompareText(Str1, Str2);
end;
2, 4 :
begin
if TryStrToInt(Item1.SubItems.Strings[lvTodo.SortColumn-1], Int1)
and TryStrToInt(Item2.SubItems.Strings[lvTodo.SortColumn-1], Int2) then
Compare := CompareValue(Int1, Int2)
else Compare := 0;
end;
else Compare := 0;
end;
if lvTodo.SortDirection = sdDescending then Compare := -Compare;
end;
procedure TIDETodoWindow.SaveDialogShow(Sender: TObject);
begin
SaveDialog.InitialDir:=GetCurrentDirUTF8;
end;
procedure TIDETodoWindow.XMLPropStorageRestoreProperties(Sender: TObject);
begin
FLoadingOptions := False;
UpdateTodos;
end;
procedure TIDETodoWindow.XMLPropStorageRestoringProperties(Sender: TObject);
begin
FLoadingOptions := True;
end;
function TIDETodoWindow.ProjectOpened(Sender: TObject; AProject: TLazProject): TModalResult;
begin
Result:=mrOK;
IDEItem:='';
UpdateTodos;
end;
procedure TIDETodoWindow.UpdateStartFilename;
var
NewStartFilename: String;
CurProject: TLazProject;
CurPkg: TIDEPackage;
begin
ResolveIDEItem(FOwnerProjPack,CurProject,CurPkg);
NewStartFilename:='';
if CurPkg<>nil then // package
NewStartFilename:=CurPkg.Filename
else if CurProject<>nil then // project
NewStartFilename:=CurProject.ProjectInfoFile;
if FStartFilename=NewStartFilename then
exit;
FStartFilename:=NewStartFilename;
UpdateTodos;
end;
procedure TIDETodoWindow.ResolveIDEItem(out CurOwner: TObject;
out CurProject: TLazProject; out CurPkg: TIDEPackage);
begin
CurOwner:=nil;
CurProject:=nil;
CurPkg:=nil;
if IsValidIdent(FIDEItem,true,true) then begin
// package
CurPkg:=PackageEditingInterface.FindPackageWithName(FIDEItem);
CurOwner:=CurPkg;
//DebugLn(['TIDETodoWindow.ResolveIDEItem: Found package ', CurPkg.Filename]);
end else begin
// project
CurProject:=LazarusIDE.ActiveProject;
CurOwner:=CurProject;
//DebugLn(['TIDETodoWindow.ResolveIDEItem: Found project ', CurProject.MainFile.Filename]);
end;
end;
procedure TIDETodoWindow.SetIdleConnected(const AValue: boolean);
begin
if FIdleConnected=AValue then exit;
FIdleConnected:=AValue;
if IdleConnected then
Application.AddOnIdleHandler(@OnIdle)
else
Application.RemoveOnIdleHandler(@OnIdle);
end;
procedure TIDETodoWindow.SetIDEItem(AValue: string);
begin
//if FIDEItem=AValue then exit; // No check, trigger update in any case.
FIDEItem:=AValue;
UpdateStartFilename;
end;
procedure TIDETodoWindow.FormCreate(Sender: TObject);
begin
FUpdating := False;
FScannedFiles := TAvlTree.Create(@CompareTLScannedFiles);
FScannedIncFiles := TStringMap.Create(False);
Caption := lisToDoList;
acRefresh.Hint := lisTodolistRefresh;
acGoto.Hint := listodoListGotoLine;
acExport.Hint := rsExportTodoIt;
acRefresh.Caption := dlgUnitDepRefresh;
acGoto.Caption := lisToDoGoto;
acExport.Caption := lisToDoExport;
acHelp.Caption := lisHelp;
lblOptions.Caption := lisOptions;
chkListed.Caption := lisToDoListed;
chkListed.Hint := lisToDoListedHint;
chkUsed.Caption := lisToDoUsed;
chkUsed.Hint := lisToDoUsedHint;
chkPackages.Caption := lisPackages;
chkPackages.Hint := Format(lisPackagesHint, [lisToDoListed, lisToDoUsed]);
chkSourceEditor.Caption := lisSourceEditor;
chkSourceEditor.Hint := lisSourceEditorHint;
with cboShowWhat do
begin
Items[0] := lisFilterItem0;
Items[1] := lisFilterItem1;
Items[2] := lisFilterItem2;
Items[3] := lisFilterItem3;
Items[4] := lisFilterItem4;
Items[5] := lisFilterItem5;
Items[6] := lisFilterItem6;
Hint := lisShowWhatHint;
end;
lblShowWhat.Caption:=lisShowWhat;
with lvTodo do
begin
Column[0].Caption := lisToDoLType;
Column[1].Caption := lisToDoLDescription;
Column[1].Width := 700;
Column[2].Caption := lisToDoLPriority;
Column[3].Caption := lisToDoLFile;
Column[4].Caption := lisToDoLLine;
Column[5].Caption := lisToDoLOwner;
Column[6].Caption := listToDoLCategory;
end;
XMLPropStorage.FileName := Concat(AppendPathDelim(LazarusIDE.GetPrimaryConfigPath),
DefaultTodoListCfgFile);
XMLPropStorage.Active := True;
end;
procedure TIDETodoWindow.acGotoExecute(Sender: TObject);
var
CurFilename: String;
aTodoItem: TTodoItem;
aListItem: TListItem;
TheLine: integer;
begin
CurFilename:='';
aListItem:= lvtodo.Selected;
if Assigned(aListItem) and Assigned(aListItem.Data) then
begin
aTodoItem := TTodoItem(aListItem.Data);
CurFileName := aTodoItem.Filename;
TheLine := aTodoItem.LineNumber;
if Assigned(OnOpenFile) then
OnOpenFile(Self,CurFilename,TheLine)
else
LazarusIDE.DoOpenFileAndJumpToPos(CurFilename,Point(1,TheLine),-1,-1,-1,
[ofOnlyIfExists,ofRegularFile,ofVirtualFile,ofDoNotLoadResource]);
end;
end;
procedure TIDETodoWindow.acHelpExecute(Sender: TObject);
begin
// usual API from IdeHelpIntf don't work
OpenURL('https://wiki.freepascal.org/IDE_Window:_ToDo_List');
end;
procedure TIDETodoWindow.acExportExecute(Sender: TObject);
begin
SaveDialog.FileName:='TodoList_'+FormatDateTime('YYYY_MM_DD',now);
if SaveDialog.Execute then
ExtractToCSV(SaveDialog.FileName, lvTodo.Items);
end;
procedure TIDETodoWindow.FormCloseQuery(Sender: TObject; var CanClose: boolean);
begin
XMLPropStorage.Save;
end;
procedure TIDETodoWindow.AddListItem(aTodoItem: TTodoItem);
function ShowThisToDoItem:boolean;
// Add this ToDoItem based on the cboShowWhat selection
begin
case cboShowWhat.ItemIndex of
0:Result := True;
1..3: Result := (TToDoType(cboShowWhat.ItemIndex - 1) = aTodoItem.ToDoType);
4:Result := aTodoItem.ToDoType in [tdToDo, tdDone];
5:Result := aTodoItem.ToDoType in [tdToDo, tdNote];
6:Result := aTodoItem.ToDoType in [tdDone, tdNote];
end;
end;
var
aListItem: TListItem;
aFilename: String;
begin
if Assigned(aTodoItem) and ShowThisToDoItem then
begin
//DebugLn(['TIDETodoWindow.AddListItem ',aTodoItem.Filename,' ',aTodoItem.LineNumber]);
aListitem := lvTodo.Items.Add;
aListitem.Data := aTodoItem;
aListItem.Caption := LIST_INDICATORS[aTodoItem.ToDoType];
aListitem.SubItems.Add(aTodoItem.Text);
aListitem.SubItems.Add(IntToStr(aTodoItem.Priority));
aFilename:=aTodoItem.Filename;
if (BaseDirectory<>'') and FilenameIsAbsolute(aFilename) then
aFilename:=CreateRelativePath(aFilename,BaseDirectory);
aListitem.SubItems.Add(aFilename);
aListitem.SubItems.Add(IntToStr(aTodoItem.LineNumber));
aListitem.SubItems.Add(aTodoItem.Owner);
aListitem.SubItems.Add(aTodoItem.Category);
end;
end;
procedure TIDETodoWindow.ScanFile(aFileName: string);
begin
ToDoListCore.ScanFile(aFileName, FScannedFiles, FScannedIncFiles);
end;
procedure TIDETodoWindow.OnIdle(Sender: TObject; var Done: Boolean);
begin
IdleConnected:=false;
UpdateTodos(true);
end;
end.