,使用::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;
}
由这个文档逆向化操作引擎,可以获得当前改动的文档的数据,并根据改动的数据更新视图,而不刷新没有更改数据的视图.从而防止了闪烁的产生.
三、简单开发实例
下面以我们开发服装CAD过程中加入的回退重做功能(文档逆向化)说明之。
1.定义回退类型
#define REUNDO_MOV 0x0001 file://衣片移动回退重做
#define REUNDO_SEL 0x0002 file://衣片选择回退重做
……….
2.保存某个操作之前和之后的数据(以衣片移动回退重做为例)
//----------申请内存----------------------//
int nByte = 4*sizeof(DWORD);
HGLOBAL hMem = GlobalAlloc(GMEM_FIXED,nByte);
LPVOID pData = (LPVOID) GlobalLock(hMem);
file://-----保存衣片移动前后的位置读入内存------//用移动前后衣片的某个坐标点表示
memcpy((DWORD*)pData, &m_oldPoint, 2*sizeof(DWORD));
memcpy((DWORD*)pData+2,&point, 2*sizeof(DWORD));
file://--------数据入栈---------------------------------------//
m_pReUndoEngine->PushData(pData,//衣片m_pReUndoEngine文档逆向化引擎对象指针
nByte,//保存数据衣片字节数
REUNDO_MOV,//回退类型
NULL,NULL);
3.当回退操作事件触发时.
//弹出回退值
int nByte = m_pReUndoEngine->GetPopDataSize();
HGLOBAL hMem = GlobalAlloc(GMEM_FIXED,nByte);//申请内存
LPVOID pData = (LPVOID) GlobalLock(hMem);
DWORD undo_type;DWORD index;
m_pReUndoEngine->PopData(&pData,NULL,&undo_type,&index);
////////////////////////////////
switch(undo_type){//回退类型
case REUNDO_SEL:
SelUndo(pData,index,&dc);break;
case REUNDO_MOV:
MovUndo(pData);break;
…………
}
void CMarkView::MovUndo(LPVOID pData) 函数功能
{
CPoint pt1,pt2;
memcpy(&pt1,(DWORD*)pData,8);
memcpy(&pt2,(DWORD*)pData+2,8);
…….由pt1 和pt2可以求出位移量,从而恢复原衣片的位置.
}
4.当重做操作事件触发时
//弹出回退值
int nByte = m_pReUndoEngine->GetRedoDataSize();
HGLOBAL hMem = GlobalAlloc(GMEM_FIXED,nByte);//申请内存
LPVOID pData = (LPVOID) GlobalLock(hMem);
DWORD undo_type;DWORD index;
m_pReUndoEngine->RedoData(&pData,NULL,&undo_type,&