Hi, I’m trying to have a UI panel that is displayed to two players, when a player clicks on a “ready” button, a toggle is marked for both of them (there is one for each player). In order to know which player was the one to click on the button, I need to get the clientid from the ServerRpcParams, however when I add this parameter to the method’s signature, I can’t use the method as a button’s onClick Listener method.
I tried to get around this by having a “middle method”, which looks like this:
However this doesn’t work either. Using the debugger I see that the ServerRpc method never gets called.
Is there a better way to know which client clicked on button which calls a ServerRpc method?
When player join, the client calls ServerRpc telling his local client id, which is unique. The server receives his id and does whatever it needs to do, then send a “callback” to the client that called it using ClientRpc parameters.
Don’t know if this is the best way to achieve this, but I hope this might help somehow.
Something should be happening, do you have an EventSystem in the scene? It might be worth adding a listener in the button inspector just to check click events are firing.
Thanks guys, it was indeed spawning the object that was missing…
Now I have a different issue since the object might be instantiated on clients which aren’t the host, but I’ll try to find a solution if needed open a different thread later since it’s a different subject
What I do is to have the object attached to the UI not be spawned, but instead call Rpcs on a different spawned object when necessary (or call functions on other objects which then call Rpcs). That way, the UI itself does not need to be a networkobject.
I’ll throw in what I do as well for more food for thought. I have associated UI elements on separate canvases and at the start of each scene a Gui manager scans in each canvas into its own UI manager, then the UI elements on each canvas are scanned, stored in lists in each UI manager and initialised with starting values.
For updating the UI I have a single update controller that has a list of update services which creates a list of interface update details that the controller passes on to the Gui manager to pass on to the correct UI manager which updates the canvas elements.
Essentially the idea is to have the UI handled in a general-purpose way, but it is rather complicated as you would expect and has a lot of limitations.
So I managed to get the UI panel to properly be spawned, and the issue remains.
When I use the debugger, both as a client and as a host, I see that the ReadyClicked() Method is being called, but it never gets into the PlayerSerReadyServerRpc().
I first tried without changing anything in the code from what I copied here, the only changes were in other classes, and then also tried with an onClick like this, skipping the ReadyClicked() method:
Which is similar to the code I use on my buttons for joining as a client or hosting:
startClientButton.onClick.AddListener(() =>
{
if (NetworkManager.Singleton.StartClient())
{
Debug.Log("Client started");
}
else
{
Debug.Log("Client could not be started");
}
});
However, that didn’t seem to work as well. The debugger never goes into the PlayerSerReadyServerRpc() method.
I’m not exactly sure where you are with this so I’ll start from the beginning. I am making some assumptions of what happens when a rpc isn’t working as I’ve not had to test it.
Check the onClick event of the button is working, I think you’ve already established it is. To make it clearer for testing use a real function as a listener rather than a lambda.
Create the server rpc with only a debug log line and call that from your listener function. If the rpc works the output will appear on the server, if it’s not working it will appear on the client. If there’s no output the rpc isn’t being called.
You can’t really debug a working server rpc as the code is executed on the server. What you will likely see is the debugger go into the NetworkManager as it will need to send a message to the server to execute the rpc there. If the rpc isn’t working the debugger should at least step through the code on the client and display the output.
The script is attached to the spawned UI panel, it contains the button as a member and adds the OnClick listener in its OnEnable method.
Does this mean the button must also be a spawned networkobject in order to call Rpcs from its OnClick?
I’m running 2 unity clients, one as a host and one as a client, and have a debugger attached to one of them, I can debug both behaviors because I can switch which unity client is the host and which one is the client each time I debug.
From what I saw now after making the ServerRpc only have a debug log line, without the
ServerRpcParams as well in the signature, it still doesn’t get called. However I also just noticed that when the client clicks on the button, I’m getting this warning in the console of the host:
[Netcode] Deferred messages were received for a trigger of type OnSpawn with key 0, but that trigger was not received within within 1 second(s).
UnityEngine.Debug:LogWarning (object)
Unity.Netcode.NetworkLog:LogWarning (string) (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Logging/NetworkLog.cs:28)
Unity.Netcode.DeferredMessageManager:PurgeTrigger (Unity.Netcode.IDeferredMessageManager/TriggerType,ulong,Unity.Netcode.DeferredMessageManager/TriggerInfo) (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Messaging/DeferredMessageManager.cs:98)
Unity.Netcode.DeferredMessageManager:CleanupStaleTriggers () (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Messaging/DeferredMessageManager.cs:83)
Unity.Netcode.NetworkManager:OnNetworkPostLateUpdate () (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Core/NetworkManager.cs:1588)
Unity.Netcode.NetworkManager:NetworkUpdate (Unity.Netcode.NetworkUpdateStage) (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Core/NetworkManager.cs:1479)
Unity.Netcode.NetworkUpdateLoop:RunNetworkUpdateStage (Unity.Netcode.NetworkUpdateStage) (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Core/NetworkUpdateLoop.cs:185)
Unity.Netcode.NetworkUpdateLoop/NetworkPostLateUpdate/<>c:<CreateLoopSystem>b__0_0 () (at Library/PackageCache/com.unity.netcode.gameobjects@1.0.1/Runtime/Core/NetworkUpdateLoop.cs:268)
I tried googling the first line here but didn’t find anything that could help me. Curiously there was one post on January where you were trying to help someone with this warning message as well, but it seems his issue was different than mine.
I’m sure I’ve seen that message after updating Netcode, and it had me stumped for a while. Try deleting the PackageCache folder and Reimport All on the Assets folder. It shouldn’t be necessary but you could try removing and re-adding the Netcode package.
I tried deleting the PackageCache folder and Reimport All on the assets folder, but the same message appears.
I also tried to remove adding the method to the button’s OnClick by script and manually add it through the unity editor, same thing happens.
It’s a shame that there’s pretty much no info about this “[Netcode] Deferred messages were received for a trigger of type OnSpawn with key 0, but that trigger was not received within within 1 second(s).” warning message. I have a feeling that it’s related to my issue about not triggering the ServerRpc.
Netcode 1.1.0 was released today, you could give that a go.
The error suggests to me the rpc is being sent to the server but its network manager isn’t able to deal with it for some reason. Without delving into the source code really only a Unity developer can tell us more. I’d recommend creating a minimal test project with just a player and try testing a rpc through that. Alternatively if you make your project available I can take a look at it.
So I finally got it to work… But in a really cumbersome way.
First of all, on a test project I tried to do basically the same thing that I tried before, with a panel containing a button and an onclick method that is a ServerRpc, and got no errors. I tried to figure out what was different from my main project but couldn’t find anything.
Then on my main project, what I did was make the button its own prefab, gave it its own script, made it networked, and instantiated & spawned it right after the main panel was instantiated, then set the main panel as its parent, and viola, the onclick ServerRpc method of the new button works smoothly!
I would have much preferred to not have the button as its own prefab tho, so I hope I could find a fix for this issue later on. For now I will have to continue working on my project with this workaround.