我之前写过一个RPG游戏<<熊猫之魂 SoulOfPanda>>
编译器使用的是SpriteBuilder,很好很强大!全部代码都由Objc完成,现在想尝试一下在其中混入Swift代码.
我的目的很简单,用Swift写一个GCMan9类,派生自Objc中的GameCharacter类,最后在Objc中使用GCMan9这个类.
GameCharacter类是游戏人物的基类,我从中派生了十几个子类分别表示游戏主角,各种NPC,各种敌人等等.
下面是GCMan1类的头文件:
#import "GameCharacter.h"
@interface GCMan1 : GameCharacter
@end
接着是实现文件:
#import "GCMan1.h"
#import "CCAnimation+Helper.h"
@implementation GCMan1
-(id)initWithGameScene:(GameScene *)gameScene{
self = [super initWithGameScene:gameScene andImageNamed:@"man1_forward_2.png"];
if (self) {
_facingForwardAnimation = [CCAnimation animation:@"man1" middle:@"forward" frameCount:3];
_facingBackAnimation = [CCAnimation animation:@"man1" middle:@"back" frameCount:3];
_facingLeftAnimation = [CCAnimation animation:@"man1" middle:@"left" frameCount:3];
_facingRightAnimation = [CCAnimation animation:@"man1" middle:@"right" frameCount:3];
self.speedPerStep = 0.4;
self.isNPC = YES;
self.gcName = @"GCMan1";
}
return self;
}
@end
可以看到GCMan1里只是根据实际角色初始化对应的纹理和动画,值得注意的是_facingForwardAnimation之类的变量是在其父类GameCharacter中定义的,如下:
@interface GameCharacter : CCSprite{
@protected
GameScene *_gameScene;
//移动的4个方向动画
CCAnimation *_facingForwardAnimation;
CCAnimation *_facingBackAnimation;
CCAnimation *_facingLeftAnimation;
CCAnimation *_facingRightAnimation;
}
这些变量增加了保护修饰,这里提一下,因为之后的Swift代码要在这里折腾一下.
按道理,如果在一个纯Objc项目中创建一个Swift文件,Xcode会提示你是否创建一个桥接文件,但是我这里并没有提示,所以我们得手动创建一个,格式是
项目名-Bridging-Header.h
里面导入需要在Swift文件中需要使用的类的头文件,在这里其内容是这样的:
#ifndef SoulOfPanda_Bridging_Header_h
#define SoulOfPanda_Bridging_Header_h
#import "GameCharacter.h"
#import "GameScene.h"
#endif /* SoulOfPanda_Bridging_Header_h */
然后新建一个GCMan9.swift文件,内容与Objc中的类似:
class GCMan9:GameCharacter{
override init(gameScene: GameScene!,andImageNamed imageName: String!) {
super.init(gameScene: gameScene,andImageNamed: imageName)
}
override init(gameScene:GameScene?){
super.init(gameScene: gameScene,andImageNamed: "man1_forward_2.png",andTintColor: UIColor.orangeColor())
//怎么设置父类中的保护变量???
speedPerStep = 0.4
isNPC = true
gcName = "GCMan9"
}
}
现在问题来了,父类中的几个@protected定义的变量在Swift中不可见!查阅了一些资料没找到解决办法,只有采用迂回方法,回到GameCharacter.m中新建一个帮助方法:
//设置GameCharacter类中的保护变量,在Swift中调用
-(void)setFacingAnimation:(NSString *)name frameCount:(int)count{
_facingForwardAnimation = [CCAnimation animation:name middle:@"forward" frameCount:count];
_facingBackAnimation = [CCAnimation animation:name middle:@"back" frameCount:count];
_facingLeftAnimation = [CCAnimation animation:name middle:@"left" frameCount:count];
_facingRightAnimation = [CCAnimation animation:name middle:@"right" frameCount:count];
}
同时别忘了在GameCharacter.h接口中导出该方法,回到GCMan9.swift文件中,将注释那一行替换为如下代码:
setFacingAnimation("man1",frameCount: 3)
因为你要在Objc中使用Swift中的类,所以你需要在对应的m文件中包含如下h文件:
#import "SoulOfPanda-Swift.h"
文件名格式很简单就是”项目名-Swift.h”
在实际测试之前,我们还要找到游戏某个场景的plist文件,将其中的NPC类名换为GCMan9:
运行一下游戏,咦怎么崩溃挂掉了…检查一下栈回溯,发现在以下方法中有问题:
//返回一个"假的"的GC对象,只能用于显示和获取数据,不能实际在场景中存在
+(instancetype)gcFakeWithName:(NSString *)name{
Class class = NSClassFromString(name);
GameCharacter *gc = [[class alloc] initWithGameScene:nil];
return gc;
}
仔细一看,发现NSClassFromString方法返回的是nil,虽然name是正确的@”GCMan9”.
测试了一下发现使用如下代码是没有问题的:
GCMan9 *man9 = [[GCMan9 alloc]initWithGameScene:gameScene];
难道是Swift中的类名在Objc中看到的有所不同么?查看Apple开发文档发现,果然如此!在Swift定义的类在Objc代码中实际看到的名字需要加上前缀,即GCMan9在Objc中看到的名字是SoulOfPanda.GCMan9,所以要把代码修改如下:
Class class = NSClassFromString(@"SoulOfPanda.GCMan9");
但问题还是没有解决,我不想一个一个的添加判断语句,这样非常的繁琐.相反我要利用Swift中提供了一个很好的特性,就是@objc伪指令,该指令告诉Swift不要在类名前添加前缀.回到GCMan9.swift在开头添加如下一句:
@objc(GCMan9)
class GCMan9:GameCharacter{
现在运行代码,终于OK啦: