home *** CD-ROM | disk | FTP | other *** search
- /*
- * DOCUMENT.CPP
- * Cosmo Chapter 18
- *
- * Implementation of the CCosmoDoc derivation of CDocument as
- * well as an implementation of CPolylineAdviseSink.
- *
- * Copyright (c)1993-1995 Microsoft Corporation, All Rights Reserved
- *
- * Kraig Brockschmidt, Microsoft
- * Internet : kraigb@microsoft.com
- * Compuserve: >INTERNET:kraigb@microsoft.com
- */
-
-
- #include "cosmo.h"
-
-
- /*
- * CCosmoDoc::CCosmoDoc
- * CCosmoDoc::~CCosmoDoc
- *
- * Constructor Parameters:
- * hInst HINSTANCE of the application.
- * pFR PCFrame of the frame object.
- * pAdv PCDocumentAdviseSink to notify on events
- */
-
- CCosmoDoc::CCosmoDoc(HINSTANCE hInst, PCFrame pFR
- , PCDocumentAdviseSink pAdv)
- : CDocument(hInst, pFR, pAdv)
- {
- m_pPL=NULL;
- m_pPLAdv=NULL;
- m_uPrevSize=SIZE_RESTORED;
-
- m_pDropTarget=NULL;
- m_fDragSource=FALSE;
-
- //CHAPTER18MOD
- m_cfEmbedSource=RegisterClipboardFormat(CFSTR_EMBEDSOURCE);
- m_cfObjectDescriptor=RegisterClipboardFormat
- (CFSTR_OBJECTDESCRIPTOR);
-
- m_pFigure=NULL;
- //End CHAPTER18MOD
- return;
- }
-
-
- CCosmoDoc::~CCosmoDoc(void)
- {
- //CHAPTER18MOD
- //Make sure the object is saved
- m_pFigure->SendAdvise(OBJECTCODE_SAVEOBJECT);
- m_pFigure->SendAdvise(OBJECTCODE_HIDEWINDOW);
- m_pFigure->SendAdvise(OBJECTCODE_CLOSED);
-
- /*
- * The AddRef here preserves a reference count on the object
- * even after our call to CoDisconnectObject. When we then
- * Release that reference count, the figure will start
- * shutdown. In some cases shutdown will begin before this
- * time in which case we need to clean up before the
- * Figure is destroyed, so we delete the figure ourselves.
- * This could also be handled with IExternalConnection.
- */
-
- if (NULL!=m_pFigure)
- {
- m_pFigure->AddRef();
- CoDisconnectObject(m_pFigure, 0L);
- m_pFigure->Release(); //Starts shutdown if necessary.
- delete m_pFigure;
- }
- //End CHAPTER18MOD
-
- //Clean up the allocations we did in Init
- if (NULL!=m_pPL)
- delete m_pPL;
-
- if (NULL!=m_pPLAdv)
- delete m_pPLAdv;
-
- return;
- }
-
-
-
-
-
-
- /*
- * CCosmoDoc::Init
- *
- * Purpose:
- * Initializes an already created document window. The client
- * actually creates the window for us, then passes that here for
- * further initialization.
- *
- * Parameters:
- * pDI PDOCUMENTINIT containing initialization
- * parameters.
- *
- * Return Value:
- * BOOL TRUE if the function succeeded, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::Init(PDOCUMENTINIT pDI)
- {
- RECT rc;
-
- //Change the stringtable range to our customization.
- pDI->idsMin=IDS_DOCUMENTMIN;
- pDI->idsMax=IDS_DOCUMENTMAX;
-
- //Do default initialization
- if (!CDocument::Init(pDI))
- return FALSE;
-
- //Add the Polyline stuff we need.
- m_pPLAdv=new CPolylineAdviseSink(this);
- m_pPL =new CPolyline(m_hInst);
-
- //Attempt to create our contained Polyline.
- GetClientRect(m_hWnd, &rc);
- InflateRect(&rc, -8, -8);
-
- if (!m_pPL->Init(m_hWnd, &rc, WS_CHILD | WS_VISIBLE
- , ID_POLYLINE, m_pPLAdv))
- return FALSE;
-
- m_pDropTarget=new CDropTarget(this);
-
- if (NULL!=m_pDropTarget)
- {
- m_pDropTarget->AddRef();
- CoLockObjectExternal(m_pDropTarget, TRUE, FALSE);
- RegisterDragDrop(m_hWnd, m_pDropTarget);
- }
-
- //CHAPTER18MOD
- m_pFigure=new CFigure(ObjectDestroyed, this);
-
- if (NULL==m_pFigure)
- return FALSE;
-
- //We created an object, so count it.
- g_cObj++;
-
- if (!m_pFigure->Init())
- return FALSE;
- //End CHAPTER18MOD
-
- return TRUE;
- }
-
-
-
-
-
-
-
- /*
- * CCosmoDoc::FMessageHook
- *
- * Purpose:
- * Processes WM_SIZE for the document so we can resize
- * the Polyline.
- *
- * Parameters:
- * <WndProc Parameters>
- * pLRes LRESULT * in which to store the return
- * value for the message.
- *
- * Return Value:
- * BOOL TRUE to prevent further processing,
- * FALSE otherwise.
- */
-
- BOOL CCosmoDoc::FMessageHook(HWND hWnd, UINT iMsg, WPARAM wParam
- , LPARAM lParam, LRESULT *pLRes)
- {
- UINT dx, dy;
- RECT rc;
-
- *pLRes=0;
-
- if (WM_SIZE==iMsg)
- {
- //Don't effect the Polyline size to or from minimized state.
- if (SIZE_MINIMIZED!=wParam && SIZE_MINIMIZED !=m_uPrevSize)
- {
- //When we change size, resize any Polyline we hold.
- dx=LOWORD(lParam);
- dy=HIWORD(lParam);
-
- /*
- * If we are getting WM_SIZE in response to a Polyline
- * notification, then don't resize the Polyline window
- * again.
- */
- if (!m_fNoSize && NULL!=m_pPL)
- {
- //Resize the polyline to fit the new client
- SetRect(&rc, 8, 8, dx-8, dy-8);
- m_pPL->RectSet(&rc, FALSE);
-
- /*
- * We consider sizing something that makes the file
- * dirty, but not until we've finished the create
- * process, which is why we set fNoDirty to FALSE
- * in WM_CREATE since we get a WM_SIZE on the first
- * creation.
- */
- if (!m_fNoDirty)
- FDirtySet(TRUE);
-
- SetRect(&rc, 0, 0, dx, dy);
-
- if (NULL!=m_pAdv)
- m_pAdv->OnSizeChange(this, &rc);
-
- m_fNoDirty=FALSE;
- }
- }
-
- m_uPrevSize=wParam;
- }
-
- if (WM_LBUTTONDOWN==iMsg)
- {
- LPDROPSOURCE pIDropSource;
- LPDATAOBJECT pIDataObject;
- HRESULT hr;
- SCODE sc;
- DWORD dwEffect;
-
- /*
- * The document has an 8 pixel border around the polyline
- * window where we'll see mouse clicks. A left mouse button
- * click here means the start of a drag-drop operation.
- *
- * Since this is a modal operation, this IDropSource
- * is entirely local.
- */
-
- pIDropSource=new CDropSource(this);
-
- if (NULL==pIDropSource)
- return FALSE;
-
- pIDropSource->AddRef();
- m_fDragSource=TRUE;
-
- //Go get the data and start the ball rolling.
- pIDataObject=TransferObjectCreate(FALSE);
-
- if (NULL!=pIDataObject)
- {
- hr=DoDragDrop(pIDataObject, pIDropSource
- , DROPEFFECT_COPY | DROPEFFECT_MOVE, &dwEffect);
-
- pIDataObject->Release();
- sc=GetScode(hr);
- }
- else
- sc=E_FAIL;
-
- /*
- * When we return from DoDragDrop, either cancel or drop.
- * First toss the IDropSource we have here, then bail out
- * on cancel, and possibly clear our data on a move drop.
- */
-
- pIDropSource->Release();
-
- /*
- * If dropped on the same document (determined using
- * this flag, then dwEffect will be DROPEFFECT_NONE (see
- * IDropTarget::Drop in DROPTGT.CPP). In any case,
- * reset this since the operation is done.
- */
-
- m_fDragSource=FALSE;
-
- if (DRAGDROP_S_DROP==sc && DROPEFFECT_MOVE==dwEffect)
- {
- m_pPL->New();
- FDirtySet(TRUE);
- }
-
- //On a canceled drop or a copy we don't do anything else
- return TRUE;
- }
-
- if (WM_DESTROY==iMsg)
- {
- /*
- * We have to revoke the drop target here because the window
- * will be destroyed and the property forcefully removed
- * before we could do this in the destructor.
- */
- if (NULL!=m_pDropTarget)
- {
- RevokeDragDrop(m_hWnd);
- CoLockObjectExternal(m_pDropTarget, FALSE, TRUE);
- ReleaseInterface(m_pDropTarget);
- }
-
- return FALSE;
- }
-
-
- /*
- * We return FALSE even on WM_SIZE so we can let the default
- * procedure handle maximized MDI child windows appropriately.
- */
- return FALSE;
- }
-
-
-
-
-
-
-
-
- /*
- * CCosmoDoc::Clear
- *
- * Purpose:
- * Sets all contents in the document back to defaults with
- * no filename.
- *
- * Paramters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CCosmoDoc::Clear(void)
- {
- //Completely reset the polyline
- m_pPL->New();
-
- CDocument::Clear();
- m_lVer=0;
- return;
- }
-
-
-
- //CHAPTER18MOD
- /*
- * CCosmoDoc::FDirtySet
- *
- * Purpose:
- * Sets or clears the document 'dirty' flag returning the previous
- * state of that same flag. We override this in Cosmo server to
- * send the OnDataChange notification as necessary.
- *
- * Parameters:
- * fDirty BOOL indicating the new contents of the dirty
- * flag.
- *
- * Return Value:
- * BOOL Previous value of the dirty flag.
- */
-
- BOOL CCosmoDoc::FDirtySet(BOOL fDirty)
- {
- BOOL fRet;
-
- fRet=CDocument::FDirtySet(fDirty);
- m_pFigure->SendAdvise(OBJECTCODE_DATACHANGED);
-
- return fRet;
- }
-
-
-
-
- /*
- * CCosmoDoc::FDirtyGet
- *
- * Purpose:
- * Override of the normal FDirtyGet such that we never return dirty
- * for an embedded object we're serving since updates constantly
- * happen and since the object will be saved on closure. This then
- * prevents any message boxes asking the user to save.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * BOOL TRUE if the document is dirty, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::FDirtyGet(void)
- {
- if (m_pFigure->FIsEmbedded())
- return FALSE;
-
- return m_fDirty;
- }
- //End CHAPTER18MOD
-
-
-
- /*
- * CCosmoDoc::Load
- *
- * Purpose:
- * Loads a given document without any user interface overwriting
- * the previous contents of the Polyline window. We do this by
- * opening the file and telling the Polyline to load itself from
- * that file.
- *
- * Parameters:
- * fChangeFile BOOL indicating if we're to update the window
- * title and the filename from using this file.
- * pszFile LPTSTR to the filename to load, NULL if the file
- * is new and untitled.
- *
- * Return Value:
- * UINT An error value from DOCERR_*
- */
-
- UINT CCosmoDoc::Load(BOOL fChangeFile, LPTSTR pszFile)
- {
- HRESULT hr;
- LPSTORAGE pIStorage;
-
- if (NULL==pszFile)
- {
- //For a new untitled document, just rename ourselves.
- Rename(NULL);
- m_lVer=VERSIONCURRENT;
- return DOCERR_NONE;
- }
-
- /*
- * If not a Compound File, open the file using STGM_CONVERT in
- * transacted mode to see old files as a storage with one stream
- * called "CONTENTS" (which is conveniently the name we use
- * in the new files). We must use STGM_TRANSACTED here or else
- * the old file will be immediately converted on disk: we only
- * want a converted image in memory from which to read. In
- * addition, note that we need STGM_READWRITE as well since
- * conversion is inherently a write operation.
- */
-
- pIStorage=NULL;
-
- if (NOERROR!=StgIsStorageFile(pszFile))
- {
- hr=StgCreateDocfile(pszFile,STGM_TRANSACTED | STGM_READWRITE
- | STGM_CONVERT | STGM_SHARE_EXCLUSIVE, 0, &pIStorage);
-
- if (FAILED(hr))
- {
- //If denied write access, try to load the old way
- if (STG_E_ACCESSDENIED==GetScode(hr))
- m_lVer=m_pPL->ReadFromFile(pszFile);
- else
- return DOCERR_COULDNOTOPEN;
- }
- }
- else
- {
- hr=StgOpenStorage(pszFile, NULL, STGM_DIRECT | STGM_READ
- | STGM_SHARE_EXCLUSIVE, NULL, 0, &pIStorage);
-
- if (FAILED(hr))
- return DOCERR_COULDNOTOPEN;
- }
-
- if (NULL!=pIStorage)
- {
- m_lVer=m_pPL->ReadFromStorage(pIStorage);
- pIStorage->Release();
- }
-
- if (POLYLINE_E_READFAILURE==m_lVer)
- return DOCERR_READFAILURE;
-
- if (POLYLINE_E_UNSUPPORTEDVERSION==m_lVer)
- return DOCERR_UNSUPPORTEDVERSION;
-
- if (fChangeFile)
- Rename(pszFile);
-
- //Importing a file makes things dirty
- FDirtySet(!fChangeFile);
-
- return DOCERR_NONE;
- }
-
-
-
-
-
-
-
- /*
- * CCosmoDoc::Save
- *
- * Purpose:
- * Writes the file to a known filename, requiring that the user has
- * previously used FileOpen or FileSaveAs to provide a filename.
- *
- * Parameters:
- * uType UINT indicating the type of file the user
- * requested to save in the File Save As dialog.
- * pszFile LPTSTR under which to save. If NULL, use the
- * current name.
- *
- * Return Value:
- * UINT An error value from DOCERR_*
- */
-
- UINT CCosmoDoc::Save(UINT uType, LPTSTR pszFile)
- {
- LONG lVer, lRet;
- UINT uTemp;
- BOOL fRename=TRUE;
- HRESULT hr;
- LPSTORAGE pIStorage;
- //CHAPTER18MOD
- BOOL fEmbedding;
-
- fEmbedding=m_pFigure->FIsEmbedded();
- //End CHAPTER18MOD
-
- if (NULL==pszFile)
- {
- fRename=FALSE;
- pszFile=m_szFile;
- }
-
- /*
- * Type 1 is the current version, type 2 is version 1.0 of the
- * Polyline so we use this to send the right version to
- * CPolyline::WriteToFile/WriteToStorage.
- */
-
- switch (uType)
- {
- case 0: //From Save, use loaded version.
- lVer=m_lVer;
- break;
-
- case 1:
- lVer=VERSIONCURRENT;
- break;
-
- case 2:
- lVer=MAKELONG(0, 1); //1.0
- break;
-
- default:
- return DOCERR_UNSUPPORTEDVERSION;
- }
-
- /*
- * If the version the user wants to save is different from the
- * version that we loaded and m_lVer is not zero (new doc),
- * then inform the user of the version change and verify.
- */
-
- //CHAPTER18MOD
- //For embedding, this is Save Copy As, so don't ask about versions.
- if (0!=m_lVer && m_lVer!=lVer && !fEmbedding)
- //End CHAPTER18MOD
- {
- TCHAR szMsg[128];
-
- wsprintf(szMsg, PSZ(IDS_VERSIONCHANGE)
- , (UINT)HIWORD(m_lVer), (UINT)LOWORD(m_lVer)
- , (UINT)HIWORD(lVer), (UINT)LOWORD(lVer));
-
- uTemp=MessageBox(m_hWnd, szMsg, PSZ(IDS_DOCUMENTCAPTION)
- , MB_YESNOCANCEL);
-
- if (IDCANCEL==uTemp)
- return DOCERR_CANCELLED;
-
- //If the user won't upgrade, revert to loaded version.
- if (IDNO==uTemp)
- lVer=m_lVer;
- }
-
- /*
- * For 1.0 files, still use the old code. For new files, use
- * storages instead
- */
- if (lVer==MAKELONG(0, 1))
- lRet=m_pPL->WriteToFile(pszFile, lVer);
- else
- {
- hr=StgCreateDocfile(pszFile, STGM_DIRECT | STGM_READWRITE
- | STGM_CREATE | STGM_SHARE_EXCLUSIVE, 0, &pIStorage);
-
- if (FAILED(hr))
- return DOCERR_COULDNOTOPEN;
-
- //Mark this as one of our class
- WriteClassStg(pIStorage, CLSID_CosmoFigure);
-
- //Write user-readable class information
- WriteFmtUserTypeStg(pIStorage, m_cf
- , PSZ(IDS_CLIPBOARDFORMAT));
-
- lRet=m_pPL->WriteToStorage(pIStorage, lVer);
- pIStorage->Release();
- }
-
- if (POLYLINE_E_NONE!=lRet)
- return DOCERR_WRITEFAILURE;
-
- //CHAPTER18MOD
- //Saving makes us clean, but this doesn't apply to embedding.
- if (!fEmbedding)
- FDirtySet(FALSE);
- //End CHAPTER18MOD
-
- //Update the known version of this document.
- m_lVer=lVer;
-
- //CHAPTER18MOD
- /*
- * If we're embedding, this is Save Copy As, so no rename.
- * Note that we also don't care about having been set to clean
- * since we're always 'clean' as an embedded object from
- * the user's perspective.
- */
- if (fRename && !fEmbedding)
- Rename(pszFile);
- //End CHAPTER18MOD
-
- return DOCERR_NONE;
- }
-
-
-
-
-
-
- /*
- * CCosmoDoc::Undo
- *
- * Purpose:
- * Reverses a previous action.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CCosmoDoc::Undo(void)
- {
- m_pPL->Undo();
- return;
- }
-
-
-
-
-
-
- /*
- * CCosmoDoc::Clip
- *
- * Purpose:
- * Places a private format, a metafile, and a bitmap of the display
- * on the clipboard, optionally implementing Cut by deleting the
- * data in the current window after rendering.
- *
- * Parameters:
- * hWndFrame HWND of the main window.
- * fCut BOOL indicating cut (TRUE) or copy (FALSE).
- *
- * Return Value:
- * BOOL TRUE if successful, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::Clip(HWND hWndFrame, BOOL fCut)
- {
- BOOL fRet=TRUE;
- LPDATAOBJECT pIDataObject;
-
- pIDataObject=TransferObjectCreate(fCut);
-
- if (NULL==pIDataObject)
- return FALSE;
-
- fRet=SUCCEEDED(OleSetClipboard(pIDataObject));
- pIDataObject->Release();
-
- //Delete our current data if "cut" succeeded.
- if (fRet && fCut)
- {
- m_pPL->New();
- FDirtySet(TRUE);
- }
-
-
- return fRet;
- }
-
-
-
-
-
- /*
- * CCosmoDoc::RenderFormat
- *
- * Purpose:
- * Renders a specific clipboard format into global memory.
- *
- * Parameters:
- * cf UINT format to render.
- *
- * Return Value:
- * HGLOBAL Global memory handle containing the data.
- */
-
- HGLOBAL CCosmoDoc::RenderFormat(UINT cf)
- {
- HGLOBAL hMem;
-
- if (cf==m_cf)
- {
- m_pPL->DataGetMem(VERSIONCURRENT, &hMem);
- return hMem;
- }
-
- switch (cf)
- {
- case CF_METAFILEPICT:
- return m_pPL->RenderMetafilePict();
-
- case CF_BITMAP:
- return (HGLOBAL)m_pPL->RenderBitmap();
- }
-
- return NULL;
- }
-
-
-
-
- //CHAPTER18MOD
- /*
- * CCosmoDoc::RenderMedium
- *
- * Purpose:
- * Like RenderFormat, this function creates a specific data format
- * based on the cf parameter. Unlike RenderFormat, we store the
- * result in a STGMEDIUM in case it has a medium other than
- * TYMED_HGLOBAL. For conveniece we'll centralize all compound
- * document formats here, hGlobal or not.
- *
- * Parameters:
- * cf UINT clipboard format of interest.
- * pSTM LSTGMEDIUM to fill. We only fill the union
- * and tymed.
- *
- * Return Value:
- * BOOL TRUE if we could render the format,
- * FALSE otherwise.
- */
-
- BOOL CCosmoDoc::RenderMedium(UINT cf, LPSTGMEDIUM pSTM)
- {
- if (NULL==pSTM)
- return FALSE;
-
- if (cf==m_cfEmbedSource)
- {
- /*
- * Embed Source data is an IStorage containing the native
- * data (same as Embedded Object). Since our data is small,
- * it makes the most sense to create an IStorage in memory
- * and put transfer that instead of a disk-based IStorage.
- */
-
- pSTM->pstg=INOLE_CreateStorageOnHGlobal(STGM_DIRECT
- | STGM_READWRITE | STGM_SHARE_EXCLUSIVE);
-
- if (NULL==pSTM->pstg)
- return FALSE;
-
- //Now save the data to the storage.
- WriteClassStg(pSTM->pstg, CLSID_CosmoFigure);
- WriteFmtUserTypeStg(pSTM->pstg, m_cf
- , PSZ(IDS_CLIPBOARDFORMAT));
-
- if (POLYLINE_E_NONE!=m_pPL->WriteToStorage(pSTM->pstg
- , VERSIONCURRENT))
- {
- /*
- * When someone releases the IStorage, STORAGE.DLL will
- * release the ILockBytes which, having fDeleteOnRelease
- * TRUE (second parameter) will release the memory.
- * That's why we don't have STGM_DELETEONRELEASE on the
- * IStorage.
- */
- pSTM->pstg->Release();
- return FALSE;
- }
-
- pSTM->tymed=TYMED_ISTORAGE;
- return TRUE;
- }
-
- if (cf==m_cfObjectDescriptor)
- {
- SIZEL szl, szlT;
- POINTL ptl;
- RECT rc;
-
- m_pPL->SizeGet(&rc);
- SETSIZEL(szlT, rc.right, rc.bottom);
- XformSizeInPixelsToHimetric(NULL, &szlT, &szl);
-
- SETPOINTL(ptl, 0, 0);
-
- pSTM->hGlobal=INOLE_AllocObjectDescriptor
- (CLSID_CosmoFigure, DVASPECT_CONTENT, szl, ptl
- , OLEMISC_RECOMPOSEONRESIZE, PSZ(IDS_OBJECTDESCRIPTION)
- , NULL);
-
- pSTM->tymed=TYMED_HGLOBAL;
- return (NULL!=pSTM->hGlobal);
- }
-
- return FALSE;
- }
-
-
- //End CHAPTER18MOD
-
-
-
- /*
- * CCosmoDoc::FQueryPaste
- *
- * Purpose:
- * Determines if we can paste data from the clipboard.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * BOOL TRUE if data is available, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::FQueryPaste(void)
- {
- LPDATAOBJECT pIDataObject;
- BOOL fRet;
-
- if (FAILED(OleGetClipboard(&pIDataObject)))
- return FALSE;
-
- fRet=FQueryPasteFromData(pIDataObject);
- pIDataObject->Release();
- return fRet;
- }
-
-
-
- /*
- * CCosmoDoc::FQueryPasteFromData
- * (Protected)
- *
- * Purpose:
- * Determines if we can paste data from a data object.
- *
- * Parameters:
- * pIDataObject LPDATAOBJECT from which we might want to paste.
- *
- * Return Value:
- * BOOL TRUE if data is available, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::FQueryPasteFromData(LPDATAOBJECT pIDataObject)
- {
- FORMATETC fe;
-
- SETDefFormatEtc(fe, m_cf, TYMED_HGLOBAL);
- return (NOERROR==pIDataObject->QueryGetData(&fe));
- }
-
-
-
-
- /*
- * CCosmoDoc::Paste
- *
- * Purpose:
- * Retrieves the private data format from the clipboard and sets it
- * to the current figure in the editor window.
- *
- * Note that if this function is called, then the clipboard format
- * is available because the Paste menu item is only enabled if the
- * format is present.
- *
- * Parameters:
- * hWndFrame HWND of the main window.
- *
- * Return Value:
- * BOOL TRUE if successful, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::Paste(HWND hWndFrame)
- {
- LPDATAOBJECT pIDataObject;
- BOOL fRet;
-
- if (FAILED(OleGetClipboard(&pIDataObject)))
- return FALSE;
-
- fRet=PasteFromData(pIDataObject);
- pIDataObject->Release();
-
- return fRet;
- }
-
-
-
-
- /*
- * CCosmoDoc::PasteFromData
- * (Protected)
- *
- * Purpose:
- * Retrieves the private data format from a data object and sets
- * it to the current figure in the editor window.
- *
- * Parameters:
- * pIDataObject LPDATAOBJECT from which to paste.
- *
- * Return Value:
- * BOOL TRUE if successful, FALSE otherwise.
- */
-
- BOOL CCosmoDoc::PasteFromData(LPDATAOBJECT pIDataObject)
- {
- FORMATETC fe;
- STGMEDIUM stm;
- BOOL fRet;
-
- SETDefFormatEtc(fe, m_cf, TYMED_HGLOBAL);
- fRet=SUCCEEDED(pIDataObject->GetData(&fe, &stm));
-
- if (fRet && NULL!=stm.hGlobal)
- {
- m_pPL->DataSetMem(stm.hGlobal, FALSE, FALSE, TRUE);
- ReleaseStgMedium(&stm);
- FDirtySet(TRUE);
- }
-
- return fRet;
- }
-
-
-
-
- /*
- * CCosmoDoc::TransferObjectCreate
- * (Protected)
- *
- * Purpose:
- * Creates a DataTransferObject and stuffs the current Polyline
- * data into it, used for both clipboard and drag-drop operations.
- *
- * Parameters:
- * fCut BOOL TRUE if we're cutting, FALSE if we're
- * copying.
- *
- * Return Value:
- * LPDATAOBJECT Pointer to the object created, NULL on failure
- */
-
- LPDATAOBJECT CCosmoDoc::TransferObjectCreate(BOOL fCut)
- {
- UINT i;
- HRESULT hr;
- STGMEDIUM stm;
- FORMATETC fe;
- LPDATAOBJECT pIDataObject=NULL;
- //CHAPTER18MOD
- const UINT cFormats=5;
- static UINT rgcf[5]={0, 0, 0, CF_METAFILEPICT, CF_BITMAP};
- static DWORD rgtm[5]={TYMED_HGLOBAL, TYMED_ISTORAGE
- , TYMED_HGLOBAL, TYMED_MFPICT, TYMED_GDI};
- //CHAPTER18MOD
-
- hr=CoCreateInstance(CLSID_DataTransferObject, NULL
- , CLSCTX_INPROC_SERVER, IID_IDataObject
- , (PPVOID)&pIDataObject);
-
- if (FAILED(hr))
- return NULL;
-
- rgcf[0]=m_cf;
- //CHAPTER18MOD
- rgcf[1]=m_cfEmbedSource;
- rgcf[2]=m_cfObjectDescriptor;
- //End CHAPTER18MOD
-
- for (i=0; i < cFormats; i++)
- {
- //CHAPTER18MOD
- /*
- * RenderFormat handles memory handles, but for compound doc
- * formats we need something more. So if RenderFormat fails
- * (which it will for i=1, try our latest addition which
- * writes to a different field in the STGMEDIUM.
- */
- stm.hGlobal=RenderFormat(rgcf[i]);
-
- if (NULL==stm.hGlobal)
- {
- if (!RenderMedium(rgcf[i], &stm))
- continue;
- }
-
- stm.tymed=rgtm[i];
- stm.pUnkForRelease=NULL;
- SETDefFormatEtc(fe, rgcf[i], rgtm[i]);
- pIDataObject->SetData(&fe, &stm, TRUE);
- //End CHAPTER18MOD
- }
-
- return pIDataObject; //Caller now responsible
- }
-
-
-
- /*
- * CCosmoDoc::DropSelectTargetWindow
- * (Protected)
- *
- * Purpose:
- * Creates a thin inverted frame around a window that we use to
- * show the window as a drop target. This is a toggle function:
- * It uses XOR to create the effect so it must be called twice to
- * leave the window as it was.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CCosmoDoc::DropSelectTargetWindow(void)
- {
- HDC hDC;
- RECT rc;
- UINT dd=3;
- HWND hWnd;
-
- hWnd=m_pPL->Window();
- hDC=GetWindowDC(hWnd);
- GetClientRect(hWnd, &rc);
-
- //Frame this window with inverted pixels
-
- //Top
- PatBlt(hDC, rc.left, rc.top, rc.right-rc.left, dd, DSTINVERT);
-
- //Bottom
- PatBlt(hDC, rc.left, rc.bottom-dd, rc.right-rc.left, dd
- , DSTINVERT);
-
- //Left excluding regions already affected by top and bottom
- PatBlt(hDC, rc.left, rc.top+dd, dd, rc.bottom-rc.top-(2*dd)
- , DSTINVERT);
-
- //Right excluding regions already affected by top and bottom
- PatBlt(hDC, rc.right-dd, rc.top+dd, dd, rc.bottom-rc.top-(2*dd)
- , DSTINVERT);
-
- ReleaseDC(hWnd, hDC);
- return;
- }
-
-
-
-
-
- /*
- * CCosmoDoc::ColorSet
- *
- * Purpose:
- * Changes a color used in our contained Polyline.
- *
- * Parameters:
- * iColor UINT index of the color to change.
- * cr COLORREF new color.
- *
- * Return Value:
- * COLORREF Previous color for the given index.
- */
-
- COLORREF CCosmoDoc::ColorSet(UINT iColor, COLORREF cr)
- {
- return m_pPL->ColorSet(iColor, cr);
- }
-
-
-
-
-
- /*
- * CCosmoDoc::ColorGet
- *
- * Purpose:
- * Retrieves a color currently in use in the Polyline.
- *
- * Parameters:
- * iColor UINT index of the color to retrieve.
- *
- * Return Value:
- * COLORREF Current color for the given index.
- */
-
- COLORREF CCosmoDoc::ColorGet(UINT iColor)
- {
- return m_pPL->ColorGet(iColor);
- }
-
-
-
-
-
-
- /*
- * CCosmoDoc::LineStyleSet
- *
- * Purpose:
- * Changes the line style currently used in the Polyline
- *
- * Parameters:
- * iStyle UINT index of the new line style to use.
- *
- * Return Value:
- * UINT Previous line style.
- */
-
-
- UINT CCosmoDoc::LineStyleSet(UINT iStyle)
- {
- return m_pPL->LineStyleSet(iStyle);
- }
-
-
-
-
-
-
-
- /*
- * CCosmoDoc::LineStyleGet
- *
- * Purpose:
- * Retrieves the line style currently used in the Polyline
- *
- * Parameters:
- * None
- *
- * Return Value:
- * UINT Current line style.
- */
-
-
- UINT CCosmoDoc::LineStyleGet(void)
- {
- if (NULL==m_pPL) //m_pPL might not be valid yet
- return 0L;
-
- return m_pPL->LineStyleGet();
- }
-
-
-
-
-
-
-
-
- /*
- * CPolylineAdviseSink::CPolylineAdviseSink
- * CPolylineAdviseSink::~CPolylineAdviseSink
- *
- * Constructor Parameters:
- * pv LPVOID to store in this object
- */
-
- CPolylineAdviseSink::CPolylineAdviseSink(LPVOID pv)
- {
- m_pv=pv;
- return;
- }
-
-
- CPolylineAdviseSink::~CPolylineAdviseSink(void)
- {
- return;
- }
-
-
-
-
-
- /*
- * CPolylineAdviseSink::OnPointChange
- *
- * Purpose:
- * Informs the document that the polyline added or removed a point.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CPolylineAdviseSink::OnPointChange(void)
- {
- PCDocument pDoc=(PCDocument)m_pv;
-
- pDoc->FDirtySet(TRUE);
- return;
- }
-
-
-
-
-
-
- /*
- * CPolylineAdviseSink::OnSizeChange
- *
- * Purpose:
- * Informs the document that the polyline changed size.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CPolylineAdviseSink::OnSizeChange(void)
- {
- PCCosmoDoc pDoc=(PCCosmoDoc)m_pv;
- RECT rc;
- DWORD dwStyle;
-
- /*
- * Polyline window is informing us that it changed size in
- * response to setting it's data. Therefore we have to
- * size ourselves accordingly but without moving the screen
- * position of the polyline window.
- */
-
- pDoc->m_fNoSize=TRUE;
-
- //Set the document window size.
- GetWindowRect(pDoc->m_pPL->Window(), &rc);
- InflateRect(&rc, 8, 8);
-
- //Adjust for a window sans menu
- dwStyle=GetWindowLong(pDoc->m_hWnd, GWL_STYLE);
- AdjustWindowRect(&rc, dwStyle, FALSE);
-
- SetWindowPos(pDoc->m_hWnd, NULL, 0, 0, rc.right-rc.left
- , rc.bottom-rc.top, SWP_NOMOVE | SWP_NOZORDER);
-
- if (NULL!=pDoc->m_pAdv)
- pDoc->m_pAdv->OnSizeChange(pDoc, &rc);
-
- pDoc->m_fNoSize=FALSE;
- pDoc->FDirtySet(TRUE);
-
- return;
- }
-
-
-
-
-
- /*
- * CPolylineAdviseSink::OnDataChange
- *
- * Purpose:
- * Informs the document that the polyline data changed.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CPolylineAdviseSink::OnDataChange(void)
- {
- PCCosmoDoc pDoc=(PCCosmoDoc)m_pv;
-
- if (NULL!=pDoc->m_pAdv)
- pDoc->m_pAdv->OnDataChange(pDoc);
-
- pDoc->FDirtySet(TRUE);
- return;
- }
-
-
-
-
-
- /*
- * CPolylineAdviseSink::OnColorChange
- *
- * Purpose:
- * Informs the document that the polyline data changed a color.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CPolylineAdviseSink::OnColorChange(void)
- {
- PCCosmoDoc pDoc=(PCCosmoDoc)m_pv;
-
- pDoc->FDirtySet(TRUE);
- return;
- }
-
-
-
-
-
- /*
- * CPolylineAdviseSink::OnLineStyleChange
- *
- * Purpose:
- * Informs the document that the polyline changed its line style.
- *
- * Parameters:
- * None
- *
- * Return Value:
- * None
- */
-
- void CPolylineAdviseSink::OnLineStyleChange(void)
- {
- PCCosmoDoc pDoc=(PCCosmoDoc)m_pv;
-
- pDoc->FDirtySet(TRUE);
- return;
- }
-