Steel Battalion controller and win 7 64-Bit drivers
#361
Posted 23 January 2016 - 03:00 PM
BTW, did you see what Skillfulest is trying to do? I gave him everything I could think of but he is still stuck. I honestly don't even know if it'll work...
#362
Posted 24 January 2016 - 01:30 PM
Skillfulist, on 20 January 2016 - 05:01 AM, said:
double combinedValue = controller.Scaled.AimingX*0.9 + controller.Scaled.SightChangeX*0.1; joystick.setAxis(1,(int)combinedValue ,HID_USAGES.HID_USAGE_X);
I think this may be what you're looking for. The Scaled structure contains the values of the SBC scaled to the input value of vJoy, so that you don't really have to calibrate anymore if you don't want to. So in order to get some precise movement, you simply add the two values. Here I use a multiplier of 0.9 for the x-axis of right stick and add it to the x-axis of the thumbstick.
Change the multipliers as you see fit, I just picked 10% of the movement from the thumbstick off the top of my head.
Also, as far as your issue with the buttons staying lit, it shouldn't be that way from default, but it may be that you've changed the second variable from:
controller.AddButtonKeyLightMapping(ButtonEnum.CockpitHatch, true, 3, SBC.Key.A, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF1, true, 3, Microsoft.DirectX.DirectInput.Key.BackSpace, true);
to:
controller.AddButtonKeyLightMapping(ButtonEnum.CockpitHatch, false, 3, SBC.Key.A, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF1, false, 3, Microsoft.DirectX.DirectInput.Key.BackSpace, true);
Edited by HackNFly, 24 January 2016 - 01:34 PM.
#363
Posted 02 February 2016 - 09:02 PM
//if(controller.GetButtonState(ButtonEnum.ToggleFilterControl)) //FILT Toggle { controller.AddButtonKeyMapping(ButtonEnum.RightJoyFire, SBC.Key.D2, true); //joystick.setButton(controller.GetButtonState(ButtonEnum.RightJoyFire),1,15); } //else { controller.AddButtonKeyMapping(ButtonEnum.RightJoyFire, SBC.Key.NoConvert, true); //joystick.setButton(controller.GetButtonState(ButtonEnum.RightJoyFire),1,0); } //if(controller.GetButtonState(ButtonEnum.ToggleOxygenSupply)) // O2 Supply Toggle { controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, SBC.Key.D2, true); } //else { controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, SBC.Key.D3, true); } currentResetValue = controller.GetButtonState((int)ButtonEnum.ToggleFuelFlowRate); if(currentResetValue != lastResetValue && currentResetValue) { controller.TestLEDs(1);//reset lights
Thanks in advance.
Golden
#364
Posted 02 February 2016 - 09:21 PM
~Hack
#365
Posted 03 February 2016 - 07:39 AM
HackNFly, on 02 February 2016 - 09:21 PM, said:
~Hack
Didn't see your reply soon enough. I'll post the full .CS file tonight.
#366
Posted 04 February 2016 - 11:23 PM
// MWO Config File // version 0.7 // by von Pilsner (thanks to HackNFly!@!!!) // // Uses default MWO keybindings as of Sept 26, 2012 // You must map the 'throttle' axis in-game though... // // For use with Steel-Batallion-64_v2_pre.zip // 64 bit driver code/glue by HackNFly http://code.google.com/p/steel-batallion-64/ // // add the folowing to user.cfg // // cl_joystick_gain = 1.35 // cl_joystick_invert_throttle = 0 // cl_joystick_invert_pitch = 1 // cl_joystick_invert_yaw = 0 // cl_joystick_invert_turn = 0 // cl_joystick_throttle_range = 0 // // updated by Santiago Saldana Sept, 27, 2012 using SBC; using System; namespace SBC { public class DynamicClass { SteelBattalionController controller; vJoy joystick; bool acquired; bool jumpPressed = false; bool stopPressed = false;//used in special handling of left pedal bool currentResetValue; bool lastResetValue;//used in assessing when to flip lights int pedalTriggerLevel = 50;//used in special handling of left pedal, SBC.Key jumpKey = SBC.Key.Space; SBC.Key stopKey = SBC.Key.X; const int refreshRate = 30;//number of milliseconds between call to mainLoop //this gets called once by main program public void Initialize() { int baseLineIntensity = 3;//just an average value for LED intensity //int emergencyLightIntensity = 15;//for stuff like eject,cockpit Hatch,Ignition, and Start controller = new SteelBattalionController(); controller.Init(50);//50 is refresh rate in milliseconds //set all buttons by default to light up only when you press them down for(int i=4;i<4+30;i++) { if (i != (int)ButtonEnum.Eject)//excluding eject since we are going to flash that one controller.AddButtonLightMapping((ButtonEnum)(i-1),(ControllerLEDEnum)(i),true,baseLineIntensity); } controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, SBC.Key.D4, true); controller.AddButtonKeyMapping(ButtonEnum.RightJoyLockOn, SBC.Key.R, true); controller.AddButtonKeyLightMapping(ButtonEnum.Eject, true, 3, SBC.Key.O, true); controller.AddButtonKeyLightMapping(ButtonEnum.Ignition, true, 3, SBC.Key.P, true); //controller.AddButtonKeyLightMapping(ButtonEnum.Start, true, 3, SBC.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonOpenClose, true, 3, SBC.Key.B, true); controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonMapZoomInOut, true, 3, SBC.Key.B, true); //controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonModeSelect, true, 3, SBC.Key.X, true); //controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonSubMonitor, true, 3, SBC.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.MainMonZoomIn, true, 3, SBC.Key.Z, true); controller.AddButtonKeyLightMapping(ButtonEnum.MainMonZoomOut, true, 3, SBC.Key.V, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionFSS, false, 13, SBC.Key.D6, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionManipulator, true, 13, SBC.Key.LeftShift, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionLineColorChange, false, 13, SBC.Key.H, true); controller.AddButtonKeyLightMapping(ButtonEnum.Washing, true, 3, SBC.Key.C, true); controller.AddButtonKeyLightMapping(ButtonEnum.Extinguisher, true, 3, SBC.Key.O, true); controller.AddButtonKeyLightMapping(ButtonEnum.Chaff, true, 13, SBC.Key.J, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionTankDetach, true, 3, SBC.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionOverride, true, 3, SBC.Key.O, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionNightScope, false, 13, SBC.Key.N, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF1, true, 3, SBC.Key.Tab, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF2, true, 3, SBC.Key.X, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF3, true, 3, SBC.Key.LeftControl, true); controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConMain, true, 3, SBC.Key.RightControl, true); controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConSub, true, 3, SBC.Key.BackSpace, true); //controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConMagazine, true, 3, SBC.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.Comm1, true, 3, SBC.Key.F6, true); controller.AddButtonKeyLightMapping(ButtonEnum.Comm2, true, 3, SBC.Key.F8, true); controller.AddButtonKeyLightMapping(ButtonEnum.Comm3, true, 3, SBC.Key.F9, true); //controller.AddButtonKeyLightMapping(ButtonEnum.Comm4, true, 3, SBC.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.Comm5, true, 3, SBC.Key.RightBracket, true); controller.AddButtonKeyMapping(ButtonEnum.LeftJoySightChange, SBC.Key.Z, true); controller.AddButtonKeyMapping(ButtonEnum.ToggleVTLocation, SBC.Key.Q, true); controller.AddButtonKeyMapping(ButtonEnum.ToggleFilterControl, SBC.Key.D6, true); joystick = new vJoy(); acquired = joystick.acquireVJD(1); joystick.resetAll();//have to reset before we use it } //this is necessary, as main program calls this to know how often to call mainLoop public int getRefreshRate() { return refreshRate; } private uint getDegrees(double x,double y) { uint temp = (uint)(System.Math.Atan(y/x)* (180/Math.PI)); if(x < 0) temp +=180; if(x > 0 && y < 0) temp += 360; temp += 90;//origin is vertical on POV not horizontal if(temp > 360)//by adding 90 we may have gone over 360 temp -=360; temp*=100; if (temp > 35999) temp = 35999; if (temp < 0) temp = 0; return temp; } public void updatePOVhat() { int thumbstickDeadZone = 44; SBC.POVdirection lastDirection = controller.POVhat; if(( (Math.Abs(controller.SightChangeX) > thumbstickDeadZone) || (Math.Abs(controller.SightChangeY) > thumbstickDeadZone) )) { if(Math.Abs(controller.SightChangeX) > Math.Abs(controller.SightChangeY)) if(controller.SightChangeX <0) controller.POVhat = SBC.POVdirection.LEFT; else controller.POVhat = SBC.POVdirection.RIGHT; else if(controller.SightChangeY <0) controller.POVhat = SBC.POVdirection.DOWN; else controller.POVhat = SBC.POVdirection.UP; } else { controller.POVhat = SBC.POVdirection.CENTER; } if(lastDirection != controller.POVhat) { switch(lastDirection) { case SBC.POVdirection.LEFT: controller.sendKeyUp(SBC.Key.Left); break; case SBC.POVdirection.RIGHT: controller.sendKeyUp(SBC.Key.Right); break; case SBC.POVdirection.DOWN: controller.sendKeyUp(SBC.Key.Up); break; case SBC.POVdirection.UP: controller.sendKeyUp(SBC.Key.Down); break; } } else { switch(controller.POVhat) { case SBC.POVdirection.LEFT: controller.sendKeyDown(SBC.Key.Left); break; case SBC.POVdirection.RIGHT: controller.sendKeyDown(SBC.Key.Right); break; case SBC.POVdirection.DOWN: controller.sendKeyDown(SBC.Key.Up); break; case SBC.POVdirection.UP: controller.sendKeyDown(SBC.Key.Down); break; } } } public void evaluateLeftPedal() { if(controller.LeftPedal > pedalTriggerLevel) { //take care of the button logic separately, to be less confusing if(!jumpPressed)//if not currently holding down jump key { if(controller.RightPedal > pedalTriggerLevel || controller.MiddlePedal > pedalTriggerLevel) { controller.sendKeyDown(jumpKey); jumpPressed = true; } } else//jump button was pressed { //adding these so that else if won't get optimized into one statement if(controller.RightPedal < pedalTriggerLevel && controller.MiddlePedal < pedalTriggerLevel) { controller.sendKeyUp(jumpKey); jumpPressed = false; } } if(!stopPressed)//if not currently holding down stop key { if(controller.RightPedal < pedalTriggerLevel && controller.MiddlePedal < pedalTriggerLevel) { controller.sendKeyDown(stopKey);//send fullstop command stopPressed = true; } } else//stop button was pressed { if(controller.RightPedal > pedalTriggerLevel || controller.MiddlePedal > pedalTriggerLevel) { controller.sendKeyUp(stopKey); stopPressed = false; } } } else { if(stopPressed) { controller.sendKeyUp(stopKey); stopPressed = false; } if(jumpPressed) { controller.sendKeyUp(jumpKey); jumpPressed = false; } } } //this gets called once every refreshRate milliseconds by main program public void mainLoop() { updatePOVhat(); joystick.setAxis(1,controller.Scaled.AimingX,HID_USAGES.HID_USAGE_X); joystick.setAxis(1,controller.Scaled.AimingY,HID_USAGES.HID_USAGE_Y); joystick.setAxis(1,-1*(controller.Scaled.RotationLever),HID_USAGES.HID_USAGE_Z); joystick.setAxis(1,-1*(controller.Scaled.SightChangeX),HID_USAGES.HID_USAGE_RX); joystick.setAxis(1,controller.Scaled.SightChangeY,HID_USAGES.HID_USAGE_RY); joystick.setAxis(1,controller.Scaled.RightMiddlePedal,HID_USAGES.HID_USAGE_RZ);//throttle joystick.setAxis(1,controller.Scaled.LeftPedal,HID_USAGES.HID_USAGE_SL0); joystick.setAxis(1,controller.Scaled.GearLever,HID_USAGES.HID_USAGE_SL1); //joystick.setContPov(1,getDegrees(controller.SightChangeX,controller.SightChangeY),1); // toggle tricks!!! if(controller.GetButtonState(ButtonEnum.ToggleFilterControl)) //FILT Toggle { controller.AddButtonKeyMapping(ButtonEnum.RightJoyFire, SBC.Key.D2, true); joystick.setButton(controller.GetButtonState(ButtonEnum.RightJoyFire),1,15); } else { controller.AddButtonKeyMapping(ButtonEnum.RightJoyFire, SBC.Key.NoConvert, true); joystick.setButton(controller.GetButtonState(ButtonEnum.RightJoyFire),1,0); } if(controller.GetButtonState(ButtonEnum.ToggleOxygenSupply)) // O2 Supply Toggle { controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, SBC.Key.D2, true); } else { controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, SBC.Key.D3, true); } currentResetValue = controller.GetButtonState((int)ButtonEnum.ToggleFuelFlowRate); if(currentResetValue != lastResetValue && currentResetValue) { controller.TestLEDs(1);//reset lights } lastResetValue = currentResetValue; evaluateLeftPedal(); joystick.sendUpdate(1); } //this gets called at the end of the program and must be present, as it cleans up resources public void shutDown() { controller.sendKeyUp(SBC.Key.LeftControl); controller.UnInit(); joystick.Release(1); } } }
And here is the error log:
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(266,7) : error CS1502: The best overloaded method match for 'SBC.SteelBattalionController.GetButtonState(int)' has some invalid arguments
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(266,33) : error CS1503: Argument 1: cannot convert from 'SBC.ButtonEnum' to 'int'
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(269,25) : error CS1502: The best overloaded method match for 'SBC.SteelBattalionController.GetButtonState(int)' has some invalid arguments
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(269,51) : error CS1503: Argument 1: cannot convert from 'SBC.ButtonEnum' to 'int'
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(274,25) : error CS1502: The best overloaded method match for 'SBC.SteelBattalionController.GetButtonState(int)' has some invalid arguments
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(274,51) : error CS1503: Argument 1: cannot convert from 'SBC.ButtonEnum' to 'int'
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(277,7) : error CS1502: The best overloaded method match for 'SBC.SteelBattalionController.GetButtonState(int)' has some invalid arguments
c:\Users\J-Golden\AppData\Local\Temp\kqw3zlo0.0.cs(277,33) : error CS1503: Argument 1: cannot convert from 'SBC.ButtonEnum' to 'int'
To get past this I remarked out lines 266, 269, 274 & 277. That's why they were remed out in the section I sent you.
#367
Posted 05 February 2016 - 12:16 PM
currentResetValue = controller.GetButtonState((int)ButtonEnum.ToggleFuelFlowRate);
change it to
currentResetValue = controller.GetButtonState(ButtonEnum.ToggleFuelFlowRate);
Looking over the interface I can't tell why those are errors are popping up though it should recognize it. I"ll look into it when I get back.
#368
Posted 09 February 2016 - 11:28 PM
I still have no clue what this means, but I hope this helps.
#369
Posted 20 February 2016 - 10:40 PM
1) I remarked out "using Microsoft.DirectX.DirectInput" just to avoid any confusion
2) I ran a replace for "Microsoft.DirectX.DirectInput" to "SBC" to upgrade input configs to the newer version of SteelBattalion-64 program
3) changed:
if(controller.GetButtonState(ButtonEnum.ToggleFilterControl)) //FILT Toggle
to
controller.GetButtonState(ButtonEnum.ToggleFilterControl) //FILT Toggle
This got rid of several errors, but still left two.
c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(271,61) : error CS1002: ; expected c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(276,4) : error CS1525: Invalid expression term 'else' c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(276,8) : error CS1002: ; expected c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(282,60) : error CS1002: ; expected c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(286,4) : error CS1525: Invalid expression term 'else' c:\Users\J-Golden\AppData\Local\Temp\3jb5g3hi.0.cs(286,8) : error CS1002: ; expected
4) inserted ";" at every Line/Column point listed with the error "; Expected"
Now the only thing I can't figure out is the, " Invalid expression term 'else' " error.
I've gone through this process with three different files created by three different people. The only glimmer of hope came from Max Gordon's Elite Dangerous code. It not only gave me points to help me with the fixes shown above, but somehow also had one set run WITHOUT ERRORS!
Find his code here: http://mwomercs.com/...ost__p__3806286
This will still need the fixes above, but you'll see that the the controls starting at lines 82 & 210 return the 'Else' error, but Line 158 does not.
Hope this helps!
#370
Posted 29 February 2016 - 10:13 PM
First the Joystick:
The biggest problem with the joystick is that to get it to work like a mouse (Usable in MWO) you get the movement you want, but several things can seriously throw it off center. To fix this you'd need a hotkey specifically to "Center Torso" losing valuable fight time, distracting you from your game and frustrating the player after a while.
So here's my idea. Why don't we set a button press of "Center Torso" to happen every time the Aiming lever is centered.
We know we can set button presses to certain ranges on an axis, like the left peddle for activating Jump Jets. The code lays it out that if the peddle is pressed to 30% and higher, it activates the JJ where as if you let off it and go underneath 30% it issues a stop command.
What if we took this one step further and instead of setting a threshold, set a range in the middle of both the X and Y axis where if you were too high or too low on either axis, it would not trigger. Basically, we are making a small box in the center of both axis range that if entered will trigger the "Center Torso" key.
It is a rough idea, and would need tweaking to find the line between activating the button press too easily (Happening when you don't want it to) to too hard (fishing for the sweet spot while in the middle of a battle).
Now for the Gear Lever:
This idea also comes from examining previous workings of the JJ and throttle peddle/Reverse Peddle. Look in the calibration screen for the Gear Lever. there is a little box at the bottom that if checked shows you the raw data. The raw data is the numerical amount for any axis controller. If you do it earlier for the joystick, you'll see the X and Y axis now show you the real time positions numerically.
Now the Gear Lever is different then the other inputs. Where they show a broad spectrum of numbers that go into the thousands, the Gear Lever output is always the same.
R = -2
N = -1
1 = 1
2 = 2
3 = 3
4 = 4
5 = 5
These never change! So by making a series of "IF" and "THAN" statements detailing each value to a certain button press, you can now set just about anything to each level of the Gear Lever. In fact, since we are using specific points (numbers) for each position that are basically not connected to each other, we could make using the Gear Lever as easy as setting up any of the other button presses.
Me? I'd create a cruse control where it went up in increments of 20%. I could make it constant, where what ever level its set at, after using the throttle or reverse, the mech will move back to the set speed on the Gear Lever.
OR
I could make it a single press and have it work like car Cruse Controls. Tap the break or speed up, the Cruse Control will disengage until the Gear Lever is shifted.
What do you all think?
#371
Posted 07 April 2016 - 10:05 PM
#372
Posted 08 April 2016 - 04:19 AM
Golden Gun, on 29 February 2016 - 10:13 PM, said:
So here's my idea. Why don't we set a button press of "Center Torso" to happen every time the Aiming lever is centered.
We know we can set button presses to certain ranges on an axis, like the left peddle for activating Jump Jets. The code lays it out that if the peddle is pressed to 30% and higher, it activates the JJ where as if you let off it and go underneath 30% it issues a stop command.
This is simply not going to be practical. The box would need to be minuscule, else you would be unable to aim at points close to the center.
I have a largely working solution for joystick absolute aim by mapping joystick to mouse, you can see my thread here for more info:
http://mwomercs.com/...ivity&mid=62105
#373
Posted 10 April 2016 - 11:55 AM
evilC, on 08 April 2016 - 04:19 AM, said:
I have a largely working solution for joystick absolute aim by mapping joystick to mouse, you can see my thread here for more info:
http://mwomercs.com/...ivity&mid=62105
I did extensive work on getting joystick to mouse to work but there were too many issues. They have all been presented earlier in this thread starting on page 13.
Those issues still exist in your AHK approach as far as I can tell. The idea of mapping "center torso" to a very small point of the center of the axis seems like it would solve all this. I have not tried it yet since I wouldn't know how to do it, and yes, a ton of tweaking would be needed, but doing it this way would be the easiest for people who aren't big on coding themselves.
#374
Posted 11 April 2016 - 03:41 AM
Let me lay it out in maths:
Lets say that a given mech has a torso twist limit of 400 mouse units, and a max twist rate of 100/units sec. ie it takes 4 seconds to twist from full left to full right.
You map your joystick to this range without any of the buffering type functionality that my script has.
You start the game, then move the stick to full right within one second.
This causes a move of 200 units (center to edge) to be issued in one second. The mech can only twist at 100 units/s, so it throws away 100 units - you end up twisted 50% right. The script is now 50% out in one movement.
How is sending a center key when the stick is centered going to fix that? It in no way addresses the fact that you issued the instruction "Move 100% right" and the result was that it moved 50% right.
What did you do your "extensive work" using? If you mapped joystick position directly to mouse output and did not use code to buffer the mouse output and only move the mouse as fast as the torso could handle, then of course you are never, ever going to be able to find anything like a workable solution - because you have not attempted to address some of the discrepancies between the input method and the output method.
Yes, there are still slight issues with my script, but it just means that we have not worked out all the parameters that we need to feed into the equations. I suspect what is throwing it out is that at the moment I simply stipulate a MAX_TWIST_RATE (How many units per tick the mouse can be moved) but I am forgetting about inertia. If the mech is currently twisting left at 100 units/sec (max rate), it cannot twist right at 100 units/sec on the next tick - it has to slow down the movement in the opposite direction first - ie I need to specify a Max Delta value as well.
It's worth noting that the advantage of using an AHK script is that we can do EMPIRICAL tests. By using code to send mouse_event API calls, we can keep a record of exactly how many units of movement in each direction we have made. I ran a script that twisted the torso left and right (It was about 50% of the move range, but that's not really relevant) at the slowest speed and let it run for an hour. When I came back, it was still 100% in synch - ie when I hit the hotkey to stop moving side to side and move the torso back to center, then hit the "recenter torso" key, the view *did not move*. So clearly there are legs in the mouse output technique, we just need to be able to find what the limits of movement are and obey them.
TL;DR: It's possible to move the joystick faster than the mech torso can physically twist. That problem is ONLY solvable through code. You CANNOT solve this problem using sensitivity or recentering - the output must be fundamentally decoupled from the input.
Edited by evilC, 11 April 2016 - 03:55 AM.
#375
Posted 11 April 2016 - 12:47 PM
Soooooooo you like to feel superior and talk down to people, is that it? I did hours of checking, measuring, testing again to get the data I presented. If you think I missed something, just say so. Stop trying to push YOUR fix as the ONLY fix.
Oh, and the "Maths" you took so long to explain is exactly WHAT I'm trying to fix.
Al the issues I have faced using the controller in MWO WOULD be fixed if I could figure out how to implement the idea. Issues with recentering, after looking around while shut down, Moving the torso too fast, and zooming in and out. The only issues I have with these problems is centering and If I can make the box small enough that it is only activated when you actually center the stick, why CAN"T this work?
You say there is no WAY this would work. Maybe it does and maybe it doesn't, but I'm sticking with the overall theme of having SB-64 be a SINGLE solution. If you don't like it, you don't have to stay or even reply.
#376
Posted 12 April 2016 - 03:22 AM
Quote
Quote
I was saying that something along the lines of what my script does (i/o limiting and buffering) is the only possible solution to the problem, and presented the code as an example of how to do it.
Quote
As I said, because it is not addressing the problem, but trying to stick a dirty-great band-aid over it and hoping it goes away, which it will not.
I gave the equations that demonstrate why this technique will not work - if you cannot wrap your head around the maths, don't have a go at me...
#377
Posted 12 April 2016 - 09:20 AM
edit:So I believe I understand the key element to your code.
; Limit the amount of movement for this tick to the MAX_TWIST_RATE delta_move[A_Index] := abs(delta_move[A_Index]) > MAX_TWIST_RATE ? MAX_TWIST_RATE * sgn(delta_move[A_Index]) : delta_move[A_Index]
Its not a bad idea. Certainly worth investigating. I also did the scripting tests you did as far as moving left and right and then coming back.
edit: numPixelExtraX/Y was added to take into account the difference in movement for mechs that allow you to have extra range by using your arms.
GoldenGun's approach would work for a spring mounted joystick, as you can be certain to return to a relatively precise position by releasing the joystick. This seems like it would be relatively difficult to pull off with the springless SBC joystick. I believe this is one of those situations where simply making use of the center torso as a joystick button will functionally work better than trying to achieve the same programmatically.
Here is the code I helped write for Mouse emulation a few years ago, although looking through some of the comments on p. 13 may help.
// MWO Config File // version 3.1 (WiP) // by von Pilsner (thanks to HackNFly!@!!!) // // NOTE: Calibrate using BvP-Simple-3.0.cs // Or your pedals may not work properly!! // // Uses default MWO keybindings and axis as of Jan 28, 2013 // // For use with Steel-Batallion-64_v2_beta.zip // 64 bit driver code/glue by HackNFly http://code.google.com/p/steel-batallion-64/ // // I suggest you add the folowing to user.cfg (remove the //'s) // // cl_joystick_gain = 5.05 // cl_joystick_sensitivity = 1.00 // cl_joystick_invert_throttle = 0 // cl_joystick_invert_pitch = 1 // cl_joystick_invert_yaw = 0 // cl_joystick_invert_turn = 0 // cl_joystick_throttle_range = 0 // // ; Joystick DeadZone, requires both the i and cl lines to make the joystick deadzone change work // ; Increase in 0.02 incriments if you get an unstable center point on either stick (0.08 works for some). // i_joystick_deadzone = 0.04 // cl_joystick_deadzone = 0.04 // using SBC; using System; namespace SBC { public class DynamicClass { SteelBattalionController controller; vJoy joystick; bool acquired; String debugString = ""; int desiredX; int desiredY; int currentX = 0; int currentY = 0; int numPixelX = 600;//number of pixels to move in X direction int numPixelY = 300;//number of pixels to move in X direction int numPixelExtraX = 25;//used at extreme edges, number of pixels per poll int numPixelExtraY = 20;//used at extreme edges const int refreshRate = 30; // Number of milliseconds between call to mainLoop const int maxAxisValue = 32768; double sideZone = 0.05;//percent of swing that will cause mouse to move continuously int jj = 0; bool startedTracking = false;//used to make switching mouse on and off not jumpy // This gets called once by main program public void Initialize() { int baseLineIntensity = 3; // Just an average value for LED intensity int emergencyLightIntensity = 15; // For stuff like eject,cockpit Hatch,Ignition, and Start controller = new SteelBattalionController(); controller.Init(30); // 50 is refresh rate in milliseconds //set all buttons by default to light up only when you press them down for (int i = 4; i < 4 + 30; i++) { if (i != (int)ButtonEnum.Eject) // Excluding eject since we are going to flash that one controller.AddButtonLightMapping((ButtonEnum)(i - 1), (ControllerLEDEnum)(i), true, baseLineIntensity); } // Button Bindings controller.AddButtonKeyMapping(ButtonEnum.RightJoyFire, Microsoft.DirectX.DirectInput.Key.D1,true); controller.AddButtonKeyMapping(ButtonEnum.RightJoyMainWeapon, Microsoft.DirectX.DirectInput.Key.D2,true); controller.AddButtonKeyMapping(ButtonEnum.RightJoyLockOn, Microsoft.DirectX.DirectInput.Key.R, true); controller.AddButtonKeyLightMapping(ButtonEnum.Eject, true, 3, Microsoft.DirectX.DirectInput.Key.O, true); controller.AddButtonKeyLightMapping(ButtonEnum.Ignition, true, 3, Microsoft.DirectX.DirectInput.Key.P, true); //controller.AddButtonKeyLightMapping(ButtonEnum.CockpitHatch, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); //controller.AddButtonKeyLightMapping(ButtonEnum.Start, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonOpenClose, true, 3, Microsoft.DirectX.DirectInput.Key.B, true); controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonMapZoomInOut, true, 3, Microsoft.DirectX.DirectInput.Key.I, true); controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonModeSelect, true, 3, Microsoft.DirectX.DirectInput.Key.Q, true); //controller.AddButtonKeyLightMapping(ButtonEnum.MultiMonSubMonitor, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.MainMonZoomIn, true, 3, Microsoft.DirectX.DirectInput.Key.Z, true); controller.AddButtonKeyLightMapping(ButtonEnum.MainMonZoomOut, true, 3, Microsoft.DirectX.DirectInput.Key.V, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionFSS, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionManipulator, true, 3, Microsoft.DirectX.DirectInput.Key.J, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionLineColorChange, true, 3, Microsoft.DirectX.DirectInput.Key.H, true); controller.AddButtonKeyLightMapping(ButtonEnum.Washing, true, 3, Microsoft.DirectX.DirectInput.Key.C, true); controller.AddButtonKeyLightMapping(ButtonEnum.Extinguisher, true, 3, Microsoft.DirectX.DirectInput.Key.Delete, true); //controller.AddButtonKeyLightMapping(ButtonEnum.Chaff, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionTankDetach, true, 3, Microsoft.DirectX.DirectInput.Key.Slash, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionOverride, true, 3, Microsoft.DirectX.DirectInput.Key.O, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionNightScope, true, 3, Microsoft.DirectX.DirectInput.Key.N, true); controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF1, true, 3, Microsoft.DirectX.DirectInput.Key.Tab, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF2, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); //controller.AddButtonKeyLightMapping(ButtonEnum.FunctionF3, true, 4, Microsoft.DirectX.DirectInput.Key.LeftControl, true); controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConMain, true, 3, Microsoft.DirectX.DirectInput.Key.RightControl, true); controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConSub, true, 3, Microsoft.DirectX.DirectInput.Key.BackSpace, true); //controller.AddButtonKeyLightMapping(ButtonEnum.WeaponConMagazine, true, 3, Microsoft.DirectX.DirectInput.Key.X, true); //controller.AddButtonKeyLightMapping(ButtonEnum.ToggleBufferMaterial, true, 3, Microsoft.DirectX.DirectInput.Key.LeftShift, true); // controller.AddButtonKeyMapping(ButtonEnum.LeftJoySightChange, Microsoft.DirectX.DirectInput.Key.V, true); controller.AddButtonKeyMapping(ButtonEnum.LeftJoySightChange, Microsoft.DirectX.DirectInput.Key.Z, true); joystick = new vJoy(); acquired = joystick.acquireVJD(1); joystick.resetAll(); //have to reset before we use it joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_SL1); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_X); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_Y); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_Z);//throttle joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_RZ); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_SL0); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_RX); joystick.setAxis(1,32768/2,HID_USAGES.HID_USAGE_RY); } //this is necessary, as main program calls this to know how often to call mainLoop public int getRefreshRate() { return refreshRate; } private uint getDegrees(double x, double y) { uint temp = (uint)(System.Math.Atan(y / x) * (180 / Math.PI)); if (x < 0) temp += 180; if (x > 0 && y < 0) temp += 360; temp += 90; //origin is vertical on POV not horizontal if (temp > 360)//by adding 90 we may have gone over 360 temp -= 360; temp *= 100; if (temp > 35999) temp = 35999; if (temp < 0) temp = 0; return temp; } // POV to Arrow Keys public void updatePOVhat() { int thumbstickDeadZone = 75; SBC.POVdirection lastDirection = controller.POVhat; if (((Math.Abs(controller.SightChangeX) > thumbstickDeadZone) || (Math.Abs(controller.SightChangeY) > thumbstickDeadZone))) { if (Math.Abs(controller.SightChangeX) > Math.Abs(controller.SightChangeY)) if (controller.SightChangeX < 0) controller.POVhat = SBC.POVdirection.LEFT; else controller.POVhat = SBC.POVdirection.RIGHT; else if (controller.SightChangeY < 0) controller.POVhat = SBC.POVdirection.DOWN; else controller.POVhat = SBC.POVdirection.UP; } else { controller.POVhat = SBC.POVdirection.CENTER; } if (lastDirection != controller.POVhat) { switch (lastDirection) { case SBC.POVdirection.LEFT: controller.sendKeyUp(Microsoft.DirectX.DirectInput.Key.Left); break; case SBC.POVdirection.RIGHT: controller.sendKeyUp(Microsoft.DirectX.DirectInput.Key.Right); break; case SBC.POVdirection.DOWN: controller.sendKeyUp(Microsoft.DirectX.DirectInput.Key.Up); break; case SBC.POVdirection.UP: controller.sendKeyUp(Microsoft.DirectX.DirectInput.Key.Down); break; } } else { switch (controller.POVhat) { case SBC.POVdirection.LEFT: controller.sendKeyDown(Microsoft.DirectX.DirectInput.Key.Left); break; case SBC.POVdirection.RIGHT: controller.sendKeyDown(Microsoft.DirectX.DirectInput.Key.Right); break; case SBC.POVdirection.DOWN: controller.sendKeyDown(Microsoft.DirectX.DirectInput.Key.Up); break; case SBC.POVdirection.UP: controller.sendKeyDown(Microsoft.DirectX.DirectInput.Key.Down); break; } } } private double expo(int value) { double tempIn = ((double)value / (double)maxAxisValue)*2 - 1;//scale to -1 to 1 tempIn *= tempIn * 0.5;//this applies expo, we get 0 to 1 regardless of sign then multiply by 0.5 since it only represents //half of the value tempIn = Math.Abs(tempIn); if(value >= maxAxisValue/2) tempIn += 0.5; else tempIn = 0.5 - tempIn; //debugString += "value:" + value.ToString() + " tempIn : " + tempIn.ToString() + "\n"; return tempIn; } public int scaleValue(int value, int low, int middle, int high, int deadzone) { double temp; debugString += value + " " + low + " " + middle + " " + high + " " + deadzone + "\n"; if(Math.Abs(value - middle) < deadzone) temp = ((middle-low)/(double)(high - low))*maxAxisValue; else { if(value < middle) temp = ((double)(value - low) / (double)(middle - low) * 0.5)*maxAxisValue; else temp = ((double)(value - middle) / (double)(high - middle) * 0.5 + 0.5)*maxAxisValue; } //clamp for extraneous values if(temp > maxAxisValue) temp = maxAxisValue; if(temp < 0) temp = 0; return (int) temp; } public int scaleValue(int value, int low, int high) { double temp; temp = ((double)(value - low) / (double)(high - low) * 0.5)*maxAxisValue; //clamp for extraneous values if(temp > maxAxisValue) temp = maxAxisValue; if(temp < 0) temp = 0; return (int) temp; } private int getDeltaS(int axisVal, int desiredVal, int currentVal,int pixelExtra) { int delta = desiredVal - currentVal; //int temp = (int) (expo(axisVal)*pixels); //debugString += "axisVal" + axisVal.ToString() + " " + expo(axisVal).ToString() + " temp " + temp.ToString() + " " //+ "desiredVal:" + desiredVal.ToString() + " currentVal : " + currentVal.ToString() + "\n"; double tempD = (double)axisVal/(double)maxAxisValue; if(tempD > (1 - sideZone))//sidezone is a percentage, i.e. 0.05 for 5 percent { tempD = (tempD - (1- sideZone)); tempD = tempD/sideZone * pixelExtra;//defined at top return (int)tempD; } if(tempD < sideZone) { tempD = (sideZone - tempD)/sideZone * pixelExtra; return (int)(-1*tempD); } return delta; } private int reverse(int val) { return (maxAxisValue - val); } /* //new optional function used for debugging purposes, comment out when running in game as it causes issues public String getDebugString() { return debugString; } */ //this gets called once every refreshRate milliseconds by main program public void mainLoop() { debugString = ""; //updatePOVhat(); int aimingX = scaleValue(controller.AimingX,1,512,1023,5); int aimingY = reverse(scaleValue(controller.AimingY,1,512,1021,5));//calibration values int rAxis = scaleValue(controller.RotationLever,-421,1,510,5);//calibration values int sCX = scaleValue(controller.SightChangeX,-461,0,470,5); int sCY = scaleValue(controller.SightChangeX,-480,-5,463,5); int lPedal = scaleValue(controller.LeftPedal,30,1022); int mPedal = scaleValue(controller.MiddlePedal,128,1021); int rPedal = scaleValue(controller.RightPedal,0,1020); desiredX = (int)(expo(aimingX)*numPixelX);//numPixels stores resolution, i.e. how much we move mouse desiredY = (int)(expo(aimingY)*numPixelY); rAxis = (int)(expo(rAxis)*maxAxisValue); debugString += "expoX " + expo(aimingX) + "\n"; debugString += "expoY " + expo(aimingY) + "\n"; debugString += "expoR " + expo(rAxis) + "\n"; if((bool)controller.GetButtonState(ButtonEnum.ToggleBufferMaterial)) { //debugString += "aimingX:" + aimingX.ToString() + " desiredX : " + desiredX.ToString() + "currentX " + currentX.ToString() + "\n"; //debugString += "aimingY:" + aimingY.ToString() + " desiredY : " + desiredY.ToString() + "currentY " + currentY.ToString() + "\n"; int deltaX = getDeltaS(aimingX,desiredX,currentX,numPixelExtraX); int deltaY = getDeltaS(aimingY,desiredY,currentY,numPixelExtraY); currentX = desiredX; currentY = desiredY; if(startedTracking)//makes it so you can flip the switch and recenter the joystick InputSimulator.MoveMouseBy(deltaX,deltaY); else startedTracking = true; } else { startedTracking = false; } // Joystick Axes debugString += "rAxis = " + rAxis + "\n"; //joystick.setAxis(1, xaxis, HID_USAGES.HID_USAGE_X); //joystick.setAxis(1, yaxis, HID_USAGES.HID_USAGE_Y); joystick.setAxis(1, rAxis, HID_USAGES.HID_USAGE_RZ); //joystick.setAxis(1, sCX, HID_USAGES.HID_USAGE_SL1); //joystick.setAxis(1, sCY, HID_USAGES.HID_USAGE_RX); joystick.setAxis(1,(rPedal - mPedal) + maxAxisValue/2,HID_USAGES.HID_USAGE_SL0);//throttle //joystick.setAxis(1, controller.GearLever, HID_USAGES.HID_USAGE_SL0); joystick.setContPov(1, getDegrees(controller.SightChangeX, controller.SightChangeY), 1); // Pedals Section if ((jj == 0) && (lPedal > (maxAxisValue*0.10))) // Left pedal pressed { controller.sendKeyDown(Microsoft.DirectX.DirectInput.Key.Space); jj = 1; } else if ((jj == 1) && (lPedal < (maxAxisValue*0.10))) // Left pedal released { controller.sendKeyUp(Microsoft.DirectX.DirectInput.Key.Space); jj = 0; } joystick.sendUpdate(1); } // This gets called at the end of the program and must be present, as it cleans up resources public void shutDown() { controller.UnInit(); joystick.Release(1); } } }
Edited by HackNFly, 12 April 2016 - 09:42 AM.
#378
Posted 12 April 2016 - 10:22 AM
Edit - Actually, if someone could share with me how to setup a T16000 to use the HAT for big movements, and the stick to only move as far as the arms will go, that would be perfect for me. I'd much prefer that type of control over trying to guess how far I have to set my stick to handle the whole mech. Basically, HAT for large movements and stick for fine aim.
Edited by GreenHell, 12 April 2016 - 10:28 AM.
#379
Posted 12 April 2016 - 10:49 AM
"I abandoned trying to emulate a mouse though due to the enormous pain of guestimating maximum twist rates for each mech"
I basically have this one licked. To detect max twist rate, I send a big move (eg +50), then the same amount in the opposite direction but as a series of small moves (ie 50x -1). I then take a screenshot, hit center torso, then take another screenshot. If the two screenshots match, then 50 is below the max twist rate, so the code increases the rate and repeats.
Detecting twist range is also done with pixel detection. I center the torso, then keep twisting until the screen doesnt change after I issue mouse movement, which gives me the twist range in mouse units.
This code is not currently in the zip in my thread, I have some more work to do on it before I release it, hopefully tonight.
You are correct about the bit of code that controls the "buffer". To avoid any confusion, here is the logic that it follows:
I will use fake values to make the math simpler:
TWIST_RATE : The number of mouse units that a mech can twist in one tick (10ms for me). Lets use a value of 20 for this.
TWIST_RANGE: The number of mouse units that a mech can twist. Lets assume that this is 100 units from center to twisted full right.
Let's also assume that a joystick reports -100 to +100 (So center to full twist right is also 100 units)
Current mouse "position" is 0
Joystick is 50% right (+50)
Corresponding mouse move to achieve that view would be +50
TWIST_RATE is 20, so we instead only move +20
End of loop
Next "tick".
Current mouse "position" is +20
Joystick has moved to 30% right
Corresponding mouse move to achieve that view would be +10
+10 is below 20, so allow the full amount that it "wants" to move.
This code is simple in AHK, because I can only poll the joystick manually anyway (Joystick axis input in AHK is not event based), so even if the stick does not change, I get a current value each "tick". If you get joystick position in an event-based manner, then implementing this may be a bit more tricky.
Looking at that code you posted, there seems to be rather a lot of logic going on for POV hats. This is an issue that I have come up against in AHK, and I found that generally the best solution was look up tables (ie prebuilt static arrays) - conversions such as degrees to cardinal directions (At least with a normal POV where the degrees are always in 45deg increments) is actually quite quick and simple with arrays. Lemme know if you need some pointers.
Edited by evilC, 12 April 2016 - 10:54 AM.
#380
Posted 12 April 2016 - 10:52 AM
GreenHell, on 12 April 2016 - 10:22 AM, said:
No, you are not alone.
GreenHell, on 12 April 2016 - 10:22 AM, said:
May not be technically possible. Is there still the key ?CTRL? that you can hold to look around the cockpit? I seem to remember the arms still worked in that mode. If that still worked AND you could still control the torso (eg with keys) while the button was held, then yes this may be possible, otherwise I dont see how.
28 user(s) are reading this topic
0 members, 28 guests, 0 anonymous users