Unity

スワイプでオブジェクトを中心に回転する処理を実装する(スマホ向け)

3Dモデルをスワイプで回転させられるような機能を実装する

1.各オブジェクトの配置

表示して眺めたい3Dオブジェクトを配置する。
次に空のゲームオブジェクトを生成し、そのオブジェクトを先に配置したオブジェクトの中心に配置する。
その下にMainCameraを配置する。位置は3Dモデルから少し離れた位置に。

2.回転スクリプトをアタッチ

空のゲームオブジェクトに以下のスクリプトをアタッチする。

using UnityEngine;
using UnityEngine.UI;
using System.Collections;
public class Rotate : MonoBehaviour {
public float maxAngle = 60.0F; //垂直方向に回転できる上限角度
public float minAngle = 300.0F; //垂直方向に回転できる下限角度
public float rotateSpeed = 30.0f; //回転速度(角度/秒)
private GameObject camera; //メインカメラ
private Vector2 beforePoint; //1フレーム前のポイント(各指)
private Vector2 nowPoint; //現フレームのポイント(各指)
private Vector2 difference; //二本指操作チェック用差分
private float horizontalAngle;
private float varticalAngle;
void Start() {
camera = this.transform.Find("MainCamera").gameObject;
}
void Update() {
//1本指でタップした場合は回転
if (Input.touchCount == 1) {
//押下時のポイントを取得
if (Input.touchCount > 0) {
if (Input.GetTouch(0).phase == TouchPhase.Began) {
beforePoint = Input.GetTouch(0).position;
}
}
//スワイプでの継続した入力があった場合、その方向へ回転させる
if (Input.GetTouch(0).phase == TouchPhase.Moved) {
nowPoint = Input.GetTouch(0).position;
//水平方向の移動があった場合、水平方向に回転
if (nowPoint.x - beforePoint.x != 0) {
horizontalAngle = nowPoint.x - beforePoint.x;
horizontalAngle *= rotateSpeed * Time.deltaTime;
//水平方向に回転させる(水平方向はワールド軸)
this.transform.Rotate(0, horizontalAngle, 0, Space.World);
}
//垂直方向の回転があった場合、垂直方向に回転
if (nowPoint.y - beforePoint.y != 0) {
varticalAngle = nowPoint.y - beforePoint.y;
varticalAngle *= rotateSpeed * Time.deltaTime;
//現在の角度を丸め込み(小数点第一位)
float tmpAngles = Mathf.Round(this.transform.rotation.eulerAngles.x * 10);
tmpAngles /= 10;
//上方向に回転
if (tmpAngles <= maxAngle) {
//指定した最大角度を超える場合は最大角度まで回転するようにする
if (varticalAngle + this.transform.rotation.eulerAngles.x > maxAngle) {
varticalAngle = maxAngle - this.transform.rotation.eulerAngles.x;
}
//垂直方向に回転させる(垂直方向はローカル軸)
this.transform.Rotate(varticalAngle, 0, 0, Space.Self);
//指定角度を超えてしまった場合の処理
}
else if (tmpAngles > maxAngle && tmpAngles <= 180) {
//下方向にしか回転できないようにする
if (varticalAngle < 0) {
this.transform.Rotate(varticalAngle, 0, 0, Space.Self);
}
}
//下方向に回転
if (tmpAngles >= minAngle) {
//指定した最大角度を超える場合は最大角度まで回転するようにする
if (varticalAngle + this.transform.rotation.eulerAngles.x < minAngle) {
varticalAngle = minAngle - this.transform.rotation.eulerAngles.x;
}
this.transform.Rotate(varticalAngle, 0, 0, Space.Self);
//指定角度を超えてしまった場合の処理
}
else if (tmpAngles < minAngle && tmpAngles > 180) {
//上方向にしか回転できないようにする
if (varticalAngle > 0) {
this.transform.Rotate(varticalAngle, 0, 0, Space.Self);
}
}
}
//現フレームのポイントを格納
beforePoint = nowPoint;
}
}
}
}
view raw Rotate.cs hosted with ❤ by GitHub

メインカメラの親となる空のゲームオブジェクトを回転させる事で、モデルの周りをメインカメラが回転するという方法。

この処理などを合わせてビューワっぽく実装したのがこちら

2本指スワイプでカメラを並行移動する処理を実装する(スマホ向け)

スワイプでオブジェクトを中心に回転する処理を実装する(スマホ向け)の続き

2本指でスワイプした時にカメラを並行移動させる処理を実装する

スワイプでオブジェクトを中心に回転する処理を実装する(スマホ向け)と同じようにオブジェクトを配置した後、MainCameraの親オブジェクトに以下のスクリプトをアタッチする

using UnityEngine;
using UnityEngine.UI;
using System.Collections;
public class ParallelMove : MonoBehaviour {
public float moveSpeed = 0.05f; //スワイプ時移動速度
private GameObject camera; //メインカメラ
private Vector2[] beforePoint; //1フレーム前のポイント(各指)
private Vector2[] nowPoint; //現フレームのポイント(各指)
private Vector2[] diffPoint; //両フレームポイント差分
private float horizontalPosition;
private float varticalPosition;
void Start() {
camera = this.transform.Find("MainCamera").gameObject;
beforePoint = new Vector2[2];
nowPoint = new Vector2[2];
}
void Update() {
//2本目の指が押下された際
if (Input.GetTouch(1).phase == TouchPhase.Began) {
beforePoint[0] = Input.GetTouch(0).position;
beforePoint[1] = Input.GetTouch(1).position;
}
//2本目の指が移動した場合
if (Input.GetTouch(1).phase == TouchPhase.Moved) {
nowPoint[0] = Input.GetTouch(0).position;
nowPoint[1] = Input.GetTouch(1).position;
//水平方向の移動
if (nowPoint[0].x - beforePoint[0].x != 0) {
//水平方向の移動差分を取得
horizontalPosition = nowPoint[0].x - beforePoint[0].x;
horizontalPosition *= moveSpeed * Time.deltaTime;
//現在の角度に応じて移動方向を算出
Vector3 direction = Quaternion.Euler(0, camera.transform.localEulerAngles.y, 0) * new Vector3(horizontalPosition, 0, 0) * (-1);
direction = transform.TransformDirection(direction);
camera.transform.position = new Vector3(camera.transform.position.x + direction.x, camera.transform.position.y + direction.y, camera.transform.position.z + direction.z);
}
//垂直方向の移動
if (nowPoint[0].y - beforePoint[0].y != 0) {
//垂直方向の移動差分を取得
varticalPosition = (nowPoint[0].y - beforePoint[0].y) * (-1);
varticalPosition *= moveSpeed * Time.deltaTime;
//現在の角度に応じて移動方向を算出
Vector3 direction = Quaternion.Euler(0, camera.transform.localEulerAngles.y, 0) * new Vector3(0, varticalPosition, 0);
direction = transform.TransformDirection(direction);
camera.transform.position = new Vector3(camera.transform.position.x + direction.x, camera.transform.position.y + direction.y, camera.transform.position.z + direction.z);
}
//現フレームのポイントを格納
beforePoint[0] = nowPoint[0];
beforePoint[1] = nowPoint[1];
}
}
}

2本の指でタッチされている状態でスワイプした場合、カメラを移動させる事でモデルが平行移動しているように見せている

この処理などを合わせてビューワっぽく実装したのがこちら

被写界深度(DOF)使用時にタップしたオブジェクトにフォーカスを当てる

被写界深度(DOF)使用時に、タップしたオブジェクトに対してフォーカスが当たるようにする処理

まずはメインカメラにStandardAssetsのImageEffect内にある被写界深度用のスクリプトをアタッチする
00018

00019

その後、オブジェクトに以下のスクリプトをアタッチする

using UnityEngine;
using System.Collections;
using UnityStandardAssets.ImageEffects;
public class TapFocus : MonoBehaviour {
void Update () {
if (Input.GetMouseButton(0)) {
//レイを生成
Ray ray = Camera.main.ScreenPointToRay(Input.mousePosition);
RaycastHit hit = new RaycastHit();
if (Physics.Raycast(ray, out hit)) {
//カメラと対象オブジェクトの距離を測るため、それぞれの位置を取得
Vector3 position1 = Camera.main.transform.position;
Vector3 position2 = hit.point;
float distance = Vector3.Distance(position1, position2);
//メインカメラにアタッチした被写界深度スクリプトのフォーカス距離を変更する
Camera.main.GetComponent<DepthOfField>().focalLength = distance;
}
}
}
}

00020

フリック、スワイプを取得する

Unityにはタップは取得できるものの、フリック、スワイプが取得できないので実装

オブジェクトに以下のスクリプトをアタッチする

using UnityEngine;
using UnityEngine.UI;
using System.Collections;
public class TapAction : MonoBehaviour {
public float flickTime = 0.15f; //フリック判定用 時間しきい値
public float flickMagnitude = 100; //フリック判定用 移動距離
private Vector2 startPosition; //タップ開始ポイント
private Vector2 endPosition; //タップ終了ポイント
private float timer = 0.0f; //フリック判定用 タイマー
void Update() {
//タップ開始時
if (Input.GetTouch(0).phase == TouchPhase.Began) {
//タップ開始ポイントを取得
startPosition = Input.GetTouch(0).position;
}
//タップ終了時
if (Input.GetTouch(0).phase == TouchPhase.Ended) {
//タップ終了ポイントを取得
endPosition = Input.GetTouch(0).position;
//タップ開始~終了ポイントの距離
Vector2 direction = endPosition - startPosition;
//距離が指定以上、タップ時間が指定以下の場合、フリックと判定
if (direction.magnitude >= flickMagnitude && timer <= flickTime) {
//x軸の距離が大きい場合は左右へのフリック
if (Mathf.Abs(direction.x) >= Mathf.Abs(direction.y)) {
if (direction.x >= 0) {
//Right Flick
}
else {
//Left Flick
}
}
//y軸の距離が大きい場合は上下のフリック
else if (Mathf.Abs(direction.x) < Mathf.Abs(direction.y)) {
if (direction.y >= 0) {
//Up Flick
}
else {
//Down Flick
}
}
}
//タイマーを初期化
timer = 0.0f;
}
//タップ中
if (Input.GetTouch(0).phase == TouchPhase.Moved) {
//タップ時間がしきい値を越えた場合、スワイプと判定
if (timer >= flickTime) {
//Swipe
}
//押下している間、タイマーを加算
timer += Time.deltaTime;
}
}
}

タップしてからの経過時間と離すまでの距離から、フリックとスワイプを判定している。
そのため、一定時間(0.15秒)が経過するまでスワイプ処理が始められないのが欠点。

CSV形式のマスタデータ読み込み

別途作成されたCSV形式のマスタデータを読み込む処理を実装

1.マスタデータをResourcesフォルダ以下に格納する

2.マスタデータ読み込みスクリプトをアタッチ
■マスタデータ用クラス

public class masterData {
public int id;
public string name;
public int hp;
public int sp;
}

■マスタデータ読み込みスクリプト

using UnityEngine;
using System.Collections;
using System.Collections.Generic;
public class loadMasterData : MonoBehaviour {
//データ格納用List
public List<masterData> list = new List<masterData>();
void Awake() {
//resourcesフォルダ内にあるsampleTextファイルをロード
TextAsset textAsset = Resources.Load("masterData") as TextAsset;
//ロードした中身をstring型に変換
string text = textAsset.text;
//1行ずつに分割
string[] row = text.Split('\n');
for (int i = 0; i < row.Length; i++) {
//各データに分割
var column = row[i].Split(',');
var data = new masterData {
id = int.Parse(column[0]),
name = column[1],
hp = int.Parse(column[2]),
sp = int.Parse(column[3])
};
//1件ずつ格納
list.Add(data);
}
Debug.Log(list[0].id + " : " + list[0].name + " : " + list[0].hp + " : " + list[0].sp);
Debug.Log(list[1].id + " : " + list[1].name + " : " + list[1].hp + " : " + list[1].sp);
}
}