響應式網站建設的思路中,一個重要的因素是怎樣處理圖片方面的問題。有很多同比縮放圖片的技術,其中有不少是簡單易行的。其中,由Richard Rutter最先嘗試的一種做法比較流行,即使用CSS的max-width屬性。這個方法在Ethan Marcotte的液態圖片一文中也有提到。
1 | img {max-width:100%;} |
只要沒有其他涉及到圖片寬度的樣式代碼覆蓋掉這一行規則,頁面上所有的圖片就會以其原始寬度進行加載,除非其容器可視部分的寬度小于圖片的原始寬度。上面的代碼確保圖片最大的寬度不會超過瀏覽器窗口或是其容器可視部分的寬度,所以當窗口或容器的可視部分開始變窄時,圖片的最大寬度值也會相應的減小,圖片本身永遠不會容器邊緣隱藏和覆蓋。實際上,就像Jason Grigsby在他的CSS Media Query for Mobile is Fool’s Gold一文中提到的,“液態圖片背后的思路,就是無論何時,都確保在其原始寬度范圍內,以最大的寬度同比完整的顯示圖片。我們不必在樣式表中為圖片設置寬度和高度,只需要讓樣式表在窗口尺寸發生變化時輔助瀏覽器對圖片進行縮放。” 一種簡而美的方法。
圖片本身的分辨率及加載時間是另外一個需要考慮的問題。雖然通過上面的方法,可以很輕松的縮放圖片,確保在移動設備的窗口中可以被完整瀏覽,但如果原始圖片本身過大,便會顯著降低圖片文件的下載速度,對存儲空間也會造成沒有必要的消耗。
響應式圖片
由Filament Group提出的“響應式圖片”技術思想,有助于解決上面提到的問題:不僅要同比的縮放圖片,還要在小設備上降低圖片自身的分辨率。可以看下demo頁面先。
這個技術的實現需要使用幾個相關文件,我們可以在Github上獲取。包括一個JavaScript文件(rwd-images.js),一個.htaccess文件,以及一些范例資源文件。具體使用方法可以參考Responsive Images的說明文檔。大致的原理是,rwd-images.js會檢測當前設備的屏幕分辨率,如果是大屏幕設備,則向頁面head部分中添加BASE標記,并將后續的圖片、腳本和樣式表加載請求定向到一個虛擬路徑“/rwd-router”。當這些請求到達服務器端,.htacces文件會決定這些請求所需要的是原始圖片還是小尺寸的“響應式圖片”,并進行相應的反饋輸出。對于小屏幕的移動設備,原始尺寸的大圖片永遠不會被用到。
這項技術支持多數的現代瀏覽器,包括IE8+、Safari、Chrome和Opera,以及這些瀏覽器的移動設備版本;在FireFox及一些舊瀏覽器中,則會優雅降級:我們仍可得到小圖片的輸出,但同時,原始大圖也會被下載。
禁用iPhone中的圖片自動縮放
在iPhone及iPod Touch中,頁面會被自動的同比例縮小至最適合屏幕大小的尺寸,x軸不會產生滾動條,用戶可以上下拖拽瀏覽全部頁面,或在需要的時候放大頁面的局部。這里會產生一個問題,即使我們運用響應式Web設計的思想,專門為iPhone的小屏輸出小圖片,它同樣會隨著整個頁面一起被同比例縮小,如下圖左側所示。
我們可以使用蘋果專有的一些meta標記來解決類似的問題。在頁面的
部分添加以下代碼(詳情可以參考Think Vitamin的相關文章): 1 | |
將initial-scale的值設定為“1”,即可覆寫默認的縮放方式,保持原始的尺寸及比例。更多關于viewport meta標記的用法,可以參考蘋果官方的文檔。
響應式網站建設打造布局結構
當頁面所需要適應的不同設備的屏幕尺寸差異過大時,除了圖片方面,我們也應該考慮到整個布局結構的響應式調整;我們可以使用獨立的樣式表,或者更有效的,使用CSS media query。這不會引起多大的麻煩,多數樣式設定不會被影響和改變,只有一些特定的元素會通過浮動、寬度和高度等的設置來改變位置。
我們可以使用一個默認主樣式表來定義頁面的主要結構元素,比如#wrapper、#content、#sidebar、#nav等的默認布局方式,以及一些全局性的配色和字體方案。
我們可以監測頁面布局隨著不同的瀏覽環境而產生的變化,如果它們變的過窄過短或是過寬過長,則通過一個子級樣式表來繼承主樣式表的設定,并專門針對某些布局結構進行樣式覆寫。我們來看下代碼示例:
下面的代碼可以放在默認主樣式表style.css中:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | /* Default styles that will carry to the child style sheet */ ? html,body{ ???background... ???font... ???color... } ? h1,h2,h3{} p, blockquote,pre,code, ol, ul{} ? /* Structural elements */ #wrapper{ ????width:80%; ????margin:0auto; ? ????background:#fff; ????padding:20px; } ? #content{ ????width:54%; ????float:left; ????margin-right:3%; } ? #sidebar-left{ ????width:20%; ????float:left; ????margin-right:3%; } ? #sidebar-right{ ????width:20%; ????float:left; } |
下面的代碼可以放在子級樣式表mobile.css中,專門針對移動設備進行樣式覆寫:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | #wrapper{ ????width:90%; } ? #content{ ????width:100%; } ? #sidebar-left{ ????width:100%; ????clear:both; ? ????/* Additional styling for our new layout */ ????border-top:1pxsolid#ccc; ????margin-top:20px; } ? #sidebar-right{ ????width:100%; ????clear:both; ? ????/* Additional styling for our new layout */ ????border-top:1pxsolid#ccc; ????margin-top:20px; } |
大致的視覺效果如下圖所示:
Media Queries
CSS3支持CSS2.1所支持的所有媒體類型,例如screen、print、handheld等,同時添加了很多涉及媒體類型的功能屬性,包括max-width(最大寬度)、device-width(設備寬度)、orientation(屏幕定向,橫屏或豎屏)和color。在CSS3發布之后出現的新玩具,如iPad或Android相關設備,都可以完美的支持這些屬性。所以我們可以通過media query為新設備設置獨特的樣式,而忽略那些不支持CSS3的臺式機中的舊瀏覽器。
在Ethan的文章中,我們能看到一段media query使用實例:
1 2 3 | ????media="screen and (max-device-width: 480px)" ????href="shetland.css"/> |
代碼本身可以很好的說明工作機制:如果頁面通過屏幕呈現(非打印一類),并且屏幕寬度不超過480px,則加載shetland.css樣式表。要了解更多關于CSS3媒體新屬性的信息,可以參考“The Orientation Media Query”一文。
我們可以創建多個樣式表,以適應不同設備類型的寬度范圍。Ethan的文章中的“Meet the media query”部分有更多的范例及解釋。更有效率的做法是,將多個media queries整合在一個樣式表文件中:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | /* Smartphones (portrait and landscape) ----------- */ @media onlyscreen and (min-device-width :320px) and (max-device-width :480px) { /* Styles */ } ? /* Smartphones (landscape) ----------- */ @media onlyscreen and (min-width :321px) { /* Styles */ } ? /* Smartphones (portrait) ----------- */ @media onlyscreen and (max-width :320px) { /* Styles */ } |
上面的代碼來自于Andy Clark創建的可以兼容各種主流設備的免費模板。這樣整合多個media queries于一個樣式表文件的方式,與通過media queries調用不同樣式表的方式之間的區別與聯系,可以參考“Hardboiled CSS3 Media Queries”一文。
CSS3 Media Queries
上面演示的一些代碼范例是CSS2.1與CSS3通吃的。現在讓我們來看看怎樣使用CSS3專有的media queries功能來創建響應式Web設計。其中某些方法在當前就有切實的使用價值,不久的將來則一定會全部派上用場。
min-width和max-width這兩個屬性很靠譜。通過min-width的設置,我們可以在瀏覽器窗口或設備屏幕寬度高于這個值的情況下,為頁面指定一個特定的樣式表;max-width則反之。
下面的幾個示例中,我們使用的是將多個media queries整合在單一樣式表中進行編碼的句法。如前文所述,這樣做更加高效,減少請求數量。
1 2 3 4 5 6 | @mediascreenand (min-width:600px) { ?????.hereIsMyClass { ??????????width:30%; ??????????float:right; ?????} } |
上面代碼中定義的類(hereIsMyClass)只有在瀏覽器或屏幕寬度超過600px時才會有效。
1 2 3 4 5 6 | @mediascreenand (max-width:600px) { ?????.aClassforSmallScreens { ??????????clear:both; ??????????font-size:1.3em; ?????} } |
?????? 而這段代碼的作用則相反:aClassforSmallScreens類只有在瀏覽器或屏幕寬度小于600px時才會有效。
可以看出min-width和max-width可以同時判斷設備屏幕尺寸與瀏覽器實際寬度。有些時候,我們希望通過media queries作用于某種特定的設備,而忽略其上運行的瀏覽器是否由于沒有最大化而在尺寸上與設備屏幕尺寸產生不一致的情況。這時,我們需要使用min-device-width與max-device-width,用來判斷設備本身的屏幕尺寸。
1 2 3 4 5 | @mediascreenand (max-device-width:480px) { ?????.classForiPhoneDisplay { ??????????font-size:1.2em; ?????} } |
1 2 3 4 5 6 | @mediascreenand (min-device-width:768px) { ?????.minimumiPadWidth { ??????????clear:both; ??????????margin-bottom:2pxsolid#ccc; ?????} } |
還有一些其他方法,可以幫我們有效的使用media queries鎖定某些指定的設備。可以參考下面兩篇出自Thomas Maier的文章:
- CSS for iPhone 4 (Retina display)
- How To: CSS for the iPad
對于iPad來說,orientation屬性尤其有用。它的值可以是landscape(橫屏)或portrait(豎屏)。
1 2 3 4 5 6 | @mediascreenand (orientation:landscape) { ?????.iPadLandscape { ??????????width:30%; ??????????float:right; ?????} } |
1 2 3 4 5 | @mediascreenand (orientation:portrait) { ?????.iPadPortrait { ??????????clear:both; ?????} } |
不幸的是,這個屬性目前確實只在iPad上有效。對于其他可以轉屏的設備,譬如iPhone,可以使用min-device-width和max-device-width來變通實現;詳情可以參考“Determine iPhone orientation using CSS”一文。
我們還可以將上述屬性組合使用,來鎖定某個屏幕尺寸范圍:
1 2 3 4 5 6 7 | @mediascreenand (min-width:800px) and (max-width:1200px) { ?????.classForaMediumScreen { ??????????background:#cc0000; ??????????width:30%; ??????????float:right; ?????} } |
上面的代碼可以作用于瀏覽器窗口或屏幕寬度在800px至1200px之間的所有設備。
其實,很多設計師和開發者仍會選擇使用多個樣式表的方式來實現media queries。如果從資源的組織和維護的角度出發,其益處確實高于效率的損耗的話,那么這樣做也完全不壞:
1 2 3 | |
所以吶,凡事沒有絕對,最好根據實際情況決定使用media queries的方式。比如,對于iPad,我們可以將多個media queries直接寫在一個樣式表中。因為iPad用戶隨時有可能切換屏幕定向,這種情況下,要保證頁面在極短的時間內響應屏幕尺寸的調整,我們必須選擇效率最高的方式。
JavaScript
JavaScript也是我們的武器之一,特別是當某些舊設備無法完美支持CSS3的media query時,它可以作為后備支援。幸運的是,已經有專門的JS庫來幫助舊瀏覽器(IE 5+,Firefox 1+,Safari 2等)支持CSS3的media queries。使用方法很簡單,下載css3-mediaqueries.js并在你的頁面中調用它。
而下面的代碼則演示了怎樣使用簡單的幾行jQuery代碼來檢測瀏覽器寬度,并為不同的情況調用不同的樣式表:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | ? |
類似這樣的解決方案還有很多。所以我們要清楚,media queries不是一個絕對唯一的答案,它只是一個以純CSS方式實現響應式Web設計思路的手段。借助JavaScript,我們則可以實現更多的變化。這篇“Combining Media Queries and JavaScript”向我們展示了JavaScript配合media queries的更多細節信息。