原文链接:
AIR Native Extension实现iOS应用内付费(In-App Purchase)全教程(二)——AIR面向iOS设备的原生扩展

本文的主要内容如下

  • AIR Native Extension介绍
  • ANE的组成部分
  • ActionScript 3.0扩展
  • Objective-C 扩展
  • 使用ADT打包ANE
  • 使用ADT打包IPA

————————————————————————————
AIR Native Extension介绍

AIR Native Extension (ANE)是AIR 3.0的一项重要特性,简单的说,它允许AIR应用程序通过扩展文件与原生应用程序类库相互通讯,从而让AIR应用实现一些只有原生程序才可以做到的功能。
在ANE出现以前,移动平台上的AIR对系统的访问非常有限,功能的实现都是封装在封闭的,由Adobe定义好的ActionScript 3.0 API内,比如Accelerometer, GeoLocator等AS类。ANE则将AIR彻底开放出来,AIR不再针对具体的功能提供封闭的API,而是允许开发者通过AIR的扩展机制自由调用使用原生语言开发的类库。这样可以让AIR应用程序享有与原生应用程序同等的机会,其意义对Flash技术来说是划时代的。
————————————————————————————
ANE的组成部分

ANE支持向Windows、Mac OSX、Android和iOS各个平台原生应用程序的扩展,本文只针对iOS平台进行介绍。在iOS平台中,ANE的组成部分基本分为ActionScript 3.0扩展类库和Objective-C原生扩展类库两个部分,这两个部分打包后生成AIR扩展文件(.ane),最后和AIR应用程序一起打包成iOS原生应用IPA文件。如下图所示。

ANE的组成部分

图1 ANE的组成部分
————————————————————————————
ActionScript 3.0扩展

ANE的AS扩展部分是一个SWC,AIR 3.0 SDK里为flash.external.ExtensionContext类添加了新的方法。如下例所示:

import flash.external.ExtensionContext;
...
private var ext:ExtensionContext;
...
ext = ExtensionContext.createExtensionContext("com.adobe.appPurchase","");

在这个例子里,ExtensionContext通过静态方法createExtensionContext()来获得一个实例,参数com.adobe.appPurchase是这个扩展的ID,它非常重要,在扩展的配置文件里和应用程序描述文件中都需要用这个ID进行配对。

调用原生类中定义的方法可以用方法call()来实现,由于是同步调用,所以函数可以有返回值。如在原生类中定义的方法finish,可以用下面的代码来调用。

var result:Object = ext.call("finish");

我们还可以给ExtensionContext类添加事件侦听,用来获取从原生类中派发回来的事件。

ext.addEventListener(StatusEvent.STATUS,onStatus);
public function onStatus(e:StatusEvent):void{ 
	switch(e.code){
             case "removeTransaction":
             ...
        }
}

————————————————————————————
Objective-C 扩展
接下来是原生类的部分,如果你注册成为苹果iOS开发者,那么你可以在苹果开发者网站上免费下载Object-C的开发工具XCode。关于如何注册成为苹果iOS开发者,请参考我的这篇文章,如何成为一个合法的iOS开发者

总的来说,Objective-C 虽然语法比较奇怪,但只要掌握了基本的规则,还是和ActionScript一样易懂。OBJC扩展类需要引入一个FlashRuntimeExtension.h类包,它实现了和ActionScript沟通的接口。

引入FlashRuntimeExtension.h之后,可以用下面的代码定义一个FREObject方法,FREObject是接口类型。这里要注意,与AS的接口包括函数返回值,都要定义成FREObject类型,比如代码中的retVal。

FREObject finishTransaction1(FREContext ctx, void* funcData, uint32_t argc, FREObject argv[]) {
	NSLog(@"Finish Transaction Called");
	BOOL matchFound = NO;
	const uint8_t* str = nil;
	uint32_t len = -1;
	......
	FREObject retVal;
	if(FRENewObjectFromBool(matchFound, &retVal) == FRE_OK){
		return retVal;
	}else{
		return nil;
	}
}

要把FREObject方法定义成接口,还需要在ContextInitializer方法内进行配置,如下:

//这里是需要定义的接口的数量
*numFunctionsToTest = 6;
 
//定义一个FRENamedFunction类型的实例func,初始化函数的个数
FRENamedFunction* func = (FRENamedFunction*)malloc(sizeof(FRENamedFunction)*6);
 
//定义一个接口,name是字符串"getProducts",函数体是getProducts
func[0].name = (const uint8_t*)"getProducts";
func[0].functionData = NULL;
func[0].function = &getProducts; 
 
func[1].name = (const uint8_t*)"startPayment";
func[1].functionData = NULL;
func[1].function = &startAppPayment; 
 
func[2].name = (const uint8_t*)"finish"; 
func[2].functionData = NULL;
func[2].function = &finishTransaction1;
 
func[3].name = (const uint8_t*)"muted"; 
func[3].functionData = NULL;
func[3].function = &muted;
 
func[4].name = (const uint8_t*)"restore"; 
func[4].functionData = NULL;
func[4].function = &restoreTrans;
 
func[5].name = (const uint8_t*)"trans"; 
func[5].functionData = NULL;
func[5].function = &getTrans;
 
*functionsToSet = func;
....

而ContextInitializer方法,是在原生扩展类的初始化函数ExtInitializer中指定的:

void ExtInitializer(void** extDataToSet, FREContextInitializer* ctxInitializerToSet, 
					FREContextFinalizer* ctxFinalizerToSet) {
	NSLog(@"Extension Initialized");
	*extDataToSet = NULL;
	*ctxInitializerToSet = &ContextInitializer;
	*ctxFinalizerToSet = &ContextFinalizer;
}

ExtInitializer是原生扩展的程序入口,它可以通过扩展配置文件extension.xml来定义:

<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<extension xmlns="http://ns.adobe.com/air/extension/2.5">
  <id>com.adobe.appPurchase</id>
  <versionNumber>1</versionNumber>
  <platforms>
    <platform name="iPhone-ARM">
            <applicationDeployment>
                <nativeLibrary>libAppPurchase.a</nativeLibrary>
                <initializer>ExtInitializer</initializer>
                <finalizer>ExtFinalizer</finalizer>
            </applicationDeployment>
        </platform>
  </platforms>
</extension>

我介绍的这个顺序,实际上就是实际程序编写的思路,先确定接口,再实现连接。 也许有朋友和我一开始接触OBJC的时候一样,对这些代码一头雾水。没有关系,在这篇教程里我只是对流程做简短的介绍,具体的代码解析会在本系列的最后一篇教程里做更详细的讲解。那么接下来让我来介绍下一个部分,打包扩展。
————————————————————————————
使用ADT打包ANE

在图1中,我介绍了.ane文件的组成,它包括了AS类库(.swc)和原生类(.a)两个部分,以及刚才我们介绍的这个扩展配置文件extension.xml。那么要打包ane我们还需要哪些文件呢?
打包ANE所需要的文件
图2 打包ANE所需要的文件

如图2所示,所选择的文件以及文件夹就是打包ANE所需要的所有文件,它包括:
1,AIR SDK打包应用程序和类库(bin,lib)
2,ActionScript扩展类包.swc,如图ANE_IAP_ASLib.swc
3,ActionScript扩展类包.swf,如图library.swf,可以通过将SWC的文件扩展名改成ZIP后解压缩得到。
4,Objective-C扩展类包.a,如图libAppPurchase.a,可以通过在Xcode中编译项目得到。
5,扩展配置文件XML,如图extension.xml
6,一个打包证书,如图selfsigned.p12,可以通过Flash CS5的AIR发布设置生成。

一切就绪后便可以使用命令行进行打包,注意路径,下例路径为当前文件夹。

bin/adt -package -storetype pkcs12 -keystore selfsigned.p12 -storepass 1234 -target ane ext/InApp.ane extension.xml -swc ANE_IAP_ASLib.swc -platform iPhone-ARM library.swf libAppPurchase.a

————————————————————————————
使用ADT打包IPA

.ane文件打包成功后,便可以用来打包IPA文件,也就是iOS应用程序包。如果你对开发iOS应用的必要流程还不很清楚,请参阅我的这篇教程,如何使用iOS开发者授权以及如何申请证书。我以前介绍过如何用Flash Professional CS5打包IPA,今天主要介绍如何用AIR SDK的打包工具ADT来生成含有ANE扩展的IPA。

使用ADT生成含有ANE扩展的IPA所需要的文件
图3 使用ADT生成含有ANE扩展的IPA所需要的文件

如图3所示,所选择的文件就是生成IPA的必要文件:
1,应用程序文件SWF,如图是ANE_IAP_Example.swf。
2,开发者设备授权文件.mobileprovision,如图是ghostbride_dev.mobileprovision。
3,开发者签名证书文件.p12,如图是jameslidevelopment.p12。
4,应用程序描述文件XML,如图是info-app.xml。
5,扩展包路径,如图是ext
6,如果应用程序有图标图片,还需要图标文件夹,如图是icon

在应用描述文件XML中,需要对扩展追加一个定义:

<extensions>
    <extensionID>com.adobe.appPurchase</extensionID> 
</extensions>

这里可以看到,在AS扩展类、扩展配置文件extension.xml和应用描述文件info-app.xml中都指定了一个统一扩展的ID: com.adobe.appPurchase。

利用下面的命令行可以打包生成Main.ipa:

bin/adt -package -target ipa-test-interpreter -provisioning-profile ghostbride_dev.mobileprovision -storetype pkcs12 -keystore jameslidevelopment.p12 -storepass 1234 Main.ipa info-app.xml ANE_IAP_Example.swf -extdir ext icon