using UnityEngine;

/**

 * File: MPJoystick.cs

 * Author: Chris Danielson of (monkeyprism.com)

 * 

// USED TO BE: Joystick.js taken from Penelope iPhone Tutorial

//

// Joystick creates a movable joystick (via GUITexture) that 

// handles touch input, taps, and phases. Dead zones can control

// where the joystick input gets picked up and can be normalized.

//

// Optionally, you can enable the touchPad property from the editor

// to treat this Joystick as a TouchPad. A TouchPad allows the finger

// to touch down at any point and it tracks the movement relatively 

// without moving the graphic

*/
[RequireComponent(typeof(GUITexture))]
public class MPJoystick : MonoBehaviour
{ class Boundary
{ public Vector2 min = Vector2.zero; public Vector2 max = Vector2.zero; }
private static MPJoystick[] joysticks; // A static collection of all joysticks private static bool enumeratedJoysticks = false; private static float tapTimeDelta = 0.3f; // Time allowed between taps
public bool touchPad; public Vector2 position = Vector2.zero; public Rect touchZone; public Vector2 deadZone = Vector2.zero; // Control when position is output public bool normalize = false; // Normalize output after the dead-zone? public int tapCount; private int lastFingerId = -; // Finger last used for this joystick private float tapTimeWindow; // How much time there is left for a tap to occur private Vector2 fingerDownPos; //private float fingerDownTime; //private float firstDeltaTime = 0.5f;
private GUITexture gui; private Rect defaultRect; // Default position / extents of the joystick graphic private Boundary guiBoundary = new Boundary(); // Boundary for joystick graphic private Vector2 guiTouchOffset; // Offset to apply to touch input [HideInInspector]
public Vector2 guiCenter; // Center of joystick
void Start()
{ gui = (GUITexture)GetComponent(typeof(GUITexture));
defaultRect = gui.pixelInset; defaultRect.x += transform.position.x * Screen.width;// + gui.pixelInset.x; // - Screen.width * 0.5; defaultRect.y += transform.position.y * Screen.height;// - Screen.height * 0.5;
transform.position = Vector3.zero; if (touchPad)
{ // If a texture has been assigned, then use the rect ferom the gui as our touchZone if (gui.texture) touchZone = defaultRect; }
else
{ guiTouchOffset.x = defaultRect.width * 0.5f; guiTouchOffset.y = defaultRect.height * 0.5f;
// Cache the center of the GUI, since it doesn't change guiCenter.x = defaultRect.x + guiTouchOffset.x; guiCenter.y = defaultRect.y + guiTouchOffset.y;
// Let's build the GUI boundary, so we can clamp joystick movement guiBoundary.min.x = defaultRect.x - guiTouchOffset.x; guiBoundary.max.x = defaultRect.x + guiTouchOffset.x; guiBoundary.min.y = defaultRect.y - guiTouchOffset.y; guiBoundary.max.y = defaultRect.y + guiTouchOffset.y; } }
public Vector2 getGUICenter()
{ return guiCenter; }
void Disable()
{ gameObject.active = false; //enumeratedJoysticks = false; }
private void ResetJoystick()
{ gui.pixelInset = defaultRect; lastFingerId = -; position = Vector2.zero; fingerDownPos = Vector2.zero; }
private bool IsFingerDown()
{ return (lastFingerId != -); }
public void LatchedFinger(int fingerId)
{ // If another joystick has latched this finger, then we must release it if (lastFingerId == fingerId) ResetJoystick(); }
void Update()
{ if (!enumeratedJoysticks)
{ // Collect all joysticks in the game, so we can relay finger latching messages joysticks = (MPJoystick[])FindObjectsOfType(typeof(MPJoystick)); enumeratedJoysticks = true; }
int count = Input.touchCount;
if (tapTimeWindow > )
tapTimeWindow -= Time.deltaTime;
else
tapCount = ; if (count == )
{
ResetJoystick();
}
else
{
for (int i = ; i < count; i++)
{
Touch touch = Input.GetTouch(i); Vector2 guiTouchPos = touch.position - guiTouchOffset;
bool shouldLatchFinger = false; if (touchPad)
{ if (touchZone.Contains(touch.position)) shouldLatchFinger = true; } else if (gui.HitTest(touch.position))
{ shouldLatchFinger = true; }
// Latch the finger if this is a new touch if (shouldLatchFinger && (lastFingerId == - || lastFingerId != touch.fingerId))
{
if (touchPad)
{ //gui.color.a = 0.15; lastFingerId = touch.fingerId; //fingerDownPos = touch.position; //fingerDownTime = Time.time; }
lastFingerId = touch.fingerId; // Accumulate taps if it is within the time window if (tapTimeWindow > ) tapCount++; else
{ tapCount = ; tapTimeWindow = tapTimeDelta; }
// Tell other joysticks we've latched this finger //for ( j : Joystick in joysticks ) foreach (MPJoystick j in joysticks)
{ if (j != this) j.LatchedFinger(touch.fingerId); } }
if (lastFingerId == touch.fingerId)
{ // Override the tap count with what the iPhone SDK reports if it is greater // This is a workaround, since the iPhone SDK does not currently track taps // for multiple touches if (touch.tapCount > tapCount) tapCount = touch.tapCount;
if (touchPad)
{ // For a touchpad, let's just set the position directly based on distance from initial touchdown position.x = Mathf.Clamp((touch.position.x - fingerDownPos.x) / (touchZone.width / ), -, ); position.y = Mathf.Clamp((touch.position.y - fingerDownPos.y) / (touchZone.height / ), -, ); }
else
{ // Change the location of the joystick graphic to match where the touch is Rect r = gui.pixelInset; r.x = Mathf.Clamp(guiTouchPos.x, guiBoundary.min.x, guiBoundary.max.x); r.y = Mathf.Clamp(guiTouchPos.y, guiBoundary.min.y, guiBoundary.max.y); gui.pixelInset = r; }
if (touch.phase == TouchPhase.Ended || touch.phase == TouchPhase.Canceled) ResetJoystick(); } } }
if (!touchPad)
{ // Get a value between -1 and 1 based on the joystick graphic location position.x = (gui.pixelInset.x + guiTouchOffset.x - guiCenter.x) / guiTouchOffset.x; position.y = (gui.pixelInset.y + guiTouchOffset.y - guiCenter.y) / guiTouchOffset.y; }
// Adjust for dead zone var absoluteX = Mathf.Abs(position.x); var absoluteY = Mathf.Abs(position.y); if (absoluteX < deadZone.x)
{ // Report the joystick as being at the center if it is within the dead zone position.x = ; } else if (normalize)
{ // Rescale the output after taking the dead zone into account position.x = Mathf.Sign(position.x) * (absoluteX - deadZone.x) / ( - deadZone.x); }
if (absoluteY < deadZone.y)
{ // Report the joystick as being at the center if it is within the dead zone position.y = ; } else if (normalize)
{ // Rescale the output after taking the dead zone into account position.y = Mathf.Sign(position.y) * (absoluteY - deadZone.y) / ( - deadZone.y); }
}
}

MPJoystick的更多相关文章

随机推荐

  1. kafka consumer 代码示例

    使用者小组 使得许多进程的多台机器 在逻辑上作为一个单个的使用者 出现. 我们使用中,一种常见的情况是,我们按照逻辑划分出多个使用者小组,每个小组都是有作为一个逻辑整体的多台使用者计算机组成的集群. ...

  2. resin + eclipse 遇到的问题

    1. 编译jsp报错: com.caucho.jsp.JspParseException: javac compiler is not available in Java(TM) SE Runtime ...

  3. 团队开发中git分支的使用

    1.Github上保持两个分支:master和develop. master是主分支,对项目进行tag或发布版本等操作,都必须在该分支上进行.最好设为不可提交只能合并的. develop是开发分支,从 ...

  4. 一分钟了解ruby中的单测

    之前用gtest写过很多c++的单测case, 对gtest的强大和灵活印象深刻:最近需要用ruby写一个小工具, 接触了下ruby, 写了代码就要写单测啊(好的单测确实对代码的健壮性和正确性保证上太 ...

  5. AC日记——[SHOI2008]小约翰的游戏John bzoj 1022

    1022 思路: nim: 代码: #include <cstdio> #include <cstdlib> #include <iostream> #includ ...

  6. (九)expect批量公钥推送

    (1)expect实现ssh非交互登录 注意:注释不能出现这脚本里面 spawn表示开启一个会话 \r:表示回车,exp_continue :表示没有出现这样,继续往下执行 interact :停留在 ...

  7. 【互动问答分享】第13期决胜云计算大数据时代Spark亚太研究院公益大讲堂

    “决胜云计算大数据时代” Spark亚太研究院100期公益大讲堂 [第13期互动问答分享] Q1:tachyon+spark框架现在有很多大公司在使用吧? Yahoo!已经在长期大规模使用: 国内也有 ...

  8. python 去掉所有空白字符【解决】

    今天用python从access数据库读取内容,组合成sql语句时,空白字符把我给搞疯了.... 所幸找到了一个好办法: ''.join(s.split())

  9. RabbitMQ (三) 工作队列之轮询分发

    上一篇讲了简单队列,实际工作中,这种队列应该很少用到,因为生产者发送消息的耗时一般都很短,但是消费者收到消息后,往往伴随着对高消息的业务逻辑处理,是个耗时的过程,这势必会导致大量的消息积压在一个消费者 ...

  10. 最小割分治(最小割树):BZOJ2229 && BZOJ4519

    定理:n个点的无向图的最小割最多n-1个. 可能从某种形式上形成了一棵树,不是很清楚. 最小割分治:先任选两个点求一边最小割,然后将两边分别递归,就能找到所有的最小割. 这两个题是一样的,直接搬din ...