📄 packageinformation.pas
字号:
{-----------------------------------------------------------------------------
The contents of this file are subject to the Mozilla Public License
Version 1.1 (the "License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.mozilla.org/MPL/MPL-1.1.html
Software distributed under the License is distributed on an "AS IS" basis,
WITHOUT WARRANTY OF ANY KIND, either expressed or implied. See the License for
the specific language governing rights and limitations under the License.
The Original Code is: PackageInformation.pas, released on 2004-05-17.
The Initial Developer of the Original Code is Andreas Hausladen
(Andreas dott Hausladen att gmx dott de)
Portions created by Andreas Hausladen are Copyright (C) 2004 Andreas Hausladen.
All Rights Reserved.
Contributor(s): -
You may retrieve the latest version of this file at the Project JEDI's JVCL
home page, located at http://jvcl.sourceforge.net
Known Issues:
-----------------------------------------------------------------------------}
// $Id: PackageInformation.pas,v 1.10 2004/12/23 00:23:58 ahuser Exp $
unit PackageInformation;
{$I jvcl.inc}
interface
uses
SysUtils, Classes, Contnrs,
JvSimpleXml;
type
{ xml Package files }
TPackageXmlInfo = class;
TRequiredPackage = class;
TContainedFile = class;
TPackageGroup = class;
/// <summary>
/// TPackageXmlInfoItem contains common parts of TRequiredPackage and
/// TContainedFile.
/// </summary>
TPackageXmlInfoItem = class(TObject)
private
FName: string;
FTargets: TStrings;
FCondition: string;
public
constructor Create(const AName, ATargets, ACondition: string);
destructor Destroy; override;
function IsIncluded(const TargetSymbol: string): Boolean;
{ IsIncluded() returns True if the item has TargetSymbol in it's Targets
list. }
property Name: string read FName;
property Targets: TStrings read FTargets;
property Condition: string read FCondition;
end;
/// <summary>
/// TRequiredPackage contains one package that is requried by a TPackageXmlInfo
/// object and it's inclusion conditions.
/// </summary>
TRequiredPackage = class(TPackageXmlInfoItem)
public
function IsRequiredByTarget(const TargetSymbol: string): Boolean;
function GetBplName(PackageGroup: TPackageGroup): string;
end;
/// <summary>
/// TContainedFile contains one file name that is contained in the
/// TPackageXmlInfo object and it's inclusion conditions.
/// </summary>
TContainedFile = class(TPackageXmlInfoItem)
private
FFormName: string;
public
constructor Create(const AName, ATargets, AFormName, ACondition: string);
function IsUsedByTarget(const TargetSymbol: string): Boolean;
property FormName: string read FFormName;
end;
/// <summary>
/// TPackageXmlInfo contains the generic .xml file for a bpl target.
/// </summary>
TPackageXmlInfo = class(TObject)
private
FFilename: string;
FName: string;
FDisplayName: string;
FDescription: string;
FClxDescription: string;
FRequires: TObjectList;
FContains: TObjectList;
FRequiresDB: Boolean;
FIsDesign: Boolean;
FIsXPlatform: Boolean;
FC5PFlags: string;
FC6PFlags: string;
FC5Libs: TStrings;
FC6Libs: TStrings;
FGUID: string;
function GetContainCount: Integer;
function GetContains(Index: Integer): TContainedFile;
function GetRequireCount: Integer;
function GetRequires(Index: Integer): TRequiredPackage;
procedure LoadFromFile(const Filename: string);
public
constructor Create(const AFilename: string);
destructor Destroy; override;
property Filename: string read FFilename;
property Name: string read FName; // "PackageName-"[R|D]
property DisplayName: string read FDisplayName; // "PackageName"
property Description: string read FDescription;
property ClxDescription: string read FClxDescription;
property RequireCount: Integer read GetRequireCount;
property Requires[Index: Integer]: TRequiredPackage read GetRequires;
property ContainCount: Integer read GetContainCount;
property Contains[Index: Integer]: TContainedFile read GetContains;
property RequiresDB: Boolean read FRequiresDB;
property IsDesign: Boolean read FIsDesign;
property IsXPlaform: Boolean read FIsXPlatform;
property C5PFlags: string read FC5PFlags;
property C6PFlags: string read FC6PFlags;
property C5Libs: TStrings read FC5Libs;
property C6Libs: TStrings read FC6Libs;
property GUID: string read FGUID; // D9 support
end;
{ Package Group }
TPackageInfo = class;
/// <summary>
/// TBpgPackageTarget contains a .bpl target and the .xml file in the
/// Info property.
/// </summary>
TBpgPackageTarget = class(TObject)
private
FOwner: TPackageGroup;
FUserData: TObject;
FTargetName: string;
FSourceName: string;
FInfo: TPackageInfo;
FRequireList: TList;
FContaineList: TList;
FAutoDeleteUserData: Boolean;
function GetRelSourceDir: string;
function GetSourceDir: string;
function GetContainCount: Integer;
function GetContains(Index: Integer): TContainedFile;
function GetRequireCount: Integer;
function GetRequires(Index: Integer): TRequiredPackage;
protected
procedure UpdateContainList; virtual;
procedure UpdateRequireList; virtual;
procedure GetDependencies; virtual; // is called after alle package targets are created
public
constructor Create(AOwner: TPackageGroup; const ATargetName, ASourceName: string); virtual;
destructor Destroy; override;
function FindRuntimePackage: TBpgPackageTarget;
property TargetName: string read FTargetName;
property SourceName: string read FSourceName;
property SourceDir: string read GetSourceDir;
property RelSourceDir: string read GetRelSourceDir;
property Info: TPackageInfo read FInfo;
// In contrast to Info.Xxx these properties only returns the
// required/contained for this target.
property RequireCount: Integer read GetRequireCount;
property Requires[Index: Integer]: TRequiredPackage read GetRequires;
property ContainCount: Integer read GetContainCount;
property Contains[Index: Integer]: TContainedFile read GetContains;
property Owner: TPackageGroup read FOwner;
property UserData: TObject read FUserData write FUserData;
property AutoDeleteUserData: Boolean read FAutoDeleteUserData write FAutoDeleteUserData default True;
end;
TBpgPackageTargetClass = class of TBpgPackageTarget;
/// <summary>
/// TPackageGroup contains the data from a .bpg (Borland Package Group) file.
/// </summary>
TPackageGroup = class(TObject)
private
FPackages: TObjectList;
FFilename: string;
FPackagesXmlDir: string;
FTargetSymbol: string;
function GetCount: Integer;
function GetPackages(Index: Integer): TBpgPackageTarget;
function GetBpgName: string;
function Add(const TargetName, SourceName: string): TBpgPackageTarget;
procedure LoadBDSGroupFile;
procedure LoadBPGFile;
protected
function GetIsVCLX: Boolean; virtual;
function GetPackageTargetClass: TBpgPackageTargetClass; virtual;
procedure LoadFile;
public
constructor Create(const AFilename, APackagesXmlDir, ATargetSymbol: string);
{ Set AFilename to '' if you want a PackageGroup instance that does not
own the TBpgPackageTarget objects. }
destructor Destroy; override;
procedure AddPackage(Pkg: TBpgPackageTarget);
function FindPackageByXmlName(const XmlName: string): TBpgPackageTarget;
{ FindPackageByXmlName returns the TBpgPackageTarget object that contains
the specified .xml file. }
function GetBplNameOf(Package: TRequiredPackage): string; virtual;
property Count: Integer read GetCount;
property Packages[Index: Integer]: TBpgPackageTarget read GetPackages; default;
property BpgName: string read GetBpgName;
property Filename: string read FFilename;
property IsVCLX: Boolean read GetIsVCLX;
property PackagesXmlDir: string read FPackagesXmlDir;
property TargetSymbol: string read FTargetSymbol;
end;
/// <summary>
/// TPackageInfo is a wrapper for TPackageXmlInfo objects that contains the
/// generic .xml file for a bpl target.
/// </summary>
TPackageInfo = class(TObject)
private
FOwner: TBpgPackageTarget;
FXmlInfo: TPackageXmlInfo;
FXmlDir: string;
function GetRequireCount: Integer;
function GetRequires(Index: Integer): TRequiredPackage;
function GetContainCount: Integer;
function GetContains(Index: Integer): TContainedFile;
function GetBplName: string;
function GetDescription: string;
function GetDisplayName: string;
function GetIsDesign: Boolean;
function GetName: string;
function GetRequiresDB: Boolean;
public
constructor Create(AOwner: TBpgPackageTarget; const AXmlDir: string);
property Name: string read GetName; // "PackageName-"[R|D]
property DisplayName: string read GetDisplayName; // "PackageName"
property BplName: string read GetBplName; // "PackageName"[D|C][5-7][R|D]
property Description: string read GetDescription;
property RequiresDB: Boolean read GetRequiresDB;
property RequireCount: Integer read GetRequireCount;
property Requires[Index: Integer]: TRequiredPackage read GetRequires;
property ContainCount: Integer read GetContainCount;
property Contains[Index: Integer]: TContainedFile read GetContains;
property IsDesign: Boolean read GetIsDesign;
property Owner: TBpgPackageTarget read FOwner;
property XmlDir: string read FXmlDir;
end;
var
BplNameToGenericNameHook: function(const BplName: string): string = nil;
ExpandPackageTargets: procedure(Targets: TStrings) = nil;
ExpandPackageTargetsObj: procedure(Targets: TStrings) of object = nil;
function BplNameToGenericName(const BplName: string): string;
{ BplNameToGenericName converts a "JvCoreD7D.XXX" to "JvCore-D" }
function GetPackageXmlInfo(const BplName, XmlDir: string): TPackageXmlInfo; overload;
{ returns a cached TPackageXmlInfo instance. }
function GetPackageXmlInfo(const XmlFilename: string): TPackageXmlInfo; overload;
{ returns a cached TPackageXmlInfo instance. }
implementation
{$IFDEF COMPILER5}
const
PathDelim = '\';
{$ENDIF COMPILER5}
var
XmlFileCache: TStringList; // cache for .xml files ( TPackageXmlInfo )
function BplNameToGenericName(const BplName: string): string;
begin
if Assigned(BplNameToGenericNameHook) then
Result := BplNameToGenericNameHook(BplName)
else
begin
// obtain package name used in the xml file
Result := ChangeFileExt(BplName, '');
Delete(Result, Length(Result) - 2, 2);
if Length(Result) > 2 then
Insert('-', Result, Length(Result)); // do not localize
end;
end;
procedure ExpandTargets(Targets: TStrings);
begin
if Assigned(ExpandPackageTargetsObj) then
ExpandPackageTargetsObj(Targets);
if Assigned(ExpandPackageTargets) then
ExpandPackageTargets(Targets);
end;
/// <summary>
/// GetPackageXmlInfo returns a cached TPackageXmlInfo instance.
/// </summary>
function GetPackageXmlInfo(const BplName, XmlDir: string): TPackageXmlInfo; overload;
var
Index: Integer;
Name: string;
begin
Name := XmlDir + PathDelim + BplNameToGenericName(BplName) + '.xml';
// already in the cache
if XmlFileCache.Find(Name, Index) then
Result := TPackageXmlInfo(XmlFileCache.Objects[Index])
else
begin
// create a new one and add it to the cache
Result := TPackageXmlInfo.Create(Name); // do not localize
XmlFileCache.AddObject(Name, Result);
end;
end;
function GetPackageXmlInfo(const XmlFilename: string): TPackageXmlInfo; overload;
var
Index: Integer;
begin
if XmlFileCache.Find(XmlFilename, Index) then
Result := TPackageXmlInfo(XmlFileCache.Objects[Index])
else
begin
// create a new one and add it to the cache
Result := TPackageXmlInfo.Create(XmlFilename);
XmlFileCache.AddObject(XmlFilename, Result);
end;
end;
function StartsWith(const Text, StartText: string; CaseInsensitive: Boolean = False): Boolean;
var
Len, i: Integer;
begin
Result := False;
Len := Length(StartText);
if Len > Length(Text) then
Exit;
if CaseInsensitive then
begin
for i := 1 to Len do
if UpCase(Text[i]) <> UpCase(StartText[i]) then
Exit;
end
else
begin
for i := 1 to Len do
if Text[i] <> StartText[i] then
Exit;
end;
Result := True;
end;
function EndsWith(const Text, EndText: string; CaseInsensitive: Boolean): Boolean;
var
Len, i, x: Integer;
begin
Result := False;
Len := Length(EndText);
x := Length(Text);
if Len > x then
Exit;
if CaseInsensitive then
begin
for i := Len downto 1 do
if UpCase(Text[x]) <> UpCase(EndText[i]) then
Exit
else
Dec(x);
end
else
begin
for i := Len downto 1 do
if Text[x] <> EndText[i] then
Exit
else
Dec(x);
end;
Result := True;
end;
function CutFirstDirectory(var Dir: string): string;
var
ps: Integer;
begin
ps := Pos(PathDelim, Dir);
if ps > 0 then
begin
Result := Copy(Dir, 1, ps - 1);
Delete(Dir, 1, ps);
end
else
begin
Result := Dir;
Dir := '';
end;
end;
function FollowRelativeFilename(const RootDir: string; RelFilename: string): string;
var
Dir: string;
begin
Result := RootDir;
while RelFilename <> '' do
begin
Dir := CutFirstDirectory(RelFilename);
if Dir = '..' then
Result := ExtractFileDir(Result)
else if Dir = '.' then
Continue
else
Result := Result + PathDelim + Dir;
end;
end;
{$IFDEF COMPILER5}
type
UTF8String = type string;
{$ENDIF COMPILER5}
function LoadUtf8File(const Filename: string): string;
var
Content: UTF8String;
Stream: TFileStream;
begin
Stream := TFileStream.Create(Filename, fmOpenRead or fmShareDenyWrite);
try
SetLength(Content, Stream.Size);
Stream.Read(Content[1], Stream.Size);
finally
Stream.Free;
end;
Delete(Content, 1, 3); // This is a little bit dirty but unless we mix UTF8,
// UTF16 and ANSI files there is no problem.
{$IFDEF COMPILER6_UP}
Result := Utf8ToAnsi(Content);
{$ELSE}
{ Delphi 5 (should) never reachs this because the Installer uses the newest
installed Delphi version and only reads the project groups of installed
Delphi/BCB/BDN versions. }
Result := Content;
{$ENDIF COMPILIER6_UP}
end;
{ TPackageXmlInfoItem }
constructor TPackageXmlInfoItem.Create(const AName, ATargets, ACondition: string);
begin
inherited Create;
FName := AName;
FTargets := TStringList.Create;
TStringList(FTargets).Duplicates := dupIgnore;
FTargets.CommaText := ATargets;
ExpandTargets(FTargets);
TStringList(FTargets).Sorted := True; // sort the targets
FCondition := ACondition;
end;
destructor TPackageXmlInfoItem.Destroy;
begin
FTargets.Free;
inherited Destroy;
end;
function TPackageXmlInfoItem.IsIncluded(const TargetSymbol: string): Boolean;
var
Index: Integer;
begin
Result := TStringList(FTargets).Find(TargetSymbol, Index);
end;
{ TRequiredPackage }
function TRequiredPackage.GetBplName(PackageGroup: TPackageGroup): string;
begin
if PackageGroup = nil then
Result := Name
else
Result := PackageGroup.GetBplNameOf(Self);
end;
function TRequiredPackage.IsRequiredByTarget(const TargetSymbol: string): Boolean;
begin
Result := IsIncluded(TargetSymbol);
end;
{ TContainedFile }
constructor TContainedFile.Create(const AName, ATargets, AFormName,
ACondition: string);
begin
inherited Create(AName, ATargets, ACondition);
FFormName := AFormName;
end;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -