windows程序設計(五)---映射模式(轉換設備坐標和邏輯坐標)
在窗口指定位置顯示圖形或文字,我們使用的坐標單位是象素,稱之為設備坐標。看下麵語句:
pDC->Rectangle(CRect(0,0,200,200));
畫一個高和寬均為200個象素的方塊,因為采用的是默認的MM_TEXT映射模式,所以在設備環境不一樣時,畫的方塊大小也不一樣,在1024*768的顯示器上看到的方塊會比640*480的顯示器上的小(在不同分辨率下的屏幕象素,在WINDOWS程序設計一書中有示例程序可以獲得,或者可以用GetClientRect函數獲得客戶區的矩形大小。在這裏就不說了,大家隻要知道就行了),在輸出到打印機時也會有類似的情況發生。如何做才能保證在不同設備上得到大小一致的方塊或者圖形、文字呢?就需要我們進行選擇模式映射,來轉換設備坐標和邏輯坐標。
Windows提供了以下幾種映射模式:
MM_TEXT
MM_LOENGLISH
MM_HIENGLISH
MM_LOMETRIC
MM_HIMETRIC
MM_TWIPS
MM_ISOTROPIC
MM_ANISOTROPIC
下麵分別講講這幾種映射模式:
MM_TEXT:
默認的映射模式,把設備坐標被映射到象素。x值向右方向遞增;y值向下方向遞增。坐標原點是屏幕左上角(0,0)。但我們可以通過調用CDC的SetViewprotOrg和SetWindowOrg函數來改變坐標原點的位置看下麵兩個例子:
//************************************************
// 例子1
void CMyView::OnDraw(CDC * pDC)
{
pDC->Rectangle(CRect(0,0,200,200));//全部采用默認畫一個寬和高為200象素的方塊
}
//**************************************************
// 例子2
void CMyView::OnDraw(CDC * pDC)
{
pDC->SetMapMode(MM_TEXT);//設定映射模式為MM_TEXT
pDC->SetWindowOrg(CPoint(100,100));//設定邏輯坐標原點為(100,100)
pDC->Rectangle(CRect(100,100,300,300));//畫一個寬和高為200象素的方塊
}
這兩個例子顯示出來的圖形是一樣的,都是從屏幕左上角開始的寬和高為200象素的方塊,可以看出例子2將邏輯坐標(100,100)映射到了設備坐標(0,0)處,這樣做有什麼用?滾動窗口使用的就是這種變換。
固定比例映射模式:
MM_LOENGLISH、MM_HIENGLISH、MM_LOMETRIC、MM_HIMETRIC、MM_TWIPS這一組是Windows提供的重要的固定比例映射模式。
它們都是x值向右方向遞增,y值向下遞減,並且無法改變。它們之間的區別在於比例因子見下:(我想書上P53頁肯定是印錯了,因為通過程序實驗x值向右方向也是遞增的)
MM_LOENGLISH 0.01英寸
MM_HIENGLISH 0.001英寸
MM_LOMETRIC 0.1mm
MM_HIMETRIC 0.01mm
MM_TWIPS 1/1440英寸 //應用於打印機,一個twip相當於1/20磅,一磅又相當於1/72英寸。
看例3
//**************************************************
// 例子3
void CMyView::OnDraw(CDC * pDC)
{
pDC->SetMapMode(MM_HIMETRIC);//設定映射模式為MM_HIMETRIC
pDC->Rectangle(CRect(0,0,4000,-4000));//畫一個寬和高為4厘米的方塊
}
還有一種是可變比例映射模式,MM_ISOTROPIC、MM_ANISOTROPIC。用這種映射模式可以做到當窗口大小發生變化時圖形的大小也會相應的發生改變,同樣當翻轉某個軸的伸展方向時圖象也會以另外一個軸為軸心進行翻轉,並且我們還可以定義任意的比例因子,怎麼樣很有用吧。
MM_ISOTROPIC、MM_ANISOTROPIC兩種映射模式的區別在於MM_ISOTROPIC模式下無論比例因子如何變化縱橫比是1:1而M_ANISOTROPIC模式則可以縱橫比獨立變化。
讓我們看例子4
//**************************************************
// 例子4
view plaincopy to clipboardprint?
01.void CMy002View::OnDraw(CDC* pDC)
02.{
03.CRect rectClient; //
04.GetClientRect(rectClient);//返回客戶區矩形的大小
05.pDC->SetMapMode(MM_ANISOTROPIC);//設定映射模式為MM_ANISOTROPIC
06.pDC->SetWindowExt(1000,1000);
07.pDC->SetViewportExt (rectClient.right ,-rectClient.bottom );
08.//用SetWindowExt和SetViewportExt函數設定窗口為1000邏輯單位高和1000邏輯單位寬
09.pDC->SetViewportOrg(rectClient.right/2,rectClient.bottom/2 );//設定邏輯坐標原點為窗口中心
10.pDC->Ellipse(CRect(-500,-500,500,500));//畫一個撐滿窗口的橢圓。
11.// TODO: add draw code for native data here
12.}
void CMy002View::OnDraw(CDC* pDC)
{
CRect rectClient; //
GetClientRect(rectClient);//返回客戶區矩形的大小
pDC->SetMapMode(MM_ANISOTROPIC);//設定映射模式為MM_ANISOTROPIC
pDC->SetWindowExt(1000,1000);
pDC->SetViewportExt (rectClient.right ,-rectClient.bottom );
//用SetWindowExt和SetViewportExt函數設定窗口為1000邏輯單位高和1000邏輯單位寬
pDC->SetViewportOrg(rectClient.right/2,rectClient.bottom/2 );//設定邏輯坐標原點為窗口中心
pDC->Ellipse(CRect(-500,-500,500,500));//畫一個撐滿窗口的橢圓。
// TODO: add draw code for native data here
}
怎麼樣,屏幕上有一個能跟隨窗口大小改變而改變的橢圓。把 pDC->SetMapMode(MM_ANISOTROPIC);這句改為pDC->SetMapMode(MM_ISOTROPIC)會怎樣?大家可以試試。那還有一個問題就是上例的比例因子是多少呢?看下麵公式(注意是以例子4為例的)
x比例因子=rectClient.right/1000 //視窗的寬除以窗口範圍
y比例因子=-rectClient.bottom/1000 //視窗的高除以窗口範圍
從Windows的鼠標消息可以獲得鼠標指針的當前坐標值(point.x和point.y)此坐標值是設備坐標。
很多MFC庫函數尤其是CRect的成員函數隻能工作在設備坐標下。
還有我們有時需要利用物理坐標,物理坐標的概念就是現實世界的實際尺寸。
設備坐標-邏輯坐標-物理坐標之間如何進行轉換便成為我們要考慮的一個問題,物理坐標和邏輯坐標是完全要我們自己來做的,但WINDOWS提供了函數來幫助我們轉換邏輯坐標和設備坐標。
CDC的LPtoDP函數可以將邏輯坐標轉換成設備坐標
CDC的DPtoLP函數可以將設備坐標轉換成邏輯坐標
下麵列出我們應該在什麼時候使用什麼樣的坐標係一定要記住:
◎CDC的所有成員函數都以邏輯坐標為參數
◎CWnd的所有成員函數都以設備坐標為參數
◎區域的定義采用設備坐標
◎所有的選中測試操作應考慮使用設備坐標。
◎需要長時間使用的值用邏輯坐標或物理坐標來保存。因設備坐標會因窗口的滾動變化而改變.
最後更新:2017-04-02 06:51:27