mirror of
https://github.com/WinampDesktop/winamp.git
synced 2024-09-24 15:54:12 +00:00
1992 lines
62 KiB
C++
1992 lines
62 KiB
C++
|
/*
|
||
|
* MPDlgs.cpp
|
||
|
* ----------
|
||
|
* Purpose: Implementation of various player setup dialogs.
|
||
|
* Notes : (currently none)
|
||
|
* Authors: OpenMPT Devs
|
||
|
* The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
|
||
|
*/
|
||
|
|
||
|
|
||
|
#include "stdafx.h"
|
||
|
#include "Mptrack.h"
|
||
|
#include "Sndfile.h"
|
||
|
#include "Mainfrm.h"
|
||
|
#include "ImageLists.h"
|
||
|
#include "Moddoc.h"
|
||
|
#include "Mpdlgs.h"
|
||
|
#include "dlg_misc.h"
|
||
|
#include "../common/mptStringBuffer.h"
|
||
|
#include "openmpt/sounddevice/SoundDevice.hpp"
|
||
|
#include "openmpt/sounddevice/SoundDeviceManager.hpp"
|
||
|
#include "../common/Dither.h"
|
||
|
|
||
|
|
||
|
OPENMPT_NAMESPACE_BEGIN
|
||
|
|
||
|
|
||
|
const TCHAR *gszChnCfgNames[3] =
|
||
|
{
|
||
|
_T("Mono"),
|
||
|
_T("Stereo"),
|
||
|
_T("Quad")
|
||
|
};
|
||
|
|
||
|
|
||
|
static double ParseTime(CString str)
|
||
|
{
|
||
|
return ConvertStrTo<double>(mpt::ToCharset(mpt::Charset::ASCII, str)) / 1000.0;
|
||
|
}
|
||
|
|
||
|
|
||
|
static CString PrintTime(double seconds)
|
||
|
{
|
||
|
int32 microseconds = mpt::saturate_round<int32>(seconds * 1000000.0);
|
||
|
int precision = 0;
|
||
|
if(microseconds < 1000)
|
||
|
{
|
||
|
precision = 3;
|
||
|
} else if(microseconds < 10000)
|
||
|
{
|
||
|
precision = 2;
|
||
|
} else if(microseconds < 100000)
|
||
|
{
|
||
|
precision = 1;
|
||
|
} else
|
||
|
{
|
||
|
precision = 0;
|
||
|
}
|
||
|
return MPT_CFORMAT("{} ms")(mpt::cfmt::fix(seconds * 1000.0, precision));
|
||
|
}
|
||
|
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(COptionsSoundcard, CPropertyPage)
|
||
|
ON_WM_HSCROLL()
|
||
|
ON_COMMAND(IDC_CHECK4, &COptionsSoundcard::OnExclusiveModeChanged)
|
||
|
ON_COMMAND(IDC_CHECK5, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK7, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK9, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK_SOUNDCARD_SHOWALL, &COptionsSoundcard::OnSoundCardShowAll)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO1, &COptionsSoundcard::OnDeviceChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO2, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_UPDATEINTERVAL, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO3, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO4, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO5, &COptionsSoundcard::OnChannelsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO6, &COptionsSoundcard::OnSampleFormatChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO10, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_EDITCHANGE(IDC_COMBO2, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_EDITCHANGE(IDC_COMBO_UPDATEINTERVAL, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO11, &COptionsSoundcard::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_BUTTON1, &COptionsSoundcard::OnSoundCardRescan)
|
||
|
ON_COMMAND(IDC_BUTTON2, &COptionsSoundcard::OnSoundCardDriverPanel)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_CHANNEL_FRONTLEFT, &COptionsSoundcard::OnChannel1Changed)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_CHANNEL_FRONTRIGHT, &COptionsSoundcard::OnChannel2Changed)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_CHANNEL_REARLEFT, &COptionsSoundcard::OnChannel3Changed)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_CHANNEL_REARRIGHT, &COptionsSoundcard::OnChannel4Changed)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_RECORDING_CHANNELS, &COptionsSoundcard::OnRecordingChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_RECORDING_SOURCE, &COptionsSoundcard::OnSettingsChanged)
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnSampleFormatChanged()
|
||
|
{
|
||
|
OnSettingsChanged();
|
||
|
UpdateDither();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnRecordingChanged()
|
||
|
{
|
||
|
DWORD_PTR inputChannels = m_CbnRecordingChannels.GetItemData(m_CbnRecordingChannels.GetCurSel());
|
||
|
m_CbnRecordingSource.EnableWindow((m_CurrentDeviceCaps.HasNamedInputSources && inputChannels > 0) ? TRUE : FALSE);
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::DoDataExchange(CDataExchange* pDX)
|
||
|
{
|
||
|
CPropertyPage::DoDataExchange(pDX);
|
||
|
//{{AFX_DATA_MAP(COptionsSoundcard)
|
||
|
DDX_Control(pDX, IDC_COMBO1, m_CbnDevice);
|
||
|
DDX_Control(pDX, IDC_COMBO2, m_CbnLatencyMS);
|
||
|
DDX_Control(pDX, IDC_COMBO_UPDATEINTERVAL, m_CbnUpdateIntervalMS);
|
||
|
DDX_Control(pDX, IDC_COMBO3, m_CbnMixingFreq);
|
||
|
DDX_Control(pDX, IDC_COMBO5, m_CbnChannels);
|
||
|
DDX_Control(pDX, IDC_COMBO6, m_CbnSampleFormat);
|
||
|
DDX_Control(pDX, IDC_COMBO10, m_CbnDither);
|
||
|
DDX_Control(pDX, IDC_BUTTON2, m_BtnDriverPanel);
|
||
|
DDX_Control(pDX, IDC_COMBO6, m_CbnSampleFormat);
|
||
|
DDX_Control(pDX, IDC_COMBO11, m_CbnStoppedMode);
|
||
|
DDX_Control(pDX, IDC_COMBO_CHANNEL_FRONTLEFT , m_CbnChannelMapping[0]);
|
||
|
DDX_Control(pDX, IDC_COMBO_CHANNEL_FRONTRIGHT, m_CbnChannelMapping[1]);
|
||
|
DDX_Control(pDX, IDC_COMBO_CHANNEL_REARLEFT , m_CbnChannelMapping[2]);
|
||
|
DDX_Control(pDX, IDC_COMBO_CHANNEL_REARRIGHT , m_CbnChannelMapping[3]);
|
||
|
DDX_Control(pDX, IDC_COMBO_RECORDING_CHANNELS, m_CbnRecordingChannels);
|
||
|
DDX_Control(pDX, IDC_COMBO_RECORDING_SOURCE, m_CbnRecordingSource);
|
||
|
DDX_Control(pDX, IDC_EDIT_STATISTICS, m_EditStatistics);
|
||
|
//}}AFX_DATA_MAP
|
||
|
}
|
||
|
|
||
|
|
||
|
COptionsSoundcard::COptionsSoundcard(SoundDevice::Identifier deviceIdentifier)
|
||
|
: CPropertyPage(IDD_OPTIONS_SOUNDCARD)
|
||
|
, m_InitialDeviceIdentifier(deviceIdentifier)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::SetInitialDevice()
|
||
|
{
|
||
|
SetDevice(m_InitialDeviceIdentifier, true);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::SetDevice(SoundDevice::Identifier dev, bool forceReload)
|
||
|
{
|
||
|
SoundDevice::Identifier olddev = m_CurrentDeviceInfo.GetIdentifier();
|
||
|
SoundDevice::Info newInfo;
|
||
|
SoundDevice::Caps newCaps;
|
||
|
SoundDevice::DynamicCaps newDynamicCaps;
|
||
|
SoundDevice::Settings newSettings;
|
||
|
newInfo = theApp.GetSoundDevicesManager()->FindDeviceInfo(dev);
|
||
|
newCaps = theApp.GetSoundDevicesManager()->GetDeviceCaps(dev, CMainFrame::GetMainFrame()->gpSoundDevice);
|
||
|
newDynamicCaps = theApp.GetSoundDevicesManager()->GetDeviceDynamicCaps(dev, TrackerSettings::Instance().GetSampleRates(), CMainFrame::GetMainFrame(), CMainFrame::GetMainFrame()->gpSoundDevice, true);
|
||
|
bool deviceChanged = (dev != olddev);
|
||
|
if(deviceChanged || forceReload)
|
||
|
{
|
||
|
newSettings = TrackerSettings::Instance().GetSoundDeviceSettings(dev);
|
||
|
} else
|
||
|
{
|
||
|
newSettings = m_Settings;
|
||
|
}
|
||
|
m_CurrentDeviceInfo = newInfo;
|
||
|
m_CurrentDeviceCaps = newCaps;
|
||
|
m_CurrentDeviceDynamicCaps = newDynamicCaps;
|
||
|
m_Settings = newSettings;
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnSoundCardShowAll()
|
||
|
{
|
||
|
TrackerSettings::Instance().m_SoundShowDeprecatedDevices = (IsDlgButtonChecked(IDC_CHECK_SOUNDCARD_SHOWALL) == BST_CHECKED);
|
||
|
SetDevice(m_CurrentDeviceInfo.GetIdentifier(), true);
|
||
|
UpdateEverything();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnSoundCardRescan()
|
||
|
{
|
||
|
{
|
||
|
// Close sound device because IDs might change when re-enumerating which could cause all kinds of havoc.
|
||
|
CMainFrame::GetMainFrame()->audioCloseDevice();
|
||
|
delete CMainFrame::GetMainFrame()->gpSoundDevice;
|
||
|
CMainFrame::GetMainFrame()->gpSoundDevice = nullptr;
|
||
|
}
|
||
|
theApp.GetSoundDevicesManager()->ReEnumerate();
|
||
|
SetDevice(m_CurrentDeviceInfo.GetIdentifier(), true);
|
||
|
UpdateEverything();
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsSoundcard::OnInitDialog()
|
||
|
{
|
||
|
CPropertyPage::OnInitDialog();
|
||
|
SetInitialDevice();
|
||
|
UpdateEverything();
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateLatency()
|
||
|
{
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_LATENCY)->EnableWindow(TRUE);
|
||
|
m_CbnLatencyMS.EnableWindow(TRUE);
|
||
|
}
|
||
|
// latency
|
||
|
{
|
||
|
static constexpr double latencies [] = {
|
||
|
0.001,
|
||
|
0.002,
|
||
|
0.003,
|
||
|
0.004,
|
||
|
0.005,
|
||
|
0.010,
|
||
|
0.015,
|
||
|
0.020,
|
||
|
0.025,
|
||
|
0.030,
|
||
|
0.040,
|
||
|
0.050,
|
||
|
0.075,
|
||
|
0.100,
|
||
|
0.150,
|
||
|
0.200,
|
||
|
0.250
|
||
|
};
|
||
|
m_CbnLatencyMS.ResetContent();
|
||
|
m_CbnLatencyMS.SetWindowText(PrintTime(m_Settings.Latency));
|
||
|
for(auto lat : latencies)
|
||
|
{
|
||
|
if(m_CurrentDeviceCaps.LatencyMin <= lat && lat <= m_CurrentDeviceCaps.LatencyMax)
|
||
|
{
|
||
|
m_CbnLatencyMS.AddString(PrintTime(lat));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_LATENCY)->EnableWindow(FALSE);
|
||
|
m_CbnLatencyMS.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateUpdateInterval()
|
||
|
{
|
||
|
{
|
||
|
m_CbnUpdateIntervalMS.EnableWindow(TRUE);
|
||
|
}
|
||
|
// update interval
|
||
|
{
|
||
|
static constexpr double updateIntervals [] = {
|
||
|
0.001,
|
||
|
0.002,
|
||
|
0.005,
|
||
|
0.010,
|
||
|
0.015,
|
||
|
0.020,
|
||
|
0.025,
|
||
|
0.050
|
||
|
};
|
||
|
m_CbnUpdateIntervalMS.ResetContent();
|
||
|
m_CbnUpdateIntervalMS.SetWindowText(PrintTime(m_Settings.UpdateInterval));
|
||
|
for(auto upd : updateIntervals)
|
||
|
{
|
||
|
if(m_CurrentDeviceCaps.UpdateIntervalMin <= upd && upd <= m_CurrentDeviceCaps.UpdateIntervalMax)
|
||
|
{
|
||
|
m_CbnUpdateIntervalMS.AddString(PrintTime(upd));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()) || !m_CurrentDeviceCaps.CanUpdateInterval)
|
||
|
{
|
||
|
m_CbnUpdateIntervalMS.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateGeneral()
|
||
|
{
|
||
|
// General
|
||
|
{
|
||
|
if(m_CurrentDeviceCaps.CanKeepDeviceRunning)
|
||
|
{
|
||
|
m_CbnStoppedMode.ResetContent();
|
||
|
m_CbnStoppedMode.AddString(_T("Close driver"));
|
||
|
m_CbnStoppedMode.AddString(_T("Pause driver"));
|
||
|
m_CbnStoppedMode.AddString(_T("Play silence"));
|
||
|
m_CbnStoppedMode.SetCurSel(TrackerSettings::Instance().m_SoundSettingsStopMode);
|
||
|
} else
|
||
|
{
|
||
|
m_CbnStoppedMode.ResetContent();
|
||
|
m_CbnStoppedMode.AddString(_T("Close driver"));
|
||
|
m_CbnStoppedMode.AddString(_T("Close driver"));
|
||
|
m_CbnStoppedMode.AddString(_T("Close driver"));
|
||
|
m_CbnStoppedMode.SetCurSel(TrackerSettings::Instance().m_SoundSettingsStopMode);
|
||
|
}
|
||
|
CheckDlgButton(IDC_CHECK7, TrackerSettings::Instance().m_SoundSettingsOpenDeviceAtStartup ? BST_CHECKED : BST_UNCHECKED);
|
||
|
}
|
||
|
bool isUnavailble = theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier());
|
||
|
m_CbnStoppedMode.EnableWindow(isUnavailble ? FALSE : (m_CurrentDeviceCaps.CanKeepDeviceRunning ? TRUE : FALSE));
|
||
|
CPropertySheet *sheet = dynamic_cast<CPropertySheet *>(GetParent());
|
||
|
if(sheet) sheet->GetDlgItem(IDOK)->EnableWindow(isUnavailble ? FALSE : TRUE);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateEverything()
|
||
|
{
|
||
|
// Sound Device
|
||
|
{
|
||
|
if(m_CurrentDeviceInfo.IsDeprecated())
|
||
|
{
|
||
|
TrackerSettings::Instance().m_SoundShowDeprecatedDevices = true;
|
||
|
}
|
||
|
CheckDlgButton(IDC_CHECK_SOUNDCARD_SHOWALL, TrackerSettings::Instance().m_SoundShowDeprecatedDevices ? BST_CHECKED : BST_UNCHECKED);
|
||
|
|
||
|
m_CbnDevice.ResetContent();
|
||
|
m_CbnDevice.SetImageList(&CMainFrame::GetMainFrame()->m_MiscIcons);
|
||
|
|
||
|
UINT iItem = 0;
|
||
|
|
||
|
for(const auto &it : *theApp.GetSoundDevicesManager())
|
||
|
{
|
||
|
|
||
|
if(!TrackerSettings::Instance().m_SoundShowDeprecatedDevices)
|
||
|
{
|
||
|
if(it.IsDeprecated())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
{
|
||
|
COMBOBOXEXITEM cbi;
|
||
|
MemsetZero(cbi);
|
||
|
cbi.iItem = iItem;
|
||
|
cbi.cchTextMax = 0;
|
||
|
cbi.mask = CBEIF_LPARAM | CBEIF_TEXT;
|
||
|
cbi.lParam = theApp.GetSoundDevicesManager()->GetGlobalID(it.GetIdentifier());
|
||
|
mpt::ustring TypeWineNative = U_("Wine-Native");
|
||
|
if(it.type == SoundDevice::TypeWAVEOUT || it.type == SoundDevice::TypePORTAUDIO_WMME)
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_WAVEOUT;
|
||
|
} else if(it.type == SoundDevice::TypeDSOUND || it.type == SoundDevice::TypePORTAUDIO_DS || it.type == U_("RtAudio-ds"))
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_DIRECTX;
|
||
|
} else if(it.type == SoundDevice::TypeASIO || it.type == U_("RtAudio-asio"))
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_ASIO;
|
||
|
} else if(it.type == SoundDevice::TypePORTAUDIO_WASAPI || it.type == U_("RtAudio-wasapi"))
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_SAMPLEMUTE; // // No real image available for now,
|
||
|
} else if(it.type == SoundDevice::TypePORTAUDIO_WDMKS)
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_CHIP; // No real image available for now,
|
||
|
} else if(it.type.find(TypeWineNative + U_("-")) == 0)
|
||
|
{
|
||
|
if(theApp.GetWineVersion() && (theApp.GetWineVersion()->HostClass() == mpt::osinfo::osclass::Linux))
|
||
|
{
|
||
|
cbi.mask |= CBEIF_IMAGE | CBEIF_SELECTEDIMAGE | CBEIF_OVERLAY;
|
||
|
cbi.iImage = IMAGE_TUX;
|
||
|
} else
|
||
|
{
|
||
|
cbi.iImage = 0;
|
||
|
}
|
||
|
} else
|
||
|
{
|
||
|
cbi.iImage = 0;
|
||
|
}
|
||
|
cbi.iSelectedImage = cbi.iImage;
|
||
|
cbi.iOverlay = cbi.iImage;
|
||
|
CString tmp = mpt::ToCString(it.GetDisplayName());
|
||
|
cbi.pszText = const_cast<TCHAR *>(tmp.GetString());
|
||
|
cbi.iIndent = 0;
|
||
|
int pos = m_CbnDevice.InsertItem(&cbi);
|
||
|
if(static_cast<SoundDevice::Manager::GlobalID>(cbi.lParam) == theApp.GetSoundDevicesManager()->GetGlobalID(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_CbnDevice.SetCurSel(pos);
|
||
|
}
|
||
|
iItem++;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
UpdateDevice();
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateDevice()
|
||
|
{
|
||
|
GetDlgItem(IDC_CHECK_SOUNDCARD_SHOWALL)->EnableWindow(m_CurrentDeviceInfo.IsDeprecated() ? FALSE : TRUE);
|
||
|
UpdateGeneral();
|
||
|
UpdateControls();
|
||
|
UpdateLatency();
|
||
|
UpdateUpdateInterval();
|
||
|
UpdateSampleRates();
|
||
|
UpdateChannels();
|
||
|
UpdateSampleFormat();
|
||
|
UpdateDither();
|
||
|
UpdateChannelMapping();
|
||
|
UpdateRecording();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateChannels()
|
||
|
{
|
||
|
{
|
||
|
m_CbnChannels.EnableWindow(TRUE);
|
||
|
}
|
||
|
m_CbnChannels.ResetContent();
|
||
|
int maxChannels = 0;
|
||
|
if(m_CurrentDeviceDynamicCaps.channelNames.size() > 0)
|
||
|
{
|
||
|
maxChannels = static_cast<int>(std::min(std::size_t(4), m_CurrentDeviceDynamicCaps.channelNames.size()));
|
||
|
} else
|
||
|
{
|
||
|
maxChannels = 4;
|
||
|
}
|
||
|
int sel = 0;
|
||
|
for(int channels = maxChannels; channels >= 1; channels /= 2)
|
||
|
{
|
||
|
int ndx = m_CbnChannels.AddString(gszChnCfgNames[(channels+2)/2-1]);
|
||
|
m_CbnChannels.SetItemData(ndx, channels);
|
||
|
if(channels == m_Settings.Channels)
|
||
|
{
|
||
|
sel = ndx;
|
||
|
}
|
||
|
}
|
||
|
m_CbnChannels.SetCurSel(sel);
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_CbnChannels.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateRecording()
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_RECORDING)->ShowWindow(TrackerSettings::Instance().m_SoundShowRecordingSettings ? SW_SHOW : SW_HIDE);
|
||
|
m_CbnRecordingChannels.ShowWindow(TrackerSettings::Instance().m_SoundShowRecordingSettings ? SW_SHOW : SW_HIDE);
|
||
|
m_CbnRecordingSource.ShowWindow(TrackerSettings::Instance().m_SoundShowRecordingSettings ? SW_SHOW : SW_HIDE);
|
||
|
m_CbnRecordingChannels.ResetContent();
|
||
|
m_CbnRecordingSource.ResetContent();
|
||
|
if(m_CurrentDeviceCaps.CanInput && ((m_CurrentDeviceCaps.HasNamedInputSources && m_CurrentDeviceDynamicCaps.inputSourceNames.size() > 0) || !m_CurrentDeviceCaps.HasNamedInputSources))
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_RECORDING)->EnableWindow(TRUE);
|
||
|
m_CbnRecordingChannels.EnableWindow(TRUE);
|
||
|
int sel = 0;
|
||
|
{
|
||
|
int ndx = m_CbnRecordingChannels.AddString(_T("off"));
|
||
|
m_CbnRecordingChannels.SetItemData(ndx, 0);
|
||
|
if(0 == m_Settings.InputChannels)
|
||
|
{
|
||
|
sel = ndx;
|
||
|
}
|
||
|
}
|
||
|
for(int channels = 4; channels >= 1; channels /= 2)
|
||
|
{
|
||
|
int ndx = m_CbnRecordingChannels.AddString(gszChnCfgNames[(channels+2)/2-1]);
|
||
|
m_CbnRecordingChannels.SetItemData(ndx, channels);
|
||
|
if(channels == m_Settings.InputChannels)
|
||
|
{
|
||
|
sel = ndx;
|
||
|
}
|
||
|
}
|
||
|
m_CbnRecordingChannels.SetCurSel(sel);
|
||
|
if(m_CurrentDeviceCaps.HasNamedInputSources)
|
||
|
{
|
||
|
m_CbnRecordingSource.EnableWindow((m_Settings.InputChannels > 0) ? TRUE : FALSE);
|
||
|
sel = -1;
|
||
|
for(size_t ch = 0; ch < m_CurrentDeviceDynamicCaps.inputSourceNames.size(); ch++)
|
||
|
{
|
||
|
const int pos = (int)::SendMessageW(m_CbnRecordingSource.m_hWnd, CB_ADDSTRING, 0, (LPARAM)m_CurrentDeviceDynamicCaps.inputSourceNames[ch].second.c_str());
|
||
|
m_CbnRecordingSource.SetItemData(pos, (DWORD_PTR)m_CurrentDeviceDynamicCaps.inputSourceNames[ch].first);
|
||
|
if(m_CurrentDeviceDynamicCaps.inputSourceNames[ch].first == m_Settings.InputSourceID)
|
||
|
{
|
||
|
sel = pos;
|
||
|
}
|
||
|
}
|
||
|
if(sel == -1 ) sel = 0;
|
||
|
m_CbnRecordingSource.SetCurSel(sel);
|
||
|
} else
|
||
|
{
|
||
|
m_CbnRecordingSource.EnableWindow(FALSE);
|
||
|
}
|
||
|
} else
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_RECORDING)->EnableWindow(FALSE);
|
||
|
m_CbnRecordingChannels.EnableWindow(FALSE);
|
||
|
int ndx = m_CbnRecordingChannels.AddString(_T("off"));
|
||
|
m_CbnRecordingChannels.SetItemData(ndx, 0);
|
||
|
m_CbnRecordingChannels.SetCurSel(ndx);
|
||
|
m_CbnRecordingSource.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateSampleFormat()
|
||
|
{
|
||
|
{
|
||
|
m_CbnSampleFormat.EnableWindow(TRUE);
|
||
|
}
|
||
|
UINT n = 0;
|
||
|
m_CbnSampleFormat.ResetContent();
|
||
|
std::vector<SampleFormat> sampleformats;
|
||
|
if(IsDlgButtonChecked(IDC_CHECK4))
|
||
|
{
|
||
|
sampleformats = m_CurrentDeviceDynamicCaps.supportedExclusiveModeSampleFormats;
|
||
|
} else
|
||
|
{
|
||
|
sampleformats = m_CurrentDeviceDynamicCaps.supportedSampleFormats;
|
||
|
}
|
||
|
m_CbnSampleFormat.EnableWindow(m_CurrentDeviceCaps.CanSampleFormat && (sampleformats.size() != 1) ? TRUE : FALSE);
|
||
|
const std::vector<SampleFormat> allSampleFormats = AllSampleFormats<std::vector<SampleFormat>>();
|
||
|
for(const auto sampleFormat : allSampleFormats)
|
||
|
{
|
||
|
if(!sampleformats.empty() && !mpt::contains(sampleformats, sampleFormat))
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
CString name;
|
||
|
if(sampleFormat.IsFloat())
|
||
|
{
|
||
|
name = MPT_CFORMAT("Float {} bit")(sampleFormat.GetBitsPerSample());
|
||
|
} else if(sampleFormat.IsUnsigned())
|
||
|
{
|
||
|
name = MPT_CFORMAT("{} Bit uint")(sampleFormat.GetBitsPerSample());
|
||
|
} else
|
||
|
{
|
||
|
name = MPT_CFORMAT("{} Bit")(sampleFormat.GetBitsPerSample());
|
||
|
}
|
||
|
UINT ndx = m_CbnSampleFormat.AddString(name);
|
||
|
m_CbnSampleFormat.SetItemData(ndx, mpt::to_underlying<SampleFormat::Enum>(sampleFormat));
|
||
|
if(sampleFormat == m_Settings.sampleFormat)
|
||
|
{
|
||
|
n = ndx;
|
||
|
}
|
||
|
}
|
||
|
m_CbnSampleFormat.SetCurSel(n);
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_CbnSampleFormat.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateDither()
|
||
|
{
|
||
|
{
|
||
|
m_CbnDither.EnableWindow(TRUE);
|
||
|
}
|
||
|
m_CbnDither.ResetContent();
|
||
|
SampleFormat sampleFormat = SampleFormat::FromInt(static_cast<int>(m_CbnSampleFormat.GetItemData(m_CbnSampleFormat.GetCurSel())));
|
||
|
if(sampleFormat.IsInt() && sampleFormat.GetBitsPerSample() < 32)
|
||
|
{
|
||
|
m_CbnDither.EnableWindow(TRUE);
|
||
|
for(std::size_t i = 0; i < DithersOpenMPT::GetNumDithers(); ++i)
|
||
|
{
|
||
|
m_CbnDither.AddString(mpt::ToCString(DithersOpenMPT::GetModeName(i) + U_(" dither")));
|
||
|
}
|
||
|
} else if(m_CurrentDeviceCaps.HasInternalDither)
|
||
|
{
|
||
|
m_CbnDither.EnableWindow(TRUE);
|
||
|
m_CbnDither.AddString(mpt::ToCString(DithersOpenMPT::GetModeName(DithersOpenMPT::GetNoDither()) + U_(" dither")));
|
||
|
m_CbnDither.AddString(mpt::ToCString(DithersOpenMPT::GetModeName(DithersOpenMPT::GetDefaultDither()) + U_(" dither")));
|
||
|
} else
|
||
|
{
|
||
|
m_CbnDither.EnableWindow(FALSE);
|
||
|
for(std::size_t i = 0; i < DithersOpenMPT::GetNumDithers(); ++i)
|
||
|
{
|
||
|
m_CbnDither.AddString(mpt::ToCString(DithersOpenMPT::GetModeName(DithersOpenMPT::GetNoDither()) + U_(" dither")));
|
||
|
}
|
||
|
}
|
||
|
if(m_Settings.DitherType < 0 || m_Settings.DitherType >= m_CbnDither.GetCount())
|
||
|
{
|
||
|
m_CbnDither.SetCurSel(1);
|
||
|
} else
|
||
|
{
|
||
|
m_CbnDither.SetCurSel(m_Settings.DitherType);
|
||
|
}
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_CbnDither.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateChannelMapping()
|
||
|
{
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_CHANNELMAPPING)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_FRONT)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_REAR)->EnableWindow(TRUE);
|
||
|
for(int mch = 0; mch < NUM_CHANNELCOMBOBOXES; mch++)
|
||
|
{
|
||
|
CComboBox *combo = &m_CbnChannelMapping[mch];
|
||
|
combo->EnableWindow(TRUE);
|
||
|
}
|
||
|
}
|
||
|
int usedChannels = static_cast<int>(m_CbnChannels.GetItemData(m_CbnChannels.GetCurSel()));
|
||
|
if(m_Settings.Channels.GetNumHostChannels() != static_cast<uint32>(usedChannels))
|
||
|
{
|
||
|
// If the channel mapping is not valid for the selected number of channels, reset it to default identity mapping.
|
||
|
m_Settings.Channels = SoundDevice::ChannelMapping(usedChannels);
|
||
|
}
|
||
|
GetDlgItem(IDC_STATIC_CHANNELMAPPING)->EnableWindow(m_CurrentDeviceCaps.CanChannelMapping ? TRUE : FALSE);
|
||
|
if(m_CurrentDeviceCaps.CanChannelMapping && usedChannels > 2)
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_FRONT)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_REAR)->EnableWindow(TRUE);
|
||
|
} else
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_FRONT)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_REAR)->EnableWindow(FALSE);
|
||
|
}
|
||
|
for(int mch = 0; mch < NUM_CHANNELCOMBOBOXES; mch++) // Host channels
|
||
|
{
|
||
|
CComboBox *combo = &m_CbnChannelMapping[mch];
|
||
|
combo->EnableWindow((m_CurrentDeviceCaps.CanChannelMapping && mch < usedChannels) ? TRUE : FALSE);
|
||
|
combo->ResetContent();
|
||
|
if(m_CurrentDeviceCaps.CanChannelMapping)
|
||
|
{
|
||
|
combo->SetItemData(combo->AddString(_T("Unassigned")), (DWORD_PTR)-1);
|
||
|
combo->SetCurSel(0);
|
||
|
if(mch < usedChannels)
|
||
|
{
|
||
|
for(size_t dch = 0; dch < m_CurrentDeviceDynamicCaps.channelNames.size(); dch++) // Device channels
|
||
|
{
|
||
|
const int pos = (int)::SendMessageW(combo->m_hWnd, CB_ADDSTRING, 0, (LPARAM)m_CurrentDeviceDynamicCaps.channelNames[dch].c_str());
|
||
|
combo->SetItemData(pos, (DWORD_PTR)dch);
|
||
|
if(static_cast<int32>(dch) == m_Settings.Channels.ToDevice(mch))
|
||
|
{
|
||
|
combo->SetCurSel(pos);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_CHANNELMAPPING)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_FRONT)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_STATIC_CHANNEL_REAR)->EnableWindow(FALSE);
|
||
|
for(int mch = 0; mch < NUM_CHANNELCOMBOBOXES; mch++)
|
||
|
{
|
||
|
CComboBox *combo = &m_CbnChannelMapping[mch];
|
||
|
combo->EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnDeviceChanged()
|
||
|
{
|
||
|
int n = m_CbnDevice.GetCurSel();
|
||
|
if(n >= 0)
|
||
|
{
|
||
|
SetDevice(theApp.GetSoundDevicesManager()->FindDeviceInfo(static_cast<SoundDevice::Manager::GlobalID>(m_CbnDevice.GetItemData(n))).GetIdentifier());
|
||
|
UpdateDevice();
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnExclusiveModeChanged()
|
||
|
{
|
||
|
UpdateSampleRates();
|
||
|
UpdateSampleFormat();
|
||
|
UpdateDither();
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnChannelsChanged()
|
||
|
{
|
||
|
UpdateChannelMapping();
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnSoundCardDriverPanel()
|
||
|
{
|
||
|
theApp.GetSoundDevicesManager()->OpenDriverSettings(
|
||
|
theApp.GetSoundDevicesManager()->FindDeviceInfo(static_cast<SoundDevice::Manager::GlobalID>(m_CbnDevice.GetItemData(m_CbnDevice.GetCurSel()))).GetIdentifier(),
|
||
|
CMainFrame::GetMainFrame(),
|
||
|
CMainFrame::GetMainFrame()->gpSoundDevice
|
||
|
);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnChannelChanged(int channel)
|
||
|
{
|
||
|
CComboBox *combo = &m_CbnChannelMapping[channel];
|
||
|
const LONG_PTR newChn = combo->GetItemData(combo->GetCurSel());
|
||
|
if(newChn == -1)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
// Ensure that no channel is used twice
|
||
|
for(int mch = 0; mch < NUM_CHANNELCOMBOBOXES; mch++) // Host channels
|
||
|
{
|
||
|
if(mch != channel)
|
||
|
{
|
||
|
combo = &m_CbnChannelMapping[mch];
|
||
|
if((int)combo->GetItemData(combo->GetCurSel()) == newChn)
|
||
|
{
|
||
|
// find an unused channel
|
||
|
bool found = false;
|
||
|
int deviceChannel = 0;
|
||
|
for(; deviceChannel < static_cast<int>(m_CurrentDeviceDynamicCaps.channelNames.size()); ++deviceChannel)
|
||
|
{
|
||
|
bool used = false;
|
||
|
for(int hostChannel = 0; hostChannel < NUM_CHANNELCOMBOBOXES; ++hostChannel)
|
||
|
{
|
||
|
if(static_cast<int>(m_CbnChannelMapping[hostChannel].GetItemData(m_CbnChannelMapping[hostChannel].GetCurSel())) == deviceChannel)
|
||
|
{
|
||
|
used = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if(!used)
|
||
|
{
|
||
|
found = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if(found)
|
||
|
{
|
||
|
combo->SetCurSel(deviceChannel+1);
|
||
|
} else
|
||
|
{
|
||
|
combo->SetCurSel(0);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
// Fill the dropdown box with a list of valid sample rates, depending on the selected sound device.
|
||
|
void COptionsSoundcard::UpdateSampleRates()
|
||
|
{
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_FORMAT)->EnableWindow(TRUE);
|
||
|
m_CbnMixingFreq.EnableWindow(TRUE);
|
||
|
}
|
||
|
|
||
|
m_CbnMixingFreq.ResetContent();
|
||
|
|
||
|
std::vector<uint32> samplerates;
|
||
|
|
||
|
if(IsDlgButtonChecked(IDC_CHECK4))
|
||
|
{
|
||
|
samplerates = m_CurrentDeviceDynamicCaps.supportedExclusiveSampleRates;
|
||
|
} else
|
||
|
{
|
||
|
samplerates = m_CurrentDeviceDynamicCaps.supportedSampleRates;
|
||
|
}
|
||
|
|
||
|
if(samplerates.empty())
|
||
|
{
|
||
|
// We have no valid list of supported playback rates! Assume all rates supported by OpenMPT are possible...
|
||
|
samplerates = TrackerSettings::Instance().GetSampleRates();
|
||
|
}
|
||
|
|
||
|
int n = 0;
|
||
|
for(size_t i = 0; i < samplerates.size(); i++)
|
||
|
{
|
||
|
int pos = m_CbnMixingFreq.AddString(MPT_CFORMAT("{} Hz")(samplerates[i]));
|
||
|
m_CbnMixingFreq.SetItemData(pos, samplerates[i]);
|
||
|
if(m_Settings.Samplerate == samplerates[i])
|
||
|
{
|
||
|
n = pos;
|
||
|
}
|
||
|
}
|
||
|
m_CbnMixingFreq.SetCurSel(n);
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
GetDlgItem(IDC_STATIC_FORMAT)->EnableWindow(FALSE);
|
||
|
m_CbnMixingFreq.EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateControls()
|
||
|
{
|
||
|
{
|
||
|
m_BtnDriverPanel.EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_CHECK4)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_CHECK5)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_CHECK9)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_STATIC_UPDATEINTERVAL)->EnableWindow(TRUE);
|
||
|
GetDlgItem(IDC_COMBO_UPDATEINTERVAL)->EnableWindow(TRUE);
|
||
|
}
|
||
|
if(!m_CurrentDeviceCaps.CanKeepDeviceRunning)
|
||
|
{
|
||
|
m_Settings.KeepDeviceRunning = false;
|
||
|
}
|
||
|
m_BtnDriverPanel.EnableWindow(m_CurrentDeviceCaps.CanDriverPanel ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_CHECK4)->EnableWindow(m_CurrentDeviceCaps.CanExclusiveMode ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_CHECK5)->EnableWindow(m_CurrentDeviceCaps.CanBoostThreadPriority ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_CHECK9)->EnableWindow(m_CurrentDeviceCaps.CanUseHardwareTiming ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_STATIC_UPDATEINTERVAL)->EnableWindow(m_CurrentDeviceCaps.CanUpdateInterval ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_COMBO_UPDATEINTERVAL)->EnableWindow(m_CurrentDeviceCaps.CanUpdateInterval ? TRUE : FALSE);
|
||
|
GetDlgItem(IDC_CHECK4)->SetWindowText(mpt::ToCString(m_CurrentDeviceCaps.ExclusiveModeDescription));
|
||
|
CheckDlgButton(IDC_CHECK4, m_CurrentDeviceCaps.CanExclusiveMode && m_Settings.ExclusiveMode ? BST_CHECKED : BST_UNCHECKED);
|
||
|
CheckDlgButton(IDC_CHECK5, m_CurrentDeviceCaps.CanBoostThreadPriority && m_Settings.BoostThreadPriority ? BST_CHECKED : BST_UNCHECKED);
|
||
|
CheckDlgButton(IDC_CHECK9, m_CurrentDeviceCaps.CanUseHardwareTiming && m_Settings.UseHardwareTiming ? BST_CHECKED : BST_UNCHECKED);
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_BtnDriverPanel.EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_CHECK4)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_CHECK5)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_CHECK9)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_STATIC_UPDATEINTERVAL)->EnableWindow(FALSE);
|
||
|
GetDlgItem(IDC_COMBO_UPDATEINTERVAL)->EnableWindow(FALSE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsSoundcard::OnSetActive()
|
||
|
{
|
||
|
CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_SOUNDCARD;
|
||
|
return CPropertyPage::OnSetActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::OnOK()
|
||
|
{
|
||
|
if(!theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
|
||
|
// General
|
||
|
{
|
||
|
TrackerSettings::Instance().m_SoundSettingsOpenDeviceAtStartup = IsDlgButtonChecked(IDC_CHECK7) != BST_UNCHECKED;
|
||
|
}
|
||
|
m_Settings.ExclusiveMode = IsDlgButtonChecked(IDC_CHECK4) != BST_UNCHECKED;
|
||
|
m_Settings.BoostThreadPriority = IsDlgButtonChecked(IDC_CHECK5) != BST_UNCHECKED;
|
||
|
m_Settings.UseHardwareTiming = IsDlgButtonChecked(IDC_CHECK9) != BST_UNCHECKED;
|
||
|
// Mixing Freq
|
||
|
{
|
||
|
m_Settings.Samplerate = static_cast<uint32>(m_CbnMixingFreq.GetItemData(m_CbnMixingFreq.GetCurSel()));
|
||
|
}
|
||
|
// Channels
|
||
|
{
|
||
|
DWORD_PTR n = m_CbnChannels.GetItemData(m_CbnChannels.GetCurSel());
|
||
|
m_Settings.Channels = static_cast<int>(n);
|
||
|
if((m_Settings.Channels != 1) && (m_Settings.Channels != 4))
|
||
|
{
|
||
|
m_Settings.Channels = 2;
|
||
|
}
|
||
|
}
|
||
|
// SampleFormat
|
||
|
{
|
||
|
DWORD_PTR n = m_CbnSampleFormat.GetItemData(m_CbnSampleFormat.GetCurSel());
|
||
|
m_Settings.sampleFormat = SampleFormat::FromInt(static_cast<int>(n));
|
||
|
}
|
||
|
// Dither
|
||
|
{
|
||
|
m_Settings.DitherType = m_CbnDither.GetCurSel();
|
||
|
}
|
||
|
// Latency
|
||
|
{
|
||
|
CString s;
|
||
|
m_CbnLatencyMS.GetWindowText(s);
|
||
|
m_Settings.Latency = ParseTime(s);
|
||
|
//Check given value.
|
||
|
if(m_Settings.Latency == 0.0) m_Settings.Latency = m_CurrentDeviceCaps.DefaultSettings.Latency;
|
||
|
m_Settings.Latency = Clamp(m_Settings.Latency, m_CurrentDeviceCaps.LatencyMin, m_CurrentDeviceCaps.LatencyMax);
|
||
|
m_CbnLatencyMS.SetWindowText(PrintTime(m_Settings.Latency));
|
||
|
}
|
||
|
// Update Interval
|
||
|
{
|
||
|
CString s;
|
||
|
m_CbnUpdateIntervalMS.GetWindowText(s);
|
||
|
m_Settings.UpdateInterval = ParseTime(s);
|
||
|
//Check given value.
|
||
|
if(m_Settings.UpdateInterval == 0.0) m_Settings.UpdateInterval = m_CurrentDeviceCaps.DefaultSettings.UpdateInterval;
|
||
|
m_Settings.UpdateInterval = Clamp(m_Settings.UpdateInterval, m_CurrentDeviceCaps.UpdateIntervalMin, m_CurrentDeviceCaps.UpdateIntervalMax);
|
||
|
m_CbnUpdateIntervalMS.SetWindowText(PrintTime(m_Settings.UpdateInterval));
|
||
|
}
|
||
|
// Channel Mapping
|
||
|
{
|
||
|
if(m_CurrentDeviceCaps.CanChannelMapping)
|
||
|
{
|
||
|
int numChannels = std::min(static_cast<int>(m_Settings.Channels), static_cast<int>(NUM_CHANNELCOMBOBOXES));
|
||
|
std::vector<int32> channels(numChannels);
|
||
|
for(int mch = 0; mch < numChannels; mch++) // Host channels
|
||
|
{
|
||
|
CComboBox *combo = &m_CbnChannelMapping[mch];
|
||
|
channels[mch] = static_cast<int32>(combo->GetItemData(combo->GetCurSel()));
|
||
|
}
|
||
|
m_Settings.Channels = channels;
|
||
|
}
|
||
|
}
|
||
|
// Recording
|
||
|
{
|
||
|
if(TrackerSettings::Instance().m_SoundShowRecordingSettings && m_CurrentDeviceCaps.CanInput && ((m_CurrentDeviceCaps.HasNamedInputSources && m_CurrentDeviceDynamicCaps.inputSourceNames.size() > 0) || !m_CurrentDeviceCaps.HasNamedInputSources))
|
||
|
{
|
||
|
DWORD_PTR n = m_CbnRecordingChannels.GetItemData(m_CbnRecordingChannels.GetCurSel());
|
||
|
m_Settings.InputChannels = static_cast<uint8>(n);
|
||
|
if((m_Settings.InputChannels != 1) && (m_Settings.InputChannels != 2) && (m_Settings.InputChannels != 4))
|
||
|
{
|
||
|
m_Settings.InputChannels = 0;
|
||
|
}
|
||
|
if(m_CurrentDeviceCaps.HasNamedInputSources)
|
||
|
{
|
||
|
DWORD_PTR sourceID = m_CbnRecordingSource.GetItemData(m_CbnRecordingSource.GetCurSel());
|
||
|
m_Settings.InputSourceID = static_cast<uint32>(sourceID);
|
||
|
} else
|
||
|
{
|
||
|
m_Settings.InputSourceID = 0;
|
||
|
}
|
||
|
} else
|
||
|
{
|
||
|
m_Settings.InputChannels = 0;
|
||
|
m_Settings.InputSourceID = 0;
|
||
|
}
|
||
|
}
|
||
|
CMainFrame::GetMainFrame()->SetupSoundCard(m_Settings, m_CurrentDeviceInfo.GetIdentifier(), (SoundDeviceStopMode)m_CbnStoppedMode.GetCurSel());
|
||
|
SetDevice(m_CurrentDeviceInfo.GetIdentifier(), true); // Poll changed ASIO sample format and channel names
|
||
|
UpdateDevice();
|
||
|
UpdateStatistics();
|
||
|
|
||
|
} else
|
||
|
{
|
||
|
|
||
|
Reporting::Error("Sound card currently not available.");
|
||
|
|
||
|
}
|
||
|
|
||
|
CPropertyPage::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsSoundcard::UpdateStatistics()
|
||
|
{
|
||
|
if (!m_EditStatistics) return;
|
||
|
CMainFrame *pMainFrm = CMainFrame::GetMainFrame();
|
||
|
if(pMainFrm->gpSoundDevice && pMainFrm->IsPlaying())
|
||
|
{
|
||
|
const SoundDevice::BufferAttributes bufferAttributes = pMainFrm->gpSoundDevice->GetEffectiveBufferAttributes();
|
||
|
const SoundDevice::Statistics stats = pMainFrm->gpSoundDevice->GetStatistics();
|
||
|
#if 0
|
||
|
const SoundDevice::TimeInfo timeInfo = pMainFrm->gpSoundDevice->GetTimeInfo();
|
||
|
const SoundDevice::StreamPosition streamPosition = pMainFrm->gpSoundDevice->GetStreamPosition();
|
||
|
#endif
|
||
|
const uint32 samplerate = pMainFrm->gpSoundDevice->GetSettings().Samplerate;
|
||
|
mpt::ustring s;
|
||
|
if(bufferAttributes.NumBuffers > 2)
|
||
|
{
|
||
|
s += MPT_UFORMAT("Buffer: {}% ({}/{})\r\n")((bufferAttributes.Latency > 0.0) ? mpt::saturate_round<int64>(stats.InstantaneousLatency / bufferAttributes.Latency * 100.0) : 0, (stats.LastUpdateInterval > 0.0) ? mpt::saturate_round<int64>(bufferAttributes.Latency / stats.LastUpdateInterval) : 0, bufferAttributes.NumBuffers);
|
||
|
} else
|
||
|
{
|
||
|
s += MPT_UFORMAT("Buffer: {}%\r\n")((bufferAttributes.Latency > 0.0) ? mpt::saturate_round<int64>(stats.InstantaneousLatency / bufferAttributes.Latency * 100.0) : 0);
|
||
|
}
|
||
|
s += MPT_UFORMAT("Latency: {} ms (current: {} ms, {} frames)\r\n")(mpt::ufmt::fix(bufferAttributes.Latency * 1000.0, 1), mpt::ufmt::fix(stats.InstantaneousLatency * 1000.0, 1), mpt::saturate_round<int64>(stats.InstantaneousLatency * samplerate));
|
||
|
s += MPT_UFORMAT("Period: {} ms (current: {} ms, {} frames)\r\n")(mpt::ufmt::fix(bufferAttributes.UpdateInterval * 1000.0, 1), mpt::ufmt::fix(stats.LastUpdateInterval * 1000.0, 1), mpt::saturate_round<int64>(stats.LastUpdateInterval * samplerate));
|
||
|
#if 0
|
||
|
s += MPT_UFORMAT("TimeInfo: latency = {} ms / speed = {} / latency = {} ms\r\n")(
|
||
|
mpt::ufmt::fix(timeInfo.Latency * 1000.0, 1),
|
||
|
mpt::ufmt::flt(timeInfo.Speed, 4),
|
||
|
mpt::ufmt::fix((timeInfo.RenderStreamPositionBefore.Seconds - streamPosition.Seconds) * 1000.0, 1));
|
||
|
#endif
|
||
|
s += stats.text;
|
||
|
m_EditStatistics.SetWindowText(mpt::ToCString(s));
|
||
|
} else
|
||
|
{
|
||
|
if(theApp.GetSoundDevicesManager()->IsDeviceUnavailable(m_CurrentDeviceInfo.GetIdentifier()))
|
||
|
{
|
||
|
m_EditStatistics.SetWindowText(_T("Device currently unavailable."));
|
||
|
} else
|
||
|
{
|
||
|
m_EditStatistics.SetWindowText(_T(""));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//////////////////
|
||
|
// COptionsMixer
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(COptionsMixer, CPropertyPage)
|
||
|
ON_WM_HSCROLL()
|
||
|
ON_WM_VSCROLL()
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_FILTER, &COptionsMixer::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_AMIGA_TYPE, &COptionsMixer::OnSettingsChanged)
|
||
|
ON_EN_UPDATE(IDC_RAMPING_IN, &COptionsMixer::OnRampingChanged)
|
||
|
ON_EN_UPDATE(IDC_RAMPING_OUT, &COptionsMixer::OnRampingChanged)
|
||
|
ON_COMMAND(IDC_CHECK_SOFTPAN, &COptionsMixer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK1, &COptionsMixer::OnAmigaChanged)
|
||
|
ON_COMMAND(IDC_BUTTON1, &COptionsMixer::OnDefaultRampSettings)
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
void COptionsMixer::DoDataExchange(CDataExchange* pDX)
|
||
|
{
|
||
|
CPropertyPage::DoDataExchange(pDX);
|
||
|
//{{AFX_DATA_MAP(COptionsSoundcard)
|
||
|
DDX_Control(pDX, IDC_COMBO_FILTER, m_CbnResampling);
|
||
|
DDX_Control(pDX, IDC_COMBO_AMIGA_TYPE, m_CbnAmigaType);
|
||
|
DDX_Control(pDX, IDC_RAMPING_IN, m_CEditRampUp);
|
||
|
DDX_Control(pDX, IDC_RAMPING_OUT, m_CEditRampDown);
|
||
|
DDX_Control(pDX, IDC_EDIT_VOLRAMP_SAMPLES_UP, m_CInfoRampUp);
|
||
|
DDX_Control(pDX, IDC_EDIT_VOLRAMP_SAMPLES_DOWN, m_CInfoRampDown);
|
||
|
DDX_Control(pDX, IDC_SLIDER_STEREOSEP, m_SliderStereoSep);
|
||
|
// check box soft pan
|
||
|
DDX_Control(pDX, IDC_SLIDER_PREAMP, m_SliderPreAmp);
|
||
|
//}}AFX_DATA_MAP
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsMixer::OnInitDialog()
|
||
|
{
|
||
|
CPropertyPage::OnInitDialog();
|
||
|
|
||
|
// Resampling type
|
||
|
{
|
||
|
const auto resamplingModes = Resampling::AllModes();
|
||
|
for(auto mode : resamplingModes)
|
||
|
{
|
||
|
int index = m_CbnResampling.AddString(CTrackApp::GetResamplingModeName(mode, 2, true));
|
||
|
m_CbnResampling.SetItemData(index, mode);
|
||
|
if(TrackerSettings::Instance().ResamplerMode == mode)
|
||
|
{
|
||
|
m_CbnResampling.SetCurSel(index);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Amiga Resampler
|
||
|
const bool enableAmigaResampler = TrackerSettings::Instance().ResamplerEmulateAmiga != Resampling::AmigaFilter::Off;
|
||
|
CheckDlgButton(IDC_CHECK1, enableAmigaResampler ? BST_CHECKED : BST_UNCHECKED);
|
||
|
m_CbnAmigaType.EnableWindow(enableAmigaResampler ? TRUE : FALSE);
|
||
|
static constexpr std::pair<const TCHAR *, Resampling::AmigaFilter> Filters[] =
|
||
|
{
|
||
|
{_T("A500 Filter"), Resampling::AmigaFilter::A500},
|
||
|
{_T("A1200 Filter"), Resampling::AmigaFilter::A1200},
|
||
|
{_T("Unfiltered"), Resampling::AmigaFilter::Unfiltered},
|
||
|
};
|
||
|
int sel = 0;
|
||
|
for(const auto & [name, filter] : Filters)
|
||
|
{
|
||
|
const int item = m_CbnAmigaType.AddString(name);
|
||
|
m_CbnAmigaType.SetItemData(item, static_cast<DWORD_PTR>(filter));
|
||
|
if(filter == TrackerSettings::Instance().ResamplerEmulateAmiga)
|
||
|
sel = item;
|
||
|
}
|
||
|
m_CbnAmigaType.SetCurSel(sel);
|
||
|
|
||
|
// volume ramping
|
||
|
{
|
||
|
m_CEditRampUp.SetWindowText(mpt::ToCString(mpt::ufmt::val(TrackerSettings::Instance().GetMixerSettings().GetVolumeRampUpMicroseconds())));
|
||
|
m_CEditRampDown.SetWindowText(mpt::ToCString(mpt::ufmt::val(TrackerSettings::Instance().GetMixerSettings().GetVolumeRampDownMicroseconds())));
|
||
|
static_cast<CSpinButtonCtrl *>(GetDlgItem(IDC_SPIN2))->SetRange32(0, int32_max);
|
||
|
static_cast<CSpinButtonCtrl *>(GetDlgItem(IDC_SPIN3))->SetRange32(0, int32_max);
|
||
|
UpdateRamping();
|
||
|
}
|
||
|
|
||
|
// Stereo Separation
|
||
|
{
|
||
|
m_SliderStereoSep.SetRange(0, 32);
|
||
|
m_SliderStereoSep.SetPos(16);
|
||
|
for (int n = 0; n <= 32; n++)
|
||
|
{
|
||
|
if ((int)TrackerSettings::Instance().MixerStereoSeparation <= 8 * n)
|
||
|
{
|
||
|
m_SliderStereoSep.SetPos(n);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
UpdateStereoSep();
|
||
|
}
|
||
|
|
||
|
// soft pan
|
||
|
{
|
||
|
CheckDlgButton(IDC_CHECK_SOFTPAN, (TrackerSettings::Instance().MixerFlags & SNDMIX_SOFTPANNING) ? BST_CHECKED : BST_UNCHECKED);
|
||
|
}
|
||
|
|
||
|
// Pre-Amplification
|
||
|
{
|
||
|
m_SliderPreAmp.SetTicFreq(5);
|
||
|
m_SliderPreAmp.SetRange(0, 40);
|
||
|
int n = (TrackerSettings::Instance().MixerPreAmp - 64) / 8;
|
||
|
if ((n < 0) || (n > 40)) n = 16;
|
||
|
m_SliderPreAmp.SetPos(n);
|
||
|
}
|
||
|
|
||
|
m_initialized = true;
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsMixer::OnSetActive()
|
||
|
{
|
||
|
CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_MIXER;
|
||
|
return CPropertyPage::OnSetActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::OnAmigaChanged()
|
||
|
{
|
||
|
const bool enableAmigaResampler = IsDlgButtonChecked(IDC_CHECK1) != BST_UNCHECKED;
|
||
|
m_CbnAmigaType.EnableWindow(enableAmigaResampler ? TRUE : FALSE);
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::OnRampingChanged()
|
||
|
{
|
||
|
if(!m_initialized)
|
||
|
return;
|
||
|
UpdateRamping();
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::OnDefaultRampSettings()
|
||
|
{
|
||
|
m_CEditRampUp.SetWindowText(mpt::ToCString(mpt::ufmt::val(MixerSettings().GetVolumeRampUpMicroseconds())));
|
||
|
m_CEditRampDown.SetWindowText(mpt::ToCString(mpt::ufmt::val(MixerSettings().GetVolumeRampDownMicroseconds())));
|
||
|
OnRampingChanged();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::OnHScroll(UINT n, UINT pos, CScrollBar *p)
|
||
|
{
|
||
|
CPropertyPage::OnHScroll(n, pos, p);
|
||
|
if(p == (CScrollBar *)&m_SliderStereoSep)
|
||
|
{
|
||
|
UpdateStereoSep();
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::UpdateRamping()
|
||
|
{
|
||
|
MixerSettings settings = TrackerSettings::Instance().GetMixerSettings();
|
||
|
CString s;
|
||
|
m_CEditRampUp.GetWindowText(s);
|
||
|
settings.SetVolumeRampUpMicroseconds(ConvertStrTo<int32>(s));
|
||
|
m_CEditRampDown.GetWindowText(s);
|
||
|
settings.SetVolumeRampDownMicroseconds(ConvertStrTo<int32>(s));
|
||
|
s.Format(_T("%i samples at %i Hz"), (int)settings.GetVolumeRampUpSamples(), (int)settings.gdwMixingFreq);
|
||
|
m_CInfoRampUp.SetWindowText(s);
|
||
|
s.Format(_T("%i samples at %i Hz"), (int)settings.GetVolumeRampDownSamples(), (int)settings.gdwMixingFreq);
|
||
|
m_CInfoRampDown.SetWindowText(s);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::UpdateStereoSep()
|
||
|
{
|
||
|
CString s;
|
||
|
s.Format(_T("%d%%"), ((8 * m_SliderStereoSep.GetPos()) * 100) / 128);
|
||
|
SetDlgItemText(IDC_TEXT_STEREOSEP, s);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsMixer::OnOK()
|
||
|
{
|
||
|
// resampler mode
|
||
|
{
|
||
|
TrackerSettings::Instance().ResamplerMode = static_cast<ResamplingMode>(m_CbnResampling.GetItemData(m_CbnResampling.GetCurSel()));
|
||
|
}
|
||
|
|
||
|
// Amiga Resampler
|
||
|
if(IsDlgButtonChecked(IDC_CHECK1) == BST_UNCHECKED)
|
||
|
TrackerSettings::Instance().ResamplerEmulateAmiga = Resampling::AmigaFilter::Off;
|
||
|
else
|
||
|
TrackerSettings::Instance().ResamplerEmulateAmiga = static_cast<Resampling::AmigaFilter>(m_CbnAmigaType.GetItemData(m_CbnAmigaType.GetCurSel()));
|
||
|
|
||
|
// volume ramping
|
||
|
{
|
||
|
MixerSettings settings = TrackerSettings::Instance().GetMixerSettings();
|
||
|
CString s;
|
||
|
m_CEditRampUp.GetWindowText(s);
|
||
|
settings.SetVolumeRampUpMicroseconds(ConvertStrTo<int>(s));
|
||
|
m_CEditRampDown.GetWindowText(s);
|
||
|
settings.SetVolumeRampDownMicroseconds(ConvertStrTo<int>(s));
|
||
|
TrackerSettings::Instance().SetMixerSettings(settings);
|
||
|
}
|
||
|
|
||
|
// stereo sep
|
||
|
{
|
||
|
TrackerSettings::Instance().MixerStereoSeparation = 8 * m_SliderStereoSep.GetPos();
|
||
|
}
|
||
|
|
||
|
// soft pan
|
||
|
{
|
||
|
if(IsDlgButtonChecked(IDC_CHECK_SOFTPAN))
|
||
|
{
|
||
|
TrackerSettings::Instance().MixerFlags = TrackerSettings::Instance().MixerFlags | SNDMIX_SOFTPANNING;
|
||
|
} else
|
||
|
{
|
||
|
TrackerSettings::Instance().MixerFlags = TrackerSettings::Instance().MixerFlags & ~SNDMIX_SOFTPANNING;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// pre amp
|
||
|
{
|
||
|
int n = m_SliderPreAmp.GetPos();
|
||
|
if ((n >= 0) && (n <= 40)) // approximately +/- 10dB
|
||
|
{
|
||
|
TrackerSettings::Instance().MixerPreAmp = 64 + (n * 8);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
CMainFrame::GetMainFrame()->SetupPlayer();
|
||
|
CMainFrame::GetMainFrame()->PostMessage(WM_MOD_INVALIDATEPATTERNS, HINT_MPTOPTIONS);
|
||
|
|
||
|
CPropertyPage::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// CEQSavePresetDlg
|
||
|
//
|
||
|
|
||
|
#ifndef NO_EQ
|
||
|
|
||
|
class CEQSavePresetDlg: public CDialog
|
||
|
{
|
||
|
protected:
|
||
|
EQPreset &m_EQ;
|
||
|
|
||
|
public:
|
||
|
CEQSavePresetDlg(EQPreset &eq, CWnd *parent = nullptr) : CDialog(IDD_SAVEPRESET, parent), m_EQ(eq) { }
|
||
|
BOOL OnInitDialog();
|
||
|
void OnOK();
|
||
|
};
|
||
|
|
||
|
|
||
|
BOOL CEQSavePresetDlg::OnInitDialog()
|
||
|
{
|
||
|
CComboBox *pCombo = (CComboBox *)GetDlgItem(IDC_COMBO1);
|
||
|
if (pCombo)
|
||
|
{
|
||
|
int ndx = 0;
|
||
|
for (UINT i=0; i<4; i++)
|
||
|
{
|
||
|
int n = pCombo->AddString(mpt::ToCString(mpt::Charset::Locale, TrackerSettings::Instance().m_EqUserPresets[i].szName));
|
||
|
pCombo->SetItemData( n, i);
|
||
|
if (!lstrcmpiA(TrackerSettings::Instance().m_EqUserPresets[i].szName, m_EQ.szName)) ndx = n;
|
||
|
}
|
||
|
pCombo->SetCurSel(ndx);
|
||
|
}
|
||
|
SetDlgItemText(IDC_EDIT1, mpt::ToCString(mpt::Charset::Locale, m_EQ.szName));
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CEQSavePresetDlg::OnOK()
|
||
|
{
|
||
|
CComboBox *pCombo = (CComboBox *)GetDlgItem(IDC_COMBO1);
|
||
|
if (pCombo)
|
||
|
{
|
||
|
int n = pCombo->GetCurSel();
|
||
|
if ((n < 0) || (n >= 4)) n = 0;
|
||
|
CString s;
|
||
|
GetDlgItemText(IDC_EDIT1, s);
|
||
|
mpt::String::WriteAutoBuf(m_EQ.szName) = mpt::ToCharset(mpt::Charset::Locale, s);
|
||
|
TrackerSettings::Instance().m_EqUserPresets[n] = m_EQ;
|
||
|
}
|
||
|
CDialog::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
void CEQSlider::Init(UINT nID, UINT n, CWnd *parent)
|
||
|
{
|
||
|
m_nSliderNo = n;
|
||
|
m_pParent = parent;
|
||
|
SubclassDlgItem(nID, parent);
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL CEQSlider::PreTranslateMessage(MSG *pMsg)
|
||
|
{
|
||
|
if ((pMsg) && (pMsg->message == WM_RBUTTONDOWN) && (m_pParent))
|
||
|
{
|
||
|
m_x = LOWORD(pMsg->lParam);
|
||
|
m_y = HIWORD(pMsg->lParam);
|
||
|
m_pParent->PostMessage(WM_COMMAND, ID_EQSLIDER_BASE+m_nSliderNo, 0);
|
||
|
}
|
||
|
return CSliderCtrl::PreTranslateMessage(pMsg);
|
||
|
}
|
||
|
|
||
|
#endif // !NO_EQ
|
||
|
|
||
|
|
||
|
//////////////////////////////////////////////////////////
|
||
|
// COptionsPlayer - DSP / EQ settings
|
||
|
|
||
|
|
||
|
#ifndef NO_EQ
|
||
|
#define EQ_MAX_FREQS 5
|
||
|
|
||
|
const UINT gEqBandFreqs[MAX_EQ_BANDS][EQ_MAX_FREQS] =
|
||
|
{
|
||
|
{ 100, 125, 150, 200, 250 },
|
||
|
{ 300, 350, 400, 450, 500 },
|
||
|
{ 600, 700, 800, 900, 1000 },
|
||
|
{ 1250, 1500, 1750, 2000, 2500 },
|
||
|
{ 3000, 3500, 4000, 4500, 5000 },
|
||
|
{ 6000, 7000, 8000, 9000, 10000 },
|
||
|
};
|
||
|
#endif // !NO_EQ
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(COptionsPlayer, CPropertyPage)
|
||
|
#ifndef NO_EQ
|
||
|
// EQ
|
||
|
ON_WM_VSCROLL()
|
||
|
ON_COMMAND(IDC_CHECK3, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_BUTTON1, &COptionsPlayer::OnEqUser1)
|
||
|
ON_COMMAND(IDC_BUTTON2, &COptionsPlayer::OnEqUser2)
|
||
|
ON_COMMAND(IDC_BUTTON3, &COptionsPlayer::OnEqUser3)
|
||
|
ON_COMMAND(IDC_BUTTON4, &COptionsPlayer::OnEqUser4)
|
||
|
ON_COMMAND(IDC_BUTTON5, &COptionsPlayer::OnSavePreset)
|
||
|
ON_COMMAND_RANGE(ID_EQSLIDER_BASE, ID_EQSLIDER_BASE + MAX_EQ_BANDS, &COptionsPlayer::OnSliderMenu)
|
||
|
ON_COMMAND_RANGE(ID_EQMENU_BASE, ID_EQMENU_BASE + EQ_MAX_FREQS, &COptionsPlayer::OnSliderFreq)
|
||
|
#endif // !NO_EQ
|
||
|
|
||
|
// DSP
|
||
|
ON_WM_HSCROLL()
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO2, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK1, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK2, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK4, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK5, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK6, &COptionsPlayer::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK7, &COptionsPlayer::OnSettingsChanged)
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
void COptionsPlayer::DoDataExchange(CDataExchange* pDX)
|
||
|
{
|
||
|
CPropertyPage::DoDataExchange(pDX);
|
||
|
//{{AFX_DATA_MAP(COptionsPlayer)
|
||
|
DDX_Control(pDX, IDC_COMBO2, m_CbnReverbPreset);
|
||
|
DDX_Control(pDX, IDC_SLIDER1, m_SbXBassDepth);
|
||
|
DDX_Control(pDX, IDC_SLIDER2, m_SbXBassRange);
|
||
|
DDX_Control(pDX, IDC_SLIDER3, m_SbReverbDepth);
|
||
|
DDX_Control(pDX, IDC_SLIDER5, m_SbSurroundDepth);
|
||
|
DDX_Control(pDX, IDC_SLIDER6, m_SbSurroundDelay);
|
||
|
DDX_Control(pDX, IDC_SLIDER4, m_SbBitCrushBits);
|
||
|
//}}AFX_DATA_MAP
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsPlayer::OnInitDialog()
|
||
|
{
|
||
|
CPropertyPage::OnInitDialog();
|
||
|
|
||
|
uint32 dwQuality = TrackerSettings::Instance().MixerDSPMask;
|
||
|
|
||
|
#ifndef NO_EQ
|
||
|
for (UINT i = 0; i < MAX_EQ_BANDS; i++)
|
||
|
{
|
||
|
m_Sliders[i].Init(IDC_SLIDER7 + i, i, this);
|
||
|
m_Sliders[i].SetRange(0, 32);
|
||
|
m_Sliders[i].SetTicFreq(4);
|
||
|
}
|
||
|
|
||
|
UpdateDialog();
|
||
|
|
||
|
if (dwQuality & SNDDSP_EQ) CheckDlgButton(IDC_CHECK3, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK3)->EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
// Effects
|
||
|
#ifndef NO_DSP
|
||
|
if (dwQuality & SNDDSP_MEGABASS) CheckDlgButton(IDC_CHECK1, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK1)->EnableWindow(FALSE);
|
||
|
#endif
|
||
|
#ifndef NO_AGC
|
||
|
if (dwQuality & SNDDSP_AGC) CheckDlgButton(IDC_CHECK2, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK2)->EnableWindow(FALSE);
|
||
|
#endif
|
||
|
#ifndef NO_DSP
|
||
|
if (dwQuality & SNDDSP_SURROUND) CheckDlgButton(IDC_CHECK4, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK4)->EnableWindow(FALSE);
|
||
|
#endif
|
||
|
#ifndef NO_DSP
|
||
|
if (dwQuality & SNDDSP_BITCRUSH) CheckDlgButton(IDC_CHECK5, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK5)->EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
m_SbBitCrushBits.SetRange(1, 24);
|
||
|
m_SbBitCrushBits.SetPos(TrackerSettings::Instance().m_BitCrushSettings.m_Bits);
|
||
|
#else
|
||
|
m_SbBitCurshBits.EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
// Bass Expansion
|
||
|
m_SbXBassDepth.SetRange(0,4);
|
||
|
m_SbXBassDepth.SetPos(8-TrackerSettings::Instance().m_MegaBassSettings.m_nXBassDepth);
|
||
|
m_SbXBassRange.SetRange(0,4);
|
||
|
m_SbXBassRange.SetPos(4 - (TrackerSettings::Instance().m_MegaBassSettings.m_nXBassRange - 1) / 5);
|
||
|
#else
|
||
|
m_SbXBassDepth.EnableWindow(FALSE);
|
||
|
m_SbXBassRange.EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_REVERB
|
||
|
// Reverb
|
||
|
m_SbReverbDepth.SetRange(1, 16);
|
||
|
m_SbReverbDepth.SetPos(TrackerSettings::Instance().m_ReverbSettings.m_nReverbDepth);
|
||
|
UINT nSel = 0;
|
||
|
for (UINT iRvb=0; iRvb<NUM_REVERBTYPES; iRvb++)
|
||
|
{
|
||
|
CString pszName = mpt::ToCString(GetReverbPresetName(iRvb));
|
||
|
if(!pszName.IsEmpty())
|
||
|
{
|
||
|
UINT n = m_CbnReverbPreset.AddString(pszName);
|
||
|
m_CbnReverbPreset.SetItemData(n, iRvb);
|
||
|
if (iRvb == TrackerSettings::Instance().m_ReverbSettings.m_nReverbType) nSel = n;
|
||
|
}
|
||
|
}
|
||
|
m_CbnReverbPreset.SetCurSel(nSel);
|
||
|
if (dwQuality & SNDDSP_REVERB) CheckDlgButton(IDC_CHECK6, BST_CHECKED);
|
||
|
#else
|
||
|
GetDlgItem(IDC_CHECK6)->EnableWindow(FALSE);
|
||
|
m_SbReverbDepth.EnableWindow(FALSE);
|
||
|
m_CbnReverbPreset.EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
// Surround
|
||
|
{
|
||
|
UINT n = TrackerSettings::Instance().m_SurroundSettings.m_nProLogicDepth;
|
||
|
if (n < 1) n = 1;
|
||
|
if (n > 16) n = 16;
|
||
|
m_SbSurroundDepth.SetRange(1, 16);
|
||
|
m_SbSurroundDepth.SetPos(n);
|
||
|
m_SbSurroundDelay.SetRange(0, 8);
|
||
|
m_SbSurroundDelay.SetPos((TrackerSettings::Instance().m_SurroundSettings.m_nProLogicDelay-5)/5);
|
||
|
}
|
||
|
#else
|
||
|
m_SbSurroundDepth.EnableWindow(FALSE);
|
||
|
m_SbSurroundDelay.EnableWindow(FALSE);
|
||
|
#endif
|
||
|
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsPlayer::OnSetActive()
|
||
|
{
|
||
|
CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_PLAYER;
|
||
|
|
||
|
SetDlgItemText(IDC_EQ_WARNING,
|
||
|
_T("Note: This EQ is applied to any and all of the modules ")
|
||
|
_T("that you load in OpenMPT; its settings are stored globally, ")
|
||
|
_T("rather than in each file. This means that you should avoid ")
|
||
|
_T("using it as part of your production process, and instead only ")
|
||
|
_T("use it to correct deficiencies in your audio hardware."));
|
||
|
|
||
|
return CPropertyPage::OnSetActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnHScroll(UINT nSBCode, UINT, CScrollBar *psb)
|
||
|
{
|
||
|
if (nSBCode == SB_ENDSCROLL) return;
|
||
|
if ((psb) && (psb->m_hWnd == m_SbReverbDepth.m_hWnd))
|
||
|
{
|
||
|
#ifndef NO_REVERB
|
||
|
UINT n = m_SbReverbDepth.GetPos();
|
||
|
if ((n) && (n <= 16)) TrackerSettings::Instance().m_ReverbSettings.m_nReverbDepth = n;
|
||
|
//if ((n) && (n <= 16)) CSoundFile::m_Reverb.m_Settings.m_nReverbDepth = n;
|
||
|
CMainFrame::GetMainFrame()->SetupPlayer();
|
||
|
#endif
|
||
|
} else
|
||
|
{
|
||
|
OnSettingsChanged();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnOK()
|
||
|
{
|
||
|
DWORD dwQuality = 0;
|
||
|
|
||
|
DWORD dwQualityMask = 0;
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
dwQualityMask |= SNDDSP_MEGABASS;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK1)) dwQuality |= SNDDSP_MEGABASS;
|
||
|
#endif
|
||
|
#ifndef NO_AGC
|
||
|
dwQualityMask |= SNDDSP_AGC;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK2)) dwQuality |= SNDDSP_AGC;
|
||
|
#endif
|
||
|
#ifndef NO_EQ
|
||
|
dwQualityMask |= SNDDSP_EQ;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK3)) dwQuality |= SNDDSP_EQ;
|
||
|
#endif
|
||
|
#ifndef NO_DSP
|
||
|
dwQualityMask |= SNDDSP_SURROUND;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK4)) dwQuality |= SNDDSP_SURROUND;
|
||
|
#endif
|
||
|
#ifndef NO_REVERB
|
||
|
dwQualityMask |= SNDDSP_REVERB;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK6)) dwQuality |= SNDDSP_REVERB;
|
||
|
#endif
|
||
|
#ifndef NO_DSP
|
||
|
dwQualityMask |= SNDDSP_BITCRUSH;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK5)) dwQuality |= SNDDSP_BITCRUSH;
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
{
|
||
|
TrackerSettings::Instance().m_BitCrushSettings.m_Bits = m_SbBitCrushBits.GetPos();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#ifndef NO_DSP
|
||
|
// Bass Expansion
|
||
|
{
|
||
|
UINT nXBassDepth = 8-m_SbXBassDepth.GetPos();
|
||
|
if (nXBassDepth < 4) nXBassDepth = 4;
|
||
|
if (nXBassDepth > 8) nXBassDepth = 8;
|
||
|
UINT nXBassRange = (4-m_SbXBassRange.GetPos()) * 5 + 1;
|
||
|
if (nXBassRange < 5) nXBassRange = 5;
|
||
|
if (nXBassRange > 21) nXBassRange = 21;
|
||
|
TrackerSettings::Instance().m_MegaBassSettings.m_nXBassDepth = nXBassDepth;
|
||
|
TrackerSettings::Instance().m_MegaBassSettings.m_nXBassRange = nXBassRange;
|
||
|
}
|
||
|
#endif
|
||
|
#ifndef NO_REVERB
|
||
|
// Reverb
|
||
|
{
|
||
|
// Reverb depth is dynamically changed
|
||
|
uint32 nReverbType = static_cast<uint32>(m_CbnReverbPreset.GetItemData(m_CbnReverbPreset.GetCurSel()));
|
||
|
if (nReverbType < NUM_REVERBTYPES) TrackerSettings::Instance().m_ReverbSettings.m_nReverbType = nReverbType;
|
||
|
}
|
||
|
#endif
|
||
|
#ifndef NO_DSP
|
||
|
// Surround
|
||
|
{
|
||
|
UINT nProLogicDepth = m_SbSurroundDepth.GetPos();
|
||
|
UINT nProLogicDelay = 5 + (m_SbSurroundDelay.GetPos() * 5);
|
||
|
TrackerSettings::Instance().m_SurroundSettings.m_nProLogicDepth = nProLogicDepth;
|
||
|
TrackerSettings::Instance().m_SurroundSettings.m_nProLogicDelay = nProLogicDelay;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
TrackerSettings::Instance().MixerDSPMask = dwQuality;
|
||
|
|
||
|
CMainFrame::GetMainFrame()->SetupPlayer();
|
||
|
CPropertyPage::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifndef NO_EQ
|
||
|
|
||
|
void COptionsPlayer::UpdateEQ(bool bReset)
|
||
|
{
|
||
|
CriticalSection cs;
|
||
|
if(CMainFrame::GetMainFrame()->GetSoundFilePlaying())
|
||
|
CMainFrame::GetMainFrame()->GetSoundFilePlaying()->SetEQGains(m_EQPreset.Gains, m_EQPreset.Freqs, bReset);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnVScroll(UINT nSBCode, UINT nPos, CScrollBar *pScrollBar)
|
||
|
{
|
||
|
CDialog::OnVScroll(nSBCode, nPos, pScrollBar);
|
||
|
for (UINT i=0; i<MAX_EQ_BANDS; i++)
|
||
|
{
|
||
|
int n = 32 - m_Sliders[i].GetPos();
|
||
|
if ((n >= 0) && (n <= 32)) m_EQPreset.Gains[i] = n;
|
||
|
}
|
||
|
UpdateEQ(FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::LoadEQPreset(const EQPreset &preset)
|
||
|
{
|
||
|
m_EQPreset = preset;
|
||
|
UpdateEQ(TRUE);
|
||
|
UpdateDialog();
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnSavePreset()
|
||
|
{
|
||
|
CEQSavePresetDlg dlg(m_EQPreset, this);
|
||
|
if (dlg.DoModal() == IDOK)
|
||
|
{
|
||
|
UpdateDialog();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static CString f2s(UINT f)
|
||
|
{
|
||
|
if (f < 1000)
|
||
|
{
|
||
|
return MPT_CFORMAT("{}Hz")(f);
|
||
|
} else
|
||
|
{
|
||
|
UINT fHi = f / 1000u;
|
||
|
UINT fLo = f % 1000u;
|
||
|
if(fLo)
|
||
|
{
|
||
|
return MPT_CFORMAT("{}.{}kHz")(fHi, mpt::cfmt::dec0<1>(fLo/100));
|
||
|
} else
|
||
|
{
|
||
|
return MPT_CFORMAT("{}kHz")(fHi);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::UpdateDialog()
|
||
|
{
|
||
|
for (UINT i=0; i<MAX_EQ_BANDS; i++)
|
||
|
{
|
||
|
int n = 32 - m_EQPreset.Gains[i];
|
||
|
if (n < 0) n = 0;
|
||
|
if (n > 32) n = 32;
|
||
|
if (n != (m_Sliders[i].GetPos() & 0xFFFF)) m_Sliders[i].SetPos(n);
|
||
|
SetDlgItemText(IDC_TEXT1 + i, f2s(m_EQPreset.Freqs[i]));
|
||
|
}
|
||
|
for(unsigned int i = 0; i < std::size(TrackerSettings::Instance().m_EqUserPresets); i++)
|
||
|
{
|
||
|
SetDlgItemText(IDC_BUTTON1 + i, mpt::ToCString(mpt::Charset::Locale, TrackerSettings::Instance().m_EqUserPresets[i].szName));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnSliderMenu(UINT nID)
|
||
|
{
|
||
|
UINT n = nID - ID_EQSLIDER_BASE;
|
||
|
if (n < MAX_EQ_BANDS)
|
||
|
{
|
||
|
HMENU hMenu = ::CreatePopupMenu();
|
||
|
m_nSliderMenu = n;
|
||
|
if (!hMenu) return;
|
||
|
const UINT *pFreqs = gEqBandFreqs[m_nSliderMenu];
|
||
|
for (UINT i = 0; i < EQ_MAX_FREQS; i++)
|
||
|
{
|
||
|
DWORD d = MF_STRING;
|
||
|
if (m_EQPreset.Freqs[m_nSliderMenu] == pFreqs[i]) d |= MF_CHECKED;
|
||
|
::AppendMenu(hMenu, d, ID_EQMENU_BASE+i, f2s(pFreqs[i]));
|
||
|
}
|
||
|
CPoint pt(m_Sliders[m_nSliderMenu].m_x, m_Sliders[m_nSliderMenu].m_y);
|
||
|
m_Sliders[m_nSliderMenu].ClientToScreen(&pt);
|
||
|
::TrackPopupMenu(hMenu, TPM_LEFTALIGN|TPM_RIGHTBUTTON, pt.x, pt.y, 0, m_hWnd, NULL);
|
||
|
::DestroyMenu(hMenu);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsPlayer::OnSliderFreq(UINT nID)
|
||
|
{
|
||
|
UINT n = nID - ID_EQMENU_BASE;
|
||
|
if ((m_nSliderMenu < MAX_EQ_BANDS) && (n < EQ_MAX_FREQS))
|
||
|
{
|
||
|
UINT f = gEqBandFreqs[m_nSliderMenu][n];
|
||
|
if (f != m_EQPreset.Freqs[m_nSliderMenu])
|
||
|
{
|
||
|
m_EQPreset.Freqs[m_nSliderMenu] = f;
|
||
|
UpdateEQ(TRUE);
|
||
|
UpdateDialog();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#endif // !NO_EQ
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////
|
||
|
// CMidiSetupDlg
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(CMidiSetupDlg, CPropertyPage)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO1, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO2, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO3, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_BUTTON1, &CMidiSetupDlg::OnRenameDevice)
|
||
|
ON_COMMAND(IDC_CHECK1, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK2, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK3, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK4, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_CHECK5, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_MIDI_TO_PLUGIN, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_MIDI_MACRO_CONTROL, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_MIDIVOL_TO_NOTEVOL, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_MIDIPLAYCONTROL, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_COMMAND(IDC_MIDIPLAYPATTERNONMIDIIN, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_EN_CHANGE(IDC_EDIT1, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_EN_CHANGE(IDC_EDIT2, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_EN_CHANGE(IDC_EDIT3, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
ON_EN_CHANGE(IDC_EDIT4, &CMidiSetupDlg::OnSettingsChanged)
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
void CMidiSetupDlg::DoDataExchange(CDataExchange* pDX)
|
||
|
{
|
||
|
CPropertyPage::DoDataExchange(pDX);
|
||
|
//{{AFX_DATA_MAP(COptionsSoundcard)
|
||
|
DDX_Control(pDX, IDC_SPIN1, m_SpinSpd);
|
||
|
DDX_Control(pDX, IDC_SPIN2, m_SpinPat);
|
||
|
DDX_Control(pDX, IDC_SPIN3, m_SpinAmp);
|
||
|
DDX_Control(pDX, IDC_COMBO1, m_InputDevice);
|
||
|
DDX_Control(pDX, IDC_COMBO2, m_ATBehaviour);
|
||
|
DDX_Control(pDX, IDC_COMBO3, m_Quantize);
|
||
|
//}}AFX_DATA_MAP
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL CMidiSetupDlg::OnInitDialog()
|
||
|
{
|
||
|
CPropertyPage::OnInitDialog();
|
||
|
// Flags
|
||
|
if (m_dwMidiSetup & MIDISETUP_RECORDVELOCITY) CheckDlgButton(IDC_CHECK1, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_RECORDNOTEOFF) CheckDlgButton(IDC_CHECK2, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_ENABLE_RECORD_DEFAULT) CheckDlgButton(IDC_CHECK3, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_TRANSPOSEKEYBOARD) CheckDlgButton(IDC_CHECK4, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_MIDITOPLUG) CheckDlgButton(IDC_MIDI_TO_PLUGIN, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_MIDIMACROCONTROL) CheckDlgButton(IDC_MIDI_MACRO_CONTROL, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_MIDIVOL_TO_NOTEVOL) CheckDlgButton(IDC_MIDIVOL_TO_NOTEVOL, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_RESPONDTOPLAYCONTROLMSGS) CheckDlgButton(IDC_MIDIPLAYCONTROL, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_PLAYPATTERNONMIDIIN) CheckDlgButton(IDC_MIDIPLAYPATTERNONMIDIIN, BST_CHECKED);
|
||
|
if (m_dwMidiSetup & MIDISETUP_MIDIMACROPITCHBEND) CheckDlgButton(IDC_CHECK5, BST_CHECKED);
|
||
|
|
||
|
// Midi In Device
|
||
|
RefreshDeviceList(m_nMidiDevice);
|
||
|
|
||
|
// Aftertouch behaviour
|
||
|
m_ATBehaviour.ResetContent();
|
||
|
static constexpr std::pair<const TCHAR *, RecordAftertouchOptions> aftertouchOptions[] =
|
||
|
{
|
||
|
{ _T("Do not record Aftertouch"), atDoNotRecord },
|
||
|
{ _T("Record as Volume Commands"), atRecordAsVolume },
|
||
|
{ _T("Record as MIDI Macros"), atRecordAsMacro },
|
||
|
};
|
||
|
|
||
|
for(const auto & [str, value] : aftertouchOptions)
|
||
|
{
|
||
|
int item = m_ATBehaviour.AddString(str);
|
||
|
m_ATBehaviour.SetItemData(item, value);
|
||
|
if(value == TrackerSettings::Instance().aftertouchBehaviour)
|
||
|
{
|
||
|
m_ATBehaviour.SetCurSel(item);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Note Velocity amp
|
||
|
SetDlgItemInt(IDC_EDIT3, TrackerSettings::Instance().midiVelocityAmp);
|
||
|
m_SpinAmp.SetRange(1, 10000);
|
||
|
|
||
|
SetDlgItemText(IDC_EDIT4, mpt::ToCString(IgnoredCCsToString(TrackerSettings::Instance().midiIgnoreCCs)));
|
||
|
|
||
|
// Midi Import settings
|
||
|
SetDlgItemInt(IDC_EDIT1, TrackerSettings::Instance().midiImportTicks);
|
||
|
SetDlgItemInt(IDC_EDIT2, TrackerSettings::Instance().midiImportPatternLen);
|
||
|
|
||
|
// Note quantization
|
||
|
m_Quantize.ResetContent();
|
||
|
static constexpr std::pair<const TCHAR *, uint32> quantizeOptions[] =
|
||
|
{
|
||
|
{ _T("1/4th Notes"), 4 }, { _T("1/6th Notes"), 6 },
|
||
|
{ _T("1/8th Notes"), 8 }, { _T("1/12th Notes"), 12 },
|
||
|
{ _T("1/16th Notes"), 16 }, { _T("1/24th Notes"), 24 },
|
||
|
{ _T("1/32nd Notes"), 32 }, { _T("1/48th Notes"), 48 },
|
||
|
{ _T("1/64th Notes"), 64 }, { _T("1/96th Notes"), 96 },
|
||
|
};
|
||
|
|
||
|
for(const auto & [str, value]: quantizeOptions)
|
||
|
{
|
||
|
int item = m_Quantize.AddString(str);
|
||
|
m_Quantize.SetItemData(item, value);
|
||
|
if(value == TrackerSettings::Instance().midiImportQuantize)
|
||
|
{
|
||
|
m_Quantize.SetCurSel(item);
|
||
|
}
|
||
|
}
|
||
|
m_SpinSpd.SetRange(2, 16);
|
||
|
m_SpinPat.SetRange(1, MAX_PATTERN_ROWS);
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void CMidiSetupDlg::RefreshDeviceList(UINT currentDevice)
|
||
|
{
|
||
|
m_InputDevice.SetRedraw(FALSE);
|
||
|
m_InputDevice.ResetContent();
|
||
|
UINT ndevs = midiInGetNumDevs();
|
||
|
for(UINT i = 0; i < ndevs; i++)
|
||
|
{
|
||
|
MIDIINCAPS mic;
|
||
|
mic.szPname[0] = 0;
|
||
|
if(midiInGetDevCaps(i, &mic, sizeof(mic)) == MMSYSERR_NOERROR)
|
||
|
{
|
||
|
int item = m_InputDevice.AddString(theApp.GetFriendlyMIDIPortName(mpt::ToCString(mpt::String::ReadWinBuf(mic.szPname)), true));
|
||
|
m_InputDevice.SetItemData(item, i);
|
||
|
if(i == currentDevice)
|
||
|
{
|
||
|
m_InputDevice.SetCurSel(item);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
m_InputDevice.SetRedraw(TRUE);
|
||
|
m_InputDevice.Invalidate(FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
void CMidiSetupDlg::OnRenameDevice()
|
||
|
{
|
||
|
int n = m_InputDevice.GetCurSel();
|
||
|
if(n >= 0)
|
||
|
{
|
||
|
UINT device = static_cast<UINT>(m_InputDevice.GetItemData(n));
|
||
|
MIDIINCAPS mic;
|
||
|
mic.szPname[0] = 0;
|
||
|
midiInGetDevCaps(device, &mic, sizeof(mic));
|
||
|
CString name = mic.szPname;
|
||
|
CString friendlyName = theApp.GetSettings().Read(U_("MIDI Input Ports"), mpt::ToUnicode(name), name);
|
||
|
CInputDlg dlg(this, _T("New name for ") + name + _T(":"), friendlyName);
|
||
|
if(dlg.DoModal() == IDOK)
|
||
|
{
|
||
|
if(dlg.resultAsString.IsEmpty() || dlg.resultAsString == name)
|
||
|
theApp.GetSettings().Remove(U_("MIDI Input Ports"), mpt::ToUnicode(name));
|
||
|
else
|
||
|
theApp.GetSettings().Write(U_("MIDI Input Ports"), mpt::ToUnicode(name), dlg.resultAsString);
|
||
|
RefreshDeviceList(device);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void CMidiSetupDlg::OnOK()
|
||
|
{
|
||
|
CMainFrame *pMainFrm = CMainFrame::GetMainFrame();
|
||
|
m_dwMidiSetup = 0;
|
||
|
m_nMidiDevice = MIDI_MAPPER;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK1)) m_dwMidiSetup |= MIDISETUP_RECORDVELOCITY;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK2)) m_dwMidiSetup |= MIDISETUP_RECORDNOTEOFF;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK3)) m_dwMidiSetup |= MIDISETUP_ENABLE_RECORD_DEFAULT;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK4)) m_dwMidiSetup |= MIDISETUP_TRANSPOSEKEYBOARD;
|
||
|
if (IsDlgButtonChecked(IDC_MIDI_TO_PLUGIN)) m_dwMidiSetup |= MIDISETUP_MIDITOPLUG;
|
||
|
if (IsDlgButtonChecked(IDC_MIDI_MACRO_CONTROL)) m_dwMidiSetup |= MIDISETUP_MIDIMACROCONTROL;
|
||
|
if (IsDlgButtonChecked(IDC_MIDIVOL_TO_NOTEVOL)) m_dwMidiSetup |= MIDISETUP_MIDIVOL_TO_NOTEVOL;
|
||
|
if (IsDlgButtonChecked(IDC_MIDIPLAYCONTROL)) m_dwMidiSetup |= MIDISETUP_RESPONDTOPLAYCONTROLMSGS;
|
||
|
if (IsDlgButtonChecked(IDC_MIDIPLAYPATTERNONMIDIIN)) m_dwMidiSetup |= MIDISETUP_PLAYPATTERNONMIDIIN;
|
||
|
if (IsDlgButtonChecked(IDC_CHECK5)) m_dwMidiSetup |= MIDISETUP_MIDIMACROPITCHBEND;
|
||
|
|
||
|
int n = m_InputDevice.GetCurSel();
|
||
|
if (n >= 0) m_nMidiDevice = static_cast<UINT>(m_InputDevice.GetItemData(n));
|
||
|
|
||
|
TrackerSettings::Instance().aftertouchBehaviour = static_cast<RecordAftertouchOptions>(m_ATBehaviour.GetItemData(m_ATBehaviour.GetCurSel()));
|
||
|
|
||
|
TrackerSettings::Instance().midiVelocityAmp = static_cast<uint16>(Clamp(GetDlgItemInt(IDC_EDIT3), 1u, 10000u));
|
||
|
|
||
|
CString cc;
|
||
|
GetDlgItemText(IDC_EDIT4, cc);
|
||
|
TrackerSettings::Instance().midiIgnoreCCs = StringToIgnoredCCs(mpt::ToUnicode(cc));
|
||
|
|
||
|
TrackerSettings::Instance().midiImportTicks = static_cast<uint8>(Clamp(GetDlgItemInt(IDC_EDIT1), uint8(2), uint8(16)));
|
||
|
TrackerSettings::Instance().midiImportPatternLen = Clamp(GetDlgItemInt(IDC_EDIT2), ROWINDEX(1), MAX_PATTERN_ROWS);
|
||
|
if(m_Quantize.GetCurSel() != -1)
|
||
|
{
|
||
|
TrackerSettings::Instance().midiImportQuantize = static_cast<uint32>(m_Quantize.GetItemData(m_Quantize.GetCurSel()));
|
||
|
}
|
||
|
|
||
|
if (pMainFrm) pMainFrm->SetupMidi(m_dwMidiSetup, m_nMidiDevice);
|
||
|
CPropertyPage::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL CMidiSetupDlg::OnSetActive()
|
||
|
{
|
||
|
CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_MIDI;
|
||
|
return CPropertyPage::OnSetActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
// Wine
|
||
|
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(COptionsWine, CPropertyPage)
|
||
|
ON_COMMAND(IDC_CHECK_WINE_ENABLE, &COptionsWine::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_WINE_PULSEAUDIO, &COptionsWine::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_WINE_PORTAUDIO, &COptionsWine::OnSettingsChanged)
|
||
|
ON_CBN_SELCHANGE(IDC_COMBO_WINE_RTAUDIO, &COptionsWine::OnSettingsChanged)
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
COptionsWine::COptionsWine()
|
||
|
: CPropertyPage(IDD_OPTIONS_WINE)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsWine::DoDataExchange(CDataExchange* pDX)
|
||
|
{
|
||
|
CPropertyPage::DoDataExchange(pDX);
|
||
|
//{{AFX_DATA_MAP(COptionsWine)
|
||
|
DDX_Control(pDX, IDC_COMBO_WINE_PULSEAUDIO, m_CbnPulseAudio);
|
||
|
DDX_Control(pDX, IDC_COMBO_WINE_PORTAUDIO, m_CbnPortAudio);
|
||
|
DDX_Control(pDX, IDC_COMBO_WINE_RTAUDIO, m_CbnRtAudio);
|
||
|
//}}AFX_DATA_MAP
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsWine::OnInitDialog()
|
||
|
{
|
||
|
CPropertyPage::OnInitDialog();
|
||
|
GetDlgItem(IDC_CHECK_WINE_ENABLE)->EnableWindow(mpt::OS::Windows::IsWine() ? TRUE : FALSE);
|
||
|
CheckDlgButton(IDC_CHECK_WINE_ENABLE, TrackerSettings::Instance().WineSupportEnabled ? BST_CHECKED : BST_UNCHECKED);
|
||
|
int index;
|
||
|
index = m_CbnPulseAudio.AddString(_T("Auto" )); m_CbnPulseAudio.SetItemData(index, 1);
|
||
|
index = m_CbnPulseAudio.AddString(_T("Enabled" )); m_CbnPulseAudio.SetItemData(index, 2);
|
||
|
index = m_CbnPulseAudio.AddString(_T("Disabled")); m_CbnPulseAudio.SetItemData(index, 0);
|
||
|
m_CbnPulseAudio.SetCurSel(0);
|
||
|
for(index = 0; index < 3; ++index)
|
||
|
{
|
||
|
if(m_CbnPulseAudio.GetItemData(index) == static_cast<uint32>(TrackerSettings::Instance().WineSupportEnablePulseAudio))
|
||
|
{
|
||
|
m_CbnPulseAudio.SetCurSel(index);
|
||
|
}
|
||
|
}
|
||
|
index = m_CbnPortAudio.AddString(_T("Auto" )); m_CbnPortAudio.SetItemData(index, 1);
|
||
|
index = m_CbnPortAudio.AddString(_T("Enabled" )); m_CbnPortAudio.SetItemData(index, 2);
|
||
|
index = m_CbnPortAudio.AddString(_T("Disabled")); m_CbnPortAudio.SetItemData(index, 0);
|
||
|
for(index = 0; index < 3; ++index)
|
||
|
{
|
||
|
if(m_CbnPortAudio.GetItemData(index) == static_cast<uint32>(TrackerSettings::Instance().WineSupportEnablePortAudio))
|
||
|
{
|
||
|
m_CbnPortAudio.SetCurSel(index);
|
||
|
}
|
||
|
}
|
||
|
index = m_CbnRtAudio.AddString(_T("Auto" )); m_CbnRtAudio.SetItemData(index, 1);
|
||
|
index = m_CbnRtAudio.AddString(_T("Enabled" )); m_CbnRtAudio.SetItemData(index, 2);
|
||
|
index = m_CbnRtAudio.AddString(_T("Disabled")); m_CbnRtAudio.SetItemData(index, 0);
|
||
|
for(index = 0; index < 3; ++index)
|
||
|
{
|
||
|
if(m_CbnRtAudio.GetItemData(index) == static_cast<uint32>(TrackerSettings::Instance().WineSupportEnableRtAudio))
|
||
|
{
|
||
|
m_CbnRtAudio.SetCurSel(index);
|
||
|
}
|
||
|
}
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsWine::OnSettingsChanged()
|
||
|
{
|
||
|
SetModified(TRUE);
|
||
|
}
|
||
|
|
||
|
|
||
|
void COptionsWine::OnOK()
|
||
|
{
|
||
|
TrackerSettings::Instance().WineSupportEnabled = IsDlgButtonChecked(IDC_CHECK_WINE_ENABLE) ? true : false;
|
||
|
TrackerSettings::Instance().WineSupportEnablePulseAudio = static_cast<int32>(m_CbnPulseAudio.GetItemData(m_CbnPulseAudio.GetCurSel()));
|
||
|
TrackerSettings::Instance().WineSupportEnablePortAudio = static_cast<int32>(m_CbnPortAudio.GetItemData(m_CbnPortAudio.GetCurSel()));
|
||
|
TrackerSettings::Instance().WineSupportEnableRtAudio = static_cast<int32>(m_CbnRtAudio.GetItemData(m_CbnRtAudio.GetCurSel()));
|
||
|
CPropertyPage::OnOK();
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL COptionsWine::OnSetActive()
|
||
|
{
|
||
|
CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_WINE;
|
||
|
return CPropertyPage::OnSetActive();
|
||
|
}
|
||
|
|
||
|
|
||
|
OPENMPT_NAMESPACE_END
|