A unified JavaScript layer for Apache Cordova projects.
cordova-js
|
|-build/
| Will contain any build modules (currently nothing here as it is all hacked
| into the JakeFile)
|
|-lib
| |-bootstrap.js
| | Code to bootstrap the Cordova platform, inject APIs and fire events
| |
| |-builder.js
| | Injects in our classes onto window and navigator (or wherever else is needed)
| |
| |-channel.js
| | A pub/sub implementation to handle custom framework events
| |
| |-cordova.js
| | Common Cordova stuff such as callback handling and
| | window/document add/removeEventListener hijacking
| |
| |-utils.js
| | General purpose JS utility stuff: closures, uuids, object
| | cloning, extending prototypes
| |
| |-exec/
| | Contains the platform specific definitions of the exec method
| |
| |-platform/
| | Definitions of each platform that help us describe where
| | and what to put on the window object, and what to run to
| | initialize the platform. A common set of globals are also
| | defined (common.js)
| |
| |-plugin/
| | | All API definitions as plugins, ones common to all
| | | platforms reside at the top level...
| | `-<platform>
| | ... and platform-specific ones reside in their respective
| | folders
Just make sure you have node.js installed. It
should come pre-installed with npm - but if you
install node and can't run npm
then head over to the website and
install it yourself. Make sure you have all of the node dependencies
installed by running the following command from the repository root:
npm install
Then all of the build tasks can be run via:
jake
This will run the "build" and "test" tasks by default. All of the available tasks are:
- build: creates platform versions of cordova-js and builds them into
the
pkg
directory. - test: runs all of the unit tests inside node.
- btest: creates a server so you can run the tests inside a browser.
- clean: cleans out the
pkg
directory
The build/packager.js
tool is a node.js script that concatenates and
wraps various .js files with a RequireJS-compatible module syntax in this
project together to generate cordova.js files that are compatible
to the various supported platforms. Check that script out to figure out
how and in what order the various files are concatenated together.
We end up with a script file that has a ton of define
calls, wrapping
each Cordova API or object into its own module. Next, the Cordova bridge is initialized with the
help of lib/bootstrap.js
. This file attaches the _self.boot
function
once the channel.onNativeReady
event is fired - which should be fired
from the native side (native should call require('cordova/channel).onNativeReady.fire()
).
Finally, the boot
method is where the magic happens. First, it grabs
the common platform definition (as defined under
lib/platform/common.js
) and injects all the objects defined in there
onto window
and other global namespaces. Next, it grabs all of the platform-specific object
definitions (as defined under lib/platform/<platform>.js
) and
overrides those onto window
. Finally, it calls the platform-specific
initialize
function (located in the platform definition). At this
point, Cordova is fully initialized and ready to roll, however, before
the deviceready
event is fired, we still wait for the
DOMContentLoaded
event to fire to make sure the page has loaded
properly.
Tests run in node or the browser, and you can launch them with :
jake test
or to run in the browser:
jake btest
Final testing should always be done with the Mobile Spec test application.
Build the .js file and drop it in as a replacement for cordova.js or cordova.js!
Load this in Ripple to play with it. You will have to use the cordova prototype branch to better simulate the phone environment and use this javascript rather than Ripples emulated code.
git clone [email protected]:blackberry-webworks/Ripple-UI.git
git checkout winnie.the.pooh
./configure
jake
and then load the upacked extension in chrome in the pkg/chromium folder. Use the cordova.proto platform in ripple.
- Write a module that encapsulates your platform's
exec
method and call it .js. Theexec
method is a JavaScript function that enables communication from the platform's JavaScript environment into the platform's native environment. Each platform uses a different mechanism to enable this bridge. We recommend you check out the other platformexec
definitions for inspiration. Drop this into thelib/exec
folder. Theexec
method has the following method signature:function(success, fail, service, action, args)
, with the following parameters:
success
: a success function callbackfail
: a failure function callbackservice
: a string identifier that the platform can resolve to a native classaction
: a string identifier that the platform can resolve to a specific method inside the class pointed to byservice
args
: an array of parameters to pass to the native method invoked by theexec
call It is required that new platform additions be as consistent as possible with the existingservice
andaction
labels.
-
Define your platform definition object and name it .js. Drop this into the
lib/platform
folder. This file should contain a JSON object with the following properties:id
: a string representing the platform. This should be the same name the .js file hasobjects
: the property names defined as children of this property are injected intowindow
. Each property can have the following child properties:path
: a string representing the module ID that will define this object. For example, the filelib/plugin/accelerometer.js
can be accessed as"cordova/plugin/accelerometer"
. More details on how the module IDs are defined are above under the "How It Works" section.children
: in a recursive fashion, can havepath
andchildren
properties of its own that are defined as children of the parent property object
initialize
: a function that fires immediately after the theobjects
(see above) are defined in the global scope
The following is a simple example of a platform definition:
{ id:"atari", initialize:function(){ console.log('firing up cordova in my atari, yo.'); }, objects:{ cordova:{ path:"cordova", children:{ joystick:{ path:"cordova/plugin/atari/joystick" } } } } }
-
You should probably add a
packager.bundle('<platform>')
call to theJakefile
. -
Make sure your native implementation executes the following JavaScript once all of the native side is initialized and ready:
require('cordova/channel').onNativeReady.fire()
.
- Related to above, come up with a consistent sensor plugin API.
Functions like
getCurrent<data>
andwatch<data>
can be abstracted into a nice plugin. Compass, Accel, Geo should all be basically the same implementation. For example, on Android geo + accel handle callingstart
(starting the listener) in the native code on its own. However, Compass requires that JS initiates astart
. This is dumb. - Media (and other plugin) implementations across platforms need to use the established cordova/exec callback method (instead of triggering globally-accessible functions to dispatch listeners). On iOS and Android, grep for "cast" in the native code - you'll see a bunch of invoked JavaScript from native, which shouldn't be there.
- Media needs updates across all platforms. Methods need fixing with respect to timing: some methods use milliseconds, some use seconds. Some methods not documented (setVolume on Android). Consolidate / implement properly across platforms.
- Storage shim on Android needs to change its win/fail callbacks to
require('cordova/plugin/android/storage').failQuery / completeQuery
(away from droiddb.fail / completeQuery) - Normalize
Entry.toURL
return values. iOS returns"file://localhost" + fullPath
, Android returns"file://" + fullPath
, BlackBerry returns justfullPath
- APIs that are not cross-platform - what
to do with these?
- Crypto on Android
- SMS, telephony, splashscreen on iOS
- Need to normalize native return values as much as possible across platforms. For example, error objects. Should we return JSON objects from native or minimal primitives (i.e. error codes as numbers)? Both are in use today, we need to decide on a standard.
- Once-over all of the cordova-docs with the APIs defined in here to make sure all is consistent. There were function signature tweaks, undocumented procedures, etc.
- implementations:
- BlackBerry: button + app + contact + file + others (need to once-over)
- all of Playbook
- everything for WP7
- everything for Bada
- tests for channel, pretty much everything under lib/plugin
- think about whether to select and load the platform specific modules at runtime or at buildtime. what about platform-specific overrides? can we at buildtime decide to include only the overrides (to save a few kb?). what about specifically denoting modules to include/exclude on a per-platform basis?
- 3rd party plugins could be interesting. Need a little bit more thought about how these will fit into the system. I am thinking a package.json type file to handle per plugin.