Titanium中Android模块开发之---移植到1.8版本的指南官方地址:Android Module Porting Guide for 1.8.0.1
Titanium中Android模块开发之---移植到1.8版本的指南
官方地址:Android Module Porting Guide for 1.8.0.1
概要
伴随着Titanium Mobile 1.8.0.1的发布, 对于Android的支持改变成支持多个Javascript引擎(V8和Rhino). 这个变化也影响了很多的APIs和第三方modules. 尽管平台的变化, 大多数内部module需要修改的地方也就是简单的方法标记,类型重命名, import 位置。
Manifest变化
开发在Titanium Mobile 1.8.0.1 (或者更高)的第三方modules需要在module manifest中设置一个新的属性:apiversion,值为2.
TiDrawableReference.fromUrl(proxy.getTiContext(), tiImage)
becomes:
TiDrawableReference.fromUrl(proxy.getActivity(), tiImage)
In the specific case of fromUrl, the following form can also be used:
TiDrawableReference.fromUrl(proxy, tiImage)
The specific alternative varies based on which API point is being modified, but generally there is a very simple alternative that can be used.
2. Use KrollFunction instead of KrollCallback
KrollFunction has been added, and KrollCallback has been removed; The dual runtime change required a common interface to be defined to replace KrollCallback.
In most cases, a direct replacement of KrollCallback with KrollFunction should suffice.
主要变化Known compatibility points that need to be changed:
1. Remove KrollInvocation as a method argment.
Example:
@Kroll.methodpublic void myMethod(final KrollInvocation invocation, KrollDict args)
becomes:
@Kroll.methodpublic void myMethod(KrollDict args)
2. Remove TiContext from your module constructor. super(TiContext) will no longer work due to the previously mentioned TiContext change. In most if not all cases, simply removing the argument will address the problem.
Example:
public BoxModule(TiContext context) { super(context);}
becomes:
public BoxModule() { super();}
3. When replacing KrollCallback with KrollFunction, you need to to pass a KrollObject argument to the call and callAsync methods.
Example:
KrollCallback success = (KrollCallback)args.get("success");...success.callAsync(event);
becomes:
KrollFunction success = (KrollFunction)args.get("success");...success.callAsync(proxy.getKrollObject(), event);
4. Change getView() on a TiViewProxy to getOrCreateView()/
Example:
View view = myViewProxy.getView();
becomes:
View view = myViewProxy.getOrCreateView();
5. Change TiDrawableReference.fromUrl(proxy.getTiContext(), tiImage) to TiDrawableReference.fromUrl(proxy.getActivity(), tiImage) due to the TiContext change. Same applies to all the "from<source>" methods in TiDrawableReference.
Example:
TiDrawableReference ref = TiDrawableReference.fromBlob(context, blob);
becomes:
TiDrawableReference ref = TiDrawableReference.fromBlob(getActivity(), blob);
6. <KrollInvocation>.getActivity() no longer exists. getActivity can be called on each proxy to get the activity for that proxy or TiApplication.getAppCurrentActivity() and TiApplication.getAppRootActivity() can be used for getting activity instances to work with. In general, system services, etc., can and should use the root activity if it exists. TiApplication.getRootOrCurrentActivity() will serve this purpose in the vast majority of situations.
Example:
Activity activity = invocation.getActivity();
becomes:
Activity activity = TiApplication.getAppRootOrCurrentActivity();
7. Calling addOnLifeCycleEvent on a module is no longer necessary. Modules are now automatically registered to receive the lifecycle events (onPause, onResume, onStart, onStop, and onDestroy).
8. <KrollEventManager>.addOnEventChangeListener() is no longer supported. The new mechanism for this is to override KrollProxy.eventListenerAdded, and move the code from the listenerAdded method into the overridden eventListenerAdded method after the call to super.eventListenerAdded.
Example:
@Overridepublic void eventListenerAdded(String type, int count, KrollProxy proxy){ super.eventListenerAdded(type, count, proxy); // this is the logic that used to live inside the listenerAdded() method if (MY_CUSTOM_EVENT.equals(type)) { invokeMyCustomMethod(); }}
9. resolveUrl has been moved to the proxy object.
Example:
_proxy.getTiContext().resolveUrl(url);
becomes:
_proxy.resolveUrl(url);
10. Change getModuleById to getModuleByName and specify the module name in the module constructor. By default the module cannot be found by calling getModuleByName. You must use the form of super() in the module constructor that allows you to specify the module name.
Example:
public myModule(TiContext tiContext) { super(tiContext);}... TiApplication appContext = TiApplication.getInstance(); MyModule myModule =(MyModule)appContext.getModuleById("ti.mymodule");
becomes:
public myModule() { super("mymodule");}... TiApplication appContext = TiApplication.getInstance(); MyModule myModule = (MyModule)appContext.getModuleByName("mymodule");
11. <TiContext>.getAndroidContext() no longer exists. If the ContextWrapper returned originally is being used to access system or app level resources, use TiApplication.getInstance() or TiApplication.getInstance().getApplicationContext() instead. To get the ContextWrapper for the top most Activity, use TiApplication.getAppCurrentActivity() instead.
Example:
myFunction(context.getAndroidContext());
becomes:
myFunction(TiApplication.getInstance().getApplicationContext());
12. getContext() on a TiProxy no longer exists because its purpose was to return a TiContext instance. This call should no longer be needed in module implementation once TiContext is no longer being passed in as an argument (the normal use case for this method).
Example:
Context context = proxy.getContext();
becomes:
Context context = TiApplication.getInstance().getApplicationContext();
13. Remove context from TiFileFactory.createTitaniumFile.
Example:
TiBaseFile file = TiFileFactory.createTitaniumFile(context, fileURL, false);
becomes:
TiBaseFile file = TiFileFactory.createTitaniumFile(fileURL, false);
14. getChangeDir is now called on the TiApplication instance.
Example:
_proxy.getContext().getCacheDir()
becomes:
TiApplication.getInstance().getCacheDir()
15. Overriding the fireEvent method requires a change to the method signature. The argument has changed from a KrollDict class to an Object.
Example:
@Overridepublic boolean fireEvent(String eventName, KrollDict data) {...}
becomes:
@Overridepublic boolean fireEvent(String eventName, Object data) {...}
16. <KrollInvocation>.getTiContext() no longer exists. TiContext is no longer needed. Some examples show this being used to get the TiApplication instance. TiApplication.getInstance() can be used instead.
17. Change usage of KrollDict in method signatures to HashMap. Dictionary values are now passed to methods as HashMap objects. If you need to access any of the KrollDict methods on the HashMap object you can create a KrollDict object from the HashMap object.
Example:
public void myMethod(KrollInvocation invocation, KrollDict args){ TiBlob blob = (TiBlob) args.get("image"); int myId = args.getInt("id);
becomes:
public void myMethod(HashMap args) { KrollDict argsDict = new KrollDict(args); TiBlob blob = (TiBlob) args.get("image"); int myID = argsDict.getInt("id");
18. runOnUiThread is no longer supported. You must explicitly manage and call your methods on the UI thread where necessary. Specifically, you can use the TiMessenger class to run code on the UI thread.
Example:
@Kroll.method(runOnUiThread = true)public void refresh() { if (_view != null) _view.refresh();}
becomes:
private static final int MSG_REFRESH = 50000;private final Handler handler = new Handler(TiMessenger.getMainMessenger().getLooper(), new Handler.Callback (){ public boolean handleMessage(Message msg) { switch (msg.what) { case MSG_REFRESH: { AsyncResult result = (AsyncResult) msg.obj; handleRefresh(); result.setResult(null); return true; } } return false; }}); private void handleRefresh(){ _view.refresh();} @Kroll.methodpublic void refresh() { if (_view != null) { if (!TiApplication.isUIThread()) { TiMessenger.sendBlockingMainMessage(handler.obtainMessage(MSG_REFRESH)); } else { handleRefresh(); } }}