在移动浏览器或 PhoneGap 应用程序之间进行检测

Posted

技术标签:

【中文标题】在移动浏览器或 PhoneGap 应用程序之间进行检测【英文标题】:Detect between a mobile browser or a PhoneGap application 【发布时间】:2012-04-27 08:44:25 【问题描述】:

是否可以检测用户是否通过浏览器或使用 javascript 的应用程序进行访问?

我正在通过一个网页和一个 PhoneGap 应用程序为多个移动操作系统开发一个混合应用程序,目标是:

    独立于部署目标使用相同的代码 仅当用户代理是应用程序时才添加 PhoneGap.js 文件

【问题讨论】:

【参考方案1】:

您可以检查当前 URL 是否包含http 协议。

var app = document.URL.indexOf( 'http://' ) === -1 && document.URL.indexOf( 'https://' ) === -1;
if ( app ) 
    // PhoneGap application
 else 
    // Web page

【讨论】:

这样你就不需要等待PhoneGap/Cordova加载后就可以查看了。 我已经测试过并且工作正常!这正是我一直在寻找的解决方案类型,仅当它是应用程序时才导入 PhoneGap。 更短的检查:var fromBrowser = document.URL.match(/^https?:/) 在我的构建中,document.URL 是 http://local/,所以 http 协议测试仍然是积极的。我必须添加以稍微更改此代码以使其适合我... 如果我们在网络上托管科尔多瓦代码并将应用程序指向它,这也不起作用。【参考方案2】:

想到的快速解决方案是,

onDeviceReady

会帮助你的。由于此 JS 调用仅由本机桥(objC 或 Java)调用,因此 safari 移动浏览器将无法检测到这一点。因此,您的设备上应用(电话间隙)源库将从 onDeviceReady 开始。

如果 Phonegap 的任何 JS 调用(如 Device.platform 或 Device.name)为 NaN 或 null,那么它显然是移动网络调用。

请检查并告诉我结果。

【讨论】:

OK 在看到您对 Zvona 的回复后,现在我明白您甚至想决定是否加载或不加载 phonegap。让我再检查一遍 此解决方案的唯一问题是在部署网页时会执行额外请求以获取 phonegap 文件。 如果您真的可以对 Phonegap 源 (JS) 进行逆向工程。触发 onDeviceReady 本机调用的一段代码。这将使其重量相对较轻。您可以在设备上加载 Phonegap 的 JS 的其余部分。一个昂贵的测试操作。如果可能,请尝试。【参考方案3】:

我想出了一种方法来做到这一点,而不是依赖于 deviceready 事件,从而保持网络代码库的完整性......

当前使用内置 deviceready 事件的问题是,当页面加载时,您无法告诉应用:“嘿,这不是在移动设备上运行的,无需等待设备准备开始”。

1.- 在代码的本机部分,例如对于 ios,在 MainViewController.m 中有一个方法 viewDidLoad,我正在发送一个 javascript 变量,我稍后会在 Web 代码中检查该变量,如果该变量在附近,我将等待为我的页面启动代码,直到一切准备就绪(例如,导航器地理位置)

在 MainViewController.m 下:

- (void) viewDidLoad

    [super viewDidLoad];
    NSString* jsString = [NSString stringWithFormat:@"isAppNative = true;"];
    [self.webView stringByEvaluatingJavaScriptFromString:jsString];

2.- index.html 代码如下:

function onBodyLoad()

    document.addEventListener("deviceready", onDeviceReady, false);


function onDeviceReady();
    myApp.run();


try
    if(isAppNative!=undefined);
catch(err)
    $(document).ready(function()
        myApp.run();
    );

【讨论】:

我们还可以更进一步,将用户代理更改为 MyAppLoader XXHDPI。这将有助于加载更好的图像。很遗憾,cordova 没有这个功能。 这个解决方案对我来说非常有效,但请注意there are a couple small changes needed for Cordova 4。 if ([self.webView isKindOfClass:[UIWebView class]]) [(UIWebView*)self.webView stringByEvaluatingJavaScriptFromString:jsString]; 【参考方案4】:

PhoneGap 具有 window.PhoneGap(或在 Cordova 中为 window.cordova 或 window.Cordova)对象集。检查该对象是否存在并施展魔法。

【讨论】:

该解决方案的问题是我必须包含或不包含PhoneGap 文件,具体取决于我是否正在构建页面或应用程序。目标是独立于部署目标使用相同的代码。 那么这听起来有问题,因为我认为 UIWebView 包含相同的 Mobile Safari。让我看看我以后有没有发现。【参考方案5】:

在加载 phonegap 应用程序的 url 的本机调用中,添加一个参数 target,其值为 phonegap。所以对android的调用变成了这样。

super.loadUrl("file:///android_asset/www/index.html?target=phonegap");
使用此代码的网站不会使用额外参数调用,因此我们现在在两个部署平台之间有一些不同。 在 javascript 中,我们检查参数是否存在,如果存在,我们为 phonegap/cordova 添加脚本标签。 var urlVars = window.location.href.split('?'); if(urlVars.length > 1 && urlVars[1].search('target=phonegap') != -1) //phonegap 用于通话 $('head').append(''); 一个小警告:此方法需要为每个不同的目标移动平台更改 phonegap 中对 index.html 的调用。对于大多数平台,我不熟悉在哪里执行此操作。

【讨论】:

【参考方案6】:

我对 phonegap 应用和我们的网络客户端使用相同的代码。这是我用来检测 phonegap 是否可用的代码:

window.phonegap = false;
$.getScript("cordova-1.7.0.js", function()
    window.phonegap = true;
);

请记住,phonegap js 文件是异步加载的。您可以通过设置漂亮的 jquery $.getScript 函数的正确选项来同步加载它。

请注意,即使在您的网络客户端中,该方法也会发出额外的 GET 请求来获取 phonegap js 文件。就我而言,它并没有影响我的网络客户端的性能;所以它最终成为了一种很好/干净的方法。至少在其他人找到快速的单线解决方案之前:)

【讨论】:

并非所有 phonegap 应用程序都是用 cordova 构建的。检测 phonegap.js 可能会更好。【参考方案7】:

在 Phonegap 应用程序中启动 web 视图后,您似乎正在加载另一个网页,对吗?如果是这样,那么您可以根据配置向请求 url 添加一个参数。

例如,假设 php

App.Config = 
  target: "phonegap"
;

<body onload="onbodyload()">

var onbodyload = function () 
  var target = App.Config.target;
  document.location = "/home?target=" + target;
;

然后在服务器端,如果目标是phonegap,则包含phonegap js。

无法使用用户代理检测差异。

【讨论】:

【参考方案8】:

我这样做的方式是使用一个全局变量,该变量被仅浏览器版本的 cordova.js 覆盖。在您的主 html 文件(通常是 index.html)中,我有以下依赖于顺序的脚本:

    <script>
        var __cordovaRunningOnBrowser__ = false
    </script>
    <script src="cordova.js"></script> <!-- must be included after __cordovaRunningOnBrowser__ is initialized -->
    <script src="index.js"></script> <!-- must be included after cordova.js so that __cordovaRunningOnBrowser__ is set correctly -->

cordova.js 里面我有:

__cordovaRunningOnBrowser__ = true

在为移动设备构建时,不会使用cordova.js(而是使用特定于平台的cordova.js 文件),因此这种方法的好处是无论协议、userAgents 怎样都100% 正确,或库变量(可能会改变)。我可能还应该在 cordova.js 中包含其他内容,但我还不知道它们是什么。

【讨论】:

我很高兴我一直滚动到底部 - 在此处列出的方法中,这是我最喜欢的。【参考方案9】:

我也一直在努力解决这个问题,我知道这是一个旧线程,但我还没有在任何地方看到我的方法,所以认为 id 分享以防它对某人有所帮助。

我在实际用户代理之后设置了一个自定义用户代理:

String useragent = settings.getUserAgentString();
settings.setUserAgentString(useragent + ";phonegap");

这只是添加了 phonegap 字符串,因此其他依赖检测您的移动用户代理的网站仍然可以工作。

然后你可以像这样加载phonegap:

if( /phonegap/i.test(navigator.userAgent) ) 

//you are on a phonegap app, $getScript etc
 else 
alert("not phonegap");

【讨论】:

【参考方案10】:

如果你尝试以下方法:

if(window._cordovaNative) 
  alert("loading cordova");
  requirejs(["...path/to/cordova.js"], function ()  
         alert("Finished loading cordova");
  );

【讨论】:

检查window._cordovaNative 为我工作!至少在安卓上。在 cordova.js 的其他地方有一条评论暗示它不适用于 Android 【参考方案11】:

在我看来,你试图为自己制造问题。您没有提到您的开发平台,但它们中的大多数都有不同的部署配置。您可以定义两种配置。并设置指示代码部署方式的变量。 在这种情况下,您无需关心部署应用的设备。

【讨论】:

【参考方案12】:

短而有效:

if (document.location.protocol == 'file:')  //Phonegap is present 

【讨论】:

【参考方案13】:

类似于B T's solution,但更简单:

我的 www 文件夹中有一个空的 cordova.js,它在构建时会被 Cordova 覆盖。不要忘记在您的应用脚本文件之前包含cordova.js(我花了一个小时才发现它们的顺序错误......)。

然后您可以检查 Cordova 对象:

document.addEventListener('DOMContentLoaded', function()
    if (window.Cordova) 
        document.addEventListener('DeviceReady', bootstrap);
     else 
        bootstrap();
    
);

function bootstrap() 
   do_something()

【讨论】:

【参考方案14】:

新解决方案:

var isPhoneGapWebView = location.href.match(/^file:/); // returns true for PhoneGap app

旧解决方案: 使用jQuery,这样运行

$(document).ready(function()
   alert(window.innerHeight);
);

以 iPhone 作为您的移动应用程序的示例,

使用 PhoneGap 或 Cordova 时,您将获得 460px 的 WebView,但在 safari 中,由于浏览器的默认页眉和页脚,您会失去一些高度。

如果window.innerHeight等于460,可以加载phonegap.js,调用onDeviceReady函数

【讨论】:

这似乎是一个很好的解决方案,我怎样才能使其通用?我正在为多个移动设备进行部署。 您可以对所有设备使用导航器对象来设置条件 我认为您可以使用 navigator.userAgent 检测浏览器,请在此处查看答案***.com/questions/3514784/… 这将是一个非常脆弱的解决方案【参考方案15】:

还没有人提到这一点,但 Cordova 现在似乎支持将浏览器添加为平台:

cordova platforms add browser

这将在运行时自动添加cordova.js,其中包含onDeviceReady 事件,因此您无需伪造它。此外,许多插件都支持浏览器,因此您的代码中不再有浏览器黑客攻击。

要在浏览器中使用您的应用程序,您应该使用cordova run browser。如果您想部署它,您可以使用与其他平台相同的命令进行部署。

编辑:忘记提及my source。

【讨论】:

【参考方案16】:

解决方案:在 Cordova 中修补 index.html 并将 cordova-platform="android" 添加到 &lt;html&gt; 标记,这样 cordova-platform 属性将仅存在于 Cordova 构建中,并且在用于 Cordova 之外的 Web 的原始 index.html 中缺失。

优点:不依赖于用户代理、url 架构或 cordova API。不需要等待 deviceready 事件。可以通过各种方式进行扩展,例如可以包含或不包含cordova-platform="browser",以便区分Cordova 之外的Web 应用程序与Cordova 的浏览器平台构建。

与 config.xml 合并

    <platform name="android">
        <hook src="scripts/patch-android-index.js" type="after_prepare" />
    </platform>

添加文件脚本/patch-android-index.js

module.exports = function(ctx) 
    var fs = ctx.requireCordovaModule('fs');
    var path = ctx.requireCordovaModule('path');

    var platformRoot = path.join(ctx.opts.projectRoot, 'platforms/android');
    var indexPath = platformRoot + '/app/src/main/assets/www/index.html';

    var indexSource = fs.readFileSync(indexPath, 'utf-8');

    indexSource = indexSource.replace('<html', '<html cordova-platform="android"');

    fs.writeFileSync(indexPath, indexSource, 'utf-8');

注意:对于非android,路径platforms/android/app/src/main/assets/www/index.html需要调整。

应用可以通过

检查cordova-platform
if (! document.documentElement.getAttribute('cordova-platform')) 
  // Not in Cordova

if (document.documentElement.getAttribute('cordova-platform') === 'android') 
  // Cordova, Android

【讨论】:

以上是关于在移动浏览器或 PhoneGap 应用程序之间进行检测的主要内容,如果未能解决你的问题,请参考以下文章

phonegap 和 intel xdk 或其他之间的区别[关闭]

Angular Animations 和 Phonegap 不在移动设备上运行

如何在phonegap子浏览器与主应用程序之间发送和接收数据

在 phonegap 应用程序中使用 iframe

Ajax 调用如何在移动应用程序 (PhoneGap) 中工作,但在 PC 浏览器中却没有?

本地网络上的 Mobile Phonegap 应用程序