• <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>
            我們知道,關于高DPI的支持, Windows XP時代就開始有了, 那時關于高DPI的支持比較簡單, 但是從Vista/Win7 到現在Win8 /Win8.1, Windows關于高DPI的支持已經發生了很大的變化, 下面我們依次簡單介紹下。

            如果說以前XP時代我們還有理由不關注高DPI,  那么在移動設備時代和大顯示器的高分辨率時代, 我們就沒有理由不關注高DPI了, 比如Surface Pro的分辨率是1920x1080, 這種情況下如果系統我們不設置高DPI, 基本上就沒法觸摸和操作了,所以現在普通程序對高DPI的支持已經成為趨勢了。 

            什么DPI? 全稱是dots per inch (DPI), 也就是每英寸的點數,在顯示器上就是每英寸的像素個數,Window上一般默認是96 dpi 作為100% 的縮放比率, 但是要注意的是該值未必是真正的顯示器物理值, 只是Windows里我們的一個參考標準。

            下面我們思考為什么DPI設置高了之后, 我們看到的字體會變大? 因為系統字體是是以固定大小(宋體10號字,物理尺寸為(10/72)英寸)設計的, 當我們DPI設置高了之后 ,說明該字體要占有更多的像素, 在屏幕分辨率不變的前提下, 看起來也就大了。所以如果我們設置高DPI,通常也意味著我們的顯示器是高分辨率, 里面的字體看起來太小了, 我們需要提高DPI來把內容放大。

            那么我們的程序如何才能支持高DPI? 對于高DPI的支持, 不同操作系統有不同的方案。通常來說如果我們程序支持高DPI, 意味著我們要對繪畫的內容進行相應的放大, 比如字體,圖片和控件等。當然, 如果我們用的是系統字體(比如GetStockObject(DEFAULT_GUI_FONT)), 那么這種情況下我們不用操心, 因為系統會對該字體在高DPI時進行相應的放大; 如果我們是用CreateFont自己創建的字體, 那就要我們自己對該字體進行放大了。

            下面我們看XP是如何對高DPI進行支持的? 

            XP對高DPI的支持比較差勁, 大部分情況下就是字體的放大, 當然我們程序也可以通過GetDeviceCaps(hDC, LOGPIXELSX)獲取DPI后自己對繪畫的內容進行縮放。

            下面我們看Vista/Win7/Win8是如何對高DPI進行支持的?

            我們知道Vista/Win7我們可以禁止DWM(Desktop Window Manager), 該模式我們稱之為Basic模式, 這種模式下的高DPI效果和XP一樣。

            對于DWM沒有禁掉的情況, Vista/Win7/Win8 對高DPI的支持又分為2種情況, 具體看下圖: 

            一種XP風格的高DPi支持, 這種方式我們上面討論過了;
            還有一種是通過 DWM 虛擬化支持的 高DPI方式, 下面我們討論下該方式: 

            該種方式的高DPI支持是通過DWM的縮放實現的, 具體過程是這樣的, 比如我們當前系統的DPI是200%, 我們程序運行時,系統會告訴你當前DPI仍然是96(100%), 所以我們程序會仍然按照100%的方式進行繪畫, 但是但是系統給我們的坐標是根據DPI縮小過后的(也就是我們對窗口調用GetWindowRect或是通過GetSystemMetrics(SM_CXSCREEN)得到的大小會比實際大小減半) , 當我們畫完之后, DWM再對整個窗口進行200% 放大后畫到屏幕上, 這樣看起來我們的程序就自動支持高DPI了。

             這種方式看起來很美妙, 但是它也有缺點, 主要是經過縮放后的內容看起來會變模糊, 比如文字會有明顯的鋸齒。

            既然DWM虛擬化用戶效果有時不是那么好, 那么我們很多時候可能會自己支持高DPI, 如何讓我們的程序禁用該效果?
            事實上我們可以對每個進程對DWM虛擬化的支持進行設置和查詢, 系統給我們提供了2個APi: SetProcessDPIAware  IsProcessDPIAware , 通過調用SetProcessDPIAware , 我們告訴系統不要對我們的程序進行DWM虛擬化。

            這里還有特殊情況也提一下: 我們在高DPI下通過窗口句柄取到的坐標信息是和目標程序是否支持DWM虛擬化相關聯的, 我們對支持DWM虛擬化的程序窗口調用GetWindowRect, 取到的坐標也是經過DWM縮放后的坐標; 對禁用DWM虛擬化程序的窗口調用GetWindowRect, 取到的坐標則是沒有經過縮放的原始坐標。

             最后我們再討論下Win8.1 對高DPI的支持, WIn8.1對高DPi以3種方式支持 Process_DPI_Awareness : 
            typedef enum _Process_DPI_Awareness { 
              Process_DPI_Unaware            = 0,
              Process_System_DPI_Aware       = 1,
              Process_Per_Monitor_DPI_Aware  = 2
            } Process_DPI_Awareness;
            下面我們依次討論這3種方式: 

            第一種Unaware, 該種方式是告訴系統, 我的程序不支持DPI aware, 請通過DWM虛擬化幫我們實現。 該方式和上面Win7/Win8對高DPI的支持的實現基本一樣,主要區別是它通過GetWindowRect取到的坐標都是經過DWM縮放后的, 無論對方窗口是不是支持DWM虛擬化。

            第二種方式是System DPI aware, 該方式下告訴系統, 我的程序會在啟動的顯示器上自己支持DPI aware, 所以不需要對我進行DWM 虛擬化。 但是當我的程序被拖動到其他DPI不一樣的顯示器時, 請對我們先進行system DWM虛擬化縮放。

            第三種方式是Per Monitor DPI aware, 該方式是告訴系統, 請永遠不要對我進行DWM虛擬化,我會自己針對不同的Monitor的DPi縮放比率進行縮放。

            再介紹下相關API:
            SetProcessDpiAwareness :設置當前進程對高DPi的支持方式
            GetProcessDpiAwareness :查詢某個進程對高DPI的支持方式
            GetDpiForMonitor : 獲取某個Monitor的DPI
            WM_DPICHANGED :當某個程序窗口被拖到另外一個DPI的Monitor時收到

            最后,簡單總結下, 從上面我們可以看到微軟在不同操作系統上對高DPI支持的改進線路,很多方面也體現了他們對老程序兼容性上的考慮, DWM虛擬化雖然很簡單, 卻丟失了用戶體驗。  

            PS, 我在我機器上測試發現,桌面程序基本上只有微軟自己的程序能做到在高DPI下完美支持, 其他大部分程序(即使如Chrome)也是通過DWM虛擬化實現的高DPI支持。當然現在WPF和Window store App基本上都是內置支持高DPI的。

            統計下, 你們的程序支持高DPI嗎? 

                              High DPI Settings in Windows
            posted on 2014-02-18 23:12 Richard Wei 閱讀(41863) 評論(10)  編輯 收藏 引用 所屬分類: windows desktop

            FeedBack:
            # re: 關于Windows高DPI的一些簡單總結
            2014-03-03 09:22 | Won
            Vista/Win7 125%與150%的處理方式,應該是XP與Vista模式的差別吧  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2014-03-04 19:14 | Richard Wei
            @Won
            確實,DWM虛擬化只有在大于125%時才會自動開啟  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2014-09-14 16:06 | Richard Wei
            有人提到中文中夾雜英文的問題,不多說了,在外企呆過的同學應該都會不自覺地有這個習慣,并且有時候錯誤的中文還不如用原始的英文。  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結[未登錄]
            2014-10-12 02:09 | gyj
            感謝樓主。  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2014-11-22 21:32 | 好文章
            windows 8下一般只有微軟的應用才完美支持hi-dpi的,“云團隊”率先在windows下完美支持hi-dpi設置,效率高,效果超好,大家可以看看 http://www.17team.cn  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2014-12-16 09:35 | craft
            效果的確不錯,動畫很舒服。  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2015-01-01 23:56 | ibuick
            Windows跟OSX比就是一垃圾貨  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結[未登錄]
            2015-01-12 15:36 | David
            @ibuick
            明明是軟件為了兼容XP用了老舊的庫導致對新系統兼容性爛。。  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2016-02-08 11:01 | Nukepayload2
            看來還是WPF之類的Direct UI有前途,不用管Dpi的事情,總是非常清晰,比例也總是非常協調。  回復  更多評論
              
            # re: 關于Windows高DPI的一些簡單總結
            2016-03-22 21:56 | xiabibi
            然而擴展顯示器也無法分別設置不同的縮放級別喝喝了  回復  更多評論
              
            国产亚洲美女精品久久久久狼| 人人狠狠综合久久88成人| 国产三级观看久久| 久久成人国产精品一区二区| 久久精品无码免费不卡| 精品久久久中文字幕人妻| 97久久精品国产精品青草| 久久天天躁狠狠躁夜夜不卡| 日日躁夜夜躁狠狠久久AV| 国产精品青草久久久久福利99| 久久久久久久97| 成人a毛片久久免费播放| 久久午夜无码鲁丝片| 久久伊人中文无码| 国产V综合V亚洲欧美久久| 久久综合久久美利坚合众国| 91精品国产91久久久久久青草| 久久久久久久波多野结衣高潮 | 精品久久久久久国产| 人人狠狠综合久久亚洲88| 亚洲人成无码久久电影网站| 婷婷综合久久中文字幕蜜桃三电影 | 久久久久亚洲AV无码专区桃色| 一本一本久久aa综合精品| 美女久久久久久| 99久久国产免费福利| 精品永久久福利一区二区| 久久久久久久女国产乱让韩 | 四虎国产精品成人免费久久| 精品无码人妻久久久久久| 777米奇久久最新地址| 久久人妻AV中文字幕| 怡红院日本一道日本久久| 久久99免费视频| av无码久久久久久不卡网站| 亚洲av伊人久久综合密臀性色 | 青青草原精品99久久精品66| 久久无码中文字幕东京热| 国产精品久久久久久久久软件| 午夜精品久久久久久| 久久人妻少妇嫩草AV蜜桃|