X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/3d3a4a25addf9729c74a8d0f7a056b57bb7f7e2a..7a5e53abb3f9ea13eba2880972306446e4e21e83:/src/common/rendcmn.cpp diff --git a/src/common/rendcmn.cpp b/src/common/rendcmn.cpp index 96a0c68318..2cedd7f4de 100644 --- a/src/common/rendcmn.cpp +++ b/src/common/rendcmn.cpp @@ -25,6 +25,8 @@ #ifndef WX_PRECOMP #include "wx/app.h" + #include "wx/log.h" + #include "wx/intl.h" #endif //WX_PRECOMP #include "wx/apptrait.h" @@ -32,6 +34,10 @@ #include "wx/ptr_scpd.h" +#if wxUSE_DYNLIB_CLASS + #include "wx/dynlib.h" +#endif // wxUSE_DYNLIB_CLASS + // ---------------------------------------------------------------------------- // wxRendererPtr: auto pointer holding the global renderer // ---------------------------------------------------------------------------- @@ -42,8 +48,6 @@ wxDEFINE_SCOPED_PTR(wxRendererNative, wxRendererPtrBase); class wxRendererPtr : public wxRendererPtrBase { public: - wxRendererPtr() : wxRendererPtrBase(NULL) { m_initialized = false; } - // return true if we have a renderer, false otherwise bool IsOk() { @@ -58,7 +62,17 @@ public: return get() != NULL; } + // return the global and unique wxRendererPtr + static wxRendererPtr& Get() + { + static wxRendererPtr s_renderer; + + return s_renderer; + } + private: + wxRendererPtr() : wxRendererPtrBase(NULL) { m_initialized = false; } + void DoInit() { wxAppTraits *traits = wxTheApp ? wxTheApp->GetTraits() : NULL; @@ -71,18 +85,122 @@ private: bool m_initialized; + // just to suppress a gcc warning + friend class wxRendererPtrDummyFriend; + DECLARE_NO_COPY_CLASS(wxRendererPtr) }; +#if wxUSE_DYNLIB_CLASS + +// ---------------------------------------------------------------------------- +// wxRendererFromDynLib: represents a renderer dynamically loaded from a DLL +// ---------------------------------------------------------------------------- + +class wxRendererFromDynLib : public wxDelegateRendererNative +{ +public: + // create the object wrapping the given renderer created from this DLL + // + // we take ownership of the pointer and will delete it (and also unload the + // DLL) when we're deleted + wxRendererFromDynLib(wxDynamicLibrary& dll, wxRendererNative *renderer) + : wxDelegateRendererNative(*renderer), + m_renderer(renderer), + m_dllHandle(dll.Detach()) + { + } + + virtual ~wxRendererFromDynLib() + { + delete m_renderer; + wxDynamicLibrary::Unload(m_dllHandle); + } + +private: + wxRendererNative *m_renderer; + wxDllType m_dllHandle; +}; + +#endif // wxUSE_DYNLIB_CLASS + // ============================================================================ // wxRendererNative implementation // ============================================================================ +wxRendererNative::~wxRendererNative() +{ + // empty but necessary +} + +// ---------------------------------------------------------------------------- +// Managing the global renderer +// ---------------------------------------------------------------------------- + /* static */ wxRendererNative& wxRendererNative::Get() { - static wxRendererPtr s_renderer; + wxRendererPtr& renderer = wxRendererPtr::Get(); - return s_renderer.IsOk() ? *s_renderer.get() : GetDefault(); + return renderer.IsOk() ? *renderer.get() : GetDefault(); } +/* static */ +wxRendererNative *wxRendererNative::Set(wxRendererNative *rendererNew) +{ + wxRendererPtr& renderer = wxRendererPtr::Get(); + + wxRendererNative *rendererOld = renderer.release(); + + renderer.reset(rendererNew); + + return rendererOld; +} + + +// ---------------------------------------------------------------------------- +// Dynamic renderers loading +// ---------------------------------------------------------------------------- + +#if wxUSE_DYNLIB_CLASS + +/* static */ +wxRendererNative *wxRendererNative::Load(const wxString& name) +{ + wxString fullname = wxDynamicLibrary::CanonicalizePluginName(name); + + wxDynamicLibrary dll(fullname); + if ( !dll.IsLoaded() ) + return NULL; + + // each theme DLL must export a wxCreateRenderer() function with this + // signature + typedef wxRendererNative *(*wxCreateRenderer_t)(); + + wxDYNLIB_FUNCTION(wxCreateRenderer_t, wxCreateRenderer, dll); + if ( !pfnwxCreateRenderer ) + return NULL; + + // create a renderer object + wxRendererNative *renderer = (*pfnwxCreateRenderer)(); + if ( !renderer ) + return NULL; + + // check that its version is compatible with ours + wxRendererVersion ver = renderer->GetVersion(); + if ( !wxRendererVersion::IsCompatible(ver) ) + { + wxLogError(_("Renderer \"%s\" has incompatible version %d.%d and couldn't be loaded."), + name.c_str(), ver.version, ver.age); + delete renderer; + + return NULL; + } + + // finally wrap the renderer in an object which will delete it and unload + // the library when it is deleted and return it to the caller + return new wxRendererFromDynLib(dll, renderer); +} + +#endif // wxUSE_DYNLIB_CLASS +