12

我有使用 javascript geolocation api 的网站,并希望它在 web 视图中打开。我在清单文件中设置了这些权限:

<uses-permission android:name="android.permission.ACCESS_GPS" />
<uses-permission android:name="android.permission.ACCESS_ASSISTED_GPS" />
<uses-permission android:name="android.permission.ACCESS_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>

在我的活动中,我还设置了 webview 设置:

webview.getSettings().setDatabaseEnabled(true);
webview.getSettings().setDomStorageEnabled(true);
webview.getSettings().setGeolocationDatabasePath("/data/data/com.my.app/databases");
webview.getSettings().setGeolocationEnabled(true);

我还处理了 javascript 地理定位对话框:

webview.setWebChromeClient(new WebChromeClient(){
    @Override
    public void onGeolocationPermissionsShowPrompt(String origin, GeolocationPermissions.Callback callback) {
        callback.invoke(origin, true, false);
    }
});

地理位置本身正在工作,但我无法将地理位置缓存到数据库中。它应该在 CachedGeoposition.db 中自行完成,但是当我启动 webview 时,我得到了这个奇怪的 SQLite 错误:

E/SQLiteLog(22653): (14) cannot open file at line 30174 of [00bb9c9ce4]
E/SQLiteLog(22653): (14) os_unix.c:30174: (2) open(/CachedGeoposition.db) -
D/WebKit  (22653): ERROR:
D/WebKit  (22653): SQLite database failed to load from /CachedGeoposition.db
D/WebKit  (22653): Cause - unable to open database file
D/WebKit  (22653):
D/WebKit  (22653): external/webkit/Source/WebCore/platform/sql/SQLiteDatabase.cp
p(71) : bool WebCore::SQLiteDatabase::open(const WTF::String&, bool)

当我在文件资源管理器中检查 CachedGeoposition.db 文件是否存在时,它始终存在,并且数据库的权限设置为 -rw-------。

我是否在设置中遗漏了一些可能导致数据库无法正确打开的内容?我是 Android 新手,现在正在尝试寻找 2 天的解决方案。任何帮助将不胜感激。谢谢你。

4

3 回答 3

2

我将发布一个对我有用的解决方案,希望它可以提供一些关于错误可能在哪里的想法。如果您可以提供有关您正在测试的设备(模拟器)/操作系统的详细信息,这也会有所帮助。

我测试过(没有错误):

  • (模拟器) Galaxy Nexus (2.3.3) (data/data/package.name/files/CachedGeoposition.db 文件创建)
  • (模拟器) Galaxy Nexus (3.0) (data/data/package.name/files/CachedGeoposition.db 文件创建)
  • HTC One S (4.1.1) (data/data/package.name/files/CachedGeoposition.db 文件创建)
  • Galaxy S3 (4.3)(看不到文件,因为我的手机没有植根,并且 4.3 上存在一些“运行方式”错误)
  • Nexus 7 (4.4.4) - KIT KAT 上面的 webview 发生了一些变化,文件不再存在,但没有显示错误(即使提供了错误的“数据库”文件夹路径)
  • (模拟器)Galaxy S4(5.0)(与 4.4.4 相同)

AndroidManifest 权限(几乎相同):

<uses-permission android:name="android.permission.ACCESS_GPS" />
<uses-permission android:name="android.permission.ACCESS_ASSISTED_GPS" />
<uses-permission android:name="android.permission.ACCESS_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>

网页视图设置:

WebSettings webSettings = webview.getSettings();
webSettings.setJavaScriptEnabled(true);
webSettings.setDatabaseEnabled(true);
webSettings.setDomStorageEnabled(true);
webSettings.setGeolocationDatabasePath(getFilesDir().getPath());
webSettings.setGeolocationEnabled(true);

Web chrome客户端(同):

webview.setWebChromeClient(new WebChromeClient(){
    @Override
    public void onGeolocationPermissionsShowPrompt(String origin, GeolocationPermissions.Callback callback) {
        callback.invoke(origin, true, false);
    }
});

测试用于获取地理位置的 html 文件:

<!DOCTYPE html>
<html>
<head>
<script>
var x = document.getElementById("demo");

function getLocation() {
    if (navigator.geolocation) {
        navigator.geolocation.getCurrentPosition(showPosition, showError);
    } else {
        x.innerHTML = "Geolocation is not supported by this browser.";
    }
}

function showPosition(position) {
    var latlon = position.coords.latitude + "," + position.coords.longitude;
    document.getElementById("locationHolder").innerHTML = latlon;
}

function showError(error) {
    switch(error.code) {
        case error.PERMISSION_DENIED:
            x.innerHTML = "User denied the request for Geolocation.";
            break;
        case error.POSITION_UNAVAILABLE:
            x.innerHTML = "Location information is unavailable";
            break;
        case error.TIMEOUT:
            x.innerHTML = "The request to get user location timed out.";
            break;
        case error.UNKNOWN_ERROR:
            x.innerHTML = "An unknown error occurred.";
            break;
    }
}
</script>
</head>

<body>

<p id="demo">Click the button to get your position.</p>

<div id="locationHolder">No location</div>
<button onclick="getLocation()">Get position</button>

</body>
</html>

此外,对于本地测试,您可以在“/assets/www/index.html”下创建一个包含 html 的文件,并使用以下代码将其加载到 webview 中:

try {
    String html = readAssetFile("www/index.html");
    webview.loadDataWithBaseURL(null, html, "text/html", "UTF-8", null);
} catch (IOException e) {
}

读取资产文件方法:

private String readAssetFile(String filePath) throws IOException {
    StringBuilder buffer = new StringBuilder();
    InputStream fileInputStream = getAssets().open(filePath);
    BufferedReader bufferReader = new BufferedReader(new InputStreamReader(fileInputStream, "UTF-8"));
    String str;

    while ((str=bufferReader.readLine()) != null) {
        buffer.append(str);
    }
    fileInputStream.close();

    return buffer.toString();
}

如果不提供“数据库”文件夹的错误硬编码路径,我将无法重现您的错误。

于 2015-02-26T13:45:51.780 回答
0

线

D/WebKit (22653):SQLite 数据库无法从 /CachedGeoposition.db 加载

很有趣

尝试在 WebSettings.setDatabasePath 中设置适当的数据库路径,这可能是用作地理数据库路径的根

于 2012-11-01T16:15:49.497 回答
0

我知道这是一个老问题,但任何有这个问题的人可能会发现关注这个功能很有趣:

mWebView.getSettings().setGeolocationDatabasePath(getFilesDir().getPath());

地理位置使用数据库在会话之间保留缓存的位置和权限。此调用设置数据库的位置。

与其将固定路径设置为参数,不如动态调用它:

getFilesDir().getPath()
于 2014-08-22T09:21:15.153 回答