1. 윈폼에서 /open [ServerIP] 명령어를 TextBox에 입력 후 Enter 버튼을 누르면 새 쓰레드를 만들어 서버를 엽니다. 포트는 적당히 설정해주시면 됩니다.
2. 서버는 while(true) { .... AcceptTcpClient(); ... } 를 통해 클라이언트를 계속 받습니다.
3. 클라이언트를 받을 때 마다 해당 클라이언트와 통신할 ChatServer 인스턴스를 만듭니다. 이 인스턴스는 생성시 서버로부터 TcpClient 데이터를 받고 Listen() { while(true} { ... read(); ... } } 를 통해 실시간 클라이언트로부터 데이터를 읽습니다.
<클라이언트>
1. ID를 입력하면 Client 폼을 띄웁니다.
2. /connect [ClientIP] [ServerIP] [ServerPort] 를 입력해 서버와 연결합니다. (Client 쪽 포트는 OS에서 알아서 처리하게끔 합니다)
3. 클라이언트가 메세지 입력 후 전송버튼을 누르면 NetworkStream.write(); 메소드로 서버측에 메세지를 전송합니다.
4. 백그라운드로 Receiver() 메소드를 담당할 쓰레드를 만들어 서버쪽에서 오는 메세지를 받습니다.
주요 기능만 구현하고 부가적인 기능(예를 들자면 서버에 연결 안된채로 클라이언트 폼을 닫을때 처리라거나..여러 예외처리들)은 생략했습니다. 이것들은 다음 글에 천천히 쓰겠습니다 허허
메세지 전송은 photonView.RPC 메소드를 이용해 각 유저들에게 ReceiveMsg 메소드를 실행케 합니다.
자기 자신에게도 메세지를 띄워야하니 ReceiveMsg(msg); 를 실행합니다.
input.ActivateInputField();는 메세지 전송 후 바로 메세지를 입력할 수 있게 포커스를 Input Field로 옮깁니다. (편의기능)
그 후 input.text를 빈칸으로 만들어줍시다.
ReceiveMsg메소드는 간단합니다. RPC 메소드로 사용할 수 있게 [PunRPC]를 메소드 위에 써주시면 되고, 하는 동작은 chatLog.text에 msg와 \n(개행문자)를 더해주고, scroll_rect.verticalNormalizedPosition = 0.0f; 로 메세지를 받을 때마다 스크롤을 가장 아래로 오게끔 합니다.
채팅부분은 끝입니다. 생각보다 굉장히 간단하죠. 나머진 편의기능으로,
chatterUpdate 메소드는 채팅 플레이어 리스트를 업데이트해줍니다. Player List 텍스트 아래에 플레이어들의 ID를 더해주는 식으로 하며, 실시간으로 출입하는 유저들의 ID를 반영합니다.
Update 메소드에선 chatterUpdate(); 메소드로 주기적으로 플레이어 리스트를 업데이트하며, input에 포커스가 맞춰져있고 엔터키가 눌려졌을 경우에도 SendButtonOnClicked(); 메소드를 실행하는 기능을 추가했습니다.
위에서 설명한 ID(Input Field), Login(Button), ConnectionStatus(Text)
그리고 로그인 화면 작업을 담당할 LobbyManager(Empty) 입니다.
LobbyManager 코드
LobbyManager.cs 스크립트는 다음과 같습니다.
C# 스크립트 생성 후 LobbyManager.cs 로 만들고, Empty object인 Lobby Manager에 추가하고, 다음과 같이 드래그앤 드롭 합니다.
Login Btn은 접속 버튼, IDtext는 Input Field의 Text, Connection Status는 접속 상태 출력 Text입니다.
마찬가지로, 접속 버튼이 눌러졌을때 실행할 메소드를 LobbyManager의 Connect()로 설정합니다.
소스코드를 각 부분별로 설명하자면,
LobbyManager는 MonoBehaviorPunCallbacks 를 상속받습니다.
Start 메소드는 시작과 동시에 PhotonNetwork.ConnectUsingSettings(); 메소드를 이용해 Photon 서버와 연결하며,
연결하는 동안 loginBtn.interactoable = false; 로 접속 버튼을 누르지 못하게 하고 연결상태를 출력합니다.
Connect 메소드는 접속 버튼을 누르면 실행할 메소드로, ID를 입력하지 않았다면 아무 반응을 하지 않습니다. ID를 입력해달라는 메세지를 띄우셔도 좋습니다. ID를 입력했다면 PhotonNetwork.LocalPlayer.NickName = IDtext.text; 코드를 통해 해당 로컬플레이어의 닉네임(ID)을 입력한 텍스트로 저장합니다.
그 후 Start 메소드에서 성공적으로 Photon 서버와 연결되었다면 PhotonNetwork.JoinRandomRoom() 메소드를 통해 채팅할 방에 입장하며, Start 메소드에서 Photon 서버와 연결이 되지 않았다면 재접속을 시도합니다.
OnConnectedToMaster 메소드는 Photon 서버(Master서버)와 연결되면 실행되는 메소드로, 접속 버튼을 사용 가능케 하며 연결 상태를 온라인으로 바꿉니다.
OnDisconnected 메소드는 Photon 서버와 연결이 끊겼을 경우 접속 버튼을 다시 사용 불가능하게 하며 연결 상태를 오프라인으로 바꿔 출력합니다. 그 후 서버와 재접속을 시도합니다.
OnJoinRandomFailed 메소드는 Connect 메소드에서 JoinRandomRoom 메소드를 실행했으나 방에 들어가지 못한 경우 실행되는 메소드로, 적합한 방이 없으면 PhotonNetwork.CreateRoom을 이용해 새 방을 만듭니다. 여기서 CreateRoom의 첫번째 인자는 방의 이름, 두번째 인자는 방의 옵션입니다. 여기선 방의 최대 인원을 4로 설정했습니다. (참고로 MaxPlayers = 0으로 하면 인원 제한을 없앨 수 있습니다.)
OnJoinedRoom 메소드는 방에 성공적으로 입장했을 경우 실행되는 메소드로, 씬을 Main(채팅을 진행할 씬)으로 옮깁니다. SceneManager.LoadScene()과 달리 PhotonNetwork.LoadLevel() 메소드를 사용한 이유는 네트워크의 정보를 그대로 가져가기 위함입니다.
완성된 로그인 씬입니다. ID를 입력하지 않고 접속버튼을 누르면 반응이 없으며, ID 입력후 접속에 성공하면 Main 씬으로 갑니다.
using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.UI; using Photon.Pun; using Photon.Realtime;
public class LobbyManager : MonoBehaviourPunCallbacks { public Button loginBtn; public Text IDtext; public Text ConnectionStatus; // Start is called before the first frame update void Start() { PhotonNetwork.ConnectUsingSettings(); loginBtn.interactable = false; ConnectionStatus.text = "Connecting to Master Server..."; }
public void Connect() { if (IDtext.text.Equals("")) { return; } else { PhotonNetwork.LocalPlayer.NickName = IDtext.text; loginBtn.interactable = false; if (PhotonNetwork.IsConnected) { ConnectionStatus.text = "connecting to room..."; PhotonNetwork.JoinRandomRoom(); } else { ConnectionStatus.text = "Offline : failed to connect.\nreconnecting..."; PhotonNetwork.ConnectUsingSettings(); }
}
} public override void OnConnectedToMaster() { loginBtn.interactable = true; ConnectionStatus.text = "Online : connected to master server"; } public override void OnDisconnected(DisconnectCause cause) { loginBtn.interactable = false; ConnectionStatus.text = "Offline : failed to connect.\nreconnecting..."; PhotonNetwork.ConnectUsingSettings(); } public override void OnJoinRandomFailed(short returnCode, string message) { ConnectionStatus.text = "No empty room. creating new room..."; PhotonNetwork.CreateRoom(null, new RoomOptions { MaxPlayers = 4 }); } public override void OnJoinedRoom() { ConnectionStatus.text = "Success to join room"; PhotonNetwork.LoadLevel("Main"); } }