1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: Assorted wxLogXXX functions, and wxLog (sink for logs)
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
17 // ----------------------------------------------------------------------------
19 // ----------------------------------------------------------------------------
21 // NB: this is needed even if wxUSE_LOG == 0
22 typedef unsigned long wxLogLevel
;
24 // the trace masks have been superseded by symbolic trace constants, they're
25 // for compatibility only and will be removed soon - do NOT use them
26 #if WXWIN_COMPATIBILITY_2_8
27 #define wxTraceMemAlloc 0x0001 // trace memory allocation (new/delete)
28 #define wxTraceMessages 0x0002 // trace window messages/X callbacks
29 #define wxTraceResAlloc 0x0004 // trace GDI resource allocation
30 #define wxTraceRefCount 0x0008 // trace various ref counting operations
33 #define wxTraceOleCalls 0x0100 // OLE interface calls
36 typedef unsigned long wxTraceMask
;
37 #endif // WXWIN_COMPATIBILITY_2_8
39 // ----------------------------------------------------------------------------
41 // ----------------------------------------------------------------------------
43 #include "wx/string.h"
44 #include "wx/strvararg.h"
46 // ----------------------------------------------------------------------------
47 // forward declarations
48 // ----------------------------------------------------------------------------
50 class WXDLLIMPEXP_FWD_BASE wxObject
;
53 class WXDLLIMPEXP_FWD_CORE wxFrame
;
58 #include "wx/arrstr.h"
62 #include <time.h> // for time_t
64 #endif // ! __WXPALMOS5__
66 #include "wx/dynarray.h"
67 #include "wx/hashmap.h"
70 #include "wx/thread.h"
71 #endif // wxUSE_THREADS
73 // wxUSE_LOG_DEBUG enables the debug log messages
74 #ifndef wxUSE_LOG_DEBUG
76 #define wxUSE_LOG_DEBUG 1
77 #else // !wxDEBUG_LEVEL
78 #define wxUSE_LOG_DEBUG 0
82 // wxUSE_LOG_TRACE enables the trace messages, they are disabled by default
83 #ifndef wxUSE_LOG_TRACE
85 #define wxUSE_LOG_TRACE 1
86 #else // !wxDEBUG_LEVEL
87 #define wxUSE_LOG_TRACE 0
89 #endif // wxUSE_LOG_TRACE
91 // wxLOG_COMPONENT identifies the component which generated the log record and
92 // can be #define'd to a user-defined value when compiling the user code to use
93 // component-based filtering (see wxLog::SetComponentLevel())
94 #ifndef wxLOG_COMPONENT
95 // this is a variable and not a macro in order to allow the user code to
96 // just #define wxLOG_COMPONENT without #undef'ining it first
97 extern WXDLLIMPEXP_DATA_BASE(const char *) wxLOG_COMPONENT
;
100 #define wxLOG_COMPONENT "wx"
104 // ----------------------------------------------------------------------------
106 // ----------------------------------------------------------------------------
108 // different standard log levels (you may also define your own)
109 enum wxLogLevelValues
111 wxLOG_FatalError
, // program can't continue, abort immediately
112 wxLOG_Error
, // a serious error, user must be informed about it
113 wxLOG_Warning
, // user is normally informed about it but may be ignored
114 wxLOG_Message
, // normal message (i.e. normal output of a non GUI app)
115 wxLOG_Status
, // informational: might go to the status line of GUI app
116 wxLOG_Info
, // informational message (a.k.a. 'Verbose')
117 wxLOG_Debug
, // never shown to the user, disabled in release mode
118 wxLOG_Trace
, // trace messages are also only enabled in debug mode
119 wxLOG_Progress
, // used for progress indicator (not yet)
120 wxLOG_User
= 100, // user defined levels start here
124 // symbolic trace masks - wxLogTrace("foo", "some trace message...") will be
125 // discarded unless the string "foo" has been added to the list of allowed
126 // ones with AddTraceMask()
128 #define wxTRACE_MemAlloc wxT("memalloc") // trace memory allocation (new/delete)
129 #define wxTRACE_Messages wxT("messages") // trace window messages/X callbacks
130 #define wxTRACE_ResAlloc wxT("resalloc") // trace GDI resource allocation
131 #define wxTRACE_RefCount wxT("refcount") // trace various ref counting operations
134 #define wxTRACE_OleCalls wxT("ole") // OLE interface calls
137 #include "wx/iosfwrap.h"
139 // ----------------------------------------------------------------------------
140 // information about a log record, i.e. unit of log output
141 // ----------------------------------------------------------------------------
143 class wxLogRecordInfo
146 // default ctor creates an uninitialized object
149 memset(this, 0, sizeof(*this));
152 // normal ctor, used by wxLogger specifies the location of the log
153 // statement; its time stamp and thread id are set up here
154 wxLogRecordInfo(const char *filename_
,
157 const char *component_
)
159 filename
= filename_
;
162 component
= component_
;
164 timestamp
= time(NULL
);
167 threadId
= wxThread::GetCurrentId();
168 #endif // wxUSE_THREADS
173 // we need to define copy ctor and assignment operator because of m_data
174 wxLogRecordInfo(const wxLogRecordInfo
& other
)
179 wxLogRecordInfo
& operator=(const wxLogRecordInfo
& other
)
181 if ( &other
!= this )
190 // dtor is non-virtual, this class is not meant to be derived from
197 // the file name and line number of the file where the log record was
198 // generated, if available or NULL and 0 otherwise
199 const char *filename
;
202 // the name of the function where the log record was generated (may be NULL
203 // if the compiler doesn't support __FUNCTION__)
206 // the name of the component which generated this message, may be NULL if
207 // not set (i.e. wxLOG_COMPONENT not defined)
208 const char *component
;
210 // time of record generation
214 // id of the thread which logged this record
215 wxThreadIdType threadId
;
216 #endif // wxUSE_THREADS
219 // store an arbitrary value in this record context
221 // wxWidgets always uses keys starting with "wx.", e.g. "wx.sys_error"
222 void StoreValue(const wxString
& key
, wxUIntPtr val
)
225 m_data
= new ExtraData
;
227 m_data
->numValues
[key
] = val
;
230 void StoreValue(const wxString
& key
, const wxString
& val
)
233 m_data
= new ExtraData
;
235 m_data
->strValues
[key
] = val
;
239 // these functions retrieve the value of either numeric or string key,
240 // return false if not found
241 bool GetNumValue(const wxString
& key
, wxUIntPtr
*val
) const
246 wxStringToNumHashMap::const_iterator it
= m_data
->numValues
.find(key
);
247 if ( it
== m_data
->numValues
.end() )
255 bool GetStrValue(const wxString
& key
, wxString
*val
) const
260 wxStringToStringHashMap::const_iterator it
= m_data
->strValues
.find(key
);
261 if ( it
== m_data
->strValues
.end() )
270 void Copy(const wxLogRecordInfo
& other
)
272 memcpy(this, &other
, sizeof(*this));
274 m_data
= new ExtraData(*other
.m_data
);
277 // extra data associated with the log record: this is completely optional
278 // and can be used to pass information from the log function to the log
279 // sink (e.g. wxLogSysError() uses this to pass the error code)
282 wxStringToNumHashMap numValues
;
283 wxStringToStringHashMap strValues
;
290 #define wxLOG_KEY_TRACE_MASK "wx.trace_mask"
292 // ----------------------------------------------------------------------------
293 // log record: a unit of log output
294 // ----------------------------------------------------------------------------
298 wxLogRecord(wxLogLevel level_
,
299 const wxString
& msg_
,
300 const wxLogRecordInfo
& info_
)
309 wxLogRecordInfo info
;
312 // ----------------------------------------------------------------------------
313 // derive from this class to redirect (or suppress, or ...) log messages
314 // normally, only a single instance of this class exists but it's not enforced
315 // ----------------------------------------------------------------------------
317 class WXDLLIMPEXP_BASE wxLog
323 // make dtor virtual for all derived classes
327 // log messages selection
328 // ----------------------
330 // these functions allow to completely disable all log messages or disable
331 // log messages at level less important than specified for the current
334 // is logging enabled at all now?
335 static bool IsEnabled()
338 if ( !wxThread::IsMain() )
339 return IsThreadLoggingEnabled();
340 #endif // wxUSE_THREADS
345 // change the flag state, return the previous one
346 static bool EnableLogging(bool enable
= true)
349 if ( !wxThread::IsMain() )
350 return EnableThreadLogging(enable
);
351 #endif // wxUSE_THREADS
353 bool doLogOld
= ms_doLog
;
358 // return the current global log level
359 static wxLogLevel
GetLogLevel() { return ms_logLevel
; }
361 // set global log level: messages with level > logLevel will not be logged
362 static void SetLogLevel(wxLogLevel logLevel
) { ms_logLevel
= logLevel
; }
364 // set the log level for the given component
365 static void SetComponentLevel(const wxString
& component
, wxLogLevel level
);
367 // return the effective log level for this component, falling back to
368 // parent component and to the default global log level if necessary
370 // NB: component argument is passed by value and not const reference in an
371 // attempt to encourage compiler to avoid an extra copy: as we modify
372 // the component internally, we'd create one anyhow and like this it
373 // can be avoided if the string is a temporary anyhow
374 static wxLogLevel
GetComponentLevel(wxString component
);
377 // is logging of messages from this component enabled at this level?
379 // usually always called with wxLOG_COMPONENT as second argument
380 static bool IsLevelEnabled(wxLogLevel level
, wxString component
)
382 return IsEnabled() && level
<= GetComponentLevel(component
);
386 // enable/disable messages at wxLOG_Verbose level (only relevant if the
387 // current log level is greater or equal to it)
389 // notice that verbose mode can be activated by the standard command-line
390 // '--verbose' option
391 static void SetVerbose(bool bVerbose
= true) { ms_bVerbose
= bVerbose
; }
393 // check if verbose messages are enabled
394 static bool GetVerbose() { return ms_bVerbose
; }
400 // flush shows all messages if they're not logged immediately (FILE
401 // and iostream logs don't need it, but wxLogGui does to avoid showing
402 // 17 modal dialogs one after another)
403 virtual void Flush();
405 // flush the active target if any and also output any pending messages from
406 // background threads
407 static void FlushActive();
409 // only one sink is active at each moment get current log target, will call
410 // wxAppTraits::CreateLogTarget() to create one if none exists
411 static wxLog
*GetActiveTarget();
413 // change log target, logger may be NULL
414 static wxLog
*SetActiveTarget(wxLog
*logger
);
417 // change log target for the current thread only, shouldn't be called from
418 // the main thread as it doesn't use thread-specific log target
419 static wxLog
*SetThreadActiveTarget(wxLog
*logger
);
420 #endif // wxUSE_THREADS
422 // suspend the message flushing of the main target until the next call
423 // to Resume() - this is mainly for internal use (to prevent wxYield()
424 // from flashing the messages)
425 static void Suspend() { ms_suspendCount
++; }
427 // must be called for each Suspend()!
428 static void Resume() { ms_suspendCount
--; }
430 // should GetActiveTarget() try to create a new log object if the
432 static void DontCreateOnDemand();
434 // Make GetActiveTarget() create a new log object again.
435 static void DoCreateOnDemand();
437 // log the count of repeating messages instead of logging the messages
439 static void SetRepetitionCounting(bool bRepetCounting
= true)
440 { ms_bRepetCounting
= bRepetCounting
; }
442 // gets duplicate counting status
443 static bool GetRepetitionCounting() { return ms_bRepetCounting
; }
445 // add string trace mask
446 static void AddTraceMask(const wxString
& str
);
448 // add string trace mask
449 static void RemoveTraceMask(const wxString
& str
);
451 // remove all string trace masks
452 static void ClearTraceMasks();
454 // get string trace masks: note that this is MT-unsafe if other threads can
455 // call AddTraceMask() concurrently
456 static const wxArrayString
& GetTraceMasks() { return ms_aTraceMasks
; }
458 // sets the time stamp string format: this is used as strftime() format
459 // string for the log targets which add time stamps to the messages; set
460 // it to empty string to disable time stamping completely.
461 static void SetTimestamp(const wxString
& ts
) { ms_timestamp
= ts
; }
463 // disable time stamping of log messages
464 static void DisableTimestamp() { SetTimestamp(wxEmptyString
); }
467 // is this trace mask in the list?
468 static bool IsAllowedTraceMask(const wxString
& mask
);
470 // get the current timestamp format string (maybe empty)
471 static const wxString
& GetTimestamp() { return ms_timestamp
; }
475 // helpers: all functions in this section are mostly for internal use only,
476 // don't call them from your code even if they are not formally deprecated
478 // put the time stamp into the string if ms_timestamp != NULL (don't
479 // change it otherwise)
480 static void TimeStamp(wxString
*str
);
482 // these methods should only be called from derived classes DoLogRecord(),
483 // DoLogTextAtLevel() and DoLogText() implementations respectively and
484 // shouldn't be called directly, use logging functions instead
485 void LogRecord(wxLogLevel level
,
487 const wxLogRecordInfo
& info
)
489 DoLogRecord(level
, msg
, info
);
492 void LogTextAtLevel(wxLogLevel level
, const wxString
& msg
)
494 DoLogTextAtLevel(level
, msg
);
497 void LogText(const wxString
& msg
)
502 // this is a helper used by wxLogXXX() functions, don't call it directly
503 // and see DoLog() for function to overload in the derived classes
504 static void OnLog(wxLogLevel level
,
506 const wxLogRecordInfo
& info
);
508 // version called when no information about the location of the log record
509 // generation is available (but the time stamp is), it mainly exists for
510 // backwards compatibility, don't use it in new code
511 static void OnLog(wxLogLevel level
, const wxString
& msg
, time_t t
);
513 // a helper calling the above overload with current time
514 static void OnLog(wxLogLevel level
, const wxString
& msg
)
516 OnLog(level
, msg
, time(NULL
));
520 // this method exists for backwards compatibility only, don't use
521 bool HasPendingMessages() const { return true; }
523 #if WXWIN_COMPATIBILITY_2_6
524 // this function doesn't do anything any more, don't call it
526 static wxChar
*SetLogBuffer(wxChar
*, size_t = 0), return NULL
;
528 #endif // WXWIN_COMPATIBILITY_2_6
530 // don't use integer masks any more, use string trace masks instead
531 #if WXWIN_COMPATIBILITY_2_8
532 wxDEPRECATED_INLINE( static void SetTraceMask(wxTraceMask ulMask
),
533 ms_ulTraceMask
= ulMask
; )
535 // this one can't be marked deprecated as it's used in our own wxLogger
536 // below but it still is deprecated and shouldn't be used
537 static wxTraceMask
GetTraceMask() { return ms_ulTraceMask
; }
538 #endif // WXWIN_COMPATIBILITY_2_8
541 // the logging functions that can be overridden: DoLogRecord() is called
542 // for every "record", i.e. a unit of log output, to be logged and by
543 // default formats the message and passes it to DoLogTextAtLevel() which in
544 // turn passes it to DoLogText() by default
546 // override this method if you want to change message formatting or do
548 virtual void DoLogRecord(wxLogLevel level
,
550 const wxLogRecordInfo
& info
);
552 // override this method to redirect output to different channels depending
553 // on its level only; if even the level doesn't matter, override
554 // DoLogText() instead
555 virtual void DoLogTextAtLevel(wxLogLevel level
, const wxString
& msg
);
557 // this function is not pure virtual as it might not be needed if you do
558 // the logging in overridden DoLogRecord() or DoLogTextAtLevel() directly
559 // but if you do not override them in your derived class you must override
560 // this one as the default implementation of it simply asserts
561 virtual void DoLogText(const wxString
& msg
);
564 // the rest of the functions are for backwards compatibility only, don't
565 // use them in new code; if you're updating your existing code you need to
566 // switch to overriding DoLogRecord/Text() above (although as long as these
567 // functions exist, log classes using them will continue to work)
568 #if WXWIN_COMPATIBILITY_2_8
569 wxDEPRECATED_BUT_USED_INTERNALLY(
570 virtual void DoLog(wxLogLevel level
, const char *szString
, time_t t
)
573 wxDEPRECATED_BUT_USED_INTERNALLY(
574 virtual void DoLog(wxLogLevel level
, const wchar_t *wzString
, time_t t
)
577 // these shouldn't be used by new code
578 wxDEPRECATED_BUT_USED_INTERNALLY_INLINE(
579 virtual void DoLogString(const char *WXUNUSED(szString
),
581 wxEMPTY_PARAMETER_VALUE
584 wxDEPRECATED_BUT_USED_INTERNALLY_INLINE(
585 virtual void DoLogString(const wchar_t *WXUNUSED(wzString
),
587 wxEMPTY_PARAMETER_VALUE
589 #endif // WXWIN_COMPATIBILITY_2_8
592 // log a message indicating the number of times the previous message was
593 // repeated if previous repetition counter is strictly positive, does
594 // nothing otherwise; return the old value of repetition counter
595 unsigned LogLastRepeatIfNeeded();
599 // called from FlushActive() to really log any buffered messages logged
600 // from the other threads
601 void FlushThreadMessages();
603 // these functions are called for non-main thread only by IsEnabled() and
604 // EnableLogging() respectively
605 static bool IsThreadLoggingEnabled();
606 static bool EnableThreadLogging(bool enable
= true);
607 #endif // wxUSE_THREADS
609 // get the active log target for the main thread, auto-creating it if
612 // this is called from GetActiveTarget() and OnLog() when they're called
613 // from the main thread
614 static wxLog
*GetMainThreadActiveTarget();
616 // called from OnLog() if it's called from the main thread or if we have a
617 // (presumably MT-safe) thread-specific logger and by FlushThreadMessages()
618 // when it plays back the buffered messages logged from the other threads
619 void CallDoLogNow(wxLogLevel level
,
621 const wxLogRecordInfo
& info
);
627 // if true, don't log the same message multiple times, only log it once
628 // with the number of times it was repeated
629 static bool ms_bRepetCounting
;
631 static wxLog
*ms_pLogger
; // currently active log sink
632 static bool ms_doLog
; // false => all logging disabled
633 static bool ms_bAutoCreate
; // create new log targets on demand?
634 static bool ms_bVerbose
; // false => ignore LogInfo messages
636 static wxLogLevel ms_logLevel
; // limit logging to levels <= ms_logLevel
638 static size_t ms_suspendCount
; // if positive, logs are not flushed
640 // format string for strftime(), if empty, time stamping log messages is
642 static wxString ms_timestamp
;
644 #if WXWIN_COMPATIBILITY_2_8
645 static wxTraceMask ms_ulTraceMask
; // controls wxLogTrace behaviour
646 #endif // WXWIN_COMPATIBILITY_2_8
648 // currently enabled trace masks
649 static wxArrayString ms_aTraceMasks
;
652 // ----------------------------------------------------------------------------
653 // "trivial" derivations of wxLog
654 // ----------------------------------------------------------------------------
656 // log everything except for the debug/trace messages (which are passed to
657 // wxMessageOutputDebug) to a buffer
658 class WXDLLIMPEXP_BASE wxLogBuffer
: public wxLog
663 // get the string contents with all messages logged
664 const wxString
& GetBuffer() const { return m_str
; }
666 // show the buffer contents to the user in the best possible way (this uses
667 // wxMessageOutputMessageBox) and clear it
668 virtual void Flush();
671 virtual void DoLogTextAtLevel(wxLogLevel level
, const wxString
& msg
);
676 wxDECLARE_NO_COPY_CLASS(wxLogBuffer
);
680 // log everything to a "FILE *", stderr by default
681 class WXDLLIMPEXP_BASE wxLogStderr
: public wxLog
684 // redirect log output to a FILE
685 wxLogStderr(FILE *fp
= NULL
);
688 // implement sink function
689 virtual void DoLogText(const wxString
& msg
);
693 wxDECLARE_NO_COPY_CLASS(wxLogStderr
);
696 #if wxUSE_STD_IOSTREAM
698 // log everything to an "ostream", cerr by default
699 class WXDLLIMPEXP_BASE wxLogStream
: public wxLog
702 // redirect log output to an ostream
703 wxLogStream(wxSTD ostream
*ostr
= (wxSTD ostream
*) NULL
);
706 // implement sink function
707 virtual void DoLogText(const wxString
& msg
);
709 // using ptr here to avoid including <iostream.h> from this file
710 wxSTD ostream
*m_ostr
;
713 #endif // wxUSE_STD_IOSTREAM
715 // ----------------------------------------------------------------------------
716 // /dev/null log target: suppress logging until this object goes out of scope
717 // ----------------------------------------------------------------------------
725 // wxFile.Open() normally complains if file can't be opened, we don't
729 if ( !file.Open("bar") )
730 ... process error ourselves ...
732 // ~wxLogNull called, old log sink restored
735 class WXDLLIMPEXP_BASE wxLogNull
738 wxLogNull() : m_flagOld(wxLog::EnableLogging(false)) { }
739 ~wxLogNull() { (void)wxLog::EnableLogging(m_flagOld
); }
742 bool m_flagOld
; // the previous value of the wxLog::ms_doLog
745 // ----------------------------------------------------------------------------
746 // chaining log target: installs itself as a log target and passes all
747 // messages to the real log target given to it in the ctor but also forwards
748 // them to the previously active one
750 // note that you don't have to call SetActiveTarget() with this class, it
751 // does it itself in its ctor
752 // ----------------------------------------------------------------------------
754 class WXDLLIMPEXP_BASE wxLogChain
: public wxLog
757 wxLogChain(wxLog
*logger
);
758 virtual ~wxLogChain();
760 // change the new log target
761 void SetLog(wxLog
*logger
);
763 // this can be used to temporarily disable (and then reenable) passing
764 // messages to the old logger (by default we do pass them)
765 void PassMessages(bool bDoPass
) { m_bPassMessages
= bDoPass
; }
767 // are we passing the messages to the previous log target?
768 bool IsPassingMessages() const { return m_bPassMessages
; }
770 // return the previous log target (may be NULL)
771 wxLog
*GetOldLog() const { return m_logOld
; }
773 // override base class version to flush the old logger as well
774 virtual void Flush();
776 // call to avoid destroying the old log target
777 void DetachOldLog() { m_logOld
= NULL
; }
780 // pass the record to the old logger if needed
781 virtual void DoLogRecord(wxLogLevel level
,
783 const wxLogRecordInfo
& info
);
786 // the current log target
789 // the previous log target
792 // do we pass the messages to the old logger?
793 bool m_bPassMessages
;
795 wxDECLARE_NO_COPY_CLASS(wxLogChain
);
798 // a chain log target which uses itself as the new logger
800 #define wxLogPassThrough wxLogInterposer
802 class WXDLLIMPEXP_BASE wxLogInterposer
: public wxLogChain
808 wxDECLARE_NO_COPY_CLASS(wxLogInterposer
);
811 // a temporary interposer which doesn't destroy the old log target
812 // (calls DetachOldLog)
814 class WXDLLIMPEXP_BASE wxLogInterposerTemp
: public wxLogChain
817 wxLogInterposerTemp();
820 wxDECLARE_NO_COPY_CLASS(wxLogInterposerTemp
);
824 // include GUI log targets:
825 #include "wx/generic/logg.h"
828 // ----------------------------------------------------------------------------
830 // ----------------------------------------------------------------------------
832 // wxLogger is a helper class used by wxLogXXX() functions implementation,
833 // don't use it directly as it's experimental and subject to change (OTOH it
834 // might become public in the future if it's deemed to be useful enough)
836 // contains information about the context from which a log message originates
837 // and provides Log() vararg method which forwards to wxLog::OnLog() and passes
838 // this context to it
842 // ctor takes the basic information about the log record
843 wxLogger(wxLogLevel level
,
844 const char *filename
,
847 const char *component
)
849 m_info(filename
, line
, func
, component
)
853 // store extra data in our log record and return this object itself (so
854 // that further calls to its functions could be chained)
855 template <typename T
>
856 wxLogger
& Store(const wxString
& key
, T val
)
858 m_info
.StoreValue(key
, val
);
862 // hack for "overloaded" wxLogXXX() functions: calling this method
863 // indicates that we may have an extra first argument preceding the format
864 // string and that if we do have it, we should store it in m_info using the
865 // given key (while by default 0 value will be used)
866 wxLogger
& MaybeStore(const wxString
& key
, wxUIntPtr value
= 0)
868 wxASSERT_MSG( m_optKey
.empty(), "can only have one optional value" );
871 m_info
.StoreValue(key
, value
);
876 // non-vararg function used by wxVLogXXX():
878 // log the message at the level specified in the ctor if this log message
880 void LogV(const wxString
& format
, va_list argptr
)
882 // remember that fatal errors can't be disabled
883 if ( m_level
== wxLOG_FatalError
||
884 wxLog::IsLevelEnabled(m_level
, m_info
.component
) )
885 DoCallOnLog(format
, argptr
);
888 // overloads used by functions with optional leading arguments (whose
889 // values are stored in the key passed to MaybeStore())
890 void LogV(long num
, const wxString
& format
, va_list argptr
)
892 Store(m_optKey
, num
);
894 LogV(format
, argptr
);
897 void LogV(void *ptr
, const wxString
& format
, va_list argptr
)
899 Store(m_optKey
, wxPtrToUInt(ptr
));
901 LogV(format
, argptr
);
904 void LogVTrace(const wxString
& mask
, const wxString
& format
, va_list argptr
)
906 if ( !wxLog::IsAllowedTraceMask(mask
) )
909 Store(wxLOG_KEY_TRACE_MASK
, mask
);
911 LogV(format
, argptr
);
915 // vararg functions used by wxLogXXX():
917 // will log the message at the level specified in the ctor
919 // notice that this function supposes that the caller already checked that
920 // the level was enabled and does no checks itself
921 WX_DEFINE_VARARG_FUNC_VOID
924 1, (const wxFormatString
&),
928 // same as Log() but with an extra numeric or pointer parameters: this is
929 // used to pass an optional value by storing it in m_info under the name
930 // passed to MaybeStore() and is required to support "overloaded" versions
931 // of wxLogStatus() and wxLogSysError()
932 WX_DEFINE_VARARG_FUNC_VOID
935 2, (long, const wxFormatString
&),
936 DoLogWithNum
, DoLogWithNumUtf8
939 // unfortunately we can't use "void *" here as we get overload ambiguities
940 // with Log(wxFormatString, ...) when the first argument is a "char *" or
941 // "wchar_t *" then -- so we only allow passing wxObject here, which is
942 // ugly but fine in practice as this overload is only used by wxLogStatus()
943 // whose first argument is a wxFrame
944 WX_DEFINE_VARARG_FUNC_VOID
947 2, (wxObject
*, const wxFormatString
&),
948 DoLogWithPtr
, DoLogWithPtrUtf8
951 // log the message at the level specified as its first argument
953 // as the macros don't have access to the level argument in this case, this
954 // function does check that the level is enabled itself
955 WX_DEFINE_VARARG_FUNC_VOID
958 2, (wxLogLevel
, const wxFormatString
&),
959 DoLogAtLevel
, DoLogAtLevelUtf8
962 // special versions for wxLogTrace() which is passed either string or
963 // integer mask as first argument determining whether the message should be
965 WX_DEFINE_VARARG_FUNC_VOID
968 2, (const wxString
&, const wxFormatString
&),
969 DoLogTrace
, DoLogTraceUtf8
972 #if WXWIN_COMPATIBILITY_2_8
973 WX_DEFINE_VARARG_FUNC_VOID
976 2, (wxTraceMask
, const wxFormatString
&),
977 DoLogTraceMask
, DoLogTraceMaskUtf8
979 #endif // WXWIN_COMPATIBILITY_2_8
982 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
983 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
984 1, (const wxString
&),
985 (wxFormatString(f1
)))
986 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
987 1, (const wxCStrData
&),
988 (wxFormatString(f1
)))
989 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
991 (wxFormatString(f1
)))
992 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
994 (wxFormatString(f1
)))
996 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
997 2, (long, const wxString
&),
998 (f1
, wxFormatString(f2
)))
999 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1000 2, (long, const wxCStrData
&),
1001 (f1
, wxFormatString(f2
)))
1002 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1003 2, (long, const char *),
1004 (f1
, wxFormatString(f2
)))
1005 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1006 2, (long, const wchar_t *),
1007 (f1
, wxFormatString(f2
)))
1009 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1010 2, (wxObject
*, const wxString
&),
1011 (f1
, wxFormatString(f2
)))
1012 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1013 2, (wxObject
*, const wxCStrData
&),
1014 (f1
, wxFormatString(f2
)))
1015 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1016 2, (wxObject
*, const char *),
1017 (f1
, wxFormatString(f2
)))
1018 WX_VARARG_WATCOM_WORKAROUND(void, Log
,
1019 2, (wxObject
*, const wchar_t *),
1020 (f1
, wxFormatString(f2
)))
1022 WX_VARARG_WATCOM_WORKAROUND(void, LogAtLevel
,
1023 2, (wxLogLevel
, const wxString
&),
1024 (f1
, wxFormatString(f2
)))
1025 WX_VARARG_WATCOM_WORKAROUND(void, LogAtLevel
,
1026 2, (wxLogLevel
, const wxCStrData
&),
1027 (f1
, wxFormatString(f2
)))
1028 WX_VARARG_WATCOM_WORKAROUND(void, LogAtLevel
,
1029 2, (wxLogLevel
, const char *),
1030 (f1
, wxFormatString(f2
)))
1031 WX_VARARG_WATCOM_WORKAROUND(void, LogAtLevel
,
1032 2, (wxLogLevel
, const wchar_t *),
1033 (f1
, wxFormatString(f2
)))
1035 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1036 2, (const wxString
&, const wxString
&),
1037 (f1
, wxFormatString(f2
)))
1038 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1039 2, (const wxString
&, const wxCStrData
&),
1040 (f1
, wxFormatString(f2
)))
1041 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1042 2, (const wxString
&, const char *),
1043 (f1
, wxFormatString(f2
)))
1044 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1045 2, (const wxString
&, const wchar_t *),
1046 (f1
, wxFormatString(f2
)))
1048 #if WXWIN_COMPATIBILITY_2_8
1049 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1050 2, (wxTraceMask
, wxTraceMask
),
1051 (f1
, wxFormatString(f2
)))
1052 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1053 2, (wxTraceMask
, const wxCStrData
&),
1054 (f1
, wxFormatString(f2
)))
1055 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1056 2, (wxTraceMask
, const char *),
1057 (f1
, wxFormatString(f2
)))
1058 WX_VARARG_WATCOM_WORKAROUND(void, LogTrace
,
1059 2, (wxTraceMask
, const wchar_t *),
1060 (f1
, wxFormatString(f2
)))
1061 #endif // WXWIN_COMPATIBILITY_2_8
1062 #endif // __WATCOMC__
1065 #if !wxUSE_UTF8_LOCALE_ONLY
1066 void DoLog(const wxChar
*format
, ...)
1069 va_start(argptr
, format
);
1070 DoCallOnLog(format
, argptr
);
1074 void DoLogWithNum(long num
, const wxChar
*format
, ...)
1076 Store(m_optKey
, num
);
1079 va_start(argptr
, format
);
1080 DoCallOnLog(format
, argptr
);
1084 void DoLogWithPtr(void *ptr
, const wxChar
*format
, ...)
1086 Store(m_optKey
, wxPtrToUInt(ptr
));
1089 va_start(argptr
, format
);
1090 DoCallOnLog(format
, argptr
);
1094 void DoLogAtLevel(wxLogLevel level
, const wxChar
*format
, ...)
1096 if ( !wxLog::IsLevelEnabled(level
, m_info
.component
) )
1100 va_start(argptr
, format
);
1101 DoCallOnLog(level
, format
, argptr
);
1105 void DoLogTrace(const wxString
& mask
, const wxChar
*format
, ...)
1107 if ( !wxLog::IsAllowedTraceMask(mask
) )
1110 Store(wxLOG_KEY_TRACE_MASK
, mask
);
1113 va_start(argptr
, format
);
1114 DoCallOnLog(format
, argptr
);
1118 #if WXWIN_COMPATIBILITY_2_8
1119 void DoLogTraceMask(wxTraceMask mask
, const wxChar
*format
, ...)
1121 if ( (wxLog::GetTraceMask() & mask
) != mask
)
1124 Store(wxLOG_KEY_TRACE_MASK
, mask
);
1127 va_start(argptr
, format
);
1128 DoCallOnLog(format
, argptr
);
1131 #endif // WXWIN_COMPATIBILITY_2_8
1132 #endif // !wxUSE_UTF8_LOCALE_ONLY
1134 #if wxUSE_UNICODE_UTF8
1135 void DoLogUtf8(const char *format
, ...)
1138 va_start(argptr
, format
);
1139 DoCallOnLog(format
, argptr
);
1143 void DoLogWithNumUtf8(long num
, const char *format
, ...)
1145 Store(m_optKey
, num
);
1148 va_start(argptr
, format
);
1149 DoCallOnLog(format
, argptr
);
1153 void DoLogWithPtrUtf8(void *ptr
, const char *format
, ...)
1155 Store(m_optKey
, wxPtrToUInt(ptr
));
1158 va_start(argptr
, format
);
1159 DoCallOnLog(format
, argptr
);
1163 void DoLogAtLevelUtf8(wxLogLevel level
, const char *format
, ...)
1165 if ( !wxLog::IsLevelEnabled(level
, m_info
.component
) )
1169 va_start(argptr
, format
);
1170 DoCallOnLog(level
, format
, argptr
);
1174 void DoLogTraceUtf8(const wxString
& mask
, const char *format
, ...)
1176 if ( !wxLog::IsAllowedTraceMask(mask
) )
1179 Store(wxLOG_KEY_TRACE_MASK
, mask
);
1182 va_start(argptr
, format
);
1183 DoCallOnLog(format
, argptr
);
1187 #if WXWIN_COMPATIBILITY_2_8
1188 void DoLogTraceMaskUtf8(wxTraceMask mask
, const char *format
, ...)
1190 if ( (wxLog::GetTraceMask() & mask
) != mask
)
1193 Store(wxLOG_KEY_TRACE_MASK
, mask
);
1196 va_start(argptr
, format
);
1197 DoCallOnLog(format
, argptr
);
1200 #endif // WXWIN_COMPATIBILITY_2_8
1201 #endif // wxUSE_UNICODE_UTF8
1203 void DoCallOnLog(wxLogLevel level
, const wxString
& format
, va_list argptr
)
1205 wxLog::OnLog(level
, wxString::FormatV(format
, argptr
), m_info
);
1208 void DoCallOnLog(const wxString
& format
, va_list argptr
)
1210 DoCallOnLog(m_level
, format
, argptr
);
1214 const wxLogLevel m_level
;
1215 wxLogRecordInfo m_info
;
1219 wxDECLARE_NO_COPY_CLASS(wxLogger
);
1222 // ============================================================================
1224 // ============================================================================
1226 // ----------------------------------------------------------------------------
1227 // get error code/error message from system in a portable way
1228 // ----------------------------------------------------------------------------
1230 // return the last system error code
1231 WXDLLIMPEXP_BASE
unsigned long wxSysErrorCode();
1233 // return the error message for given (or last if 0) error code
1234 WXDLLIMPEXP_BASE
const wxChar
* wxSysErrorMsg(unsigned long nErrCode
= 0);
1236 // ----------------------------------------------------------------------------
1237 // define wxLog<level>() functions which can be used by application instead of
1238 // stdio, iostream &c for log messages for easy redirection
1239 // ----------------------------------------------------------------------------
1242 The code below is unreadable because it (unfortunately unavoidably)
1243 contains a lot of macro magic but all it does is to define wxLogXXX() such
1244 that you can call them as vararg functions to log a message at the
1245 corresponding level.
1247 More precisely, it defines:
1249 - wxLog{FatalError,Error,Warning,Message,Verbose,Debug}() functions
1250 taking the format string and additional vararg arguments if needed.
1251 - wxLogGeneric(wxLogLevel level, const wxString& format, ...) which
1252 takes the log level explicitly.
1253 - wxLogSysError(const wxString& format, ...) and wxLogSysError(long
1254 err, const wxString& format, ...) which log a wxLOG_Error severity
1255 message with the error message corresponding to the system error code
1256 err or the last error.
1257 - wxLogStatus(const wxString& format, ...) which logs the message into
1258 the status bar of the main application window and its overload
1259 wxLogStatus(wxFrame *frame, const wxString& format, ...) which logs it
1260 into the status bar of the specified frame.
1261 - wxLogTrace(Mask mask, const wxString& format, ...) which only logs
1262 the message is the specified mask is enabled. This comes in two kinds:
1263 Mask can be a wxString or a long. Both are deprecated.
1265 In addition, wxVLogXXX() versions of all the functions above are also
1266 defined. They take a va_list argument instead of "...".
1269 // creates wxLogger object for the current location
1270 #define wxMAKE_LOGGER(level) \
1271 wxLogger(wxLOG_##level, __FILE__, __LINE__, __WXFUNCTION__, wxLOG_COMPONENT)
1273 // this macro generates the expression which logs whatever follows it in
1274 // parentheses at the level specified as argument
1275 #define wxDO_LOG(level) wxMAKE_LOGGER(level).Log
1277 // this is the non-vararg equivalent
1278 #define wxDO_LOGV(level, format, argptr) \
1279 wxMAKE_LOGGER(level).LogV(format, argptr)
1281 // this macro declares wxLog<level>() macro which logs whatever follows it if
1282 // logging at specified level is enabled (notice that if it is false, the
1283 // following arguments are not even evaluated which is good as it avoids
1284 // unnecessary overhead)
1286 // Note: the strange if/else construct is needed to make the following code
1289 // wxLogError("!!!");
1293 // work as expected, without it the second "else" would match the "if"
1294 // inside wxLogError(). Unfortunately code like
1297 // wxLogError("!!!");
1299 // now provokes "suggest explicit braces to avoid ambiguous 'else'"
1300 // warnings from g++ 4.3 and later with -Wparentheses on but they can be
1301 // easily fixed by adding curly braces around wxLogError() and at least
1302 // the code still does do the right thing.
1303 #define wxDO_LOG_IF_ENABLED(level) \
1304 if ( !wxLog::IsLevelEnabled(wxLOG_##level, wxLOG_COMPONENT) ) \
1309 // wxLogFatalError() is special as it can't be disabled
1310 #define wxLogFatalError wxDO_LOG(FatalError)
1311 #define wxVLogFatalError(format, argptr) wxDO_LOGV(FatalError, format, argptr)
1313 #define wxLogError wxDO_LOG_IF_ENABLED(Error)
1314 #define wxVLogError(format, argptr) wxDO_LOGV(Error, format, argptr)
1316 #define wxLogWarning wxDO_LOG_IF_ENABLED(Warning)
1317 #define wxVLogWarning(format, argptr) wxDO_LOGV(Warning, format, argptr)
1319 #define wxLogMessage wxDO_LOG_IF_ENABLED(Message)
1320 #define wxVLogMessage(format, argptr) wxDO_LOGV(Message, format, argptr)
1322 // this one is special as it only logs if we're in verbose mode
1323 #define wxLogVerbose \
1324 if ( !(wxLog::IsLevelEnabled(wxLOG_Info, wxLOG_COMPONENT) && \
1325 wxLog::GetVerbose()) ) \
1329 #define wxVLogVerbose(format, argptr) \
1330 if ( !(wxLog::IsLevelEnabled(wxLOG_Info, wxLOG_COMPONENT) && \
1331 wxLog::GetVerbose()) ) \
1334 wxDO_LOGV(Info, format, argptr)
1336 // deprecated synonyms for wxLogVerbose() and wxVLogVerbose()
1337 #define wxLogInfo wxLogVerbose
1338 #define wxVLogInfo wxVLogVerbose
1341 // another special case: the level is passed as first argument of the function
1342 // and so is not available to the macro
1344 // notice that because of this, arguments of wxLogGeneric() are currently
1345 // always evaluated, unlike for the other log functions
1346 #define wxLogGeneric wxMAKE_LOGGER(Max).LogAtLevel
1347 #define wxVLogGeneric(level, format, argptr) \
1348 if ( !wxLog::IsLevelEnabled(wxLOG_##level, wxLOG_COMPONENT) ) \
1351 wxDO_LOGV(level, format, argptr)
1354 // wxLogSysError() needs to stash the error code value in the log record info
1355 // so it needs special handling too; additional complications arise because the
1356 // error code may or not be present as the first argument
1358 // notice that we unfortunately can't avoid the call to wxSysErrorCode() even
1359 // though it may be unneeded if an explicit error code is passed to us because
1360 // the message might not be logged immediately (e.g. it could be queued for
1361 // logging from the main thread later) and so we can't to wait until it is
1362 // logged to determine whether we have last error or not as it will be too late
1363 // and it will have changed already by then (in fact it even changes when
1364 // wxString::Format() is called because of vsnprintf() inside it so it can
1365 // change even much sooner)
1366 #define wxLOG_KEY_SYS_ERROR_CODE "wx.sys_error"
1368 #define wxLogSysError \
1369 if ( !wxLog::IsLevelEnabled(wxLOG_Error, wxLOG_COMPONENT) ) \
1372 wxMAKE_LOGGER(Error).MaybeStore(wxLOG_KEY_SYS_ERROR_CODE, \
1373 wxSysErrorCode()).Log
1375 // unfortunately we can't have overloaded macros so we can't define versions
1376 // both with and without error code argument and have to rely on LogV()
1377 // overloads in wxLogger to select between them
1378 #define wxVLogSysError \
1379 wxMAKE_LOGGER(Error).MaybeStore(wxLOG_KEY_SYS_ERROR_CODE, \
1380 wxSysErrorCode()).LogV
1383 // wxLogStatus() is similar to wxLogSysError() as it allows to optionally
1384 // specify the frame to which the message should go
1385 #define wxLOG_KEY_FRAME "wx.frame"
1387 #define wxLogStatus \
1388 if ( !wxLog::IsLevelEnabled(wxLOG_Status, wxLOG_COMPONENT) ) \
1391 wxMAKE_LOGGER(Status).MaybeStore(wxLOG_KEY_FRAME).Log
1393 #define wxVLogStatus(format, argptr) \
1394 wxMAKE_LOGGER(Status).MaybeStore(wxLOG_KEY_FRAME).LogV
1400 #undef wxUSE_LOG_DEBUG
1401 #define wxUSE_LOG_DEBUG 0
1403 #undef wxUSE_LOG_TRACE
1404 #define wxUSE_LOG_TRACE 0
1406 #if defined(__WATCOMC__) || defined(__MINGW32__)
1407 // Mingw has similar problem with wxLogSysError:
1408 #define WX_WATCOM_OR_MINGW_ONLY_CODE( x ) x
1410 #define WX_WATCOM_OR_MINGW_ONLY_CODE( x )
1413 // define macros for defining log functions which do nothing at all
1415 // WX_WATCOM_ONLY_CODE is needed to work around
1416 // http://bugzilla.openwatcom.org/show_bug.cgi?id=351
1417 #define wxDEFINE_EMPTY_LOG_FUNCTION(level) \
1418 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 1, (const wxFormatString&)) \
1419 WX_WATCOM_ONLY_CODE( \
1420 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 1, (const char*)) \
1421 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 1, (const wchar_t*)) \
1422 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 1, (const wxCStrData&)) \
1424 inline void wxVLog##level(const wxFormatString& WXUNUSED(format), \
1425 va_list WXUNUSED(argptr)) { } \
1427 #define wxDEFINE_EMPTY_LOG_FUNCTION2(level, argclass) \
1428 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 2, (argclass, const wxFormatString&)) \
1429 WX_WATCOM_OR_MINGW_ONLY_CODE( \
1430 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 2, (argclass, const char*)) \
1431 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 2, (argclass, const wchar_t*)) \
1432 WX_DEFINE_VARARG_FUNC_NOP(wxLog##level, 2, (argclass, const wxCStrData&)) \
1434 inline void wxVLog##level(argclass WXUNUSED(arg), \
1435 const wxFormatString& WXUNUSED(format), \
1436 va_list WXUNUSED(argptr)) {}
1438 wxDEFINE_EMPTY_LOG_FUNCTION(FatalError
);
1439 wxDEFINE_EMPTY_LOG_FUNCTION(Error
);
1440 wxDEFINE_EMPTY_LOG_FUNCTION(SysError
);
1441 wxDEFINE_EMPTY_LOG_FUNCTION2(SysError
, long);
1442 wxDEFINE_EMPTY_LOG_FUNCTION(Warning
);
1443 wxDEFINE_EMPTY_LOG_FUNCTION(Message
);
1444 wxDEFINE_EMPTY_LOG_FUNCTION(Info
);
1445 wxDEFINE_EMPTY_LOG_FUNCTION(Verbose
);
1447 wxDEFINE_EMPTY_LOG_FUNCTION2(Generic
, wxLogLevel
);
1450 wxDEFINE_EMPTY_LOG_FUNCTION(Status
);
1451 wxDEFINE_EMPTY_LOG_FUNCTION2(Status
, wxFrame
*);
1454 // Empty Class to fake wxLogNull
1455 class WXDLLIMPEXP_BASE wxLogNull
1461 // Dummy macros to replace some functions.
1462 #define wxSysErrorCode() (unsigned long)0
1463 #define wxSysErrorMsg( X ) (const wxChar*)NULL
1465 // Fake symbolic trace masks... for those that are used frequently
1466 #define wxTRACE_OleCalls wxEmptyString // OLE interface calls
1468 #endif // wxUSE_LOG/!wxUSE_LOG
1471 // debug functions can be completely disabled in optimized builds
1473 // if these log functions are disabled, we prefer to define them as (empty)
1474 // variadic macros as this completely removes them and their argument
1475 // evaluation from the object code but if this is not supported by compiler we
1476 // use empty inline functions instead (defining them as nothing would result in
1477 // compiler warnings)
1479 // note that making wxVLogDebug/Trace() themselves (empty inline) functions is
1480 // a bad idea as some compilers are stupid enough to not inline even empty
1481 // functions if their parameters are complicated enough, but by defining them
1482 // as an empty inline function we ensure that even dumbest compilers optimise
1485 // but Borland gives "W8019: Code has no effect" for wxLogNop() so we need
1486 // to define it differently for it to avoid these warnings (same problem as
1487 // with wxUnusedVar())
1488 #define wxLogNop() { }
1490 inline void wxLogNop() { }
1494 #define wxLogDebug wxDO_LOG_IF_ENABLED(Debug)
1495 #define wxVLogDebug(format, argptr) wxDO_LOGV(Debug, format, argptr)
1496 #else // !wxUSE_LOG_DEBUG
1497 #define wxVLogDebug(fmt, valist) wxLogNop()
1499 #ifdef HAVE_VARIADIC_MACROS
1500 #define wxLogDebug(fmt, ...) wxLogNop()
1501 #else // !HAVE_VARIADIC_MACROS
1502 WX_DEFINE_VARARG_FUNC_NOP(wxLogDebug
, 1, (const wxFormatString
&))
1504 #endif // wxUSE_LOG_DEBUG/!wxUSE_LOG_DEBUG
1507 #define wxLogTrace \
1508 if ( !wxLog::IsLevelEnabled(wxLOG_Trace, wxLOG_COMPONENT) ) \
1511 wxMAKE_LOGGER(Trace).LogTrace
1512 #define wxVLogTrace \
1513 if ( !wxLog::IsLevelEnabled(wxLOG_Trace, wxLOG_COMPONENT) ) \
1516 wxMAKE_LOGGER(Trace).LogVTrace
1517 #else // !wxUSE_LOG_TRACE
1518 #define wxVLogTrace(mask, fmt, valist) wxLogNop()
1520 #ifdef HAVE_VARIADIC_MACROS
1521 #define wxLogTrace(mask, fmt, ...) wxLogNop()
1522 #else // !HAVE_VARIADIC_MACROS
1523 #if WXWIN_COMPATIBILITY_2_8
1524 WX_DEFINE_VARARG_FUNC_NOP(wxLogTrace
, 2, (wxTraceMask
, const wxFormatString
&))
1526 WX_DEFINE_VARARG_FUNC_NOP(wxLogTrace
, 2, (const wxString
&, const wxFormatString
&))
1528 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
1529 WX_DEFINE_VARARG_FUNC_NOP(wxLogTrace
, 2, (const char*, const char*))
1530 WX_DEFINE_VARARG_FUNC_NOP(wxLogTrace
, 2, (const wchar_t*, const wchar_t*))
1532 #endif // HAVE_VARIADIC_MACROS/!HAVE_VARIADIC_MACROS
1533 #endif // wxUSE_LOG_TRACE/!wxUSE_LOG_TRACE
1535 // wxLogFatalError helper: show the (fatal) error to the user in a safe way,
1536 // i.e. without using wxMessageBox() for example because it could crash
1537 void WXDLLIMPEXP_BASE
1538 wxSafeShowMessage(const wxString
& title
, const wxString
& text
);
1540 // ----------------------------------------------------------------------------
1541 // debug only logging functions: use them with API name and error code
1542 // ----------------------------------------------------------------------------
1545 // make life easier for people using VC++ IDE: clicking on the message
1546 // will take us immediately to the place of the failed API
1548 #define wxLogApiError(api, rc) \
1549 wxLogDebug(wxT("%s(%d): '%s' failed with error 0x%08lx (%s)."), \
1550 __FILE__, __LINE__, api, \
1551 (long)rc, wxSysErrorMsg(rc))
1553 #define wxLogApiError(api, rc) \
1554 wxLogDebug(wxT("In file %s at line %d: '%s' failed with ") \
1555 wxT("error 0x%08lx (%s)."), \
1556 __FILE__, __LINE__, api, \
1557 (long)rc, wxSysErrorMsg(rc))
1558 #endif // VC++/!VC++
1560 #define wxLogLastError(api) wxLogApiError(api, wxSysErrorCode())
1562 #else // !wxUSE_LOG_DEBUG
1563 #define wxLogApiError(api, err) wxLogNop()
1564 #define wxLogLastError(api) wxLogNop()
1565 #endif // wxUSE_LOG_DEBUG/!wxUSE_LOG_DEBUG
1567 // wxCocoa has additiional trace masks
1568 #if defined(__WXCOCOA__)
1569 #include "wx/cocoa/log.h"
1572 #ifdef WX_WATCOM_ONLY_CODE
1573 #undef WX_WATCOM_ONLY_CODE
1576 // macro which disables debug logging in release builds: this is done by
1577 // default by IMPLEMENT_APP() so usually it doesn't need to be used explicitly
1579 #define wxDISABLE_DEBUG_LOGGING_IN_RELEASE_BUILD() \
1580 wxLog::SetLogLevel(wxLOG_Info)
1582 #define wxDISABLE_DEBUG_LOGGING_IN_RELEASE_BUILD()
1583 #endif // NDEBUG/!NDEBUG
1585 #endif // _WX_LOG_H_