PhotonNetwork, syncing objects?

Trying to build controls that will allow a box within the scene to be moved by either the Host or Client. The box is different from the players because it has already been placed in the scene and is not being instantiated. I’m hoping to avoid instantiating the box, as it represents vehicles and other random objects that have been placed around the map.

::Current Script::

using UnityEngine;
using System.Collections;

public class tellToMove : MonoBehaviour {
    bool firstUpdate;
    Vector3 realPosition;

    void Update () {
        if (Input.GetKey (KeyCode.Keypad8)){
            realPosition += Vector3.forward;
        }
        if (Input.GetKey (KeyCode.Keypad2)){
            realPosition -= Vector3.forward;
        }

        if (PhotonNetwork.inRoom){
            if(!firstUpdate){
                firstUpdate = true;
                realPosition = transform.position;
            }

            GetComponent<PhotonView>().RPC ("moveMe", PhotonTargets.AllBuffered, realPosition);
        }
    }

    [RPC]
    public void moveMe(Vector3 realPosition){
        //print ("moveMe");
        transform.position = Vector3.Lerp (transform.position, realPosition, Time.fixedDeltaTime);
    }

    public void OnPhotonSerializeView(PhotonStream stream, PhotonMessageInfo info){
        if(stream.isWriting){
            //print ("Writing");
            stream.SendNext(transform.position);
        }
        else{
            //print ("Reading");
            realPosition = (Vector3)stream.ReceiveNext();
        }
    }
}

With this, if I observe the gameobject transform with the attached photonView I’m able to move the box on either machine but it seems to be fighting with data ((jumps back and forth strangely)). If I observe the above script with photonView I can still move the box on either machine but now it only has a full range of motion with the Host, it seems the client can only move it x-distance from where the host left it, and both seem to jump strangely as before…

Unfamiliar with networking, I’m not sure how best to describe what I’m viewing, hopefully someone has an idea.

OK I think with photon network if you use the photon view then the transform should be shared amongst players, so any inputs that affect a transform will be automatically networked and updated.

Also if the move amount is per the Update() then you should scale it by the Time.deltaTime variable not the fixedDelta time. DeltaTime is per a frame, fixedDelta time is per a physics tick (about 6 times faster).

I’m still having issues with this actually,
I can control the box with both players, now it seems that if machineA is moving the box machineA sees the box move while machineB sees machineA’s player move to where the box would be and follow along it’s path of control dictated by machineA, while their box (machineB) remains still. And the opposite of this if you’re controlling with machineB.

In this video link, you can see everything’s good with the players, until I move the box

I feel i’m closer but still missing something pretty important. Changing the photonView ownership is probably where I’ve gone wrong, how to fix this…I do not know. In all honesty I thought this would be more straight forward? This seems like a pretty generic thing to want to do.

using UnityEngine;
using System.Collections;

public class tellToMove : Photon.MonoBehaviour {
    bool firstUpdate;
    Vector3 realPosition;
   
    void Update () {
        if (PhotonNetwork.inRoom){
            if(!firstUpdate){
                firstUpdate = true;
                realPosition = transform.position;
            }

            if (Input.GetKey (KeyCode.Keypad8)){
                photonView.ownerId = PhotonNetwork.player.ID;
                realPosition += Vector3.right;
            }
            if (Input.GetKey (KeyCode.Keypad2)){
                photonView.ownerId = PhotonNetwork.player.ID;
                realPosition -= Vector3.right;
            }

            transform.position = Vector3.Lerp (transform.position, realPosition, Time.deltaTime);
        }
    }

    void OnPhotonSerializeView(PhotonStream stream, PhotonMessageInfo info){
        if (stream.isWriting){
            stream.SendNext(realPosition);
        }
        else{
            realPosition = (Vector3)stream.ReceiveNext();
        }
    }
}

No I can’t think why you would need something that multiple players can control at once, unless your doing a Ouija board. :wink:

I would expect that most game objects can only be controlled by the closest player and once one player is in control another cannot take it over.

You could change it so that both players don’t control the position but change the physics velocity.

I would drop using the OnPhotonSerializeView.

Just save realposition when you receive the RPC and then Lerp it locally in Update. If you do so remember to update its position for newly joined players, by sending it from the master client to the newly joined client.

However the problem with changing the position from more players is that the cube will likely not be in the exact same position on all clients as they will receive the messages at different times, so when a lot of changes from the players occur it might start looking like some crazy movement is going on, just saying :smile:

Ha yeah Ouija board would be pretty awesome though

I’m open to anything really, this is my first attempt at networking, I’ve only been able to find tutorials on moving players, none for moving objects unfortunately. Basic get in/out of a vehicle, make sure my vehicle movements are updated across server. I could be attacking it all wrong, i don’t even know. but since the player moves around fine I thought I’d just convert that code to work for the box. Quickly realized it doesn’t, been trying things since

Sorry guy’s I’m clearly not picking up what you’re laying down…

This is exactly what I’m doing:

Open New scene
- Create a cube
- Place ‘PhotonView.cs’ on cube
- Place ‘viewMe.cs’ on cube
- Drag/Drop ‘viewMe.cs’ into the Observe field of ‘PhotonView.cs’
- Update the ‘viewMe.cs’ script with a script that’s placed on the player called ‘setNumber.cs’ . ((This should only be enabled on 1 player at a time, for now…I just want to see something work)).

viewMe.cs
(This is on the cube that’s in each players scene, not instantiated)

```csharp
**using UnityEngine;
using System.Collections;

public class viewMe : Photon.MonoBehaviour{
public int setThisNumber;
}**
```

setNumber.cs
(This is on the instantiated players)

using UnityEngine;
using System.Collections;

public class setNumber : Photon.MonoBehaviour {
    void Update (){
        if (Input.GetKey (KeyCode.UpArrow)) {
            GameObject.Find ("Cube").GetComponent<viewMe>().setThisNumber = 1;
        }
        if (Input.GetKey (KeyCode.DownArrow)) {
            GameObject.Find ("Cube").GetComponent<viewMe>().setThisNumber = 2;
        }
    }
}

I really hate asking repeat questions. It all seems pretty straight forward though, this is EXACTLY what I’m doing. Step by step, everything. Hopefully you can see where I’m making my errors and correct me in my no0b ways :slight_smile:

Right now you are only updating setThisNumber locally.

Create a method in viewMe in which you pass on a number as an argument, then let that method call a RPC and update the number.

Also save a reference for the cubes viewMe in Start() så you don’t call Find AND GetComponent each time.

Have you considered creating an array of controllable objects and, if a player is moving it, send it’s data via that player’s OnPhotonSerializeView and just updating everyone else’s array of moveable objects? So essentially you’d use an RPC to build the array initially and to add/remove objects so that everyone has the same array, and then you’re only updating if it’s being used. This way you’ll not need to treat it any differently than any other object in your scene, and in fact could techincally add any object in your scene to that array and players could interact with it and have it sync across clients.

Please follow the Marco Polo Tutorial. It covers what you describe as “Fight for Control”. You basically have to make sure that your key/controller input is only applied to the one character instance you control (and not to each character that is in the room, total). The solution is to check isMine of a PhotonView on the instanced GameObjects.

See:

Hey all, sorry for late reply been doing life things.

Thank you for the responses! I’ve worked my way through the Marco Polo tutorial, I get decently lost around “Switch it”, I feel I followed pretty well up to this point.

I think this is similar to what I had attempted with my original code post, I’ve since incorporated some of your suggestions and I can view changes on both machines made by either player. The downside is the changes viewed on receiving machines is a bit choppy, I think this is what BFGames was commenting towards?

This is what I’m working with now, using the RPC approach :

NetworkManager (placed on gameobject in scene)

using UnityEngine;
using System.Collections;
using System.Collections.Generic;

public class NetworkManager_Test : MonoBehaviour {

    public GameObject testPlayer;
    public GameObject[] vehicles; //Array of moveable objects
    public Vector3[] realPosition;

    void Start (){
        //PhotonNetwork.logLevel = PhotonLogLevel.Full;
        PhotonNetwork.ConnectUsingSettings("0.1");
        vehicles = GameObject.FindGameObjectsWithTag ("Vehicle");
        realPosition = new Vector3[vehicles.Length];

        for(int i = 0; i < realPosition.Length; i++){
            realPosition[i] = vehicles[i].transform.position;
        }
    }

    void OnJoinedLobby(){
        PhotonNetwork.JoinRandomRoom();
    }

    void OnPhotonRandomJoinFailed(){
        PhotonNetwork.CreateRoom(null);
    }

    void OnJoinedRoom(){
        print ("Joined Room");
        PhotonNetwork.Instantiate ("testPlayer", Vector3.zero, Quaternion.identity, 0);
    }

    void OnGUI(){
        GUILayout.Label(PhotonNetwork.connectionStateDetailed.ToString());
    }

    //TODO: Doesn't appear to change anything.
    /*
    public void OnPhotonSerializeView(PhotonStream stream, PhotonMessageInfo info){
        for (int i = 0; i < vehicles.Length; i++) {
            if(stream.isWriting){
                print ("Writing");

                realPosition[i] = vehicles[i].transform.position;
                //stream.SendNext(this.realPosition);
            }
            else{
                print ("Reading");
                //vehicles[i].transform.position = (Vector3)stream.ReceiveNext();
            }
        }
    }*/
}

Instantiated PlayerController:

using UnityEngine;
using System.Collections;

public class testPlayerController : Photon.MonoBehaviour{
    GameObject NetworkManager;

    void Start(){
        NetworkManager = GameObject.Find ("GameObject");
    }

    void Update(){
        if (Input.GetKey (KeyCode.Keypad8)) {
            Vector3 newPos = Vector3.Lerp (GameObject.Find ("GameObject").GetComponent<NetworkManager_Test>().vehicles[0].transform.position, GameObject.Find ("GameObject").GetComponent<NetworkManager_Test>().vehicles[0].transform.position + Vector3.forward*10f, Time.deltaTime);
            GetComponent<PhotonView>().RPC ("vehiclePosition", PhotonTargets.AllBuffered, newPos);
        }
        if (Input.GetKey (KeyCode.Keypad2)) {
            Vector3 newPos = Vector3.Lerp (GameObject.Find ("GameObject").GetComponent<NetworkManager_Test>().vehicles[0].transform.position, GameObject.Find ("GameObject").GetComponent<NetworkManager_Test>().vehicles[0].transform.position - Vector3.forward*10f, Time.deltaTime);
            GetComponent<PhotonView>().RPC ("vehiclePosition", PhotonTargets.AllBuffered, newPos);
        }
    }

    [RPC]
    void vehiclePosition(Vector3 newPos){
        NetworkManager.GetComponent<NetworkManager_Test>().vehicles[0].transform.position = newPos;
    }
}

Quite rough I know, I can clean it up when I better understand what’s going on…

Actually in photon the scene object by default own by the master client and only master client can change their position if you want to change their position by client then either you must change their ownership or you can make it a network object using PhotonNetwork.Instantiate(Gameobject),it means you need to spawn the object when we load the scene.

if the masterClient left the room the instantiated gameobject will disappear too i need that one time so that game object can still keep running even if the masterclient leave the game

The Master Client is allowed to InstantiateSceneObject().
Those should stay alive.

1 Like