@ -1,83 +0,0 @@ |
|||||
<!-- |
|
||||
# |
|
||||
# Licensed to the Apache Software Foundation (ASF) under one |
|
||||
# or more contributor license agreements. See the NOTICE file |
|
||||
# distributed with this work for additional information |
|
||||
# regarding copyright ownership. The ASF licenses this file |
|
||||
# to you under the Apache License, Version 2.0 (the |
|
||||
# "License"); you may not use this file except in compliance |
|
||||
# with the License. You may obtain a copy of the License at |
|
||||
# |
|
||||
# http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
# |
|
||||
# Unless required by applicable law or agreed to in writing, |
|
||||
# software distributed under the License is distributed on an |
|
||||
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
# KIND, either express or implied. See the License for the |
|
||||
# specific language governing permissions and limitations |
|
||||
# under the License. |
|
||||
# |
|
||||
--> |
|
||||
# Cordova Hooks |
|
||||
|
|
||||
This directory may contain scripts used to customize cordova commands. This |
|
||||
directory used to exist at `.cordova/hooks`, but has now been moved to the |
|
||||
project root. Any scripts you add to these directories will be executed before |
|
||||
and after the commands corresponding to the directory name. Useful for |
|
||||
integrating your own build systems or integrating with version control systems. |
|
||||
|
|
||||
__Remember__: Make your scripts executable. |
|
||||
|
|
||||
## Hook Directories |
|
||||
The following subdirectories will be used for hooks: |
|
||||
|
|
||||
after_build/ |
|
||||
after_compile/ |
|
||||
after_docs/ |
|
||||
after_emulate/ |
|
||||
after_platform_add/ |
|
||||
after_platform_rm/ |
|
||||
after_platform_ls/ |
|
||||
after_plugin_add/ |
|
||||
after_plugin_ls/ |
|
||||
after_plugin_rm/ |
|
||||
after_plugin_search/ |
|
||||
after_prepare/ |
|
||||
after_run/ |
|
||||
after_serve/ |
|
||||
before_build/ |
|
||||
before_compile/ |
|
||||
before_docs/ |
|
||||
before_emulate/ |
|
||||
before_platform_add/ |
|
||||
before_platform_rm/ |
|
||||
before_platform_ls/ |
|
||||
before_plugin_add/ |
|
||||
before_plugin_ls/ |
|
||||
before_plugin_rm/ |
|
||||
before_plugin_search/ |
|
||||
before_prepare/ |
|
||||
before_run/ |
|
||||
before_serve/ |
|
||||
pre_package/ <-- Windows 8 and Windows Phone only. |
|
||||
|
|
||||
## Script Interface |
|
||||
|
|
||||
All scripts are run from the project's root directory and have the root directory passes as the first argument. All other options are passed to the script using environment variables: |
|
||||
|
|
||||
* CORDOVA_VERSION - The version of the Cordova-CLI. |
|
||||
* CORDOVA_PLATFORMS - Comma separated list of platforms that the command applies to (e.g.: android, ios). |
|
||||
* CORDOVA_PLUGINS - Comma separated list of plugin IDs that the command applies to (e.g.: org.apache.cordova.file, org.apache.cordova.file-transfer) |
|
||||
* CORDOVA_HOOK - Path to the hook that is being executed. |
|
||||
* CORDOVA_CMDLINE - The exact command-line arguments passed to cordova (e.g.: cordova run ios --emulate) |
|
||||
|
|
||||
If a script returns a non-zero exit code, then the parent cordova command will be aborted. |
|
||||
|
|
||||
|
|
||||
## Writing hooks |
|
||||
|
|
||||
We highly recommend writting your hooks using Node.js so that they are |
|
||||
cross-platform. Some good examples are shown here: |
|
||||
|
|
||||
[http://devgirl.org/2013/11/12/three-hooks-your-cordovaphonegap-project-needs/](http://devgirl.org/2013/11/12/three-hooks-your-cordovaphonegap-project-needs/) |
|
||||
|
|
@ -1,94 +0,0 @@ |
|||||
#!/usr/bin/env node
|
|
||||
|
|
||||
// Add Platform Class
|
|
||||
// v1.0
|
|
||||
// Automatically adds the platform class to the body tag
|
|
||||
// after the `prepare` command. By placing the platform CSS classes
|
|
||||
// directly in the HTML built for the platform, it speeds up
|
|
||||
// rendering the correct layout/style for the specific platform
|
|
||||
// instead of waiting for the JS to figure out the correct classes.
|
|
||||
|
|
||||
var fs = require('fs'); |
|
||||
var path = require('path'); |
|
||||
|
|
||||
var rootdir = process.argv[2]; |
|
||||
|
|
||||
function addPlatformBodyTag(indexPath, platform) { |
|
||||
// add the platform class to the body tag
|
|
||||
try { |
|
||||
var platformClass = 'platform-' + platform; |
|
||||
var cordovaClass = 'platform-cordova platform-webview'; |
|
||||
|
|
||||
var html = fs.readFileSync(indexPath, 'utf8'); |
|
||||
|
|
||||
var bodyTag = findBodyTag(html); |
|
||||
if(!bodyTag) return; // no opening body tag, something's wrong
|
|
||||
|
|
||||
if(bodyTag.indexOf(platformClass) > -1) return; // already added
|
|
||||
|
|
||||
var newBodyTag = bodyTag; |
|
||||
|
|
||||
var classAttr = findClassAttr(bodyTag); |
|
||||
if(classAttr) { |
|
||||
// body tag has existing class attribute, add the classname
|
|
||||
var endingQuote = classAttr.substring(classAttr.length-1); |
|
||||
var newClassAttr = classAttr.substring(0, classAttr.length-1); |
|
||||
newClassAttr += ' ' + platformClass + ' ' + cordovaClass + endingQuote; |
|
||||
newBodyTag = bodyTag.replace(classAttr, newClassAttr); |
|
||||
|
|
||||
} else { |
|
||||
// add class attribute to the body tag
|
|
||||
newBodyTag = bodyTag.replace('>', ' class="' + platformClass + ' ' + cordovaClass + '">'); |
|
||||
} |
|
||||
|
|
||||
html = html.replace(bodyTag, newBodyTag); |
|
||||
|
|
||||
fs.writeFileSync(indexPath, html, 'utf8'); |
|
||||
|
|
||||
process.stdout.write('add to body class: ' + platformClass + '\n'); |
|
||||
} catch(e) { |
|
||||
process.stdout.write(e); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
function findBodyTag(html) { |
|
||||
// get the body tag
|
|
||||
try{ |
|
||||
return html.match(/<body(?=[\s>])(.*?)>/gi)[0]; |
|
||||
}catch(e){} |
|
||||
} |
|
||||
|
|
||||
function findClassAttr(bodyTag) { |
|
||||
// get the body tag's class attribute
|
|
||||
try{ |
|
||||
return bodyTag.match(/ class=["|'](.*?)["|']/gi)[0]; |
|
||||
}catch(e){} |
|
||||
} |
|
||||
|
|
||||
if (rootdir) { |
|
||||
|
|
||||
// go through each of the platform directories that have been prepared
|
|
||||
var platforms = (process.env.CORDOVA_PLATFORMS ? process.env.CORDOVA_PLATFORMS.split(',') : []); |
|
||||
|
|
||||
for(var x=0; x<platforms.length; x++) { |
|
||||
// open up the index.html file at the www root
|
|
||||
try { |
|
||||
var platform = platforms[x].trim().toLowerCase(); |
|
||||
var indexPath; |
|
||||
|
|
||||
if(platform == 'android') { |
|
||||
indexPath = path.join('platforms', platform, 'assets', 'www', 'index.html'); |
|
||||
} else { |
|
||||
indexPath = path.join('platforms', platform, 'www', 'index.html'); |
|
||||
} |
|
||||
|
|
||||
if(fs.existsSync(indexPath)) { |
|
||||
addPlatformBodyTag(indexPath, platform); |
|
||||
} |
|
||||
|
|
||||
} catch(e) { |
|
||||
process.stdout.write(e); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
} |
|
@ -1,13 +0,0 @@ |
|||||
<!DOCTYPE html> |
|
||||
<html lang="en"> |
|
||||
<head> |
|
||||
<meta charset="utf-8"> |
|
||||
<meta http-equiv="refresh" content="0; url=www/index.html" /> |
|
||||
<title>title</title> |
|
||||
<link rel="stylesheet" href="style.css"> |
|
||||
<script src="script.js"></script> |
|
||||
</head> |
|
||||
<body> |
|
||||
<!-- page content --> |
|
||||
</body> |
|
||||
</html> |
|
@ -1,14 +0,0 @@ |
|||||
# Non-project-specific build files: |
|
||||
build.xml |
|
||||
local.properties |
|
||||
/gradlew |
|
||||
/gradlew.bat |
|
||||
/gradle |
|
||||
# Ant builds |
|
||||
ant-build |
|
||||
ant-gen |
|
||||
# Eclipse builds |
|
||||
gen |
|
||||
out |
|
||||
# Gradle builds |
|
||||
/build |
|
@ -1,14 +0,0 @@ |
|||||
<?xml version='1.0' encoding='utf-8'?> |
|
||||
<manifest android:hardwareAccelerated="true" android:versionCode="1" android:versionName="0.0.1" package="com.ionicframework.app751563" xmlns:android="http://schemas.android.com/apk/res/android"> |
|
||||
<supports-screens android:anyDensity="true" android:largeScreens="true" android:normalScreens="true" android:resizeable="true" android:smallScreens="true" android:xlargeScreens="true" /> |
|
||||
<uses-permission android:name="android.permission.INTERNET" /> |
|
||||
<application android:hardwareAccelerated="true" android:icon="@drawable/icon" android:label="@string/app_name" android:supportsRtl="true"> |
|
||||
<activity android:configChanges="orientation|keyboardHidden|keyboard|screenSize|locale" android:label="@string/activity_name" android:launchMode="singleTop" android:name="MainActivity" android:theme="@android:style/Theme.DeviceDefault.NoActionBar" android:windowSoftInputMode="adjustResize"> |
|
||||
<intent-filter android:label="@string/launcher_name"> |
|
||||
<action android:name="android.intent.action.MAIN" /> |
|
||||
<category android:name="android.intent.category.LAUNCHER" /> |
|
||||
</intent-filter> |
|
||||
</activity> |
|
||||
</application> |
|
||||
<uses-sdk android:minSdkVersion="16" android:targetSdkVersion="23" /> |
|
||||
</manifest> |
|
@ -1,23 +0,0 @@ |
|||||
<?xml version="1.0" encoding="utf-8"?> |
|
||||
<!-- |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
--> |
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android" |
|
||||
package="org.apache.cordova" android:versionName="1.0" android:versionCode="1"> |
|
||||
<uses-sdk android:minSdkVersion="14" /> |
|
||||
</manifest> |
|
@ -1,61 +0,0 @@ |
|||||
/* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
|
|
||||
|
|
||||
buildscript { |
|
||||
repositories { |
|
||||
mavenCentral() |
|
||||
} |
|
||||
|
|
||||
dependencies { |
|
||||
classpath 'com.android.tools.build:gradle:2.1.0' |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
|
|
||||
apply plugin: 'com.android.library' |
|
||||
|
|
||||
ext { |
|
||||
apply from: 'cordova.gradle' |
|
||||
cdvCompileSdkVersion = privateHelpers.getProjectTarget() |
|
||||
cdvBuildToolsVersion = privateHelpers.findLatestInstalledBuildTools() |
|
||||
} |
|
||||
|
|
||||
android { |
|
||||
compileSdkVersion cdvCompileSdkVersion |
|
||||
buildToolsVersion cdvBuildToolsVersion |
|
||||
publishNonDefault true |
|
||||
|
|
||||
compileOptions { |
|
||||
sourceCompatibility JavaVersion.VERSION_1_6 |
|
||||
targetCompatibility JavaVersion.VERSION_1_6 |
|
||||
} |
|
||||
|
|
||||
sourceSets { |
|
||||
main { |
|
||||
manifest.srcFile 'AndroidManifest.xml' |
|
||||
java.srcDirs = ['src'] |
|
||||
resources.srcDirs = ['src'] |
|
||||
aidl.srcDirs = ['src'] |
|
||||
renderscript.srcDirs = ['src'] |
|
||||
res.srcDirs = ['res'] |
|
||||
assets.srcDirs = ['assets'] |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,201 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
import java.util.regex.Pattern |
|
||||
import groovy.swing.SwingBuilder |
|
||||
|
|
||||
String doEnsureValueExists(filePath, props, key) { |
|
||||
if (props.get(key) == null) { |
|
||||
throw new GradleException(filePath + ': Missing key required "' + key + '"') |
|
||||
} |
|
||||
return props.get(key) |
|
||||
} |
|
||||
|
|
||||
String doGetProjectTarget() { |
|
||||
def props = new Properties() |
|
||||
file('project.properties').withReader { reader -> |
|
||||
props.load(reader) |
|
||||
} |
|
||||
return doEnsureValueExists('project.properties', props, 'target') |
|
||||
} |
|
||||
|
|
||||
String[] getAvailableBuildTools() { |
|
||||
def buildToolsDir = new File(getAndroidSdkDir(), "build-tools") |
|
||||
buildToolsDir.list() |
|
||||
.findAll { it ==~ /[0-9.]+/ } |
|
||||
.sort { a, b -> compareVersions(b, a) } |
|
||||
} |
|
||||
|
|
||||
String doFindLatestInstalledBuildTools(String minBuildToolsVersion) { |
|
||||
def availableBuildToolsVersions |
|
||||
try { |
|
||||
availableBuildToolsVersions = getAvailableBuildTools() |
|
||||
} catch (e) { |
|
||||
println "An exception occurred while trying to find the Android build tools." |
|
||||
throw e |
|
||||
} |
|
||||
if (availableBuildToolsVersions.length > 0) { |
|
||||
def highestBuildToolsVersion = availableBuildToolsVersions[0] |
|
||||
if (compareVersions(highestBuildToolsVersion, minBuildToolsVersion) < 0) { |
|
||||
throw new RuntimeException( |
|
||||
"No usable Android build tools found. Highest installed version is " + |
|
||||
highestBuildToolsVersion + "; minimum version required is " + |
|
||||
minBuildToolsVersion + ".") |
|
||||
} |
|
||||
highestBuildToolsVersion |
|
||||
} else { |
|
||||
throw new RuntimeException( |
|
||||
"No installed build tools found. Install the Android build tools version " + |
|
||||
minBuildToolsVersion + " or higher.") |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// Return the first non-zero result of subtracting version list elements |
|
||||
// pairwise. If they are all identical, return the difference in length of |
|
||||
// the two lists. |
|
||||
int compareVersionList(Collection aParts, Collection bParts) { |
|
||||
def pairs = ([aParts, bParts]).transpose() |
|
||||
pairs.findResult(aParts.size()-bParts.size()) {it[0] - it[1] != 0 ? it[0] - it[1] : null} |
|
||||
} |
|
||||
|
|
||||
// Compare two version strings, such as "19.0.0" and "18.1.1.0". If all matched |
|
||||
// elements are identical, the longer version is the largest by this method. |
|
||||
// Examples: |
|
||||
// "19.0.0" > "19" |
|
||||
// "19.0.1" > "19.0.0" |
|
||||
// "19.1.0" > "19.0.1" |
|
||||
// "19" > "18.999.999" |
|
||||
int compareVersions(String a, String b) { |
|
||||
def aParts = a.tokenize('.').collect {it.toInteger()} |
|
||||
def bParts = b.tokenize('.').collect {it.toInteger()} |
|
||||
compareVersionList(aParts, bParts) |
|
||||
} |
|
||||
|
|
||||
String getAndroidSdkDir() { |
|
||||
def rootDir = project.rootDir |
|
||||
def androidSdkDir = null |
|
||||
String envVar = System.getenv("ANDROID_HOME") |
|
||||
def localProperties = new File(rootDir, 'local.properties') |
|
||||
String systemProperty = System.getProperty("android.home") |
|
||||
if (envVar != null) { |
|
||||
androidSdkDir = envVar |
|
||||
} else if (localProperties.exists()) { |
|
||||
Properties properties = new Properties() |
|
||||
localProperties.withInputStream { instr -> |
|
||||
properties.load(instr) |
|
||||
} |
|
||||
def sdkDirProp = properties.getProperty('sdk.dir') |
|
||||
if (sdkDirProp != null) { |
|
||||
androidSdkDir = sdkDirProp |
|
||||
} else { |
|
||||
sdkDirProp = properties.getProperty('android.dir') |
|
||||
if (sdkDirProp != null) { |
|
||||
androidSdkDir = (new File(rootDir, sdkDirProp)).getAbsolutePath() |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
if (androidSdkDir == null && systemProperty != null) { |
|
||||
androidSdkDir = systemProperty |
|
||||
} |
|
||||
if (androidSdkDir == null) { |
|
||||
throw new RuntimeException( |
|
||||
"Unable to determine Android SDK directory.") |
|
||||
} |
|
||||
androidSdkDir |
|
||||
} |
|
||||
|
|
||||
def doExtractIntFromManifest(name) { |
|
||||
def manifestFile = file(android.sourceSets.main.manifest.srcFile) |
|
||||
def pattern = Pattern.compile(name + "=\"(\\d+)\"") |
|
||||
def matcher = pattern.matcher(manifestFile.getText()) |
|
||||
matcher.find() |
|
||||
return Integer.parseInt(matcher.group(1)) |
|
||||
} |
|
||||
|
|
||||
def doExtractStringFromManifest(name) { |
|
||||
def manifestFile = file(android.sourceSets.main.manifest.srcFile) |
|
||||
def pattern = Pattern.compile(name + "=\"(\\S+)\"") |
|
||||
def matcher = pattern.matcher(manifestFile.getText()) |
|
||||
matcher.find() |
|
||||
return matcher.group(1) |
|
||||
} |
|
||||
|
|
||||
def doPromptForPassword(msg) { |
|
||||
if (System.console() == null) { |
|
||||
def ret = null |
|
||||
new SwingBuilder().edt { |
|
||||
dialog(modal: true, title: 'Enter password', alwaysOnTop: true, resizable: false, locationRelativeTo: null, pack: true, show: true) { |
|
||||
vbox { |
|
||||
label(text: msg) |
|
||||
def input = passwordField() |
|
||||
button(defaultButton: true, text: 'OK', actionPerformed: { |
|
||||
ret = input.password; |
|
||||
dispose(); |
|
||||
}) |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
if (!ret) { |
|
||||
throw new GradleException('User canceled build') |
|
||||
} |
|
||||
return new String(ret) |
|
||||
} else { |
|
||||
return System.console().readPassword('\n' + msg); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
def doGetConfigXml() { |
|
||||
def xml = file("res/xml/config.xml").getText() |
|
||||
// Disable namespace awareness since Cordova doesn't use them properly |
|
||||
return new XmlParser(false, false).parseText(xml) |
|
||||
} |
|
||||
|
|
||||
def doGetConfigPreference(name, defaultValue) { |
|
||||
name = name.toLowerCase() |
|
||||
def root = doGetConfigXml() |
|
||||
|
|
||||
def ret = defaultValue |
|
||||
root.preference.each { it -> |
|
||||
def attrName = it.attribute("name") |
|
||||
if (attrName && attrName.toLowerCase() == name) { |
|
||||
ret = it.attribute("value") |
|
||||
} |
|
||||
} |
|
||||
return ret |
|
||||
} |
|
||||
|
|
||||
// Properties exported here are visible to all plugins. |
|
||||
ext { |
|
||||
// These helpers are shared, but are not guaranteed to be stable / unchanged. |
|
||||
privateHelpers = {} |
|
||||
privateHelpers.getProjectTarget = { doGetProjectTarget() } |
|
||||
privateHelpers.findLatestInstalledBuildTools = { doFindLatestInstalledBuildTools('19.1.0') } |
|
||||
privateHelpers.extractIntFromManifest = { name -> doExtractIntFromManifest(name) } |
|
||||
privateHelpers.extractStringFromManifest = { name -> doExtractStringFromManifest(name) } |
|
||||
privateHelpers.promptForPassword = { msg -> doPromptForPassword(msg) } |
|
||||
privateHelpers.ensureValueExists = { filePath, props, key -> doEnsureValueExists(filePath, props, key) } |
|
||||
|
|
||||
// These helpers can be used by plugins / projects and will not change. |
|
||||
cdvHelpers = {} |
|
||||
// Returns a XmlParser for the config.xml. Added in 4.1.0. |
|
||||
cdvHelpers.getConfigXml = { doGetConfigXml() } |
|
||||
// Returns the value for the desired <preference>. Added in 4.1.0. |
|
||||
cdvHelpers.getConfigPreference = { name, defaultValue -> doGetConfigPreference(name, defaultValue) } |
|
||||
} |
|
||||
|
|
@ -1,16 +0,0 @@ |
|||||
# This file is automatically generated by Android Tools. |
|
||||
# Do not modify this file -- YOUR CHANGES WILL BE ERASED! |
|
||||
# |
|
||||
# This file must be checked in Version Control Systems. |
|
||||
# |
|
||||
# To customize properties used by the Ant build system use, |
|
||||
# "ant.properties", and override values to adapt the script to your |
|
||||
# project structure. |
|
||||
|
|
||||
# Indicates whether an apk should be generated for each density. |
|
||||
split.density=false |
|
||||
# Project target. |
|
||||
target=android-23 |
|
||||
apk-configurations= |
|
||||
renderscript.opt.level=O0 |
|
||||
android.library=true |
|
@ -1,69 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
/** |
|
||||
* The Class AuthenticationToken defines the userName and password to be used for authenticating a web resource |
|
||||
*/ |
|
||||
public class AuthenticationToken { |
|
||||
private String userName; |
|
||||
private String password; |
|
||||
|
|
||||
/** |
|
||||
* Gets the user name. |
|
||||
* |
|
||||
* @return the user name |
|
||||
*/ |
|
||||
public String getUserName() { |
|
||||
return userName; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Sets the user name. |
|
||||
* |
|
||||
* @param userName |
|
||||
* the new user name |
|
||||
*/ |
|
||||
public void setUserName(String userName) { |
|
||||
this.userName = userName; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Gets the password. |
|
||||
* |
|
||||
* @return the password |
|
||||
*/ |
|
||||
public String getPassword() { |
|
||||
return password; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Sets the password. |
|
||||
* |
|
||||
* @param password |
|
||||
* the new password |
|
||||
*/ |
|
||||
public void setPassword(String password) { |
|
||||
this.password = password; |
|
||||
} |
|
||||
|
|
||||
|
|
||||
|
|
||||
|
|
||||
} |
|
@ -1,144 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.json.JSONArray; |
|
||||
|
|
||||
import android.util.Log; |
|
||||
|
|
||||
import org.apache.cordova.CordovaWebView; |
|
||||
import org.apache.cordova.PluginResult; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
public class CallbackContext { |
|
||||
private static final String LOG_TAG = "CordovaPlugin"; |
|
||||
|
|
||||
private String callbackId; |
|
||||
private CordovaWebView webView; |
|
||||
protected boolean finished; |
|
||||
private int changingThreads; |
|
||||
|
|
||||
public CallbackContext(String callbackId, CordovaWebView webView) { |
|
||||
this.callbackId = callbackId; |
|
||||
this.webView = webView; |
|
||||
} |
|
||||
|
|
||||
public boolean isFinished() { |
|
||||
return finished; |
|
||||
} |
|
||||
|
|
||||
public boolean isChangingThreads() { |
|
||||
return changingThreads > 0; |
|
||||
} |
|
||||
|
|
||||
public String getCallbackId() { |
|
||||
return callbackId; |
|
||||
} |
|
||||
|
|
||||
public void sendPluginResult(PluginResult pluginResult) { |
|
||||
synchronized (this) { |
|
||||
if (finished) { |
|
||||
Log.w(LOG_TAG, "Attempted to send a second callback for ID: " + callbackId + "\nResult was: " + pluginResult.getMessage()); |
|
||||
return; |
|
||||
} else { |
|
||||
finished = !pluginResult.getKeepCallback(); |
|
||||
} |
|
||||
} |
|
||||
webView.sendPluginResult(pluginResult, callbackId); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
* |
|
||||
* @param message The message to add to the success result. |
|
||||
*/ |
|
||||
public void success(JSONObject message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
* |
|
||||
* @param message The message to add to the success result. |
|
||||
*/ |
|
||||
public void success(String message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
* |
|
||||
* @param message The message to add to the success result. |
|
||||
*/ |
|
||||
public void success(JSONArray message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
* |
|
||||
* @param message The message to add to the success result. |
|
||||
*/ |
|
||||
public void success(byte[] message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
* |
|
||||
* @param message The message to add to the success result. |
|
||||
*/ |
|
||||
public void success(int message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for success callbacks that just returns the Status.OK by default |
|
||||
*/ |
|
||||
public void success() { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.OK)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for error callbacks that just returns the Status.ERROR by default |
|
||||
* |
|
||||
* @param message The message to add to the error result. |
|
||||
*/ |
|
||||
public void error(JSONObject message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.ERROR, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for error callbacks that just returns the Status.ERROR by default |
|
||||
* |
|
||||
* @param message The message to add to the error result. |
|
||||
*/ |
|
||||
public void error(String message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.ERROR, message)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Helper for error callbacks that just returns the Status.ERROR by default |
|
||||
* |
|
||||
* @param message The message to add to the error result. |
|
||||
*/ |
|
||||
public void error(int message) { |
|
||||
sendPluginResult(new PluginResult(PluginResult.Status.ERROR, message)); |
|
||||
} |
|
||||
} |
|
@ -1,65 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.util.Pair; |
|
||||
import android.util.SparseArray; |
|
||||
|
|
||||
/** |
|
||||
* Provides a collection that maps unique request codes to CordovaPlugins and Integers. |
|
||||
* Used to ensure that when plugins make requests for runtime permissions, those requests do not |
|
||||
* collide with requests from other plugins that use the same request code value. |
|
||||
*/ |
|
||||
public class CallbackMap { |
|
||||
private int currentCallbackId = 0; |
|
||||
private SparseArray<Pair<CordovaPlugin, Integer>> callbacks; |
|
||||
|
|
||||
public CallbackMap() { |
|
||||
this.callbacks = new SparseArray<Pair<CordovaPlugin, Integer>>(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Stores a CordovaPlugin and request code and returns a new unique request code to use |
|
||||
* in a permission request. |
|
||||
* |
|
||||
* @param receiver The plugin that is making the request |
|
||||
* @param requestCode The original request code used by the plugin |
|
||||
* @return A unique request code that can be used to retrieve this callback |
|
||||
* with getAndRemoveCallback() |
|
||||
*/ |
|
||||
public synchronized int registerCallback(CordovaPlugin receiver, int requestCode) { |
|
||||
int mappedId = this.currentCallbackId++; |
|
||||
callbacks.put(mappedId, new Pair<CordovaPlugin, Integer>(receiver, requestCode)); |
|
||||
return mappedId; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Retrieves and removes a callback stored in the map using the mapped request code |
|
||||
* obtained from registerCallback() |
|
||||
* |
|
||||
* @param mappedId The request code obtained from registerCallback() |
|
||||
* @return The CordovaPlugin and orignal request code that correspond to the |
|
||||
* given mappedCode |
|
||||
*/ |
|
||||
public synchronized Pair<CordovaPlugin, Integer> getAndRemoveCallback(int mappedId) { |
|
||||
Pair<CordovaPlugin, Integer> callback = callbacks.get(mappedId); |
|
||||
callbacks.remove(mappedId); |
|
||||
return callback; |
|
||||
} |
|
||||
} |
|
@ -1,72 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.List; |
|
||||
|
|
||||
import android.app.Activity; |
|
||||
import android.util.Log; |
|
||||
|
|
||||
@Deprecated // Use Whitelist, CordovaPrefences, etc. directly. |
|
||||
public class Config { |
|
||||
private static final String TAG = "Config"; |
|
||||
|
|
||||
static ConfigXmlParser parser; |
|
||||
|
|
||||
private Config() { |
|
||||
} |
|
||||
|
|
||||
public static void init(Activity action) { |
|
||||
parser = new ConfigXmlParser(); |
|
||||
parser.parse(action); |
|
||||
//TODO: Add feature to bring this back. Some preferences should be overridden by intents, but not all |
|
||||
parser.getPreferences().setPreferencesBundle(action.getIntent().getExtras()); |
|
||||
} |
|
||||
|
|
||||
// Intended to be used for testing only; creates an empty configuration. |
|
||||
public static void init() { |
|
||||
if (parser == null) { |
|
||||
parser = new ConfigXmlParser(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public static String getStartUrl() { |
|
||||
if (parser == null) { |
|
||||
return "file:///android_asset/www/index.html"; |
|
||||
} |
|
||||
return parser.getLaunchUrl(); |
|
||||
} |
|
||||
|
|
||||
public static String getErrorUrl() { |
|
||||
return parser.getPreferences().getString("errorurl", null); |
|
||||
} |
|
||||
|
|
||||
public static List<PluginEntry> getPluginEntries() { |
|
||||
return parser.getPluginEntries(); |
|
||||
} |
|
||||
|
|
||||
public static CordovaPreferences getPreferences() { |
|
||||
return parser.getPreferences(); |
|
||||
} |
|
||||
|
|
||||
public static boolean isInitialized() { |
|
||||
return parser != null; |
|
||||
} |
|
||||
} |
|
@ -1,145 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.io.IOException; |
|
||||
import java.util.ArrayList; |
|
||||
import java.util.Locale; |
|
||||
import java.util.regex.Matcher; |
|
||||
import java.util.regex.Pattern; |
|
||||
|
|
||||
import org.xmlpull.v1.XmlPullParser; |
|
||||
import org.xmlpull.v1.XmlPullParserException; |
|
||||
|
|
||||
import android.content.Context; |
|
||||
|
|
||||
public class ConfigXmlParser { |
|
||||
private static String TAG = "ConfigXmlParser"; |
|
||||
|
|
||||
private String launchUrl = "file:///android_asset/www/index.html"; |
|
||||
private CordovaPreferences prefs = new CordovaPreferences(); |
|
||||
private ArrayList<PluginEntry> pluginEntries = new ArrayList<PluginEntry>(20); |
|
||||
|
|
||||
public CordovaPreferences getPreferences() { |
|
||||
return prefs; |
|
||||
} |
|
||||
|
|
||||
public ArrayList<PluginEntry> getPluginEntries() { |
|
||||
return pluginEntries; |
|
||||
} |
|
||||
|
|
||||
public String getLaunchUrl() { |
|
||||
return launchUrl; |
|
||||
} |
|
||||
|
|
||||
public void parse(Context action) { |
|
||||
// First checking the class namespace for config.xml |
|
||||
int id = action.getResources().getIdentifier("config", "xml", action.getClass().getPackage().getName()); |
|
||||
if (id == 0) { |
|
||||
// If we couldn't find config.xml there, we'll look in the namespace from AndroidManifest.xml |
|
||||
id = action.getResources().getIdentifier("config", "xml", action.getPackageName()); |
|
||||
if (id == 0) { |
|
||||
LOG.e(TAG, "res/xml/config.xml is missing!"); |
|
||||
return; |
|
||||
} |
|
||||
} |
|
||||
parse(action.getResources().getXml(id)); |
|
||||
} |
|
||||
|
|
||||
boolean insideFeature = false; |
|
||||
String service = "", pluginClass = "", paramType = ""; |
|
||||
boolean onload = false; |
|
||||
|
|
||||
public void parse(XmlPullParser xml) { |
|
||||
int eventType = -1; |
|
||||
|
|
||||
while (eventType != XmlPullParser.END_DOCUMENT) { |
|
||||
if (eventType == XmlPullParser.START_TAG) { |
|
||||
handleStartTag(xml); |
|
||||
} |
|
||||
else if (eventType == XmlPullParser.END_TAG) |
|
||||
{ |
|
||||
handleEndTag(xml); |
|
||||
} |
|
||||
try { |
|
||||
eventType = xml.next(); |
|
||||
} catch (XmlPullParserException e) { |
|
||||
e.printStackTrace(); |
|
||||
} catch (IOException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void handleStartTag(XmlPullParser xml) { |
|
||||
String strNode = xml.getName(); |
|
||||
if (strNode.equals("feature")) { |
|
||||
//Check for supported feature sets aka. plugins (Accelerometer, Geolocation, etc) |
|
||||
//Set the bit for reading params |
|
||||
insideFeature = true; |
|
||||
service = xml.getAttributeValue(null, "name"); |
|
||||
} |
|
||||
else if (insideFeature && strNode.equals("param")) { |
|
||||
paramType = xml.getAttributeValue(null, "name"); |
|
||||
if (paramType.equals("service")) // check if it is using the older service param |
|
||||
service = xml.getAttributeValue(null, "value"); |
|
||||
else if (paramType.equals("package") || paramType.equals("android-package")) |
|
||||
pluginClass = xml.getAttributeValue(null,"value"); |
|
||||
else if (paramType.equals("onload")) |
|
||||
onload = "true".equals(xml.getAttributeValue(null, "value")); |
|
||||
} |
|
||||
else if (strNode.equals("preference")) { |
|
||||
String name = xml.getAttributeValue(null, "name").toLowerCase(Locale.ENGLISH); |
|
||||
String value = xml.getAttributeValue(null, "value"); |
|
||||
prefs.set(name, value); |
|
||||
} |
|
||||
else if (strNode.equals("content")) { |
|
||||
String src = xml.getAttributeValue(null, "src"); |
|
||||
if (src != null) { |
|
||||
setStartUrl(src); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void handleEndTag(XmlPullParser xml) { |
|
||||
String strNode = xml.getName(); |
|
||||
if (strNode.equals("feature")) { |
|
||||
pluginEntries.add(new PluginEntry(service, pluginClass, onload)); |
|
||||
|
|
||||
service = ""; |
|
||||
pluginClass = ""; |
|
||||
insideFeature = false; |
|
||||
onload = false; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private void setStartUrl(String src) { |
|
||||
Pattern schemeRegex = Pattern.compile("^[a-z-]+://"); |
|
||||
Matcher matcher = schemeRegex.matcher(src); |
|
||||
if (matcher.find()) { |
|
||||
launchUrl = src; |
|
||||
} else { |
|
||||
if (src.charAt(0) == '/') { |
|
||||
src = src.substring(1); |
|
||||
} |
|
||||
launchUrl = "file:///android_asset/www/" + src; |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,508 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.ArrayList; |
|
||||
import java.util.Locale; |
|
||||
|
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import android.app.Activity; |
|
||||
import android.app.AlertDialog; |
|
||||
import android.annotation.SuppressLint; |
|
||||
import android.content.DialogInterface; |
|
||||
import android.content.Intent; |
|
||||
import android.content.res.Configuration; |
|
||||
import android.graphics.Color; |
|
||||
import android.media.AudioManager; |
|
||||
import android.os.Build; |
|
||||
import android.os.Bundle; |
|
||||
import android.util.Log; |
|
||||
import android.view.Menu; |
|
||||
import android.view.MenuItem; |
|
||||
import android.view.View; |
|
||||
import android.view.ViewGroup; |
|
||||
import android.view.Window; |
|
||||
import android.view.WindowManager; |
|
||||
import android.webkit.WebViewClient; |
|
||||
import android.widget.FrameLayout; |
|
||||
|
|
||||
/** |
|
||||
* This class is the main Android activity that represents the Cordova |
|
||||
* application. It should be extended by the user to load the specific |
|
||||
* html file that contains the application. |
|
||||
* |
|
||||
* As an example: |
|
||||
* |
|
||||
* <pre> |
|
||||
* package org.apache.cordova.examples; |
|
||||
* |
|
||||
* import android.os.Bundle; |
|
||||
* import org.apache.cordova.*; |
|
||||
* |
|
||||
* public class Example extends CordovaActivity { |
|
||||
* @Override |
|
||||
* public void onCreate(Bundle savedInstanceState) { |
|
||||
* super.onCreate(savedInstanceState); |
|
||||
* super.init(); |
|
||||
* // Load your application |
|
||||
* loadUrl(launchUrl); |
|
||||
* } |
|
||||
* } |
|
||||
* </pre> |
|
||||
* |
|
||||
* Cordova xml configuration: Cordova uses a configuration file at |
|
||||
* res/xml/config.xml to specify its settings. See "The config.xml File" |
|
||||
* guide in cordova-docs at http://cordova.apache.org/docs for the documentation |
|
||||
* for the configuration. The use of the set*Property() methods is |
|
||||
* deprecated in favor of the config.xml file. |
|
||||
* |
|
||||
*/ |
|
||||
public class CordovaActivity extends Activity { |
|
||||
public static String TAG = "CordovaActivity"; |
|
||||
|
|
||||
// The webview for our app |
|
||||
protected CordovaWebView appView; |
|
||||
|
|
||||
private static int ACTIVITY_STARTING = 0; |
|
||||
private static int ACTIVITY_RUNNING = 1; |
|
||||
private static int ACTIVITY_EXITING = 2; |
|
||||
|
|
||||
// Keep app running when pause is received. (default = true) |
|
||||
// If true, then the JavaScript and native code continue to run in the background |
|
||||
// when another application (activity) is started. |
|
||||
protected boolean keepRunning = true; |
|
||||
|
|
||||
// Flag to keep immersive mode if set to fullscreen |
|
||||
protected boolean immersiveMode; |
|
||||
|
|
||||
// Read from config.xml: |
|
||||
protected CordovaPreferences preferences; |
|
||||
protected String launchUrl; |
|
||||
protected ArrayList<PluginEntry> pluginEntries; |
|
||||
protected CordovaInterfaceImpl cordovaInterface; |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is first created. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onCreate(Bundle savedInstanceState) { |
|
||||
LOG.i(TAG, "Apache Cordova native platform version " + CordovaWebView.CORDOVA_VERSION + " is starting"); |
|
||||
LOG.d(TAG, "CordovaActivity.onCreate()"); |
|
||||
|
|
||||
// need to activate preferences before super.onCreate to avoid "requestFeature() must be called before adding content" exception |
|
||||
loadConfig(); |
|
||||
if (!preferences.getBoolean("ShowTitle", false)) { |
|
||||
getWindow().requestFeature(Window.FEATURE_NO_TITLE); |
|
||||
} |
|
||||
|
|
||||
if (preferences.getBoolean("SetFullscreen", false)) { |
|
||||
Log.d(TAG, "The SetFullscreen configuration is deprecated in favor of Fullscreen, and will be removed in a future version."); |
|
||||
preferences.set("Fullscreen", true); |
|
||||
} |
|
||||
if (preferences.getBoolean("Fullscreen", false)) { |
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { |
|
||||
immersiveMode = true; |
|
||||
} else { |
|
||||
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, |
|
||||
WindowManager.LayoutParams.FLAG_FULLSCREEN); |
|
||||
} |
|
||||
} else { |
|
||||
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN, |
|
||||
WindowManager.LayoutParams.FLAG_FORCE_NOT_FULLSCREEN); |
|
||||
} |
|
||||
|
|
||||
super.onCreate(savedInstanceState); |
|
||||
|
|
||||
cordovaInterface = makeCordovaInterface(); |
|
||||
if (savedInstanceState != null) { |
|
||||
cordovaInterface.restoreInstanceState(savedInstanceState); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
protected void init() { |
|
||||
appView = makeWebView(); |
|
||||
createViews(); |
|
||||
if (!appView.isInitialized()) { |
|
||||
appView.init(cordovaInterface, pluginEntries, preferences); |
|
||||
} |
|
||||
cordovaInterface.onCordovaInit(appView.getPluginManager()); |
|
||||
|
|
||||
// Wire the hardware volume controls to control media if desired. |
|
||||
String volumePref = preferences.getString("DefaultVolumeStream", ""); |
|
||||
if ("media".equals(volumePref.toLowerCase(Locale.ENGLISH))) { |
|
||||
setVolumeControlStream(AudioManager.STREAM_MUSIC); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@SuppressWarnings("deprecation") |
|
||||
protected void loadConfig() { |
|
||||
ConfigXmlParser parser = new ConfigXmlParser(); |
|
||||
parser.parse(this); |
|
||||
preferences = parser.getPreferences(); |
|
||||
preferences.setPreferencesBundle(getIntent().getExtras()); |
|
||||
launchUrl = parser.getLaunchUrl(); |
|
||||
pluginEntries = parser.getPluginEntries(); |
|
||||
Config.parser = parser; |
|
||||
} |
|
||||
|
|
||||
//Suppressing warnings in AndroidStudio |
|
||||
@SuppressWarnings({"deprecation", "ResourceType"}) |
|
||||
protected void createViews() { |
|
||||
//Why are we setting a constant as the ID? This should be investigated |
|
||||
appView.getView().setId(100); |
|
||||
appView.getView().setLayoutParams(new FrameLayout.LayoutParams( |
|
||||
ViewGroup.LayoutParams.MATCH_PARENT, |
|
||||
ViewGroup.LayoutParams.MATCH_PARENT)); |
|
||||
|
|
||||
setContentView(appView.getView()); |
|
||||
|
|
||||
if (preferences.contains("BackgroundColor")) { |
|
||||
int backgroundColor = preferences.getInteger("BackgroundColor", Color.BLACK); |
|
||||
// Background of activity: |
|
||||
appView.getView().setBackgroundColor(backgroundColor); |
|
||||
} |
|
||||
|
|
||||
appView.getView().requestFocusFromTouch(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Construct the default web view object. |
|
||||
* <p/> |
|
||||
* Override this to customize the webview that is used. |
|
||||
*/ |
|
||||
protected CordovaWebView makeWebView() { |
|
||||
return new CordovaWebViewImpl(makeWebViewEngine()); |
|
||||
} |
|
||||
|
|
||||
protected CordovaWebViewEngine makeWebViewEngine() { |
|
||||
return CordovaWebViewImpl.createEngine(this, preferences); |
|
||||
} |
|
||||
|
|
||||
protected CordovaInterfaceImpl makeCordovaInterface() { |
|
||||
return new CordovaInterfaceImpl(this) { |
|
||||
@Override |
|
||||
public Object onMessage(String id, Object data) { |
|
||||
// Plumb this to CordovaActivity.onMessage for backwards compatibility |
|
||||
return CordovaActivity.this.onMessage(id, data); |
|
||||
} |
|
||||
}; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Load the url into the webview. |
|
||||
*/ |
|
||||
public void loadUrl(String url) { |
|
||||
if (appView == null) { |
|
||||
init(); |
|
||||
} |
|
||||
|
|
||||
// If keepRunning |
|
||||
this.keepRunning = preferences.getBoolean("KeepRunning", true); |
|
||||
|
|
||||
appView.loadUrlIntoView(url, true); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the system is about to start resuming a previous activity. |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onPause() { |
|
||||
super.onPause(); |
|
||||
LOG.d(TAG, "Paused the activity."); |
|
||||
|
|
||||
if (this.appView != null) { |
|
||||
// CB-9382 If there is an activity that started for result and main activity is waiting for callback |
|
||||
// result, we shoudn't stop WebView Javascript timers, as activity for result might be using them |
|
||||
boolean keepRunning = this.keepRunning || this.cordovaInterface.activityResultCallback != null; |
|
||||
this.appView.handlePause(keepRunning); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity receives a new intent |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onNewIntent(Intent intent) { |
|
||||
super.onNewIntent(intent); |
|
||||
//Forward to plugins |
|
||||
if (this.appView != null) |
|
||||
this.appView.onNewIntent(intent); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity will start interacting with the user. |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onResume() { |
|
||||
super.onResume(); |
|
||||
LOG.d(TAG, "Resumed the activity."); |
|
||||
|
|
||||
if (this.appView == null) { |
|
||||
return; |
|
||||
} |
|
||||
// Force window to have focus, so application always |
|
||||
// receive user input. Workaround for some devices (Samsung Galaxy Note 3 at least) |
|
||||
this.getWindow().getDecorView().requestFocus(); |
|
||||
|
|
||||
this.appView.handleResume(this.keepRunning); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is no longer visible to the user. |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onStop() { |
|
||||
super.onStop(); |
|
||||
LOG.d(TAG, "Stopped the activity."); |
|
||||
|
|
||||
if (this.appView == null) { |
|
||||
return; |
|
||||
} |
|
||||
this.appView.handleStop(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is becoming visible to the user. |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onStart() { |
|
||||
super.onStart(); |
|
||||
LOG.d(TAG, "Started the activity."); |
|
||||
|
|
||||
if (this.appView == null) { |
|
||||
return; |
|
||||
} |
|
||||
this.appView.handleStart(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* The final call you receive before your activity is destroyed. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onDestroy() { |
|
||||
LOG.d(TAG, "CordovaActivity.onDestroy()"); |
|
||||
super.onDestroy(); |
|
||||
|
|
||||
if (this.appView != null) { |
|
||||
appView.handleDestroy(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when view focus is changed |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onWindowFocusChanged(boolean hasFocus) { |
|
||||
super.onWindowFocusChanged(hasFocus); |
|
||||
if (hasFocus && immersiveMode) { |
|
||||
final int uiOptions = View.SYSTEM_UI_FLAG_LAYOUT_STABLE |
|
||||
| View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION |
|
||||
| View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN |
|
||||
| View.SYSTEM_UI_FLAG_HIDE_NAVIGATION |
|
||||
| View.SYSTEM_UI_FLAG_FULLSCREEN |
|
||||
| View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY; |
|
||||
|
|
||||
getWindow().getDecorView().setSystemUiVisibility(uiOptions); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@SuppressLint("NewApi") |
|
||||
@Override |
|
||||
public void startActivityForResult(Intent intent, int requestCode, Bundle options) { |
|
||||
// Capture requestCode here so that it is captured in the setActivityResultCallback() case. |
|
||||
cordovaInterface.setActivityResultRequestCode(requestCode); |
|
||||
super.startActivityForResult(intent, requestCode, options); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when an activity you launched exits, giving you the requestCode you started it with, |
|
||||
* the resultCode it returned, and any additional data from it. |
|
||||
* |
|
||||
* @param requestCode The request code originally supplied to startActivityForResult(), |
|
||||
* allowing you to identify who this result came from. |
|
||||
* @param resultCode The integer result code returned by the child activity through its setResult(). |
|
||||
* @param intent An Intent, which can return result data to the caller (various data can be attached to Intent "extras"). |
|
||||
*/ |
|
||||
@Override |
|
||||
protected void onActivityResult(int requestCode, int resultCode, Intent intent) { |
|
||||
LOG.d(TAG, "Incoming Result. Request code = " + requestCode); |
|
||||
super.onActivityResult(requestCode, resultCode, intent); |
|
||||
cordovaInterface.onActivityResult(requestCode, resultCode, intent); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Report an error to the host application. These errors are unrecoverable (i.e. the main resource is unavailable). |
|
||||
* The errorCode parameter corresponds to one of the ERROR_* constants. |
|
||||
* |
|
||||
* @param errorCode The error code corresponding to an ERROR_* value. |
|
||||
* @param description A String describing the error. |
|
||||
* @param failingUrl The url that failed to load. |
|
||||
*/ |
|
||||
public void onReceivedError(final int errorCode, final String description, final String failingUrl) { |
|
||||
final CordovaActivity me = this; |
|
||||
|
|
||||
// If errorUrl specified, then load it |
|
||||
final String errorUrl = preferences.getString("errorUrl", null); |
|
||||
if ((errorUrl != null) && (!failingUrl.equals(errorUrl)) && (appView != null)) { |
|
||||
// Load URL on UI thread |
|
||||
me.runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
me.appView.showWebPage(errorUrl, false, true, null); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
// If not, then display error dialog |
|
||||
else { |
|
||||
final boolean exit = !(errorCode == WebViewClient.ERROR_HOST_LOOKUP); |
|
||||
me.runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
if (exit) { |
|
||||
me.appView.getView().setVisibility(View.GONE); |
|
||||
me.displayError("Application Error", description + " (" + failingUrl + ")", "OK", exit); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Display an error dialog and optionally exit application. |
|
||||
*/ |
|
||||
public void displayError(final String title, final String message, final String button, final boolean exit) { |
|
||||
final CordovaActivity me = this; |
|
||||
me.runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
try { |
|
||||
AlertDialog.Builder dlg = new AlertDialog.Builder(me); |
|
||||
dlg.setMessage(message); |
|
||||
dlg.setTitle(title); |
|
||||
dlg.setCancelable(false); |
|
||||
dlg.setPositiveButton(button, |
|
||||
new AlertDialog.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
dialog.dismiss(); |
|
||||
if (exit) { |
|
||||
finish(); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
dlg.create(); |
|
||||
dlg.show(); |
|
||||
} catch (Exception e) { |
|
||||
finish(); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Hook in Cordova for menu plugins |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean onCreateOptionsMenu(Menu menu) { |
|
||||
if (appView != null) { |
|
||||
appView.getPluginManager().postMessage("onCreateOptionsMenu", menu); |
|
||||
} |
|
||||
return super.onCreateOptionsMenu(menu); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean onPrepareOptionsMenu(Menu menu) { |
|
||||
if (appView != null) { |
|
||||
appView.getPluginManager().postMessage("onPrepareOptionsMenu", menu); |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean onOptionsItemSelected(MenuItem item) { |
|
||||
if (appView != null) { |
|
||||
appView.getPluginManager().postMessage("onOptionsItemSelected", item); |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when a message is sent to plugin. |
|
||||
* |
|
||||
* @param id The message id |
|
||||
* @param data The message data |
|
||||
* @return Object or null |
|
||||
*/ |
|
||||
public Object onMessage(String id, Object data) { |
|
||||
if ("onReceivedError".equals(id)) { |
|
||||
JSONObject d = (JSONObject) data; |
|
||||
try { |
|
||||
this.onReceivedError(d.getInt("errorCode"), d.getString("description"), d.getString("url")); |
|
||||
} catch (JSONException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
} else if ("exit".equals(id)) { |
|
||||
finish(); |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
protected void onSaveInstanceState(Bundle outState) { |
|
||||
cordovaInterface.onSaveInstanceState(outState); |
|
||||
super.onSaveInstanceState(outState); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the device configuration changes while your activity is running. |
|
||||
* |
|
||||
* @param newConfig The new device configuration |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onConfigurationChanged(Configuration newConfig) { |
|
||||
super.onConfigurationChanged(newConfig); |
|
||||
if (this.appView == null) { |
|
||||
return; |
|
||||
} |
|
||||
PluginManager pm = this.appView.getPluginManager(); |
|
||||
if (pm != null) { |
|
||||
pm.onConfigurationChanged(newConfig); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the user grants permissions |
|
||||
* |
|
||||
* @param requestCode |
|
||||
* @param permissions |
|
||||
* @param grantResults |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onRequestPermissionsResult(int requestCode, String permissions[], |
|
||||
int[] grantResults) { |
|
||||
try |
|
||||
{ |
|
||||
cordovaInterface.onRequestPermissionResult(requestCode, permissions, grantResults); |
|
||||
} |
|
||||
catch (JSONException e) |
|
||||
{ |
|
||||
LOG.d(TAG, "JSONException: Parameters fed into the method are not valid"); |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
|
|
||||
} |
|
@ -1,113 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.json.JSONArray; |
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import android.util.Base64; |
|
||||
|
|
||||
public class CordovaArgs { |
|
||||
private JSONArray baseArgs; |
|
||||
|
|
||||
public CordovaArgs(JSONArray args) { |
|
||||
this.baseArgs = args; |
|
||||
} |
|
||||
|
|
||||
|
|
||||
// Pass through the basics to the base args. |
|
||||
public Object get(int index) throws JSONException { |
|
||||
return baseArgs.get(index); |
|
||||
} |
|
||||
|
|
||||
public boolean getBoolean(int index) throws JSONException { |
|
||||
return baseArgs.getBoolean(index); |
|
||||
} |
|
||||
|
|
||||
public double getDouble(int index) throws JSONException { |
|
||||
return baseArgs.getDouble(index); |
|
||||
} |
|
||||
|
|
||||
public int getInt(int index) throws JSONException { |
|
||||
return baseArgs.getInt(index); |
|
||||
} |
|
||||
|
|
||||
public JSONArray getJSONArray(int index) throws JSONException { |
|
||||
return baseArgs.getJSONArray(index); |
|
||||
} |
|
||||
|
|
||||
public JSONObject getJSONObject(int index) throws JSONException { |
|
||||
return baseArgs.getJSONObject(index); |
|
||||
} |
|
||||
|
|
||||
public long getLong(int index) throws JSONException { |
|
||||
return baseArgs.getLong(index); |
|
||||
} |
|
||||
|
|
||||
public String getString(int index) throws JSONException { |
|
||||
return baseArgs.getString(index); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
public Object opt(int index) { |
|
||||
return baseArgs.opt(index); |
|
||||
} |
|
||||
|
|
||||
public boolean optBoolean(int index) { |
|
||||
return baseArgs.optBoolean(index); |
|
||||
} |
|
||||
|
|
||||
public double optDouble(int index) { |
|
||||
return baseArgs.optDouble(index); |
|
||||
} |
|
||||
|
|
||||
public int optInt(int index) { |
|
||||
return baseArgs.optInt(index); |
|
||||
} |
|
||||
|
|
||||
public JSONArray optJSONArray(int index) { |
|
||||
return baseArgs.optJSONArray(index); |
|
||||
} |
|
||||
|
|
||||
public JSONObject optJSONObject(int index) { |
|
||||
return baseArgs.optJSONObject(index); |
|
||||
} |
|
||||
|
|
||||
public long optLong(int index) { |
|
||||
return baseArgs.optLong(index); |
|
||||
} |
|
||||
|
|
||||
public String optString(int index) { |
|
||||
return baseArgs.optString(index); |
|
||||
} |
|
||||
|
|
||||
public boolean isNull(int index) { |
|
||||
return baseArgs.isNull(index); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
// The interesting custom helpers. |
|
||||
public byte[] getArrayBuffer(int index) throws JSONException { |
|
||||
String encoded = baseArgs.getString(index); |
|
||||
return Base64.decode(encoded, Base64.DEFAULT); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
|
|
@ -1,184 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.security.SecureRandom; |
|
||||
|
|
||||
import org.json.JSONArray; |
|
||||
import org.json.JSONException; |
|
||||
|
|
||||
import android.util.Log; |
|
||||
|
|
||||
/** |
|
||||
* Contains APIs that the JS can call. All functions in here should also have |
|
||||
* an equivalent entry in CordovaChromeClient.java, and be added to |
|
||||
* cordova-js/lib/android/plugin/android/promptbasednativeapi.js |
|
||||
*/ |
|
||||
public class CordovaBridge { |
|
||||
private static final String LOG_TAG = "CordovaBridge"; |
|
||||
private PluginManager pluginManager; |
|
||||
private NativeToJsMessageQueue jsMessageQueue; |
|
||||
private volatile int expectedBridgeSecret = -1; // written by UI thread, read by JS thread. |
|
||||
|
|
||||
public CordovaBridge(PluginManager pluginManager, NativeToJsMessageQueue jsMessageQueue) { |
|
||||
this.pluginManager = pluginManager; |
|
||||
this.jsMessageQueue = jsMessageQueue; |
|
||||
} |
|
||||
|
|
||||
public String jsExec(int bridgeSecret, String service, String action, String callbackId, String arguments) throws JSONException, IllegalAccessException { |
|
||||
if (!verifySecret("exec()", bridgeSecret)) { |
|
||||
return null; |
|
||||
} |
|
||||
// If the arguments weren't received, send a message back to JS. It will switch bridge modes and try again. See CB-2666. |
|
||||
// We send a message meant specifically for this case. It starts with "@" so no other message can be encoded into the same string. |
|
||||
if (arguments == null) { |
|
||||
return "@Null arguments."; |
|
||||
} |
|
||||
|
|
||||
jsMessageQueue.setPaused(true); |
|
||||
try { |
|
||||
// Tell the resourceApi what thread the JS is running on. |
|
||||
CordovaResourceApi.jsThread = Thread.currentThread(); |
|
||||
|
|
||||
pluginManager.exec(service, action, callbackId, arguments); |
|
||||
String ret = null; |
|
||||
if (!NativeToJsMessageQueue.DISABLE_EXEC_CHAINING) { |
|
||||
ret = jsMessageQueue.popAndEncode(false); |
|
||||
} |
|
||||
return ret; |
|
||||
} catch (Throwable e) { |
|
||||
e.printStackTrace(); |
|
||||
return ""; |
|
||||
} finally { |
|
||||
jsMessageQueue.setPaused(false); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void jsSetNativeToJsBridgeMode(int bridgeSecret, int value) throws IllegalAccessException { |
|
||||
if (!verifySecret("setNativeToJsBridgeMode()", bridgeSecret)) { |
|
||||
return; |
|
||||
} |
|
||||
jsMessageQueue.setBridgeMode(value); |
|
||||
} |
|
||||
|
|
||||
public String jsRetrieveJsMessages(int bridgeSecret, boolean fromOnlineEvent) throws IllegalAccessException { |
|
||||
if (!verifySecret("retrieveJsMessages()", bridgeSecret)) { |
|
||||
return null; |
|
||||
} |
|
||||
return jsMessageQueue.popAndEncode(fromOnlineEvent); |
|
||||
} |
|
||||
|
|
||||
private boolean verifySecret(String action, int bridgeSecret) throws IllegalAccessException { |
|
||||
if (!jsMessageQueue.isBridgeEnabled()) { |
|
||||
if (bridgeSecret == -1) { |
|
||||
Log.d(LOG_TAG, action + " call made before bridge was enabled."); |
|
||||
} else { |
|
||||
Log.d(LOG_TAG, "Ignoring " + action + " from previous page load."); |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
// Bridge secret wrong and bridge not due to it being from the previous page. |
|
||||
if (expectedBridgeSecret < 0 || bridgeSecret != expectedBridgeSecret) { |
|
||||
Log.e(LOG_TAG, "Bridge access attempt with wrong secret token, possibly from malicious code. Disabling exec() bridge!"); |
|
||||
clearBridgeSecret(); |
|
||||
throw new IllegalAccessException(); |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** Called on page transitions */ |
|
||||
void clearBridgeSecret() { |
|
||||
expectedBridgeSecret = -1; |
|
||||
} |
|
||||
|
|
||||
public boolean isSecretEstablished() { |
|
||||
return expectedBridgeSecret != -1; |
|
||||
} |
|
||||
|
|
||||
/** Called by cordova.js to initialize the bridge. */ |
|
||||
int generateBridgeSecret() { |
|
||||
SecureRandom randGen = new SecureRandom(); |
|
||||
expectedBridgeSecret = randGen.nextInt(Integer.MAX_VALUE); |
|
||||
return expectedBridgeSecret; |
|
||||
} |
|
||||
|
|
||||
public void reset() { |
|
||||
jsMessageQueue.reset(); |
|
||||
clearBridgeSecret(); |
|
||||
} |
|
||||
|
|
||||
public String promptOnJsPrompt(String origin, String message, String defaultValue) { |
|
||||
if (defaultValue != null && defaultValue.length() > 3 && defaultValue.startsWith("gap:")) { |
|
||||
JSONArray array; |
|
||||
try { |
|
||||
array = new JSONArray(defaultValue.substring(4)); |
|
||||
int bridgeSecret = array.getInt(0); |
|
||||
String service = array.getString(1); |
|
||||
String action = array.getString(2); |
|
||||
String callbackId = array.getString(3); |
|
||||
String r = jsExec(bridgeSecret, service, action, callbackId, message); |
|
||||
return r == null ? "" : r; |
|
||||
} catch (JSONException e) { |
|
||||
e.printStackTrace(); |
|
||||
} catch (IllegalAccessException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
return ""; |
|
||||
} |
|
||||
// Sets the native->JS bridge mode. |
|
||||
else if (defaultValue != null && defaultValue.startsWith("gap_bridge_mode:")) { |
|
||||
try { |
|
||||
int bridgeSecret = Integer.parseInt(defaultValue.substring(16)); |
|
||||
jsSetNativeToJsBridgeMode(bridgeSecret, Integer.parseInt(message)); |
|
||||
} catch (NumberFormatException e){ |
|
||||
e.printStackTrace(); |
|
||||
} catch (IllegalAccessException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
return ""; |
|
||||
} |
|
||||
// Polling for JavaScript messages |
|
||||
else if (defaultValue != null && defaultValue.startsWith("gap_poll:")) { |
|
||||
int bridgeSecret = Integer.parseInt(defaultValue.substring(9)); |
|
||||
try { |
|
||||
String r = jsRetrieveJsMessages(bridgeSecret, "1".equals(message)); |
|
||||
return r == null ? "" : r; |
|
||||
} catch (IllegalAccessException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
return ""; |
|
||||
} |
|
||||
else if (defaultValue != null && defaultValue.startsWith("gap_init:")) { |
|
||||
// Protect against random iframes being able to talk through the bridge. |
|
||||
// Trust only pages which the app would have been allowed to navigate to anyway. |
|
||||
if (pluginManager.shouldAllowBridgeAccess(origin)) { |
|
||||
// Enable the bridge |
|
||||
int bridgeMode = Integer.parseInt(defaultValue.substring(9)); |
|
||||
jsMessageQueue.setBridgeMode(bridgeMode); |
|
||||
// Tell JS the bridge secret. |
|
||||
int secret = generateBridgeSecret(); |
|
||||
return ""+secret; |
|
||||
} else { |
|
||||
Log.e(LOG_TAG, "gap_init called from restricted origin: " + origin); |
|
||||
} |
|
||||
return ""; |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
} |
|
@ -1,96 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.security.Principal; |
|
||||
import java.security.PrivateKey; |
|
||||
import java.security.cert.X509Certificate; |
|
||||
|
|
||||
import android.webkit.ClientCertRequest; |
|
||||
|
|
||||
/** |
|
||||
* Implementation of the ICordovaClientCertRequest for Android WebView. |
|
||||
*/ |
|
||||
public class CordovaClientCertRequest implements ICordovaClientCertRequest { |
|
||||
|
|
||||
private final ClientCertRequest request; |
|
||||
|
|
||||
public CordovaClientCertRequest(ClientCertRequest request) { |
|
||||
this.request = request; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Cancel this request |
|
||||
*/ |
|
||||
public void cancel() |
|
||||
{ |
|
||||
request.cancel(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Returns the host name of the server requesting the certificate. |
|
||||
*/ |
|
||||
public String getHost() |
|
||||
{ |
|
||||
return request.getHost(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Returns the acceptable types of asymmetric keys (can be null). |
|
||||
*/ |
|
||||
public String[] getKeyTypes() |
|
||||
{ |
|
||||
return request.getKeyTypes(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Returns the port number of the server requesting the certificate. |
|
||||
*/ |
|
||||
public int getPort() |
|
||||
{ |
|
||||
return request.getPort(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Returns the acceptable certificate issuers for the certificate matching the private key (can be null). |
|
||||
*/ |
|
||||
public Principal[] getPrincipals() |
|
||||
{ |
|
||||
return request.getPrincipals(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Ignore the request for now. Do not remember user's choice. |
|
||||
*/ |
|
||||
public void ignore() |
|
||||
{ |
|
||||
request.ignore(); |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Proceed with the specified private key and client certificate chain. Remember the user's positive choice and use it for future requests. |
|
||||
* |
|
||||
* @param privateKey The privateKey |
|
||||
* @param chain The certificate chain |
|
||||
*/ |
|
||||
public void proceed(PrivateKey privateKey, X509Certificate[] chain) |
|
||||
{ |
|
||||
request.proceed(privateKey, chain); |
|
||||
} |
|
||||
} |
|
@ -1,152 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.app.AlertDialog; |
|
||||
import android.content.Context; |
|
||||
import android.content.DialogInterface; |
|
||||
import android.view.KeyEvent; |
|
||||
import android.widget.EditText; |
|
||||
|
|
||||
/** |
|
||||
* Helper class for WebViews to implement prompt(), alert(), confirm() dialogs. |
|
||||
*/ |
|
||||
public class CordovaDialogsHelper { |
|
||||
private final Context context; |
|
||||
private AlertDialog lastHandledDialog; |
|
||||
|
|
||||
public CordovaDialogsHelper(Context context) { |
|
||||
this.context = context; |
|
||||
} |
|
||||
|
|
||||
public void showAlert(String message, final Result result) { |
|
||||
AlertDialog.Builder dlg = new AlertDialog.Builder(context); |
|
||||
dlg.setMessage(message); |
|
||||
dlg.setTitle("Alert"); |
|
||||
//Don't let alerts break the back button |
|
||||
dlg.setCancelable(true); |
|
||||
dlg.setPositiveButton(android.R.string.ok, |
|
||||
new AlertDialog.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
result.gotResult(true, null); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setOnCancelListener( |
|
||||
new DialogInterface.OnCancelListener() { |
|
||||
public void onCancel(DialogInterface dialog) { |
|
||||
result.gotResult(false, null); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setOnKeyListener(new DialogInterface.OnKeyListener() { |
|
||||
//DO NOTHING |
|
||||
public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) { |
|
||||
if (keyCode == KeyEvent.KEYCODE_BACK) |
|
||||
{ |
|
||||
result.gotResult(true, null); |
|
||||
return false; |
|
||||
} |
|
||||
else |
|
||||
return true; |
|
||||
} |
|
||||
}); |
|
||||
lastHandledDialog = dlg.show(); |
|
||||
} |
|
||||
|
|
||||
public void showConfirm(String message, final Result result) { |
|
||||
AlertDialog.Builder dlg = new AlertDialog.Builder(context); |
|
||||
dlg.setMessage(message); |
|
||||
dlg.setTitle("Confirm"); |
|
||||
dlg.setCancelable(true); |
|
||||
dlg.setPositiveButton(android.R.string.ok, |
|
||||
new DialogInterface.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
result.gotResult(true, null); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setNegativeButton(android.R.string.cancel, |
|
||||
new DialogInterface.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
result.gotResult(false, null); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setOnCancelListener( |
|
||||
new DialogInterface.OnCancelListener() { |
|
||||
public void onCancel(DialogInterface dialog) { |
|
||||
result.gotResult(false, null); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setOnKeyListener(new DialogInterface.OnKeyListener() { |
|
||||
//DO NOTHING |
|
||||
public boolean onKey(DialogInterface dialog, int keyCode, KeyEvent event) { |
|
||||
if (keyCode == KeyEvent.KEYCODE_BACK) |
|
||||
{ |
|
||||
result.gotResult(false, null); |
|
||||
return false; |
|
||||
} |
|
||||
else |
|
||||
return true; |
|
||||
} |
|
||||
}); |
|
||||
lastHandledDialog = dlg.show(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Tell the client to display a prompt dialog to the user. |
|
||||
* If the client returns true, WebView will assume that the client will |
|
||||
* handle the prompt dialog and call the appropriate JsPromptResult method. |
|
||||
* |
|
||||
* Since we are hacking prompts for our own purposes, we should not be using them for |
|
||||
* this purpose, perhaps we should hack console.log to do this instead! |
|
||||
*/ |
|
||||
public void showPrompt(String message, String defaultValue, final Result result) { |
|
||||
// Returning false would also show a dialog, but the default one shows the origin (ugly). |
|
||||
AlertDialog.Builder dlg = new AlertDialog.Builder(context); |
|
||||
dlg.setMessage(message); |
|
||||
final EditText input = new EditText(context); |
|
||||
if (defaultValue != null) { |
|
||||
input.setText(defaultValue); |
|
||||
} |
|
||||
dlg.setView(input); |
|
||||
dlg.setCancelable(false); |
|
||||
dlg.setPositiveButton(android.R.string.ok, |
|
||||
new DialogInterface.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
String userText = input.getText().toString(); |
|
||||
result.gotResult(true, userText); |
|
||||
} |
|
||||
}); |
|
||||
dlg.setNegativeButton(android.R.string.cancel, |
|
||||
new DialogInterface.OnClickListener() { |
|
||||
public void onClick(DialogInterface dialog, int which) { |
|
||||
result.gotResult(false, null); |
|
||||
} |
|
||||
}); |
|
||||
lastHandledDialog = dlg.show(); |
|
||||
} |
|
||||
|
|
||||
public void destroyLastDialog(){ |
|
||||
if (lastHandledDialog != null){ |
|
||||
lastHandledDialog.cancel(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public interface Result { |
|
||||
public void gotResult(boolean success, String value); |
|
||||
} |
|
||||
} |
|
@ -1,51 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.webkit.HttpAuthHandler; |
|
||||
|
|
||||
/** |
|
||||
* Specifies interface for HTTP auth handler object which is used to handle auth requests and |
|
||||
* specifying user credentials. |
|
||||
*/ |
|
||||
public class CordovaHttpAuthHandler implements ICordovaHttpAuthHandler { |
|
||||
|
|
||||
private final HttpAuthHandler handler; |
|
||||
|
|
||||
public CordovaHttpAuthHandler(HttpAuthHandler handler) { |
|
||||
this.handler = handler; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Instructs the WebView to cancel the authentication request. |
|
||||
*/ |
|
||||
public void cancel () { |
|
||||
this.handler.cancel(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Instructs the WebView to proceed with the authentication with the given credentials. |
|
||||
* |
|
||||
* @param username |
|
||||
* @param password |
|
||||
*/ |
|
||||
public void proceed (String username, String password) { |
|
||||
this.handler.proceed(username, password); |
|
||||
} |
|
||||
} |
|
@ -1,88 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.app.Activity; |
|
||||
import android.content.Intent; |
|
||||
|
|
||||
import org.apache.cordova.CordovaPlugin; |
|
||||
|
|
||||
import java.util.concurrent.ExecutorService; |
|
||||
|
|
||||
/** |
|
||||
* The Activity interface that is implemented by CordovaActivity. |
|
||||
* It is used to isolate plugin development, and remove dependency on entire Cordova library. |
|
||||
*/ |
|
||||
public interface CordovaInterface { |
|
||||
|
|
||||
/** |
|
||||
* Launch an activity for which you would like a result when it finished. When this activity exits, |
|
||||
* your onActivityResult() method will be called. |
|
||||
* |
|
||||
* @param command The command object |
|
||||
* @param intent The intent to start |
|
||||
* @param requestCode The request code that is passed to callback to identify the activity |
|
||||
*/ |
|
||||
abstract public void startActivityForResult(CordovaPlugin command, Intent intent, int requestCode); |
|
||||
|
|
||||
/** |
|
||||
* Set the plugin to be called when a sub-activity exits. |
|
||||
* |
|
||||
* @param plugin The plugin on which onActivityResult is to be called |
|
||||
*/ |
|
||||
abstract public void setActivityResultCallback(CordovaPlugin plugin); |
|
||||
|
|
||||
/** |
|
||||
* Get the Android activity. |
|
||||
* |
|
||||
* @return the Activity |
|
||||
*/ |
|
||||
public abstract Activity getActivity(); |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Called when a message is sent to plugin. |
|
||||
* |
|
||||
* @param id The message id |
|
||||
* @param data The message data |
|
||||
* @return Object or null |
|
||||
*/ |
|
||||
public Object onMessage(String id, Object data); |
|
||||
|
|
||||
/** |
|
||||
* Returns a shared thread pool that can be used for background tasks. |
|
||||
*/ |
|
||||
public ExecutorService getThreadPool(); |
|
||||
|
|
||||
/** |
|
||||
* Sends a permission request to the activity for one permission. |
|
||||
*/ |
|
||||
public void requestPermission(CordovaPlugin plugin, int requestCode, String permission); |
|
||||
|
|
||||
/** |
|
||||
* Sends a permission request to the activity for a group of permissions |
|
||||
*/ |
|
||||
public void requestPermissions(CordovaPlugin plugin, int requestCode, String [] permissions); |
|
||||
|
|
||||
/** |
|
||||
* Check for a permission. Returns true if the permission is granted, false otherwise. |
|
||||
*/ |
|
||||
public boolean hasPermission(String permission); |
|
||||
|
|
||||
} |
|
@ -1,242 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.app.Activity; |
|
||||
import android.content.Intent; |
|
||||
import android.content.pm.PackageManager; |
|
||||
import android.os.Build; |
|
||||
import android.os.Bundle; |
|
||||
import android.util.Log; |
|
||||
import android.util.Pair; |
|
||||
|
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import java.util.concurrent.ExecutorService; |
|
||||
import java.util.concurrent.Executors; |
|
||||
|
|
||||
/** |
|
||||
* Default implementation of CordovaInterface. |
|
||||
*/ |
|
||||
public class CordovaInterfaceImpl implements CordovaInterface { |
|
||||
private static final String TAG = "CordovaInterfaceImpl"; |
|
||||
protected Activity activity; |
|
||||
protected ExecutorService threadPool; |
|
||||
protected PluginManager pluginManager; |
|
||||
|
|
||||
protected ActivityResultHolder savedResult; |
|
||||
protected CallbackMap permissionResultCallbacks; |
|
||||
protected CordovaPlugin activityResultCallback; |
|
||||
protected String initCallbackService; |
|
||||
protected int activityResultRequestCode; |
|
||||
protected boolean activityWasDestroyed = false; |
|
||||
protected Bundle savedPluginState; |
|
||||
|
|
||||
public CordovaInterfaceImpl(Activity activity) { |
|
||||
this(activity, Executors.newCachedThreadPool()); |
|
||||
} |
|
||||
|
|
||||
public CordovaInterfaceImpl(Activity activity, ExecutorService threadPool) { |
|
||||
this.activity = activity; |
|
||||
this.threadPool = threadPool; |
|
||||
this.permissionResultCallbacks = new CallbackMap(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void startActivityForResult(CordovaPlugin command, Intent intent, int requestCode) { |
|
||||
setActivityResultCallback(command); |
|
||||
try { |
|
||||
activity.startActivityForResult(intent, requestCode); |
|
||||
} catch (RuntimeException e) { // E.g.: ActivityNotFoundException |
|
||||
activityResultCallback = null; |
|
||||
throw e; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void setActivityResultCallback(CordovaPlugin plugin) { |
|
||||
// Cancel any previously pending activity. |
|
||||
if (activityResultCallback != null) { |
|
||||
activityResultCallback.onActivityResult(activityResultRequestCode, Activity.RESULT_CANCELED, null); |
|
||||
} |
|
||||
activityResultCallback = plugin; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public Activity getActivity() { |
|
||||
return activity; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public Object onMessage(String id, Object data) { |
|
||||
if ("exit".equals(id)) { |
|
||||
activity.finish(); |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public ExecutorService getThreadPool() { |
|
||||
return threadPool; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Dispatches any pending onActivityResult callbacks and sends the resume event if the |
|
||||
* Activity was destroyed by the OS. |
|
||||
*/ |
|
||||
public void onCordovaInit(PluginManager pluginManager) { |
|
||||
this.pluginManager = pluginManager; |
|
||||
if (savedResult != null) { |
|
||||
onActivityResult(savedResult.requestCode, savedResult.resultCode, savedResult.intent); |
|
||||
} else if(activityWasDestroyed) { |
|
||||
// If there was no Activity result, we still need to send out the resume event if the |
|
||||
// Activity was destroyed by the OS |
|
||||
activityWasDestroyed = false; |
|
||||
if(pluginManager != null) |
|
||||
{ |
|
||||
CoreAndroid appPlugin = (CoreAndroid) pluginManager.getPlugin(CoreAndroid.PLUGIN_NAME); |
|
||||
if(appPlugin != null) { |
|
||||
JSONObject obj = new JSONObject(); |
|
||||
try { |
|
||||
obj.put("action", "resume"); |
|
||||
} catch (JSONException e) { |
|
||||
LOG.e(TAG, "Failed to create event message", e); |
|
||||
} |
|
||||
appPlugin.sendResumeEvent(new PluginResult(PluginResult.Status.OK, obj)); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Routes the result to the awaiting plugin. Returns false if no plugin was waiting. |
|
||||
*/ |
|
||||
public boolean onActivityResult(int requestCode, int resultCode, Intent intent) { |
|
||||
CordovaPlugin callback = activityResultCallback; |
|
||||
if(callback == null && initCallbackService != null) { |
|
||||
// The application was restarted, but had defined an initial callback |
|
||||
// before being shut down. |
|
||||
savedResult = new ActivityResultHolder(requestCode, resultCode, intent); |
|
||||
if (pluginManager != null) { |
|
||||
callback = pluginManager.getPlugin(initCallbackService); |
|
||||
if(callback != null) { |
|
||||
callback.onRestoreStateForActivityResult(savedPluginState.getBundle(callback.getServiceName()), |
|
||||
new ResumeCallback(callback.getServiceName(), pluginManager)); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
activityResultCallback = null; |
|
||||
|
|
||||
if (callback != null) { |
|
||||
Log.d(TAG, "Sending activity result to plugin"); |
|
||||
initCallbackService = null; |
|
||||
savedResult = null; |
|
||||
callback.onActivityResult(requestCode, resultCode, intent); |
|
||||
return true; |
|
||||
} |
|
||||
Log.w(TAG, "Got an activity result, but no plugin was registered to receive it" + (savedResult != null ? " yet!" : ".")); |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Call this from your startActivityForResult() overload. This is required to catch the case |
|
||||
* where plugins use Activity.startActivityForResult() + CordovaInterface.setActivityResultCallback() |
|
||||
* rather than CordovaInterface.startActivityForResult(). |
|
||||
*/ |
|
||||
public void setActivityResultRequestCode(int requestCode) { |
|
||||
activityResultRequestCode = requestCode; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Saves parameters for startActivityForResult(). |
|
||||
*/ |
|
||||
public void onSaveInstanceState(Bundle outState) { |
|
||||
if (activityResultCallback != null) { |
|
||||
String serviceName = activityResultCallback.getServiceName(); |
|
||||
outState.putString("callbackService", serviceName); |
|
||||
} |
|
||||
if(pluginManager != null){ |
|
||||
outState.putBundle("plugin", pluginManager.onSaveInstanceState()); |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Call this from onCreate() so that any saved startActivityForResult parameters will be restored. |
|
||||
*/ |
|
||||
public void restoreInstanceState(Bundle savedInstanceState) { |
|
||||
initCallbackService = savedInstanceState.getString("callbackService"); |
|
||||
savedPluginState = savedInstanceState.getBundle("plugin"); |
|
||||
activityWasDestroyed = true; |
|
||||
} |
|
||||
|
|
||||
private static class ActivityResultHolder { |
|
||||
private int requestCode; |
|
||||
private int resultCode; |
|
||||
private Intent intent; |
|
||||
|
|
||||
public ActivityResultHolder(int requestCode, int resultCode, Intent intent) { |
|
||||
this.requestCode = requestCode; |
|
||||
this.resultCode = resultCode; |
|
||||
this.intent = intent; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the user grants permissions |
|
||||
* |
|
||||
* @param requestCode |
|
||||
* @param permissions |
|
||||
* @param grantResults |
|
||||
*/ |
|
||||
public void onRequestPermissionResult(int requestCode, String[] permissions, |
|
||||
int[] grantResults) throws JSONException { |
|
||||
Pair<CordovaPlugin, Integer> callback = permissionResultCallbacks.getAndRemoveCallback(requestCode); |
|
||||
if(callback != null) { |
|
||||
callback.first.onRequestPermissionResult(callback.second, permissions, grantResults); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void requestPermission(CordovaPlugin plugin, int requestCode, String permission) { |
|
||||
String[] permissions = new String [1]; |
|
||||
permissions[0] = permission; |
|
||||
requestPermissions(plugin, requestCode, permissions); |
|
||||
} |
|
||||
|
|
||||
public void requestPermissions(CordovaPlugin plugin, int requestCode, String [] permissions) { |
|
||||
int mappedRequestCode = permissionResultCallbacks.registerCallback(plugin, requestCode); |
|
||||
getActivity().requestPermissions(permissions, mappedRequestCode); |
|
||||
} |
|
||||
|
|
||||
public boolean hasPermission(String permission) |
|
||||
{ |
|
||||
if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) |
|
||||
{ |
|
||||
int result = activity.checkSelfPermission(permission); |
|
||||
return PackageManager.PERMISSION_GRANTED == result; |
|
||||
} |
|
||||
else |
|
||||
{ |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,422 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.apache.cordova.CordovaArgs; |
|
||||
import org.apache.cordova.CordovaWebView; |
|
||||
import org.apache.cordova.CordovaInterface; |
|
||||
import org.apache.cordova.CallbackContext; |
|
||||
import org.json.JSONArray; |
|
||||
import org.json.JSONException; |
|
||||
|
|
||||
import android.content.Intent; |
|
||||
import android.content.pm.PackageManager; |
|
||||
import android.content.res.Configuration; |
|
||||
import android.net.Uri; |
|
||||
import android.os.Build; |
|
||||
import android.os.Bundle; |
|
||||
|
|
||||
import java.io.FileNotFoundException; |
|
||||
import java.io.IOException; |
|
||||
|
|
||||
/** |
|
||||
* Plugins must extend this class and override one of the execute methods. |
|
||||
*/ |
|
||||
public class CordovaPlugin { |
|
||||
public CordovaWebView webView; |
|
||||
public CordovaInterface cordova; |
|
||||
protected CordovaPreferences preferences; |
|
||||
private String serviceName; |
|
||||
|
|
||||
/** |
|
||||
* Call this after constructing to initialize the plugin. |
|
||||
* Final because we want to be able to change args without breaking plugins. |
|
||||
*/ |
|
||||
public final void privateInitialize(String serviceName, CordovaInterface cordova, CordovaWebView webView, CordovaPreferences preferences) { |
|
||||
assert this.cordova == null; |
|
||||
this.serviceName = serviceName; |
|
||||
this.cordova = cordova; |
|
||||
this.webView = webView; |
|
||||
this.preferences = preferences; |
|
||||
initialize(cordova, webView); |
|
||||
pluginInitialize(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called after plugin construction and fields have been initialized. |
|
||||
* Prefer to use pluginInitialize instead since there is no value in |
|
||||
* having parameters on the initialize() function. |
|
||||
*/ |
|
||||
public void initialize(CordovaInterface cordova, CordovaWebView webView) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called after plugin construction and fields have been initialized. |
|
||||
*/ |
|
||||
protected void pluginInitialize() { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Returns the plugin's service name (what you'd use when calling pluginManger.getPlugin()) |
|
||||
*/ |
|
||||
public String getServiceName() { |
|
||||
return serviceName; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Executes the request. |
|
||||
* |
|
||||
* This method is called from the WebView thread. To do a non-trivial amount of work, use: |
|
||||
* cordova.getThreadPool().execute(runnable); |
|
||||
* |
|
||||
* To run on the UI thread, use: |
|
||||
* cordova.getActivity().runOnUiThread(runnable); |
|
||||
* |
|
||||
* @param action The action to execute. |
|
||||
* @param rawArgs The exec() arguments in JSON form. |
|
||||
* @param callbackContext The callback context used when calling back into JavaScript. |
|
||||
* @return Whether the action was valid. |
|
||||
*/ |
|
||||
public boolean execute(String action, String rawArgs, CallbackContext callbackContext) throws JSONException { |
|
||||
JSONArray args = new JSONArray(rawArgs); |
|
||||
return execute(action, args, callbackContext); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Executes the request. |
|
||||
* |
|
||||
* This method is called from the WebView thread. To do a non-trivial amount of work, use: |
|
||||
* cordova.getThreadPool().execute(runnable); |
|
||||
* |
|
||||
* To run on the UI thread, use: |
|
||||
* cordova.getActivity().runOnUiThread(runnable); |
|
||||
* |
|
||||
* @param action The action to execute. |
|
||||
* @param args The exec() arguments. |
|
||||
* @param callbackContext The callback context used when calling back into JavaScript. |
|
||||
* @return Whether the action was valid. |
|
||||
*/ |
|
||||
public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException { |
|
||||
CordovaArgs cordovaArgs = new CordovaArgs(args); |
|
||||
return execute(action, cordovaArgs, callbackContext); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Executes the request. |
|
||||
* |
|
||||
* This method is called from the WebView thread. To do a non-trivial amount of work, use: |
|
||||
* cordova.getThreadPool().execute(runnable); |
|
||||
* |
|
||||
* To run on the UI thread, use: |
|
||||
* cordova.getActivity().runOnUiThread(runnable); |
|
||||
* |
|
||||
* @param action The action to execute. |
|
||||
* @param args The exec() arguments, wrapped with some Cordova helpers. |
|
||||
* @param callbackContext The callback context used when calling back into JavaScript. |
|
||||
* @return Whether the action was valid. |
|
||||
*/ |
|
||||
public boolean execute(String action, CordovaArgs args, CallbackContext callbackContext) throws JSONException { |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the system is about to start resuming a previous activity. |
|
||||
* |
|
||||
* @param multitasking Flag indicating if multitasking is turned on for app |
|
||||
*/ |
|
||||
public void onPause(boolean multitasking) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity will start interacting with the user. |
|
||||
* |
|
||||
* @param multitasking Flag indicating if multitasking is turned on for app |
|
||||
*/ |
|
||||
public void onResume(boolean multitasking) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is becoming visible to the user. |
|
||||
*/ |
|
||||
public void onStart() { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is no longer visible to the user. |
|
||||
*/ |
|
||||
public void onStop() { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity receives a new intent. |
|
||||
*/ |
|
||||
public void onNewIntent(Intent intent) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* The final call you receive before your activity is destroyed. |
|
||||
*/ |
|
||||
public void onDestroy() { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the Activity is being destroyed (e.g. if a plugin calls out to an external |
|
||||
* Activity and the OS kills the CordovaActivity in the background). The plugin should save its |
|
||||
* state in this method only if it is awaiting the result of an external Activity and needs |
|
||||
* to preserve some information so as to handle that result; onRestoreStateForActivityResult() |
|
||||
* will only be called if the plugin is the recipient of an Activity result |
|
||||
* |
|
||||
* @return Bundle containing the state of the plugin or null if state does not need to be saved |
|
||||
*/ |
|
||||
public Bundle onSaveInstanceState() { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when a plugin is the recipient of an Activity result after the CordovaActivity has |
|
||||
* been destroyed. The Bundle will be the same as the one the plugin returned in |
|
||||
* onSaveInstanceState() |
|
||||
* |
|
||||
* @param state Bundle containing the state of the plugin |
|
||||
* @param callbackContext Replacement Context to return the plugin result to |
|
||||
*/ |
|
||||
public void onRestoreStateForActivityResult(Bundle state, CallbackContext callbackContext) {} |
|
||||
|
|
||||
/** |
|
||||
* Called when a message is sent to plugin. |
|
||||
* |
|
||||
* @param id The message id |
|
||||
* @param data The message data |
|
||||
* @return Object to stop propagation or null |
|
||||
*/ |
|
||||
public Object onMessage(String id, Object data) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when an activity you launched exits, giving you the requestCode you started it with, |
|
||||
* the resultCode it returned, and any additional data from it. |
|
||||
* |
|
||||
* @param requestCode The request code originally supplied to startActivityForResult(), |
|
||||
* allowing you to identify who this result came from. |
|
||||
* @param resultCode The integer result code returned by the child activity through its setResult(). |
|
||||
* @param intent An Intent, which can return result data to the caller (various data can be |
|
||||
* attached to Intent "extras"). |
|
||||
*/ |
|
||||
public void onActivityResult(int requestCode, int resultCode, Intent intent) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Hook for blocking the loading of external resources. |
|
||||
* |
|
||||
* This will be called when the WebView's shouldInterceptRequest wants to |
|
||||
* know whether to open a connection to an external resource. Return false |
|
||||
* to block the request: if any plugin returns false, Cordova will block |
|
||||
* the request. If all plugins return null, the default policy will be |
|
||||
* enforced. If at least one plugin returns true, and no plugins return |
|
||||
* false, then the request will proceed. |
|
||||
* |
|
||||
* Note that this only affects resource requests which are routed through |
|
||||
* WebViewClient.shouldInterceptRequest, such as XMLHttpRequest requests and |
|
||||
* img tag loads. WebSockets and media requests (such as <video> and <audio> |
|
||||
* tags) are not affected by this method. Use CSP headers to control access |
|
||||
* to such resources. |
|
||||
*/ |
|
||||
public Boolean shouldAllowRequest(String url) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Hook for blocking navigation by the Cordova WebView. This applies both to top-level and |
|
||||
* iframe navigations. |
|
||||
* |
|
||||
* This will be called when the WebView's needs to know whether to navigate |
|
||||
* to a new page. Return false to block the navigation: if any plugin |
|
||||
* returns false, Cordova will block the navigation. If all plugins return |
|
||||
* null, the default policy will be enforced. It at least one plugin returns |
|
||||
* true, and no plugins return false, then the navigation will proceed. |
|
||||
*/ |
|
||||
public Boolean shouldAllowNavigation(String url) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Hook for allowing page to call exec(). By default, this returns the result of |
|
||||
* shouldAllowNavigation(). It's generally unsafe to allow untrusted content to be loaded |
|
||||
* into a CordovaWebView, even within an iframe, so it's best not to touch this. |
|
||||
*/ |
|
||||
public Boolean shouldAllowBridgeAccess(String url) { |
|
||||
return shouldAllowNavigation(url); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Hook for blocking the launching of Intents by the Cordova application. |
|
||||
* |
|
||||
* This will be called when the WebView will not navigate to a page, but |
|
||||
* could launch an intent to handle the URL. Return false to block this: if |
|
||||
* any plugin returns false, Cordova will block the navigation. If all |
|
||||
* plugins return null, the default policy will be enforced. If at least one |
|
||||
* plugin returns true, and no plugins return false, then the URL will be |
|
||||
* opened. |
|
||||
*/ |
|
||||
public Boolean shouldOpenExternalUrl(String url) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Allows plugins to handle a link being clicked. Return true here to cancel the navigation. |
|
||||
* |
|
||||
* @param url The URL that is trying to be loaded in the Cordova webview. |
|
||||
* @return Return true to prevent the URL from loading. Default is false. |
|
||||
*/ |
|
||||
public boolean onOverrideUrlLoading(String url) { |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Hook for redirecting requests. Applies to WebView requests as well as requests made by plugins. |
|
||||
* To handle the request directly, return a URI in the form: |
|
||||
* |
|
||||
* cdvplugin://pluginId/... |
|
||||
* |
|
||||
* And implement handleOpenForRead(). |
|
||||
* To make this easier, use the toPluginUri() and fromPluginUri() helpers: |
|
||||
* |
|
||||
* public Uri remapUri(Uri uri) { return toPluginUri(uri); } |
|
||||
* |
|
||||
* public CordovaResourceApi.OpenForReadResult handleOpenForRead(Uri uri) throws IOException { |
|
||||
* Uri origUri = fromPluginUri(uri); |
|
||||
* ... |
|
||||
* } |
|
||||
*/ |
|
||||
public Uri remapUri(Uri uri) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called to handle CordovaResourceApi.openForRead() calls for a cdvplugin://pluginId/ URL. |
|
||||
* Should never return null. |
|
||||
* Added in cordova-android@4.0.0 |
|
||||
*/ |
|
||||
public CordovaResourceApi.OpenForReadResult handleOpenForRead(Uri uri) throws IOException { |
|
||||
throw new FileNotFoundException("Plugin can't handle uri: " + uri); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Refer to remapUri() |
|
||||
* Added in cordova-android@4.0.0 |
|
||||
*/ |
|
||||
protected Uri toPluginUri(Uri origUri) { |
|
||||
return new Uri.Builder() |
|
||||
.scheme(CordovaResourceApi.PLUGIN_URI_SCHEME) |
|
||||
.authority(serviceName) |
|
||||
.appendQueryParameter("origUri", origUri.toString()) |
|
||||
.build(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Refer to remapUri() |
|
||||
* Added in cordova-android@4.0.0 |
|
||||
*/ |
|
||||
protected Uri fromPluginUri(Uri pluginUri) { |
|
||||
return Uri.parse(pluginUri.getQueryParameter("origUri")); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the WebView does a top-level navigation or refreshes. |
|
||||
* |
|
||||
* Plugins should stop any long-running processes and clean up internal state. |
|
||||
* |
|
||||
* Does nothing by default. |
|
||||
*/ |
|
||||
public void onReset() { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the system received an HTTP authentication request. Plugin can use |
|
||||
* the supplied HttpAuthHandler to process this auth challenge. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback |
|
||||
* @param handler The HttpAuthHandler used to set the WebView's response |
|
||||
* @param host The host requiring authentication |
|
||||
* @param realm The realm for which authentication is required |
|
||||
* |
|
||||
* @return Returns True if plugin will resolve this auth challenge, otherwise False |
|
||||
* |
|
||||
*/ |
|
||||
public boolean onReceivedHttpAuthRequest(CordovaWebView view, ICordovaHttpAuthHandler handler, String host, String realm) { |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when he system received an SSL client certificate request. Plugin can use |
|
||||
* the supplied ClientCertRequest to process this certificate challenge. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback |
|
||||
* @param request The client certificate request |
|
||||
* |
|
||||
* @return Returns True if plugin will resolve this auth challenge, otherwise False |
|
||||
* |
|
||||
*/ |
|
||||
public boolean onReceivedClientCertRequest(CordovaWebView view, ICordovaClientCertRequest request) { |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the device configuration changes while your activity is running. |
|
||||
* |
|
||||
* @param newConfig The new device configuration |
|
||||
*/ |
|
||||
public void onConfigurationChanged(Configuration newConfig) { |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the Plugin Manager when we need to actually request permissions |
|
||||
* |
|
||||
* @param requestCode Passed to the activity to track the request |
|
||||
* |
|
||||
* @return Returns the permission that was stored in the plugin |
|
||||
*/ |
|
||||
|
|
||||
public void requestPermissions(int requestCode) { |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Called by the WebView implementation to check for geolocation permissions, can be used |
|
||||
* by other Java methods in the event that a plugin is using this as a dependency. |
|
||||
* |
|
||||
* @return Returns true if the plugin has all the permissions it needs to operate. |
|
||||
*/ |
|
||||
|
|
||||
public boolean hasPermisssion() { |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the user grants permissions |
|
||||
* |
|
||||
* @param requestCode |
|
||||
* @param permissions |
|
||||
* @param grantResults |
|
||||
*/ |
|
||||
public void onRequestPermissionResult(int requestCode, String[] permissions, |
|
||||
int[] grantResults) throws JSONException { |
|
||||
|
|
||||
} |
|
||||
} |
|
@ -1,101 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.HashMap; |
|
||||
import java.util.Locale; |
|
||||
import java.util.Map; |
|
||||
|
|
||||
import org.apache.cordova.LOG; |
|
||||
|
|
||||
import android.app.Activity; |
|
||||
import android.os.Bundle; |
|
||||
|
|
||||
public class CordovaPreferences { |
|
||||
private HashMap<String, String> prefs = new HashMap<String, String>(20); |
|
||||
private Bundle preferencesBundleExtras; |
|
||||
|
|
||||
public void setPreferencesBundle(Bundle extras) { |
|
||||
preferencesBundleExtras = extras; |
|
||||
} |
|
||||
|
|
||||
public void set(String name, String value) { |
|
||||
prefs.put(name.toLowerCase(Locale.ENGLISH), value); |
|
||||
} |
|
||||
|
|
||||
public void set(String name, boolean value) { |
|
||||
set(name, "" + value); |
|
||||
} |
|
||||
|
|
||||
public void set(String name, int value) { |
|
||||
set(name, "" + value); |
|
||||
} |
|
||||
|
|
||||
public void set(String name, double value) { |
|
||||
set(name, "" + value); |
|
||||
} |
|
||||
|
|
||||
public Map<String, String> getAll() { |
|
||||
return prefs; |
|
||||
} |
|
||||
|
|
||||
public boolean getBoolean(String name, boolean defaultValue) { |
|
||||
name = name.toLowerCase(Locale.ENGLISH); |
|
||||
String value = prefs.get(name); |
|
||||
if (value != null) { |
|
||||
return Boolean.parseBoolean(value); |
|
||||
} |
|
||||
return defaultValue; |
|
||||
} |
|
||||
|
|
||||
// Added in 4.0.0 |
|
||||
public boolean contains(String name) { |
|
||||
return getString(name, null) != null; |
|
||||
} |
|
||||
|
|
||||
public int getInteger(String name, int defaultValue) { |
|
||||
name = name.toLowerCase(Locale.ENGLISH); |
|
||||
String value = prefs.get(name); |
|
||||
if (value != null) { |
|
||||
// Use Integer.decode() can't handle it if the highest bit is set. |
|
||||
return (int)(long)Long.decode(value); |
|
||||
} |
|
||||
return defaultValue; |
|
||||
} |
|
||||
|
|
||||
public double getDouble(String name, double defaultValue) { |
|
||||
name = name.toLowerCase(Locale.ENGLISH); |
|
||||
String value = prefs.get(name); |
|
||||
if (value != null) { |
|
||||
return Double.valueOf(value); |
|
||||
} |
|
||||
return defaultValue; |
|
||||
} |
|
||||
|
|
||||
public String getString(String name, String defaultValue) { |
|
||||
name = name.toLowerCase(Locale.ENGLISH); |
|
||||
String value = prefs.get(name); |
|
||||
if (value != null) { |
|
||||
return value; |
|
||||
} |
|
||||
return defaultValue; |
|
||||
} |
|
||||
|
|
||||
} |
|
@ -1,471 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.content.ContentResolver; |
|
||||
import android.content.Context; |
|
||||
import android.content.res.AssetFileDescriptor; |
|
||||
import android.content.res.AssetManager; |
|
||||
import android.database.Cursor; |
|
||||
import android.net.Uri; |
|
||||
import android.os.Looper; |
|
||||
import android.util.Base64; |
|
||||
import android.webkit.MimeTypeMap; |
|
||||
|
|
||||
import java.io.ByteArrayInputStream; |
|
||||
import java.io.File; |
|
||||
import java.io.FileInputStream; |
|
||||
import java.io.FileNotFoundException; |
|
||||
import java.io.FileOutputStream; |
|
||||
import java.io.IOException; |
|
||||
import java.io.InputStream; |
|
||||
import java.io.OutputStream; |
|
||||
import java.io.UnsupportedEncodingException; |
|
||||
import java.net.HttpURLConnection; |
|
||||
import java.net.URL; |
|
||||
import java.nio.channels.FileChannel; |
|
||||
import java.util.Locale; |
|
||||
|
|
||||
/** |
|
||||
* What this class provides: |
|
||||
* 1. Helpers for reading & writing to URLs. |
|
||||
* - E.g. handles assets, resources, content providers, files, data URIs, http[s] |
|
||||
* - E.g. Can be used to query for mime-type & content length. |
|
||||
* |
|
||||
* 2. To allow plugins to redirect URLs (via remapUrl). |
|
||||
* - All plugins should call remapUrl() on URLs they receive from JS *before* |
|
||||
* passing the URL onto other utility functions in this class. |
|
||||
* - For an example usage of this, refer to the org.apache.cordova.file plugin. |
|
||||
* |
|
||||
* Future Work: |
|
||||
* - Consider using a Cursor to query content URLs for their size (like the file plugin does). |
|
||||
* - Allow plugins to remapUri to "cdv-plugin://plugin-name/foo", which CordovaResourceApi |
|
||||
* would then delegate to pluginManager.getPlugin(plugin-name).openForRead(url) |
|
||||
* - Currently, plugins *can* do this by remapping to a data: URL, but it's inefficient |
|
||||
* for large payloads. |
|
||||
*/ |
|
||||
public class CordovaResourceApi { |
|
||||
@SuppressWarnings("unused") |
|
||||
private static final String LOG_TAG = "CordovaResourceApi"; |
|
||||
|
|
||||
public static final int URI_TYPE_FILE = 0; |
|
||||
public static final int URI_TYPE_ASSET = 1; |
|
||||
public static final int URI_TYPE_CONTENT = 2; |
|
||||
public static final int URI_TYPE_RESOURCE = 3; |
|
||||
public static final int URI_TYPE_DATA = 4; |
|
||||
public static final int URI_TYPE_HTTP = 5; |
|
||||
public static final int URI_TYPE_HTTPS = 6; |
|
||||
public static final int URI_TYPE_PLUGIN = 7; |
|
||||
public static final int URI_TYPE_UNKNOWN = -1; |
|
||||
|
|
||||
public static final String PLUGIN_URI_SCHEME = "cdvplugin"; |
|
||||
|
|
||||
private static final String[] LOCAL_FILE_PROJECTION = { "_data" }; |
|
||||
|
|
||||
public static Thread jsThread; |
|
||||
|
|
||||
private final AssetManager assetManager; |
|
||||
private final ContentResolver contentResolver; |
|
||||
private final PluginManager pluginManager; |
|
||||
private boolean threadCheckingEnabled = true; |
|
||||
|
|
||||
|
|
||||
public CordovaResourceApi(Context context, PluginManager pluginManager) { |
|
||||
this.contentResolver = context.getContentResolver(); |
|
||||
this.assetManager = context.getAssets(); |
|
||||
this.pluginManager = pluginManager; |
|
||||
} |
|
||||
|
|
||||
public void setThreadCheckingEnabled(boolean value) { |
|
||||
threadCheckingEnabled = value; |
|
||||
} |
|
||||
|
|
||||
public boolean isThreadCheckingEnabled() { |
|
||||
return threadCheckingEnabled; |
|
||||
} |
|
||||
|
|
||||
|
|
||||
public static int getUriType(Uri uri) { |
|
||||
assertNonRelative(uri); |
|
||||
String scheme = uri.getScheme(); |
|
||||
if (ContentResolver.SCHEME_CONTENT.equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_CONTENT; |
|
||||
} |
|
||||
if (ContentResolver.SCHEME_ANDROID_RESOURCE.equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_RESOURCE; |
|
||||
} |
|
||||
if (ContentResolver.SCHEME_FILE.equalsIgnoreCase(scheme)) { |
|
||||
if (uri.getPath().startsWith("/android_asset/")) { |
|
||||
return URI_TYPE_ASSET; |
|
||||
} |
|
||||
return URI_TYPE_FILE; |
|
||||
} |
|
||||
if ("data".equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_DATA; |
|
||||
} |
|
||||
if ("http".equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_HTTP; |
|
||||
} |
|
||||
if ("https".equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_HTTPS; |
|
||||
} |
|
||||
if (PLUGIN_URI_SCHEME.equalsIgnoreCase(scheme)) { |
|
||||
return URI_TYPE_PLUGIN; |
|
||||
} |
|
||||
return URI_TYPE_UNKNOWN; |
|
||||
} |
|
||||
|
|
||||
public Uri remapUri(Uri uri) { |
|
||||
assertNonRelative(uri); |
|
||||
Uri pluginUri = pluginManager.remapUri(uri); |
|
||||
return pluginUri != null ? pluginUri : uri; |
|
||||
} |
|
||||
|
|
||||
public String remapPath(String path) { |
|
||||
return remapUri(Uri.fromFile(new File(path))).getPath(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Returns a File that points to the resource, or null if the resource |
|
||||
* is not on the local filesystem. |
|
||||
*/ |
|
||||
public File mapUriToFile(Uri uri) { |
|
||||
assertBackgroundThread(); |
|
||||
switch (getUriType(uri)) { |
|
||||
case URI_TYPE_FILE: |
|
||||
return new File(uri.getPath()); |
|
||||
case URI_TYPE_CONTENT: { |
|
||||
Cursor cursor = contentResolver.query(uri, LOCAL_FILE_PROJECTION, null, null, null); |
|
||||
if (cursor != null) { |
|
||||
try { |
|
||||
int columnIndex = cursor.getColumnIndex(LOCAL_FILE_PROJECTION[0]); |
|
||||
if (columnIndex != -1 && cursor.getCount() > 0) { |
|
||||
cursor.moveToFirst(); |
|
||||
String realPath = cursor.getString(columnIndex); |
|
||||
if (realPath != null) { |
|
||||
return new File(realPath); |
|
||||
} |
|
||||
} |
|
||||
} finally { |
|
||||
cursor.close(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
public String getMimeType(Uri uri) { |
|
||||
switch (getUriType(uri)) { |
|
||||
case URI_TYPE_FILE: |
|
||||
case URI_TYPE_ASSET: |
|
||||
return getMimeTypeFromPath(uri.getPath()); |
|
||||
case URI_TYPE_CONTENT: |
|
||||
case URI_TYPE_RESOURCE: |
|
||||
return contentResolver.getType(uri); |
|
||||
case URI_TYPE_DATA: { |
|
||||
return getDataUriMimeType(uri); |
|
||||
} |
|
||||
case URI_TYPE_HTTP: |
|
||||
case URI_TYPE_HTTPS: { |
|
||||
try { |
|
||||
HttpURLConnection conn = (HttpURLConnection)new URL(uri.toString()).openConnection(); |
|
||||
conn.setDoInput(false); |
|
||||
conn.setRequestMethod("HEAD"); |
|
||||
String mimeType = conn.getHeaderField("Content-Type"); |
|
||||
if (mimeType != null) { |
|
||||
mimeType = mimeType.split(";")[0]; |
|
||||
} |
|
||||
return mimeType; |
|
||||
} catch (IOException e) { |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//This already exists |
|
||||
private String getMimeTypeFromPath(String path) { |
|
||||
String extension = path; |
|
||||
int lastDot = extension.lastIndexOf('.'); |
|
||||
if (lastDot != -1) { |
|
||||
extension = extension.substring(lastDot + 1); |
|
||||
} |
|
||||
// Convert the URI string to lower case to ensure compatibility with MimeTypeMap (see CB-2185). |
|
||||
extension = extension.toLowerCase(Locale.getDefault()); |
|
||||
if (extension.equals("3ga")) { |
|
||||
return "audio/3gpp"; |
|
||||
} else if (extension.equals("js")) { |
|
||||
// Missing from the map :(. |
|
||||
return "text/javascript"; |
|
||||
} |
|
||||
return MimeTypeMap.getSingleton().getMimeTypeFromExtension(extension); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Opens a stream to the given URI, also providing the MIME type & length. |
|
||||
* @return Never returns null. |
|
||||
* @throws Throws an InvalidArgumentException for relative URIs. Relative URIs should be |
|
||||
* resolved before being passed into this function. |
|
||||
* @throws Throws an IOException if the URI cannot be opened. |
|
||||
* @throws Throws an IllegalStateException if called on a foreground thread. |
|
||||
*/ |
|
||||
public OpenForReadResult openForRead(Uri uri) throws IOException { |
|
||||
return openForRead(uri, false); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Opens a stream to the given URI, also providing the MIME type & length. |
|
||||
* @return Never returns null. |
|
||||
* @throws Throws an InvalidArgumentException for relative URIs. Relative URIs should be |
|
||||
* resolved before being passed into this function. |
|
||||
* @throws Throws an IOException if the URI cannot be opened. |
|
||||
* @throws Throws an IllegalStateException if called on a foreground thread and skipThreadCheck is false. |
|
||||
*/ |
|
||||
public OpenForReadResult openForRead(Uri uri, boolean skipThreadCheck) throws IOException { |
|
||||
if (!skipThreadCheck) { |
|
||||
assertBackgroundThread(); |
|
||||
} |
|
||||
switch (getUriType(uri)) { |
|
||||
case URI_TYPE_FILE: { |
|
||||
FileInputStream inputStream = new FileInputStream(uri.getPath()); |
|
||||
String mimeType = getMimeTypeFromPath(uri.getPath()); |
|
||||
long length = inputStream.getChannel().size(); |
|
||||
return new OpenForReadResult(uri, inputStream, mimeType, length, null); |
|
||||
} |
|
||||
case URI_TYPE_ASSET: { |
|
||||
String assetPath = uri.getPath().substring(15); |
|
||||
AssetFileDescriptor assetFd = null; |
|
||||
InputStream inputStream; |
|
||||
long length = -1; |
|
||||
try { |
|
||||
assetFd = assetManager.openFd(assetPath); |
|
||||
inputStream = assetFd.createInputStream(); |
|
||||
length = assetFd.getLength(); |
|
||||
} catch (FileNotFoundException e) { |
|
||||
// Will occur if the file is compressed. |
|
||||
inputStream = assetManager.open(assetPath); |
|
||||
} |
|
||||
String mimeType = getMimeTypeFromPath(assetPath); |
|
||||
return new OpenForReadResult(uri, inputStream, mimeType, length, assetFd); |
|
||||
} |
|
||||
case URI_TYPE_CONTENT: |
|
||||
case URI_TYPE_RESOURCE: { |
|
||||
String mimeType = contentResolver.getType(uri); |
|
||||
AssetFileDescriptor assetFd = contentResolver.openAssetFileDescriptor(uri, "r"); |
|
||||
InputStream inputStream = assetFd.createInputStream(); |
|
||||
long length = assetFd.getLength(); |
|
||||
return new OpenForReadResult(uri, inputStream, mimeType, length, assetFd); |
|
||||
} |
|
||||
case URI_TYPE_DATA: { |
|
||||
OpenForReadResult ret = readDataUri(uri); |
|
||||
if (ret == null) { |
|
||||
break; |
|
||||
} |
|
||||
return ret; |
|
||||
} |
|
||||
case URI_TYPE_HTTP: |
|
||||
case URI_TYPE_HTTPS: { |
|
||||
HttpURLConnection conn = (HttpURLConnection)new URL(uri.toString()).openConnection(); |
|
||||
conn.setDoInput(true); |
|
||||
String mimeType = conn.getHeaderField("Content-Type"); |
|
||||
if (mimeType != null) { |
|
||||
mimeType = mimeType.split(";")[0]; |
|
||||
} |
|
||||
int length = conn.getContentLength(); |
|
||||
InputStream inputStream = conn.getInputStream(); |
|
||||
return new OpenForReadResult(uri, inputStream, mimeType, length, null); |
|
||||
} |
|
||||
case URI_TYPE_PLUGIN: { |
|
||||
String pluginId = uri.getHost(); |
|
||||
CordovaPlugin plugin = pluginManager.getPlugin(pluginId); |
|
||||
if (plugin == null) { |
|
||||
throw new FileNotFoundException("Invalid plugin ID in URI: " + uri); |
|
||||
} |
|
||||
return plugin.handleOpenForRead(uri); |
|
||||
} |
|
||||
} |
|
||||
throw new FileNotFoundException("URI not supported by CordovaResourceApi: " + uri); |
|
||||
} |
|
||||
|
|
||||
public OutputStream openOutputStream(Uri uri) throws IOException { |
|
||||
return openOutputStream(uri, false); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Opens a stream to the given URI. |
|
||||
* @return Never returns null. |
|
||||
* @throws Throws an InvalidArgumentException for relative URIs. Relative URIs should be |
|
||||
* resolved before being passed into this function. |
|
||||
* @throws Throws an IOException if the URI cannot be opened. |
|
||||
*/ |
|
||||
public OutputStream openOutputStream(Uri uri, boolean append) throws IOException { |
|
||||
assertBackgroundThread(); |
|
||||
switch (getUriType(uri)) { |
|
||||
case URI_TYPE_FILE: { |
|
||||
File localFile = new File(uri.getPath()); |
|
||||
File parent = localFile.getParentFile(); |
|
||||
if (parent != null) { |
|
||||
parent.mkdirs(); |
|
||||
} |
|
||||
return new FileOutputStream(localFile, append); |
|
||||
} |
|
||||
case URI_TYPE_CONTENT: |
|
||||
case URI_TYPE_RESOURCE: { |
|
||||
AssetFileDescriptor assetFd = contentResolver.openAssetFileDescriptor(uri, append ? "wa" : "w"); |
|
||||
return assetFd.createOutputStream(); |
|
||||
} |
|
||||
} |
|
||||
throw new FileNotFoundException("URI not supported by CordovaResourceApi: " + uri); |
|
||||
} |
|
||||
|
|
||||
public HttpURLConnection createHttpConnection(Uri uri) throws IOException { |
|
||||
assertBackgroundThread(); |
|
||||
return (HttpURLConnection)new URL(uri.toString()).openConnection(); |
|
||||
} |
|
||||
|
|
||||
// Copies the input to the output in the most efficient manner possible. |
|
||||
// Closes both streams. |
|
||||
public void copyResource(OpenForReadResult input, OutputStream outputStream) throws IOException { |
|
||||
assertBackgroundThread(); |
|
||||
try { |
|
||||
InputStream inputStream = input.inputStream; |
|
||||
if (inputStream instanceof FileInputStream && outputStream instanceof FileOutputStream) { |
|
||||
FileChannel inChannel = ((FileInputStream)input.inputStream).getChannel(); |
|
||||
FileChannel outChannel = ((FileOutputStream)outputStream).getChannel(); |
|
||||
long offset = 0; |
|
||||
long length = input.length; |
|
||||
if (input.assetFd != null) { |
|
||||
offset = input.assetFd.getStartOffset(); |
|
||||
} |
|
||||
// transferFrom()'s 2nd arg is a relative position. Need to set the absolute |
|
||||
// position first. |
|
||||
inChannel.position(offset); |
|
||||
outChannel.transferFrom(inChannel, 0, length); |
|
||||
} else { |
|
||||
final int BUFFER_SIZE = 8192; |
|
||||
byte[] buffer = new byte[BUFFER_SIZE]; |
|
||||
|
|
||||
for (;;) { |
|
||||
int bytesRead = inputStream.read(buffer, 0, BUFFER_SIZE); |
|
||||
|
|
||||
if (bytesRead <= 0) { |
|
||||
break; |
|
||||
} |
|
||||
outputStream.write(buffer, 0, bytesRead); |
|
||||
} |
|
||||
} |
|
||||
} finally { |
|
||||
input.inputStream.close(); |
|
||||
if (outputStream != null) { |
|
||||
outputStream.close(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void copyResource(Uri sourceUri, OutputStream outputStream) throws IOException { |
|
||||
copyResource(openForRead(sourceUri), outputStream); |
|
||||
} |
|
||||
|
|
||||
// Added in 3.5.0. |
|
||||
public void copyResource(Uri sourceUri, Uri dstUri) throws IOException { |
|
||||
copyResource(openForRead(sourceUri), openOutputStream(dstUri)); |
|
||||
} |
|
||||
|
|
||||
private void assertBackgroundThread() { |
|
||||
if (threadCheckingEnabled) { |
|
||||
Thread curThread = Thread.currentThread(); |
|
||||
if (curThread == Looper.getMainLooper().getThread()) { |
|
||||
throw new IllegalStateException("Do not perform IO operations on the UI thread. Use CordovaInterface.getThreadPool() instead."); |
|
||||
} |
|
||||
if (curThread == jsThread) { |
|
||||
throw new IllegalStateException("Tried to perform an IO operation on the WebCore thread. Use CordovaInterface.getThreadPool() instead."); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private String getDataUriMimeType(Uri uri) { |
|
||||
String uriAsString = uri.getSchemeSpecificPart(); |
|
||||
int commaPos = uriAsString.indexOf(','); |
|
||||
if (commaPos == -1) { |
|
||||
return null; |
|
||||
} |
|
||||
String[] mimeParts = uriAsString.substring(0, commaPos).split(";"); |
|
||||
if (mimeParts.length > 0) { |
|
||||
return mimeParts[0]; |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
private OpenForReadResult readDataUri(Uri uri) { |
|
||||
String uriAsString = uri.getSchemeSpecificPart(); |
|
||||
int commaPos = uriAsString.indexOf(','); |
|
||||
if (commaPos == -1) { |
|
||||
return null; |
|
||||
} |
|
||||
String[] mimeParts = uriAsString.substring(0, commaPos).split(";"); |
|
||||
String contentType = null; |
|
||||
boolean base64 = false; |
|
||||
if (mimeParts.length > 0) { |
|
||||
contentType = mimeParts[0]; |
|
||||
} |
|
||||
for (int i = 1; i < mimeParts.length; ++i) { |
|
||||
if ("base64".equalsIgnoreCase(mimeParts[i])) { |
|
||||
base64 = true; |
|
||||
} |
|
||||
} |
|
||||
String dataPartAsString = uriAsString.substring(commaPos + 1); |
|
||||
byte[] data; |
|
||||
if (base64) { |
|
||||
data = Base64.decode(dataPartAsString, Base64.DEFAULT); |
|
||||
} else { |
|
||||
try { |
|
||||
data = dataPartAsString.getBytes("UTF-8"); |
|
||||
} catch (UnsupportedEncodingException e) { |
|
||||
data = dataPartAsString.getBytes(); |
|
||||
} |
|
||||
} |
|
||||
InputStream inputStream = new ByteArrayInputStream(data); |
|
||||
return new OpenForReadResult(uri, inputStream, contentType, data.length, null); |
|
||||
} |
|
||||
|
|
||||
private static void assertNonRelative(Uri uri) { |
|
||||
if (!uri.isAbsolute()) { |
|
||||
throw new IllegalArgumentException("Relative URIs are not supported."); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public static final class OpenForReadResult { |
|
||||
public final Uri uri; |
|
||||
public final InputStream inputStream; |
|
||||
public final String mimeType; |
|
||||
public final long length; |
|
||||
public final AssetFileDescriptor assetFd; |
|
||||
|
|
||||
public OpenForReadResult(Uri uri, InputStream inputStream, String mimeType, long length, AssetFileDescriptor assetFd) { |
|
||||
this.uri = uri; |
|
||||
this.inputStream = inputStream; |
|
||||
this.mimeType = mimeType; |
|
||||
this.length = length; |
|
||||
this.assetFd = assetFd; |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,142 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.List; |
|
||||
import java.util.Map; |
|
||||
|
|
||||
import android.content.Context; |
|
||||
import android.content.Intent; |
|
||||
import android.view.View; |
|
||||
import android.webkit.WebChromeClient.CustomViewCallback; |
|
||||
|
|
||||
/** |
|
||||
* Main interface for interacting with a Cordova webview - implemented by CordovaWebViewImpl. |
|
||||
* This is an interface so that it can be easily mocked in tests. |
|
||||
* Methods may be added to this interface without a major version bump, as plugins & embedders |
|
||||
* are not expected to implement it. |
|
||||
*/ |
|
||||
public interface CordovaWebView { |
|
||||
public static final String CORDOVA_VERSION = "5.2.2"; |
|
||||
|
|
||||
void init(CordovaInterface cordova, List<PluginEntry> pluginEntries, CordovaPreferences preferences); |
|
||||
|
|
||||
boolean isInitialized(); |
|
||||
|
|
||||
View getView(); |
|
||||
|
|
||||
void loadUrlIntoView(String url, boolean recreatePlugins); |
|
||||
|
|
||||
void stopLoading(); |
|
||||
|
|
||||
boolean canGoBack(); |
|
||||
|
|
||||
void clearCache(); |
|
||||
|
|
||||
/** Use parameter-less overload */ |
|
||||
@Deprecated |
|
||||
void clearCache(boolean b); |
|
||||
|
|
||||
void clearHistory(); |
|
||||
|
|
||||
boolean backHistory(); |
|
||||
|
|
||||
void handlePause(boolean keepRunning); |
|
||||
|
|
||||
void onNewIntent(Intent intent); |
|
||||
|
|
||||
void handleResume(boolean keepRunning); |
|
||||
|
|
||||
void handleStart(); |
|
||||
|
|
||||
void handleStop(); |
|
||||
|
|
||||
void handleDestroy(); |
|
||||
|
|
||||
/** |
|
||||
* Send JavaScript statement back to JavaScript. |
|
||||
* |
|
||||
* Deprecated (https://issues.apache.org/jira/browse/CB-6851) |
|
||||
* Instead of executing snippets of JS, you should use the exec bridge |
|
||||
* to create a Java->JS communication channel. |
|
||||
* To do this: |
|
||||
* 1. Within plugin.xml (to have your JS run before deviceready): |
|
||||
* <js-module><runs/></js-module> |
|
||||
* 2. Within your .js (call exec on start-up): |
|
||||
* require('cordova/channel').onCordovaReady.subscribe(function() { |
|
||||
* require('cordova/exec')(win, null, 'Plugin', 'method', []); |
|
||||
* function win(message) { |
|
||||
* ... process message from java here ... |
|
||||
* } |
|
||||
* }); |
|
||||
* 3. Within your .java: |
|
||||
* PluginResult dataResult = new PluginResult(PluginResult.Status.OK, CODE); |
|
||||
* dataResult.setKeepCallback(true); |
|
||||
* savedCallbackContext.sendPluginResult(dataResult); |
|
||||
*/ |
|
||||
@Deprecated |
|
||||
void sendJavascript(String statememt); |
|
||||
|
|
||||
/** |
|
||||
* Load the specified URL in the Cordova webview or a new browser instance. |
|
||||
* |
|
||||
* NOTE: If openExternal is false, only whitelisted URLs can be loaded. |
|
||||
* |
|
||||
* @param url The url to load. |
|
||||
* @param openExternal Load url in browser instead of Cordova webview. |
|
||||
* @param clearHistory Clear the history stack, so new page becomes top of history |
|
||||
* @param params Parameters for new app |
|
||||
*/ |
|
||||
void showWebPage(String url, boolean openExternal, boolean clearHistory, Map<String, Object> params); |
|
||||
|
|
||||
/** |
|
||||
* Deprecated in 4.0.0. Use your own View-toggling logic. |
|
||||
*/ |
|
||||
@Deprecated |
|
||||
boolean isCustomViewShowing(); |
|
||||
|
|
||||
/** |
|
||||
* Deprecated in 4.0.0. Use your own View-toggling logic. |
|
||||
*/ |
|
||||
@Deprecated |
|
||||
void showCustomView(View view, CustomViewCallback callback); |
|
||||
|
|
||||
/** |
|
||||
* Deprecated in 4.0.0. Use your own View-toggling logic. |
|
||||
*/ |
|
||||
@Deprecated |
|
||||
void hideCustomView(); |
|
||||
|
|
||||
CordovaResourceApi getResourceApi(); |
|
||||
|
|
||||
void setButtonPlumbedToJs(int keyCode, boolean override); |
|
||||
boolean isButtonPlumbedToJs(int keyCode); |
|
||||
|
|
||||
void sendPluginResult(PluginResult cr, String callbackId); |
|
||||
|
|
||||
PluginManager getPluginManager(); |
|
||||
CordovaWebViewEngine getEngine(); |
|
||||
CordovaPreferences getPreferences(); |
|
||||
ICordovaCookieManager getCookieManager(); |
|
||||
|
|
||||
String getUrl(); |
|
||||
|
|
||||
// TODO: Work on deleting these by removing refs from plugins. |
|
||||
Context getContext(); |
|
||||
void loadUrl(String url); |
|
||||
Object postMessage(String id, Object data); |
|
||||
} |
|
@ -1,81 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.view.KeyEvent; |
|
||||
import android.view.View; |
|
||||
|
|
||||
/** |
|
||||
* Interface for all Cordova engines. |
|
||||
* No methods will be added to this class (in order to be compatible with existing engines). |
|
||||
* Instead, we will create a new interface: e.g. CordovaWebViewEngineV2 |
|
||||
*/ |
|
||||
public interface CordovaWebViewEngine { |
|
||||
void init(CordovaWebView parentWebView, CordovaInterface cordova, Client client, |
|
||||
CordovaResourceApi resourceApi, PluginManager pluginManager, |
|
||||
NativeToJsMessageQueue nativeToJsMessageQueue); |
|
||||
|
|
||||
CordovaWebView getCordovaWebView(); |
|
||||
ICordovaCookieManager getCookieManager(); |
|
||||
View getView(); |
|
||||
|
|
||||
void loadUrl(String url, boolean clearNavigationStack); |
|
||||
|
|
||||
void stopLoading(); |
|
||||
|
|
||||
/** Return the currently loaded URL */ |
|
||||
String getUrl(); |
|
||||
|
|
||||
void clearCache(); |
|
||||
|
|
||||
/** After calling clearHistory(), canGoBack() should be false. */ |
|
||||
void clearHistory(); |
|
||||
|
|
||||
boolean canGoBack(); |
|
||||
|
|
||||
/** Returns whether a navigation occurred */ |
|
||||
boolean goBack(); |
|
||||
|
|
||||
/** Pauses / resumes the WebView's event loop. */ |
|
||||
void setPaused(boolean value); |
|
||||
|
|
||||
/** Clean up all resources associated with the WebView. */ |
|
||||
void destroy(); |
|
||||
|
|
||||
/** |
|
||||
* Used to retrieve the associated CordovaWebView given a View without knowing the type of Engine. |
|
||||
* E.g. ((CordovaWebView.EngineView)activity.findViewById(android.R.id.webView)).getCordovaWebView(); |
|
||||
*/ |
|
||||
public interface EngineView { |
|
||||
CordovaWebView getCordovaWebView(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Contains methods that an engine uses to communicate with the parent CordovaWebView. |
|
||||
* Methods may be added in future cordova versions, but never removed. |
|
||||
*/ |
|
||||
public interface Client { |
|
||||
Boolean onDispatchKeyEvent(KeyEvent event); |
|
||||
void clearLoadTimeoutTimer(); |
|
||||
void onPageStarted(String newUrl); |
|
||||
void onReceivedError(int errorCode, String description, String failingUrl); |
|
||||
void onPageFinishedLoading(String url); |
|
||||
boolean onNavigationAttempt(String url); |
|
||||
} |
|
||||
} |
|
@ -1,614 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.content.Context; |
|
||||
import android.content.Intent; |
|
||||
import android.net.Uri; |
|
||||
import android.util.Log; |
|
||||
import android.view.Gravity; |
|
||||
import android.view.KeyEvent; |
|
||||
import android.view.View; |
|
||||
import android.view.ViewGroup; |
|
||||
import android.webkit.WebChromeClient; |
|
||||
import android.widget.FrameLayout; |
|
||||
|
|
||||
import org.apache.cordova.engine.SystemWebViewEngine; |
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import java.lang.reflect.Constructor; |
|
||||
import java.util.ArrayList; |
|
||||
import java.util.HashSet; |
|
||||
import java.util.List; |
|
||||
import java.util.Map; |
|
||||
import java.util.Set; |
|
||||
|
|
||||
/** |
|
||||
* Main class for interacting with a Cordova webview. Manages plugins, events, and a CordovaWebViewEngine. |
|
||||
* Class uses two-phase initialization. You must call init() before calling any other methods. |
|
||||
*/ |
|
||||
public class CordovaWebViewImpl implements CordovaWebView { |
|
||||
|
|
||||
public static final String TAG = "CordovaWebViewImpl"; |
|
||||
|
|
||||
private PluginManager pluginManager; |
|
||||
|
|
||||
protected final CordovaWebViewEngine engine; |
|
||||
private CordovaInterface cordova; |
|
||||
|
|
||||
// Flag to track that a loadUrl timeout occurred |
|
||||
private int loadUrlTimeout = 0; |
|
||||
|
|
||||
private CordovaResourceApi resourceApi; |
|
||||
private CordovaPreferences preferences; |
|
||||
private CoreAndroid appPlugin; |
|
||||
private NativeToJsMessageQueue nativeToJsMessageQueue; |
|
||||
private EngineClient engineClient = new EngineClient(); |
|
||||
private boolean hasPausedEver; |
|
||||
|
|
||||
// The URL passed to loadUrl(), not necessarily the URL of the current page. |
|
||||
String loadedUrl; |
|
||||
|
|
||||
/** custom view created by the browser (a video player for example) */ |
|
||||
private View mCustomView; |
|
||||
private WebChromeClient.CustomViewCallback mCustomViewCallback; |
|
||||
|
|
||||
private Set<Integer> boundKeyCodes = new HashSet<Integer>(); |
|
||||
|
|
||||
public static CordovaWebViewEngine createEngine(Context context, CordovaPreferences preferences) { |
|
||||
String className = preferences.getString("webview", SystemWebViewEngine.class.getCanonicalName()); |
|
||||
try { |
|
||||
Class<?> webViewClass = Class.forName(className); |
|
||||
Constructor<?> constructor = webViewClass.getConstructor(Context.class, CordovaPreferences.class); |
|
||||
return (CordovaWebViewEngine) constructor.newInstance(context, preferences); |
|
||||
} catch (Exception e) { |
|
||||
throw new RuntimeException("Failed to create webview. ", e); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public CordovaWebViewImpl(CordovaWebViewEngine cordovaWebViewEngine) { |
|
||||
this.engine = cordovaWebViewEngine; |
|
||||
} |
|
||||
|
|
||||
// Convenience method for when creating programmatically (not from Config.xml). |
|
||||
public void init(CordovaInterface cordova) { |
|
||||
init(cordova, new ArrayList<PluginEntry>(), new CordovaPreferences()); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void init(CordovaInterface cordova, List<PluginEntry> pluginEntries, CordovaPreferences preferences) { |
|
||||
if (this.cordova != null) { |
|
||||
throw new IllegalStateException(); |
|
||||
} |
|
||||
this.cordova = cordova; |
|
||||
this.preferences = preferences; |
|
||||
pluginManager = new PluginManager(this, this.cordova, pluginEntries); |
|
||||
resourceApi = new CordovaResourceApi(engine.getView().getContext(), pluginManager); |
|
||||
nativeToJsMessageQueue = new NativeToJsMessageQueue(); |
|
||||
nativeToJsMessageQueue.addBridgeMode(new NativeToJsMessageQueue.NoOpBridgeMode()); |
|
||||
nativeToJsMessageQueue.addBridgeMode(new NativeToJsMessageQueue.LoadUrlBridgeMode(engine, cordova)); |
|
||||
|
|
||||
if (preferences.getBoolean("DisallowOverscroll", false)) { |
|
||||
engine.getView().setOverScrollMode(View.OVER_SCROLL_NEVER); |
|
||||
} |
|
||||
engine.init(this, cordova, engineClient, resourceApi, pluginManager, nativeToJsMessageQueue); |
|
||||
// This isn't enforced by the compiler, so assert here. |
|
||||
assert engine.getView() instanceof CordovaWebViewEngine.EngineView; |
|
||||
|
|
||||
pluginManager.addService(CoreAndroid.PLUGIN_NAME, "org.apache.cordova.CoreAndroid"); |
|
||||
pluginManager.init(); |
|
||||
|
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean isInitialized() { |
|
||||
return cordova != null; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void loadUrlIntoView(final String url, boolean recreatePlugins) { |
|
||||
LOG.d(TAG, ">>> loadUrl(" + url + ")"); |
|
||||
if (url.equals("about:blank") || url.startsWith("javascript:")) { |
|
||||
engine.loadUrl(url, false); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
recreatePlugins = recreatePlugins || (loadedUrl == null); |
|
||||
|
|
||||
if (recreatePlugins) { |
|
||||
// Don't re-initialize on first load. |
|
||||
if (loadedUrl != null) { |
|
||||
appPlugin = null; |
|
||||
pluginManager.init(); |
|
||||
} |
|
||||
loadedUrl = url; |
|
||||
} |
|
||||
|
|
||||
// Create a timeout timer for loadUrl |
|
||||
final int currentLoadUrlTimeout = loadUrlTimeout; |
|
||||
final int loadUrlTimeoutValue = preferences.getInteger("LoadUrlTimeoutValue", 20000); |
|
||||
|
|
||||
// Timeout error method |
|
||||
final Runnable loadError = new Runnable() { |
|
||||
public void run() { |
|
||||
stopLoading(); |
|
||||
LOG.e(TAG, "CordovaWebView: TIMEOUT ERROR!"); |
|
||||
|
|
||||
// Handle other errors by passing them to the webview in JS |
|
||||
JSONObject data = new JSONObject(); |
|
||||
try { |
|
||||
data.put("errorCode", -6); |
|
||||
data.put("description", "The connection to the server was unsuccessful."); |
|
||||
data.put("url", url); |
|
||||
} catch (JSONException e) { |
|
||||
// Will never happen. |
|
||||
} |
|
||||
pluginManager.postMessage("onReceivedError", data); |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
// Timeout timer method |
|
||||
final Runnable timeoutCheck = new Runnable() { |
|
||||
public void run() { |
|
||||
try { |
|
||||
synchronized (this) { |
|
||||
wait(loadUrlTimeoutValue); |
|
||||
} |
|
||||
} catch (InterruptedException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
|
|
||||
// If timeout, then stop loading and handle error |
|
||||
if (loadUrlTimeout == currentLoadUrlTimeout) { |
|
||||
cordova.getActivity().runOnUiThread(loadError); |
|
||||
} |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
final boolean _recreatePlugins = recreatePlugins; |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
if (loadUrlTimeoutValue > 0) { |
|
||||
cordova.getThreadPool().execute(timeoutCheck); |
|
||||
} |
|
||||
engine.loadUrl(url, _recreatePlugins); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
@Override |
|
||||
public void loadUrl(String url) { |
|
||||
loadUrlIntoView(url, true); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void showWebPage(String url, boolean openExternal, boolean clearHistory, Map<String, Object> params) { |
|
||||
LOG.d(TAG, "showWebPage(%s, %b, %b, HashMap)", url, openExternal, clearHistory); |
|
||||
|
|
||||
// If clearing history |
|
||||
if (clearHistory) { |
|
||||
engine.clearHistory(); |
|
||||
} |
|
||||
|
|
||||
// If loading into our webview |
|
||||
if (!openExternal) { |
|
||||
// Make sure url is in whitelist |
|
||||
if (pluginManager.shouldAllowNavigation(url)) { |
|
||||
// TODO: What about params? |
|
||||
// Load new URL |
|
||||
loadUrlIntoView(url, true); |
|
||||
} else { |
|
||||
LOG.w(TAG, "showWebPage: Refusing to load URL into webview since it is not in the <allow-navigation> whitelist. URL=" + url); |
|
||||
} |
|
||||
} |
|
||||
if (!pluginManager.shouldOpenExternalUrl(url)) { |
|
||||
LOG.w(TAG, "showWebPage: Refusing to send intent for URL since it is not in the <allow-intent> whitelist. URL=" + url); |
|
||||
return; |
|
||||
} |
|
||||
try { |
|
||||
Intent intent = new Intent(Intent.ACTION_VIEW); |
|
||||
// To send an intent without CATEGORY_BROWSER, a custom plugin should be used. |
|
||||
intent.addCategory(Intent.CATEGORY_BROWSABLE); |
|
||||
Uri uri = Uri.parse(url); |
|
||||
// Omitting the MIME type for file: URLs causes "No Activity found to handle Intent". |
|
||||
// Adding the MIME type to http: URLs causes them to not be handled by the downloader. |
|
||||
if ("file".equals(uri.getScheme())) { |
|
||||
intent.setDataAndType(uri, resourceApi.getMimeType(uri)); |
|
||||
} else { |
|
||||
intent.setData(uri); |
|
||||
} |
|
||||
cordova.getActivity().startActivity(intent); |
|
||||
} catch (android.content.ActivityNotFoundException e) { |
|
||||
LOG.e(TAG, "Error loading url " + url, e); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
@Deprecated |
|
||||
public void showCustomView(View view, WebChromeClient.CustomViewCallback callback) { |
|
||||
// This code is adapted from the original Android Browser code, licensed under the Apache License, Version 2.0 |
|
||||
Log.d(TAG, "showing Custom View"); |
|
||||
// if a view already exists then immediately terminate the new one |
|
||||
if (mCustomView != null) { |
|
||||
callback.onCustomViewHidden(); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
// Store the view and its callback for later (to kill it properly) |
|
||||
mCustomView = view; |
|
||||
mCustomViewCallback = callback; |
|
||||
|
|
||||
// Add the custom view to its container. |
|
||||
ViewGroup parent = (ViewGroup) engine.getView().getParent(); |
|
||||
parent.addView(view, new FrameLayout.LayoutParams( |
|
||||
ViewGroup.LayoutParams.MATCH_PARENT, |
|
||||
ViewGroup.LayoutParams.MATCH_PARENT, |
|
||||
Gravity.CENTER)); |
|
||||
|
|
||||
// Hide the content view. |
|
||||
engine.getView().setVisibility(View.GONE); |
|
||||
|
|
||||
// Finally show the custom view container. |
|
||||
parent.setVisibility(View.VISIBLE); |
|
||||
parent.bringToFront(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
@Deprecated |
|
||||
public void hideCustomView() { |
|
||||
// This code is adapted from the original Android Browser code, licensed under the Apache License, Version 2.0 |
|
||||
if (mCustomView == null) return; |
|
||||
Log.d(TAG, "Hiding Custom View"); |
|
||||
|
|
||||
// Hide the custom view. |
|
||||
mCustomView.setVisibility(View.GONE); |
|
||||
|
|
||||
// Remove the custom view from its container. |
|
||||
ViewGroup parent = (ViewGroup) engine.getView().getParent(); |
|
||||
parent.removeView(mCustomView); |
|
||||
mCustomView = null; |
|
||||
mCustomViewCallback.onCustomViewHidden(); |
|
||||
|
|
||||
// Show the content view. |
|
||||
engine.getView().setVisibility(View.VISIBLE); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
@Deprecated |
|
||||
public boolean isCustomViewShowing() { |
|
||||
return mCustomView != null; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
@Deprecated |
|
||||
public void sendJavascript(String statement) { |
|
||||
nativeToJsMessageQueue.addJavaScript(statement); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void sendPluginResult(PluginResult cr, String callbackId) { |
|
||||
nativeToJsMessageQueue.addPluginResult(cr, callbackId); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public PluginManager getPluginManager() { |
|
||||
return pluginManager; |
|
||||
} |
|
||||
@Override |
|
||||
public CordovaPreferences getPreferences() { |
|
||||
return preferences; |
|
||||
} |
|
||||
@Override |
|
||||
public ICordovaCookieManager getCookieManager() { |
|
||||
return engine.getCookieManager(); |
|
||||
} |
|
||||
@Override |
|
||||
public CordovaResourceApi getResourceApi() { |
|
||||
return resourceApi; |
|
||||
} |
|
||||
@Override |
|
||||
public CordovaWebViewEngine getEngine() { |
|
||||
return engine; |
|
||||
} |
|
||||
@Override |
|
||||
public View getView() { |
|
||||
return engine.getView(); |
|
||||
} |
|
||||
@Override |
|
||||
public Context getContext() { |
|
||||
return engine.getView().getContext(); |
|
||||
} |
|
||||
|
|
||||
private void sendJavascriptEvent(String event) { |
|
||||
if (appPlugin == null) { |
|
||||
appPlugin = (CoreAndroid)pluginManager.getPlugin(CoreAndroid.PLUGIN_NAME); |
|
||||
} |
|
||||
|
|
||||
if (appPlugin == null) { |
|
||||
LOG.w(TAG, "Unable to fire event without existing plugin"); |
|
||||
return; |
|
||||
} |
|
||||
appPlugin.fireJavascriptEvent(event); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void setButtonPlumbedToJs(int keyCode, boolean override) { |
|
||||
switch (keyCode) { |
|
||||
case KeyEvent.KEYCODE_VOLUME_DOWN: |
|
||||
case KeyEvent.KEYCODE_VOLUME_UP: |
|
||||
case KeyEvent.KEYCODE_BACK: |
|
||||
case KeyEvent.KEYCODE_MENU: |
|
||||
// TODO: Why are search and menu buttons handled separately? |
|
||||
if (override) { |
|
||||
boundKeyCodes.add(keyCode); |
|
||||
} else { |
|
||||
boundKeyCodes.remove(keyCode); |
|
||||
} |
|
||||
return; |
|
||||
default: |
|
||||
throw new IllegalArgumentException("Unsupported keycode: " + keyCode); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean isButtonPlumbedToJs(int keyCode) { |
|
||||
return boundKeyCodes.contains(keyCode); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public Object postMessage(String id, Object data) { |
|
||||
return pluginManager.postMessage(id, data); |
|
||||
} |
|
||||
|
|
||||
// Engine method proxies: |
|
||||
@Override |
|
||||
public String getUrl() { |
|
||||
return engine.getUrl(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void stopLoading() { |
|
||||
// Clear timeout flag |
|
||||
loadUrlTimeout++; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean canGoBack() { |
|
||||
return engine.canGoBack(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void clearCache() { |
|
||||
engine.clearCache(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
@Deprecated |
|
||||
public void clearCache(boolean b) { |
|
||||
engine.clearCache(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void clearHistory() { |
|
||||
engine.clearHistory(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean backHistory() { |
|
||||
return engine.goBack(); |
|
||||
} |
|
||||
|
|
||||
/////// LifeCycle methods /////// |
|
||||
@Override |
|
||||
public void onNewIntent(Intent intent) { |
|
||||
if (this.pluginManager != null) { |
|
||||
this.pluginManager.onNewIntent(intent); |
|
||||
} |
|
||||
} |
|
||||
@Override |
|
||||
public void handlePause(boolean keepRunning) { |
|
||||
if (!isInitialized()) { |
|
||||
return; |
|
||||
} |
|
||||
hasPausedEver = true; |
|
||||
pluginManager.onPause(keepRunning); |
|
||||
sendJavascriptEvent("pause"); |
|
||||
|
|
||||
// If app doesn't want to run in background |
|
||||
if (!keepRunning) { |
|
||||
// Pause JavaScript timers. This affects all webviews within the app! |
|
||||
engine.setPaused(true); |
|
||||
} |
|
||||
} |
|
||||
@Override |
|
||||
public void handleResume(boolean keepRunning) { |
|
||||
if (!isInitialized()) { |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
// Resume JavaScript timers. This affects all webviews within the app! |
|
||||
engine.setPaused(false); |
|
||||
this.pluginManager.onResume(keepRunning); |
|
||||
|
|
||||
// In order to match the behavior of the other platforms, we only send onResume after an |
|
||||
// onPause has occurred. The resume event might still be sent if the Activity was killed |
|
||||
// while waiting for the result of an external Activity once the result is obtained |
|
||||
if (hasPausedEver) { |
|
||||
sendJavascriptEvent("resume"); |
|
||||
} |
|
||||
} |
|
||||
@Override |
|
||||
public void handleStart() { |
|
||||
if (!isInitialized()) { |
|
||||
return; |
|
||||
} |
|
||||
pluginManager.onStart(); |
|
||||
} |
|
||||
@Override |
|
||||
public void handleStop() { |
|
||||
if (!isInitialized()) { |
|
||||
return; |
|
||||
} |
|
||||
pluginManager.onStop(); |
|
||||
} |
|
||||
@Override |
|
||||
public void handleDestroy() { |
|
||||
if (!isInitialized()) { |
|
||||
return; |
|
||||
} |
|
||||
// Cancel pending timeout timer. |
|
||||
loadUrlTimeout++; |
|
||||
|
|
||||
// Forward to plugins |
|
||||
this.pluginManager.onDestroy(); |
|
||||
|
|
||||
// TODO: about:blank is a bit special (and the default URL for new frames) |
|
||||
// We should use a blank data: url instead so it's more obvious |
|
||||
this.loadUrl("about:blank"); |
|
||||
|
|
||||
// TODO: Should not destroy webview until after about:blank is done loading. |
|
||||
engine.destroy(); |
|
||||
hideCustomView(); |
|
||||
} |
|
||||
|
|
||||
protected class EngineClient implements CordovaWebViewEngine.Client { |
|
||||
@Override |
|
||||
public void clearLoadTimeoutTimer() { |
|
||||
loadUrlTimeout++; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onPageStarted(String newUrl) { |
|
||||
LOG.d(TAG, "onPageDidNavigate(" + newUrl + ")"); |
|
||||
boundKeyCodes.clear(); |
|
||||
pluginManager.onReset(); |
|
||||
pluginManager.postMessage("onPageStarted", newUrl); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onReceivedError(int errorCode, String description, String failingUrl) { |
|
||||
clearLoadTimeoutTimer(); |
|
||||
JSONObject data = new JSONObject(); |
|
||||
try { |
|
||||
data.put("errorCode", errorCode); |
|
||||
data.put("description", description); |
|
||||
data.put("url", failingUrl); |
|
||||
} catch (JSONException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
pluginManager.postMessage("onReceivedError", data); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onPageFinishedLoading(String url) { |
|
||||
LOG.d(TAG, "onPageFinished(" + url + ")"); |
|
||||
|
|
||||
clearLoadTimeoutTimer(); |
|
||||
|
|
||||
// Broadcast message that page has loaded |
|
||||
pluginManager.postMessage("onPageFinished", url); |
|
||||
|
|
||||
// Make app visible after 2 sec in case there was a JS error and Cordova JS never initialized correctly |
|
||||
if (engine.getView().getVisibility() != View.VISIBLE) { |
|
||||
Thread t = new Thread(new Runnable() { |
|
||||
public void run() { |
|
||||
try { |
|
||||
Thread.sleep(2000); |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
pluginManager.postMessage("spinner", "stop"); |
|
||||
} |
|
||||
}); |
|
||||
} catch (InterruptedException e) { |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
t.start(); |
|
||||
} |
|
||||
|
|
||||
// Shutdown if blank loaded |
|
||||
if (url.equals("about:blank")) { |
|
||||
pluginManager.postMessage("exit", null); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public Boolean onDispatchKeyEvent(KeyEvent event) { |
|
||||
int keyCode = event.getKeyCode(); |
|
||||
boolean isBackButton = keyCode == KeyEvent.KEYCODE_BACK; |
|
||||
if (event.getAction() == KeyEvent.ACTION_DOWN) { |
|
||||
if (isBackButton && mCustomView != null) { |
|
||||
return true; |
|
||||
} else if (boundKeyCodes.contains(keyCode)) { |
|
||||
return true; |
|
||||
} else if (isBackButton) { |
|
||||
return engine.canGoBack(); |
|
||||
} |
|
||||
} else if (event.getAction() == KeyEvent.ACTION_UP) { |
|
||||
if (isBackButton && mCustomView != null) { |
|
||||
hideCustomView(); |
|
||||
return true; |
|
||||
} else if (boundKeyCodes.contains(keyCode)) { |
|
||||
String eventName = null; |
|
||||
switch (keyCode) { |
|
||||
case KeyEvent.KEYCODE_VOLUME_DOWN: |
|
||||
eventName = "volumedownbutton"; |
|
||||
break; |
|
||||
case KeyEvent.KEYCODE_VOLUME_UP: |
|
||||
eventName = "volumeupbutton"; |
|
||||
break; |
|
||||
case KeyEvent.KEYCODE_SEARCH: |
|
||||
eventName = "searchbutton"; |
|
||||
break; |
|
||||
case KeyEvent.KEYCODE_MENU: |
|
||||
eventName = "menubutton"; |
|
||||
break; |
|
||||
case KeyEvent.KEYCODE_BACK: |
|
||||
eventName = "backbutton"; |
|
||||
break; |
|
||||
} |
|
||||
if (eventName != null) { |
|
||||
sendJavascriptEvent(eventName); |
|
||||
return true; |
|
||||
} |
|
||||
} else if (isBackButton) { |
|
||||
return engine.goBack(); |
|
||||
} |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean onNavigationAttempt(String url) { |
|
||||
// Give plugins the chance to handle the url |
|
||||
if (pluginManager.onOverrideUrlLoading(url)) { |
|
||||
return true; |
|
||||
} else if (pluginManager.shouldAllowNavigation(url)) { |
|
||||
return false; |
|
||||
} else if (pluginManager.shouldOpenExternalUrl(url)) { |
|
||||
showWebPage(url, true, false, null); |
|
||||
return true; |
|
||||
} |
|
||||
LOG.w(TAG, "Blocked (possibly sub-frame) navigation to non-allowed URL: " + url); |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,360 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.json.JSONArray; |
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import android.content.BroadcastReceiver; |
|
||||
import android.content.Context; |
|
||||
import android.content.Intent; |
|
||||
import android.content.IntentFilter; |
|
||||
import android.telephony.TelephonyManager; |
|
||||
import android.view.KeyEvent; |
|
||||
|
|
||||
import java.util.HashMap; |
|
||||
|
|
||||
/** |
|
||||
* This class exposes methods in Cordova that can be called from JavaScript. |
|
||||
*/ |
|
||||
class CoreAndroid extends CordovaPlugin { |
|
||||
|
|
||||
public static final String PLUGIN_NAME = "CoreAndroid"; |
|
||||
protected static final String TAG = "CordovaApp"; |
|
||||
private BroadcastReceiver telephonyReceiver; |
|
||||
private CallbackContext messageChannel; |
|
||||
private PluginResult pendingResume; |
|
||||
private final Object messageChannelLock = new Object(); |
|
||||
|
|
||||
/** |
|
||||
* Send an event to be fired on the Javascript side. |
|
||||
* |
|
||||
* @param action The name of the event to be fired |
|
||||
*/ |
|
||||
public void fireJavascriptEvent(String action) { |
|
||||
sendEventMessage(action); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Sets the context of the Command. This can then be used to do things like |
|
||||
* get file paths associated with the Activity. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void pluginInitialize() { |
|
||||
this.initTelephonyReceiver(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Executes the request and returns PluginResult. |
|
||||
* |
|
||||
* @param action The action to execute. |
|
||||
* @param args JSONArry of arguments for the plugin. |
|
||||
* @param callbackContext The callback context from which we were invoked. |
|
||||
* @return A PluginResult object with a status and message. |
|
||||
*/ |
|
||||
public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException { |
|
||||
PluginResult.Status status = PluginResult.Status.OK; |
|
||||
String result = ""; |
|
||||
|
|
||||
try { |
|
||||
if (action.equals("clearCache")) { |
|
||||
this.clearCache(); |
|
||||
} |
|
||||
else if (action.equals("show")) { |
|
||||
// This gets called from JavaScript onCordovaReady to show the webview. |
|
||||
// I recommend we change the name of the Message as spinner/stop is not |
|
||||
// indicative of what this actually does (shows the webview). |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
webView.getPluginManager().postMessage("spinner", "stop"); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
else if (action.equals("loadUrl")) { |
|
||||
this.loadUrl(args.getString(0), args.optJSONObject(1)); |
|
||||
} |
|
||||
else if (action.equals("cancelLoadUrl")) { |
|
||||
//this.cancelLoadUrl(); |
|
||||
} |
|
||||
else if (action.equals("clearHistory")) { |
|
||||
this.clearHistory(); |
|
||||
} |
|
||||
else if (action.equals("backHistory")) { |
|
||||
this.backHistory(); |
|
||||
} |
|
||||
else if (action.equals("overrideButton")) { |
|
||||
this.overrideButton(args.getString(0), args.getBoolean(1)); |
|
||||
} |
|
||||
else if (action.equals("overrideBackbutton")) { |
|
||||
this.overrideBackbutton(args.getBoolean(0)); |
|
||||
} |
|
||||
else if (action.equals("exitApp")) { |
|
||||
this.exitApp(); |
|
||||
} |
|
||||
else if (action.equals("messageChannel")) { |
|
||||
synchronized(messageChannelLock) { |
|
||||
messageChannel = callbackContext; |
|
||||
if (pendingResume != null) { |
|
||||
sendEventMessage(pendingResume); |
|
||||
pendingResume = null; |
|
||||
} |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
callbackContext.sendPluginResult(new PluginResult(status, result)); |
|
||||
return true; |
|
||||
} catch (JSONException e) { |
|
||||
callbackContext.sendPluginResult(new PluginResult(PluginResult.Status.JSON_EXCEPTION)); |
|
||||
return false; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
//-------------------------------------------------------------------------- |
|
||||
// LOCAL METHODS |
|
||||
//-------------------------------------------------------------------------- |
|
||||
|
|
||||
/** |
|
||||
* Clear the resource cache. |
|
||||
*/ |
|
||||
public void clearCache() { |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
webView.clearCache(true); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Load the url into the webview. |
|
||||
* |
|
||||
* @param url |
|
||||
* @param props Properties that can be passed in to the Cordova activity (i.e. loadingDialog, wait, ...) |
|
||||
* @throws JSONException |
|
||||
*/ |
|
||||
public void loadUrl(String url, JSONObject props) throws JSONException { |
|
||||
LOG.d("App", "App.loadUrl("+url+","+props+")"); |
|
||||
int wait = 0; |
|
||||
boolean openExternal = false; |
|
||||
boolean clearHistory = false; |
|
||||
|
|
||||
// If there are properties, then set them on the Activity |
|
||||
HashMap<String, Object> params = new HashMap<String, Object>(); |
|
||||
if (props != null) { |
|
||||
JSONArray keys = props.names(); |
|
||||
for (int i = 0; i < keys.length(); i++) { |
|
||||
String key = keys.getString(i); |
|
||||
if (key.equals("wait")) { |
|
||||
wait = props.getInt(key); |
|
||||
} |
|
||||
else if (key.equalsIgnoreCase("openexternal")) { |
|
||||
openExternal = props.getBoolean(key); |
|
||||
} |
|
||||
else if (key.equalsIgnoreCase("clearhistory")) { |
|
||||
clearHistory = props.getBoolean(key); |
|
||||
} |
|
||||
else { |
|
||||
Object value = props.get(key); |
|
||||
if (value == null) { |
|
||||
|
|
||||
} |
|
||||
else if (value.getClass().equals(String.class)) { |
|
||||
params.put(key, (String)value); |
|
||||
} |
|
||||
else if (value.getClass().equals(Boolean.class)) { |
|
||||
params.put(key, (Boolean)value); |
|
||||
} |
|
||||
else if (value.getClass().equals(Integer.class)) { |
|
||||
params.put(key, (Integer)value); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// If wait property, then delay loading |
|
||||
|
|
||||
if (wait > 0) { |
|
||||
try { |
|
||||
synchronized(this) { |
|
||||
this.wait(wait); |
|
||||
} |
|
||||
} catch (InterruptedException e) { |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
} |
|
||||
this.webView.showWebPage(url, openExternal, clearHistory, params); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Clear page history for the app. |
|
||||
*/ |
|
||||
public void clearHistory() { |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
webView.clearHistory(); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Go to previous page displayed. |
|
||||
* This is the same as pressing the backbutton on Android device. |
|
||||
*/ |
|
||||
public void backHistory() { |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
webView.backHistory(); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Override the default behavior of the Android back button. |
|
||||
* If overridden, when the back button is pressed, the "backKeyDown" JavaScript event will be fired. |
|
||||
* |
|
||||
* @param override T=override, F=cancel override |
|
||||
*/ |
|
||||
public void overrideBackbutton(boolean override) { |
|
||||
LOG.i("App", "WARNING: Back Button Default Behavior will be overridden. The backbutton event will be fired!"); |
|
||||
webView.setButtonPlumbedToJs(KeyEvent.KEYCODE_BACK, override); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Override the default behavior of the Android volume buttons. |
|
||||
* If overridden, when the volume button is pressed, the "volume[up|down]button" JavaScript event will be fired. |
|
||||
* |
|
||||
* @param button volumeup, volumedown |
|
||||
* @param override T=override, F=cancel override |
|
||||
*/ |
|
||||
public void overrideButton(String button, boolean override) { |
|
||||
LOG.i("App", "WARNING: Volume Button Default Behavior will be overridden. The volume event will be fired!"); |
|
||||
if (button.equals("volumeup")) { |
|
||||
webView.setButtonPlumbedToJs(KeyEvent.KEYCODE_VOLUME_UP, override); |
|
||||
} |
|
||||
else if (button.equals("volumedown")) { |
|
||||
webView.setButtonPlumbedToJs(KeyEvent.KEYCODE_VOLUME_DOWN, override); |
|
||||
} |
|
||||
else if (button.equals("menubutton")) { |
|
||||
webView.setButtonPlumbedToJs(KeyEvent.KEYCODE_MENU, override); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Return whether the Android back button is overridden by the user. |
|
||||
* |
|
||||
* @return boolean |
|
||||
*/ |
|
||||
public boolean isBackbuttonOverridden() { |
|
||||
return webView.isButtonPlumbedToJs(KeyEvent.KEYCODE_BACK); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Exit the Android application. |
|
||||
*/ |
|
||||
public void exitApp() { |
|
||||
this.webView.getPluginManager().postMessage("exit", null); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Listen for telephony events: RINGING, OFFHOOK and IDLE |
|
||||
* Send these events to all plugins using |
|
||||
* CordovaActivity.onMessage("telephone", "ringing" | "offhook" | "idle") |
|
||||
*/ |
|
||||
private void initTelephonyReceiver() { |
|
||||
IntentFilter intentFilter = new IntentFilter(); |
|
||||
intentFilter.addAction(TelephonyManager.ACTION_PHONE_STATE_CHANGED); |
|
||||
//final CordovaInterface mycordova = this.cordova; |
|
||||
this.telephonyReceiver = new BroadcastReceiver() { |
|
||||
|
|
||||
@Override |
|
||||
public void onReceive(Context context, Intent intent) { |
|
||||
|
|
||||
// If state has changed |
|
||||
if ((intent != null) && intent.getAction().equals(TelephonyManager.ACTION_PHONE_STATE_CHANGED)) { |
|
||||
if (intent.hasExtra(TelephonyManager.EXTRA_STATE)) { |
|
||||
String extraData = intent.getStringExtra(TelephonyManager.EXTRA_STATE); |
|
||||
if (extraData.equals(TelephonyManager.EXTRA_STATE_RINGING)) { |
|
||||
LOG.i(TAG, "Telephone RINGING"); |
|
||||
webView.getPluginManager().postMessage("telephone", "ringing"); |
|
||||
} |
|
||||
else if (extraData.equals(TelephonyManager.EXTRA_STATE_OFFHOOK)) { |
|
||||
LOG.i(TAG, "Telephone OFFHOOK"); |
|
||||
webView.getPluginManager().postMessage("telephone", "offhook"); |
|
||||
} |
|
||||
else if (extraData.equals(TelephonyManager.EXTRA_STATE_IDLE)) { |
|
||||
LOG.i(TAG, "Telephone IDLE"); |
|
||||
webView.getPluginManager().postMessage("telephone", "idle"); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
// Register the receiver |
|
||||
webView.getContext().registerReceiver(this.telephonyReceiver, intentFilter); |
|
||||
} |
|
||||
|
|
||||
private void sendEventMessage(String action) { |
|
||||
JSONObject obj = new JSONObject(); |
|
||||
try { |
|
||||
obj.put("action", action); |
|
||||
} catch (JSONException e) { |
|
||||
LOG.e(TAG, "Failed to create event message", e); |
|
||||
} |
|
||||
sendEventMessage(new PluginResult(PluginResult.Status.OK, obj)); |
|
||||
} |
|
||||
|
|
||||
private void sendEventMessage(PluginResult payload) { |
|
||||
payload.setKeepCallback(true); |
|
||||
if (messageChannel != null) { |
|
||||
messageChannel.sendPluginResult(payload); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/* |
|
||||
* Unregister the receiver |
|
||||
* |
|
||||
*/ |
|
||||
public void onDestroy() |
|
||||
{ |
|
||||
webView.getContext().unregisterReceiver(this.telephonyReceiver); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Used to send the resume event in the case that the Activity is destroyed by the OS |
|
||||
* |
|
||||
* @param resumeEvent PluginResult containing the payload for the resume event to be fired |
|
||||
*/ |
|
||||
public void sendResumeEvent(PluginResult resumeEvent) { |
|
||||
// This operation must be synchronized because plugin results that trigger resume |
|
||||
// events can be processed asynchronously |
|
||||
synchronized(messageChannelLock) { |
|
||||
if (messageChannel != null) { |
|
||||
sendEventMessage(resumeEvent); |
|
||||
} else { |
|
||||
// Might get called before the page loads, so we need to store it until the |
|
||||
// messageChannel gets created |
|
||||
this.pendingResume = resumeEvent; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,31 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.json.JSONException; |
|
||||
|
|
||||
/* |
|
||||
* Any exposed Javascript API MUST implement these three things! |
|
||||
*/ |
|
||||
public interface ExposedJsApi { |
|
||||
public String exec(int bridgeSecret, String service, String action, String callbackId, String arguments) throws JSONException, IllegalAccessException; |
|
||||
public void setNativeToJsBridgeMode(int bridgeSecret, int value) throws IllegalAccessException; |
|
||||
public String retrieveJsMessages(int bridgeSecret, boolean fromOnlineEvent) throws IllegalAccessException; |
|
||||
} |
|
@ -1,66 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.security.Principal; |
|
||||
import java.security.PrivateKey; |
|
||||
import java.security.cert.X509Certificate; |
|
||||
|
|
||||
/** |
|
||||
* Specifies interface for handling certificate requests. |
|
||||
*/ |
|
||||
public interface ICordovaClientCertRequest { |
|
||||
/** |
|
||||
* Cancel this request |
|
||||
*/ |
|
||||
public void cancel(); |
|
||||
|
|
||||
/* |
|
||||
* Returns the host name of the server requesting the certificate. |
|
||||
*/ |
|
||||
public String getHost(); |
|
||||
|
|
||||
/* |
|
||||
* Returns the acceptable types of asymmetric keys (can be null). |
|
||||
*/ |
|
||||
public String[] getKeyTypes(); |
|
||||
|
|
||||
/* |
|
||||
* Returns the port number of the server requesting the certificate. |
|
||||
*/ |
|
||||
public int getPort(); |
|
||||
|
|
||||
/* |
|
||||
* Returns the acceptable certificate issuers for the certificate matching the private key (can be null). |
|
||||
*/ |
|
||||
public Principal[] getPrincipals(); |
|
||||
|
|
||||
/* |
|
||||
* Ignore the request for now. Do not remember user's choice. |
|
||||
*/ |
|
||||
public void ignore(); |
|
||||
|
|
||||
/* |
|
||||
* Proceed with the specified private key and client certificate chain. Remember the user's positive choice and use it for future requests. |
|
||||
* |
|
||||
* @param privateKey The privateKey |
|
||||
* @param chain The certificate chain |
|
||||
*/ |
|
||||
public void proceed(PrivateKey privateKey, X509Certificate[] chain); |
|
||||
} |
|
@ -1,33 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
public interface ICordovaCookieManager { |
|
||||
|
|
||||
public void setCookiesEnabled(boolean accept); |
|
||||
|
|
||||
public void setCookie(final String url, final String value); |
|
||||
|
|
||||
public String getCookie(final String url); |
|
||||
|
|
||||
public void clearCookies(); |
|
||||
|
|
||||
public void flush(); |
|
||||
}; |
|
@ -1,38 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
/** |
|
||||
* Specifies interface for HTTP auth handler object which is used to handle auth requests and |
|
||||
* specifying user credentials. |
|
||||
*/ |
|
||||
public interface ICordovaHttpAuthHandler { |
|
||||
/** |
|
||||
* Instructs the WebView to cancel the authentication request. |
|
||||
*/ |
|
||||
public void cancel (); |
|
||||
|
|
||||
/** |
|
||||
* Instructs the WebView to proceed with the authentication with the given credentials. |
|
||||
* |
|
||||
* @param username The user name |
|
||||
* @param password The password |
|
||||
*/ |
|
||||
public void proceed (String username, String password); |
|
||||
} |
|
@ -1,234 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import android.util.Log; |
|
||||
|
|
||||
/** |
|
||||
* Log to Android logging system. |
|
||||
* |
|
||||
* Log message can be a string or a printf formatted string with arguments. |
|
||||
* See http://developer.android.com/reference/java/util/Formatter.html |
|
||||
*/ |
|
||||
public class LOG { |
|
||||
|
|
||||
public static final int VERBOSE = Log.VERBOSE; |
|
||||
public static final int DEBUG = Log.DEBUG; |
|
||||
public static final int INFO = Log.INFO; |
|
||||
public static final int WARN = Log.WARN; |
|
||||
public static final int ERROR = Log.ERROR; |
|
||||
|
|
||||
// Current log level |
|
||||
public static int LOGLEVEL = Log.ERROR; |
|
||||
|
|
||||
/** |
|
||||
* Set the current log level. |
|
||||
* |
|
||||
* @param logLevel |
|
||||
*/ |
|
||||
public static void setLogLevel(int logLevel) { |
|
||||
LOGLEVEL = logLevel; |
|
||||
Log.i("CordovaLog", "Changing log level to " + logLevel); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Set the current log level. |
|
||||
* |
|
||||
* @param logLevel |
|
||||
*/ |
|
||||
public static void setLogLevel(String logLevel) { |
|
||||
if ("VERBOSE".equals(logLevel)) LOGLEVEL = VERBOSE; |
|
||||
else if ("DEBUG".equals(logLevel)) LOGLEVEL = DEBUG; |
|
||||
else if ("INFO".equals(logLevel)) LOGLEVEL = INFO; |
|
||||
else if ("WARN".equals(logLevel)) LOGLEVEL = WARN; |
|
||||
else if ("ERROR".equals(logLevel)) LOGLEVEL = ERROR; |
|
||||
Log.i("CordovaLog", "Changing log level to " + logLevel + "(" + LOGLEVEL + ")"); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Determine if log level will be logged |
|
||||
* |
|
||||
* @param logLevel |
|
||||
* @return true if the parameter passed in is greater than or equal to the current log level |
|
||||
*/ |
|
||||
public static boolean isLoggable(int logLevel) { |
|
||||
return (logLevel >= LOGLEVEL); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Verbose log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
*/ |
|
||||
public static void v(String tag, String s) { |
|
||||
if (LOG.VERBOSE >= LOGLEVEL) Log.v(tag, s); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Debug log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
*/ |
|
||||
public static void d(String tag, String s) { |
|
||||
if (LOG.DEBUG >= LOGLEVEL) Log.d(tag, s); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Info log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
*/ |
|
||||
public static void i(String tag, String s) { |
|
||||
if (LOG.INFO >= LOGLEVEL) Log.i(tag, s); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Warning log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
*/ |
|
||||
public static void w(String tag, String s) { |
|
||||
if (LOG.WARN >= LOGLEVEL) Log.w(tag, s); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Error log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
*/ |
|
||||
public static void e(String tag, String s) { |
|
||||
if (LOG.ERROR >= LOGLEVEL) Log.e(tag, s); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Verbose log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param e |
|
||||
*/ |
|
||||
public static void v(String tag, String s, Throwable e) { |
|
||||
if (LOG.VERBOSE >= LOGLEVEL) Log.v(tag, s, e); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Debug log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param e |
|
||||
*/ |
|
||||
public static void d(String tag, String s, Throwable e) { |
|
||||
if (LOG.DEBUG >= LOGLEVEL) Log.d(tag, s, e); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Info log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param e |
|
||||
*/ |
|
||||
public static void i(String tag, String s, Throwable e) { |
|
||||
if (LOG.INFO >= LOGLEVEL) Log.i(tag, s, e); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Warning log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param e |
|
||||
*/ |
|
||||
public static void w(String tag, String s, Throwable e) { |
|
||||
if (LOG.WARN >= LOGLEVEL) Log.w(tag, s, e); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Error log message. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param e |
|
||||
*/ |
|
||||
public static void e(String tag, String s, Throwable e) { |
|
||||
if (LOG.ERROR >= LOGLEVEL) Log.e(tag, s, e); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Verbose log message with printf formatting. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param args |
|
||||
*/ |
|
||||
public static void v(String tag, String s, Object... args) { |
|
||||
if (LOG.VERBOSE >= LOGLEVEL) Log.v(tag, String.format(s, args)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Debug log message with printf formatting. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param args |
|
||||
*/ |
|
||||
public static void d(String tag, String s, Object... args) { |
|
||||
if (LOG.DEBUG >= LOGLEVEL) Log.d(tag, String.format(s, args)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Info log message with printf formatting. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param args |
|
||||
*/ |
|
||||
public static void i(String tag, String s, Object... args) { |
|
||||
if (LOG.INFO >= LOGLEVEL) Log.i(tag, String.format(s, args)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Warning log message with printf formatting. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param args |
|
||||
*/ |
|
||||
public static void w(String tag, String s, Object... args) { |
|
||||
if (LOG.WARN >= LOGLEVEL) Log.w(tag, String.format(s, args)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Error log message with printf formatting. |
|
||||
* |
|
||||
* @param tag |
|
||||
* @param s |
|
||||
* @param args |
|
||||
*/ |
|
||||
public static void e(String tag, String s, Object... args) { |
|
||||
if (LOG.ERROR >= LOGLEVEL) Log.e(tag, String.format(s, args)); |
|
||||
} |
|
||||
|
|
||||
} |
|
@ -1,501 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.ArrayList; |
|
||||
import java.util.LinkedList; |
|
||||
|
|
||||
import android.util.Log; |
|
||||
|
|
||||
/** |
|
||||
* Holds the list of messages to be sent to the WebView. |
|
||||
*/ |
|
||||
public class NativeToJsMessageQueue { |
|
||||
private static final String LOG_TAG = "JsMessageQueue"; |
|
||||
|
|
||||
// Set this to true to force plugin results to be encoding as |
|
||||
// JS instead of the custom format (useful for benchmarking). |
|
||||
// Doesn't work for multipart messages. |
|
||||
private static final boolean FORCE_ENCODE_USING_EVAL = false; |
|
||||
|
|
||||
// Disable sending back native->JS messages during an exec() when the active |
|
||||
// exec() is asynchronous. Set this to true when running bridge benchmarks. |
|
||||
static final boolean DISABLE_EXEC_CHAINING = false; |
|
||||
|
|
||||
// Arbitrarily chosen upper limit for how much data to send to JS in one shot. |
|
||||
// This currently only chops up on message boundaries. It may be useful |
|
||||
// to allow it to break up messages. |
|
||||
private static int MAX_PAYLOAD_SIZE = 50 * 1024 * 10240; |
|
||||
|
|
||||
/** |
|
||||
* When true, the active listener is not fired upon enqueue. When set to false, |
|
||||
* the active listener will be fired if the queue is non-empty. |
|
||||
*/ |
|
||||
private boolean paused; |
|
||||
|
|
||||
/** |
|
||||
* The list of JavaScript statements to be sent to JavaScript. |
|
||||
*/ |
|
||||
private final LinkedList<JsMessage> queue = new LinkedList<JsMessage>(); |
|
||||
|
|
||||
/** |
|
||||
* The array of listeners that can be used to send messages to JS. |
|
||||
*/ |
|
||||
private ArrayList<BridgeMode> bridgeModes = new ArrayList<BridgeMode>(); |
|
||||
|
|
||||
/** |
|
||||
* When null, the bridge is disabled. This occurs during page transitions. |
|
||||
* When disabled, all callbacks are dropped since they are assumed to be |
|
||||
* relevant to the previous page. |
|
||||
*/ |
|
||||
private BridgeMode activeBridgeMode; |
|
||||
|
|
||||
public void addBridgeMode(BridgeMode bridgeMode) { |
|
||||
bridgeModes.add(bridgeMode); |
|
||||
} |
|
||||
|
|
||||
public boolean isBridgeEnabled() { |
|
||||
return activeBridgeMode != null; |
|
||||
} |
|
||||
|
|
||||
public boolean isEmpty() { |
|
||||
return queue.isEmpty(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Changes the bridge mode. |
|
||||
*/ |
|
||||
public void setBridgeMode(int value) { |
|
||||
if (value < -1 || value >= bridgeModes.size()) { |
|
||||
Log.d(LOG_TAG, "Invalid NativeToJsBridgeMode: " + value); |
|
||||
} else { |
|
||||
BridgeMode newMode = value < 0 ? null : bridgeModes.get(value); |
|
||||
if (newMode != activeBridgeMode) { |
|
||||
Log.d(LOG_TAG, "Set native->JS mode to " + (newMode == null ? "null" : newMode.getClass().getSimpleName())); |
|
||||
synchronized (this) { |
|
||||
activeBridgeMode = newMode; |
|
||||
if (newMode != null) { |
|
||||
newMode.reset(); |
|
||||
if (!paused && !queue.isEmpty()) { |
|
||||
newMode.onNativeToJsMessageAvailable(this); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Clears all messages and resets to the default bridge mode. |
|
||||
*/ |
|
||||
public void reset() { |
|
||||
synchronized (this) { |
|
||||
queue.clear(); |
|
||||
setBridgeMode(-1); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private int calculatePackedMessageLength(JsMessage message) { |
|
||||
int messageLen = message.calculateEncodedLength(); |
|
||||
String messageLenStr = String.valueOf(messageLen); |
|
||||
return messageLenStr.length() + messageLen + 1; |
|
||||
} |
|
||||
|
|
||||
private void packMessage(JsMessage message, StringBuilder sb) { |
|
||||
int len = message.calculateEncodedLength(); |
|
||||
sb.append(len) |
|
||||
.append(' '); |
|
||||
message.encodeAsMessage(sb); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Combines and returns queued messages combined into a single string. |
|
||||
* Combines as many messages as possible, while staying under MAX_PAYLOAD_SIZE. |
|
||||
* Returns null if the queue is empty. |
|
||||
*/ |
|
||||
public String popAndEncode(boolean fromOnlineEvent) { |
|
||||
synchronized (this) { |
|
||||
if (activeBridgeMode == null) { |
|
||||
return null; |
|
||||
} |
|
||||
activeBridgeMode.notifyOfFlush(this, fromOnlineEvent); |
|
||||
if (queue.isEmpty()) { |
|
||||
return null; |
|
||||
} |
|
||||
int totalPayloadLen = 0; |
|
||||
int numMessagesToSend = 0; |
|
||||
for (JsMessage message : queue) { |
|
||||
int messageSize = calculatePackedMessageLength(message); |
|
||||
if (numMessagesToSend > 0 && totalPayloadLen + messageSize > MAX_PAYLOAD_SIZE && MAX_PAYLOAD_SIZE > 0) { |
|
||||
break; |
|
||||
} |
|
||||
totalPayloadLen += messageSize; |
|
||||
numMessagesToSend += 1; |
|
||||
} |
|
||||
|
|
||||
StringBuilder sb = new StringBuilder(totalPayloadLen); |
|
||||
for (int i = 0; i < numMessagesToSend; ++i) { |
|
||||
JsMessage message = queue.removeFirst(); |
|
||||
packMessage(message, sb); |
|
||||
} |
|
||||
|
|
||||
if (!queue.isEmpty()) { |
|
||||
// Attach a char to indicate that there are more messages pending. |
|
||||
sb.append('*'); |
|
||||
} |
|
||||
String ret = sb.toString(); |
|
||||
return ret; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Same as popAndEncode(), except encodes in a form that can be executed as JS. |
|
||||
*/ |
|
||||
public String popAndEncodeAsJs() { |
|
||||
synchronized (this) { |
|
||||
int length = queue.size(); |
|
||||
if (length == 0) { |
|
||||
return null; |
|
||||
} |
|
||||
int totalPayloadLen = 0; |
|
||||
int numMessagesToSend = 0; |
|
||||
for (JsMessage message : queue) { |
|
||||
int messageSize = message.calculateEncodedLength() + 50; // overestimate. |
|
||||
if (numMessagesToSend > 0 && totalPayloadLen + messageSize > MAX_PAYLOAD_SIZE && MAX_PAYLOAD_SIZE > 0) { |
|
||||
break; |
|
||||
} |
|
||||
totalPayloadLen += messageSize; |
|
||||
numMessagesToSend += 1; |
|
||||
} |
|
||||
boolean willSendAllMessages = numMessagesToSend == queue.size(); |
|
||||
StringBuilder sb = new StringBuilder(totalPayloadLen + (willSendAllMessages ? 0 : 100)); |
|
||||
// Wrap each statement in a try/finally so that if one throws it does |
|
||||
// not affect the next. |
|
||||
for (int i = 0; i < numMessagesToSend; ++i) { |
|
||||
JsMessage message = queue.removeFirst(); |
|
||||
if (willSendAllMessages && (i + 1 == numMessagesToSend)) { |
|
||||
message.encodeAsJsMessage(sb); |
|
||||
} else { |
|
||||
sb.append("try{"); |
|
||||
message.encodeAsJsMessage(sb); |
|
||||
sb.append("}finally{"); |
|
||||
} |
|
||||
} |
|
||||
if (!willSendAllMessages) { |
|
||||
sb.append("window.setTimeout(function(){cordova.require('cordova/plugin/android/polling').pollOnce();},0);"); |
|
||||
} |
|
||||
for (int i = willSendAllMessages ? 1 : 0; i < numMessagesToSend; ++i) { |
|
||||
sb.append('}'); |
|
||||
} |
|
||||
String ret = sb.toString(); |
|
||||
return ret; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Add a JavaScript statement to the list. |
|
||||
*/ |
|
||||
public void addJavaScript(String statement) { |
|
||||
enqueueMessage(new JsMessage(statement)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Add a JavaScript statement to the list. |
|
||||
*/ |
|
||||
public void addPluginResult(PluginResult result, String callbackId) { |
|
||||
if (callbackId == null) { |
|
||||
Log.e(LOG_TAG, "Got plugin result with no callbackId", new Throwable()); |
|
||||
return; |
|
||||
} |
|
||||
// Don't send anything if there is no result and there is no need to |
|
||||
// clear the callbacks. |
|
||||
boolean noResult = result.getStatus() == PluginResult.Status.NO_RESULT.ordinal(); |
|
||||
boolean keepCallback = result.getKeepCallback(); |
|
||||
if (noResult && keepCallback) { |
|
||||
return; |
|
||||
} |
|
||||
JsMessage message = new JsMessage(result, callbackId); |
|
||||
if (FORCE_ENCODE_USING_EVAL) { |
|
||||
StringBuilder sb = new StringBuilder(message.calculateEncodedLength() + 50); |
|
||||
message.encodeAsJsMessage(sb); |
|
||||
message = new JsMessage(sb.toString()); |
|
||||
} |
|
||||
|
|
||||
enqueueMessage(message); |
|
||||
} |
|
||||
|
|
||||
private void enqueueMessage(JsMessage message) { |
|
||||
synchronized (this) { |
|
||||
if (activeBridgeMode == null) { |
|
||||
Log.d(LOG_TAG, "Dropping Native->JS message due to disabled bridge"); |
|
||||
return; |
|
||||
} |
|
||||
queue.add(message); |
|
||||
if (!paused) { |
|
||||
activeBridgeMode.onNativeToJsMessageAvailable(this); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void setPaused(boolean value) { |
|
||||
if (paused && value) { |
|
||||
// This should never happen. If a use-case for it comes up, we should |
|
||||
// change pause to be a counter. |
|
||||
Log.e(LOG_TAG, "nested call to setPaused detected.", new Throwable()); |
|
||||
} |
|
||||
paused = value; |
|
||||
if (!value) { |
|
||||
synchronized (this) { |
|
||||
if (!queue.isEmpty() && activeBridgeMode != null) { |
|
||||
activeBridgeMode.onNativeToJsMessageAvailable(this); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public static abstract class BridgeMode { |
|
||||
public abstract void onNativeToJsMessageAvailable(NativeToJsMessageQueue queue); |
|
||||
public void notifyOfFlush(NativeToJsMessageQueue queue, boolean fromOnlineEvent) {} |
|
||||
public void reset() {} |
|
||||
} |
|
||||
|
|
||||
/** Uses JS polls for messages on a timer.. */ |
|
||||
public static class NoOpBridgeMode extends BridgeMode { |
|
||||
@Override public void onNativeToJsMessageAvailable(NativeToJsMessageQueue queue) { |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** Uses webView.loadUrl("javascript:") to execute messages. */ |
|
||||
public static class LoadUrlBridgeMode extends BridgeMode { |
|
||||
private final CordovaWebViewEngine engine; |
|
||||
private final CordovaInterface cordova; |
|
||||
|
|
||||
public LoadUrlBridgeMode(CordovaWebViewEngine engine, CordovaInterface cordova) { |
|
||||
this.engine = engine; |
|
||||
this.cordova = cordova; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onNativeToJsMessageAvailable(final NativeToJsMessageQueue queue) { |
|
||||
cordova.getActivity().runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
String js = queue.popAndEncodeAsJs(); |
|
||||
if (js != null) { |
|
||||
engine.loadUrl("javascript:" + js, false); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** Uses online/offline events to tell the JS when to poll for messages. */ |
|
||||
public static class OnlineEventsBridgeMode extends BridgeMode { |
|
||||
private final OnlineEventsBridgeModeDelegate delegate; |
|
||||
private boolean online; |
|
||||
private boolean ignoreNextFlush; |
|
||||
|
|
||||
public interface OnlineEventsBridgeModeDelegate { |
|
||||
void setNetworkAvailable(boolean value); |
|
||||
void runOnUiThread(Runnable r); |
|
||||
} |
|
||||
|
|
||||
public OnlineEventsBridgeMode(OnlineEventsBridgeModeDelegate delegate) { |
|
||||
this.delegate = delegate; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void reset() { |
|
||||
delegate.runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
online = false; |
|
||||
// If the following call triggers a notifyOfFlush, then ignore it. |
|
||||
ignoreNextFlush = true; |
|
||||
delegate.setNetworkAvailable(true); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onNativeToJsMessageAvailable(final NativeToJsMessageQueue queue) { |
|
||||
delegate.runOnUiThread(new Runnable() { |
|
||||
public void run() { |
|
||||
if (!queue.isEmpty()) { |
|
||||
ignoreNextFlush = false; |
|
||||
delegate.setNetworkAvailable(online); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
// Track when online/offline events are fired so that we don't fire excess events. |
|
||||
@Override |
|
||||
public void notifyOfFlush(final NativeToJsMessageQueue queue, boolean fromOnlineEvent) { |
|
||||
if (fromOnlineEvent && !ignoreNextFlush) { |
|
||||
online = !online; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private static class JsMessage { |
|
||||
final String jsPayloadOrCallbackId; |
|
||||
final PluginResult pluginResult; |
|
||||
JsMessage(String js) { |
|
||||
if (js == null) { |
|
||||
throw new NullPointerException(); |
|
||||
} |
|
||||
jsPayloadOrCallbackId = js; |
|
||||
pluginResult = null; |
|
||||
} |
|
||||
JsMessage(PluginResult pluginResult, String callbackId) { |
|
||||
if (callbackId == null || pluginResult == null) { |
|
||||
throw new NullPointerException(); |
|
||||
} |
|
||||
jsPayloadOrCallbackId = callbackId; |
|
||||
this.pluginResult = pluginResult; |
|
||||
} |
|
||||
|
|
||||
static int calculateEncodedLengthHelper(PluginResult pluginResult) { |
|
||||
switch (pluginResult.getMessageType()) { |
|
||||
case PluginResult.MESSAGE_TYPE_BOOLEAN: // f or t |
|
||||
case PluginResult.MESSAGE_TYPE_NULL: // N |
|
||||
return 1; |
|
||||
case PluginResult.MESSAGE_TYPE_NUMBER: // n |
|
||||
return 1 + pluginResult.getMessage().length(); |
|
||||
case PluginResult.MESSAGE_TYPE_STRING: // s |
|
||||
return 1 + pluginResult.getStrMessage().length(); |
|
||||
case PluginResult.MESSAGE_TYPE_BINARYSTRING: |
|
||||
return 1 + pluginResult.getMessage().length(); |
|
||||
case PluginResult.MESSAGE_TYPE_ARRAYBUFFER: |
|
||||
return 1 + pluginResult.getMessage().length(); |
|
||||
case PluginResult.MESSAGE_TYPE_MULTIPART: |
|
||||
int ret = 1; |
|
||||
for (int i = 0; i < pluginResult.getMultipartMessagesSize(); i++) { |
|
||||
int length = calculateEncodedLengthHelper(pluginResult.getMultipartMessage(i)); |
|
||||
int argLength = String.valueOf(length).length(); |
|
||||
ret += argLength + 1 + length; |
|
||||
} |
|
||||
return ret; |
|
||||
case PluginResult.MESSAGE_TYPE_JSON: |
|
||||
default: |
|
||||
return pluginResult.getMessage().length(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
int calculateEncodedLength() { |
|
||||
if (pluginResult == null) { |
|
||||
return jsPayloadOrCallbackId.length() + 1; |
|
||||
} |
|
||||
int statusLen = String.valueOf(pluginResult.getStatus()).length(); |
|
||||
int ret = 2 + statusLen + 1 + jsPayloadOrCallbackId.length() + 1; |
|
||||
return ret + calculateEncodedLengthHelper(pluginResult); |
|
||||
} |
|
||||
|
|
||||
static void encodeAsMessageHelper(StringBuilder sb, PluginResult pluginResult) { |
|
||||
switch (pluginResult.getMessageType()) { |
|
||||
case PluginResult.MESSAGE_TYPE_BOOLEAN: |
|
||||
sb.append(pluginResult.getMessage().charAt(0)); // t or f. |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_NULL: // N |
|
||||
sb.append('N'); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_NUMBER: // n |
|
||||
sb.append('n') |
|
||||
.append(pluginResult.getMessage()); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_STRING: // s |
|
||||
sb.append('s'); |
|
||||
sb.append(pluginResult.getStrMessage()); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_BINARYSTRING: // S |
|
||||
sb.append('S'); |
|
||||
sb.append(pluginResult.getMessage()); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_ARRAYBUFFER: // A |
|
||||
sb.append('A'); |
|
||||
sb.append(pluginResult.getMessage()); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_MULTIPART: |
|
||||
sb.append('M'); |
|
||||
for (int i = 0; i < pluginResult.getMultipartMessagesSize(); i++) { |
|
||||
PluginResult multipartMessage = pluginResult.getMultipartMessage(i); |
|
||||
sb.append(String.valueOf(calculateEncodedLengthHelper(multipartMessage))); |
|
||||
sb.append(' '); |
|
||||
encodeAsMessageHelper(sb, multipartMessage); |
|
||||
} |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_JSON: |
|
||||
default: |
|
||||
sb.append(pluginResult.getMessage()); // [ or { |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
void encodeAsMessage(StringBuilder sb) { |
|
||||
if (pluginResult == null) { |
|
||||
sb.append('J') |
|
||||
.append(jsPayloadOrCallbackId); |
|
||||
return; |
|
||||
} |
|
||||
int status = pluginResult.getStatus(); |
|
||||
boolean noResult = status == PluginResult.Status.NO_RESULT.ordinal(); |
|
||||
boolean resultOk = status == PluginResult.Status.OK.ordinal(); |
|
||||
boolean keepCallback = pluginResult.getKeepCallback(); |
|
||||
|
|
||||
sb.append((noResult || resultOk) ? 'S' : 'F') |
|
||||
.append(keepCallback ? '1' : '0') |
|
||||
.append(status) |
|
||||
.append(' ') |
|
||||
.append(jsPayloadOrCallbackId) |
|
||||
.append(' '); |
|
||||
|
|
||||
encodeAsMessageHelper(sb, pluginResult); |
|
||||
} |
|
||||
|
|
||||
void encodeAsJsMessage(StringBuilder sb) { |
|
||||
if (pluginResult == null) { |
|
||||
sb.append(jsPayloadOrCallbackId); |
|
||||
} else { |
|
||||
int status = pluginResult.getStatus(); |
|
||||
boolean success = (status == PluginResult.Status.OK.ordinal()) || (status == PluginResult.Status.NO_RESULT.ordinal()); |
|
||||
sb.append("cordova.callbackFromNative('") |
|
||||
.append(jsPayloadOrCallbackId) |
|
||||
.append("',") |
|
||||
.append(success) |
|
||||
.append(",") |
|
||||
.append(status) |
|
||||
.append(",["); |
|
||||
switch (pluginResult.getMessageType()) { |
|
||||
case PluginResult.MESSAGE_TYPE_BINARYSTRING: |
|
||||
sb.append("atob('") |
|
||||
.append(pluginResult.getMessage()) |
|
||||
.append("')"); |
|
||||
break; |
|
||||
case PluginResult.MESSAGE_TYPE_ARRAYBUFFER: |
|
||||
sb.append("cordova.require('cordova/base64').toArrayBuffer('") |
|
||||
.append(pluginResult.getMessage()) |
|
||||
.append("')"); |
|
||||
break; |
|
||||
default: |
|
||||
sb.append(pluginResult.getMessage()); |
|
||||
} |
|
||||
sb.append("],") |
|
||||
.append(pluginResult.getKeepCallback()) |
|
||||
.append(");"); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,70 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import org.apache.cordova.CordovaPlugin; |
|
||||
|
|
||||
/** |
|
||||
* This class represents a service entry object. |
|
||||
*/ |
|
||||
public final class PluginEntry { |
|
||||
|
|
||||
/** |
|
||||
* The name of the service that this plugin implements |
|
||||
*/ |
|
||||
public final String service; |
|
||||
|
|
||||
/** |
|
||||
* The plugin class name that implements the service. |
|
||||
*/ |
|
||||
public final String pluginClass; |
|
||||
|
|
||||
/** |
|
||||
* The pre-instantiated plugin to use for this entry. |
|
||||
*/ |
|
||||
public final CordovaPlugin plugin; |
|
||||
|
|
||||
/** |
|
||||
* Flag that indicates the plugin object should be created when PluginManager is initialized. |
|
||||
*/ |
|
||||
public final boolean onload; |
|
||||
|
|
||||
/** |
|
||||
* Constructs with a CordovaPlugin already instantiated. |
|
||||
*/ |
|
||||
public PluginEntry(String service, CordovaPlugin plugin) { |
|
||||
this(service, plugin.getClass().getName(), true, plugin); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* @param service The name of the service |
|
||||
* @param pluginClass The plugin class name |
|
||||
* @param onload Create plugin object when HTML page is loaded |
|
||||
*/ |
|
||||
public PluginEntry(String service, String pluginClass, boolean onload) { |
|
||||
this(service, pluginClass, onload, null); |
|
||||
} |
|
||||
|
|
||||
private PluginEntry(String service, String pluginClass, boolean onload, CordovaPlugin plugin) { |
|
||||
this.service = service; |
|
||||
this.pluginClass = pluginClass; |
|
||||
this.onload = onload; |
|
||||
this.plugin = plugin; |
|
||||
} |
|
||||
} |
|
@ -1,527 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.Collection; |
|
||||
import java.util.LinkedHashMap; |
|
||||
|
|
||||
import org.json.JSONException; |
|
||||
|
|
||||
import android.content.Intent; |
|
||||
import android.content.res.Configuration; |
|
||||
import android.net.Uri; |
|
||||
import android.os.Bundle; |
|
||||
import android.os.Debug; |
|
||||
import android.util.Log; |
|
||||
|
|
||||
/** |
|
||||
* PluginManager is exposed to JavaScript in the Cordova WebView. |
|
||||
* |
|
||||
* Calling native plugin code can be done by calling PluginManager.exec(...) |
|
||||
* from JavaScript. |
|
||||
*/ |
|
||||
public class PluginManager { |
|
||||
private static String TAG = "PluginManager"; |
|
||||
private static final int SLOW_EXEC_WARNING_THRESHOLD = Debug.isDebuggerConnected() ? 60 : 16; |
|
||||
|
|
||||
// List of service entries |
|
||||
private final LinkedHashMap<String, CordovaPlugin> pluginMap = new LinkedHashMap<String, CordovaPlugin>(); |
|
||||
private final LinkedHashMap<String, PluginEntry> entryMap = new LinkedHashMap<String, PluginEntry>(); |
|
||||
|
|
||||
private final CordovaInterface ctx; |
|
||||
private final CordovaWebView app; |
|
||||
private boolean isInitialized; |
|
||||
|
|
||||
private CordovaPlugin permissionRequester; |
|
||||
|
|
||||
public PluginManager(CordovaWebView cordovaWebView, CordovaInterface cordova, Collection<PluginEntry> pluginEntries) { |
|
||||
this.ctx = cordova; |
|
||||
this.app = cordovaWebView; |
|
||||
setPluginEntries(pluginEntries); |
|
||||
} |
|
||||
|
|
||||
public Collection<PluginEntry> getPluginEntries() { |
|
||||
return entryMap.values(); |
|
||||
} |
|
||||
|
|
||||
public void setPluginEntries(Collection<PluginEntry> pluginEntries) { |
|
||||
if (isInitialized) { |
|
||||
this.onPause(false); |
|
||||
this.onDestroy(); |
|
||||
pluginMap.clear(); |
|
||||
entryMap.clear(); |
|
||||
} |
|
||||
for (PluginEntry entry : pluginEntries) { |
|
||||
addService(entry); |
|
||||
} |
|
||||
if (isInitialized) { |
|
||||
startupPlugins(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Init when loading a new HTML page into webview. |
|
||||
*/ |
|
||||
public void init() { |
|
||||
LOG.d(TAG, "init()"); |
|
||||
isInitialized = true; |
|
||||
this.onPause(false); |
|
||||
this.onDestroy(); |
|
||||
pluginMap.clear(); |
|
||||
this.startupPlugins(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Create plugins objects that have onload set. |
|
||||
*/ |
|
||||
private void startupPlugins() { |
|
||||
for (PluginEntry entry : entryMap.values()) { |
|
||||
// Add a null entry to for each non-startup plugin to avoid ConcurrentModificationException |
|
||||
// When iterating plugins. |
|
||||
if (entry.onload) { |
|
||||
getPlugin(entry.service); |
|
||||
} else { |
|
||||
pluginMap.put(entry.service, null); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Receives a request for execution and fulfills it by finding the appropriate |
|
||||
* Java class and calling it's execute method. |
|
||||
* |
|
||||
* PluginManager.exec can be used either synchronously or async. In either case, a JSON encoded |
|
||||
* string is returned that will indicate if any errors have occurred when trying to find |
|
||||
* or execute the class denoted by the clazz argument. |
|
||||
* |
|
||||
* @param service String containing the service to run |
|
||||
* @param action String containing the action that the class is supposed to perform. This is |
|
||||
* passed to the plugin execute method and it is up to the plugin developer |
|
||||
* how to deal with it. |
|
||||
* @param callbackId String containing the id of the callback that is execute in JavaScript if |
|
||||
* this is an async plugin call. |
|
||||
* @param rawArgs An Array literal string containing any arguments needed in the |
|
||||
* plugin execute method. |
|
||||
*/ |
|
||||
public void exec(final String service, final String action, final String callbackId, final String rawArgs) { |
|
||||
CordovaPlugin plugin = getPlugin(service); |
|
||||
if (plugin == null) { |
|
||||
Log.d(TAG, "exec() call to unknown plugin: " + service); |
|
||||
PluginResult cr = new PluginResult(PluginResult.Status.CLASS_NOT_FOUND_EXCEPTION); |
|
||||
app.sendPluginResult(cr, callbackId); |
|
||||
return; |
|
||||
} |
|
||||
CallbackContext callbackContext = new CallbackContext(callbackId, app); |
|
||||
try { |
|
||||
long pluginStartTime = System.currentTimeMillis(); |
|
||||
boolean wasValidAction = plugin.execute(action, rawArgs, callbackContext); |
|
||||
long duration = System.currentTimeMillis() - pluginStartTime; |
|
||||
|
|
||||
if (duration > SLOW_EXEC_WARNING_THRESHOLD) { |
|
||||
Log.w(TAG, "THREAD WARNING: exec() call to " + service + "." + action + " blocked the main thread for " + duration + "ms. Plugin should use CordovaInterface.getThreadPool()."); |
|
||||
} |
|
||||
if (!wasValidAction) { |
|
||||
PluginResult cr = new PluginResult(PluginResult.Status.INVALID_ACTION); |
|
||||
callbackContext.sendPluginResult(cr); |
|
||||
} |
|
||||
} catch (JSONException e) { |
|
||||
PluginResult cr = new PluginResult(PluginResult.Status.JSON_EXCEPTION); |
|
||||
callbackContext.sendPluginResult(cr); |
|
||||
} catch (Exception e) { |
|
||||
Log.e(TAG, "Uncaught exception from plugin", e); |
|
||||
callbackContext.error(e.getMessage()); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Get the plugin object that implements the service. |
|
||||
* If the plugin object does not already exist, then create it. |
|
||||
* If the service doesn't exist, then return null. |
|
||||
* |
|
||||
* @param service The name of the service. |
|
||||
* @return CordovaPlugin or null |
|
||||
*/ |
|
||||
public CordovaPlugin getPlugin(String service) { |
|
||||
CordovaPlugin ret = pluginMap.get(service); |
|
||||
if (ret == null) { |
|
||||
PluginEntry pe = entryMap.get(service); |
|
||||
if (pe == null) { |
|
||||
return null; |
|
||||
} |
|
||||
if (pe.plugin != null) { |
|
||||
ret = pe.plugin; |
|
||||
} else { |
|
||||
ret = instantiatePlugin(pe.pluginClass); |
|
||||
} |
|
||||
ret.privateInitialize(service, ctx, app, app.getPreferences()); |
|
||||
pluginMap.put(service, ret); |
|
||||
} |
|
||||
return ret; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Add a plugin class that implements a service to the service entry table. |
|
||||
* This does not create the plugin object instance. |
|
||||
* |
|
||||
* @param service The service name |
|
||||
* @param className The plugin class name |
|
||||
*/ |
|
||||
public void addService(String service, String className) { |
|
||||
PluginEntry entry = new PluginEntry(service, className, false); |
|
||||
this.addService(entry); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Add a plugin class that implements a service to the service entry table. |
|
||||
* This does not create the plugin object instance. |
|
||||
* |
|
||||
* @param entry The plugin entry |
|
||||
*/ |
|
||||
public void addService(PluginEntry entry) { |
|
||||
this.entryMap.put(entry.service, entry); |
|
||||
if (entry.plugin != null) { |
|
||||
entry.plugin.privateInitialize(entry.service, ctx, app, app.getPreferences()); |
|
||||
pluginMap.put(entry.service, entry.plugin); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the system is about to start resuming a previous activity. |
|
||||
* |
|
||||
* @param multitasking Flag indicating if multitasking is turned on for app |
|
||||
*/ |
|
||||
public void onPause(boolean multitasking) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onPause(multitasking); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the system received an HTTP authentication request. Plugins can use |
|
||||
* the supplied HttpAuthHandler to process this auth challenge. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback |
|
||||
* @param handler The HttpAuthHandler used to set the WebView's response |
|
||||
* @param host The host requiring authentication |
|
||||
* @param realm The realm for which authentication is required |
|
||||
* |
|
||||
* @return Returns True if there is a plugin which will resolve this auth challenge, otherwise False |
|
||||
* |
|
||||
*/ |
|
||||
public boolean onReceivedHttpAuthRequest(CordovaWebView view, ICordovaHttpAuthHandler handler, String host, String realm) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null && plugin.onReceivedHttpAuthRequest(app, handler, host, realm)) { |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when he system received an SSL client certificate request. Plugin can use |
|
||||
* the supplied ClientCertRequest to process this certificate challenge. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback |
|
||||
* @param request The client certificate request |
|
||||
* |
|
||||
* @return Returns True if plugin will resolve this auth challenge, otherwise False |
|
||||
* |
|
||||
*/ |
|
||||
public boolean onReceivedClientCertRequest(CordovaWebView view, ICordovaClientCertRequest request) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null && plugin.onReceivedClientCertRequest(app, request)) { |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity will start interacting with the user. |
|
||||
* |
|
||||
* @param multitasking Flag indicating if multitasking is turned on for app |
|
||||
*/ |
|
||||
public void onResume(boolean multitasking) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onResume(multitasking); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is becoming visible to the user. |
|
||||
*/ |
|
||||
public void onStart() { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onStart(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity is no longer visible to the user. |
|
||||
*/ |
|
||||
public void onStop() { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onStop(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* The final call you receive before your activity is destroyed. |
|
||||
*/ |
|
||||
public void onDestroy() { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onDestroy(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Send a message to all plugins. |
|
||||
* |
|
||||
* @param id The message id |
|
||||
* @param data The message data |
|
||||
* @return Object to stop propagation or null |
|
||||
*/ |
|
||||
public Object postMessage(String id, Object data) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
Object obj = plugin.onMessage(id, data); |
|
||||
if (obj != null) { |
|
||||
return obj; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
return ctx.onMessage(id, data); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the activity receives a new intent. |
|
||||
*/ |
|
||||
public void onNewIntent(Intent intent) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onNewIntent(intent); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the webview is going to request an external resource. |
|
||||
* |
|
||||
* This delegates to the installed plugins, and returns true/false for the |
|
||||
* first plugin to provide a non-null result. If no plugins respond, then |
|
||||
* the default policy is applied. |
|
||||
* |
|
||||
* @param url The URL that is being requested. |
|
||||
* @return Returns true to allow the resource to load, |
|
||||
* false to block the resource. |
|
||||
*/ |
|
||||
public boolean shouldAllowRequest(String url) { |
|
||||
for (PluginEntry entry : this.entryMap.values()) { |
|
||||
CordovaPlugin plugin = pluginMap.get(entry.service); |
|
||||
if (plugin != null) { |
|
||||
Boolean result = plugin.shouldAllowRequest(url); |
|
||||
if (result != null) { |
|
||||
return result; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// Default policy: |
|
||||
if (url.startsWith("blob:") || url.startsWith("data:") || url.startsWith("about:blank")) { |
|
||||
return true; |
|
||||
} |
|
||||
// TalkBack requires this, so allow it by default. |
|
||||
if (url.startsWith("https://ssl.gstatic.com/accessibility/javascript/android/")) { |
|
||||
return true; |
|
||||
} |
|
||||
if (url.startsWith("file://")) { |
|
||||
//This directory on WebKit/Blink based webviews contains SQLite databases! |
|
||||
//DON'T CHANGE THIS UNLESS YOU KNOW WHAT YOU'RE DOING! |
|
||||
return !url.contains("/app_webview/"); |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the webview is going to change the URL of the loaded content. |
|
||||
* |
|
||||
* This delegates to the installed plugins, and returns true/false for the |
|
||||
* first plugin to provide a non-null result. If no plugins respond, then |
|
||||
* the default policy is applied. |
|
||||
* |
|
||||
* @param url The URL that is being requested. |
|
||||
* @return Returns true to allow the navigation, |
|
||||
* false to block the navigation. |
|
||||
*/ |
|
||||
public boolean shouldAllowNavigation(String url) { |
|
||||
for (PluginEntry entry : this.entryMap.values()) { |
|
||||
CordovaPlugin plugin = pluginMap.get(entry.service); |
|
||||
if (plugin != null) { |
|
||||
Boolean result = plugin.shouldAllowNavigation(url); |
|
||||
if (result != null) { |
|
||||
return result; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// Default policy: |
|
||||
return url.startsWith("file://") || url.startsWith("about:blank"); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Called when the webview is requesting the exec() bridge be enabled. |
|
||||
*/ |
|
||||
public boolean shouldAllowBridgeAccess(String url) { |
|
||||
for (PluginEntry entry : this.entryMap.values()) { |
|
||||
CordovaPlugin plugin = pluginMap.get(entry.service); |
|
||||
if (plugin != null) { |
|
||||
Boolean result = plugin.shouldAllowBridgeAccess(url); |
|
||||
if (result != null) { |
|
||||
return result; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// Default policy: |
|
||||
return url.startsWith("file://"); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the webview is going not going to navigate, but may launch |
|
||||
* an Intent for an URL. |
|
||||
* |
|
||||
* This delegates to the installed plugins, and returns true/false for the |
|
||||
* first plugin to provide a non-null result. If no plugins respond, then |
|
||||
* the default policy is applied. |
|
||||
* |
|
||||
* @param url The URL that is being requested. |
|
||||
* @return Returns true to allow the URL to launch an intent, |
|
||||
* false to block the intent. |
|
||||
*/ |
|
||||
public Boolean shouldOpenExternalUrl(String url) { |
|
||||
for (PluginEntry entry : this.entryMap.values()) { |
|
||||
CordovaPlugin plugin = pluginMap.get(entry.service); |
|
||||
if (plugin != null) { |
|
||||
Boolean result = plugin.shouldOpenExternalUrl(url); |
|
||||
if (result != null) { |
|
||||
return result; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
// Default policy: |
|
||||
// External URLs are not allowed |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the URL of the webview changes. |
|
||||
* |
|
||||
* @param url The URL that is being changed to. |
|
||||
* @return Return false to allow the URL to load, return true to prevent the URL from loading. |
|
||||
*/ |
|
||||
public boolean onOverrideUrlLoading(String url) { |
|
||||
for (PluginEntry entry : this.entryMap.values()) { |
|
||||
CordovaPlugin plugin = pluginMap.get(entry.service); |
|
||||
if (plugin != null && plugin.onOverrideUrlLoading(url)) { |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called when the app navigates or refreshes. |
|
||||
*/ |
|
||||
public void onReset() { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onReset(); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
Uri remapUri(Uri uri) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
Uri ret = plugin.remapUri(uri); |
|
||||
if (ret != null) { |
|
||||
return ret; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Create a plugin based on class name. |
|
||||
*/ |
|
||||
private CordovaPlugin instantiatePlugin(String className) { |
|
||||
CordovaPlugin ret = null; |
|
||||
try { |
|
||||
Class<?> c = null; |
|
||||
if ((className != null) && !("".equals(className))) { |
|
||||
c = Class.forName(className); |
|
||||
} |
|
||||
if (c != null & CordovaPlugin.class.isAssignableFrom(c)) { |
|
||||
ret = (CordovaPlugin) c.newInstance(); |
|
||||
} |
|
||||
} catch (Exception e) { |
|
||||
e.printStackTrace(); |
|
||||
System.out.println("Error adding plugin " + className + "."); |
|
||||
} |
|
||||
return ret; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Called by the system when the device configuration changes while your activity is running. |
|
||||
* |
|
||||
* @param newConfig The new device configuration |
|
||||
*/ |
|
||||
public void onConfigurationChanged(Configuration newConfig) { |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
plugin.onConfigurationChanged(newConfig); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public Bundle onSaveInstanceState() { |
|
||||
Bundle state = new Bundle(); |
|
||||
for (CordovaPlugin plugin : this.pluginMap.values()) { |
|
||||
if (plugin != null) { |
|
||||
Bundle pluginState = plugin.onSaveInstanceState(); |
|
||||
if(pluginState != null) { |
|
||||
state.putBundle(plugin.getServiceName(), pluginState); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
return state; |
|
||||
} |
|
||||
} |
|
@ -1,198 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.util.List; |
|
||||
|
|
||||
import org.json.JSONArray; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import android.util.Base64; |
|
||||
|
|
||||
public class PluginResult { |
|
||||
private final int status; |
|
||||
private final int messageType; |
|
||||
private boolean keepCallback = false; |
|
||||
private String strMessage; |
|
||||
private String encodedMessage; |
|
||||
private List<PluginResult> multipartMessages; |
|
||||
|
|
||||
public PluginResult(Status status) { |
|
||||
this(status, PluginResult.StatusMessages[status.ordinal()]); |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, String message) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = message == null ? MESSAGE_TYPE_NULL : MESSAGE_TYPE_STRING; |
|
||||
this.strMessage = message; |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, JSONArray message) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_JSON; |
|
||||
encodedMessage = message.toString(); |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, JSONObject message) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_JSON; |
|
||||
encodedMessage = message.toString(); |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, int i) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_NUMBER; |
|
||||
this.encodedMessage = ""+i; |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, float f) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_NUMBER; |
|
||||
this.encodedMessage = ""+f; |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, boolean b) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_BOOLEAN; |
|
||||
this.encodedMessage = Boolean.toString(b); |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, byte[] data) { |
|
||||
this(status, data, false); |
|
||||
} |
|
||||
|
|
||||
public PluginResult(Status status, byte[] data, boolean binaryString) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = binaryString ? MESSAGE_TYPE_BINARYSTRING : MESSAGE_TYPE_ARRAYBUFFER; |
|
||||
this.encodedMessage = Base64.encodeToString(data, Base64.NO_WRAP); |
|
||||
} |
|
||||
|
|
||||
// The keepCallback and status of multipartMessages are ignored. |
|
||||
public PluginResult(Status status, List<PluginResult> multipartMessages) { |
|
||||
this.status = status.ordinal(); |
|
||||
this.messageType = MESSAGE_TYPE_MULTIPART; |
|
||||
this.multipartMessages = multipartMessages; |
|
||||
} |
|
||||
|
|
||||
public void setKeepCallback(boolean b) { |
|
||||
this.keepCallback = b; |
|
||||
} |
|
||||
|
|
||||
public int getStatus() { |
|
||||
return status; |
|
||||
} |
|
||||
|
|
||||
public int getMessageType() { |
|
||||
return messageType; |
|
||||
} |
|
||||
|
|
||||
public String getMessage() { |
|
||||
if (encodedMessage == null) { |
|
||||
encodedMessage = JSONObject.quote(strMessage); |
|
||||
} |
|
||||
return encodedMessage; |
|
||||
} |
|
||||
|
|
||||
public int getMultipartMessagesSize() { |
|
||||
return multipartMessages.size(); |
|
||||
} |
|
||||
|
|
||||
public PluginResult getMultipartMessage(int index) { |
|
||||
return multipartMessages.get(index); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* If messageType == MESSAGE_TYPE_STRING, then returns the message string. |
|
||||
* Otherwise, returns null. |
|
||||
*/ |
|
||||
public String getStrMessage() { |
|
||||
return strMessage; |
|
||||
} |
|
||||
|
|
||||
public boolean getKeepCallback() { |
|
||||
return this.keepCallback; |
|
||||
} |
|
||||
|
|
||||
@Deprecated // Use sendPluginResult instead of sendJavascript. |
|
||||
public String getJSONString() { |
|
||||
return "{\"status\":" + this.status + ",\"message\":" + this.getMessage() + ",\"keepCallback\":" + this.keepCallback + "}"; |
|
||||
} |
|
||||
|
|
||||
@Deprecated // Use sendPluginResult instead of sendJavascript. |
|
||||
public String toCallbackString(String callbackId) { |
|
||||
// If no result to be sent and keeping callback, then no need to sent back to JavaScript |
|
||||
if ((status == PluginResult.Status.NO_RESULT.ordinal()) && keepCallback) { |
|
||||
return null; |
|
||||
} |
|
||||
|
|
||||
// Check the success (OK, NO_RESULT & !KEEP_CALLBACK) |
|
||||
if ((status == PluginResult.Status.OK.ordinal()) || (status == PluginResult.Status.NO_RESULT.ordinal())) { |
|
||||
return toSuccessCallbackString(callbackId); |
|
||||
} |
|
||||
|
|
||||
return toErrorCallbackString(callbackId); |
|
||||
} |
|
||||
|
|
||||
@Deprecated // Use sendPluginResult instead of sendJavascript. |
|
||||
public String toSuccessCallbackString(String callbackId) { |
|
||||
return "cordova.callbackSuccess('"+callbackId+"',"+this.getJSONString()+");"; |
|
||||
} |
|
||||
|
|
||||
@Deprecated // Use sendPluginResult instead of sendJavascript. |
|
||||
public String toErrorCallbackString(String callbackId) { |
|
||||
return "cordova.callbackError('"+callbackId+"', " + this.getJSONString()+ ");"; |
|
||||
} |
|
||||
|
|
||||
public static final int MESSAGE_TYPE_STRING = 1; |
|
||||
public static final int MESSAGE_TYPE_JSON = 2; |
|
||||
public static final int MESSAGE_TYPE_NUMBER = 3; |
|
||||
public static final int MESSAGE_TYPE_BOOLEAN = 4; |
|
||||
public static final int MESSAGE_TYPE_NULL = 5; |
|
||||
public static final int MESSAGE_TYPE_ARRAYBUFFER = 6; |
|
||||
// Use BINARYSTRING when your string may contain null characters. |
|
||||
// This is required to work around a bug in the platform :(. |
|
||||
public static final int MESSAGE_TYPE_BINARYSTRING = 7; |
|
||||
public static final int MESSAGE_TYPE_MULTIPART = 8; |
|
||||
|
|
||||
public static String[] StatusMessages = new String[] { |
|
||||
"No result", |
|
||||
"OK", |
|
||||
"Class not found", |
|
||||
"Illegal access", |
|
||||
"Instantiation error", |
|
||||
"Malformed url", |
|
||||
"IO error", |
|
||||
"Invalid action", |
|
||||
"JSON error", |
|
||||
"Error" |
|
||||
}; |
|
||||
|
|
||||
public enum Status { |
|
||||
NO_RESULT, |
|
||||
OK, |
|
||||
CLASS_NOT_FOUND_EXCEPTION, |
|
||||
ILLEGAL_ACCESS_EXCEPTION, |
|
||||
INSTANTIATION_EXCEPTION, |
|
||||
MALFORMED_URL_EXCEPTION, |
|
||||
IO_EXCEPTION, |
|
||||
INVALID_ACTION, |
|
||||
JSON_EXCEPTION, |
|
||||
ERROR |
|
||||
} |
|
||||
} |
|
@ -1,76 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
|
|
||||
import org.json.JSONException; |
|
||||
import org.json.JSONObject; |
|
||||
|
|
||||
import java.util.ArrayList; |
|
||||
import java.util.List; |
|
||||
|
|
||||
public class ResumeCallback extends CallbackContext { |
|
||||
private final String TAG = "CordovaResumeCallback"; |
|
||||
private String serviceName; |
|
||||
private PluginManager pluginManager; |
|
||||
|
|
||||
public ResumeCallback(String serviceName, PluginManager pluginManager) { |
|
||||
super("resumecallback", null); |
|
||||
this.serviceName = serviceName; |
|
||||
this.pluginManager = pluginManager; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void sendPluginResult(PluginResult pluginResult) { |
|
||||
synchronized (this) { |
|
||||
if (finished) { |
|
||||
LOG.w(TAG, serviceName + " attempted to send a second callback to ResumeCallback\nResult was: " + pluginResult.getMessage()); |
|
||||
return; |
|
||||
} else { |
|
||||
finished = true; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
JSONObject event = new JSONObject(); |
|
||||
JSONObject pluginResultObject = new JSONObject(); |
|
||||
|
|
||||
try { |
|
||||
pluginResultObject.put("pluginServiceName", this.serviceName); |
|
||||
pluginResultObject.put("pluginStatus", PluginResult.StatusMessages[pluginResult.getStatus()]); |
|
||||
|
|
||||
event.put("action", "resume"); |
|
||||
event.put("pendingResult", pluginResultObject); |
|
||||
} catch (JSONException e) { |
|
||||
LOG.e(TAG, "Unable to create resume object for Activity Result"); |
|
||||
} |
|
||||
|
|
||||
PluginResult eventResult = new PluginResult(PluginResult.Status.OK, event); |
|
||||
|
|
||||
// We send a list of results to the js so that we don't have to decode |
|
||||
// the PluginResult passed to this CallbackContext into JSON twice. |
|
||||
// The results are combined into an event payload before the event is |
|
||||
// fired on the js side of things (see platform.js) |
|
||||
List<PluginResult> result = new ArrayList<PluginResult>(); |
|
||||
result.add(eventResult); |
|
||||
result.add(pluginResult); |
|
||||
|
|
||||
CoreAndroid appPlugin = (CoreAndroid) pluginManager.getPlugin(CoreAndroid.PLUGIN_NAME); |
|
||||
appPlugin.sendResumeEvent(new PluginResult(PluginResult.Status.OK, result)); |
|
||||
} |
|
||||
} |
|
@ -1,170 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova; |
|
||||
|
|
||||
import java.net.MalformedURLException; |
|
||||
import java.util.ArrayList; |
|
||||
import java.util.Iterator; |
|
||||
import java.util.regex.Matcher; |
|
||||
import java.util.regex.Pattern; |
|
||||
|
|
||||
import org.apache.cordova.LOG; |
|
||||
|
|
||||
import android.net.Uri; |
|
||||
|
|
||||
public class Whitelist { |
|
||||
private static class URLPattern { |
|
||||
public Pattern scheme; |
|
||||
public Pattern host; |
|
||||
public Integer port; |
|
||||
public Pattern path; |
|
||||
|
|
||||
private String regexFromPattern(String pattern, boolean allowWildcards) { |
|
||||
final String toReplace = "\\.[]{}()^$?+|"; |
|
||||
StringBuilder regex = new StringBuilder(); |
|
||||
for (int i=0; i < pattern.length(); i++) { |
|
||||
char c = pattern.charAt(i); |
|
||||
if (c == '*' && allowWildcards) { |
|
||||
regex.append("."); |
|
||||
} else if (toReplace.indexOf(c) > -1) { |
|
||||
regex.append('\\'); |
|
||||
} |
|
||||
regex.append(c); |
|
||||
} |
|
||||
return regex.toString(); |
|
||||
} |
|
||||
|
|
||||
public URLPattern(String scheme, String host, String port, String path) throws MalformedURLException { |
|
||||
try { |
|
||||
if (scheme == null || "*".equals(scheme)) { |
|
||||
this.scheme = null; |
|
||||
} else { |
|
||||
this.scheme = Pattern.compile(regexFromPattern(scheme, false), Pattern.CASE_INSENSITIVE); |
|
||||
} |
|
||||
if ("*".equals(host)) { |
|
||||
this.host = null; |
|
||||
} else if (host.startsWith("*.")) { |
|
||||
this.host = Pattern.compile("([a-z0-9.-]*\\.)?" + regexFromPattern(host.substring(2), false), Pattern.CASE_INSENSITIVE); |
|
||||
} else { |
|
||||
this.host = Pattern.compile(regexFromPattern(host, false), Pattern.CASE_INSENSITIVE); |
|
||||
} |
|
||||
if (port == null || "*".equals(port)) { |
|
||||
this.port = null; |
|
||||
} else { |
|
||||
this.port = Integer.parseInt(port,10); |
|
||||
} |
|
||||
if (path == null || "/*".equals(path)) { |
|
||||
this.path = null; |
|
||||
} else { |
|
||||
this.path = Pattern.compile(regexFromPattern(path, true)); |
|
||||
} |
|
||||
} catch (NumberFormatException e) { |
|
||||
throw new MalformedURLException("Port must be a number"); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public boolean matches(Uri uri) { |
|
||||
try { |
|
||||
return ((scheme == null || scheme.matcher(uri.getScheme()).matches()) && |
|
||||
(host == null || host.matcher(uri.getHost()).matches()) && |
|
||||
(port == null || port.equals(uri.getPort())) && |
|
||||
(path == null || path.matcher(uri.getPath()).matches())); |
|
||||
} catch (Exception e) { |
|
||||
LOG.d(TAG, e.toString()); |
|
||||
return false; |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private ArrayList<URLPattern> whiteList; |
|
||||
|
|
||||
public static final String TAG = "Whitelist"; |
|
||||
|
|
||||
public Whitelist() { |
|
||||
this.whiteList = new ArrayList<URLPattern>(); |
|
||||
} |
|
||||
|
|
||||
/* Match patterns (from http://developer.chrome.com/extensions/match_patterns.html) |
|
||||
* |
|
||||
* <url-pattern> := <scheme>://<host><path> |
|
||||
* <scheme> := '*' | 'http' | 'https' | 'file' | 'ftp' | 'chrome-extension' |
|
||||
* <host> := '*' | '*.' <any char except '/' and '*'>+ |
|
||||
* <path> := '/' <any chars> |
|
||||
* |
|
||||
* We extend this to explicitly allow a port attached to the host, and we allow |
|
||||
* the scheme to be omitted for backwards compatibility. (Also host is not required |
|
||||
* to begin with a "*" or "*.".) |
|
||||
*/ |
|
||||
public void addWhiteListEntry(String origin, boolean subdomains) { |
|
||||
if (whiteList != null) { |
|
||||
try { |
|
||||
// Unlimited access to network resources |
|
||||
if (origin.compareTo("*") == 0) { |
|
||||
LOG.d(TAG, "Unlimited access to network resources"); |
|
||||
whiteList = null; |
|
||||
} |
|
||||
else { // specific access |
|
||||
Pattern parts = Pattern.compile("^((\\*|[A-Za-z-]+):(//)?)?(\\*|((\\*\\.)?[^*/:]+))?(:(\\d+))?(/.*)?"); |
|
||||
Matcher m = parts.matcher(origin); |
|
||||
if (m.matches()) { |
|
||||
String scheme = m.group(2); |
|
||||
String host = m.group(4); |
|
||||
// Special case for two urls which are allowed to have empty hosts |
|
||||
if (("file".equals(scheme) || "content".equals(scheme)) && host == null) host = "*"; |
|
||||
String port = m.group(8); |
|
||||
String path = m.group(9); |
|
||||
if (scheme == null) { |
|
||||
// XXX making it stupid friendly for people who forget to include protocol/SSL |
|
||||
whiteList.add(new URLPattern("http", host, port, path)); |
|
||||
whiteList.add(new URLPattern("https", host, port, path)); |
|
||||
} else { |
|
||||
whiteList.add(new URLPattern(scheme, host, port, path)); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} catch (Exception e) { |
|
||||
LOG.d(TAG, "Failed to add origin %s", origin); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Determine if URL is in approved list of URLs to load. |
|
||||
* |
|
||||
* @param uri |
|
||||
* @return true if wide open or whitelisted |
|
||||
*/ |
|
||||
public boolean isUrlWhiteListed(String uri) { |
|
||||
// If there is no whitelist, then it's wide open |
|
||||
if (whiteList == null) return true; |
|
||||
|
|
||||
Uri parsedUri = Uri.parse(uri); |
|
||||
// Look for match in white list |
|
||||
Iterator<URLPattern> pit = whiteList.iterator(); |
|
||||
while (pit.hasNext()) { |
|
||||
URLPattern p = pit.next(); |
|
||||
if (p.matches(parsedUri)) { |
|
||||
return true; |
|
||||
} |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
} |
|
@ -1,69 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import android.annotation.TargetApi; |
|
||||
import android.os.Build; |
|
||||
import android.webkit.CookieManager; |
|
||||
import android.webkit.WebView; |
|
||||
|
|
||||
import org.apache.cordova.ICordovaCookieManager; |
|
||||
|
|
||||
class SystemCookieManager implements ICordovaCookieManager { |
|
||||
|
|
||||
protected final WebView webView; |
|
||||
private final CookieManager cookieManager; |
|
||||
|
|
||||
//Added because lint can't see the conditional RIGHT ABOVE this |
|
||||
@TargetApi(Build.VERSION_CODES.LOLLIPOP) |
|
||||
public SystemCookieManager(WebView webview) { |
|
||||
webView = webview; |
|
||||
cookieManager = CookieManager.getInstance(); |
|
||||
|
|
||||
//REALLY? Nobody has seen this UNTIL NOW? |
|
||||
cookieManager.setAcceptFileSchemeCookies(true); |
|
||||
|
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { |
|
||||
cookieManager.setAcceptThirdPartyCookies(webView, true); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
public void setCookiesEnabled(boolean accept) { |
|
||||
cookieManager.setAcceptCookie(accept); |
|
||||
} |
|
||||
|
|
||||
public void setCookie(final String url, final String value) { |
|
||||
cookieManager.setCookie(url, value); |
|
||||
} |
|
||||
|
|
||||
public String getCookie(final String url) { |
|
||||
return cookieManager.getCookie(url); |
|
||||
} |
|
||||
|
|
||||
public void clearCookies() { |
|
||||
cookieManager.removeAllCookie(); |
|
||||
} |
|
||||
|
|
||||
public void flush() { |
|
||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { |
|
||||
cookieManager.flush(); |
|
||||
} |
|
||||
} |
|
||||
}; |
|
@ -1,53 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import android.webkit.JavascriptInterface; |
|
||||
|
|
||||
import org.apache.cordova.CordovaBridge; |
|
||||
import org.apache.cordova.ExposedJsApi; |
|
||||
import org.json.JSONException; |
|
||||
|
|
||||
/** |
|
||||
* Contains APIs that the JS can call. All functions in here should also have |
|
||||
* an equivalent entry in CordovaChromeClient.java, and be added to |
|
||||
* cordova-js/lib/android/plugin/android/promptbasednativeapi.js |
|
||||
*/ |
|
||||
class SystemExposedJsApi implements ExposedJsApi { |
|
||||
private final CordovaBridge bridge; |
|
||||
|
|
||||
SystemExposedJsApi(CordovaBridge bridge) { |
|
||||
this.bridge = bridge; |
|
||||
} |
|
||||
|
|
||||
@JavascriptInterface |
|
||||
public String exec(int bridgeSecret, String service, String action, String callbackId, String arguments) throws JSONException, IllegalAccessException { |
|
||||
return bridge.jsExec(bridgeSecret, service, action, callbackId, arguments); |
|
||||
} |
|
||||
|
|
||||
@JavascriptInterface |
|
||||
public void setNativeToJsBridgeMode(int bridgeSecret, int value) throws IllegalAccessException { |
|
||||
bridge.jsSetNativeToJsBridgeMode(bridgeSecret, value); |
|
||||
} |
|
||||
|
|
||||
@JavascriptInterface |
|
||||
public String retrieveJsMessages(int bridgeSecret, boolean fromOnlineEvent) throws IllegalAccessException { |
|
||||
return bridge.jsRetrieveJsMessages(bridgeSecret, fromOnlineEvent); |
|
||||
} |
|
||||
} |
|
@ -1,293 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import java.util.Arrays; |
|
||||
import android.annotation.TargetApi; |
|
||||
import android.app.Activity; |
|
||||
import android.content.Context; |
|
||||
import android.content.ActivityNotFoundException; |
|
||||
import android.content.Intent; |
|
||||
import android.net.Uri; |
|
||||
import android.os.Build; |
|
||||
import android.util.Log; |
|
||||
import android.view.Gravity; |
|
||||
import android.view.View; |
|
||||
import android.view.ViewGroup.LayoutParams; |
|
||||
import android.webkit.ConsoleMessage; |
|
||||
import android.webkit.GeolocationPermissions.Callback; |
|
||||
import android.webkit.JsPromptResult; |
|
||||
import android.webkit.JsResult; |
|
||||
import android.webkit.ValueCallback; |
|
||||
import android.webkit.WebChromeClient; |
|
||||
import android.webkit.WebStorage; |
|
||||
import android.webkit.WebView; |
|
||||
import android.webkit.PermissionRequest; |
|
||||
import android.widget.LinearLayout; |
|
||||
import android.widget.ProgressBar; |
|
||||
import android.widget.RelativeLayout; |
|
||||
|
|
||||
import org.apache.cordova.CordovaDialogsHelper; |
|
||||
import org.apache.cordova.CordovaPlugin; |
|
||||
import org.apache.cordova.LOG; |
|
||||
|
|
||||
/** |
|
||||
* This class is the WebChromeClient that implements callbacks for our web view. |
|
||||
* The kind of callbacks that happen here are on the chrome outside the document, |
|
||||
* such as onCreateWindow(), onConsoleMessage(), onProgressChanged(), etc. Related |
|
||||
* to but different than CordovaWebViewClient. |
|
||||
*/ |
|
||||
public class SystemWebChromeClient extends WebChromeClient { |
|
||||
|
|
||||
private static final int FILECHOOSER_RESULTCODE = 5173; |
|
||||
private static final String LOG_TAG = "SystemWebChromeClient"; |
|
||||
private long MAX_QUOTA = 100 * 1024 * 1024; |
|
||||
protected final SystemWebViewEngine parentEngine; |
|
||||
|
|
||||
// the video progress view |
|
||||
private View mVideoProgressView; |
|
||||
|
|
||||
private CordovaDialogsHelper dialogsHelper; |
|
||||
private Context appContext; |
|
||||
|
|
||||
private WebChromeClient.CustomViewCallback mCustomViewCallback; |
|
||||
private View mCustomView; |
|
||||
|
|
||||
public SystemWebChromeClient(SystemWebViewEngine parentEngine) { |
|
||||
this.parentEngine = parentEngine; |
|
||||
appContext = parentEngine.webView.getContext(); |
|
||||
dialogsHelper = new CordovaDialogsHelper(appContext); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Tell the client to display a javascript alert dialog. |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean onJsAlert(WebView view, String url, String message, final JsResult result) { |
|
||||
dialogsHelper.showAlert(message, new CordovaDialogsHelper.Result() { |
|
||||
@Override public void gotResult(boolean success, String value) { |
|
||||
if (success) { |
|
||||
result.confirm(); |
|
||||
} else { |
|
||||
result.cancel(); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Tell the client to display a confirm dialog to the user. |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean onJsConfirm(WebView view, String url, String message, final JsResult result) { |
|
||||
dialogsHelper.showConfirm(message, new CordovaDialogsHelper.Result() { |
|
||||
@Override |
|
||||
public void gotResult(boolean success, String value) { |
|
||||
if (success) { |
|
||||
result.confirm(); |
|
||||
} else { |
|
||||
result.cancel(); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Tell the client to display a prompt dialog to the user. |
|
||||
* If the client returns true, WebView will assume that the client will |
|
||||
* handle the prompt dialog and call the appropriate JsPromptResult method. |
|
||||
* |
|
||||
* Since we are hacking prompts for our own purposes, we should not be using them for |
|
||||
* this purpose, perhaps we should hack console.log to do this instead! |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean onJsPrompt(WebView view, String origin, String message, String defaultValue, final JsPromptResult result) { |
|
||||
// Unlike the @JavascriptInterface bridge, this method is always called on the UI thread. |
|
||||
String handledRet = parentEngine.bridge.promptOnJsPrompt(origin, message, defaultValue); |
|
||||
if (handledRet != null) { |
|
||||
result.confirm(handledRet); |
|
||||
} else { |
|
||||
dialogsHelper.showPrompt(message, defaultValue, new CordovaDialogsHelper.Result() { |
|
||||
@Override |
|
||||
public void gotResult(boolean success, String value) { |
|
||||
if (success) { |
|
||||
result.confirm(value); |
|
||||
} else { |
|
||||
result.cancel(); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Handle database quota exceeded notification. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onExceededDatabaseQuota(String url, String databaseIdentifier, long currentQuota, long estimatedSize, |
|
||||
long totalUsedQuota, WebStorage.QuotaUpdater quotaUpdater) |
|
||||
{ |
|
||||
LOG.d(LOG_TAG, "onExceededDatabaseQuota estimatedSize: %d currentQuota: %d totalUsedQuota: %d", estimatedSize, currentQuota, totalUsedQuota); |
|
||||
quotaUpdater.updateQuota(MAX_QUOTA); |
|
||||
} |
|
||||
|
|
||||
// console.log in api level 7: http://developer.android.com/guide/developing/debug-tasks.html |
|
||||
// Expect this to not compile in a future Android release! |
|
||||
@SuppressWarnings("deprecation") |
|
||||
@Override |
|
||||
public void onConsoleMessage(String message, int lineNumber, String sourceID) |
|
||||
{ |
|
||||
//This is only for Android 2.1 |
|
||||
if(android.os.Build.VERSION.SDK_INT == android.os.Build.VERSION_CODES.ECLAIR_MR1) |
|
||||
{ |
|
||||
LOG.d(LOG_TAG, "%s: Line %d : %s", sourceID, lineNumber, message); |
|
||||
super.onConsoleMessage(message, lineNumber, sourceID); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@TargetApi(8) |
|
||||
@Override |
|
||||
public boolean onConsoleMessage(ConsoleMessage consoleMessage) |
|
||||
{ |
|
||||
if (consoleMessage.message() != null) |
|
||||
LOG.d(LOG_TAG, "%s: Line %d : %s" , consoleMessage.sourceId() , consoleMessage.lineNumber(), consoleMessage.message()); |
|
||||
return super.onConsoleMessage(consoleMessage); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
/** |
|
||||
* Instructs the client to show a prompt to ask the user to set the Geolocation permission state for the specified origin. |
|
||||
* |
|
||||
* This also checks for the Geolocation Plugin and requests permission from the application to use Geolocation. |
|
||||
* |
|
||||
* @param origin |
|
||||
* @param callback |
|
||||
*/ |
|
||||
public void onGeolocationPermissionsShowPrompt(String origin, Callback callback) { |
|
||||
super.onGeolocationPermissionsShowPrompt(origin, callback); |
|
||||
callback.invoke(origin, true, false); |
|
||||
//Get the plugin, it should be loaded |
|
||||
CordovaPlugin geolocation = parentEngine.pluginManager.getPlugin("Geolocation"); |
|
||||
if(geolocation != null && !geolocation.hasPermisssion()) |
|
||||
{ |
|
||||
geolocation.requestPermissions(0); |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
|
|
||||
// API level 7 is required for this, see if we could lower this using something else |
|
||||
@Override |
|
||||
public void onShowCustomView(View view, WebChromeClient.CustomViewCallback callback) { |
|
||||
parentEngine.getCordovaWebView().showCustomView(view, callback); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void onHideCustomView() { |
|
||||
parentEngine.getCordovaWebView().hideCustomView(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
/** |
|
||||
* Ask the host application for a custom progress view to show while |
|
||||
* a <video> is loading. |
|
||||
* @return View The progress view. |
|
||||
*/ |
|
||||
public View getVideoLoadingProgressView() { |
|
||||
|
|
||||
if (mVideoProgressView == null) { |
|
||||
// Create a new Loading view programmatically. |
|
||||
|
|
||||
// create the linear layout |
|
||||
LinearLayout layout = new LinearLayout(parentEngine.getView().getContext()); |
|
||||
layout.setOrientation(LinearLayout.VERTICAL); |
|
||||
RelativeLayout.LayoutParams layoutParams = new RelativeLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT); |
|
||||
layoutParams.addRule(RelativeLayout.CENTER_IN_PARENT); |
|
||||
layout.setLayoutParams(layoutParams); |
|
||||
// the proress bar |
|
||||
ProgressBar bar = new ProgressBar(parentEngine.getView().getContext()); |
|
||||
LinearLayout.LayoutParams barLayoutParams = new LinearLayout.LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT); |
|
||||
barLayoutParams.gravity = Gravity.CENTER; |
|
||||
bar.setLayoutParams(barLayoutParams); |
|
||||
layout.addView(bar); |
|
||||
|
|
||||
mVideoProgressView = layout; |
|
||||
} |
|
||||
return mVideoProgressView; |
|
||||
} |
|
||||
|
|
||||
// <input type=file> support: |
|
||||
// openFileChooser() is for pre KitKat and in KitKat mr1 (it's known broken in KitKat). |
|
||||
// For Lollipop, we use onShowFileChooser(). |
|
||||
public void openFileChooser(ValueCallback<Uri> uploadMsg) { |
|
||||
this.openFileChooser(uploadMsg, "*/*"); |
|
||||
} |
|
||||
|
|
||||
public void openFileChooser( ValueCallback<Uri> uploadMsg, String acceptType ) { |
|
||||
this.openFileChooser(uploadMsg, acceptType, null); |
|
||||
} |
|
||||
|
|
||||
public void openFileChooser(final ValueCallback<Uri> uploadMsg, String acceptType, String capture) |
|
||||
{ |
|
||||
Intent intent = new Intent(Intent.ACTION_GET_CONTENT); |
|
||||
intent.addCategory(Intent.CATEGORY_OPENABLE); |
|
||||
intent.setType("*/*"); |
|
||||
parentEngine.cordova.startActivityForResult(new CordovaPlugin() { |
|
||||
@Override |
|
||||
public void onActivityResult(int requestCode, int resultCode, Intent intent) { |
|
||||
Uri result = intent == null || resultCode != Activity.RESULT_OK ? null : intent.getData(); |
|
||||
Log.d(LOG_TAG, "Receive file chooser URL: " + result); |
|
||||
uploadMsg.onReceiveValue(result); |
|
||||
} |
|
||||
}, intent, FILECHOOSER_RESULTCODE); |
|
||||
} |
|
||||
|
|
||||
@TargetApi(Build.VERSION_CODES.LOLLIPOP) |
|
||||
@Override |
|
||||
public boolean onShowFileChooser(WebView webView, final ValueCallback<Uri[]> filePathsCallback, final WebChromeClient.FileChooserParams fileChooserParams) { |
|
||||
Intent intent = fileChooserParams.createIntent(); |
|
||||
try { |
|
||||
parentEngine.cordova.startActivityForResult(new CordovaPlugin() { |
|
||||
@Override |
|
||||
public void onActivityResult(int requestCode, int resultCode, Intent intent) { |
|
||||
Uri[] result = WebChromeClient.FileChooserParams.parseResult(resultCode, intent); |
|
||||
Log.d(LOG_TAG, "Receive file chooser URL: " + result); |
|
||||
filePathsCallback.onReceiveValue(result); |
|
||||
} |
|
||||
}, intent, FILECHOOSER_RESULTCODE); |
|
||||
} catch (ActivityNotFoundException e) { |
|
||||
Log.w("No activity found to handle file chooser intent.", e); |
|
||||
filePathsCallback.onReceiveValue(null); |
|
||||
} |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
@TargetApi(Build.VERSION_CODES.LOLLIPOP) |
|
||||
@Override |
|
||||
public void onPermissionRequest(final PermissionRequest request) { |
|
||||
Log.d(LOG_TAG, "onPermissionRequest: " + Arrays.toString(request.getResources())); |
|
||||
request.grant(request.getResources()); |
|
||||
} |
|
||||
|
|
||||
public void destroyLastDialog(){ |
|
||||
dialogsHelper.destroyLastDialog(); |
|
||||
} |
|
||||
} |
|
@ -1,88 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import android.content.Context; |
|
||||
import android.util.AttributeSet; |
|
||||
import android.view.KeyEvent; |
|
||||
import android.webkit.WebChromeClient; |
|
||||
import android.webkit.WebView; |
|
||||
import android.webkit.WebViewClient; |
|
||||
|
|
||||
import org.apache.cordova.CordovaInterface; |
|
||||
import org.apache.cordova.CordovaWebView; |
|
||||
import org.apache.cordova.CordovaWebViewEngine; |
|
||||
|
|
||||
/** |
|
||||
* Custom WebView subclass that enables us to capture events needed for Cordova. |
|
||||
*/ |
|
||||
public class SystemWebView extends WebView implements CordovaWebViewEngine.EngineView { |
|
||||
private SystemWebViewClient viewClient; |
|
||||
SystemWebChromeClient chromeClient; |
|
||||
private SystemWebViewEngine parentEngine; |
|
||||
private CordovaInterface cordova; |
|
||||
|
|
||||
public SystemWebView(Context context) { |
|
||||
this(context, null); |
|
||||
} |
|
||||
|
|
||||
public SystemWebView(Context context, AttributeSet attrs) { |
|
||||
super(context, attrs); |
|
||||
} |
|
||||
|
|
||||
// Package visibility to enforce that only SystemWebViewEngine should call this method. |
|
||||
void init(SystemWebViewEngine parentEngine, CordovaInterface cordova) { |
|
||||
this.cordova = cordova; |
|
||||
this.parentEngine = parentEngine; |
|
||||
if (this.viewClient == null) { |
|
||||
setWebViewClient(new SystemWebViewClient(parentEngine)); |
|
||||
} |
|
||||
|
|
||||
if (this.chromeClient == null) { |
|
||||
setWebChromeClient(new SystemWebChromeClient(parentEngine)); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public CordovaWebView getCordovaWebView() { |
|
||||
return parentEngine != null ? parentEngine.getCordovaWebView() : null; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void setWebViewClient(WebViewClient client) { |
|
||||
viewClient = (SystemWebViewClient)client; |
|
||||
super.setWebViewClient(client); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void setWebChromeClient(WebChromeClient client) { |
|
||||
chromeClient = (SystemWebChromeClient)client; |
|
||||
super.setWebChromeClient(client); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean dispatchKeyEvent(KeyEvent event) { |
|
||||
Boolean ret = parentEngine.client.onDispatchKeyEvent(event); |
|
||||
if (ret != null) { |
|
||||
return ret.booleanValue(); |
|
||||
} |
|
||||
return super.dispatchKeyEvent(event); |
|
||||
} |
|
||||
} |
|
@ -1,374 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import android.annotation.TargetApi; |
|
||||
import android.content.pm.ApplicationInfo; |
|
||||
import android.content.pm.PackageManager; |
|
||||
import android.content.pm.PackageManager.NameNotFoundException; |
|
||||
import android.graphics.Bitmap; |
|
||||
import android.net.Uri; |
|
||||
import android.net.http.SslError; |
|
||||
import android.os.Build; |
|
||||
import android.webkit.ClientCertRequest; |
|
||||
import android.webkit.HttpAuthHandler; |
|
||||
import android.webkit.SslErrorHandler; |
|
||||
import android.webkit.WebResourceResponse; |
|
||||
import android.webkit.WebView; |
|
||||
import android.webkit.WebViewClient; |
|
||||
|
|
||||
import org.apache.cordova.AuthenticationToken; |
|
||||
import org.apache.cordova.CordovaClientCertRequest; |
|
||||
import org.apache.cordova.CordovaHttpAuthHandler; |
|
||||
import org.apache.cordova.CordovaResourceApi; |
|
||||
import org.apache.cordova.LOG; |
|
||||
import org.apache.cordova.PluginManager; |
|
||||
|
|
||||
import java.io.FileNotFoundException; |
|
||||
import java.io.IOException; |
|
||||
import java.util.Hashtable; |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* This class is the WebViewClient that implements callbacks for our web view. |
|
||||
* The kind of callbacks that happen here are regarding the rendering of the |
|
||||
* document instead of the chrome surrounding it, such as onPageStarted(), |
|
||||
* shouldOverrideUrlLoading(), etc. Related to but different than |
|
||||
* CordovaChromeClient. |
|
||||
*/ |
|
||||
public class SystemWebViewClient extends WebViewClient { |
|
||||
|
|
||||
private static final String TAG = "SystemWebViewClient"; |
|
||||
protected final SystemWebViewEngine parentEngine; |
|
||||
private boolean doClearHistory = false; |
|
||||
boolean isCurrentlyLoading; |
|
||||
|
|
||||
/** The authorization tokens. */ |
|
||||
private Hashtable<String, AuthenticationToken> authenticationTokens = new Hashtable<String, AuthenticationToken>(); |
|
||||
|
|
||||
public SystemWebViewClient(SystemWebViewEngine parentEngine) { |
|
||||
this.parentEngine = parentEngine; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Give the host application a chance to take over the control when a new url |
|
||||
* is about to be loaded in the current WebView. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback. |
|
||||
* @param url The url to be loaded. |
|
||||
* @return true to override, false for default behavior |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean shouldOverrideUrlLoading(WebView view, String url) { |
|
||||
return parentEngine.client.onNavigationAttempt(url); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* On received http auth request. |
|
||||
* The method reacts on all registered authentication tokens. There is one and only one authentication token for any host + realm combination |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onReceivedHttpAuthRequest(WebView view, HttpAuthHandler handler, String host, String realm) { |
|
||||
|
|
||||
// Get the authentication token (if specified) |
|
||||
AuthenticationToken token = this.getAuthenticationToken(host, realm); |
|
||||
if (token != null) { |
|
||||
handler.proceed(token.getUserName(), token.getPassword()); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
// Check if there is some plugin which can resolve this auth challenge |
|
||||
PluginManager pluginManager = this.parentEngine.pluginManager; |
|
||||
if (pluginManager != null && pluginManager.onReceivedHttpAuthRequest(null, new CordovaHttpAuthHandler(handler), host, realm)) { |
|
||||
parentEngine.client.clearLoadTimeoutTimer(); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
// By default handle 401 like we'd normally do! |
|
||||
super.onReceivedHttpAuthRequest(view, handler, host, realm); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* On received client cert request. |
|
||||
* The method forwards the request to any running plugins before using the default implementation. |
|
||||
* |
|
||||
* @param view |
|
||||
* @param request |
|
||||
*/ |
|
||||
@Override |
|
||||
@TargetApi(21) |
|
||||
public void onReceivedClientCertRequest (WebView view, ClientCertRequest request) |
|
||||
{ |
|
||||
|
|
||||
// Check if there is some plugin which can resolve this certificate request |
|
||||
PluginManager pluginManager = this.parentEngine.pluginManager; |
|
||||
if (pluginManager != null && pluginManager.onReceivedClientCertRequest(null, new CordovaClientCertRequest(request))) { |
|
||||
parentEngine.client.clearLoadTimeoutTimer(); |
|
||||
return; |
|
||||
} |
|
||||
|
|
||||
// By default pass to WebViewClient |
|
||||
super.onReceivedClientCertRequest(view, request); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Notify the host application that a page has started loading. |
|
||||
* This method is called once for each main frame load so a page with iframes or framesets will call onPageStarted |
|
||||
* one time for the main frame. This also means that onPageStarted will not be called when the contents of an |
|
||||
* embedded frame changes, i.e. clicking a link whose target is an iframe. |
|
||||
* |
|
||||
* @param view The webview initiating the callback. |
|
||||
* @param url The url of the page. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onPageStarted(WebView view, String url, Bitmap favicon) { |
|
||||
super.onPageStarted(view, url, favicon); |
|
||||
isCurrentlyLoading = true; |
|
||||
// Flush stale messages & reset plugins. |
|
||||
parentEngine.bridge.reset(); |
|
||||
parentEngine.client.onPageStarted(url); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Notify the host application that a page has finished loading. |
|
||||
* This method is called only for main frame. When onPageFinished() is called, the rendering picture may not be updated yet. |
|
||||
* |
|
||||
* |
|
||||
* @param view The webview initiating the callback. |
|
||||
* @param url The url of the page. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onPageFinished(WebView view, String url) { |
|
||||
super.onPageFinished(view, url); |
|
||||
// Ignore excessive calls, if url is not about:blank (CB-8317). |
|
||||
if (!isCurrentlyLoading && !url.startsWith("about:")) { |
|
||||
return; |
|
||||
} |
|
||||
isCurrentlyLoading = false; |
|
||||
|
|
||||
/** |
|
||||
* Because of a timing issue we need to clear this history in onPageFinished as well as |
|
||||
* onPageStarted. However we only want to do this if the doClearHistory boolean is set to |
|
||||
* true. You see when you load a url with a # in it which is common in jQuery applications |
|
||||
* onPageStared is not called. Clearing the history at that point would break jQuery apps. |
|
||||
*/ |
|
||||
if (this.doClearHistory) { |
|
||||
view.clearHistory(); |
|
||||
this.doClearHistory = false; |
|
||||
} |
|
||||
parentEngine.client.onPageFinishedLoading(url); |
|
||||
|
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Report an error to the host application. These errors are unrecoverable (i.e. the main resource is unavailable). |
|
||||
* The errorCode parameter corresponds to one of the ERROR_* constants. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback. |
|
||||
* @param errorCode The error code corresponding to an ERROR_* value. |
|
||||
* @param description A String describing the error. |
|
||||
* @param failingUrl The url that failed to load. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void onReceivedError(WebView view, int errorCode, String description, String failingUrl) { |
|
||||
// Ignore error due to stopLoading(). |
|
||||
if (!isCurrentlyLoading) { |
|
||||
return; |
|
||||
} |
|
||||
LOG.d(TAG, "CordovaWebViewClient.onReceivedError: Error code=%s Description=%s URL=%s", errorCode, description, failingUrl); |
|
||||
|
|
||||
// If this is a "Protocol Not Supported" error, then revert to the previous |
|
||||
// page. If there was no previous page, then punt. The application's config |
|
||||
// is likely incorrect (start page set to sms: or something like that) |
|
||||
if (errorCode == WebViewClient.ERROR_UNSUPPORTED_SCHEME) { |
|
||||
parentEngine.client.clearLoadTimeoutTimer(); |
|
||||
|
|
||||
if (view.canGoBack()) { |
|
||||
view.goBack(); |
|
||||
return; |
|
||||
} else { |
|
||||
super.onReceivedError(view, errorCode, description, failingUrl); |
|
||||
} |
|
||||
} |
|
||||
parentEngine.client.onReceivedError(errorCode, description, failingUrl); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Notify the host application that an SSL error occurred while loading a resource. |
|
||||
* The host application must call either handler.cancel() or handler.proceed(). |
|
||||
* Note that the decision may be retained for use in response to future SSL errors. |
|
||||
* The default behavior is to cancel the load. |
|
||||
* |
|
||||
* @param view The WebView that is initiating the callback. |
|
||||
* @param handler An SslErrorHandler object that will handle the user's response. |
|
||||
* @param error The SSL error object. |
|
||||
*/ |
|
||||
@TargetApi(8) |
|
||||
@Override |
|
||||
public void onReceivedSslError(WebView view, SslErrorHandler handler, SslError error) { |
|
||||
|
|
||||
final String packageName = parentEngine.cordova.getActivity().getPackageName(); |
|
||||
final PackageManager pm = parentEngine.cordova.getActivity().getPackageManager(); |
|
||||
|
|
||||
ApplicationInfo appInfo; |
|
||||
try { |
|
||||
appInfo = pm.getApplicationInfo(packageName, PackageManager.GET_META_DATA); |
|
||||
if ((appInfo.flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0) { |
|
||||
// debug = true |
|
||||
handler.proceed(); |
|
||||
return; |
|
||||
} else { |
|
||||
// debug = false |
|
||||
super.onReceivedSslError(view, handler, error); |
|
||||
} |
|
||||
} catch (NameNotFoundException e) { |
|
||||
// When it doubt, lock it out! |
|
||||
super.onReceivedSslError(view, handler, error); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Sets the authentication token. |
|
||||
* |
|
||||
* @param authenticationToken |
|
||||
* @param host |
|
||||
* @param realm |
|
||||
*/ |
|
||||
public void setAuthenticationToken(AuthenticationToken authenticationToken, String host, String realm) { |
|
||||
if (host == null) { |
|
||||
host = ""; |
|
||||
} |
|
||||
if (realm == null) { |
|
||||
realm = ""; |
|
||||
} |
|
||||
this.authenticationTokens.put(host.concat(realm), authenticationToken); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Removes the authentication token. |
|
||||
* |
|
||||
* @param host |
|
||||
* @param realm |
|
||||
* |
|
||||
* @return the authentication token or null if did not exist |
|
||||
*/ |
|
||||
public AuthenticationToken removeAuthenticationToken(String host, String realm) { |
|
||||
return this.authenticationTokens.remove(host.concat(realm)); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Gets the authentication token. |
|
||||
* |
|
||||
* In order it tries: |
|
||||
* 1- host + realm |
|
||||
* 2- host |
|
||||
* 3- realm |
|
||||
* 4- no host, no realm |
|
||||
* |
|
||||
* @param host |
|
||||
* @param realm |
|
||||
* |
|
||||
* @return the authentication token |
|
||||
*/ |
|
||||
public AuthenticationToken getAuthenticationToken(String host, String realm) { |
|
||||
AuthenticationToken token = null; |
|
||||
token = this.authenticationTokens.get(host.concat(realm)); |
|
||||
|
|
||||
if (token == null) { |
|
||||
// try with just the host |
|
||||
token = this.authenticationTokens.get(host); |
|
||||
|
|
||||
// Try the realm |
|
||||
if (token == null) { |
|
||||
token = this.authenticationTokens.get(realm); |
|
||||
} |
|
||||
|
|
||||
// if no host found, just query for default |
|
||||
if (token == null) { |
|
||||
token = this.authenticationTokens.get(""); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
return token; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Clear all authentication tokens. |
|
||||
*/ |
|
||||
public void clearAuthenticationTokens() { |
|
||||
this.authenticationTokens.clear(); |
|
||||
} |
|
||||
|
|
||||
@TargetApi(Build.VERSION_CODES.HONEYCOMB) |
|
||||
@Override |
|
||||
public WebResourceResponse shouldInterceptRequest(WebView view, String url) { |
|
||||
try { |
|
||||
// Check the against the whitelist and lock out access to the WebView directory |
|
||||
// Changing this will cause problems for your application |
|
||||
if (!parentEngine.pluginManager.shouldAllowRequest(url)) { |
|
||||
LOG.w(TAG, "URL blocked by whitelist: " + url); |
|
||||
// Results in a 404. |
|
||||
return new WebResourceResponse("text/plain", "UTF-8", null); |
|
||||
} |
|
||||
|
|
||||
CordovaResourceApi resourceApi = parentEngine.resourceApi; |
|
||||
Uri origUri = Uri.parse(url); |
|
||||
// Allow plugins to intercept WebView requests. |
|
||||
Uri remappedUri = resourceApi.remapUri(origUri); |
|
||||
|
|
||||
if (!origUri.equals(remappedUri) || needsSpecialsInAssetUrlFix(origUri) || needsKitKatContentUrlFix(origUri)) { |
|
||||
CordovaResourceApi.OpenForReadResult result = resourceApi.openForRead(remappedUri, true); |
|
||||
return new WebResourceResponse(result.mimeType, "UTF-8", result.inputStream); |
|
||||
} |
|
||||
// If we don't need to special-case the request, let the browser load it. |
|
||||
return null; |
|
||||
} catch (IOException e) { |
|
||||
if (!(e instanceof FileNotFoundException)) { |
|
||||
LOG.e(TAG, "Error occurred while loading a file (returning a 404).", e); |
|
||||
} |
|
||||
// Results in a 404. |
|
||||
return new WebResourceResponse("text/plain", "UTF-8", null); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private static boolean needsKitKatContentUrlFix(Uri uri) { |
|
||||
return android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT && "content".equals(uri.getScheme()); |
|
||||
} |
|
||||
|
|
||||
private static boolean needsSpecialsInAssetUrlFix(Uri uri) { |
|
||||
if (CordovaResourceApi.getUriType(uri) != CordovaResourceApi.URI_TYPE_ASSET) { |
|
||||
return false; |
|
||||
} |
|
||||
if (uri.getQuery() != null || uri.getFragment() != null) { |
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
if (!uri.toString().contains("%")) { |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
switch(android.os.Build.VERSION.SDK_INT){ |
|
||||
case android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH: |
|
||||
case android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1: |
|
||||
return true; |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
} |
|
@ -1,334 +0,0 @@ |
|||||
/* |
|
||||
Licensed to the Apache Software Foundation (ASF) under one |
|
||||
or more contributor license agreements. See the NOTICE file |
|
||||
distributed with this work for additional information |
|
||||
regarding copyright ownership. The ASF licenses this file |
|
||||
to you under the Apache License, Version 2.0 (the |
|
||||
"License"); you may not use this file except in compliance |
|
||||
with the License. You may obtain a copy of the License at |
|
||||
|
|
||||
http://www.apache.org/licenses/LICENSE-2.0 |
|
||||
|
|
||||
Unless required by applicable law or agreed to in writing, |
|
||||
software distributed under the License is distributed on an |
|
||||
"AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
KIND, either express or implied. See the License for the |
|
||||
specific language governing permissions and limitations |
|
||||
under the License. |
|
||||
*/ |
|
||||
|
|
||||
package org.apache.cordova.engine; |
|
||||
|
|
||||
import android.annotation.SuppressLint; |
|
||||
import android.annotation.TargetApi; |
|
||||
import android.content.BroadcastReceiver; |
|
||||
import android.content.Context; |
|
||||
import android.content.Intent; |
|
||||
import android.content.IntentFilter; |
|
||||
import android.content.pm.ApplicationInfo; |
|
||||
import android.os.Build; |
|
||||
import android.util.Log; |
|
||||
import android.view.View; |
|
||||
import android.webkit.WebSettings; |
|
||||
import android.webkit.WebSettings.LayoutAlgorithm; |
|
||||
import android.webkit.WebView; |
|
||||
|
|
||||
import org.apache.cordova.CordovaBridge; |
|
||||
import org.apache.cordova.CordovaInterface; |
|
||||
import org.apache.cordova.CordovaPreferences; |
|
||||
import org.apache.cordova.CordovaResourceApi; |
|
||||
import org.apache.cordova.CordovaWebView; |
|
||||
import org.apache.cordova.CordovaWebViewEngine; |
|
||||
import org.apache.cordova.ICordovaCookieManager; |
|
||||
import org.apache.cordova.NativeToJsMessageQueue; |
|
||||
import org.apache.cordova.PluginManager; |
|
||||
|
|
||||
import java.lang.reflect.InvocationTargetException; |
|
||||
import java.lang.reflect.Method; |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Glue class between CordovaWebView (main Cordova logic) and SystemWebView (the actual View). |
|
||||
* We make the Engine separate from the actual View so that: |
|
||||
* A) We don't need to worry about WebView methods clashing with CordovaWebViewEngine methods |
|
||||
* (e.g.: goBack() is void for WebView, and boolean for CordovaWebViewEngine) |
|
||||
* B) Separating the actual View from the Engine makes API surfaces smaller. |
|
||||
* Class uses two-phase initialization. However, CordovaWebView is responsible for calling .init(). |
|
||||
*/ |
|
||||
public class SystemWebViewEngine implements CordovaWebViewEngine { |
|
||||
public static final String TAG = "SystemWebViewEngine"; |
|
||||
|
|
||||
protected final SystemWebView webView; |
|
||||
protected final SystemCookieManager cookieManager; |
|
||||
protected CordovaPreferences preferences; |
|
||||
protected CordovaBridge bridge; |
|
||||
protected CordovaWebViewEngine.Client client; |
|
||||
protected CordovaWebView parentWebView; |
|
||||
protected CordovaInterface cordova; |
|
||||
protected PluginManager pluginManager; |
|
||||
protected CordovaResourceApi resourceApi; |
|
||||
protected NativeToJsMessageQueue nativeToJsMessageQueue; |
|
||||
private BroadcastReceiver receiver; |
|
||||
|
|
||||
/** Used when created via reflection. */ |
|
||||
public SystemWebViewEngine(Context context, CordovaPreferences preferences) { |
|
||||
this(new SystemWebView(context), preferences); |
|
||||
} |
|
||||
|
|
||||
public SystemWebViewEngine(SystemWebView webView) { |
|
||||
this(webView, null); |
|
||||
} |
|
||||
|
|
||||
public SystemWebViewEngine(SystemWebView webView, CordovaPreferences preferences) { |
|
||||
this.preferences = preferences; |
|
||||
this.webView = webView; |
|
||||
cookieManager = new SystemCookieManager(webView); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void init(CordovaWebView parentWebView, CordovaInterface cordova, CordovaWebViewEngine.Client client, |
|
||||
CordovaResourceApi resourceApi, PluginManager pluginManager, |
|
||||
NativeToJsMessageQueue nativeToJsMessageQueue) { |
|
||||
if (this.cordova != null) { |
|
||||
throw new IllegalStateException(); |
|
||||
} |
|
||||
// Needed when prefs are not passed by the constructor |
|
||||
if (preferences == null) { |
|
||||
preferences = parentWebView.getPreferences(); |
|
||||
} |
|
||||
this.parentWebView = parentWebView; |
|
||||
this.cordova = cordova; |
|
||||
this.client = client; |
|
||||
this.resourceApi = resourceApi; |
|
||||
this.pluginManager = pluginManager; |
|
||||
this.nativeToJsMessageQueue = nativeToJsMessageQueue; |
|
||||
webView.init(this, cordova); |
|
||||
|
|
||||
initWebViewSettings(); |
|
||||
|
|
||||
nativeToJsMessageQueue.addBridgeMode(new NativeToJsMessageQueue.OnlineEventsBridgeMode(new NativeToJsMessageQueue.OnlineEventsBridgeMode.OnlineEventsBridgeModeDelegate() { |
|
||||
@Override |
|
||||
public void setNetworkAvailable(boolean value) { |
|
||||
webView.setNetworkAvailable(value); |
|
||||
} |
|
||||
@Override |
|
||||
public void runOnUiThread(Runnable r) { |
|
||||
SystemWebViewEngine.this.cordova.getActivity().runOnUiThread(r); |
|
||||
} |
|
||||
})); |
|
||||
bridge = new CordovaBridge(pluginManager, nativeToJsMessageQueue); |
|
||||
exposeJsInterface(webView, bridge); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public CordovaWebView getCordovaWebView() { |
|
||||
return parentWebView; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public ICordovaCookieManager getCookieManager() { |
|
||||
return cookieManager; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public View getView() { |
|
||||
return webView; |
|
||||
} |
|
||||
|
|
||||
@SuppressLint({"NewApi", "SetJavaScriptEnabled"}) |
|
||||
@SuppressWarnings("deprecation") |
|
||||
private void initWebViewSettings() { |
|
||||
webView.setInitialScale(0); |
|
||||
webView.setVerticalScrollBarEnabled(false); |
|
||||
// Enable JavaScript |
|
||||
final WebSettings settings = webView.getSettings(); |
|
||||
settings.setJavaScriptEnabled(true); |
|
||||
settings.setJavaScriptCanOpenWindowsAutomatically(true); |
|
||||
settings.setLayoutAlgorithm(LayoutAlgorithm.NORMAL); |
|
||||
|
|
||||
// Set the nav dump for HTC 2.x devices (disabling for ICS, deprecated entirely for Jellybean 4.2) |
|
||||
try { |
|
||||
Method gingerbread_getMethod = WebSettings.class.getMethod("setNavDump", new Class[] { boolean.class }); |
|
||||
|
|
||||
String manufacturer = android.os.Build.MANUFACTURER; |
|
||||
Log.d(TAG, "CordovaWebView is running on device made by: " + manufacturer); |
|
||||
if(android.os.Build.VERSION.SDK_INT < android.os.Build.VERSION_CODES.HONEYCOMB && |
|
||||
android.os.Build.MANUFACTURER.contains("HTC")) |
|
||||
{ |
|
||||
gingerbread_getMethod.invoke(settings, true); |
|
||||
} |
|
||||
} catch (NoSuchMethodException e) { |
|
||||
Log.d(TAG, "We are on a modern version of Android, we will deprecate HTC 2.3 devices in 2.8"); |
|
||||
} catch (IllegalArgumentException e) { |
|
||||
Log.d(TAG, "Doing the NavDump failed with bad arguments"); |
|
||||
} catch (IllegalAccessException e) { |
|
||||
Log.d(TAG, "This should never happen: IllegalAccessException means this isn't Android anymore"); |
|
||||
} catch (InvocationTargetException e) { |
|
||||
Log.d(TAG, "This should never happen: InvocationTargetException means this isn't Android anymore."); |
|
||||
} |
|
||||
|
|
||||
//We don't save any form data in the application |
|
||||
settings.setSaveFormData(false); |
|
||||
settings.setSavePassword(false); |
|
||||
|
|
||||
// Jellybean rightfully tried to lock this down. Too bad they didn't give us a whitelist |
|
||||
// while we do this |
|
||||
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN) { |
|
||||
settings.setAllowUniversalAccessFromFileURLs(true); |
|
||||
} |
|
||||
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR1) { |
|
||||
settings.setMediaPlaybackRequiresUserGesture(false); |
|
||||
} |
|
||||
// Enable database |
|
||||
// We keep this disabled because we use or shim to get around DOM_EXCEPTION_ERROR_16 |
|
||||
String databasePath = webView.getContext().getApplicationContext().getDir("database", Context.MODE_PRIVATE).getPath(); |
|
||||
settings.setDatabaseEnabled(true); |
|
||||
settings.setDatabasePath(databasePath); |
|
||||
|
|
||||
|
|
||||
//Determine whether we're in debug or release mode, and turn on Debugging! |
|
||||
ApplicationInfo appInfo = webView.getContext().getApplicationContext().getApplicationInfo(); |
|
||||
if ((appInfo.flags & ApplicationInfo.FLAG_DEBUGGABLE) != 0 && |
|
||||
android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.KITKAT) { |
|
||||
enableRemoteDebugging(); |
|
||||
} |
|
||||
|
|
||||
settings.setGeolocationDatabasePath(databasePath); |
|
||||
|
|
||||
// Enable DOM storage |
|
||||
settings.setDomStorageEnabled(true); |
|
||||
|
|
||||
// Enable built-in geolocation |
|
||||
settings.setGeolocationEnabled(true); |
|
||||
|
|
||||
// Enable AppCache |
|
||||
// Fix for CB-2282 |
|
||||
settings.setAppCacheMaxSize(5 * 1048576); |
|
||||
settings.setAppCachePath(databasePath); |
|
||||
settings.setAppCacheEnabled(true); |
|
||||
|
|
||||
// Fix for CB-1405 |
|
||||
// Google issue 4641 |
|
||||
String defaultUserAgent = settings.getUserAgentString(); |
|
||||
|
|
||||
// Fix for CB-3360 |
|
||||
String overrideUserAgent = preferences.getString("OverrideUserAgent", null); |
|
||||
if (overrideUserAgent != null) { |
|
||||
settings.setUserAgentString(overrideUserAgent); |
|
||||
} else { |
|
||||
String appendUserAgent = preferences.getString("AppendUserAgent", null); |
|
||||
if (appendUserAgent != null) { |
|
||||
settings.setUserAgentString(defaultUserAgent + " " + appendUserAgent); |
|
||||
} |
|
||||
} |
|
||||
// End CB-3360 |
|
||||
|
|
||||
IntentFilter intentFilter = new IntentFilter(); |
|
||||
intentFilter.addAction(Intent.ACTION_CONFIGURATION_CHANGED); |
|
||||
if (this.receiver == null) { |
|
||||
this.receiver = new BroadcastReceiver() { |
|
||||
@Override |
|
||||
public void onReceive(Context context, Intent intent) { |
|
||||
settings.getUserAgentString(); |
|
||||
} |
|
||||
}; |
|
||||
webView.getContext().registerReceiver(this.receiver, intentFilter); |
|
||||
} |
|
||||
// end CB-1405 |
|
||||
} |
|
||||
|
|
||||
@TargetApi(Build.VERSION_CODES.KITKAT) |
|
||||
private void enableRemoteDebugging() { |
|
||||
try { |
|
||||
WebView.setWebContentsDebuggingEnabled(true); |
|
||||
} catch (IllegalArgumentException e) { |
|
||||
Log.d(TAG, "You have one job! To turn on Remote Web Debugging! YOU HAVE FAILED! "); |
|
||||
e.printStackTrace(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
private static void exposeJsInterface(WebView webView, CordovaBridge bridge) { |
|
||||
if ((Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR1)) { |
|
||||
Log.i(TAG, "Disabled addJavascriptInterface() bridge since Android version is old."); |
|
||||
// Bug being that Java Strings do not get converted to JS strings automatically. |
|
||||
// This isn't hard to work-around on the JS side, but it's easier to just |
|
||||
// use the prompt bridge instead. |
|
||||
return; |
|
||||
} |
|
||||
SystemExposedJsApi exposedJsApi = new SystemExposedJsApi(bridge); |
|
||||
webView.addJavascriptInterface(exposedJsApi, "_cordovaNative"); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
/** |
|
||||
* Load the url into the webview. |
|
||||
*/ |
|
||||
@Override |
|
||||
public void loadUrl(final String url, boolean clearNavigationStack) { |
|
||||
webView.loadUrl(url); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public String getUrl() { |
|
||||
return webView.getUrl(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void stopLoading() { |
|
||||
webView.stopLoading(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void clearCache() { |
|
||||
webView.clearCache(true); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void clearHistory() { |
|
||||
webView.clearHistory(); |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public boolean canGoBack() { |
|
||||
return webView.canGoBack(); |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Go to previous page in history. (We manage our own history) |
|
||||
* |
|
||||
* @return true if we went back, false if we are already at top |
|
||||
*/ |
|
||||
@Override |
|
||||
public boolean goBack() { |
|
||||
// Check webview first to see if there is a history |
|
||||
// This is needed to support curPage#diffLink, since they are added to parentEngine's history, but not our history url array (JQMobile behavior) |
|
||||
if (webView.canGoBack()) { |
|
||||
webView.goBack(); |
|
||||
return true; |
|
||||
} |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void setPaused(boolean value) { |
|
||||
if (value) { |
|
||||
webView.pauseTimers(); |
|
||||
} else { |
|
||||
webView.resumeTimers(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
@Override |
|
||||
public void destroy() { |
|
||||
webView.chromeClient.destroyLastDialog(); |
|
||||
webView.destroy(); |
|
||||
// unregister the receiver |
|
||||
if (receiver != null) { |
|
||||
try { |
|
||||
webView.getContext().unregisterReceiver(receiver); |
|
||||
} catch (Exception e) { |
|
||||
Log.e(TAG, "Error unregistering configuration receiver: " + e.getMessage(), e); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
} |
|
@ -1,100 +0,0 @@ |
|||||
{ |
|
||||
"prepare_queue": { |
|
||||
"installed": [], |
|
||||
"uninstalled": [] |
|
||||
}, |
|
||||
"config_munge": { |
|
||||
"files": { |
|
||||
"res/xml/config.xml": { |
|
||||
"parents": { |
|
||||
"/*": [ |
|
||||
{ |
|
||||
"xml": "<feature name=\"Device\"><param name=\"android-package\" value=\"org.apache.cordova.device.Device\" /></feature>", |
|
||||
"count": 1 |
|
||||
}, |
|
||||
{ |
|
||||
"xml": "<feature name=\"StatusBar\"><param name=\"android-package\" value=\"org.apache.cordova.statusbar.StatusBar\" /><param name=\"onload\" value=\"true\" /></feature>", |
|
||||
"count": 1 |
|
||||
}, |
|
||||
{ |
|
||||
"xml": "<feature name=\"Whitelist\"><param name=\"android-package\" value=\"org.apache.cordova.whitelist.WhitelistPlugin\" /><param name=\"onload\" value=\"true\" /></feature>", |
|
||||
"count": 1 |
|
||||
}, |
|
||||
{ |
|
||||
"xml": "<feature name=\"SplashScreen\"><param name=\"android-package\" value=\"org.apache.cordova.splashscreen.SplashScreen\" /><param name=\"onload\" value=\"true\" /></feature>", |
|
||||
"count": 1 |
|
||||
}, |
|
||||
{ |
|
||||
"xml": "<feature name=\"Keyboard\"><param name=\"android-package\" value=\"io.ionic.keyboard.IonicKeyboard\" /><param name=\"onload\" value=\"true\" /></feature>", |
|
||||
"count": 1 |
|
||||
} |
|
||||
] |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
}, |
|
||||
"installed_plugins": { |
|
||||
"cordova-plugin-device": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
}, |
|
||||
"cordova-plugin-statusbar": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
}, |
|
||||
"cordova-plugin-whitelist": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
}, |
|
||||
"cordova-plugin-splashscreen": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
}, |
|
||||
"cordova-plugin-console": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
}, |
|
||||
"ionic-plugin-keyboard": { |
|
||||
"PACKAGE_NAME": "com.ionicframework.app751563" |
|
||||
} |
|
||||
}, |
|
||||
"dependent_plugins": {}, |
|
||||
"modules": [ |
|
||||
{ |
|
||||
"id": "cordova-plugin-device.device", |
|
||||
"file": "plugins/cordova-plugin-device/www/device.js", |
|
||||
"pluginId": "cordova-plugin-device", |
|
||||
"clobbers": [ |
|
||||
"device" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "cordova-plugin-statusbar.statusbar", |
|
||||
"file": "plugins/cordova-plugin-statusbar/www/statusbar.js", |
|
||||
"pluginId": "cordova-plugin-statusbar", |
|
||||
"clobbers": [ |
|
||||
"window.StatusBar" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "cordova-plugin-splashscreen.SplashScreen", |
|
||||
"file": "plugins/cordova-plugin-splashscreen/www/splashscreen.js", |
|
||||
"pluginId": "cordova-plugin-splashscreen", |
|
||||
"clobbers": [ |
|
||||
"navigator.splashscreen" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "ionic-plugin-keyboard.keyboard", |
|
||||
"file": "plugins/ionic-plugin-keyboard/www/android/keyboard.js", |
|
||||
"pluginId": "ionic-plugin-keyboard", |
|
||||
"clobbers": [ |
|
||||
"cordova.plugins.Keyboard" |
|
||||
], |
|
||||
"runs": true |
|
||||
} |
|
||||
], |
|
||||
"plugin_metadata": { |
|
||||
"cordova-plugin-device": "1.1.3", |
|
||||
"cordova-plugin-statusbar": "2.2.0", |
|
||||
"cordova-plugin-whitelist": "1.3.0", |
|
||||
"cordova-plugin-splashscreen": "4.0.0", |
|
||||
"cordova-plugin-console": "1.0.4", |
|
||||
"ionic-plugin-keyboard": "2.2.1" |
|
||||
} |
|
||||
} |
|
@ -1,36 +0,0 @@ |
|||||
/* |
|
||||
* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
* or more contributor license agreements. See the NOTICE file |
|
||||
* distributed with this work for additional information |
|
||||
* regarding copyright ownership. The ASF licenses this file |
|
||||
* to you under the Apache License, Version 2.0 (the |
|
||||
* "License"); you may not use this file except in compliance |
|
||||
* with the License. You may obtain a copy of the License at |
|
||||
* |
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||
* |
|
||||
* Unless required by applicable law or agreed to in writing, |
|
||||
* software distributed under the License is distributed on an |
|
||||
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
* KIND, either express or implied. See the License for the |
|
||||
* specific language governing permissions and limitations |
|
||||
* under the License. |
|
||||
*/ |
|
||||
|
|
||||
/** |
|
||||
* Exports the ExposedJsApi.java object if available, otherwise exports the PromptBasedNativeApi. |
|
||||
*/ |
|
||||
|
|
||||
var nativeApi = this._cordovaNative || require('cordova/android/promptbasednativeapi'); |
|
||||
var currentApi = nativeApi; |
|
||||
|
|
||||
module.exports = { |
|
||||
get: function() { return currentApi; }, |
|
||||
setPreferPrompt: function(value) { |
|
||||
currentApi = value ? require('cordova/android/promptbasednativeapi') : nativeApi; |
|
||||
}, |
|
||||
// Used only by tests.
|
|
||||
set: function(value) { |
|
||||
currentApi = value; |
|
||||
} |
|
||||
}; |
|
@ -1,35 +0,0 @@ |
|||||
/* |
|
||||
* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
* or more contributor license agreements. See the NOTICE file |
|
||||
* distributed with this work for additional information |
|
||||
* regarding copyright ownership. The ASF licenses this file |
|
||||
* to you under the Apache License, Version 2.0 (the |
|
||||
* "License"); you may not use this file except in compliance |
|
||||
* with the License. You may obtain a copy of the License at |
|
||||
* |
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||
* |
|
||||
* Unless required by applicable law or agreed to in writing, |
|
||||
* software distributed under the License is distributed on an |
|
||||
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
* KIND, either express or implied. See the License for the |
|
||||
* specific language governing permissions and limitations |
|
||||
* under the License. |
|
||||
*/ |
|
||||
|
|
||||
/** |
|
||||
* Implements the API of ExposedJsApi.java, but uses prompt() to communicate. |
|
||||
* This is used pre-JellyBean, where addJavascriptInterface() is disabled. |
|
||||
*/ |
|
||||
|
|
||||
module.exports = { |
|
||||
exec: function(bridgeSecret, service, action, callbackId, argsJson) { |
|
||||
return prompt(argsJson, 'gap:'+JSON.stringify([bridgeSecret, service, action, callbackId])); |
|
||||
}, |
|
||||
setNativeToJsBridgeMode: function(bridgeSecret, value) { |
|
||||
prompt(value, 'gap_bridge_mode:' + bridgeSecret); |
|
||||
}, |
|
||||
retrieveJsMessages: function(bridgeSecret, fromOnlineEvent) { |
|
||||
return prompt(+fromOnlineEvent, 'gap_poll:' + bridgeSecret); |
|
||||
} |
|
||||
}; |
|
@ -1,283 +0,0 @@ |
|||||
/* |
|
||||
* |
|
||||
* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
* or more contributor license agreements. See the NOTICE file |
|
||||
* distributed with this work for additional information |
|
||||
* regarding copyright ownership. The ASF licenses this file |
|
||||
* to you under the Apache License, Version 2.0 (the |
|
||||
* "License"); you may not use this file except in compliance |
|
||||
* with the License. You may obtain a copy of the License at |
|
||||
* |
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||
* |
|
||||
* Unless required by applicable law or agreed to in writing, |
|
||||
* software distributed under the License is distributed on an |
|
||||
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
* KIND, either express or implied. See the License for the |
|
||||
* specific language governing permissions and limitations |
|
||||
* under the License. |
|
||||
* |
|
||||
*/ |
|
||||
|
|
||||
/** |
|
||||
* Execute a cordova command. It is up to the native side whether this action |
|
||||
* is synchronous or asynchronous. The native side can return: |
|
||||
* Synchronous: PluginResult object as a JSON string |
|
||||
* Asynchronous: Empty string "" |
|
||||
* If async, the native side will cordova.callbackSuccess or cordova.callbackError, |
|
||||
* depending upon the result of the action. |
|
||||
* |
|
||||
* @param {Function} success The success callback |
|
||||
* @param {Function} fail The fail callback |
|
||||
* @param {String} service The name of the service to use |
|
||||
* @param {String} action Action to be run in cordova |
|
||||
* @param {String[]} [args] Zero or more arguments to pass to the method |
|
||||
*/ |
|
||||
var cordova = require('cordova'), |
|
||||
nativeApiProvider = require('cordova/android/nativeapiprovider'), |
|
||||
utils = require('cordova/utils'), |
|
||||
base64 = require('cordova/base64'), |
|
||||
channel = require('cordova/channel'), |
|
||||
jsToNativeModes = { |
|
||||
PROMPT: 0, |
|
||||
JS_OBJECT: 1 |
|
||||
}, |
|
||||
nativeToJsModes = { |
|
||||
// Polls for messages using the JS->Native bridge.
|
|
||||
POLLING: 0, |
|
||||
// For LOAD_URL to be viable, it would need to have a work-around for
|
|
||||
// the bug where the soft-keyboard gets dismissed when a message is sent.
|
|
||||
LOAD_URL: 1, |
|
||||
// For the ONLINE_EVENT to be viable, it would need to intercept all event
|
|
||||
// listeners (both through addEventListener and window.ononline) as well
|
|
||||
// as set the navigator property itself.
|
|
||||
ONLINE_EVENT: 2 |
|
||||
}, |
|
||||
jsToNativeBridgeMode, // Set lazily.
|
|
||||
nativeToJsBridgeMode = nativeToJsModes.ONLINE_EVENT, |
|
||||
pollEnabled = false, |
|
||||
bridgeSecret = -1; |
|
||||
|
|
||||
var messagesFromNative = []; |
|
||||
var isProcessing = false; |
|
||||
var resolvedPromise = typeof Promise == 'undefined' ? null : Promise.resolve(); |
|
||||
var nextTick = resolvedPromise ? function(fn) { resolvedPromise.then(fn); } : function(fn) { setTimeout(fn); }; |
|
||||
|
|
||||
function androidExec(success, fail, service, action, args) { |
|
||||
if (bridgeSecret < 0) { |
|
||||
// If we ever catch this firing, we'll need to queue up exec()s
|
|
||||
// and fire them once we get a secret. For now, I don't think
|
|
||||
// it's possible for exec() to be called since plugins are parsed but
|
|
||||
// not run until until after onNativeReady.
|
|
||||
throw new Error('exec() called without bridgeSecret'); |
|
||||
} |
|
||||
// Set default bridge modes if they have not already been set.
|
|
||||
// By default, we use the failsafe, since addJavascriptInterface breaks too often
|
|
||||
if (jsToNativeBridgeMode === undefined) { |
|
||||
androidExec.setJsToNativeBridgeMode(jsToNativeModes.JS_OBJECT); |
|
||||
} |
|
||||
|
|
||||
// Process any ArrayBuffers in the args into a string.
|
|
||||
for (var i = 0; i < args.length; i++) { |
|
||||
if (utils.typeName(args[i]) == 'ArrayBuffer') { |
|
||||
args[i] = base64.fromArrayBuffer(args[i]); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
var callbackId = service + cordova.callbackId++, |
|
||||
argsJson = JSON.stringify(args); |
|
||||
|
|
||||
if (success || fail) { |
|
||||
cordova.callbacks[callbackId] = {success:success, fail:fail}; |
|
||||
} |
|
||||
|
|
||||
var msgs = nativeApiProvider.get().exec(bridgeSecret, service, action, callbackId, argsJson); |
|
||||
// If argsJson was received by Java as null, try again with the PROMPT bridge mode.
|
|
||||
// This happens in rare circumstances, such as when certain Unicode characters are passed over the bridge on a Galaxy S2. See CB-2666.
|
|
||||
if (jsToNativeBridgeMode == jsToNativeModes.JS_OBJECT && msgs === "@Null arguments.") { |
|
||||
androidExec.setJsToNativeBridgeMode(jsToNativeModes.PROMPT); |
|
||||
androidExec(success, fail, service, action, args); |
|
||||
androidExec.setJsToNativeBridgeMode(jsToNativeModes.JS_OBJECT); |
|
||||
} else if (msgs) { |
|
||||
messagesFromNative.push(msgs); |
|
||||
// Always process async to avoid exceptions messing up stack.
|
|
||||
nextTick(processMessages); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
androidExec.init = function() { |
|
||||
bridgeSecret = +prompt('', 'gap_init:' + nativeToJsBridgeMode); |
|
||||
channel.onNativeReady.fire(); |
|
||||
}; |
|
||||
|
|
||||
function pollOnceFromOnlineEvent() { |
|
||||
pollOnce(true); |
|
||||
} |
|
||||
|
|
||||
function pollOnce(opt_fromOnlineEvent) { |
|
||||
if (bridgeSecret < 0) { |
|
||||
// This can happen when the NativeToJsMessageQueue resets the online state on page transitions.
|
|
||||
// We know there's nothing to retrieve, so no need to poll.
|
|
||||
return; |
|
||||
} |
|
||||
var msgs = nativeApiProvider.get().retrieveJsMessages(bridgeSecret, !!opt_fromOnlineEvent); |
|
||||
if (msgs) { |
|
||||
messagesFromNative.push(msgs); |
|
||||
// Process sync since we know we're already top-of-stack.
|
|
||||
processMessages(); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
function pollingTimerFunc() { |
|
||||
if (pollEnabled) { |
|
||||
pollOnce(); |
|
||||
setTimeout(pollingTimerFunc, 50); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
function hookOnlineApis() { |
|
||||
function proxyEvent(e) { |
|
||||
cordova.fireWindowEvent(e.type); |
|
||||
} |
|
||||
// The network module takes care of firing online and offline events.
|
|
||||
// It currently fires them only on document though, so we bridge them
|
|
||||
// to window here (while first listening for exec()-releated online/offline
|
|
||||
// events).
|
|
||||
window.addEventListener('online', pollOnceFromOnlineEvent, false); |
|
||||
window.addEventListener('offline', pollOnceFromOnlineEvent, false); |
|
||||
cordova.addWindowEventHandler('online'); |
|
||||
cordova.addWindowEventHandler('offline'); |
|
||||
document.addEventListener('online', proxyEvent, false); |
|
||||
document.addEventListener('offline', proxyEvent, false); |
|
||||
} |
|
||||
|
|
||||
hookOnlineApis(); |
|
||||
|
|
||||
androidExec.jsToNativeModes = jsToNativeModes; |
|
||||
androidExec.nativeToJsModes = nativeToJsModes; |
|
||||
|
|
||||
androidExec.setJsToNativeBridgeMode = function(mode) { |
|
||||
if (mode == jsToNativeModes.JS_OBJECT && !window._cordovaNative) { |
|
||||
mode = jsToNativeModes.PROMPT; |
|
||||
} |
|
||||
nativeApiProvider.setPreferPrompt(mode == jsToNativeModes.PROMPT); |
|
||||
jsToNativeBridgeMode = mode; |
|
||||
}; |
|
||||
|
|
||||
androidExec.setNativeToJsBridgeMode = function(mode) { |
|
||||
if (mode == nativeToJsBridgeMode) { |
|
||||
return; |
|
||||
} |
|
||||
if (nativeToJsBridgeMode == nativeToJsModes.POLLING) { |
|
||||
pollEnabled = false; |
|
||||
} |
|
||||
|
|
||||
nativeToJsBridgeMode = mode; |
|
||||
// Tell the native side to switch modes.
|
|
||||
// Otherwise, it will be set by androidExec.init()
|
|
||||
if (bridgeSecret >= 0) { |
|
||||
nativeApiProvider.get().setNativeToJsBridgeMode(bridgeSecret, mode); |
|
||||
} |
|
||||
|
|
||||
if (mode == nativeToJsModes.POLLING) { |
|
||||
pollEnabled = true; |
|
||||
setTimeout(pollingTimerFunc, 1); |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
function buildPayload(payload, message) { |
|
||||
var payloadKind = message.charAt(0); |
|
||||
if (payloadKind == 's') { |
|
||||
payload.push(message.slice(1)); |
|
||||
} else if (payloadKind == 't') { |
|
||||
payload.push(true); |
|
||||
} else if (payloadKind == 'f') { |
|
||||
payload.push(false); |
|
||||
} else if (payloadKind == 'N') { |
|
||||
payload.push(null); |
|
||||
} else if (payloadKind == 'n') { |
|
||||
payload.push(+message.slice(1)); |
|
||||
} else if (payloadKind == 'A') { |
|
||||
var data = message.slice(1); |
|
||||
payload.push(base64.toArrayBuffer(data)); |
|
||||
} else if (payloadKind == 'S') { |
|
||||
payload.push(window.atob(message.slice(1))); |
|
||||
} else if (payloadKind == 'M') { |
|
||||
var multipartMessages = message.slice(1); |
|
||||
while (multipartMessages !== "") { |
|
||||
var spaceIdx = multipartMessages.indexOf(' '); |
|
||||
var msgLen = +multipartMessages.slice(0, spaceIdx); |
|
||||
var multipartMessage = multipartMessages.substr(spaceIdx + 1, msgLen); |
|
||||
multipartMessages = multipartMessages.slice(spaceIdx + msgLen + 1); |
|
||||
buildPayload(payload, multipartMessage); |
|
||||
} |
|
||||
} else { |
|
||||
payload.push(JSON.parse(message)); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
// Processes a single message, as encoded by NativeToJsMessageQueue.java.
|
|
||||
function processMessage(message) { |
|
||||
var firstChar = message.charAt(0); |
|
||||
if (firstChar == 'J') { |
|
||||
// This is deprecated on the .java side. It doesn't work with CSP enabled.
|
|
||||
eval(message.slice(1)); |
|
||||
} else if (firstChar == 'S' || firstChar == 'F') { |
|
||||
var success = firstChar == 'S'; |
|
||||
var keepCallback = message.charAt(1) == '1'; |
|
||||
var spaceIdx = message.indexOf(' ', 2); |
|
||||
var status = +message.slice(2, spaceIdx); |
|
||||
var nextSpaceIdx = message.indexOf(' ', spaceIdx + 1); |
|
||||
var callbackId = message.slice(spaceIdx + 1, nextSpaceIdx); |
|
||||
var payloadMessage = message.slice(nextSpaceIdx + 1); |
|
||||
var payload = []; |
|
||||
buildPayload(payload, payloadMessage); |
|
||||
cordova.callbackFromNative(callbackId, success, status, payload, keepCallback); |
|
||||
} else { |
|
||||
console.log("processMessage failed: invalid message: " + JSON.stringify(message)); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
function processMessages() { |
|
||||
// Check for the reentrant case.
|
|
||||
if (isProcessing) { |
|
||||
return; |
|
||||
} |
|
||||
if (messagesFromNative.length === 0) { |
|
||||
return; |
|
||||
} |
|
||||
isProcessing = true; |
|
||||
try { |
|
||||
var msg = popMessageFromQueue(); |
|
||||
// The Java side can send a * message to indicate that it
|
|
||||
// still has messages waiting to be retrieved.
|
|
||||
if (msg == '*' && messagesFromNative.length === 0) { |
|
||||
nextTick(pollOnce); |
|
||||
return; |
|
||||
} |
|
||||
processMessage(msg); |
|
||||
} finally { |
|
||||
isProcessing = false; |
|
||||
if (messagesFromNative.length > 0) { |
|
||||
nextTick(processMessages); |
|
||||
} |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
function popMessageFromQueue() { |
|
||||
var messageBatch = messagesFromNative.shift(); |
|
||||
if (messageBatch == '*') { |
|
||||
return '*'; |
|
||||
} |
|
||||
|
|
||||
var spaceIdx = messageBatch.indexOf(' '); |
|
||||
var msgLen = +messageBatch.slice(0, spaceIdx); |
|
||||
var message = messageBatch.substr(spaceIdx + 1, msgLen); |
|
||||
messageBatch = messageBatch.slice(spaceIdx + msgLen + 1); |
|
||||
if (messageBatch) { |
|
||||
messagesFromNative.unshift(messageBatch); |
|
||||
} |
|
||||
return message; |
|
||||
} |
|
||||
|
|
||||
module.exports = androidExec; |
|
@ -1,125 +0,0 @@ |
|||||
/* |
|
||||
* |
|
||||
* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
* or more contributor license agreements. See the NOTICE file |
|
||||
* distributed with this work for additional information |
|
||||
* regarding copyright ownership. The ASF licenses this file |
|
||||
* to you under the Apache License, Version 2.0 (the |
|
||||
* "License"); you may not use this file except in compliance |
|
||||
* with the License. You may obtain a copy of the License at |
|
||||
* |
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||
* |
|
||||
* Unless required by applicable law or agreed to in writing, |
|
||||
* software distributed under the License is distributed on an |
|
||||
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
* KIND, either express or implied. See the License for the |
|
||||
* specific language governing permissions and limitations |
|
||||
* under the License. |
|
||||
* |
|
||||
*/ |
|
||||
|
|
||||
// The last resume event that was received that had the result of a plugin call.
|
|
||||
var lastResumeEvent = null; |
|
||||
|
|
||||
module.exports = { |
|
||||
id: 'android', |
|
||||
bootstrap: function() { |
|
||||
var channel = require('cordova/channel'), |
|
||||
cordova = require('cordova'), |
|
||||
exec = require('cordova/exec'), |
|
||||
modulemapper = require('cordova/modulemapper'); |
|
||||
|
|
||||
// Get the shared secret needed to use the bridge.
|
|
||||
exec.init(); |
|
||||
|
|
||||
// TODO: Extract this as a proper plugin.
|
|
||||
modulemapper.clobbers('cordova/plugin/android/app', 'navigator.app'); |
|
||||
|
|
||||
var APP_PLUGIN_NAME = Number(cordova.platformVersion.split('.')[0]) >= 4 ? 'CoreAndroid' : 'App'; |
|
||||
|
|
||||
// Inject a listener for the backbutton on the document.
|
|
||||
var backButtonChannel = cordova.addDocumentEventHandler('backbutton'); |
|
||||
backButtonChannel.onHasSubscribersChange = function() { |
|
||||
// If we just attached the first handler or detached the last handler,
|
|
||||
// let native know we need to override the back button.
|
|
||||
exec(null, null, APP_PLUGIN_NAME, "overrideBackbutton", [this.numHandlers == 1]); |
|
||||
}; |
|
||||
|
|
||||
// Add hardware MENU and SEARCH button handlers
|
|
||||
cordova.addDocumentEventHandler('menubutton'); |
|
||||
cordova.addDocumentEventHandler('searchbutton'); |
|
||||
|
|
||||
function bindButtonChannel(buttonName) { |
|
||||
// generic button bind used for volumeup/volumedown buttons
|
|
||||
var volumeButtonChannel = cordova.addDocumentEventHandler(buttonName + 'button'); |
|
||||
volumeButtonChannel.onHasSubscribersChange = function() { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "overrideButton", [buttonName, this.numHandlers == 1]); |
|
||||
}; |
|
||||
} |
|
||||
// Inject a listener for the volume buttons on the document.
|
|
||||
bindButtonChannel('volumeup'); |
|
||||
bindButtonChannel('volumedown'); |
|
||||
|
|
||||
// The resume event is not "sticky", but it is possible that the event
|
|
||||
// will contain the result of a plugin call. We need to ensure that the
|
|
||||
// plugin result is delivered even after the event is fired (CB-10498)
|
|
||||
var cordovaAddEventListener = document.addEventListener; |
|
||||
|
|
||||
document.addEventListener = function(evt, handler, capture) { |
|
||||
cordovaAddEventListener(evt, handler, capture); |
|
||||
|
|
||||
if (evt === 'resume' && lastResumeEvent) { |
|
||||
handler(lastResumeEvent); |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
// Let native code know we are all done on the JS side.
|
|
||||
// Native code will then un-hide the WebView.
|
|
||||
channel.onCordovaReady.subscribe(function() { |
|
||||
exec(onMessageFromNative, null, APP_PLUGIN_NAME, 'messageChannel', []); |
|
||||
exec(null, null, APP_PLUGIN_NAME, "show", []); |
|
||||
}); |
|
||||
} |
|
||||
}; |
|
||||
|
|
||||
function onMessageFromNative(msg) { |
|
||||
var cordova = require('cordova'); |
|
||||
var action = msg.action; |
|
||||
|
|
||||
switch (action) |
|
||||
{ |
|
||||
// Button events
|
|
||||
case 'backbutton': |
|
||||
case 'menubutton': |
|
||||
case 'searchbutton': |
|
||||
// App life cycle events
|
|
||||
case 'pause': |
|
||||
// Volume events
|
|
||||
case 'volumedownbutton': |
|
||||
case 'volumeupbutton': |
|
||||
cordova.fireDocumentEvent(action); |
|
||||
break; |
|
||||
case 'resume': |
|
||||
if(arguments.length > 1 && msg.pendingResult) { |
|
||||
if(arguments.length === 2) { |
|
||||
msg.pendingResult.result = arguments[1]; |
|
||||
} else { |
|
||||
// The plugin returned a multipart message
|
|
||||
var res = []; |
|
||||
for(var i = 1; i < arguments.length; i++) { |
|
||||
res.push(arguments[i]); |
|
||||
} |
|
||||
msg.pendingResult.result = res; |
|
||||
} |
|
||||
|
|
||||
// Save the plugin result so that it can be delivered to the js
|
|
||||
// even if they miss the initial firing of the event
|
|
||||
lastResumeEvent = msg; |
|
||||
} |
|
||||
cordova.fireDocumentEvent(action, msg); |
|
||||
break; |
|
||||
default: |
|
||||
throw new Error('Unknown event action ' + action); |
|
||||
} |
|
||||
} |
|
@ -1,108 +0,0 @@ |
|||||
/* |
|
||||
* |
|
||||
* Licensed to the Apache Software Foundation (ASF) under one |
|
||||
* or more contributor license agreements. See the NOTICE file |
|
||||
* distributed with this work for additional information |
|
||||
* regarding copyright ownership. The ASF licenses this file |
|
||||
* to you under the Apache License, Version 2.0 (the |
|
||||
* "License"); you may not use this file except in compliance |
|
||||
* with the License. You may obtain a copy of the License at |
|
||||
* |
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||
* |
|
||||
* Unless required by applicable law or agreed to in writing, |
|
||||
* software distributed under the License is distributed on an |
|
||||
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
|
||||
* KIND, either express or implied. See the License for the |
|
||||
* specific language governing permissions and limitations |
|
||||
* under the License. |
|
||||
* |
|
||||
*/ |
|
||||
|
|
||||
var exec = require('cordova/exec'); |
|
||||
var APP_PLUGIN_NAME = Number(require('cordova').platformVersion.split('.')[0]) >= 4 ? 'CoreAndroid' : 'App'; |
|
||||
|
|
||||
module.exports = { |
|
||||
/** |
|
||||
* Clear the resource cache. |
|
||||
*/ |
|
||||
clearCache:function() { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "clearCache", []); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Load the url into the webview or into new browser instance. |
|
||||
* |
|
||||
* @param url The URL to load |
|
||||
* @param props Properties that can be passed in to the activity: |
|
||||
* wait: int => wait msec before loading URL |
|
||||
* loadingDialog: "Title,Message" => display a native loading dialog |
|
||||
* loadUrlTimeoutValue: int => time in msec to wait before triggering a timeout error |
|
||||
* clearHistory: boolean => clear webview history (default=false) |
|
||||
* openExternal: boolean => open in a new browser (default=false) |
|
||||
* |
|
||||
* Example: |
|
||||
* navigator.app.loadUrl("http://server/myapp/index.html", {wait:2000, loadingDialog:"Wait,Loading App", loadUrlTimeoutValue: 60000}); |
|
||||
*/ |
|
||||
loadUrl:function(url, props) { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "loadUrl", [url, props]); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Cancel loadUrl that is waiting to be loaded. |
|
||||
*/ |
|
||||
cancelLoadUrl:function() { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "cancelLoadUrl", []); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Clear web history in this web view. |
|
||||
* Instead of BACK button loading the previous web page, it will exit the app. |
|
||||
*/ |
|
||||
clearHistory:function() { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "clearHistory", []); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Go to previous page displayed. |
|
||||
* This is the same as pressing the backbutton on Android device. |
|
||||
*/ |
|
||||
backHistory:function() { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "backHistory", []); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Override the default behavior of the Android back button. |
|
||||
* If overridden, when the back button is pressed, the "backKeyDown" JavaScript event will be fired. |
|
||||
* |
|
||||
* Note: The user should not have to call this method. Instead, when the user |
|
||||
* registers for the "backbutton" event, this is automatically done. |
|
||||
* |
|
||||
* @param override T=override, F=cancel override |
|
||||
*/ |
|
||||
overrideBackbutton:function(override) { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "overrideBackbutton", [override]); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Override the default behavior of the Android volume button. |
|
||||
* If overridden, when the volume button is pressed, the "volume[up|down]button" |
|
||||
* JavaScript event will be fired. |
|
||||
* |
|
||||
* Note: The user should not have to call this method. Instead, when the user |
|
||||
* registers for the "volume[up|down]button" event, this is automatically done. |
|
||||
* |
|
||||
* @param button volumeup, volumedown |
|
||||
* @param override T=override, F=cancel override |
|
||||
*/ |
|
||||
overrideButton:function(button, override) { |
|
||||
exec(null, null, APP_PLUGIN_NAME, "overrideButton", [button, override]); |
|
||||
}, |
|
||||
|
|
||||
/** |
|
||||
* Exit and terminate the application. |
|
||||
*/ |
|
||||
exitApp:function() { |
|
||||
return exec(null, null, APP_PLUGIN_NAME, "exitApp", []); |
|
||||
} |
|
||||
}; |
|
@ -1,48 +0,0 @@ |
|||||
cordova.define('cordova/plugin_list', function(require, exports, module) { |
|
||||
module.exports = [ |
|
||||
{ |
|
||||
"id": "cordova-plugin-device.device", |
|
||||
"file": "plugins/cordova-plugin-device/www/device.js", |
|
||||
"pluginId": "cordova-plugin-device", |
|
||||
"clobbers": [ |
|
||||
"device" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "cordova-plugin-statusbar.statusbar", |
|
||||
"file": "plugins/cordova-plugin-statusbar/www/statusbar.js", |
|
||||
"pluginId": "cordova-plugin-statusbar", |
|
||||
"clobbers": [ |
|
||||
"window.StatusBar" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "cordova-plugin-splashscreen.SplashScreen", |
|
||||
"file": "plugins/cordova-plugin-splashscreen/www/splashscreen.js", |
|
||||
"pluginId": "cordova-plugin-splashscreen", |
|
||||
"clobbers": [ |
|
||||
"navigator.splashscreen" |
|
||||
] |
|
||||
}, |
|
||||
{ |
|
||||
"id": "ionic-plugin-keyboard.keyboard", |
|
||||
"file": "plugins/ionic-plugin-keyboard/www/android/keyboard.js", |
|
||||
"pluginId": "ionic-plugin-keyboard", |
|
||||
"clobbers": [ |
|
||||
"cordova.plugins.Keyboard" |
|
||||
], |
|
||||
"runs": true |
|
||||
} |
|
||||
]; |
|
||||
module.exports.metadata = |
|
||||
// TOP OF METADATA
|
|
||||
{ |
|
||||
"cordova-plugin-device": "1.1.3", |
|
||||
"cordova-plugin-statusbar": "2.2.0", |
|
||||
"cordova-plugin-whitelist": "1.3.0", |
|
||||
"cordova-plugin-splashscreen": "4.0.0", |
|
||||
"cordova-plugin-console": "1.0.4", |
|
||||
"ionic-plugin-keyboard": "2.2.1" |
|
||||
}; |
|
||||
// BOTTOM OF METADATA
|
|
||||
}); |
|
@ -1,46 +0,0 @@ |
|||||
/* Empty. Add your own CSS if you like */ |
|
||||
.o-imgMenu{ |
|
||||
width: 25px!important; |
|
||||
height: 25px!important; |
|
||||
} |
|
||||
.o-imgTitle{ |
|
||||
width: 40px!important; |
|
||||
height: 40px!important; |
|
||||
} |
|
||||
.o-bold{ |
|
||||
font-weight: bold!important; |
|
||||
} |
|
||||
.o-float-right{ |
|
||||
float: right; |
|
||||
} |
|
||||
.o-text-right{ |
|
||||
text-align: right; |
|
||||
} |
|
||||
.o-mini-text{ |
|
||||
font-size: 12px; |
|
||||
display: inline-block; |
|
||||
} |
|
||||
.o-img-joined{ |
|
||||
width: 20px!important; |
|
||||
height: 20px!important; |
|
||||
} |
|
||||
.o-img-new{ |
|
||||
width: 30px!important; |
|
||||
height: 30px!important; |
|
||||
} |
|
||||
.o-badgeCollectivized{ |
|
||||
background: #33CD5F; |
|
||||
padding: 5px; |
|
||||
border-radius: 5px; |
|
||||
font-size: 12px; |
|
||||
color: #ffffff; |
|
||||
font-weight: bold; |
|
||||
} |
|
||||
.o-badge-calm{ |
|
||||
background: #11C1F3; |
|
||||
padding: 5px; |
|
||||
border-radius: 7px; |
|
||||
font-size: 12px; |
|
||||
color: #ffffff; |
|
||||
font-weight: bold; |
|
||||
} |
|