init
This commit is contained in:
commit
939a7aff4c
32
.editorconfig
Normal file
32
.editorconfig
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
; http://editorconfig.org
|
||||||
|
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
|
charset = utf-8
|
||||||
|
end_of_line = lf
|
||||||
|
insert_final_newline = true
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
[*.txt]
|
||||||
|
insert_final_newline = false
|
||||||
|
trim_trailing_whitespace = false
|
||||||
|
|
||||||
|
[*.py]
|
||||||
|
indent_size = 4
|
||||||
|
|
||||||
|
[*.m]
|
||||||
|
indent_size = 4
|
||||||
|
|
||||||
|
[Makefile]
|
||||||
|
indent_style = tab
|
||||||
|
indent_size = 8
|
||||||
|
|
||||||
|
[*.{js,json}]
|
||||||
|
indent_style = space
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
[*.md]
|
||||||
|
trim_trailing_whitespace = false
|
180
.gitignore
vendored
Normal file
180
.gitignore
vendored
Normal file
@ -0,0 +1,180 @@
|
|||||||
|
# Created by .ignore support plugin (hsz.mobi)
|
||||||
|
### Archives template
|
||||||
|
# It's better to unpack these files and commit the raw source because
|
||||||
|
# git has its own built in compression methods.
|
||||||
|
*.7z
|
||||||
|
*.jar
|
||||||
|
*.rar
|
||||||
|
*.zip
|
||||||
|
*.gz
|
||||||
|
*.bzip
|
||||||
|
*.bz2
|
||||||
|
*.xz
|
||||||
|
*.lzma
|
||||||
|
*.cab
|
||||||
|
|
||||||
|
#packing-only formats
|
||||||
|
*.iso
|
||||||
|
*.tar
|
||||||
|
|
||||||
|
#package management formats
|
||||||
|
*.dmg
|
||||||
|
*.xpi
|
||||||
|
*.gem
|
||||||
|
*.egg
|
||||||
|
*.deb
|
||||||
|
*.rpm
|
||||||
|
*.msi
|
||||||
|
*.msm
|
||||||
|
*.msp
|
||||||
|
### Windows template
|
||||||
|
# Windows image file caches
|
||||||
|
Thumbs.db
|
||||||
|
ehthumbs.db
|
||||||
|
|
||||||
|
# Folder config file
|
||||||
|
Desktop.ini
|
||||||
|
|
||||||
|
# Recycle Bin used on file shares
|
||||||
|
$RECYCLE.BIN/
|
||||||
|
|
||||||
|
# Windows Installer files
|
||||||
|
*.cab
|
||||||
|
*.msi
|
||||||
|
*.msm
|
||||||
|
*.msp
|
||||||
|
|
||||||
|
# Windows shortcuts
|
||||||
|
*.lnk
|
||||||
|
### OSX template
|
||||||
|
.DS_Store
|
||||||
|
.AppleDouble
|
||||||
|
.LSOverride
|
||||||
|
|
||||||
|
# Icon must end with two \r
|
||||||
|
Icon
|
||||||
|
|
||||||
|
# Thumbnails
|
||||||
|
._*
|
||||||
|
|
||||||
|
# Files that might appear in the root of a volume
|
||||||
|
.DocumentRevisions-V100
|
||||||
|
.fseventsd
|
||||||
|
.Spotlight-V100
|
||||||
|
.TemporaryItems
|
||||||
|
.Trashes
|
||||||
|
.VolumeIcon.icns
|
||||||
|
|
||||||
|
# Directories potentially created on remote AFP share
|
||||||
|
.AppleDB
|
||||||
|
.AppleDesktop
|
||||||
|
Network Trash Folder
|
||||||
|
Temporary Items
|
||||||
|
.apdisk
|
||||||
|
### JetBrains template
|
||||||
|
# Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio
|
||||||
|
|
||||||
|
*.iml
|
||||||
|
|
||||||
|
## Directory-based project format:
|
||||||
|
.idea/
|
||||||
|
# if you remove the above rule, at least ignore the following:
|
||||||
|
|
||||||
|
# User-specific stuff:
|
||||||
|
# .idea/workspace.xml
|
||||||
|
# .idea/tasks.xml
|
||||||
|
# .idea/dictionaries
|
||||||
|
|
||||||
|
# Sensitive or high-churn files:
|
||||||
|
# .idea/dataSources.ids
|
||||||
|
# .idea/dataSources.xml
|
||||||
|
# .idea/sqlDataSources.xml
|
||||||
|
# .idea/dynamic.xml
|
||||||
|
# .idea/uiDesigner.xml
|
||||||
|
|
||||||
|
# Gradle:
|
||||||
|
# .idea/gradle.xml
|
||||||
|
# .idea/libraries
|
||||||
|
|
||||||
|
# Mongo Explorer plugin:
|
||||||
|
# .idea/mongoSettings.xml
|
||||||
|
|
||||||
|
## File-based project format:
|
||||||
|
*.ipr
|
||||||
|
*.iws
|
||||||
|
|
||||||
|
## Plugin-specific files:
|
||||||
|
|
||||||
|
# IntelliJ
|
||||||
|
/out/
|
||||||
|
|
||||||
|
# mpeltonen/sbt-idea plugin
|
||||||
|
.idea_modules/
|
||||||
|
|
||||||
|
# JIRA plugin
|
||||||
|
atlassian-ide-plugin.xml
|
||||||
|
|
||||||
|
# Crashlytics plugin (for Android Studio and IntelliJ)
|
||||||
|
com_crashlytics_export_strings.xml
|
||||||
|
crashlytics.properties
|
||||||
|
crashlytics-build.properties
|
||||||
|
### Node template
|
||||||
|
# Logs
|
||||||
|
logs
|
||||||
|
*.log
|
||||||
|
npm-debug.log*
|
||||||
|
|
||||||
|
# Runtime data
|
||||||
|
pids
|
||||||
|
*.pid
|
||||||
|
*.seed
|
||||||
|
|
||||||
|
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||||
|
lib-cov
|
||||||
|
|
||||||
|
# Coverage directory used by tools like istanbul
|
||||||
|
coverage
|
||||||
|
|
||||||
|
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
|
||||||
|
.grunt
|
||||||
|
|
||||||
|
# node-waf configuration
|
||||||
|
.lock-wscript
|
||||||
|
|
||||||
|
# Compiled binary addons (http://nodejs.org/api/addons.html)
|
||||||
|
build/Release
|
||||||
|
|
||||||
|
# Dependency directory
|
||||||
|
# https://docs.npmjs.com/misc/faq#should-i-check-my-node-modules-folder-into-git
|
||||||
|
node_modules
|
||||||
|
bower_components
|
||||||
|
|
||||||
|
### VisualStudioCode template
|
||||||
|
.settings
|
||||||
|
|
||||||
|
### Xcode template
|
||||||
|
# Xcode
|
||||||
|
#
|
||||||
|
# gitignore contributors: remember to update Global/Xcode.gitignore, Objective-C.gitignore & Swift.gitignore
|
||||||
|
|
||||||
|
## Build generated
|
||||||
|
build/
|
||||||
|
DerivedData
|
||||||
|
|
||||||
|
## Various settings
|
||||||
|
*.pbxuser
|
||||||
|
!default.pbxuser
|
||||||
|
*.mode1v3
|
||||||
|
!default.mode1v3
|
||||||
|
*.mode2v3
|
||||||
|
!default.mode2v3
|
||||||
|
*.perspectivev3
|
||||||
|
!default.perspectivev3
|
||||||
|
xcuserdata
|
||||||
|
|
||||||
|
## Other
|
||||||
|
*.xccheckout
|
||||||
|
*.moved-aside
|
||||||
|
*.xcuserstate
|
||||||
|
|
||||||
|
dist
|
46
.jscsrc
Normal file
46
.jscsrc
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
{
|
||||||
|
"disallowKeywords": ["with"],
|
||||||
|
"disallowKeywordsOnNewLine": ["else"],
|
||||||
|
"disallowMixedSpacesAndTabs": true,
|
||||||
|
"disallowMultipleVarDecl": "exceptUndefined",
|
||||||
|
"disallowNewlineBeforeBlockStatements": true,
|
||||||
|
"disallowQuotedKeysInObjects": true,
|
||||||
|
"disallowSpaceAfterObjectKeys": true,
|
||||||
|
"disallowSpaceAfterPrefixUnaryOperators": true,
|
||||||
|
"disallowSpacesInFunction": {
|
||||||
|
"beforeOpeningRoundBrace": true
|
||||||
|
},
|
||||||
|
"disallowSpacesInsideParentheses": true,
|
||||||
|
"disallowTrailingWhitespace": true,
|
||||||
|
"maximumLineLength": 160,
|
||||||
|
"requireCamelCaseOrUpperCaseIdentifiers": false,
|
||||||
|
"requireCapitalizedComments": true,
|
||||||
|
"requireCapitalizedConstructors": true,
|
||||||
|
"requireCurlyBraces": true,
|
||||||
|
"requireSpaceAfterKeywords": [
|
||||||
|
"if",
|
||||||
|
"else",
|
||||||
|
"for",
|
||||||
|
"while",
|
||||||
|
"do",
|
||||||
|
"switch",
|
||||||
|
"case",
|
||||||
|
"return",
|
||||||
|
"try",
|
||||||
|
"catch",
|
||||||
|
"typeof"
|
||||||
|
],
|
||||||
|
"requireSpaceAfterLineComment": true,
|
||||||
|
"requireSpaceAfterBinaryOperators": true,
|
||||||
|
"requireSpaceBeforeBinaryOperators": true,
|
||||||
|
"requireSpaceBeforeBlockStatements": true,
|
||||||
|
"requireSpaceBeforeObjectValues": true,
|
||||||
|
"requireSpacesInFunction": {
|
||||||
|
"beforeOpeningCurlyBrace": true
|
||||||
|
},
|
||||||
|
"requireTrailingComma": false,
|
||||||
|
"requireEarlyReturn": false,
|
||||||
|
"validateIndentation": 2,
|
||||||
|
"validateLineBreaks": "LF",
|
||||||
|
"validateQuoteMarks": "'"
|
||||||
|
}
|
37
.jshintrc
Normal file
37
.jshintrc
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
{
|
||||||
|
"predef": [
|
||||||
|
"Promise",
|
||||||
|
"$"
|
||||||
|
],
|
||||||
|
"globals": {
|
||||||
|
"$": false,
|
||||||
|
"MicroEvent": false
|
||||||
|
},
|
||||||
|
"node":true,
|
||||||
|
"browser": true,
|
||||||
|
"boss": true,
|
||||||
|
"curly": true,
|
||||||
|
"debug": false,
|
||||||
|
"devel": true,
|
||||||
|
"eqeqeq": true,
|
||||||
|
"evil": true,
|
||||||
|
"forin": false,
|
||||||
|
"immed": false,
|
||||||
|
"laxbreak": false,
|
||||||
|
"newcap": true,
|
||||||
|
"noarg": true,
|
||||||
|
"noempty": false,
|
||||||
|
"nonew": false,
|
||||||
|
"nomen": false,
|
||||||
|
"onevar": false,
|
||||||
|
"plusplus": false,
|
||||||
|
"regexp": false,
|
||||||
|
"undef": true,
|
||||||
|
"sub": true,
|
||||||
|
"strict": false,
|
||||||
|
"white": false,
|
||||||
|
"eqnull": true,
|
||||||
|
"esnext": true,
|
||||||
|
"unused": true,
|
||||||
|
"supernew":true
|
||||||
|
}
|
23
bower.json
Normal file
23
bower.json
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
{
|
||||||
|
"name": "sodashserver",
|
||||||
|
"description": "Smart Office Dashboard Server",
|
||||||
|
"main": "index.js",
|
||||||
|
"authors": [
|
||||||
|
"Martin Donnelly"
|
||||||
|
],
|
||||||
|
"license": "ISC",
|
||||||
|
"homepage": "",
|
||||||
|
"private": true,
|
||||||
|
"ignore": [
|
||||||
|
"**/.*",
|
||||||
|
"node_modules",
|
||||||
|
"bower_components",
|
||||||
|
"test",
|
||||||
|
"tests"
|
||||||
|
],
|
||||||
|
"dependencies": {
|
||||||
|
"jquery": "^2.2.3",
|
||||||
|
"mui": "^0.5.3",
|
||||||
|
"bluebird": "^3.4.0"
|
||||||
|
}
|
||||||
|
}
|
29
config.xml
Normal file
29
config.xml
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
<?xml version='1.0' encoding='utf-8'?>
|
||||||
|
<widget id="org.censis.sensortoy" version="0.0.1" xmlns="http://www.w3.org/ns/widgets" xmlns:cdv="http://cordova.apache.org/ns/1.0">
|
||||||
|
<name>Sensortoy</name>
|
||||||
|
<description>
|
||||||
|
A sample Apache Cordova application that responds to the deviceready event.
|
||||||
|
</description>
|
||||||
|
<author email="dev@cordova.apache.org" href="http://cordova.io">
|
||||||
|
Apache Cordova Team
|
||||||
|
</author>
|
||||||
|
<content src="index.html" />
|
||||||
|
<plugin name="cordova-plugin-whitelist" spec="1" />
|
||||||
|
<access origin="*" />
|
||||||
|
<allow-intent href="http://*/*" />
|
||||||
|
<allow-intent href="https://*/*" />
|
||||||
|
<allow-intent href="tel:*" />
|
||||||
|
<allow-intent href="sms:*" />
|
||||||
|
<allow-intent href="mailto:*" />
|
||||||
|
<allow-intent href="geo:*" />
|
||||||
|
<platform name="android">
|
||||||
|
<allow-intent href="market:*" />
|
||||||
|
</platform>
|
||||||
|
<platform name="ios">
|
||||||
|
<allow-intent href="itms:*" />
|
||||||
|
<allow-intent href="itms-apps:*" />
|
||||||
|
</platform>
|
||||||
|
<engine name="ios" spec="~4.1.1" />
|
||||||
|
<engine name="android" spec="~5.1.1" />
|
||||||
|
<plugin name="cordova-plugin-ble-central" spec="~1.1.0" />
|
||||||
|
</widget>
|
23
hooks/README.md
Normal file
23
hooks/README.md
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
<!--
|
||||||
|
#
|
||||||
|
# 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
|
||||||
|
|
||||||
|
Cordova Hooks represent special scripts which could be added by application and plugin developers or even by your own build system to customize cordova commands. See Hooks Guide for more details: http://cordova.apache.org/docs/en/edge/guide_appdev_hooks_index.md.html#Hooks%20Guide.
|
53
output.txt
Normal file
53
output.txt
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
cc2650_accelerometer.js:83 Gyro <br/>X: -0.15259<br/>Y: -0.2899169921875<br/>Z: -0.0152587890625<br/>Accel <br/>X: 0.00390625<br/>Y: 0.0062255859375<br/>Z: 0.25042724609375<br/>Mag <br/>X: 64<br/>Y: 102<br/>Z: 4103<br/>
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 34 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15394
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000100010
|
||||||
|
cc2650_luxometer.js:68 111111111000100010
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
||||||
|
cc2650_barometer.js:59 ArrayBuffer {}
|
||||||
|
cc2650_barometer.js:76 Barometer: Temperature <br/>26.14°CPressure <br/>1004.11hPa
|
||||||
|
cc2650_accelerometer.js:40 ArrayBuffer {}
|
||||||
|
cc2650_accelerometer.js:83 Gyro <br/>X: -0.24414<br/>Y: -0.38909912109375<br/>Z: 0.1068115234375<br/>Accel <br/>X: 0.00372314453125<br/>Y: 0.0062255859375<br/>Z: 0.2489013671875<br/>Mag <br/>X: 61<br/>Y: 102<br/>Z: 4078<br/>
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 30 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15390
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000011110
|
||||||
|
cc2650_luxometer.js:68 111111110111111110
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
||||||
|
cc2650_barometer.js:59 ArrayBuffer {}
|
||||||
|
cc2650_barometer.js:76 Barometer: Temperature <br/>26.14°CPressure <br/>1004.1hPa
|
||||||
|
cc2650_accelerometer.js:40 ArrayBuffer {}
|
||||||
|
cc2650_accelerometer.js:83 Gyro <br/>X: 0.05341<br/>Y: -0.25177001953125<br/>Z: 0.2288818359375<br/>Accel <br/>X: 0.00421142578125<br/>Y: 0.0068359375<br/>Z: 0.25067138671875<br/>Mag <br/>X: 69<br/>Y: 112<br/>Z: 4107<br/>
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 26 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15386
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000011010
|
||||||
|
cc2650_luxometer.js:68 111111110110111010
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 22 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15382
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000010110
|
||||||
|
cc2650_luxometer.js:68 111111110101110110
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
||||||
|
cc2650_barometer.js:59 ArrayBuffer {}
|
||||||
|
cc2650_barometer.js:76 Barometer: Temperature <br/>26.15°CPressure <br/>1004.09hPa
|
||||||
|
cc2650_accelerometer.js:40 ArrayBuffer {}
|
||||||
|
cc2650_accelerometer.js:83 Gyro <br/>X: -0.12970<br/>Y: -0.3204345703125<br/>Z: 0.0457763671875<br/>Accel <br/>X: 0.0035400390625<br/>Y: 0.00775146484375<br/>Z: 0.24798583984375<br/>Mag <br/>X: 58<br/>Y: 127<br/>Z: 4063<br/>
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 18 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15378
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000010010
|
||||||
|
cc2650_luxometer.js:68 111111110100110010
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
||||||
|
cc2650_barometer.js:59 ArrayBuffer {}
|
||||||
|
cc2650_barometer.js:76 Barometer: Temperature <br/>26.14°CPressure <br/>1004.09hPa
|
||||||
|
cc2650_accelerometer.js:40 ArrayBuffer {}
|
||||||
|
cc2650_accelerometer.js:83 Gyro <br/>X: -0.06104<br/>Y: -0.4119873046875<br/>Z: 0.11444091796875<br/>Accel <br/>X: 0.00494384765625<br/>Y: 0.00634765625<br/>Z: 0.2486572265625<br/>Mag <br/>X: 81<br/>Y: 104<br/>Z: 4074<br/>
|
||||||
|
cc2650_luxometer.js:58 ArrayBuffer {}
|
||||||
|
cc2650_luxometer.js:63 lux a 30 60
|
||||||
|
cc2650_luxometer.js:64 lux b 15390
|
||||||
|
cc2650_luxometer.js:65 lux b 11110000011110
|
||||||
|
cc2650_luxometer.js:68 111111110111111110
|
||||||
|
cc2650_luxometer.js:73 Luxometer: undefined
|
14
platforms/android/.gitignore
vendored
Normal file
14
platforms/android/.gitignore
vendored
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# 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
|
@ -0,0 +1 @@
|
|||||||
|
#Wed May 18 14:10:20 BST 2016
|
Binary file not shown.
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/fileHashes.bin
Normal file
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/fileHashes.bin
Normal file
Binary file not shown.
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/fileSnapshots.bin
Normal file
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/fileSnapshots.bin
Normal file
Binary file not shown.
Binary file not shown.
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/taskArtifacts.bin
Normal file
BIN
platforms/android/.gradle/2.2.1/taskArtifacts/taskArtifacts.bin
Normal file
Binary file not shown.
17
platforms/android/AndroidManifest.xml
Normal file
17
platforms/android/AndroidManifest.xml
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<?xml version='1.0' encoding='utf-8'?>
|
||||||
|
<manifest android:hardwareAccelerated="true" android:versionCode="1" android:versionName="0.0.1" package="org.censis.sensortoy" 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="14" android:targetSdkVersion="23" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||||
|
<uses-permission android:name="android.permission.BLUETOOTH" />
|
||||||
|
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
||||||
|
</manifest>
|
23
platforms/android/CordovaLib/AndroidManifest.xml
Executable file
23
platforms/android/CordovaLib/AndroidManifest.xml
Executable file
@ -0,0 +1,23 @@
|
|||||||
|
<?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>
|
61
platforms/android/CordovaLib/build.gradle
Normal file
61
platforms/android/CordovaLib/build.gradle
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
/* 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:1.5.0'
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
apply plugin: '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']
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
201
platforms/android/CordovaLib/cordova.gradle
Normal file
201
platforms/android/CordovaLib/cordova.gradle
Normal file
@ -0,0 +1,201 @@
|
|||||||
|
/*
|
||||||
|
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. Please 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) }
|
||||||
|
}
|
||||||
|
|
16
platforms/android/CordovaLib/project.properties
Normal file
16
platforms/android/CordovaLib/project.properties
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
# 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
|
@ -0,0 +1,69 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,144 @@
|
|||||||
|
/*
|
||||||
|
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));
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,72 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,145 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
508
platforms/android/CordovaLib/src/org/apache/cordova/CordovaActivity.java
Executable file
508
platforms/android/CordovaLib/src/org/apache/cordova/CordovaActivity.java
Executable file
@ -0,0 +1,508 @@
|
|||||||
|
/*
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,113 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
@ -0,0 +1,184 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,96 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,152 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,51 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
88
platforms/android/CordovaLib/src/org/apache/cordova/CordovaInterface.java
Executable file
88
platforms/android/CordovaLib/src/org/apache/cordova/CordovaInterface.java
Executable file
@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,243 @@
|
|||||||
|
/*
|
||||||
|
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 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 CordovaPlugin activityResultCallback;
|
||||||
|
protected CordovaPlugin permissionResultCallback;
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
@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 {
|
||||||
|
if(permissionResultCallback != null)
|
||||||
|
{
|
||||||
|
permissionResultCallback.onRequestPermissionResult(requestCode, permissions, grantResults);
|
||||||
|
permissionResultCallback = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void requestPermission(CordovaPlugin plugin, int requestCode, String permission) {
|
||||||
|
permissionResultCallback = plugin;
|
||||||
|
String[] permissions = new String [1];
|
||||||
|
permissions[0] = permission;
|
||||||
|
getActivity().requestPermissions(permissions, requestCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void requestPermissions(CordovaPlugin plugin, int requestCode, String [] permissions)
|
||||||
|
{
|
||||||
|
permissionResultCallback = plugin;
|
||||||
|
getActivity().requestPermissions(permissions, requestCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,422 @@
|
|||||||
|
/*
|
||||||
|
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 {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,101 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,471 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,142 @@
|
|||||||
|
/*
|
||||||
|
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.1.1";
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
@ -0,0 +1,81 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,613 @@
|
|||||||
|
/*
|
||||||
|
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) {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
360
platforms/android/CordovaLib/src/org/apache/cordova/CoreAndroid.java
Executable file
360
platforms/android/CordovaLib/src/org/apache/cordova/CoreAndroid.java
Executable file
@ -0,0 +1,360 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,31 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
@ -0,0 +1,66 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
@ -0,0 +1,33 @@
|
|||||||
|
/*
|
||||||
|
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();
|
||||||
|
};
|
@ -0,0 +1,38 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
234
platforms/android/CordovaLib/src/org/apache/cordova/LOG.java
Executable file
234
platforms/android/CordovaLib/src/org/apache/cordova/LOG.java
Executable file
@ -0,0 +1,234 @@
|
|||||||
|
/*
|
||||||
|
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));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
501
platforms/android/CordovaLib/src/org/apache/cordova/NativeToJsMessageQueue.java
Executable file
501
platforms/android/CordovaLib/src/org/apache/cordova/NativeToJsMessageQueue.java
Executable file
@ -0,0 +1,501 @@
|
|||||||
|
/*
|
||||||
|
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(");");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
70
platforms/android/CordovaLib/src/org/apache/cordova/PluginEntry.java
Executable file
70
platforms/android/CordovaLib/src/org/apache/cordova/PluginEntry.java
Executable file
@ -0,0 +1,70 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
527
platforms/android/CordovaLib/src/org/apache/cordova/PluginManager.java
Executable file
527
platforms/android/CordovaLib/src/org/apache/cordova/PluginManager.java
Executable file
@ -0,0 +1,527 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,198 @@
|
|||||||
|
/*
|
||||||
|
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
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
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));
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,170 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,66 @@
|
|||||||
|
/*
|
||||||
|
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();
|
||||||
|
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
@ -0,0 +1,53 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,293 @@
|
|||||||
|
/*
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,374 @@
|
|||||||
|
/*
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,334 @@
|
|||||||
|
/*
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
70
platforms/android/android.json
Normal file
70
platforms/android/android.json
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
{
|
||||||
|
"prepare_queue": {
|
||||||
|
"installed": [],
|
||||||
|
"uninstalled": []
|
||||||
|
},
|
||||||
|
"config_munge": {
|
||||||
|
"files": {
|
||||||
|
"res/xml/config.xml": {
|
||||||
|
"parents": {
|
||||||
|
"/*": [
|
||||||
|
{
|
||||||
|
"xml": "<feature name=\"Whitelist\"><param name=\"android-package\" value=\"org.apache.cordova.whitelist.WhitelistPlugin\" /><param name=\"onload\" value=\"true\" /></feature>",
|
||||||
|
"count": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"/widget": [
|
||||||
|
{
|
||||||
|
"xml": "<feature name=\"BLE\"><param name=\"android-package\" value=\"com.megster.cordova.ble.central.BLECentralPlugin\" /></feature>",
|
||||||
|
"count": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"AndroidManifest.xml": {
|
||||||
|
"parents": {
|
||||||
|
"/manifest": [
|
||||||
|
{
|
||||||
|
"xml": "<uses-permission android:name=\"android.permission.ACCESS_COARSE_LOCATION\" />",
|
||||||
|
"count": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"xml": "<uses-permission android:name=\"android.permission.BLUETOOTH\" />",
|
||||||
|
"count": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"xml": "<uses-permission android:name=\"android.permission.BLUETOOTH_ADMIN\" />",
|
||||||
|
"count": 1
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"installed_plugins": {
|
||||||
|
"cordova-plugin-whitelist": {
|
||||||
|
"PACKAGE_NAME": "org.censis.sensortoy"
|
||||||
|
},
|
||||||
|
"cordova-plugin-compat": {
|
||||||
|
"PACKAGE_NAME": "org.censis.sensortoy"
|
||||||
|
},
|
||||||
|
"cordova-plugin-ble-central": {
|
||||||
|
"PACKAGE_NAME": "org.censis.sensortoy"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"dependent_plugins": {},
|
||||||
|
"modules": [
|
||||||
|
{
|
||||||
|
"file": "plugins/cordova-plugin-ble-central/www/ble.js",
|
||||||
|
"id": "cordova-plugin-ble-central.ble",
|
||||||
|
"clobbers": [
|
||||||
|
"ble"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"plugin_metadata": {
|
||||||
|
"cordova-plugin-whitelist": "1.2.2",
|
||||||
|
"cordova-plugin-compat": "1.0.0",
|
||||||
|
"cordova-plugin-ble-central": "1.1.0"
|
||||||
|
}
|
||||||
|
}
|
36
platforms/android/assets/www/cordova-js-src/android/nativeapiprovider.js
vendored
Normal file
36
platforms/android/assets/www/cordova-js-src/android/nativeapiprovider.js
vendored
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* 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;
|
||||||
|
}
|
||||||
|
};
|
35
platforms/android/assets/www/cordova-js-src/android/promptbasednativeapi.js
vendored
Normal file
35
platforms/android/assets/www/cordova-js-src/android/promptbasednativeapi.js
vendored
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
/*
|
||||||
|
* 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);
|
||||||
|
}
|
||||||
|
};
|
283
platforms/android/assets/www/cordova-js-src/exec.js
vendored
Normal file
283
platforms/android/assets/www/cordova-js-src/exec.js
vendored
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* 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;
|
125
platforms/android/assets/www/cordova-js-src/platform.js
vendored
Normal file
125
platforms/android/assets/www/cordova-js-src/platform.js
vendored
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* 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);
|
||||||
|
}
|
||||||
|
}
|
108
platforms/android/assets/www/cordova-js-src/plugin/android/app.js
vendored
Normal file
108
platforms/android/assets/www/cordova-js-src/plugin/android/app.js
vendored
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* 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", []);
|
||||||
|
}
|
||||||
|
};
|
2167
platforms/android/assets/www/cordova.js
vendored
Normal file
2167
platforms/android/assets/www/cordova.js
vendored
Normal file
File diff suppressed because it is too large
Load Diff
19
platforms/android/assets/www/cordova_plugins.js
vendored
Normal file
19
platforms/android/assets/www/cordova_plugins.js
vendored
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
cordova.define('cordova/plugin_list', function(require, exports, module) {
|
||||||
|
module.exports = [
|
||||||
|
{
|
||||||
|
"file": "plugins/cordova-plugin-ble-central/www/ble.js",
|
||||||
|
"id": "cordova-plugin-ble-central.ble",
|
||||||
|
"clobbers": [
|
||||||
|
"ble"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
module.exports.metadata =
|
||||||
|
// TOP OF METADATA
|
||||||
|
{
|
||||||
|
"cordova-plugin-whitelist": "1.2.2",
|
||||||
|
"cordova-plugin-compat": "1.0.0",
|
||||||
|
"cordova-plugin-ble-central": "1.1.0"
|
||||||
|
};
|
||||||
|
// BOTTOM OF METADATA
|
||||||
|
});
|
280
platforms/android/assets/www/css/app.css
Normal file
280
platforms/android/assets/www/css/app.css
Normal file
@ -0,0 +1,280 @@
|
|||||||
|
body {
|
||||||
|
font-family: Ubuntu, "Helvetica Neue", Helvetica, arial, sans-serif;
|
||||||
|
background-color: #004c6d;
|
||||||
|
}
|
||||||
|
|
||||||
|
#weatherIcon {
|
||||||
|
margin-left:25%;
|
||||||
|
margin-right:25%;
|
||||||
|
height:70px;
|
||||||
|
width:70px;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#lightR, #projR { color: red !important; }
|
||||||
|
|
||||||
|
#lightG, #projG { color: green !important; }
|
||||||
|
|
||||||
|
#lightB, #projB { color: blue !important; }
|
||||||
|
|
||||||
|
#lightW, #projW { background-color: #aabbcc; }
|
||||||
|
|
||||||
|
.lightBG, .heatingBG, .projectorBG {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*.lightBG {
|
||||||
|
background-color: rgba(255, 255, 0, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.heatingBG {
|
||||||
|
background-color: rgba(255, 0, 255, 0.3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.projectorBG {
|
||||||
|
background-color: rgba(0, 255, 255, 0.3);
|
||||||
|
}*/
|
||||||
|
|
||||||
|
.mui-panel {
|
||||||
|
background-color: #015579;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h105 {
|
||||||
|
height: 100px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mdHeading {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mui--text-title {
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item_content {
|
||||||
|
height: 100px;
|
||||||
|
/* border: 1px solid grey;*/
|
||||||
|
min-height: 100px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item_content a.title {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item_content div.body, .item_content div.site, .item_content div.tags {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: #313131;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time, .date, .temp {
|
||||||
|
font-family: 'Ubuntu Condensed', sans-serif;
|
||||||
|
font-size: 80px;
|
||||||
|
color: #bad649;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time span.hour:after {
|
||||||
|
content: ":";
|
||||||
|
}
|
||||||
|
|
||||||
|
.date {
|
||||||
|
font-size: 35px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.temp::after {
|
||||||
|
content: "°c";
|
||||||
|
}
|
||||||
|
|
||||||
|
.item_content div.tags {
|
||||||
|
color: blue;
|
||||||
|
}
|
||||||
|
|
||||||
|
.noConnection {
|
||||||
|
color: rgb(244, 150, 26);
|
||||||
|
}
|
||||||
|
|
||||||
|
#caltext {
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Smartphones (portrait and landscape) ----------- */
|
||||||
|
@media only screen and (min-device-width: 320px) and (max-device-width: 480px) {
|
||||||
|
/* Styles */
|
||||||
|
.time, .date, .temp {
|
||||||
|
font-family: 'Ubuntu Condensed', sans-serif;
|
||||||
|
font-size: 33px;
|
||||||
|
/*color: #ff0000;*/
|
||||||
|
}
|
||||||
|
|
||||||
|
.time {
|
||||||
|
font-size: 50px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time span.hour:after {
|
||||||
|
content: "\a";
|
||||||
|
white-space: pre;
|
||||||
|
}
|
||||||
|
|
||||||
|
.temp {
|
||||||
|
font-size: 70px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.temp::after {
|
||||||
|
content: "°";
|
||||||
|
}
|
||||||
|
|
||||||
|
.wd-we {
|
||||||
|
font-size: 75%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo {
|
||||||
|
font-size: 85%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo.mo-1, .mo.mo-10 {
|
||||||
|
font-size: 70%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo.mo-2 {
|
||||||
|
font-size: 65%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo.mo-8 {
|
||||||
|
font-size: 80%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo.mo-9 {
|
||||||
|
font-size: 55%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.mo.mo-11, .mo.mo-12 {
|
||||||
|
font-size: 60%;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Smartphones (landscape) ----------- */
|
||||||
|
@media only screen and (min-width: 321px) {
|
||||||
|
/* Styles */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Smartphones (portrait) ----------- */
|
||||||
|
@media only screen and (max-width: 320px) {
|
||||||
|
/* Styles */
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner {
|
||||||
|
margin: 25px auto 0;
|
||||||
|
width: 70px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner > div {
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
background-color: rgb(244, 150, 26);
|
||||||
|
border-radius: 100%;
|
||||||
|
display: inline-block;
|
||||||
|
-webkit-animation: sk-bouncedelay 1.4s infinite ease-in-out both;
|
||||||
|
animation: sk-bouncedelay 1.4s infinite ease-in-out both;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner .bounce1 {
|
||||||
|
-webkit-animation-delay: -0.32s;
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.spinner .bounce2 {
|
||||||
|
-webkit-animation-delay: -0.16s;
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@-webkit-keyframes sk-bouncedelay {
|
||||||
|
0%, 80%, 100% { -webkit-transform: scale(0) }
|
||||||
|
40% { -webkit-transform: scale(1.0) }
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes sk-bouncedelay {
|
||||||
|
0%, 80%, 100% {
|
||||||
|
-webkit-transform: scale(0);
|
||||||
|
transform: scale(0);
|
||||||
|
}
|
||||||
|
40% {
|
||||||
|
-webkit-transform: scale(1.0);
|
||||||
|
transform: scale(1.0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.material-icons {
|
||||||
|
color: #e5f7fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
.material-icons.md-18 { font-size: 18px; }
|
||||||
|
|
||||||
|
.material-icons.md-24 { font-size: 24px; }
|
||||||
|
|
||||||
|
.material-icons.md-36 { font-size: 36px; }
|
||||||
|
|
||||||
|
.material-icons.md-48 { font-size: 48px; }
|
||||||
|
|
||||||
|
.material-icons.md-100 { font-size: 100px; }
|
||||||
|
|
||||||
|
/* Rules for using icons as black on a light background. */
|
||||||
|
.material-icons.md-dark { color: rgba(0, 0, 0, 0.54); }
|
||||||
|
|
||||||
|
.material-icons.md-dark.md-inactive { color: rgba(0, 0, 0, 0.26); }
|
||||||
|
|
||||||
|
/* Rules for using icons as white on a dark background. */
|
||||||
|
.material-icons.md-light { color: rgba(255, 255, 255, 1); }
|
||||||
|
|
||||||
|
.material-icons.md-light.md-inactive { color: rgba(255, 255, 255, 0.3); }
|
||||||
|
|
||||||
|
.material-icons.md-bulb {
|
||||||
|
content: ""
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
fan : toys
|
||||||
|
|
||||||
|
<i class="material-icons"></i>
|
||||||
|
|
||||||
|
|
||||||
|
bulb : lightbulb_outline
|
||||||
|
<i class="material-icons"></i>
|
||||||
|
|
||||||
|
calendar: event_note
|
||||||
|
<i class="material-icons"></i>
|
||||||
|
|
||||||
|
projector: cast
|
||||||
|
<i class="material-icons"></i>
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
.md-display {
|
||||||
|
opacity: 1;
|
||||||
|
transition: opacity 0.3s, visibility 0.3s;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.lostConnection {
|
||||||
|
opacity: 0.5;
|
||||||
|
transition: opacity 0.3s, visibility 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.mui-ellipsis-2 {
|
||||||
|
display: -webkit-box;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
word-wrap: break-word;
|
||||||
|
white-space: normal!important;
|
||||||
|
-webkit-line-clamp: 1;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
}
|
115
platforms/android/assets/www/css/index.css
Normal file
115
platforms/android/assets/www/css/index.css
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
* {
|
||||||
|
-webkit-tap-highlight-color: rgba(0,0,0,0); /* make transparent link selection, adjust last value opacity 0 to 1.0 */
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
-webkit-touch-callout: none; /* prevent callout to copy image, etc when tap to hold */
|
||||||
|
-webkit-text-size-adjust: none; /* prevent webkit from resizing text to fit */
|
||||||
|
-webkit-user-select: none; /* prevent copy paste, to allow, change 'none' to 'text' */
|
||||||
|
background-color:#E4E4E4;
|
||||||
|
background-image:linear-gradient(top, #A7A7A7 0%, #E4E4E4 51%);
|
||||||
|
background-image:-webkit-linear-gradient(top, #A7A7A7 0%, #E4E4E4 51%);
|
||||||
|
background-image:-ms-linear-gradient(top, #A7A7A7 0%, #E4E4E4 51%);
|
||||||
|
background-image:-webkit-gradient(
|
||||||
|
linear,
|
||||||
|
left top,
|
||||||
|
left bottom,
|
||||||
|
color-stop(0, #A7A7A7),
|
||||||
|
color-stop(0.51, #E4E4E4)
|
||||||
|
);
|
||||||
|
background-attachment:fixed;
|
||||||
|
font-family:'HelveticaNeue-Light', 'HelveticaNeue', Helvetica, Arial, sans-serif;
|
||||||
|
font-size:12px;
|
||||||
|
height:100%;
|
||||||
|
margin:0px;
|
||||||
|
padding:0px;
|
||||||
|
text-transform:uppercase;
|
||||||
|
width:100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Portrait layout (default) */
|
||||||
|
.app {
|
||||||
|
background:url(../img/logo.png) no-repeat center top; /* 170px x 200px */
|
||||||
|
position:absolute; /* position in the center of the screen */
|
||||||
|
left:50%;
|
||||||
|
top:50%;
|
||||||
|
height:50px; /* text area height */
|
||||||
|
width:225px; /* text area width */
|
||||||
|
text-align:center;
|
||||||
|
padding:180px 0px 0px 0px; /* image height is 200px (bottom 20px are overlapped with text) */
|
||||||
|
margin:-115px 0px 0px -112px; /* offset vertical: half of image height and text area height */
|
||||||
|
/* offset horizontal: half of text area width */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Landscape layout (with min-width) */
|
||||||
|
@media screen and (min-aspect-ratio: 1/1) and (min-width:400px) {
|
||||||
|
.app {
|
||||||
|
background-position:left center;
|
||||||
|
padding:75px 0px 75px 170px; /* padding-top + padding-bottom + text area = image height */
|
||||||
|
margin:-90px 0px 0px -198px; /* offset vertical: half of image height */
|
||||||
|
/* offset horizontal: half of image width and text area width */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
h1 {
|
||||||
|
font-size:24px;
|
||||||
|
font-weight:normal;
|
||||||
|
margin:0px;
|
||||||
|
overflow:visible;
|
||||||
|
padding:0px;
|
||||||
|
text-align:center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event {
|
||||||
|
border-radius:4px;
|
||||||
|
-webkit-border-radius:4px;
|
||||||
|
color:#FFFFFF;
|
||||||
|
font-size:12px;
|
||||||
|
margin:0px 30px;
|
||||||
|
padding:2px 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.listening {
|
||||||
|
background-color:#333333;
|
||||||
|
display:block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.event.received {
|
||||||
|
background-color:#4B946A;
|
||||||
|
display:none;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fade {
|
||||||
|
from { opacity: 1.0; }
|
||||||
|
50% { opacity: 0.4; }
|
||||||
|
to { opacity: 1.0; }
|
||||||
|
}
|
||||||
|
|
||||||
|
@-webkit-keyframes fade {
|
||||||
|
from { opacity: 1.0; }
|
||||||
|
50% { opacity: 0.4; }
|
||||||
|
to { opacity: 1.0; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.blink {
|
||||||
|
animation:fade 3000ms infinite;
|
||||||
|
-webkit-animation:fade 3000ms infinite;
|
||||||
|
}
|
15
platforms/android/assets/www/css/material-icons.css
Normal file
15
platforms/android/assets/www/css/material-icons.css
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
.material-icons {
|
||||||
|
font-family: 'Material Icons';
|
||||||
|
font-weight: normal;
|
||||||
|
font-style: normal;
|
||||||
|
font-size: 24px;
|
||||||
|
line-height: 1;
|
||||||
|
letter-spacing: normal;
|
||||||
|
text-transform: none;
|
||||||
|
display: inline-block;
|
||||||
|
white-space: nowrap;
|
||||||
|
word-wrap: normal;
|
||||||
|
direction: ltr;
|
||||||
|
text-rendering: optimizeLegibility;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
1912
platforms/android/assets/www/css/mui.css
Normal file
1912
platforms/android/assets/www/css/mui.css
Normal file
File diff suppressed because it is too large
Load Diff
1912
platforms/android/assets/www/css/mui.custom.css
Normal file
1912
platforms/android/assets/www/css/mui.custom.css
Normal file
File diff suppressed because it is too large
Load Diff
1
platforms/android/assets/www/css/mui.min.css
vendored
Normal file
1
platforms/android/assets/www/css/mui.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
BIN
platforms/android/assets/www/img/logo.png
Normal file
BIN
platforms/android/assets/www/img/logo.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 21 KiB |
91
platforms/android/assets/www/index.html
Normal file
91
platforms/android/assets/www/index.html
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<!--
|
||||||
|
Customize this policy to fit your own app's needs. For more guidance, see:
|
||||||
|
https://github.com/apache/cordova-plugin-whitelist/blob/master/README.md#content-security-policy
|
||||||
|
Some notes:
|
||||||
|
* gap: is required only on iOS (when using UIWebView) and is needed for JS->native communication
|
||||||
|
* https://ssl.gstatic.com is required only on Android and is needed for TalkBack to function properly
|
||||||
|
* Disables use of inline scripts in order to mitigate risk of XSS vulnerabilities. To change this:
|
||||||
|
* Enable inline JS: add 'unsafe-inline' to default-src
|
||||||
|
-->
|
||||||
|
<meta http-equiv="Content-Security-Policy" content="default-src 'self' data: gap: https://ssl.gstatic.com 'unsafe-eval'; style-src 'self' 'unsafe-inline'; media-src *">
|
||||||
|
<meta name="format-detection" content="telephone=no">
|
||||||
|
<meta name="msapplication-tap-highlight" content="no">
|
||||||
|
<meta name="viewport" content="user-scalable=no, initial-scale=1, maximum-scale=1, minimum-scale=1, width=device-width">
|
||||||
|
<link rel="stylesheet"
|
||||||
|
href="http://fonts.googleapis.com/css?family=Ubuntu:300,400,500,700"
|
||||||
|
type="text/css">
|
||||||
|
<link href='https://fonts.googleapis.com/css?family=Ubuntu+Condensed'
|
||||||
|
rel='stylesheet' type='text/css'>
|
||||||
|
<link href="https://fonts.googleapis.com/icon?family=Material+Icons"
|
||||||
|
rel="stylesheet">
|
||||||
|
<!-- endbuild -->
|
||||||
|
<!-- build:css -->
|
||||||
|
<link href="css/mui.css" rel="stylesheet" type="text/css"/>
|
||||||
|
<link href="css/app.css" rel="stylesheet" type="text/css"/>
|
||||||
|
<title>Sensor Toy</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id='app' class="mui-container">
|
||||||
|
|
||||||
|
<div class="mui-panel">
|
||||||
|
<button class="mui-btn mui-btn--primary" id="scan">Scan</button>
|
||||||
|
</div>
|
||||||
|
<table class="mui-table mui--text-white" id="results">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>ID</th>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>RSI</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody id="tbody">
|
||||||
|
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="frames" class="mui-container">
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript" src="cordova.js"></script>
|
||||||
|
<script type="text/javascript" src="libs/jquery/dist/jquery.js"></script>
|
||||||
|
<script type="text/javascript" src="libs/bluebird/js/browser/bluebird.js"></script>
|
||||||
|
|
||||||
|
<script type="text/javascript" src="js/standards/capability.js"></script>
|
||||||
|
<script type="text/javascript" src="js/standards/battery.js"></script>
|
||||||
|
<script type="text/javascript" src="js/standards/button.js"></script>
|
||||||
|
|
||||||
|
|
||||||
|
<script type="text/javascript" src="js/device/CC2650/cc2650_accelerometer.js"></script>
|
||||||
|
<script type="text/javascript" src="js/device/CC2650/cc2650_barometer.js"></script>
|
||||||
|
<script type="text/javascript" src="js/device/CC2650/cc2650_luxometer.js"></script>
|
||||||
|
|
||||||
|
|
||||||
|
<script type="text/javascript" src="js/index.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -0,0 +1,169 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:13
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* global CAPABILITY, inheritsFrom */
|
||||||
|
/* global ble */
|
||||||
|
/* jshint browser: true , devel: true*/
|
||||||
|
|
||||||
|
var CC2650_ACCEL = function(deviceId) {
|
||||||
|
'use strict';
|
||||||
|
this.name = 'Accelerometer';
|
||||||
|
this.deviceID = deviceId;
|
||||||
|
this.capabilityID = 'F000AA80-0451-4000-B000-000000000000';
|
||||||
|
this.serviceDef = {
|
||||||
|
service: 'F000AA80-0451-4000-B000-000000000000',
|
||||||
|
data: 'F000AA81-0451-4000-B000-000000000000', // Read/notify 3 bytes X : Y : Z
|
||||||
|
notification: 'F0002902-0451-4000-B000-000000000000',
|
||||||
|
configuration: 'F000AA82-0451-4000-B000-000000000000', // Read/write 1 byte
|
||||||
|
period: 'F000AA83-0451-4000-B000-000000000000' // Read/write 1 byte Period = [Input*10]ms
|
||||||
|
};
|
||||||
|
this.frames = {};
|
||||||
|
|
||||||
|
this.$id = {};
|
||||||
|
this.$result = {};
|
||||||
|
|
||||||
|
|
||||||
|
this.sensorMpu9250GyroConvert = function(data) {
|
||||||
|
return data / (65536 / 500);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.sensorMpu9250AccConvert = function(data) {
|
||||||
|
// Change /2 to match accel range...i.e. 16 g would be /16
|
||||||
|
return data / (32768 / 2);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onAccelerometerData = function(data) {
|
||||||
|
console.log(data);
|
||||||
|
var message;
|
||||||
|
var a = new Int16Array(data);
|
||||||
|
|
||||||
|
//0 gyro x
|
||||||
|
//1 gyro y
|
||||||
|
//2 gyro z
|
||||||
|
//3 accel x
|
||||||
|
//4 accel y
|
||||||
|
//5 accel z
|
||||||
|
//6 mag x
|
||||||
|
//7 mag y
|
||||||
|
//8 mag z
|
||||||
|
|
||||||
|
// TODO get a template to line this up
|
||||||
|
// TODO round or format numbers for better display
|
||||||
|
message = 'Gyro <br/>' +
|
||||||
|
'X: ' + this.sensorMpu9250GyroConvert(a[0]).toFixed(5) + '<br/>' +
|
||||||
|
'Y: ' + this.sensorMpu9250GyroConvert(a[1]) + '<br/>' +
|
||||||
|
'Z: ' + this.sensorMpu9250GyroConvert(a[2]) + '<br/>' +
|
||||||
|
'Accel <br/>' +
|
||||||
|
'X: ' + this.sensorMpu9250AccConvert(a[3]) + '<br/>' +
|
||||||
|
'Y: ' + this.sensorMpu9250AccConvert(a[4]) + '<br/>' +
|
||||||
|
'Z: ' + this.sensorMpu9250AccConvert(a[5]) + '<br/>' +
|
||||||
|
'Mag <br/>' +
|
||||||
|
'X: ' + a[3] + '<br/>' +
|
||||||
|
'Y: ' + a[4] + '<br/>' +
|
||||||
|
'Z: ' + a[5] + '<br/>' ;
|
||||||
|
|
||||||
|
this.state = message;
|
||||||
|
|
||||||
|
this.$result[this.frames.gyroID + '-x'].text(this.sensorMpu9250GyroConvert(a[0]).toFixed(5));
|
||||||
|
this.$result[this.frames.gyroID + '-y'].text(this.sensorMpu9250GyroConvert(a[1]).toFixed(5));
|
||||||
|
this.$result[this.frames.gyroID + '-z'].text(this.sensorMpu9250GyroConvert(a[2]).toFixed(5));
|
||||||
|
|
||||||
|
this.$result[this.frames.accelID + '-x'].text(this.sensorMpu9250AccConvert(a[3]).toFixed(5));
|
||||||
|
this.$result[this.frames.accelID + '-y'].text(this.sensorMpu9250AccConvert(a[4]).toFixed(5));
|
||||||
|
this.$result[this.frames.accelID + '-z'].text(this.sensorMpu9250AccConvert(a[5]).toFixed(5));
|
||||||
|
|
||||||
|
this.$result[this.frames.magID + '-x'].text(a[3]);
|
||||||
|
this.$result[this.frames.magID + '-y'].text(a[4]);
|
||||||
|
this.$result[this.frames.magID + '-z'].text(a[5]);
|
||||||
|
|
||||||
|
console.log(this.state);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.startService = function() {
|
||||||
|
'use strict';
|
||||||
|
if (this.deviceID !== null) {
|
||||||
|
|
||||||
|
console.log('Starting CC2650 Accelerometer Service on ', this.deviceID);
|
||||||
|
console.log(this.serviceDef);
|
||||||
|
//Ble.startNotification(this.deviceID, , this.onButtonData.bind(this), this.onError);
|
||||||
|
|
||||||
|
ble.startNotification(this.deviceID, this.serviceDef.service, this.serviceDef.data, this.onAccelerometerData.bind(this), this.onError);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// Turn accelerometer on
|
||||||
|
var configData = new Uint16Array(1);
|
||||||
|
// Turn on gyro, accel, and mag, 2G range, Disable wake on motion
|
||||||
|
configData[0] = 0x007F;
|
||||||
|
ble.write(this.deviceID, this.serviceDef.service, this.serviceDef.configuration, configData.buffer,
|
||||||
|
function() { console.log('Started accelerometer.'); }, this.onError);
|
||||||
|
|
||||||
|
var periodData = new Uint8Array(1);
|
||||||
|
periodData[0] = 0x0A;
|
||||||
|
// Ble.write(deviceId, accelerometer.service, accelerometer.period, periodData.buffer,
|
||||||
|
// function() { console.log("Configured accelerometer period."); },app.onError);
|
||||||
|
|
||||||
|
this.setInternalID();
|
||||||
|
this.insertFrame('gyro');
|
||||||
|
this.insertFrame('accel');
|
||||||
|
this.insertFrame('mag');
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
this.insertFrame = function(mode) {
|
||||||
|
|
||||||
|
var frame;
|
||||||
|
var title;
|
||||||
|
var modeID = mode + 'ID';
|
||||||
|
this.frames[modeID] = this.frameID + '-' + mode;
|
||||||
|
|
||||||
|
var titles = {gyro: 'Gyroscope', accel: 'Accelerometer', mag: 'Magnetometer'};
|
||||||
|
|
||||||
|
console.log('FrameID: ' , this.frames[modeID]);
|
||||||
|
|
||||||
|
title = [titles[mode], ' - ', this.deviceID].join(' ');
|
||||||
|
|
||||||
|
frame = $('<div />', {
|
||||||
|
class: 'mui-panel', id: this.frames[modeID]
|
||||||
|
});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-row'}).append($('<div />', { class: 'mui-col-xs-12 mui--text-title mui-ellipsis-2', text: title})).appendTo(frame);
|
||||||
|
|
||||||
|
$('#frames').append(frame);
|
||||||
|
this.$id[modeID] = $('#' + this.frames[modeID]);
|
||||||
|
|
||||||
|
// Call the parent displayForm first...
|
||||||
|
|
||||||
|
var row = $('<div />', {class: 'mui-row'});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'X'}).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Y'}).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Z'}).appendTo(row);
|
||||||
|
|
||||||
|
this.$id[modeID].append(row);
|
||||||
|
|
||||||
|
|
||||||
|
row = $('<div />', {class: 'mui-row'});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', text: '--', id: this.frames[modeID] + '-x' }).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', text: '--', id: this.frames[modeID] + '-y'}).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', text: '--', id: this.frames[modeID] + '-z'}).appendTo(row);
|
||||||
|
|
||||||
|
this.$id[modeID].append(row);
|
||||||
|
|
||||||
|
this.$result[this.frames[modeID] + '-x'] = $('#' + this.frames[modeID] + '-x');
|
||||||
|
this.$result[this.frames[modeID] + '-y'] = $('#' + this.frames[modeID] + '-y');
|
||||||
|
this.$result[this.frames[modeID] + '-z'] = $('#' + this.frames[modeID] + '-z');
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
inheritsFrom(CC2650_ACCEL, CAPABILITY);
|
@ -0,0 +1,106 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:13
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* global CAPABILITY, inheritsFrom */
|
||||||
|
/* global ble */
|
||||||
|
/* jshint browser: true , devel: true*/
|
||||||
|
|
||||||
|
var CC2650_BAR = function(deviceId) {
|
||||||
|
'use strict';
|
||||||
|
this.name = 'Barometer';
|
||||||
|
this.deviceID = deviceId;
|
||||||
|
this.capabilityID = 'F000AA40-0451-4000-B000-000000000000';
|
||||||
|
this.serviceDef = {
|
||||||
|
service: 'F000AA40-0451-4000-B000-000000000000',
|
||||||
|
data: 'F000AA41-0451-4000-B000-000000000000',
|
||||||
|
notification: 'F0002902-0451-4000-B000-000000000000',
|
||||||
|
configuration: 'F000AA42-0451-4000-B000-000000000000',
|
||||||
|
period: 'F000AA43-0451-4000-B000-000000000000'
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.$result = {temp: null, pressure: null};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
this.startService = function() {
|
||||||
|
'use strict';
|
||||||
|
if (this.deviceID !== null) {
|
||||||
|
|
||||||
|
console.log('Starting CC2650 Barometer Service on ', this.deviceID);
|
||||||
|
console.log(this.serviceDef);
|
||||||
|
this.insertFrame();
|
||||||
|
|
||||||
|
ble.startNotification(this.deviceID, this.serviceDef.service, this.serviceDef.data, this.onBarometerData.bind(this), this.onError);
|
||||||
|
|
||||||
|
//Turn on barometer
|
||||||
|
var barometerConfig = new Uint8Array(1);
|
||||||
|
barometerConfig[0] = 0x01;
|
||||||
|
ble.write(this.deviceID, this.serviceDef.service, this.serviceDef.configuration, barometerConfig.buffer,
|
||||||
|
function() { console.log('Started barometer.'); },this.onError);
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.sensorBarometerConvert = function(data) {
|
||||||
|
return (data / 100);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onBarometerData = function(data) {
|
||||||
|
var pStr;
|
||||||
|
var tStr;
|
||||||
|
console.log(data);
|
||||||
|
var message;
|
||||||
|
var a = new Uint8Array(data);
|
||||||
|
|
||||||
|
//0-2 Temp
|
||||||
|
//3-5 Pressure
|
||||||
|
tStr = this.sensorBarometerConvert(a[0] | (a[1] << 8) | (a[2] << 16)) + '°C';
|
||||||
|
pStr = this.sensorBarometerConvert(a[3] | (a[4] << 8) | (a[5] << 16)) + 'hPa';
|
||||||
|
|
||||||
|
message = 'Temperature <br/>' + tStr +
|
||||||
|
'Pressure <br/>' + pStr ;
|
||||||
|
|
||||||
|
this.$result.temp.text(tStr);
|
||||||
|
this.$result.pressure.text(pStr);
|
||||||
|
|
||||||
|
this.state = message;
|
||||||
|
|
||||||
|
console.log('Barometer:', this.state);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.insertFrame = function() {
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
console.log('Overloading...');
|
||||||
|
// Call the parent displayForm first...
|
||||||
|
this.superClass_.insertFrame.call(self);
|
||||||
|
|
||||||
|
var temp = this.frameID + '-t';
|
||||||
|
var pressure = this.frameID + '-p';
|
||||||
|
var row = $('<div />', {class: 'mui-row'});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Temp:'}).appendTo(row);
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', id: temp}).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Pressure:'}).appendTo(row);
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', id: pressure}).appendTo(row);
|
||||||
|
|
||||||
|
this.$id.append(row);
|
||||||
|
this.$result.temp = $('#' + temp);
|
||||||
|
this.$result.pressure = $('#' + pressure);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
inheritsFrom(CC2650_BAR, CAPABILITY);
|
@ -0,0 +1,103 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:13
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* global CAPABILITY, inheritsFrom */
|
||||||
|
/* global ble */
|
||||||
|
/* jshint browser: true , devel: true*/
|
||||||
|
|
||||||
|
var CC2650_LUX = function(deviceId) {
|
||||||
|
'use strict';
|
||||||
|
this.name = 'Luxometer';
|
||||||
|
this.deviceID = deviceId;
|
||||||
|
this.capabilityID = 'F000AA70-0451-4000-B000-000000000000';
|
||||||
|
this.serviceDef = {
|
||||||
|
service: 'F000AA70-0451-4000-B000-000000000000',
|
||||||
|
data: 'F000AA71-0451-4000-B000-000000000000',
|
||||||
|
notification: 'F0002902-0451-4000-B000-000000000000',
|
||||||
|
configuration: 'F000AA72-0451-4000-B000-000000000000',
|
||||||
|
period: 'F000AA73-0451-4000-B000-000000000000'
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.$result = {temp: null, pressure: null};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
this.startService = function() {
|
||||||
|
'use strict';
|
||||||
|
if (this.deviceID !== null) {
|
||||||
|
|
||||||
|
console.log('Starting CC2650 Luxometer Service on ', this.deviceID);
|
||||||
|
console.log(this.serviceDef);
|
||||||
|
//this.insertFrame();
|
||||||
|
|
||||||
|
ble.startNotification(this.deviceID, this.serviceDef.service, this.serviceDef.data, this.onLuxData.bind(this), this.onError);
|
||||||
|
|
||||||
|
//Turn on barometer
|
||||||
|
var luxConfig = new Uint8Array(1);
|
||||||
|
luxConfig[0] = 0x01;
|
||||||
|
ble.write(this.deviceID, this.serviceDef.service, this.serviceDef.configuration, luxConfig.buffer,
|
||||||
|
function() { console.log('Started luxometer.'); },this.onError);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.sensorBarometerConvert = function(data) {
|
||||||
|
return (data / 100);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onLuxData = function(data) {
|
||||||
|
var pStr;
|
||||||
|
var tStr;
|
||||||
|
console.log(data);
|
||||||
|
var message;
|
||||||
|
var a = new Uint8Array(data);
|
||||||
|
var b = new Uint16Array(data);
|
||||||
|
|
||||||
|
console.log('lux a',a[0],a[1]);
|
||||||
|
console.log('lux b',b[0]);
|
||||||
|
console.log('lux b',b[0].toString(2));
|
||||||
|
|
||||||
|
var m = b[0] | b[0] << 4;
|
||||||
|
console.log(m.toString(2));
|
||||||
|
|
||||||
|
//0-2 Temp
|
||||||
|
//3-5 Pressure
|
||||||
|
|
||||||
|
console.log('Luxometer:', this.state);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.insertFrame = function() {
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
console.log('Overloading...');
|
||||||
|
// Call the parent displayForm first...
|
||||||
|
this.superClass_.insertFrame.call(self);
|
||||||
|
|
||||||
|
var temp = this.frameID + '-t';
|
||||||
|
var pressure = this.frameID + '-p';
|
||||||
|
var row = $('<div />', {class: 'mui-row'});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Temp:'}).appendTo(row);
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', id: temp}).appendTo(row);
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-accent', text: 'Pressure:'}).appendTo(row);
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-3 mui--text-white', id: pressure}).appendTo(row);
|
||||||
|
|
||||||
|
this.$id.append(row);
|
||||||
|
this.$result.temp = $('#' + temp);
|
||||||
|
this.$result.pressure = $('#' + pressure);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
inheritsFrom(CC2650_LUX, CAPABILITY);
|
224
platforms/android/assets/www/js/index.js
Normal file
224
platforms/android/assets/www/js/index.js
Normal file
@ -0,0 +1,224 @@
|
|||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* global BATTERY, BUTTON */
|
||||||
|
|
||||||
|
var app = {
|
||||||
|
activeServices: [],
|
||||||
|
serviceList: {
|
||||||
|
1800: 'Generic Access',
|
||||||
|
1801: 'Generic Attribute',
|
||||||
|
'180a': 'Device Information',
|
||||||
|
|
||||||
|
FFE0: 'Button',
|
||||||
|
'F000AA00-0451-4000-B000-000000000000': 'Temperature',
|
||||||
|
'F000AA20-0451-4000-B000-000000000000': 'Humidity',
|
||||||
|
'F000AA80-0451-4000-B000-000000000000': 'Accelerometer',
|
||||||
|
'F000AA40-0451-4000-B000-000000000000': 'Barometer'
|
||||||
|
|
||||||
|
},
|
||||||
|
list: {},
|
||||||
|
button: {
|
||||||
|
service: 'FFE0',
|
||||||
|
data: 'FFE1', // Bit 2: side key, Bit 1- right key, Bit 0 –left key
|
||||||
|
},
|
||||||
|
accelerometer: {
|
||||||
|
service: 'F000AA80-0451-4000-B000-000000000000',
|
||||||
|
data: 'F000AA81-0451-4000-B000-000000000000', // Read/notify 3 bytes X : Y : Z
|
||||||
|
notification: 'F0002902-0451-4000-B000-000000000000',
|
||||||
|
configuration: 'F000AA82-0451-4000-B000-000000000000', // Read/write 1 byte
|
||||||
|
period: 'F000AA83-0451-4000-B000-000000000000' // Read/write 1 byte Period = [Input*10]ms
|
||||||
|
},
|
||||||
|
barometer: {
|
||||||
|
service: 'F000AA40-0451-4000-B000-000000000000',
|
||||||
|
data: 'F000AA41-0451-4000-B000-000000000000',
|
||||||
|
notification: 'F0002902-0451-4000-B000-000000000000',
|
||||||
|
configuration: 'F000AA42-0451-4000-B000-000000000000',
|
||||||
|
period: 'F000AA43-0451-4000-B000-000000000000'
|
||||||
|
|
||||||
|
},
|
||||||
|
// Application Constructor
|
||||||
|
initialize: function() {
|
||||||
|
this.bindEvents();
|
||||||
|
},
|
||||||
|
doScan: function() {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
$('#tbody').empty();
|
||||||
|
ble.startScan([], function(device) {
|
||||||
|
console.log(JSON.stringify(device));
|
||||||
|
|
||||||
|
var newId = device.id.replace(/:/g, '');
|
||||||
|
console.log(newId);
|
||||||
|
|
||||||
|
this.list[newId] = device.id;
|
||||||
|
|
||||||
|
var newTR = $('<tr id="' + newId + '" class="clickRow">');
|
||||||
|
|
||||||
|
newTR.append($('<td>').text(device.id));
|
||||||
|
|
||||||
|
if (device.hasOwnProperty('name')) {
|
||||||
|
newTR.append($('<td>').text(device.name));
|
||||||
|
} else {
|
||||||
|
newTR.append($('<td>').text('*** Unknown'));
|
||||||
|
}
|
||||||
|
|
||||||
|
newTR.append($('<td>').text(device.rssi));
|
||||||
|
|
||||||
|
$('#tbody').append(newTR);
|
||||||
|
|
||||||
|
$('#output').append(JSON.stringify(device) + '<br/>');
|
||||||
|
|
||||||
|
}.bind(this), function(e) {
|
||||||
|
'use strict';
|
||||||
|
console.error(e);
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
setTimeout(ble.stopScan,
|
||||||
|
5000,
|
||||||
|
function() { console.log('Scan complete'); },
|
||||||
|
function() { console.log('stopScan failed'); }
|
||||||
|
);
|
||||||
|
|
||||||
|
},
|
||||||
|
// Bind Event Listeners
|
||||||
|
//
|
||||||
|
// Bind any events that are required on startup. Common events are:
|
||||||
|
// 'load', 'deviceready', 'offline', and 'online'.
|
||||||
|
bindEvents: function() {
|
||||||
|
var self = this;
|
||||||
|
document.addEventListener('deviceready', this.onDeviceReady, false);
|
||||||
|
$('#scan').on('click', function() {
|
||||||
|
'use strict';
|
||||||
|
this.doScan();
|
||||||
|
}.bind(this));
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
$('#tbody').on('click','tr', function() {
|
||||||
|
'use strict';
|
||||||
|
var tID = $(this).context.id;
|
||||||
|
|
||||||
|
var id = self.list[tID];
|
||||||
|
|
||||||
|
console.log(tID, id);
|
||||||
|
|
||||||
|
self.connect(id);
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
},
|
||||||
|
// Deviceready Event Handler
|
||||||
|
//
|
||||||
|
// The scope of 'this' is the event. In order to call the 'receivedEvent'
|
||||||
|
// function, we must explicitly call 'app.receivedEvent(...);'
|
||||||
|
onDeviceReady: function() {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}, serviceDiscovery: function(services) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
console.log(services);
|
||||||
|
|
||||||
|
},
|
||||||
|
sensorMpu9250GyroConvert: function(data) {
|
||||||
|
return data / (65536 / 500);
|
||||||
|
},
|
||||||
|
|
||||||
|
sensorMpu9250AccConvert: function(data) {
|
||||||
|
// Change /2 to match accel range...i.e. 16 g would be /16
|
||||||
|
return data / (32768 / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
,connect: function(deviceId) {
|
||||||
|
|
||||||
|
$('#results').slideUp();
|
||||||
|
console.log('Connect to ', deviceId);
|
||||||
|
|
||||||
|
var onConnect = function(a) {
|
||||||
|
var services = [];
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
console.log('A:', a);
|
||||||
|
|
||||||
|
services = a.services;
|
||||||
|
|
||||||
|
for (var t = 0; t < services.length;t++) {
|
||||||
|
|
||||||
|
var ident = services[t].toUpperCase();
|
||||||
|
|
||||||
|
switch (ident) {
|
||||||
|
case '180F':
|
||||||
|
var batteryStat = new BATTERY(deviceId);
|
||||||
|
batteryStat.startService();
|
||||||
|
batteryStat.readBatteryState();
|
||||||
|
app.activeServices.push(batteryStat);
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'FFE0':
|
||||||
|
var buttonState = new BUTTON(deviceId);
|
||||||
|
buttonState.startService();
|
||||||
|
app.activeServices.push(buttonState);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'F000AA80-0451-4000-B000-000000000000':
|
||||||
|
var cc2650_accel = new CC2650_ACCEL(deviceId);
|
||||||
|
cc2650_accel.startService();
|
||||||
|
app.activeServices.push(cc2650_accel);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'F000AA40-0451-4000-B000-000000000000':
|
||||||
|
var cc2650_bar = new CC2650_BAR(deviceId);
|
||||||
|
cc2650_bar.startService();
|
||||||
|
app.activeServices.push(cc2650_bar);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'F000AA70-0451-4000-B000-000000000000':
|
||||||
|
var cc2650_lux = new CC2650_LUX(deviceId);
|
||||||
|
cc2650_lux.startService();
|
||||||
|
app.activeServices.push(cc2650_lux);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
console.log('Unknown service: ', ident);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
ble.connect(deviceId, onConnect, function(e) {
|
||||||
|
'use strict';
|
||||||
|
console.error(e);
|
||||||
|
});
|
||||||
|
|
||||||
|
},
|
||||||
|
onError: function(reason) {
|
||||||
|
console.error('ERROR: ' + reason); // Real apps should use notification.alert
|
||||||
|
},
|
||||||
|
updateGyro: function(g) {
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
app.initialize();
|
52
platforms/android/assets/www/js/standards/battery.js
Normal file
52
platforms/android/assets/www/js/standards/battery.js
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:13
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* global CAPABILITY */
|
||||||
|
/* global ble */
|
||||||
|
/* jshint browser: true , devel: true*/
|
||||||
|
|
||||||
|
var BATTERY = function() {
|
||||||
|
this.name = 'Battery';
|
||||||
|
this.capabilityID = '180F';
|
||||||
|
this.serviceDef = {
|
||||||
|
service: '180F',
|
||||||
|
level: '2A19'
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onBatteryLevelChange = function(data) {
|
||||||
|
console.log(data);
|
||||||
|
var a = new Uint8Array(data);
|
||||||
|
this.state = a[0];
|
||||||
|
console.log('onBatteryLevelChange', this.state);
|
||||||
|
};
|
||||||
|
this.readBatteryState = function() {
|
||||||
|
console.log('readBatteryState');
|
||||||
|
ble.read(this.deviceID, this.serviceDef.service, this.serviceDef.level, this.onReadBatteryLevel.bind(this), this.onError);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onReadBatteryLevel = function(data) {
|
||||||
|
console.log(data);
|
||||||
|
var a = new Uint8Array(data);
|
||||||
|
this.state = a[0];
|
||||||
|
console.log('onReadBatteryLevel', this.state);
|
||||||
|
};
|
||||||
|
|
||||||
|
this.startService = function() {
|
||||||
|
'use strict';
|
||||||
|
if (this.deviceID !== null) {
|
||||||
|
console.log('Starting Battery Service');
|
||||||
|
ble.startNotification(this.deviceID, this.serviceDef.service, this.serviceDef.level, this.onBatteryLevelChange.bind(this), this.onError);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.insertFrame();
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
inheritsFrom(BATTERY, CAPABILITY);
|
96
platforms/android/assets/www/js/standards/button.js
Normal file
96
platforms/android/assets/www/js/standards/button.js
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:13
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
/* global CAPABILITY, inheritsFrom */
|
||||||
|
/* global ble */
|
||||||
|
/* jshint browser: true , devel: true*/
|
||||||
|
|
||||||
|
var BUTTON = function(deviceId) {
|
||||||
|
'use strict';
|
||||||
|
this.name = 'Button';
|
||||||
|
this.deviceID = deviceId;
|
||||||
|
this.capabilityID = 'FFE0';
|
||||||
|
this.serviceDef = {
|
||||||
|
service: 'FFE0',
|
||||||
|
data: 'FFE1' // Bit 2: side key, Bit 1- right key, Bit 0 –left key
|
||||||
|
};
|
||||||
|
|
||||||
|
this.buttonMatrix = {
|
||||||
|
LEFT_BUTTON: 1, // 0001
|
||||||
|
RIGHT_BUTTON: 2, // 0010
|
||||||
|
REED_SWITCH: 4 // 0100
|
||||||
|
};
|
||||||
|
|
||||||
|
this.onButtonData = function(data) {
|
||||||
|
console.log('+ onButtonData');
|
||||||
|
console.log(data);
|
||||||
|
var state = new Uint8Array(data);
|
||||||
|
var message = '';
|
||||||
|
|
||||||
|
if (state === 0) {
|
||||||
|
message = 'No buttons are pressed.';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state & this.buttonMatrix.LEFT_BUTTON) {
|
||||||
|
message += 'Left button is pressed. ';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state & this.buttonMatrix.RIGHT_BUTTON) {
|
||||||
|
message += 'Right button is pressed.';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state & this.buttonMatrix.REED_SWITCH) {
|
||||||
|
message += 'Reed switch is activated.<br/>';
|
||||||
|
}
|
||||||
|
|
||||||
|
this.state = message;
|
||||||
|
|
||||||
|
if (this.$result !== null)
|
||||||
|
{
|
||||||
|
this.$result.text(this.state);
|
||||||
|
}
|
||||||
|
console.log('ButtonState: ', this.state);
|
||||||
|
|
||||||
|
console.log('- onButtonData');
|
||||||
|
};
|
||||||
|
|
||||||
|
this.startService = function() {
|
||||||
|
'use strict';
|
||||||
|
if (this.deviceID !== null) {
|
||||||
|
|
||||||
|
console.log('Starting Button Service on ', this.deviceID);
|
||||||
|
console.log(this.serviceDef);
|
||||||
|
ble.startNotification(this.deviceID, this.serviceDef.service, this.serviceDef.data, this.onButtonData.bind(this), this.onError);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
this.insertFrame();
|
||||||
|
};
|
||||||
|
|
||||||
|
this.insertFrame = function() {
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
console.log('Overloading...');
|
||||||
|
// Call the parent displayForm first...
|
||||||
|
this.superClass_.insertFrame.call(self);
|
||||||
|
|
||||||
|
var detail = this.frameID + '-d';
|
||||||
|
var row = $('<div />', {class: 'mui-row'});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-4 mui--text-accent', text: 'Status:'}).appendTo(row);
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-col-xs-8 mui--text-white', id: detail}).appendTo(row);
|
||||||
|
|
||||||
|
this.$id.append(row);
|
||||||
|
this.$result = $('#'+detail);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
inheritsFrom(BUTTON, CAPABILITY);
|
78
platforms/android/assets/www/js/standards/capability.js
Normal file
78
platforms/android/assets/www/js/standards/capability.js
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* User: Martin Donnelly
|
||||||
|
* Date: 2016-05-20
|
||||||
|
* Time: 10:32
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
'use strict';
|
||||||
|
var CAPABILITY = function(p) {
|
||||||
|
this.name = '-CAPABILITY-';
|
||||||
|
this.capabilityID = p.capabilityID || null;
|
||||||
|
this.state = '';
|
||||||
|
this.serviceDef = p.service || null;
|
||||||
|
|
||||||
|
this.internalID = null;
|
||||||
|
this.frameID = null;
|
||||||
|
this.$id = null;
|
||||||
|
|
||||||
|
this.deviceID = null;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
CAPABILITY.prototype.setInternalID = function() {
|
||||||
|
this.internalID = (Math.floor(Math.random() * Number.MAX_SAFE_INTEGER) + 1).toString(36);
|
||||||
|
this.frameID = 'f-' + this.internalID;
|
||||||
|
};
|
||||||
|
|
||||||
|
CAPABILITY.prototype.setDeviceID = function(dID) {
|
||||||
|
this.deviceID = dID;
|
||||||
|
};
|
||||||
|
|
||||||
|
CAPABILITY.prototype.getDeviceID = function() {
|
||||||
|
return this.deviceID ;
|
||||||
|
};
|
||||||
|
|
||||||
|
CAPABILITY.prototype.insertFrame = function() {
|
||||||
|
|
||||||
|
this.setInternalID();
|
||||||
|
|
||||||
|
console.log('FrameID: ' , this.frameID);
|
||||||
|
|
||||||
|
var title = [this.name, ' - ', this.deviceID].join(' ');
|
||||||
|
|
||||||
|
var frame = $('<div />', {
|
||||||
|
class: 'mui-panel',
|
||||||
|
id: this.frameID});
|
||||||
|
|
||||||
|
$('<div />', { class: 'mui-row'}).append($('<div />', { class: 'mui-col-xs-12 mui--text-title', text: title})).appendTo(frame);
|
||||||
|
|
||||||
|
$('#frames').append(frame);
|
||||||
|
this.$id = $('#' + this.frameID);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
CAPABILITY.prototype.inherits = function(a, b) {
|
||||||
|
var c = function() {};
|
||||||
|
c.prototype = b.prototype;
|
||||||
|
a.superClass_ = b.prototype;
|
||||||
|
a.prototype = new c;
|
||||||
|
a.prototype.constructor = a;
|
||||||
|
};
|
||||||
|
|
||||||
|
CAPABILITY.prototype.onError = function(e) {
|
||||||
|
console.error(e);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
var inheritsFrom = function(a, b) {
|
||||||
|
var c = function() {};
|
||||||
|
c.prototype = b.prototype;
|
||||||
|
a.superClass_ = b.prototype;
|
||||||
|
a.prototype = new c;
|
||||||
|
a.prototype.constructor = a;
|
||||||
|
a.prototype.superClass_ = b.prototype;
|
||||||
|
};
|
55
platforms/android/assets/www/js/test.js
Normal file
55
platforms/android/assets/www/js/test.js
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
/**
|
||||||
|
* Created by martin on 2016-05-20.
|
||||||
|
*/
|
||||||
|
var DEVICEBASE = function(p) {
|
||||||
|
|
||||||
|
this.controller = p.controller || null;
|
||||||
|
this.settings = p.settings || {};
|
||||||
|
this.data = {};
|
||||||
|
this.standards = {
|
||||||
|
|
||||||
|
button: {
|
||||||
|
service: 'FFE0',
|
||||||
|
data: 'FFE1' // Bit 2: side key, Bit 1- right key, Bit 0 –left key
|
||||||
|
},
|
||||||
|
heartRate: {
|
||||||
|
service: '180d',
|
||||||
|
measurement: '2a37'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
this.custom = {};
|
||||||
|
this.capabilities = [];
|
||||||
|
this.deviceID = null;
|
||||||
|
|
||||||
|
this.inherits = function(a, b) {
|
||||||
|
var c = function() {};
|
||||||
|
c.prototype = b.prototype;
|
||||||
|
a.superClass_ = b.prototype;
|
||||||
|
a.prototype = new c;
|
||||||
|
a.prototype.constructor = a;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
this.register = function(c) {
|
||||||
|
c.registerPage(this);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
this.bytesToString = function(buffer) {
|
||||||
|
return String.fromCharCode.apply(null, new Uint8Array(buffer));
|
||||||
|
};
|
||||||
|
|
||||||
|
// ASCII only
|
||||||
|
this.stringToBytes = function(string) {
|
||||||
|
var array = new Uint8Array(string.length);
|
||||||
|
for (var i = 0, l = string.length; i < l; i++) {
|
||||||
|
array[i] = string.charCodeAt(i);
|
||||||
|
}
|
||||||
|
return array.buffer;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
this.register(this.controller);
|
||||||
|
};
|
44
platforms/android/assets/www/libs/bluebird/.bower.json
Normal file
44
platforms/android/assets/www/libs/bluebird/.bower.json
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
{
|
||||||
|
"name": "bluebird",
|
||||||
|
"version": "3.4.0",
|
||||||
|
"homepage": "https://github.com/petkaantonov/bluebird",
|
||||||
|
"authors": [
|
||||||
|
"Petka Antonov <petka_antonov@hotmail.com>"
|
||||||
|
],
|
||||||
|
"description": "Bluebird is a full featured promise library with unmatched performance.",
|
||||||
|
"main": "js/browser/bluebird.js",
|
||||||
|
"license": "MIT",
|
||||||
|
"ignore": [
|
||||||
|
"**/.*",
|
||||||
|
"benchmark",
|
||||||
|
"bower_components",
|
||||||
|
"./browser",
|
||||||
|
"node_modules",
|
||||||
|
"test"
|
||||||
|
],
|
||||||
|
"keywords": [
|
||||||
|
"promise",
|
||||||
|
"performance",
|
||||||
|
"promises",
|
||||||
|
"promises-a",
|
||||||
|
"promises-aplus",
|
||||||
|
"async",
|
||||||
|
"await",
|
||||||
|
"deferred",
|
||||||
|
"deferreds",
|
||||||
|
"future",
|
||||||
|
"flow control",
|
||||||
|
"dsl",
|
||||||
|
"fluent interface"
|
||||||
|
],
|
||||||
|
"_release": "3.4.0",
|
||||||
|
"_resolution": {
|
||||||
|
"type": "version",
|
||||||
|
"tag": "v3.4.0",
|
||||||
|
"commit": "3aeb023d26df1e738e37a162a8f81b36afe1c6b9"
|
||||||
|
},
|
||||||
|
"_source": "https://github.com/petkaantonov/bluebird.git",
|
||||||
|
"_target": "^3.4.0",
|
||||||
|
"_originalSource": "bluebird",
|
||||||
|
"_direct": true
|
||||||
|
}
|
1
platforms/android/assets/www/libs/bluebird/API.md
Normal file
1
platforms/android/assets/www/libs/bluebird/API.md
Normal file
@ -0,0 +1 @@
|
|||||||
|
[http://bluebirdjs.com/docs/api-reference.html](http://bluebirdjs.com/docs/api-reference.html)
|
@ -0,0 +1,9 @@
|
|||||||
|
|
||||||
|
# Questions and issues
|
||||||
|
|
||||||
|
Please see [The Support Page](http://bluebirdjs.com/docs/support.html)
|
||||||
|
The [github issue tracker](https://github.com/petkaantonov/bluebird/issues) is **_only_** for bug reports and feature requests.
|
||||||
|
|
||||||
|
# Contributing to the library
|
||||||
|
|
||||||
|
Contributions are welcome and appreciated. See the [Contribution Page](http://bluebirdjs.com/docs/contribute.html) on bluebirdjs.com
|
21
platforms/android/assets/www/libs/bluebird/LICENSE
Normal file
21
platforms/android/assets/www/libs/bluebird/LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2013-2015 Petka Antonov
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
51
platforms/android/assets/www/libs/bluebird/README.md
Normal file
51
platforms/android/assets/www/libs/bluebird/README.md
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
<a href="http://promisesaplus.com/">
|
||||||
|
<img src="http://promisesaplus.com/assets/logo-small.png" alt="Promises/A+ logo"
|
||||||
|
title="Promises/A+ 1.1 compliant" align="right" />
|
||||||
|
</a>
|
||||||
|
[![Build Status](https://travis-ci.org/petkaantonov/bluebird.svg?branch=master)](https://travis-ci.org/petkaantonov/bluebird)
|
||||||
|
[![coverage-98%](http://img.shields.io/badge/coverage-98%-brightgreen.svg?style=flat)](http://petkaantonov.github.io/bluebird/coverage/debug/index.html)
|
||||||
|
|
||||||
|
**Got a question?** Join us on [stackoverflow](http://stackoverflow.com/questions/tagged/bluebird), the [mailing list](https://groups.google.com/forum/#!forum/bluebird-js) or chat on [IRC](https://webchat.freenode.net/?channels=#promises)
|
||||||
|
|
||||||
|
# Introduction
|
||||||
|
|
||||||
|
Bluebird is a fully featured promise library with focus on innovative features and performance
|
||||||
|
|
||||||
|
See the [**bluebird website**](http://bluebirdjs.com/docs/getting-started.html) for further documentation, references and instructions. See the [**API reference**](http://bluebirdjs.com/docs/api-reference.html) here.
|
||||||
|
|
||||||
|
For bluebird 2.x documentation and files, see the [2.x tree](https://github.com/petkaantonov/bluebird/tree/2.x).
|
||||||
|
|
||||||
|
# Questions and issues
|
||||||
|
|
||||||
|
The [github issue tracker](https://github.com/petkaantonov/bluebird/issues) is **_only_** for bug reports and feature requests. Anything else, such as questions for help in using the library, should be posted in [StackOverflow](http://stackoverflow.com/questions/tagged/bluebird) under tags `promise` and `bluebird`.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Thanks
|
||||||
|
|
||||||
|
Thanks to BrowserStack for providing us with a free account which lets us support old browsers like IE8.
|
||||||
|
|
||||||
|
# License
|
||||||
|
|
||||||
|
The MIT License (MIT)
|
||||||
|
|
||||||
|
Copyright (c) 2013-2016 Petka Antonov
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in
|
||||||
|
all copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
THE SOFTWARE.
|
||||||
|
|
37
platforms/android/assets/www/libs/bluebird/bench
Executable file
37
platforms/android/assets/www/libs/bluebird/bench
Executable file
@ -0,0 +1,37 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
./build --release --no-debug
|
||||||
|
benchmark=$1
|
||||||
|
nodepath=${2:-node}
|
||||||
|
shift 2;
|
||||||
|
cwd=${PWD}
|
||||||
|
|
||||||
|
trap 'cd "$cwd"' EXIT
|
||||||
|
|
||||||
|
if [ "$benchmark" = "doxbee" ]; then
|
||||||
|
cd "$cwd/benchmark"
|
||||||
|
npm install
|
||||||
|
echo "Doxbee sequential"
|
||||||
|
$nodepath performance.js --n 10000 --t 1 ./doxbee-sequential/*.js --harmony "$@"
|
||||||
|
exit 0
|
||||||
|
elif [ "$benchmark" = "doxbee-errors" ]; then
|
||||||
|
cd "$cwd/benchmark"
|
||||||
|
npm install
|
||||||
|
echo "Doxbee sequential with 10% errors"
|
||||||
|
$nodepath performance.js --n 10000 --t 1 --e 0.1 ./doxbee-sequential-errors/*.js --harmony "$@"
|
||||||
|
exit 0
|
||||||
|
elif [ "$benchmark" = "parallel" ]; then
|
||||||
|
cd "$cwd/benchmark"
|
||||||
|
npm install
|
||||||
|
echo "Madeup parallel"
|
||||||
|
$nodepath performance.js --n 10000 --t 1 --p 25 ./madeup-parallel/*.js --harmony "$@"
|
||||||
|
exit 0
|
||||||
|
elif [ "$benchmark" = "analysis" ]; then
|
||||||
|
cd "$cwd/benchmark"
|
||||||
|
npm install
|
||||||
|
echo "analysis"
|
||||||
|
$nodepath performance.js --n 10000 --t 1 ./analysis/*.js --harmony "$@"
|
||||||
|
exit 0
|
||||||
|
else
|
||||||
|
echo "Invalid benchmark name $benchmark"
|
||||||
|
exit -1
|
||||||
|
fi
|
34
platforms/android/assets/www/libs/bluebird/bower.json
Normal file
34
platforms/android/assets/www/libs/bluebird/bower.json
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
{
|
||||||
|
"name": "bluebird",
|
||||||
|
"version": "3.4.0",
|
||||||
|
"homepage": "https://github.com/petkaantonov/bluebird",
|
||||||
|
"authors": [
|
||||||
|
"Petka Antonov <petka_antonov@hotmail.com>"
|
||||||
|
],
|
||||||
|
"description": "Bluebird is a full featured promise library with unmatched performance.",
|
||||||
|
"main": "js/browser/bluebird.js",
|
||||||
|
"license": "MIT",
|
||||||
|
"ignore": [
|
||||||
|
"**/.*",
|
||||||
|
"benchmark",
|
||||||
|
"bower_components",
|
||||||
|
"./browser",
|
||||||
|
"node_modules",
|
||||||
|
"test"
|
||||||
|
],
|
||||||
|
"keywords": [
|
||||||
|
"promise",
|
||||||
|
"performance",
|
||||||
|
"promises",
|
||||||
|
"promises-a",
|
||||||
|
"promises-aplus",
|
||||||
|
"async",
|
||||||
|
"await",
|
||||||
|
"deferred",
|
||||||
|
"deferreds",
|
||||||
|
"future",
|
||||||
|
"flow control",
|
||||||
|
"dsl",
|
||||||
|
"fluent interface"
|
||||||
|
]
|
||||||
|
}
|
2
platforms/android/assets/www/libs/bluebird/build
Executable file
2
platforms/android/assets/www/libs/bluebird/build
Executable file
@ -0,0 +1,2 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
node tools/build.js "$@"
|
1
platforms/android/assets/www/libs/bluebird/changelog.md
Normal file
1
platforms/android/assets/www/libs/bluebird/changelog.md
Normal file
@ -0,0 +1 @@
|
|||||||
|
[http://bluebirdjs.com/docs/changelog.html](http://bluebirdjs.com/docs/changelog.html)
|
@ -0,0 +1 @@
|
|||||||
|
[http://bluebirdjs.com/docs/deprecated-apis.html](http://bluebirdjs.com/docs/deprecated-apis.html)
|
8
platforms/android/assets/www/libs/bluebird/docs/Gemfile
Normal file
8
platforms/android/assets/www/libs/bluebird/docs/Gemfile
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
source "https://rubygems.org"
|
||||||
|
|
||||||
|
gem "jekyll"
|
||||||
|
gem "jekyll-redirect-from"
|
||||||
|
gem "sanitize", '4.0.1'
|
||||||
|
gem "redcarpet"
|
||||||
|
gem "pygments.rb"
|
||||||
|
gem 'wdm', '>= 0.1.0' if Gem.win_platform?
|
@ -0,0 +1,5 @@
|
|||||||
|
Requires ruby and [jekyll](http://jekyllrb.com/). See the gem file for dependencies.
|
||||||
|
|
||||||
|
Change directory to `bluebird/docs` and run `jekyll serve`. The docs will be hosted under `/docs` directory in relation to the web root. Typically something like `http://localhost:4000/docs/`
|
||||||
|
|
||||||
|
|
35
platforms/android/assets/www/libs/bluebird/docs/_config.yml
Normal file
35
platforms/android/assets/www/libs/bluebird/docs/_config.yml
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
name: bluebird
|
||||||
|
description: Bluebird is a fully featured JavaScript promises library with unmatched performance.
|
||||||
|
url: "http://bluebirdjs.com"
|
||||||
|
baseurl: ""
|
||||||
|
title: bluebird
|
||||||
|
timezone: Helsinki/Finland
|
||||||
|
highlighter: pygments
|
||||||
|
exclude:
|
||||||
|
- Gemfile
|
||||||
|
- Gemfile.lock
|
||||||
|
- helpers.rb
|
||||||
|
defaults:
|
||||||
|
- scope:
|
||||||
|
path: docs
|
||||||
|
type: pages
|
||||||
|
values:
|
||||||
|
layout: page
|
||||||
|
markdown: redcarpet
|
||||||
|
redcarpet:
|
||||||
|
extensions:
|
||||||
|
- fenced_code_blocks
|
||||||
|
version: 3.4.0
|
||||||
|
gems:
|
||||||
|
- jekyll-redirect-from
|
||||||
|
destination: ../gh-pages/
|
||||||
|
keep_files:
|
||||||
|
- .git
|
||||||
|
- .gitignore
|
||||||
|
- logo.png
|
||||||
|
- CNAME
|
||||||
|
- coverage
|
||||||
|
safe: false
|
||||||
|
encoding: utf-8
|
||||||
|
host: 0.0.0.0
|
||||||
|
|
@ -0,0 +1,9 @@
|
|||||||
|
---
|
||||||
|
layout: default
|
||||||
|
---
|
||||||
|
|
||||||
|
<div class="post">
|
||||||
|
<article class="post-content">
|
||||||
|
{{ content }}
|
||||||
|
</article>
|
||||||
|
</div>
|
@ -0,0 +1,136 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<!--[if lt IE 7]> <html class="no-js lt-ie9 lt-ie8 lt-ie7"> <![endif]-->
|
||||||
|
<!--[if IE 7]> <html class="no-js lt-ie9 lt-ie8"> <![endif]-->
|
||||||
|
<!--[if IE 8]> <html class="no-js lt-ie9"> <![endif]-->
|
||||||
|
<!--[if gt IE 8]><!-->
|
||||||
|
<html class="no-js">
|
||||||
|
<!--<![endif]-->
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
<link href='http://fonts.googleapis.com/css?family=Raleway:400,300,200,700,500,100,800,600,900' rel='stylesheet' type='text/css'>
|
||||||
|
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.2/css/bootstrap.min.css" />
|
||||||
|
<title>{% if page.title %}{{ page.title }} | {{ site.title }}{% else %}{{ site.title }}{% endif %}</title>
|
||||||
|
<meta name="description" content="{% if page.excerpt %}{{ page.excerpt | strip_html | strip_newlines | truncate: 160 }}{% else %}{{ site.description }}{% endif %}">
|
||||||
|
<link href="//netdna.bootstrapcdn.com/font-awesome/4.0.3/css/font-awesome.css" rel="stylesheet">
|
||||||
|
<link rel="stylesheet" href="{{ "/css/mono-blue.css" | prepend: site.baseurl }}" type='text/css' />
|
||||||
|
<link rel="stylesheet" href="{{ "/css/hover-min.css" | prepend: site.baseurl }}" media="all">
|
||||||
|
<link rel="stylesheet" href="{{ "/css/main.css" | prepend: site.baseurl }}">
|
||||||
|
<link rel="canonical" href="{{ page.url | replace:'index.html','' | prepend: site.baseurl | prepend: site.url }}">
|
||||||
|
<link rel="icon" href="{{ "/img/favicon.png" | prepend: site.baseurl }}" type="image/png" />
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
|
||||||
|
<nav class="navbar" role="navigation">
|
||||||
|
<div class="container">
|
||||||
|
<div class="navbar-header">
|
||||||
|
<button type="button" class="navbar-toggle collapsed" data-toggle="collapse" data-target="#navbar" aria-expanded="false" aria-controls="navbar">
|
||||||
|
<span class="sr-only">Toggle navigation</span>
|
||||||
|
<span class="icon-bar"></span>
|
||||||
|
<span class="icon-bar"></span>
|
||||||
|
<span class="icon-bar"></span>
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<a href="/" class="title">
|
||||||
|
<img src="{{ "/img/logo.png" | prepend: site.baseurl }}" class="hidden-xs" />
|
||||||
|
|
||||||
|
<span class="tagline">bluebird</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div id="navbar" class="navbar-collapse navbar-bluebird navbar-right collapse">
|
||||||
|
<ul class="nav navbar-nav bb-nav">
|
||||||
|
{% if page.path == 'docs/support.md' %}
|
||||||
|
{% assign name = 'support' %}
|
||||||
|
{% elsif page.path == 'docs/install.md' %}
|
||||||
|
{% assign name = 'install' %}
|
||||||
|
{% else %}
|
||||||
|
{% assign name = 'docs' %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<li class="{% if name == 'docs' %}active{% endif %}"><a href="{{ "/docs/getting-started.html" | prepend: site.baseurl }}">Docs</a></li>
|
||||||
|
<li class="{% if name == 'support' %}active{% endif %}"><a href="{{ "/docs/support.html" | prepend: site.baseurl }}">Support</a></li>
|
||||||
|
<li class="{% if name == 'install' %}active{% endif %}"><a href="{{ "/docs/install.html" | prepend: site.baseurl }}">Install</a></li>
|
||||||
|
<li><a href="https://github.com/petkaantonov/bluebird/">Github</a></li>
|
||||||
|
</ul>
|
||||||
|
</div><!--/.navbar-collapse -->
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-sm-3">
|
||||||
|
<ul class="nav left-nav">
|
||||||
|
<li><a href="{{ "/docs/getting-started.html" | prepend: site.baseurl }}">Getting Started</a></li>
|
||||||
|
<li><a href="{{ "/docs/features.html" | prepend: site.baseurl }}">Features</a></li>
|
||||||
|
<li><a href="{{ "/docs/changelog.html" | prepend: site.baseurl }}">Changelog</a></li>
|
||||||
|
<li><a href="{{ "/docs/api-reference.html" | prepend: site.baseurl }}"><strong>API Reference</strong></a></li>
|
||||||
|
<li><a href="{{ "/docs/new-in-bluebird-3.html" | prepend: site.baseurl }}"><strong>New in 3.0</strong></a></li>
|
||||||
|
<li><a href="{{ "/docs/warning-explanations.html" | prepend: site.baseurl }}">Warning Explanations</a></li>
|
||||||
|
<li><a href="{{ "/docs/error-explanations.html" | prepend: site.baseurl }}">Error Explanations</a></li>
|
||||||
|
<li><a href="{{ "/docs/contribute.html" | prepend: site.baseurl }}">Contribute</a></li>
|
||||||
|
<li><a href="{{ "/docs/benchmarks.html" | prepend: site.baseurl }}">Benchmarks</a></li>
|
||||||
|
<li><a href="{{ "/docs/deprecated-apis.html" | prepend: site.baseurl }}">Deprecated APIs</a></li>
|
||||||
|
<li><a href="{{ "/docs/download-api-reference.html" | prepend: site.baseurl }}">Download API Reference</a></li>
|
||||||
|
<li><hr></li>
|
||||||
|
<li>Why?
|
||||||
|
<ul class="nav nav-child">
|
||||||
|
<li><a href="{{ "/docs/why-promises.html" | prepend: site.baseurl }}">Why Promises?</a></li>
|
||||||
|
<li><a href="{{ "/docs/why-bluebird.html" | prepend: site.baseurl }}">Why bluebird?</a></li>
|
||||||
|
<li><a href="{{ "/docs/why-performance.html" | prepend: site.baseurl }}">Why Performance?</a></li>
|
||||||
|
<li><a href="{{ "/docs/what-about-generators.html" | prepend: site.baseurl }}">What About Generators?</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
<li><hr></li>
|
||||||
|
<li>
|
||||||
|
Tutorials
|
||||||
|
<ul class="nav nav-child">
|
||||||
|
<li><a href="{{ "/docs/async-dialogs.html" | prepend: site.baseurl }}">Async Dialogs</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
<li><hr></li>
|
||||||
|
<li>
|
||||||
|
Guides
|
||||||
|
<ul class="nav nav-child">
|
||||||
|
<li><a href="{{ "/docs/beginners-guide.html" | prepend: site.baseurl }}">Beginner's Guide</a></li>
|
||||||
|
<li><a href="{{ "/docs/anti-patterns.html" | prepend: site.baseurl }}">Anti-patterns</a></li>
|
||||||
|
<li><a href="{{ "/docs/working-with-callbacks.html" | prepend: site.baseurl }}">Working with Callbacks</a></li>
|
||||||
|
<li><a href="{{ "/docs/coming-from-other-languages.html" | prepend: site.baseurl }}">Coming from Other Languages</a></li>
|
||||||
|
<li><a href="{{ "/docs/coming-from-other-libraries.html" | prepend: site.baseurl }}">Coming from Other Libraries</a></li>
|
||||||
|
</ul>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-9">
|
||||||
|
{% if page.path %}
|
||||||
|
<div class="post-info">
|
||||||
|
<a href="{{ page.path | prepend: "https://github.com/petkaantonov/bluebird/edit/master/docs/" }}">
|
||||||
|
<i class="fa fa-edit"></i>
|
||||||
|
Edit on Github</a>
|
||||||
|
<br>
|
||||||
|
<i>Updated {{ page.path | file_date | date_to_string }}</i>
|
||||||
|
</div>
|
||||||
|
<div class="clearfix"></div>
|
||||||
|
{% endif %}
|
||||||
|
{{ content }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<footer></footer>
|
||||||
|
<script src="https://code.jquery.com/jquery-2.1.3.min.js"></script>
|
||||||
|
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.2/js/bootstrap.min.js"></script>
|
||||||
|
<script src="//cdn.jsdelivr.net/bluebird/{{ site.version }}/bluebird.js"></script>
|
||||||
|
<script>
|
||||||
|
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
|
||||||
|
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
|
||||||
|
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
|
||||||
|
})(window,document,'script','//www.google-analytics.com/analytics.js','ga');
|
||||||
|
|
||||||
|
ga('create', 'UA-46253177-1', 'auto');
|
||||||
|
ga('send', 'pageview');
|
||||||
|
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
@ -0,0 +1,14 @@
|
|||||||
|
---
|
||||||
|
layout: default
|
||||||
|
---
|
||||||
|
<div class="post">
|
||||||
|
|
||||||
|
<header class="post-header">
|
||||||
|
<h1 class="post-title">{{ page.title }}</h1>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<article class="post-content">
|
||||||
|
{{ content }}
|
||||||
|
</article>
|
||||||
|
|
||||||
|
</div>
|
@ -0,0 +1,19 @@
|
|||||||
|
module Jekyll
|
||||||
|
module MyFilters
|
||||||
|
def file_date(input)
|
||||||
|
File.mtime(input)
|
||||||
|
end
|
||||||
|
|
||||||
|
def check_active(page_path, link_type)
|
||||||
|
if (link_type == "support" and page_path =~ /support/) or
|
||||||
|
(link_type == "install" and page_path =~ /install/) or
|
||||||
|
(link_type == "docs")
|
||||||
|
"active"
|
||||||
|
else
|
||||||
|
""
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
Liquid::Template.register_filter(Jekyll::MyFilters)
|
@ -0,0 +1,55 @@
|
|||||||
|
require "redcarpet"
|
||||||
|
require "pygments"
|
||||||
|
require_relative "../helpers"
|
||||||
|
|
||||||
|
|
||||||
|
class Redcarpet::Render::HTML
|
||||||
|
def header(title, level)
|
||||||
|
anchor = Helpers.clean(title)
|
||||||
|
|
||||||
|
return <<-eos
|
||||||
|
<h#{level}>
|
||||||
|
<a class="header-anchor"
|
||||||
|
name="#{anchor}"
|
||||||
|
aria-hidden="true"
|
||||||
|
href="##{anchor}"><i class="fa fa-link"></i></a>
|
||||||
|
#{title}
|
||||||
|
</h#{level}>
|
||||||
|
eos
|
||||||
|
end
|
||||||
|
# Hacks to get markdown working inside html blocks ....
|
||||||
|
def block_html(html)
|
||||||
|
html.gsub(/<markdown>([\d\D]+?)<\/markdown>/) {|_|
|
||||||
|
extensions = {fenced_code_blocks: true}
|
||||||
|
renderer = Redcarpet::Markdown.new(WithHighlights, extensions)
|
||||||
|
renderer.render(Regexp.last_match[1])
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
def link(link, title, content)
|
||||||
|
if link == "unfinished-article"
|
||||||
|
return <<-eos
|
||||||
|
<div class="info-box">
|
||||||
|
This article is partially or completely unfinished.
|
||||||
|
You are welcome to create <a href="https://github.com/petkaantonov/bluebird/edit/master/docs/docs/#{content}.md">pull requests</a>
|
||||||
|
to help completing this article.
|
||||||
|
</div>
|
||||||
|
eos
|
||||||
|
elsif link == "."
|
||||||
|
if content =~ /#\d+/
|
||||||
|
url = "https://github.com/petkaantonov/bluebird/issues/" + content[1..-1]
|
||||||
|
else
|
||||||
|
url = "/docs/api/" + Helpers.clean(content) + ".html"
|
||||||
|
end
|
||||||
|
return "<a href='#{url}'><code>#{content}</code></a>"
|
||||||
|
else
|
||||||
|
return "<a href='#{link}' title='#{title}'>#{content}</a>"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
class WithHighlights < Redcarpet::Render::HTML
|
||||||
|
def block_code(code, language)
|
||||||
|
Pygments.highlight(code, :lexer => language)
|
||||||
|
end
|
||||||
|
end
|
181
platforms/android/assets/www/libs/bluebird/docs/css/main.css
Normal file
181
platforms/android/assets/www/libs/bluebird/docs/css/main.css
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
body {
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
body > .container {
|
||||||
|
padding-bottom: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-bluebird {
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
font-size: 18px;
|
||||||
|
margin-top: 50px;
|
||||||
|
text-transform:uppercase;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav {
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav.bb-nav > li > a{
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
border-bottom: 1px solid rgba(0,0,0,0);
|
||||||
|
}
|
||||||
|
.nav.bb-nav > li > a:focus, .nav.bb-nav > li > a:hover,
|
||||||
|
.nav.bb-nav > li.active > a {
|
||||||
|
background-color:initial;
|
||||||
|
border-bottom:1px solid #BE7306;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav a {
|
||||||
|
color: #BE7306;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav.left-nav li a {
|
||||||
|
padding: 2px 0px;
|
||||||
|
font-size: 14px;
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav.left-nav li a:focus, .nav.left-nav li a:hover {
|
||||||
|
background: none;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-bluebird li {
|
||||||
|
text-align:center;
|
||||||
|
}
|
||||||
|
h1,h2,h3,h4,h5,h6{
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
/*color:#5275B4;*/
|
||||||
|
color: #1B4288;
|
||||||
|
}
|
||||||
|
.api-code-section h2, code, pre {
|
||||||
|
font-family: 'Consolas', 'Lucida Console', 'Courier New', 'monospace'
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
margin-top:10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-bar {
|
||||||
|
background-color:#5275B4;
|
||||||
|
}
|
||||||
|
.promises-showroom .arrow {
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 100px;
|
||||||
|
color:#EC9313;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre{
|
||||||
|
background-color: initial;
|
||||||
|
border: none;
|
||||||
|
margin:0px;
|
||||||
|
padding:0px;
|
||||||
|
}
|
||||||
|
.promises-showroom .hljs{
|
||||||
|
border-radius: 6px;
|
||||||
|
padding-left: 30px;
|
||||||
|
padding-bottom: 15px;
|
||||||
|
}
|
||||||
|
.take-action{
|
||||||
|
position:relative;
|
||||||
|
min-height:300px;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.take-action .cloud-bg{
|
||||||
|
width: 100%;
|
||||||
|
height: 300px;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.take-action .action-buttons{
|
||||||
|
position:absolute;
|
||||||
|
top:50%;
|
||||||
|
text-align:center;
|
||||||
|
width:100%;
|
||||||
|
}
|
||||||
|
.btn-bb {
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
.btn-beak{
|
||||||
|
background-color: #EC9313;
|
||||||
|
border-bottom: 4px solid #BE7306;
|
||||||
|
}
|
||||||
|
.btn-wings {
|
||||||
|
background-color: #265584;
|
||||||
|
border-bottom: 4px solid #1A3A59;
|
||||||
|
}
|
||||||
|
.btn-bb:hover{
|
||||||
|
color:white;
|
||||||
|
}
|
||||||
|
.undo-margin{
|
||||||
|
margin-left:-15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.main-line{
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
margin-bottom: 59px;
|
||||||
|
margin-top: -74px;
|
||||||
|
font-size: 20px;
|
||||||
|
color:white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagline {
|
||||||
|
font-family: 'Raleway', sans-serif;
|
||||||
|
font-size:26px;
|
||||||
|
color:#5275B4;
|
||||||
|
font-weight:200;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-child {
|
||||||
|
margin-left:20px;
|
||||||
|
}
|
||||||
|
.title:hover{
|
||||||
|
text-decoration:none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-anchor {
|
||||||
|
opacity: 0;
|
||||||
|
|
||||||
|
-webkit-transition: opacity 0.2s ease-in-out 0.1s;
|
||||||
|
-moz-transition: opacity 0.2s ease-in-out 0.1s;
|
||||||
|
-ms-transition: opacity 0.2s ease-in-out 0.1s;
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-anchor:active, .header-anchor:focus {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1:hover .header-anchor,
|
||||||
|
h2:hover .header-anchor,
|
||||||
|
h3:hover .header-anchor,
|
||||||
|
h4:hover .header-anchor,
|
||||||
|
h5:hover .header-anchor,
|
||||||
|
h6:hover .header-anchor {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.api-reference-menu {
|
||||||
|
-moz-column-count: 2;
|
||||||
|
-webkit-column-count: 2;
|
||||||
|
columns: 2;
|
||||||
|
margin-bottom: 50px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.post-info {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-box {
|
||||||
|
margin-top: 20px;
|
||||||
|
color: #8a6d3b;
|
||||||
|
background-color: #fcf8e3;
|
||||||
|
border-color: #faebcc;
|
||||||
|
padding: 15px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
border: 1px dashed #8a6d3b;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user