mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-25 07:43:54 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			1077 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1077 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /**********************************************************************
 | |
| 
 | |
|   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 <wx/wxprec.h>
 | |
| #include <wx/image.h>
 | |
| #include <wx/file.h>
 | |
| #include <wx/ffile.h>
 | |
| #include <wx/mstream.h>
 | |
| #include <wx/msgdlg.h>
 | |
| 
 | |
| #include "Project.h"
 | |
| #include "toolbars/ToolBar.h"
 | |
| #include "toolbars/ToolManager.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/arrimpl.cpp>
 | |
| 
 | |
| 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...
 | |
| static 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();
 | |
|    for( int ii = 0; ii < ToolBarCount; ++ii )
 | |
|    {
 | |
|       ToolBar *pToolBar = p->GetToolManager()->GetToolBar(ii);
 | |
|       if( pToolBar )
 | |
|          pToolBar->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<nBytes;i++)
 | |
|    {
 | |
|       alpha[i] = mk[0];
 | |
|       mk+=3;
 | |
|    }
 | |
| 
 | |
|    Img1.SetAlpha( alpha);
 | |
| 
 | |
|    //dmazzoni: the top line does not work on wxGTK
 | |
|    //wxBitmap Result( Img1, 32 );
 | |
|    //wxBitmap Result( Img1 );
 | |
| 
 | |
|    return Img1;
 | |
| }
 | |
| 
 | |
| void ThemeBase::RegisterImage( int &iIndex, char const ** pXpm, const wxString & Name )
 | |
| {
 | |
| 
 | |
|    wxASSERT( iIndex == -1 ); // Don't initialise same bitmap twice!
 | |
|    wxBitmap Bmp( pXpm ); // a 24 bit bitmap.
 | |
|    wxImage Img( Bmp.ConvertToImage() );
 | |
|    Img.InitAlpha();
 | |
| 
 | |
|    //dmazzoni: the top line does not work on wxGTK
 | |
|    //wxBitmap Bmp2( Img, 32 );
 | |
|    //wxBitmap Bmp2( Img );
 | |
| 
 | |
|    RegisterImage( iIndex, Img, Name );
 | |
| }
 | |
| 
 | |
| void ThemeBase::RegisterImage( int &iIndex, const wxImage &Image, const wxString & Name )
 | |
| {
 | |
|    wxASSERT( iIndex == -1 ); // Don't initialise same bitmap twice!
 | |
|    mImages.Add( Image );
 | |
| 
 | |
| #ifdef __APPLE__
 | |
|    // On Mac, bitmaps with alpha don't work.
 | |
|    // So we convert to a mask and use that.
 | |
|    // It isn't quite as good, as alpha gives smoother edges.
 | |
|    //[Does not affect the large control buttons, as for those we do
 | |
|    // the blending ourselves anyway.]
 | |
|    wxImage TempImage( Image );
 | |
|    TempImage.ConvertAlphaToMask();
 | |
|    mBitmaps.Add( wxBitmap( TempImage ) );
 | |
| #else
 | |
|    mBitmaps.Add( wxBitmap( Image ) );
 | |
| #endif
 | |
| 
 | |
|    mBitmapNames.Add( Name );
 | |
|    mBitmapFlags.Add( mFlow.mFlags );
 | |
|    iIndex = mBitmaps.GetCount()-1;
 | |
| }
 | |
| 
 | |
| void ThemeBase::RegisterColour( int &iIndex, const wxColour &Clr, const wxString & Name )
 | |
| {
 | |
|    wxASSERT( iIndex == -1 ); // Don't initialise same colour twice!
 | |
|    mColours.Add( Clr );
 | |
|    mColourNames.Add( Name );
 | |
|    iIndex = mColours.GetCount()-1;
 | |
| }
 | |
| 
 | |
| void FlowPacker::Init(int width)
 | |
| {
 | |
|    mFlags = resFlagPaired;
 | |
|    mOldFlags = mFlags;
 | |
|    mxCacheWidth = width;
 | |
| 
 | |
|    myPos = 0;
 | |
|    myPosBase =0;
 | |
|    myHeight = 0;
 | |
|    iImageGroupSize = 1;
 | |
|    SetNewGroup(1);
 | |
| }
 | |
| 
 | |
| void FlowPacker::SetNewGroup( int iGroupSize )
 | |
| {
 | |
|    myPosBase +=myHeight * iImageGroupSize;
 | |
|    mxPos =0;
 | |
|    mOldFlags = mFlags;
 | |
|    iImageGroupSize = iGroupSize;
 | |
|    iImageGroupIndex = -1;
 | |
|    mComponentWidth=0;
 | |
| }
 | |
| 
 | |
| void FlowPacker::GetNextPosition( int xSize, int ySize )
 | |
| {
 | |
|    // if the height has increased, then we are on a NEW group.
 | |
|    if(( ySize > 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 final : public wxOutputStream
 | |
| {
 | |
| public:
 | |
|    SourceOutputStream(){;};
 | |
|    int OpenFile(const wxString & Filename);
 | |
|    virtual ~SourceOutputStream();
 | |
| 
 | |
| protected:
 | |
|    size_t OnSysWrite(const void *buffer, size_t bufsize) override;
 | |
|    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 bmpAudacityLogo. Use double width + 10.
 | |
| const int ImageCacheWidth = 440;
 | |
| 
 | |
| 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("<img src=\"ImageCache.png\" usemap=\"#map1\">" ));
 | |
|    wxLogDebug( wxT("<map name=\"map1\">") );
 | |
| #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("<area title=\"Bitmap:%s\" shape=rect coords=\"%i,%i,%i,%i\">"),
 | |
|             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<iColSize;x++)
 | |
|       {
 | |
|          for(y=0;y<iColSize;y++)
 | |
|          {
 | |
|             ImageCache.SetAlpha( mFlow.mxPos + x, mFlow.myPos+y, 255);
 | |
|          }
 | |
|       }
 | |
| #ifdef IMAGE_MAP
 | |
|       // No href in html.  Uses title not alt.
 | |
|       wxRect R( mFlow.Rect() );
 | |
|       wxLogDebug( wxT("<area title=\"Colour:%s\" shape=rect coords=\"%i,%i,%i,%i\">"),
 | |
|          mColourNames[i].c_str(),
 | |
|          R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom() );
 | |
| #endif
 | |
|    }
 | |
| 
 | |
| #ifdef IMAGE_MAP
 | |
|    wxLogDebug( "</map>" );
 | |
| #endif
 | |
| 
 | |
|    // IF nBinarySave, THEN saving to a normal PNG file.
 | |
|    if( bBinarySave )
 | |
|    {
 | |
|       const 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;
 | |
|       const 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("<html>\r\n"));
 | |
|    File.Write( wxT("<body>\r\n"));
 | |
|    File.Write( wxT("<img src=\"ImageCache.png\" usemap=\"#map1\">\r\n" ));
 | |
|    File.Write( wxT("<map name=\"map1\">\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("<area title=\"Bitmap:%s\" shape=rect coords=\"%i,%i,%i,%i\">\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("<area title=\"Colour:%s\" shape=rect coords=\"%i,%i,%i,%i\">\r\n"),
 | |
|          mColourNames[i].c_str(),
 | |
|          R.GetLeft(), R.GetTop(), R.GetRight(), R.GetBottom()) );
 | |
|    }
 | |
|    File.Write( wxT("</map>\r\n") );
 | |
|    File.Write( wxT("</body>\r\n"));
 | |
|    File.Write( wxT("</html>\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 )
 | |
|    {
 | |
|       const 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 ))
 | |
|       {
 | |
|          /* i18n-hint: Do not translate png.  It is the name of a file format.*/
 | |
|          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 ))
 | |
|             {
 | |
|                /* i18n-hint: Do not translate png.  It is the name of a file format.*/
 | |
|                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());
 | |
| }
 | |
| 
 | |
| // The next two functions are for future use.
 | |
| #if 0
 | |
| wxCursor & ThemeBase::Cursor( int iIndex )
 | |
| {
 | |
|    wxASSERT( iIndex >= 0 );
 | |
|    EnsureInitialised();
 | |
|    // Purposeful null deref.  Function is for future use.
 | |
|    // If anyone tries to use it now they will get an error.
 | |
|    return *(wxCursor*)NULL;
 | |
| }
 | |
| 
 | |
| wxFont   & ThemeBase::Font( int iIndex )
 | |
| {
 | |
|    wxASSERT( iIndex >= 0 );
 | |
|    EnsureInitialised();
 | |
|    // Purposeful null deref.  Function is for future use.
 | |
|    // If anyone tries to use it now they will get an error.
 | |
|    return *(wxFont*)NULL;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /// Replaces both the image and the bitmap.
 | |
| void ThemeBase::ReplaceImage( int iIndex, wxImage * pImage )
 | |
| {
 | |
|    Image( iIndex ) = *pImage;
 | |
|    Bitmap( iIndex ) = wxBitmap( *pImage );
 | |
| }
 | |
| 
 | |
| 
 |