InputManager for Key clicks - JME3 bug needs fixing

Does anyone else experience this.
You click ‘1’ and the input Manager will send 3 events.

1st. ‘1’ pressed
2nd. ‘1’ pressed
3rd. ‘1’ not pressed

Every key that is clicked, I get it twice.I looked inside the input manager, and my class is not registered more than once, it actually got 3 keys events for every key that is clicked?

Does anyone else experience this.

Also, I wanted to make sure I’m doing this right. Or understanding is correct.

If 1 have baseApp running that listens to key inputs, when it gets a pressed key, it ignores it and waits for the “not” pressed event. When the baseApp handles the ‘1’ click “not” pressed. I mark the event as “CONSUMED” and that baseApp launched another class that takes input and registers to listener class. As soon as it runs it get the “NOT” pressed key ‘1’ event. It was not consumed, in that time frame.

So I’m assuming the event is not consumed, I see code in there to not call an event if it is consumed. So I’m thinking I’m doing something wrong?

Any help.

ps>
I think what is happening, is it launched the new GUI for input and registers as a listener before the event is done looping??? Is that possible.

Do you a sample code that shows the behavior you are describing? I do not seem to have that issue on my end and perhaps I can be of help.

That is what I was about to do. writ something simple to see if I am getting the double hit on everything.

Yep. No I don’t see the double key when writing a simple application. Only under my big project.

Are both applications using the same version of lwjgl?

…I have a vague recollection that the event delivery changed slightly (and surprisingly) in lwjgl3. Like, you’ll get one pressed event without the ascii character and one with… where as lwjgl2 would give you just one with the ascii code in it. (Useful if you want to implement typing into an edit field.)

@pspeed you’ve jogged my memory - as I recall, lwjgl3 has separate “key events” and “character events,” and my guess is the observed behavior is jME’s way of mapping that convention into jME input events.

Edit: Obsidian UI is my latest project dealing with this. You can see how I differentiated between “standard” key presses and character keys in the Obsidian-jME embedding library here:

if(evt.getKeyChar() != 0 && evt.isPressed()){
  uiEvt = ui.getInput().fireCharacterEvent(new char[]{evt.getKeyChar()});
  consumeEvent(evt, uiEvt);
 }

uiEvt = ui.getInput().fireKeyEvent(mapKey(evt.getKeyCode()), evt.isPressed() || evt.isRepeating());
consumeEvent(evt, uiEvt);

Note that based on the above, this code probably ends up effectively duplicating some key press start events since it fires the key event for both the character event and a “plain” key event from lwjgl.

2 Likes

Lwjgl3 also has 3 events for each key. Down, pressed and release to capture long presses.

2 Likes

I am using lwjgl3 for my projects. I was usually getting duplicates when I registered an action listener twice by accident and not removing them. To avoid that in Depthris I went to have two implementations of RawInpulListener (one for menu and one for game) and I made sure which needed to be active.

I have not tested using Lemur’s input system

Thanks, I don’t use a JME gui system. I have my own, and debugging through the code I see that many classes where doing rawinputListener when it was not required. I started doing the “Gui” vs “Game” option also. There are only 2 “Game” gui that need raw input because it is text input, but the rest are just menu items.

I removed rawInputListeners from the game but this is what I see.

ClassABaseApp implements RawInputListener

	@Override
	public void onKeyEvent(KeyInputEvent evt)
	{
		System.out.println("got key "+(""+evt.getKeyChar())+ " pressed "+evt.isPressed());
		
//    	if (moduleState != ModuleState.RUNNING)
//    		System.out.println("State "+moduleState);
		
	evt.setConsumed();
    	if (moduleState != ModuleState.PAUSED)
    	{
    		if (!evt.isPressed())
    		{
    			
    			key = ""+evt.getKeyChar();
    System.out.println("key "+key+".");

    			switch (evt.getKeyCode())
    			{
    				case 1:
    					key = "ESC";
    					break;
    				case 56:
    					key = "ALT";
    					break;
    				case 208:
    					key = "down";
    					break;
    				case 205:
    					key = "right";
    					break;
    				case 203: 
    					key = "left";
    					break;
    				case 200:
    					key = "up";
    					break;
    				default:
    					key = new String( ""+evt.getKeyChar()).toUpperCase();
    					break;
    			}
        		GameEngine.getGameLogic().playSound("keyclick", false);
        		updateModule();
//        		key = "";
    		}
    	}
	}

OnKeyEvent above receives 3 input events>
‘1’ pressed
Key(CODE=57, PRESSED)
‘1’ not pressed

Not sure what the 2nd event is, but it shows up all the time.

onKeyEvent processes ‘1’ not pressed. launches another gui through the ‘updateModule()’ function. This gui uses (below) for keyMappings

public class SelectItemMenuControl
{
	public void init()
	{
		GuiGlobals.inputManager.addMapping('1', new KeyTrigger(KeyInput.KEY_1));
		GuiGlobals.inputManager.addListener(this, '1');
       }
	public void onAction(String key, boolean isPressed, float tpf)
	{
        ......
    }
}

It register keyMappings of a key ‘1’. Same key.

Then it attaches this gui to the guiNode. and the ClassABaseApp removes itself from the RawInputListener (verified it stops receiving input)

But as soon as JME starts to process this new GUI. it immedietaly calls onAction() with a
‘1’ not pressed

So I end up with in first class

‘1’ pressed
Key(CODE=57, PRESSED)
‘1’ not pressed

then
‘1’ not pressed

in second class from hitting the key once.

It is like the ‘1’ not pressed is not being “CONSUMED”, i have the ‘evt.setConsumed();’ set in the first class and the gui would not get launched without hitting the consumed function called and verified that it changes the consumed to true.

This has allowed me to see a bunch of extra things happening that not needed to be happening behind closed doors.

Thanks for you input

I did a test, I changed

ClassABaseApp  extends BaseAppState implements ActionListener
{
	protected void intModule(Globals.MODULES module, Globals.GameStates state) 
	{
   // gets called on startup of the appState (it is called from initialize, hang over from my engine to JME)
		addKeyListening("1", new KeyTrigger(KeyInput.KEY_1));
		addKeyListening("2", new KeyTrigger(KeyInput.KEY_2));
		addKeyListening("3", new KeyTrigger(KeyInput.KEY_3));
		addKeyListening("4", new KeyTrigger(KeyInput.KEY_4));
		addKeyListening("5", new KeyTrigger(KeyInput.KEY_5));
		addKeyListening("6", new KeyTrigger(KeyInput.KEY_6));
		addKeyListening("7", new KeyTrigger(KeyInput.KEY_7));
		addKeyListening("8", new KeyTrigger(KeyInput.KEY_8));
		addKeyListening("9", new KeyTrigger(KeyInput.KEY_9));
		addKeyListening("N", new KeyTrigger(KeyInput.KEY_N));
		addKeyListening("Y", new KeyTrigger(KeyInput.KEY_Y));
		addKeyListening("SPACE", new KeyTrigger(KeyInput.KEY_SPACE));
		addKeyListening("ESC", new KeyTrigger(KeyInput.KEY_ESCAPE));
    }

	@Override
	public void onAction(String name, boolean isPressed, float tpf)
	{
		System.out.println("name="+name+". pressed "+isPressed);
		
	}

}

I changed the ClassABaseApp from RawInputListener to ActionListener.

When ClassA processes the input and launched the new GUI. No extra input. it is all related to RawInputListener, I’m assuming is not clearing out the last input. I need to verify what is happening with this.

But it works not using the RawInputListener.

Doing more debugging this is what I’m seeing for LWJGL3 library.

Inside the GlfwKeyInput class , it handles the callbacks for the inputs.

    private void initCallbacks() {
        glfwSetKeyCallback(context.getWindowHandle(), keyCallback = new GLFWKeyCallback() {
            @Override
            public void invoke(final long window, final int key, final int scancode, final int action, final int mods) {

                if (key < 0 || key > GLFW_KEY_LAST) {
                    return;
                }

                int jmeKey = GlfwKeyMap.toJmeKeyCode(key);

                final KeyInputEvent event = new KeyInputEvent(jmeKey, '\0', GLFW_PRESS == action, GLFW_REPEAT == action);
                event.setTime(getInputTimeNanos());

                keyInputEvents.add(event);
            }
        });

        glfwSetCharCallback(context.getWindowHandle(), charCallback = new GLFWCharCallback() {
            @Override
            public void invoke(final long window, final int codepoint) {

                final char keyChar = (char) codepoint;

                final KeyInputEvent pressed = new KeyInputEvent(KeyInput.KEY_UNKNOWN, keyChar, true, false);
                pressed.setTime(getInputTimeNanos());

                keyInputEvents.add(pressed);

                final KeyInputEvent released = new KeyInputEvent(KeyInput.KEY_UNKNOWN, keyChar, false, false);
                released.setTime(getInputTimeNanos());

                keyInputEvents.add(released);
            }
        });
    }

So When I click on ‘1’ on the keyboard. JME3 is causing the issue, not LWJGL3. LWJGL3, sends 2 callbacks to public void invoke(final long window, final int key, final int scancode, final int action, final int mods) like it should, with the “action” changed, one with a value of 1 = GLFW_PRESS and then makes another call to the callback with a “action” value of 0 = GLFW_RELEASE That is all that should happen.

But JME register for glfwSetCharCallback and so after the first callback to glfwSetKeyCallback JME gets 1 callback to glfwSetCharCallback. But inside that code it add 2 more InputEvents and add them to the arraylist. So At the end of a person hitting the keyboard once for ‘1’. There is now 4 inputs inside JME3.

This callback and JME3 ending up putting 4 inputkey events inside is what is causing this issue for JME. In my case, the ClassA gets 3 of them, and then unregister as a listener and the new gui starts to listen before the 4 one comes in, and so the last 1 is sent off to the new GUI, it thinks it got a keyboard input.

But it didn’t. for ever 1 key pressed there ARE 4 EVENTS.

This is an issue for JME3. I believe this is due to the new input callback requirements in LWJGL3 library.

Now I will let you guys figure this out. I see 2 options to solve this issue.

  1. Have 2 input key events, one for RawInputs, and the second for ActionListeners, Then inside the InputManager class in the function of private void processQueue(), when it process the key events and calls the Raw Listeners and then the Action Listeners, it could be based off 2 array list. Raw Listeners get their input from glfwSetKeyCallback and Action Listeners get theirs from glfwSetCharCallback.
  2. Could remove the glfwSetCharCallback and inside the glfwSetKeyCallback, assign the keyChar to the KeyInputEvent. Then the second callback is not needed at all.

I think one of those two ways would work. What are your thoughts?

Give me a direction and I can fix it. I’m going to fix it inside my special JME3 build that I’m using INFO: Running on jMonkeyEngine 3.7.0-SNAPSHOT, would like to have it fixed inside the main JME.

Thanks,

2 Likes

…and every text editor in every GUI everywhere would break.

No, it would not break. It would not appear to be different at all to any JME3 code.

Here is a screen shot from a text game I wrote in JME3, using the new option 2. I just typed in “LOOK” (its a text adventure game). and you can see the look appears in the scroll text and the response to the command showed up.

Also, I didn’t test it in Lemur, this was toneguid library, that no one supports anymore.

Option 2 works because it ends up loading inputQueue inside InputManager like before, there is not visible difference to any part of JME3 because it is put into the key event array like before.

But, I know this community is not very open to suggestions and this is why I created my own JME build. and Now I see from above several people have had to do work around because of the issue in JME3-LJWGL3 library.

That is fine, if you the community wants to keep the bug live then that is fine. I’ve had this issue for every suggestion along the way. Getting LJGL3 as a standard, adding support for multiple devices, RawInputListener issues for adding Multiple listener of the same class (while ActionListener check and doesn’t assign again,) everyone said it should check and the programmer just needs to know what they are doing even though ActionListener checks and I’ve decided to not fight the people that want JME to reflect their own and only view of the engine.

Just trying to help out the community, I know there is a handful of people and everyone else has little say in bug or features.

Thanks for listening,

If anyone wants to have their JME3 bugs free of Phantom inputs, you should correct this for your build.

You implied RawInputListener wouldn’t have character codes anymore.

If your change won’t break RawInputListener-using code then I may have misunderstood this:

Edit: in other words, if RawInputListeners still get character values then I didn’t understand what you are saying.

This is all I did. I have to run a ton of test, but I ran my CrossyRoads, Alternate Reality, Pyramid 2000(text adventure) and Time Pilot.

I’ll go through the examples and make sure it doesn’t break them.

RawInputListener uses the jmeKey and not the character code. My change updates the keyChar to reflect the keyChar, before JME was putting null basically in keyChar.

The glfwSetCharCallback didn’t use the rawcode, it converted it to keyChar and passed that in.

Look at it and let me know if it looks like something you want.

    private void initCallbacks() {
        glfwSetKeyCallback(context.getWindowHandle(), keyCallback = new GLFWKeyCallback() {
            @Override
            public void invoke(final long window, final int key, final int scancode, final int action, final int mods) {

                if (key < 0 || key > GLFW_KEY_LAST) {
                    return;
                }

                final char keyChar = (char) key;
                int jmeKey = GlfwKeyMap.toJmeKeyCode(key);

                final KeyInputEvent event = new KeyInputEvent(jmeKey, keyChar, GLFW_PRESS == action, GLFW_REPEAT == action);
                event.setTime(getInputTimeNanos());

                keyInputEvents.add(event);
            }
        });

//        glfwSetCharCallback(context.getWindowHandle(), charCallback = new GLFWCharCallback() {
//            @Override
//            public void invoke(final long window, final int codepoint) {
//
//                final char keyChar = (char) codepoint;
//
//                final KeyInputEvent pressed = new KeyInputEvent(KeyInput.KEY_UNKNOWN, keyChar, true, false);
//                pressed.setTime(getInputTimeNanos());
//
//                keyInputEvents.add(pressed);
//
//                final KeyInputEvent released = new KeyInputEvent(KeyInput.KEY_UNKNOWN, keyChar, false, false);
//                released.setTime(getInputTimeNanos());
//
//                keyInputEvents.add(released);
//            }
//        });
    }

2 Likes