How to update a gameobject at other clients

Hi,

Can anybody give me a small C# sample of how to update a network object (where the original can be shifted around by a fps).

I came as far as the following C# code (MyAvatar is a cube packed as a prefab and located in the Resources directory):

using UnityEngine;
using System;
using System.Diagnostics;
using System.Collections;

public class ServerScript : MonoBehaviour {

	public void OnServerInitialized() {
		GameObject avatar = Resources.Load("MyAvatar") as GameObject;

		Vector3 baseposition = new Vector3(10,1,10);

		GameObject go= Network.Instantiate(avatar, baseposition, transform.rotation, 0) as GameObject;
		go.name = "server_object";
	}
	
	public void OnConnectedToServer() {
		networkView.RPC( "RequestOwner", RPCMode.Server ); 

		GameObject avatar = Resources.Load("MyAvatar") as GameObject;
		
		Vector3 baseposition = new Vector3(10,1,10);

		GameObject go = Network.Instantiate(avatar, baseposition, transform.rotation, 0) as GameObject;
		go.name = "client_object";
	}
	
	public void OnGUI () {
		if (GUI.Button (new Rect (10,10,100,50), "Start server")) {
				
			Network.InitializeServer(32, 25000);			
		}
		if (GUI.Button (new Rect (10,70,100,50), "Connect")) {

			Network.Connect("127.0.0.1", 25000);		
		}
	}
}

When i execute this code each client gets it’s own avatar but it can only be pushed in its own world (remote does not show up a thing).

From the documentation, it seems like Network.Instantiate sets up everything and does the synchronizing automatically.

It does, but you still need to define what data you want to send over the net…
Read this and focus on the part called “Choosing data to send”.

The network view has to observe something to sync it with the other clients. So if you want to transfer the position and rotation to the rest of the clients, you can probably just set up the Observed property to observe the Transform component of MyAvatar. Oh, and set the “State Synchronization” property to either “Reliable Delta Compressed” or “Unreliable”.
If I’m not mistaken, this will work as is.

If you want to do more fancy stuff, you can set the Observe property to watch a script and then from inside the script use OnSerializeNetworkView():

void OnSerializeNetworkView(BitStream stream, NetworkMessageInfo info)
	{
		// Always send transform (depending on reliability of the network view)
		if (stream.isWriting)
		{
			Vector3 pos = transform.localPosition;
			Quaternion rot = transform.localRotation;
			stream.Serialize(ref pos);
			stream.Serialize(ref rot);
		}
		// When receiving, buffer the information
		else
		{
			// Receive latest state information
			Vector3 pos = Vector3.zero;
			Quaternion rot = Quaternion.identity;
			stream.Serialize(ref pos);
			stream.Serialize(ref rot);
			transform.localRotation = rot;
			transform.localPosition = pos;
		}
	}

Also, check out the network examples.

Thanks a lot!

The Network View did the trick. I set the Observed property to Transform and that did it.

And I noticed that I created the cube inside the OnServerInitialized() which was wrong!

After calling it (btw why is it called twice?) the client can connect but the remote object is not created (at least so it seemed). If i create it later (for now in a GUIbutton) all works as expected.

Last question: how to set Observed to a scripts’ .

Code sofar:

  1. OnSerializeNetworkView is not used,
  2. Observed is set to Transform of the Prefab,
  3. I’m also not sure about the ownercode working. The Jscript sample had @RPC but that does not work in C#:
using UnityEngine;
using System;
using System.Diagnostics;
using System.Collections;

public class ServerScript : MonoBehaviour {

	private NetworkPlayer networkPlayer;
	private int currentPlayerID;
	private GameObject avatar;
	private GameObject go;

//Debug Code
	
    public static DateTime FirstTick = DateTime.Now;
	
	public static void Trace() {
		_Trace(new StackFrame(1, true).GetMethod().ToString(), null);
   }
	
	public static void Trace(string msg) {
		_Trace(new StackFrame(1, true).GetMethod().ToString() , msg);
   }
	
	public static void _Trace(string method, string msg) {
		string Duration = ((TimeSpan)(DateTime.Now - FirstTick)).ToString();
		Duration = Duration.Substring(0, Duration.Length - 4); //Removed some detail (msec resolution).
		if (msg==null || msg.Length==0) {
			print (Duration + ": "+method);
		} else {
			print (Duration + ": "+method + " - " +msg);
		}
   }

//Debug Code
	
	// Use this for initialization, called after Awake().
	public void Start () {
		Trace();
	}
	
	// Update is called once per frame.
	public void Update () {
	  //
	}

	//Put global code here (whats put outside of functions in JScript. 
	public void Awake () {
		Trace();
	}
	
	public void OnServerInitialized() {
		Trace();
	
		//Where does this end up?
		//UnityEngine.Debug.Log("OnServerInitialized");
	}
	
	public void OnConnectedToServer() {
		Trace();
	}
	
	public void OnGUI () {
		if (GUI.Button (new Rect (10,10,100,50), "Start server")) {
			Trace("Start Server");
				
			Network.InitializeServer(32, 25000);			
		}
		
		if (GUI.Button (new Rect (10,70,100,50), "Connect")) {
			Trace("Connect");

			Network.Connect("127.0.0.1", 25000);		
		}

		if (GUI.Button (new Rect (10,140,100,50), "Create")) {
			Trace("Create");

			GameObject avatar = Resources.Load("MyAvatar") as GameObject;

			Vector3 baseposition = new Vector3(10,1,10);

			go = Network.Instantiate(avatar, baseposition, transform.rotation, 0) as GameObject;
			go.name = "server_object";
		}

		if (GUI.Button (new Rect (10,200,100,50), "Move")) {
			Trace("Move");

			go.transform.Translate(0,0,1);
		
			//Network.Connect("127.0.0.1", 25000, "HolyMoly");		
		}
	}
	
	public void RequestOwner(NetworkMessageInfo info)  { 
		Trace();
		networkView.RPC( "SetOwner", info.sender, info.sender ); 
	} 

	public void SetOwner(NetworkPlayer a_owner) 	{ 
		networkPlayer = a_owner; 
		currentPlayerID = int.Parse( networkPlayer.ToString() ); 
		
		Trace("Current PlayerID = "+currentPlayerID.ToString());
	}
	
	void OnSerializeNetworkView(BitStream stream, NetworkMessageInfo info) { 
      // Always send transform (depending on reliability of the network view) 
      if (stream.isWriting) 
      { 
         Vector3 pos = transform.localPosition; 
         Quaternion rot = transform.localRotation; 
         stream.Serialize(ref pos); 
         stream.Serialize(ref rot); 
      } 
      // When receiving, buffer the information 
      else 
      { 
         // Receive latest state information 
         Vector3 pos = Vector3.zero; 
         Quaternion rot = Quaternion.identity; 
         stream.Serialize(ref pos); 
         stream.Serialize(ref rot); 
         transform.localRotation = rot; 
         transform.localPosition = pos; 
      } 
   }  
}

Glad it helped you.
About the OnServerInitialized() running twice - it shouldn’t. I think you have some confusion here about instantiation, and because of it you think it runs twice.
Network.Instantiate() creates your game object on all connecting clients. It also buffers it so that new clients will receive this object too.
So when you had the Network.Instantiate both in OnServerInitialized() and in OnConnectedToServer(), it ran once when you started the server, and then once again when the client connected to it. Each time, it created a new instance of the object. When the client connected, it first received the buffered call from the server to create the object (1st instance of the object), and then it ran the code in OnConnectedToServer() and created a new instance both on the client, and on the server. It also buffered it, so the next client that will connect will receive two buffered calls and will create one of its own.

About ownership - in Unity, when you use Network.Instantiate the owner of that object is the client that ran the function. Lets say you connect to a server and you need to create your character. You (as a client) run Network.Instantiate and it will create an instance of your character on all the other clients and on the server. From now on, whenever someone will run networkView.owner or networkView.isMine, the owner of that object will be you (the client that created it). I’m not sure about this, but I think you cannot change ownership from a script. There are other solutions for that (that don’t use Network.Instantiate).

About RPC - “@RPC” is the syntax in Javascript. in C# it’s “[RPC]”.

And the reason OnSerializeNetworkView() is not used in your code is because your network view is observing the Transform component. If you want to use it, either change the observing property of the existing network view to observe the script that has the OnSerializeNetworkView() function in it, or create a new network view on the same game object that observes it. Obviously, the sample script I posted already handles position and rotation, so you don’t need both the OnSerializeNetworkView() and the observing of Transform. They both do the same thing, just in two different ways. (Oh, and the script doesn’t handle the scale as it is).