// SoundPage.cpp : implementation file
//

#include "stdafx.h"
#include "source_sounds_builder.h"
#include "source_sounds_builderDlg.h"
#include "SoundPage.h"

#include "../src/sound/driver/buffer.h"
#include "../src/sound/audio_mixer_user.h"

#include "nel/sound/u_listener.h"

#include <nel/misc/common.h>
#include <nel/misc/vector.h>
using namespace NLMISC;

#include <math.h>

using namespace std;


#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif


#define CONE_X 340
#define CONE_Y 280
#define CONE_R 30


uint XCenter;
uint YCenter;
uint Radius;


const char *PriorityStr [NbSoundPriorities] = { "Highest", "High", "Medium", "Low" };


//----------------------------------------------------------
// Name: ConvertLinearSliderPosToLogScale()
// Desc: Converts a linear slider position to a quasi logrithmic scale
//-----------------------------------------------------------------------------
float ConvertLinearSliderPosToLogScale( LONG lSliderPos )
{
    if( lSliderPos > 0 && lSliderPos <= 10 )
    {
        return lSliderPos*0.01f;
    }
    else if( lSliderPos > 10 && lSliderPos <= 20 )
    {
        return (lSliderPos-10)*0.1f;
    }
    else if( lSliderPos > 20 && lSliderPos <= 30 )
    {
        return (lSliderPos-20)*1.0f;
    }
    else if( lSliderPos > 30 && lSliderPos <= 40 )
    {
        return (lSliderPos-30)*10.0f;
    }

    return 0.0f;
}




//-----------------------------------------------------------------------------
// Name: ConvertLinearSliderPosToLogScale()
// Desc: Converts a quasi logrithmic scale to a slider position
//-----------------------------------------------------------------------------
LONG ConvertLogScaleToLinearSliderPosTo( FLOAT fValue )
{
    if( fValue > 0.0f && fValue <= 0.1f )
    {
        return (LONG)(fValue/0.01f);
    }
    else if( fValue > 0.1f && fValue <= 1.0f )
    {
        return (LONG)(fValue/0.1f) + 10;
    }
    else if( fValue > 1.0f && fValue <= 10.0f )
    {
        return (LONG)(fValue/1.0f) + 20;
    }
    else if( fValue > 10.0f && fValue <= 100.0f )
    {
        return (LONG)(fValue/10.0f) + 30;
    }

    return 0;
}



/////////////////////////////////////////////////////////////////////////////
// CSoundPage dialog


CSoundPage::CSoundPage(CWnd* pParent /*=NULL*/)
	: CDialog(CSoundPage::IDD, pParent)
{
	//{{AFX_DATA_INIT(CSoundPage)
	m_Filename = _T("");
	m_Gain = 1.0f;
	m_Pos3D = FALSE;
	m_MinDist = 1.0f;
	m_MaxDist = 1000000.0f;
	m_InnerAngleDeg = 360;
	m_OuterAngleDeg = 360;
	m_OuterGain = 1.0f;
	m_Looped = FALSE;
	m_Stereo = _T("");
	m_Pitch = 1.0f;
	m_Looping = false;
	m_SoundName = _T("");
	//}}AFX_DATA_INIT

	_CurrentSound = NULL;
	_Tree = NULL;
	_Source = NULL;
}


/*
 *
 */
BOOL CSoundPage::OnInitDialog() 
{
	CDialog::OnInitDialog();
	
	// Load driver
	CWaitCursor waitcursor;
	_AudioMixer = UAudioMixer::createAudioMixer();
	try
	{
		_AudioMixer->init();
		//_AudioMixer->getListener()->setPos( CVector(0.0f,0.0001f,0.0f) );
	}
	catch( Exception& e )
	{
		CString s;
		s.Format( "No sound driver: %s\n\nSound playback will be unavailable" , e.what() );
		AfxMessageBox( s );
		_AudioMixer = NULL;
	}

	// Fill priorities
	uint i;
	for ( i=0; i!=NbSoundPriorities; ++i )
	{
		m_Priority.AddString( PriorityStr[i] );
	}

	// We want to load nss files even if the corresponding waves are missing
	CSound::allowMissingWave( true );

	// Cone drawing: make it work with normal and big fonts (depending on system settings)
	CRect parentrect, sliderrect;
	this->GetWindowRect( &parentrect );
	GetDlgItem( IDC_SliderOuterAngle )->GetWindowRect( &sliderrect );
	XCenter = sliderrect.right - parentrect.left + 15 + CONE_R;
	YCenter = sliderrect.top - parentrect.top + 10;
	Radius = CONE_R;

	waitcursor.Restore();

	_NameFont = new CFont();
	LOGFONT logfont;
	GetFont()->GetLogFont( &logfont );
	logfont.lfWeight = FW_BOLD;
	_NameFont->CreateFontIndirect( &logfont );
	
	GetDlgItem( IDC_SoundName )->SetFont( _NameFont );

	((CSliderCtrl*)GetDlgItem( IDC_SliderGain ))->SetRange( 0, 40 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderPitch ))->SetRange( 0, 40 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderMinDist ))->SetRange( 0, 1000 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderMaxDist ))->SetRange( 0, 1000 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderInnerAngle ))->SetRange( 0, 360 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterAngle ))->SetRange( 0, 360 );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterGain ))->SetRange( 0, 40 );

	return TRUE;  // return TRUE unless you set the focus to a control
	              // EXCEPTION: OCX Property Pages should return FALSE
}


void CSoundPage::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
	//{{AFX_DATA_MAP(CSoundPage)
	DDX_Control(pDX, IDC_CbPriority, m_Priority);
	DDX_Text(pDX, IDC_EditFilename, m_Filename);
	DDX_Text(pDX, IDC_EditGain, m_Gain);
	DDV_MinMaxFloat(pDX, m_Gain, 0.f, 1.f);
	DDX_Check(pDX, IDC_Pos3D, m_Pos3D);
	DDX_Text(pDX, IDC_EditMinDist, m_MinDist);
	DDV_MinMaxFloat(pDX, m_MinDist, 0.f, 1.e+006f);
	DDX_Text(pDX, IDC_EditMaxDist, m_MaxDist);
	DDV_MinMaxFloat(pDX, m_MaxDist, 0.f, 1.e+006f);
	DDX_Text(pDX, IDC_EditInnerAngle, m_InnerAngleDeg);
	DDV_MinMaxUInt(pDX, m_InnerAngleDeg, 0, 360);
	DDX_Text(pDX, IDC_EditOuterAngle, m_OuterAngleDeg);
	DDV_MinMaxUInt(pDX, m_OuterAngleDeg, 0, 360);
	DDX_Text(pDX, IDC_EditOuterGain, m_OuterGain);
	DDV_MinMaxFloat(pDX, m_OuterGain, 0.f, 1.f);
	DDX_Check(pDX, IDC_Looped, m_Looped);
	DDX_Text(pDX, IDC_Stereo, m_Stereo);
	DDX_Text(pDX, IDC_EditPitch, m_Pitch);
	DDV_MinMaxFloat(pDX, m_Pitch, 1.e-011f, 1.f);
	DDX_Check(pDX, IDC_Looping, m_Looping);
	DDX_Text(pDX, IDC_SoundName, m_SoundName);
	//}}AFX_DATA_MAP
}


BEGIN_MESSAGE_MAP(CSoundPage, CDialog)
	//{{AFX_MSG_MAP(CSoundPage)
	ON_BN_CLICKED(IDC_Pos3D, OnPos3D)
	ON_BN_CLICKED(IDC_ChooseFile, OnChooseFile)
	ON_BN_CLICKED(IDC_Remove, OnRemove)
	ON_BN_CLICKED(IDC_PlaySound, OnPlaySound)
	ON_WM_CLOSE()
	ON_BN_CLICKED(IDC_Looped, OnLooped)
	ON_EN_CHANGE(IDC_EditInnerAngle, OnChangeEditInnerAngle)
	ON_EN_CHANGE(IDC_EditOuterAngle, OnChangeEditOuterAngle)
	ON_WM_PAINT()
	ON_EN_CHANGE(IDC_EditOuterGain, OnChangeEditOuterGain)
	ON_WM_HSCROLL()
	ON_EN_CHANGE(IDC_EditMinDist, OnChangeEditMinDist)
	ON_EN_CHANGE(IDC_EditMaxDist, OnChangeEditMaxDist)
	ON_BN_CLICKED(IDC_ButtonHelp, OnButtonHelp)
	ON_EN_CHANGE(IDC_EditGain, OnChangeEditGain)
	ON_BN_CLICKED(IDC_ButtonTestOuterGain, OnButtonTestOuterGain)
	ON_EN_CHANGE(IDC_EditPitch, OnChangeEditPitch)
	ON_BN_CLICKED(IDC_Cancel, OnCancel)
	ON_BN_CLICKED(IDC_Home, OnHome)
	//}}AFX_MSG_MAP
END_MESSAGE_MAP()


/*
 *
 */
const char *StereoToCStr( CSound* snd, bool *st )
{
	static const char empty [] = "";
	static const char mono [] = "Mono";
	static const char stereo [] = "Stereo";
	if ( snd->getBuffer() == NULL )
	{
		*st = false;
		return empty;
	}
	else
	{
		if ( snd->getBuffer()->isStereo() )
		{
			*st = true;
			return stereo;
		}
		else
		{
			*st = false;
			return mono;
		}
	}
}


/*
 *
 */
void		CSoundPage::UpdateStereo()
{
	bool stereo;
	m_Stereo = StereoToCStr( _CurrentSound, &stereo );
	if ( stereo )
	{
		m_Pos3D = false;
	}
	GetDlgItem( IDC_Pos3D )->EnableWindow( ! stereo );
}


/*
 *
 */
void		CSoundPage::getPropertiesFromSound()
{
	m_SoundName = _CurrentSound->getName().c_str();
	m_Filename = _CurrentSound->getFilename().c_str();
	m_Gain = _CurrentSound->getGain();
	m_Pitch = _CurrentSound->getPitch();
	m_Priority.SetCurSel( _CurrentSound->getPriority() );
	m_Looping = _CurrentSound->getLooping();
	m_Pos3D = _CurrentSound->isDetailed();
	((CSliderCtrl*)GetDlgItem( IDC_SliderGain ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_Gain*100.0f ) );
	((CSliderCtrl*)GetDlgItem( IDC_SliderPitch ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_Pitch*100.0f ) );
	UpdateStereo();

	if ( m_Pos3D )
	{
		m_MinDist = _CurrentSound->getMinDistance();
		m_MaxDist = _CurrentSound->getMaxDistance();
		m_InnerAngleDeg = (uint)radToDeg( _CurrentSound->getConeInnerAngle() );
		m_OuterAngleDeg = (uint)radToDeg( _CurrentSound->getConeOuterAngle() );
		m_OuterGain = _CurrentSound->getConeOuterGain();
	}
	((CSliderCtrl*)GetDlgItem( IDC_SliderMinDist ))->SetPos( (int)m_MinDist );
	((CSliderCtrl*)GetDlgItem( IDC_SliderMaxDist ))->SetPos( (int)m_MaxDist );
	((CSliderCtrl*)GetDlgItem( IDC_SliderInnerAngle ))->SetPos( m_InnerAngleDeg );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterAngle ))->SetPos( m_OuterAngleDeg );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterGain ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_OuterGain*100.0f ) );
	UpdateData( false );

	Invalidate();
	OnPos3D(); // enable/disable 3d properties
}


/*
 *
 */
void CSoundPage::UpdateCurrentSound()
{
	CString name = ((CSource_sounds_builderDlg*)GetOwner())->SoundName( _HItem );
	if ( ! m_Pos3D )
	{
		_CurrentSound->setProperties( string(name), string(m_Filename), m_Gain, m_Pitch, (TSoundPriority)(m_Priority.GetCurSel()), m_Looping!=0, m_Pos3D!=0 );
	}
	else
	{
		_CurrentSound->setProperties( string(name), string(m_Filename), m_Gain, m_Pitch, (TSoundPriority)(m_Priority.GetCurSel()), m_Looping!=0, m_Pos3D!=0,
			m_MinDist, m_MaxDist, degToRad((float)m_InnerAngleDeg), degToRad((float)m_OuterAngleDeg), m_OuterGain );
	}
	// Argument checking is already done by the dialog wizard
}




/////////////////////////////////////////////////////////////////////////////
// CSoundPage message handlers


/*
 *
 */
void CSoundPage::setCurrentSound( CSound *sound, HTREEITEM hitem )
{
	_CurrentSound = sound;
	_HItem = hitem;
	if ( _Source != NULL )
	{
	 	_Source->stop();
	}
	m_Looped = false;
	UpdateData( false );
}


/*
 *
 */
void CSoundPage::OnPos3D() 
{
	UpdateData( true );
	GetDlgItem( IDC_Pos3DGroup )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_MinDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_MaxDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_ConeInnerAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_ConeOuterAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_ConeOuterGain )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_EditMinDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_EditMaxDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_EditInnerAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_EditOuterAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_EditOuterGain )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_SliderMinDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_SliderMaxDist )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_SliderInnerAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_SliderOuterAngle )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_SliderOuterGain )->EnableWindow( m_Pos3D );
	GetDlgItem( IDC_ButtonTestOuterGain )->EnableWindow( m_Pos3D );

	DrawCones();
}


/*
 *
 */
void CSoundPage::apply()
{
	if ( _CurrentSound )
	{
		if ( _Source != NULL )
		{
			_Source->stop();
		}

		UpdateData( true );

		UpdateCurrentSound();

		//(static_cast<CSource_sounds_builderDlg*>(GetOwner()))->setModified();

		nlassert( _Tree );

		if ( m_Filename != "" )
		{
			CString s = ((CSource_sounds_builderDlg*)GetOwner())->SoundName( _HItem ) + " (" + m_Filename + ")";
			_Tree->SetItemText( _HItem, s );
		}

		//_Tree->SelectItem( NULL );

		GetOwner()->SetFocus();
	}
}


/*
 *
 */
void CSoundPage::rename( CString s )
{
	// Quick way to do it (should be simplified)
	
	getPropertiesFromSound();
	nlassert( _Tree );
	_Tree->SetItemText( _HItem, s );
	UpdateCurrentSound();

	m_SoundName = s;
	UpdateData( false );
}


/*
 * Reverts the changes
 */
void CSoundPage::OnCancel()
{
	getPropertiesFromSound();
}


/*
 * Unselects the current sound
 */
void CSoundPage::cancel()
{
	if ( _Source != NULL )
	{
		_Source->stop();
	}

	_CurrentSound = NULL;
}


/*
 *
 */
void CSoundPage::OnHome() 
{
	nlassert( _Tree );
	_Tree->SelectItem( _Tree->GetRootItem() );
	GetOwner()->SetFocus();
}


/*
 *
 */
void CSoundPage::OnChooseFile() 
{
	// Prompt filename
	CFileDialog opendlg( true, "wav", "", OFN_OVERWRITEPROMPT, "PCM Wave files (*.wav)|*.wav||", this );
	if ( opendlg.DoModal()==IDOK )
	{
		UpdateData( true );
		m_Filename = opendlg.GetFileName();
		try 
		{
			loadSound();
		}
		catch ( Exception& e )
		{
			CString s;
			s.Format( "%s", e.what() );
			AfxMessageBox( s );
		}
		UpdateData( false );
	}
}


/*
 *
 */
bool CSoundPage::loadSound()
{
	UpdateCurrentSound();
	nlassert( _CurrentSound );
	if ( (m_Filename != "") && (_AudioMixer != NULL) )
	{
		_CurrentSound->loadBuffer( string(m_Filename) );
		UpdateStereo();
		return true;
	}
	else
	{
		return false;
	}
}


/*
 *
 */
void CSoundPage::OnRemove() 
{
	removeSound();
}


/*
 *
 */
void CSoundPage::removeSound()
{
	/*if ( AfxMessageBox( "Are you sure to remove the current sound from the list ?", MB_YESNO | MB_ICONQUESTION ) == IDYES )
	{*/
	if ( _CurrentSound )
	{
		nlassert( _Tree );
		_Tree->DeleteItem( _HItem );
	}
	/*}*/
}


/*
 *
 */
void CSoundPage::Play( bool outsidecone ) 
{
	CWaitCursor waitcursor;

	// Load sound
	try 
	{
		if ( loadSound() )
		{
			UpdateData( false );
			
			// Play source
			if ( _Source == NULL )
			{
				_Source = _AudioMixer->createSource( _CurrentSound );
			}
			else
			{
				_Source->stop();
				_Source->setSound( _CurrentSound );
			}
			_Source->setLooping( m_Looped!=0 );
			if ( outsidecone )
			{
				// The listener is just behind the source
				_Source->setPos( CVector(0.0f,0.1f,0.0f) );
				_Source->setDirection( CVector(0.0f,1.0f,0.0f) ); // directional
			}
			else
			{
				// The listener is at the source pos
				_Source->setPos( CVector(0.0f,0.0f,0.0f) );
				_Source->setDirection( CVector(0.0f,0.0f,0.0f) ); // non-directional
			}
			_Source->play();
		}

		waitcursor.Restore();
	}
	catch ( Exception& e )
	{
		waitcursor.Restore();

		CString s;
		s.Format( "%s", e.what() );
		AfxMessageBox( s );
	}
}


/*
 *
 */
void CSoundPage::OnPlaySound()
{
	UpdateData( true );
	m_Looped = false;
	Play( false );
}


/*
 *
 */
void CSoundPage::OnLooped() 
{
	UpdateData( true );
	if ( m_Looped )
	{
		Play( false );
	}
	else
	{
		if ( _Source != NULL )
		{
			_Source->stop();
		}
	}
}


/*
 *
 */
void CSoundPage::OnButtonTestOuterGain() 
{
	UpdateData( true );
	m_Looped = false;
	Play( true );
}


/*
 *
 */
void CSoundPage::OnClose() 
{
	if ( _AudioMixer != NULL )
	{
		if ( _Source != NULL )
		{
			delete _Source;
		}
		delete _AudioMixer;
	}
}


/*
 *
 */
void CSoundPage::DrawCones()
{
	if ( m_InnerAngleDeg > m_OuterAngleDeg )
	{
		GetDlgItem( IDC_BadCone )->SetWindowText( "Inner > Outer !" );
	}
	else
	{
		GetDlgItem( IDC_BadCone )->SetWindowText( "" );
	}

	CRect rect( XCenter-Radius, YCenter-Radius, XCenter+Radius+1, YCenter+Radius+1 );
	InvalidateRect( &rect, true );
}


/*
 *
 */
void CSoundPage::OnChangeEditGain() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderGain ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_Gain*100.0f ) );
}


/*
 *
 */
void CSoundPage::OnChangeEditPitch() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderPitch ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_Pitch*100.0f ) );
}


/*
 *
 */
void CSoundPage::OnChangeEditMinDist() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderMinDist ))->SetPos( (int)m_MinDist );
}


/*
 *
 */
void CSoundPage::OnChangeEditMaxDist() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderMaxDist ))->SetPos( (int)m_MaxDist );
}


/*
 *
 */
void CSoundPage::OnChangeEditInnerAngle() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderInnerAngle ))->SetPos( m_InnerAngleDeg );
	DrawCones();
}


/*
 *
 */
void CSoundPage::OnChangeEditOuterAngle() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterAngle ))->SetPos( m_OuterAngleDeg );
	DrawCones();
}


/*
 *
 */
void CSoundPage::OnChangeEditOuterGain() 
{
	UpdateData( true );
	((CSliderCtrl*)GetDlgItem( IDC_SliderOuterGain ))->SetPos( ConvertLogScaleToLinearSliderPosTo( m_OuterGain*100.0f) );
	DrawCones();
}


/*
 *
 */
void CSoundPage::OnHScroll( UINT nSBCode, UINT nPos, CScrollBar* pScrollBar )
{
	CSliderCtrl *slider = (CSliderCtrl*)pScrollBar; // MFC sucks

	nPos = slider->GetPos();
	{
		if ( slider == GetDlgItem( IDC_SliderGain ) )
		{
			m_Gain = ConvertLinearSliderPosToLogScale( nPos ) / 100.0f;
			if ( _Source != NULL )
			{
				_Source->setGain( m_Gain );
			}
		}
		else if ( slider == GetDlgItem( IDC_SliderPitch ) )
		{
			m_Pitch = ConvertLinearSliderPosToLogScale( nPos ) / 100.0f;
			if ( _Source != NULL )
			{
				_Source->setPitch( m_Pitch );
			}
		}
		else if ( slider == GetDlgItem( IDC_SliderMinDist ) )
		{
			m_MinDist = (float)nPos;
		}
		else if ( slider == GetDlgItem( IDC_SliderMaxDist ) )
		{
			m_MaxDist = (float)nPos;
		}
		else if ( slider == GetDlgItem( IDC_SliderInnerAngle ) )
		{
			m_InnerAngleDeg = nPos;
		}
		else if ( slider == GetDlgItem( IDC_SliderOuterAngle ) )
		{
			m_OuterAngleDeg = nPos;
		}
		else if ( slider == GetDlgItem( IDC_SliderOuterGain ) )
		{
			m_OuterGain = ConvertLinearSliderPosToLogScale( nPos ) / 100.0f;
		}
		UpdateData( false );
		DrawCones();
	}
}


/*
 *
 */
void CSoundPage::OnPaint() 
{
	CPaintDC dc(this); // device context for painting

	float innerangle = degToRad((float)m_InnerAngleDeg);
	float outerangle = degToRad((float)m_OuterAngleDeg);

	/*// Erase background (done by InvalidateRect())
	CRect rect( XCenter-Radius, YCenter-Radius, XCenter+Radius+1, YCenter+Radius+1 );
	CBrush brush;
	brush.CreateSolidBrush( GetSysColor( COLOR_BTNFACE ) );
	dc.FillRect( &rect, &brush );*/

	if ( m_Pos3D )
	{
		COLORREF innercolor = RGB(255,0,0);
		uint8 c = (uint)(255.0f * (1.0f - (float)(ConvertLogScaleToLinearSliderPosTo( m_OuterGain*100.0f ))/40.0f )); // linear, not logarithmic
		COLORREF outercolor = RGB(255,c,c);
		COLORREF tcolor = RGB(255,c/1.5,c/1.5); // not progressive
		if ( m_OuterGain == 0.0f ) // change white to background color
		{
			outercolor = GetSysColor( COLOR_BTNFACE );
		}

		uint dx = (uint)(Radius*sin(outerangle/2.0f));
		uint y = YCenter-(uint)(Radius*cos(outerangle/2.0f));

		// Outside
		CPen outpen( PS_SOLID, 1, outercolor );
		CBrush outbrush;
		outbrush.CreateSolidBrush( outercolor );
		dc.SelectObject( &outpen );
		dc.SelectObject( &outbrush );
		dc.Pie( XCenter-Radius, YCenter-Radius, XCenter+Radius+1, YCenter+Radius+1,	XCenter-dx, y, XCenter+dx, y );

		// Transition
		if ( (dx != 0) || (outerangle > 3.14) )
		{
			CPen tpen( PS_SOLID, 1, tcolor );
			CBrush tbrush;
			tbrush.CreateSolidBrush( tcolor );
			dc.SelectObject( &tpen );
			dc.SelectObject( &tbrush );
			dc.Pie( XCenter-Radius, YCenter-Radius, XCenter+Radius+1, YCenter+Radius+1,	XCenter+dx, y, XCenter-dx, y );
		}

		dx = (uint)(Radius*sin(innerangle/2.0f));
		y = YCenter-(uint)(Radius*cos(innerangle/2.0f));

		// Inner
		if ( (dx != 0) || (innerangle > 3.14) )
		{
			CPen inpen( PS_SOLID, 1, innercolor );
			CBrush inbrush;
			inbrush.CreateSolidBrush( innercolor );
			dc.SelectObject( &inpen );
			dc.SelectObject( &inbrush );
			dc.Pie( XCenter-Radius, YCenter-Radius, XCenter+Radius+1, YCenter+Radius+1,	XCenter+dx, y, XCenter-dx, y );
		}
	}
}


/*
 *
 */
BOOL CSoundPage::DestroyWindow() 
{
	delete _NameFont;

	return CDialog::DestroyWindow();
}


/*
 *
 */
void CSoundPage::OnButtonHelp() 
{
	MessageBox( "\
Gain: Range: [0, 1]. A gain factor is logarithmic ; 1.0 means no attenuation (full volume) ; 0.5 \
means an attenuation of 6 dB ; 0 means silence.\n\n\
Pitch: Range: ]0, 1]. 1.0 means normal ; dividing by 2 means pitching one octave down. Pitching up \
is not supported. 0 is not a legal value.\n\n\
Min Dist: Distance threshold below which gain is clamped (does not increase anymore).\n\
Unit: meters. Range: 0 - 1000000 (the maximum value 1e+006 is considered as infinite).\n\n\
Max Dist: Distance threshold above which gain is clamped (does not decrease anymore).\n\
Unit: meters. Range: 0 - 1000000 m (the maximum value 1e+006 is considered as infinite).\n\n\
Cone Inner Angle: Inside angle of the sound cone where the main gain is applied. \
The default of 360 means that the inner angle covers the entire world, which is equivalent to \
an omnidirectional source.\n\
Unit: degrees. Range: 0 - 360.\n\n\
Cone Outer Angle: Outer angle of the sound cone where the outer gain is applied \
to the main gain. The default of 360 means that the outer angle covers the entire world. If \
the inner angle is also 360, then there is no transition zone for angle-dependent (progressive) \
attenuation.\n\
Unit: degress. Range: 0 - 360.\n\n\
Cone Outer Gain: The factor with which the main gain is multiplied to determine the effective \
gain outside the cone defined by the outer angle. To test the outer gain, you can play the \
sound source as if you were outside the cone.\n\
Range: 0 - 1.\
", "Help about types" );	
}