1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: XRC resources
4 // Author: Vaclav Slavik
7 // Copyright: (c) 2000 Vaclav Slavik
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
12 #pragma implementation "xmlres.h"
15 // For compilers that support precompilation, includes "wx.h".
16 #include "wx/wxprec.h"
22 #include "wx/dialog.h"
25 #include "wx/wfstream.h"
26 #include "wx/filesys.h"
27 #include "wx/filename.h"
30 #include "wx/tokenzr.h"
31 #include "wx/fontenum.h"
32 #include "wx/module.h"
33 #include "wx/bitmap.h"
35 #include "wx/fontmap.h"
36 #include "wx/artprov.h"
38 #include "wx/xml/xml.h"
39 #include "wx/xrc/xmlres.h"
41 #include "wx/arrimpl.cpp"
42 WX_DEFINE_OBJARRAY(wxXmlResourceDataRecords
);
45 wxXmlResource
*wxXmlResource::ms_instance
= NULL
;
47 /*static*/ wxXmlResource
*wxXmlResource::Get()
50 ms_instance
= new wxXmlResource
;
54 /*static*/ wxXmlResource
*wxXmlResource::Set(wxXmlResource
*res
)
56 wxXmlResource
*old
= ms_instance
;
61 wxXmlResource::wxXmlResource(int flags
)
67 wxXmlResource::wxXmlResource(const wxString
& filemask
, int flags
)
74 wxXmlResource::~wxXmlResource()
80 bool wxXmlResource::Load(const wxString
& filemask
)
83 wxXmlResourceDataRecord
*drec
;
84 bool iswild
= wxIsWild(filemask
);
89 # define wxXmlFindFirst fsys.FindFirst(filemask, wxFILE)
90 # define wxXmlFindNext fsys.FindNext()
92 # define wxXmlFindFirst wxFindFirstFile(filemask, wxFILE)
93 # define wxXmlFindNext wxFindNextFile()
101 // NB: Load() accepts both filenames and URLs (should probably be
102 // changed to filenames only, but embedded resources currently
103 // rely on its ability to handle URLs - FIXME). This check
104 // serves as a quick way to determine whether found name is
105 // filename and not URL:
106 if (wxFileName::FileExists(fnd
))
108 // Make the name absolute filename, because the app may
109 // change working directory later:
114 fnd
= fn
.GetFullPath();
117 fnd
= wxFileSystem::FileNameToURL(fnd
);
122 if (fnd
.Lower().Matches(wxT("*.zip")) ||
123 fnd
.Lower().Matches(wxT("*.xrs")))
125 rt
= rt
&& Load(fnd
+ wxT("#zip:*.xrc"));
130 drec
= new wxXmlResourceDataRecord
;
140 # undef wxXmlFindFirst
141 # undef wxXmlFindNext
142 return rt
&& UpdateResources();
146 IMPLEMENT_ABSTRACT_CLASS(wxXmlResourceHandler
, wxObject
)
148 void wxXmlResource::AddHandler(wxXmlResourceHandler
*handler
)
150 m_handlers
.Append(handler
);
151 handler
->SetParentResource(this);
154 void wxXmlResource::InsertHandler(wxXmlResourceHandler
*handler
)
156 m_handlers
.Insert(handler
);
157 handler
->SetParentResource(this);
162 void wxXmlResource::ClearHandlers()
164 WX_CLEAR_LIST(wxList
, m_handlers
);
168 wxMenu
*wxXmlResource::LoadMenu(const wxString
& name
)
170 return (wxMenu
*)CreateResFromNode(FindResource(name
, wxT("wxMenu")), NULL
, NULL
);
175 wxMenuBar
*wxXmlResource::LoadMenuBar(wxWindow
*parent
, const wxString
& name
)
177 return (wxMenuBar
*)CreateResFromNode(FindResource(name
, wxT("wxMenuBar")), parent
, NULL
);
183 wxToolBar
*wxXmlResource::LoadToolBar(wxWindow
*parent
, const wxString
& name
)
185 return (wxToolBar
*)CreateResFromNode(FindResource(name
, wxT("wxToolBar")), parent
, NULL
);
190 wxDialog
*wxXmlResource::LoadDialog(wxWindow
*parent
, const wxString
& name
)
192 return (wxDialog
*)CreateResFromNode(FindResource(name
, wxT("wxDialog")), parent
, NULL
);
195 bool wxXmlResource::LoadDialog(wxDialog
*dlg
, wxWindow
*parent
, const wxString
& name
)
197 return CreateResFromNode(FindResource(name
, wxT("wxDialog")), parent
, dlg
) != NULL
;
202 wxPanel
*wxXmlResource::LoadPanel(wxWindow
*parent
, const wxString
& name
)
204 return (wxPanel
*)CreateResFromNode(FindResource(name
, wxT("wxPanel")), parent
, NULL
);
207 bool wxXmlResource::LoadPanel(wxPanel
*panel
, wxWindow
*parent
, const wxString
& name
)
209 return CreateResFromNode(FindResource(name
, wxT("wxPanel")), parent
, panel
) != NULL
;
212 wxFrame
*wxXmlResource::LoadFrame(wxWindow
* parent
, const wxString
& name
)
214 return (wxFrame
*)CreateResFromNode(FindResource(name
, wxT("wxFrame")), parent
, NULL
);
217 bool wxXmlResource::LoadFrame(wxFrame
* frame
, wxWindow
*parent
, const wxString
& name
)
219 return CreateResFromNode(FindResource(name
, wxT("wxFrame")), parent
, frame
) != NULL
;
222 wxBitmap
wxXmlResource::LoadBitmap(const wxString
& name
)
224 wxBitmap
*bmp
= (wxBitmap
*)CreateResFromNode(
225 FindResource(name
, wxT("wxBitmap")), NULL
, NULL
);
228 if (bmp
) { rt
= *bmp
; delete bmp
; }
232 wxIcon
wxXmlResource::LoadIcon(const wxString
& name
)
234 wxIcon
*icon
= (wxIcon
*)CreateResFromNode(
235 FindResource(name
, wxT("wxIcon")), NULL
, NULL
);
238 if (icon
) { rt
= *icon
; delete icon
; }
243 wxObject
*wxXmlResource::LoadObject(wxWindow
*parent
, const wxString
& name
, const wxString
& classname
)
245 return CreateResFromNode(FindResource(name
, classname
), parent
, NULL
);
248 bool wxXmlResource::LoadObject(wxObject
*instance
, wxWindow
*parent
, const wxString
& name
, const wxString
& classname
)
250 return CreateResFromNode(FindResource(name
, classname
), parent
, instance
) != NULL
;
254 bool wxXmlResource::AttachUnknownControl(const wxString
& name
,
255 wxWindow
*control
, wxWindow
*parent
)
258 parent
= control
->GetParent();
259 wxWindow
*container
= parent
->FindWindow(name
+ wxT("_container"));
262 wxLogError(_("Cannot find container for unknown control '%s'."), name
.c_str());
265 return control
->Reparent(container
);
269 static void ProcessPlatformProperty(wxXmlNode
*node
)
274 wxXmlNode
*c
= node
->GetChildren();
278 if (!c
->GetPropVal(wxT("platform"), &s
))
282 wxStringTokenizer
tkn(s
, wxT(" |"));
284 while (tkn
.HasMoreTokens())
286 s
= tkn
.GetNextToken();
288 if (s
== wxT("win")) isok
= true;
290 #if defined(__MAC__) || defined(__APPLE__)
291 if (s
== wxT("mac")) isok
= true;
292 #elif defined(__UNIX__)
293 if (s
== wxT("unix")) isok
= true;
296 if (s
== wxT("os2")) isok
= true;
306 ProcessPlatformProperty(c
);
311 wxXmlNode
*c2
= c
->GetNext();
312 node
->RemoveChild(c
);
321 bool wxXmlResource::UpdateResources()
325 # if wxUSE_FILESYSTEM
326 wxFSFile
*file
= NULL
;
331 wxString
encoding(wxT("UTF-8"));
332 #if !wxUSE_UNICODE && wxUSE_INTL
333 if ( (GetFlags() & wxXRC_USE_LOCALE
) == 0 )
335 // In case we are not using wxLocale to translate strings, convert the
336 // strings GUI's charset. This must not be done when wxXRC_USE_LOCALE
337 // is on, because it could break wxGetTranslation lookup.
338 encoding
= wxLocale::GetSystemEncodingName();
342 for (size_t i
= 0; i
< m_data
.GetCount(); i
++)
344 modif
= (m_data
[i
].Doc
== NULL
);
346 if (!modif
&& !(m_flags
& wxXRC_NO_RELOADING
))
348 # if wxUSE_FILESYSTEM
349 file
= fsys
.OpenFile(m_data
[i
].File
);
350 modif
= file
&& file
->GetModificationTime() > m_data
[i
].Time
;
353 wxLogError(_("Cannot open file '%s'."), m_data
[i
].File
.c_str());
359 modif
= wxDateTime(wxFileModificationTime(m_data
[i
].File
)) > m_data
[i
].Time
;
365 wxLogTrace(_T("xrc"),
366 _T("opening file '%s'"), m_data
[i
].File
.c_str());
368 wxInputStream
*stream
= NULL
;
370 # if wxUSE_FILESYSTEM
371 file
= fsys
.OpenFile(m_data
[i
].File
);
373 stream
= file
->GetStream();
375 stream
= new wxFileInputStream(m_data
[i
].File
);
380 delete m_data
[i
].Doc
;
381 m_data
[i
].Doc
= new wxXmlDocument
;
383 if (!stream
|| !m_data
[i
].Doc
->Load(*stream
, encoding
))
385 wxLogError(_("Cannot load resources from file '%s'."),
386 m_data
[i
].File
.c_str());
387 wxDELETE(m_data
[i
].Doc
);
390 else if (m_data
[i
].Doc
->GetRoot()->GetName() != wxT("resource"))
392 wxLogError(_("Invalid XRC resource '%s': doesn't have root node 'resource'."), m_data
[i
].File
.c_str());
393 wxDELETE(m_data
[i
].Doc
);
400 wxString verstr
= m_data
[i
].Doc
->GetRoot()->GetPropVal(
401 wxT("version"), wxT("0.0.0.0"));
402 if (wxSscanf(verstr
.c_str(), wxT("%i.%i.%i.%i"),
403 &v1
, &v2
, &v3
, &v4
) == 4)
404 version
= v1
*256*256*256+v2
*256*256+v3
*256+v4
;
409 if (m_version
!= version
)
411 wxLogError(_("Resource files must have same version number!"));
415 ProcessPlatformProperty(m_data
[i
].Doc
->GetRoot());
417 m_data
[i
].Time
= file
->GetModificationTime();
419 m_data
[i
].Time
= wxDateTime(wxFileModificationTime(m_data
[i
].File
));
423 # if wxUSE_FILESYSTEM
436 wxXmlNode
*wxXmlResource::DoFindResource(wxXmlNode
*parent
,
437 const wxString
& name
,
438 const wxString
& classname
,
444 // first search for match at the top-level nodes (as this is
445 // where the resource is most commonly looked for):
446 for (node
= parent
->GetChildren(); node
; node
= node
->GetNext())
448 if ( node
->GetType() == wxXML_ELEMENT_NODE
&&
449 (node
->GetName() == wxT("object") ||
450 node
->GetName() == wxT("object_ref")) &&
451 node
->GetPropVal(wxT("name"), &dummy
) && dummy
== name
)
453 wxString
cls(node
->GetPropVal(wxT("class"), wxEmptyString
));
454 if (!classname
|| cls
== classname
)
456 // object_ref may not have 'class' property:
457 if (cls
.empty() && node
->GetName() == wxT("object_ref"))
459 wxString refName
= node
->GetPropVal(wxT("ref"), wxEmptyString
);
462 wxXmlNode
* refNode
= FindResource(refName
, wxEmptyString
, true);
464 refNode
->GetPropVal(wxT("class"), wxEmptyString
) == classname
)
473 for (node
= parent
->GetChildren(); node
; node
= node
->GetNext())
475 if ( node
->GetType() == wxXML_ELEMENT_NODE
&&
476 (node
->GetName() == wxT("object") ||
477 node
->GetName() == wxT("object_ref")) )
479 wxXmlNode
* found
= DoFindResource(node
, name
, classname
, true);
488 wxXmlNode
*wxXmlResource::FindResource(const wxString
& name
,
489 const wxString
& classname
,
492 UpdateResources(); //ensure everything is up-to-date
495 for (size_t f
= 0; f
< m_data
.GetCount(); f
++)
497 if ( m_data
[f
].Doc
== NULL
|| m_data
[f
].Doc
->GetRoot() == NULL
)
500 wxXmlNode
* found
= DoFindResource(m_data
[f
].Doc
->GetRoot(),
501 name
, classname
, recursive
);
505 m_curFileSystem
.ChangePathTo(m_data
[f
].File
);
511 wxLogError(_("XRC resource '%s' (class '%s') not found!"),
512 name
.c_str(), classname
.c_str());
516 static void MergeNodes(wxXmlNode
& dest
, wxXmlNode
& with
)
519 for (wxXmlProperty
*prop
= with
.GetProperties(); prop
; prop
= prop
->GetNext())
521 wxXmlProperty
*dprop
;
522 for (dprop
= dest
.GetProperties(); dprop
; dprop
= dprop
->GetNext())
525 if ( dprop
->GetName() == prop
->GetName() )
527 dprop
->SetValue(prop
->GetValue());
533 dest
.AddProperty(prop
->GetName(), prop
->GetValue());
536 // Merge child nodes:
537 for (wxXmlNode
* node
= with
.GetChildren(); node
; node
= node
->GetNext())
539 wxString name
= node
->GetPropVal(wxT("name"), wxEmptyString
);
542 for (dnode
= dest
.GetChildren(); dnode
; dnode
= dnode
->GetNext() )
544 if ( dnode
->GetName() == node
->GetName() &&
545 dnode
->GetPropVal(wxT("name"), wxEmptyString
) == name
&&
546 dnode
->GetType() == node
->GetType() )
548 MergeNodes(*dnode
, *node
);
554 dest
.AddChild(new wxXmlNode(*node
));
557 if ( dest
.GetType() == wxXML_TEXT_NODE
&& with
.GetContent().Length() )
558 dest
.SetContent(with
.GetContent());
561 wxObject
*wxXmlResource::CreateResFromNode(wxXmlNode
*node
, wxObject
*parent
,
563 wxXmlResourceHandler
*handlerToUse
)
565 if (node
== NULL
) return NULL
;
567 // handling of referenced resource
568 if ( node
->GetName() == wxT("object_ref") )
570 wxString refName
= node
->GetPropVal(wxT("ref"), wxEmptyString
);
571 wxXmlNode
* refNode
= FindResource(refName
, wxEmptyString
, true);
575 wxLogError(_("Referenced object node with ref=\"%s\" not found!"),
580 wxXmlNode
copy(*refNode
);
581 MergeNodes(copy
, *node
);
583 return CreateResFromNode(©
, parent
, instance
);
586 wxXmlResourceHandler
*handler
;
590 if (handlerToUse
->CanHandle(node
))
592 return handlerToUse
->CreateResource(node
, parent
, instance
);
595 else if (node
->GetName() == wxT("object"))
597 wxList::compatibility_iterator ND
= m_handlers
.GetFirst();
600 handler
= (wxXmlResourceHandler
*)ND
->GetData();
601 if (handler
->CanHandle(node
))
603 return handler
->CreateResource(node
, parent
, instance
);
609 wxLogError(_("No handler found for XML node '%s', class '%s'!"),
610 node
->GetName().c_str(),
611 node
->GetPropVal(wxT("class"), wxEmptyString
).c_str());
616 #include "wx/listimpl.cpp"
617 WX_DECLARE_LIST(wxXmlSubclassFactory
, wxXmlSubclassFactoriesList
);
618 WX_DEFINE_LIST(wxXmlSubclassFactoriesList
);
620 wxXmlSubclassFactoriesList
*wxXmlResource::ms_subclassFactories
= NULL
;
622 /*static*/ void wxXmlResource::AddSubclassFactory(wxXmlSubclassFactory
*factory
)
624 if (!ms_subclassFactories
)
626 ms_subclassFactories
= new wxXmlSubclassFactoriesList
;
628 ms_subclassFactories
->Append(factory
);
631 class wxXmlSubclassFactoryCXX
: public wxXmlSubclassFactory
634 ~wxXmlSubclassFactoryCXX() {}
636 wxObject
*Create(const wxString
& className
)
638 wxClassInfo
* classInfo
= wxClassInfo::FindClass(className
);
641 return classInfo
->CreateObject();
651 wxXmlResourceHandler::wxXmlResourceHandler()
652 : m_node(NULL
), m_parent(NULL
), m_instance(NULL
),
653 m_parentAsWindow(NULL
), m_instanceAsWindow(NULL
)
658 wxObject
*wxXmlResourceHandler::CreateResource(wxXmlNode
*node
, wxObject
*parent
, wxObject
*instance
)
660 wxXmlNode
*myNode
= m_node
;
661 wxString myClass
= m_class
;
662 wxObject
*myParent
= m_parent
, *myInstance
= m_instance
;
663 wxWindow
*myParentAW
= m_parentAsWindow
, *myInstanceAW
= m_instanceAsWindow
;
665 m_instance
= instance
;
666 if (!m_instance
&& node
->HasProp(wxT("subclass")) &&
667 !(m_resource
->GetFlags() & wxXRC_NO_SUBCLASSING
))
669 wxString subclass
= node
->GetPropVal(wxT("subclass"), wxEmptyString
);
670 if (!subclass
.empty())
672 for (wxXmlSubclassFactoriesList::compatibility_iterator i
= wxXmlResource::ms_subclassFactories
->GetFirst();
675 m_instance
= i
->GetData()->Create(subclass
);
682 wxString name
= node
->GetPropVal(wxT("name"), wxEmptyString
);
683 wxLogError(_("Subclass '%s' not found for resource '%s', not subclassing!"),
684 subclass
.c_str(), name
.c_str());
690 m_class
= node
->GetPropVal(wxT("class"), wxEmptyString
);
692 m_parentAsWindow
= wxDynamicCast(m_parent
, wxWindow
);
693 m_instanceAsWindow
= wxDynamicCast(m_instance
, wxWindow
);
695 wxObject
*returned
= DoCreateResource();
699 m_parent
= myParent
; m_parentAsWindow
= myParentAW
;
700 m_instance
= myInstance
; m_instanceAsWindow
= myInstanceAW
;
706 void wxXmlResourceHandler::AddStyle(const wxString
& name
, int value
)
708 m_styleNames
.Add(name
);
709 m_styleValues
.Add(value
);
714 void wxXmlResourceHandler::AddWindowStyles()
716 XRC_ADD_STYLE(wxCLIP_CHILDREN
);
717 XRC_ADD_STYLE(wxSIMPLE_BORDER
);
718 XRC_ADD_STYLE(wxSUNKEN_BORDER
);
719 XRC_ADD_STYLE(wxDOUBLE_BORDER
);
720 XRC_ADD_STYLE(wxRAISED_BORDER
);
721 XRC_ADD_STYLE(wxSTATIC_BORDER
);
722 XRC_ADD_STYLE(wxNO_BORDER
);
723 XRC_ADD_STYLE(wxTRANSPARENT_WINDOW
);
724 XRC_ADD_STYLE(wxWANTS_CHARS
);
725 XRC_ADD_STYLE(wxNO_FULL_REPAINT_ON_RESIZE
);
726 XRC_ADD_STYLE(wxFULL_REPAINT_ON_RESIZE
);
727 XRC_ADD_STYLE(wxWS_EX_BLOCK_EVENTS
);
732 bool wxXmlResourceHandler::HasParam(const wxString
& param
)
734 return (GetParamNode(param
) != NULL
);
738 int wxXmlResourceHandler::GetStyle(const wxString
& param
, int defaults
)
740 wxString s
= GetParamValue(param
);
742 if (!s
) return defaults
;
744 wxStringTokenizer
tkn(s
, wxT("| \t\n"), wxTOKEN_STRTOK
);
748 while (tkn
.HasMoreTokens())
750 fl
= tkn
.GetNextToken();
751 index
= m_styleNames
.Index(fl
);
752 if (index
!= wxNOT_FOUND
)
753 style
|= m_styleValues
[index
];
755 wxLogError(_("Unknown style flag ") + fl
);
762 wxString
wxXmlResourceHandler::GetText(const wxString
& param
, bool translate
)
764 wxXmlNode
*parNode
= GetParamNode(param
);
765 wxString
str1(GetNodeContent(parNode
));
770 // VS: First version of XRC resources used $ instead of & (which is
771 // illegal in XML), but later I realized that '_' fits this purpose
772 // much better (because &File means "File with F underlined").
773 if (m_resource
->CompareVersion(2,3,0,1) < 0)
778 for (dt
= str1
.c_str(); *dt
; dt
++)
780 // Remap amp_char to &, map double amp_char to amp_char (for things
781 // like "&File..." -- this is illegal in XML, so we use "_File..."):
784 if ( *(++dt
) == amp_char
)
787 str2
<< wxT('&') << *dt
;
789 // Remap \n to CR, \r to LF, \t to TAB:
790 else if (*dt
== wxT('\\'))
793 case wxT('n') : str2
<< wxT('\n'); break;
794 case wxT('t') : str2
<< wxT('\t'); break;
795 case wxT('r') : str2
<< wxT('\r'); break;
796 default : str2
<< wxT('\\') << *dt
; break;
801 if (m_resource
->GetFlags() & wxXRC_USE_LOCALE
)
803 if (translate
&& parNode
&&
804 parNode
->GetPropVal(wxT("translate"), wxEmptyString
) != wxT("0"))
806 return wxGetTranslation(str2
);
813 // The string is internally stored as UTF-8, we have to convert
814 // it into system's default encoding so that it can be displayed:
815 return wxString(str2
.mb_str(wxConvUTF8
), wxConvLocal
);
821 // If wxXRC_USE_LOCALE is not set, then the string is already in
822 // system's default encoding in ANSI build, so we don't have to
823 // do anything special here.
830 long wxXmlResourceHandler::GetLong(const wxString
& param
, long defaultv
)
833 wxString str1
= GetParamValue(param
);
835 if (!str1
.ToLong(&value
))
843 int wxXmlResourceHandler::GetID()
845 return wxXmlResource::GetXRCID(GetName());
850 wxString
wxXmlResourceHandler::GetName()
852 return m_node
->GetPropVal(wxT("name"), wxT("-1"));
857 bool wxXmlResourceHandler::GetBool(const wxString
& param
, bool defaultv
)
859 wxString v
= GetParamValue(param
);
861 if (!v
) return defaultv
;
862 else return (v
== wxT("1"));
867 wxColour
wxXmlResourceHandler::GetColour(const wxString
& param
)
869 wxString v
= GetParamValue(param
);
870 unsigned long tmp
= 0;
872 if (v
.Length() != 7 || v
[0u] != wxT('#') ||
873 wxSscanf(v
.c_str(), wxT("#%lX"), &tmp
) != 1)
875 wxLogError(_("XRC resource: Incorrect colour specification '%s' for property '%s'."),
876 v
.c_str(), param
.c_str());
880 return wxColour((unsigned char) ((tmp
& 0xFF0000) >> 16) ,
881 (unsigned char) ((tmp
& 0x00FF00) >> 8),
882 (unsigned char) ((tmp
& 0x0000FF)));
887 wxBitmap
wxXmlResourceHandler::GetBitmap(const wxString
& param
,
888 const wxArtClient
& defaultArtClient
,
891 /* If the bitmap is specified as stock item, query wxArtProvider for it: */
892 wxXmlNode
*bmpNode
= GetParamNode(param
);
895 wxString sid
= bmpNode
->GetPropVal(wxT("stock_id"), wxEmptyString
);
898 wxString scl
= bmpNode
->GetPropVal(wxT("stock_client"), defaultArtClient
);
900 wxArtProvider::GetBitmap(wxART_MAKE_ART_ID_FROM_STR(sid
),
901 wxART_MAKE_CLIENT_ID_FROM_STR(scl
),
908 /* ...or load the bitmap from file: */
909 wxString name
= GetParamValue(param
);
910 if (name
.IsEmpty()) return wxNullBitmap
;
912 wxFSFile
*fsfile
= GetCurFileSystem().OpenFile(name
);
915 wxLogError(_("XRC resource: Cannot create bitmap from '%s'."),
919 wxImage
img(*(fsfile
->GetStream()));
922 wxImage
img(GetParamValue(wxT("bitmap")));
927 wxLogError(_("XRC resource: Cannot create bitmap from '%s'."), param
.c_str());
930 if (!(size
== wxDefaultSize
)) img
.Rescale(size
.x
, size
.y
);
931 return wxBitmap(img
);
937 wxIcon
wxXmlResourceHandler::GetIcon(const wxString
& param
,
938 const wxArtClient
& defaultArtClient
,
942 icon
.CopyFromBitmap(GetBitmap(param
, defaultArtClient
, size
));
948 wxXmlNode
*wxXmlResourceHandler::GetParamNode(const wxString
& param
)
950 wxCHECK_MSG(m_node
, NULL
, wxT("You can't access handler data before it was initialized!"));
952 wxXmlNode
*n
= m_node
->GetChildren();
956 if (n
->GetType() == wxXML_ELEMENT_NODE
&& n
->GetName() == param
)
964 wxString
wxXmlResourceHandler::GetNodeContent(wxXmlNode
*node
)
967 if (n
== NULL
) return wxEmptyString
;
968 n
= n
->GetChildren();
972 if (n
->GetType() == wxXML_TEXT_NODE
||
973 n
->GetType() == wxXML_CDATA_SECTION_NODE
)
974 return n
->GetContent();
977 return wxEmptyString
;
982 wxString
wxXmlResourceHandler::GetParamValue(const wxString
& param
)
985 return GetNodeContent(m_node
);
987 return GetNodeContent(GetParamNode(param
));
992 wxSize
wxXmlResourceHandler::GetSize(const wxString
& param
)
994 wxString s
= GetParamValue(param
);
995 if (s
.IsEmpty()) s
= wxT("-1,-1");
999 is_dlg
= s
[s
.Length()-1] == wxT('d');
1000 if (is_dlg
) s
.RemoveLast();
1002 if (!s
.BeforeFirst(wxT(',')).ToLong(&sx
) ||
1003 !s
.AfterLast(wxT(',')).ToLong(&sy
))
1005 wxLogError(_("Cannot parse coordinates from '%s'."), s
.c_str());
1006 return wxDefaultSize
;
1011 if (m_instanceAsWindow
)
1012 return wxDLG_UNIT(m_instanceAsWindow
, wxSize(sx
, sy
));
1013 else if (m_parentAsWindow
)
1014 return wxDLG_UNIT(m_parentAsWindow
, wxSize(sx
, sy
));
1017 wxLogError(_("Cannot convert dialog units: dialog unknown."));
1018 return wxDefaultSize
;
1021 else return wxSize(sx
, sy
);
1026 wxPoint
wxXmlResourceHandler::GetPosition(const wxString
& param
)
1028 wxSize sz
= GetSize(param
);
1029 return wxPoint(sz
.x
, sz
.y
);
1034 wxCoord
wxXmlResourceHandler::GetDimension(const wxString
& param
, wxCoord defaultv
)
1036 wxString s
= GetParamValue(param
);
1037 if (s
.IsEmpty()) return defaultv
;
1041 is_dlg
= s
[s
.Length()-1] == wxT('d');
1042 if (is_dlg
) s
.RemoveLast();
1046 wxLogError(_("Cannot parse dimension from '%s'."), s
.c_str());
1052 if (m_instanceAsWindow
)
1053 return wxDLG_UNIT(m_instanceAsWindow
, wxSize(sx
, 0)).x
;
1054 else if (m_parentAsWindow
)
1055 return wxDLG_UNIT(m_parentAsWindow
, wxSize(sx
, 0)).x
;
1058 wxLogError(_("Cannot convert dialog units: dialog unknown."));
1067 wxFont
wxXmlResourceHandler::GetFont(const wxString
& param
)
1069 wxXmlNode
*font_node
= GetParamNode(param
);
1070 if (font_node
== NULL
)
1072 wxLogError(_("Cannot find font node '%s'."), param
.c_str());
1076 wxXmlNode
*oldnode
= m_node
;
1079 long size
= GetLong(wxT("size"), 12);
1081 wxString style
= GetParamValue(wxT("style"));
1082 wxString weight
= GetParamValue(wxT("weight"));
1083 int istyle
= wxNORMAL
, iweight
= wxNORMAL
;
1084 if (style
== wxT("italic")) istyle
= wxITALIC
;
1085 else if (style
== wxT("slant")) istyle
= wxSLANT
;
1086 if (weight
== wxT("bold")) iweight
= wxBOLD
;
1087 else if (weight
== wxT("light")) iweight
= wxLIGHT
;
1089 wxString family
= GetParamValue(wxT("family"));
1090 int ifamily
= wxDEFAULT
;
1091 if (family
== wxT("decorative")) ifamily
= wxDECORATIVE
;
1092 else if (family
== wxT("roman")) ifamily
= wxROMAN
;
1093 else if (family
== wxT("script")) ifamily
= wxSCRIPT
;
1094 else if (family
== wxT("swiss")) ifamily
= wxSWISS
;
1095 else if (family
== wxT("modern")) ifamily
= wxMODERN
;
1097 bool underlined
= GetBool(wxT("underlined"), false);
1099 wxString encoding
= GetParamValue(wxT("encoding"));
1100 wxFontMapper mapper
;
1101 wxFontEncoding enc
= wxFONTENCODING_DEFAULT
;
1102 if (!encoding
.IsEmpty())
1103 enc
= mapper
.CharsetToEncoding(encoding
);
1104 if (enc
== wxFONTENCODING_SYSTEM
)
1105 enc
= wxFONTENCODING_DEFAULT
;
1107 wxString faces
= GetParamValue(wxT("face"));
1108 wxString facename
= wxEmptyString
;
1109 wxFontEnumerator enu
;
1110 enu
.EnumerateFacenames();
1111 wxStringTokenizer
tk(faces
, wxT(","));
1112 while (tk
.HasMoreTokens())
1114 int index
= enu
.GetFacenames()->Index(tk
.GetNextToken(), false);
1115 if (index
!= wxNOT_FOUND
)
1117 facename
= (*enu
.GetFacenames())[index
];
1124 wxFont
font(size
, ifamily
, istyle
, iweight
, underlined
, facename
, enc
);
1129 void wxXmlResourceHandler::SetupWindow(wxWindow
*wnd
)
1131 //FIXME : add cursor
1133 if (HasParam(wxT("exstyle")))
1134 // Have to OR it with existing style, since
1135 // some implementations (e.g. wxGTK) use the extra style
1137 wnd
->SetExtraStyle(wnd
->GetExtraStyle() | GetStyle(wxT("exstyle")));
1138 if (HasParam(wxT("bg")))
1139 wnd
->SetBackgroundColour(GetColour(wxT("bg")));
1140 if (HasParam(wxT("fg")))
1141 wnd
->SetForegroundColour(GetColour(wxT("fg")));
1142 if (GetBool(wxT("enabled"), 1) == 0)
1144 if (GetBool(wxT("focused"), 0) == 1)
1146 if (GetBool(wxT("hidden"), 0) == 1)
1149 if (HasParam(wxT("tooltip")))
1150 wnd
->SetToolTip(GetText(wxT("tooltip")));
1152 if (HasParam(wxT("font")))
1153 wnd
->SetFont(GetFont());
1157 void wxXmlResourceHandler::CreateChildren(wxObject
*parent
, bool this_hnd_only
)
1159 wxXmlNode
*n
= m_node
->GetChildren();
1163 if (n
->GetType() == wxXML_ELEMENT_NODE
&&
1164 (n
->GetName() == wxT("object") || n
->GetName() == wxT("object_ref")))
1166 m_resource
->CreateResFromNode(n
, parent
, NULL
,
1167 this_hnd_only
? this : NULL
);
1174 void wxXmlResourceHandler::CreateChildrenPrivately(wxObject
*parent
, wxXmlNode
*rootnode
)
1177 if (rootnode
== NULL
) root
= m_node
; else root
= rootnode
;
1178 wxXmlNode
*n
= root
->GetChildren();
1182 if (n
->GetType() == wxXML_ELEMENT_NODE
&& CanHandle(n
))
1184 CreateResource(n
, parent
, NULL
);
1196 // --------------- XRCID implementation -----------------------------
1198 #define XRCID_TABLE_SIZE 1024
1208 static XRCID_record
*XRCID_Records
[XRCID_TABLE_SIZE
] = {NULL
};
1210 static int XRCID_Lookup(const wxChar
*str_id
, int value_if_not_found
= -2)
1212 static int XRCID_LastID
= wxID_HIGHEST
;
1216 for (const wxChar
*c
= str_id
; *c
!= wxT('\0'); c
++) index
+= (int)*c
;
1217 index
%= XRCID_TABLE_SIZE
;
1219 XRCID_record
*oldrec
= NULL
;
1220 for (XRCID_record
*rec
= XRCID_Records
[index
]; rec
; rec
= rec
->next
)
1222 if (wxStrcmp(rec
->key
, str_id
) == 0)
1229 XRCID_record
**rec_var
= (oldrec
== NULL
) ?
1230 &XRCID_Records
[index
] : &oldrec
->next
;
1231 *rec_var
= new XRCID_record
;
1232 (*rec_var
)->key
= wxStrdup(str_id
);
1233 (*rec_var
)->next
= NULL
;
1236 if (value_if_not_found
!= -2)
1237 (*rec_var
)->id
= value_if_not_found
;
1240 int asint
= wxStrtol(str_id
, &end
, 10);
1241 if (*str_id
&& *end
== 0)
1243 // if str_id was integer, keep it verbosely:
1244 (*rec_var
)->id
= asint
;
1248 (*rec_var
)->id
= ++XRCID_LastID
;
1252 return (*rec_var
)->id
;
1255 /*static*/ int wxXmlResource::GetXRCID(const wxChar
*str_id
)
1257 return XRCID_Lookup(str_id
);
1261 static void CleanXRCID_Record(XRCID_record
*rec
)
1265 CleanXRCID_Record(rec
->next
);
1271 static void CleanXRCID_Records()
1273 for (int i
= 0; i
< XRCID_TABLE_SIZE
; i
++)
1275 CleanXRCID_Record(XRCID_Records
[i
]);
1276 XRCID_Records
[i
] = NULL
;
1280 static void AddStdXRCID_Records()
1282 #define stdID(id) XRCID_Lookup(wxT(#id), id)
1284 stdID(wxID_OPEN
); stdID(wxID_CLOSE
); stdID(wxID_NEW
);
1285 stdID(wxID_SAVE
); stdID(wxID_SAVEAS
); stdID(wxID_REVERT
);
1286 stdID(wxID_EXIT
); stdID(wxID_UNDO
); stdID(wxID_REDO
);
1287 stdID(wxID_HELP
); stdID(wxID_PRINT
); stdID(wxID_PRINT_SETUP
);
1288 stdID(wxID_PREVIEW
); stdID(wxID_ABOUT
); stdID(wxID_HELP_CONTENTS
);
1289 stdID(wxID_HELP_COMMANDS
); stdID(wxID_HELP_PROCEDURES
);
1290 stdID(wxID_CUT
); stdID(wxID_COPY
); stdID(wxID_PASTE
);
1291 stdID(wxID_CLEAR
); stdID(wxID_FIND
); stdID(wxID_DUPLICATE
);
1292 stdID(wxID_SELECTALL
); stdID(wxID_OK
); stdID(wxID_CANCEL
);
1293 stdID(wxID_APPLY
); stdID(wxID_YES
); stdID(wxID_NO
);
1294 stdID(wxID_STATIC
); stdID(wxID_FORWARD
); stdID(wxID_BACKWARD
);
1295 stdID(wxID_DEFAULT
); stdID(wxID_MORE
); stdID(wxID_SETUP
);
1296 stdID(wxID_RESET
); stdID(wxID_HELP_CONTEXT
);
1297 stdID(wxID_CLOSE_ALL
);
1305 // --------------- module and globals -----------------------------
1307 class wxXmlResourceModule
: public wxModule
1309 DECLARE_DYNAMIC_CLASS(wxXmlResourceModule
)
1311 wxXmlResourceModule() {}
1314 AddStdXRCID_Records();
1315 wxXmlResource::AddSubclassFactory(new wxXmlSubclassFactoryCXX
);
1320 delete wxXmlResource::Set(NULL
);
1321 if(wxXmlResource::ms_subclassFactories
)
1322 WX_CLEAR_LIST(wxXmlSubclassFactoriesList
, *wxXmlResource::ms_subclassFactories
);
1323 wxDELETE(wxXmlResource::ms_subclassFactories
);
1324 CleanXRCID_Records();
1328 IMPLEMENT_DYNAMIC_CLASS(wxXmlResourceModule
, wxModule
)
1331 // When wxXml is loaded dynamically after the application is already running
1332 // then the built-in module system won't pick this one up. Add it manually.
1333 void wxXmlInitResourceModule()
1335 wxModule
* module = new wxXmlResourceModule
;
1337 wxModule::RegisterModule(module);