Как вернуться из SKScene в UIViewController (MenuVC)?

У меня такой же вопрос, как и у парня в этой теме Как перейти от SKScene к UIViewController ?. Во-первых, я должен сказать, что я совсем новичок, и все это для меня ново.

Мое главное меню (первый экран, когда мое приложение загрузилось) находится здесь:

class MenuVC: UIViewController {

... }

Взаимодействия моего приложения (игры) можно было бы сделать здесь:

    //
//  GameScene.swift
//  Doodle Pong
//
//  Created by Daniel Kloe on 28.02.17.
//  Copyright © 2017 Daniel Kloe. All rights reserved.
//

import SpriteKit
import GameplayKit
import Foundation
import UIKit


class GameScene: SKScene {

//let storyboard = UIStoryboard(name: "Main", bundle: nil)

//let MenuVC = storyboard.instantiateViewController(withIdentifier: "MenuVC")

//MenuVC.view.frame = (self.view?.frame)!

//MenuVC.view.layoutIfNeeded()


//UIView.transition(with: self.view!, duration: 0.3, options: .transitionFlipFromRight, animations:

//{

//self.view?.window?.rootViewController = vc

//}, completion: { completed in

//})



var ball = SKSpriteNode()
var enemy = SKSpriteNode()
var main = SKSpriteNode()
var score = [Int] ()

var topLbl = SKLabelNode()
var btmLbl = SKLabelNode()

override func didMove(to view: SKView) {

    topLbl = self.childNode(withName: "topLabel") as! SKLabelNode
    btmLbl = self.childNode(withName: "btmLabel") as! SKLabelNode

    ball = self.childNode(withName: "ball") as! SKSpriteNode
    enemy = self.childNode(withName: "enemy") as! SKSpriteNode
    main = self.childNode(withName: "main") as! SKSpriteNode


    let border = SKPhysicsBody(edgeLoopFrom: self.frame)
    border.friction = 0
    border.restitution = 1
    self.physicsBody = border

    startGame()

}

override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {

    for touch in touches{


       // let gameSceneTemp = GameScene(fileNamed: "GameScene")
       // self.scene?.view?.presentScene(gameSceneTemp, transition: SKTransition.doorsCloseHorizontal(withDuration: 0.01))

        let location = touch.location(in: self) //die location wird mit dem Berühren auf dem Bildschirm beschrieben

        if currentGameType == .TwoPlayer{

            if location.y > 0{

                enemy.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt
            }

            if location.y < 0{

                main.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt

            }

        }

        else{

            main.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt

        }


    }
}

func startGame(){

    score = [0,0]
    topLbl.text = "\(score[1])"
    btmLbl.text = "\(score[0])"

    //let delay = SKAction.wait(forDuration: 2000)
    //self.run(delay) //evtl. Wartezeit 2s, funktioniert noch nicht richtig

    ball.physicsBody?.applyImpulse(CGVector(dx: 20, dy: 20))

}

func addScore(playerWhoWon : SKSpriteNode){


    ball.position = CGPoint(x: 0, y: 0)
    ball.physicsBody?.velocity = CGVector(dx: 0, dy: 0)


    if playerWhoWon == enemy{

        score[1] += 1
        ball.physicsBody?.applyImpulse(CGVector(dx: 32, dy: 27))


    }

    else if playerWhoWon == main{


        score[0] += 1
        ball.physicsBody?.applyImpulse(CGVector(dx: -32, dy: -27))

    }

    topLbl.text = "\(score[1])"
    btmLbl.text = "\(score[0])"


}

override func touchesMoved(_ touches: Set<UITouch>, with event: UIEvent?) {

    for touch in touches{

        let location = touch.location(in: self) //die location wird mit dem Berühren auf dem Bildschirm beschrieben

        if currentGameType == .TwoPlayer{

            if location.y > 0{

                enemy.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt
            }

            if location.y < 0{

                 main.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt

            }

        }

        else{

            self.topLbl.zRotation = CGFloat(2 * M_PI)
           // UIView.animate(withDuration: 0, animations: ({

               // self.topLbl.zRotation = CGFloat(2 * M_PI) //CGFloat(.pi / 4.0)

           // }))

            main.run(SKAction.moveTo(x: location.x, duration: 0.01)) //Der "main Balken" wird zu den location bewegt

        }

    }


}

override func update(_ currentTime: TimeInterval) {

    switch currentGameType{

    case .Easy:
        enemy.run(SKAction.moveTo(x: ball.position.x, duration: 1.0))
        break

    case .Medium:
        enemy.run(SKAction.moveTo(x: ball.position.x, duration: 0.5))
        break

    case .Extreme:
        enemy.run(SKAction.moveTo(x: ball.position.x, duration: 0.08))
        break

    case .TwoPlayer:

        break

    }

     //Called before each frame is rendered
    if ball.position.y <= main.position.y - 20{
        addScore(playerWhoWon: enemy)

    }

     else if ball.position.y >= enemy.position.y + 20{
       addScore(playerWhoWon: main)


    }

}



@IBAction func backToMainMenu(_ sender: Any) {
     //self.view?.window?.rootViewController
    self.view?.window!.rootViewController?.dismiss(animated: false, completion: nil)
}


}

В моей функции «backToMainMenu» я пытаюсь вернуться в MenuVC, но получаю предупреждение «Выражение типа« UIViewController? не используется», и если я попробую это в симуляторе, симулятор вылетит.

Я благодарен за любую помощь :).


person dani chloé    schedule 11.03.2019    source источник


Ответы (1)


пытаться:

self.view.window!.rootViewController?.dismiss(animated: false, completion: nil)

Это должно отклонить все представления выше rootViewController.

редактировать:

Это может сработать для вас, поскольку мы определяем ViewController, убедитесь, что в вашей раскадровке ваш MenuVC имеет идентификатор:

let storyboard = UIStoryboard(name: "Main", bundle: nil)

        let vc = storyboard.instantiateViewController(withIdentifier: "MenuVC") // Make sure is the same identifier as in the storyboard.

        vc.view.frame = (self.view?.frame)!

        vc.view.layoutIfNeeded()


        UIView.transition(with: self.view!, duration: 0.3, options: .transitionFlipFromRight, animations:

            {

                self.view?.window?.rootViewController = vc

        }, completion: { completed in

        })

    }

Редактировать:

Кажется, лучший способ сделать это — разместить класс SKScene в другом контроллере представления. Я предлагаю создать еще один ViewController, где будет ваш SKScene, например SKViewController. Вставьте свой MenuVC в navigationController, нажав на него в StoryBoard, щелкнув «Редактор» (в меню вверху) > «Встроить» > «Контроллер навигации». Отредактируйте свой MenuVC так, чтобы вместо открытия сцены вы просто перешли к SKViewController и открыли там сцену. Для этого просто создайте переход от MenuVC в раскадровке, например, с идентификатором «MenuToSKScene», и когда вы хотите открыть сцену, из вашего MenuVC просто выполните:

  self.performsegue(with identifier: "MenuToSKScene", sender: self)

Затем сделайте то, что вы делаете в настоящее время в MenuVC в SKViewController, чтобы открыть сцену.

Это отправит вас к SKViewController, у которого есть ваш класс SKScene. Затем, когда вы захотите вернуться в меню, все, что вам нужно сделать, это:

 self.dismiss(animated: true, completion: nil)

или создайте переход от SKViewController к MenuVC с идентификатором "SKSceneToMenu" и вызовите:

self.performsegue(with identifier: "SKSceneToMenu", sender: self)

По сути, не открывайте SKScene из MenuVC, а создайте отдельный viewController SKViewController, который вы открываете, когда хотите открыть свою сцену, чтобы вы могли легко вернуться в меню.

person Peter Ruppert    schedule 11.03.2019
comment
Мне пришлось немного изменить код на --› self.view?.window!.rootViewController?.dismiss(animated: false, complete: nil). Теперь синтаксис кажется правильным, но если я нажму кнопку главного меню, приложение вылетит (-> 0x108929e28 ‹+72›: movq -0x20 (% rbp), % rdi). - person dani chloé; 11.03.2019
comment
Ваш MenuVC находится в навигационном контроллере? - person Peter Ruppert; 11.03.2019
comment
@danichloé попробуйте решение выше в моем редактировании, оно должно работать из SKScene, определяющего ViewController, а затем с использованием Transition. - person Peter Ruppert; 11.03.2019
comment
Спасибо. Мой MenuVC находится в контроллере представления, который имеет отношение к контроллеру навигации. Вы имеете в виду ... убедитесь, что в вашей раскадровке у вашего MenuVC есть идентификатор, который я должен вставить в контроллер представления моего MenuVC идентификатор раскадровки? - person dani chloé; 11.03.2019
comment
@edey да, верно, где (withIdentifier: "MenuVC") совпадает с идентификатором раскадровки, который вы дали MenuVC (я просто поставил MenuVC в качестве идентификатора) - person Peter Ruppert; 11.03.2019
comment
Ок, отлично. Я поместил ваш код в свой класс GameScene: SKScene {.., но я получаю это предупреждение. Невозможно использовать член экземпляра «раскадровка» в инициализаторе свойства; инициализаторы свойств запускаются до того, как «я» станет доступным. И если я попробую self.storyboard.... Я получу значение типа '(GameScene) -> () -> GameScene' не имеет члена 'storyboard'. Я думаю, что мы близки к этому, может быть, есть небольшая проблема, которую я не понимаю :P - person dani chloé; 11.03.2019
comment
попробуйте добавить import UIKit и import Foundation вверху? - person Peter Ruppert; 11.03.2019
comment
Сейчас добавил, но проблема похоже та же. class GameScene: SKScene { let storyboard = UIStoryboard(name: Main, bundle: nil) let MenuVC = storyboard.instantiateViewController(withIdentifier: MenuVC) // Невозможно использовать член экземпляра «раскадровка» в инициализаторе свойства; инициализаторы свойств запускаются до того, как становится доступным 'self' MenuVC.view.frame = (self.view?.frame)! // Ожидаемое объявление MenuVC.view.layoutIfNeeded() ..... - person dani chloé; 11.03.2019
comment
Я думаю, это связано с твоим @IBAction func Это пуговица? Могу ли я обновить свой вопрос с помощью полного класса SKScene? Я думаю, вам нужно сделать кнопку, создать отдельную функцию, а затем добавить эту функцию в качестве цели кнопки, например, viewDidLoad(). - person Peter Ruppert; 11.03.2019
comment
@danichloé внес изменения, кажется, лучшим решением является использование другого ViewController для размещения SKScene, после чего вы можете легко переключаться между ними. - person Peter Ruppert; 12.03.2019
comment
comment
@danichloé рад, что вы нашли решение! Удачи! - person Peter Ruppert; 13.03.2019