?? ccrystaltextbuffer.cpp
字號:
if (pszText[nTextPos] == _T('\n'))
{
nTextPos ++;
}
else
{
ASSERT(FALSE); // Invalid line-end format passed
}
pszText += nTextPos;
}
if (pszRestChars != NULL)
delete pszRestChars;
context.m_ptEnd.x = nEndChar;
context.m_ptEnd.y = nEndLine;
if (bNewLines)
UpdateViews(pSource, &context, UPDATE_HORZRANGE | UPDATE_VERTRANGE, nLine);
else
UpdateViews(pSource, &context, UPDATE_SINGLELINE | UPDATE_HORZRANGE, nLine);
if (! m_bModified)
SetModified(TRUE);
return TRUE;
}
BOOL CCrystalTextBuffer::CanUndo()
{
ASSERT(m_nUndoPosition >= 0 && m_nUndoPosition <= m_aUndoBuf.GetSize());
return m_nUndoPosition > 0;
}
BOOL CCrystalTextBuffer::CanRedo()
{
ASSERT(m_nUndoPosition >= 0 && m_nUndoPosition <= m_aUndoBuf.GetSize());
return m_nUndoPosition < m_aUndoBuf.GetSize();
}
POSITION CCrystalTextBuffer::GetUndoDescription(CString &desc, POSITION pos /*= NULL*/)
{
ASSERT(CanUndo()); // Please call CanUndo() first
ASSERT((m_aUndoBuf[0].m_dwFlags & UNDO_BEGINGROUP) != 0);
int nPosition;
if (pos == NULL)
{
// Start from beginning
nPosition = m_nUndoPosition;
}
else
{
nPosition = (int) pos;
ASSERT(nPosition > 0 && nPosition < m_nUndoPosition);
ASSERT((m_aUndoBuf[nPosition].m_dwFlags & UNDO_BEGINGROUP) != 0);
}
// Advance to next undo group
nPosition --;
while ((m_aUndoBuf[nPosition].m_dwFlags & UNDO_BEGINGROUP) == 0)
nPosition --;
// Read description
if (! GetActionDescription(m_aUndoBuf[nPosition].m_nAction, desc))
desc.Empty(); // Use empty string as description
// Now, if we stop at zero position, this will be the last action,
// since we return (POSITION) nPosition
return (POSITION) nPosition;
}
POSITION CCrystalTextBuffer::GetRedoDescription(CString &desc, POSITION pos /*= NULL*/)
{
ASSERT(CanRedo()); // Please call CanRedo() before!
ASSERT((m_aUndoBuf[0].m_dwFlags & UNDO_BEGINGROUP) != 0);
ASSERT((m_aUndoBuf[m_nUndoPosition].m_dwFlags & UNDO_BEGINGROUP) != 0);
int nPosition;
if (pos == NULL)
{
// Start from beginning
nPosition = m_nUndoPosition;
}
else
{
nPosition = (int) pos;
ASSERT(nPosition > m_nUndoPosition);
ASSERT((m_aUndoBuf[nPosition].m_dwFlags & UNDO_BEGINGROUP) != 0);
}
// Read description
if (! GetActionDescription(m_aUndoBuf[nPosition].m_nAction, desc))
desc.Empty(); // Use empty string as description
// Advance to next undo group
nPosition ++;
while (nPosition < m_aUndoBuf.GetSize() &&
(m_aUndoBuf[nPosition].m_dwFlags & UNDO_BEGINGROUP) == 0)
nPosition --;
if (nPosition >= m_aUndoBuf.GetSize())
return NULL; // No more redo actions!
return (POSITION) nPosition;
}
BOOL CCrystalTextBuffer::Undo(CPoint &ptCursorPos)
{
ASSERT(CanUndo());
ASSERT((m_aUndoBuf[0].m_dwFlags & UNDO_BEGINGROUP) != 0);
for (;;)
{
m_nUndoPosition --;
const SUndoRecord &ur = m_aUndoBuf[m_nUndoPosition];
if (ur.m_dwFlags & UNDO_INSERT)
{
#ifdef _ADVANCED_BUGCHECK
// Try to ensure that we undoing correctly...
// Just compare the text as it was before Undo operation
CString text;
GetText(ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.m_ptEndPos.y, ur.m_ptEndPos.x, text);
ASSERT(lstrcmp(text, ur.GetText()) == 0);
#endif
VERIFY(InternalDeleteText(NULL, ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.m_ptEndPos.y, ur.m_ptEndPos.x));
ptCursorPos = ur.m_ptStartPos;
}
else
{
int nEndLine, nEndChar;
VERIFY(InternalInsertText(NULL, ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.GetText(), nEndLine, nEndChar));
#ifdef _ADVANCED_BUGCHECK
ASSERT(ur.m_ptEndPos.y == nEndLine);
ASSERT(ur.m_ptEndPos.x == nEndChar);
#endif
ptCursorPos = ur.m_ptEndPos;
}
if (ur.m_dwFlags & UNDO_BEGINGROUP)
break;
}
if (m_bModified && m_nSyncPosition == m_nUndoPosition)
SetModified(FALSE);
if (! m_bModified && m_nSyncPosition != m_nUndoPosition)
SetModified(TRUE);
return TRUE;
}
BOOL CCrystalTextBuffer::Redo(CPoint &ptCursorPos)
{
ASSERT(CanRedo());
ASSERT((m_aUndoBuf[0].m_dwFlags & UNDO_BEGINGROUP) != 0);
ASSERT((m_aUndoBuf[m_nUndoPosition].m_dwFlags & UNDO_BEGINGROUP) != 0);
for (;;)
{
const SUndoRecord &ur = m_aUndoBuf[m_nUndoPosition];
if (ur.m_dwFlags & UNDO_INSERT)
{
int nEndLine, nEndChar;
VERIFY(InternalInsertText(NULL, ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.GetText(), nEndLine, nEndChar));
#ifdef _ADVANCED_BUGCHECK
ASSERT(ur.m_ptEndPos.y == nEndLine);
ASSERT(ur.m_ptEndPos.x == nEndChar);
#endif
ptCursorPos = ur.m_ptEndPos;
}
else
{
#ifdef _ADVANCED_BUGCHECK
CString text;
GetText(ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.m_ptEndPos.y, ur.m_ptEndPos.x, text);
ASSERT(lstrcmp(text, ur.GetText()) == 0);
#endif
VERIFY(InternalDeleteText(NULL, ur.m_ptStartPos.y, ur.m_ptStartPos.x, ur.m_ptEndPos.y, ur.m_ptEndPos.x));
ptCursorPos = ur.m_ptStartPos;
}
m_nUndoPosition ++;
if (m_nUndoPosition == m_aUndoBuf.GetSize())
break;
if ((m_aUndoBuf[m_nUndoPosition].m_dwFlags & UNDO_BEGINGROUP) != 0)
break;
}
if (m_bModified && m_nSyncPosition == m_nUndoPosition)
SetModified(FALSE);
if (! m_bModified && m_nSyncPosition != m_nUndoPosition)
SetModified(TRUE);
return TRUE;
}
// [JRT] Support For Descriptions On Undo/Redo Actions
void CCrystalTextBuffer::AddUndoRecord(BOOL bInsert, const CPoint &ptStartPos, const CPoint &ptEndPos, LPCTSTR pszText, int nActionType)
{
// Forgot to call BeginUndoGroup()?
ASSERT(m_bUndoGroup);
ASSERT(m_aUndoBuf.GetSize() == 0 || (m_aUndoBuf[0].m_dwFlags & UNDO_BEGINGROUP) != 0);
// Strip unnecessary undo records (edit after undo)
int nBufSize = m_aUndoBuf.GetSize();
if (m_nUndoPosition < nBufSize)
{
for (int I = m_nUndoPosition; I < nBufSize; I++)
m_aUndoBuf[I].FreeText();
m_aUndoBuf.SetSize(m_nUndoPosition);
}
// If undo buffer size is close to critical, remove the oldest records
ASSERT(m_aUndoBuf.GetSize() <= m_nUndoBufSize);
nBufSize = m_aUndoBuf.GetSize();
if (nBufSize >= m_nUndoBufSize)
{
int nIndex = 0;
for (;;)
{
m_aUndoBuf[nIndex].FreeText();
nIndex ++;
if (nIndex == nBufSize || (m_aUndoBuf[nIndex].m_dwFlags & UNDO_BEGINGROUP) != 0)
break;
}
m_aUndoBuf.RemoveAt(0, nIndex);
}
ASSERT(m_aUndoBuf.GetSize() < m_nUndoBufSize);
// Add new record
SUndoRecord ur;
ur.m_dwFlags = bInsert ? UNDO_INSERT : 0;
ur.m_nAction = nActionType;
if (m_bUndoBeginGroup)
{
ur.m_dwFlags |= UNDO_BEGINGROUP;
m_bUndoBeginGroup = FALSE;
}
ur.m_ptStartPos = ptStartPos;
ur.m_ptEndPos = ptEndPos;
ur.SetText(pszText);
m_aUndoBuf.Add(ur);
m_nUndoPosition = m_aUndoBuf.GetSize();
ASSERT(m_aUndoBuf.GetSize() <= m_nUndoBufSize);
}
BOOL CCrystalTextBuffer::InsertText(CCrystalEditView *pSource, int nLine, int nPos, LPCTSTR pszText,
int &nEndLine, int &nEndChar, int nAction)
{
if (! InternalInsertText(pSource, nLine, nPos, pszText, nEndLine, nEndChar))
return FALSE;
BOOL bGroupFlag = FALSE;
if (! m_bUndoGroup)
{
BeginUndoGroup();
bGroupFlag = TRUE;
}
AddUndoRecord(TRUE, CPoint(nPos, nLine), CPoint(nEndChar, nEndLine), pszText, nAction);
if (bGroupFlag)
FlushUndoGroup(pSource);
return TRUE;
}
BOOL CCrystalTextBuffer::DeleteText(CCrystalEditView *pSource, int nStartLine, int nStartChar,
int nEndLine, int nEndChar, int nAction)
{
CString sTextToDelete;
GetText(nStartLine, nStartChar, nEndLine, nEndChar, sTextToDelete);
if (! InternalDeleteText(pSource, nStartLine, nStartChar, nEndLine, nEndChar))
return FALSE;
BOOL bGroupFlag = FALSE;
if (! m_bUndoGroup)
{
BeginUndoGroup();
bGroupFlag = TRUE;
}
AddUndoRecord(FALSE, CPoint(nStartChar, nStartLine), CPoint(nEndChar, nEndLine), sTextToDelete, nAction);
if (bGroupFlag)
FlushUndoGroup(pSource);
return TRUE;
}
BOOL CCrystalTextBuffer::GetActionDescription(int nAction, CString &desc)
{
/* HINSTANCE hOldResHandle = AfxGetResourceHandle();
#ifdef CRYSEDIT_RES_HANDLE
AfxSetResourceHandle(CRYSEDIT_RES_HANDLE);
#else
if (CCrystalEditView::s_hResourceInst != NULL)
AfxSetResourceHandle(CCrystalEditView::s_hResourceInst);
#endif*/
BOOL bSuccess = FALSE;
switch (nAction)
{
case CE_ACTION_UNKNOWN:
bSuccess = desc.LoadString(IDS_EDITOP_UNKNOWN);
break;
case CE_ACTION_PASTE:
bSuccess = desc.LoadString(IDS_EDITOP_PASTE);
break;
case CE_ACTION_DELSEL:
bSuccess = desc.LoadString(IDS_EDITOP_DELSELECTION);
break;
case CE_ACTION_CUT:
bSuccess = desc.LoadString(IDS_EDITOP_CUT);
break;
case CE_ACTION_TYPING:
bSuccess = desc.LoadString(IDS_EDITOP_TYPING);
break;
case CE_ACTION_BACKSPACE:
bSuccess = desc.LoadString(IDS_EDITOP_BACKSPACE);
break;
case CE_ACTION_INDENT:
bSuccess = desc.LoadString(IDS_EDITOP_INDENT);
break;
case CE_ACTION_DRAGDROP:
bSuccess = desc.LoadString(IDS_EDITOP_DRAGDROP);
break;
case CE_ACTION_REPLACE:
bSuccess = desc.LoadString(IDS_EDITOP_REPLACE);
break;
case CE_ACTION_DELETE:
bSuccess = desc.LoadString(IDS_EDITOP_DELETE);
break;
case CE_ACTION_AUTOINDENT:
bSuccess = desc.LoadString(IDS_EDITOP_AUTOINDENT);
break;
}
// AfxSetResourceHandle(hOldResHandle);
return bSuccess;
}
void CCrystalTextBuffer::SetModified(BOOL bModified /*= TRUE*/)
{
m_bModified = bModified;
}
void CCrystalTextBuffer::BeginUndoGroup(BOOL bMergeWithPrevious /*= FALSE*/)
{
ASSERT(! m_bUndoGroup);
m_bUndoGroup = TRUE;
m_bUndoBeginGroup = m_nUndoPosition == 0 || ! bMergeWithPrevious;
}
void CCrystalTextBuffer::FlushUndoGroup(CCrystalEditView *pSource)
{
ASSERT(m_bUndoGroup);
if (pSource != NULL)
{
ASSERT(m_nUndoPosition == m_aUndoBuf.GetSize());
if (m_nUndoPosition > 0)
{
m_bUndoBeginGroup = TRUE;
// pSource->OnEditOperation(m_aUndoBuf[m_nUndoPosition - 1].m_nAction, m_aUndoBuf[m_nUndoPosition - 1].GetText());
}
}
m_bUndoGroup = FALSE;
}
int CCrystalTextBuffer::FindNextBookmarkLine(int nCurrentLine)
{
BOOL bWrapIt = TRUE;
DWORD dwFlags = GetLineFlags(nCurrentLine);
if ((dwFlags & LF_BOOKMARKS) != 0)
nCurrentLine++;
int nSize = m_aLines.GetSize();
for (;;)
{
while (nCurrentLine < nSize)
{
if ((m_aLines[nCurrentLine].m_dwFlags & LF_BOOKMARKS) != 0)
return nCurrentLine;
// Keep going
nCurrentLine++;
}
// End of text reached
if (!bWrapIt)
return -1;
// Start from the beginning of text
bWrapIt = FALSE;
nCurrentLine = 0;
}
return -1;
}
int CCrystalTextBuffer::FindPrevBookmarkLine(int nCurrentLine)
{
BOOL bWrapIt = TRUE;
DWORD dwFlags = GetLineFlags(nCurrentLine);
if ((dwFlags & LF_BOOKMARKS) != 0)
nCurrentLine--;
int nSize = m_aLines.GetSize();
for (;;)
{
while (nCurrentLine >= 0)
{
if ((m_aLines[nCurrentLine].m_dwFlags & LF_BOOKMARKS) != 0)
return nCurrentLine;
// Keep moving up
nCurrentLine--;
}
// Beginning of text reached
if (!bWrapIt)
return -1;
// Start from the end of text
bWrapIt = FALSE;
nCurrentLine = nSize - 1;
}
return -1;
}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -