c++ - 如何在 C++ 中获取 JoyStick Z 旋转消息

标签 c++ windows joystick borland-c++

我的 C++ 代码可以正确拦截我的 Saitek Cyborg 操纵杆的操纵杆 x 和 y 位置。

这个操纵杆还可以绕 handle 轴旋转,我正在尝试获取此 Action 的消息。

对于 x 和 y,消息 ID 是 MM_JOY1MOVE 和 MM_JOY2MOVE,它们允许我捕获摇杆的 x 和 y 位置。

对于 z 轴,有 MM_JOY1ZMOVE 和 MM_JOY2ZMOVE,但它们都没有捕获扭转运动。

查看mmsystem.h,发现如下定义:

....
#define MM_JOY1MOVE         0x3A0           /* joystick */
#define MM_JOY2MOVE         0x3A1
#define MM_JOY1ZMOVE        0x3A2
#define MM_JOY2ZMOVE        0x3A3
#define MM_JOY1BUTTONDOWN   0x3B5
#define MM_JOY2BUTTONDOWN   0x3B6
#define MM_JOY1BUTTONUP     0x3B7
#define MM_JOY2BUTTONUP     0x3B8
...

但这些都不起作用。

问题是:z-twist 运动的 messageID 是什么,或者我怎样才能找到它?

最佳答案

我不确定你所说的 Z 扭曲是什么意思(如果它是模拟轴或只是 POV 按钮)但我用 JOYINFOEXJOYCAPS< 处理操纵杆 像这样:

//---------------------------------------------------------------------------
//--- Spektres Joystick class ver: 1.1 --------------------------------------
//---------------------------------------------------------------------------
#ifndef _joystick_h
#define _joystick_h
//---------------------------------------------------------------------------
//#include <dinput.h>
#include <Mmsystem.h>

//---------------------------------------------------------------------------
//---------------------------------------------------------------------------
//---------------------------------------------------------------------------
const int _joystick_max=16;
//---------------------------------------------------------------------------
class joystick
        {
public: int _enable_double;
        int _x,_y,_z,_r,_u,_v,_pov; // status non zero ?
        double x,y,z,r,u,v,pov;     // pos
        double x0,y0,z0,r0,u0,v0;   // min
        double dx,dy,dz,dr,du,dv;   // 2/(max-min)
        double deadzone;
        DWORD  buttons;

        JOYINFOEX joy_info;
        JOYCAPS joy_cp;
        int joy_list[_joystick_max],joy_num;
        int joy_id;
        joystick() { deadzone=0.1; _enable_double=1; joy_id=-1; joy_num=0; init_list(); }
        joystick(joystick& a) { *this=a; }
        ~joystick()    {}
        joystick* operator = (const joystick *a) { *this=*a; return this; }
        //joystick* operator = (const joystick &a) { ...copy... return this; }
        void init_list();
        void refresh();
        void refresh_cp();
        };
//---------------------------------------------------------------------------
void joystick::init_list()
        {
        int i,j;
        joy_num=0;
        j=JOYSTICKID1;
        for (i=0;i<_joystick_max;i++)
            {
            joy_info.dwSize = sizeof(JOYINFOEX);
            joy_info.dwFlags=JOY_RETURNALL;
            if (joyGetPosEx(i,&joy_info)==JOYERR_NOERROR)
                {
                joy_list[joy_num]=i;
                joy_num++;
                j++;
                }
            }
        j=0;
        for (i=0;i<joy_num;i++)
         if (joy_list[i]==joy_id)
          { j=1; break; }
        if (!j)
            {
            joy_id=-1;
            if (joy_num>0) joy_id=joy_list[0];
            }
        refresh();
        refresh_cp();
        }
//---------------------------------------------------------------------------
void joystick::refresh()
        {
        if (joy_id==-1) return;
        joy_info.dwSize = sizeof(JOYINFOEX);
        joy_info.dwFlags=JOY_RETURNALL;
        joyGetPosEx(joy_id,&joy_info);
        if (_enable_double)
            {
            x=double(joy_info.dwXpos-x0)*dx-1.0;
            y=double(joy_info.dwYpos-y0)*dy-1.0;
            z=double(joy_info.dwZpos-z0)*dz-1.0;
            r=double(joy_info.dwRpos-r0)*dr-1.0;
            u=double(joy_info.dwUpos-u0)*du-1.0;
            v=double(joy_info.dwVpos-v0)*dv-1.0;
            pov=double(joy_info.dwPOV)*0.01;
            _x=0; if (x<=-deadzone) { _x=1; x+=deadzone; } if (x>=+deadzone) { _x=1; x-=deadzone; }
            _y=0; if (y<=-deadzone) { _y=1; y+=deadzone; } if (y>=+deadzone) { _y=1; y-=deadzone; }
            _z=0; if (z<=-deadzone) { _z=1; z+=deadzone; } if (z>=+deadzone) { _z=1; z-=deadzone; }
            _r=0; if (r<=-deadzone) { _r=1; r+=deadzone; } if (r>=+deadzone) { _r=1; r-=deadzone; }
            _u=0; if (u<=-deadzone) { _u=1; u+=deadzone; } if (u>=+deadzone) { _u=1; u-=deadzone; }
            _v=0; if (v<=-deadzone) { _v=1; v+=deadzone; } if (v>=+deadzone) { _v=1; v-=deadzone; }
            _pov=1; if (joy_info.dwPOV==65535) _pov=0;
            buttons=joy_info.dwButtons;
            }
        }
//---------------------------------------------------------------------------
void joystick::refresh_cp()
        {
        if (joy_id==-1) return;
        joyGetDevCaps(joy_id,&joy_cp,sizeof(JOYCAPS));
        double q=2.0+deadzone+deadzone;
        x0=joy_cp.wXmin; dx=joy_cp.wXmax-x0; if (dx) dx=q/dx;
        y0=joy_cp.wYmin; dy=joy_cp.wYmax-y0; if (dy) dy=q/dy;
        z0=joy_cp.wZmin; dz=joy_cp.wZmax-z0; if (dz) dz=q/dz;
        r0=joy_cp.wRmin; dr=joy_cp.wRmax-r0; if (dr) dr=q/dr;
        u0=joy_cp.wUmin; du=joy_cp.wUmax-u0; if (du) du=q/du;
        v0=joy_cp.wVmin; dv=joy_cp.wVmax-v0; if (dv) dv=q/dv;
        }
//---------------------------------------------------------------------------
//---------------------------------------------------------------------------
//---------------------------------------------------------------------------
#endif
//---------------------------------------------------------------------------
//---------------------------------------------------------------------------
//---------------------------------------------------------------------------

用法:

// some (locally) global variable
joystick joy; 

// do this in some timer or whatever
joy.refresh();

// this just prints the joystick state on VCL Form->Canvas you can rewrite it to what you want:
int x,y,dy;
dy=20;
x=10;
y=10-dy;
Canvas->FillRect(TRect(x-20,y,x+200,y+15*dy));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("x: %i",joy.joy_info.dwXpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("y: %i",joy.joy_info.dwYpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("z: %i",joy.joy_info.dwZpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("r: %i",joy.joy_info.dwRpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("u: %i",joy.joy_info.dwUpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("v: %i",joy.joy_info.dwVpos));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("b: %i",joy.joy_info.dwButtons));
Canvas->TextOutA(x,y+=dy,AnsiString().sprintf("z: %i",joy.joy_info.dwPOV));

这可以处理 6 个模拟轴 + POV 按钮 + 所有你想要的按钮 ...

它是为 Win2000 编写的,但仍然可以在 Win7 x64 中使用我的 4 轴 + POV + 7 个按钮 Genius 操纵杆:)(有除去灰尘它已经多年没有把它拿在手上了)

附言。

这是非常旧的源代码,所以我需要添加这个 bds 2006 C hidden memory manager conflicts只是为了确定。

关于c++ - 如何在 C++ 中获取 JoyStick Z 旋转消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37058265/

相关文章:

winapi - C++ 将操纵杆方向输入发送到程序

c++ - 这种立即使用且未定义的宏有什么缺点?

c++ - 此素数测试函数的时间复杂度

c++ - 为什么按下的键的 dwControlKeyState 与常量不匹配?

c++ - Windows模拟的奇怪行为

c++ - SDL_Jostick 正在发送奇怪的整数

python - Pygame 不返回操纵杆轴移动而不显示

c++ - 从记事本进程内存中读取文本

c++ - 如何根据结构的字段之一对结构指针列表进行排序?

windows - 如何自动提交Plink "Access granted. Press Return to begin session"提示