VS2019實現C++的第一個MFC程序

一、創建項目

然後點下一步,配置項目,這裡我命名的是myfisrtmfc

點擊創建按鈕,然後彈出下面的對話框。

對上面的MFC應用程序進行配置,如下:

點擊完成,生成如下界面。

第一次編譯生成的默認項目,之後得到下面的界面

點擊VS2019的界面,“解決方案資源管理器”

到這裡,項目建成,並且編譯通過。

二、添加自定義的功能(以比較通用的畫圖為例)

點擊資源視圖,這裡的控件將是後面需要操作的。

雙擊IDR_MAINFRAME,可以在這裡面添加畫圖功能。

也可以在Ribbon裡面添加畫圖功能

然後點擊工具箱->RIbbon編輯器:

雙擊Ribbon下的面板控件

修改名稱為形狀,並添加一個按鈕控件,修改名字為矩形

修改矩形的雜項,ID改為ID_RECTANGLE

右鍵矩形按鍵,選擇添加事件處理程序

得到如下彈窗

配置這個彈窗如下:

點擊確定後,我們得到下面的代碼

以下內容參考https://www.jb51.net/article/214347.htm

第一次使用c++,mfc很多函數都不熟悉,就直接套用瞭。

這裡我們新建一個graph.cpp源文件

#include "framework.h"
#include "pch.h"
 
IMPLEMENT_SERIAL(graph, CObject, 1)
graph::graph(int l, int u, int r, int d)
{
    left = l;
    up = u;
    right = r;
    down = d;
    state = 0;
    fcolor = 0xffffff;
}
 
void graph::Offset(int cx, int cy)
{
    left += cx;
    right += cx;
    up += cy;
    down += cy;
}
 
void graph::onPress(int x, int y)
{
    sx = x; sy = y;
    state = 0;
    //選中圖形
    if (left < x && x < right &&
        up < y && y < down) {
        state = 1;
        return;
    }
    if (left - f_width / 2 < x && x < left + f_width / 2)    state |= 2;    //    選中左邊
    if (up - f_width / 2 < y && y < up + f_width / 2)    state |= 4;//選中上邊
    if (right - f_width / 2 < x && x < right + f_width / 2)    state |= 8;//選中右邊
    if (down - f_width / 2 < y && y < down + f_width / 2)    state |= 16;    //    選中下邊
 
}
 
void graph::onRelease(int x, int y)
{
    state = 0;
}
 
 
void graph::SetBorderColor(int color)
{
    fcolor = color;
}
 
void graph::SetFillColor(int color)
{
    bcolor = color;
}
int graph::onMove(int x, int y)
{
    int cx, cy;
    cx = x - sx; cy = y - sy;
    sx = x; sy = y;
 
    if (state == 1) {
        Offset(cx, cy);        //  位移量cx,cy
    }
 
    if (2 == (state & 2)) {
        left = x;
 
    }
 
    if (4 == (state & 4)) {
        up = y;
 
    }
 
    if (8 == (state & 8)) {
        right = x;
 
    }
 
    if (16 == (state & 16)) {
        down = y;
 
    }
    return state == 0 ? 0 : 1;
}
void graph::Serialize(CArchive& ar)
{
    CObject::Serialize(ar);
    if (ar.IsLoading()) {
        ar >> left >> right >> up >> down >> f_width >> fcolor >> bcolor;
    }
    else
    {
        ar << left << right << up << down << f_width << fcolor << bcolor;
    }
}
graph::~graph()
{
}
void graph::onDraw(CDC* pDC) {
    CBrush b(fcolor);
    pDC->SelectObject(&b);
    CRect r(left, up, right, down);
    pDC->FillRect(&r, &b);
    CPen p(PS_SOLID, 1, bcolor);
    pDC->SelectObject(&p);
    pDC->Rectangle(left, up, right, down);
    pDC->MoveTo(left, up);
    pDC->DrawText(_T("空圖形"), -1, new CRect(left, up, right, down), DT_CENTER | DT_VCENTER | DT_SINGLELINE);
}

在項目中添加頭文件graphz.h

在graph.h中添加下面的代碼:

#pragma once
 
class graph :
	public CObject
{
protected:
	//邊框
	DECLARE_SERIAL(graph)
	int left, up, right, down;
	//選中狀態
	unsigned int state;
	int sx, sy;
	int  f_width = 5;
	int fcolor = 0xffffff, bcolor = 0;
 
public:
	graph() :graph(50, 50, 100, 100) {
 
	}
	graph(int l, int u, int r, int d);
	void Offset(int cx, int cy);
	void  onPress(int x, int y);	//  鼠標按下
	int  onMove(int cx, int cy);	//  鼠標移動
	void  onRelease(int x, int y);	//  鼠標釋放
	virtual void onDraw(CDC* pDC);
	virtual int getGraphID() { return 0; }
	virtual void Serialize(CArchive& ar);
	void SetFillColor(int color);
	void SetBorderColor(int color);
	~graph();
 
};

在framework.h中添加graph.h

#include "graph.h"

我們要畫矩形,這裡添加矩形的相關代碼,

跟上面的步驟一樣,見一個rectangle.h和rectangle.cpp

rectangle.cpp

#include "framework.h"
#include "pch.h"
rectangle::rectangle(int l, int u, int r, int d) :graph(l, u, r, d)
{
    state = 0;
    fcolor = 0xffffff;
 
}
 
void rectangle::onDraw(CDC* pDC)
{
    CBrush b(fcolor);
    pDC->SelectObject(&b);
    CRect r(left, up, right, down);
    pDC->FillRect(&r, &b);
    CPen p(PS_SOLID, 1, bcolor);
    pDC->SelectObject(&p);
    pDC->Rectangle(left, up, right, down);
    pDC->MoveTo(left, up);
}
 
rectangle::~rectangle()
{
}

rectangle.h

#include "graph.h"
class rectangle :
    public graph
{
public:
    //DECLARE_SERIAL(graph)
    //void Serialize(CArchive& ar);
    rectangle() :graph(50, 50, 100, 100) {}
    rectangle(int l, int u, int r, int d);
    void onDraw(CDC* pDC);
    int getGraphID() { return 2; }
    ~rectangle();
};

然後myfirstmfcDoc.h中添加list

std::list<graph*> graphList;

因為調用瞭list,所以在framework.h中添加

#include <list>

這裡要調用用OnRectangle()函數,之前生成的函數,我們現在添加下面的代碼:

    CmyfisrtmfcDoc* pDoc = GetDocument();
    ASSERT_VALID(pDoc);
    if (!pDoc)
        return;
    pDoc->graphList.push_front(new rectangle(50, 50, 100, 100));
 
    Invalidate();

修改myfirstmfcView.cpp中的OnDraw函數為如下:

void CmyfisrtmfcView::OnDraw(CDC* pDC)
{
    CmyfisrtmfcDoc* pDoc = GetDocument();
    ASSERT_VALID(pDoc);
    if (!pDoc)
        return;
 
    // TODO: 在此處為本機數據添加繪制代碼
    std::list<graph*>::iterator v;
    for (v = pDoc->graphList.begin(); v != pDoc->graphList.end(); ++v) {
        (*v)->onDraw(pDC);
    }
}

接下來通過類向導添加消息

添加鼠標左鍵按下消息,左鍵松開消息,鼠標移動消息

在生成的按鍵按下函數中

void CmyfisrtmfcView::OnLButtonDown(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息處理程序代碼和/或調用默認值
    CmyfisrtmfcDoc* pDoc = GetDocument();
    ASSERT_VALID(pDoc);
    if (!pDoc)
        return;
 
    // TODO: 在此處為本機數據添加繪制代碼
    std::list<graph*>::iterator v;
    for (v = pDoc->graphList.begin(); v != pDoc->graphList.end(); ++v) {
        (*v)->onPress(point.x, point.y);
    }
    Invalidate();
    //CView::OnLButtonDown(nFlags, point);
}

跟上面一樣的方式

自從生成的代碼在myfirstmfcView中如下:

void CmyfisrtmfcView::OnLButtonUp(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息處理程序代碼和/或調用默認值
    CmyfisrtmfcDoc* pDoc = GetDocument();
    ASSERT_VALID(pDoc);
    if (!pDoc)
        return;
 
    // TODO: 在此處為本機數據添加繪制代碼
    std::list<graph*>::iterator v;
    for (v = pDoc->graphList.begin(); v != pDoc->graphList.end(); ++v) {
        (*v)->onRelease(point.x, point.y);
    }
 
    //CView::OnLButtonUp(nFlags, point);
}
 
 
void CmyfisrtmfcView::OnMouseMove(UINT nFlags, CPoint point)
{
    // TODO: 在此添加消息處理程序代碼和/或調用默認值
    CmyfisrtmfcDoc* pDoc = GetDocument();
    ASSERT_VALID(pDoc);
    if (!pDoc)
        return;
 
    // TODO: 在此處為本機數據添加繪制代碼
    std::list<graph*>::iterator v;
    for (v = pDoc->graphList.begin(); v != pDoc->graphList.end(); ++v) {
        (*v)->onMove(point.x, point.y);
    }
    Invalidate();
//    CView::OnMouseMove(nFlags, point);
}

到這裡就完成瞭全部工作,可以進行編譯瞭。

生成下面的圖形,矩形可以移動,可拉伸

點擊項目中的屬性,在配置屬性中選擇高級,MFC使用 靜態庫,在編譯一次,生成.exe可以其他電腦上不依賴動態庫也能打開瞭。

總結:

1.學會瞭如何添加項目工程

2.學會瞭添加用戶自己的源文件和頭文件,並且與項目關聯

3.學會瞭類向導

4.學會瞭按鍵控件的生成,和通過消息ID跟函數關聯起來

參考文獻:

(1)vs2019 MFC實現office界面的畫圖小項目(超超級詳細)

(2)在vs2019中使用MFC快速構建簡單windows窗口程序

到此這篇關於VS2019實現C++的第一個MFC程序的文章就介紹到這瞭,更多相關C++第一個MFC程序內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: