相關鏈接:
C++ GUI 繪圖控件目錄 MFC
Qt |
也許這是vc下最好最方便的繪圖類,它有TeeChart的繪圖和操作風格,不用當心註冊破解的問題,因爲它是開源的。不用打包註冊,因爲它是封裝成類的,能方便擴展繼承。vc6.0到vs2010都能使用,而且非常簡單。
此類發表於codeproject
在使用它的時候,展示一下它的效果吧:
如果你想需要上面這些效果的,果斷選它吧!
下面用圖文並茂的方式,來詳細介紹這個繪圖控件
首先,下載這個控件,最新可以從這裏獲取codeproject
1 ChartCtrl類的導入
文件夾內容如圖所示
2.創建控件
2.1 對話框編輯器創建
對於一些不需要改變大小的對話框來說,在對話框編輯器裏拖曳創建控件是最舒服的方法了,這個ChartCtrl可以用用戶控件來創建首先在對話框上放置一個Custom Control
給對話框添加變量,和傳統的方法一樣。這裏需要注意的是,由於文件都放置在工程文件的一個文件夾下,包含頭文件時需要指明路徑
#include "ChartCtrl/ChartCtrl.h"
CChartCtrl m_ChartCtrl2;
void CSpeedChartCtrlDemoDlg::DoDataExchange(CDataExchange* pDX)
{
CDialogEx::DoDataExchange(pDX);
DDX_Control(pDX, IDC_ChartCtrl1, m_ChartCtrl1);
}
編譯運行,繪圖控件就出來了2.2 動態創建
#include "ChartCtrl/ChartCtrl.h"
然後添加成員變量CChartCtrl m_ChartCtrl2;
添加IDC_ChartCtrl2,爲1001,注意記得把_APS_NEXT_CONTROL_VALUE改成下一個資源號
在OnInitDialog裏創建
CRect rect,rectChart;
GetDlgItem(IDC_ChartCtrl1)->GetWindowRect(&rect);
ScreenToClient(rect);
rectChart = rect;
rectChart.top = rect.bottom + 3;
rectChart.bottom = rectChart.top + rect.Height();
m_ChartCtrl2.Create(this,rectChart,IDC_ChartCtrl2);
m_ChartCtrl2.ShowWindow(SW_SHOWNORMAL);
這樣就可以創建了,下圖兩個控件分別通過對話框編輯器創建和動態創建,代碼在附件下載裏
此時什麼也不會顯示,需要添加座標軸
3.創建座標軸
CChartAxis *pAxis= NULL;
pAxis = m_ChartCtrl1.CreateStandardAxis(CChartCtrl::BottomAxis);
pAxis->SetAutomatic(true);
pAxis = m_ChartCtrl1.CreateStandardAxis(CChartCtrl::LeftAxis);
pAxis->SetAutomatic(true);
這樣就建立兩個座標軸了,如圖所示CChartDateTimeAxis* pDateAxis= NULL;
pDateAxis = NULL;
pDateAxis = m_ChartCtrl2.CreateDateTimeAxis(CChartCtrl::BottomAxis);
pDateAxis->SetAutomatic(true);
pDateAxis->SetTickLabelFormat(false,_T("%m月%d日"));
pAxis = m_ChartCtrl2.CreateStandardAxis(CChartCtrl::LeftAxis);
pAxis->SetAutomatic(true);
4.創建標題
在添加標題時,先要說說ChartCtrl的字符串,ChartCtrl的字符串實際是stl的string和wstring,爲了對應unicode,作者對這兩種字符進行了一個宏定義,就像TCHAR一樣,定義如下:
#include<string>
#include <sstream>
#if defined _UNICODE ||defined UNICODE
typedef std::wstring TChartString;
typedef std::wstringstream TChartStringStream;
#else
typedef std::string TChartString;
typedef std::stringstream TChartStringStream;
#endif
所以在多字節情況下,就是string。由於MFC大部分都是用CString,CString也是經過宏定義,所以可以比較輕鬆的和TChartString轉換,另外TChartStringStream遠比CString的Format靈活和直觀,建議大家研究研究!
加入如下代碼:
TChartString str1;
str1 = _T("IDC_ChartCtrl1 - m_ChartCtrl1");
m_ChartCtrl1.GetTitle()->AddString(str1);
CString str2(_T(""));
str2 = _T("IDC_ChartCtrl2 - m_ChartCtrl2");
m_ChartCtrl2.GetTitle()->AddString(TChartString(str2));
TChartString 可以直接用“=”對字符串賦值
設置座標軸的標題,首先需要獲取座標GetLeftAxis,GetBottomAxis ……
獲取座標後,獲得座標的文字標籤GetLabel,然後進行修改
如下兩種寫法,一種比較安全繁瑣,一種就直接過去就可以,看個人喜好
CChartAxisLabel* pLabel = NULL;
CChartAxis *pAxis = NULL;
TChartString str1 = _T("左座標軸");
CChartAxisLabel* pLabel = NULL;
pAxis = m_ChartCtrl1.GetLeftAxis();
if(pAxis)
pLabel = pAxis->GetLabel();
if(pLabel)
pLabel->SetText(str1);
m_ChartCtrl2.GetLeftAxis()->GetLabel()->SetText(str1);
str1 = _T("數值座標軸");
pAxis = m_ChartCtrl1.GetBottomAxis();
if(pAxis)
pLabel = pAxis->GetLabel();
if(pLabel)
pLabel->SetText(str1);
str1 = _T("時間座標軸");
m_ChartCtrl2.GetBottomAxis()->GetLabel()->SetText(str1);
設置完效果如圖
標題還可以更改顏色,這裏不再重複描述。
5.畫圖
5.1 創建線圖
m_ChartCtrl1.EnableRefresh(false);
m_ChartCtrl2.EnableRefresh(false);
//////////////////////////////////////////////////////////////////////////
//畫圖測試
//////////////////////////////////////////////////////////////////////////
double x[1000], y[1000];
for (int i=0; i<1000; i++)
{
x[i] = i;
y[i] = sin(float(i));
}
CChartLineSerie *pLineSerie1;
m_ChartCtrl1.RemoveAllSeries();//先清空
pLineSerie1 = m_ChartCtrl1.CreateLineSerie();
pLineSerie1->SetSeriesOrdering(poNoOrdering);//設置爲無序
pLineSerie1->AddPoints(x, y,1000);
pLineSerie1->SetName(_T("這是IDC_ChartCtrl1的第一條線"));//SetName的作用將在後面講到
//////////////////////////////////////////////////////////////////////////
//時間軸畫圖
//////////////////////////////////////////////////////////////////////////
COleDateTime t1(COleDateTime::GetCurrentTime());
COleDateTimeSpan tsp(1,0,0,0);
for (int i=0; i<1000; i++)
{
x[i] = t1.m_dt;
y[i] = sin(float(i));
t1 += tsp;
}
CChartLineSerie *pLineSerie2;
m_ChartCtrl2.RemoveAllSeries();//先清空
pLineSerie2 = m_ChartCtrl2.CreateLineSerie();
pLineSerie2->SetSeriesOrdering(poNoOrdering);//設置爲無序
pLineSerie2->AddPoints(x, y,1000);
pLineSerie2->SetName(_T("這是IDC_ChartCtrl2的第一條線"));//SetName的作用將在後面講到
m_ChartCtrl1.EnableRefresh(true);
m_ChartCtrl2.EnableRefresh(true);
5.2 添加曲線
m_ChartCtrl1.EnableRefresh(false);
m_ChartCtrl2.EnableRefresh(false);
//////////////////////////////////////////////////////////////////////////
//畫圖測試
//////////////////////////////////////////////////////////////////////////
double x[1000], y[1000];
for (int i=0; i<1000; i++)
{
x[i] = i;
y[i] = sin(float(i)*m_ChartCtrl1.GetSeriesCount());
}
CChartLineSerie *pLineSerie1;
// m_ChartCtrl1.RemoveAllSeries();//不清空
pLineSerie1 = m_ChartCtrl1.CreateLineSerie();
pLineSerie1->SetSeriesOrdering(poNoOrdering);//設置爲無序
pLineSerie1->AddPoints(x, y,1000);
TChartStringStream strs1;
strs1 << _T("這是IDC_ChartCtrl1的第")
<< m_ChartCtrl1.GetSeriesCount()
<< _T("條曲線");
pLineSerie1->SetName(strs1.str());
//////////////////////////////////////////////////////////////////////////
//時間軸畫圖
//////////////////////////////////////////////////////////////////////////
COleDateTime t1(COleDateTime::GetCurrentTime());
COleDateTimeSpan tsp(1,0,0,0);
for (int i=0; i<1000; i++)
{
x[i] = t1.m_dt;
y[i] = sin(float(i)*m_ChartCtrl2.GetSeriesCount());
t1 += tsp;
}
CChartLineSerie *pLineSerie2;
// m_ChartCtrl2.RemoveAllSeries();//不清空
pLineSerie2 = m_ChartCtrl2.CreateLineSerie();
pLineSerie2->SetSeriesOrdering(poNoOrdering);//設置爲無序
pLineSerie2->AddPoints(x, y,1000);
TChartStringStream strs2;
strs2 << _T("這是IDC_ChartCtrl2的第")
<< m_ChartCtrl2.GetSeriesCount()
<< _T("條曲線");
pLineSerie2->SetName(strs2.str());
m_ChartCtrl1.EnableRefresh(true);
m_ChartCtrl2.EnableRefresh(true);
5.3 動態曲線
for(int i(0);i<99;++i)
{
pdx[i] = pdx[i+1];
pdy[i] = pdy[i+1];
}
pdy[99] = ……需要顯示的新數據……
ChartCtrl提供了兩種繪圖函數,AddPoints和AddPoint,這兩種函數在繪製動態圖時會有所區別,區別見上圖,上面的是AddPoints,繪圖長度固定,AddPoint效果見下圖,圖線會不停積累。
這是我在上研究生時寫的一篇文章,當時放進草稿箱裏一直沒發出來,今天無意看到,決定把他完成,demo代碼已經找不到了,可能在我以前實驗室的電腦裏吧~工作後也很少用mfc了,現在偶爾用用qt過把癮,大家如果對繪圖工控有需求的話,可以使用qt的qwt控件,也比較簡單,這個控件幫了我很多忙,在此向作者表示感謝,大家可以查看其源代碼學習其編程思想。
下載地址:http://www.codeproject.com/Articles/14075/High-speed-Charting-Control
csdn資源下載地址:http://download.csdn.net/detail/czyt1988/6880917