1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/xrc/xmlres.cpp
3 // Purpose: XRC resources
4 // Author: Vaclav Slavik
7 // Copyright: (c) 2000 Vaclav Slavik
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx.h".
12 #include "wx/wxprec.h"
20 #include "wx/xrc/xmlres.h"
27 #include "wx/dialog.h"
28 #include "wx/settings.h"
29 #include "wx/bitmap.h"
31 #include "wx/module.h"
32 #include "wx/wxcrtvararg.h"
39 #include "wx/vector.h"
40 #include "wx/wfstream.h"
41 #include "wx/filesys.h"
42 #include "wx/filename.h"
43 #include "wx/tokenzr.h"
44 #include "wx/fontenum.h"
45 #include "wx/fontmap.h"
46 #include "wx/artprov.h"
48 #include "wx/xml/xml.h"
51 class wxXmlResourceDataRecord
54 wxXmlResourceDataRecord() : Doc(NULL
) {
56 Time
= wxDateTime::Now();
59 ~wxXmlResourceDataRecord() {delete Doc
;}
68 class wxXmlResourceDataRecords
: public wxVector
<wxXmlResourceDataRecord
*>
70 // this is a class so that it can be forward-declared
74 wxXmlResource
*wxXmlResource::ms_instance
= NULL
;
76 /*static*/ wxXmlResource
*wxXmlResource::Get()
79 ms_instance
= new wxXmlResource
;
83 /*static*/ wxXmlResource
*wxXmlResource::Set(wxXmlResource
*res
)
85 wxXmlResource
*old
= ms_instance
;
90 wxXmlResource::wxXmlResource(int flags
, const wxString
& domain
)
94 m_data
= new wxXmlResourceDataRecords
;
98 wxXmlResource::wxXmlResource(const wxString
& filemask
, int flags
, const wxString
& domain
)
102 m_data
= new wxXmlResourceDataRecords
;
107 wxXmlResource::~wxXmlResource()
111 for ( wxXmlResourceDataRecords::iterator i
= m_data
->begin();
112 i
!= m_data
->end(); ++i
)
119 void wxXmlResource::SetDomain(const wxString
& domain
)
126 wxString
wxXmlResource::ConvertFileNameToURL(const wxString
& filename
)
128 wxString
fnd(filename
);
130 // NB: as Load() and Unload() accept both filenames and URLs (should
131 // probably be changed to filenames only, but embedded resources
132 // currently rely on its ability to handle URLs - FIXME) we need to
133 // determine whether found name is filename and not URL and this is the
134 // fastest/simplest way to do it
135 if (wxFileName::FileExists(fnd
))
137 // Make the name absolute filename, because the app may
138 // change working directory later:
143 fnd
= fn
.GetFullPath();
146 fnd
= wxFileSystem::FileNameToURL(fnd
);
156 bool wxXmlResource::IsArchive(const wxString
& filename
)
158 const wxString fnd
= filename
.Lower();
160 return fnd
.Matches(wxT("*.zip")) || fnd
.Matches(wxT("*.xrs"));
163 #endif // wxUSE_FILESYSTEM
165 bool wxXmlResource::Load(const wxString
& filemask
)
168 bool iswild
= wxIsWild(filemask
);
173 # define wxXmlFindFirst fsys.FindFirst(filemask, wxFILE)
174 # define wxXmlFindNext fsys.FindNext()
176 # define wxXmlFindFirst wxFindFirstFile(filemask, wxFILE)
177 # define wxXmlFindNext wxFindNextFile()
180 fnd
= wxXmlFindFirst
;
185 fnd
= ConvertFileNameToURL(fnd
);
188 if ( IsArchive(fnd
) )
190 rt
= rt
&& Load(fnd
+ wxT("#zip:*.xrc"));
192 else // a single resource URL
193 #endif // wxUSE_FILESYSTEM
195 wxXmlResourceDataRecord
*drec
= new wxXmlResourceDataRecord
;
197 Data().push_back(drec
);
205 # undef wxXmlFindFirst
206 # undef wxXmlFindNext
207 return rt
&& UpdateResources();
210 bool wxXmlResource::Unload(const wxString
& filename
)
212 wxASSERT_MSG( !wxIsWild(filename
),
213 _T("wildcards not supported by wxXmlResource::Unload()") );
215 wxString fnd
= ConvertFileNameToURL(filename
);
217 const bool isArchive
= IsArchive(fnd
);
220 #endif // wxUSE_FILESYSTEM
222 bool unloaded
= false;
223 for ( wxXmlResourceDataRecords::iterator i
= Data().begin();
224 i
!= Data().end(); ++i
)
229 if ( (*i
)->File
.StartsWith(fnd
) )
231 // don't break from the loop, we can have other matching files
233 else // a single resource URL
234 #endif // wxUSE_FILESYSTEM
236 if ( (*i
)->File
== fnd
)
242 // no sense in continuing, there is only one file with this URL
252 IMPLEMENT_ABSTRACT_CLASS(wxXmlResourceHandler
, wxObject
)
254 void wxXmlResource::AddHandler(wxXmlResourceHandler
*handler
)
256 m_handlers
.push_back(handler
);
257 handler
->SetParentResource(this);
260 void wxXmlResource::InsertHandler(wxXmlResourceHandler
*handler
)
262 m_handlers
.insert(m_handlers
.begin(), handler
);
263 handler
->SetParentResource(this);
268 void wxXmlResource::ClearHandlers()
270 for ( wxVector
<wxXmlResourceHandler
*>::iterator i
= m_handlers
.begin();
271 i
!= m_handlers
.end(); ++i
)
277 wxMenu
*wxXmlResource::LoadMenu(const wxString
& name
)
279 return (wxMenu
*)CreateResFromNode(FindResource(name
, wxT("wxMenu")), NULL
, NULL
);
284 wxMenuBar
*wxXmlResource::LoadMenuBar(wxWindow
*parent
, const wxString
& name
)
286 return (wxMenuBar
*)CreateResFromNode(FindResource(name
, wxT("wxMenuBar")), parent
, NULL
);
292 wxToolBar
*wxXmlResource::LoadToolBar(wxWindow
*parent
, const wxString
& name
)
294 return (wxToolBar
*)CreateResFromNode(FindResource(name
, wxT("wxToolBar")), parent
, NULL
);
299 wxDialog
*wxXmlResource::LoadDialog(wxWindow
*parent
, const wxString
& name
)
301 return (wxDialog
*)CreateResFromNode(FindResource(name
, wxT("wxDialog")), parent
, NULL
);
304 bool wxXmlResource::LoadDialog(wxDialog
*dlg
, wxWindow
*parent
, const wxString
& name
)
306 return CreateResFromNode(FindResource(name
, wxT("wxDialog")), parent
, dlg
) != NULL
;
311 wxPanel
*wxXmlResource::LoadPanel(wxWindow
*parent
, const wxString
& name
)
313 return (wxPanel
*)CreateResFromNode(FindResource(name
, wxT("wxPanel")), parent
, NULL
);
316 bool wxXmlResource::LoadPanel(wxPanel
*panel
, wxWindow
*parent
, const wxString
& name
)
318 return CreateResFromNode(FindResource(name
, wxT("wxPanel")), parent
, panel
) != NULL
;
321 wxFrame
*wxXmlResource::LoadFrame(wxWindow
* parent
, const wxString
& name
)
323 return (wxFrame
*)CreateResFromNode(FindResource(name
, wxT("wxFrame")), parent
, NULL
);
326 bool wxXmlResource::LoadFrame(wxFrame
* frame
, wxWindow
*parent
, const wxString
& name
)
328 return CreateResFromNode(FindResource(name
, wxT("wxFrame")), parent
, frame
) != NULL
;
331 wxBitmap
wxXmlResource::LoadBitmap(const wxString
& name
)
333 wxBitmap
*bmp
= (wxBitmap
*)CreateResFromNode(
334 FindResource(name
, wxT("wxBitmap")), NULL
, NULL
);
337 if (bmp
) { rt
= *bmp
; delete bmp
; }
341 wxIcon
wxXmlResource::LoadIcon(const wxString
& name
)
343 wxIcon
*icon
= (wxIcon
*)CreateResFromNode(
344 FindResource(name
, wxT("wxIcon")), NULL
, NULL
);
347 if (icon
) { rt
= *icon
; delete icon
; }
352 wxObject
*wxXmlResource::LoadObject(wxWindow
*parent
, const wxString
& name
, const wxString
& classname
)
354 return CreateResFromNode(FindResource(name
, classname
), parent
, NULL
);
357 bool wxXmlResource::LoadObject(wxObject
*instance
, wxWindow
*parent
, const wxString
& name
, const wxString
& classname
)
359 return CreateResFromNode(FindResource(name
, classname
), parent
, instance
) != NULL
;
363 bool wxXmlResource::AttachUnknownControl(const wxString
& name
,
364 wxWindow
*control
, wxWindow
*parent
)
367 parent
= control
->GetParent();
368 wxWindow
*container
= parent
->FindWindow(name
+ wxT("_container"));
371 wxLogError(_("Cannot find container for unknown control '%s'."), name
.c_str());
374 return control
->Reparent(container
);
378 static void ProcessPlatformProperty(wxXmlNode
*node
)
383 wxXmlNode
*c
= node
->GetChildren();
387 if (!c
->GetAttribute(wxT("platform"), &s
))
391 wxStringTokenizer
tkn(s
, wxT(" |"));
393 while (tkn
.HasMoreTokens())
395 s
= tkn
.GetNextToken();
397 if (s
== wxT("win")) isok
= true;
399 #if defined(__MAC__) || defined(__APPLE__)
400 if (s
== wxT("mac")) isok
= true;
401 #elif defined(__UNIX__)
402 if (s
== wxT("unix")) isok
= true;
405 if (s
== wxT("os2")) isok
= true;
415 ProcessPlatformProperty(c
);
420 wxXmlNode
*c2
= c
->GetNext();
421 node
->RemoveChild(c
);
430 bool wxXmlResource::UpdateResources()
434 # if wxUSE_FILESYSTEM
435 wxFSFile
*file
= NULL
;
440 wxString
encoding(wxT("UTF-8"));
441 #if !wxUSE_UNICODE && wxUSE_INTL
442 if ( (GetFlags() & wxXRC_USE_LOCALE
) == 0 )
444 // In case we are not using wxLocale to translate strings, convert the
445 // strings GUI's charset. This must not be done when wxXRC_USE_LOCALE
446 // is on, because it could break wxGetTranslation lookup.
447 encoding
= wxLocale::GetSystemEncodingName();
451 for ( wxXmlResourceDataRecords::iterator i
= Data().begin();
452 i
!= Data().end(); ++i
)
454 wxXmlResourceDataRecord
* const rec
= *i
;
456 modif
= (rec
->Doc
== NULL
);
458 if (!modif
&& !(m_flags
& wxXRC_NO_RELOADING
))
460 # if wxUSE_FILESYSTEM
461 file
= fsys
.OpenFile(rec
->File
);
463 modif
= file
&& file
->GetModificationTime() > rec
->Time
;
464 # else // wxUSE_DATETIME
466 # endif // wxUSE_DATETIME
469 wxLogError(_("Cannot open file '%s'."), rec
->File
);
474 # else // wxUSE_FILESYSTEM
476 modif
= wxDateTime(wxFileModificationTime(rec
->File
)) > rec
->Time
;
477 # else // wxUSE_DATETIME
479 # endif // wxUSE_DATETIME
480 # endif // wxUSE_FILESYSTEM
485 wxLogTrace(_T("xrc"), _T("opening file '%s'"), rec
->File
);
487 wxInputStream
*stream
= NULL
;
489 # if wxUSE_FILESYSTEM
490 file
= fsys
.OpenFile(rec
->File
);
492 stream
= file
->GetStream();
494 stream
= new wxFileInputStream(rec
->File
);
500 rec
->Doc
= new wxXmlDocument
;
502 if (!stream
|| !rec
->Doc
->Load(*stream
, encoding
))
504 wxLogError(_("Cannot load resources from file '%s'."),
509 else if (rec
->Doc
->GetRoot()->GetName() != wxT("resource"))
511 wxLogError(_("Invalid XRC resource '%s': doesn't have root node 'resource'."), rec
->File
);
519 wxString verstr
= rec
->Doc
->GetRoot()->GetAttribute(
520 wxT("version"), wxT("0.0.0.0"));
521 if (wxSscanf(verstr
.c_str(), wxT("%i.%i.%i.%i"),
522 &v1
, &v2
, &v3
, &v4
) == 4)
523 version
= v1
*256*256*256+v2
*256*256+v3
*256+v4
;
528 if (m_version
!= version
)
530 wxLogError(_("Resource files must have same version number!"));
534 ProcessPlatformProperty(rec
->Doc
->GetRoot());
537 rec
->Time
= file
->GetModificationTime();
538 #else // wxUSE_FILESYSTEM
539 rec
->Time
= wxDateTime(wxFileModificationTime(rec
->File
));
540 #endif // wxUSE_FILESYSTEM
541 #endif // wxUSE_DATETIME
544 # if wxUSE_FILESYSTEM
557 wxXmlNode
*wxXmlResource::DoFindResource(wxXmlNode
*parent
,
558 const wxString
& name
,
559 const wxString
& classname
,
565 // first search for match at the top-level nodes (as this is
566 // where the resource is most commonly looked for):
567 for (node
= parent
->GetChildren(); node
; node
= node
->GetNext())
569 if ( node
->GetType() == wxXML_ELEMENT_NODE
&&
570 (node
->GetName() == wxT("object") ||
571 node
->GetName() == wxT("object_ref")) &&
572 node
->GetAttribute(wxT("name"), &dummy
) && dummy
== name
)
574 wxString
cls(node
->GetAttribute(wxT("class"), wxEmptyString
));
575 if (!classname
|| cls
== classname
)
577 // object_ref may not have 'class' attribute:
578 if (cls
.empty() && node
->GetName() == wxT("object_ref"))
580 wxString refName
= node
->GetAttribute(wxT("ref"), wxEmptyString
);
583 wxXmlNode
* refNode
= FindResource(refName
, wxEmptyString
, true);
585 refNode
->GetAttribute(wxT("class"), wxEmptyString
) == classname
)
594 for (node
= parent
->GetChildren(); node
; node
= node
->GetNext())
596 if ( node
->GetType() == wxXML_ELEMENT_NODE
&&
597 (node
->GetName() == wxT("object") ||
598 node
->GetName() == wxT("object_ref")) )
600 wxXmlNode
* found
= DoFindResource(node
, name
, classname
, true);
609 wxXmlNode
*wxXmlResource::FindResource(const wxString
& name
,
610 const wxString
& classname
,
613 UpdateResources(); //ensure everything is up-to-date
616 for ( wxXmlResourceDataRecords::const_iterator f
= Data().begin();
617 f
!= Data().end(); ++f
)
619 wxXmlResourceDataRecord
* const rec
= *f
;
620 if ( rec
->Doc
== NULL
|| rec
->Doc
->GetRoot() == NULL
)
623 wxXmlNode
* found
= DoFindResource(rec
->Doc
->GetRoot(),
624 name
, classname
, recursive
);
628 m_curFileSystem
.ChangePathTo(rec
->File
);
634 wxLogError(_("XRC resource '%s' (class '%s') not found!"),
635 name
.c_str(), classname
.c_str());
639 static void MergeNodes(wxXmlNode
& dest
, wxXmlNode
& with
)
642 for ( wxXmlAttribute
*attr
= with
.GetAttributes();
643 attr
; attr
= attr
->GetNext() )
645 wxXmlAttribute
*dattr
;
646 for (dattr
= dest
.GetAttributes(); dattr
; dattr
= dattr
->GetNext())
649 if ( dattr
->GetName() == attr
->GetName() )
651 dattr
->SetValue(attr
->GetValue());
657 dest
.AddAttribute(attr
->GetName(), attr
->GetValue());
660 // Merge child nodes:
661 for (wxXmlNode
* node
= with
.GetChildren(); node
; node
= node
->GetNext())
663 wxString name
= node
->GetAttribute(wxT("name"), wxEmptyString
);
666 for (dnode
= dest
.GetChildren(); dnode
; dnode
= dnode
->GetNext() )
668 if ( dnode
->GetName() == node
->GetName() &&
669 dnode
->GetAttribute(wxT("name"), wxEmptyString
) == name
&&
670 dnode
->GetType() == node
->GetType() )
672 MergeNodes(*dnode
, *node
);
679 static const wxChar
*AT_END
= wxT("end");
680 wxString insert_pos
= node
->GetAttribute(wxT("insert_at"), AT_END
);
681 if ( insert_pos
== AT_END
)
683 dest
.AddChild(new wxXmlNode(*node
));
685 else if ( insert_pos
== wxT("begin") )
687 dest
.InsertChild(new wxXmlNode(*node
), dest
.GetChildren());
692 if ( dest
.GetType() == wxXML_TEXT_NODE
&& with
.GetContent().length() )
693 dest
.SetContent(with
.GetContent());
696 wxObject
*wxXmlResource::CreateResFromNode(wxXmlNode
*node
, wxObject
*parent
,
698 wxXmlResourceHandler
*handlerToUse
)
700 if (node
== NULL
) return NULL
;
702 // handling of referenced resource
703 if ( node
->GetName() == wxT("object_ref") )
705 wxString refName
= node
->GetAttribute(wxT("ref"), wxEmptyString
);
706 wxXmlNode
* refNode
= FindResource(refName
, wxEmptyString
, true);
710 wxLogError(_("Referenced object node with ref=\"%s\" not found!"),
715 wxXmlNode
copy(*refNode
);
716 MergeNodes(copy
, *node
);
718 return CreateResFromNode(©
, parent
, instance
);
723 if (handlerToUse
->CanHandle(node
))
725 return handlerToUse
->CreateResource(node
, parent
, instance
);
728 else if (node
->GetName() == wxT("object"))
730 for ( wxVector
<wxXmlResourceHandler
*>::iterator h
= m_handlers
.begin();
731 h
!= m_handlers
.end(); ++h
)
733 wxXmlResourceHandler
*handler
= *h
;
734 if (handler
->CanHandle(node
))
735 return handler
->CreateResource(node
, parent
, instance
);
739 wxLogError(_("No handler found for XML node '%s', class '%s'!"),
740 node
->GetName().c_str(),
741 node
->GetAttribute(wxT("class"), wxEmptyString
).c_str());
746 class wxXmlSubclassFactories
: public wxVector
<wxXmlSubclassFactory
*>
748 // this is a class so that it can be forward-declared
751 wxXmlSubclassFactories
*wxXmlResource::ms_subclassFactories
= NULL
;
753 /*static*/ void wxXmlResource::AddSubclassFactory(wxXmlSubclassFactory
*factory
)
755 if (!ms_subclassFactories
)
757 ms_subclassFactories
= new wxXmlSubclassFactories
;
759 ms_subclassFactories
->push_back(factory
);
762 class wxXmlSubclassFactoryCXX
: public wxXmlSubclassFactory
765 ~wxXmlSubclassFactoryCXX() {}
767 wxObject
*Create(const wxString
& className
)
769 wxClassInfo
* classInfo
= wxClassInfo::FindClass(className
);
772 return classInfo
->CreateObject();
781 wxXmlResourceHandler::wxXmlResourceHandler()
782 : m_node(NULL
), m_parent(NULL
), m_instance(NULL
),
783 m_parentAsWindow(NULL
)
788 wxObject
*wxXmlResourceHandler::CreateResource(wxXmlNode
*node
, wxObject
*parent
, wxObject
*instance
)
790 wxXmlNode
*myNode
= m_node
;
791 wxString myClass
= m_class
;
792 wxObject
*myParent
= m_parent
, *myInstance
= m_instance
;
793 wxWindow
*myParentAW
= m_parentAsWindow
;
795 m_instance
= instance
;
796 if (!m_instance
&& node
->HasAttribute(wxT("subclass")) &&
797 !(m_resource
->GetFlags() & wxXRC_NO_SUBCLASSING
))
799 wxString subclass
= node
->GetAttribute(wxT("subclass"), wxEmptyString
);
800 if (!subclass
.empty())
802 for (wxXmlSubclassFactories::iterator i
= wxXmlResource::ms_subclassFactories
->begin();
803 i
!= wxXmlResource::ms_subclassFactories
->end(); ++i
)
805 m_instance
= (*i
)->Create(subclass
);
812 wxString name
= node
->GetAttribute(wxT("name"), wxEmptyString
);
813 wxLogError(_("Subclass '%s' not found for resource '%s', not subclassing!"),
814 subclass
.c_str(), name
.c_str());
820 m_class
= node
->GetAttribute(wxT("class"), wxEmptyString
);
822 m_parentAsWindow
= wxDynamicCast(m_parent
, wxWindow
);
824 wxObject
*returned
= DoCreateResource();
828 m_parent
= myParent
; m_parentAsWindow
= myParentAW
;
829 m_instance
= myInstance
;
835 void wxXmlResourceHandler::AddStyle(const wxString
& name
, int value
)
837 m_styleNames
.Add(name
);
838 m_styleValues
.Add(value
);
843 void wxXmlResourceHandler::AddWindowStyles()
845 XRC_ADD_STYLE(wxCLIP_CHILDREN
);
847 // the border styles all have the old and new names, recognize both for now
848 XRC_ADD_STYLE(wxSIMPLE_BORDER
); XRC_ADD_STYLE(wxBORDER_SIMPLE
);
849 XRC_ADD_STYLE(wxSUNKEN_BORDER
); XRC_ADD_STYLE(wxBORDER_SUNKEN
);
850 XRC_ADD_STYLE(wxDOUBLE_BORDER
); XRC_ADD_STYLE(wxBORDER_DOUBLE
);
851 XRC_ADD_STYLE(wxRAISED_BORDER
); XRC_ADD_STYLE(wxBORDER_RAISED
);
852 XRC_ADD_STYLE(wxSTATIC_BORDER
); XRC_ADD_STYLE(wxBORDER_STATIC
);
853 XRC_ADD_STYLE(wxNO_BORDER
); XRC_ADD_STYLE(wxBORDER_NONE
);
855 XRC_ADD_STYLE(wxTRANSPARENT_WINDOW
);
856 XRC_ADD_STYLE(wxWANTS_CHARS
);
857 XRC_ADD_STYLE(wxTAB_TRAVERSAL
);
858 XRC_ADD_STYLE(wxNO_FULL_REPAINT_ON_RESIZE
);
859 XRC_ADD_STYLE(wxFULL_REPAINT_ON_RESIZE
);
860 XRC_ADD_STYLE(wxALWAYS_SHOW_SB
);
861 XRC_ADD_STYLE(wxWS_EX_BLOCK_EVENTS
);
862 XRC_ADD_STYLE(wxWS_EX_VALIDATE_RECURSIVELY
);
867 bool wxXmlResourceHandler::HasParam(const wxString
& param
)
869 return (GetParamNode(param
) != NULL
);
873 int wxXmlResourceHandler::GetStyle(const wxString
& param
, int defaults
)
875 wxString s
= GetParamValue(param
);
877 if (!s
) return defaults
;
879 wxStringTokenizer
tkn(s
, wxT("| \t\n"), wxTOKEN_STRTOK
);
883 while (tkn
.HasMoreTokens())
885 fl
= tkn
.GetNextToken();
886 index
= m_styleNames
.Index(fl
);
887 if (index
!= wxNOT_FOUND
)
888 style
|= m_styleValues
[index
];
890 wxLogError(_("Unknown style flag ") + fl
);
897 wxString
wxXmlResourceHandler::GetText(const wxString
& param
, bool translate
)
899 wxXmlNode
*parNode
= GetParamNode(param
);
900 wxString
str1(GetNodeContent(parNode
));
903 // "\\" wasn't translated to "\" prior to 2.5.3.0:
904 const bool escapeBackslash
= (m_resource
->CompareVersion(2,5,3,0) >= 0);
906 // VS: First version of XRC resources used $ instead of & (which is
907 // illegal in XML), but later I realized that '_' fits this purpose
908 // much better (because &File means "File with F underlined").
909 const wxChar amp_char
= (m_resource
->CompareVersion(2,3,0,1) < 0)
912 for ( wxString::const_iterator dt
= str1
.begin(); dt
!= str1
.end(); ++dt
)
914 // Remap amp_char to &, map double amp_char to amp_char (for things
915 // like "&File..." -- this is illegal in XML, so we use "_File..."):
916 if ( *dt
== amp_char
)
918 if ( *(++dt
) == amp_char
)
921 str2
<< wxT('&') << *dt
;
923 // Remap \n to CR, \r to LF, \t to TAB, \\ to \:
924 else if ( *dt
== wxT('\\') )
926 switch ( (*(++dt
)).GetValue() )
941 // "\\" wasn't translated to "\" prior to 2.5.3.0:
942 if ( escapeBackslash
)
947 // else fall-through to default: branch below
950 str2
<< wxT('\\') << *dt
;
960 if (m_resource
->GetFlags() & wxXRC_USE_LOCALE
)
962 if (translate
&& parNode
&&
963 parNode
->GetAttribute(wxT("translate"), wxEmptyString
) != wxT("0"))
965 return wxGetTranslation(str2
, m_resource
->GetDomain());
972 // The string is internally stored as UTF-8, we have to convert
973 // it into system's default encoding so that it can be displayed:
974 return wxString(str2
.wc_str(wxConvUTF8
), wxConvLocal
);
979 // If wxXRC_USE_LOCALE is not set, then the string is already in
980 // system's default encoding in ANSI build, so we don't have to
981 // do anything special here.
987 long wxXmlResourceHandler::GetLong(const wxString
& param
, long defaultv
)
990 wxString str1
= GetParamValue(param
);
992 if (!str1
.ToLong(&value
))
998 float wxXmlResourceHandler::GetFloat(const wxString
& param
, float defaultv
)
1000 wxString str
= GetParamValue(param
);
1003 // strings in XRC always use C locale but wxString::ToDouble() uses the
1004 // current one, so transform the string to it supposing that the only
1005 // difference between them is the decimal separator
1007 // TODO: use wxString::ToCDouble() when we have it
1008 str
.Replace(wxT("."), wxLocale::GetInfo(wxLOCALE_DECIMAL_POINT
,
1009 wxLOCALE_CAT_NUMBER
));
1010 #endif // wxUSE_INTL
1013 if (!str
.ToDouble(&value
))
1016 return wx_truncate_cast(float, value
);
1020 int wxXmlResourceHandler::GetID()
1022 return wxXmlResource::GetXRCID(GetName());
1027 wxString
wxXmlResourceHandler::GetName()
1029 return m_node
->GetAttribute(wxT("name"), wxT("-1"));
1034 bool wxXmlResourceHandler::GetBool(const wxString
& param
, bool defaultv
)
1036 wxString v
= GetParamValue(param
);
1038 if (!v
) return defaultv
;
1040 return (v
== wxT("1"));
1044 static wxColour
GetSystemColour(const wxString
& name
)
1048 #define SYSCLR(clr) \
1049 if (name == _T(#clr)) return wxSystemSettings::GetColour(clr);
1050 SYSCLR(wxSYS_COLOUR_SCROLLBAR
)
1051 SYSCLR(wxSYS_COLOUR_BACKGROUND
)
1052 SYSCLR(wxSYS_COLOUR_DESKTOP
)
1053 SYSCLR(wxSYS_COLOUR_ACTIVECAPTION
)
1054 SYSCLR(wxSYS_COLOUR_INACTIVECAPTION
)
1055 SYSCLR(wxSYS_COLOUR_MENU
)
1056 SYSCLR(wxSYS_COLOUR_WINDOW
)
1057 SYSCLR(wxSYS_COLOUR_WINDOWFRAME
)
1058 SYSCLR(wxSYS_COLOUR_MENUTEXT
)
1059 SYSCLR(wxSYS_COLOUR_WINDOWTEXT
)
1060 SYSCLR(wxSYS_COLOUR_CAPTIONTEXT
)
1061 SYSCLR(wxSYS_COLOUR_ACTIVEBORDER
)
1062 SYSCLR(wxSYS_COLOUR_INACTIVEBORDER
)
1063 SYSCLR(wxSYS_COLOUR_APPWORKSPACE
)
1064 SYSCLR(wxSYS_COLOUR_HIGHLIGHT
)
1065 SYSCLR(wxSYS_COLOUR_HIGHLIGHTTEXT
)
1066 SYSCLR(wxSYS_COLOUR_BTNFACE
)
1067 SYSCLR(wxSYS_COLOUR_3DFACE
)
1068 SYSCLR(wxSYS_COLOUR_BTNSHADOW
)
1069 SYSCLR(wxSYS_COLOUR_3DSHADOW
)
1070 SYSCLR(wxSYS_COLOUR_GRAYTEXT
)
1071 SYSCLR(wxSYS_COLOUR_BTNTEXT
)
1072 SYSCLR(wxSYS_COLOUR_INACTIVECAPTIONTEXT
)
1073 SYSCLR(wxSYS_COLOUR_BTNHIGHLIGHT
)
1074 SYSCLR(wxSYS_COLOUR_BTNHILIGHT
)
1075 SYSCLR(wxSYS_COLOUR_3DHIGHLIGHT
)
1076 SYSCLR(wxSYS_COLOUR_3DHILIGHT
)
1077 SYSCLR(wxSYS_COLOUR_3DDKSHADOW
)
1078 SYSCLR(wxSYS_COLOUR_3DLIGHT
)
1079 SYSCLR(wxSYS_COLOUR_INFOTEXT
)
1080 SYSCLR(wxSYS_COLOUR_INFOBK
)
1081 SYSCLR(wxSYS_COLOUR_LISTBOX
)
1082 SYSCLR(wxSYS_COLOUR_HOTLIGHT
)
1083 SYSCLR(wxSYS_COLOUR_GRADIENTACTIVECAPTION
)
1084 SYSCLR(wxSYS_COLOUR_GRADIENTINACTIVECAPTION
)
1085 SYSCLR(wxSYS_COLOUR_MENUHILIGHT
)
1086 SYSCLR(wxSYS_COLOUR_MENUBAR
)
1090 return wxNullColour
;
1093 wxColour
wxXmlResourceHandler::GetColour(const wxString
& param
, const wxColour
& defaultv
)
1095 wxString v
= GetParamValue(param
);
1102 // wxString -> wxColour conversion
1105 // the colour doesn't use #RRGGBB format, check if it is symbolic
1107 clr
= GetSystemColour(v
);
1111 wxLogError(_("XRC resource: Incorrect colour specification '%s' for attribute '%s'."),
1112 v
.c_str(), param
.c_str());
1113 return wxNullColour
;
1121 wxBitmap
wxXmlResourceHandler::GetBitmap(const wxString
& param
,
1122 const wxArtClient
& defaultArtClient
,
1125 /* If the bitmap is specified as stock item, query wxArtProvider for it: */
1126 wxXmlNode
*bmpNode
= GetParamNode(param
);
1129 wxString sid
= bmpNode
->GetAttribute(wxT("stock_id"), wxEmptyString
);
1132 wxString scl
= bmpNode
->GetAttribute(wxT("stock_client"), wxEmptyString
);
1134 scl
= defaultArtClient
;
1136 scl
= wxART_MAKE_CLIENT_ID_FROM_STR(scl
);
1139 wxArtProvider::GetBitmap(wxART_MAKE_ART_ID_FROM_STR(sid
),
1141 if ( stockArt
.Ok() )
1146 /* ...or load the bitmap from file: */
1147 wxString name
= GetParamValue(param
);
1148 if (name
.empty()) return wxNullBitmap
;
1149 #if wxUSE_FILESYSTEM
1150 wxFSFile
*fsfile
= GetCurFileSystem().OpenFile(name
, wxFS_READ
| wxFS_SEEKABLE
);
1153 wxLogError(_("XRC resource: Cannot create bitmap from '%s'."),
1155 return wxNullBitmap
;
1157 wxImage
img(*(fsfile
->GetStream()));
1165 wxLogError(_("XRC resource: Cannot create bitmap from '%s'."),
1167 return wxNullBitmap
;
1169 if (!(size
== wxDefaultSize
)) img
.Rescale(size
.x
, size
.y
);
1170 return wxBitmap(img
);
1173 #if wxUSE_ANIMATIONCTRL
1174 wxAnimation
wxXmlResourceHandler::GetAnimation(const wxString
& param
)
1178 /* load the animation from file: */
1179 wxString name
= GetParamValue(param
);
1180 if (name
.empty()) return wxNullAnimation
;
1181 #if wxUSE_FILESYSTEM
1182 wxFSFile
*fsfile
= GetCurFileSystem().OpenFile(name
, wxFS_READ
| wxFS_SEEKABLE
);
1185 wxLogError(_("XRC resource: Cannot create animation from '%s'."),
1187 return wxNullAnimation
;
1189 ani
.Load(*(fsfile
->GetStream()));
1197 wxLogError(_("XRC resource: Cannot create animation from '%s'."),
1199 return wxNullAnimation
;
1204 #endif // wxUSE_ANIMATIONCTRL
1208 wxIcon
wxXmlResourceHandler::GetIcon(const wxString
& param
,
1209 const wxArtClient
& defaultArtClient
,
1213 icon
.CopyFromBitmap(GetBitmap(param
, defaultArtClient
, size
));
1219 wxXmlNode
*wxXmlResourceHandler::GetParamNode(const wxString
& param
)
1221 wxCHECK_MSG(m_node
, NULL
, wxT("You can't access handler data before it was initialized!"));
1223 wxXmlNode
*n
= m_node
->GetChildren();
1227 if (n
->GetType() == wxXML_ELEMENT_NODE
&& n
->GetName() == param
)
1236 bool wxXmlResourceHandler::IsOfClass(wxXmlNode
*node
, const wxString
& classname
)
1238 return node
->GetAttribute(wxT("class"), wxEmptyString
) == classname
;
1243 wxString
wxXmlResourceHandler::GetNodeContent(wxXmlNode
*node
)
1245 wxXmlNode
*n
= node
;
1246 if (n
== NULL
) return wxEmptyString
;
1247 n
= n
->GetChildren();
1251 if (n
->GetType() == wxXML_TEXT_NODE
||
1252 n
->GetType() == wxXML_CDATA_SECTION_NODE
)
1253 return n
->GetContent();
1256 return wxEmptyString
;
1261 wxString
wxXmlResourceHandler::GetParamValue(const wxString
& param
)
1264 return GetNodeContent(m_node
);
1266 return GetNodeContent(GetParamNode(param
));
1271 wxSize
wxXmlResourceHandler::GetSize(const wxString
& param
,
1272 wxWindow
*windowToUse
)
1274 wxString s
= GetParamValue(param
);
1275 if (s
.empty()) s
= wxT("-1,-1");
1279 is_dlg
= s
[s
.length()-1] == wxT('d');
1280 if (is_dlg
) s
.RemoveLast();
1282 if (!s
.BeforeFirst(wxT(',')).ToLong(&sx
) ||
1283 !s
.AfterLast(wxT(',')).ToLong(&sy
))
1285 wxLogError(_("Cannot parse coordinates from '%s'."), s
.c_str());
1286 return wxDefaultSize
;
1293 return wxDLG_UNIT(windowToUse
, wxSize(sx
, sy
));
1295 else if (m_parentAsWindow
)
1297 return wxDLG_UNIT(m_parentAsWindow
, wxSize(sx
, sy
));
1301 wxLogError(_("Cannot convert dialog units: dialog unknown."));
1302 return wxDefaultSize
;
1306 return wxSize(sx
, sy
);
1311 wxPoint
wxXmlResourceHandler::GetPosition(const wxString
& param
)
1313 wxSize sz
= GetSize(param
);
1314 return wxPoint(sz
.x
, sz
.y
);
1319 wxCoord
wxXmlResourceHandler::GetDimension(const wxString
& param
,
1321 wxWindow
*windowToUse
)
1323 wxString s
= GetParamValue(param
);
1324 if (s
.empty()) return defaultv
;
1328 is_dlg
= s
[s
.length()-1] == wxT('d');
1329 if (is_dlg
) s
.RemoveLast();
1333 wxLogError(_("Cannot parse dimension from '%s'."), s
.c_str());
1341 return wxDLG_UNIT(windowToUse
, wxSize(sx
, 0)).x
;
1343 else if (m_parentAsWindow
)
1345 return wxDLG_UNIT(m_parentAsWindow
, wxSize(sx
, 0)).x
;
1349 wxLogError(_("Cannot convert dialog units: dialog unknown."));
1358 // Get system font index using indexname
1359 static wxFont
GetSystemFont(const wxString
& name
)
1363 #define SYSFNT(fnt) \
1364 if (name == _T(#fnt)) return wxSystemSettings::GetFont(fnt);
1365 SYSFNT(wxSYS_OEM_FIXED_FONT
)
1366 SYSFNT(wxSYS_ANSI_FIXED_FONT
)
1367 SYSFNT(wxSYS_ANSI_VAR_FONT
)
1368 SYSFNT(wxSYS_SYSTEM_FONT
)
1369 SYSFNT(wxSYS_DEVICE_DEFAULT_FONT
)
1370 SYSFNT(wxSYS_DEFAULT_PALETTE
)
1371 SYSFNT(wxSYS_SYSTEM_FIXED_FONT
)
1372 SYSFNT(wxSYS_DEFAULT_GUI_FONT
)
1379 wxFont
wxXmlResourceHandler::GetFont(const wxString
& param
)
1381 wxXmlNode
*font_node
= GetParamNode(param
);
1382 if (font_node
== NULL
)
1384 wxLogError(_("Cannot find font node '%s'."), param
.c_str());
1388 wxXmlNode
*oldnode
= m_node
;
1395 bool hasSize
= HasParam(wxT("size"));
1397 isize
= GetLong(wxT("size"), -1);
1400 int istyle
= wxNORMAL
;
1401 bool hasStyle
= HasParam(wxT("style"));
1404 wxString style
= GetParamValue(wxT("style"));
1405 if (style
== wxT("italic"))
1407 else if (style
== wxT("slant"))
1412 int iweight
= wxNORMAL
;
1413 bool hasWeight
= HasParam(wxT("weight"));
1416 wxString weight
= GetParamValue(wxT("weight"));
1417 if (weight
== wxT("bold"))
1419 else if (weight
== wxT("light"))
1424 bool hasUnderlined
= HasParam(wxT("underlined"));
1425 bool underlined
= hasUnderlined
? GetBool(wxT("underlined"), false) : false;
1427 // family and facename
1428 int ifamily
= wxDEFAULT
;
1429 bool hasFamily
= HasParam(wxT("family"));
1432 wxString family
= GetParamValue(wxT("family"));
1433 if (family
== wxT("decorative")) ifamily
= wxDECORATIVE
;
1434 else if (family
== wxT("roman")) ifamily
= wxROMAN
;
1435 else if (family
== wxT("script")) ifamily
= wxSCRIPT
;
1436 else if (family
== wxT("swiss")) ifamily
= wxSWISS
;
1437 else if (family
== wxT("modern")) ifamily
= wxMODERN
;
1438 else if (family
== wxT("teletype")) ifamily
= wxTELETYPE
;
1443 bool hasFacename
= HasParam(wxT("face"));
1446 wxString faces
= GetParamValue(wxT("face"));
1447 wxStringTokenizer
tk(faces
, wxT(","));
1449 wxArrayString
facenames(wxFontEnumerator::GetFacenames());
1450 while (tk
.HasMoreTokens())
1452 int index
= facenames
.Index(tk
.GetNextToken(), false);
1453 if (index
!= wxNOT_FOUND
)
1455 facename
= facenames
[index
];
1459 #else // !wxUSE_FONTENUM
1460 // just use the first face name if we can't check its availability:
1461 if (tk
.HasMoreTokens())
1462 facename
= tk
.GetNextToken();
1463 #endif // wxUSE_FONTENUM/!wxUSE_FONTENUM
1467 wxFontEncoding enc
= wxFONTENCODING_DEFAULT
;
1468 bool hasEncoding
= HasParam(wxT("encoding"));
1471 wxString encoding
= GetParamValue(wxT("encoding"));
1472 wxFontMapper mapper
;
1473 if (!encoding
.empty())
1474 enc
= mapper
.CharsetToEncoding(encoding
);
1475 if (enc
== wxFONTENCODING_SYSTEM
)
1476 enc
= wxFONTENCODING_DEFAULT
;
1479 // is this font based on a system font?
1480 wxFont font
= GetSystemFont(GetParamValue(wxT("sysfont")));
1484 if (hasSize
&& isize
!= -1)
1485 font
.SetPointSize(isize
);
1486 else if (HasParam(wxT("relativesize")))
1487 font
.SetPointSize(int(font
.GetPointSize() *
1488 GetFloat(wxT("relativesize"))));
1491 font
.SetStyle(istyle
);
1493 font
.SetWeight(iweight
);
1495 font
.SetUnderlined(underlined
);
1497 font
.SetFamily(ifamily
);
1499 font
.SetFaceName(facename
);
1501 font
.SetDefaultEncoding(enc
);
1503 else // not based on system font
1505 font
= wxFont(isize
== -1 ? wxNORMAL_FONT
->GetPointSize() : isize
,
1506 ifamily
, istyle
, iweight
,
1507 underlined
, facename
, enc
);
1515 void wxXmlResourceHandler::SetupWindow(wxWindow
*wnd
)
1517 //FIXME : add cursor
1519 if (HasParam(wxT("exstyle")))
1520 // Have to OR it with existing style, since
1521 // some implementations (e.g. wxGTK) use the extra style
1523 wnd
->SetExtraStyle(wnd
->GetExtraStyle() | GetStyle(wxT("exstyle")));
1524 if (HasParam(wxT("bg")))
1525 wnd
->SetBackgroundColour(GetColour(wxT("bg")));
1526 if (HasParam(wxT("fg")))
1527 wnd
->SetForegroundColour(GetColour(wxT("fg")));
1528 if (GetBool(wxT("enabled"), 1) == 0)
1530 if (GetBool(wxT("focused"), 0) == 1)
1532 if (GetBool(wxT("hidden"), 0) == 1)
1535 if (HasParam(wxT("tooltip")))
1536 wnd
->SetToolTip(GetText(wxT("tooltip")));
1538 if (HasParam(wxT("font")))
1539 wnd
->SetFont(GetFont());
1540 if (HasParam(wxT("help")))
1541 wnd
->SetHelpText(GetText(wxT("help")));
1545 void wxXmlResourceHandler::CreateChildren(wxObject
*parent
, bool this_hnd_only
)
1547 wxXmlNode
*n
= m_node
->GetChildren();
1551 if (n
->GetType() == wxXML_ELEMENT_NODE
&&
1552 (n
->GetName() == wxT("object") || n
->GetName() == wxT("object_ref")))
1554 m_resource
->CreateResFromNode(n
, parent
, NULL
,
1555 this_hnd_only
? this : NULL
);
1562 void wxXmlResourceHandler::CreateChildrenPrivately(wxObject
*parent
, wxXmlNode
*rootnode
)
1565 if (rootnode
== NULL
) root
= m_node
; else root
= rootnode
;
1566 wxXmlNode
*n
= root
->GetChildren();
1570 if (n
->GetType() == wxXML_ELEMENT_NODE
&& CanHandle(n
))
1572 CreateResource(n
, parent
, NULL
);
1584 // --------------- XRCID implementation -----------------------------
1586 #define XRCID_TABLE_SIZE 1024
1591 /* Hold the id so that once an id is allocated for a name, it
1592 does not get created again by NewControlId at least
1593 until we are done with it */
1599 static XRCID_record
*XRCID_Records
[XRCID_TABLE_SIZE
] = {NULL
};
1601 static int XRCID_Lookup(const char *str_id
, int value_if_not_found
= wxID_NONE
)
1605 for (const char *c
= str_id
; *c
!= '\0'; c
++) index
+= (int)*c
;
1606 index
%= XRCID_TABLE_SIZE
;
1608 XRCID_record
*oldrec
= NULL
;
1609 for (XRCID_record
*rec
= XRCID_Records
[index
]; rec
; rec
= rec
->next
)
1611 if (wxStrcmp(rec
->key
, str_id
) == 0)
1618 XRCID_record
**rec_var
= (oldrec
== NULL
) ?
1619 &XRCID_Records
[index
] : &oldrec
->next
;
1620 *rec_var
= new XRCID_record
;
1621 (*rec_var
)->key
= wxStrdup(str_id
);
1622 (*rec_var
)->next
= NULL
;
1625 if (value_if_not_found
!= wxID_NONE
)
1626 (*rec_var
)->id
= value_if_not_found
;
1629 int asint
= wxStrtol(str_id
, &end
, 10);
1630 if (*str_id
&& *end
== 0)
1632 // if str_id was integer, keep it verbosely:
1633 (*rec_var
)->id
= asint
;
1637 (*rec_var
)->id
= wxWindowBase::NewControlId();
1641 return (*rec_var
)->id
;
1644 static void AddStdXRCID_Records();
1647 int wxXmlResource::DoGetXRCID(const char *str_id
, int value_if_not_found
)
1649 static bool s_stdIDsAdded
= false;
1651 if ( !s_stdIDsAdded
)
1653 s_stdIDsAdded
= true;
1654 AddStdXRCID_Records();
1657 return XRCID_Lookup(str_id
, value_if_not_found
);
1661 static void CleanXRCID_Record(XRCID_record
*rec
)
1665 CleanXRCID_Record(rec
->next
);
1672 static void CleanXRCID_Records()
1674 for (int i
= 0; i
< XRCID_TABLE_SIZE
; i
++)
1676 CleanXRCID_Record(XRCID_Records
[i
]);
1677 XRCID_Records
[i
] = NULL
;
1681 static void AddStdXRCID_Records()
1683 #define stdID(id) XRCID_Lookup(#id, id)
1687 stdID(wxID_SEPARATOR
);
1700 stdID(wxID_PRINT_SETUP
);
1701 stdID(wxID_PAGE_SETUP
);
1702 stdID(wxID_PREVIEW
);
1704 stdID(wxID_HELP_CONTENTS
);
1705 stdID(wxID_HELP_COMMANDS
);
1706 stdID(wxID_HELP_PROCEDURES
);
1707 stdID(wxID_HELP_CONTEXT
);
1708 stdID(wxID_CLOSE_ALL
);
1709 stdID(wxID_PREFERENCES
);
1716 stdID(wxID_DUPLICATE
);
1717 stdID(wxID_SELECTALL
);
1719 stdID(wxID_REPLACE
);
1720 stdID(wxID_REPLACE_ALL
);
1721 stdID(wxID_PROPERTIES
);
1722 stdID(wxID_VIEW_DETAILS
);
1723 stdID(wxID_VIEW_LARGEICONS
);
1724 stdID(wxID_VIEW_SMALLICONS
);
1725 stdID(wxID_VIEW_LIST
);
1726 stdID(wxID_VIEW_SORTDATE
);
1727 stdID(wxID_VIEW_SORTNAME
);
1728 stdID(wxID_VIEW_SORTSIZE
);
1729 stdID(wxID_VIEW_SORTTYPE
);
1745 stdID(wxID_FORWARD
);
1746 stdID(wxID_BACKWARD
);
1747 stdID(wxID_DEFAULT
);
1751 stdID(wxID_CONTEXT_HELP
);
1752 stdID(wxID_YESTOALL
);
1753 stdID(wxID_NOTOALL
);
1762 stdID(wxID_REFRESH
);
1767 stdID(wxID_JUSTIFY_CENTER
);
1768 stdID(wxID_JUSTIFY_FILL
);
1769 stdID(wxID_JUSTIFY_RIGHT
);
1770 stdID(wxID_JUSTIFY_LEFT
);
1771 stdID(wxID_UNDERLINE
);
1773 stdID(wxID_UNINDENT
);
1774 stdID(wxID_ZOOM_100
);
1775 stdID(wxID_ZOOM_FIT
);
1776 stdID(wxID_ZOOM_IN
);
1777 stdID(wxID_ZOOM_OUT
);
1778 stdID(wxID_UNDELETE
);
1779 stdID(wxID_REVERT_TO_SAVED
);
1780 stdID(wxID_SYSTEM_MENU
);
1781 stdID(wxID_CLOSE_FRAME
);
1782 stdID(wxID_MOVE_FRAME
);
1783 stdID(wxID_RESIZE_FRAME
);
1784 stdID(wxID_MAXIMIZE_FRAME
);
1785 stdID(wxID_ICONIZE_FRAME
);
1786 stdID(wxID_RESTORE_FRAME
);
1795 // --------------- module and globals -----------------------------
1797 // normally we would do the cleanup from wxXmlResourceModule::OnExit() but it
1798 // can happen that some XRC records have been created because of the use of
1799 // XRCID() in event tables, which happens during static objects initialization,
1800 // but then the application initialization failed and so the wx modules were
1801 // neither initialized nor cleaned up -- this static object does the cleanup in
1803 static struct wxXRCStaticCleanup
1805 ~wxXRCStaticCleanup() { CleanXRCID_Records(); }
1808 class wxXmlResourceModule
: public wxModule
1810 DECLARE_DYNAMIC_CLASS(wxXmlResourceModule
)
1812 wxXmlResourceModule() {}
1815 wxXmlResource::AddSubclassFactory(new wxXmlSubclassFactoryCXX
);
1820 delete wxXmlResource::Set(NULL
);
1821 if(wxXmlResource::ms_subclassFactories
)
1823 for ( wxXmlSubclassFactories::iterator i
= wxXmlResource::ms_subclassFactories
->begin();
1824 i
!= wxXmlResource::ms_subclassFactories
->end(); ++i
)
1828 wxDELETE(wxXmlResource::ms_subclassFactories
);
1830 CleanXRCID_Records();
1834 IMPLEMENT_DYNAMIC_CLASS(wxXmlResourceModule
, wxModule
)
1837 // When wxXml is loaded dynamically after the application is already running
1838 // then the built-in module system won't pick this one up. Add it manually.
1839 void wxXmlInitResourceModule()
1841 wxModule
* module = new wxXmlResourceModule
;
1843 wxModule::RegisterModule(module);