doublecmd/src/uwdxmodule.pas
2024-02-23 12:17:17 +03:00

1505 lines
43 KiB
ObjectPascal

{
Double Commander
-------------------------------------------------------------------------
WDX-API implementation.
(TC WDX-API v1.5)
Copyright (C) 2008 Dmitry Kolomiets (B4rr4cuda@rambler.ru)
Copyright (C) 2008-2024 Alexander Koblov (alexx2000@mail.ru)
Some ideas were found in sources of WdxGuide by Alexey Torgashin
and SuperWDX by Pavel Dubrovsky and Dmitry Vorotilin.
This program 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 program 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.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
}
unit uWDXModule;
{$mode delphi}{$H+}
interface
uses
//Lazarus, Free-Pascal, etc.
Classes, SysUtils, dynlibs,
//DC
uLng, uWdxPrototypes, WdxPlugin, uDetectStr, lua, uFile, DCXmlConfig,
uExtension;
const
WDX_MAX_LEN = 2048;
type
{ TWdxField }
TWdxField = class
private
OUnits: String; // Units (original)
public
FName: String; // Field name (english)
LName: String; // Field name (localized)
FType: Integer;
FUnits: TStringArray; // Units (english)
LUnits: TStringArray; // Units (localized)
function GetUnitIndex(UnitName: String): Integer;
end;
{ TWDXModule }
TWDXModule = class(TDcxModule)
private
FFieldsList: TStringList;
FParser: TParserControl;
protected
FFileName: String;
FMutex: TRTLCriticalSection;
protected
procedure Translate;
function GetADetectStr: String; virtual;
procedure SetADetectStr(const AValue: String); virtual;
procedure AddField(const AName, AUnits: String; AType: Integer);
protected
function GetAName: String; virtual; abstract;
function GetAFileName: String; virtual; abstract;
procedure SetAName(AValue: String); virtual; abstract;
procedure SetAFileName(AValue: String); virtual; abstract;
public
//---------------------
constructor Create; virtual;
destructor Destroy; override;
//---------------------
function LoadModule: Boolean; virtual; abstract;
procedure UnloadModule; virtual; abstract;
function IsLoaded: Boolean; virtual; abstract;
//---------------------
function FieldList: TStringList; virtual;
function WdxFieldType(n: Integer): String;
function GetFieldIndex(FieldName: String): Integer; virtual;
function FileParamVSDetectStr(const aFile: TFile): Boolean; virtual;
//------------------------------------------------------
procedure CallContentGetSupportedField; virtual; abstract;
procedure CallContentSetDefaultParams; virtual; abstract;
procedure CallContentStopGetValue(FileName: String); virtual; abstract;
//---------------------
function CallContentGetDefaultSortOrder(FieldIndex: Integer): Boolean; virtual; abstract;
function CallContentGetDetectString: String; virtual; abstract;
function CallContentGetValueV(FileName: String; FieldName: String; UnitName: String; flags: Integer): Variant; overload; virtual;
function CallContentGetValueV(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): Variant; overload; virtual; abstract;
function CallContentGetValue(FileName: String; FieldName: String; UnitName: String; flags: Integer): String; overload; virtual;
function CallContentGetValue(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): String; overload; virtual; abstract;
function CallContentGetValue(FileName: String; FieldIndex: Integer; var UnitIndex: Integer): String; overload; virtual; abstract;
function CallContentGetSupportedFieldFlags(FieldIndex: Integer): Integer; virtual; abstract;
{ContentSetValue
ContentEditValue
ContentSendStateInformation}
//------------------------------------------------------
property Name: String read GetAName write SetAName;
property FileName: String read GetAFileName write SetAFileName;
property DetectStr: String read GetADetectStr write SetADetectStr;
//---------------------
end;
{ TPluginWDX }
TPluginWDX = class(TWDXModule)
protected
FForce: Boolean;
FName: String;
protected
function GetAName: String; override;
function GetAFileName: String; override;
procedure SetAName(AValue: String); override;
procedure SetAFileName(AValue: String); override;
protected
//a) Mandatory (must be implemented)
ContentGetSupportedField: TContentGetSupportedField;
ContentGetValue: TContentGetValue;
//b) Optional (must NOT be implemented if unsupported!)
ContentGetDetectString: TContentGetDetectString;
ContentSetDefaultParams: TContentSetDefaultParams;
ContentStopGetValue: TContentStopGetValue;
ContentGetDefaultSortOrder: TContentGetDefaultSortOrder;
ContentPluginUnloading: TContentPluginUnloading;
ContentGetSupportedFieldFlags: TContentGetSupportedFieldFlags;
ContentSetValue: TContentSetValue;
ContentEditValue: TContentEditValue;
ContentSendStateInformation: TContentSendStateInformation;
//c) Unicode
ContentGetValueW: TContentGetValueW;
ContentStopGetValueW: TContentStopGetValueW;
ContentSetValueW: TContentSetValueW;
ContentSendStateInformationW: TContentSendStateInformationW;
public
//---------------------
function LoadModule: Boolean; override;
procedure UnloadModule; override;
function IsLoaded: Boolean; override;
//---------------------
procedure CallContentGetSupportedField; override;
procedure CallContentSetDefaultParams; override;
procedure CallContentStopGetValue(FileName: String); override;
//---------------------
function CallContentGetDefaultSortOrder(FieldIndex: Integer): Boolean; override;
function CallContentGetDetectString: String; override;
function CallContentGetValueV(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): Variant; overload; override;
function CallContentGetValue(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): String; overload; override;
function CallContentGetValue(FileName: String; FieldIndex: Integer; var UnitIndex: Integer): String; overload; override;
function CallContentGetSupportedFieldFlags(FieldIndex: Integer): Integer; override;
{ContentSetValue
ContentEditValue
ContentSendStateInformation}
//------------------------------------------------------
property ModuleHandle: TLibHandle read FModuleHandle;
property Force: Boolean read FForce write FForce;
//---------------------
end;
{ TLuaWdx }
TLuaWdx = class(TWdxModule)
private
L: Plua_State;
FForce: Boolean;
FName: String;
protected
function GetAName: String; override;
function GetAFileName: String; override;
procedure SetAName(AValue: String); override;
procedure SetAFileName(AValue: String); override;
function DoScript(AName: String): Integer;
function WdxLuaContentGetSupportedField(Index: Integer; var xFieldName, xUnits: String): Integer;
procedure WdxLuaContentPluginUnloading;
public
constructor Create; override;
//---------------------
function LoadModule: Boolean; override;
procedure UnloadModule; override;
function IsLoaded: Boolean; override;
//---------------------
procedure CallContentGetSupportedField; override;
procedure CallContentSetDefaultParams; override;
procedure CallContentStopGetValue(FileName: String); override;
//---------------------
function CallContentGetDefaultSortOrder(FieldIndex: Integer): Boolean; override;
function CallContentGetDetectString: String; override;
function CallContentGetValueV(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): Variant; overload; override;
function CallContentGetValue(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): String; overload; override;
function CallContentGetValue(FileName: String; FieldIndex: Integer; var UnitIndex: Integer): String; overload; override;
function CallContentGetSupportedFieldFlags(FieldIndex: Integer): Integer; override;
//---------------------
property Force: Boolean read FForce write FForce;
end;
{ TEmbeddedWDX }
TEmbeddedWDX = class(TWDXModule)
protected
function GetAName: String; override;
function GetAFileName: String; override;
procedure SetAName({%H-}AValue: String); override;
procedure SetAFileName({%H-}AValue: String); override;
protected
procedure AddField(const AName, XName: String; AType: Integer);
public
//---------------------
constructor Create; override;
//---------------------
function LoadModule: Boolean; override;
procedure UnloadModule; override;
function IsLoaded: Boolean; override;
//---------------------
function GetFieldIndex(FieldName: String): Integer; override;
end;
{ TWDXModuleList }
TWDXModuleList = class
private
Flist: TStringList;
function GetCount: Integer;
public
//---------------------
constructor Create;
destructor Destroy; override;
//---------------------
procedure Assign(Source: TWDXModuleList);
function IndexOfName(const AName: String): Integer;
//---------------------
procedure Clear;
procedure Exchange(Index1, Index2: Integer);
procedure Move(CurIndex, NewIndex: Integer);
procedure Load(AConfig: TXmlConfig; ANode: TXmlNode); overload;
procedure Save(AConfig: TXmlConfig; ANode: TXmlNode); overload;
function ComputeSignature(seed: dword): dword;
procedure DeleteItem(Index: Integer);
//---------------------
function Add(Item: TWDXModule): Integer; overload;
function Add(FileName: String): Integer; overload;
function Add(AName, FileName, DetectStr: String): Integer; overload;
function IsLoaded(AName: String): Boolean; overload;
function IsLoaded(Index: Integer): Boolean; overload;
function LoadModule(AName: String): Boolean; overload;
function LoadModule(Index: Integer): Boolean; overload;
function GetWdxModule(Index: Integer): TWDXModule; overload;
function GetWdxModule(AName: String): TWDXModule; overload;
//---------------------
//property WdxList:TStringList read Flist;
property Count: Integer read GetCount;
end;
function StrToVar(const Value: String; FieldType: Integer): Variant;
implementation
uses
//Lazarus, Free-Pascal, etc.
Math, StrUtils, LazUTF8, FileUtil,
//DC
DCClassesUtf8, DCStrUtils,
uComponentsSignature, uGlobs, uGlobsPaths, uDebug, uDCUtils, uOSUtils,
DCBasicTypes, DCOSUtils, DCDateTimeUtils, DCConvertEncoding, uLuaPas;
const
WdxIniFileName = 'wdx.ini';
type
TWdxModuleClass = class of TWdxModule;
// Language code conversion table
// Double Commander <-> Total Commander
const
WdxLangTable: array[0..30, 0..1] of String =
(
('be', 'BEL'),
('bg', 'BUL'),
('ca', 'CAT'),
('zh_CN', 'CHN'),
('cs', 'CZ' ),
('da', 'DAN'),
('de', 'DEU'),
('nl', 'DUT'),
('el', 'ELL'),
('es', 'ESP'),
('fr', 'FRA'),
('hr', 'HR' ),
('hu', 'HUN'),
('it', 'ITA'),
('ja', 'JPN'),
('ko', 'KOR'),
('nb', 'NOR'),
('nn', 'NOR'),
('pl', 'POL'),
('pt', 'POR'),
('pt_BR', 'PTG'),
('ro', 'ROM'),
('ru', 'RUS'),
('sk', 'SK' ),
('sr', 'SRB'),
('sr@latin', 'SRL'),
('sl', 'SVN'),
('sv', 'SWE'),
('tr', 'TUR'),
('zh_TW', 'TW' ),
('uk', 'UKR')
);
function GetWdxLang(const Code: String): String;
var
Index: Integer;
begin
for Index:= Low(WdxLangTable) to High(WdxLangTable) do
begin
if CompareStr(WdxLangTable[Index, 0], Code) = 0 then
begin
Exit(WdxLangTable[Index, 1]);
end;
end;
Result:= Code;
end;
function StrToVar(const Value: String; FieldType: Integer): Variant;
begin
case FieldType of
ft_fieldempty: Result := Unassigned;
ft_numeric_32: Result := StrToInt(Value);
ft_numeric_64: Result := StrToInt64(Value);
ft_numeric_floating: Result := StrToFloat(Value);
ft_date: Result := StrToDate(Value);
ft_time: Result := StrToTime(Value);
ft_datetime: Result := StrToDateTime(Value);
ft_boolean: Result := ((LowerCase(Value) = 'true') OR (Value = rsSimpleWordTrue));
ft_multiplechoice,
ft_string,
ft_fulltext,
ft_stringw: Result := Value;
else
Result := Unassigned;
end;
end;
{ TWDXModuleList }
function TWDXModuleList.GetCount: Integer;
begin
if Assigned(Flist) then
Result := Flist.Count
else
Result := 0;
end;
constructor TWDXModuleList.Create;
begin
Flist := TStringList.Create;
end;
destructor TWDXModuleList.Destroy;
begin
Clear;
FreeAndNil(Flist);
inherited Destroy;
end;
procedure TWDXModuleList.Assign(Source: TWDXModuleList);
var
I: Integer;
Module: TWDXModule;
begin
if Assigned(Source) then
begin
Clear;
for I := 0 to Source.Flist.Count - 1 do
begin
with TWdxModule(Source.Flist.Objects[I]) do
begin
Module:= TWdxModuleClass(ClassType).Create;
Module.Name:= Name;
Module.FileName:= FileName;
Module.DetectStr:= DetectStr;
Add(Module);
end;
end;
end;
end;
function TWDXModuleList.IndexOfName(const AName: String): Integer;
begin
Result := Flist.IndexOf(UpCase(AName));
end;
procedure TWDXModuleList.Clear;
var
i: Integer;
begin
for i := 0 to Flist.Count - 1 do
TWDXModule(Flist.Objects[i]).Free;
Flist.Clear;
end;
procedure TWDXModuleList.Exchange(Index1, Index2: Integer);
begin
FList.Exchange(Index1, Index2);
end;
procedure TWDXModuleList.Move(CurIndex, NewIndex: Integer);
begin
FList.Move(CurIndex, NewIndex);
end;
procedure TWDXModuleList.Load(AConfig: TXmlConfig; ANode: TXmlNode);
var
AName, APath: String;
AWdxModule: TWDXModule;
begin
Self.Clear;
ANode := ANode.FindNode('WdxPlugins');
if Assigned(ANode) then
begin
ANode := ANode.FirstChild;
while Assigned(ANode) do
begin
if ANode.CompareName('WdxPlugin') = 0 then
begin
if AConfig.TryGetValue(ANode, 'Name', AName) and
AConfig.TryGetValue(ANode, 'Path', APath) then
begin
// Create a correct object based on plugin file extension.
if UpCase(ExtractFileExt(APath)) = '.LUA' then
AWdxModule := TLuaWdx.Create
else
AWdxModule := TPluginWDX.Create;
AWdxModule.Name := AName;
AWdxModule.FileName := APath;
AWdxModule.DetectStr := AConfig.GetValue(ANode, 'DetectString', '');
Flist.AddObject(UpCase(AName), AWdxModule);
end
else
DCDebug('Invalid entry in configuration: ' + AConfig.GetPathFromNode(ANode) + '.');
end;
ANode := ANode.NextSibling;
end;
end;
end;
procedure TWDXModuleList.Save(AConfig: TXmlConfig; ANode: TXmlNode);
var
i: Integer;
SubNode: TXmlNode;
begin
ANode := AConfig.FindNode(ANode, 'WdxPlugins', True);
AConfig.ClearNode(ANode);
For i := 0 to Flist.Count - 1 do
begin
if not (Flist.Objects[I] is TEmbeddedWDX) then
begin
SubNode := AConfig.AddNode(ANode, 'WdxPlugin');
AConfig.AddValue(SubNode, 'Name', TWDXModule(Flist.Objects[I]).Name);
AConfig.AddValue(SubNode, 'Path', TWDXModule(Flist.Objects[I]).FileName);
AConfig.AddValue(SubNode, 'DetectString', TWDXModule(Flist.Objects[I]).DetectStr);
end;
end;
end;
{ TWDXModuleList.ComputeSignature }
function TWDXModuleList.ComputeSignature(seed: dword): dword;
var
iIndex: integer;
begin
result := seed;
for iIndex := 0 to pred(Count) do
begin
result := ComputeSignatureString(result, TWDXModule(Flist.Objects[iIndex]).Name);
result := ComputeSignatureString(result, TWDXModule(Flist.Objects[iIndex]).FileName);
result := ComputeSignatureString(result, TWDXModule(Flist.Objects[iIndex]).DetectStr);
end;
end;
procedure TWDXModuleList.DeleteItem(Index: Integer);
begin
if (Index > -1) and (Index < Flist.Count) then
begin
TWDXModule(Flist.Objects[Index]).Free;
Flist.Delete(Index);
end;
end;
function TWDXModuleList.Add(Item: TWDXModule): Integer;
begin
Result := Flist.AddObject(UpCase(item.Name), Item);
end;
function TWDXModuleList.Add(FileName: String): Integer;
var
s: String;
begin
s := ExtractFileName(FileName);
if pos('.', s) > 0 then
Delete(s, pos('.', s), length(s));
if UpCase(ExtractFileExt(FileName)) = '.LUA' then
Result := Flist.AddObject(UpCase(s), TLuaWdx.Create)
else
Result := Flist.AddObject(UpCase(s), TPluginWDX.Create);
TWDXModule(Flist.Objects[Result]).Name := s;
TWDXModule(Flist.Objects[Result]).FileName := FileName;
if TWDXModule(Flist.Objects[Result]).LoadModule then
begin
TWDXModule(Flist.Objects[Result]).DetectStr := TWDXModule(Flist.Objects[Result]).CallContentGetDetectString;
TWDXModule(Flist.Objects[Result]).UnloadModule;
end;
end;
function TWDXModuleList.Add(AName, FileName, DetectStr: String): Integer;
begin
if UpCase(ExtractFileExt(FileName)) = '.LUA' then
Result := Flist.AddObject(UpCase(AName), TLuaWdx.Create)
else
Result := Flist.AddObject(UpCase(AName), TPluginWDX.Create);
TWDXModule(Flist.Objects[Result]).Name := AName;
TWDXModule(Flist.Objects[Result]).DetectStr := DetectStr;
TWDXModule(Flist.Objects[Result]).FileName := FileName;
end;
function TWDXModuleList.IsLoaded(AName: String): Boolean;
var
x: Integer;
begin
x := Flist.IndexOf(AName);
if x = -1 then
Result := False
else
begin
Result := GetWdxModule(x).IsLoaded;
end;
end;
function TWDXModuleList.IsLoaded(Index: Integer): Boolean;
begin
Result := GetWdxModule(Index).IsLoaded;
end;
function TWDXModuleList.LoadModule(AName: String): Boolean;
var
x: Integer;
begin
x := Flist.IndexOf(UpCase(AName));
if x = -1 then
Result := False
else
begin
Result := GetWdxModule(x).LoadModule;
end;
end;
function TWDXModuleList.LoadModule(Index: Integer): Boolean;
begin
Result := GetWdxModule(Index).LoadModule;
end;
function TWDXModuleList.GetWdxModule(Index: Integer): TWDXModule;
begin
Result := TWDXModule(Flist.Objects[Index]);
end;
function TWDXModuleList.GetWdxModule(AName: String): TWDXModule;
var
tmp: Integer;
begin
tmp := Flist.IndexOf(upcase(AName));
if tmp < 0 then Exit(nil);
Result := TWDXModule(Flist.Objects[tmp])
end;
{ TPluginWDX }
function TPluginWDX.IsLoaded: Boolean;
begin
Result := FModuleHandle <> NilHandle;
end;
function TPluginWDX.GetAName: String;
begin
Result := FName;
end;
function TPluginWDX.GetAFileName: String;
begin
Result := FFileName;
end;
procedure TPluginWDX.SetAName(AValue: String);
begin
FName := AValue;
end;
procedure TPluginWDX.SetAFileName(AValue: String);
begin
FFileName := AValue;
end;
function TPluginWDX.LoadModule: Boolean;
var
AHandle: TLibHandle;
begin
EnterCriticalSection(FMutex);
try
if FModuleHandle <> NilHandle then Exit(True);
AHandle := mbLoadLibrary(mbExpandFileName(Self.FileName));
Result := (AHandle <> NilHandle);
if not Result then Exit;
{ Mandatory }
ContentGetSupportedField := TContentGetSupportedField(GetProcAddress(AHandle, 'ContentGetSupportedField'));
ContentGetValue := TContentGetValue(GetProcAddress(AHandle, 'ContentGetValue'));
{ Optional (must NOT be implemented if unsupported!) }
ContentGetDetectString := TContentGetDetectString(GetProcAddress(AHandle, 'ContentGetDetectString'));
ContentSetDefaultParams := TContentSetDefaultParams(GetProcAddress(AHandle, 'ContentSetDefaultParams'));
ContentStopGetValue := TContentStopGetValue(GetProcAddress(AHandle, 'ContentStopGetValue'));
ContentGetDefaultSortOrder := TContentGetDefaultSortOrder(GetProcAddress(AHandle, 'ContentGetDefaultSortOrder'));
ContentPluginUnloading := TContentPluginUnloading(GetProcAddress(AHandle, 'ContentPluginUnloading'));
ContentGetSupportedFieldFlags := TContentGetSupportedFieldFlags(GetProcAddress(AHandle, 'ContentGetSupportedFieldFlags'));
ContentSetValue := TContentSetValue(GetProcAddress(AHandle, 'ContentSetValue'));
ContentEditValue := TContentEditValue(GetProcAddress(AHandle, 'ContentEditValue'));
ContentSendStateInformation := TContentSendStateInformation(GetProcAddress(AHandle, 'ContentSendStateInformation'));
{ Unicode }
ContentGetValueW := TContentGetValueW(GetProcAddress(AHandle, 'ContentGetValueW'));
ContentStopGetValueW := TContentStopGetValueW(GetProcAddress(AHandle, 'ContentStopGetValueW'));
ContentSetValueW := TContentSetValueW(GetProcAddress(AHandle, 'ContentSetValueW'));
ContentSendStateInformationW := TContentSendStateInformationW(GetProcAddress(AHandle, 'ContentSendStateInformationW'));
CallContentSetDefaultParams;
CallContentGetSupportedField;
if Length(Self.DetectStr) = 0 then
Self.DetectStr := CallContentGetDetectString;
FModuleHandle := AHandle;
finally
LeaveCriticalSection(FMutex);
end;
end;
procedure TPluginWDX.CallContentSetDefaultParams;
var
dps: tContentDefaultParamStruct;
begin
if assigned(ContentSetDefaultParams) then
begin
dps.DefaultIniName := mbFileNameToSysEnc(gpCfgDir + WdxIniFileName);
dps.PluginInterfaceVersionHi := 1;
dps.PluginInterfaceVersionLow := 50;
dps.size := SizeOf(tContentDefaultParamStruct);
ContentSetDefaultParams(@dps);
end;
end;
procedure TPluginWDX.CallContentStopGetValue(FileName: String);
begin
if Assigned(ContentStopGetValueW) then
ContentStopGetValueW(PWideChar(CeUtf8ToUtf16(FileName)))
else if Assigned(ContentStopGetValue) then
ContentStopGetValue(PAnsiChar(CeUtf8ToSys(FileName)));
end;
function TPluginWDX.CallContentGetDefaultSortOrder(FieldIndex: Integer): Boolean;
var
x: Integer;
begin
if Assigned(ContentGetDefaultSortOrder) then
begin
x := ContentGetDefaultSortOrder(FieldIndex);
case x of
1: Result := False; //a..z 1..9
-1: Result := True; //z..a 9..1
end;
end;
end;
procedure TPluginWDX.UnloadModule;
var
AHandle: TLibHandle;
begin
EnterCriticalSection(FMutex);
try
if Assigned(ContentPluginUnloading) then
ContentPluginUnloading;
if FModuleHandle <> NilHandle then
begin
AHandle:= FModuleHandle;
FModuleHandle := NilHandle;
FreeLibrary(AHandle);
end;
{ Mandatory }
ContentGetSupportedField := nil;
ContentGetValue := nil;
{ Optional (must NOT be implemented if unsupported!) }
ContentGetDetectString := nil;
ContentSetDefaultParams := nil;
ContentStopGetValue := nil;
ContentGetDefaultSortOrder := nil;
ContentPluginUnloading := nil;
ContentGetSupportedFieldFlags := nil;
ContentSetValue := nil;
ContentEditValue := nil;
ContentSendStateInformation := nil;
{ Unicode }
ContentGetValueW := nil;
ContentStopGetValueW := nil;
ContentSetValueW := nil;
ContentSendStateInformationW := nil;
finally
LeaveCriticalSection(FMutex);
end;
end;
procedure TPluginWDX.CallContentGetSupportedField;
const
MAX_LEN = 256;
var
sFieldName: String;
Index, Rez: Integer;
xFieldName, xUnits: array[0..Pred(MAX_LEN)] of AnsiChar;
begin
FFieldsList.Clear;
if Assigned(ContentGetSupportedField) then
begin
Index := 0;
xUnits[0] := #0;
xFieldName[0] := #0;
repeat
Rez := ContentGetSupportedField(Index, xFieldName, xUnits, MAX_LEN);
if Rez > ft_nomorefields then
begin
sFieldName := CeSysToUtf8(xFieldName);
AddField(sFieldName, xUnits, Rez);
end;
Inc(Index);
until (Rez <= ft_nomorefields);
Translate;
end;
end;
function TPluginWDX.CallContentGetDetectString: String;
const
MAX_LEN = 2048; // See contentplugin.hlp for details
begin
if not Assigned(ContentGetDetectString) then
Result := EmptyStr
else begin
Result := StringOfChar(#0, MAX_LEN);
ContentGetDetectString(PAnsiChar(Result), MAX_LEN);
Result := Trim(PAnsiChar(Result));
end;
end;
function TPluginWDX.CallContentGetValueV(FileName: String; FieldIndex,
UnitIndex: Integer; flags: Integer): Variant;
var
Rez: Integer;
ATime: TDateTime;
Buf: array[0..WDX_MAX_LEN] of Byte;
fnval: Integer absolute buf;
fnval64: Int64 absolute buf;
ffval: Double absolute buf;
fdate: TDateFormat absolute buf;
ftime: TTimeFormat absolute buf;
wtime: TWinFileTime absolute buf;
begin
EnterCriticalSection(FMutex);
try
if Assigned(ContentGetValueW) then
Rez := ContentGetValueW(PWideChar(CeUtf8ToUtf16(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), flags)
else if Assigned(ContentGetValue) then
Rez := ContentGetValue(PAnsiChar(mbFileNameToSysEnc(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), flags);
case Rez of
ft_fieldempty: Result := Unassigned;
ft_numeric_32: Result := fnval;
ft_numeric_64: Result := fnval64;
ft_numeric_floating: Result := ffval;
ft_date:
begin
if TryEncodeDate(fdate.wYear, fdate.wMonth, fdate.wDay, ATime) then
Result := ATime
else
Result := Unassigned;
end;
ft_time:
begin
if TryEncodeTime(ftime.wHour, ftime.wMinute, ftime.wSecond, 0, ATime) then
Result := ATime
else
Result := Unassigned;
end;
ft_datetime: Result := WinFileTimeToDateTime(wtime);
ft_boolean: Result := Boolean(fnval);
ft_multiplechoice,
ft_string,
ft_fulltext: Result := CeSysToUtf8(AnsiString(PAnsiChar(@Buf[0])));
ft_stringw,
ft_fulltextw: Result := UTF16ToUTF8(UnicodeString(PWideChar(@Buf[0])));
else
Result := Unassigned;
end;
finally
LeaveCriticalSection(FMutex);
end;
end;
function TPluginWDX.CallContentGetValue(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): String;
var
Rez: Integer;
Buf: array[0..WDX_MAX_LEN] of Byte;
fnval: Integer absolute buf;
fnval64: Int64 absolute buf;
ffval: Double absolute buf;
fdate: TDateFormat absolute buf;
ftime: TTimeFormat absolute buf;
wtime: TWinFileTime absolute buf;
begin
EnterCriticalSection(FMutex);
try
if Assigned(ContentGetValueW) then
Rez := ContentGetValueW(PWideChar(CeUtf8ToUtf16(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), flags)
else if Assigned(ContentGetValue) then
Rez := ContentGetValue(PAnsiChar(mbFileNameToSysEnc(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), flags);
case Rez of
ft_fieldempty: Result := '';
ft_numeric_32: Result := IntToStr(fnval);
ft_numeric_64: Result := IntToStr(fnval64);
ft_numeric_floating: Result := FloatToStr(ffval);
ft_date: Result := Format('%2.2d.%2.2d.%4.4d', [fdate.wDay, fdate.wMonth, fdate.wYear]);
ft_time: Result := Format('%2.2d:%2.2d:%2.2d', [ftime.wHour, ftime.wMinute, ftime.wSecond]);
ft_datetime: Result := DateTimeToStr(WinFileTimeToDateTime(wtime));
ft_boolean: Result := ifThen((fnval = 0), rsSimpleWordFalse, rsSimpleWordTrue);
ft_multiplechoice,
ft_string,
ft_fulltext: Result := CeSysToUtf8(AnsiString(PAnsiChar(@Buf[0])));
ft_stringw,
ft_fulltextw: Result := UTF16ToUTF8(UnicodeString(PWideChar(@Buf[0])));
//TODO: FT_DELAYED,ft_ondemand
else
Result := '';
end;
finally
LeaveCriticalSection(FMutex);
end;
end;
function TPluginWDX.CallContentGetValue(FileName: String; FieldIndex: Integer; var UnitIndex: Integer): String;
var
Rez: Integer;
ValueA: AnsiString;
ValueW: UnicodeString;
Buf: array[0..WDX_MAX_LEN] of Byte;
begin
EnterCriticalSection(FMutex);
try
if Assigned(ContentGetValueW) then
Rez := ContentGetValueW(PWideChar(CeUtf8ToUtf16(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), 0)
else if Assigned(ContentGetValue) then
Rez := ContentGetValue(PAnsiChar(mbFileNameToSysEnc(FileName)), FieldIndex, UnitIndex, @Buf, SizeOf(buf), 0);
case Rez of
ft_fieldempty:
Result := EmptyStr;
ft_fulltext:
begin
ValueA:= AnsiString(PAnsiChar(@Buf[0]));
Inc(UnitIndex, Length(ValueA));
Result := CeSysToUtf8(ValueA);
end;
ft_fulltextw:
begin
ValueW:= UnicodeString(PWideChar(@Buf[0]));
Inc(UnitIndex, Length(ValueW) * SizeOf(WideChar));
Result := UTF16ToUTF8(ValueW);
end;
else begin
Result := EmptyStr;
end;
end;
finally
LeaveCriticalSection(FMutex);
end;
end;
function TPluginWDX.CallContentGetSupportedFieldFlags(FieldIndex: Integer): Integer;
begin
if assigned(ContentGetSupportedFieldFlags) then
Result := ContentGetSupportedFieldFlags(FieldIndex);
end;
{ TLuaWdx }
function TLuaWdx.GetAName: String;
begin
Result := FName;
end;
function TLuaWdx.GetAFileName: String;
begin
Result := FFileName;
end;
procedure TLuaWdx.SetAName(AValue: String);
begin
FName := AValue;
end;
procedure TLuaWdx.SetAFileName(AValue: String);
begin
FFileName := AValue;
end;
function TLuaWdx.DoScript(AName: String): Integer;
begin
Result := LUA_ERRRUN;
if not Assigned(L) then Exit;
Result := luaL_dofile(L, PChar(AName));
if Result <> 0 then begin
DCDebug('TLuaWdx.DoScript: ', lua_tostring(L, -1));
end;
end;
constructor TLuaWdx.Create;
begin
inherited Create;
if not IsLuaLibLoaded then
LoadLuaLib(mbExpandFileName(gLuaLib)); //Todo вынести загрузку либы в VmClass
end;
function TLuaWdx.LoadModule: Boolean;
var
sAbsolutePathFilename: string;
begin
EnterCriticalSection(FMutex);
try
Result := False;
if (not IsLuaLibLoaded) or (L <> nil) then
exit;
L := lua_open;
if not Assigned(L) then
exit;
luaL_openlibs(L);
RegisterPackages(L);
sAbsolutePathFilename := mbExpandFileName(FFilename);
SetPackagePath(L, ExtractFilePath(sAbsolutePathFilename));
if DoScript(sAbsolutePathFilename) = 0 then
Result := True
else
Result := False;
CallContentSetDefaultParams;
CallContentGetSupportedField;
if Length(Self.DetectStr) = 0 then
Self.DetectStr := CallContentGetDetectString;
finally
LeaveCriticalSection(FMutex);
end;
end;
procedure TLuaWdx.UnloadModule;
begin
WdxLuaContentPluginUnloading;
if Assigned(L) then
begin
lua_close(L);
L := nil;
end;
end;
function TLuaWdx.IsLoaded: Boolean;
begin
Result := IsLuaLibLoaded and Assigned(L);
end;
function TLuaWdx.WdxLuaContentGetSupportedField(Index: Integer; var xFieldName, xUnits: String): Integer;
begin
Result := ft_nomorefields;
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentGetSupportedField');
if not lua_isfunction(L, -1) then
exit;
lua_pushinteger(L, Index);
LuaPCall(L, 1, 3);
xFieldName := lua_tostring(L, -3);
xUnits := lua_tostring(L, -2);
Result := Integer(lua_tointeger(L, -1));
lua_pop(L, 3);
end;
procedure TLuaWdx.WdxLuaContentPluginUnloading;
begin
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentPluginUnloading');
if not lua_isfunction(L, -1) then
exit;
LuaPCall(L, 0, 0);
end;
procedure TLuaWdx.CallContentGetSupportedField;
var
Index, Rez: Integer;
xFieldName, xUnits: String;
begin
FFieldsList.Clear;
Index := 0;
repeat
Rez := WdxLuaContentGetSupportedField(Index, xFieldName, xUnits);
DCDebug('WDX:CallGetSupFields:' + IntToStr(Rez));
if Rez <> ft_nomorefields then
begin
AddField(xFieldName, xUnits, Rez);
end;
Inc(Index);
until Rez = ft_nomorefields;
Translate;
end;
procedure TLuaWdx.CallContentSetDefaultParams;
begin
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentSetDefaultParams');
if not lua_isfunction(L, -1) then
exit;
lua_pushstring(L, PAnsiChar(gpCfgDir + WdxIniFileName));
lua_pushinteger(L, 1);
lua_pushinteger(L, 50);
LuaPCall(L, 3, 0);
end;
procedure TLuaWdx.CallContentStopGetValue(FileName: String);
begin
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentStopGetValue');
if not lua_isfunction(L, -1) then
exit;
lua_pushstring(L, PAnsiChar(FileName));
LuaPCall(L, 1, 0);
end;
function TLuaWdx.CallContentGetDefaultSortOrder(FieldIndex: Integer): Boolean;
var
x: Integer;
begin
Result := False;
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentGetDefaultSortOrder');
if not lua_isfunction(L, -1) then
exit;
lua_pushinteger(L, FieldIndex);
LuaPCall(L, 1, 1);
x := lua_tointeger(L, -1);
case x of
1: Result := False; //a..z 1..9
-1: Result := True; //z..a 9..1
end;
lua_pop(L, 1);
end;
function TLuaWdx.CallContentGetDetectString: String;
begin
Result := '';
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentGetDetectString');
if not lua_isfunction(L, -1) then
exit;
LuaPCall(L, 0, 1);
Result := lua_tostring(L, -1);
lua_pop(L, 1);
end;
function TLuaWdx.CallContentGetValueV(FileName: String; FieldIndex,
UnitIndex: Integer; flags: Integer): Variant;
begin
EnterCriticalSection(FMutex);
try
Result := Unassigned;
if not Assigned(L) then
Exit;
lua_getglobal(L, 'ContentGetValue');
if not lua_isfunction(L, -1) then
Exit;
lua_pushstring(L, PAnsiChar(FileName));
lua_pushinteger(L, FieldIndex);
lua_pushinteger(L, UnitIndex);
lua_pushinteger(L, flags);
LuaPCall(L, 4, 1);
if not lua_isnil(L, -1) then
begin
case TWdxField(FieldList.Objects[FieldIndex]).FType of
ft_string,
ft_fulltext,
ft_multiplechoice:
Result := lua_tostring(L, -1);
ft_numeric_32:
Result := Int32(lua_tointeger(L, -1));
ft_numeric_64:
Result := Int64(lua_tointeger(L, -1));
ft_boolean:
Result := lua_toboolean(L, -1);
ft_numeric_floating:
Result := lua_tonumber(L, -1);
ft_datetime:
Result := WinFileTimeToDateTime(TWinFileTime(lua_tointeger(L, -1)));
end;
end;
lua_pop(L, 1);
finally
LeaveCriticalSection(FMutex);
end;
end;
function TLuaWdx.CallContentGetValue(FileName: String; FieldIndex, UnitIndex: Integer; flags: Integer): String;
begin
EnterCriticalSection(FMutex);
try
Result := '';
if not Assigned(L) then
Exit;
lua_getglobal(L, 'ContentGetValue');
if not lua_isfunction(L, -1) then
Exit;
lua_pushstring(L, PAnsiChar(FileName));
lua_pushinteger(L, FieldIndex);
lua_pushinteger(L, UnitIndex);
lua_pushinteger(L, flags);
LuaPCall(L, 4, 1);
if not lua_isnil(L, -1) then
begin
case TWdxField(FieldList.Objects[FieldIndex]).FType of
ft_string,
ft_fulltext,
ft_multiplechoice:
Result := lua_tostring(L, -1);
ft_numeric_32:
Result := IntToStr(Int32(lua_tointeger(L, -1)));
ft_numeric_64:
Result := IntToStr(Int64(lua_tointeger(L, -1)));
ft_numeric_floating:
Result := FloatToStr(lua_tonumber(L, -1));
ft_boolean:
Result := IfThen(lua_toboolean(L, -1), rsSimpleWordTrue, rsSimpleWordFalse);
ft_datetime:
Result := DateTimeToStr(WinFileTimeToDateTime(TWinFileTime(lua_tointeger(L, -1))));
end;
end;
lua_pop(L, 1);
finally
LeaveCriticalSection(FMutex);
end;
end;
function TLuaWdx.CallContentGetValue(FileName: String; FieldIndex: Integer; var UnitIndex: Integer): String;
begin
EnterCriticalSection(FMutex);
try
Result := EmptyStr;
if not Assigned(L) then
Exit;
lua_getglobal(L, 'ContentGetValue');
if not lua_isfunction(L, -1) then
Exit;
lua_pushstring(L, PAnsiChar(FileName));
lua_pushinteger(L, FieldIndex);
lua_pushinteger(L, UnitIndex);
lua_pushinteger(L, 0);
LuaPCall(L, 4, 1);
if not lua_isnil(L, -1) then
begin
Result := lua_tostring(L, -1);
Inc(UnitIndex, Length(Result));
end;
lua_pop(L, 1);
finally
LeaveCriticalSection(FMutex);
end;
end;
function TLuaWdx.CallContentGetSupportedFieldFlags(FieldIndex: Integer): Integer;
begin
Result := 0;
if not assigned(L) then
exit;
lua_getglobal(L, 'ContentGetSupportedFieldFlags');
if not lua_isfunction(L, -1) then
exit;
lua_pushinteger(L, FieldIndex);
LuaPCall(L, 1, 1);
Result := lua_tointeger(L, -1);
lua_pop(L, 1);
end;
{ TEmbeddedWDX }
function TEmbeddedWDX.GetAName: String;
begin
Result:= EmptyStr;
end;
function TEmbeddedWDX.GetAFileName: String;
begin
Result:= ParamStrUTF8(0);
end;
procedure TEmbeddedWDX.SetAName(AValue: String);
begin
end;
procedure TEmbeddedWDX.SetAFileName(AValue: String);
begin
end;
procedure TEmbeddedWDX.AddField(const AName, XName: String; AType: Integer);
var
I: Integer;
begin
I := FFieldsList.AddObject(AName, TWdxField.Create);
with TWdxField(FFieldsList.Objects[I]) do
begin
FName := AName;
LName := XName;
FType := AType;
end;
end;
constructor TEmbeddedWDX.Create;
begin
inherited Create;
CallContentGetSupportedField;
end;
function TEmbeddedWDX.LoadModule: Boolean;
begin
Result:= True;
end;
procedure TEmbeddedWDX.UnloadModule;
begin
end;
function TEmbeddedWDX.IsLoaded: Boolean;
begin
Result:= True;
end;
function TEmbeddedWDX.GetFieldIndex(FieldName: String): Integer;
var
Index: Integer;
begin
Result:= inherited GetFieldIndex(FieldName);
if Result < 0 then
begin
for Index:= 0 to FFieldsList.Count - 1 do
begin
if AnsiSameText(FieldName, TWdxField(FFieldsList.Objects[Index]).LName) then
Exit(Index);
end;
end;
end;
{ TWDXModule }
procedure TWDXModule.Translate;
var
I: Integer;
SUnits: String;
Ini: TIniFileEx;
UserLang: String;
AFileName: String;
AUnits: TStringArray;
begin
AFileName:= mbExpandFileName(ChangeFileExt(Self.FileName, '.lng'));
if mbFileExists(AFileName) then
begin
UserLang:= GetWdxLang(ExtractDelimited(2, gpoFileName, ['.']));
if Length(UserLang) > 0 then
try
Ini:= TIniFileEx.Create(AFileName, fmOpenRead);
try
for I:= 0 to FFieldsList.Count - 1 do
begin
with TWdxField(FFieldsList.Objects[I]) do
begin
LName:= CeRawToUtf8(Ini.ReadString(UserLang, FName, FName));
if Length(OUnits) > 0 then
begin
SUnits:= CeRawToUtf8(Ini.ReadString(UserLang, OUnits, OUnits));
AUnits:= SplitString(sUnits, '|');
// Check that translation is valid
if Length(AUnits) = Length(FUnits) then
LUnits:= CopyArray(AUnits);
end;
end;
end;
finally
Ini.Free;
end;
except
// Skip
end;
end;
end;
function TWDXModule.GetADetectStr: String;
begin
Result:= FParser.DetectStr;
end;
procedure TWDXModule.SetADetectStr(const AValue: String);
begin
FParser.DetectStr:= AValue;
end;
procedure TWDXModule.AddField(const AName, AUnits: String; AType: Integer);
var
WdxField: TWdxField;
begin
WdxField:= TWdxField.Create;
FFieldsList.AddObject(AName, WdxField);
with WdxField do
begin
FName := AName;
LName := FName;
OUnits := AUnits;
FUnits := SplitString(OUnits, '|');
LUnits := CopyArray(FUnits);
FType := AType;
end;
end;
constructor TWDXModule.Create;
begin
InitCriticalSection(FMutex);
FParser:= TParserControl.Create;
FFieldsList:= TStringList.Create;
FFieldsList.OwnsObjects:= True;
end;
destructor TWDXModule.Destroy;
begin
FParser.Free;
FFieldsList.Free;
Self.UnloadModule;
inherited Destroy;
DoneCriticalSection(FMutex);
end;
function TWDXModule.FieldList: TStringList;
begin
Result:= FFieldsList;
end;
function TWDXModule.WdxFieldType(n: Integer): String;
begin
case n of
FT_NUMERIC_32: Result := 'FT_NUMERIC_32';
FT_NUMERIC_64: Result := 'FT_NUMERIC_64';
FT_NUMERIC_FLOATING: Result := 'FT_NUMERIC_FLOATING';
FT_DATE: Result := 'FT_DATE';
FT_TIME: Result := 'FT_TIME';
FT_DATETIME: Result := 'FT_DATETIME';
FT_BOOLEAN: Result := 'FT_BOOLEAN';
FT_MULTIPLECHOICE: Result := 'FT_MULTIPLECHOICE';
FT_STRING: Result := 'FT_STRING';
FT_FULLTEXT: Result := 'FT_FULLTEXT';
FT_NOSUCHFIELD: Result := 'FT_NOSUCHFIELD';
FT_FILEERROR: Result := 'FT_FILEERROR';
FT_FIELDEMPTY: Result := 'FT_FIELDEMPTY';
FT_DELAYED: Result := 'FT_DELAYED';
else
Result := '?';
end;
end;
function TWDXModule.GetFieldIndex(FieldName: String): Integer;
begin
Result := FFieldsList.IndexOf(FieldName);
end;
function TWDXModule.FileParamVSDetectStr(const aFile: TFile): Boolean;
begin
EnterCriticalSection(FMutex);
try
Result := FParser.TestFileResult(aFile);
finally
LeaveCriticalsection(FMutex);
end;
end;
function TWDXModule.CallContentGetValueV(FileName: String; FieldName: String;
UnitName: String; flags: Integer): Variant;
var
FieldIndex,
UnitIndex: Integer;
begin
FieldIndex := GetFieldIndex(FieldName);
if FieldIndex <> -1 then
begin
UnitIndex := TWdxField(FieldList.Objects[FieldIndex]).GetUnitIndex(UnitName);
Result := CallContentGetValueV(FileName, FieldIndex, UnitIndex, flags);
end
else
Result := Unassigned;
end;
function TWDXModule.CallContentGetValue(FileName: String; FieldName: String;
UnitName: String; flags: Integer): String;
var
FieldIndex,
UnitIndex: Integer;
begin
FieldIndex := GetFieldIndex(FieldName);
if FieldIndex <> -1 then
begin
UnitIndex := TWdxField(FieldList.Objects[FieldIndex]).GetUnitIndex(UnitName);
Result := CallContentGetValue(FileName, FieldIndex, UnitIndex, flags);
end
else
Result := EmptyStr;
end;
{ TWdxField }
function TWdxField.GetUnitIndex(UnitName: String): Integer;
var
Index: Integer;
begin
for Index:= 0 to High(FUnits) do
begin
if SameText(UnitName, FUnits[Index]) then
Exit(Index);
end;
Result := IfThen(FType = FT_MULTIPLECHOICE, -1, 0);
end;
end.