/********************************************************************** Audacity: A Digital Audio Editor Theme.cpp James Crook Audacity is free software. This file is licensed under the wxWidgets license, see License.txt ********************************************************************//** \class Theme \brief Based on ThemeBase, Theme manages image and icon resources. Theme is a class which manages theme resources. It maps sets of ids to the resources and to names of the resources, so that they can be loaded/saved from files. Theme adds the Audacity specific images to ThemeBase. \see \ref Themability *//*****************************************************************//** \class ThemeBase \brief Theme management - Image loading and saving. Base for the Theme class. ThemeBase is a generic non-Audacity specific class. \see \ref Themability *//*****************************************************************//** \class FlowPacker \brief Packs rectangular boxes into a rectangle, using simple first fit. This class is currently used by Theme to pack its images into the image cache. Perhaps someday we will improve FlowPacker and make it more flexible, and use it for toolbar and window layouts too. *//*****************************************************************//** \class SourceOutputStream \brief Allows us to capture output of the Save .png and 'pipe' it into our own output function which gives a series of numbers. This class is currently used by Theme to pack its images into the image cache. Perhaps someday we will improve FlowPacker and make it more flexible, and use it for toolbar and window layouts too. *//*****************************************************************/ #include "Audacity.h" #include #include #include #include #include #include "Project.h" #include "toolbars/ToolBar.h" #include "toolbars/ControlToolBar.h" #include "ImageManipulation.h" #include "Theme.h" #include "Experimental.h" #include "AllThemeResources.h" // can remove this later, only needed for 'XPMS_RETIRED'. #include "FileNames.h" #include "Prefs.h" #include WX_DEFINE_USER_EXPORTED_OBJARRAY( ArrayOfImages ) WX_DEFINE_USER_EXPORTED_OBJARRAY( ArrayOfBitmaps ) WX_DEFINE_USER_EXPORTED_OBJARRAY( ArrayOfColours ) // JKC: First get the MAC specific images. // As we've disabled USE_AQUA_THEME, we need to name each file we use. // // JKC: Mac Hackery. // These #defines are very temporary. We want to ensure the Mac XPM names don't collide with // the PC XPM names, so we do some #defines and later undo them. // Use the same trick wherever we need to avoid name collisions. // All this will vanish when the XPMs are eliminated. // Indeed XPMS_RETIRED the #ifndef ensures we're already not using any of it. #ifndef XPMS_RETIRED // This step should mean that we get PC/Linux images only // except where we EXPLICITLY request otherwise. #undef USE_AQUA_THEME // This step ensures we treat the cursors as 32x32 even on Mac. // We're not yet creating the cursors from the theme, so // all this ensures is that the sizing on PC and Mac stays in step. #define CURSORS_SIZE32 #define DownButton MacDownButton #define HiliteButton MacHiliteButton #define UpButton MacUpButton #define Down MacDown #define Hilite MacHilite #define Up MacUp #define Slider MacSlider #define SliderThumb MacSliderThumb #include "../images/Aqua/HiliteButtonSquare.xpm" #include "../images/Aqua/UpButtonSquare.xpm" #include "../images/Aqua/DownButtonSquare.xpm" #include "../images/Aqua/Slider.xpm" #include "../images/Aqua/SliderThumb.xpm" #include "../images/Aqua/Down.xpm" #include "../images/Aqua/Hilite.xpm" #include "../images/Aqua/Up.xpm" #if 0 // These ones aren't used... #include "../images/Aqua/DownButtonStripes.xpm" #include "../images/Aqua/DownButtonWhite.xpm" #include "../images/Aqua/HiliteButtonStripes.xpm" #include "../images/Aqua/HiliteButtonWhite.xpm" #include "../images/Aqua/UpButtonStripes.xpm" #include "../images/Aqua/UpButtonWhite.xpm" #endif #undef DownButton #undef UpButton #undef HiliteButton #undef Down #undef Hilite #undef Up #undef Slider #undef SliderThumb //-- OK now on to includes for Linux/PC images. #include "../images/PostfishButtons.h" #include "../images/ControlButtons.h" #define HAVE_SHARED_BUTTONS #include "../images/EditButtons.h" #include "../images/MixerImages.h" #include "../images/Cursors.h" #include "../images/ToolBarButtons.h" #include "../images/TranscriptionButtons.h" #include "../images/ToolsButtons.h" #include "../images/ExpandingToolBar/ToolBarToggle.xpm" #include "../images/ExpandingToolBar/ToolBarTarget.xpm" #include "../images/ExpandingToolBar/ToolBarGrabber.xpm" #define Slider VolumeSlider #define SliderThumb VolumeSliderThumb #include "../images/ControlButtons/Slider.xpm" #include "../images/ControlButtons/SliderThumb.xpm" #undef Slider #undef SliderThumb // A different slider's thumb. #include "../images/SliderThumb.xpm" #include "../images/SliderThumbAlpha.xpm" // Include files to get the default images //#include "../images/Aqua.xpm" #include "../images/Arrow.xpm" #include "../images/GlyphImages.h" #include "../images/UploadImages.h" #include "../images/AudacityLogoWithName.xpm" //#include "../images/AudacityLogo.xpm" #include "../images/AudacityLogo48x48.xpm" #endif // This declares the variables such as // int BmpRecordButton = -1; #define THEME_DECLARATIONS #include "AllThemeResources.h" // Include the ImageCache... unsigned char ImageCacheAsData[] = { #include "ThemeAsCeeCode.h" }; // theTheme is a global variable. AUDACITY_DLL_API Theme theTheme; Theme::Theme(void) { mbInitialised=false; } Theme::~Theme(void) { } void Theme::EnsureInitialised() { if( mbInitialised ) return; RegisterImages(); RegisterColours(); #ifdef EXPERIMENTAL_EXTRA_THEME_RESOURCES extern void RegisterExtraThemeResources(); RegisterExtraThemeResources(); #endif bool bLoadThemeAtStart; gPrefs->Read( wxT("/Theme/LoadAtStart"), &bLoadThemeAtStart, false ); LoadThemeAtStartUp( bLoadThemeAtStart ); } void Theme::ApplyUpdatedImages() { AudacityProject *p = GetActiveProject(); if( p->GetControlToolBar() ) { p->GetControlToolBar()->ReCreateButtons(); } } void Theme::RegisterImages() { if( mbInitialised ) return; mbInitialised = true; // This initialises the variables e.g // RegisterImage( bmpRecordButton, some image, wxT("RecordButton")); #define THEME_INITS #include "AllThemeResources.h" } void Theme::RegisterColours() { } ThemeBase::ThemeBase(void) { } ThemeBase::~ThemeBase(void) { } /// This function is called to load the initial Theme images. /// There are many possible choices for what this function /// should do, as we have (potentially) four sources of images. /// - (deprecated) programmed in XPMs. /// - Programmed in in-built theme. /// - External image Cache file. /// - External component files. /// /// We currently still have the deprecated XPMs, so we have /// those being used if the user decides not to load themes. /// /// @param bLookForExternalFiles uses file iff true. void ThemeBase::LoadThemeAtStartUp( bool bLookForExternalFiles ) { EnsureInitialised(); const bool cbBinaryRead =true; const bool cbOkIfNotFound = true; // IF not interested in external files, // THEN just use the internal default set. if( !bLookForExternalFiles ) { // IF the XPMs have been retired, THEN we'd better use the built-in cache // at start up. // ELSE do nothing, we already have XPM based images. #ifdef XPMS_RETIRED ReadThemeInternal(); #endif return; } // ELSE IF can't read the external image cache. else if( !ReadImageCache( cbBinaryRead, cbOkIfNotFound ) ) { // THEN get the default set. ReadThemeInternal(); // JKC: Now we could go on and load the individual images // on top of the default images using the commented out // code that follows... // // However, I think it is better to get the user to // build a new image cache, which they can do easily // from the Theme preferences tab. #if 0 // and now add any available component images. LoadComponents( cbOkIfNotFound ); // JKC: I'm usure about doing this next step automatically. // Suppose the disk is write protected? // Is having the image cache created automatically // going to confuse users? Do we need version specific names? // and now save the combined image as a cache for later use. // We should load the images a little faster in future as a result. CreateImageCache(); #endif } // Next line is not required as we haven't yet built the GUI // when this function is (or should be) called. // ApplyUpdatedImages(); } wxImage ThemeBase::MaskedImage( char const ** pXpm, char const ** pMask ) { wxBitmap Bmp1( pXpm ); wxBitmap Bmp2( pMask ); // wxLogDebug( wxT("Image 1: %i Image 2: %i"), // Bmp1.GetDepth(), Bmp2.GetDepth() ); // We want a 24-bit-depth bitmap if all is working, but on some // platforms it might just return -1 (which means best available // or not relevant). // JKC: \todo check that we're not relying on 24 bit elsewhere. wxASSERT( Bmp1.GetDepth()==-1 || Bmp1.GetDepth()==24); wxASSERT( Bmp1.GetDepth()==-1 || Bmp2.GetDepth()==24); int i,nBytes; nBytes = Bmp1.GetWidth() * Bmp1.GetHeight(); wxImage Img1( Bmp1.ConvertToImage()); wxImage Img2( Bmp2.ConvertToImage()); // unsigned char *src = Img1.GetData(); unsigned char *mk = Img2.GetData(); unsigned char *alpha = (unsigned char*)malloc( nBytes ); // Extract alpha channel from second XPM. for(i=0;i myHeight )||(mFlags != mOldFlags )) { SetNewGroup( ((mFlags & resFlagPaired)!=0) ? 2 : 1 ); myHeight = ySize; } iImageGroupIndex++; if( iImageGroupIndex == iImageGroupSize ) { iImageGroupIndex = 0; mxPos += mComponentWidth; } if(mxPos > (mxCacheWidth - xSize )) { SetNewGroup(iImageGroupSize); iImageGroupIndex++; myHeight = ySize; } myPos = myPosBase + iImageGroupIndex * myHeight; mComponentWidth = xSize; mComponentHeight = ySize; } wxRect FlowPacker::Rect() { return wxRect( mxPos, myPos, mComponentWidth, mComponentHeight); } void FlowPacker::RectMid( int &x, int &y ) { x = mxPos + mComponentWidth/2; y = myPos + mComponentHeight/2; } /// \brief Helper class based on wxOutputStream used to get a png file in text format /// /// The trick used here is that wxWidgets can write a PNG image to a stream. /// By writing to a custom stream, we get to see each byte of data in turn, convert /// it to text, put in commas, and then write that out to our own text stream. class SourceOutputStream : public wxOutputStream { public: SourceOutputStream(){;}; int OpenFile(const wxString & Filename); virtual ~SourceOutputStream(); protected: virtual size_t OnSysWrite(const void *buffer, size_t bufsize); wxFile File; int nBytes; }; /// Opens the file and also adds a standard comment at the start of it. int SourceOutputStream::OpenFile(const wxString & Filename) { nBytes = 0; bool bOk; bOk = File.Open( Filename, wxFile::write ); if( bOk ) { File.Write( wxT("// ThemeAsCeeCode.h\r\n") ); File.Write( wxT("//\r\n") ); File.Write( wxT("// This file was Auto-Generated.\r\n") ); File.Write( wxT("// It is included by Theme.cpp.\r\n") ); File.Write( wxT("// Only check this into SVN if you've read and understood the guidelines!\r\n\r\n ") ); } return bOk; } /// This is the 'callback' function called with each write of PNG data /// to the stream. This is where we conveet to text and add commas. size_t SourceOutputStream::OnSysWrite(const void *buffer, size_t bufsize) { wxString Temp; for(int i=0;i<(int)bufsize;i++) { // Write one byte with a comma Temp = wxString::Format( wxT("%i,"),(int)(((unsigned char*)buffer)[i]) ); File.Write( Temp ); nBytes++; // New line if more than 20 bytes written since last time. if( (nBytes %20)==0 ) { File.Write( wxT("\r\n ")); } } return bufsize; } /// Destructor. We close our text stream in here. SourceOutputStream::~SourceOutputStream() { File.Write( wxT("\r\n") ); File.Close(); } // Must be wide enough for AudacityLogoWithName. Use double width + 10. //vvv const int ImageCacheWidth = (2 * LOGOWITHNAME_WIDTH) + 10; const int ImageCacheWidth = 440;// Must be wide enough for AudacityLogo which is 215, use double width. const int ImageCacheHeight = 836; void ThemeBase::CreateImageCache( bool bBinarySave ) { EnsureInitialised(); wxBusyCursor busy; wxImage ImageCache( ImageCacheWidth, ImageCacheHeight ); ImageCache.SetRGB( wxRect( 0,0,ImageCacheWidth, ImageCacheHeight), 1,1,1);//Not-quite black. // Ensure we have an alpha channel... if( !ImageCache.HasAlpha() ) { ImageCache.InitAlpha(); } int i; mFlow.Init( ImageCacheWidth ); //#define IMAGE_MAP #ifdef IMAGE_MAP wxLogDebug( wxT("" )); wxLogDebug( wxT("") ); #endif // Save the bitmaps for(i=0;i<(int)mImages.GetCount();i++) { wxImage &SrcImage = mImages[i]; mFlow.mFlags = mBitmapFlags[i]; if( (mBitmapFlags[i] & resFlagInternal)==0) { mFlow.GetNextPosition( SrcImage.GetWidth(), SrcImage.GetHeight()); PasteSubImage( &ImageCache, &SrcImage, mFlow.mxPos, mFlow.myPos ); #ifdef IMAGE_MAP // No href in html. Uses title not alt. wxRect R( mFlow.Rect() ); wxLogDebug( wxT(""), mBitmapNames[i].c_str(), R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom() ); #endif } } // Now save the colours. int x,y; mFlow.SetNewGroup(1); const int iColSize=10; mFlow.myHeight = iColSize+1; for(i=0;i<(int)mColours.GetCount();i++) { mFlow.GetNextPosition( iColSize, iColSize ); wxColour c = mColours[i]; ImageCache.SetRGB( mFlow.Rect(), c.Red(), c.Green(), c.Blue() ); // YUCK! No function in wxWidgets to set a rectangle of alpha... for(x=0;x"), mColourNames[i].c_str(), R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom() ); #endif } #ifdef IMAGE_MAP wxLogDebug( "" ); #endif // IF nBinarySave, THEN saving to a normal PNG file. if( bBinarySave ) { wxString FileName = FileNames::ThemeCachePng(); // Perhaps we should prompt the user if they are overwriting // an existing theme cache? #if 0 if( wxFileExist( FileName )) { wxMessageBox( wxString::Format( wxT("Theme cache file:\n %s\nalready exists.\nAre you sure you want to replace it?"), FileName.c_str() )); return; } #endif if( !ImageCache.SaveFile( FileName, wxBITMAP_TYPE_PNG )) { wxMessageBox( wxString::Format( _("Audacity could not write file:\n %s."), FileName.c_str() )); return; } wxMessageBox( wxString::Format( wxT("Theme written to:\n %s."), FileName.c_str() )); } // ELSE saving to a C code textual version. else { SourceOutputStream OutStream; wxString FileName = FileNames::ThemeCacheAsCee( ); if( !OutStream.OpenFile( FileName )) { wxMessageBox( wxString::Format( _("Audacity could not open file:\n %s\nfor writing."), FileName.c_str() )); return; } if( !ImageCache.SaveFile(OutStream, wxBITMAP_TYPE_PNG ) ) { wxMessageBox( wxString::Format( _("Audacity could not write images to file:\n %s."), FileName.c_str() )); return; } wxMessageBox( wxString::Format( wxT("Theme as Cee code written to:\n %s."), FileName.c_str() )); } } /// Writes an html file with an image map of the ImageCache /// Very handy for seeing what each part is for. void ThemeBase::WriteImageMap( ) { EnsureInitialised(); wxBusyCursor busy; int i; mFlow.Init( ImageCacheWidth ); wxFFile File( FileNames::ThemeCacheHtm(), wxT("wb") );// I'll put in new lines explicitly. if( !File.IsOpened() ) return; File.Write( wxT("\r\n")); File.Write( wxT("\r\n")); File.Write( wxT("\r\n" )); File.Write( wxT("\r\n") ); for(i=0;i<(int)mImages.GetCount();i++) { wxImage &SrcImage = mImages[i]; mFlow.mFlags = mBitmapFlags[i]; if( (mBitmapFlags[i] & resFlagInternal)==0) { mFlow.GetNextPosition( SrcImage.GetWidth(), SrcImage.GetHeight()); // No href in html. Uses title not alt. wxRect R( mFlow.Rect() ); File.Write( wxString::Format( wxT("\r\n"), mBitmapNames[i].c_str(), R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom()) ); } } // Now save the colours. mFlow.SetNewGroup(1); const int iColSize=10; for(i=0;i<(int)mColours.GetCount();i++) { mFlow.GetNextPosition( iColSize, iColSize ); // No href in html. Uses title not alt. wxRect R( mFlow.Rect() ); File.Write( wxString::Format( wxT("\r\n"), mColourNames[i].c_str(), R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom()) ); } File.Write( wxT("\r\n") ); File.Write( wxT("\r\n")); File.Write( wxT("\r\n")); // File will be closed automatically. } /// Writes a series of Macro definitions that can be used in the include file. void ThemeBase::WriteImageDefs( ) { EnsureInitialised(); wxBusyCursor busy; int i; wxFFile File( FileNames::ThemeImageDefsAsCee(), wxT("wb") ); if( !File.IsOpened() ) return; teResourceFlags PrevFlags = (teResourceFlags)-1; for(i=0;i<(int)mImages.GetCount();i++) { wxImage &SrcImage = mImages[i]; // No href in html. Uses title not alt. if( PrevFlags != mBitmapFlags[i] ) { PrevFlags = (teResourceFlags)mBitmapFlags[i]; int t = (int)PrevFlags; wxString Temp; if( t==0 ) Temp = wxT(" resFlagNone "); if( t & resFlagPaired ) Temp += wxT(" resFlagPaired "); if( t & resFlagCursor ) Temp += wxT(" resFlagCursor "); if( t & resFlagNewLine ) Temp += wxT(" resFlagNewLine "); if( t & resFlagInternal ) Temp += wxT(" resFlagInternal "); Temp.Replace( wxT(" "), wxT(" | ") ); File.Write( wxString::Format( wxT("\r\n SET_THEME_FLAGS( %s );\r\n"), Temp.c_str() )); } File.Write( wxString::Format( wxT(" DEFINE_IMAGE( bmp%s, wxImage( %i, %i ), wxT(\"%s\"));\r\n"), mBitmapNames[i].c_str(), SrcImage.GetWidth(), SrcImage.GetHeight(), mBitmapNames[i].c_str() )); } } /// Reads an image cache including images, cursors and colours. /// @param bBinaryRead if true means read from an external binary file. /// otherwise the data is taken from a compiled in block of memory. /// @param bOkIfNotFound if true means do not report absent file. /// @return true iff we loaded the images. bool ThemeBase::ReadImageCache( bool bBinaryRead, bool bOkIfNotFound) { EnsureInitialised(); wxImage ImageCache; wxBusyCursor busy; // Ensure we have an alpha channel... // if( !ImageCache.HasAlpha() ) // { // ImageCache.InitAlpha(); // } // IF bBinary read THEN a normal read from a PNG file if( bBinaryRead ) { wxString FileName = FileNames::ThemeCachePng(); if( !wxFileExists( FileName )) { if( bOkIfNotFound ) return false; // did not load the images, so return false. wxMessageBox( wxString::Format( _("Audacity could not find file:\n %s.\nTheme not loaded."), FileName.c_str() )); return false; } if( !ImageCache.LoadFile( FileName, wxBITMAP_TYPE_PNG )) { wxMessageBox( wxString::Format( _("Audacity could not load file:\n %s.\nBad png format perhaps?"), FileName.c_str() )); return false; } } // ELSE we are reading from internal storage. else { wxMemoryInputStream InternalStream( (char *)ImageCacheAsData, sizeof(ImageCacheAsData)); if( !ImageCache.LoadFile( InternalStream, wxBITMAP_TYPE_PNG )) { // If we get this message, it means that the data in file // was not a valid png image. // Most likely someone edited it by mistake, // Or some experiment is being tried with new formats for it. wxMessageBox(_("Audacity could not read its default theme.\nPlease report the problem.")); return false; } } int i; mFlow.Init(ImageCacheWidth); // Load the bitmaps for(i=0;i<(int)mImages.GetCount();i++) { wxImage &Image = mImages[i]; mFlow.mFlags = mBitmapFlags[i]; if( (mBitmapFlags[i] & resFlagInternal)==0) { mFlow.GetNextPosition( Image.GetWidth(),Image.GetHeight() ); // wxLogDebug(wxT("Copy at %i %i (%i,%i)"), mxPos, myPos, xWidth1, yHeight1 ); Image = GetSubImageWithAlpha( ImageCache, mFlow.Rect()); mBitmaps[i] = wxBitmap(Image); } } // return true; //To not load colours.. // Now load the colours. int x,y; mFlow.SetNewGroup(1); wxColour TempColour; const int iColSize=10; mFlow.myHeight = iColSize+1; for(i=0;i<(int)mColours.GetCount();i++) { mFlow.GetNextPosition( iColSize, iColSize ); mFlow.RectMid( x, y ); // Only change the colour if the alpha is opaque. // This allows us to add new colours more easily. if( ImageCache.GetAlpha(x,y ) > 128 ) { TempColour = wxColour( ImageCache.GetRed( x,y), ImageCache.GetGreen( x,y), ImageCache.GetBlue(x,y)); /// \todo revisit this hack which makes adding new colours easier /// but which prevents a colour of (1,1,1) from being added. /// find an alternative way to make adding new colours easier. /// e.g. initialise the background to translucent, perhaps. if( TempColour != wxColour(1,1,1) ) mColours[i] = TempColour; } } return true; } void ThemeBase::LoadComponents( bool bOkIfNotFound ) { // IF directory doesn't exist THEN return early. if( !wxDirExists( FileNames::ThemeComponentsDir() )) return; wxBusyCursor busy; int i; int n=0; wxString FileName; for(i=0;i<(int)mImages.GetCount();i++) { if( (mBitmapFlags[i] & resFlagInternal)==0) { FileName = FileNames::ThemeComponent( mBitmapNames[i] ); if( wxFileExists( FileName )) { if( !mImages[i].LoadFile( FileName, wxBITMAP_TYPE_PNG )) { wxMessageBox( wxString::Format( _("Audacity could not load file:\n %s.\nBad png format perhaps?"), FileName.c_str() )); return; } /// JKC: \bug (wxWidgets) A png may have been saved with alpha, but when you /// load it, it comes back with a mask instead! (well I guess it is more /// efficient). Anyway, we want alpha and not a mask, so we call InitAlpha, /// and that transfers the mask into the alpha channel, and we're done. if( ! mImages[i].HasAlpha() ) { // wxLogDebug( wxT("File %s lacked alpha"), mBitmapNames[i].c_str() ); mImages[i].InitAlpha(); } mBitmaps[i] = wxBitmap( mImages[i] ); n++; } } } if( n==0 ) { if( bOkIfNotFound ) return; wxMessageBox(wxString::Format(_("None of the expected theme component files\n were found in:\n %s."), FileNames::ThemeComponentsDir().c_str())); } } void ThemeBase::SaveComponents() { // IF directory doesn't exist THEN create it if( !wxDirExists( FileNames::ThemeComponentsDir() )) { /// \bug 1 in wxWidgets documentation; wxMkDir returns false if /// directory didn't exist, even if it successfully creates it. /// so we create and then test if it exists instead. /// \bug 2 in wxWidgets documentation; wxMkDir has only one argument /// under MSW #ifdef __WXMSW__ wxMkDir( FileNames::ThemeComponentsDir().fn_str() ); #else wxMkDir( FileNames::ThemeComponentsDir().fn_str(), 0700 ); #endif if( !wxDirExists( FileNames::ThemeComponentsDir() )) { wxMessageBox( wxString::Format( _("Could not create directory:\n %s"), FileNames::ThemeComponentsDir().c_str() )); return; } } wxBusyCursor busy; int i; int n=0; wxString FileName; for(i=0;i<(int)mImages.GetCount();i++) { if( (mBitmapFlags[i] & resFlagInternal)==0) { FileName = FileNames::ThemeComponent( mBitmapNames[i] ); if( !wxFileExists( FileName )) { if( !mImages[i].SaveFile( FileName, wxBITMAP_TYPE_PNG )) { wxMessageBox( wxString::Format( _("Audacity could not save file:\n %s"), FileName.c_str() )); return; } n++; } } } if( n==0 ) { wxMessageBox( wxString::Format( _("All required files in:\n %s\nwere already present."), FileNames::ThemeComponentsDir().c_str() )); return; } wxMessageBox( wxString::Format( wxT("Theme written to:\n %s."), FileNames::ThemeComponentsDir().c_str() )); } void ThemeBase::ReadThemeInternal() { // false indicates not using standard binary method. ReadImageCache( false ); } void ThemeBase::SaveThemeAsCode() { // false indicates not using standard binary method. CreateImageCache( false ); } wxImage ThemeBase::MakeImageWithAlpha( wxBitmap & Bmp ) { // BUG in wxWidgets. Conversion from BMP to image does not preserve alpha. wxImage image( Bmp.ConvertToImage() ); return image; } wxColour & ThemeBase::Colour( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); return mColours[iIndex]; } void ThemeBase::SetBrushColour( wxBrush & Brush, int iIndex ) { wxASSERT( iIndex >= 0 ); Brush.SetColour( Colour( iIndex )); } void ThemeBase::SetPenColour( wxPen & Pen, int iIndex ) { wxASSERT( iIndex >= 0 ); Pen.SetColour( Colour( iIndex )); } wxBitmap & ThemeBase::Bitmap( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); return mBitmaps[iIndex]; } wxImage & ThemeBase::Image( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); return mImages[iIndex]; } wxSize ThemeBase::ImageSize( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); wxImage & Image = mImages[iIndex]; return wxSize( Image.GetWidth(), Image.GetHeight()); } wxCursor & ThemeBase::Cursor( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); return *(wxCursor*)NULL; } wxFont & ThemeBase::Font( int iIndex ) { wxASSERT( iIndex >= 0 ); EnsureInitialised(); return *(wxFont*)NULL; } /// Replaces both the image and the bitmap. void ThemeBase::ReplaceImage( int iIndex, wxImage * pImage ) { Image( iIndex ) = *pImage; Bitmap( iIndex ) = wxBitmap( *pImage ); }