更靠譜的橫豎屏檢測方法

来源:http://www.cnblogs.com/zhansingsong/archive/2016/09/12/5866692.html
-Advertisement-
Play Games

前不久,做了一個H5項目,需要在橫豎屏變化時,做一些處理。毫無疑問,需要使用orientationchange來監聽橫豎屏的變化。 方案一: 代碼添加上後,就各種相容性問題。這裡相容性問題出現在兩個地方: orientationchange event.orientation|screen.orie ...


前不久,做了一個H5項目,需要在橫豎屏變化時,做一些處理。毫無疑問,需要使用orientationchange來監聽橫豎屏的變化。

 方案一:

// 監聽 orientation changes
window.addEventListener("orientationchange", function(event) {
    // 根據event.orientation|screen.orientation.angle等於0|180、90|-90度來判斷橫豎屏
}, false);

代碼添加上後,就各種相容性問題。這裡相容性問題出現在兩個地方:

  • orientationchange
  • event.orientation|screen.orientation.angle

如下是orientationchange事件的相容性:

如下是screen.orientation的相容性:

 

 

 方案二:

上述方案不行,只能另行他法了。google一下,瞭解到可以通過resize配合(window.inner/outerWidth, window.inner/outerHeight)來實現:

window.addEventListener("resize", function(event) {
var orientation=(window.innerWidth > window.innerHeight)? "landscape":"portrait";
if(oritentation === 'portrait'){
// do something ……
} else {
// do something else ……
}
}, false);

這種方案基本滿足大部分項目的需求,但是還是有些不足之處:

  • 只要windowsize變化,就會不斷觸發觸發resize事件。可以使用setTimeout來優化一下
  • 如果有多個地方需要監聽橫豎屏,就需要註冊多個window.addEventListener("resize", function(event) {……})。能不能通過訂閱與發佈模式來改進一下,只註冊一個resize負責監聽橫豎屏變化,只要橫豎發生變化就發佈通知訂閱的對象。其他需要監聽橫豎屏的地方只需訂閱一下即可。

關鍵代碼如下:

    var resizeCB = function(){
      if(win.innerWidth > win.innerHeight){//初始化判斷
        meta.init = 'landscape';
        meta.current = 'landscape';
      } else {
        meta.init = 'portrait';
        meta.current = 'portrait';
      }
      return function(){
        if(win.innerWidth > win.innerHeight){
          if(meta.current !== 'landscape'){
            meta.current = 'landscape';
            event.trigger('__orientationChange__', meta);
          }
        } else {
          if(meta.current !== 'portrait'){
            meta.current = 'portrait';
            event.trigger('__orientationChange__', meta);
          }
        }
      }
    }();

完整代碼猛擊這裡

 

 方案三:

不過個人覺得通過window.innerWidth > window.innerHeight來實現的是一種偽檢測,有點不可靠。 可不可以通過瀏覽器來實現檢測?如基於CSS3@media媒體查詢來實現。

如下@media相容性:


如上上圖所示,移動端瀏覽器都支持CSS3 media。

實現思路:

  • 創建包含標識橫豎屏狀態的特定css樣式
  • 通過JS向頁面中註入CSS代碼
  • resize回調函數中獲取橫豎屏的狀態

這裡我選擇<html></html>的節點font-family作為檢測樣式屬性。理由如下:

  • 選擇<html></html>主要為了避免reflowrepaint
  • 選擇font-family樣式,主要是因為font-family有如下特性:
  • 優先使用排在前面的字體。
  • 如果找不到該種字體,或者該種字體不包括所要渲染的文字,則使用下一種字體。
  • 如果所列出的字體,都無法滿足需要,則讓操作系統自行決定使用哪種字體。

這樣我們就可以指定特定標識來標識橫豎屏的狀態,不過需要將指定的標識放置在其他字體的前面,這樣就不會引起hmtl字體的變化。

關鍵代碼如下:

    // callback
    var resizeCB = function() {
        var hstyle = win.getComputedStyle(html, null),
            ffstr = hstyle['font-family'],
            pstr = "portrait, " + ffstr,
            lstr = "landscape, " + ffstr,
            // 拼接css
            cssstr = '@media (orientation: portrait) { .orientation{font-family:' + pstr + ';} } @media (orientation: landscape) {  .orientation{font-family:' + lstr + ';}}';
        // 載入樣式        
        loadStyleString(cssstr);
        // 添加類
        html.className = 'orientation' + html.className;
        if (hstyle['font-family'] === pstr) { //初始化判斷
            meta.init = 'portrait';
            meta.current = 'portrait';
        } else {
            meta.init = 'landscape';
            meta.current = 'landscape';
        }
        return function() {
            if (hstyle['font-family'] === pstr) {
                if (meta.current !== 'portrait') {
                    meta.current = 'portrait';
                    event.trigger('__orientationChange__', meta);
                }
            } else {
                if (meta.current !== 'landscape') {
                    meta.current = 'landscape';
                    event.trigger('__orientationChange__', meta);
                }
            }
        }
    }();

完整代碼猛擊這裡


測試效果

  •  portrait效果:

  • landscape效果:

 

方案四:

可以再改進一下,在支持orientationchange時,就使用原生的orientationchange,不支持則使用方案三

關鍵代碼如下:

// 是否支持orientationchange事件
var isOrientation = ('orientation' in window && 'onorientationchange' in window);
// callback
var orientationCB = function(e) {
    if (win.orientation === 180 || win.orientation === 0) {
        meta.init = 'portrait';
        meta.current = 'portrait';
    }
    if (win.orientation === 90 || win.orientation === -90) {
        meta.init = 'landscape';
        meta.current = 'landscape';
    }
    return function() {
        if (win.orientation === 180 || win.orientation === 0) {
            meta.current = 'portrait';
        }
        if (win.orientation === 90 || win.orientation === -90) {
            meta.current = 'landscape';
        }
        event.trigger(eventType, meta);
    }
};
var callback = isOrientation ? orientationCB() : (function() {
    resizeCB();
    return function() {
        timer && win.clearTimeout(timer);
        timer = win.setTimeout(resizeCB, 300);
    }
})();
// 監聽
win.addEventListener(isOrientation ? eventType : 'resize', callback, false);

完整代碼猛擊這裡

 

方案五:

目前,上述幾種方案都是通過自定製的訂閱與發佈事件模式來實現的。這裡可以基於瀏覽器的事件機制,來模擬orientationchange。即對orientationchange的不相容進行修複。

關鍵代碼如下:

var eventType = 'orientationchange';
// 觸發原生orientationchange
var fire = function() {
    var e;
    if (document.createEvent) {
        e = document.createEvent('HTMLEvents');
        e.initEvent(eventType, true, false);
        win.dispatchEvent(e);
    } else {
        e = document.createEventObject();
        e.eventType = eventType;
        if (win[eventType]) {
            win[eventType]();
        } else if (win['on' + eventType]) {
            win['on' + eventType]();
        } else {
            win.fireEvent(eventType, e);
        }
    }
}

完整代碼猛擊這裡

通過上述5種方案,自己對移動端橫豎屏檢測有了更進一步的認識,有些東西只有自己親身經歷過才知道為什麼要這麼寫,自己也把其中緣由記錄在文章中,希望對大家有幫助。經過5種方案的演變得到了最終orientationchange-fix,github地址:https://github.com/zhansingsong/orientationchange-fix

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.singleShot的用法 代碼: QTextEdit *testEdit = new QTextEdit("hello world"); testEdit->setMaximumHeight(20); QTimer::singleShot( 5*1000, testEdit, SLOT(cle ...
  • 單例模式 定義:確保一個類只有一個實例,而且自行實例化並向整個系統提供這個實例。 類型:創建類模式 類圖: 單例模式應該是23種設計模式中最簡單的一種模式了。它有以下幾個要素: 私有的構造方法 指向自己實例的私有靜態引用 以自己實例為返回值的靜態的公有的方法 單例模式根據實例化對象時機的不同分為兩種 ...
  • 主要參考了 [IMOOC-SpringMVC 起步](http://www.imooc.com/video/7237) 視頻教程和 [SpringMVC從入門到精通 系列 - HansonQ](http://www.imooc.com/article/3804) ,還有自己的一些總結。MVC 簡介、... ...
  • SWFObject 2提供兩種優化flash播放器的嵌入方法:基於標記的方法和依賴於js的方法。 SWFObject 2提供一個js的API,為嵌入SWF文件和獲取Flash播放器的相關信息提供了一個完整的工具箱。 只用了一個很小的js文件 (10Kb / GZIPed: 3.9Kb)。 是SWFO ...
  • 一、簡介 該demo通過do_Painterview這個組件實現畫板的基本功能,模仿的是Appstore上的叫“白板”的應用,可以更改字體顏色,字體粗細,然後用手指進行繪製,可以回退,清屏,保存到相冊等操作。 二、效果圖 三、討論地址 http://bbs.deviceone.net/forum.p ...
  • 在手機端瀏覽網頁時,經常使用一個功能,當我們瀏覽京東或者淘寶時,頁面滑動到底部,我們看到數據自動載入到列表。之前並不知道這些功能是怎麼實現的,於是自己在PC瀏覽器上模擬實現這樣的功能。先看看瀏覽效果: 當滾動條滾動到頁面底部時,提示“正在載入…”。 當頁面已經載入了所有數據後,滾動到頁面底部會提示“ ...
  • 位元組操作 buffer對象 構造函數 new Buffer(size) 參數為數值,表示分配空間的長度 new Buffer(string,[encoding]) 參數為字元串,表示存入Buffer的數據,編碼可選 new Buffer(array) 參數為數組,表示存入Buffer的數據 靜態方法 ...
  • 我們首先定義一個json數組對象如下: 一. 根據對象屬性值得到相應對象 二. 刪除其中一個對象 三. 修改其中一個對象的屬性值 四. 往數組中添加一個對象 ——註: 以上的所有操作都會對原數組產生直接影響。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...