Was reminded tonight that I should have put up the slides for this a while ago:
Game Dev Boot Camp Slides (ODP)
PDF Slides (Warning: Animation makes these look kind of shit)
Was reminded tonight that I should have put up the slides for this a while ago:
Game Dev Boot Camp Slides (ODP)
PDF Slides (Warning: Animation makes these look kind of shit)
I gave another talk at the Newfoundland Gaming Expo. As I mentioned during that talk, I figured out last year that “Making Games” is too big a subject, so I decided on “Game Programming” for this year’s talk. Which is ALSO too big a subject, so next year should be halfway focused. Maybe.
Here are the slides for this year’s NGX2016 Talk – Game Programming With Unity
I struggled to find any information about this online, so I’ll write a quick post about how I’m solving this with the prototype for Contension in hopes that it will help someone out there at some point.
The prototype has a ContensionGame object which derives from NetworkManager, which, if you’re not familiar with UNET, is basically the thing that coordinates the network traffic of the application, kind of a very abstract client/server class.
using UnityEngine; using UnityEngine.Networking; using System.Collections; using System.Collections.Generic; public class MultiplayerGame : ContensionGame // ContensionGame is a NetworkManager { public List<uint> _readySignals; public void Launch() { StartHost(); } public void Connect(string ipAddress) { networkAddress = ipAddress; StartClient(); Debug.Log("connected"); } public void AddReady(uint id) { if(!_readySignals.Contains(id)) { _readySignals.Add(id); if(_readySignals.Count > 1) { ServerChangeScene(this.onlineScene); } } } void Awake() { DontDestroyOnLoad(this); _readySignals = new List<uint>(); } }
Simple enough – in a normal multiplayer game, we wait for all the players to connect (tracked with _readySignals), and once we have two or more we go to the “main” scene. This isn’t exactly how you’d do things with a full game; for one thing, you’d have more complex scene loading, and for another you’d probably have more robust reconnection logic, but it gets the job done for prototyping.
The real work of starting a multiplayer level, however, is done in the Player GameObject, primarily by the TeamSpawner script component. This object actually spawns our units in the appropriate areas on the map.
Network code can be hard to think about, but in Contension I’m using an authoritative server, which just means that the client won’t actually be doing a whole lot in terms of judging when and how units move or come into conflict. The premise of the game doesn’t work super well if you allow clients to make those judgements, though I’ll probably have to revisit that down the road.
The basic things you need to know to understand this are:
using UnityEngine; using UnityEngine.Networking; using System.Collections; using System.Collections.Generic; [RequireComponent(typeof(NetworkIdentity))] public class TeamSpawner : NetworkBehaviour { public GameObject ContenderPrefab; [SyncVar]string _teamTag; List<Contender.Description> _contenderDescriptions; bool _spawned; void Start() { DontDestroyOnLoad(this); } public override void OnStartServer () { if(MoreThanOnePlayerWithMyTag()) { _teamTag = "Team2"; } if(isServer) { _tagged = true; } } public override void OnStartClient() { _teamTag = tag; } public override void OnStartLocalPlayer () { if(!isServer) { CmdSendTag(); } base.OnStartLocalPlayer (); } [Command] public void CmdSendTag() { RpcSetTag(this.tag); } [ClientRpc] public void RpcSetTag(string newTag) { tag = newTag; _tagged = true; } internal void SubmitTeam (IEnumerable<TeamSetup.DescriptionWrapper> team) { ClearTeam(); foreach(TeamSetup.DescriptionWrapper description in team) { AddDescription(description.Role, description.Commitment, description.Speed); } CmdSignalReady(); } [Command] void CmdSignalReady() { GetComponent<ReadySignal>().Send(); } private void AddDescription(Contender.Roles role, Contender.Commitments commitment, Contender.Speeds speed) { CmdAddDescription(role, commitment, speed); } [Command] void CmdAddDescription(Contender.Roles role, Contender.Commitments commitment, Contender.Speeds speed) { ContenderDescriptions.Add(new Contender.Description(role, commitment, speed)); } void OnLevelWasLoaded() { _spawned = false; } void Update () { if(isLocalPlayer && _tagged && !_spawned && _contenderDescriptions != null) { TeamSpawnArea[] spawnAreas = FindObjectsOfType<TeamSpawnArea>(); foreach(TeamSpawnArea area in spawnAreas) { if(area.tag == this.tag) { // Simple local perspective hack - the camera is rotated 180 if the player spawns in the // top of the map instead of the bottom transform.position = area.Center; if(transform.position.y > 0 && GetComponent<AiPlayer>() == null) { Camera.main.transform.Rotate (new Vector3(0,0,180)); } if(isServer) { SpawnTeam (tag); } else { CmdSpawnTeam(tag); } _spawned = true; } } } } [Command] public void CmdSpawnTeam (string tag) { SpawnTeam(tag); } private void SpawnTeam(string tag) { TeamSpawnArea[] spawnAreas = FindObjectsOfType<TeamSpawnArea>(); TeamSpawnArea teamArea = spawnAreas[0]; foreach(TeamSpawnArea area in spawnAreas) { if(area.tag == tag) { teamArea = area; break; } } foreach(Contender.Description description in _contenderDescriptions) { Vector2 SpawnLocation = PickSpawnPoint(teamArea); GameObject obj = (GameObject)Instantiate(ContenderPrefab, SpawnLocation, Quaternion.identity); Contender contender = obj.GetComponent<Contender>(); contender.Initialize(tag, netId.Value, description); NetworkServer.Spawn(obj); } } }
One of the basic problems with UNET, however, is it doesn’t natively support different player prefabs (read: types) for different players. This means that you can’t just set the player type and forget about it if you want to reuse the multiplayer code for your single player game. In a larger studio that might not be a concern, but I’m doing this on my own right now and that means I need to try to restrict how many things I have to worry about.
My solution to this (again, this is prototype code!) is pretty quick and dirty. Basically I’ve set the “main” playerPrefab to be my AI player class, and then added the human player as a spawnable prefab. As soon as the game starts, the AI player connects, which causes the game to spawn a second client with a hardcoded team.
using UnityEngine; using UnityEngine.Networking; using System.Collections; using System.Collections.Generic; public class SinglePlayerGame : ContensionGame { bool _playerAdded; // Use this for initialization void Start () { StartHost(); } public override void OnServerAddPlayer(NetworkConnection conn, short playerControllerId) { GameObject Player; if(playerControllerId == 0) { Player = (GameObject)GameObject.Instantiate(playerPrefab, Vector2.zero, Quaternion.identity);; } else { Player = (GameObject)GameObject.Instantiate(spawnPrefabs[0], Vector2.zero, Quaternion.identity); } NetworkServer.AddPlayerForConnection(conn, Player, playerControllerId); if(playerControllerId != 0) { TeamSpawner PlayerTeam = Player.GetComponent<TeamSpawner>(); List<TeamSetup.DescriptionWrapper> Units = new List<TeamSetup.DescriptionWrapper>(); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.ManyOnOne, Contender.Commitments.Balanced, Contender.Speeds.Average))); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.ManyOnOne, Contender.Commitments.Balanced, Contender.Speeds.Average))); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.OneOnMany, Contender.Commitments.Balanced, Contender.Speeds.Average))); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.OneOnMany, Contender.Commitments.Balanced, Contender.Speeds.Average))); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.OneOnOne, Contender.Commitments.Balanced, Contender.Speeds.Average))); Units.Add( new TeamSetup.DescriptionWrapper( new Contender.Description(Contender.Roles.OneOnOne, Contender.Commitments.Balanced, Contender.Speeds.Average))); PlayerTeam.SubmitTeam(Units); } } // Update is called once per frame void Update () { if(!_playerAdded && ClientScene.ready) { _playerAdded = true; ClientScene.AddPlayer(2); } } }
For two AI players (for example, when building an AI demo or training simulator), you can do a similar thing but simply spawn a second AI player prefab instead of the human player.
I’ve also realized while writing this article that I can do a better team tagging solution based on the map’s available spawn areas. Which is neat!
So you want to write code for a living, but you also have a wee bit of graphic artist in you? Maybe shaders are your glovely medium!
Here’s a small sample of the insanity on display at ShaderToy.
Garage – I haven’t parsed this completely, but I think that entire scene – including the cars – may be a single shader. Note that the car goes both up AND down the decks.
Tentacle Thing – More traditional, akin to the hair shaders supposedly used on the Monsters movies from Pixar, but still amazing to see it running in realtime in a browser.
Seascape – What can I say, except maybe why the hell aren’t there games that have this as part of their experience???
Flame – Simple, but also easy to mess with (try changing the numbers in the mainImage function and hit the Play button below the code window to see what I mean).
It’s hard to imagine the world of the elder statesmen in this field; they were stuck decoding research papers from a variety of obscure journals. But it’s also nice to know that there is such an incredibly rich ocean of knowledge out there to scour while learning to do cool things.
There’s a moment in here (27s-34s) that actually caused me to shout with delighted surprise. You know you’re a bit of a programming nerd when your own AI playing itself is the best part of the development process.
Unity kicked my ass several ways to Sunday tonight, but despite that, I eventually got it to do what I wanted!
New stuff happening. I’ll be giving a talk about programming and technical creation in Unity at NGX2016. I’m working on separating the Contension network code from the conflict code so I can start doing up some AI behaviours. I’ve got a bunch of candidate attributes to try out in-game. And I’ve got a rough idea of the narrative arc of the game.
Next up, I’ll be trying to figure out how to apply the Non Combat episode of Extra Credits to my presentation of the game.