The Extend Function
Let's consider the following code snippet:
var myViewGroup = android.view.ViewGroup.extend({
onMeasure: function (widthMeasureSpec, heightMeasureSpec) {
// call the base class implementation
this.super.onMeasure(widthMeasureSpec, heightMeasureSpec);
// do something additionally
// myMeasureMethod(widthMeasureSpec, heightMeasureSpec);
},
onLayout: function (changed, left, top, right, bottom): void {
// call the base class implementation
this.super.onLayout(changed, left, top, right, bottom);
// do something additionally
// myOnLayoutMethod(left, top, right, bottom);
}
})
At a low level, the Runtime adds the extend
function on each Android Class Proxy. The function is associated with a V8 method callback, which, when triggered, will first validate the arguments passed and then the Binding (Proxy) generator will attempt to create a new dynamic Android Type. This type will allow the JavaScript defined method overrides to be properly called.
Extend Function Syntax
The full extend
syntax is as follows:
extend(<className>, <implementationObject>);
The className Parameter
This parameter is optional and specifies the name for the new class. In case not specified (which is the most common scenario), the Runtime will attempt to generate an implicit unique class name, based on the following criteria:
- The base class name
- The name of the file where the type is declared
- The line and the column at which the
extend
function is called
Based on these criteria, a typical implicit class name will look like
android_view_ViewGroup_myFile_l10_c20
. There are several corner cases, as described in the Gotchas section, where generating implicit class names may lead to errors.
The implementationObject Parameter
This is a JavaScript Object Literal that provides the methods to be overridden in the newly generated class. If we consider the example at the top, the implementation object is:
var implObject = {
onMeasure: function (widthMeasureSpec, heightMeasureSpec) {
// call the base class implementation
this.super.onMeasure(widthMeasureSpec, heightMeasureSpec);
// do something additionally
// myMeasureMethod(widthMeasureSpec, heightMeasureSpec);
},
onLayout: function (changed, left, top, right, bottom): void {
// call the base class implementation
this.super.onLayout(changed, left, top, right, bottom);
// do something additionally
// myOnLayoutMethod(left, top, right, bottom);
}
}
The New Class (Type)
The new type inherits the specified class (android.view.ViewGroup
in this concrete example) and defines overrides as specified by the JavaScript object literal, passed to the extend
function:
public class android_view_ViewGroup_myFile_l10_c20 extends android.view.ViewGroup {
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
// populate the arguments list in an array to pass to the JavaScript world
java.lang.Object[] params = new java.lang.Object[2];
params[0] = widthMeasureSpec;
params[1] = heightMeasureSpec;
// call the associated JavaScript method
return excuteJavaScriptMethod(this, “onMeasure”, params);
}
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
// populate the arguments list in an array to pass to the JavaScript world
java.lang.Object[] params = new java.lang.Object[5];
params[0] = changed;
params[1] = left;
params[2] = top;
params[3] = right;
params[4] = bottom;
// call the associated JavaScript method
return excuteJavaScriptMethod(this, “onLayout”, params);
}
}
Note: The arguments in the above example are converted to JavaScript data types as described in the Java to JavaScript and the Kotlin to JavaScript data conversion article