2006-02-25 20:50:29 +00:00
/*
AppleWin : An Apple //e emulator for Windows
Copyright ( C ) 1994 - 1996 , Michael O ' Brien
Copyright ( C ) 1999 - 2001 , Oliver Schmidt
Copyright ( C ) 2002 - 2005 , Tom Charlesworth
2007-04-01 15:24:52 +00:00
Copyright ( C ) 2006 - 2007 , Tom Charlesworth , Michael Pohoreski
2006-02-25 20:50:29 +00:00
AppleWin is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
AppleWin is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with AppleWin ; if not , write to the Free Software
Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
/* Description: Joystick emulation via keyboard, PC joystick or mouse
*
* Author : Various
*/
// TC: Extended for 2nd joystick:
// Apple joystick #0 can be emulated with: NONE, JOYSTICKID1, KEYBOARD, MOUSE
// Apple joystick #1 can be emulated with: NONE, JOYSTICKID2, KEYBOARD, MOUSE
// If Apple joystick #0 is using {KEYBOARD | MOUSE} then joystick #1 can't use it.
// If Apple joystick #1 is using KEYBOARD, then disable the standard keys that control Apple switches #0/#1.
// - So that in a 2 player game, player 2 can't cheat by controlling player 1's buttons.
// If Apple joystick #1 is not NONE, then Apple joystick #0 only gets the use of Apple switch #0.
// - When using 2 joysticks, button #1 is used by joystick #1 (Archon).
// Apple joystick #1's button now controls Apple switches #1 and #2.
// - This is because the 2-joystick version of Mario Bros expects the 2nd joystick to control Apple switch #2.
# include "StdAfx.h"
2014-08-13 21:30:35 +01:00
2020-11-11 21:15:27 +00:00
# include "Joystick.h"
2014-08-13 21:30:35 +01:00
# include "CPU.h"
# include "Memory.h"
2015-12-05 16:50:27 +00:00
# include "YamlHelper.h"
2020-12-20 15:32:51 +00:00
# include "Interface.h"
2022-12-16 01:04:29 -08:00
# include "CopyProtectionDongles.h"
2014-08-13 21:30:35 +01:00
2017-06-07 10:14:25 +12:00
enum { DEVICE_NONE = 0 , DEVICE_JOYSTICK , DEVICE_KEYBOARD , DEVICE_MOUSE , DEVICE_JOYSTICK_THUMBSTICK2 } ;
2006-02-25 20:50:29 +00:00
2013-12-31 22:40:10 +00:00
// Indexed by joytype[n]
2017-06-07 10:14:25 +12:00
static const DWORD joyinfo [ 6 ] = { DEVICE_NONE ,
2013-12-31 22:40:10 +00:00
DEVICE_JOYSTICK ,
DEVICE_KEYBOARD , // Cursors (prev: Numpad-Standard)
DEVICE_KEYBOARD , // Numpad (prev: Numpad-Centering)
2017-06-07 10:14:25 +12:00
DEVICE_MOUSE ,
DEVICE_JOYSTICK_THUMBSTICK2 } ;
2006-02-25 20:50:29 +00:00
// Key pad [1..9]; Key pad 0,Key pad '.'; Left ALT,Right ALT
enum JOYKEY { JK_DOWNLEFT = 0 ,
JK_DOWN ,
JK_DOWNRIGHT ,
JK_LEFT ,
JK_CENTRE ,
JK_RIGHT ,
JK_UPLEFT ,
JK_UP ,
JK_UPRIGHT ,
JK_BUTTON0 ,
JK_BUTTON1 ,
JK_OPENAPPLE ,
JK_CLOSEDAPPLE ,
JK_MAX
} ;
const UINT PDL_MIN = 0 ;
const UINT PDL_CENTRAL = 127 ;
const UINT PDL_MAX = 255 ;
static BOOL keydown [ JK_MAX ] = { FALSE } ;
static POINT keyvalue [ 9 ] = { { PDL_MIN , PDL_MAX } , { PDL_CENTRAL , PDL_MAX } , { PDL_MAX , PDL_MAX } ,
{ PDL_MIN , PDL_CENTRAL } , { PDL_CENTRAL , PDL_CENTRAL } , { PDL_MAX , PDL_CENTRAL } ,
{ PDL_MIN , PDL_MIN } , { PDL_CENTRAL , PDL_MIN } , { PDL_MAX , PDL_MIN } } ;
static BOOL joybutton [ 3 ] = { 0 , 0 , 0 } ;
static int joyshrx [ 2 ] = { 8 , 8 } ;
static int joyshry [ 2 ] = { 8 , 8 } ;
static int joysubx [ 2 ] = { 0 , 0 } ;
static int joysuby [ 2 ] = { 0 , 0 } ;
2013-12-31 22:40:10 +00:00
// Value persisted to Registry for REGVALUE_JOYSTICK0_EMU_TYPE
2015-08-21 21:45:21 +01:00
static DWORD joytype [ 2 ] = { J0C_JOYSTICK1 , J1C_DISABLED } ; // Emulation Type for joysticks #0 & #1
2006-02-25 20:50:29 +00:00
static BOOL setbutton [ 3 ] = { 0 , 0 , 0 } ; // Used when a mouse button is pressed/released
2021-10-04 22:08:37 +01:00
static int xpos [ 2 ] = { PDL_MAX , PDL_MAX } ;
static int ypos [ 2 ] = { PDL_MAX , PDL_MAX } ;
2006-02-25 20:50:29 +00:00
2021-10-04 22:08:37 +01:00
static UINT64 g_paddleInactiveCycle [ 4 ] = { 0 } ; // Abs cycle that each paddle becomes inactive after PTRIG strobe
2006-02-25 20:50:29 +00:00
static short g_nPdlTrimX = 0 ;
static short g_nPdlTrimY = 0 ;
2014-02-16 14:39:26 +00:00
enum { JOYPORT_LEFTRIGHT = 0 , JOYPORT_UPDOWN = 1 } ;
static UINT g_bJoyportEnabled = 0 ; // Set to use Joyport to drive the 3 button inputs
static UINT g_uJoyportActiveStick = 0 ;
static UINT g_uJoyportReadMode = JOYPORT_LEFTRIGHT ;
2018-11-10 15:55:20 +00:00
static bool g_bHookAltKeys = true ;
2023-05-05 21:36:38 +01:00
static int JOYSTICK_1 = - 1 ;
static int JOYSTICK_2 = - 1 ;
2018-11-10 15:55:20 +00:00
//===========================================================================
void JoySetHookAltKeys ( bool hook )
{
g_bHookAltKeys = hook ;
}
2023-05-05 21:36:38 +01:00
int GetJoystick1 ( void )
{
return JOYSTICK_1 ;
}
int GetJoystick2 ( void )
{
return JOYSTICK_2 ;
}
2006-02-25 20:50:29 +00:00
//===========================================================================
2023-05-05 21:36:38 +01:00
static void CheckJoystick0 ( )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
if ( JOYSTICK_1 < 0 )
return ;
2006-02-25 20:50:29 +00:00
static DWORD lastcheck = 0 ;
DWORD currtime = GetTickCount ( ) ;
if ( ( currtime - lastcheck > = 10 ) | | joybutton [ 0 ] | | joybutton [ 1 ] )
{
lastcheck = currtime ;
JOYINFO info ;
2023-01-06 20:37:56 +01:00
if ( joyGetPos ( JOYSTICK_1 , & info ) = = JOYERR_NOERROR )
2006-02-25 20:50:29 +00:00
{
joybutton [ 0 ] = ( ( info . wButtons & JOY_BUTTON1 ) ! = 0 ) ;
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_NONE ) // Only consider 2nd button if NOT emulating a 2nd Apple joystick
2006-02-25 20:50:29 +00:00
joybutton [ 1 ] = ( ( info . wButtons & JOY_BUTTON2 ) ! = 0 ) ;
xpos [ 0 ] = ( info . wXpos - joysubx [ 0 ] ) > > joyshrx [ 0 ] ;
ypos [ 0 ] = ( info . wYpos - joysuby [ 0 ] ) > > joyshry [ 0 ] ;
// NB. This does not work for analogue joysticks (not self-centreing) - except if Trim=0
if ( xpos [ 0 ] = = 127 | | xpos [ 0 ] = = 128 ) xpos [ 0 ] + = g_nPdlTrimX ;
if ( ypos [ 0 ] = = 127 | | ypos [ 0 ] = = 128 ) ypos [ 0 ] + = g_nPdlTrimY ;
}
}
}
2023-05-05 21:36:38 +01:00
static void CheckJoystick1 ( )
2006-02-25 20:50:29 +00:00
{
static DWORD lastcheck = 0 ;
DWORD currtime = GetTickCount ( ) ;
if ( ( currtime - lastcheck > = 10 ) | | joybutton [ 2 ] )
{
lastcheck = currtime ;
JOYINFO info ;
2023-05-05 21:36:38 +01:00
MMRESULT result = JOYERR_NOERROR ;
2017-06-07 10:14:25 +12:00
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK_THUMBSTICK2 )
{
// Use results of joystick 1 thumbstick 2 and button 2 for joystick 1 and button 1
JOYINFOEX infoEx ;
infoEx . dwSize = sizeof ( infoEx ) ;
infoEx . dwFlags = JOY_RETURNBUTTONS | JOY_RETURNZ | JOY_RETURNR ;
2023-01-06 20:37:56 +01:00
result = joyGetPosEx ( JOYSTICK_1 , & infoEx ) ;
2017-06-07 10:14:25 +12:00
if ( result = = JOYERR_NOERROR )
{
info . wButtons = ( infoEx . dwButtons & JOY_BUTTON2 ) ? JOY_BUTTON1 : 0 ;
info . wXpos = infoEx . dwZpos ;
info . wYpos = infoEx . dwRpos ;
}
}
2023-05-05 21:36:38 +01:00
else
{
result = joyGetPos ( JOYSTICK_2 , & info ) ; // NB. joyGetPos(-1, &info) returns JOYERR_PARMS (bad parameters)
}
if ( result = = JOYERR_NOERROR )
2006-02-25 20:50:29 +00:00
{
joybutton [ 2 ] = ( ( info . wButtons & JOY_BUTTON1 ) ! = 0 ) ;
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 1 ] ] ! = DEVICE_NONE )
2006-02-25 20:50:29 +00:00
joybutton [ 1 ] = ( ( info . wButtons & JOY_BUTTON1 ) ! = 0 ) ; // Re-map this button when emulating a 2nd Apple joystick
xpos [ 1 ] = ( info . wXpos - joysubx [ 1 ] ) > > joyshrx [ 1 ] ;
ypos [ 1 ] = ( info . wYpos - joysuby [ 1 ] ) > > joyshry [ 1 ] ;
// NB. This does not work for analogue joysticks (not self-centreing) - except if Trim=0
if ( xpos [ 1 ] = = 127 | | xpos [ 1 ] = = 128 ) xpos [ 1 ] + = g_nPdlTrimX ;
if ( ypos [ 1 ] = = 127 | | ypos [ 1 ] = = 128 ) ypos [ 1 ] + = g_nPdlTrimY ;
}
}
}
//
// ----- ALL GLOBALLY ACCESSIBLE FUNCTIONS ARE BELOW THIS LINE -----
//
//===========================================================================
2008-05-17 23:20:33 +00:00
void JoyInitialize ( )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
//
2023-05-05 21:36:38 +01:00
// Detect First and Second connected JOYSTICK in WinMM API. JOYSTICKID1 == 0 but is not always the first connected joystick.
2023-01-06 20:37:56 +01:00
//
2006-02-25 20:50:29 +00:00
2023-01-06 20:37:56 +01:00
JOYSTICK_1 = - 1 ;
JOYSTICK_2 = - 1 ;
2006-02-25 20:50:29 +00:00
2023-01-06 20:37:56 +01:00
bool firstFound = false ;
2023-05-05 21:36:38 +01:00
const UINT numDevs = joyGetNumDevs ( ) ;
for ( UINT i = 0 ; i < numDevs ; i + + )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
JOYCAPS caps ;
int ret = joyGetDevCaps ( i , & caps , sizeof ( JOYCAPS ) ) ;
if ( ret ! = JOYERR_NOERROR )
continue ;
JOYINFO info ;
ret = joyGetPos ( i , & info ) ;
if ( ret ! = JOYERR_NOERROR )
continue ;
if ( firstFound )
{
JOYSTICK_2 = i ;
break ;
}
JOYSTICK_1 = i ;
firstFound = true ;
}
//
// Init for emulated joystick #0:
//
if ( joyinfo [ joytype [ 0 ] ] = = DEVICE_JOYSTICK )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
JOYCAPS caps ;
if ( JOYSTICK_1 > = 0 & & joyGetDevCaps ( JOYSTICK_1 , & caps , sizeof ( JOYCAPS ) ) = = JOYERR_NOERROR )
{
joyshrx [ 0 ] = 0 ;
joyshry [ 0 ] = 0 ;
joysubx [ 0 ] = ( int ) caps . wXmin ;
joysuby [ 0 ] = ( int ) caps . wYmin ;
UINT xrange = caps . wXmax - caps . wXmin ;
UINT yrange = caps . wYmax - caps . wYmin ;
while ( xrange > 256 )
{
xrange > > = 1 ;
+ + joyshrx [ 0 ] ;
}
while ( yrange > 256 )
{
yrange > > = 1 ;
+ + joyshry [ 0 ] ;
}
}
else
{
joytype [ 0 ] = J0C_KEYBD_NUMPAD ;
}
2006-02-25 20:50:29 +00:00
}
2023-01-06 20:37:56 +01:00
//
// Init for emulated joystick #1:
//
2006-02-25 20:50:29 +00:00
2023-05-05 21:36:38 +01:00
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
JOYCAPS caps ;
2023-05-05 21:36:38 +01:00
if ( JOYSTICK_2 > = 0 & & joyGetDevCaps ( JOYSTICK_2 , & caps , sizeof ( JOYCAPS ) ) = = JOYERR_NOERROR )
2023-01-06 20:37:56 +01:00
{
joyshrx [ 1 ] = 0 ;
joyshry [ 1 ] = 0 ;
joysubx [ 1 ] = ( int ) caps . wXmin ;
joysuby [ 1 ] = ( int ) caps . wYmin ;
UINT xrange = caps . wXmax - caps . wXmin ;
UINT yrange = caps . wYmax - caps . wYmin ;
while ( xrange > 256 )
{
xrange > > = 1 ;
+ + joyshrx [ 1 ] ;
}
while ( yrange > 256 )
{
yrange > > = 1 ;
+ + joyshry [ 1 ] ;
}
}
else
{
joytype [ 1 ] = J1C_DISABLED ;
}
}
2023-05-05 21:36:38 +01:00
else if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK_THUMBSTICK2 )
2006-02-25 20:50:29 +00:00
{
2023-01-06 20:37:56 +01:00
JOYCAPS caps ;
2023-05-05 21:36:38 +01:00
if ( JOYSTICK_1 > = 0 & & joyGetDevCaps ( JOYSTICK_1 , & caps , sizeof ( JOYCAPS ) ) = = JOYERR_NOERROR )
2023-01-06 20:37:56 +01:00
{
joyshrx [ 1 ] = 0 ;
joyshry [ 1 ] = 0 ;
joysubx [ 1 ] = ( int ) caps . wZmin ;
joysuby [ 1 ] = ( int ) caps . wRmin ;
UINT xrange = caps . wZmax - caps . wZmin ;
UINT yrange = caps . wRmax - caps . wRmin ;
while ( xrange > 256 )
{
xrange > > = 1 ;
+ + joyshrx [ 1 ] ;
}
while ( yrange > 256 )
{
yrange > > = 1 ;
+ + joyshry [ 1 ] ;
}
}
else
{
joytype [ 1 ] = J1C_DISABLED ;
}
2006-02-25 20:50:29 +00:00
}
}
//===========================================================================
2020-01-09 22:04:26 +00:00
static UINT g_buttonVirtKey [ 2 ] = { VK_MENU , VK_MENU | KF_EXTENDED } ; // VK_MENU == ALT Key
2020-01-07 21:59:06 +00:00
void JoySetButtonVirtualKey ( UINT button , UINT virtKey )
{
_ASSERT ( button < 2 ) ;
if ( button > = 2 ) return ;
g_buttonVirtKey [ button ] = virtKey ;
}
//===========================================================================
2013-12-06 21:10:41 +00:00
# define SUPPORT_CURSOR_KEYS
2013-05-20 20:51:45 +00:00
2018-11-10 15:55:20 +00:00
BOOL JoyProcessKey ( int virtkey , bool extended , bool down , bool autorep )
2006-02-25 20:50:29 +00:00
{
2013-05-20 20:51:45 +00:00
static struct
{
UINT32 Left : 1 ;
UINT32 Up : 1 ;
UINT32 Right : 1 ;
UINT32 Down : 1 ;
} CursorKeys = { 0 } ;
2020-01-09 22:04:26 +00:00
const UINT virtKeyWithExtended = ( ( UINT ) virtkey ) | ( extended ? KF_EXTENDED : 0 ) ;
2013-12-31 22:40:10 +00:00
if ( ( joyinfo [ joytype [ 0 ] ] ! = DEVICE_KEYBOARD ) & &
( joyinfo [ joytype [ 1 ] ] ! = DEVICE_KEYBOARD ) & &
2020-01-09 22:04:26 +00:00
( virtKeyWithExtended ! = g_buttonVirtKey [ 0 ] ) & &
( virtKeyWithExtended ! = g_buttonVirtKey [ 1 ] ) )
2013-05-20 20:51:45 +00:00
{
return 0 ;
}
2006-02-25 20:50:29 +00:00
2018-11-10 15:55:20 +00:00
if ( ! g_bHookAltKeys & & virtkey = = VK_MENU ) // GH#583
return 0 ;
2013-05-20 20:51:45 +00:00
//
2006-02-25 20:50:29 +00:00
2013-05-20 20:51:45 +00:00
BOOL keychange = 0 ;
2013-12-06 21:10:41 +00:00
bool bIsCursorKey = false ;
2006-02-25 20:50:29 +00:00
2021-04-10 19:09:36 +01:00
if ( virtKeyWithExtended = = g_buttonVirtKey [ 0 ] )
2020-01-07 21:59:06 +00:00
{
keychange = 1 ;
keydown [ JK_OPENAPPLE ] = down ;
}
2021-04-10 19:09:36 +01:00
else if ( virtKeyWithExtended = = g_buttonVirtKey [ 1 ] )
2006-02-25 20:50:29 +00:00
{
2013-05-20 20:51:45 +00:00
keychange = 1 ;
2020-01-07 21:59:06 +00:00
keydown [ JK_CLOSEDAPPLE ] = down ;
2006-02-25 20:50:29 +00:00
}
2013-05-20 20:51:45 +00:00
else if ( ! extended )
2006-02-25 20:50:29 +00:00
{
2013-12-31 22:40:10 +00:00
if ( JoyUsingKeyboardNumpad ( ) )
2013-05-20 20:51:45 +00:00
{
2013-12-31 22:40:10 +00:00
keychange = 1 ;
if ( ( virtkey > = VK_NUMPAD1 ) & & ( virtkey < = VK_NUMPAD9 ) ) // NumLock on
2013-05-20 20:51:45 +00:00
{
2013-12-31 22:40:10 +00:00
keydown [ virtkey - VK_NUMPAD1 ] = down ;
}
2021-04-10 19:09:36 +01:00
else // NumLock off (except for '0' and '.')
2013-12-31 22:40:10 +00:00
{
switch ( virtkey )
{
case VK_END : keydown [ JK_DOWNLEFT ] = down ; break ;
case VK_DOWN : keydown [ JK_DOWN ] = down ; break ;
case VK_NEXT : keydown [ JK_DOWNRIGHT ] = down ; break ;
case VK_LEFT : keydown [ JK_LEFT ] = down ; break ;
case VK_CLEAR : keydown [ JK_CENTRE ] = down ; break ;
case VK_RIGHT : keydown [ JK_RIGHT ] = down ; break ;
case VK_HOME : keydown [ JK_UPLEFT ] = down ; break ;
case VK_UP : keydown [ JK_UP ] = down ; break ;
case VK_PRIOR : keydown [ JK_UPRIGHT ] = down ; break ;
2021-04-10 19:09:36 +01:00
case VK_INSERT : // fall through... (NB. extended=0 for NumPad's Insert)
case VK_NUMPAD0 : keydown [ JK_BUTTON0 ] = down ; break ; // NumLock on
case VK_DELETE : // fall through... (NB. extended=0 for NumPad's Delete)
case VK_DECIMAL : keydown [ JK_BUTTON1 ] = down ; break ; // NumLock on
2013-12-31 22:40:10 +00:00
default : keychange = 0 ; break ;
}
2013-05-20 20:51:45 +00:00
}
}
2006-02-25 20:50:29 +00:00
}
2013-05-20 20:51:45 +00:00
# ifdef SUPPORT_CURSOR_KEYS
else if ( extended )
{
2013-12-31 22:40:10 +00:00
if ( JoyUsingKeyboardCursors ( ) & & ( virtkey = = VK_LEFT | | virtkey = = VK_UP | | virtkey = = VK_RIGHT | | virtkey = = VK_DOWN ) )
2013-05-20 20:51:45 +00:00
{
2013-12-31 22:40:10 +00:00
keychange = 1 ; // This prevents cursors keys being available to the Apple II (eg. Lode Runner uses cursor left/right for game speed & Ctrl-J/K for joystick/keyboard)
2013-12-06 21:10:41 +00:00
bIsCursorKey = true ;
2013-05-20 20:51:45 +00:00
switch ( virtkey )
{
case VK_LEFT : CursorKeys . Left = down ; break ;
case VK_UP : CursorKeys . Up = down ; break ;
case VK_RIGHT : CursorKeys . Right = down ; break ;
case VK_DOWN : CursorKeys . Down = down ; break ;
}
}
}
# endif
2006-02-25 20:50:29 +00:00
2013-05-20 20:51:45 +00:00
if ( ! keychange )
return 0 ;
2006-02-25 20:50:29 +00:00
2013-05-20 20:51:45 +00:00
//
2021-11-14 17:40:15 +00:00
if ( ( down & & ! autorep ) | | ( GetPropertySheet ( ) . GetJoystickCenteringControl ( ) = = JOYSTICK_MODE_CENTERING ) )
2006-02-25 20:50:29 +00:00
{
2013-05-20 20:51:45 +00:00
int xkeys = 0 ;
int ykeys = 0 ;
int xtotal = 0 ;
int ytotal = 0 ;
for ( int keynum = JK_DOWNLEFT ; keynum < = JK_UPRIGHT ; keynum + + )
2006-02-25 20:50:29 +00:00
{
2013-05-20 20:51:45 +00:00
if ( keydown [ keynum ] )
{
if ( ( keynum % 3 ) ! = 1 ) // Not middle col (ie. not VK_DOWN, VK_CLEAR, VK_UP)
{
xkeys + + ;
xtotal + = keyvalue [ keynum ] . x ;
}
if ( ( keynum / 3 ) ! = 1 ) // Not middle row (ie. not VK_LEFT, VK_CLEAR, VK_RIGHT)
{
ykeys + + ;
ytotal + = keyvalue [ keynum ] . y ;
}
}
}
2006-02-25 20:50:29 +00:00
2013-05-20 20:51:45 +00:00
if ( CursorKeys . Left )
{
xkeys + + ; xtotal + = keyvalue [ JK_LEFT ] . x ;
}
if ( CursorKeys . Right )
{
xkeys + + ; xtotal + = keyvalue [ JK_RIGHT ] . x ;
}
if ( CursorKeys . Up )
{
ykeys + + ; ytotal + = keyvalue [ JK_UP ] . y ;
}
if ( CursorKeys . Down )
{
ykeys + + ; ytotal + = keyvalue [ JK_DOWN ] . y ;
}
2013-12-31 22:40:10 +00:00
// Joystick # which is being emulated by keyboard
int nJoyNum = ( joyinfo [ joytype [ 0 ] ] = = DEVICE_KEYBOARD ) ? 0 : 1 ;
2013-05-20 20:51:45 +00:00
if ( xkeys )
xpos [ nJoyNum ] = xtotal / xkeys ;
else
xpos [ nJoyNum ] = PDL_CENTRAL + g_nPdlTrimX ;
if ( ykeys )
ypos [ nJoyNum ] = ytotal / ykeys ;
else
ypos [ nJoyNum ] = PDL_CENTRAL + g_nPdlTrimY ;
}
2006-02-25 20:50:29 +00:00
2020-12-20 15:32:51 +00:00
if ( bIsCursorKey & & GetPropertySheet ( ) . GetJoystickCursorControl ( ) )
2013-12-06 21:10:41 +00:00
{
// Allow AppleII keyboard to see this cursor keypress too
return 0 ;
}
2013-05-20 20:51:45 +00:00
return 1 ;
2006-02-25 20:50:29 +00:00
}
//===========================================================================
2013-12-06 21:10:41 +00:00
static void DoAutofire ( UINT uButton , BOOL & pressed )
{
2021-04-10 19:09:36 +01:00
static BOOL toggle [ 3 ] = { 0 , 0 , 0 } ;
static BOOL lastPressed [ 3 ] = { 0 , 0 , 0 } ;
2013-12-06 21:10:41 +00:00
BOOL nowPressed = pressed ;
2020-12-20 15:32:51 +00:00
if ( GetPropertySheet ( ) . GetAutofire ( uButton ) & & pressed )
2013-12-06 21:10:41 +00:00
{
2020-11-21 17:05:02 +00:00
toggle [ uButton ] = ( ! lastPressed [ uButton ] ) ? TRUE : ! toggle [ uButton ] ;
2013-12-06 21:10:41 +00:00
pressed = pressed & & toggle [ uButton ] ;
}
lastPressed [ uButton ] = nowPressed ;
}
2018-03-03 21:27:50 +00:00
BYTE __stdcall JoyportReadButton ( WORD address , ULONG nExecutedCycles )
2014-02-16 14:39:26 +00:00
{
BOOL pressed = 0 ;
if ( g_uJoyportActiveStick = = 0 )
{
switch ( address )
{
case 0x61 :
2021-11-14 17:40:15 +00:00
pressed = ( joybutton [ 0 ] | | setbutton [ 0 ] /*|| keydown[JK_OPENAPPLE]*/ ) ;
2014-02-16 14:39:26 +00:00
if ( joyinfo [ joytype [ 1 ] ] ! = DEVICE_KEYBOARD ) // BUG? joytype[1] should be [0] ?
pressed = ( pressed | | keydown [ JK_BUTTON0 ] ) ;
break ;
case 0x62 : // Left or Up
if ( g_uJoyportReadMode = = JOYPORT_LEFTRIGHT ) // LEFT
{
if ( xpos [ 0 ] = = 0 ) // TODO: More range for mouse control?
pressed = 1 ;
}
else // UP
{
if ( ypos [ 0 ] = = 0 ) // TODO: More range for mouse control?
pressed = 1 ;
}
break ;
case 0x63 : // Right or Down
if ( g_uJoyportReadMode = = JOYPORT_LEFTRIGHT ) // RIGHT
{
if ( xpos [ 0 ] > = 255 ) // TODO: More range for mouse control?
pressed = 1 ;
}
else // DOWN
{
if ( ypos [ 0 ] > = 255 ) // TODO: More range for mouse control?
pressed = 1 ;
}
break ;
}
}
else // TODO: stick #1
{
}
pressed = pressed ? 0 : 1 ; // Invert as Joyport signals are active low
2018-03-03 21:27:50 +00:00
return MemReadFloatingBus ( pressed , nExecutedCycles ) ;
2014-02-16 14:39:26 +00:00
}
2021-04-10 19:09:36 +01:00
static BOOL CheckButton0Pressed ( void )
{
2021-11-14 17:40:15 +00:00
BOOL pressed = joybutton [ 0 ] | |
2021-04-10 19:09:36 +01:00
setbutton [ 0 ] | |
keydown [ JK_OPENAPPLE ] ;
if ( joyinfo [ joytype [ 1 ] ] ! = DEVICE_KEYBOARD ) // NB. always joytype[1] regardless if button is 0 or 1
pressed = pressed | | keydown [ JK_BUTTON0 ] ;
return pressed ;
}
static BOOL CheckButton1Pressed ( void )
{
2021-11-14 17:40:15 +00:00
BOOL pressed = joybutton [ 1 ] | |
2021-04-10 19:09:36 +01:00
setbutton [ 1 ] | |
keydown [ JK_CLOSEDAPPLE ] ;
if ( joyinfo [ joytype [ 1 ] ] ! = DEVICE_KEYBOARD ) // NB. always joytype[1] regardless if button is 0 or 1
pressed = pressed | | keydown [ JK_BUTTON1 ] ;
return pressed ;
}
2018-03-03 21:27:50 +00:00
BYTE __stdcall JoyReadButton ( WORD pc , WORD address , BYTE , BYTE , ULONG nExecutedCycles )
2006-02-25 20:50:29 +00:00
{
2013-12-06 21:10:41 +00:00
address & = 0xFF ;
2006-02-25 20:50:29 +00:00
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 0 ] ] = = DEVICE_JOYSTICK )
2013-12-06 21:10:41 +00:00
CheckJoystick0 ( ) ;
2017-06-07 10:14:25 +12:00
if ( ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK ) | | ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK_THUMBSTICK2 ) )
2013-12-06 21:10:41 +00:00
CheckJoystick1 ( ) ;
2014-02-16 14:39:26 +00:00
if ( g_bJoyportEnabled )
{
// Some extra logic to stop the Joyport forcing a self-test at CTRL+RESET
if ( ( address ! = 0x62 ) | | ( address = = 0x62 & & pc ! = 0xC242 & & pc ! = 0xC2BE ) ) // Original //e ($C242), Enhanced //e ($C2BE)
2018-03-03 21:27:50 +00:00
return JoyportReadButton ( address , nExecutedCycles ) ;
2014-02-16 14:39:26 +00:00
}
2021-04-10 19:09:36 +01:00
const bool swapButtons0and1 = GetPropertySheet ( ) . GetButtonsSwapState ( ) ;
BOOL pressed = FALSE ;
2013-12-06 21:10:41 +00:00
switch ( address )
{
case 0x61 :
2021-04-10 19:09:36 +01:00
{
pressed = ! swapButtons0and1 ? CheckButton0Pressed ( ) : CheckButton1Pressed ( ) ;
const UINT button0 = ! swapButtons0and1 ? 0 : 1 ;
DoAutofire ( button0 , pressed ) ;
2023-07-18 05:02:55 +09:00
if ( CopyProtectionDonglePB0 ( ) > = 0 ) //If a copy protection dongle needs PB0, this overrides the joystick
2022-12-16 01:04:29 -08:00
pressed = CopyProtectionDonglePB0 ( ) ;
2021-04-10 19:09:36 +01:00
}
2013-12-06 21:10:41 +00:00
break ;
case 0x62 :
2021-04-10 19:09:36 +01:00
{
pressed = ! swapButtons0and1 ? CheckButton1Pressed ( ) : CheckButton0Pressed ( ) ;
const UINT button1 = ! swapButtons0and1 ? 1 : 0 ;
DoAutofire ( button1 , pressed ) ;
2022-12-16 01:04:29 -08:00
if ( CopyProtectionDonglePB1 ( ) > = 0 ) //If a copy protection dongle needs PB1, this overrides the joystick
pressed = CopyProtectionDonglePB1 ( ) ;
2021-04-10 19:09:36 +01:00
}
2013-12-06 21:10:41 +00:00
break ;
case 0x63 :
2022-12-16 01:04:29 -08:00
if ( CopyProtectionDonglePB2 ( ) > = 0 ) //If a copy protection dongle needs PB2, this overrides the joystick
pressed = CopyProtectionDonglePB2 ( ) ;
else if ( IS_APPLE2 & & ( joyinfo [ joytype [ 1 ] ] = = DEVICE_NONE ) )
2013-12-06 21:10:41 +00:00
{
// Apple II/II+ with no joystick has the "SHIFT key mod"
// See Sather's Understanding The Apple II p7-36
pressed = ! ( GetKeyState ( VK_SHIFT ) < 0 ) ;
}
else
{
2021-11-14 17:40:15 +00:00
pressed = ( joybutton [ 2 ] | | setbutton [ 2 ] ) ;
2013-12-06 21:10:41 +00:00
DoAutofire ( 2 , pressed ) ;
}
break ;
}
2018-03-03 21:27:50 +00:00
return MemReadFloatingBus ( pressed , nExecutedCycles ) ;
2006-02-25 20:50:29 +00:00
}
//===========================================================================
// PREAD: ; $FB1E
// AD 70 C0 : (4) LDA $C070
// A0 00 : (2) LDA #$00
// EA : (2) NOP
// EA : (2) NOP
// Lbl1: ; 11 cycles is the normal duration of the loop
// BD 64 C0 : (4) LDA $C064,X
// 10 04 : (2) BPL Lbl2 ; NB. 3 cycles if branch taken (not likely)
// C8 : (2) INY
2013-05-20 20:51:45 +00:00
// D0 F8 : (3) BNE Lbl1 ; NB. 2 cycles if branch not taken (not likely)
2006-02-25 20:50:29 +00:00
// 88 : (2) DEY
// Lbl2:
// 60 : (6) RTS
//
static const double PDL_CNTR_INTERVAL = 2816.0 / 255.0 ; // 11.04 (From KEGS)
2018-03-03 21:27:50 +00:00
BYTE __stdcall JoyReadPosition ( WORD programcounter , WORD address , BYTE , BYTE , ULONG nExecutedCycles )
2006-02-25 20:50:29 +00:00
{
2018-03-03 21:27:50 +00:00
CpuCalcCycles ( nExecutedCycles ) ;
2006-02-25 20:50:29 +00:00
2021-10-04 22:08:37 +01:00
BOOL nPdlCntrActive = g_nCumulativeCycles < = g_paddleInactiveCycle [ address & 3 ] ;
2006-02-25 20:50:29 +00:00
2023-07-18 05:02:55 +09:00
// If no joystick connected, then this is always active (GH#778) && no copy-protection dongle connected
2021-10-04 22:08:37 +01:00
const UINT joyNum = ( address & 2 ) ? 1 : 0 ; // $C064..$C067
2023-07-18 05:02:55 +09:00
if ( joyinfo [ joytype [ joyNum ] ] = = DEVICE_NONE & & CopyProtectionDonglePDL ( address & 3 ) < 0 )
2020-04-26 16:19:50 +01:00
nPdlCntrActive = TRUE ;
2018-03-03 21:27:50 +00:00
return MemReadFloatingBus ( nPdlCntrActive , nExecutedCycles ) ;
2006-02-25 20:50:29 +00:00
}
//===========================================================================
2008-05-17 23:20:33 +00:00
void JoyReset ( )
2006-02-25 20:50:29 +00:00
{
int loop = 0 ;
while ( loop < JK_MAX )
keydown [ loop + + ] = FALSE ;
}
//===========================================================================
2023-07-18 05:02:55 +09:00
static void SetPaddleInactiveCycle ( UINT pdl , UINT pdlPos )
{
g_paddleInactiveCycle [ pdl ] = g_nCumulativeCycles + ( UINT64 ) ( ( double ) pdlPos * PDL_CNTR_INTERVAL ) ;
}
2019-04-06 18:23:42 +01:00
void JoyResetPosition ( ULONG nExecutedCycles )
2006-02-25 20:50:29 +00:00
{
2018-03-03 21:27:50 +00:00
CpuCalcCycles ( nExecutedCycles ) ;
2006-02-25 20:50:29 +00:00
2023-06-20 04:29:34 +09:00
bool isJoystick [ 2 ] = { false , false } ;
if ( joyinfo [ joytype [ 0 ] ] = = DEVICE_JOYSTICK )
{
2006-02-25 20:50:29 +00:00
CheckJoystick0 ( ) ;
2023-06-20 04:29:34 +09:00
isJoystick [ 0 ] = true ;
}
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK | | joyinfo [ joytype [ 1 ] ] = = DEVICE_JOYSTICK_THUMBSTICK2 )
{
2006-02-25 20:50:29 +00:00
CheckJoystick1 ( ) ;
2023-06-20 04:29:34 +09:00
isJoystick [ 1 ] = true ;
}
2021-10-04 22:08:37 +01:00
// If any of the timers are still running then strobe has no effect (GH#985)
for ( UINT pdl = 0 ; pdl < 4 ; pdl + + )
{
if ( g_nCumulativeCycles < = g_paddleInactiveCycle [ pdl ] )
continue ;
const UINT joyNum = ( pdl & 2 ) ? 1 : 0 ;
UINT pdlPos = ( pdl & 1 ) ? ypos [ joyNum ] : xpos [ joyNum ] ;
2023-06-20 04:29:34 +09:00
// "Square the circle" for controllers with analog sticks (but compatible with digital D-pads too) - GH#429
if ( isJoystick [ joyNum ] )
{
// Convert to unit circle, centred at (0,0)
2023-06-20 20:16:58 +01:00
const double scalar = 0.5 * 255.0 ;
const double offset = 1.0 ;
const double x = ( ( double ) xpos [ joyNum ] ) / scalar - offset ;
const double y = ( ( double ) ypos [ joyNum ] ) / scalar - offset ;
double axis = ! ( pdl & 1 ) ? x : y ;
2023-06-20 04:29:34 +09:00
2023-06-20 20:16:58 +01:00
if ( x * y ! = 0.0 )
2023-06-20 04:29:34 +09:00
{
// rescale the circle to the square
2023-06-20 20:16:58 +01:00
const double ratio2 = ( y * y ) / ( x * x ) ;
2023-06-20 04:29:34 +09:00
const double c = min ( ratio2 , 1.0 / ratio2 ) ;
const double coeff = sqrt ( 1.0 + c ) ;
axis * = coeff ;
}
if ( axis < - 1.0 ) axis = - 1.0 ;
else if ( axis > 1.0 ) axis = 1.0 ;
2023-06-20 20:16:58 +01:00
pdlPos = static_cast < int > ( ( axis + offset ) * scalar ) ;
2023-06-20 04:29:34 +09:00
}
2022-09-19 14:44:13 +01:00
// This is from KEGS. It helps games like Championship Lode Runner, Boulderdash & Learning with Leeper(GH#1128)
2021-10-04 22:08:37 +01:00
if ( pdlPos > = 255 )
2022-09-19 14:44:13 +01:00
pdlPos = 287 ;
2021-10-04 22:08:37 +01:00
2023-07-18 05:02:55 +09:00
SetPaddleInactiveCycle ( pdl , pdlPos ) ;
}
// Protection dongle overrides the PDL timer
// . eg. needed when Robocom PDL3 is still timing-out from the extended-255 count (eg. 287)
// . really if it were at 255 (ie. not connected), then on enabling the dongle it switches to (eg) 23 it should timeout immediately
for ( UINT pdl = 0 ; pdl < 4 ; pdl + + )
{
int pdlPosDongle = CopyProtectionDonglePDL ( pdl ) ;
if ( pdlPosDongle > = 0 )
SetPaddleInactiveCycle ( pdl , pdlPosDongle ) ;
2021-10-04 22:08:37 +01:00
}
2006-02-25 20:50:29 +00:00
}
//===========================================================================
// Called when mouse is being used as a joystick && mouse button changes
2008-05-17 23:20:33 +00:00
void JoySetButton ( eBUTTON number , eBUTTONSTATE down )
2006-02-25 20:50:29 +00:00
{
if ( number > 1 ) // Sanity check on mouse button #
return ;
// If 2nd joystick is enabled, then both joysticks only have 1 button
2013-12-31 22:40:10 +00:00
if ( ( joyinfo [ joytype [ 1 ] ] ! = DEVICE_NONE ) & & ( number ! = 0 ) )
2006-02-25 20:50:29 +00:00
return ;
// If it is 2nd joystick that is being emulated with mouse, then re-map button #
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_MOUSE )
2006-02-25 20:50:29 +00:00
{
2007-08-06 21:38:35 +00:00
number = BUTTON1 ; // 2nd joystick controls Apple button #1
2006-02-25 20:50:29 +00:00
}
setbutton [ number ] = down ;
}
//===========================================================================
2012-08-12 19:10:47 +00:00
BOOL JoySetEmulationType ( HWND window , DWORD newtype , int nJoystickNumber , const bool bMousecardActive )
2006-02-25 20:50:29 +00:00
{
if ( joytype [ nJoystickNumber ] = = newtype )
return 1 ; // Already set to this type. Return OK.
2017-06-07 10:14:25 +12:00
if ( joyinfo [ newtype ] = = DEVICE_JOYSTICK | | joyinfo [ newtype ] = = DEVICE_JOYSTICK_THUMBSTICK2 )
2006-02-25 20:50:29 +00:00
{
JOYCAPS caps ;
2023-05-05 21:36:38 +01:00
int nJoy2ID = joyinfo [ newtype ] = = DEVICE_JOYSTICK_THUMBSTICK2 ? JOYSTICK_1 : JOYSTICK_2 ;
int nJoyID = nJoystickNumber = = JN_JOYSTICK0 ? JOYSTICK_1 : nJoy2ID ;
2023-01-06 20:37:56 +01:00
if ( nJoyID < 0 | | joyGetDevCaps ( nJoyID , & caps , sizeof ( JOYCAPS ) ) ! = JOYERR_NOERROR )
2017-06-07 10:14:25 +12:00
{
2006-02-25 20:50:29 +00:00
MessageBox ( window ,
TEXT ( " The emulator is unable to read your PC joystick. " )
TEXT ( " Ensure that your game port is configured properly, " )
TEXT ( " that the joystick is firmly plugged in, and that " )
TEXT ( " you have a joystick driver installed. " ) ,
2017-06-07 10:14:25 +12:00
TEXT ( " Configuration " ) ,
MB_ICONEXCLAMATION | MB_SETFOREGROUND ) ;
return 0 ;
}
if ( ( joyinfo [ newtype ] = = DEVICE_JOYSTICK_THUMBSTICK2 ) & & ( caps . wNumAxes < 4 ) )
{
MessageBox ( window ,
TEXT ( " The emulator is unable to read thumbstick 2. " )
TEXT ( " Ensure that your game port is configured properly, " )
TEXT ( " that the joystick is firmly plugged in, and that " )
TEXT ( " you have a joystick driver installed. " ) ,
2006-02-25 20:50:29 +00:00
TEXT ( " Configuration " ) ,
MB_ICONEXCLAMATION | MB_SETFOREGROUND ) ;
return 0 ;
}
}
2013-12-31 22:40:10 +00:00
else if ( ( joyinfo [ newtype ] = = DEVICE_MOUSE ) & &
( joyinfo [ joytype [ nJoystickNumber ] ] ! = DEVICE_MOUSE ) )
2006-02-25 20:50:29 +00:00
{
2012-08-12 19:10:47 +00:00
if ( bMousecardActive )
2007-08-06 21:38:35 +00:00
{
2012-08-12 19:10:47 +00:00
// Shouldn't be necessary, since Property Sheet's logic should prevent this option being given to the user.
2007-08-06 21:38:35 +00:00
MessageBox ( window ,
TEXT ( " Mouse interface card is enabled - unable to use mouse for joystick emulation. " ) ,
TEXT ( " Configuration " ) ,
MB_ICONEXCLAMATION | MB_SETFOREGROUND ) ;
return 0 ;
}
2006-02-25 20:50:29 +00:00
MessageBox ( window ,
TEXT ( " To begin emulating a joystick with your mouse, move " )
TEXT ( " the mouse cursor over the emulated screen of a running " )
TEXT ( " program and click the left mouse button. During the " )
TEXT ( " time the mouse is emulating a joystick, you will not " )
TEXT ( " be able to use it to perform mouse functions, and the " )
TEXT ( " mouse cursor will not be visible. To end joystick " )
TEXT ( " emulation and regain the mouse cursor, click the left " )
TEXT ( " mouse button while pressing Ctrl. " ) ,
TEXT ( " Configuration " ) ,
MB_ICONINFORMATION | MB_SETFOREGROUND ) ;
}
2013-12-31 22:40:10 +00:00
else if ( joyinfo [ newtype ] = = DEVICE_KEYBOARD )
{
if ( newtype = = J0C_KEYBD_CURSORS | | newtype = = J1C_KEYBD_CURSORS )
{
MessageBox ( window ,
TEXT ( " Using cursor keys to emulate a joystick can cause conflicts. \n \n " )
2014-03-02 17:19:47 +00:00
TEXT ( " Be aware that 'cursor-up' = CTRL+K, and 'cursor-down' = CTRL+J. \n " )
2013-12-31 22:40:10 +00:00
TEXT ( " EG. Lode Runner uses CTRL+K/J to switch between keyboard/joystick modes " )
2014-03-02 17:19:47 +00:00
TEXT ( " (and cursor-left/right to control speed). \n \n " )
2013-12-31 22:40:10 +00:00
TEXT ( " Also if cursor keys are blocked from being read from the Apple keyboard " )
TEXT ( " then even simple AppleSoft command-line editing (cursor left/right) will not work. " ) ,
TEXT ( " Configuration " ) ,
MB_ICONINFORMATION | MB_SETFOREGROUND ) ;
}
}
2006-02-25 20:50:29 +00:00
joytype [ nJoystickNumber ] = newtype ;
JoyInitialize ( ) ;
JoyReset ( ) ;
return 1 ;
}
//===========================================================================
// Called when mouse is being used as a joystick && mouse position changes
2008-05-17 23:20:33 +00:00
void JoySetPosition ( int xvalue , int xrange , int yvalue , int yrange )
2006-02-25 20:50:29 +00:00
{
2013-12-31 22:40:10 +00:00
int nJoyNum = ( joyinfo [ joytype [ 0 ] ] = = DEVICE_MOUSE ) ? 0 : 1 ;
2006-02-25 20:50:29 +00:00
xpos [ nJoyNum ] = ( xvalue * 255 ) / xrange ;
ypos [ nJoyNum ] = ( yvalue * 255 ) / yrange ;
}
//===========================================================================
2014-09-14 21:23:54 +01:00
2008-05-17 23:20:33 +00:00
BOOL JoyUsingMouse ( )
2006-02-25 20:50:29 +00:00
{
2013-12-31 22:40:10 +00:00
return ( joyinfo [ joytype [ 0 ] ] = = DEVICE_MOUSE ) | | ( joyinfo [ joytype [ 1 ] ] = = DEVICE_MOUSE ) ;
2006-02-25 20:50:29 +00:00
}
2013-12-06 21:10:41 +00:00
BOOL JoyUsingKeyboard ( )
{
2013-12-31 22:40:10 +00:00
return ( joyinfo [ joytype [ 0 ] ] = = DEVICE_KEYBOARD ) | | ( joyinfo [ joytype [ 1 ] ] = = DEVICE_KEYBOARD ) ;
2013-12-06 21:10:41 +00:00
}
2013-12-31 22:40:10 +00:00
BOOL JoyUsingKeyboardCursors ( )
{
return ( joytype [ 0 ] = = J0C_KEYBD_CURSORS ) | | ( joytype [ 1 ] = = J1C_KEYBD_CURSORS ) ;
}
BOOL JoyUsingKeyboardNumpad ( )
{
return ( joytype [ 0 ] = = J0C_KEYBD_NUMPAD ) | | ( joytype [ 1 ] = = J1C_KEYBD_NUMPAD ) ;
}
2006-02-25 20:50:29 +00:00
//===========================================================================
2008-05-17 23:20:33 +00:00
void JoyDisableUsingMouse ( )
{
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 0 ] ] = = DEVICE_MOUSE )
joytype [ 0 ] = J0C_DISABLED ;
2008-05-17 23:20:33 +00:00
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_MOUSE )
joytype [ 1 ] = J1C_DISABLED ;
2008-05-17 23:20:33 +00:00
}
//===========================================================================
2015-08-21 21:45:21 +01:00
void JoySetJoyType ( UINT num , DWORD type )
{
_ASSERT ( num < = JN_JOYSTICK1 ) ;
if ( num > JN_JOYSTICK1 )
return ;
2017-07-03 21:48:21 +01:00
if ( num = = JN_JOYSTICK0 ) // GH#434
{
_ASSERT ( type < J0C_MAX ) ;
if ( type > = J0C_MAX )
return ;
}
else
{
_ASSERT ( type < J1C_MAX ) ;
if ( type > = J1C_MAX )
return ;
}
2015-08-21 21:45:21 +01:00
joytype [ num ] = type ;
// Refresh centre positions whenever 'joytype' changes
JoySetTrim ( JoyGetTrim ( true ) , true ) ;
JoySetTrim ( JoyGetTrim ( false ) , false ) ;
}
DWORD JoyGetJoyType ( UINT num )
{
_ASSERT ( num < = JN_JOYSTICK1 ) ;
if ( num > JN_JOYSTICK1 )
return J0C_DISABLED ;
return joytype [ num ] ;
}
//===========================================================================
2006-02-25 20:50:29 +00:00
void JoySetTrim ( short nValue , bool bAxisX )
{
if ( bAxisX )
g_nPdlTrimX = nValue ;
else
g_nPdlTrimY = nValue ;
int nJoyNum = - 1 ;
2013-12-31 22:40:10 +00:00
if ( joyinfo [ joytype [ 0 ] ] = = DEVICE_KEYBOARD )
2006-02-25 20:50:29 +00:00
nJoyNum = 0 ;
2013-12-31 22:40:10 +00:00
else if ( joyinfo [ joytype [ 1 ] ] = = DEVICE_KEYBOARD )
2006-02-25 20:50:29 +00:00
nJoyNum = 1 ;
if ( nJoyNum > = 0 )
{
xpos [ nJoyNum ] = PDL_CENTRAL + g_nPdlTrimX ;
ypos [ nJoyNum ] = PDL_CENTRAL + g_nPdlTrimY ;
}
}
short JoyGetTrim ( bool bAxisX )
{
return bAxisX ? g_nPdlTrimX : g_nPdlTrimY ;
}
//===========================================================================
2014-02-16 14:39:26 +00:00
// Joyport truth-table:
//
// AN0 AN1 /PB0 /PB1 /PB2
// ------------------------------------
// 0 0 Trigger-1 Left-1 Right-1
// 0 1 Trigger-1 Up-1 Down-1
// 1 0 Trigger-2 Left-2 Right-2
// 1 1 Trigger-2 Up-2 Down-2
#if 0
void JoyportEnable ( const bool bEnable )
{
2021-05-23 21:28:26 +01:00
if ( IS_APPLE2C ( ) )
2014-02-16 14:39:26 +00:00
g_bJoyportEnabled = false ;
else
g_bJoyportEnabled = bEnable ? 1 : 0 ;
}
# endif
void JoyportControl ( const UINT uControl )
{
if ( ! g_bJoyportEnabled )
return ;
switch ( uControl )
{
case 0 : // AN0 clr
g_uJoyportActiveStick = 0 ;
break ;
case 1 : // AN0 set
g_uJoyportActiveStick = 1 ;
break ;
case 2 : // AN1 clr
g_uJoyportReadMode = JOYPORT_LEFTRIGHT ;
break ;
case 3 : // AN1 set
g_uJoyportReadMode = JOYPORT_UPDOWN ;
break ;
}
}
//===========================================================================
2015-12-05 16:50:27 +00:00
# define SS_YAML_KEY_COUNTERRESETCYCLE "Counter Reset Cycle"
# define SS_YAML_KEY_JOY0TRIMX "Joystick0 TrimX"
# define SS_YAML_KEY_JOY0TRIMY "Joystick0 TrimY"
# define SS_YAML_KEY_JOY1TRIMX "Joystick1 TrimX"
# define SS_YAML_KEY_JOY1TRIMY "Joystick1 TrimY"
2021-10-04 22:08:37 +01:00
# define SS_YAML_KEY_PDL_INACTIVE_CYCLE "Paddle%1d Inactive Cycle"
2015-12-05 16:50:27 +00:00
2022-03-01 07:52:18 +11:00
static const std : : string & JoyGetSnapshotStructName ( void )
2015-12-05 16:50:27 +00:00
{
static const std : : string name ( " Joystick " ) ;
return name ;
}
void JoySaveSnapshot ( YamlSaveHelper & yamlSaveHelper )
{
YamlSaveHelper : : Label state ( yamlSaveHelper , " %s: \n " , JoyGetSnapshotStructName ( ) . c_str ( ) ) ;
2016-02-24 21:51:20 +00:00
yamlSaveHelper . SaveInt ( SS_YAML_KEY_JOY0TRIMX , JoyGetTrim ( true ) ) ;
yamlSaveHelper . SaveInt ( SS_YAML_KEY_JOY0TRIMY , JoyGetTrim ( false ) ) ;
2015-12-05 16:50:27 +00:00
yamlSaveHelper . Save ( " %s: %d # not implemented yet \n " , SS_YAML_KEY_JOY1TRIMX , 0 ) ; // not implemented yet
yamlSaveHelper . Save ( " %s: %d # not implemented yet \n " , SS_YAML_KEY_JOY1TRIMY , 0 ) ; // not implemented yet
2021-10-04 22:08:37 +01:00
for ( UINT n = 0 ; n < 4 ; n + + )
{
2022-02-27 04:54:06 +11:00
std : : string str = StrFormat ( SS_YAML_KEY_PDL_INACTIVE_CYCLE , n ) ;
yamlSaveHelper . SaveHexUint64 ( str . c_str ( ) , g_paddleInactiveCycle [ n ] ) ;
2021-10-04 22:08:37 +01:00
}
2015-12-05 16:50:27 +00:00
}
2021-10-04 22:08:37 +01:00
void JoyLoadSnapshot ( YamlLoadHelper & yamlLoadHelper , UINT version )
2015-12-05 16:50:27 +00:00
{
if ( ! yamlLoadHelper . GetSubMap ( JoyGetSnapshotStructName ( ) ) )
return ;
2016-02-24 22:38:59 +00:00
JoySetTrim ( yamlLoadHelper . LoadInt ( SS_YAML_KEY_JOY0TRIMX ) , true ) ;
JoySetTrim ( yamlLoadHelper . LoadInt ( SS_YAML_KEY_JOY0TRIMY ) , false ) ;
yamlLoadHelper . LoadInt ( SS_YAML_KEY_JOY1TRIMX ) ; // dump value
yamlLoadHelper . LoadInt ( SS_YAML_KEY_JOY1TRIMY ) ; // dump value
2015-12-05 16:50:27 +00:00
2021-10-04 22:08:37 +01:00
if ( version > = 7 )
{
for ( UINT n = 0 ; n < 4 ; n + + )
{
2022-02-27 04:54:06 +11:00
std : : string str = StrFormat ( SS_YAML_KEY_PDL_INACTIVE_CYCLE , n ) ;
2021-10-04 22:08:37 +01:00
g_paddleInactiveCycle [ n ] = yamlLoadHelper . LoadUint64 ( str ) ;
}
}
else
{
UINT64 resetCycle = yamlLoadHelper . LoadUint64 ( SS_YAML_KEY_COUNTERRESETCYCLE ) ;
for ( UINT n = 0 ; n < 4 ; n + + )
g_paddleInactiveCycle [ n ] = resetCycle ;
}
2015-12-05 16:50:27 +00:00
yamlLoadHelper . PopMap ( ) ;
}