commit
7327368999
46 changed files with 5501 additions and 661 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -329,3 +329,4 @@ ASALocalRun/
|
|||
|
||||
# MFractors (Xamarin productivity tool) working folder
|
||||
.mfractor/
|
||||
/SRMP/_info.txt
|
||||
|
|
BIN
Libs/Assembly-CSharp_publicized.dll
Normal file
BIN
Libs/Assembly-CSharp_publicized.dll
Normal file
Binary file not shown.
BIN
Libs/Newtonsoft.Json.dll
Normal file
BIN
Libs/Newtonsoft.Json.dll
Normal file
Binary file not shown.
BIN
Libs/SRML.Editor.dll
Normal file
BIN
Libs/SRML.Editor.dll
Normal file
Binary file not shown.
BIN
Libs/SRML.dll
Normal file
BIN
Libs/SRML.dll
Normal file
Binary file not shown.
3810
Libs/SRML.xml
Normal file
3810
Libs/SRML.xml
Normal file
File diff suppressed because it is too large
Load diff
BIN
Libs/System.Runtime.Serialization.dll
Normal file
BIN
Libs/System.Runtime.Serialization.dll
Normal file
Binary file not shown.
47
README.md
47
README.md
|
@ -1,21 +1,42 @@
|
|||
# srmp-public
|
||||
This is the code for the Slime Rancher MultiPlayer Mod (SRMP) by SatyPardus.
|
||||
|
||||
It's bad. It's really really bad.
|
||||
The user manual which includes information about compatability information, download and installation instructions and standard troubleshooting can be found [here](/manual.md).
|
||||
|
||||
There is so much to clean up in this project, I would not even know where to begin.
|
||||
Kinda the reason why it's already uploaded. No idea where and how to clean it up.
|
||||
|
||||
But this is the source code of my Slime Rancher Multiplayer mod.
|
||||
It went through many iterations and each new one had stuff copy pasted.
|
||||
## Bug Status
|
||||
Note: Bug list compiled from last known bug list of version 1488
|
||||
|
||||
It's a mess within a mess.
|
||||
Oh god, forgive me.
|
||||
FIXED:
|
||||
- Multiplayer window doesn't show up on resolutions < 1920x1080
|
||||
- Exchange sometimes skips rewards if multiple players put items in at the same time
|
||||
- Exchange chest disappears without rewards
|
||||
-
|
||||
IN PROGRESS: N/A
|
||||
|
||||
---------------------------------------------------------------------------------
|
||||
Known Bugs:
|
||||
- Plort collectors don't work properly sometimes (Playing effect but not pulling anything)
|
||||
- Nutcracker doesn't spit out right amount, or spits out "babies"
|
||||
- Slimes sometimes appear "angry" for other players
|
||||
- Game stutters when placing/removing gadgets
|
||||
- First time using the mod can apparently break a lot of things (falling through world, slimes not eating) - Restarting the game fixes it
|
||||
- DLCs don't seem to be loaded correctly when leaving and joining
|
||||
- DLC are not initialized on game start, making the "You need following DLCs:" message pop up (Can be fixed by loading the "Manage DLCs" menu and trying again)
|
||||
- Gordos don't drop things sometimes
|
||||
- Slimes sometimes dont produce plorts
|
||||
- Drones get stuck in place sometimes
|
||||
- Chat sometimes empty for remote players
|
||||
- Upgrades sometimes does not get applied to All players
|
||||
|
||||
But for real. Have fun digging through my unoptimized and undocumented code.
|
||||
I never thought I actually gonna release this, yet here we are.
|
||||
Would have dressed differently if I would have known.
|
||||
|
||||
Don't judge me tho. It still .... "works".
|
||||
Just not as good as it could!
|
||||
## Current Status
|
||||
[@Twirlbug](https://github.com/Twirlbug)
|
||||
- Currently working on going through the code, adding notes and fixing some of the bugs in my free time.
|
||||
- I adore this mod and want to give both credit and a huge thank you to Saty for the origional creation of the mod.
|
||||
I am slowly working my way through the list of bugs as seen above.
|
||||
|
||||
### Notation Status
|
||||
Files in the following folders still need more notation:
|
||||
- Networking
|
||||
- Packets
|
||||
- Patches
|
||||
|
|
21
SRMP.sln
21
SRMP.sln
|
@ -1,32 +1,37 @@
|
|||
|
||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||
# Visual Studio 15
|
||||
VisualStudioVersion = 15.0.28307.1267
|
||||
# Visual Studio Version 17
|
||||
VisualStudioVersion = 17.6.33815.320
|
||||
MinimumVisualStudioVersion = 10.0.40219.1
|
||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SRMP", "SRMP\SRMP.csproj", "{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}"
|
||||
EndProject
|
||||
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution Items", "{623884D3-E475-43F6-A6DF-64B8860E3EDD}"
|
||||
ProjectSection(SolutionItems) = preProject
|
||||
SRMP\_info.txt = SRMP\_info.txt
|
||||
EndProjectSection
|
||||
EndProject
|
||||
Global
|
||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||
Debug|Any CPU = Debug|Any CPU
|
||||
Release|Any CPU = Release|Any CPU
|
||||
SRML|Any CPU = SRML|Any CPU
|
||||
SRML NoVer|Any CPU = SRML NoVer|Any CPU
|
||||
Standalone|Any CPU = Standalone|Any CPU
|
||||
SRML|Any CPU = SRML|Any CPU
|
||||
Standalone NoVer|Any CPU = Standalone NoVer|Any CPU
|
||||
Standalone|Any CPU = Standalone|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML|Any CPU.ActiveCfg = SRML|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML|Any CPU.Build.0 = SRML|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML NoVer|Any CPU.ActiveCfg = SRML NoVer|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML NoVer|Any CPU.Build.0 = SRML NoVer|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone|Any CPU.ActiveCfg = Standalone|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone|Any CPU.Build.0 = Standalone|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML|Any CPU.ActiveCfg = SRML|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.SRML|Any CPU.Build.0 = SRML|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone NoVer|Any CPU.ActiveCfg = Standalone NoVer|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone NoVer|Any CPU.Build.0 = Standalone NoVer|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone|Any CPU.ActiveCfg = Standalone|Any CPU
|
||||
{E1BF7CDA-F2AE-4042-A992-DCBF62E79238}.Standalone|Any CPU.Build.0 = Standalone|Any CPU
|
||||
EndGlobalSection
|
||||
GlobalSection(SolutionProperties) = preSolution
|
||||
HideSolutionNode = FALSE
|
||||
|
|
27
SRMP/Console/Console Commands.txt
Normal file
27
SRMP/Console/Console Commands.txt
Normal file
|
@ -0,0 +1,27 @@
|
|||
The following console Commands work in the pop up console
|
||||
|
||||
cheat money [amount]
|
||||
Add or Remove the given amount from the curret money score
|
||||
|
||||
cheat keys [amount]
|
||||
Add or Remove the given amount of keys from the current amount
|
||||
|
||||
cheat allgadgets
|
||||
Unlocks all of the gagets in the game
|
||||
|
||||
cheat spawn [id] ([amount])
|
||||
spawns given item for the given ammount
|
||||
|
||||
tp [(optional)TargetPlayer] [DestinationPlayer]
|
||||
Teleports the player in spot 1 to player/destination entered
|
||||
if spot 1 is empty the palayer that entered in the command in infered as the target
|
||||
listplayers
|
||||
Prints the list of current players to the console
|
||||
|
||||
sleep [Hours]
|
||||
Sleep command followed by the time in in game hours to "sleep"". Fast forwards the game the alotted hours.
|
||||
|
||||
console [enable/disable] [logtype]
|
||||
Sets whether a specific log type should be displaying in the console.
|
||||
All type default to enabled.
|
||||
Log Types: Error, Assert, Warning, Log, Exception
|
|
@ -1,4 +1,5 @@
|
|||
using System;
|
||||
using Microsoft.Win32;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
@ -10,6 +11,45 @@ namespace SRMultiplayer
|
|||
public string inputString = "";
|
||||
public event Action<string> OnInputText;
|
||||
|
||||
/// <summary>
|
||||
/// Handle updates the console
|
||||
/// This is to catch key presses and process them
|
||||
/// </summary>
|
||||
public void Update()
|
||||
{
|
||||
if (Console.KeyAvailable)
|
||||
{
|
||||
ConsoleKeyInfo consoleKeyInfo = Console.ReadKey();
|
||||
|
||||
switch (consoleKeyInfo.Key)
|
||||
{
|
||||
case ConsoleKey.Enter: this.OnEnter(); break;
|
||||
case ConsoleKey.Backspace: this.OnBackspace(); break;
|
||||
case ConsoleKey.Escape: this.OnEscape(); break;
|
||||
case ConsoleKey.UpArrow: this.GetCommand(1); break;
|
||||
case ConsoleKey.DownArrow: this.GetCommand(-1); break;
|
||||
default:
|
||||
//check if pressed key is a character
|
||||
bool flag = consoleKeyInfo.KeyChar > '\0';
|
||||
if (flag)
|
||||
{
|
||||
//if so at it tothe input line
|
||||
this.inputString += consoleKeyInfo.KeyChar.ToString();
|
||||
this.RedrawInputLine();
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
#region Console Processors
|
||||
/// <summary>
|
||||
/// Clears out the current console line
|
||||
/// </summary>
|
||||
public void ClearLine()
|
||||
{
|
||||
Console.CursorLeft = 0;
|
||||
|
@ -17,7 +57,9 @@ namespace SRMultiplayer
|
|||
Console.CursorTop--;
|
||||
Console.CursorLeft = 0;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Used to redraw the input line incase we had to hold for console write lines
|
||||
/// </summary>
|
||||
public void RedrawInputLine()
|
||||
{
|
||||
bool flag = Console.CursorLeft > 0;
|
||||
|
@ -29,7 +71,12 @@ namespace SRMultiplayer
|
|||
Console.Write("> ");
|
||||
Console.Write(this.inputString);
|
||||
}
|
||||
#endregion region
|
||||
|
||||
#region Key Press Events
|
||||
/// <summary>
|
||||
/// Process Backspace pressed
|
||||
/// </summary>
|
||||
internal void OnBackspace()
|
||||
{
|
||||
bool flag = this.inputString.Length <= 0;
|
||||
|
@ -39,7 +86,9 @@ namespace SRMultiplayer
|
|||
this.RedrawInputLine();
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Process Escape pressed
|
||||
/// </summary>
|
||||
internal void OnEscape()
|
||||
{
|
||||
this.ClearLine();
|
||||
|
@ -47,6 +96,9 @@ namespace SRMultiplayer
|
|||
this.RedrawInputLine();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Process Enter pressed
|
||||
/// </summary>
|
||||
internal void OnEnter()
|
||||
{
|
||||
this.ClearLine();
|
||||
|
@ -57,47 +109,56 @@ namespace SRMultiplayer
|
|||
bool flag = this.OnInputText != null;
|
||||
if (flag)
|
||||
{
|
||||
//on text inputed reset the search loc and cycle the search tree
|
||||
searchLoc = -1; //searchLoc set to -1 to always go to place 0 on first arrow
|
||||
AddToCommandTree(obj);
|
||||
|
||||
this.OnInputText(obj);
|
||||
}
|
||||
}
|
||||
|
||||
public void Update()
|
||||
#endregion
|
||||
|
||||
#region Command History Retrieval
|
||||
List<string> cmdTree = new List<string>();
|
||||
int maxCommands = 10;
|
||||
/// <summary>
|
||||
/// Manages the Command tree for the up and down arrows
|
||||
/// </summary>
|
||||
/// <param name="cmdText"></param>
|
||||
internal void AddToCommandTree(string cmdText)
|
||||
{
|
||||
bool flag = !Console.KeyAvailable;
|
||||
if (!flag)
|
||||
{
|
||||
ConsoleKeyInfo consoleKeyInfo = Console.ReadKey();
|
||||
bool flag2 = consoleKeyInfo.Key == ConsoleKey.Enter;
|
||||
if (flag2)
|
||||
{
|
||||
this.OnEnter();
|
||||
cmdTree.Insert(0, cmdText);
|
||||
|
||||
//if tree gets larger than 10 remove the 11th item
|
||||
if (cmdTree.Count > maxCommands) { cmdTree.RemoveAt(10); };
|
||||
}
|
||||
else
|
||||
|
||||
|
||||
//handle internal cycling of last 10 commands
|
||||
int searchLoc = -1;
|
||||
/// <summary>
|
||||
/// Gets the command at the designaed slot from the current possition
|
||||
/// </summary>
|
||||
/// <param name="diff">Possition from the current Command </param>
|
||||
internal void GetCommand(int diff)
|
||||
{
|
||||
bool flag3 = consoleKeyInfo.Key == ConsoleKey.Backspace;
|
||||
if (flag3)
|
||||
if (cmdTree.Count > 0)
|
||||
{
|
||||
this.OnBackspace();
|
||||
}
|
||||
else
|
||||
{
|
||||
bool flag4 = consoleKeyInfo.Key == ConsoleKey.Escape;
|
||||
if (flag4)
|
||||
{
|
||||
this.OnEscape();
|
||||
}
|
||||
else
|
||||
{
|
||||
bool flag5 = consoleKeyInfo.KeyChar > '\0';
|
||||
if (flag5)
|
||||
{
|
||||
this.inputString += consoleKeyInfo.KeyChar.ToString();
|
||||
searchLoc = searchLoc + diff;
|
||||
//prevent below 0 or over max position
|
||||
if (searchLoc > (cmdTree.Count - 1)) searchLoc = (cmdTree.Count - 1);
|
||||
if (searchLoc < 0) searchLoc = 0;
|
||||
|
||||
//if a new location is found enter the search text in the input and redraw it.
|
||||
this.inputString = cmdTree[searchLoc];
|
||||
this.RedrawInputLine();
|
||||
}
|
||||
else
|
||||
{
|
||||
Console.WriteLine("cmdTree is empty");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
#endregion
|
||||
}
|
||||
}
|
|
@ -14,7 +14,9 @@ namespace SRMultiplayer
|
|||
private TextWriter oldOutput;
|
||||
|
||||
private const int STD_OUTPUT_HANDLE = -11;
|
||||
|
||||
/// <summary>
|
||||
/// Create the console window
|
||||
/// </summary>
|
||||
public void Initialize()
|
||||
{
|
||||
bool flag = !ConsoleWindow.AttachConsole(uint.MaxValue);
|
||||
|
|
482
SRMP/Console/SRMPConsole.cs
Normal file
482
SRMP/Console/SRMPConsole.cs
Normal file
|
@ -0,0 +1,482 @@
|
|||
using JetBrains.Annotations;
|
||||
using MonomiPark.SlimeRancher.Regions;
|
||||
using SRMultiplayer.Networking;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Net.Sockets;
|
||||
using UnityEngine;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
enum LogItems
|
||||
{
|
||||
CLIENT,
|
||||
SERVER,
|
||||
PLAYERAMMO,
|
||||
EXCHANGE
|
||||
}
|
||||
|
||||
|
||||
public class SRMPConsole : MonoBehaviour
|
||||
{
|
||||
ConsoleWindow console = new ConsoleWindow();
|
||||
ConsoleInput input = new ConsoleInput();
|
||||
|
||||
//
|
||||
// Create console window, register callbacks
|
||||
//
|
||||
void Awake()
|
||||
{
|
||||
DontDestroyOnLoad(gameObject);
|
||||
|
||||
console.Initialize();
|
||||
console.SetTitle("Slime Rancher");
|
||||
|
||||
input.OnInputText += OnInputText;
|
||||
|
||||
Application.logMessageReceived += Application_logMessageReceived;
|
||||
|
||||
SRMP.Log("Console Started");
|
||||
}
|
||||
|
||||
//used to prevent duplicate messages displaying and list what number duplicate it is
|
||||
string LastMessage = "";
|
||||
int duplicateCount = 0;
|
||||
|
||||
//types of console types that can be enabled/disabled
|
||||
//server automatically starts with all active
|
||||
List<LogType> blockMessages = new List<LogType>(); //keeps a list of message types that have been disabled
|
||||
List<LogItems> blockLogs = new List<LogItems>(); //keeps a list of log message types that have been disabled
|
||||
bool DisplayTrace = false;
|
||||
private void Application_logMessageReceived(string condition, string stackTrace, LogType type)
|
||||
{
|
||||
// We're half way through typing something, so clear this line ..
|
||||
if (Console.CursorLeft != 0)
|
||||
input.ClearLine();
|
||||
|
||||
//construct message
|
||||
string message = condition;
|
||||
if (!string.IsNullOrEmpty(stackTrace) && DisplayTrace)
|
||||
{
|
||||
//add stack strace if included
|
||||
message += Environment.NewLine + stackTrace;
|
||||
}
|
||||
|
||||
if (message == LastMessage)
|
||||
{
|
||||
//do not process duplicate marks if the last item was not written
|
||||
if (duplicateCount > 0) duplicateCount++;
|
||||
}
|
||||
else
|
||||
{
|
||||
//add write line for duplicate notices if necessary
|
||||
if (duplicateCount > 0)
|
||||
{
|
||||
Console.ForegroundColor = ConsoleColor.Gray;
|
||||
Console.WriteLine("Output Duplicated: " + duplicateCount);
|
||||
}
|
||||
|
||||
//format color for message
|
||||
if (type == LogType.Warning)
|
||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
||||
else if (type == LogType.Error)
|
||||
Console.ForegroundColor = ConsoleColor.Red;
|
||||
else
|
||||
Console.ForegroundColor = ConsoleColor.White;
|
||||
|
||||
// mark new message
|
||||
LastMessage = message;
|
||||
|
||||
bool displayLog = true;
|
||||
|
||||
//if log type is blocked turn off display
|
||||
if (blockMessages.Contains(type)) displayLog = false;
|
||||
|
||||
//check data for specific inner types
|
||||
string data = condition;
|
||||
|
||||
//remove the srmp tag and the time to check inner tags
|
||||
if (type == LogType.Log && data.StartsWith("[SRMP]"))
|
||||
{
|
||||
data = data.Substring(17);
|
||||
|
||||
//if is still able to display make sure the log message blocker isnt blocking it
|
||||
if (displayLog)
|
||||
{
|
||||
//try to pase the log message
|
||||
if (Enum.TryParse(data.Split("]"[0])[0].Substring(1), true, out LogItems logMessage))
|
||||
{
|
||||
if (blockLogs.Contains(logMessage)) displayLog = false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
//always allow console reply to display
|
||||
if (data.StartsWith("[Console]")) displayLog = true;
|
||||
|
||||
//only write the message type if its not blocked
|
||||
if (displayLog)
|
||||
{
|
||||
//write the new line if not blocked
|
||||
duplicateCount = 0;
|
||||
Console.WriteLine(message);
|
||||
}
|
||||
else
|
||||
{
|
||||
//for testing log disabled display
|
||||
//Console.ForegroundColor = ConsoleColor.Magenta;
|
||||
Console.WriteLine(type.ToString() + " Dismissed");
|
||||
|
||||
//mark dupilcate count to -1
|
||||
//prevent duplicate count for supressed messages from displaying
|
||||
duplicateCount = -1;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
// If we were typing something re-add it.
|
||||
input.RedrawInputLine();
|
||||
}
|
||||
|
||||
//create a log call that marks all console sent replys
|
||||
void ConsoleLog(string message)
|
||||
{
|
||||
SRMP.Log(message, "Console");
|
||||
}
|
||||
|
||||
|
||||
|
||||
//
|
||||
// Text has been entered into the console
|
||||
// Run it as a console command
|
||||
//
|
||||
void OnInputText(string obj)
|
||||
{
|
||||
var args = obj.Split(' ');
|
||||
var cmd = args.FirstOrDefault();
|
||||
args = args.Skip(1).ToArray();
|
||||
|
||||
switch (cmd.ToLower())
|
||||
{
|
||||
case "cheat":
|
||||
{
|
||||
if (args.Length > 0)
|
||||
{
|
||||
if (args[0].Equals("money"))
|
||||
{
|
||||
if (args.Length != 2 || !int.TryParse(args[1], out int money))
|
||||
{
|
||||
ConsoleLog("Usage: cheat money <amount>");
|
||||
}
|
||||
else
|
||||
{
|
||||
if (money > 0)
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.AddCurrency(money);
|
||||
else
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.SpendCurrency(money);
|
||||
}
|
||||
}
|
||||
else if (args[0].Equals("enable"))
|
||||
{
|
||||
//TestUI.Instance.cheat = !TestUI.Instance.cheat;
|
||||
}
|
||||
else if (args[0].Equals("keys"))
|
||||
{
|
||||
if (args.Length != 2 || !int.TryParse(args[1], out int value))
|
||||
{
|
||||
ConsoleLog("Usage: cheat keys <amount>");
|
||||
}
|
||||
else
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.keys = value;
|
||||
}
|
||||
}
|
||||
else if (args[0].Equals("allgadgets"))
|
||||
{
|
||||
foreach (var data in (Gadget.Id[])Enum.GetValues(typeof(Gadget.Id)))
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.GadgetDirector.model.gadgets[data] = int.MaxValue;
|
||||
}
|
||||
foreach (var data in Identifiable.CRAFT_CLASS.Union(Identifiable.PLORT_CLASS))
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.GadgetDirector.model.craftMatCounts[data] = int.MaxValue;
|
||||
}
|
||||
}
|
||||
else if (args[0].Equals("spawn"))
|
||||
{
|
||||
if (args.Length == 2)
|
||||
{
|
||||
if (Enum.TryParse<Identifiable.Id>(args[1].ToUpper(), out Identifiable.Id id))
|
||||
{
|
||||
GameObject prefab = SRSingleton<GameContext>.Instance.LookupDirector.GetPrefab(id);
|
||||
if (prefab != null)
|
||||
{
|
||||
SRBehaviour.InstantiateActor(prefab, SRSingleton<SceneContext>.Instance.GameModel.player.currRegionSetId, SRSingleton<SceneContext>.Instance.GameModel.player.GetPos() + new Vector3(0, 4, 0), Quaternion.identity, false);
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog(id + " can not be spawned");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var data = Enum.GetNames(typeof(Identifiable.Id)).Where(n => n.ToLower().Contains(args[1].ToLower()));
|
||||
SRMP.Log(args[1] + " not found. " + (data.Count() > 0 ? " Did you mean one of these?" : ""));
|
||||
foreach (var name in data)
|
||||
{
|
||||
ConsoleLog(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
else if (args.Length == 3 && int.TryParse(args[2], out int amount))
|
||||
{
|
||||
if (Enum.TryParse<Identifiable.Id>(args[1].ToUpper(), out Identifiable.Id id))
|
||||
{
|
||||
GameObject prefab = SRSingleton<GameContext>.Instance.LookupDirector.GetPrefab(id);
|
||||
if (prefab != null)
|
||||
{
|
||||
for (int i = 0; i < amount; i++)
|
||||
{
|
||||
SRBehaviour.InstantiateActor(prefab, SRSingleton<SceneContext>.Instance.GameModel.player.currRegionSetId, SRSingleton<SceneContext>.Instance.GameModel.player.GetPos() + new Vector3(0, 4, 0), Quaternion.identity, false);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log(id + " can not be spawned");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var data = Enum.GetNames(typeof(Identifiable.Id)).Where(n => n.ToLower().Contains(args[1].ToLower()));
|
||||
ConsoleLog(args[1] + " not found. " + (data.Count() > 0 ? " Did you mean one of these?" : ""));
|
||||
foreach (var name in data)
|
||||
{
|
||||
ConsoleLog(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Usage: cheat spawn <id> (<amount>)");
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Available sub commands:");
|
||||
ConsoleLog("cheat money <amount>");
|
||||
ConsoleLog("cheat keys <amount>");
|
||||
ConsoleLog("cheat spawn <id> (<amount>)");
|
||||
ConsoleLog("cheat allgadgets");
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "tp":
|
||||
{
|
||||
if (args.Length < 1 || args.Length > 2)
|
||||
{
|
||||
ConsoleLog("Usage: tp <(optional)target:username> <destination>");
|
||||
return;
|
||||
}
|
||||
|
||||
//mark target and destination seperately
|
||||
string target = args.Length == 2 ? args[0] : "";
|
||||
string destination = args.Length == 2 ? args[1] : args[0];
|
||||
|
||||
//mark target transform location
|
||||
PacketPlayerPosition packet = null;
|
||||
string destSummary = "";
|
||||
|
||||
//first check distination
|
||||
switch (destination.ToLower())
|
||||
{
|
||||
case "home":
|
||||
|
||||
var home = SRSingleton<SceneContext>.Instance.GetWakeUpDestination();
|
||||
packet = new PacketPlayerPosition()
|
||||
{
|
||||
Position = home.transform.position,
|
||||
Rotation = home.transform.eulerAngles.y,
|
||||
RegionSet = (byte)home.GetRegionSetId()
|
||||
};
|
||||
|
||||
destSummary = "Home";
|
||||
break;
|
||||
default: //check for a player name
|
||||
var play = Globals.Players.Values.FirstOrDefault(p => p.Username.Equals(destination, StringComparison.CurrentCultureIgnoreCase));
|
||||
|
||||
if (play == null)
|
||||
{
|
||||
ConsoleLog("Destination not found");
|
||||
return;
|
||||
}
|
||||
packet = new PacketPlayerPosition()
|
||||
{
|
||||
Position = play.transform.position,
|
||||
Rotation = play.transform.eulerAngles.y,
|
||||
RegionSet = (byte)play.CurrentRegionSet
|
||||
};
|
||||
destSummary = play.Username;
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
if (packet != null)
|
||||
{
|
||||
//set as not load destination
|
||||
packet.OnLoad = false;
|
||||
|
||||
NetworkPlayer targetPlayer = null;
|
||||
if (target.Length > 0)
|
||||
{
|
||||
targetPlayer = Globals.Players.Values.FirstOrDefault(p => p.Username.Equals(target, StringComparison.CurrentCultureIgnoreCase));
|
||||
}
|
||||
else
|
||||
{
|
||||
targetPlayer = Globals.LocalPlayer;
|
||||
}
|
||||
//check if target is local user
|
||||
if (targetPlayer == null)
|
||||
{
|
||||
ConsoleLog("Target Player not found");
|
||||
return;
|
||||
}
|
||||
|
||||
ConsoleLog("Teleporting " + targetPlayer.Username + " to " + destSummary);
|
||||
|
||||
if (!targetPlayer.IsLocal)
|
||||
{
|
||||
//if a target is located and is not the local player send the teleport command
|
||||
packet.ID = targetPlayer.ID;
|
||||
packet.WeaponY = targetPlayer.GetWeaponLocation();
|
||||
packet.Send();
|
||||
return;
|
||||
}
|
||||
else
|
||||
{
|
||||
//if we have have made it here the tp in question is for the instances player
|
||||
//so move that player
|
||||
SRSingleton<SceneContext>.Instance.player.transform.position = packet.Position;
|
||||
SRSingleton<SceneContext>.Instance.player.transform.eulerAngles = new Vector3(0, packet.Rotation, 0);
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.SetCurrRegionSet((RegionRegistry.RegionSetId)packet.RegionSet);
|
||||
// play the teleport animation
|
||||
SRSingleton<Overlay>.Instance.PlayTeleport();
|
||||
}
|
||||
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Destination not found");
|
||||
}
|
||||
|
||||
}
|
||||
break;
|
||||
case "listplayers":
|
||||
{
|
||||
ConsoleLog("Players:");
|
||||
foreach (var player in Globals.Players.Values)
|
||||
{
|
||||
SRMP.Log(player.Username);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "sleep":
|
||||
{
|
||||
if (args.Length == 1)
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.TimeDirector.FastForwardTo(SRSingleton<SceneContext>.Instance.TimeDirector.HoursFromNow(float.Parse(args[0])));
|
||||
ConsoleLog("Sleeoing for " + args[0] + " hours");
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Usage: sleep <hours>");
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "console": //add toggle option for turning on and off logging types
|
||||
|
||||
if (args.Length > 1 && (args[0] == "enable" || args[0] == "disable"))
|
||||
{
|
||||
bool enable = args[0] == "enable";
|
||||
//double check type
|
||||
if (Enum.TryParse(args[1], true, out LogType logType)) //check for main log type
|
||||
{
|
||||
if (enable)
|
||||
{
|
||||
if (!blockMessages.Contains(logType)) blockMessages.Remove(logType);
|
||||
ConsoleLog(logType.ToString() + " Messages Enabled");
|
||||
}
|
||||
else
|
||||
{
|
||||
if (blockMessages.Contains(logType)) blockMessages.Add(logType);
|
||||
ConsoleLog("[Console] " + logType.ToString() + " Messages Disabled");
|
||||
}
|
||||
}
|
||||
else if (Enum.TryParse(args[1], true, out LogItems logMessage)) //check for log message item types
|
||||
{
|
||||
if (enable)
|
||||
{
|
||||
if (!blockLogs.Contains(logMessage)) blockLogs.Remove(logMessage);
|
||||
ConsoleLog(logMessage.ToString() + " Log Messages Enabled");
|
||||
}
|
||||
else
|
||||
{
|
||||
if (blockLogs.Contains(logMessage)) blockLogs.Add(logMessage);
|
||||
ConsoleLog(logMessage.ToString() + " Log Messages Disabled");
|
||||
}
|
||||
}
|
||||
else if (args[1].Equals("stacktrace", StringComparison.InvariantCultureIgnoreCase) || args[1].Equals("stack_trace", StringComparison.InvariantCultureIgnoreCase))
|
||||
{
|
||||
if (enable)
|
||||
{
|
||||
DisplayTrace = true;
|
||||
ConsoleLog("Stack Trace Information Enabled");
|
||||
}
|
||||
else
|
||||
{
|
||||
DisplayTrace = false;
|
||||
ConsoleLog("Stack Trace Information Disabled");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Invalid Feed back Type");
|
||||
ConsoleLog("Valid Types: ");
|
||||
ConsoleLog(string.Join(", ", Enum.GetValues(typeof(LogType))));
|
||||
ConsoleLog(string.Join(", ", Enum.GetValues(typeof(LogItems))));
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ConsoleLog("Usage: console <enable/disable> <feedbackType>");
|
||||
}
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// Update the input every frame
|
||||
// This gets new key input and calls the OnInputText callback
|
||||
//
|
||||
void Update()
|
||||
{
|
||||
input.Update();
|
||||
}
|
||||
|
||||
//
|
||||
// It's important to call console.ShutDown in OnDestroy
|
||||
// because compiling will error out in the editor if you don't
|
||||
// because we redirected output. This sets it back to normal.
|
||||
//
|
||||
void OnDestroy()
|
||||
{
|
||||
console.Shutdown();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -13,7 +13,11 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
private float fadeTime;
|
||||
private List<ChatMessage> messages = new List<ChatMessage>();
|
||||
private Vector2 chatScroll;
|
||||
int maxWidth = 290;
|
||||
|
||||
/// <summary>
|
||||
/// create a chat message to be displayed
|
||||
/// </summary>
|
||||
public class ChatMessage
|
||||
{
|
||||
public string Text;
|
||||
|
@ -27,52 +31,68 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
Time = DateTime.Now;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// On chat ui update triggered, handle it
|
||||
/// </summary>
|
||||
private void Update()
|
||||
{
|
||||
//only display chat if in multiplayer mode
|
||||
if (!Globals.IsMultiplayer)
|
||||
{
|
||||
openChat = false;
|
||||
message = "";
|
||||
return;
|
||||
}
|
||||
|
||||
//if enter is pressed start chat typing mode
|
||||
if (Input.GetKeyUp(KeyCode.Return))
|
||||
{
|
||||
StartCoroutine(FocusChat());
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Crreate the Chat gui
|
||||
/// </summary>
|
||||
private void OnGUI()
|
||||
{
|
||||
//only mess with the gui in multiplayer mode
|
||||
if (!Globals.IsMultiplayer) return;
|
||||
|
||||
if (openChat)
|
||||
{
|
||||
GUILayout.BeginArea(new Rect(20, Screen.height / 2, 500, 300), GUI.skin.box);
|
||||
//draw chat area
|
||||
GUILayout.BeginArea(new Rect(20, Screen.height / 2, maxWidth + 10, 300), GUI.skin.box);
|
||||
chatScroll = GUILayout.BeginScrollView(chatScroll);
|
||||
var skin = GUI.skin.box;
|
||||
skin.wordWrap = true;
|
||||
skin.alignment = TextAnchor.MiddleLeft;
|
||||
|
||||
//add each mesage into the chat box scroller
|
||||
foreach (var msg in messages)
|
||||
{
|
||||
GUILayout.Label(wrapString(msg.Text, 490), skin, GUILayout.MaxWidth(490));
|
||||
GUILayout.Label(wrapString(msg.Text, maxWidth), skin, GUILayout.MaxWidth(maxWidth));
|
||||
}
|
||||
GUILayout.EndScrollView();
|
||||
|
||||
//add display for text input area
|
||||
GUI.SetNextControlName("ChatInput");
|
||||
message = GUILayout.TextField(message);
|
||||
GUILayout.EndArea();
|
||||
|
||||
//focus the input
|
||||
GUI.FocusControl("ChatInput");
|
||||
|
||||
//watch for changes to the text input
|
||||
Event e = Event.current;
|
||||
if (e.rawType == EventType.KeyUp && e.keyCode == KeyCode.Return)
|
||||
{
|
||||
//on send close chat
|
||||
openChat = !openChat;
|
||||
if (!string.IsNullOrWhiteSpace(message))
|
||||
{
|
||||
//if server send message to all plauers
|
||||
if (Globals.IsServer)
|
||||
{
|
||||
//if server send it to the server
|
||||
AddChatMessage(Globals.Username + ": " + message);
|
||||
new PacketPlayerChat()
|
||||
{
|
||||
|
@ -81,6 +101,7 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
}
|
||||
else
|
||||
{
|
||||
//if player send it to the server
|
||||
new PacketPlayerChat()
|
||||
{
|
||||
message = message
|
||||
|
@ -90,13 +111,16 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
else //if chat isnt open yet
|
||||
{
|
||||
GUILayout.BeginArea(new Rect(20, Screen.height / 2, 500, 300));
|
||||
//draw chat area
|
||||
GUILayout.BeginArea(new Rect(20, Screen.height / 2, maxWidth+ 10, 300));
|
||||
chatScroll = GUILayout.BeginScrollView(chatScroll);
|
||||
var skin = GUI.skin.box;
|
||||
skin.wordWrap = true;
|
||||
skin.alignment = TextAnchor.MiddleLeft;
|
||||
|
||||
//add each mesage into the chat box scroller
|
||||
foreach (var msg in messages)
|
||||
{
|
||||
if (msg.FadeTime > 0f)
|
||||
|
@ -105,26 +129,34 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
var c = GUI.color;
|
||||
c.a = (msg.FadeTime / 5f);
|
||||
GUI.color = c;
|
||||
GUILayout.Label(wrapString(msg.Text, 490), skin, GUILayout.MaxWidth(490));
|
||||
GUILayout.Label(wrapString(msg.Text, maxWidth), skin, GUILayout.MaxWidth(maxWidth));
|
||||
}
|
||||
}
|
||||
GUILayout.EndScrollView();
|
||||
GUILayout.EndArea();
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Add a chat message to be displayed
|
||||
/// </summary>
|
||||
/// <param name="message">Message to display</param>
|
||||
public void AddChatMessage(string message)
|
||||
{
|
||||
fadeTime = 10f;
|
||||
messages.Add(new ChatMessage(message));
|
||||
chatScroll = new Vector2(0, 100000);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Trigger a full clear of the chat
|
||||
/// </summary>
|
||||
public void Clear()
|
||||
{
|
||||
messages.Clear();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Trigger focus on the chat box for when the uer hits enter
|
||||
/// </summary>
|
||||
private IEnumerator FocusChat()
|
||||
{
|
||||
yield return new WaitForEndOfFrame();
|
||||
|
@ -133,6 +165,12 @@ public class ChatUI : SRSingleton<ChatUI>
|
|||
GUI.FocusControl("ChatInput");
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Create the wrapped string to display in the chat box
|
||||
/// </summary>
|
||||
/// <param name="msg">Message to display</param>
|
||||
/// <param name="width">Width of the chat box</param>
|
||||
/// <returns></returns>
|
||||
string wrapString(string msg, int width)
|
||||
{
|
||||
string[] words = msg.Split(" "[0]);
|
|
@ -8,18 +8,38 @@ using SRMultiplayer.Networking;
|
|||
|
||||
public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
||||
{
|
||||
private Rect windowRect = new Rect(Screen.width - 300, 20, 300, 500);
|
||||
private Rect windowRect = new Rect(Screen.width - 300 - 20, 20, 300, 500);
|
||||
private Vector2 playersScroll = Vector2.zero;
|
||||
private string ipaddress = "localhost";
|
||||
private string port = "16500";
|
||||
private string servercode = "";
|
||||
private bool menuOpen;
|
||||
|
||||
//use internal name with getter and setter to allow the menu panel to remember the users last choice
|
||||
private int _menuOpen;
|
||||
private int menuOpen //menu open hold the current menu state (0 colapsed, 1 minimized, 2 open)
|
||||
{
|
||||
get
|
||||
{
|
||||
return _menuOpen;
|
||||
}
|
||||
set
|
||||
{
|
||||
_menuOpen = value;
|
||||
|
||||
//save users setting
|
||||
PlayerPrefs.SetInt("SRMP_Menu", menuOpen);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private string username;
|
||||
private float lastCodeUse;
|
||||
private ConnectError error;
|
||||
private ConnectHelp help;
|
||||
private string errorMessage;
|
||||
|
||||
|
||||
|
||||
public enum ConnectError
|
||||
{
|
||||
None,
|
||||
|
@ -36,18 +56,30 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
Hosting,
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Creation of gui with side panel is last display state
|
||||
/// </summary>
|
||||
public override void Awake()
|
||||
{
|
||||
base.Awake();
|
||||
|
||||
//set default ui location width adapting numbers for smaller resolutions
|
||||
float width = 300;
|
||||
if (Screen.width / 4 < width) width = Screen.width / 4;
|
||||
windowRect = new Rect(Screen.width - width - 20, 20, width, 500);
|
||||
|
||||
Globals.Username = PlayerPrefs.GetString("SRMP_Username", "");
|
||||
ipaddress = PlayerPrefs.GetString("SRMP_IP", "localhost");
|
||||
port = PlayerPrefs.GetString("SRMP_Port", "16500");
|
||||
|
||||
menuOpen = true;
|
||||
menuOpen = PlayerPrefs.GetInt("SRMP_Menu", 2); ; //start panel open by default
|
||||
|
||||
username = Globals.Username;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Update of panel display
|
||||
/// </summary>
|
||||
private void Update()
|
||||
{
|
||||
if (lastCodeUse > 0f)
|
||||
|
@ -60,30 +92,181 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
}
|
||||
|
||||
|
||||
//this sets presskey senarios
|
||||
if (Input.GetKeyDown(KeyCode.F4))
|
||||
{
|
||||
menuOpen = !menuOpen;
|
||||
//use the f4 key to swap between collapsed and minimized
|
||||
menuOpen = menuOpen == 2 ? 0 : 2;
|
||||
}
|
||||
if (Input.GetKeyDown(KeyCode.F3))
|
||||
{
|
||||
//use the f3 key to set minimized or maximized
|
||||
menuOpen = menuOpen < 2 ? 2 : 1;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handle draw event of the gui
|
||||
/// </summary>
|
||||
private void OnGUI()
|
||||
{
|
||||
if (!menuOpen || (!Levels.isMainMenu() && !Globals.IsMultiplayer && Globals.PauseState != PauseState.Pause)) return;
|
||||
//verify on a window that the menu can be drawn on
|
||||
if ((!Levels.isMainMenu() && !Globals.IsMultiplayer && Globals.PauseState != PauseState.Pause)) return;
|
||||
if (Globals.IsMultiplayer && Globals.PauseState != PauseState.Pause) return;
|
||||
|
||||
//if yes draw the window for the given state
|
||||
if (SceneManager.GetActiveScene().buildIndex >= 2)
|
||||
{
|
||||
windowRect = GUI.Window(1, windowRect, MultiplayerWindow, "SRMP v" + Globals.Version);
|
||||
//check to make sure the panel is taking less than 25% of the screen if possible
|
||||
float width = 300;
|
||||
if(Screen.width/ 4 < width) width = Screen.width/4;
|
||||
windowRect.width = width;
|
||||
|
||||
//check to see if the windows needs to move due to it being off the screen from size change
|
||||
//also prevent the user from dragging it off the screen
|
||||
if (windowRect.x + 20 + windowRect.width > Screen.width) windowRect.x = Screen.width - windowRect.width - 20;
|
||||
if (windowRect.y + 20 + windowRect.height > Screen.height) windowRect.y = (Screen.height - windowRect.height - 20) >= 20 ? (Screen.height - windowRect.height - 20) : 20;
|
||||
if (windowRect.x < 20) windowRect.x = 20;
|
||||
if (windowRect.y < 20) windowRect.y = 20;
|
||||
|
||||
|
||||
|
||||
//drawn in the window
|
||||
switch (menuOpen)
|
||||
{
|
||||
case 0: //collapsed
|
||||
windowRect.height = 50;
|
||||
windowRect = GUILayout.Window(1, windowRect, ClosedWindow, "SRMP v" + Globals.Version);
|
||||
break;
|
||||
case 1: //minimized
|
||||
windowRect.height = 100;
|
||||
windowRect = GUILayout.Window(1, windowRect, MiniWindow, "SRMP v" + Globals.Version);
|
||||
break;
|
||||
case 2: //open
|
||||
windowRect.height = 500;
|
||||
windowRect = GUILayout.Window(1, windowRect, MultiplayerWindow, "SRMP v" + Globals.Version);
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
/// <summary>
|
||||
/// display for the function keyps section of the gui
|
||||
/// </summary>
|
||||
private void FunctionKeys()
|
||||
{
|
||||
if (menuOpen != 0)
|
||||
{
|
||||
GUILayout.Label("Press Button or Key To Change Style");
|
||||
}
|
||||
GUILayout.BeginHorizontal();
|
||||
if (GUILayout.Button(menuOpen == 1 ? "F3 - Full" : "F3 - Mini"))
|
||||
{
|
||||
menuOpen = menuOpen == 1 ? 2 : 1;
|
||||
}
|
||||
GUILayout.FlexibleSpace();
|
||||
if (GUILayout.Button(menuOpen == 0 ? "F4 - Full" : "F4 - Colapsed"))
|
||||
{
|
||||
menuOpen = menuOpen == 0 ? 2 : 0;
|
||||
}
|
||||
GUILayout.EndHorizontal();
|
||||
}
|
||||
/// <summary>
|
||||
/// Sets the window display for collapsed
|
||||
/// Only activated if the id is the id for the window
|
||||
/// </summary>
|
||||
private void ClosedWindow(int id)
|
||||
{
|
||||
if (id != 1) return;
|
||||
//display f3 and f4 commands
|
||||
FunctionKeys();
|
||||
|
||||
GUI.DragWindow(new Rect(0, 0, 10000, 10000));
|
||||
}
|
||||
/// <summary>
|
||||
/// Sets the window display for summary mode
|
||||
/// Only activated if the id is the id for the window
|
||||
/// </summary>
|
||||
private void MiniWindow(int id)
|
||||
{
|
||||
if (id != 1) return;
|
||||
|
||||
//display f3 and f4 commands
|
||||
FunctionKeys();
|
||||
|
||||
//show username and current connection status
|
||||
GUIStyle standard = new GUIStyle(GUI.skin.label);
|
||||
GUIStyle red = new GUIStyle(GUI.skin.label);
|
||||
red.normal.textColor = Color.red;
|
||||
|
||||
|
||||
if (string.IsNullOrWhiteSpace(Globals.Username))
|
||||
{
|
||||
GUILayout.Label("Username: Not Set", red);
|
||||
}
|
||||
else
|
||||
{
|
||||
GUILayout.Label("Username: " + Globals.Username);
|
||||
|
||||
if (Globals.IsServer)
|
||||
{
|
||||
GUILayout.Label("Status: Host");
|
||||
}
|
||||
else if (Globals.IsClient)
|
||||
{
|
||||
|
||||
GUIStyle green = new GUIStyle(GUI.skin.label);
|
||||
green.normal.textColor = Color.green;
|
||||
GUILayout.Label("Status: Client", green);
|
||||
}
|
||||
else
|
||||
{
|
||||
bool canHost = true;
|
||||
//check if user can host the session
|
||||
if (!int.TryParse(port, out int numport) || numport < 1000 || numport > 65000)
|
||||
{
|
||||
canHost = false;
|
||||
}
|
||||
|
||||
GUILayout.BeginHorizontal();
|
||||
GUILayout.Label("Status: Disconnected", red);
|
||||
|
||||
GUILayout.FlexibleSpace();
|
||||
if (canHost)
|
||||
{
|
||||
//only show host if not main menu
|
||||
if (!Levels.isMainMenu())
|
||||
{
|
||||
if (GUILayout.Button("Host"))
|
||||
{
|
||||
NetworkServer.Instance.StartServer(numport);
|
||||
SaveSettings();
|
||||
}
|
||||
}
|
||||
}
|
||||
GUILayout.EndHorizontal();
|
||||
if (!canHost)
|
||||
{
|
||||
GUILayout.Label("Invalid Port Settings", red);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
GUI.DragWindow(new Rect(0, 0, 10000, 10000));
|
||||
}
|
||||
/// <summary>
|
||||
/// Sets the window display for full display mode
|
||||
/// Only activated if the id is the id for the window
|
||||
/// </summary>
|
||||
private void MultiplayerWindow(int id)
|
||||
{
|
||||
if (id != 1) return;
|
||||
|
||||
GUILayout.Label("You can close this menu with F4");
|
||||
GUILayout.Space(20);
|
||||
//display f3 and f4 commands
|
||||
FunctionKeys();
|
||||
|
||||
//now display standard
|
||||
if (string.IsNullOrWhiteSpace(Globals.Username))
|
||||
{
|
||||
UsernameGUI();
|
||||
|
@ -129,6 +312,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
GUI.DragWindow(new Rect(0, 0, 10000, 10000));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the active server info part of the gui
|
||||
/// </summary>
|
||||
private void ServerGUI()
|
||||
{
|
||||
GUILayout.Label("You are the server");
|
||||
|
@ -151,7 +337,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
GUILayout.EndScrollView();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the client info part of the gui
|
||||
/// </summary>
|
||||
private void ClientGUI()
|
||||
{
|
||||
GUILayout.Label("You are a client");
|
||||
|
@ -169,7 +357,11 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
GUILayout.EndScrollView();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the connection information of the gui
|
||||
/// this section includes user information,
|
||||
/// how to and other imbedded sections for handling display
|
||||
/// </summary>
|
||||
private void ConnectGUI()
|
||||
{
|
||||
GUILayout.Label("Username: " + Globals.Username);
|
||||
|
@ -250,6 +442,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the hosting info part of the gui
|
||||
/// </summary>
|
||||
private void HostGUI()
|
||||
{
|
||||
GUILayout.Label("Username: " + Globals.Username);
|
||||
|
@ -277,7 +472,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the Help info part of the gui with instructions for hosting
|
||||
/// </summary>
|
||||
private void HelpGUI()
|
||||
{
|
||||
switch (help)
|
||||
|
@ -301,7 +498,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the Error summaries in the gui
|
||||
/// </summary>
|
||||
private void ErrorGUI()
|
||||
{
|
||||
switch (error)
|
||||
|
@ -354,7 +553,9 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Display the Current user information
|
||||
/// </summary>
|
||||
private void UsernameGUI()
|
||||
{
|
||||
GUILayout.BeginHorizontal();
|
||||
|
@ -376,14 +577,18 @@ public class MultiplayerUI : SRSingleton<MultiplayerUI>
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Saves the gui settings for the user
|
||||
/// </summary>
|
||||
private void SaveSettings()
|
||||
{
|
||||
PlayerPrefs.SetString("SRMP_Username", Globals.Username);
|
||||
PlayerPrefs.SetString("SRMP_IP", ipaddress);
|
||||
PlayerPrefs.GetString("SRMP_Port", port);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handles connection resonces display when connection is lost from the server
|
||||
/// </summary>
|
||||
public void ConnectResponse(ConnectError connectError, string message = "")
|
||||
{
|
||||
lastCodeUse = 0f;
|
|
@ -8,6 +8,24 @@
|
|||
//using UnityEngine;
|
||||
//using UnityEngine.SceneManagement;
|
||||
|
||||
/*
|
||||
* TestUi is a version of the multiplayer ui that adds the following information
|
||||
* Added
|
||||
Chat
|
||||
Cheats
|
||||
|
||||
|
||||
* More stats
|
||||
Quick silver race is active
|
||||
client status
|
||||
server status
|
||||
server code
|
||||
send and recieve sizes
|
||||
packet sizes
|
||||
*/
|
||||
|
||||
|
||||
|
||||
//namespace SRMultiplayer
|
||||
//{
|
||||
// class TestUI : SRSingleton<TestUI>
|
||||
|
@ -551,8 +569,8 @@
|
|||
// GUILayout.Label($"Server Code: {Globals.ServerCode}");
|
||||
// }
|
||||
|
||||
// //GUILayout.Label("Received Messages: " + SRSingleton<NetworkClient>.Instance.Statistics.ReceivedMessages);
|
||||
// //GUILayout.Label("Send Messages: " + SRSingleton<NetworkClient>.Instance.Statistics.SentMessages);
|
||||
// GUILayout.Label("Received Messages: " + SRSingleton<NetworkClient>.Instance.Statistics.ReceivedMessages);
|
||||
// GUILayout.Label("Send Messages: " + SRSingleton<NetworkClient>.Instance.Statistics.SentMessages);
|
||||
|
||||
// if (NetworkServer.Instance != null && NetworkServer.Instance.Status == NetworkServer.ServerStatus.Running)
|
||||
// {
|
|
@ -11,12 +11,12 @@ namespace SRMultiplayer
|
|||
{
|
||||
public static class Globals
|
||||
{
|
||||
|
||||
//setup global objects for refrence usage
|
||||
public static int Version;
|
||||
public static UserData UserData;
|
||||
public static GameObject BeatrixModel;
|
||||
public static RuntimeAnimatorController BeatrixController;
|
||||
public static GameObject IngameMultiplayerMenuPrefab;
|
||||
public static GameObject MainMultiplayerMenuPrefab;
|
||||
public static Dictionary<byte, NetworkPlayer> Players = new Dictionary<byte, NetworkPlayer>();
|
||||
public static string Username;
|
||||
public static string ServerCode;
|
||||
|
@ -53,6 +53,10 @@ namespace SRMultiplayer
|
|||
public static List<string> LemonTrees = new List<string>();
|
||||
public static Dictionary<PacketType, long> PacketSize = new Dictionary<PacketType, long>();
|
||||
|
||||
/// <summary>
|
||||
/// get list of current installed mods
|
||||
/// Excluding supporting files
|
||||
/// </summary>
|
||||
public static List<string> Mods
|
||||
{
|
||||
get
|
||||
|
|
|
@ -14,6 +14,10 @@ using UnityEngine;
|
|||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
/// <summary>
|
||||
/// Handles mod being loaded from SRML
|
||||
/// Takes the place of the Main Standalone file, but caters to the SRML
|
||||
/// </summary>
|
||||
public class MainSRML : ModEntryPoint
|
||||
{
|
||||
private static GameObject m_GameObject;
|
||||
|
@ -35,10 +39,12 @@ namespace SRMultiplayer
|
|||
|
||||
SRMP.Log("Loading SRMP SRML Version");
|
||||
|
||||
//create the mod directory in the install folder if needed
|
||||
if (!Directory.Exists(SRMP.ModDataPath))
|
||||
{
|
||||
Directory.CreateDirectory(SRMP.ModDataPath);
|
||||
}
|
||||
//create the user data file if not created yet
|
||||
if (!File.Exists(Path.Combine(SRMP.ModDataPath, "userdata.json")))
|
||||
{
|
||||
Globals.UserData = new UserData()
|
||||
|
@ -50,7 +56,7 @@ namespace SRMultiplayer
|
|||
File.WriteAllText(Path.Combine(SRMP.ModDataPath, "userdata.json"), JsonConvert.SerializeObject(Globals.UserData));
|
||||
SRMP.Log("Created userdata with UUID " + Globals.UserData.UUID);
|
||||
}
|
||||
else
|
||||
else //if alreayd created load in the data
|
||||
{
|
||||
Globals.UserData = JsonConvert.DeserializeObject<UserData>(File.ReadAllText(Path.Combine(SRMP.ModDataPath, "userdata.json")));
|
||||
if(Globals.UserData.IgnoredMods == null)
|
||||
|
@ -60,6 +66,7 @@ namespace SRMultiplayer
|
|||
SRMP.Log("Loaded userdata with UUID " + Globals.UserData.UUID);
|
||||
}
|
||||
|
||||
//create the mods main game objects and start connecting everything
|
||||
string[] args = System.Environment.GetCommandLineArgs();
|
||||
|
||||
m_GameObject = new GameObject("SRMP");
|
||||
|
@ -71,12 +78,16 @@ namespace SRMultiplayer
|
|||
m_GameObject.AddComponent<ChatUI>();
|
||||
m_GameObject.AddComponent<SRMPConsole>();
|
||||
|
||||
//mark all mod objects and do not destroy
|
||||
GameObject.DontDestroyOnLoad(m_GameObject);
|
||||
|
||||
//get current mod version
|
||||
Globals.Version = Assembly.GetExecutingAssembly().GetName().Version.Revision;
|
||||
|
||||
//mark the mod as a background task
|
||||
Application.runInBackground = true;
|
||||
|
||||
//initialize connect to the harmony patcher
|
||||
HarmonyPatcher.GetInstance().PatchAll(Assembly.GetExecutingAssembly());
|
||||
}
|
||||
|
||||
|
|
|
@ -8,7 +8,9 @@ using UnityCoreMod;
|
|||
using UnityEngine;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
{ // <summary>
|
||||
/// Handles mod being loaded from directly without the mod loader
|
||||
/// </summary>
|
||||
public class MainSaty : IUnityMod
|
||||
{
|
||||
private static GameObject m_GameObject;
|
||||
|
@ -19,10 +21,12 @@ namespace SRMultiplayer
|
|||
|
||||
SRMP.Log("Loading SRMP Standalone Version");
|
||||
|
||||
//create the mod directory in the install folder if needed
|
||||
if (!Directory.Exists(SRMP.ModDataPath))
|
||||
{
|
||||
Directory.CreateDirectory(SRMP.ModDataPath);
|
||||
}
|
||||
//create the user data file if not created yet
|
||||
if (!File.Exists(Path.Combine(SRMP.ModDataPath, "userdata.json")))
|
||||
{
|
||||
Globals.UserData = new UserData()
|
||||
|
@ -34,8 +38,9 @@ namespace SRMultiplayer
|
|||
File.WriteAllText(Path.Combine(SRMP.ModDataPath, "userdata.json"), JsonConvert.SerializeObject(Globals.UserData));
|
||||
SRMP.Log("Created userdata with UUID " + Globals.UserData.UUID);
|
||||
}
|
||||
else
|
||||
else //if alreayd created load in the data
|
||||
{
|
||||
|
||||
Globals.UserData = JsonConvert.DeserializeObject<UserData>(File.ReadAllText(Path.Combine(SRMP.ModDataPath, "userdata.json")));
|
||||
if(Globals.UserData.IgnoredMods == null)
|
||||
{
|
||||
|
@ -44,6 +49,7 @@ namespace SRMultiplayer
|
|||
SRMP.Log("Loaded userdata with UUID " + Globals.UserData.UUID);
|
||||
}
|
||||
|
||||
//create the mods main game objects and start connecting everything
|
||||
string[] args = System.Environment.GetCommandLineArgs();
|
||||
|
||||
m_GameObject = new GameObject("SRMP");
|
||||
|
@ -55,13 +61,17 @@ namespace SRMultiplayer
|
|||
m_GameObject.AddComponent<ChatUI>();
|
||||
m_GameObject.AddComponent<SRMPConsole>();
|
||||
|
||||
//mark all mod objects and do not destroy
|
||||
GameObject.DontDestroyOnLoad(m_GameObject);
|
||||
|
||||
//get current mod version
|
||||
Globals.Version = Assembly.GetExecutingAssembly().GetName().Version.Revision;
|
||||
|
||||
//initialize harmony and init the patches
|
||||
var harmony = new Harmony("saty.mod.srmp");
|
||||
harmony.PatchAll(Assembly.GetExecutingAssembly());
|
||||
|
||||
//mark the mod as a background task
|
||||
Application.runInBackground = true;
|
||||
}
|
||||
|
||||
|
|
|
@ -1,16 +1,12 @@
|
|||
using Assets.Script.Util.Extensions;
|
||||
using DG.Tweening;
|
||||
using DG.Tweening;
|
||||
using Lidgren.Network;
|
||||
using MonomiPark.SlimeRancher.DataModel;
|
||||
using MonomiPark.SlimeRancher.Persist;
|
||||
using MonomiPark.SlimeRancher.Regions;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using UnityEngine;
|
||||
|
||||
namespace SRMultiplayer.Networking
|
||||
|
@ -24,22 +20,24 @@ namespace SRMultiplayer.Networking
|
|||
Globals.PacketSize[type] += im.LengthBytes;
|
||||
switch (type)
|
||||
{
|
||||
//Player amimations
|
||||
case PacketType.PlayerAnimation: OnPlayerAnimation(new PacketPlayerAnimation(im)); break;
|
||||
|
||||
//Players
|
||||
case PacketType.PlayerJoined: OnPlayerJoined(new PacketPlayerJoined(im)); break;
|
||||
case PacketType.PlayerLeft: OnPlayerLeft(new PacketPlayerLeft(im)); break;
|
||||
case PacketType.PlayerLoaded: OnPlayerLoaded(new PacketPlayerLoaded(im)); break;
|
||||
case PacketType.PlayerPosition: OnPlayerPosition(new PacketPlayerPosition(im)); break;
|
||||
case PacketType.PlayerAnimationLayer: OnPlayerAnimationLayer(im); break;
|
||||
case PacketType.PlayerAnimationParameters: OnPlayerAnimationParameters(im); break;
|
||||
case PacketType.PlayerAnimationSpeed: OnPlayerAnimationSpeed(im); break;
|
||||
case PacketType.PlayerFX: OnPlayerFX(new PacketPlayerFX(im)); break;
|
||||
case PacketType.PlayerCurrency: OnPlayerCurrency(new PacketPlayerCurrency(im)); break;
|
||||
case PacketType.PlayerCurrencyDisplay: OnPlayerCurrencyDisplay(new PacketPlayerCurrencyDisplay(im)); break;
|
||||
case PacketType.PlayerUpgrade: OnPlayerUpgrade(new PacketPlayerUpgrade(im)); break;
|
||||
case PacketType.PlayerUpgradeUnlock: OnPlayerUpgradeUnlock(new PacketPlayerUpgradeUnlock(im)); break;
|
||||
case PacketType.PlayerChat: OnPlayerChat(new PacketPlayerChat(im)); break;
|
||||
|
||||
// Region
|
||||
case PacketType.RegionOwner: OnRegionOwner(new PacketRegionOwner(im)); break;
|
||||
|
||||
//Actors
|
||||
case PacketType.Actors: OnActors(new PacketActors(im)); break;
|
||||
case PacketType.ActorSpawn: OnActorSpawn(new PacketActorSpawn(im)); break;
|
||||
|
@ -153,7 +151,7 @@ namespace SRMultiplayer.Networking
|
|||
case PacketType.RaceTime: OnRaceTime(new PacketRaceTime(im)); break;
|
||||
case PacketType.RaceTrigger: OnRaceTrigger(new PacketRaceTrigger(im)); break;
|
||||
default:
|
||||
SRMP.Log($"Got unhandled packet: {type}");
|
||||
SRMP.Log($"Got unhandled packet: {type} " + Enum.GetName(typeof(PacketType), type));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -337,30 +335,51 @@ namespace SRMultiplayer.Networking
|
|||
#region Exchanges
|
||||
private static void OnExchangeTryAccept(PacketExchangeTryAccept packet)
|
||||
{
|
||||
//get the exchange type
|
||||
var type = (ExchangeDirector.OfferType)packet.Type;
|
||||
//check if current scene (view) contains the item in question
|
||||
if (SRSingleton<SceneContext>.Instance.ExchangeDirector.worldModel.currOffers.ContainsKey(type))
|
||||
{
|
||||
//handle the scene changes for the given offer
|
||||
var offer = SRSingleton<SceneContext>.Instance.ExchangeDirector.worldModel.currOffers[type];
|
||||
//cycle through requested items
|
||||
foreach (ExchangeDirector.RequestedItemEntry requestedItemEntry in offer.requests)
|
||||
{
|
||||
//check if the item can be accespted
|
||||
//is on the board and not already completed
|
||||
if (requestedItemEntry.id == (Identifiable.Id)packet.ID && !requestedItemEntry.IsComplete())
|
||||
{
|
||||
//mark submit to log
|
||||
SRMP.Log($"Exchange TryAccept for {(Identifiable.Id)packet.ID} ({(ExchangeDirector.OfferType)packet.Type}", "SERVER");
|
||||
//mark progress
|
||||
requestedItemEntry.progress++;
|
||||
|
||||
//if the given item completes the necesary quantity
|
||||
if (offer.IsComplete())
|
||||
{
|
||||
foreach (var rewarder in Resources.FindObjectsOfTypeAll<RancherProgressAwarder>())
|
||||
{
|
||||
rewarder.AwardIfType(type);
|
||||
}
|
||||
SRSingleton<SceneContext>.Instance.ExchangeDirector.ClearOffer(type);
|
||||
|
||||
//trigger fireworks
|
||||
//get ExchangeEjector
|
||||
foreach (var eject in Resources.FindObjectsOfTypeAll<ExchangeEjector>())
|
||||
{
|
||||
//send off fireworks
|
||||
SRBehaviour.InstantiateDynamic(eject.awardFX, eject.awardAt.position, eject.awardAt.rotation);
|
||||
}
|
||||
|
||||
//dont clear out the offer yet, we arent done with it
|
||||
//SRSingleton<SceneContext>.Instance.ExchangeDirector.ClearOffer(type);
|
||||
}
|
||||
|
||||
//trigger offer status changed
|
||||
SRSingleton<SceneContext>.Instance.ExchangeDirector.OfferDidChange();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnExchangePrepareDaily(PacketExchangePrepareDaily packet)
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.ExchangeDirector.worldModel.pendingOfferRancherIds = packet.pendingOfferRancherIds;
|
||||
|
@ -1906,30 +1925,23 @@ namespace SRMultiplayer.Networking
|
|||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerAnimationSpeed(NetIncomingMessage im)
|
||||
private static void OnPlayerAnimation(PacketPlayerAnimation packet)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
if (Globals.Players.TryGetValue(id, out NetworkPlayer player) && player.HasLoaded)
|
||||
{
|
||||
player.ReadAnimatorSpeed(im);
|
||||
}
|
||||
}
|
||||
//handle character animation triggers
|
||||
PacketPlayerAnimation.AnimationType type = (PacketPlayerAnimation.AnimationType)packet.Type;
|
||||
|
||||
private static void OnPlayerAnimationParameters(NetIncomingMessage im)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
if (Globals.Players.TryGetValue(id, out NetworkPlayer player) && player.HasLoaded)
|
||||
{
|
||||
player.ReadParameters(im);
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerAnimationLayer(NetIncomingMessage im)
|
||||
|
||||
if (Globals.Players.TryGetValue(packet.ID, out NetworkPlayer player) && player.HasLoaded)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
if (Globals.Players.TryGetValue(id, out NetworkPlayer player) && player.HasLoaded)
|
||||
if(type == PacketPlayerAnimation.AnimationType.Speed)
|
||||
player.ReadAnimatorSpeed(packet.internalData);
|
||||
else if (type == PacketPlayerAnimation.AnimationType.Parameters)
|
||||
{
|
||||
player.ReadAnimatorLayer(im);
|
||||
player.ReadParameters(packet.internalData);
|
||||
}
|
||||
else
|
||||
player.ReadAnimatorLayer(packet.internalData);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1941,6 +1953,9 @@ namespace SRMultiplayer.Networking
|
|||
{
|
||||
if (player.IsLocal)
|
||||
{
|
||||
if (packet.OnLoad)
|
||||
{
|
||||
|
||||
var euler = SRSingleton<SceneContext>.Instance.player.GetComponentInChildren<WeaponVacuum>().transform.eulerAngles;
|
||||
euler.x = packet.WeaponY;
|
||||
SRSingleton<SceneContext>.Instance.player.GetComponentInChildren<WeaponVacuum>().transform.eulerAngles = euler;
|
||||
|
@ -1948,6 +1963,8 @@ namespace SRMultiplayer.Networking
|
|||
SRSingleton<SceneContext>.Instance.player.transform.eulerAngles = new Vector3(0, packet.Rotation, 0);
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.SetCurrRegionSet((RegionRegistry.RegionSetId)packet.RegionSet);
|
||||
|
||||
|
||||
//only reload inventory if this is a load up packet and NOT a tp packet
|
||||
if (!Globals.IsServer)
|
||||
{
|
||||
try
|
||||
|
@ -1995,6 +2012,18 @@ namespace SRMultiplayer.Networking
|
|||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log("Player is being teleported", "CLIENT");
|
||||
|
||||
SRSingleton<SceneContext>.Instance.player.transform.position = packet.Position;
|
||||
SRSingleton<SceneContext>.Instance.player.transform.eulerAngles = new Vector3(0, packet.Rotation, 0);
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.SetCurrRegionSet((RegionRegistry.RegionSetId)packet.RegionSet);
|
||||
|
||||
SRSingleton<Overlay>.Instance.PlayTeleport();
|
||||
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
player.PositionRotationUpdate(packet.Position, packet.Rotation, false);
|
||||
player.UpdateWeaponRotation(packet.WeaponY);
|
|
@ -2,6 +2,7 @@
|
|||
using Lidgren.Network;
|
||||
using MonomiPark.SlimeRancher.DataModel;
|
||||
using MonomiPark.SlimeRancher.Regions;
|
||||
using Newtonsoft.Json.Linq;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
@ -21,12 +22,11 @@ namespace SRMultiplayer.Networking
|
|||
Globals.PacketSize[type] += im.LengthBytes;
|
||||
switch (type)
|
||||
{
|
||||
//Player animation
|
||||
case PacketType.PlayerAnimation: OnPlayerAnimation(new PacketPlayerAnimation(im), player); break;
|
||||
//Player
|
||||
case PacketType.PlayerLoaded: OnPlayerLoaded(new PacketPlayerLoaded(im), player); break;
|
||||
case PacketType.PlayerPosition: OnPlayerPosition(new PacketPlayerPosition(im), player); break;
|
||||
case PacketType.PlayerAnimationLayer: OnPlayerAnimationLayer(im, player); break;
|
||||
case PacketType.PlayerAnimationParameters: OnPlayerAnimationParameters(im, player); break;
|
||||
case PacketType.PlayerAnimationSpeed: OnPlayerAnimationSpeed(im, player); break;
|
||||
case PacketType.PlayerCurrency: OnPlayerCurrency(new PacketPlayerCurrency(im), player); break;
|
||||
case PacketType.PlayerCurrencyDisplay: OnPlayerCurrencyDisplay(new PacketPlayerCurrencyDisplay(im), player); break;
|
||||
case PacketType.PlayerUpgrade: OnPlayerUpgrade(new PacketPlayerUpgrade(im), player); break;
|
||||
|
@ -132,7 +132,7 @@ namespace SRMultiplayer.Networking
|
|||
case PacketType.RaceTime: OnRaceTime(new PacketRaceTime(im), player); break;
|
||||
case PacketType.RaceTrigger: OnRaceTrigger(new PacketRaceTrigger(im), player); break;
|
||||
default:
|
||||
SRMP.Log($"Got unhandled packet from {player}: {type}");
|
||||
SRMP.Log($"Got unhandled packet from {player}: {type}" + Enum.GetName(typeof(PacketType), type));
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -318,7 +318,17 @@ namespace SRMultiplayer.Networking
|
|||
{
|
||||
rewarder.AwardIfType(type);
|
||||
}
|
||||
SRSingleton<SceneContext>.Instance.ExchangeDirector.ClearOffer(type);
|
||||
|
||||
//trigger fireworks
|
||||
//get ExchangeEjector
|
||||
foreach (var eject in Resources.FindObjectsOfTypeAll<ExchangeEjector>())
|
||||
{
|
||||
//send off fireworks
|
||||
SRBehaviour.InstantiateDynamic(eject.awardFX, eject.awardAt.position, eject.awardAt.rotation);
|
||||
}
|
||||
|
||||
//dont clear out the offer yet, we arent done with it
|
||||
//SRSingleton<SceneContext>.Instance.ExchangeDirector.ClearOffer(type);
|
||||
}
|
||||
SRSingleton<SceneContext>.Instance.ExchangeDirector.OfferDidChange();
|
||||
}
|
||||
|
@ -1629,55 +1639,53 @@ namespace SRMultiplayer.Networking
|
|||
packet.SendToAllExcept(player);
|
||||
}
|
||||
|
||||
private static void OnPlayerAnimationSpeed(NetIncomingMessage im, NetworkPlayer player)
|
||||
private static void OnPlayerAnimation(PacketPlayerAnimation packet, NetworkPlayer player)
|
||||
{
|
||||
if (player.HasLoaded)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
player.ReadAnimatorSpeed(im);
|
||||
|
||||
NetOutgoingMessage om = NetworkServer.Instance.CreateMessage();
|
||||
om.Write(im);
|
||||
NetworkServer.Instance.SendToAll(om, player);
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerAnimationParameters(NetIncomingMessage im, NetworkPlayer player)
|
||||
switch (packet.Type)
|
||||
{
|
||||
case (byte)PacketPlayerAnimation.AnimationType.Speed:
|
||||
player.ReadAnimatorSpeed(packet.internalData);
|
||||
break;
|
||||
case (byte)PacketPlayerAnimation.AnimationType.Layer:
|
||||
player.ReadAnimatorLayer(packet.internalData);
|
||||
break;
|
||||
case (byte)PacketPlayerAnimation.AnimationType.Parameters:
|
||||
player.ReadParameters(packet.internalData);
|
||||
break;
|
||||
}
|
||||
|
||||
//make the incoming message an out going message
|
||||
packet.SendToAllExcept(player, NetDeliveryMethod.Unreliable);
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerPosition(PacketPlayerPosition packet, NetworkPlayer netPlayer)
|
||||
{
|
||||
//get player id from packet in case of a teleport
|
||||
NetworkPlayer player = Globals.Players.Values.FirstOrDefault(p => p.ID.Equals(packet.ID));
|
||||
|
||||
if (player.HasLoaded)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
player.ReadParameters(im);
|
||||
|
||||
NetOutgoingMessage om = NetworkServer.Instance.CreateMessage();
|
||||
om.Write(im);
|
||||
NetworkServer.Instance.SendToAll(om, player);
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerAnimationLayer(NetIncomingMessage im, NetworkPlayer player)
|
||||
if (player.IsLocal) //if the server player is the one being moved, teleport them
|
||||
{
|
||||
if(player.HasLoaded)
|
||||
{
|
||||
byte id = im.ReadByte();
|
||||
player.ReadAnimatorLayer(im);
|
||||
SRSingleton<SceneContext>.Instance.player.transform.position = packet.Position;
|
||||
SRSingleton<SceneContext>.Instance.player.transform.eulerAngles = new Vector3(0, packet.Rotation, 0);
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.SetCurrRegionSet((RegionRegistry.RegionSetId)packet.RegionSet);
|
||||
|
||||
NetOutgoingMessage om = NetworkServer.Instance.CreateMessage();
|
||||
om.Write(im);
|
||||
NetworkServer.Instance.SendToAll(om, player);
|
||||
SRSingleton<Overlay>.Instance.PlayTeleport();
|
||||
}
|
||||
}
|
||||
|
||||
private static void OnPlayerPosition(PacketPlayerPosition packet, NetworkPlayer player)
|
||||
{
|
||||
if (player.HasLoaded)
|
||||
else //else process player movement
|
||||
{
|
||||
player.PositionRotationUpdate(packet.Position, packet.Rotation, false);
|
||||
player.UpdateWeaponRotation(packet.WeaponY);
|
||||
player.CurrentRegionSet = (RegionRegistry.RegionSetId)packet.RegionSet;
|
||||
|
||||
packet.ID = player.ID;
|
||||
packet.SendToAllExcept(player, NetDeliveryMethod.Unreliable);
|
||||
packet.SendToAllExcept(netPlayer, NetDeliveryMethod.Unreliable);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
||||
|
|
@ -340,6 +340,7 @@ namespace SRMultiplayer.Networking
|
|||
{
|
||||
Globals.HandlePacket = true;
|
||||
PacketType type = (PacketType)im.ReadUInt16();
|
||||
|
||||
var player = Globals.Players.Values.FirstOrDefault(p => p.Connection != null && p.Connection.RemoteUniqueIdentifier == im.SenderConnection.RemoteUniqueIdentifier);
|
||||
if (player != null)
|
||||
{
|
|
@ -1,8 +1,10 @@
|
|||
using MonomiPark.SlimeRancher.Regions;
|
||||
using DG.Tweening;
|
||||
using MonomiPark.SlimeRancher.Regions;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Net.Sockets;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using UnityEngine;
|
||||
|
@ -237,6 +239,7 @@ namespace SRMultiplayer.Networking
|
|||
var exchangeBreakOnImpact = GetComponentInChildren<ExchangeBreakOnImpact>();
|
||||
if (exchangeBreakOnImpact != null)
|
||||
{
|
||||
SRMP.Log("Exchange Box Broke!", "ACTOR");
|
||||
SRBehaviour.SpawnAndPlayFX(exchangeBreakOnImpact.breakFX, exchangeBreakOnImpact.gameObject.transform.position, exchangeBreakOnImpact.gameObject.transform.rotation);
|
||||
}
|
||||
var breakOnImpact = GetComponentInChildren<BreakOnImpactBase>();
|
||||
|
|
|
@ -33,36 +33,51 @@ namespace SRMultiplayer.Networking
|
|||
|
||||
public void Burst()
|
||||
{
|
||||
//if object is in active mark the reach target
|
||||
if (gameObject.activeInHierarchy)
|
||||
{
|
||||
StartCoroutine(ReachedTarget());
|
||||
}
|
||||
else
|
||||
{
|
||||
//if not just dismiss the gordo completely
|
||||
Gordo.gameObject.SetActive(false);
|
||||
Gordo.SetEatenCount(-1);
|
||||
}
|
||||
}
|
||||
|
||||
//process the gordo burst reaction
|
||||
private IEnumerator ReachedTarget()
|
||||
{
|
||||
//start the burst and begin sounds and animations
|
||||
Gordo.WillStartBurst();
|
||||
Gordo.GetComponent<GordoFaceAnimator>().SetTrigger("Strain");
|
||||
SECTR_AudioSystem.Play(Gordo.strainCue, Gordo.transform.position, false);
|
||||
//wait for amination/sounds to finish
|
||||
yield return new WaitForSeconds(2f);
|
||||
SECTR_AudioSystem.Play(Gordo.burstCue, Gordo.transform.position, false);
|
||||
|
||||
//if the gordo has a destroy effect process it
|
||||
if (Gordo.destroyFX != null)
|
||||
{
|
||||
//play the spawn behavior for destroy events for the gordo that is bursting
|
||||
GameObject gameObject = SRBehaviour.SpawnAndPlayFX(Gordo.destroyFX, Gordo.transform.position + Vector3.up * 2f, Gordo.transform.rotation);
|
||||
//get the gordo slime type
|
||||
Identifiable component = Gordo.gameObject.GetComponent<Identifiable>();
|
||||
//get the color of the current gordo
|
||||
Color[] colors = SlimeUtil.GetColors(Gordo.gameObject, (component != null) ? component.id : Identifiable.Id.NONE, true);
|
||||
//get the slime children spawned by the gordo
|
||||
RecolorSlimeMaterial[] componentsInChildren = gameObject.GetComponentsInChildren<RecolorSlimeMaterial>();
|
||||
for (int i = 0; i < componentsInChildren.Length; i++)
|
||||
{
|
||||
//foreach slime in the count spawned by the gordo, set their coloring
|
||||
componentsInChildren[i].SetColors(colors[0], colors[1], colors[2]);
|
||||
}
|
||||
}
|
||||
//trigger the burst completed event
|
||||
Gordo.DidCompleteBurst();
|
||||
|
||||
//despawn the bursted gordo from game
|
||||
Gordo.gameObject.SetActive(false);
|
||||
Gordo.SetEatenCount(-1);
|
||||
yield break;
|
||||
|
|
|
@ -1,12 +1,10 @@
|
|||
using Lidgren.Network;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using UnityEngine;
|
||||
using static SRMultiplayer.Packets.PacketPlayerAnimation;
|
||||
|
||||
namespace SRMultiplayer.Networking
|
||||
{
|
||||
|
@ -62,33 +60,42 @@ namespace SRMultiplayer.Networking
|
|||
continue;
|
||||
}
|
||||
|
||||
NetOutgoingMessage writer = CreateMessage();
|
||||
writer.Write((ushort)PacketType.PlayerAnimationLayer);
|
||||
writer.Write(Globals.LocalID);
|
||||
WriteAnimatorLayer(writer, stateHash, normalizedTime, i, layerWeight[i]);
|
||||
Send(writer);
|
||||
//NetOutgoingMessage writer = CreateMessage();
|
||||
|
||||
//add the object to the writer but dont send it yet
|
||||
var packet = new PacketPlayerAnimation()
|
||||
{
|
||||
Type = (byte)PacketPlayerAnimation.AnimationType.Layer,
|
||||
ID = Globals.LocalID
|
||||
};
|
||||
|
||||
//add extra parameters
|
||||
WriteAnimatorLayer(packet, stateHash, normalizedTime, i, layerWeight[i]);
|
||||
|
||||
//send the changes
|
||||
packet.Send();
|
||||
}
|
||||
|
||||
CheckSpeed();
|
||||
}
|
||||
|
||||
void WriteAnimatorLayer(NetOutgoingMessage writer, int stateHash, float normalizedTime, int layerNumber, float layerWeight)
|
||||
void WriteAnimatorLayer(PacketPlayerAnimation writer, int stateHash, float normalizedTime, int layerNumber, float layerWeight)
|
||||
{
|
||||
writer.Write(stateHash);
|
||||
writer.Write(normalizedTime);
|
||||
writer.Write(layerNumber);
|
||||
writer.Write(layerWeight);
|
||||
writer.Add(stateHash);
|
||||
writer.Add(normalizedTime);
|
||||
writer.Add(layerNumber);
|
||||
writer.Add(layerWeight);
|
||||
WriteParameters(writer);
|
||||
}
|
||||
|
||||
public void ReadAnimatorLayer(NetIncomingMessage im)
|
||||
public void ReadAnimatorLayer(Queue<animateData> im)
|
||||
{
|
||||
if (m_Animator == null) return;
|
||||
|
||||
int stateHash = im.ReadInt32();
|
||||
float normalizedTime = im.ReadFloat();
|
||||
int layerNumber = im.ReadInt32();
|
||||
float layerWeight = im.ReadFloat();
|
||||
int stateHash = im.Dequeue().iData.Value;
|
||||
float normalizedTime = im.Dequeue().fData.Value;
|
||||
int layerNumber = im.Dequeue().iData.Value;
|
||||
float layerWeight = im.Dequeue().fData.Value;
|
||||
|
||||
if (stateHash != 0 && m_Animator.enabled)
|
||||
{
|
||||
|
@ -106,24 +113,33 @@ namespace SRMultiplayer.Networking
|
|||
if (Mathf.Abs(previousSpeed - newSpeed) > 0.001f)
|
||||
{
|
||||
previousSpeed = newSpeed;
|
||||
NetOutgoingMessage writer = CreateMessage();
|
||||
writer.Write((ushort)PacketType.PlayerAnimationSpeed);
|
||||
writer.Write(Globals.LocalID);
|
||||
WriteAnimatorSpeed(writer, newSpeed);
|
||||
Send(writer);
|
||||
}
|
||||
}
|
||||
//NetOutgoingMessage writer = CreateMessage();
|
||||
|
||||
void WriteAnimatorSpeed(NetOutgoingMessage writer, float newSpeed)
|
||||
//add the object to the writer but dont send it yet
|
||||
var packet = new PacketPlayerAnimation()
|
||||
{
|
||||
writer.Write(newSpeed);
|
||||
Type = (byte)PacketPlayerAnimation.AnimationType.Speed,
|
||||
ID = Globals.LocalID
|
||||
};
|
||||
|
||||
//add extra parameters
|
||||
WriteAnimatorSpeed(packet, newSpeed);
|
||||
|
||||
//send the speed change
|
||||
packet.Send();
|
||||
}
|
||||
}
|
||||
|
||||
public void ReadAnimatorSpeed(NetIncomingMessage im)
|
||||
void WriteAnimatorSpeed(PacketPlayerAnimation writer, float newSpeed)
|
||||
{
|
||||
writer.Add(newSpeed);
|
||||
}
|
||||
|
||||
public void ReadAnimatorSpeed(Queue<animateData> im)
|
||||
{
|
||||
if (m_Animator == null) return;
|
||||
|
||||
var newSpeed = im.ReadFloat();
|
||||
var newSpeed = im.Dequeue().fData.Value;
|
||||
// set m_Animator
|
||||
m_Animator.speed = newSpeed;
|
||||
m_AnimatorSpeed = newSpeed;
|
||||
|
@ -179,12 +195,17 @@ namespace SRMultiplayer.Networking
|
|||
{
|
||||
nextSendTime = now + syncInterval;
|
||||
|
||||
NetOutgoingMessage writer = CreateMessage();
|
||||
writer.Write((ushort)PacketType.PlayerAnimationParameters);
|
||||
writer.Write(Globals.LocalID);
|
||||
if (WriteParameters(writer))
|
||||
//add the object to the writer but dont send it yet
|
||||
var packet = new PacketPlayerAnimation()
|
||||
{
|
||||
Send(writer);
|
||||
Type = (byte)PacketPlayerAnimation.AnimationType.Parameters,
|
||||
ID = Globals.LocalID
|
||||
};
|
||||
|
||||
//add extra parameters
|
||||
if (WriteParameters(packet))
|
||||
{
|
||||
packet.Send();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -226,10 +247,10 @@ namespace SRMultiplayer.Networking
|
|||
return dirtyBits;
|
||||
}
|
||||
|
||||
bool WriteParameters(NetOutgoingMessage writer, bool forceAll = false)
|
||||
bool WriteParameters(PacketPlayerAnimation writer, bool forceAll = false)
|
||||
{
|
||||
ulong dirtyBits = forceAll ? (~0ul) : NextDirtyBits();
|
||||
writer.Write(dirtyBits);
|
||||
writer.Add(dirtyBits);
|
||||
for (int i = 0; i < parameters.Length; i++)
|
||||
{
|
||||
if ((dirtyBits & (1ul << i)) == 0)
|
||||
|
@ -239,53 +260,55 @@ namespace SRMultiplayer.Networking
|
|||
if (par.type == AnimatorControllerParameterType.Int)
|
||||
{
|
||||
int newIntValue = m_Animator.GetInteger(par.nameHash);
|
||||
writer.Write(newIntValue);
|
||||
writer.Add(newIntValue);
|
||||
}
|
||||
else if (par.type == AnimatorControllerParameterType.Float)
|
||||
{
|
||||
float newFloatValue = m_Animator.GetFloat(par.nameHash);
|
||||
writer.Write(newFloatValue);
|
||||
writer.Add(newFloatValue);
|
||||
}
|
||||
else if (par.type == AnimatorControllerParameterType.Bool)
|
||||
{
|
||||
bool newBoolValue = m_Animator.GetBool(par.nameHash);
|
||||
writer.Write(newBoolValue);
|
||||
writer.Add(newBoolValue);
|
||||
}
|
||||
}
|
||||
return dirtyBits != 0;
|
||||
}
|
||||
|
||||
public void ReadParameters(NetIncomingMessage reader)
|
||||
{
|
||||
public void ReadParameters(Queue<animateData> im)
|
||||
{ //make sure
|
||||
if (m_Animator == null) return;
|
||||
|
||||
bool m_AnimatorEnabled = m_Animator.enabled;
|
||||
// need to read values from NetworkReader even if m_Animator is disabled
|
||||
|
||||
ulong dirtyBits = reader.ReadUInt64();
|
||||
// need to read values from NetworkReader even if m_Animator is disabled
|
||||
ulong dirtyBits = im.Dequeue().uData.Value;
|
||||
for (int i = 0; i < parameters.Length; i++)
|
||||
{
|
||||
if ((dirtyBits & (1ul << i)) == 0)
|
||||
continue;
|
||||
|
||||
|
||||
AnimatorControllerParameter par = parameters[i];
|
||||
|
||||
if (par.type == AnimatorControllerParameterType.Int)
|
||||
{
|
||||
int newIntValue = reader.ReadInt32();
|
||||
int? newIntValue = im.Dequeue().iData;
|
||||
if (m_AnimatorEnabled)
|
||||
m_Animator.SetInteger(par.nameHash, newIntValue);
|
||||
m_Animator.SetInteger(par.nameHash, newIntValue.Value);
|
||||
}
|
||||
else if (par.type == AnimatorControllerParameterType.Float)
|
||||
{
|
||||
float newFloatValue = reader.ReadSingle();
|
||||
float? newFloatValue = im.Dequeue().fData;
|
||||
if (m_AnimatorEnabled)
|
||||
m_Animator.SetFloat(par.nameHash, newFloatValue);
|
||||
m_Animator.SetFloat(par.nameHash, newFloatValue.Value);
|
||||
}
|
||||
else if (par.type == AnimatorControllerParameterType.Bool)
|
||||
{
|
||||
bool newBoolValue = reader.ReadBoolean();
|
||||
bool? newBoolValue = im.Dequeue().bData;
|
||||
if (m_AnimatorEnabled)
|
||||
m_Animator.SetBool(par.nameHash, newBoolValue);
|
||||
m_Animator.SetBool(par.nameHash, newBoolValue.Value);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -150,6 +150,11 @@ namespace SRMultiplayer.Networking
|
|||
}
|
||||
}
|
||||
|
||||
public float GetWeaponLocation()
|
||||
{
|
||||
return m_ActualWeaponY;
|
||||
}
|
||||
|
||||
private void OnAnimatorIK()
|
||||
{
|
||||
if(m_Animator != null && m_LeftHandTarget != null)
|
||||
|
|
133
SRMP/Packets/Actors/PacketPlayerAnimation.cs
Normal file
133
SRMP/Packets/Actors/PacketPlayerAnimation.cs
Normal file
|
@ -0,0 +1,133 @@
|
|||
using Lidgren.Network;
|
||||
using MonomiPark.SlimeRancher.DataModel;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Data;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using static SRMultiplayer.Packets.PacketAccessDoors;
|
||||
|
||||
namespace SRMultiplayer.Packets
|
||||
{
|
||||
[Packet(PacketType.PlayerAnimation)]
|
||||
public class PacketPlayerAnimation : Packet
|
||||
{
|
||||
public enum AnimationType : int
|
||||
{
|
||||
Layer,
|
||||
Speed,
|
||||
Parameters
|
||||
|
||||
}
|
||||
|
||||
public byte ID;
|
||||
public byte Type;
|
||||
public struct animateData
|
||||
{
|
||||
public byte type
|
||||
{
|
||||
get
|
||||
{
|
||||
if (fData.HasValue) return 0;
|
||||
if (bData.HasValue) return 1;
|
||||
if (iData.HasValue) return 2;
|
||||
if (uData.HasValue) return 3;
|
||||
return 8;
|
||||
}
|
||||
}
|
||||
public float? fData;
|
||||
public bool? bData;
|
||||
public int? iData;
|
||||
public ulong? uData;
|
||||
}
|
||||
|
||||
public Queue<animateData> internalData { get; set; } = new Queue<animateData>();
|
||||
|
||||
public void Add<T>(T obj)
|
||||
{
|
||||
switch (obj)
|
||||
{
|
||||
case float itm:
|
||||
internalData.Enqueue(new animateData() { fData = itm });
|
||||
break;
|
||||
case bool itm:
|
||||
internalData.Enqueue(new animateData() { bData = itm });
|
||||
break;
|
||||
case int itm:
|
||||
internalData.Enqueue(new animateData() { iData = itm });
|
||||
break;
|
||||
case ulong itm:
|
||||
internalData.Enqueue(new animateData() { uData = itm });
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// mark construction inheritance incase we need it
|
||||
/// </summary>
|
||||
public PacketPlayerAnimation() : base() { }
|
||||
/// <summary>
|
||||
/// mark construction inheritance so the deserialization automatically happens for is
|
||||
/// since the base decalres this
|
||||
/// </summary>
|
||||
public PacketPlayerAnimation(NetIncomingMessage im) : base(im) { }
|
||||
|
||||
public override void Serialize(NetOutgoingMessage om)
|
||||
{
|
||||
base.Serialize(om);
|
||||
|
||||
om.Write(internalData.Count);
|
||||
foreach (var data in internalData)
|
||||
{
|
||||
om.Write(data.type);
|
||||
switch (data.type)
|
||||
{
|
||||
case 0:
|
||||
om.Write(data.fData.Value);
|
||||
break;
|
||||
case 1:
|
||||
om.Write(data.bData.Value);
|
||||
break;
|
||||
case 2:
|
||||
om.Write(data.iData.Value);
|
||||
break;
|
||||
case 3:
|
||||
om.Write(data.uData.Value);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public override void Deserialize(NetIncomingMessage im)
|
||||
{
|
||||
base.Deserialize(im);
|
||||
|
||||
internalData = new Queue<animateData>();
|
||||
int Count = im.ReadInt32();
|
||||
for (int i = 0; i < Count; i++)
|
||||
{
|
||||
var data = new animateData();
|
||||
byte type = im.ReadByte();
|
||||
switch (type)
|
||||
{
|
||||
case 0:
|
||||
data.fData = im.ReadFloat();
|
||||
break;
|
||||
case 1:
|
||||
data.bData = im.ReadBoolean();
|
||||
break;
|
||||
case 2:
|
||||
data.iData = im.ReadInt32();
|
||||
break;
|
||||
case 3:
|
||||
data.uData = im.ReadUInt64();
|
||||
break;
|
||||
}
|
||||
|
||||
internalData.Enqueue(data);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -15,6 +15,7 @@ namespace SRMultiplayer.Packets
|
|||
public float Rotation;
|
||||
public float WeaponY;
|
||||
public byte RegionSet;
|
||||
public bool OnLoad = true;
|
||||
|
||||
public PacketPlayerPosition() { }
|
||||
public PacketPlayerPosition(NetIncomingMessage im) { Deserialize(im); }
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
using Lidgren.Network;
|
||||
using SRMultiplayer.Networking;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
@ -9,7 +10,15 @@ namespace SRMultiplayer.Packets
|
|||
public interface IPacket
|
||||
{
|
||||
PacketType GetPacketType();
|
||||
/// <summary>
|
||||
/// Searilizes the given packet item
|
||||
/// </summary>
|
||||
/// <param name="om">Outgoing Message that the packet should be added to</param>
|
||||
void Serialize(NetOutgoingMessage om);
|
||||
/// <summary>
|
||||
/// Deserializes the given packet item
|
||||
/// </summary>
|
||||
/// <param name="om">Incoming Message that the packet should be deserialized from</param>
|
||||
void Deserialize(NetIncomingMessage im);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -9,16 +9,42 @@ namespace SRMultiplayer.Packets
|
|||
{
|
||||
public abstract class Packet : IPacket
|
||||
{
|
||||
/// <summary>
|
||||
/// Parameterless constructor to be used through inheritance
|
||||
/// </summary>
|
||||
public Packet() { }
|
||||
|
||||
/// <summary>
|
||||
/// Incoming message based construtor that deserielizes the item for the message
|
||||
/// </summary>
|
||||
public Packet(NetIncomingMessage im) { Deserialize(im); }
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Searilizes the given packet item
|
||||
/// </summary>
|
||||
/// <param name="om">Outgoing Message that the packet should be added to</param>
|
||||
public virtual void Serialize(NetOutgoingMessage om)
|
||||
{
|
||||
om.Write((ushort)GetPacketType());
|
||||
//writes the object type using the gettype name
|
||||
//this allows the object to assume its object packet type
|
||||
//om.Write(this.GetType().AssemblyQualifiedName);
|
||||
//
|
||||
|
||||
om.WriteAllFields(this, System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Instance);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Deserializes the given packet item
|
||||
/// </summary>
|
||||
/// <param name="om">Incoming Message that the packet should be deserialized from</param>
|
||||
public virtual void Deserialize(NetIncomingMessage im)
|
||||
{
|
||||
im.ReadAllFields(this, System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Instance);
|
||||
}
|
||||
|
||||
public virtual void Serialize(NetOutgoingMessage om)
|
||||
{
|
||||
om.Write((ushort)GetPacketType());
|
||||
om.WriteAllFields(this, System.Reflection.BindingFlags.Public | System.Reflection.BindingFlags.Instance);
|
||||
}
|
||||
|
||||
|
||||
public PacketType GetPacketType()
|
||||
{
|
||||
|
|
|
@ -7,9 +7,7 @@
|
|||
PlayerLeft,
|
||||
PlayerPosition,
|
||||
PlayerLoaded,
|
||||
PlayerAnimationLayer,
|
||||
PlayerAnimationSpeed,
|
||||
PlayerAnimationParameters,
|
||||
PlayerAnimation,
|
||||
PlayerFX,
|
||||
PlayAudio,
|
||||
ActorSpawn,
|
||||
|
@ -97,6 +95,7 @@
|
|||
ExchangeTryAccept,
|
||||
ExchangeClear,
|
||||
ExchangeOffers,
|
||||
ExchangeBreak,
|
||||
GordoEat,
|
||||
Oasis,
|
||||
OasisLive,
|
||||
|
|
|
@ -1,9 +1,13 @@
|
|||
using HarmonyLib;
|
||||
using MonomiPark.SlimeRancher.Regions;
|
||||
using SRMultiplayer.Networking;
|
||||
using SRMultiplayer.Packets;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Runtime.InteropServices.WindowsRuntime;
|
||||
using System.Text;
|
||||
using UnityEngine;
|
||||
|
||||
namespace SRMultiplayer.Patches
|
||||
{
|
||||
|
@ -11,11 +15,19 @@ namespace SRMultiplayer.Patches
|
|||
[HarmonyPatch("BreakOpen")]
|
||||
class ExchangeBreakOnImpact_BreakOpen
|
||||
{
|
||||
|
||||
static bool Prefix(ExchangeBreakOnImpact __instance)
|
||||
{
|
||||
//if multiplayer, only the server can trigger the impact
|
||||
if (!Globals.IsMultiplayer) return true;
|
||||
|
||||
|
||||
|
||||
var entity = __instance.GetComponent<NetworkActor>();
|
||||
|
||||
SRMP.Log("Exchange Box Break: " + (entity != null && entity.IsLocal), "EXCHANGE");
|
||||
|
||||
//if the box exists and the entity is local process the break
|
||||
return (entity != null && entity.IsLocal);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -22,7 +22,18 @@ namespace SRMultiplayer.Patches
|
|||
{
|
||||
if (Globals.Actors.ContainsKey(netActor.ID))
|
||||
{
|
||||
//check if this is an exchange box
|
||||
var exchangeBreakOnImpact = netActor.GetComponentInChildren<ExchangeBreakOnImpact>();
|
||||
if (exchangeBreakOnImpact != null)
|
||||
{
|
||||
//exchange box was found processing it with the ondestroy command instead of just removing it!
|
||||
//netActor.OnDestroyEffect();
|
||||
//Destroyer.DestroyActor(netActor.gameObject, "NetworkHandlerServer.OnActorDestroy");
|
||||
}
|
||||
|
||||
//make sure the actor still gets cleaned up
|
||||
Globals.Actors.Remove(netActor.ID);
|
||||
|
||||
new PacketActorDestroy()
|
||||
{
|
||||
ID = netActor.ID
|
||||
|
|
|
@ -17,7 +17,9 @@ namespace SRMultiplayer.Patches
|
|||
{
|
||||
if (!Globals.IsMultiplayer) return;
|
||||
|
||||
//only handle the client if the client is the one disconnecting
|
||||
NetworkClient.Instance.Disconnect();
|
||||
//if server ahndle the shutdown
|
||||
NetworkServer.Instance.Disconnect();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,14 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
public enum PauseState
|
||||
{
|
||||
Pause,
|
||||
Playing
|
||||
}
|
||||
}
|
|
@ -15,7 +15,7 @@ using System.Resources;
|
|||
[assembly: AssemblyCulture("")]
|
||||
|
||||
// Version informationr(
|
||||
[assembly: AssemblyVersion("0.0.0.1488")]
|
||||
[assembly: AssemblyFileVersion("0.0.0.1488")]
|
||||
[assembly: AssemblyVersion("0.0.0.1510")]
|
||||
[assembly: AssemblyFileVersion("0.0.0.1510")]
|
||||
[assembly: NeutralResourcesLanguageAttribute( "en-US" )]
|
||||
|
||||
|
|
58
SRMP/SRMP.cs
58
SRMP/SRMP.cs
|
@ -20,26 +20,41 @@ namespace SRMultiplayer
|
|||
|
||||
private float m_LastTimeSync;
|
||||
|
||||
/// <summary>
|
||||
/// Acts as the initializer for the Mod
|
||||
/// </summary>
|
||||
public override void Awake()
|
||||
{
|
||||
base.Awake();
|
||||
|
||||
//attach scene manager to trigger event when in a menu or loading up a game
|
||||
SceneManager.activeSceneChanged += SceneManager_activeSceneChanged;
|
||||
//attach log messager to log all game errors and exceptions into the SRMP Logs
|
||||
Application.logMessageReceived += Application_logMessageReceived;
|
||||
|
||||
//load up mod specific resources
|
||||
var myLoadedAssetBundle = AssetBundle.LoadFromMemory(Utils.ExtractResource("SRMultiplayer.srmultiplayer.dat"));
|
||||
if (myLoadedAssetBundle == null)
|
||||
{
|
||||
SRMP.Log("Failed to load AssetBundle!");
|
||||
return;
|
||||
}
|
||||
//load up the Player moment animator for the Beatrix model
|
||||
Globals.BeatrixController = myLoadedAssetBundle.LoadAsset<RuntimeAnimatorController>("Controller");
|
||||
Globals.IngameMultiplayerMenuPrefab = myLoadedAssetBundle.LoadAsset<GameObject>("IngameMultiplayerMenu");
|
||||
Globals.MainMultiplayerMenuPrefab = myLoadedAssetBundle.LoadAsset<GameObject>("MainMultiplayerMenu");
|
||||
}
|
||||
|
||||
//unused prefab menus, these menus functions are handled in the floating gui
|
||||
//Globals.IngameMultiplayerMenuPrefab = myLoadedAssetBundle.LoadAsset<GameObject>("IngameMultiplayerMenu");
|
||||
//Globals.MainMultiplayerMenuPrefab = myLoadedAssetBundle.LoadAsset<GameObject>("MainMultiplayerMenu");
|
||||
}
|
||||
/// <summary>
|
||||
/// Subscriber to the Applicaiton log and process it on to the Mods console
|
||||
/// </summary>
|
||||
/// <param name="condition">Log condition</param>
|
||||
/// <param name="stackTrace">Stack trace of log strigger (if applicable)</param>
|
||||
/// <param name="type">Log Type</param>
|
||||
private void Application_logMessageReceived(string condition, string stackTrace, LogType type)
|
||||
{
|
||||
//if Error or Exception hand the error of to the Mods log/console to display
|
||||
if(type == LogType.Error || type == LogType.Exception)
|
||||
{
|
||||
SRMP.Log(condition);
|
||||
|
@ -54,6 +69,10 @@ namespace SRMultiplayer
|
|||
//menuObj.AddComponent<NetworkClientUI>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// After triggering base destroy
|
||||
/// trigger disconnect and shut down the server
|
||||
/// </summary>
|
||||
public override void OnDestroy()
|
||||
{
|
||||
base.OnDestroy();
|
||||
|
@ -62,18 +81,25 @@ namespace SRMultiplayer
|
|||
NetworkServer.Instance.Disconnect();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// On Game quit trigger disconnect and shut down the server
|
||||
/// </summary>
|
||||
private void OnApplicationQuit()
|
||||
{
|
||||
NetworkClient.Instance.Disconnect();
|
||||
NetworkServer.Instance.Disconnect();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// On Update triggered sync up game time
|
||||
/// </summary>
|
||||
private void Update()
|
||||
{
|
||||
if(Globals.GameLoaded)
|
||||
{
|
||||
if(Globals.IsServer)
|
||||
{
|
||||
//every 30 seconds send a time updater out to all clients
|
||||
if(Time.time - m_LastTimeSync > 30)
|
||||
{
|
||||
m_LastTimeSync = Time.time;
|
||||
|
@ -98,17 +124,28 @@ namespace SRMultiplayer
|
|||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handle scene changed events triggered by the game
|
||||
/// </summary>
|
||||
/// <param name="from">Scene previously</param>
|
||||
/// <param name="to">New Scene</param>
|
||||
private void SceneManager_activeSceneChanged(Scene from, Scene to)
|
||||
{
|
||||
//trigger handlers for returning or going to the main menu
|
||||
if (to.buildIndex == 2) OnMainMenuLoaded();
|
||||
if (to.buildIndex == 3) OnGameLoaded();
|
||||
//trigger handlers for loading the game
|
||||
else if (to.buildIndex == 3) OnGameLoaded();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handle user changing to the main menu, whether it is start up or from saving/ being kicked out of the game
|
||||
/// </summary>
|
||||
private void OnMainMenuLoaded()
|
||||
{
|
||||
//var menuObj = Instantiate(Globals.MainMultiplayerMenuPrefab, null, false);
|
||||
//menuObj.AddComponent<NetworkClientUI>();
|
||||
|
||||
//innitialize all necessary global variables
|
||||
Globals.LocalID = 0;
|
||||
Globals.DisableAchievements = false;
|
||||
Globals.GameLoaded = false;
|
||||
|
@ -135,6 +172,8 @@ namespace SRMultiplayer
|
|||
Globals.Nutcrackers.Clear();
|
||||
Globals.RaceTriggers.Clear();
|
||||
NetworkAmmo.All.Clear();
|
||||
|
||||
//clean up any lingering players in the global list
|
||||
foreach (var player in Globals.Players.Values.ToList())
|
||||
{
|
||||
if(player != null && player.gameObject != null)
|
||||
|
@ -144,9 +183,14 @@ namespace SRMultiplayer
|
|||
}
|
||||
Globals.Players.Clear();
|
||||
|
||||
|
||||
//reset the chat
|
||||
ChatUI.Instance.Clear();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handle the user loading into the multiplayer game
|
||||
/// </summary>
|
||||
private void OnGameLoaded()
|
||||
{
|
||||
System.Diagnostics.Stopwatch stopwatch = new System.Diagnostics.Stopwatch();
|
||||
|
@ -365,6 +409,12 @@ namespace SRMultiplayer
|
|||
|
||||
private static FileStream m_LogFileStream;
|
||||
private static StreamWriter m_LogWriter;
|
||||
/// <summary>
|
||||
/// Custom message logging of a given message
|
||||
/// </summary>
|
||||
/// <param name="msg">Main message to be logged</param>
|
||||
/// <param name="prefix">Prefix to be displayed before the message. Prefix will be after time marker and before the message
|
||||
/// It will also be incapsulated in []</param>
|
||||
public static void Log(string msg, string prefix = null)
|
||||
{
|
||||
if(m_LogFileStream == null)
|
||||
|
|
100
SRMP/SRMP.csproj
100
SRMP/SRMP.csproj
|
@ -9,14 +9,14 @@
|
|||
<AppDesignerFolder>Properties</AppDesignerFolder>
|
||||
<RootNamespace>SRMultiplayer</RootNamespace>
|
||||
<AssemblyName>SRMP</AssemblyName>
|
||||
<TargetFrameworkVersion>v4.5</TargetFrameworkVersion>
|
||||
<TargetFrameworkVersion>v4.7.2</TargetFrameworkVersion>
|
||||
<FileAlignment>512</FileAlignment>
|
||||
<Deterministic>true</Deterministic>
|
||||
<TargetFrameworkProfile />
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|AnyCPU' ">
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
<DebugType>full</DebugType>
|
||||
<DebugType>portable</DebugType>
|
||||
<Optimize>false</Optimize>
|
||||
<OutputPath>..\Builds\SRMP\</OutputPath>
|
||||
<DefineConstants>__CONSTRAINED__;Standalone</DefineConstants>
|
||||
|
@ -74,24 +74,25 @@
|
|||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Reference Include="0Harmony">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\0Harmony.dll</HintPath>
|
||||
<HintPath>.\0Harmony.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="Assembly-CSharp_publicized">
|
||||
<HintPath>..\Libs\Assembly-CSharp_publicized.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="DOTween">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\DOTween.dll</HintPath>
|
||||
<HintPath>.\DOTween.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="InControl">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\InControl.dll</HintPath>
|
||||
<HintPath>.\InControl.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="Newtonsoft.Json, Version=6.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
|
||||
<Reference Include="Newtonsoft.Json, Version=9.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
|
||||
<SpecificVersion>False</SpecificVersion>
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\Newtonsoft.Json.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="SRML, Version=0.1.8.2, Culture=neutral, processorArchitecture=MSIL">
|
||||
<SpecificVersion>False</SpecificVersion>
|
||||
<HintPath>..\Libs\SRML.dll</HintPath>
|
||||
<HintPath>..\Libs\Newtonsoft.Json.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="System" />
|
||||
<Reference Include="System.Core" />
|
||||
|
@ -103,54 +104,73 @@
|
|||
<Reference Include="System.Net.Http" />
|
||||
<Reference Include="System.Xml" />
|
||||
<Reference Include="Unity.TextMeshPro">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\Unity.TextMeshPro.dll</HintPath>
|
||||
<HintPath>.\Unity.TextMeshPro.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityCoreMod, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||
<SpecificVersion>False</SpecificVersion>
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityCoreMod.dll</HintPath>
|
||||
<HintPath>.\UnityCoreMod.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.AnimationModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.AnimationModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.AnimationModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.AssetBundleModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.AssetBundleModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.AssetBundleModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.CoreModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.CoreModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.CoreModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.IMGUIModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.IMGUIModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.IMGUIModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.InputLegacyModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.InputLegacyModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.InputLegacyModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.InputModule">
|
||||
<HintPath>.\UnityEngine.InputModule.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.JSONSerializeModule">
|
||||
<HintPath>.\UnityEngine.JSONSerializeModule.dll</HintPath>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.PhysicsModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.PhysicsModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.PhysicsModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.TextCoreModule, Version=0.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||
<SpecificVersion>False</SpecificVersion>
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.TextCoreModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.TextCoreModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.TextRenderingModule, Version=0.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||
<SpecificVersion>False</SpecificVersion>
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.TextRenderingModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.TextRenderingModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.UI">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.UI.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.UI.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
<Reference Include="UnityEngine.UIModule">
|
||||
<HintPath>C:\Program Files (x86)\Steam\steamapps\common\Slime Rancher\SlimeRancher_Data\Managed\UnityEngine.UIModule.dll</HintPath>
|
||||
<HintPath>.\UnityEngine.UIModule.dll</HintPath>
|
||||
<Private>False</Private>
|
||||
</Reference>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="ChatUI.cs" />
|
||||
<Compile Include="Compression.cs" />
|
||||
<Compile Include="Console\ConsoleInput.cs" />
|
||||
<Compile Include="Console\ConsoleWindow.cs" />
|
||||
<Compile Include="Extensions.cs" />
|
||||
<Compile Include="Custom UI\ChatUI.cs" />
|
||||
<Compile Include="Custom UI\MultiplayerUI.cs" />
|
||||
<Compile Include="Custom UI\TestUI.cs" />
|
||||
<Compile Include="Globals.cs" />
|
||||
<Compile Include="Lidgren.Network\Encryption\NetAESEncryption.cs" />
|
||||
<Compile Include="Lidgren.Network\Encryption\NetBlockEncryptionBase.cs" />
|
||||
|
@ -228,11 +248,13 @@
|
|||
<Compile Include="Lidgren.Network\Platform\PlatformWinRT.cs" />
|
||||
<Compile Include="MainSRML.cs" />
|
||||
<Compile Include="MainStandalone.cs" />
|
||||
<Compile Include="MultiplayerUI.cs" />
|
||||
<Compile Include="Networking\Communication\NetworkClient.cs" />
|
||||
<Compile Include="Networking\Communication\NetworkHandlerClient.cs" />
|
||||
<Compile Include="Networking\Communication\NetworkHandlerServer.cs" />
|
||||
<Compile Include="Networking\Communication\NetworkServer.cs" />
|
||||
<Compile Include="Networking\NetworkAccessDoor.cs" />
|
||||
<Compile Include="Networking\NetworkActor.cs" />
|
||||
<Compile Include="Networking\NetworkAmmo.cs" />
|
||||
<Compile Include="Networking\NetworkClient.cs" />
|
||||
<Compile Include="Networking\NetworkClientUI.cs" />
|
||||
<Compile Include="Networking\NetworkDirectedActorSpawner.cs" />
|
||||
<Compile Include="Networking\NetworkDrone.cs" />
|
||||
|
@ -241,8 +263,6 @@
|
|||
<Compile Include="Networking\NetworkFireColumn.cs" />
|
||||
<Compile Include="Networking\NetworkGadgetSite.cs" />
|
||||
<Compile Include="Networking\NetworkGordo.cs" />
|
||||
<Compile Include="Networking\NetworkHandlerClient.cs" />
|
||||
<Compile Include="Networking\NetworkHandlerServer.cs" />
|
||||
<Compile Include="Networking\NetworkHostUI.cs" />
|
||||
<Compile Include="Networking\NetworkKookadobaPatchNode.cs" />
|
||||
<Compile Include="Networking\NetworkLandplot.cs" />
|
||||
|
@ -256,11 +276,11 @@
|
|||
<Compile Include="Networking\NetworkPuzzleSlot.cs" />
|
||||
<Compile Include="Networking\NetworkRaceTrigger.cs" />
|
||||
<Compile Include="Networking\NetworkRegion.cs" />
|
||||
<Compile Include="Networking\NetworkServer.cs" />
|
||||
<Compile Include="Networking\NetworkSpawnResource.cs" />
|
||||
<Compile Include="Networking\NetworkTreasurePod.cs" />
|
||||
<Compile Include="Networking\NetworkWorldStateMasterSwitch.cs" />
|
||||
<Compile Include="Packets\AccessDoors\PacketAccessDoors.cs" />
|
||||
<Compile Include="Packets\Actors\PacketPlayerAnimation.cs" />
|
||||
<Compile Include="Packets\Exchanges\PacketExchangeClear.cs" />
|
||||
<Compile Include="Packets\Exchanges\PacketExchangeOffer.cs" />
|
||||
<Compile Include="Packets\Exchanges\PacketExchangeOffers.cs" />
|
||||
|
@ -462,19 +482,20 @@
|
|||
<Compile Include="Patches\Patch_WeaponVacuum.cs" />
|
||||
<Compile Include="Patches\Patch_WorldStateMasterSwitch.cs" />
|
||||
<Compile Include="Patches\Patch_ZoneDirector.cs" />
|
||||
<Compile Include="PauseState.cs" />
|
||||
<Compile Include="Properties\AssemblyInfo.cs">
|
||||
<AutoGen>True</AutoGen>
|
||||
<DesignTime>True</DesignTime>
|
||||
<DependentUpon>AssemblyInfo.tt</DependentUpon>
|
||||
</Compile>
|
||||
<Compile Include="SRMP.cs" />
|
||||
<Compile Include="SRMPConsole.cs" />
|
||||
<Compile Include="TestUI.cs" />
|
||||
<Compile Include="UserData.cs" />
|
||||
<Compile Include="Utils.cs" />
|
||||
<Compile Include="Console\SRMPConsole.cs" />
|
||||
<Compile Include="Utils\Compression.cs" />
|
||||
<Compile Include="Utils\Extensions.cs" />
|
||||
<Compile Include="Utils\Objects.cs" />
|
||||
<Compile Include="Utils\Utils.cs" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<Content Include="Console\Console Commands.txt" />
|
||||
<Content Include="modinfo.tt">
|
||||
<Generator>TextTemplatingFileGenerator</Generator>
|
||||
<LastGenOutput>modinfo.json</LastGenOutput>
|
||||
|
@ -487,6 +508,11 @@
|
|||
<ItemGroup>
|
||||
<Service Include="{508349B6-6B84-4DF5-91F0-309BEEBAD82D}" />
|
||||
</ItemGroup>
|
||||
<ItemGroup Condition=" '$(Configuration)|$(Platform)' == 'SRML|AnyCPU' ">
|
||||
<Reference Include="SRML">
|
||||
<HintPath>..\Libs\SRML.dll</HintPath>
|
||||
</Reference>
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<EmbeddedResource Include="modinfo.json">
|
||||
<AutoGen>True</AutoGen>
|
||||
|
|
|
@ -1,238 +0,0 @@
|
|||
using MonomiPark.SlimeRancher.Regions;
|
||||
using SRMultiplayer.Networking;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using UnityEngine;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
public class SRMPConsole : MonoBehaviour
|
||||
{
|
||||
ConsoleWindow console = new ConsoleWindow();
|
||||
ConsoleInput input = new ConsoleInput();
|
||||
|
||||
//
|
||||
// Create console window, register callbacks
|
||||
//
|
||||
void Awake()
|
||||
{
|
||||
DontDestroyOnLoad(gameObject);
|
||||
|
||||
console.Initialize();
|
||||
console.SetTitle("Slime Rancher");
|
||||
|
||||
input.OnInputText += OnInputText;
|
||||
|
||||
Application.logMessageReceived += Application_logMessageReceived;
|
||||
|
||||
SRMP.Log("Console Started");
|
||||
}
|
||||
|
||||
private void Application_logMessageReceived(string condition, string stackTrace, LogType type)
|
||||
{
|
||||
if (type == LogType.Warning)
|
||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
||||
else if (type == LogType.Error)
|
||||
Console.ForegroundColor = ConsoleColor.Red;
|
||||
else
|
||||
Console.ForegroundColor = ConsoleColor.White;
|
||||
|
||||
// We're half way through typing something, so clear this line ..
|
||||
if (Console.CursorLeft != 0)
|
||||
input.ClearLine();
|
||||
|
||||
Console.WriteLine(condition);
|
||||
if (!string.IsNullOrEmpty(stackTrace))
|
||||
Console.WriteLine(stackTrace);
|
||||
|
||||
// If we were typing something re-add it.
|
||||
input.RedrawInputLine();
|
||||
}
|
||||
|
||||
//
|
||||
// Text has been entered into the console
|
||||
// Run it as a console command
|
||||
//
|
||||
void OnInputText(string obj)
|
||||
{
|
||||
var args = obj.Split(' ');
|
||||
var cmd = args.FirstOrDefault();
|
||||
args = args.Skip(1).ToArray();
|
||||
|
||||
switch (cmd.ToLower())
|
||||
{
|
||||
case "cheat":
|
||||
{
|
||||
if (args.Length > 0)
|
||||
{
|
||||
if (args[0].Equals("money"))
|
||||
{
|
||||
if (args.Length != 2 || !int.TryParse(args[1], out int money))
|
||||
{
|
||||
SRMP.Log("Usage: cheat money <amount>");
|
||||
}
|
||||
else
|
||||
{
|
||||
if(money > 0)
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.AddCurrency(money);
|
||||
else
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.SpendCurrency(money);
|
||||
}
|
||||
}
|
||||
else if(args[0].Equals("enable"))
|
||||
{
|
||||
//TestUI.Instance.cheat = !TestUI.Instance.cheat;
|
||||
}
|
||||
else if (args[0].Equals("keys"))
|
||||
{
|
||||
if (args.Length != 2 || !int.TryParse(args[1], out int value))
|
||||
{
|
||||
SRMP.Log("Usage: cheat keys <amount>");
|
||||
}
|
||||
else
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.PlayerState.model.keys = value;
|
||||
}
|
||||
}
|
||||
else if (args[0].Equals("allgadgets"))
|
||||
{
|
||||
foreach (var data in (Gadget.Id[])Enum.GetValues(typeof(Gadget.Id)))
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.GadgetDirector.model.gadgets[data] = int.MaxValue;
|
||||
}
|
||||
foreach (var data in Identifiable.CRAFT_CLASS.Union(Identifiable.PLORT_CLASS))
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.GadgetDirector.model.craftMatCounts[data] = int.MaxValue;
|
||||
}
|
||||
}
|
||||
else if (args[0].Equals("spawn"))
|
||||
{
|
||||
if (args.Length == 2)
|
||||
{
|
||||
if (Enum.TryParse<Identifiable.Id>(args[1].ToUpper(), out Identifiable.Id id))
|
||||
{
|
||||
GameObject prefab = SRSingleton<GameContext>.Instance.LookupDirector.GetPrefab(id);
|
||||
if (prefab != null)
|
||||
{
|
||||
SRBehaviour.InstantiateActor(prefab, SRSingleton<SceneContext>.Instance.GameModel.player.currRegionSetId, SRSingleton<SceneContext>.Instance.GameModel.player.GetPos() + new Vector3(0, 4, 0), Quaternion.identity, false);
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log(id + " can not be spawned");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var data = Enum.GetNames(typeof(Identifiable.Id)).Where(n => n.ToLower().Contains(args[1].ToLower()));
|
||||
SRMP.Log(args[1] + " not found. " + (data.Count() > 0 ? " Did you mean one of these?" : ""));
|
||||
foreach (var name in data)
|
||||
{
|
||||
SRMP.Log(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
else if (args.Length == 3 && int.TryParse(args[2], out int amount))
|
||||
{
|
||||
if (Enum.TryParse<Identifiable.Id>(args[1].ToUpper(), out Identifiable.Id id))
|
||||
{
|
||||
GameObject prefab = SRSingleton<GameContext>.Instance.LookupDirector.GetPrefab(id);
|
||||
if (prefab != null)
|
||||
{
|
||||
for (int i = 0; i < amount; i++)
|
||||
{
|
||||
SRBehaviour.InstantiateActor(prefab, SRSingleton<SceneContext>.Instance.GameModel.player.currRegionSetId, SRSingleton<SceneContext>.Instance.GameModel.player.GetPos() + new Vector3(0, 4, 0), Quaternion.identity, false);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log(id + " can not be spawned");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
var data = Enum.GetNames(typeof(Identifiable.Id)).Where(n => n.ToLower().Contains(args[1].ToLower()));
|
||||
SRMP.Log(args[1] + " not found. " + (data.Count() > 0 ? " Did you mean one of these?" : ""));
|
||||
foreach (var name in data)
|
||||
{
|
||||
SRMP.Log(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log("Usage: cheat spawn <id> (<amount>)");
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log("Available sub commands:");
|
||||
SRMP.Log("cheat money <amount>");
|
||||
SRMP.Log("cheat keys <amount>");
|
||||
SRMP.Log("cheat spawn <id> (<amount>)");
|
||||
SRMP.Log("cheat allgadgets");
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "tp":
|
||||
{
|
||||
if (args.Length == 1)
|
||||
{
|
||||
var player = Globals.Players.Values.FirstOrDefault(p => p.Username.Equals(args[0], StringComparison.CurrentCultureIgnoreCase));
|
||||
if (player != null)
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.player.transform.position = player.transform.position;
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log("Player not found");
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
SRMP.Log("Usage: tp <username>");
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "listplayers":
|
||||
{
|
||||
SRMP.Log("Players:");
|
||||
foreach (var player in Globals.Players.Values)
|
||||
{
|
||||
SRMP.Log(player.Username);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case "sleep":
|
||||
{
|
||||
if (args.Length == 1)
|
||||
{
|
||||
SRSingleton<SceneContext>.Instance.TimeDirector.FastForwardTo(SRSingleton<SceneContext>.Instance.TimeDirector.HoursFromNow(float.Parse(args[0])));
|
||||
}
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
//
|
||||
// Update the input every frame
|
||||
// This gets new key input and calls the OnInputText callback
|
||||
//
|
||||
void Update()
|
||||
{
|
||||
input.Update();
|
||||
}
|
||||
|
||||
//
|
||||
// It's important to call console.ShutDown in OnDestroy
|
||||
// because compiling will error out in the editor if you don't
|
||||
// because we redirected output. This sets it back to normal.
|
||||
//
|
||||
void OnDestroy()
|
||||
{
|
||||
console.Shutdown();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,16 +0,0 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
[Serializable]
|
||||
public struct UserData
|
||||
{
|
||||
public Guid UUID;
|
||||
public bool CheckDLC;
|
||||
public List<string> IgnoredMods;
|
||||
}
|
||||
}
|
|
@ -10,6 +10,10 @@ using UnityEngine;
|
|||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
/// <summary>
|
||||
/// Extends multiple objects to add extras functionality
|
||||
///
|
||||
/// </summary>
|
||||
public static class Extensions
|
||||
{
|
||||
public static void Rebuild(this RefineryUI ui)
|
||||
|
@ -40,6 +44,7 @@ namespace SRMultiplayer
|
|||
}
|
||||
}
|
||||
|
||||
#region Ammo Slot Extensions
|
||||
public static void WriteAmmoSlot(this NetOutgoingMessage om, Ammo.Slot slot)
|
||||
{
|
||||
om.Write(slot != null);
|
||||
|
@ -78,7 +83,9 @@ namespace SRMultiplayer
|
|||
}
|
||||
return null;
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Packet Handling Extensions
|
||||
public static void Send(this Packet packet, NetDeliveryMethod method = NetDeliveryMethod.ReliableOrdered, int sequence = 0)
|
||||
{
|
||||
if(!Globals.IsClient)
|
||||
|
@ -160,6 +167,10 @@ namespace SRMultiplayer
|
|||
NetworkServer.Instance.SendTo(packet, cons, method, sequence);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Component Handling Extensions
|
||||
|
||||
public static T CopyComponent<T>(this T original, GameObject destination) where T : Component
|
||||
{
|
||||
System.Type type = original.GetType();
|
||||
|
@ -245,5 +256,6 @@ namespace SRMultiplayer
|
|||
}
|
||||
return null;
|
||||
}
|
||||
#endregion
|
||||
}
|
||||
}
|
29
SRMP/Utils/Objects.cs
Normal file
29
SRMP/Utils/Objects.cs
Normal file
|
@ -0,0 +1,29 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace SRMultiplayer
|
||||
{
|
||||
/// <summary>
|
||||
/// Used marking the game time movement status
|
||||
/// </summary>
|
||||
public enum PauseState
|
||||
{
|
||||
Pause,
|
||||
Playing
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Handles basic user data to be used for connection communication
|
||||
/// Uesr ID, mod count and dlc checkers
|
||||
/// </summary>
|
||||
[Serializable]
|
||||
public struct UserData
|
||||
{
|
||||
public Guid UUID;
|
||||
public bool CheckDLC;
|
||||
public List<string> IgnoredMods;
|
||||
}
|
||||
}
|
|
@ -11,6 +11,11 @@ namespace SRMultiplayer
|
|||
{
|
||||
public static class Utils
|
||||
{
|
||||
/// <summary>
|
||||
/// Loads up any resources embedded in the mod
|
||||
/// </summary>
|
||||
/// <param name="filename">Name of resource</param>
|
||||
/// <returns>Contents of resource</returns>
|
||||
public static byte[] ExtractResource(String filename)
|
||||
{
|
||||
System.Reflection.Assembly a = System.Reflection.Assembly.GetExecutingAssembly();
|
||||
|
@ -22,7 +27,9 @@ namespace SRMultiplayer
|
|||
return ba;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sets what layer the given item is at
|
||||
/// </summary>
|
||||
public static void SetLayer(GameObject obj, int layer)
|
||||
{
|
||||
obj.layer = layer;
|
||||
|
@ -31,18 +38,27 @@ namespace SRMultiplayer
|
|||
SetLayer(child.gameObject, layer);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Check to see if the provided values are close enough to be the same
|
||||
/// </summary>
|
||||
public static bool CloseEnoughForMe(double value1, double value2, double acceptableDifference)
|
||||
{
|
||||
return Math.Abs(value1 - value2) <= acceptableDifference;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Check to see if the provided values are close enough to be the same
|
||||
/// </summary>
|
||||
public static bool CloseEnoughForMe(float value1, float value2, float acceptableDifference)
|
||||
{
|
||||
return Mathf.Abs(value1 - value2) <= acceptableDifference;
|
||||
}
|
||||
|
||||
private static System.Random m_Random = new System.Random();
|
||||
|
||||
/// <summary>
|
||||
/// Gets a new random actor id for netowork actors
|
||||
/// </summary>
|
||||
/// <returns>Random integer between in Min and int max that is not in the current sessions of NetworkActors</returns>
|
||||
public static int GetRandomActorID()
|
||||
{
|
||||
int id = m_Random.Next(int.MinValue, int.MaxValue);
|
||||
|
@ -55,6 +71,11 @@ namespace SRMultiplayer
|
|||
|
||||
static readonly string[] SizeSuffixes = { "bytes", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB" };
|
||||
const long byteConversion = 1000;
|
||||
/// <summary>
|
||||
/// Takes a count of bytes and turns it into a human readable version
|
||||
/// </summary>
|
||||
/// <param name="value">Count of Bytes as a long</param>
|
||||
/// <returns>String statement of the bytes </returns>
|
||||
public static string GetHumanReadableFileSize(long value)
|
||||
{
|
||||
|
|
@ -1,11 +1,9 @@
|
|||
|
||||
{
|
||||
"id": "srmp",
|
||||
"name": "Slime Rancher Multiplayer",
|
||||
"version": "0.0.1488",
|
||||
"version": "0.0.1510",
|
||||
"author": "SatyPardus",
|
||||
"dependencies": [
|
||||
|
||||
]
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue