The BepInEx console will not appear when launching like it does for other games on Thunderstore (you can turn it back on in your BepInEx.cfg file). If your PEAK crashes on startup, add -dx12 to your launch parameters.
 
        PhotonCustomPropsUtils
Helps modders easily synchronize room/player properties across all connected clients through Photon's Custom Properties.
By Snosz
| Last updated | 2 months ago | 
| Total downloads | 168353 | 
| Total rating | 1 | 
| Categories | Mods All Clients | 
| Dependency string | Snosz-PhotonCustomPropsUtils-1.1.0 | 
| Dependants | 60 other packages depend on this package | 
This mod requires the following mods to function
 
                        BepInEx-BepInExPack_PEAK
BepInEx pack for PEAK. Preconfigured and ready to use.
Preferred version: 5.4.2403README
Photon Custom Props Utils
Helps modders easily synchronize room/player properties across all connected clients.
Utilizes Photon RoomProperties and PlayerProperties to make it easier to synchronize properties for all players.
To Modders
The best way to reach me is through the Photon Custom Props Utils thread on the Peak modding discord server.
Guide
First add this mod dll as a project reference for your mod. Then you can add a dependency to the mod to ensure your mod runs after PhotonCustomPropsUtils.
[BepInDependency("com.snosz.photoncustompropsutils", BepInDependency.DependencyFlags.HardDependency)]
Or if you dont want to set a dependancy you can just check for IsReady and listen for OnReady.
if (PhotonCustomPropsUtilsPlugin.IsReady)
    DoPhotonStuff();
else
    PhotonCustomPropsUtilsPlugin.OnReady += DoPhotonStuff;
Then you can create your scoped CustomPropsManager by passing your plugin GUID like so:
manager = PhotonCustomPropsUtilsPlugin.GetManager("com.author.modname");
This ensures that any properties you listen to are specific to your mod which helps avoid naming conflicts where multiple mods might have similar property names.
To start listening for a property change you can use either RegisterRoomProperty, or RegisterPlayerProperty. You can also set the event type, depending on if its a player or room event you can use RoomEventType.All (for both OnJoinedRoom, and OnRoomPropertyChanged) or either one seperately if you want to handle that logic seperately.
PlayerEventType is similar and can be PlayerEventType.All, OnPlayerPropertyChanged, and OnJoinedRoom.
For easy usage you can specify the expected type to be received which means you dont have to worry about casting it yourself.
// This will be called for either OnJoinedRoom (which runs for the local player), or OnRoomPropertyChanged.
manager.RegisterRoomProperty<bool>("myValue", RoomEventType.All, value =>
{
    storedValue = value; //This is where you handle what to do with the updated property
});
// This will be called for OnPlayerPropertyChanged.
manager.RegisterPlayerProperty<bool>("myValue", PlayerEventType.OnPlayerPropertyChanged, (targetPlayer, value) =>
{
    storedValue = value; //This is where you handle what to do with the updated property
});
To set a new value for a property you can just call SetRoomProperty
manager.SetRoomProperty("myValue", newValue);
To listen for when the local player has joined a room:
manager.RegisterOnJoinedRoom(localPlayer => 
{
	// Do thing
});
To get all player properties, or all players with a specific property:
var allPlayersProperties = manager.GetAllPlayerProperties<float[]>("targetProperty");
Please note that this should not be used in excess such as setting a property every frame. It should be only done in response to client behaviour such as pushing a button.
For further reading about Photon custom properties, the official Photon docs have both information on how it can be used, and the allowed types that are supported.
https://doc.photonengine.com/realtime/current/gameplay/custom-properties
https://doc.photonengine.com/fusion/current/manual/data-transfer/rpcs