Mapping iPhoneInput.acceleration to object

Hello,

I am trying to map the iPhoneInput.acceleration to the rotation of a game object.

I would like that this gameObject maps the iphone space rotation.

The following code works with the screen facing upwards but no downwards…

transform.rotation.x=iPhoneInput.acceleration.x/1.45;
transform.rotation.z=iPhoneInput.acceleration.y/1.45;

Any ideas ? :wink:

I’m only rotating on Z but this may help you get started:

using UnityEngine;
using System.Collections;

public class RotateTransform : MonoBehaviour
{
	public float smooth = 5;
	float lastZ = float.PositiveInfinity;
	iPhoneOrientation lastKnownOrientation;

	void Update()
	{
		if (Application.platform == RuntimePlatform.IPhonePlayer)
		{
			float z = iPhoneInput.acceleration.x * -90;

			iPhoneOrientation orientation = iPhoneInput.orientation;
			if (orientation == iPhoneOrientation.Unknown)
				orientation = lastKnownOrientation;
			lastKnownOrientation = orientation;

			if (orientation == iPhoneOrientation.PortraitUpsideDown ||
				orientation == iPhoneOrientation.LandscapeLeft ||
				orientation == iPhoneOrientation.LandscapeRight)
				z = 180 - z;

			// smooth angle...
			if (smooth > 0  lastZ != float.PositiveInfinity)
				z = Mathf.Lerp(lastZ, z, smooth * Time.deltaTime);
			lastZ = z;

			transform.rotation = Quaternion.Euler(0, 0, z);
		}
		else
		{
			// keyboard control
			float z = Input.GetAxis("Horizontal");
			transform.Rotate(0, 0, z);
		}
	}
}

Basically depending on the orientation, I may translate say 75° to 105°… problem is, there are dead zones, for example between Portrait and LandscapeRight it’ll turn to Unknown, so I try to compensate that by remembering the last known orientation, but it’s not bullet-proof so sometimes you’ll see your object snap. I’m pretty sure it could be more clever so those snaps would occur even less, but when it got satisfactory I just stopped caring :wink: ! If I ever end up with a very clean version I’ll put it on the wiki…

And lastly, I smooth the angle a bit otherwise the object jitters (the accelerometer is “too precise”).
Oh yeah and you can ignore the keyboard part if you don’t need it (I have a webplayer version of my app so I just use left/right arrow to simulate tilting the phone).

Some debug code I use to have a clearer view of what’s going on:

	void OnGUI()
	{
		GUILayout.Label("orientation : " + iPhoneInput.orientation);
		GUILayout.Label("last orient : " + lastKnownOrientation);
		GUILayout.Label("acc : " + iPhoneInput.acceleration.x.ToString("0.00"));
		GUILayout.Label("rot : " + transform.eulerAngles.z.ToString("0.00"));
		float z = Mathf.Abs(transform.eulerAngles.z - 180);
		GUILayout.Label("abs(rot - 180) : " + z.ToString("0.00"));
	}

Merci Ben !!!

Your scripts have been usefull.

I used a different approach for rotating around only one axis

#pragma strict

private var  IphoneAcc : Vector3;
private var  IphoneAccFiltered : Vector3;



private var LowPassFilterFactor : float = .1; 
private var lowPassValue : Vector3 = Vector3.zero; // initialized with 1st sample

private var TempRot : Vector3;
/////	
function LowPassFilter(newSample : Vector3) {
		lowPassValue = Vector3.Lerp(lowPassValue, newSample, LowPassFilterFactor);
		return lowPassValue;
}
/////

function FixedUpdate () {

	IphoneAcc = iPhoneInput.acceleration;
	
	IphoneAccFiltered = LowPassFilter(IphoneAcc);
	
	if(IphoneAccFiltered.z<0) {
	TempRot.x=IphoneAccFiltered.x;
	//TempRot.z=IphoneAccFiltered.y;	
	} else {
	TempRot.x=1+(1-IphoneAccFiltered.x);
	//TempRot.z=2-IphoneAccFiltered.y;
						
		}

	transform.rotation=Quaternion.Euler(TempRot*90);

}

(the lowPass filter is used to smooth the accelerometer)

It works for one Axis ( x or z in my case). If you try with both axis, it does not work anymore…

I think that i have miss something in understanding the way iphone Z acceleration works…

So i will keep trying, by mixing your script and mine :wink:

Keep going with your fantastic games !

Merci encore,

It really all depends on the orientation of the phone, and the way an accelerometer works. If you’re standing up with the phone towards you, rotation around the Z-axis (you.transform.forward, ie the direction of your eyes) will be picked up very precisely. You’ll also have good X reading, but nothing on Y.
But if you sit down and lay the phone on a table, you won’t get any interesting input on Z at all! On the other hand, you’ll have excellent X and Y input.

It’s better to think of it as an airplane, I think X and Y are called pitch and yaw, and Z would be roll. For excellent this is the way the Wiimote plugin gives you input.

And you’re right to make your own filter, I forgot to mention it yesterday but I see you’ve read the docs.
I definitely want to try that too, since the orientation given by the OS is not very reliable…

Ahah excellent, j’avais pas remarqué que tu étais Français! Hobbyist? pro? C’est toujours intéressant de connaître les développeurs Unity “du coin”, on n’est qu’une poignée.
</French connection off>

thanks for explanations !

And thanks also for the Wiimote plugin link !
I am use to WiiFlash plugin but did not know that such a plugin exists for Unity ( It’s true that i had not search yet…)

C'est vrai qu'on est pas beaucoup. Mais avec la 2.5, la communautée devrait vite grandir ;-) En fait, je suis un vieux flasheur totalement converti à Unity. J'ai découvert récemment ce soft, donc je n'ai pas ton expérience mais je progresse tout les jours. J'étais même à unity2008 à Copenhague... Si tu le souhaites, je serais ravi qu'on fasse plus ample connaissance ;-)

If I use this approach, then in my balancing app, the user will be aware how to maintain balance. But I don’t want user to get the player balanced just by placing the iPhone in particular position.

Actually, I want to spin the gameObject with tilt. More the tilt more the spin to get stoped.

How to do that and What about this code :?:

transform.RotateAround (transform.position, Vector3.forward, iPhoneInput.acceleration.y);