最近想做格鬥遊戲,那麼就要有搖杆控件,不想去看別人的代碼就自己寫了個搖杆控件,實現起來很簡單。

話不多說,看代碼:

#ifndef __Joystick__
#define __Joystick__ #include "cocos2d.h"
USING_NS_CC; enum JoystickEnum
{
DEFAULT,
D_UP,
D_DOWN,
D_LEFT,
D_RIGHT,
D_LEFT_UP,
D_LEFT_DOWN,
D_RIGHT_UP,
D_RIGHT_DOWN
}; class Joystick : public Layer
{
public:
/** 啟動搖杆器 */
void onRun();
/** 清除數據 */
void onDisable();
/** 設置死亡半徑,即超出半徑將摇杆器失效 */
void setDieRadius(float radius);
/** 設置無效區域半徑(如果在無效區域內,將重置) */
void setFailRadius(float radius);
/** 是否顯示底盤和觸點 */
void setVisibleJoystick(bool visible);
/** 是否自由變換搖杆器的位置,即在屏幕上每一次按下鼠標時的座標將是搖杆器的座標,移動時將不改變搖杆器座標,直到下次按下鼠標 */
void setAutoPosition(bool value);
/** 回調函數指針 */
std::function<void(JoystickEnum)> onDirection;
/** 靜態創建函數(需要傳入底盤和觸點圖片路徑) */
static Joystick* create(std::string chassisPath,std::string dotPath);
/** 初始化搖杆器(需要傳入底盤和觸點圖片路徑) */
void initWithJoystick(std::string chassisPath,std::string dotPath);
protected:
/** 有效區域半徑 */
float _radius;
/** 失效區域半徑 */
float _failradius;
/** 是否移出有效區域 */
bool isMoveOut;
/** 是否存在有效區域半徑 */
bool isDieRadius;
/** 是否自由變換搖杆器座標 */
bool isAutoPosition;
/** 方向 */
JoystickEnum _direction;
/** 底盤 */
Sprite* _chassis;
/** 觸點 */
Sprite* _touchDot;
EventListenerTouchOneByOne* listener; bool onTouchBegan(Touch* touch,Event* event);
void onTouchMoved(Touch* touch,Event* event);
void onTouchEnded(Touch* touch,Event* event);
/**
1、設置觸點,并判斷是否在無效區域內(如果在無效區域內,將重置)
2、發送角度變化(如果不在無效區域內) */
void setTouchDotPosition(Vec2 vec1,Vec2 vec2);
/**
1、計算搖杆器八方向
2、發送角度變化,回調弧度變化函數 */
void changeAngle( Vec2 position );
/** 回調註冊的監聽函數 */
void callDirectionFun();
/** 重置(當弧度不是 DEFAULT時才重置) */
void resetState(); }; #endif
#include "Joystick.h"

Joystick* Joystick::create(std::string chassisPath,std::string dotPath)
{
auto joystick = new Joystick();
joystick->initWithJoystick(chassisPath,dotPath);
return joystick;
} void Joystick::initWithJoystick(std::string chassisPath,std::string dotPath)
{
_chassis = Sprite::create(chassisPath);
this->addChild(_chassis,);
_touchDot = Sprite::create(dotPath);
this->addChild(_touchDot,); isDieRadius = false;
isAutoPosition = false;
isMoveOut = false;
_direction = DEFAULT;
} void Joystick::onRun()
{
listener = EventListenerTouchOneByOne::create();
listener->setSwallowTouches(false);
listener->onTouchBegan = CC_CALLBACK_2(Joystick::onTouchBegan,this);
listener->onTouchMoved = CC_CALLBACK_2(Joystick::onTouchMoved,this);
listener->onTouchEnded = CC_CALLBACK_2(Joystick::onTouchEnded,this);
this->_eventDispatcher->addEventListenerWithSceneGraphPriority(listener,this);
} bool Joystick::onTouchBegan(Touch* touch,Event* event)
{
Vec2 locationInNode = this->convertToNodeSpace(touch->getLocation());
if( isAutoPosition )
{
this->setPosition(touch->getLocation());
return true;
}
if( isAutoPosition==false && isDieRadius )
{
if( locationInNode.getLength() > _radius )
{
return false;
}
}
_touchDot->setPosition(locationInNode);
if( locationInNode.getLength() > _failradius )
{
changeAngle(locationInNode);
}
return true;
}
void Joystick::onTouchMoved(Touch* touch,Event* event)
{
Vec2 locationInNode = this->convertToNodeSpace(touch->getLocation());
if( isDieRadius )
{
if( locationInNode.getLength() < _radius )
{
if( isMoveOut )
{
_touchDot->setPosition(locationInNode);
isMoveOut = false;
}
setTouchDotPosition(locationInNode,_touchDot->getPosition() + touch->getDelta());
return;
}
}
else
{
setTouchDotPosition(locationInNode,_touchDot->getPosition() + touch->getDelta());
return;
} isMoveOut = true;
_touchDot->setPosition(,);
resetState();
}
void Joystick::onTouchEnded(Touch* touch,Event* event)
{
_touchDot->setPosition(,);
isMoveOut = false;
resetState();
} void Joystick::setTouchDotPosition(Vec2 vec1,Vec2 vec2)
{
_touchDot->setPosition(vec2);
if( _failradius> )
{
if( vec1.getLength() < _failradius )
{
resetState();
return;
}
}
changeAngle(vec1);
} void Joystick::setDieRadius(float radius)
{
_radius = radius;
isDieRadius = true;
} void Joystick::setAutoPosition(bool value)
{
isAutoPosition = value;
} void Joystick::setFailRadius(float radius)
{
_failradius = radius;
} void Joystick::onDisable()
{
this->_eventDispatcher->removeEventListener(listener);
isDieRadius = false;
isAutoPosition = false;
isMoveOut = false;
} void Joystick::changeAngle( Vec2 position )
{
auto angle = CC_RADIANS_TO_DEGREES(position.getAngle());
if(angle > -22.5 && angle < 22.5)
{
_direction=D_RIGHT;
}
else if(angle > 22.5 && angle < 67.5)
{
_direction=D_RIGHT_UP;
}
else if(angle > 67.5 && angle < 112.5)
{
_direction=D_UP;
}
else if(angle > 112.5 && angle < 157.5)
{
_direction=D_LEFT_UP;
}
else if((angle > 157.5 && angle < )||(angle < -157.5 && angle > -))
{
_direction=D_LEFT;
}
else if(angle < -112.5 && angle > -157.5)
{
_direction=D_LEFT_DOWN;
}
else if(angle < -67.5 && angle > -112.5)
{
_direction=D_DOWN;
}
else if(angle < -22.5 && angle > -67.5)
{
_direction=D_RIGHT_DOWN;
}
callDirectionFun();
} void Joystick::callDirectionFun()
{
if( onDirection )
{
onDirection(_direction);
}
} void Joystick::resetState()
{
if(_direction != DEFAULT)
{
_direction = DEFAULT;
callDirectionFun();
}
} void Joystick::setVisibleJoystick(bool visible)
{
_chassis->setVisible(visible);
_touchDot->setVisible(visible);
}

當然,如果有用到的朋友可以自己修改。這隻是最簡單的實現。

下面有效果圖,不過加載比較慢:

普通模式

  auto joystick = Joystick::create("rocker.png","rocker_joy.png");//通過兩張圖片初始化控件
this->addChild(joystick);
joystick->setPosition(VisibleRect::leftBottom()+Vec2(,));//設置初始位置
  joystick->onDirection=CC_CALLBACK_1(SceneMain::onDirection,this);//角度變化更新(onDirection(JoystickEnum enums))
joystick->onRun();//啟動

存在死亡半徑模式:(超出死亡半徑將觸點重置初始位置,移動失效)

auto joystick = Joystick::create("rocker.png","rocker_joy.png");//通過兩張圖片初始化控件
this->addChild(joystick);
joystick->setPosition(VisibleRect::leftBottom()+Vec2(,));//設置初始位置
joystick->setDieRadius();//設置死亡半徑(外圈)
joystick->onDirection=CC_CALLBACK_1(SceneMain::onDirection,this);//角度變化更新(onDirection(JoystickEnum enums))
joystick->onRun();//啟動

設置失效半徑:(在失效半徑內將不會觸發角度改變事件)

auto joystick = Joystick::create("rocker.png","rocker_joy.png");//通過兩張圖片初始化控件
this->addChild(joystick);
joystick->setPosition(VisibleRect::leftBottom()+Vec2(,));//設置初始位置
joystick->setDieRadius();//設置死亡半徑(外圈)
joystick->setFailRadius();//設置失效半徑(內圈)
joystick->onDirection=CC_CALLBACK_1(SceneMain::onDirection,this);//角度變化更新(onDirection(JoystickEnum enums))
joystick->onRun();//啟動

設置自由變換位置:以鼠標按下的座標為初始位置

auto joystick = Joystick::create("rocker.png","rocker_joy.png");//通過兩張圖片初始化控件
this->addChild(joystick);
joystick->setPosition(VisibleRect::leftBottom()+Vec2(,));//設置初始位置
joystick->setAutoPosition(true);//是否自由改變座標
joystick->onDirection=CC_CALLBACK_1(SceneMain::onDirection,this);//角度變化更新
joystick->onRun();//啟動

設置隱藏搖杆:(不顯示搖杆底盤和觸點,一般會設置成自由改變位置)

auto joystick = Joystick::create("rocker.png","rocker_joy.png");//通過兩張圖片初始化控件
this->addChild(joystick);
joystick->setPosition(VisibleRect::leftBottom()+Vec2(,));//設置初始位置
joystick->setAutoPosition(true);//是否自由改變座標
joystick->setVisibleJoystick(false);//是否顯示
joystick->onDirection=CC_CALLBACK_1(SceneMain::onDirection,this);//角度變化更新
joystick->onRun();//啟動

Cocos2d-x 3.2 学习笔记(十)Joystick 搖杆控件的更多相关文章

  1. Android学习笔记(七)——常见控件

    //此系列博文是<第一行Android代码>的学习笔记,如有错漏,欢迎指正! Android 给我们提供了大量的 UI控件,下面我们简单试试几种常用的控件. 一.TextView 在布局文 ...

  2. Java学习笔记——可视化Swing中JTable控件绑定SQL数据源的两种方法

    在 MyEclipse 的可视化 Swing 中,有 JTable 控件. JTable 用来显示和编辑常规二维单元表. 那么,如何将 数据库SQL中的数据绑定至JTable中呢? 在这里,提供两种方 ...

  3. Unity3D学习笔记(一)GUI控件的调用

    GUI控件:1.在Start中初始化,在OnGUI中调整.2.公有变量才会出现在Inspector面板.3.GUI控件的初始化和处理在OnGUI内完成.4.JavaScript的中文为UTF-8编码可 ...

  4. C# 学习笔记 三层架构系列(控件一)

    下面是我两周的学习总结:这是我写给自己的,如果哪位朋友有幸看到这篇文章就是缘分.如果所说的内容不对,就请纠正.勿喷!!! 想要将两周的学习知识通过文字.通过代码.通过图片储备起来,以防自己那天思维短路 ...

  5. COCOS学习笔记--Button类及其相关控件属性

    一.Button介绍 Button就是button.Cocos中提供了Button类对button进行相关的操作.我们看一下Button类继承关系图: 能够看到.Button是继承自Widget类,W ...

  6. .net core 学习笔记(1)-分页控件的使用

    最近有个小项目,用.net core开发练练手,碰到的第一个问题就是分页控件的问题,自己写太费时间,上网查了下,发现有人已经封装好了的,就拿过来用了,分页控件github:https://github ...

  7. Android学习笔记50:使用WebView控件浏览网页

    在Android中,可以使用Webview控件来浏览网页.通过使用该控件,我们可以自制一个简单的浏览器,运行效果如图1所示. 图1 运行效果 1.WebView 在使用WebView控件时,首先需要在 ...

  8. 【jQuery UI 1.8 The User Interface Library for jQuery】.学习笔记.10.Button 和 Autocomplete控件

    Button ,可以使用 <button> <input> <a>. <input> 中的不同类型,submit , radio , checkbox ...

  9. [Android学习笔记]继承自ViewGroup的控件的过程学习

    ViewGroup文档 http://developer.android.com/training/index.html 继承自ViewGroup需要重写onLayout方法用来为子View设定位置信 ...

随机推荐

  1. JAVA设计模式--工厂方法模式

    工厂方法设计模式 抽象工厂角色: 这是工厂方法模式的核心,它与应用程序无关.是具体工厂角色必须实现的接口或者必须继承的父类.在java中它由抽象类或者接口来实现.具体工厂角色:它含有和具体业务逻辑有关 ...

  2. XAF 官方问题笔记

    1.How to print report direct, and print preview via an simple action? 如何通过一个按钮直接打印报表或者打印预览报表 https:/ ...

  3. NGUI 3.0.7的新锚点系统设置不好就会造成显示错误的错觉

    每次设置NGUI控件的锚点时,都需要刷新一下窗口,不然就会造成显示错误的错觉. 同时,NGUI控件设置锚点的参考对象为一个物体时还需要调整其控件大小,不然也会造成显示错误的错觉.

  4. phpcms v9 标签含义整理

    {template "content","header"}   ----------  调用根目录下phpcms\template\content\header ...

  5. 手机浏览器,微信中播放amr录音

    由于微信公众号开发中,临时素材只有三天的有效期,但是客户要求所有录音永久保存,永久素材数量又有限制,故只能把录音保存到服务器上.但是存到服务器上有一个问题,手机微信中无法直接播放amr录音.无意中发现 ...

  6. C#操作内存读写方法的主要实现代码

    C#操作内存读写方法是什么呢?让我们来看看具体的实例实现: using System.Runtime.InteropServices; using System.Text; publicclass F ...

  7. Blend 2015 教程 (三) 模板

    前一篇讲述了一些基本样式的修改方法,并搭建了Style层的基本框架,本篇将进一步修改ListBox的样式. 1. 首先选中ListBox控件,在美工板导航栏中点击ListBox,选择 编辑其他模板-编 ...

  8. 图解:Arcgis Server 安装

    必须保证IIS配置正常,否则arcserver安装不会成功. 选择安装路径,还是尽量不要在有括号的文件夹下. 设置服务名,最好使用默认的. 点击完成后会要求进行服务配置. 在windows serve ...

  9. STC12C5A60S2笔记6(中断)

    1. 基本特性 1) 中断源 STC12C5A60S2共有十个中断源,每个中断源可设置4类优先级:当相同优先级下各中断优先级由高到低依次如下: 1.1)INT0(外部中断0) 中断向量地址 0003H ...

  10. 本地DNS安装

    在centOS里最常用的DNS服务工具应该是bind了.下面就以bind为例做一个DNS服务. 首先查看bind 是否已经安装 Rpm -qa | gerp bind 如果没有的话就用yum 安装一下 ...