Android webview地理位置数据库无法打开

And*_*ský 12 sqlite android geolocation webview

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

<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"/>
Run Code Online (Sandbox Code Playgroud)

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

webview.getSettings().setDatabaseEnabled(true);
webview.getSettings().setDomStorageEnabled(true);
webview.getSettings().setGeolocationDatabasePath("/data/data/com.my.app/databases");
webview.getSettings().setGeolocationEnabled(true);
Run Code Online (Sandbox Code Playgroud)

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

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

地理位置本身正在运行,但我无法将地理位置缓存到数据库中.它应该自己进入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)
Run Code Online (Sandbox Code Playgroud)

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

我在设置中遗漏了什么可能导致数据库无法正确打开?我是Android新手,现在试图找到解决方案2天.任何帮助将不胜感激.谢谢.

Spi*_*iri 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)(无法看到该文件,因为我的手机未 root 并且 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"/>
Run Code Online (Sandbox Code Playgroud)

网页视图设置:

WebSettings webSettings = webview.getSettings();
webSettings.setJavaScriptEnabled(true);
webSettings.setDatabaseEnabled(true);
webSettings.setDomStorageEnabled(true);
webSettings.setGeolocationDatabasePath(getFilesDir().getPath());
webSettings.setGeolocationEnabled(true);
Run Code Online (Sandbox Code Playgroud)

Web chrome 客户端(相同):

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

测试 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>
Run Code Online (Sandbox Code Playgroud)

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

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

读取资源文件方法:

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();
}
Run Code Online (Sandbox Code Playgroud)

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