|
|
/*
* PASTESPL.C * * Implements the OleUIPasteSpecial function which invokes the complete * Paste Special dialog. * * Copyright (c)1992 Microsoft Corporation, All Rights Reserved */
#define STRICT 1
#include "ole2ui.h"
#include "pastespl.h"
#include "common.h"
#include "utility.h"
#include "resimage.h"
#include "iconbox.h"
#include "geticon.h"
#include "icon.h"
#include "regdb.h"
#include <stdlib.h>
OLEDBGDATA
/*
* OleUIPasteSpecial * * Purpose: * Invokes the standard OLE Paste Special dialog box which allows the user * to select the format of the clipboard object to be pasted or paste linked. * * Parameters: * lpPS LPOLEUIPasteSpecial pointing to the in-out structure * for this dialog. * * Return Value: * UINT One of the following codes or one of the standard error codes (OLEUI_ERR_*) * defined in OLE2UI.H, indicating success or error: * OLEUI_OK User selected OK * OLEUI_CANCEL User cancelled the dialog * OLEUI_IOERR_SRCDATAOBJECTINVALID lpSrcDataObject field of OLEUIPASTESPECIAL invalid * OLEUI_IOERR_ARRPASTEENTRIESINVALID arrPasteEntries field of OLEUIPASTESPECIAL invalid * OLEUI_IOERR_ARRLINKTYPESINVALID arrLinkTypes field of OLEUIPASTESPECIAL invalid * OLEUI_PSERR_CLIPBOARDCHANGED Clipboard contents changed while dialog was up */
STDAPI_(UINT) OleUIPasteSpecial(LPOLEUIPASTESPECIAL lpPS) { UINT uRet; HGLOBAL hMemDlg=NULL;
uRet=UStandardValidation((LPOLEUISTANDARD)lpPS, sizeof(OLEUIPASTESPECIAL) , &hMemDlg);
if (uRet != OLEUI_SUCCESS) return uRet;
//Validate PasteSpecial specific fields
if (NULL == lpPS->lpSrcDataObj || IsBadReadPtr(lpPS->lpSrcDataObj, sizeof(IDataObject))) uRet = OLEUI_IOERR_SRCDATAOBJECTINVALID; if (NULL == lpPS->arrPasteEntries || IsBadReadPtr(lpPS->arrPasteEntries, sizeof(OLEUIPASTEENTRY))) uRet = OLEUI_IOERR_ARRPASTEENTRIESINVALID; if (NULL != lpPS->arrLinkTypes && IsBadReadPtr(lpPS->arrLinkTypes, sizeof(UINT))) uRet = OLEUI_IOERR_ARRLINKTYPESINVALID;
if (0!=lpPS->cClsidExclude) { if (NULL!=lpPS->lpClsidExclude && IsBadReadPtr(lpPS->lpClsidExclude , lpPS->cClsidExclude*sizeof(CLSID))) uRet=OLEUI_IOERR_LPCLSIDEXCLUDEINVALID; }
if (uRet >= OLEUI_ERR_STANDARDMIN) { if (NULL != hMemDlg) FreeResource(hMemDlg); return uRet; }
//Now that we've validated everything, we can invoke the dialog.
uRet = UStandardInvocation(PasteSpecialDialogProc, (LPOLEUISTANDARD)lpPS , hMemDlg, MAKEINTRESOURCE(IDD_PASTESPECIAL));
/*
* IF YOU ARE CREATING ANYTHING BASED ON THE RESULTS, DO IT HERE. */
return uRet; }
/*
* PasteSpecialDialogProc * * Purpose: * Implements the OLE Paste Special dialog as invoked through the * OleUIPasteSpecial function. * * Parameters: * Standard * * Return Value: * Standard */
BOOL CALLBACK EXPORT PasteSpecialDialogProc(HWND hDlg, UINT iMsg, WPARAM wParam, LPARAM lParam) { LPOLEUIPASTESPECIAL lpOPS; LPPASTESPECIAL lpPS; BOOL fHook=FALSE; HCURSOR hCursorOld;
//Declare Win16/Win32 compatible WM_COMMAND parameters.
COMMANDPARAMS(wID, wCode, hWndMsg);
//This will fail under WM_INITDIALOG, where we allocate it.
lpPS=(LPPASTESPECIAL)LpvStandardEntry(hDlg, iMsg, wParam, lParam, &fHook);
//If the hook processed the message, we're done.
if (0!=fHook) return fHook;
// Process help message from Change Icon
if (iMsg == uMsgHelp) { PostMessage(lpPS->lpOPS->hWndOwner, uMsgHelp, wParam, lParam); return FALSE; }
//Process the temination message
if (iMsg==uMsgEndDialog) { HWND hwndNextViewer;
// Free the icon/icon-title metafile corresponding to Paste/PasteList option which is not selected
if (lpPS->fLink) OleUIMetafilePictIconFree(lpPS->hMetaPictOD); else OleUIMetafilePictIconFree(lpPS->hMetaPictLSD);
// Free data associated with each list box entry
FreeListData(GetDlgItem(hDlg, ID_PS_PASTELIST)); FreeListData(GetDlgItem(hDlg, ID_PS_PASTELINKLIST));
//Free any specific allocations before calling StandardCleanup
if (lpPS->hObjDesc) GlobalFree(lpPS->hObjDesc); if (lpPS->hLinkSrcDesc) GlobalFree(lpPS->hLinkSrcDesc); if (lpPS->hBuff) GlobalFree(lpPS->hBuff);
// Change the clipboard notification chain
hwndNextViewer = GetProp(hDlg, NEXTCBVIEWER); if (hwndNextViewer != HWND_BROADCAST) { SetProp(hDlg, NEXTCBVIEWER, HWND_BROADCAST); ChangeClipboardChain(hDlg, hwndNextViewer); } RemoveProp(hDlg, NEXTCBVIEWER);
StandardCleanup(lpPS, hDlg); EndDialog(hDlg, wParam); return TRUE; }
switch (iMsg) { case WM_INITDIALOG: hCursorOld = HourGlassOn(); FPasteSpecialInit(hDlg, wParam, lParam); HourGlassOff(hCursorOld); return FALSE;
case WM_DRAWCLIPBOARD: { HWND hwndNextViewer = GetProp(hDlg, NEXTCBVIEWER); HWND hDlg_ChgIcon;
if (hwndNextViewer == HWND_BROADCAST) break;
if (hwndNextViewer) { SendMessage(hwndNextViewer, iMsg, wParam, lParam); // Refresh next viewer in case it got modified
// by the SendMessage() (likely if multiple
// PasteSpecial dialogs are up simultaneously)
hwndNextViewer = GetProp(hDlg, NEXTCBVIEWER); } SetProp(hDlg, NEXTCBVIEWER, HWND_BROADCAST); ChangeClipboardChain(hDlg, hwndNextViewer);
/* OLE2NOTE: if the ChangeIcon dialog is currently up, then
** we need to defer bringing down PasteSpecial dialog ** until after ChangeIcon dialog returns. if the ** ChangeIcon dialog is NOT up, then we can bring down ** the PasteSpecial dialog immediately. */ if ((hDlg_ChgIcon=(HWND)GetProp(hDlg,PROP_HWND_CHGICONDLG))!=NULL) { // ChangeIcon dialog is UP
lpPS->fClipboardChanged = TRUE; } else { // ChangeIcon dialog is NOT up
// Free icon and icon title metafile
SendDlgItemMessage( hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGEFREE, 0, 0L);
SendMessage( hDlg, uMsgEndDialog, OLEUI_PSERR_CLIPBOARDCHANGED,0L); } break; }
case WM_CHANGECBCHAIN: { HWND hwndNextViewer = GetProp(hDlg, NEXTCBVIEWER);
if (wParam == (WORD)hwndNextViewer) SetProp(hDlg, NEXTCBVIEWER, (hwndNextViewer = (HWND)LOWORD(lParam))); else if (hwndNextViewer && hwndNextViewer != HWND_BROADCAST) SendMessage(hwndNextViewer, iMsg, wParam, lParam); break; }
case WM_COMMAND: switch (wID) { case ID_PS_PASTE: FTogglePasteType(hDlg, lpPS, PSF_SELECTPASTE); break;
case ID_PS_PASTELINK: FTogglePasteType(hDlg, lpPS, PSF_SELECTPASTELINK); break;
case ID_PS_DISPLAYLIST: switch (wCode) { case LBN_SELCHANGE: ChangeListSelection(hDlg, lpPS, hWndMsg); break;
case LBN_DBLCLK: // Same as pressing OK
SendCommand(hDlg, IDOK, BN_CLICKED, hWndMsg); break; } break;
case ID_PS_DISPLAYASICON: ToggleDisplayAsIcon(hDlg, lpPS); break;
case ID_PS_CHANGEICON: ChangeIcon(hDlg, lpPS); if (lpPS->fClipboardChanged) { // Free icon and icon title metafile
SendDlgItemMessage( hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGEFREE,0,0L); SendMessage( hDlg, uMsgEndDialog, OLEUI_PSERR_CLIPBOARDCHANGED, 0L); } break;
case IDOK: { BOOL fDestAspectIcon = ((lpPS->dwFlags & PSF_CHECKDISPLAYASICON) ? TRUE : FALSE); lpOPS = lpPS->lpOPS; // Return current flags
lpOPS->dwFlags = lpPS->dwFlags; // Return index of arrPasteEntries[] corresponding to format selected by user
lpOPS->nSelectedIndex = lpPS->nSelectedIndex; // Return if user selected Paste or PasteLink
lpOPS->fLink = lpPS->fLink;
/* if user selected same ASPECT as displayed in the
** source, then sizel passed in the ** ObjectDescriptor/LinkSrcDescriptor is ** applicable. otherwise, the sizel does not apply. */ if (lpPS->fLink) { if (lpPS->fSrcAspectIconLSD == fDestAspectIcon) lpOPS->sizel = lpPS->sizelLSD; else lpOPS->sizel.cx = lpOPS->sizel.cy = 0; } else { if (lpPS->fSrcAspectIconOD == fDestAspectIcon) lpOPS->sizel = lpPS->sizelOD; else lpOPS->sizel.cx = lpOPS->sizel.cy = 0; } // Return metafile with icon and icon title that the user selected
lpOPS->hMetaPict=(HGLOBAL)SendDlgItemMessage(hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGEGET, 0, 0L); SendMessage(hDlg, uMsgEndDialog, OLEUI_OK, 0L); break; } case IDCANCEL: // Free icon and icon title metafile
SendDlgItemMessage( hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGEFREE, 0, 0L); SendMessage(hDlg, uMsgEndDialog, OLEUI_CANCEL, 0L); break;
case ID_OLEUIHELP: PostMessage(lpPS->lpOPS->hWndOwner, uMsgHelp, (WPARAM)hDlg, MAKELPARAM(IDD_PASTESPECIAL, 0)); break; } break; } return FALSE; }
/*
* FPasteSpecialInit * * Purpose: * WM_INITIDIALOG handler for the Paste Special dialog box. * * Parameters: * hDlg HWND of the dialog * wParam WPARAM of the message * lParam LPARAM of the message * * Return Value: * BOOL Value to return for WM_INITDIALOG. */
BOOL FPasteSpecialInit(HWND hDlg, WPARAM wParam, LPARAM lParam) { LPPASTESPECIAL lpPS; LPOLEUIPASTESPECIAL lpOPS; HFONT hFont; BOOL fPasteAvailable, fPasteLinkAvailable; STGMEDIUM medium; LPOBJECTDESCRIPTOR lpOD; LPLINKSRCDESCRIPTOR lpLSD; int n; CLIPFORMAT cfFormat;
// Copy the structure at lParam into our instance memory.
lpPS = (LPPASTESPECIAL)LpvStandardInit(hDlg, sizeof(PASTESPECIAL), TRUE, &hFont);
// PvStandardInit sent a termination to us already.
if (NULL == lpPS) return FALSE;
lpOPS=(LPOLEUIPASTESPECIAL)lParam;
// Copy other information from lpOPS that we might modify.
lpPS->lpOPS = lpOPS; lpPS->dwFlags = lpOPS->dwFlags;
// Initialize user selections in the Paste and PasteLink listboxes
lpPS->nPasteListCurSel = 0; lpPS->nPasteLinkListCurSel = 0;
// If we got a font, send it to the necessary controls.
if (NULL!=hFont) { SendDlgItemMessage(hDlg, ID_PS_SOURCETEXT, WM_SETFONT, (WPARAM)hFont, 0L); SendDlgItemMessage(hDlg, ID_PS_RESULTTEXT, WM_SETFONT, (WPARAM)hFont, 0L); }
// Hide the help button if required
if (!(lpPS->lpOPS->dwFlags & PSF_SHOWHELP)) StandardShowDlgItem(hDlg, ID_OLEUIHELP, SW_HIDE);
// Hide all DisplayAsIcon related controls if it should be disabled
if ( lpPS->dwFlags & PSF_DISABLEDISPLAYASICON ) { StandardShowDlgItem(hDlg, ID_PS_DISPLAYASICON, SW_HIDE); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_HIDE); StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_HIDE); }
// PSF_CHECKDISPLAYASICON is an OUT flag. Clear it if has been set on the way in.
lpPS->dwFlags = lpPS->dwFlags & ~PSF_CHECKDISPLAYASICON;
// Change the caption if required
if (NULL != lpOPS->lpszCaption) SetWindowText(hDlg, lpOPS->lpszCaption);
// Load 'Unknown Source' and 'Unknown Type' strings
n = LoadString(ghInst, IDS_PSUNKNOWNTYPE, lpPS->szUnknownType, PS_UNKNOWNSTRLEN); if (n) n = LoadString(ghInst, IDS_PSUNKNOWNSRC, lpPS->szUnknownSource, PS_UNKNOWNSTRLEN); if (!n) { PostMessage(hDlg, uMsgEndDialog, OLEUI_ERR_LOADSTRING, 0L); return FALSE; } lpPS->szAppName[0]=TEXT('\0');
// GetData CF_OBJECTDESCRIPTOR. If the object on the clipboard in an OLE1 object (offering CF_OWNERLINK)
// or has been copied to clipboard by FileMaager (offering CF_FILENAME), an OBJECTDESCRIPTOR will be
// created will be created from CF_OWNERLINK or CF_FILENAME. See OBJECTDESCRIPTOR for more info.
if (lpPS->hObjDesc = OleStdFillObjectDescriptorFromData(lpOPS->lpSrcDataObj, &medium, &cfFormat)) { lpOD = GlobalLock(lpPS->hObjDesc);
// Get FullUserTypeName, SourceOfCopy and CLSID
if (lpOD->dwFullUserTypeName) lpPS->szFullUserTypeNameOD = (LPTSTR)lpOD+lpOD->dwFullUserTypeName; else lpPS->szFullUserTypeNameOD = lpPS->szUnknownType;
if (lpOD->dwSrcOfCopy) { lpPS->szSourceOfDataOD = (LPTSTR)lpOD+lpOD->dwSrcOfCopy; // If CF_FILENAME was offered, source of copy is a path name. Fit the path to the
// static control that will display it.
if (cfFormat == cfFileName) lpPS->szSourceOfDataOD = ChopText(GetDlgItem(hDlg, ID_PS_SOURCETEXT), 0, lpPS->szSourceOfDataOD); } else lpPS->szSourceOfDataOD = lpPS->szUnknownSource;
lpPS->clsidOD = lpOD->clsid; lpPS->sizelOD = lpOD->sizel;
// Does source specify DVASPECT_ICON?
if (lpOD->dwDrawAspect & DVASPECT_ICON) lpPS->fSrcAspectIconOD = TRUE; else lpPS->fSrcAspectIconOD = FALSE;
// Does source specify OLEMISC_ONLYICONIC?
if (lpOD->dwStatus & OLEMISC_ONLYICONIC) lpPS->fSrcOnlyIconicOD = TRUE; else lpPS->fSrcOnlyIconicOD = FALSE;
// Get application name of source from auxusertype3 in the registration database
if (0==OleStdGetAuxUserType(&lpPS->clsidOD, 3, lpPS->szAppName, OLEUI_CCHKEYMAX_SIZE, NULL)) { // Use "the application which created it" as the name of the application
if (0==LoadString(ghInst, IDS_PSUNKNOWNAPP, lpPS->szAppName, PS_UNKNOWNSTRLEN)) { PostMessage(hDlg, uMsgEndDialog, OLEUI_ERR_LOADSTRING, 0L); return FALSE; } }
// Retrieve an icon from the object
if (lpPS->fSrcAspectIconOD) { lpPS->hMetaPictOD = OleStdGetData( lpOPS->lpSrcDataObj, (CLIPFORMAT) CF_METAFILEPICT, NULL, DVASPECT_ICON, &medium );
} // If object does not offer icon, obtain it from the CLSID
if (NULL == lpPS->hMetaPictOD) { #ifdef OLE201
lpPS->hMetaPictOD = GetIconOfClass( ghInst, &lpPS->clsidOD, NULL, TRUE); // Use the short user type name (auxusertype3)
#endif
lpPS->hMetaPictOD = NULL;
} }
// Does object offer CF_LINKSRCDESCRIPTOR?
if (lpPS->hLinkSrcDesc = OleStdGetData( lpOPS->lpSrcDataObj, (CLIPFORMAT) cfLinkSrcDescriptor, NULL, DVASPECT_CONTENT, &medium)) { // Get FullUserTypeName, SourceOfCopy and CLSID
lpLSD = GlobalLock(lpPS->hLinkSrcDesc); if (lpLSD->dwFullUserTypeName) lpPS->szFullUserTypeNameLSD = (LPTSTR)lpLSD+lpLSD->dwFullUserTypeName; else lpPS->szFullUserTypeNameLSD = lpPS->szUnknownType;
if (lpLSD->dwSrcOfCopy) lpPS->szSourceOfDataLSD = (LPTSTR)lpLSD+lpLSD->dwSrcOfCopy; else lpPS->szSourceOfDataLSD = lpPS->szUnknownSource;
// if no ObjectDescriptor, then use LinkSourceDescriptor source string
if (!lpPS->hObjDesc) lpPS->szSourceOfDataOD = lpPS->szSourceOfDataLSD;
lpPS->clsidLSD = lpLSD->clsid; lpPS->sizelLSD = lpLSD->sizel;
// Does source specify DVASPECT_ICON?
if (lpLSD->dwDrawAspect & DVASPECT_ICON) lpPS->fSrcAspectIconLSD = TRUE; else lpPS->fSrcAspectIconLSD = FALSE;
// Does source specify OLEMISC_ONLYICONIC?
if (lpLSD->dwStatus & OLEMISC_ONLYICONIC) lpPS->fSrcOnlyIconicLSD = TRUE; else lpPS->fSrcOnlyIconicLSD = FALSE;
// Retrieve an icon from the object
if (lpPS->fSrcAspectIconLSD) { lpPS->hMetaPictLSD = OleStdGetData( lpOPS->lpSrcDataObj, CF_METAFILEPICT, NULL, DVASPECT_ICON, &medium );
} // If object does not offer icon, obtain it from the CLSID
if (NULL == lpPS->hMetaPictLSD) { TCHAR szLabel[OLEUI_CCHLABELMAX]; HWND hIconWnd; RECT IconRect; int nWidth; LPTSTR lpszLabel;
hIconWnd = GetDlgItem(hDlg, ID_PS_ICONDISPLAY);
GetClientRect(hIconWnd, &IconRect);
nWidth = ((IconRect.right-IconRect.left) * 3) / 2; // width is 1.5 times width of iconbox
LSTRCPYN(szLabel, lpPS->szSourceOfDataLSD, OLEUI_CCHLABELMAX); szLabel[OLEUI_CCHLABELMAX-1] = TEXT('\0');
lpszLabel = ChopText(hIconWnd, nWidth, (LPTSTR)szLabel);
#ifdef OLE201
lpPS->hMetaPictLSD = GetIconOfClass( ghInst, &lpPS->clsidLSD, lpszLabel, /* use chopped source string as label */ FALSE /* not applicable */ ); #endif
lpPS->hMetaPictLSD = NULL;
} } else if (lpPS->hObjDesc) // Does not offer CF_LINKSRCDESCRIPTOR but offers CF_OBJECTDESCRIPTOR
{ // Copy the values of OBJECTDESCRIPTOR
lpPS->szFullUserTypeNameLSD = lpPS->szFullUserTypeNameOD; lpPS->szSourceOfDataLSD = lpPS->szSourceOfDataOD; lpPS->clsidLSD = lpPS->clsidOD; lpPS->sizelLSD = lpPS->sizelOD; lpPS->fSrcAspectIconLSD = lpPS->fSrcAspectIconOD; lpPS->fSrcOnlyIconicLSD = lpPS->fSrcOnlyIconicOD;
// Don't copy the hMetaPict; instead get a separate copy
if (lpPS->fSrcAspectIconLSD) { lpPS->hMetaPictLSD = OleStdGetData( lpOPS->lpSrcDataObj, CF_METAFILEPICT, NULL, DVASPECT_ICON, &medium ); } if (NULL == lpPS->hMetaPictLSD) { TCHAR szLabel[OLEUI_CCHLABELMAX]; HWND hIconWnd; RECT IconRect; int nWidth; LPTSTR lpszLabel;
hIconWnd = GetDlgItem(hDlg, ID_PS_ICONDISPLAY);
GetClientRect(hIconWnd, &IconRect);
nWidth = ((IconRect.right-IconRect.left) * 3) / 2; // width is 1.5 times width of iconbox
LSTRCPYN(szLabel, lpPS->szSourceOfDataLSD, OLEUI_CCHLABELMAX); szLabel[OLEUI_CCHLABELMAX-1] = TEXT('\0');
lpszLabel = ChopText(hIconWnd, nWidth, (LPTSTR)szLabel);
#ifdef OLE201
lpPS->hMetaPictLSD = GetIconOfClass( ghInst, &lpPS->clsidLSD, lpszLabel, /* Use chopped source string as label */ FALSE /* Not applicable */ ); #endif
lpPS->hMetaPictLSD = NULL;
} }
// Not an OLE object
if (lpPS->hObjDesc == NULL && lpPS->hLinkSrcDesc == NULL) { lpPS->szFullUserTypeNameLSD = lpPS->szFullUserTypeNameOD = lpPS->szUnknownType; lpPS->szSourceOfDataLSD = lpPS->szSourceOfDataOD = lpPS->szUnknownSource; lpPS->hMetaPictLSD = lpPS->hMetaPictOD = NULL; }
// Allocate scratch memory to construct item names in the paste and pastelink listboxes
lpPS->hBuff = AllocateScratchMem(lpPS); if (lpPS->hBuff == NULL) { PostMessage(hDlg, uMsgEndDialog, OLEUI_ERR_GLOBALMEMALLOC, 0L); return FALSE; }
// Select the Paste Link Button if specified. Otherwise select
// Paste Button by default
if (lpPS->dwFlags & PSF_SELECTPASTELINK) lpPS->dwFlags = (lpPS->dwFlags & ~PSF_SELECTPASTE) | PSF_SELECTPASTELINK; else lpPS->dwFlags =(lpPS->dwFlags & ~PSF_SELECTPASTELINK) | PSF_SELECTPASTE;
// Mark which PasteEntry formats are available from source data object
OleStdMarkPasteEntryList( lpOPS->lpSrcDataObj,lpOPS->arrPasteEntries,lpOPS->cPasteEntries);
// Check if items are available to be pasted
fPasteAvailable = FFillPasteList(hDlg, lpPS); if (!fPasteAvailable) { lpPS->dwFlags &= ~PSF_SELECTPASTE; EnableWindow(GetDlgItem(hDlg, ID_PS_PASTE), FALSE); }
// Check if items are available to be paste-linked
fPasteLinkAvailable = FFillPasteLinkList(hDlg, lpPS); if (!fPasteLinkAvailable) { lpPS->dwFlags &= ~PSF_SELECTPASTELINK; EnableWindow(GetDlgItem(hDlg, ID_PS_PASTELINK), FALSE); }
// If one of Paste or PasteLink is disabled, select the other one
// regardless of what the input flags say
if (fPasteAvailable && !fPasteLinkAvailable) lpPS->dwFlags |= PSF_SELECTPASTE; if (fPasteLinkAvailable && !fPasteAvailable) lpPS->dwFlags |= PSF_SELECTPASTELINK;
if (lpPS->dwFlags & PSF_SELECTPASTE) { // FTogglePaste will set the PSF_SELECTPASTE flag, so clear it.
lpPS->dwFlags &= ~PSF_SELECTPASTE; CheckRadioButton(hDlg, ID_PS_PASTE, ID_PS_PASTELINK, ID_PS_PASTE); FTogglePasteType(hDlg, lpPS, PSF_SELECTPASTE); } else if (lpPS->dwFlags & PSF_SELECTPASTELINK) { // FTogglePaste will set the PSF_SELECTPASTELINK flag, so clear it.
lpPS->dwFlags &= ~PSF_SELECTPASTELINK; CheckRadioButton(hDlg, ID_PS_PASTE, ID_PS_PASTELINK, ID_PS_PASTELINK); FTogglePasteType(hDlg, lpPS, PSF_SELECTPASTELINK); } else // Items are not available to be be Pasted or Paste-Linked
{ // Enable or disable DisplayAsIcon and set the result text and image
EnableDisplayAsIcon(hDlg, lpPS); SetPasteSpecialHelpResults(hDlg, lpPS); }
// Give initial focus to the list box
SetFocus(GetDlgItem(hDlg, ID_PS_DISPLAYLIST));
// Set property to handle clipboard change notifications
SetProp(hDlg, NEXTCBVIEWER, HWND_BROADCAST); SetProp(hDlg, NEXTCBVIEWER, SetClipboardViewer(hDlg));
lpPS->fClipboardChanged = FALSE;
/*
* PERFORM OTHER INITIALIZATION HERE. */
// Call the hook with lCustData in lParam
UStandardHook(lpPS, hDlg, WM_INITDIALOG, wParam, lpOPS->lCustData); return TRUE; }
/*
* FTogglePasteType * * Purpose: * Toggles between Paste and Paste Link. The Paste list and PasteLink * list are always invisible. The Display List is filled from either * the Paste list or the PasteLink list depending on which Paste radio * button is selected. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * dwOption Paste or PasteSpecial option * * Return Value: * BOOL Returns TRUE if the option has already been selected. * Otherwise the option is selected and FALSE is returned */
BOOL FTogglePasteType(HWND hDlg, LPPASTESPECIAL lpPS, DWORD dwOption) { DWORD dwTemp; HWND hList, hListDisplay; DWORD dwData; int i, nItems; LPTSTR lpsz;
// Skip all this if the button is already selected
if (lpPS->dwFlags & dwOption) return TRUE;
dwTemp = PSF_SELECTPASTE | PSF_SELECTPASTELINK; lpPS->dwFlags = (lpPS->dwFlags & ~dwTemp) | dwOption;
// Hide IconDisplay. This prevents flashing if the icon display is changed
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_HIDE);
hListDisplay = GetDlgItem(hDlg, ID_PS_DISPLAYLIST);
// If Paste was selected
if (lpPS->dwFlags & PSF_SELECTPASTE) { // Set the Source of the object in the clipboard
SetDlgItemText(hDlg, ID_PS_SOURCETEXT, lpPS->szSourceOfDataOD);
// If an icon is available
if (lpPS->hMetaPictOD) // Set the icon display
SendDlgItemMessage(hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGESET, (WPARAM)lpPS->hMetaPictOD, 0L);
hList = GetDlgItem(hDlg, ID_PS_PASTELIST); // We are switching from PasteLink to Paste. Remember current selection
// in PasteLink list so it can be restored.
lpPS->nPasteLinkListCurSel = (int)SendMessage(hListDisplay, LB_GETCURSEL, 0, 0L); if (lpPS->nPasteLinkListCurSel == LB_ERR) lpPS->nPasteLinkListCurSel = 0; // Remember if user selected Paste or PasteLink
lpPS->fLink = FALSE; } else // If PasteLink was selected
{ // Set the Source of the object in the clipboard
SetDlgItemText(hDlg, ID_PS_SOURCETEXT, lpPS->szSourceOfDataLSD);
// If an icon is available
if (lpPS->hMetaPictLSD) // Set the icon display
SendDlgItemMessage(hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGESET, (WPARAM)lpPS->hMetaPictLSD, 0L);
hList = GetDlgItem(hDlg, ID_PS_PASTELINKLIST); // We are switching from Paste to PasteLink. Remember current selection
// in Paste list so it can be restored.
lpPS->nPasteListCurSel = (int)SendMessage(hListDisplay, LB_GETCURSEL, 0, 0L); if (lpPS->nPasteListCurSel == LB_ERR) lpPS->nPasteListCurSel = 0; // Remember if user selected Paste or PasteLink
lpPS->fLink = TRUE; }
// Turn drawing off while the Display List is being filled
SendMessage(hListDisplay, WM_SETREDRAW, (WPARAM)FALSE, 0L);
// Move data to Display list box
SendMessage(hListDisplay, LB_RESETCONTENT, 0, 0L); nItems = (int) SendMessage(hList, LB_GETCOUNT, 0, 0L); lpsz = (LPTSTR)GlobalLock(lpPS->hBuff); for (i = 0; i < nItems; i++) { SendMessage(hList, LB_GETTEXT, (WPARAM)i, (LPARAM)lpsz); dwData = SendMessage(hList, LB_GETITEMDATA, (WPARAM)i, 0L); SendMessage(hListDisplay, LB_INSERTSTRING, (WPARAM)i, (LPARAM)lpsz); SendMessage(hListDisplay, LB_SETITEMDATA, (WPARAM)i, dwData); } GlobalUnlock(lpPS->hBuff);
// Restore the selection in the Display List from user's last selection
if (lpPS->dwFlags & PSF_SELECTPASTE) SendMessage(hListDisplay, LB_SETCURSEL, lpPS->nPasteListCurSel, 0L); else SendMessage(hListDisplay, LB_SETCURSEL, lpPS->nPasteLinkListCurSel, 0L);
// Paint Display List
SendMessage(hListDisplay, WM_SETREDRAW, (WPARAM)TRUE, 0L); InvalidateRect(hListDisplay, NULL, TRUE); UpdateWindow(hListDisplay);
// Auto give the focus to the Display List
SetFocus(hListDisplay);
// Enable/Disable DisplayAsIcon and set the help result text and bitmap corresponding to
// the current selection
ChangeListSelection(hDlg, lpPS, hListDisplay);
return FALSE; }
/*
* ChangeListSelection * * Purpose: * When the user changes the selection in the list, DisplayAsIcon is enabled or disabled, * Result text and bitmap are updated and the index of the arrPasteEntries[] corresponding * to the current format selection is saved. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * hList HWND of the List * * Return Value: * No return value */
void ChangeListSelection(HWND hDlg, LPPASTESPECIAL lpPS, HWND hList) { LPPASTELISTITEMDATA lpItemData; int nCurSel;
EnableDisplayAsIcon(hDlg, lpPS); SetPasteSpecialHelpResults(hDlg, lpPS);
// Remember index of arrPasteEntries[] corresponding to the current selection
nCurSel = (int)SendMessage(hList, LB_GETCURSEL, 0, 0L); if (nCurSel == LB_ERR) return; lpItemData = (LPPASTELISTITEMDATA) SendMessage(hList, LB_GETITEMDATA, (WPARAM)nCurSel, 0L); if ((LRESULT)lpItemData == LB_ERR) return; lpPS->nSelectedIndex = lpItemData->nPasteEntriesIndex; }
/*
* EnableDisplayAsIcon * * Purpose: * Enable or disable the DisplayAsIcon button depending on whether * the current selection can be displayed as an icon or not. The following table describes * the state of DisplayAsIcon. The calling application is termed CONTAINER, the source * of data on the clipboard is termed SOURCE. * Y = Yes; N = No; Blank = State does not matter; * ===================================================================== * SOURCE SOURCE CONTAINER DisplayAsIcon * specifies specifies specifies Initial State * DVASPECT_ICON OLEMISC_ONLYICONIC OLEUIPASTE_ENABLEICON * * N Unchecked&Disabled * Y Y Checked&Disabled * Y N Y Checked&Enabled * N N Y Unchecked&Enabled * ===================================================================== * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * No return value */
void EnableDisplayAsIcon(HWND hDlg, LPPASTESPECIAL lpPS) { int nIndex; BOOL fCntrEnableIcon; BOOL fSrcOnlyIconic = (lpPS->fLink) ? lpPS->fSrcOnlyIconicLSD : lpPS->fSrcOnlyIconicOD; BOOL fSrcAspectIcon = (lpPS->fLink) ? lpPS->fSrcAspectIconLSD : lpPS->fSrcAspectIconOD; HWND hList; LPPASTELISTITEMDATA lpItemData; HGLOBAL hMetaPict = (lpPS->fLink) ? lpPS->hMetaPictLSD : lpPS->hMetaPictOD;
hList = GetDlgItem(hDlg, ID_PS_DISPLAYLIST);
// Get data corresponding to the current selection in the listbox
nIndex = (int)SendMessage(hList, LB_GETCURSEL, 0, 0); if (nIndex != LB_ERR) { lpItemData = (LPPASTELISTITEMDATA) SendMessage(hList, LB_GETITEMDATA, (WPARAM)nIndex, 0L); if ((LRESULT)lpItemData != LB_ERR) fCntrEnableIcon = lpItemData->fCntrEnableIcon; else fCntrEnableIcon = FALSE; } else fCntrEnableIcon = FALSE;
// If there is an icon available
if (hMetaPict != NULL) { if (!fCntrEnableIcon) // Does CONTAINER specify OLEUIPASTE_ENABLEICON?
{ // Uncheck & Disable DisplayAsIcon
lpPS->dwFlags &= ~PSF_CHECKDISPLAYASICON; CheckDlgButton(hDlg, ID_PS_DISPLAYASICON, FALSE); EnableWindow(GetDlgItem(hDlg, ID_PS_DISPLAYASICON), FALSE);
// Hide IconDisplay and ChangeIcon button
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_HIDE); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_HIDE); } else if (fSrcOnlyIconic) // Does SOURCE specify OLEMISC_ONLYICONIC?
{ // Check & Disable DisplayAsIcon
lpPS->dwFlags |= PSF_CHECKDISPLAYASICON; CheckDlgButton(hDlg, ID_PS_DISPLAYASICON, TRUE); EnableWindow(GetDlgItem(hDlg, ID_PS_DISPLAYASICON), FALSE);
// Show IconDisplay and ChangeIcon button
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_SHOWNORMAL); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_SHOWNORMAL); } else if (fSrcAspectIcon) // Does SOURCE specify DVASPECT_ICON?
{ // Check & Enable DisplayAsIcon
lpPS->dwFlags |= PSF_CHECKDISPLAYASICON; CheckDlgButton(hDlg, ID_PS_DISPLAYASICON, TRUE); EnableWindow(GetDlgItem(hDlg, ID_PS_DISPLAYASICON), TRUE);
// Show IconDisplay and ChangeIcon button
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_SHOWNORMAL); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_SHOWNORMAL); } else { //Uncheck and Enable DisplayAsIcon
lpPS->dwFlags &= ~PSF_CHECKDISPLAYASICON; CheckDlgButton(hDlg, ID_PS_DISPLAYASICON, FALSE); EnableWindow(GetDlgItem(hDlg, ID_PS_DISPLAYASICON), TRUE);
// Hide IconDisplay and ChangeIcon button
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_HIDE); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_HIDE);
} } else // No icon available
{ // Unchecked & Disabled
lpPS->dwFlags &= ~PSF_CHECKDISPLAYASICON; CheckDlgButton(hDlg, ID_PS_DISPLAYASICON, FALSE); EnableWindow(GetDlgItem(hDlg, ID_PS_DISPLAYASICON), FALSE);
// Hide IconDisplay and ChangeIcon button
StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, SW_HIDE); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, SW_HIDE); } }
/*
* ToggleDisplayAsIcon * * Purpose: * Toggles the DisplayAsIcon button. Hides or shows the Icon Display and * the ChangeIcon button and changes the help result text and bitmap. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * None * */
void ToggleDisplayAsIcon(HWND hDlg, LPPASTESPECIAL lpPS) { BOOL fCheck; int i;
fCheck = IsDlgButtonChecked(hDlg, ID_PS_DISPLAYASICON);
if (fCheck) lpPS->dwFlags |= PSF_CHECKDISPLAYASICON; else lpPS->dwFlags &= ~PSF_CHECKDISPLAYASICON;
// Set the help result text and bitmap
SetPasteSpecialHelpResults(hDlg, lpPS);
// Show or hide the Icon Display and ChangeIcon button depending
// on the check state
i = (fCheck) ? SW_SHOWNORMAL : SW_HIDE; StandardShowDlgItem(hDlg, ID_PS_ICONDISPLAY, i); StandardShowDlgItem(hDlg, ID_PS_CHANGEICON, i); }
/*
* ChangeIcon * * Purpose: * Brings up the ChangeIcon dialog which allows the user to change * the icon and label. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * None * */
void ChangeIcon(HWND hDlg, LPPASTESPECIAL lpPS) { OLEUICHANGEICON ci; UINT uRet; CLSID clsid = (lpPS->fLink) ? lpPS->clsidLSD : lpPS->clsidOD;
//Initialize the structure
_fmemset((LPOLEUICHANGEICON)&ci, 0, sizeof(ci));
ci.hMetaPict = (HGLOBAL)SendDlgItemMessage(hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGEGET, 0, 0L); ci.cbStruct = sizeof(ci); ci.hWndOwner = hDlg; ci.clsid = clsid; ci.dwFlags = CIF_SELECTCURRENT;
// Only show help in the ChangeIcon dialog if we're showing it in this dialog.
if (lpPS->dwFlags & PSF_SHOWHELP) ci.dwFlags |= CIF_SHOWHELP;
// Let the hook in to customize Change Icon if desired.
uRet = UStandardHook(lpPS, hDlg, uMsgChangeIcon, 0, (LONG)(LPSTR)&ci);
if (0 == uRet) uRet=(UINT)(OLEUI_OK==OleUIChangeIcon(&ci));
// Update the display if necessary.
if (0!=uRet) { /*
* OleUIChangeIcon will have already freed our * current hMetaPict that we passed in when OK is * pressed in that dialog. So we use 0L as lParam * here so the IconBox doesn't try to free the * metafilepict again. */ SendDlgItemMessage(hDlg, ID_PS_ICONDISPLAY, IBXM_IMAGESET, (WPARAM)ci.hMetaPict, 0L); // Remember the new icon chosen by the user. Note that Paste and PasteLink have separate
// icons - changing one does not change the other.
if (lpPS->fLink) lpPS->hMetaPictLSD = ci.hMetaPict; else lpPS->hMetaPictOD = ci.hMetaPict; } }
/*
*SetPasteSpecialHelpResults * * Purpose: * Sets the help result text and bitmap according to the current * list selection. The following state table indicates which ResultText * and ResultImage are selected. If %s in the lpstrFormatName is present, * it is assumed that an object is being pasted/paste-linked, otherwise it * is assumed that data is being pasted/paste-linked. * Y = Yes; N = No; Blank = State does not matter; * The numbers in the the ResultText and ResultImage columns refer to the table * entries that follow. * ===================================================================== * Paste/ lpstrFormatName in DisplayAsIcon Result Result * PasteLink arrPasteEntry[]contains %s checked Text Image * (Is Object == Y, Is Data == N) * Paste N 1 1 * Paste Y N 2 2 * Paste Y Y 3 3 * PasteLink N 4 4 * PasteLink Y N 5 4 * PasteLink Y Y 6 5 * ===================================================================== * Result Text: * * 1. "Inserts the contents of the Clipboard into your document as <native type name, * and optionally an additional help sentence>" * 2. "Inserts the contents of the Clipboard into your document so that you may * activate it using <object app name>" * 3. "Inserts the contents of the Clipboard into your document so that you may * activate it using <object app name>. It will be displayed as an icon." * 4. "Inserts the contents of the Clipboard into your document as <native type name>. * Paste Link creates a link to the source file so that changes to the source file * will be reflected in your document." * 5. "Inserts a picture of the Clipboard contents into your document. Paste Link * creates a link to the source file so that changes to the source file will be * reflected in your document." * 6. "Inserts an icon into your document which represents the Clipboard contents. * Paste Link creates a link to the source file so that changes to the source file * will be reflected in your document." * ===================================================================== * Result Image: * * 1. Clipboard Image * 2. Paste image, non-iconic. * 3. Paste image, iconic. * 4. Paste Link image, non-iconic * 5. Paste Link image, iconic * ==================================================================== * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * No return value */ void SetPasteSpecialHelpResults(HWND hDlg, LPPASTESPECIAL lpPS) { LPTSTR psz1, psz2, psz3, psz4; UINT i, iString, iImage, cch; int nPasteEntriesIndex; BOOL fDisplayAsIcon; BOOL fIsObject; HWND hList; LPPASTELISTITEMDATA lpItemData; LPOLEUIPASTESPECIAL lpOPS = lpPS->lpOPS; LPTSTR szFullUserTypeName = (lpPS->fLink) ? lpPS->szFullUserTypeNameLSD : lpPS->szFullUserTypeNameOD; LPTSTR szInsert;
hList = GetDlgItem(hDlg, ID_PS_DISPLAYLIST);
i=(UINT)SendMessage(hList, LB_GETCURSEL, 0, 0L); if (i != LB_ERR) { lpItemData = (LPPASTELISTITEMDATA)SendMessage(hList, LB_GETITEMDATA, i, 0L); if ((LRESULT)lpItemData == LB_ERR) return; nPasteEntriesIndex = lpItemData->nPasteEntriesIndex; // Check if there is a '%s' in the lpstrFormatName, then an object is being
// pasted/pastelinked. Otherwise Data is being pasted-pastelinked.
fIsObject = FHasPercentS(lpOPS->arrPasteEntries[nPasteEntriesIndex].lpstrFormatName, lpPS); } else return;
// Is DisplayAsIcon checked?
fDisplayAsIcon=(0L!=(lpPS->dwFlags & PSF_CHECKDISPLAYASICON));
szInsert = szFullUserTypeName;
if (lpPS->dwFlags & PSF_SELECTPASTE) // If user selected Paste
{ if (fIsObject) { iString = fDisplayAsIcon ? IDS_PSPASTEOBJECTASICON : IDS_PSPASTEOBJECT; iImage = fDisplayAsIcon ? RESULTIMAGE_EMBEDICON : RESULTIMAGE_EMBED; szInsert = lpPS->szAppName; } else { iString = IDS_PSPASTEDATA; iImage = RESULTIMAGE_PASTE; } } else if (lpPS->dwFlags & PSF_SELECTPASTELINK) // User selected PasteLink
{ if (fIsObject) { iString = fDisplayAsIcon ? IDS_PSPASTELINKOBJECTASICON : IDS_PSPASTELINKOBJECT; iImage = fDisplayAsIcon ? RESULTIMAGE_LINKICON : RESULTIMAGE_LINK; } else { iString = IDS_PSPASTELINKDATA; iImage = RESULTIMAGE_LINK; }
} else // Should never occur.
{ iString = IDS_PSNONOLE; iImage = RESULTIMAGE_PASTE; }
// hBuff contains enough space for the 4 buffers required to build up the help
// result text.
cch = (UINT)GlobalSize(lpPS->hBuff)/4;
psz1=(LPTSTR)GlobalLock(lpPS->hBuff); psz2=psz1+cch; psz3=psz2+cch; psz4=psz3+cch;
// Default is an empty string.
*psz1=0;
if (0!=LoadString(ghInst, iString, psz1, cch)) { // Insert the FullUserTypeName of the source object into the partial result text
// specified by the container.
wsprintf(psz3, lpOPS->arrPasteEntries[nPasteEntriesIndex].lpstrResultText, (LPTSTR)szInsert); // Insert the above partial result text into the standard result text.
wsprintf(psz4, psz1, (LPTSTR)psz3); psz1=psz4; }
// If LoadString failed, we simply clear out the results (*psz1=0 above)
SetDlgItemText(hDlg, ID_PS_RESULTTEXT, psz1);
// Change the result bitmap
SendDlgItemMessage(hDlg, ID_PS_RESULTIMAGE, RIM_IMAGESET, iImage, 0L);
GlobalUnlock(lpPS->hBuff); }
/*
* FAddPasteListItem * * Purpose: * Adds an item to the list box * * Parameters: * hList HWND List into which item is to be added * fInsertFirst BOOL Insert in the beginning of the list? * nPasteEntriesIndex int Index of Paste Entry array this list item corresponsds to * lpPS Paste Special Dialog Structure * pIMalloc LPMALLOC Memory Allocator * lpszBuf LPSTR Scratch buffer to build up string for list entry * lpszFullUserTypeName LPSTR full user type name for object entry * * Return Value: * BOOL TRUE if sucessful. * FALSE if unsucessful. */ BOOL FAddPasteListItem( HWND hList, BOOL fInsertFirst, int nPasteEntriesIndex, LPPASTESPECIAL lpPS, LPMALLOC pIMalloc, LPTSTR lpszBuf, LPTSTR lpszFullUserTypeName) { LPOLEUIPASTESPECIAL lpOPS = lpPS->lpOPS; LPPASTELISTITEMDATA lpItemData; int nIndex;
// Allocate memory for each list box item
lpItemData = (LPPASTELISTITEMDATA)pIMalloc->lpVtbl->Alloc( pIMalloc, (DWORD)sizeof(PASTELISTITEMDATA)); if (NULL == lpItemData) return FALSE;
// Fill data associated with each list box item
lpItemData->nPasteEntriesIndex = nPasteEntriesIndex; lpItemData->fCntrEnableIcon = ((lpOPS->arrPasteEntries[nPasteEntriesIndex].dwFlags & OLEUIPASTE_ENABLEICON) ? TRUE : FALSE);
// Build list box entry string, insert the string and add the data the corresponds to it
wsprintf( (LPTSTR)lpszBuf, lpOPS->arrPasteEntries[nPasteEntriesIndex].lpstrFormatName, (LPTSTR)lpszFullUserTypeName );
// only add to listbox if not a duplicate
if (LB_ERR!=SendMessage(hList,LB_FINDSTRING, 0, (LPARAM)(LPTSTR)lpszBuf)) { // item is already in list; SKIP this one
pIMalloc->lpVtbl->Free(pIMalloc, (LPVOID)lpItemData); return TRUE; // this is NOT an error
}
nIndex = (int)SendMessage( hList, (fInsertFirst ? LB_INSERTSTRING : LB_ADDSTRING), 0, (LPARAM)(LPTSTR)lpszBuf ); SendMessage( hList, LB_SETITEMDATA, nIndex, (LPARAM)(LPPASTELISTITEMDATA)lpItemData ); return TRUE; }
/*
* FFillPasteList * * Purpose: * Fills the invisible paste list with the formats offered by the clipboard object and * asked for by the container. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * BOOL TRUE if sucessful and if formats could be found. * FALSE if unsucessful or if no formats could be found. */ BOOL FFillPasteList(HWND hDlg, LPPASTESPECIAL lpPS) { LPOLEUIPASTESPECIAL lpOPS = lpPS->lpOPS; LPMALLOC pIMalloc = NULL; LPTSTR lpszBuf = (LPTSTR)GlobalLock(lpPS->hBuff); HWND hList; int i, j; int nItems = 0; int nDefFormat = -1; BOOL fTryObjFmt = FALSE; BOOL fInsertFirst; BOOL fExclude; HRESULT hrErr;
hrErr = CoGetMalloc(MEMCTX_TASK, &pIMalloc); if (hrErr != NOERROR) goto error;
hList = GetDlgItem(hDlg, ID_PS_PASTELIST);
// Loop over the target's priority list of formats
for (i = 0; i < lpOPS->cPasteEntries; i++) { if (lpOPS->arrPasteEntries[i].dwFlags != OLEUIPASTE_PASTEONLY && !(lpOPS->arrPasteEntries[i].dwFlags & OLEUIPASTE_PASTE)) continue;
fInsertFirst = FALSE;
if (lpOPS->arrPasteEntries[i].fmtetc.cfFormat==cfFileName || lpOPS->arrPasteEntries[i].fmtetc.cfFormat==cfEmbeddedObject || lpOPS->arrPasteEntries[i].fmtetc.cfFormat==cfEmbedSource) { if (! fTryObjFmt) { fTryObjFmt = TRUE; // only use 1st object format
fInsertFirst = TRUE; // OLE obj format should always be 1st
//Check if this CLSID is in the exclusion list.
fExclude=FALSE;
for (j=0; j < (int)lpOPS->cClsidExclude; j++) { if (IsEqualCLSID(&lpPS->clsidOD, (LPCLSID)(lpOPS->lpClsidExclude+j))) { fExclude=TRUE; break; } }
if (fExclude) continue; // don't add the object entry to list
} else { continue; // already added an object format to list
} }
// add to list if entry is marked TRUE
if (lpOPS->arrPasteEntries[i].dwScratchSpace) { if (nDefFormat < 0) nDefFormat = (fInsertFirst ? 0 : nItems); else if (fInsertFirst) nDefFormat++; // adjust for obj fmt inserted 1st in list
if (!FAddPasteListItem(hList, fInsertFirst, i, lpPS, pIMalloc, lpszBuf, lpPS->szFullUserTypeNameOD)) goto error; nItems++; } }
// initialize selection to first format matched in list
if (nDefFormat >= 0) lpPS->nPasteListCurSel = nDefFormat;
// Clean up
if (pIMalloc) pIMalloc->lpVtbl->Release(pIMalloc); if (lpszBuf) GlobalUnlock(lpPS->hBuff);
// If no items have been added to the list box (none of the formats
// offered by the source matched those acceptable to the container),
// return FALSE
if (nItems > 0) return TRUE; else return FALSE;
error: if (pIMalloc) pIMalloc->lpVtbl->Release(pIMalloc); if (lpszBuf) GlobalUnlock(lpPS->hBuff); FreeListData(hList);
return FALSE; }
/*
* FFillPasteLinkList * * Purpose: * Fills the invisible paste link list with the formats offered by the clipboard object and * asked for by the container. * * Parameters: * hDlg HWND of the dialog * lpPS Paste Special Dialog Structure * * Return Value: * BOOL TRUE if sucessful and if formats could be found. * FALSE if unsucessful or if no formats could be found. */ BOOL FFillPasteLinkList(HWND hDlg, LPPASTESPECIAL lpPS) { LPOLEUIPASTESPECIAL lpOPS = lpPS->lpOPS; LPDATAOBJECT lpSrcDataObj = lpOPS->lpSrcDataObj; LPENUMFORMATETC lpEnumFmtEtc = NULL; LPMALLOC pIMalloc = NULL; LPTSTR lpszBuf = (LPTSTR)GlobalLock(lpPS->hBuff); OLEUIPASTEFLAG pasteFlag; UINT arrLinkTypesSupported[PS_MAXLINKTYPES]; // Array of flags that
// indicate which link types
// are supported by source.
FORMATETC fmtetc; int i, j; int nItems = 0; BOOL fLinkTypeSupported = FALSE; HWND hList; int nDefFormat = -1; BOOL fTryObjFmt = FALSE; BOOL fInsertFirst; HRESULT hrErr;
hrErr = CoGetMalloc(MEMCTX_TASK, &pIMalloc); if (hrErr != NOERROR) goto error;
// Remember which link type formats are offered by lpSrcDataObj.
_fmemset(&fmtetc, 0, sizeof(FORMATETC)); for (i = 0; i < lpOPS->cLinkTypes; i++) { if (lpOPS->arrLinkTypes[i] = cfLinkSource) { OLEDBG_BEGIN2(TEXT("OleQueryLinkFromData called\r\n")) hrErr = OleQueryLinkFromData(lpSrcDataObj); OLEDBG_END2 if(NOERROR == hrErr) { arrLinkTypesSupported[i] = 1; fLinkTypeSupported = TRUE; } else arrLinkTypesSupported[i] = 0; } else { fmtetc.cfFormat = lpOPS->arrLinkTypes[i]; fmtetc.dwAspect = DVASPECT_CONTENT; fmtetc.tymed = 0xFFFFFFFF; // All tymed values
fmtetc.lindex = -1; OLEDBG_BEGIN2(TEXT("IDataObject::QueryGetData called\r\n")) hrErr = lpSrcDataObj->lpVtbl->QueryGetData(lpSrcDataObj,&fmtetc); OLEDBG_END2 if(NOERROR == hrErr) { arrLinkTypesSupported[i] = 1; fLinkTypeSupported = TRUE; } else arrLinkTypesSupported[i] = 0; } } // No link types are offered by lpSrcDataObj
if (! fLinkTypeSupported) { nItems = 0; goto cleanup; }
hList = GetDlgItem(hDlg, ID_PS_PASTELINKLIST);
// Enumerate the formats acceptable to container
for (i = 0; i < lpOPS->cPasteEntries; i++) { fLinkTypeSupported = FALSE;
// If container will accept any link type offered by source object
if (lpOPS->arrPasteEntries[i].dwFlags & OLEUIPASTE_LINKANYTYPE) fLinkTypeSupported = TRUE; else { // Check if any of the link types offered by the source
// object are acceptable to the container
// This code depends on the LINKTYPE enum values being powers of 2
for (pasteFlag = OLEUIPASTE_LINKTYPE1, j = 0; j < lpOPS->cLinkTypes; pasteFlag*=2, j++) { if ((lpOPS->arrPasteEntries[i].dwFlags & pasteFlag) && arrLinkTypesSupported[j]) { fLinkTypeSupported = TRUE; break; } } }
fInsertFirst = FALSE;
if (lpOPS->arrPasteEntries[i].fmtetc.cfFormat==cfFileName || lpOPS->arrPasteEntries[i].fmtetc.cfFormat==cfLinkSource) { if (! fTryObjFmt) { fTryObjFmt = TRUE; // only use 1st object format
fInsertFirst = TRUE; // OLE obj format should always be 1st
} else { continue; // already added an object format to list
} }
// add to list if entry is marked TRUE
if (fLinkTypeSupported && lpOPS->arrPasteEntries[i].dwScratchSpace) { if (nDefFormat < 0) nDefFormat = (fInsertFirst ? 0 : nItems); else if (fInsertFirst) nDefFormat++; // adjust for obj fmt inserted 1st in list
if (!FAddPasteListItem(hList, fInsertFirst, i, lpPS, pIMalloc, lpszBuf, lpPS->szFullUserTypeNameLSD)) goto error; nItems++; } } // end FOR
nItems = (int)SendMessage(hList, LB_GETCOUNT, 0, 0L);
// initialize selection to first format matched in list
if (nDefFormat >= 0) lpPS->nPasteLinkListCurSel = nDefFormat;
cleanup: // Clean up
if (pIMalloc) pIMalloc->lpVtbl->Release(pIMalloc); if (lpszBuf) GlobalUnlock(lpPS->hBuff);
// If no items have been added to the list box (none of the formats
// offered by the source matched those acceptable to the destination),
// return FALSE
if (nItems > 0) return TRUE; else return FALSE;
error: if (pIMalloc) pIMalloc->lpVtbl->Release(pIMalloc); if (lpszBuf) GlobalUnlock(lpPS->hBuff); FreeListData(hList);
return FALSE; }
/*
* FreeListData * * Purpose: * Free the local memory associated with each list box item * * Parameters: * hList HWND of the list * * Return Value: * None */ void FreeListData(HWND hList) { int nItems, i; LPPASTELISTITEMDATA lpItemData; LPMALLOC pIMalloc; HRESULT hrErr;
hrErr = CoGetMalloc(MEMCTX_TASK, &pIMalloc); if (hrErr != NOERROR) return;
nItems = (int) SendMessage(hList, LB_GETCOUNT, 0, 0L); for (i = 0; i < nItems; i++) { lpItemData = (LPPASTELISTITEMDATA)SendMessage(hList, LB_GETITEMDATA, (WPARAM)i, 0L); if ((LRESULT)lpItemData != LB_ERR) pIMalloc->lpVtbl->Free(pIMalloc, (LPVOID)lpItemData); } pIMalloc->lpVtbl->Release(pIMalloc); }
/*
* FHasPercentS * * Purpose: * Determines if string contains %s. * * Parameters: * lpsz LPCSTR string in which occurence of '%s' is looked for * * Return Value: * BOOL TRUE if %s is found, else FALSE. */
BOOL FHasPercentS(LPCTSTR lpsz, LPPASTESPECIAL lpPS) { int n = 0; LPTSTR lpszTmp;
if (!lpsz) return FALSE; // Copy input string to buffer. This allows caller to pass a
// code-based string. Code segments may be swapped out in low memory situations
// and so code-based strings need to be copied before string elements can be accessed.
lpszTmp = (LPTSTR)GlobalLock(lpPS->hBuff); lstrcpy(lpszTmp, lpsz);
while (*lpszTmp) { if (*lpszTmp == TEXT('%')) { #ifdef WIN32
// AnsiNext is obsolete in Win32
lpszTmp = CharNext(lpszTmp); #else
lpszTmp = AnsiNext(lpszTmp); #endif
if (*lpszTmp == TEXT('s')) // If %s, return
{ GlobalUnlock(lpPS->hBuff); return TRUE; } else if (*lpszTmp == TEXT('%')) // if %%, skip to next character
#ifdef WIN32
// AnsiNext is obsolete in Win32
lpszTmp = CharNext(lpszTmp); #else
lpszTmp = AnsiNext(lpszTmp); #endif
} else #ifdef WIN32
lpszTmp = CharNext(lpszTmp); #else
lpszTmp = AnsiNext(lpszTmp); #endif
}
GlobalUnlock(lpPS->hBuff); return FALSE; }
/*
* AllocateScratchMem * * Purpose: * Allocates scratch memory for use by the PasteSpecial dialog. The memory is * is used as the buffer for building up strings using wsprintf. Strings are built up * using the buffer while inserting items into the Paste & PasteLink lists and while * setting the help result text. It must be big enough to handle the string that results after * replacing the %s in the lpstrFormatName and lpstrResultText in arrPasteEntries[] * by the FullUserTypeName. It must also be big enough to build the dialog's result text * after %s substitutions by the FullUserTypeName or the ApplicationName. * * Parameters: * lpPS Paste Special Dialog Structure * * Return Value: * HGLOBAL Handle to allocated global memory */
HGLOBAL AllocateScratchMem(LPPASTESPECIAL lpPS) { LPOLEUIPASTESPECIAL lpOPS = lpPS->lpOPS; int nLen, i; int nSubstitutedText = 0; int nAlloc = 0;
// Get the maximum length of the FullUserTypeNames specified by OBJECTDESCRIPTOR
// and the LINKSRCDESCRIPTOR and the Application Name. Any of these may be substituted
// for %s in the result-text/list entries.
if (lpPS->szFullUserTypeNameOD) nSubstitutedText = lstrlen(lpPS->szFullUserTypeNameOD); if (lpPS->szFullUserTypeNameLSD) nSubstitutedText = __max(nSubstitutedText, lstrlen(lpPS->szFullUserTypeNameLSD)); if (lpPS->szAppName) nSubstitutedText = __max(nSubstitutedText, lstrlen(lpPS->szAppName));
// Get the maximum length of lpstrFormatNames & lpstrResultText in arrPasteEntries
nLen = 0; for (i = 0; i < lpOPS->cPasteEntries; i++) { nLen = __max(nLen, lstrlen(lpOPS->arrPasteEntries[i].lpstrFormatName)); nLen = __max(nLen, lstrlen(lpOPS->arrPasteEntries[i].lpstrResultText)); }
// Get the maximum length of lpstrFormatNames and lpstrResultText after %s has
// been substituted (At most one %s can appear in each string).
// Add 1 to hold NULL terminator.
nAlloc = (nLen+nSubstitutedText+1)*sizeof(TCHAR);
// Allocate scratch memory to be used to build strings
// nAlloc is big enough to hold any of the lpstrResultText or lpstrFormatName in arrPasteEntries[]
// after %s substitution.
// We also need space to build up the help result text. 512 is the maximum length of the
// standard dialog help text before substitutions. 512+nAlloc is the maximum length
// after %s substition.
// SetPasteSpecialHelpResults() requires 4 such buffers to build up the result text
return GlobalAlloc(GHND, (DWORD)4*(512+nAlloc)); }
|