【转载】VC 通过剪贴板读取word数据并存为jpg

Posted on 2009-06-27 14:01  WindFly  阅读(1902)  评论(0编辑  收藏  举报

【载自】http://www.cnblogs.com/anfeind/archive/2009/05/26/1489403.html

 

vc从剪贴板中读取word中的数据。word在剪贴板中的数据类型是HENHMETAFILE,把主要代码贴在下面

1、读取数据转存emf文件

void CTestClipBoardDlg::OnBnClickedButton1()

{

// TODO: 在此添加控件通知处理程序代码


if(!OpenClipboard())

{

return;

}

for(int i=1; i<16; i++)

{

UINT   nRet1   =   EnumClipboardFormats(i); 

CString str;

str.Format("%d\n",i);

if(nRet1)

TRACE("[ok]"+str);

else

TRACE("[error]"+str);

}

DWORD   dwError;


UINT nRet = EnumClipboardFormats(CF_ENHMETAFILE);

if(   nRet   ) 

    { 

         BOOL   ba=::IsClipboardFormatAvailable(CF_ENHMETAFILE);   

HANDLE   hClip=::GetClipboardData(CF_ENHMETAFILE);

hEnhMetaFile=(HENHMETAFILE)hClip;  

dwError=GetLastError();   //   dwError=6,无效的句柄。   

CTime time = CTime::GetCurrentTime();

CString str = "d:\\aaa.emf";

//str.Format("d:\\aaa.emf", time.Format("%H%M%S"));

HENHMETAFILE   hMetaFile=CopyEnhMetaFile(hEnhMetaFile, str);//保存到文件 

 

//关闭CMetafileDC并获得它的句柄


DeleteEnhMetaFile(hMetaFile);

dwError=GetLastError();   //   dwError=6,无效的句柄。 

CClientDC dc(this);

CRect client(0,0,600,400);

dc.PlayMetaFile(hEnhMetaFile,client);


    } 

      ::CloseClipboard();

}

 

2.load   emf文件转存jpg

 BOOL CConvertEMFToBMP::ConvertEMFToBMP(const char * pszEMFFile,const char* pszBMPFile,BOOL bScaleImage/*=false*/)   

{   


//Prepare the BMP file name   

char szBMPFile[255] = "";   

strcpy(szBMPFile,pszBMPFile);   



//Convert the EMF file to BMP File   

BOOL bRet = ConvertToBMP(pszEMFFile,(const char*)szBMPFile,bScaleImage);   


//if BMP conversion failed return false   

if(bRet == FALSE)   

{   

return FALSE;   

}else

{

CImage m_img;

m_img.Load(_T("d:\\aaa.bmp"));

m_img.Save(_T("d:\\aaa.jpg"));


}



//return true/false accordingly   

return bRet ? TRUE:FALSE;   

}      HANDLE CConvertEMFToBMP::DDBToDIB( HBITMAP bitmap, DWORD dwCompression, HPALETTE pPal )    

{   

BITMAP          bm;   

BITMAPINFOHEADER    bi;   

LPBITMAPINFOHEADER  lpbi;   

DWORD           dwLen;   

HANDLE          hDIB;   

HANDLE          handle;   

HDC             hDC;   

HPALETTE        hPal;   



// The function has no arg for bitfields   

if( dwCompression == BI_BITFIELDS )   

return NULL;   


// If a palette has not been supplied use defaul palette   

hPal = (HPALETTE) pPal;   

if (hPal==NULL)   

hPal = (HPALETTE) ::GetStockObject(DEFAULT_PALETTE);   


// Get bitmap information   

::GetObject(bitmap,sizeof(bm),(LPSTR)&bm);   


// Initialize the bitmapinfoheader   

bi.biSize       = sizeof(BITMAPINFOHEADER);   

bi.biWidth      = bm.bmWidth;   

bi.biHeight         = bm.bmHeight;   

bi.biPlanes         = 1;   

bi.biBitCount       = bm.bmPlanes * bm.bmBitsPixel;   

bi.biCompression    = dwCompression;   

bi.biSizeImage      = 0;   

bi.biXPelsPerMeter  = 0;   

bi.biYPelsPerMeter  = 0;   

bi.biClrUsed        = 0;   

bi.biClrImportant   = 0;   


// Compute the size of the  infoheader and the color table   

int nColors = (1 << bi.biBitCount);   

if( nColors > 256 )    

nColors = 0;   

dwLen  = bi.biSize + nColors * sizeof(RGBQUAD);   


// We need a device context to get the DIB from   

hDC = ::GetDC(NULL);   

hPal = SelectPalette(hDC,hPal,FALSE);   

RealizePalette(hDC);   


// Allocate enough memory to hold bitmapinfoheader and color table   

hDIB = GlobalAlloc(GMEM_FIXED,dwLen);   


if (!hDIB)   

{   

SelectPalette(hDC,hPal,FALSE);   

::ReleaseDC(NULL,hDC);   

return NULL;   

}   


lpbi = (LPBITMAPINFOHEADER)hDIB;   


*lpbi = bi;   


// Call GetDIBits with a NULL lpBits param, so the device driver    

// will calculate the biSizeImage field    

GetDIBits(hDC, (HBITMAP)bitmap, 0L, (DWORD)bi.biHeight,   

(LPBYTE)NULL, (LPBITMAPINFO)lpbi, (DWORD)DIB_RGB_COLORS);   


bi = *lpbi;   


// If the driver did not fill in the biSizeImage field, then compute it   

// Each scan line of the image is aligned on a DWORD (32bit) boundary   

if (bi.biSizeImage == 0)   

{   

bi.biSizeImage = ((((bi.biWidth * bi.biBitCount) + 31) & ~31) / 8)    

* bi.biHeight;   


// If a compression scheme is used the result may infact be larger   

// Increase the size to account for this.   

if (dwCompression != BI_RGB)   

bi.biSizeImage = (bi.biSizeImage * 3) / 2;   

}   


// Realloc the buffer so that it can hold all the bits   

dwLen += bi.biSizeImage;   

if (handle = GlobalReAlloc(hDIB, dwLen, GMEM_MOVEABLE))   

{   

hDIB = handle;   

}   

else   

{   

GlobalFree(hDIB);   


// Reselect the original palette   

SelectPalette(hDC,hPal,FALSE);   

::ReleaseDC(NULL,hDC);   

return NULL;   

}   


// Get the bitmap bits   

lpbi = (LPBITMAPINFOHEADER)hDIB;   


// FINALLY get the DIB   

BOOL bGotBits = GetDIBits( hDC, bitmap,   

0L,             // Start scan line   

(DWORD)bi.biHeight,     // # of scan lines   

(LPBYTE)lpbi            // address for bitmap bits   

+ (bi.biSize + nColors * sizeof(RGBQUAD)),   

(LPBITMAPINFO)lpbi,     // address of bitmapinfo   

(DWORD)DIB_RGB_COLORS);     // Use RGB for color table   


if( !bGotBits )   

{   

GlobalFree(hDIB);   


SelectPalette(hDC,hPal,FALSE);   

::ReleaseDC(NULL,hDC);   

return NULL;   

}   


SelectPalette(hDC,hPal,FALSE);   

::ReleaseDC(NULL,hDC);   

return hDIB;   

}   



/*==========================================================================  

Name                :WriteDIB  

Purpose             :Writes the DIB information to the file  

szFile      - Name of file to write to  

hDIB            - Handle of the DIB  

Arguments           :LPTSTR szFile, HANDLE hDIB  

Return values       :TRUE on success  

============================================================================*/   


BOOL CConvertEMFToBMP::WriteDIB( char * szFile, HANDLE hDIB)   

{   

BITMAPFILEHEADER    hdr;   

LPBITMAPINFOHEADER  lpbi;   


if (!hDIB)   

return FALSE;   


FILE* file;   

file = fopen(szFile,"wb");   

if(file == NULL)   

return FALSE;   


lpbi = (LPBITMAPINFOHEADER)hDIB;    


int nColors = 0;    

if(lpbi->biBitCount <= 8)    

{    

nColors = (1 << lpbi->biBitCount);    

}    



// Fill in the fields of the file header    

hdr.bfType      = ((WORD) ('M' << 8) | 'B');  // is always "BM"   

hdr.bfSize      = GlobalSize (hDIB) + sizeof( hdr );   

hdr.bfReserved1     = 0;   

hdr.bfReserved2     = 0;   

hdr.bfOffBits       = (DWORD) (sizeof( hdr ) + lpbi->biSize +   

nColors * sizeof(RGBQUAD));   


// Write the file header    

fwrite( &hdr, sizeof(hdr),1,file);   


// Write the DIB header and the bits    

fwrite( lpbi, GlobalSize(hDIB),1,file);   


//Close the file and return   

fclose(file);   


return TRUE;   

}   




BOOL CConvertEMFToBMP::SetBackColorToWhite(HDC pDC)   

{   

// Set brush to desired background color   

HBRUSH backBrush= (HBRUSH)(#ffffff);   



// Save old brush   

HBRUSH pOldBrush = (HBRUSH)::SelectObject(pDC,backBrush);   


RECT rect ;   

::GetClipBox(pDC,&rect);     // Erase the area needed   


//paint the given rectangle using the brush that is currently selected    

//into the specified device context   

::PatBlt(pDC,rect.left, rect.top, abs(rect.left - rect.right),abs(rect.top-rect.bottom ),PATCOPY);   


//Select back the old brush   

::SelectObject(pDC,pOldBrush);   


return TRUE;       

}   




/*==========================================================================  

Return values       :BOOL (TRUE on Success)  

============================================================================*/   


BOOL CConvertEMFToBMP::ConvertToBMP(const char * strFileName,const char* strBMPFile,BOOL bScaleImage)   

{   


//Declartions   

HENHMETAFILE hemf;   

HBITMAP     bitmap;   

HDC         memDC;   

ENHMETAHEADER   emh;   


//Get the DC of the Window   

HDC dc = ::GetDC(NULL);   


//Get the Handle from the enhanced metafile   

hemf = GetEnhMetaFile(L"d:\\aaa.emf");


// Get the header from the enhanced metafile.   

ZeroMemory( &emh, sizeof(ENHMETAHEADER) );   

emh.nSize = sizeof(ENHMETAHEADER);   

if( GetEnhMetaFileHeader( hemf, sizeof( ENHMETAHEADER ), &emh ) == 0 )   

{   

DeleteEnhMetaFile( hemf );   

return FALSE;   

}   


//Declare variables for calculation of metafile rect   

RECT    rect;   

float   PixelsX, PixelsY, MMX, MMY;   


float fAspectRatio;   

long lWidth,lHeight;   


// Get the characteristics of the output device.   

PixelsX = (float)GetDeviceCaps( dc, HORZRES );   

PixelsY = (float)GetDeviceCaps( dc, VERTRES );   

MMX = (float)GetDeviceCaps( dc, HORZSIZE );   

MMY = (float)GetDeviceCaps( dc, VERTSIZE );   


// Calculate the rect in which to draw the metafile based on the   

// intended size and the current output device resolution.   

// Remember that the intended size is given in 0.01 mm units, so   

// convert those to device units on the target device.   

rect.top = (int)((float)(emh.rclFrame.top) * PixelsY / (MMY*100.0f));   

rect.left = (int)((float)(emh.rclFrame.left) * PixelsX / (MMX*100.0f));   

rect.right = (int)((float)(emh.rclFrame.right) * PixelsX / (MMX*100.0f));   

rect.bottom = (int)((float)(emh.rclFrame.bottom) * PixelsY / (MMY*100.0f));   


//Calculate the Width and Height of the metafile   

lWidth = (long)((float)(abs(rect.left - rect.right)));   

lHeight =(long)((float)(abs(rect.top-rect.bottom )));   



fAspectRatio = (float)lWidth/(float)lHeight;   



if(bScaleImage) //If miniature option is ON, change the width and height accordingly   

{   

if(fAspectRatio > 1 ) //width is more than height    

{   

//Make width as constant and calculate Height   

lWidth = X_MINIATUREFRAME;   

lHeight = (long)((float)Y_MINIATUREFRAME / fAspectRatio);   


}   

else //width is less than height(or equal to height)   

{   

//Make Height as constant and calculate Width   

lHeight = Y_MINIATUREFRAME;   

lWidth = (long)((float)X_MINIATUREFRAME * fAspectRatio);   

}   

}   



//Populate the rect structure   

rect.left = 0;   

rect.top = 0;   

rect.right = lWidth;   

rect.bottom = lHeight;   



//Create a Memory DC compatible to WindowDC   

memDC=::CreateCompatibleDC(dc);    



//Create a bitmap compatible to Window DC   

bitmap = ::CreateCompatibleBitmap(dc,lWidth,lHeight);   



DWORD dwRetError = GetLastError();   



//Select the bitmap into the Mem DC   

::SelectObject(memDC,bitmap);   


//Paint the background of the DC to White   

SetBackColorToWhite(memDC);   



//Now play the enhanced metafile into the memory DC; ignore its return value   

//it may be false even if successful   

PlayEnhMetaFile(memDC,hemf,&rect);     

DWORD dwRet = GetLastError();   


// Create logical palette if device support a palette   

HPALETTE pal;   

if( GetDeviceCaps(dc,RASTERCAPS) & RC_PALETTE )   

{   

UINT nSize = sizeof(LOGPALETTE) + (sizeof(PALETTEENTRY) * 256);   

LOGPALETTE *pLP = (LOGPALETTE *) new BYTE[nSize];   

pLP->palVersion = 0x300;   


pLP->palNumEntries =    

GetSystemPaletteEntries( dc, 0, 255, pLP->palPalEntry );   


// Create the palette   

pal = ::CreatePalette(pLP );   


delete[] pLP;   

}   


// Convert the bitmap to a DIB   

HANDLE hDIB = DDBToDIB(bitmap, BI_RGB, NULL );   


if( hDIB == NULL )   

{   

DeleteEnhMetaFile( hemf );   

return FALSE;   

}   



// Write it to file   

WriteDIB((char*)strBMPFile, hDIB );   


// Free the memory allocated by DDBToDIB for the DIB   

::GlobalFree( hDIB );   


::DeleteEnhMetaFile( hemf );   


::ReleaseDC(NULL,dc);   

return TRUE;   

}

 

(转) 转vc操作剪贴板的一些笔记

 本文主要介绍了VC++/MFC中如下内容的剪贴板操作:

1、文本内容的操作
2
WMF数据的操作
3
、位图的操作
4
、设置使用自定义格式
5
、感知剪贴板内容的改变
6
、自动将数据粘贴到另一应用程序窗口

一、文本内容的操作
下面的代码示范了如何将文本内容复制到剪贴板(Unicode编码的先转化为ASCII)

CString source;
//
文本内容保存在source变量中

if( OpenClipboard() )
{
 HGLOBAL clipbuffer;
 char * buffer;
 EmptyClipboard();
 clipbuffer = GlobalAlloc(GMEM_DDESHARE, source.GetLength()+1);
 buffer = (char*)GlobalLock(clipbuffer);
 strcpy(buffer, LPCSTR(source));
 GlobalUnlock(clipbuffer);
 SetClipboardData(CF_TEXT,clipbuffer);
 CloseClipboard();
}

下面的代码显示了如何从剪贴板上获得文本内容:

char * buffer = NULL;
//
打开剪贴板

CString fromClipboard;
if ( OpenClipboard() )
{
 HANDLE hData = GetClipboardData(CF_TEXT);
 char * buffer = (char*)GlobalLock(hData);
 fromClipboard = buffer;
 GlobalUnlock(hData);
 CloseClipboard();
}

二、WMF数据的操作

  在剪贴板上读写图象数据是非常有用的功能,并且实现起来也很简单。下面的代码显示了如何将扩展图元文件复制到剪贴板:

  if(OpenClipboard());
{
 EmptyClipboard();

 //创建图元文件DC
 CMetaFileDC * cDC = new CMetaFileDC();
 cDC->CreateEnhanced(GetDC(),NULL,NULL,"the_name");

 //调用绘图例程

 //关闭CMetafileDC并获得它的句柄
 HENHMETAFILE handle = cDC->CloseEnhanced();

 //复制到剪贴板
 SetClipBoardData(CF_ENHMETAFILE,handle);
 CloseClipboard();

 //删除dc
 delete cDC;
}

下面的代码演示了从剪贴板获得图元文件并将其绘制到client DC上:

if(OpenClipboard())
{
 //
获得剪贴板数据

 HENMETAFILE handle = (HENMETAFILE)GetClipboardData(CF_ENHMETAFILE);

 //显示
 CClientDC dc(this);
 CRect client(0,0,200,200);
 dc.PlayMetaFile(handle,client);

 //关闭剪贴板
 CloseClipboard();
}
三、位图的操作


位图的操作稍微复杂一点,下面这个例子显示了如何在剪贴板保存位图:

if(OpenClipboard())
{
 EmptyClipboard();
 CBitmap * junk = new CBitmap();
 CClientDC cdc(this);
 CDC dc;
 dc.CreateCompatibleDC(&cdc);
 CRect client(0,0,200,200);
 junk->CreateCompatibleBitmap(&cdc,client.Width(),client.Height());
 dc.SelectObject(junk);

 DrawImage(&dc,CString("Bitmap"));

 //复制数据到剪贴板
 SetClipboardData(CF_BITMAP,junk->m_hObject);
 CloseClipboard();

 delete junk;
}

下面的代码显示了如何从剪贴板上获得位图数据:

if(OpenClipboard())
{
 //
获得剪贴板数据

 HBITMAP handle = (HBITMAP)GetClipboardData(CF_BITMAP);
 CBitmap * bm = CBitmap::FromHandle(handle);

 CClientDC cdc(this);
 CDC dc;
 dc.CreateCompatibleDC(&cdc);
 dc.SelectObject(bm);
 cdc.BitBlt(0,0,200,200,&dc,0,0,SRCCOPY);

 CloseClipboard();
}

四、设置并使用自定义格式

使用RegisterClipboardFormat()函数,可以复制和粘贴任何你需要的数据类型。比如我们有以下一个数据类型:

struct MyFormatData
{
 long val1;
 int val2;
};

我们要把它复制到剪贴板,可以使用如下的代码:

UINT format = RegisterClipBoardFormat("MY_CUSTOM_FORMAT");
if(OpenClipboard())
{
 MyFormatData data;
 data.val1 = 100;
 data.val2 = 200;

 HGLOBAL clipbuffer;
 EmptyClipboard();
 clipbuffer = GlobalAlloc(GMEM_DDESHARE, sizeof(MyFormatData));
 MyFormatData * buffer = (MyFormatData*)GlobalLock(clipbuffer);

 //保存到内存
 *buffer = data;

 //保存到剪贴板
 GlobalUnlock(clipbuffer);
 SetClipboardData(format,clipbuffer);
 CloseClipboard();
}

读取数据使用以下代码:

UINT format = RegisterClipboardFormat("MY_CUSTOM_FORMAT");
MyFormatData data;
if(Openclipboard())
{
 HANDLE hData =GetClipboardData(format);
 MyFormatData * buffer = (MyFormatData*)GlobalLock(hData);

 data = *buffer;

 GlobalUnlock(hData);
 CloseClipboard();
}

五、感知剪贴板内容的改变

通过Windows消息可以感知剪贴板内容是否发生改变,代码如下:

//In your initialization code call:

SetClipboardViewer(); //将我们的程序添加到剪贴板观察链

//In your message map add:
ON_MESSAGE(WM_DRAWCLIPBOARD, OnClipChange) //
添加Message handle

//Which is declared as:

afx_msg void OnClipChange();

Finally implement:
void CDetectClipboardChangeDlg::OnClipChange() 
{
 CTime time = CTime::GetCurrentTime();
 SetDlgItemText(IDC_CHANGED_DATE,time.Format("%a, %b %d, %Y -- %H:%M:%S"));

 DisplayClipboardText();
}

六、自动将数据粘贴到另一应用程序窗口

只需获得相应窗口的句柄,并发送一个消息就可以了:

SendMessage(m_hTextWnd, WM_PASTE, 0, 0);

 

Copyright © 2025 WindFly
Powered by .NET 9.0 on Kubernetes