summaryrefslogtreecommitdiff
path: root/src/corelib
diff options
context:
space:
mode:
authorgraemeg <graemeg@ae50a9b5-8222-0410-bf8d-8a13f76226bf>2008-04-11 08:39:19 +0000
committergraemeg <graemeg@ae50a9b5-8222-0410-bf8d-8a13f76226bf>2008-04-11 08:39:19 +0000
commit95b760d80613d0873a9ed92e4bf29f1ef9b27564 (patch)
treef11414640f0197adaf78bc319b46dcf50e4c1061 /src/corelib
parent2d5b6397c7f583fcaad81bf98c6af5623e759592 (diff)
downloadfpGUI-95b760d80613d0873a9ed92e4bf29f1ef9b27564.tar.xz
* Fixed a bug where exceptions were silently handled without notification. This was in the messagequeue code.
* Implemented a default application wide exception handler. Applications will not simply crash and quit. * Added TfpgApplication.HandleException * Added TfpgApplication.OnException event developers can hook into * Added TfpgApplication.ShowException - the default exception handler. * Added TfpgApplication.StopOnException - should the application quit or not when an unhandled exception occurres.
Diffstat (limited to 'src/corelib')
-rw-r--r--src/corelib/fpgfx.pas81
-rw-r--r--src/corelib/gfx_msgqueue.inc30
2 files changed, 86 insertions, 25 deletions
diff --git a/src/corelib/fpgfx.pas b/src/corelib/fpgfx.pas
index 78a00317..5aeafe77 100644
--- a/src/corelib/fpgfx.pas
+++ b/src/corelib/fpgfx.pas
@@ -69,7 +69,8 @@ type
TMouseWheelEvent = procedure(Sender: TObject; AShift: TShiftState; AWheelDelta: Single; const AMousePos: TPoint) of object;
{ Painting }
TPaintEvent = procedure(Sender: TObject{; const ARect: TfpgRect}) of object;
-
+ { Exceptions }
+ TExceptionEvent = procedure(Sender: TObject; E: Exception) of object;
type
@@ -187,6 +188,8 @@ type
TfpgApplication = class(TfpgApplicationImpl)
private
+ FOnException: TExceptionEvent;
+ FStopOnException: Boolean;
procedure SetupLocalizationStrings;
protected
FDisplayParams: string;
@@ -209,9 +212,13 @@ type
procedure ProcessMessages;
procedure SetMessageHook(AWidget: TObject; const AMsgCode: integer; AListener: TObject);
procedure UnsetMessageHook(AWidget: TObject; const AMsgCode: integer; AListener: TObject);
+ procedure HandleException(Sender: TObject);
+ procedure ShowException(E: Exception);
+ property DefaultFont: TfpgFont read FDefaultFont;
property ScreenWidth: integer read FScreenWidth;
property ScreenHeight: integer read FScreenHeight;
- property DefaultFont: TfpgFont read FDefaultFont;
+ property StopOnException: Boolean read FStopOnException write FStopOnException;
+ property OnException: TExceptionEvent read FOnException write FOnException;
end;
@@ -326,7 +333,8 @@ uses
gfx_stdimages,
gfx_extinterpolation, // only so that it get auto compiled
gfx_translations,
- gfx_constants;
+ gfx_constants,
+ gui_dialogs;
var
fpgTimers: TList;
@@ -570,9 +578,28 @@ begin
end;
procedure DumpStack;
+Var
+ Message : String;
+ i : longint;
begin
writeln(' Stack trace:');
- Dump_Stack(StdOut, get_frame);
+// Dump_Stack(StdOut, get_frame);
+
+ Writeln(stdout,'An unhandled exception occurred at $',HexStr(Ptrint(ExceptAddr),sizeof(PtrInt)*2),' :');
+ if ExceptObject is exception then
+ begin
+ Message:=Exception(ExceptObject).ClassName+' : '+Exception(ExceptObject).Message;
+ Writeln(stdout,Message);
+ end
+ else
+ Writeln(stdout,'Exception object ',ExceptObject.ClassName,' is not of class Exception.');
+ Writeln(stdout,BackTraceStrFunc(ExceptAddr));
+ if (ExceptFrameCount>0) then
+ begin
+ for i:=0 to ExceptFrameCount-1 do
+ Writeln(stdout,BackTraceStrFunc(ExceptFrames[i]));
+ end;
+ Writeln(stdout,'');
end;
{ TfpgTimer }
@@ -728,6 +755,7 @@ begin
FScreenHeight := -1;
FModalFormStack := TList.Create;
FMessageHookList := TFPList.Create;
+ FStopOnException := False;
try
inherited Create(aparams);
@@ -741,7 +769,7 @@ begin
FDefaultFont := GetFont(FPG_DEFAULT_FONT_DESC);
except
on E: Exception do
- writeln(E.Message);
+ Sysutils.ShowException(ExceptObject, ExceptAddr);
end;
end;
@@ -831,12 +859,18 @@ begin
if IsInitialized then
InternalInit
else
- raise Exception.Create('Failed in initialize the Application object!');
+ raise Exception.Create('Failed to initialize the Application object!');
end;
procedure TfpgApplication.Run;
begin
- RunMessageLoop;
+ repeat
+ try
+ RunMessageLoop;
+ except
+ HandleException(Self);
+ end;
+ until Terminated;
end;
procedure TfpgApplication.SetupLocalizationStrings;
@@ -958,6 +992,35 @@ begin
end;
end;
+procedure TfpgApplication.HandleException(Sender: TObject);
+begin
+ if not (ExceptObject is Exception) then
+ SysUtils.ShowException(ExceptObject, ExceptAddr)
+ else
+ begin
+ if not (ExceptObject is EAbort) then // EAborts are silent. They show no message.
+ begin
+ if Assigned(FOnException) then
+ FOnException(Sender, Exception(ExceptObject))
+ else
+ begin
+// SysUtils.ShowException(ExceptObject, ExceptAddr);
+ ShowException(Exception(ExceptObject));
+// DumpStack;
+ end;
+ end;
+ end; { if/else }
+
+ // Note: We should not terminate when we receive EAbort exceptions.
+ if (not (ExceptObject is EAbort)) and StopOnException then
+ Terminated := True;
+end;
+
+procedure TfpgApplication.ShowException(E: Exception);
+begin
+ TfpgMessageDialog.Critical('An unexpected error occurred.', E.Message);
+end;
+
procedure TfpgApplication.WaitWindowMessage(atimeoutms: integer);
begin
if IsMultiThread then
@@ -970,9 +1033,7 @@ end;
procedure TfpgApplication.RunMessageLoop;
begin
- repeat
- WaitWindowMessage(1000);
- until Terminated;
+ WaitWindowMessage(1000);
end;
{ TfpgFont }
diff --git a/src/corelib/gfx_msgqueue.inc b/src/corelib/gfx_msgqueue.inc
index 92458dde..20fed3ff 100644
--- a/src/corelib/gfx_msgqueue.inc
+++ b/src/corelib/gfx_msgqueue.inc
@@ -157,12 +157,12 @@ begin
m.Dest := Dest;
m.Params := aparams;
- try
+// try
m.Dest.Dispatch(m)
- except
- on E: Exception do
- {$IFDEF DEBUG}writeln('fpgSendMessage Caught Exception: ' + E.Message){$ENDIF};
- end;
+// except
+// on E: Exception do
+// {$IFDEF DEBUG}writeln('fpgSendMessage Caught Exception: ' + E.Message){$ENDIF};
+// end;
end;
procedure fpgSendMessage(Sender, Dest: TObject; MsgCode: integer); overload;
@@ -176,12 +176,12 @@ begin
m.Sender := Sender;
m.Dest := Dest;
- try
+// try
m.Dest.Dispatch(m)
- except
- on E: Exception do
- {$IFDEF DEBUG}writeln('fpgSendMessage Caught Exception: ' + E.Message){$ENDIF};
- end;
+// except
+// on E: Exception do
+// {$IFDEF DEBUG}writeln('fpgSendMessage Caught Exception: ' + E.Message){$ENDIF};
+// end;
end;
procedure fpgDeliverMessage(var msg: TfpgMessageRec);
@@ -193,7 +193,7 @@ begin
msg.Dest.Free
else
begin
- try
+// try
msg.Dest.Dispatch(msg);
if fpgApplication.FMessageHookList.Count > 0 then
begin
@@ -204,10 +204,10 @@ begin
oItem.Listener.Dispatch(msg);
end;
end;
- except
- on E: Exception do
- {$IFDEF DEBUG}writeln('fpgDeliverMessage Caught Exception: ' + E.Message){$ENDIF};
- end;
+// except
+// on E: Exception do
+// {$IFDEF DEBUG}writeln('fpgDeliverMessage Caught Exception: ' + E.Message){$ENDIF};
+// end;
end;
end;