二、实现文档回退重做的引擎
建一文档逆向化堆栈引擎.主要代码为:
1.建立临时文件.(m_TempPath可以按照某种规则形成路径)
if(m_File.Open((LPCTSTR)m_TempPath, CFile::modeCreate|CFile::modeReadWrite|CFile::shareExclusive))
{
m_File.SeekToBegin();
m_UndoCount = 0; file://当前可重做的步数
m_RedoCount = 0; file://当前可回退的步数
}
|
2.保存回退数据模块.
// 保存一个Undo数据块(由用户提供)
int CRedoUndoEngine::PushData(
LPVOID pData, // 由用户提供的内存块首地址,其中含有用户定义的待保存的数据。
// (注:如果函数成功,此内存块将会被本函数释放,因此,该内存块必须是用::GlobalAlloc()函数分配的)
DWORD size, // pData指向的内存块尺寸
DWORD param1, // 用户提供的对该内存块的说明参数,含义由用户定义
DWORD param2, // 用户提供的对该内存块的说明参数,含义由用户定义
int *pIndex // 如果成功,本函数将返回压入的Undo块在栈中的索引值。 如果不需要此返回值,可用NULL作为参数
)
{
// 删除Redo数据
if (m_RedoCount)
{
while(m_RedoCount--)
delete (LPISEEUNDOINFO)m_UndoDataList.RemoveTail();
m_RedoCount = 0;
}
// 填写Undo数据的索引信息(lpISeeUndoInfo为一个保存数据的结构体)
lpISeeUndoInfo->m_index = m_UndoCount; // 索引
lpISeeUndoInfo->m_UserData1 = param1; // 用户定义的标识性数据1
lpISeeUndoInfo->m_UserData2 = param2; // 用户定义的标识性数据2
lpISeeUndoInfo->m_DataSize = size; // 用户的Undo数据块尺寸
lpISeeUndoInfo->m_FilePosition = _get_current_overwrite_pos();
// 加新的Undo数据到Undo栈的尾部
m_UndoDataList.AddTail((void*)lpISeeUndoInfo);
// 将用户的Undo数据写入临时文件
m_File.Seek(lpISeeUndoInfo->m_FilePosition, CFile::begin);
m_File.Write((const void *)pData, size);
并使Undo块计数加1
m_UndoCount++;
// 此时用户传过来的数据块已经无用,删除!
::GlobalFree(pData);
return 1;
}
|
3.弹出重做数据模块.
// 弹出一个Redo数据块
int CIUndoEngine::RedoData(
LPVOID *ppData, // 用于接收本函数返回的含有最近一个Redo数据的内存块首地址的指针
// (注:此内存块交由调用者释放,使用::GlobalFree()函数)
DWORD *pSize, // ppData内存块的尺寸(in byte) ,如果不需要此数据可用NULL作为参数
DWORD *pParam1, // 返回用户对该Redo块的附加信息,如果不需要此数据可用NULL作为参数
DWORD *pParam2, // 返回用户对该Redo块的附加信息,如果不需要此数据可用NULL作为参数
int *pIndex // 返回本Redo块的索引,如果不需要此数据可用NULL作为参数
)
{
if (!m_RedoCount)
return 0;
// 锁定待弹出的Redo索引信息块的地址
POSITION pos = m_UndoDataList.FindIndex(m_UndoCount);
ASSERT(pos);
LPISEEUNDOINFO lpISeeUndoInfo= (LPISEEUNDOINFO)m_UndoDataList.GetAt(pos);
ASSERT(lpISeeUndoInfo);
ASSERT(lpISeeUndoInfo->m_index == m_UndoCount);
if (!(*ppData))
return -1;
// 读出用户保存在临时文件中的Undo数据(也即Redo数据)
m_File.Seek((LONG)lpISeeUndoInfo->m_FilePosition, CFile::begin);
m_File.Read(*ppData, lpISeeUndoInfo->m_DataSize);
m_UndoCount++; // 可用Undo数据块个数加1
m_RedoCount--; // 可用Redo数据块个数减1
if (pSize)
*pSize = lpISeeUndoInfo->m_DataSize;
if (pParam1)
*pParam1= lpISeeUndoInfo->m_UserData1;
if (pParam2)
*pParam2= lpISeeUndoInfo->m_UserData2;
if (pIndex)
*pIndex = m_RedoCount;// 注:此处的索引是Redo的索引,而不是Undo的
return 1;
}
|
由这个文档逆向化操作引擎,可以获得当前改动的文档的数据,并根据改动的数据更新视图,而不刷新没有更改数据的视图.从而防止了闪烁的产生. 
说明:本教程来源互联网或网友上传或出版商,仅为学习研究或媒体推广,wanshiok.com不保证资料的完整性。
2/2 首页 上一页 1 2 |