swift – 在特定场景中构建SpriteKit / GameKit排行榜

前端之家收集整理的这篇文章主要介绍了swift – 在特定场景中构建SpriteKit / GameKit排行榜前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
我对 Swift很陌生,而且我在游戏中实现排行榜时遇到了一些麻烦.我刚看了一个教程:’游戏中心排行榜! (Xcode中的Swift 2)’其中GameCenter信息都通过应用程序的一个视图.在我的游戏中,我希望用户能够玩游戏,然后只有当他们使用特定的SKScene时,他们才能访问GameCenter.

因此,例如,在GameOverScene上,他们将进行用户身份验证,并且还可以上传他们的高分.我想我也错过了GameViewController(所有教程逻辑所在的位置)和我制作的众多场景之间的一些差异.

这是我的代码,我尝试在GameOverScene上使用GKGameCenterControllerDelegate并创建各种功能以到达GameCenter.当用户在视图中点击某个标签时进行调用:(这显然不起作用,因为我正在尝试访问这样的行上的场景:self.presentViewController(view!,animated:true,completion:nil)

  1. class GameOverScene: SKScene,GKGameCenterControllerDelegate {
  2.  
  3. init(size: CGSize,thescore:Int) {
  4. score = thescore
  5. super.init(size: size)
  6. }
  7. ...
  8.  
  9. override func didMoveToView(view: SKView) {
  10.  
  11. authPlayer()
  12.  
  13. leaderboardLabel.text = "Tap for Leaderboard"
  14. leaderboardLabel.fontSize = 12
  15. leaderboardLabel.fontColor = SKColor.redColor()
  16. leaderboardLabel.position = CGPoint(x: size.width*0.85,y: size.height*0.1)
  17. addChild(leaderboardLabel)
  18.  
  19. ...
  20.  
  21. override func touchesBegan(touches: Set<UITouch>,withEvent event: UIEvent?) {
  22.  
  23. for touch : AnyObject in touches {
  24. let location = touch.locationInNode(self)
  25.  
  26. if(CGRectContainsPoint(leaderBoardLabel.frame,location)){
  27. saveHighscore(score)
  28. showLeaderBoard()
  29. }
  30. }
  31. }
  32.  
  33.  
  34. func authPlayer(){
  35.  
  36. //Create a play
  37. let localPlayer = GKLocalPlayer.localPlayer()
  38.  
  39. //See if signed in or not
  40. localPlayer.authenticateHandler = {
  41. //A view controller and an error handler
  42. (view,error) in
  43.  
  44. //If there is a view to work with
  45. if view != nil {
  46. self.presentViewController(view!,animated:true,completion: nil) //we dont want a completion handler
  47. }
  48.  
  49. else{
  50. print(GKLocalPlayer.localPlayer().authenticated)
  51. }
  52. }
  53. }
  54.  
  55.  
  56. //Call this when ur highscore should be saved
  57. func saveHighscore(number:Int){
  58.  
  59. if(GKLocalPlayer.localPlayer().authenticated){
  60.  
  61. let scoreReporter = GKscore(leaderboardIdentifier: "scoreBoard")
  62. scoreReporter.value = Int64(number)
  63.  
  64. let scoreArray: [GKscore] = [scoreReporter]
  65.  
  66. GKscore.reportscores(scoreArray,withCompletionHandler: nil)
  67.  
  68. }
  69.  
  70. }
  71.  
  72.  
  73. func showLeaderBoard(){
  74.  
  75. let viewController = self.view.window?.rootViewController
  76. let gcvc = GKGameCenterViewController()
  77.  
  78. gcvc.gameCenterDelegate = self
  79.  
  80. viewController?.presentViewController(gcvc,animated: true,completion: nil)
  81.  
  82.  
  83. }
  84.  
  85.  
  86. func gameCenterViewControllerDidFinish(gameCenterViewController: GKGameCenterViewController) {
  87. gameCenterViewController.dismissViewControllerAnimated(true,completion: nil)
  88. }

关于如何进行此操作的任何建议都会很棒,我认为我可能会让整个场景/视图控制器混乱并导致问题.谢谢!

这个答案部分地延续了我们在评论中所停留的位置,因为你没有发布你的整个代码,我无法确切地知道你的挂断在哪里,但这是我和一个单独的指南放在一起的(这是一个稍微的您发布的不同版本的代码):

大部分代码的作者:

https://www.reddit.com/r/swift/comments/3q5owv/how_to_add_a_leaderboard_in_spritekit_and_swift_20/

GameViewController.swift:

  1. import UIKit
  2. import SpriteKit
  3. import GameKit
  4.  
  5. class GameViewController: UIViewController {
  6.  
  7. func authenticateLocalPlayer() {
  8. let localPlayer = GKLocalPlayer.localPlayer()
  9. localPlayer.authenticateHandler = {(viewController,error) -> Void in
  10.  
  11. if (viewController != nil) {
  12. self.presentViewController(viewController!,completion: nil)
  13. }
  14. else {
  15. print((GKLocalPlayer.localPlayer().authenticated))
  16. }
  17. }
  18. }
  19.  
  20. override func viewDidLoad() {
  21. super.viewDidLoad()
  22.  
  23. /////authentication//////
  24. authenticateLocalPlayer()
  25.  
  26. //... The rest of the default code
  27. }
  28.  
  29. //... The rest of the default code
  30. }

GameScene.swift(或任何你想使用GC的场景):

  1. import SpriteKit
  2. import GameKit
  3. import UIKit
  4.  
  5. // Global scope (I generally put these in a new file called Global.swift)
  6. var score = 0
  7.  
  8.  
  9. //sends the highest score to leaderboard
  10. func saveHighscore(gamescore: Int) {
  11. print ("You have a high score!")
  12. print("\n Attempting to authenticating with GC...")
  13.  
  14. if GKLocalPlayer.localPlayer().authenticated {
  15. print("\n Success! Sending highscore of \(score) to leaderboard")
  16.  
  17. //---------PUT YOUR ID HERE:
  18. // |
  19. // |
  20. // V
  21. let my_leaderboard_id = "YOUR_LEADERBOARD_ID"
  22. let scoreReporter = GKscore(leaderboardIdentifier: my_leaderboard_id)
  23.  
  24. scoreReporter.value = Int64(gamescore)
  25. let scoreArray: [GKscore] = [scoreReporter]
  26.  
  27. GKscore.reportscores(scoreArray,withCompletionHandler: {error -> Void in
  28. if error != nil {
  29. print("An error has occured:")
  30. print("\n \(error) \n")
  31. }
  32. })
  33. }
  34. }
  35.  
  36. // Your scene:
  37. class GameScene: SKScene,GKGameCenterControllerDelegate {
  38.  
  39. // Local scope variables (for this scene):
  40.  
  41. // Declare a new node,then initialize it
  42. let call_gc_node = SKLabelNode(fontNamed:"Chalkduster")
  43. let add_score_node = SKLabelNode(fontNamed: "Helvetica")
  44.  
  45.  
  46. override func didMoveToView(view: SKView) {
  47.  
  48. // Give our GameCenter node some stuff
  49. initGCNode: do {
  50.  
  51. // Set the name of the node (we will reference this later)
  52. call_gc_node.name = "callGC"
  53.  
  54. // Default inits
  55. call_gc_node.text = "Send your Highscore of \(score) into Game Center"
  56. call_gc_node.fontSize = 25
  57. call_gc_node.position = CGPoint(
  58. x:CGRectGetMidX(self.frame),y:CGRectGetMidY(self.frame))
  59.  
  60. // Self here is the instance (object) of our class,GameScene
  61. // This adds it to our view
  62. self.addChild(call_gc_node)
  63. }
  64.  
  65. // Give our Add label some stuff
  66. initADDLabel: do {
  67.  
  68. // Set the name of the node (we will reference this later)
  69. add_score_node.name = "addGC"
  70.  
  71. // Basic inits
  72. add_score_node.text = "ADD TO score!"
  73. add_score_node.fontSize = 25
  74. add_score_node.position = call_gc_node.position
  75.  
  76. // Align our label some
  77. add_score_node.runAction(SKAction.moveByX(0,y: 50,duration: 0.01))
  78.  
  79. // Add it to the view
  80. self.addChild(add_score_node)
  81. }
  82.  
  83. }
  84.  
  85.  
  86. override func touchesBegan(touches: Set<UITouch>,withEvent event: UIEvent?) {
  87. for touch in touches {
  88.  
  89. // Get the position of our click
  90. let TPOINT = touch.locationInNode(self)
  91.  
  92. // Get the name (string) of the node that was touched
  93. let
  94. node_that_was_touched: String?
  95. = nodeAtPoint(TPOINT).name
  96.  
  97.  
  98. // Prepare for switch statement,when we unwrap the optional,we don't want nil
  99. guard (node_that_was_touched != nil)
  100. else { print("-> before switch: found nil--not entering Switch");
  101. return
  102. }
  103.  
  104.  
  105. // Find out which node we clicked based on node.name?,then do stuff:
  106. switch node_that_was_touched! {
  107.  
  108. case "callGC":
  109. // We clicked the GC label:
  110.  
  111. GameOver: do {
  112.  
  113. print("GAME OVER!")
  114.  
  115. // If we have a high-score,send it to leaderboard:
  116. overrideHighestscore(score)
  117.  
  118. // Reset our score (for the next playthrough)
  119. score = 0
  120.  
  121. // Show us our stuff!
  122. showLeader()
  123. }
  124.  
  125. case "addGC":
  126. // we clicked the Add label:
  127.  
  128. // Update our *current score*
  129. score += 1
  130.  
  131.  
  132. default: print("no matches found")
  133. }
  134.  
  135. }
  136.  
  137. }
  138.  
  139.  
  140. override func update(currentTime: CFTimeInterval) {
  141. /* Called before each frame is rendered */
  142. call_gc_node.text = "Send your Highscore of \(score) into Game Center"
  143.  
  144. }
  145.  
  146.  
  147. // Gamecenter
  148. func gameCenterViewControllerDidFinish(gameCenterViewController: GKGameCenterViewController) {
  149. gameCenterViewController.dismissViewControllerAnimated(true,completion: nil)
  150. }
  151.  
  152. //shows leaderboard screen
  153. func showLeader() {
  154. let viewControllerVar = self.view?.window?.rootViewController
  155. let gKGCViewController = GKGameCenterViewController()
  156. gKGCViewController.gameCenterDelegate = self
  157. viewControllerVar?.presentViewController(gKGCViewController,completion: nil)
  158. }
  159.  
  160. // Your "game over" function call
  161. func overrideHighestscore(gamescore: Int) {
  162. NSUserDefaults.standardUserDefaults().integerForKey("highscore")
  163. if gamescore > NSUserDefaults.standardUserDefaults().integerForKey("highscore")
  164. {
  165. NSUserDefaults.standardUserDefaults().setInteger(gamescore,forKey: "highscore")
  166. NSUserDefaults.standardUserDefaults().synchronize()
  167.  
  168. saveHighscore(gamescore)
  169. }
  170. }
  171. }

要特别注意

29: let my_leaderboard_id = "YOUR_LEADERBOARD_ID"

我在那里放了一些愚蠢的ASCII艺术,以确保你不会错过它.您必须从GameCenter设置中输入您的实际排行榜ID.

您还必须添加GameCenter库,然后执行iTunes连接以在弹出窗口中实际查看您的高分.

我认为你最初的问题是没有理解SpriteKit甚至iOS视图如何工作的后端(这完全没问题,因为Apple让人很容易进入并且很容易).但是,如您所见,遵循指南/教程可能很困难,因为您的实施会有所不同.

以下是一些很好的信息:

SK中每个帧发生的情况图:

所以你看,SKScene是一个包含节点和动作等所有有趣内容的类,并且是一切(对你很重要)发生的地方.您可以通过编辑器生成这些场景,但是您可能需要创建一个新的.swift文件(因为每个场景都有自己的逻辑).

编辑器只是初始化一堆东西的“捷径”,老实说,你可以用很少的代码制作完整的游戏(但你很快发现你想要更多)

所以在这段代码中,你声明了GameScene或PauseScreen(它们基本上只是类声明,继承自SKScene),你很快就会发现这一行谈论ISNT场景:

override func didMoveToView(view: SKView)
.. it’s calling a SKView… what is that,and where did it come from?

(Read about SKView here,and look at its inheritance):

07002

我们在GameViewController文件中找到这个SKView声明(这只是一个类),注意它与普通的iOS应用程序大致相同,因为它继承了UIViewController:

  1. override func viewDidLoad() {
  2. super.viewDidLoad()
  3. if let scene = GameScene(fileNamed:"GameScene") {
  4. // Configure the view.
  5. let skView = self.view as! SKView
  6. skView.showsFPS = true
  7. skView.showsNodeCount = true
  8.  
  9. /* Sprite Kit applies additional optimizations to improve rendering performance */
  10. skView.ignoresSiblingOrder = true
  11.  
  12. /* Set the scale mode to scale to fit the window */
  13. scene.scaleMode = .AspectFill
  14.  
  15. skView.presentScene(scene)
  16. }

同样,该方法在GameViewController.swift中声明,基本上就是这样:
class GameViewController:UIViewController

那么所有这些与iOS应用和SpriteKit有什么关系呢?好吧,他们都被捣碎在彼此之上:

IOS app解剖:

基本上,从右到左,你有一个Window,它是(如果错误的话,我是错误的)AppDelegate,然后是ViewController,然后是你的View,里面有所有很酷的东西(Storyboards就在View里面,就像SKScenes位于View ….标签,节点或按钮内,都位于各自的类中((视图)))

这都是继承的三明治.

查看Apple网站了解更多信息.

https://developer.apple.com/library/safari/documentation/UserExperience/Conceptual/MobileHIG/ContentViews.html#//apple_ref/doc/uid/TP40006556-CH13-SW1

https://developer.apple.com/spritekit/

https://developer.apple.com/library/ios/documentation/SpriteKit/Reference/SpriteKitFramework_Ref/

https://developer.apple.com/library/safari/documentation/UserExperience/Conceptual/MobileHIG/Anatomy.html

基本上,一切都是继承自类继承的类的类,依此类推……它可能会变得混乱.你也可以通过CMD点击它们在Xcode中看到这些遗产,这会将你跳转到源文件.

Goodluck与你在SpriteKit的学习和冒险:)

猜你在找的Swift相关文章