草庐IT

ios - 带有游戏套件的移动平台

coder 2023-09-16 原文

创建一个 2d 游戏,并希望添加移动平台,我可以控制平台的模式和旋转。

例子:


我希望这个平台顺时针运动,当它到达顶部和底部时我希望它相应地旋转,就好像它面向它前进的方向(所以平台基本上会旋转 180 度,因为它顶部和底部呈拱形)。

我不能使用 SKActions,因为我需要物理才能正常工作。

我的想法是,我可以使用具有行为和目标的代理来执行此操作。不确定我是否也需要寻路。

我正在研究如何使用这些功能,但文档和缺乏教程很难破译。我希望有人可以通过提供一个示例来说明这是如何工作的,从而节省我一些反复试验的时间。

提前致谢!

最佳答案

使用 SKActions 或手动调整位置很难按照你想要的方式工作,但尝试一下总是值得的,因为模拟它需要 2 分钟,看看...

我建议做一些类似 Path 类的事情,每帧向平台发送速度命令 ...

实际上,这可能是学习状态机的一个很好的练习。

MOVE RIGHT STATE: if X position > starting position X + 200, enter state "move down"

MOVE DOWN STATE: if Y position < starting position Y - 200, enter state "move left"

MOVE LEFT STATE: if X position < starting position X, enter state "move up"

MOVE UP STATE: if Y position > starting position Y, enter state "move right"

.. 你可以想办法给它更多的曲率,而不仅仅是直角(当改变方向时)

否则,您必须将其转换为类/结构/组件,并为每个平台提供它自己的实例。

///

另一种选择是将物理问题排除在等式之外,并创建一个 playerIsOnPlatform 属性...然后您在每一帧手动调整玩家的位置...(或者可能是 SKConstraint)

这将需要更多关于跳跃等的代码,并很快将事情变成意大利面条(我上次尝试过)

但是,我能够成功地克隆它,使用正确的命中检测沿着这条路线:

https://www.youtube.com/watch?v=cnhlFZeIR7Q

更新:

这是一个工作项目:
https://github.com/fluidityt/exo2/tree/master

样板快速的东西:

import SpriteKit
import GameplayKit


// Workaround to not having any references to the scene off top my head..
// Simply add `gScene = self` in your didMoveToViews... or add a base scene and `super.didMoveToView()`
var gScene = GameScene()

class GameScene: SKScene {

  override func didMove(to view: SKView) {
    gScene = self
  }

  var entities = [GKEntity]()
  var graphs = [String : GKGraph]()

  private var lastUpdateTime : TimeInterval = 0

  func gkUpdate(_ currentTime: TimeInterval) -> TimeInterval {
    if (self.lastUpdateTime == 0) {
      self.lastUpdateTime = currentTime
    }

    let dt = currentTime - self.lastUpdateTime

    for entity in self.entities {
      entity.update(deltaTime: dt)
    }

    return currentTime
  }

  override func update(_ currentTime: TimeInterval) {
    self.lastUpdateTime = gkUpdate(currentTime)
  }
}

这是非常基本的组件:

class Platforms_BoxPathComponent: GKComponent {

  private enum MovingDirection: String { case up, down, left, right }

  private var node: SKSpriteNode!

  private var state: MovingDirection = .right

  private lazy var startingPos: CGPoint = { self.node.position }()

  @GKInspectable var startingDirection: String = "down"
  @GKInspectable var uniqueName: String = "platform"
  @GKInspectable var xPathSize: CGFloat = 400
  @GKInspectable var yPathSize: CGFloat = 400

  // Moves in clockwise:
  private var isTooFarRight: Bool { return self.node.position.x > (self.startingPos.x + self.xPathSize) }
  private var isTooFarDown:  Bool { return self.node.position.y < (self.startingPos.y - self.yPathSize) }
  private var isTooFarLeft:  Bool { return self.node.position.x < self.startingPos.x }
  private var isTooFarUp:    Bool { return self.node.position.y > self.startingPos.y }

  override func didAddToEntity() {
    print("adding component")
    // can't add node here because nodes aren't part of scene yet :(
    // possibly do a thread?
  }

  override func update(deltaTime seconds: TimeInterval) {
    if node == nil {
      node = gScene.childNode(withName: uniqueName) as! SKSpriteNode

      // for some reason this is glitching out and needed to be redeclared..
      // showing 0 despite clearly not being 0, both here and in the SKS editor:
      xPathSize = 300
    }

    let amount: CGFloat = 2 // Amount to move platform (could also be for for velocity)

    // Moves in clockwise:
    switch state {

    case .up:
      if isTooFarUp {
        state = .right
        fallthrough
      } else { node.position.y += amount }

    case .right:
      if isTooFarRight {
        state = .down
        fallthrough
      } else { node.position.x += amount }

    case .down:
      if isTooFarDown {
        state = .left
        fallthrough
      } else { node.position.y -= amount }

    case .left:
      if isTooFarLeft {
        state = .up
        fallthrough
      } else { node.position.x -= amount }

    default:
      print("this is not really a default, just a restarting of the loop :)")
      node.position.y += amount
    }
  }
}

这是您在 sks 编辑器中执行的操作:

关于ios - 带有游戏套件的移动平台,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46431953/

有关ios - 带有游戏套件的移动平台的更多相关文章

  1. ruby - 多次弹出/移动 ruby​​ 数组 - 2

    我的代码目前看起来像这样numbers=[1,2,3,4,5]defpop_threepop=[]3.times{pop有没有办法在一行中完成pop_three方法中的内容?我基本上想做类似numbers.slice(0,3)的事情,但要删除切片中的数组项。嗯...嗯,我想我刚刚意识到我可以试试slice! 最佳答案 是numbers.pop(3)或者numbers.shift(3)如果你想要另一边。 关于ruby-多次弹出/移动ruby​​数组,我们在StackOverflow上找到一

  2. ruby - 如何验证 IO.copy_stream 是否成功 - 2

    这里有一个很好的答案解释了如何在Ruby中下载文件而不将其加载到内存中:https://stackoverflow.com/a/29743394/4852737require'open-uri'download=open('http://example.com/image.png')IO.copy_stream(download,'~/image.png')我如何验证下载文件的IO.copy_stream调用是否真的成功——这意味着下载的文件与我打算下载的文件完全相同,而不是下载一半的损坏文件?documentation说IO.copy_stream返回它复制的字节数,但是当我还没有下

  3. ruby-on-rails - 如何重命名或移动 Rails 的 README_FOR_APP - 2

    当我在我的Rails应用程序根目录中运行rakedoc:app时,API文档是使用/doc/README_FOR_APP作为主页生成的。我想向该文件添加.rdoc扩展名,以便它在GitHub上正确呈现。更好的是,我想将它移动到应用程序根目录(/README.rdoc)。有没有办法通过修改包含的rake/rdoctask任务在我的Rakefile中执行此操作?是否有某个地方可以查找可以修改的主页文件的名称?还是我必须编写一个新的Rake任务?额外的问题:Rails应用程序的两个单独文件/README和/doc/README_FOR_APP背后的逻辑是什么?为什么不只有一个?

  4. Ruby 文件 IO 定界符? - 2

    我正在尝试解析一个文本文件,该文件每行包含可变数量的单词和数字,如下所示:foo4.500bar3.001.33foobar如何读取由空格而不是换行符分隔的文件?有什么方法可以设置File("file.txt").foreach方法以使用空格而不是换行符作为分隔符? 最佳答案 接受的答案将slurp文件,这可能是大文本文件的问题。更好的解决方案是IO.foreach.它是惯用的,将按字符流式传输文件:File.foreach(filename,""){|string|putsstring}包含“thisisanexample”结果的

  5. Get https://registry-1.docker.io/v2/: net/http: request canceled while waiting - 2

    1.错误信息:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:requestcanceledwhilewaitingforconnection(Client.Timeoutexceededwhileawaitingheaders)或者:Errorresponsefromdaemon:Gethttps://registry-1.docker.io/v2/:net/http:TLShandshaketimeout2.报错原因:docker使用的镜像网址默认为国外,下载容易超时,需要修改成国内镜像地址(首先阿里

  6. ruby-on-rails - rbenv:从 RVM 移动到 rbenv 后,在 Jenkins 执行 shell 中找不到命令 - 2

    我从Ubuntu服务器上的RVM转移到rbenv。当我使用RVM时,使用bundle没有问题。转移到rbenv后,我在Jenkins的执行shell中收到“找不到命令”错误。我内爆并删除了RVM,并从~/.bashrc'中删除了所有与RVM相关的行。使用后我仍然收到此错误:rvmimploderm~/.rvm-rfrm~/.rvmrcgeminstallbundlerecho'exportPATH="$HOME/.rbenv/bin:$PATH"'>>~/.bashrcecho'eval"$(rbenvinit-)"'>>~/.bashrc.~/.bashrcrbenvversions

  7. ruby-on-rails - 带有 Zeus 的 RSpec 3.1,我应该在 spec_helper 中要求 'rspec/rails' 吗? - 2

    使用rspec-rails3.0+,测试设置分为spec_helper和rails_helper我注意到生成的spec_helper不需要'rspec/rails'。这会导致zeus崩溃:spec_helper.rb:5:in`':undefinedmethod`configure'forRSpec:Module(NoMethodError)对thisissue最常见的回应是需要'rspec/rails'。但这是否会破坏仅使用spec_helper拆分rails规范和PORO规范的全部目的?或者这无关紧要,因为Zeus无论如何都会预加载Rails?我应该在我的spec_helper中做

  8. Ruby:如何使用带有散列的 'send' 方法调用方法? - 2

    假设我有一个类A,里面有一些方法。假设stringmethodName是这些方法之一,我已经知道我想给它什么参数。它们在散列中{'param1'=>value1,'param2'=>value2}所以我有:params={'param1'=>value1,'param2'=>value2}a=A.new()a.send(methodName,value1,value2)#callmethodnamewithbothparams我希望能够通过传递我的哈希以某种方式调用该方法。这可能吗? 最佳答案 确保methodName是一个符号,而

  9. ruby-on-rails - 带有 Pry 的 Rails 控制台 - 2

    当我进入Rails控制台时,我已将pry设置为加载代替irb。我找不到该页面或不记得如何将其恢复为默认行为,因为它似乎干扰了我的Rubymine调试器。有什么建议吗? 最佳答案 我刚发现问题,pry-railsgem。忘记了它的目的是让“railsconsole”打开pry。 关于ruby-on-rails-带有Pry的Rails控制台,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/question

  10. ruby - 为什么不能使用类IO的实例方法noecho? - 2

    print"Enteryourpassword:"pass=STDIN.noecho(&:gets)puts"Yourpasswordis#{pass}!"输出:Enteryourpassword:input.rb:2:in`':undefinedmethod`noecho'for#>(NoMethodError) 最佳答案 一开始require'io/console'后来的Ruby1.9.3 关于ruby-为什么不能使用类IO的实例方法noecho?,我们在StackOverflow上

随机推荐