{ This file is part of the Free Pascal run time library. Copyright (c) 2003 by the Free Pascal development team CustomApplication class. See the file COPYING.FPC, included in this distribution, for details about the copyright. 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. **********************************************************************} {$mode objfpc} {$h+} unit CustApp; Interface uses SysUtils,Classes; Type TExceptionEvent = Procedure (Sender : TObject; E : Exception) Of Object; TEventLogTypes = Set of TEventType; { TCustomApplication } TCustomApplication = Class(TComponent) Private FEventLogFilter: TEventLogTypes; FOnException: TExceptionEvent; FTerminated : Boolean; FHelpFile, FTitle : String; FOptionChar : Char; FCaseSensitiveOptions : Boolean; FStopOnException : Boolean; function GetEnvironmentVar(VarName : String): String; function GetExeName: string; Function GetLocation : String; function GetTitle: string; Protected procedure SetTitle(const AValue: string); Virtual; Function GetConsoleApplication : boolean; Virtual; Procedure DoRun; Virtual; Function GetParams(Index : Integer) : String;virtual; function GetParamCount: Integer;Virtual; Procedure DoLog(EventType : TEventType; const Msg : String); virtual; Public constructor Create(AOwner: TComponent); override; destructor Destroy; override; // Some Delphi methods. procedure HandleException(Sender: TObject); virtual; procedure Initialize; virtual; procedure Run; procedure ShowException(E: Exception);virtual; procedure Terminate; virtual; // Extra methods. function FindOptionIndex(Const S : String; Var Longopt : Boolean) : Integer; Function GetOptionValue(Const S : String) : String; Function GetOptionValue(Const C: Char; Const S : String) : String; Function HasOption(Const S : String) : Boolean; Function HasOption(Const C : Char; Const S : String) : Boolean; Function CheckOptions(Const ShortOptions : String; Const Longopts : TStrings; Opts,NonOpts : TStrings; AllErrors : Boolean = False) : String; Function CheckOptions(Const ShortOptions : String; Const Longopts : Array of string; Opts,NonOpts : TStrings; AllErrors : Boolean = False) : String; Function CheckOptions(Const ShortOptions : String; Const Longopts : TStrings; AllErrors : Boolean = False) : String; Function CheckOptions(Const ShortOptions : String; Const LongOpts : Array of string; AllErrors : Boolean = False) : String; Function CheckOptions(Const ShortOptions : String; Const LongOpts : String; AllErrors : Boolean = False) : String; Procedure GetEnvironmentList(List : TStrings;NamesOnly : Boolean); Procedure GetEnvironmentList(List : TStrings); Procedure Log(EventType : TEventType; const Msg : String); // Delphi properties property ExeName: string read GetExeName; property HelpFile: string read FHelpFile write FHelpFile; property Terminated: Boolean read FTerminated; property Title: string read FTitle write SetTitle; property OnException: TExceptionEvent read FOnException write FOnException; // Extra properties Property ConsoleApplication : Boolean Read GetConsoleApplication; Property Location : String Read GetLocation; Property Params [Index : integer] : String Read GetParams; Property ParamCount : Integer Read GetParamCount; Property EnvironmentVariable[envName : String] : String Read GetEnvironmentVar; Property OptionChar : Char Read FoptionChar Write FOptionChar; Property CaseSensitiveOptions : Boolean Read FCaseSensitiveOptions Write FCaseSensitiveOptions; Property StopOnException : Boolean Read FStopOnException Write FStopOnException; Property EventLogFilter : TEventLogTypes Read FEventLogFilter Write FEventLogFilter; end; var CustomApplication : TCustomApplication = nil; Implementation {$ifdef darwin} uses MacOSAll; {$endif} { TCustomApplication } function TCustomApplication.GetExeName: string; {$if defined(darwin)} var mainBundle: CFBundleRef; executableUrl: CFURLRef; executableFSPath: CFStringRef; utf16len: ptrint; error: boolean; begin error:=false; { Get main bundle. This even works most of the time for command line applications } mainbundle:=CFBundleGetMainBundle; if assigned(mainbundle) then begin { get the URL pointing to the executable of the bundle } executableUrl:=CFBundleCopyExecutableURL(mainBundle); if assigned(executableUrl) then begin { convert the url to a POSIX path } executableFSPath:=CFURLCopyFileSystemPath(executableUrl,kCFURLPOSIXPathStyle); CFRelease(executableUrl); { convert to UTF-8 -- this is not really clean since in theory the ansi-encoding could be different, but a) all file i/o routines on Darwin expect utf-8-encoded strings b) there is no easy way to convert the Unix LANG encoding setting to an equivalent CoreFoundation encoding } utf16len:=CFStringGetLength(executableFSPath); // +1 for extra terminating #0 in the worst case, so the pos below // will always find the #0 setlength(result,utf16len*3+1); if CFStringGetCString(executableFSPath,@result[1],length(result),kCFStringEncodingUTF8) then { truncate to actual length, #0 cannot appear in a file path } setlength(result,pos(#0,result)-1) else error:=true; CFRelease(executableFSPath); end else error:=true; end else error:=true; if error then { can't do better than this } Result:=Paramstr(0); end; {$else darwin} begin Result:=Paramstr(0); end; {$endif darwin} Procedure SysGetEnvironmentList(List : TStrings;NamesOnly : Boolean); var s : string; i,l,j,count : longint; begin count:=GetEnvironmentVariableCount; if count>0 then for j:=1 to count do begin s:=GetEnvironmentString(j); l:=Length(s); If NamesOnly then begin I:=pos('=',s); If (I>0) then S:=Copy(S,1,I-1); end; List.Add(S); end; end; function TCustomApplication.GetEnvironmentVar(VarName : String): String; begin Result:=GetEnvironmentVariable(VarName); end; procedure TCustomApplication.GetEnvironmentList(List: TStrings; NamesOnly: Boolean); begin // Routine must be in custapp.inc SysGetEnvironmentList(List,NamesOnly); end; procedure TCustomApplication.GetEnvironmentList(List: TStrings); begin GetEnvironmentList(List,False); end; function TCustomApplication.GetLocation: String; begin Result:=ExtractFilePath(GetExeName); end; function TCustomApplication.GetParamCount: Integer; begin Result:=System.ParamCount; end; function TCustomApplication.GetTitle: string; begin Result:=FTitle; end; function TCustomApplication.GetParams(Index: Integer): String; begin Result:=ParamStr(Index); end; procedure TCustomApplication.SetTitle(const AValue: string); begin FTitle:=AValue; end; function TCustomApplication.GetConsoleApplication: boolean; begin Result:=IsConsole; end; procedure TCustomApplication.DoRun; begin // Do nothing. Override in descendent classes. end; procedure TCustomApplication.DoLog(EventType: TEventType; const Msg: String); begin // Do nothing, override in descendants end; procedure TCustomApplication.Log(EventType: TEventType; const Msg: String); begin If (FEventLogFilter=[]) or (EventType in FEventLogFilter) then DoLog(EventType,Msg); end; constructor TCustomApplication.Create(AOwner: TComponent); begin inherited Create(AOwner); FOptionChar:='-'; FCaseSensitiveOptions:=True; FStopOnException:=False; end; destructor TCustomApplication.Destroy; begin inherited Destroy; end; procedure TCustomApplication.HandleException(Sender: TObject); begin If Not (ExceptObject is Exception) then SysUtils.showexception(ExceptObject,ExceptAddr) else begin If Not Assigned(FOnexception) then ShowException(Exception(ExceptObject)) else FOnException(Sender,Exception(ExceptObject)); end; If FStopOnException then FTerminated:=True; end; procedure TCustomApplication.Initialize; begin FTerminated:=False; end; procedure TCustomApplication.Run; begin Repeat Try DoRun; except HandleException(Self); end; Until FTerminated; end; procedure TCustomApplication.ShowException(E: Exception); begin Sysutils.ShowException(E,ExceptAddr) end; procedure TCustomApplication.Terminate; begin FTerminated:=True; end; function TCustomApplication.GetOptionValue(const S: String): String; begin Result:=GetoptionValue(#255,S); end; function TCustomApplication.GetOptionValue(const C: Char; const S: String ): String; Var B : Boolean; I,P : integer; O : String; begin Result:=''; I:=FindOptionIndex(C,B); If (I=-1) then I:=FindoptionIndex(S,B); If (I<>-1) then begin If B then begin // Long options have form --option=value O:=Params[I]; P:=Pos('=',O); If (P=0) then P:=Length(O); Delete(O,1,P); Result:=O; end else begin // short options have form '-o value' If (I'-') then Result:=Params[I+1]; end; end; end; function TCustomApplication.HasOption(const S: String): Boolean; Var B : Boolean; begin Result:=FindOptionIndex(S,B)<>-1; end; function TCustomApplication.FindOptionIndex(const S: String; var Longopt: Boolean): Integer; Var SO,O : String; I,P : Integer; begin If Not CaseSensitiveOptions then SO:=UpperCase(S) else SO:=S; Result:=-1; I:=ParamCount; While (Result=-1) and (I>0) do begin O:=Params[i]; If (Length(O)>0) and (O[1]=FOptionChar) then begin Delete(O,1,1); LongOpt:=(Length(O)>0) and (O[1]=FOptionChar); If LongOpt then begin Delete(O,1,1); P:=Pos('=',O); If (P<>0) then O:=Copy(O,1,P-1); end; If Not CaseSensitiveOptions then O:=UpperCase(O); If (O=SO) then Result:=i; end; Dec(i); end; end; function TCustomApplication.HasOption(const C: Char; const S: String): Boolean; Var B : Boolean; begin Result:=(FindOptionIndex(C,B)<>-1) or (FindOptionIndex(S,B)<>-1); end; function TCustomApplication.CheckOptions(const ShortOptions: String; const Longopts: TStrings; AllErrors: Boolean): String; begin Result:=CheckOptions(ShortOptions,LongOpts,Nil,Nil,AllErrors); end; ResourceString SErrInvalidOption = 'Invalid option at position %d: "%s"'; SErrNoOptionAllowed = 'Option at position %d does not allow an argument: %s'; SErrOptionNeeded = 'Option at position %d needs an argument : %s'; function TCustomApplication.CheckOptions(const ShortOptions: String; const Longopts: TStrings; Opts, NonOpts: TStrings; AllErrors: Boolean ): String; Var I,J,L,P : Integer; O,OV,SO : String; UsedArg,HaveArg : Boolean; Function FindLongOpt(S : String) : boolean; Var I : integer; begin Result:=Assigned(LongOpts); if Not Result then exit; If CaseSensitiveOptions then begin I:=LongOpts.Count-1; While (I>=0) and (LongOpts[i]<>S) do Dec(i); end else begin S:=UpperCase(S); I:=LongOpts.Count-1; While (I>=0) and (UpperCase(LongOpts[i])<>S) do Dec(i); end; Result:=(I<>-1); end; Procedure AddToResult(Const Msg : string); begin If (Result<>'') then Result:=Result+sLineBreak; Result:=Result+Msg; end; begin If CaseSensitiveOptions then SO:=Shortoptions else SO:=LowerCase(Shortoptions); Result:=''; I:=1; While (I<=ParamCount) and ((Result='') or AllErrors) do begin O:=Paramstr(I); If (Length(O)=0) or (O[1]<>FOptionChar) then begin If Assigned(NonOpts) then NonOpts.Add(O) end else begin If (Length(O)<2) then AddToResult(Format(SErrInvalidOption,[i,O])) else begin HaveArg:=False; OV:=''; // Long option ? If (O[2]=FOptionChar) then begin Delete(O,1,2); J:=Pos('=',O); If J<>0 then begin HaveArg:=true; OV:=O; Delete(OV,1,J); O:=Copy(O,1,J-1); end; // Switch Option If FindLongopt(O) then begin If HaveArg then AddToResult(Format(SErrNoOptionAllowed,[I,O])) end else begin // Required argument If FindLongOpt(O+':') then begin If Not HaveArg then AddToResult(Format(SErrOptionNeeded,[I,O])); end else begin // Optional Argument. If not FindLongOpt(O+'::') then AddToResult(Format(SErrInvalidOption,[I,O])); end; end; end else // Short Option. begin HaveArg:=(I0) and (ParamStr(I+1)[1]<>FOptionChar); UsedArg:=False; If HaveArg then OV:=Paramstr(I+1); If Not CaseSensitiveOptions then O:=LowerCase(O); L:=Length(O); J:=2; While ((Result='') or AllErrors) and (J<=L) do begin P:=Pos(O[J],ShortOptions); If (P=0) or (O[j]=':') then AddToResult(Format(SErrInvalidOption,[I,O[J]])) else begin If (P':') Then If (J