/* * AdvancedConfigDlg.cpp * --------------------- * Purpose: Implementation of the advanced settings dialog. * 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 "Mainfrm.h" #include "AdvancedConfigDlg.h" #include "Settings.h" #include "dlg_misc.h" OPENMPT_NAMESPACE_BEGIN BEGIN_MESSAGE_MAP(COptionsAdvanced, CPropertyPage) ON_NOTIFY(NM_DBLCLK, IDC_LIST1, &COptionsAdvanced::OnOptionDblClick) #ifndef MPT_MFC_FULL ON_NOTIFY(NM_CUSTOMDRAW, IDC_LIST1, &COptionsAdvanced::OnCustomDrawList) #endif ON_EN_CHANGE(IDC_EDIT1, &COptionsAdvanced::OnFindStringChanged) ON_COMMAND(IDC_BUTTON1, &COptionsAdvanced::OnSaveNow) END_MESSAGE_MAP() void COptionsAdvanced::DoDataExchange(CDataExchange* pDX) { CDialog::DoDataExchange(pDX); //{{AFX_DATA_MAP(CModTypeDlg) DDX_Control(pDX, IDC_LIST1, m_List); //}}AFX_DATA_MAP } BOOL COptionsAdvanced::PreTranslateMessage(MSG *msg) { if(msg->message == WM_KEYDOWN && msg->wParam == VK_RETURN) { OnOptionDblClick(nullptr, nullptr); return TRUE; } return FALSE; } BOOL COptionsAdvanced::OnInitDialog() { CPropertyPage::OnInitDialog(); m_List.SetExtendedStyle(m_List.GetExtendedStyle() | LVS_EX_GRIDLINES | LVS_EX_FULLROWSELECT); ListView_EnableGroupView(m_List.m_hWnd, FALSE); // try to set known state int enableGroupsResult1 = static_cast(ListView_EnableGroupView(m_List.m_hWnd, TRUE)); int enableGroupsResult2 = static_cast(ListView_EnableGroupView(m_List.m_hWnd, TRUE)); // Looks like we have to check enabling and check that a second enabling does // not change anything. // Just checking if enabling fails with -1 does not work for older control // versions because they just do not know the window message at all and return // 0, always. At least Wine does behave this way. if(enableGroupsResult1 == 1 && enableGroupsResult2 == 0) { m_listGrouped = true; } else { // Did not behave as documented or expected, the actual state of the // control is unknown by now. // Play safe and set and assume the traditional ungrouped mode again. ListView_EnableGroupView(m_List.m_hWnd, FALSE); m_listGrouped = false; } if(m_listGrouped) { static constexpr ListCtrl::Header headers[] = { { _T("Setting"), 150, LVCFMT_LEFT }, { _T("Type"), 40, LVCFMT_LEFT }, { _T("Value"), 140, LVCFMT_LEFT }, { _T("Default"), 62, LVCFMT_LEFT }, }; m_List.SetHeaders(headers); } else { static constexpr ListCtrl::Header headers[] = { { _T("Setting"), 200, LVCFMT_LEFT }, { _T("Type"), 40, LVCFMT_LEFT }, { _T("Value"), 100, LVCFMT_LEFT }, { _T("Default"), 52, LVCFMT_LEFT }, }; m_List.SetHeaders(headers); } ReInit(); return TRUE; } void COptionsAdvanced::ReInit() { m_List.SetRedraw(FALSE); m_List.DeleteAllItems(); if(m_listGrouped) { ListView_RemoveAllGroups(m_List.m_hWnd); } m_List.SetItemCount(static_cast(theApp.GetSettings().size())); m_indexToPath.clear(); m_indexToPath.reserve(theApp.GetSettings().size()); m_groups.clear(); int numGroups = 0; mpt::ustring findStr = mpt::ToLowerCase(GetWindowTextUnicode(*GetDlgItem(IDC_EDIT1))); LVITEMW lvi; lvi.mask = LVIF_TEXT | LVIF_PARAM; lvi.mask |= (m_listGrouped ? LVIF_GROUPID : 0); lvi.iSubItem = 0; lvi.state = 0; lvi.stateMask = 0; lvi.cchTextMax = 0; lvi.iImage = 0; lvi.iIndent = 0; lvi.iGroupId = 0; int i = 0; for(const auto &[path, state] : theApp.GetSettings()) { // In MPT_USTRING_MODE_WIDE mode, // this loop is heavily optimized to avoid as much string copies as possible // in order to perform ok-ish in debug builds. // MPT_USTRING_MODE_UTF8 is not optimized as we (currently) do not build in // this mode by default. const mpt::ustring §ion = path.GetRefSection(); const mpt::ustring &key = path.GetRefKey(); const SettingValue &value = state.GetRefValue(); const SettingValue &defaultValue = state.GetRefDefault(); if(!findStr.empty()) { mpt::ustring str = path.FormatAsString() + U_("=") + value.FormatValueAsString(); str = mpt::ToLowerCase(str); if(str.find(findStr) == mpt::ustring::npos) { continue; } } int index; lvi.iItem = i++; lvi.lParam = m_indexToPath.size(); if(m_listGrouped) { auto gi = m_groups.find(section); if(gi == m_groups.end()) { LVGROUP group; #if _WIN32_WINNT >= 0x0600 group.cbSize = LVGROUP_V5_SIZE; #else group.cbSize = sizeof(group); #endif group.mask = LVGF_HEADER | LVGF_GROUPID; #if MPT_USTRING_MODE_WIDE group.pszHeader = const_cast(section.c_str()); #else const std::wstring wsection = mpt::ToWide(section); group.pszHeader = const_cast(wsection.c_str()); #endif group.cchHeader = 0; group.pszFooter = nullptr; group.cchFooter = 0; group.iGroupId = lvi.iGroupId = numGroups++; group.stateMask = LVGS_COLLAPSIBLE; group.state = LVGS_COLLAPSIBLE; group.uAlign = LVGA_HEADER_LEFT; ListView_InsertGroup(m_List.m_hWnd, -1, &group); m_groups.insert(std::make_pair(section, lvi.iGroupId)); } else { lvi.iGroupId = gi->second; } #if MPT_USTRING_MODE_WIDE lvi.pszText = const_cast(key.c_str()); #else const std::wstring wkey = mpt::ToWide(key); lvi.pszText = const_cast(wkey.c_str()); #endif index = static_cast(m_List.SendMessage(LVM_INSERTITEMW, 0, (LPARAM)(&lvi))); } else { const mpt::ustring sectionAndKey = path.FormatAsString(); #if MPT_USTRING_MODE_WIDE lvi.pszText = const_cast(sectionAndKey.c_str()); #else const std::wstring wsectionAndKey = mpt::ToWide(sectionAndKey); lvi.pszText = const_cast(wsectionAndKey.c_str()); #endif index = static_cast(m_List.SendMessage(LVM_INSERTITEMW, 0, (LPARAM)(&lvi))); } #if MPT_USTRING_MODE_WIDE m_List.SetItemText(index, 1, value.FormatTypeAsString().c_str()); m_List.SetItemText(index, 2, value.FormatValueAsString().c_str()); m_List.SetItemText(index, 3, defaultValue.FormatValueAsString().c_str()); #else m_List.SetItemText(index, 1, mpt::ToCString(value.FormatTypeAsString())); m_List.SetItemText(index, 2, mpt::ToCString(value.FormatValueAsString())); m_List.SetItemText(index, 3, mpt::ToCString(defaultValue.FormatValueAsString())); #endif m_indexToPath.push_back(path); } m_List.SetItemCount(i); m_List.SetRedraw(TRUE); m_List.Invalidate(FALSE); } void COptionsAdvanced::OnOK() { CSoundFile::SetDefaultNoteNames(); CMainFrame *pMainFrm = CMainFrame::GetMainFrame(); if (pMainFrm) pMainFrm->PostMessage(WM_MOD_INVALIDATEPATTERNS, HINT_MPTOPTIONS); CPropertyPage::OnOK(); } BOOL COptionsAdvanced::OnSetActive() { ReInit(); CMainFrame::m_nLastOptionsPage = OPTIONS_PAGE_ADVANCED; return CPropertyPage::OnSetActive(); } #ifdef MPT_MFC_FULL COLORREF CAdvancedSettingsList::OnGetCellBkColor(int nRow, int /* nColumn */ ) { const bool isDefault = theApp.GetSettings().GetMap().find(m_indexToPath[GetItemData(nRow)])->second.IsDefault(); COLORREF defColor = GetBkColor(); COLORREF txtColor = GetTextColor(); COLORREF modColor = RGB(GetRValue(defColor) * 0.9 + GetRValue(txtColor) * 0.1, GetGValue(defColor) * 0.9 + GetGValue(txtColor) * 0.1, GetBValue(defColor) * 0.9 + GetBValue(txtColor) * 0.1); return isDefault ? defColor : modColor; } COLORREF CAdvancedSettingsList::OnGetCellTextColor(int nRow, int nColumn) { return CMFCListCtrlEx::OnGetCellTextColor(nRow, nColumn); } #else // !MPT_MFC_FULL void COptionsAdvanced::OnCustomDrawList(NMHDR* pNMHDR, LRESULT* pResult) { NMLVCUSTOMDRAW *pLVCD = reinterpret_cast(pNMHDR); *pResult = CDRF_DODEFAULT; switch(pLVCD->nmcd.dwDrawStage) { case CDDS_PREPAINT: *pResult = CDRF_NOTIFYITEMDRAW; break; case CDDS_ITEMPREPAINT: { const bool isDefault = theApp.GetSettings().GetMap().find(m_indexToPath[pLVCD->nmcd.lItemlParam])->second.IsDefault(); COLORREF defColor = m_List.GetBkColor(); COLORREF txtColor = m_List.GetTextColor(); COLORREF modColor = RGB(GetRValue(defColor) * 0.9 + GetRValue(txtColor) * 0.1, GetGValue(defColor) * 0.9 + GetGValue(txtColor) * 0.1, GetBValue(defColor) * 0.9 + GetBValue(txtColor) * 0.1); pLVCD->clrTextBk = isDefault ? defColor : modColor; } break; } } #endif // MPT_MFC_FULL void COptionsAdvanced::OnOptionDblClick(NMHDR *, LRESULT *) { const int index = m_List.GetSelectionMark(); if(index < 0) return; const SettingPath path = m_indexToPath[m_List.GetItemData(index)]; SettingValue val = theApp.GetSettings().GetMap().find(path)->second; if(val.GetType() == SettingTypeBool) { val = !val.as(); } else { CInputDlg inputDlg(this, _T("Enter new value for ") + mpt::ToCString(path.FormatAsString()), mpt::ToCString(val.FormatValueAsString())); if(inputDlg.DoModal() != IDOK) { return; } val.SetFromString(inputDlg.resultAsString); } theApp.GetSettings().Write(path, val); #if MPT_USTRING_MODE_WIDE m_List.SetItemText(index, 2, val.FormatValueAsString().c_str()); #else m_List.SetItemText(index, 2, mpt::ToCString(val.FormatValueAsString())); #endif m_List.SetSelectionMark(index); OnSettingsChanged(); } void COptionsAdvanced::OnSaveNow() { TrackerSettings::Instance().SaveSettings(); theApp.GetSettings().WriteSettings(); } OPENMPT_NAMESPACE_END