方法一:
1、聲明成員變量CBrush m_brush;
2、在InitDialog中添加代碼:
?
CBitmap bmp; bmp.LoadBitmap(IDB_BITMAP1); //IDB_BITMAP1是圖片資源ID m_brush.CreatePatternBrush(&bmp);
|
3、重載對話框的OnCtlColor,改最后的返回值:
?
1 |
return(HBRUSH)m_brush;
|
方法二:
把下面這段代碼加進OnPaint()里就行了
?
CPaintDC dc(this); CBitmap bitmap; bitmap.LoadBitmap(IDB_BITMAP1); //這個IDB_BITMAP1要自己添加 CBrush brush; brush.CreatePatternBrush(&bitmap); CBrush* pOldBrush = dc.SelectObject(&brush); dc.Rectangle(0,0,200,200); // 這些參數可以調整圖片添加位置和大小 dc.SelectObject(pOldBrush);
|
方法三:使用StretchBlt()函數,具有圖像自適應窗體功能
?
CPaintDC dc(this); CBitmap bitmap; bitmap.LoadBitmap(IDB_BITMAP1); //這個IDB_BITMAP1要自己添加 CBrush brush; brush.CreatePatternBrush(&bitmap); CBrush* pOldBrush = dc.SelectObject(&brush); dc.Rectangle(0,0,200,200); // 這些參數可以調整圖片添加位置和大小 dc.SelectObject(pOldBrush);
|
問題
有的程序員希望在自己的應用程序中以有趣味的位圖來代替對話框中令人討厭的灰色背景,希望位圖在對話框中看起來象牆紙而且並不影響對話框中的控制或靜態文本的顯示。
許多程序員找不到一個改變窗口背景的簡單方法,是否有方法利用 Windows API 函數來改變對話框的背景為某個位圖呢?
方法
改變對話框的背景為某個位圖並不困難,關鍵是需要清楚對話框和窗口是如何設置背景顏色的,以及程序員應該如何修改對話框和窗口改變顯示的行為。
當 Windows 准備改變對話框背景的顏色時,通常發送兩個消息給對話框。第一個消息是 WM_ERASEBKGND,此消息指示對話框繪制對話框的背景顏色,以“抹去”屏幕上對話框顯示區域的任何顯示。
第二個消息是 WM_CTLCOLOR,發送此消息給對話框或窗口來表示 Windows 需要知道對話框中控制的顏色。
在本節中,將重置對消息 WM_ERASEBKGND 的處理,以便將位圖繪制在窗口的背景上。另外,將重置對消息 WM_CTLCOLOR 的處理,以避免對話框中的控制“剪補”位圖。最后的結果是對話框的背景位圖繪制在對話框背景上,控制在背景位圖的“上面”。
步驟
按照下列步驟實現一個例子程序。運行此例子程序,選擇菜單 Dialog 和菜單項 Bitmap Background,將彈出一個對話框,顯示背景位圖和幾個控制。
實現例子程序的具體步驟如下:
1.在 Visual C++ 中,利用 AppWizard 創建新的項目文件,並命名此項目文件為 Ld145。
2.進入資源編輯器並創建新的對話框模板。在對話框中,添加幾個靜態文本域和編輯域,以及幾個單選按鈕和列表框。對話框的實際組成並不重要,只要能夠覆蓋部分位圖就可以了。
3.選擇 ClassWizard,為剛創建的對話框模板創建對話框類,新類命名為 CBitmaPBkgdDlg。
4.在資源編輯器中創建新的位圖。
5.進入 ClassWizard,從下拉列表中選擇 CBitmapBkgdDlg,從對象列表中選擇對象 CBitmapBkgdDlg,從消息列表中選擇消息 WM_INITDIALOG,點擊按鈕 Add Function,在 CBitmapBkgdDlg 的方法 OnInitDialog 中添加下列代碼:
BOOL CBitmapBkgdDlg::OnInitDialog()
{
CBitmap * pBmpOld;
RECT rectClient;
VERIFY(m_brush=(HBRUSH)GetStockObject(HOLLOW_BRUSH));
VERIFY(m_Bitmap.LoadBitmap(IDB_BITMAP1));
m_Bitmap.GetObject(sizeof(BITMAP),&m_bmInfo);
GetClientRect(&rectClient);
m_size.cx=rectClient.right;
m_size.cy=rectClient.bottom;
m_pt.x=rectClient.left;
m_pt.y=rectClient.top;
CClientDC dc(this);
VERIFY(m_dcMem.CreateCompatibleDC(&dc));
VERIFY(pBmpOld=m_dcMem.SelectObject(&m_Bitmap));
VERIFY(m_hBmpOld=(HBITMAP)pBmpOld->GetSafeHandle());
return TRUE; // return TRUE unless you set the focus to a control
}
6.接着,在 ClassWizard 中,從對象列表中選擇對象 CBitmapBkgdDlg,從消息列表中選擇消息 WM_CTLCOLOR,點擊按鈕 Add Function,在 CBitmapBkgdDlg 的方法 OnCtlColor 中添加下列代碼:
HBRUSH CBitmapBkgdDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
{
pDC->SetBkMode(TRANSPARENT);
return m_brush;
}
7.接着,在 ClassWizard 中,從對象列表中選擇對象 CBitmapBkgdDlg,從消息列表中選擇消息 WM_DESTROY,點擊按鈕 Add Function,在 CBitmapBkgdDlg 的方法 OnDestroy 中添加下列代碼:
void CBitmapBkgdDlg::OnDestroy()
{
CDialog::OnDestroy();
ASSERT(m_hBmpOld);
VERIFY(m_dcMem.SelectObject(CBitmap::FromHandle(m_hBmpOld)));
m_Bitmap.DeleteObject();
}
8.編輯 CBitmapBkgdDlg 的消息映射如下,添加的新行用暗紅色字體表示:
BEGIN_MESSAGE_MAP(CBitmapBkgdDlg, CDialog)
//{{AFX_MSG_MAP(CBitmapBkgdDlg)
ON_WM_CTLCOLOR()
ON_WM_DESTROY()
ON_WM_ERASEBKGND()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
9.在 CBitmapBkgdDlg 的源文件 BitmapBkgdDlg.cpp 中添加下列新方法:
BOOL CBitmapBkgdDlg::OnEraseBkgnd(CDC * pDC)
{
pDC->StretchBlt(m_pt.x,m_pt.y,m_size.cx,m_size.cy,&m_dcMem,
0,0,m_bmInfo.bmWidth-1,m_bmInfo.bmHeight-1,SRCCOPY);
return TRUE;
}
10.在 CBitmapBkgdDlg 的頭文件 BitmapBkgdDlg.h 中做下列修改,用暗紅色字體表示。
class CBitmapBkgdDlg : public CDialog
{
protected:
CDC m_dcMem;
CBitmap m_Bitmap;
HBITMAP m_hBmpOld;
HBRUSH m_brush;
BITMAP m_bmInfo;
CPoint m_pt;
CSize m_size;
// Construction
public:
CBitmapBkgdDlg(CWnd* pParent = NULL); // standard constructor
// Dialog Data
//{{AFX_DATA(CBitmapBkgdDlg)
enum { IDD = IDD_DIALOG1 };
// NOTE: the ClassWizard will add data members here
//}}AFX_DATA
// Overrides
// ClassWizard generated virtual function overrides
//{{AFX_VIRTUAL(CBitmapBkgdDlg)
protected:
virtual void DoDataExchange(CDataExchange* pDX); // DDX/DDV support
//}}AFX_VIRTUAL
// Implementation
protected:
// Generated message map functions
//{{AFX_MSG(CBitmapBkgdDlg)
virtual BOOL OnInitDialog();
afx_msg HBRUSH OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor);
afx_msg void OnDestroy();
virtual BOOL OnEraseBkgnd(CDC * pDC);
//}}AFX_MSG
DECLARE_MESSAGE_MAP()
};
11.進入資源編輯器,在菜單 IDR_MAINFRAME 中添加新的菜單,標題為 Dialog。在菜單 Dialog 中添加新的菜單項,標題為 Bitmap Background,標識符為 ID_BITMAP_BKGND,退出資源編輯器,保存資源文件。
12.進入 ClassWizard,從下拉列表中選擇對象 CMainFrame,從對象列表中選擇對象 ID_BITMAP_BKGND,從消息列表中選擇消息 COMMAND,點擊按鈕 Add Function,新函數命名為 OnBitmapBkgnd。在 CMainFrame 的方法 OnBitmapBkgnd 中添加下列代碼:
void CMainFrame::OnBitmapBkgnd()
{
CBitmapBkgdDlg dlg;
dlg.DoModal();
}
13.在源文件 MainFrm.cpp 的頂部添加下列行:
#include "BitmapBkgdDlg.h"
14.編譯並運行此例子程序。
用法
當 Windows 初始化對話框時,它發送消息 WM_ERASEBKGND 給對話框的窗口句柄。程序員可以捕捉此消息,以便在應用程序中抹去對話框的背景。在本節中,首先捕捉此消息,接着調用 API 函數 StretchBlt 來將位圖(從資源文件中裝入)拷貝到對話框的背景上。
在對話框的方法 OnCtlColor 中,通過設置背景模式為透明來確保對話框中的控制不會“剪補”儉圖,從而使得位圖看起來好像是繪制在對話框中的,沒有靜態控制的背景所引起的空白。
在工具條上使用16色以上位圖操作說明書
本操作說明書既適用於一般的工具條,也適用於三維CAD中使用的向導工具條。
步驟:
1、 首先在VC中制做16色的工具條。因為工具條資源中包含ID、尺寸等信息,為簡便起見,本說明書在16色工具條的基礎上附加16色以上位圖,即首先應有完整的16色工具條。
2、 相應於工具條資源,添加位圖資源。如工具條為IDR_MYTOOLBAR,則位圖資源ID建議為:IDB_MYTOOLBAR_NORMAL。注意,把文件名也相應改變,如改為:mytoolbar_normal.bmp等。也利后續操作。不要在VC中編緝該資源。
3、 創建工具條后,用以下方法附加工具條位圖資源。
CImageList imageList;
CBitmap bitmap;
bitmap.LoadBitmap(m_nToolBarIDNormal);
BITMAP bm;
::GetObject((HBITMAP)bitmap, sizeof(BITMAP), (LPVOID)&bm);
imageList.Create(16, 15, ILC_COLORDDB|ILC_MASK, bm.bmWidth/16, 1);
imageList.Add(&bitmap, RGB(255,0,255));
CImageList *pOldImageList = m_wndToolBar.GetToolBarCtrl().SetImageList(&imageList);
If(pOldImageList)
pOldImageList-> DeleteImageList();//一定有,否則引起資源泄漏稅
imageList.Detach();
DISABLE和HOT位圖資源附加方法同上。
注意:設置背景顏色(RGB(255, 0, 255)應和位圖的實際顏色一致,否則會出現背景。
4、 在畫筆中,設置位圖高度為15像素。把工具條圖標依次排列,設置背景色,保存為第二步中設定的文件名。
5、 編譯程序,完成。
遇到過一個問題,有的按鈕上動態生成了控件,大小不一.
已經解決了:
UINT nID ;
UINT nStyle;
int iImage ;
int nCount = m_wndFormatBar.GetToolBarCtrl().GetButtonCount();
for(int i = 5 ; i < nCount ; i++)
{
m_wndFormatBar.GetButtonInfo(i,nID,nStyle,iImage);
m_wndFormatBar.SetButtonInfo(i,nID,nStyle,i - 5);
}
現在菜單標題前的位圖與工具欄的位圖不一致,大家有沒有簡單的方法?
我要改n個程序.
好累啊!
int WINAPI icePub_dispImg(HWND hWnd,char *strImgFilename,int x,int y)
輸入:hWnd 窗體或控件句柄
strImgFilename 待顯示的圖片文件名
x x坐標
y y坐標
輸出:
{
typedef int (WINAPI ICEPUB_DISPIMG)(HWND hWnd,char *strImgFilename,int x,int y);
ICEPUB_DISPIMG *icePub_dispImg = 0;
HINSTANCE hDLLDrv = LoadLibrary("icePubDll.dll");
if(hDLLDrv)
{
icePub_dispImg = (ICEPUB_DISPIMG *)GetProcAddress(hDLLDrv, "icePub_dispImg");
}
if(icePub_dispImg != NULL)
icePub_dispImg(m_hWnd,"b.png",10,10);
// icePub_dispImg(m_ctrlText.m_hWnd,"b.jpg",0,0);
if(hDLLDrv)
FreeLibrary(hDLLDrv);
}