###前提・実現したいこと
Unityでクレーンゲームのシュミレーターを作成しています。
アームユニットを下降させる際にrigidbody(Is Kinematic)を用いているのですが、
アームユニットが物体に衝突しても、トリガーを設定していないのでそのまま下降していき非現実的な挙動となってしまいます。(y座標の最低高度は指定してあります。)
現実のクレーンゲームだと紐を巻き取ったりして制御しているそうです。(負荷がかかると紐がたるんで下降がストップする)
どのようなアルゴリズムを用いれば良いでしょうか?
まだプログラムに関するリテラシーが浅いので質問させていただきました。
###試したこと
_★_
◯◯◯
◯◆◯
◯_◯
◯◯◯
★がトリガー(Is trigger)
◯がrigidbody(Use Gravity)を適用した物体(繋がっています)
◆がユニット
上記で◯が★に接触した場合に下降を停止するようにしていますが、ほかの方法の方が良いのではという勘です。
気になる質問をクリップする
クリップした質問は、後からいつでもMYページで確認できます。
またクリップした質問に回答があった際、通知やメールを受け取ることができます。
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
回答1件
0
ベストアンサー
申し訳ないです、せっかく図で説明していただきましたが、いまいち状況が把握しきれず、勝手な解釈での回答です(クレーンが景品に下降していく際、何も対策しないとクレーンが景品の山にめり込んでいってしまうので、個々の景品の上面にクレーン停止用トリガー領域を設置する方法をとったが、これ以外に案はないだろうか...ということでしょうか?)。
トリガー方式も悪くないと思うのですが、違ったアプローチもご所望でしたら、一案としてクレーンをキネマティックなRigidbodyと非キネマティックなRigidbodyを組み合わせた構造にして、両者をFixedJointなど適当なジョイントで繋ぎ、そのジョイントのcurrentForceを監視して負荷のかかり方を検出する...というのを思いつきましたが、いかがでしょう。
クレーンが空中にあるときは、非キネマティック部分にかかる重力によってジョイントに下向きの負荷がかかるが、景品の上に非キネマティック部分が乗っかると、景品からの抗力で下方向の負荷が消失するはず...という目論見で実験してみました。
下図のような構造のクレーンを作って、
クレーンに下記スクリプトをアタッチしました。
C#
1using System.Collections; 2using System.Linq; 3using UnityEngine; 4 5public class Crane : MonoBehaviour 6{ 7 public Transform armTransform; // アームのTransform...アームはキネマティックとし、Transformを操作して移動させる 8 public Joint LoadDetectorJoint; // アームと円盤(非キネマティック)を繋ぐジョイント...ここにかかる負荷を見てアームの衝突を検知する(今回はFixedJointを使用しました) 9 public HingeJoint[] ClawHinges; // クローと円盤を繋ぐヒンジ...Use Springをオンにし、適当なSpring(大きいほど握力が強まる)を設定しておく 10 11 private bool playing; 12 private float homePosition; 13 14 private void Start() 15 { 16 this.homePosition = armTransform.position.y; // アームを上げるとき、この高さまで戻す 17 } 18 19 private void Update() 20 { 21 if (!this.playing && Input.GetKeyDown(KeyCode.Space)) 22 { 23 this.StartCoroutine(this.StartGame()); // スペースキー押下で一連のモーションを開始 24 } 25 } 26 27 private IEnumerator StartGame() 28 { 29 this.playing = true; 30 31 Debug.Log("Start!"); 32 33 yield return this.StartCoroutine(this.MoveClawsMotion(60.0f)); // クローを開く 34 yield return new WaitForSeconds(1.0f); // 少し待機 35 yield return this.StartCoroutine(this.MoveDownMotion()); // アームを下げる 36 yield return new WaitForSeconds(1.0f); // 少し待機 37 yield return this.StartCoroutine(this.MoveClawsMotion(0.0f)); // クローを閉じる 38 yield return new WaitForSeconds(1.0f); // 少し待機 39 yield return this.StartCoroutine(this.MoveUpMotion()); // アームを上げる 40 yield return new WaitForSeconds(1.0f); // 少し待機 41 yield return this.StartCoroutine(this.MoveClawsMotion(60.0f)); // クローを開く 42 yield return new WaitForSeconds(1.0f); // 少し待機 43 yield return this.StartCoroutine(this.MoveClawsMotion(0.0f)); // クローを閉じる 44 45 this.playing = false; 46 } 47 48 private IEnumerator MoveDownMotion() 49 { 50 const int loadMovingAverageLength = 30; // 負荷を積算するフレーム数(試しに「1」などとしてみると、アーム降下開始時の加速によって負荷が消失し、すぐにアームが止まってしまうと思います) 51 const float loadThreshold = 0.0f; // アームを停止する負荷値 52 const float acceleration = 0.5f; // アームの下降加速度 53 const float velocityMax = 1.0f; // アームの最大下降速度 54 55 var loads = Enumerable.Repeat<float>(this.LoadDetectorJoint.currentForce.y, loadMovingAverageLength).ToArray(); 56 var loadSum = loads.Sum(); 57 var loadIndex = 0; 58 59 var velocity = 0.0f; 60 61 while (true) 62 { 63 var previousLoad = loads[loadIndex]; 64 var newLoad = this.LoadDetectorJoint.currentForce.y; // currentForceのY成分を負荷の指標とする 65 66 loadSum += (newLoad - previousLoad); 67 68 var averageLoad = loadSum / loadMovingAverageLength; // 過去loadMovingAverageLengthフレーム分の負荷を平均して、それで停止するべきか判定する 69 70 Debug.LogFormat("Load:{0}", averageLoad); 71 72 if (averageLoad >= loadThreshold) 73 { 74 Debug.Log("Stop!"); 75 break; 76 } 77 78 velocity = Mathf.Max(velocity - acceleration * Time.deltaTime, -velocityMax); 79 this.armTransform.Translate(0.0f, velocity * Time.deltaTime, 0.0f, Space.World); 80 loadIndex = (loadIndex + 1) % loadMovingAverageLength; 81 82 yield return null; 83 } 84 } 85 86 private IEnumerator MoveUpMotion() 87 { 88 const float acceleration = 0.5f; // アームの上昇加速度 89 const float velocityMax = 1.0f; // アームの最大上昇速度 90 91 var velocity = 0.0f; 92 93 while (this.armTransform.position.y < this.homePosition) 94 { 95 velocity = Mathf.Min(velocity + acceleration * Time.deltaTime, velocityMax); 96 this.armTransform.Translate(0.0f, velocity * Time.deltaTime, 0.0f, Space.World); 97 98 yield return null; 99 } 100 101 this.armTransform.position = new Vector3(0.0f, this.homePosition, 0.0f); 102 } 103 104 private IEnumerator MoveClawsMotion(float angle, float timeout = 3.0f) 105 { 106 const float angularSpeed = 30.0f; // 毎秒30°で開閉 107 108 var time = 0.0f; 109 110 while (time < timeout) // 景品を掴もうとしているときなど、クローに負荷がかかっているといつまで経っても目標角に到達できない場合があるので、一定時間が経過したら強制的にモーションを終了 111 { 112 var reached = true; 113 var angleDeltaMax = angularSpeed * Time.deltaTime; 114 115 foreach (var clawHinge in this.ClawHinges) 116 { 117 var spring = clawHinge.spring; 118 119 spring.targetPosition = Mathf.MoveTowardsAngle(spring.targetPosition, angle, angleDeltaMax); 120 clawHinge.spring = spring; 121 reached &= Mathf.Approximately(spring.targetPosition, angle); 122 } 123 124 if (reached) 125 { 126 break; // すべてのクローが目標角に到達したらモーションを終了 127 } 128 129 time += Time.deltaTime; 130 131 yield return null; 132 } 133 } 134} 135
円盤が景品に乗っかっている状態以外でも、アームの加速時など一時的に負荷がなくなる場合があるので、一定フレームの間の平均の負荷でアーム停止の判定をしています。
アームの下降に伴う負荷の変化は下図のようになりました(あくまでも一例で、クレーンの各部位や景品の質量設定などの要因でクレーンの挙動は変わってくると思います)。
[追記]
補足として、今回はcurrentForceのyをそのまま使っていますが、ジョイントでRigidbody同士を接続した場合、currentForceは接続相手の座標系に従うみたいですので、今回のように重力の負荷を調べる場合は、適宜ワールド座標系に直す必要があるかもしれません。
投稿2017/11/27 22:30
編集2017/11/28 00:01総合スコア10811
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
あなたの回答
tips
太字
斜体
打ち消し線
見出し
引用テキストの挿入
コードの挿入
リンクの挿入
リストの挿入
番号リストの挿入
表の挿入
水平線の挿入
プレビュー
質問の解決につながる回答をしましょう。 サンプルコードなど、より具体的な説明があると質問者の理解の助けになります。 また、読む側のことを考えた、分かりやすい文章を心がけましょう。