• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            The Fourth Dimension Space

            枯葉北風(fēng)寒,忽然年以殘,念往昔,語(yǔ)默心酸。二十光陰無(wú)一物,韶光賤,寐難安; 不畏形影單,道途阻且慢,哪曲折,如渡飛湍。斬浪劈波酬壯志,同把酒,共言歡! -如夢(mèng)令

            Qt學(xué)習(xí)之路(28): 坐標(biāo)變換

            經(jīng)過(guò)前面的章節(jié),我們已經(jīng)能夠畫出一些東西來(lái),主要就是使用QPainter的相關(guān)函數(shù)。今天,我們要看的是QPainter的坐標(biāo)系統(tǒng)。
            同很多坐標(biāo)系統(tǒng)一樣,QPainter的默認(rèn)坐標(biāo)的原點(diǎn)(0, 0)位于屏幕的左上角,X軸正方向是水平向右,Y軸正方向是豎直向下。在這個(gè)坐標(biāo)系統(tǒng)中,每個(gè)像素占據(jù)1 x 1的空間。你可以把它想象成是一張坐標(biāo)值,其中的每個(gè)小格都是1個(gè)像素。這么說(shuō)來(lái),一個(gè)像素的中心實(shí)際上是一個(gè)“半像素坐標(biāo)系”,也就是說(shuō),像素(x, y)的中心位置其實(shí)是在(x + 0.5, y + 0.5)的位置上。因此,如果我們使用QPainter在(100, 100)處繪制一個(gè)像素,那么,這個(gè)像素的中心坐標(biāo)是(100.5, 100.5)。
             
            這種細(xì)微的差別在實(shí)際應(yīng)用中,特別是對(duì)坐標(biāo)要求精確的系統(tǒng)中是很重要的。首先,只有在禁止反走樣,也就是默認(rèn)狀態(tài)下,才會(huì)有這0.5像素的偏移;如果使用了反走樣,那么,我們畫(100, 100)位置的像素時(shí),QPainter會(huì)在(99.5, 99.5),(99.5, 100.5),(100.5, 99.5)和(100.5, 100.5)四個(gè)位置繪制一個(gè)亮色的像素,這么產(chǎn)生的效果就是在這四個(gè)像素的焦點(diǎn)處(100, 100)產(chǎn)生了一個(gè)像素。如果不需要這個(gè)特性,就需要將QPainter的坐標(biāo)系平移(0.5, 0.5)。
             
            這一特性在繪制直線、矩形等圖形的時(shí)候都會(huì)用到。下圖給出了在沒有反走樣技術(shù)時(shí),使用drawRect(2, 2, 6, 5)繪制一個(gè)矩形的示例。在No Pen的情況下,請(qǐng)注意矩形左上角的像素是在(2, 2),其中心位置是在(2.5, 2.5)的位置。然后注意下有不同的Pen的值的繪制樣式,在Pen寬為1時(shí),實(shí)際畫出的矩形的面積是7 x 6的(圖出自C++ GUI Programming with Qt4, 2nd Edition):
            在具有反走樣時(shí),使用drawRect(2, 2, 6, 5)的效果如下(圖出自C++ GUI Programming with Qt4, 2nd Edition):
            注意我們前面說(shuō)過(guò),通過(guò)平移QPainter的坐標(biāo)系來(lái)消除著0.5像素的差異。下面給出了使用drawRect(2.5, 2.5, 6, 5)在反走樣情況下繪制的矩形(圖出自C++ GUI Programming with Qt4, 2nd Edition):
            請(qǐng)對(duì)比與上圖的區(qū)別。
             
            在上述的QPainter的默認(rèn)坐標(biāo)系下,QPainter提供了視口(viewport)窗口(window)機(jī)制,用于繪制與繪制設(shè)備的大小和分辨率無(wú)關(guān)的圖形。視口和窗口是緊密的聯(lián)系在一起的,它們一般都是矩形。視口是由物理坐標(biāo)確定其大小,而窗口則是由邏輯坐標(biāo)決定。我們?cè)谑褂肣Painter進(jìn)行繪制時(shí),傳給QPainter的是邏輯坐標(biāo),然后,Qt的繪圖機(jī)制會(huì)使用坐標(biāo)變換將邏輯坐標(biāo)轉(zhuǎn)換成物理坐標(biāo)后進(jìn)行繪制。
             
            通常,視口和窗口的坐標(biāo)是一致的。比如一個(gè)600 x 800的widget(這是一個(gè)widget,或許是一個(gè)對(duì)話框,或許是一個(gè)面板等等),默認(rèn)情況下,視口和窗口都是一個(gè)320 x 200的矩形,原點(diǎn)都在(0, 0),此時(shí),視口和窗口的坐標(biāo)是相同的。
             
            注意到QPainter提供了setWindow()和setViewport()函數(shù),用來(lái)設(shè)置視口和窗口的矩形大小。比如,在上面所述的320 x 200的widget中,我們要設(shè)置一個(gè)從(-50, -50)到(+50, +50),原點(diǎn)在中心的矩形窗口,就可以使用
             
            painter.setWindow(-50, -50, 100, 100);
             
            其中,(-50, -50)指明了原點(diǎn),100, 100指明了窗口的長(zhǎng)和寬。這里的“指明原點(diǎn)”意思是,邏輯坐標(biāo)的(-50, -50)對(duì)應(yīng)著物理坐標(biāo)的(0, 0);“長(zhǎng)和寬”說(shuō)明,邏輯坐標(biāo)系下的長(zhǎng)100,寬100實(shí)際上對(duì)應(yīng)物理坐標(biāo)系的長(zhǎng)320,寬200。
             
            或許你已經(jīng)發(fā)現(xiàn)這么一個(gè)好處,我們可以隨時(shí)改變window的范圍,而不改變底層物理坐標(biāo)系。這就是前面所說(shuō)的,視口與窗口的作用:“繪制與繪制設(shè)備的大小和分辨率無(wú)關(guān)的圖形”,如下圖所示(圖出自C++ GUI Programming with Qt4, 2nd Edition):
             
             
            除了視口與窗口的變化,QPainter還提供了一個(gè)“世界坐標(biāo)系”,同樣也可以變換圖形。所不同的是,視口與窗口實(shí)際上是統(tǒng)一圖形在兩個(gè)坐標(biāo)系下的表達(dá),而世界坐標(biāo)系的變換是通過(guò)改變坐標(biāo)系來(lái)平移、縮放、旋轉(zhuǎn)、剪切圖形。為了清楚起見,我們來(lái)看下面一個(gè)例子:
             
            void PaintedWidget::paintEvent(QPaintEvent *event)
            {
                    QPainter painter(this);
                    QFont font("Courier", 24);
                    painter.setFont(font);
                    painter.drawText(50, 50, "Hello, world!");
                    QTransform transform;
                    transform.rotate(+45.0);
                    painter.setWorldTransform(transform);
                    painter.drawText(60, 60, "Hello, world!");
            }
             
            為了顯示方便,我在這里使用了QFont改變了字體。QPainter的drawText()函數(shù)提供了繪制文本的功能。它有幾種重載形式,我們使用了其中的一種,即制定文本的坐標(biāo)然后繪制。需要注意的是,這里的坐標(biāo)是文字左下角的坐標(biāo)(特別提醒這一點(diǎn),因?yàn)楹芏嗬L圖系統(tǒng),比如Java2D都是把左上角作為坐標(biāo)點(diǎn)的)!下面是運(yùn)行結(jié)果:
             
            我們使用QTransform做了一個(gè)rotate變換。這個(gè)變換就是旋轉(zhuǎn),而且是順時(shí)針旋轉(zhuǎn)45度。然后我們使用這個(gè)變換設(shè)置了QPainter的世界坐標(biāo)系,注意到QPainter是一個(gè)狀態(tài)機(jī),所以這種變換并不會(huì)改變之前的狀態(tài),因此只有第二個(gè)Hello, world!被旋轉(zhuǎn)了。確切的說(shuō),被旋轉(zhuǎn)的是坐標(biāo)系而不是這個(gè)文字!請(qǐng)注意體會(huì)這兩種說(shuō)法的不同。

            本文出自 “豆子空間” 博客,請(qǐng)務(wù)必保留此出處http://devbean.blog.51cto.com/448512/239585

            posted on 2010-12-27 16:24 abilitytao 閱讀(848) 評(píng)論(0)  編輯 收藏 引用


            只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問(wèn)   Chat2DB   管理


            久久国产精品国语对白| 一本综合久久国产二区| 久久精品国产亚洲AV香蕉| 99久久国产综合精品五月天喷水 | 亚洲精品无码久久久久sm| 爱做久久久久久| 国产福利电影一区二区三区久久久久成人精品综合 | 国产精品久久久久久久久软件| 久久综合丝袜日本网| 久久久无码精品亚洲日韩按摩 | 亚洲愉拍99热成人精品热久久 | 欧美精品国产综合久久| 久久笫一福利免费导航| 伊人久久久AV老熟妇色| 久久久精品2019免费观看| 浪潮AV色综合久久天堂| 久久人人爽人人爽人人片av高请| 国产亚洲精品美女久久久| 国产午夜精品理论片久久影视| 精品多毛少妇人妻AV免费久久| 久久精品一区二区三区中文字幕 | 亚洲国产精品成人久久蜜臀 | 精品国产91久久久久久久a | 欧美日韩精品久久久免费观看| 97精品依人久久久大香线蕉97| 亚洲AV日韩精品久久久久久久| 久久精品国产99久久无毒不卡| 久久99国产精品一区二区| 美女写真久久影院| 久久青青色综合| 久久精品国产99国产精品澳门| 亚洲精品NV久久久久久久久久| 老色鬼久久亚洲AV综合| 97精品伊人久久久大香线蕉| 日日狠狠久久偷偷色综合0| 欧美亚洲国产精品久久久久| 久久国产精品-国产精品| 亚洲&#228;v永久无码精品天堂久久 | 国产精品中文久久久久久久| 亚洲香蕉网久久综合影视| 久久综合九色综合97_久久久|