From: Vadim Zeitlin <vadim@wxwidgets.org> Date: Fri, 29 Jul 2005 13:19:32 +0000 (+0000) Subject: refactoring in preparation for further changes: moved data in a private struct, let... X-Git-Url: https://git.saurik.com/wxWidgets.git/commitdiff_plain/ddc5c471d4d8d185e9f2bb9c91c92ddd3a2afff8 refactoring in preparation for further changes: moved data in a private struct, let Windows load resources instead of doing it ourselves, use GlobalPtr[Lock] classes git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@34987 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775 --- diff --git a/include/wx/msw/sound.h b/include/wx/msw/sound.h index 2e7e5ff48e..d0a69a5d15 100644 --- a/include/wx/msw/sound.h +++ b/include/wx/msw/sound.h @@ -1,5 +1,5 @@ ///////////////////////////////////////////////////////////////////////////// -// Name: sound.h +// Name: wx/msw/sound.h // Purpose: wxSound class // Author: Julian Smart // Modified by: @@ -18,38 +18,39 @@ #if wxUSE_SOUND -#include "wx/object.h" - class WXDLLIMPEXP_ADV wxSound : public wxSoundBase { public: - wxSound(); - wxSound(const wxString& fileName, bool isResource = false); - wxSound(int size, const wxByte* data); - ~wxSound(); + wxSound(); + wxSound(const wxString& fileName, bool isResource = false); + wxSound(int size, const wxByte* data); + virtual ~wxSound(); -public: - // Create from resource or file - bool Create(const wxString& fileName, bool isResource = false); - // Create from data - bool Create(int size, const wxByte* data); + // Create from resource or file + bool Create(const wxString& fileName, bool isResource = false); - bool IsOk() const { return (m_waveData ? true : false); }; + // Create from data + bool Create(int size, const wxByte* data); - static void Stop(); + bool IsOk() const { return m_data != NULL; } + + static void Stop(); protected: - bool Free(); + void Init() { m_data = NULL; } + bool CheckCreatedOk(); + void Free(); - bool DoPlay(unsigned flags) const; + virtual bool DoPlay(unsigned flags) const; private: - wxByte* m_waveData; - int m_waveLength; - bool m_isResource; + // data of this object + class wxSoundData *m_data; DECLARE_NO_COPY_CLASS(wxSound) }; -#endif -#endif + +#endif // wxUSE_SOUND + +#endif // _WX_SOUND_H_ diff --git a/src/msw/sound.cpp b/src/msw/sound.cpp index 9f430540c2..2cf3619b60 100644 --- a/src/msw/sound.cpp +++ b/src/msw/sound.cpp @@ -2,161 +2,212 @@ // Name: sound.cpp // Purpose: wxSound // Author: Julian Smart -// Modified by: +// Modified by: 2005-07-29: Vadim Zeitlin: redesign // Created: 04/01/98 // RCS-ID: $Id$ // Copyright: (c) Julian Smart // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// +// ============================================================================ +// declarations +// ============================================================================ + +// ---------------------------------------------------------------------------- +// headers +// ---------------------------------------------------------------------------- + #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) -#pragma implementation "sound.h" + #pragma implementation "sound.h" #endif // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #if defined(__BORLANDC__) -#pragma hdrstop + #pragma hdrstop #endif #if wxUSE_SOUND -#ifndef WX_PRECOMP -#include "wx/wx.h" -#endif - -#include "wx/file.h" #include "wx/sound.h" #include "wx/msw/private.h" -#include <windowsx.h> - #include <mmsystem.h> -wxSound::wxSound() - : m_waveData(NULL), m_waveLength(0), m_isResource(false) +// ---------------------------------------------------------------------------- +// wxSoundData +// ---------------------------------------------------------------------------- + +// ABC for different sound data representations +class wxSoundData { -} +public: + wxSoundData() { } -wxSound::wxSound(const wxString& sFileName, bool isResource) - : m_waveData(NULL), m_waveLength(0), m_isResource(isResource) + // return true if we had been successfully initialized + virtual bool IsOk() const = 0; + + // get the flag corresponding to our content for PlaySound() + virtual DWORD GetSoundFlag() const = 0; + + // get the data to be passed to PlaySound() + virtual LPCTSTR GetSoundData() const = 0; + + virtual ~wxSoundData() { } +}; + +// class for in-memory sound data +class wxSoundDataMemory : public wxSoundData { - Create(sFileName, isResource); -} +public: + // we copy the data + wxSoundDataMemory(int size, const wxByte *buf); + + void *GetPtr() const { return m_waveDataPtr; } + + virtual bool IsOk() const { return GetPtr() != NULL; } + virtual DWORD GetSoundFlag() const { return SND_MEMORY; } + virtual LPCTSTR GetSoundData() const { return (LPCTSTR)GetPtr(); } -wxSound::wxSound(int size, const wxByte* data) - : m_waveData(NULL), m_waveLength(0), m_isResource(false) +private: + GlobalPtr m_waveData; + GlobalPtrLock m_waveDataPtr; + + DECLARE_NO_COPY_CLASS(wxSoundDataMemory) +}; + +// class for sound files and resources +class wxSoundDataFile : public wxSoundData +{ +public: + wxSoundDataFile(const wxString& filename, bool isResource); + + virtual bool IsOk() const { return !m_name.empty(); } + virtual DWORD GetSoundFlag() const + { + return m_isResource ? SND_RESOURCE : SND_FILENAME; + } + virtual LPCTSTR GetSoundData() const { return m_name.c_str(); } + +private: + const wxString m_name; + const bool m_isResource; + + DECLARE_NO_COPY_CLASS(wxSoundDataFile) +}; + +// ============================================================================ +// implementation +// ============================================================================ + +// ---------------------------------------------------------------------------- +// wxSoundData-derived classes +// ---------------------------------------------------------------------------- + +wxSoundDataMemory::wxSoundDataMemory(int size, const wxByte *buf) + : m_waveData(size), + m_waveDataPtr(m_waveData) { - Create(size, data); + if ( IsOk() ) + ::CopyMemory(m_waveDataPtr, buf, size); } -wxSound::~wxSound() +wxSoundDataFile::wxSoundDataFile(const wxString& filename, bool isResource) + : m_name(filename), + m_isResource(isResource) { - Free(); + // check for file/resource existence? } -bool wxSound::Create(const wxString& fileName, bool isResource) +// ---------------------------------------------------------------------------- +// wxSound +// ---------------------------------------------------------------------------- + +wxSound::wxSound() { - Free(); + Init(); +} - if (isResource) - { - m_isResource = true; +wxSound::wxSound(const wxString& filename, bool isResource) +{ + Init(); + Create(filename, isResource); +} - HRSRC hresInfo; - hresInfo = ::FindResource((HMODULE) wxhInstance, fileName, wxT("WAVE")); - if (!hresInfo) - return false; +wxSound::wxSound(int size, const wxByte *data) +{ + Init(); + Create(size, data); +} - HGLOBAL waveData = ::LoadResource((HMODULE) wxhInstance, hresInfo); +wxSound::~wxSound() +{ + Free(); +} - if (waveData) +void wxSound::Free() +{ + if ( m_data ) { - m_waveData= (wxByte*)::LockResource(waveData); - m_waveLength = (int) ::SizeofResource((HMODULE) wxhInstance, hresInfo); + delete m_data; + m_data = NULL; } +} - return (m_waveData ? true : false); - } - else - { - m_isResource = false; - - wxFile fileWave; - if (!fileWave.Open(fileName, wxFile::read)) - return false; +bool wxSound::CheckCreatedOk() +{ + if ( m_data && !m_data->IsOk() ) + Free(); - m_waveLength = (int) fileWave.Length(); + return m_data != NULL; +} - m_waveData = (wxByte*)GlobalLock(GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE, m_waveLength)); - if (!m_waveData) - return false; +bool wxSound::Create(const wxString& filename, bool isResource) +{ + Free(); - fileWave.Read(m_waveData, m_waveLength); + m_data = new wxSoundDataFile(filename, isResource); - return true; - } + return CheckCreatedOk(); } bool wxSound::Create(int size, const wxByte* data) { - Free(); - m_isResource = true; - m_waveLength=size; - m_waveData = (wxByte*)GlobalLock(GlobalAlloc(GMEM_MOVEABLE | GMEM_SHARE, m_waveLength)); - if (!m_waveData) - return false; - - for (int i=0; i<size; i++) m_waveData[i] = data[i]; - return true; + Free(); + + m_data = new wxSoundDataMemory(size, data); + + return CheckCreatedOk(); } bool wxSound::DoPlay(unsigned flags) const { - if (!IsOk()) - return false; - - return (::PlaySound((LPCTSTR)m_waveData, NULL, - SND_MEMORY | SND_NODEFAULT | - ((flags & wxSOUND_ASYNC) ? SND_ASYNC : SND_SYNC) | - ((flags & wxSOUND_LOOP) ? (SND_LOOP | SND_ASYNC) : 0)) - != 0); -} + if ( !IsOk() || !m_data->IsOk() ) + return false; -bool wxSound::Free() -{ - if (m_waveData) - { -#ifdef __WXWINCE__ - HGLOBAL waveData = (HGLOBAL) m_waveData; -#else - HGLOBAL waveData = GlobalHandle(m_waveData); -#endif + DWORD flagsMSW = m_data->GetSoundFlag(); + HMODULE hmod = flagsMSW == SND_RESOURCE ? wxGetInstance() : NULL; - if (waveData) - { -#ifndef __WXWINCE__ - if (m_isResource) - ::FreeResource(waveData); - else -#endif - { - GlobalUnlock(waveData); - GlobalFree(waveData); - } - - m_waveData = NULL; - m_waveLength = 0; - return true; - } + // we don't want replacement default sound + flagsMSW |= SND_NODEFAULT; + + // NB: wxSOUND_SYNC is 0, don't test for it + flagsMSW |= (flags & wxSOUND_ASYNC) ? SND_ASYNC : SND_SYNC; + if ( flags & wxSOUND_LOOP ) + { + // looping only works with async flag + flagsMSW |= SND_LOOP | SND_ASYNC; } - return false; + + return ::PlaySound(m_data->GetSoundData(), hmod, flagsMSW) != FALSE; } -/*static*/ void wxSound::Stop() +/* static */ +void wxSound::Stop() { ::PlaySound(NULL, NULL, 0); } #endif // wxUSE_SOUND +