345 lines
8.2 KiB
C++
Raw Normal View History

2024-09-24 14:54:57 +02:00
/*
* Ctrl_com.cpp
* ------------
* Purpose: Song comments tab, upper panel.
* Notes : (currently none)
* Authors: Olivier Lapicque
* OpenMPT Devs
* The OpenMPT source code is released under the BSD license. Read LICENSE for more details.
*/
#include "stdafx.h"
#include "Mptrack.h"
#include "Mainfrm.h"
#include "Moddoc.h"
#include "Globals.h"
#include "Ctrl_com.h"
#include "view_com.h"
#include "InputHandler.h"
#include "../soundlib/mod_specifications.h"
//#define MPT_COMMENTS_LONG_LINES_WRAP
//#define MPT_COMMENTS_LONG_LINES_TRUNCATE
#define MPT_COMMENTS_MARGIN 4
OPENMPT_NAMESPACE_BEGIN
BEGIN_MESSAGE_MAP(CCtrlComments, CModControlDlg)
//{{AFX_MSG_MAP(CCtrlComments)
ON_EN_UPDATE(IDC_EDIT_COMMENTS, &CCtrlComments::OnCommentsUpdated)
ON_EN_CHANGE(IDC_EDIT_COMMENTS, &CCtrlComments::OnCommentsChanged)
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
void CCtrlComments::DoDataExchange(CDataExchange* pDX)
{
CModControlDlg::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CCtrlComments)
DDX_Control(pDX, IDC_EDIT_COMMENTS, m_EditComments);
//}}AFX_DATA_MAP
}
CCtrlComments::CCtrlComments(CModControlView &parent, CModDoc &document) : CModControlDlg(parent, document)
{
}
CRuntimeClass *CCtrlComments::GetAssociatedViewClass()
{
return RUNTIME_CLASS(CViewComments);
}
void CCtrlComments::OnActivatePage(LPARAM)
{
// Don't stop generating VU meter messages
m_modDoc.SetNotifications(Notification::Default);
m_modDoc.SetFollowWnd(m_hWnd);
m_EditComments.SetFocus();
}
void CCtrlComments::OnDeactivatePage()
{
CModControlDlg::OnDeactivatePage();
}
BOOL CCtrlComments::OnInitDialog()
{
CModControlDlg::OnInitDialog();
// Initialize comments
UINT margin = Util::ScalePixels(MPT_COMMENTS_MARGIN, m_EditComments.m_hWnd);
m_EditComments.SetMargins(margin, margin);
UpdateView(CommentHint().ModType());
m_EditComments.SetFocus();
m_EditComments.FmtLines(FALSE);
m_bInitialized = TRUE;
return FALSE;
}
void CCtrlComments::RecalcLayout()
{
CRect rcClient, rect;
int cx0, cy0;
if ((!m_hWnd) || (!m_EditComments.m_hWnd)) return;
GetClientRect(&rcClient);
m_EditComments.GetWindowRect(&rect);
ScreenToClient(&rect);
cx0 = rect.Width();
cy0 = rect.Height();
rect.bottom = rcClient.bottom - 3;
rect.right = rcClient.right - rect.left;
if ((rect.right > rect.left) && (rect.bottom > rect.top))
{
int cx = rect.Width(), cy = rect.Height();
if(m_sndFile.GetModSpecifications().commentLineLengthMax != 0)
{
int cxmax = Util::ScalePixels(GetSystemMetrics(SM_CXBORDER) + MPT_COMMENTS_MARGIN + m_sndFile.GetModSpecifications().commentLineLengthMax * charWidth + MPT_COMMENTS_MARGIN + GetSystemMetrics(SM_CXVSCROLL) + GetSystemMetrics(SM_CXBORDER) - 1, m_EditComments.m_hWnd);
if (cx > cxmax && cxmax != 0) cx = cxmax;
//SetWindowLong(m_EditComments.m_hWnd, GWL_STYLE, GetWindowLong(m_EditComments.m_hWnd, GWL_STYLE) & ~WS_HSCROLL);
} else
{
//SetWindowLong(m_EditComments.m_hWnd, GWL_STYLE, GetWindowLong(m_EditComments.m_hWnd, GWL_STYLE) | WS_HSCROLL);
}
if ((cx != cx0) || (cy != cy0)) m_EditComments.SetWindowPos(NULL, 0,0, cx, cy, SWP_NOMOVE|SWP_NOZORDER|SWP_DRAWFRAME);
}
}
void CCtrlComments::UpdateView(UpdateHint hint, CObject *pHint)
{
CommentHint commentHint = hint.ToType<CommentHint>();
if (pHint == this || !commentHint.GetType()[HINT_MODCOMMENTS | HINT_MPTOPTIONS | HINT_MODTYPE]) return;
if (m_nLockCount) return;
m_nLockCount++;
static FontSetting previousFont;
FontSetting font = TrackerSettings::Instance().commentsFont;
// Point size to pixels
int32 fontSize = -MulDiv(font.size, m_nDPIy, 720);
if(previousFont != font)
{
previousFont = font;
CMainFrame::GetCommentsFont() = ::CreateFont(fontSize, 0, 0, 0, font.flags[FontSetting::Bold] ? FW_BOLD : FW_NORMAL,
font.flags[FontSetting::Italic] ? TRUE :FALSE, FALSE, FALSE,
DEFAULT_CHARSET, OUT_DEFAULT_PRECIS,
CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY,
FIXED_PITCH | FF_MODERN, mpt::ToCString(font.name));
}
m_EditComments.SendMessage(WM_SETFONT, (WPARAM)CMainFrame::GetCommentsFont());
CDC * pDC = m_EditComments.GetDC();
pDC->SelectObject(CMainFrame::GetCommentsFont());
TEXTMETRIC tm;
pDC->GetTextMetrics(&tm);
charWidth = tm.tmAveCharWidth;
m_EditComments.ReleaseDC(pDC);
RecalcLayout();
m_EditComments.SetRedraw(FALSE);
std::string text = m_sndFile.m_songMessage.GetFormatted(SongMessage::leCRLF);
for(std::size_t i = 0; i < text.length(); ++i)
{
// replace control characters
char c = text[i];
if(c > '\0' && c < ' ' && c != '\r' && c != '\n')
{
c = ' ';
}
text[i] = c;
}
CString new_text = mpt::ToCString(m_sndFile.GetCharsetInternal(), text);
CString old_text;
m_EditComments.GetWindowText(old_text);
if(new_text != old_text)
{
m_EditComments.SetWindowText(new_text);
}
if(commentHint.GetType() & HINT_MODTYPE)
{
m_EditComments.SetReadOnly(!m_sndFile.GetModSpecifications().hasComments);
}
m_EditComments.SetRedraw(TRUE);
m_nLockCount--;
}
void CCtrlComments::OnCommentsUpdated()
{
#if defined(MPT_COMMENTS_LONG_LINES_TRUNCATE) || defined(MPT_COMMENTS_LONG_LINES_WRAP)
if(m_Reformatting)
{
return;
}
if(!m_sndFile.GetModSpecifications().hasComments)
{
return;
}
if(m_sndFile.GetModSpecifications().commentLineLengthMax == 0)
{
return;
}
m_Reformatting = true;
const std::size_t maxline = m_sndFile.GetModSpecifications().commentLineLengthMax;
int beg = 0;
int end = 0;
m_EditComments.GetSel(beg, end);
CString text;
m_EditComments.GetWindowText(text);
std::string lines_new;
lines_new.reserve(text.GetLength());
bool modified = false;
std::size_t pos = 0;
#if defined(MPT_COMMENTS_LONG_LINES_WRAP)
std::string lines = text.GetString();
std::size_t line_length = 0;
for(std::size_t i = 0; i < lines.length(); ++i)
{
if(lines[i] == '\r')
{
// nothing
} else if (lines[i] == '\n')
{
line_length = 0;
} else
{
line_length += 1;
}
if(line_length > maxline)
{
modified = true;
lines_new.push_back('\r');
lines_new.push_back('\n');
if(beg >= 0)
{
if(beg >= pos)
{
beg += 2;
}
}
if(end >= 0)
{
if(end >= pos)
{
end += 2;
}
}
pos += 2;
line_length = 1;
}
lines_new.push_back(lines[i]);
pos++;
}
#elif defined(MPT_COMMENTS_LONG_LINES_TRUNCATE)
std::vector<std::string> lines = mpt::String::Split<std::string>(std::string(text.GetString()), std::string("\r\n"));
for(std::size_t i = 0; i < lines.size(); ++i)
{
if(i > 0)
{
pos += 2;
}
if(lines[i].length() > maxline)
{
modified = true;
pos += maxline;
for(std::size_t n = 0; n < lines[i].length() - maxline; ++n)
{
if(beg >= 0)
{
if(beg > pos)
{
beg--;
}
}
if(end >= 0)
{
if(end > pos)
{
end--;
}
}
}
lines[i] = lines[i].substr(0, maxline);
} else
{
pos += lines[i].length();
}
}
lines_new = mpt::String::Combine(lines, std::string("\r\n"));
#endif
if(modified)
{
text = lines_new.c_str();
m_EditComments.SetWindowText(text);
m_EditComments.SetSel(beg, end);
}
m_Reformatting = false;
#endif
}
void CCtrlComments::OnCommentsChanged()
{
if(m_nLockCount)
return;
if ((!m_bInitialized) || (!m_EditComments.m_hWnd) || (!m_EditComments.GetModify())) return;
CString text;
m_EditComments.GetWindowText(text);
m_EditComments.SetModify(FALSE);
if(m_sndFile.m_songMessage.SetFormatted(mpt::ToCharset(m_sndFile.GetCharsetInternal(), text), SongMessage::leCRLF))
{
m_modDoc.SetModified();
m_modDoc.UpdateAllViews(nullptr, CommentHint(), this);
}
}
BOOL CCtrlComments::PreTranslateMessage(MSG *pMsg)
{
if(pMsg->message == WM_KEYDOWN && pMsg->wParam == 'A' && GetKeyState(VK_CONTROL) < 0)
{
// Ctrl-A is not handled by multiline edit boxes
if(::GetFocus() == m_EditComments.m_hWnd)
m_EditComments.SetSel(0, -1);
} else if(pMsg->message == WM_KEYDOWN && pMsg->wParam == VK_TAB && CMainFrame::GetMainFrame()->GetInputHandler()->GetModifierMask() == ModNone)
{
int selStart, selEnd;
m_EditComments.GetSel(selStart, selEnd);
int posInLine = (selStart - m_EditComments.LineIndex(m_EditComments.LineFromChar(selStart)));
CString tabs(_T(' '), 4 - (posInLine % 4));
m_EditComments.ReplaceSel(tabs, TRUE);
m_EditComments.SetSel(-1, -1, TRUE);
return TRUE;
}
return CModControlDlg::PreTranslateMessage(pMsg);
}
OPENMPT_NAMESPACE_END