mirror of
https://github.com/moonlight-stream/moonlight-embedded.git
synced 2026-04-21 07:40:36 +00:00
fixed up gamepad mapping a bit. added options to invert and to act as a trigger
This commit is contained in:
@@ -5,6 +5,8 @@ import java.awt.Dimension;
|
|||||||
import java.awt.Toolkit;
|
import java.awt.Toolkit;
|
||||||
import java.awt.event.ActionEvent;
|
import java.awt.event.ActionEvent;
|
||||||
import java.awt.event.ActionListener;
|
import java.awt.event.ActionListener;
|
||||||
|
import java.awt.event.ItemEvent;
|
||||||
|
import java.awt.event.ItemListener;
|
||||||
import java.awt.event.WindowAdapter;
|
import java.awt.event.WindowAdapter;
|
||||||
import java.awt.event.WindowEvent;
|
import java.awt.event.WindowEvent;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -12,6 +14,7 @@ import java.util.List;
|
|||||||
import javax.swing.Box;
|
import javax.swing.Box;
|
||||||
import javax.swing.BoxLayout;
|
import javax.swing.BoxLayout;
|
||||||
import javax.swing.JButton;
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JCheckBox;
|
||||||
import javax.swing.JFrame;
|
import javax.swing.JFrame;
|
||||||
import javax.swing.JOptionPane;
|
import javax.swing.JOptionPane;
|
||||||
import javax.swing.JPanel;
|
import javax.swing.JPanel;
|
||||||
@@ -29,49 +32,75 @@ import com.limelight.settings.GamepadSettingsManager;
|
|||||||
|
|
||||||
public class SettingsFrame extends JFrame {
|
public class SettingsFrame extends JFrame {
|
||||||
private static final long serialVersionUID = 1L;
|
private static final long serialVersionUID = 1L;
|
||||||
|
|
||||||
private boolean configChanged = false;
|
private boolean configChanged = false;
|
||||||
private boolean shouldStartHandler = false;
|
private boolean shouldStartHandler = false;
|
||||||
|
|
||||||
|
private Thread mappingThread;
|
||||||
private GamepadMapping config;
|
private GamepadMapping config;
|
||||||
|
|
||||||
public SettingsFrame() {
|
public SettingsFrame() {
|
||||||
super("Gamepad Settings");
|
super("Gamepad Settings");
|
||||||
System.out.println("Creating Settings Frame");
|
System.out.println("Creating Settings Frame");
|
||||||
this.setSize(500, 500);
|
this.setSize(900, 500);
|
||||||
this.setResizable(false);
|
this.setResizable(false);
|
||||||
this.setAlwaysOnTop(true);
|
this.setAlwaysOnTop(true);
|
||||||
config = GamepadSettingsManager.getSettings();
|
config = GamepadSettingsManager.getSettings();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void build() {
|
public void build() {
|
||||||
Container c = this.getContentPane();
|
Container c = this.getContentPane();
|
||||||
JPanel mainPanel = new JPanel();
|
JPanel mainPanel = new JPanel();
|
||||||
mainPanel.setLayout(new BoxLayout(mainPanel, BoxLayout.X_AXIS));
|
mainPanel.setLayout(new BoxLayout(mainPanel, BoxLayout.X_AXIS));
|
||||||
|
|
||||||
Box leftColumn = Box.createVerticalBox();
|
Box leftColumn = Box.createVerticalBox();
|
||||||
Box rightColumn = Box.createVerticalBox();
|
Box rightColumn = Box.createVerticalBox();
|
||||||
|
|
||||||
leftColumn.add(Box.createVerticalStrut(10));
|
leftColumn.add(Box.createVerticalStrut(10));
|
||||||
rightColumn.add(Box.createVerticalStrut(10));
|
rightColumn.add(Box.createVerticalStrut(10));
|
||||||
|
|
||||||
ControllerComponent[] components = ControllerComponent.values();
|
ControllerComponent[] components = ControllerComponent.values();
|
||||||
for (int i = 0; i < components.length; i++) {
|
for (int i = 0; i < components.length; i++) {
|
||||||
Box componentBox = Box.createHorizontalBox();
|
Box componentBox = Box.createHorizontalBox();
|
||||||
|
|
||||||
componentBox.add(Box.createHorizontalStrut(10));
|
componentBox.add(Box.createHorizontalStrut(10));
|
||||||
componentBox.add(components[i].getLabel());
|
componentBox.add(components[i].getLabel());
|
||||||
componentBox.add(Box.createHorizontalGlue());
|
componentBox.add(Box.createHorizontalGlue());
|
||||||
componentBox.add(components[i].getMapButton());
|
componentBox.add(components[i].getMapButton());
|
||||||
|
componentBox.add(Box.createHorizontalStrut(5));
|
||||||
|
componentBox.add(components[i].getInvertBox());
|
||||||
|
componentBox.add(Box.createHorizontalStrut(5));
|
||||||
|
componentBox.add(components[i].getTriggerBox());
|
||||||
componentBox.add(Box.createHorizontalStrut(10));
|
componentBox.add(Box.createHorizontalStrut(10));
|
||||||
|
|
||||||
Dimension buttonSize = new Dimension(50,30);
|
Dimension buttonSize = new Dimension(110,32);
|
||||||
components[i].getMapButton().setMaximumSize(buttonSize);
|
components[i].getMapButton().setMaximumSize(buttonSize);
|
||||||
components[i].getMapButton().setMinimumSize(buttonSize);
|
components[i].getMapButton().setMinimumSize(buttonSize);
|
||||||
components[i].getMapButton().setPreferredSize(buttonSize);
|
components[i].getMapButton().setPreferredSize(buttonSize);
|
||||||
|
|
||||||
components[i].getMapButton().addActionListener(createListener());
|
components[i].getMapButton().addActionListener(createListener());
|
||||||
components[i].getMapButton().setText(config.getMapping(components[i]));
|
components[i].getMapButton().setText(config.getMapping(components[i]));
|
||||||
|
|
||||||
|
components[i].getInvertBox().addItemListener(new ItemListener() {
|
||||||
|
@Override
|
||||||
|
public void itemStateChanged(ItemEvent e) {
|
||||||
|
JCheckBox clicked = (JCheckBox)e.getItem();
|
||||||
|
ControllerComponent contComp = ControllerComponent.valueOf(clicked.getName());
|
||||||
|
contComp.invert(e.getStateChange() == ItemEvent.SELECTED);
|
||||||
|
configChanged = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
components[i].getTriggerBox().addItemListener(new ItemListener() {
|
||||||
|
@Override
|
||||||
|
public void itemStateChanged(ItemEvent e) {
|
||||||
|
JCheckBox clicked = (JCheckBox)e.getItem();
|
||||||
|
ControllerComponent contComp = ControllerComponent.valueOf(clicked.getName());
|
||||||
|
contComp.trigger(e.getStateChange() == ItemEvent.SELECTED);
|
||||||
|
configChanged = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
if (i > components.length / 2) {
|
if (i > components.length / 2) {
|
||||||
rightColumn.add(componentBox);
|
rightColumn.add(componentBox);
|
||||||
@@ -85,23 +114,25 @@ public class SettingsFrame extends JFrame {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
rightColumn.add(Box.createVerticalGlue());
|
rightColumn.add(Box.createVerticalGlue());
|
||||||
leftColumn.add(Box.createVerticalGlue());
|
leftColumn.add(Box.createVerticalGlue());
|
||||||
|
|
||||||
mainPanel.add(Box.createHorizontalStrut(20));
|
mainPanel.add(Box.createHorizontalStrut(10));
|
||||||
mainPanel.add(leftColumn);
|
mainPanel.add(leftColumn);
|
||||||
mainPanel.add(Box.createHorizontalGlue());
|
mainPanel.add(Box.createHorizontalStrut(75));
|
||||||
mainPanel.add(rightColumn);
|
mainPanel.add(rightColumn);
|
||||||
|
mainPanel.add(Box.createHorizontalStrut(10));
|
||||||
mainPanel.add(Box.createHorizontalStrut(20));
|
|
||||||
|
|
||||||
c.add(mainPanel);
|
c.add(mainPanel);
|
||||||
|
|
||||||
this.addWindowListener(new WindowAdapter() {
|
this.addWindowListener(new WindowAdapter() {
|
||||||
@Override
|
@Override
|
||||||
public void windowClosing(WindowEvent e) {
|
public void windowClosing(WindowEvent e) {
|
||||||
super.windowClosing(e);
|
super.windowClosing(e);
|
||||||
|
if (mappingThread != null && mappingThread.isAlive()) {
|
||||||
|
mappingThread.interrupt();
|
||||||
|
}
|
||||||
if (configChanged) {
|
if (configChanged) {
|
||||||
updateConfigs();
|
updateConfigs();
|
||||||
ControllerListener.startUp();
|
ControllerListener.startUp();
|
||||||
@@ -111,72 +142,97 @@ public class SettingsFrame extends JFrame {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
Dimension dim = Toolkit.getDefaultToolkit().getScreenSize();
|
Dimension dim = Toolkit.getDefaultToolkit().getScreenSize();
|
||||||
this.setLocation(dim.width/2-this.getSize().width/2, dim.height/2-this.getSize().height/2);
|
this.setLocation(dim.width/2-this.getSize().width/2, dim.height/2-this.getSize().height/2);
|
||||||
this.setVisible(true);
|
this.setVisible(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
private ActionListener createListener() {
|
private ActionListener createListener() {
|
||||||
return new ActionListener() {
|
return new ActionListener() {
|
||||||
@Override
|
@Override
|
||||||
public void actionPerformed(ActionEvent e) {
|
public void actionPerformed(ActionEvent e) {
|
||||||
//#allthejank
|
//#allthejank
|
||||||
ControllerComponent contComp = ControllerComponent.valueOf(((JButton)e.getSource()).getName());
|
ControllerComponent contComp = ControllerComponent.valueOf(((JButton)e.getSource()).getName());
|
||||||
|
|
||||||
List<Gamepad> gamepads = GamepadHandler.getGamepads();
|
List<Gamepad> gamepads = GamepadHandler.getGamepads();
|
||||||
|
|
||||||
if (gamepads.isEmpty()) {
|
if (gamepads.isEmpty()) {
|
||||||
JOptionPane.showMessageDialog(SettingsFrame.this, "No Gamepad Detected");
|
JOptionPane.showMessageDialog(SettingsFrame.this, "No Gamepad Detected");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
map(contComp, gamepads.get(0));
|
||||||
contComp.getMapButton().setText("Select Input");
|
|
||||||
|
|
||||||
ControllerListener.stopListening();
|
|
||||||
|
|
||||||
if (GamepadHandler.isRunning()) {
|
|
||||||
GamepadHandler.stopHandler();
|
|
||||||
shouldStartHandler = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
final Gamepad listenPad = gamepads.get(0);
|
|
||||||
|
|
||||||
Component newMapping = null;
|
|
||||||
|
|
||||||
while (newMapping == null) {
|
|
||||||
listenPad.poll();
|
|
||||||
EventQueue queue = listenPad.getEvents();
|
|
||||||
Event event = new Event();
|
|
||||||
|
|
||||||
while (queue.getNextEvent(event)) {
|
|
||||||
if (Math.abs(event.getValue()) > .75F) {
|
|
||||||
newMapping = event.getComponent();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//spin off a new thread to handle any other events we got
|
|
||||||
new Thread(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
listenPad.poll();
|
|
||||||
listenPad.handleEvents(null);
|
|
||||||
}
|
|
||||||
}).start();
|
|
||||||
|
|
||||||
ControllerComponent oldConfig = config.getControllerComponent(newMapping);
|
|
||||||
if (oldConfig != null) {
|
|
||||||
oldConfig.getMapButton().setText("");
|
|
||||||
}
|
|
||||||
config.insertMapping(contComp, newMapping);
|
|
||||||
contComp.getMapButton().setText(newMapping.getName());
|
|
||||||
configChanged = true;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void map(final ControllerComponent contComp, final Gamepad pad) {
|
||||||
|
if (mappingThread == null || !mappingThread.isAlive()) {
|
||||||
|
contComp.getMapButton().setSelected(true);
|
||||||
|
ControllerListener.stopListening();
|
||||||
|
|
||||||
|
if (GamepadHandler.isRunning()) {
|
||||||
|
GamepadHandler.stopHandler();
|
||||||
|
shouldStartHandler = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
contComp.getMapButton().setText("Select Input");
|
||||||
|
|
||||||
|
mappingThread = new Thread(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
Component newMapping = null;
|
||||||
|
|
||||||
|
while (newMapping == null) {
|
||||||
|
pad.poll();
|
||||||
|
EventQueue queue = pad.getEvents();
|
||||||
|
Event event = new Event();
|
||||||
|
|
||||||
|
while (queue.getNextEvent(event)) {
|
||||||
|
if (Math.abs(event.getValue()) > .75F) {
|
||||||
|
newMapping = event.getComponent();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Thread.sleep(100);
|
||||||
|
} catch (InterruptedException e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// start a new thread to go through all of the remaining events
|
||||||
|
Thread consumeEvents = new Thread(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
pad.poll();
|
||||||
|
EventQueue queue = pad.getEvents();
|
||||||
|
Event event = new Event();
|
||||||
|
|
||||||
|
while (queue.getNextEvent(event)) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
consumeEvents.setName("Consume Events Thread");
|
||||||
|
consumeEvents.start();
|
||||||
|
|
||||||
|
ControllerComponent oldConfig = config.getControllerComponent(newMapping);
|
||||||
|
if (oldConfig != null) {
|
||||||
|
config.removeMapping(newMapping);
|
||||||
|
oldConfig.getMapButton().setText("");
|
||||||
|
}
|
||||||
|
|
||||||
|
config.insertMapping(contComp, newMapping);
|
||||||
|
contComp.getMapButton().setText(newMapping.getName());
|
||||||
|
configChanged = true;
|
||||||
|
contComp.getMapButton().setSelected(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
mappingThread.setName("Gamepad Mapping Thread");
|
||||||
|
mappingThread.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
private void updateConfigs() {
|
private void updateConfigs() {
|
||||||
GamepadSettingsManager.writeSettings(config);
|
GamepadSettingsManager.writeSettings(config);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ package com.limelight.input;
|
|||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
|
||||||
import javax.swing.JButton;
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JCheckBox;
|
||||||
import javax.swing.JLabel;
|
import javax.swing.JLabel;
|
||||||
|
|
||||||
public enum ControllerComponent implements Serializable {
|
public enum ControllerComponent implements Serializable {
|
||||||
@@ -15,13 +16,24 @@ public enum ControllerComponent implements Serializable {
|
|||||||
|
|
||||||
private JLabel label;
|
private JLabel label;
|
||||||
private JButton mapButton;
|
private JButton mapButton;
|
||||||
|
private JCheckBox invertBox;
|
||||||
|
private JCheckBox triggerBox;
|
||||||
private boolean analog;
|
private boolean analog;
|
||||||
|
private boolean invert;
|
||||||
|
private boolean trigger;
|
||||||
|
|
||||||
private ControllerComponent(String name, boolean analog) {
|
private ControllerComponent(String name, boolean analog) {
|
||||||
this.label = new JLabel(name);
|
this.label = new JLabel(name);
|
||||||
this.mapButton = new JButton();
|
this.mapButton = new JButton();
|
||||||
this.mapButton.setName(this.name());
|
this.mapButton.setName(this.name());
|
||||||
|
this.invertBox = new JCheckBox("Invert");
|
||||||
|
this.invertBox.setName(this.name());
|
||||||
|
this.triggerBox = new JCheckBox("Trigger");
|
||||||
|
this.triggerBox.setName(this.name());
|
||||||
|
this.triggerBox.setToolTipText("If this component should act as a trigger.");
|
||||||
this.analog = analog;
|
this.analog = analog;
|
||||||
|
this.invert = false;
|
||||||
|
this.trigger = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public JLabel getLabel() {
|
public JLabel getLabel() {
|
||||||
@@ -32,7 +44,31 @@ public enum ControllerComponent implements Serializable {
|
|||||||
return mapButton;
|
return mapButton;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public JCheckBox getInvertBox() {
|
||||||
|
return invertBox;
|
||||||
|
}
|
||||||
|
|
||||||
|
public JCheckBox getTriggerBox() {
|
||||||
|
return triggerBox;
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isAnalog() {
|
public boolean isAnalog() {
|
||||||
return analog;
|
return analog;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean isTrigger() {
|
||||||
|
return trigger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void trigger(boolean isTrigger) {
|
||||||
|
trigger = isTrigger;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void invert(boolean invert) {
|
||||||
|
this.invert = invert;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean invert() {
|
||||||
|
return invert;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -107,13 +107,24 @@ public class Gamepad {
|
|||||||
ControllerComponent contComp = config.getControllerComponent(comp);
|
ControllerComponent contComp = config.getControllerComponent(comp);
|
||||||
if (contComp != null) {
|
if (contComp != null) {
|
||||||
if (contComp.isAnalog()) {
|
if (contComp.isAnalog()) {
|
||||||
handleAnalog(contComp, value);
|
handleAnalog(contComp, sanitizeValue(contComp, value));
|
||||||
} else {
|
} else {
|
||||||
handleButtons(contComp, value);
|
handleButtons(contComp, sanitizeValue(contComp, value));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private float sanitizeValue(ControllerComponent contComp, float value) {
|
||||||
|
float sanitized = value;
|
||||||
|
if (contComp.invert()) {
|
||||||
|
sanitized = -sanitized;
|
||||||
|
}
|
||||||
|
if (contComp.isTrigger()) {
|
||||||
|
sanitized = (sanitized + 1)/2;
|
||||||
|
}
|
||||||
|
return sanitized;
|
||||||
|
}
|
||||||
|
|
||||||
private void toggle(short button, boolean press) {
|
private void toggle(short button, boolean press) {
|
||||||
if (press) {
|
if (press) {
|
||||||
inputMap |= button;
|
inputMap |= button;
|
||||||
@@ -123,8 +134,6 @@ public class Gamepad {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void handleAnalog(ControllerComponent contComp, float value) {
|
private void handleAnalog(ControllerComponent contComp, float value) {
|
||||||
|
|
||||||
|
|
||||||
switch (contComp) {
|
switch (contComp) {
|
||||||
case LS_X:
|
case LS_X:
|
||||||
leftStickX = (short)Math.round(value * 0x7FFF);
|
leftStickX = (short)Math.round(value * 0x7FFF);
|
||||||
@@ -139,10 +148,10 @@ public class Gamepad {
|
|||||||
rightStickY = (short)Math.round(value * 0x7FFF);
|
rightStickY = (short)Math.round(value * 0x7FFF);
|
||||||
break;
|
break;
|
||||||
case LT:
|
case LT:
|
||||||
leftTrigger = (byte)Math.round((value + 1) / 2 * 0xFF);
|
leftTrigger = (byte)Math.round(value * 0xFF);
|
||||||
break;
|
break;
|
||||||
case RT:
|
case RT:
|
||||||
rightTrigger = (byte)Math.round((value + 1) / 2 * 0xFF);
|
rightTrigger = (byte)Math.round(value * 0xFF);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
System.out.println("A mapping error has occured. Ignoring: " + contComp.name());
|
System.out.println("A mapping error has occured. Ignoring: " + contComp.name());
|
||||||
|
|||||||
@@ -23,6 +23,10 @@ public class GamepadMapping implements Serializable {
|
|||||||
return mapping.get(comp.getIdentifier().getName());
|
return mapping.get(comp.getIdentifier().getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void removeMapping(Component comp) {
|
||||||
|
mapping.remove(comp.getIdentifier().getName());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the mapping for the specified component.</br>
|
* Gets the mapping for the specified component.</br>
|
||||||
* NOTE: Use sparingly takes O(N) time.
|
* NOTE: Use sparingly takes O(N) time.
|
||||||
|
|||||||
Reference in New Issue
Block a user