我正在尝试为Delphi 5编写一个插件,该插件将在DOF中存储我们测试exe的路径,以便在项目与该项目的测试之间存在直接关联。当我将自己的模块添加到DOF文件时,类似
[DUint Plugin]
IntegrationTestExe=Somepath
UnitTestExeList=Some;Path;
每当我去手动或通过代码添加此项目时,保存项目时,添加的行都会被删除。我认为这可能是因为IDE只是不允许在DOF中使用自定义模块。
但是,我们使用名为EurekaLog的第三方插件。 EurekaLog将其自己的变量注入到DOF中,当您保存时,这些变量不会被删除。我复制了很多代码,以便可以测试EurekaLog代码是否可以正常工作(通过魔术),但是他们的代码只是将其模块写入了DOF,而没有做其他特别的事情。
有人知道在EurekaLog中是如何做到的吗?我是否需要在某个地方注册我的模块,以便IDE知道不删除它?
答案 0 :(得分:0)
更新经过一些试验,似乎将设置保存到DOF实际上比将其保存到DSK文件更可靠。
将另一个TEdit添加到窗体并创建LoadDOFSettings和SaveDOFSettings 类似于现有的LoadSettings和SaveSettings,并在收到时调用它们 DesktopLoad和DesktoSave通知。无需通过Timer1事件调用SaveDOFSettings,因为DOF似乎没有发生重命名。
原始答案 我建议在阅读此答案之前,先做一个文件|在IDE中关闭全部, 创建一个新程序包,将下面的单元添加到其中,然后将其安装在IDE中。
该软件包的目的是双重的,首先是说明如何在DSK文件中保存自定义设置,其次是让您了解有关项目的哪些事件信息 您可以通过ToolsAPI单元中的服务从IDE中获取文件。
安装软件包后,请密切注意其表单,该表单会在您打开,处理和关闭项目时在上方的备忘录中显示文件通知。有 需要注意的几件事:
打开项目时,收到的最后一条通知是有关其DSK文件已打开的消息。
并非每种文件类型都是通知的主题。特别是,您不会收到任何专门针对DOF文件的通知,因此,如果您要写入该文件并稍后再读取该文件,则必须做出有关何时安全(或不安全)的假设,这就是可能是为什么您遇到了您要问的问题。
在项目上执行“全部关闭”时,收到通知的最后文件更改是正在写入的DSK。要注意的是,它最初是写入到具有相同名称但扩展名为。$$$的文件中。此后不久,但您无法确切地说,这个。$$$文件被重命名为.DSK。
由下面的代码创建的表单具有一个编辑框,DSK文件的edMyValue' which can be used to set a value in a section of the DSK file called
MySettings and which is reloaded the next time the project is opened. The writing of the
MySettings`部分由TTimer触发,延迟2秒以使IDE时间按照我的描述编写并重命名DSK文件。显然,这为比赛条件提供了机会。
您可能想参考
http://www.gexperts.org/open-tools-api-faq/#dsk
(GExperts是IDE加载项工具,自Delphi成立以来就一直存在)
本文的部分讨论当前项目的.DSK文件。像DOF一样,它是INI文件格式,具有类似
的部分[Closed Files]
[Modules]
[EditWindow0]
[View0]
您会看到它说
检查ofnProjectDesktopLoad和ofnProjectDesktopSave NotifyCode值。当您看到其中之一时,可以使用TIniFile等类从FileName参数指示的文件中保存/加载值。
由于重命名业务,也许比文章建议的要难一些。
玩得开心!
unit IDEEventsu;
interface
uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, StdCtrls, ExtCtrls, ToolsAPI, Grids, IniFiles;
type
TFileEventsForm = class(TForm)
Panel1: TPanel;
Memo1: TMemo;
edMyValue: TEdit;
btnClear: TButton;
Timer1: TTimer;
Memo2: TMemo;
procedure btnClearClick(Sender: TObject);
procedure Timer1Timer(Sender: TObject);
private
function GetCurrentProject: IOTAProject;
public
// The following are using interfaces accessible via the ToolsAPI
Services: IOTAServices;
ProjectGroup : IOTAProjectGroup;
Project: IOTAProject;
Options : IOTAProjectOptions;
ModServices: IOTAModuleServices;
Module: IOTAModule;
NotifierIndex: Integer; // This is used to disconnect our notifier from the IDE
IsSetUp : Boolean;
SetUpCount : Integer;
DskFileName : String;
procedure SetUp;
procedure SaveSettings;
procedure LoadSettings;
end;
var
FileEventsForm: TFileEventsForm;
procedure Register;
[...]
uses
typinfo;
type
TIdeNotifier = class(TNotifierObject, IOTANotifier, IOTAIDENotifier)
// This is the class we use to receive file notication events from the IDE via the
// interfaces in ToolsAPI.Pas
//
// It needs to implement the IOTANotifier and IOTAIDENotifier interfaces and,
// once registered with the IDE, the IDE calls its methods as a kind of call-back
// mechanism so that it gets notified of file events
//
// Note that this file also provides a form for displaying the received event
// notifications and that the IOTANotifier and IOTAIDENotifier interfaces could
// just as easily be implemented by the form itself
protected
procedure AfterCompile(Succeeded: Boolean);
procedure BeforeCompile(const Project: IOTAProject; var Cancel: Boolean);
procedure FileNotification(NotifyCode: TOTAFileNotification;
const FileName: string; var Cancel: Boolean);
end;
procedure Register;
// This is necessary to register the package in the IDE
var
Notifier : TIdeNotifier;
begin
FileEventsForm:= TFileEventsForm.Create(Nil);
FileEventsForm.Services := BorlandIDEServices as IOTAServices;
Notifier := TIdeNotifier.Create;
Notifier.Form := FileEventsForm;
FileEventsForm.NotifierIndex := FileEventsForm.Services.AddNotifier(TIdeNotifier.Create);
end;
procedure CloseDown;
begin
FileEventsForm.Services.RemoveNotifier(FileEventsForm.NotifierIndex);
FileEventsForm.Close;
FileEventsForm.Free;
end;
function NotifyCodeString(NotifyCode : TOTAFileNotification) : String;
begin
Result := Copy(GetEnumName(TypeInfo(TOTAFileNotification), Ord(NotifyCode)), 4, MaxInt);
end;
procedure TIdeNotifier.AfterCompile(Succeeded: Boolean);
begin
end;
procedure TIdeNotifier.BeforeCompile(const Project: IOTAProject; var Cancel: Boolean);
begin
end;
procedure TIdeNotifier.FileNotification(NotifyCode: TOTAFileNotification;
const FileName: string; var Cancel: Boolean);
begin
if True {NotifyCode in [ofnProjectDesktopLoad, ofnActiveProjectChanged]} then begin
FileEventsForm.Show;
FileEventsForm.Memo1.Lines.Add(Format('%s file: %s', [NotifyCodeString(NotifyCode), FileName]));
case NotifyCode of
ofnProjectDesktopLoad,
ofnDefaultDesktopLoad : begin
FileEventsForm.DskFileName := FileName;
FileEventsForm.LoadSettings;
end;
ofnProjectDesktopSave,
ofnDefaultDesktopSave : begin
if True{CompareText(ExtractFileExt(FileName), '.DSK') = 0} then begin
FileEventsForm.Caption := FileName;
FileEventsForm.Timer1.Enabled := True; // causes DSK file to be updated after Timer1.Interval (=2000ms)
end;
end;
end; { case }
end;
end;
procedure TFileEventsForm.btnClearClick(Sender: TObject);
begin
Memo1.Lines.Clear;
end;
function TFileEventsForm.GetCurrentProject: IOTAProject;
var
i: Integer;
begin
Result := nil;
ModServices := BorlandIDEServices as IOTAModuleServices;
for i := 0 to ModServices.ModuleCount - 1 do
begin
Module := ModServices.Modules[i];
if Supports(Module, IOTAProjectGroup, ProjectGroup) then begin
Result := ProjectGroup.ActiveProject;
Options := Result.ProjectOptions;
Exit;
end
else if Supports(Module, IOTAProject, Project) then
begin // In the case of unbound packages, return the 1st
if Result = nil then begin
Result := Project;
Options := Result.ProjectOptions;
end;
end;
end;
end;
procedure TFileEventsForm.SetUp;
begin
Project := GetCurrentProject;
Inc(SetUpCount);
Caption := 'Setup done ' + IntToStr(SetUpCount);
IsSetUp := True;
end;
procedure TFileEventsForm.LoadSettings;
var
Ini : TMemIniFile;
S : String;
begin
Ini := TMemIniFile.Create(DSKFileName);
try
S := Ini.ReadString('MySettings', 'Name', 'no value');
edMyValue.Text := S;
finally
Ini.Free;
end;
end;
procedure TFileEventsForm.SaveSettings;
var
Ini : TMemIniFile;
S : String;
begin
S := DSKFileName;
Caption := 'Saving: ' + S;
Ini := TMemIniFile.Create(S);
try
Ini.WriteString('MySettings', 'Name', edMyValue.Text);
Ini.UpdateFile;
Ini.ReadSections(Memo2.Lines);
Memo2.Lines.Add('This file : ' + DSKFileName);
edMyValue.Text := '?';
finally
Ini.Free;
end;
end;
procedure TFileEventsForm.Timer1Timer(Sender: TObject);
begin
Timer1.Enabled := False;
SaveSettings;
end;
initialization
finalization
CloseDown;
end.