Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
db06cf19ad | ||
|
|
aaf9bf22dd | ||
|
|
d706e173b8 | ||
|
|
6a8a09e2f6 |
51
MxValheim/EventSystem/Experience.cs
Normal file
51
MxValheim/EventSystem/Experience.cs
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
using MxValheim.Patch.HUD;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using UnityEngine;
|
||||||
|
|
||||||
|
namespace MxValheim.EventSystem
|
||||||
|
{
|
||||||
|
internal class Experience
|
||||||
|
{
|
||||||
|
public static int GetRequiredExpByLevel(int level)
|
||||||
|
{
|
||||||
|
return (level * 100);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void AddExperience(string playerid, int amount)
|
||||||
|
{
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_RequestProfile", playerid);
|
||||||
|
int level = Profiles.playerLevel;
|
||||||
|
int exp = Profiles.playerExperience;
|
||||||
|
exp += amount;
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_ServerSetUserExperience", playerid, exp);
|
||||||
|
exp = Profiles.playerExperience;
|
||||||
|
int maxexp = GetRequiredExpByLevel(level);
|
||||||
|
UpdateProgressBar(playerid);
|
||||||
|
if (exp >= maxexp)
|
||||||
|
{
|
||||||
|
exp -= maxexp;
|
||||||
|
level++;
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_ServerSetUserLevel", playerid, level);
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_ServerSetUserExperience", playerid, exp);
|
||||||
|
UpdateProgressBar(playerid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void UpdateProgressBar(string playerid)
|
||||||
|
{
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_RequestProfile", playerid);
|
||||||
|
float expPercentage = ((float)Profiles.playerExperience / GetRequiredExpByLevel(Profiles.playerLevel));
|
||||||
|
|
||||||
|
if (Clock.m_barFill != null)
|
||||||
|
{
|
||||||
|
Clock.m_barFill.anchorMax = new Vector2(expPercentage, 1);
|
||||||
|
Clock.m_barFill.offsetMin = Vector2.zero;
|
||||||
|
Clock.m_barFill.offsetMax = Vector2.zero;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
161
MxValheim/EventSystem/Patch.cs
Normal file
161
MxValheim/EventSystem/Patch.cs
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
using HarmonyLib;
|
||||||
|
using Splatform;
|
||||||
|
using System;
|
||||||
|
using System.Collections;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using System.Windows.Controls;
|
||||||
|
using UnityEngine;
|
||||||
|
using UnityEngine.UI;
|
||||||
|
using static MxValheimMod;
|
||||||
|
using Canvas = UnityEngine.Canvas;
|
||||||
|
using Image = UnityEngine.UI.Image;
|
||||||
|
|
||||||
|
namespace MxValheim.EventSystem
|
||||||
|
{
|
||||||
|
internal class EventSystem_Patch
|
||||||
|
{
|
||||||
|
// --- 3. HARMONY PATCHES ---
|
||||||
|
|
||||||
|
[HarmonyPatch(typeof(Player), nameof(Player.Awake))]
|
||||||
|
public static class PlayerAwakePatch
|
||||||
|
{
|
||||||
|
static void Postfix(Player __instance)
|
||||||
|
{
|
||||||
|
// Only trigger for the actual human player, not NPCs/Dummies
|
||||||
|
if (__instance == Player.m_localPlayer)
|
||||||
|
{
|
||||||
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
ZLog.Log($"[ProfileSystem] Spawning finished. Requesting profile for {pid}");
|
||||||
|
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_RequestProfile", pid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[HarmonyPatch(typeof(RandEventSystem), nameof(RandEventSystem.FixedUpdate))]
|
||||||
|
public static class DebugRaidFrequency
|
||||||
|
{
|
||||||
|
static void Prefix(RandEventSystem __instance)
|
||||||
|
{
|
||||||
|
// Use a small timer check so we don't spam the logic every single frame
|
||||||
|
// though for debugging, forcing these values is fine.
|
||||||
|
//__instance.m_eventIntervalMin = 10f;
|
||||||
|
//__instance.m_eventChance = 100f;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[HarmonyPatch(typeof(RandEventSystem), nameof(RandEventSystem.FixedUpdate))]
|
||||||
|
public static class TimerFreezePatch
|
||||||
|
{
|
||||||
|
static void Prefix(RandEventSystem __instance)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[HarmonyPatch(typeof(Hud), nameof(Hud.Update))]
|
||||||
|
public static class Hud_MonitorPatch
|
||||||
|
{
|
||||||
|
private static RandomEvent _lastKnownEvent = null;
|
||||||
|
|
||||||
|
static void Postfix()
|
||||||
|
{
|
||||||
|
// 1. Safety check: Ensure the system exists
|
||||||
|
if (RandEventSystem.instance == null) return;
|
||||||
|
|
||||||
|
RandomEvent activeEvent = RandEventSystem.instance.GetActiveEvent();
|
||||||
|
|
||||||
|
// 2. Logic: Detection
|
||||||
|
if (activeEvent != null && _lastKnownEvent == null)
|
||||||
|
{
|
||||||
|
Debug.Log($"MxEvent: HUD Heartbeat detected Raid Start: {activeEvent.m_name}");
|
||||||
|
_lastKnownEvent = activeEvent;
|
||||||
|
|
||||||
|
// Force HUD Creation and Display
|
||||||
|
//UpdateWave(true);
|
||||||
|
}
|
||||||
|
else if (activeEvent == null && _lastKnownEvent != null)
|
||||||
|
{
|
||||||
|
Debug.Log("MxEvent: HUD Heartbeat detected Raid End.");
|
||||||
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
Heightmap.Biome currentBiome = EnvMan.instance.GetCurrentBiome();
|
||||||
|
Reward.GiveItemStatic("Coins",Reward.CalculateRewardCoin(Profiles.playerLevel, currentBiome));
|
||||||
|
System.Random rand = new System.Random();
|
||||||
|
Reward.GiveItemStatic("Feathers", rand.Next(1,10));
|
||||||
|
if (rand.Next(100) < 50)
|
||||||
|
{
|
||||||
|
Reward.GiveItemRandom(Reward.GetRewardPrefab(Profiles.playerLevel, "food"), 1);
|
||||||
|
}
|
||||||
|
if (rand.Next(100) < 25)
|
||||||
|
{
|
||||||
|
Reward.GiveItemRandom(Reward.GetRewardPrefab(Profiles.playerLevel, "heal"), 1);
|
||||||
|
}
|
||||||
|
if (rand.Next(100) < 10)
|
||||||
|
{
|
||||||
|
Reward.GiveItemRandom(Reward.GetRewardPrefab(Profiles.playerLevel, "metal"), 1);
|
||||||
|
}
|
||||||
|
_lastKnownEvent = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[HarmonyPatch(typeof(Character), nameof(Character.ApplyDamage))]
|
||||||
|
public static class DeathNotifier
|
||||||
|
{
|
||||||
|
static void Postfix(Character __instance)
|
||||||
|
{
|
||||||
|
// Check if the character is dead or just died
|
||||||
|
if (__instance.GetHealth() <= 0f)
|
||||||
|
{
|
||||||
|
ZNetView nview = __instance.GetComponent<ZNetView>();
|
||||||
|
|
||||||
|
// This check should now pass because ApplyDamage happens before the object is invalidated
|
||||||
|
if (nview == null || !nview.IsValid() || !nview.IsOwner()) return;
|
||||||
|
|
||||||
|
// 2. Check if a raid is active
|
||||||
|
RandomEvent activeEvent = RandEventSystem.instance.GetActiveEvent();
|
||||||
|
if (activeEvent == null) return;
|
||||||
|
Debug.Log("MxEvent: Confirmed Active Event!");
|
||||||
|
|
||||||
|
Vector3 playerPos = Player.m_localPlayer.transform.position;
|
||||||
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
Debug.Log($"EventSystem:ApplyDamage User {pid}");
|
||||||
|
float distance = Vector3.Distance(playerPos, activeEvent.m_pos);
|
||||||
|
Debug.Log($"MxEvent: Distance: {distance}");
|
||||||
|
|
||||||
|
if (distance <= 96f)
|
||||||
|
{
|
||||||
|
int level = __instance.GetLevel();
|
||||||
|
int exp =
|
||||||
|
(level == 1) ? 2:
|
||||||
|
(level == 2) ? 3:
|
||||||
|
(level == 3) ? 4:
|
||||||
|
5;
|
||||||
|
|
||||||
|
Experience.AddExperience(pid, exp);
|
||||||
|
|
||||||
|
/* Send the update
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(
|
||||||
|
ZRoutedRpc.Everybody,
|
||||||
|
"RPC_UpdateRaidHUD",
|
||||||
|
$"PHASE: {CurrentPhase}",
|
||||||
|
KillsInCurrentPhase,
|
||||||
|
GetTargetForPhase(CurrentPhase)
|
||||||
|
);*/
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hide HUD when raid ends
|
||||||
|
[HarmonyPatch(typeof(RandEventSystem), nameof(RandEventSystem.ResetRandomEvent))]
|
||||||
|
[HarmonyPostfix]
|
||||||
|
static void OnRaidEnd()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
167
MxValheim/EventSystem/Profiles.cs
Normal file
167
MxValheim/EventSystem/Profiles.cs
Normal file
@@ -0,0 +1,167 @@
|
|||||||
|
using Newtonsoft.Json;
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace MxValheim.EventSystem
|
||||||
|
{
|
||||||
|
internal class Profiles
|
||||||
|
{
|
||||||
|
public class ProfileRoot
|
||||||
|
{
|
||||||
|
public string playerid { get; set; }
|
||||||
|
public string player_name { get; set; }
|
||||||
|
public int level { get; set; }
|
||||||
|
public int experience { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public static bool userExists = false;
|
||||||
|
public static int playerLevel = 1;
|
||||||
|
public static int playerExperience = 0;
|
||||||
|
|
||||||
|
|
||||||
|
public static void ServerUserExists(string playerid)
|
||||||
|
{
|
||||||
|
// Check to ensure this is actually the server running this
|
||||||
|
ZNet zn = new ZNet();
|
||||||
|
if (!zn.IsDedicated()) return;
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
|
||||||
|
if (File.Exists(profilePath)) userExists = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void ServerReceiveCreateUser(long sender, string playerid, string playerName)
|
||||||
|
{
|
||||||
|
// Check to ensure this is actually the server running this
|
||||||
|
ZNet zn = new ZNet();
|
||||||
|
if (!zn.IsDedicated()) return;
|
||||||
|
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
|
||||||
|
ProfileRoot pr = new ProfileRoot
|
||||||
|
{
|
||||||
|
playerid = playerid,
|
||||||
|
player_name = playerName,
|
||||||
|
level = 1,
|
||||||
|
experience = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
var raw = JsonConvert.SerializeObject(pr, Formatting.Indented);
|
||||||
|
File.WriteAllText(profilePath, raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void ServerHandleProfileRequest(long sender, string playerid)
|
||||||
|
{
|
||||||
|
// Check to ensure this is actually the server running this
|
||||||
|
ZNet zn = new ZNet();
|
||||||
|
if (!zn.IsDedicated()) return;
|
||||||
|
|
||||||
|
ZNetPeer peer = ZNet.instance.GetPeer(sender);
|
||||||
|
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
|
||||||
|
if (File.Exists(profilePath))
|
||||||
|
{
|
||||||
|
string rawJson = File.ReadAllText(profilePath);
|
||||||
|
// Send ONLY to the 'sender' who requested it
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(sender, "RPC_ClientReceiveProfile", rawJson);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ProfileRoot pr = new ProfileRoot
|
||||||
|
{
|
||||||
|
playerid = playerid,
|
||||||
|
player_name = peer.m_playerName,
|
||||||
|
level = 1,
|
||||||
|
experience = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
var raw = JsonConvert.SerializeObject(pr, Formatting.Indented);
|
||||||
|
File.WriteAllText(profilePath, raw);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void RequestCreateUser(string playerid, Player player)
|
||||||
|
{
|
||||||
|
string playerName = player.GetHoverName();
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_ServerCreateUser", playerid, playerName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void ClientLoadProfile(long sender, string json)
|
||||||
|
{
|
||||||
|
ProfileRoot loadedProfile = JsonConvert.DeserializeObject<ProfileRoot>(json);
|
||||||
|
|
||||||
|
playerLevel = loadedProfile.level;
|
||||||
|
playerExperience = loadedProfile.experience;
|
||||||
|
|
||||||
|
//ZLog.Log($"[Client] Profile loaded! Level: {loadedProfile.level}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void CreateUser(string playerid, Player player)
|
||||||
|
{
|
||||||
|
// We only send the raw data; we let the server handle the file I/O
|
||||||
|
string playerName = player?.GetHoverName() ?? "Unknown";
|
||||||
|
|
||||||
|
// Invoke the RPC on the Server (0 is the server ID)
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_ServerCreateUser", playerid, playerName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void ServerSetUserExperience(long sender, string playerid, int exp)
|
||||||
|
{
|
||||||
|
// Check to ensure this is actually the server running this
|
||||||
|
ZNet zn = new ZNet();
|
||||||
|
if (!zn.IsDedicated()) return;
|
||||||
|
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
ProfileRoot pro = JsonConvert.DeserializeObject<ProfileRoot>(File.ReadAllText(profilePath));
|
||||||
|
ProfileRoot pr = new ProfileRoot
|
||||||
|
{
|
||||||
|
playerid = playerid,
|
||||||
|
player_name = pro.player_name,
|
||||||
|
level = pro.level,
|
||||||
|
experience = exp
|
||||||
|
};
|
||||||
|
var raw = JsonConvert.SerializeObject(pr, Formatting.Indented);
|
||||||
|
File.WriteAllText(profilePath, raw);
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_RequestProfile", playerid);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void SetUserExperience(string playerid, int exp)
|
||||||
|
{
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
ProfileRoot pro = JsonConvert.DeserializeObject<ProfileRoot>(File.ReadAllText(profilePath));
|
||||||
|
ProfileRoot pr = new ProfileRoot
|
||||||
|
{
|
||||||
|
playerid = playerid,
|
||||||
|
player_name = pro.player_name,
|
||||||
|
level = pro.level,
|
||||||
|
experience = exp
|
||||||
|
};
|
||||||
|
var raw = JsonConvert.SerializeObject(pr, Formatting.Indented);
|
||||||
|
File.WriteAllText(profilePath, raw);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void ServerSetUserLevel(long sender, string playerid, int level)
|
||||||
|
{
|
||||||
|
// Check to ensure this is actually the server running this
|
||||||
|
ZNet zn = new ZNet();
|
||||||
|
if (!zn.IsDedicated()) return;
|
||||||
|
|
||||||
|
string profilePath = Path.Combine(MxValheimMod.internalDataEventSystemPath, $"{playerid}.json");
|
||||||
|
ProfileRoot pro = JsonConvert.DeserializeObject<ProfileRoot>(File.ReadAllText(profilePath));
|
||||||
|
ProfileRoot pr = new ProfileRoot
|
||||||
|
{
|
||||||
|
playerid = playerid,
|
||||||
|
player_name = pro.player_name,
|
||||||
|
level = level,
|
||||||
|
experience = pro.experience
|
||||||
|
};
|
||||||
|
var raw = JsonConvert.SerializeObject(pr, Formatting.Indented);
|
||||||
|
File.WriteAllText(profilePath, raw);
|
||||||
|
ZRoutedRpc.instance.InvokeRoutedRPC(0, "RPC_RequestProfile", playerid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
87
MxValheim/EventSystem/Reward.cs
Normal file
87
MxValheim/EventSystem/Reward.cs
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using UnityEngine;
|
||||||
|
|
||||||
|
namespace MxValheim.EventSystem
|
||||||
|
{
|
||||||
|
internal class Reward
|
||||||
|
{
|
||||||
|
public static int CalculateRewardCoin(int level, Heightmap.Biome biome)
|
||||||
|
{
|
||||||
|
Dictionary<int, int> biomeBase = new Dictionary<int, int>()
|
||||||
|
{
|
||||||
|
{ 1, 10 },
|
||||||
|
{ 2, 20 },
|
||||||
|
{ 4, 30 },
|
||||||
|
{ 8, 40 },
|
||||||
|
{ 16, 50 },
|
||||||
|
{ 512, 60 },
|
||||||
|
{ 64, 70 },
|
||||||
|
};
|
||||||
|
|
||||||
|
int rewardCoin = biomeBase[((int)biome)] * level;
|
||||||
|
|
||||||
|
return rewardCoin;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string GetRewardPrefab(int level, string type)
|
||||||
|
{
|
||||||
|
string[] foodList = { "OnionSoup", "CarrotSoup", "DeerStew", "Salad", "SerpentStew", "WolfMeatSkewer" };
|
||||||
|
string[] healList = { "MeadHealthMinor", "MeadHealthMedium", "MeadHealthMajor", "MeadHealthLingering" };
|
||||||
|
string[] metalList = { "CopperOre", "TinOre", "IronScrap", "SilverOre", "BlackMetalScrap", "FlametalOreNew" };
|
||||||
|
|
||||||
|
System.Random rng = new System.Random();
|
||||||
|
int foodTier = rng.Next(1, foodList.Length);
|
||||||
|
int healTier = rng.Next(1, healList.Length);
|
||||||
|
int metalTier = rng.Next(1, metalList.Length);
|
||||||
|
|
||||||
|
if (type == "food") return foodList[foodTier];
|
||||||
|
if (type == "heal") return healList[healTier];
|
||||||
|
if (type == "metal") return metalList[metalTier];
|
||||||
|
return "Wood";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void GiveItemRandom(string itemName, int qty)
|
||||||
|
{
|
||||||
|
System.Random rng = new System.Random();
|
||||||
|
|
||||||
|
Player player = Player.m_localPlayer;
|
||||||
|
if (player == null) return;
|
||||||
|
|
||||||
|
GameObject prefab = ObjectDB.instance.GetItemPrefab(itemName);
|
||||||
|
if (prefab != null)
|
||||||
|
{
|
||||||
|
player.GetInventory().AddItem(itemName, qty, 1, 0, 0, "");
|
||||||
|
|
||||||
|
player.Message(MessageHud.MessageType.TopLeft, $"Received {qty}x {itemName}");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug.LogWarning($"Item {itemName} not found in ObjectDB!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void GiveItemStatic(string itemName, int qty)
|
||||||
|
{
|
||||||
|
Player player = Player.m_localPlayer;
|
||||||
|
if (player == null) return;
|
||||||
|
|
||||||
|
GameObject prefab = ObjectDB.instance.GetItemPrefab(itemName);
|
||||||
|
if (prefab != null)
|
||||||
|
{
|
||||||
|
player.GetInventory().AddItem(itemName, qty, 1, 0, 0, "");
|
||||||
|
|
||||||
|
player.Message(MessageHud.MessageType.TopLeft, $"Received {qty}x {itemName}");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Debug.LogWarning($"Item {itemName} not found in ObjectDB!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,12 @@
|
|||||||
using BepInEx;
|
using BepInEx;
|
||||||
using BepInEx.Configuration;
|
using BepInEx.Configuration;
|
||||||
using HarmonyLib;
|
using HarmonyLib;
|
||||||
|
using MxValheim.EventSystem;
|
||||||
using MxValheim.KillFeed;
|
using MxValheim.KillFeed;
|
||||||
using MxValheim.Patch;
|
using MxValheim.Patch;
|
||||||
using MxValheim.Patch.HUD;
|
using MxValheim.Patch.HUD;
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
|
using Splatform;
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
@@ -26,7 +28,7 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
|
|
||||||
private const string ModGUID = "ovh.mxdev.mxvalheim";
|
private const string ModGUID = "ovh.mxdev.mxvalheim";
|
||||||
private const string ModName = "MxValheim";
|
private const string ModName = "MxValheim";
|
||||||
public const string ModVersion = "1.5.8";
|
private const string ModVersion = "1.6.1";
|
||||||
|
|
||||||
public static ConfigEntry<bool> Config_Locked;
|
public static ConfigEntry<bool> Config_Locked;
|
||||||
public static ConfigEntry<int> Config_OreMultiplier;
|
public static ConfigEntry<int> Config_OreMultiplier;
|
||||||
@@ -39,11 +41,15 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
|
|
||||||
public static string modPath = Path.Combine(Paths.PluginPath, "MxValheim");
|
public static string modPath = Path.Combine(Paths.PluginPath, "MxValheim");
|
||||||
public static string internalConfigsPath = Path.Combine(modPath, "Configs");
|
public static string internalConfigsPath = Path.Combine(modPath, "Configs");
|
||||||
|
public static string internalDataPath = Path.Combine(modPath, "Data");
|
||||||
|
public static string internalDataEventSystemPath = Path.Combine(internalDataPath, "EventSystem");
|
||||||
public static string WeightConfigPath => Path.Combine(internalConfigsPath, "items_weight.json");
|
public static string WeightConfigPath => Path.Combine(internalConfigsPath, "items_weight.json");
|
||||||
public static string AutoDoorConfigPath => Path.Combine(internalConfigsPath, "auto_doors.json");
|
public static string AutoDoorConfigPath => Path.Combine(internalConfigsPath, "auto_doors.json");
|
||||||
public static DoorRoot CachedConfig;
|
public static DoorRoot CachedConfig;
|
||||||
public static Dictionary<string, float> WeightSettings = new Dictionary<string, float>();
|
public static Dictionary<string, float> WeightSettings = new Dictionary<string, float>();
|
||||||
|
|
||||||
|
public static bool initExpBar = true;
|
||||||
|
|
||||||
// Data structures
|
// Data structures
|
||||||
public class KillData
|
public class KillData
|
||||||
{
|
{
|
||||||
@@ -79,8 +85,8 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
{
|
{
|
||||||
Instance = this;
|
Instance = this;
|
||||||
|
|
||||||
Config_OreMultiplier = Config.Bind("General","OreMultiplier",3,"How many items should drop for every 1 ore/scrap found.");
|
Config_OreMultiplier = Config.Bind("General", "OreMultiplier", 3, "How many items should drop for every 1 ore/scrap found.");
|
||||||
Config_rangeMultiplier = Config.Bind("General", "CraftingRangeMultiplier",2.0f,"Multiplier for the workbench build/crafting range. Default is 2x.");
|
Config_rangeMultiplier = Config.Bind("General", "CraftingRangeMultiplier", 2.0f, "Multiplier for the workbench build/crafting range. Default is 2x.");
|
||||||
Config_bowDrawSpeedBonusPerLevel = Config.Bind("General", "BowDrawSpeedBonusPercentPerLevel", 1.0f, "Shorten the bow draw speed by this percent for every bow upgrade level.");
|
Config_bowDrawSpeedBonusPerLevel = Config.Bind("General", "BowDrawSpeedBonusPercentPerLevel", 1.0f, "Shorten the bow draw speed by this percent for every bow upgrade level.");
|
||||||
Config_rainDamage = Config.Bind("General", "RainDamage", true, "Set to true to stop rain damage, false to return to vanilla behavior.");
|
Config_rainDamage = Config.Bind("General", "RainDamage", true, "Set to true to stop rain damage, false to return to vanilla behavior.");
|
||||||
Config_boatSpeed = Config.Bind("General", "BoatSpeedMultiplier", 2.0f, "Your boat/raft will move without wind at a speed multiplied by this value.");
|
Config_boatSpeed = Config.Bind("General", "BoatSpeedMultiplier", 2.0f, "Your boat/raft will move without wind at a speed multiplied by this value.");
|
||||||
@@ -115,17 +121,7 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
string text = __instance.m_input.text;
|
string text = __instance.m_input.text;
|
||||||
if (text.ToLower() == "listicons")
|
if (text.ToLower() == "listicons")
|
||||||
{
|
{
|
||||||
var spriteAsset = Resources.FindObjectsOfTypeAll<TMP_SpriteAsset>().FirstOrDefault(x => x.name == "icons"); ;
|
|
||||||
|
|
||||||
if (spriteAsset != null)
|
|
||||||
{
|
|
||||||
Debug.Log($"--- Listing all sprites in {spriteAsset.name} ---");
|
|
||||||
for (int i = 0; i < spriteAsset.spriteCharacterTable.Count; i++)
|
|
||||||
{
|
|
||||||
var sprite = spriteAsset.spriteCharacterTable[i];
|
|
||||||
Debug.Log($"Index: {i} | Name: {sprite.name}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -141,13 +137,25 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
ZRoutedRpc.instance.Register<string, string, string, string, int>("RPC_MxKillMsg",
|
ZRoutedRpc.instance.Register<string, string, string, string, int>("RPC_MxKillMsg",
|
||||||
new RoutedMethod<string, string, string, string, int>.Method(kfp.OnReceiveKillMsg));
|
new RoutedMethod<string, string, string, string, int>.Method(kfp.OnReceiveKillMsg));
|
||||||
|
|
||||||
Debug.Log("MxValheimMod: RPC Registered successfully with explicit delegate.");
|
ZRoutedRpc.instance.Register<string, string>("RPC_ServerCreateUser", Profiles.ServerReceiveCreateUser);
|
||||||
|
ZRoutedRpc.instance.Register<string>("RPC_RequestProfile", Profiles.ServerHandleProfileRequest);
|
||||||
|
ZRoutedRpc.instance.Register<string>("RPC_ClientReceiveProfile", Profiles.ClientLoadProfile);
|
||||||
|
ZRoutedRpc.instance.Register<string>("RPC_UpdateProgressBar", Profiles.ClientLoadProfile);
|
||||||
|
ZRoutedRpc.instance.Register<string, int>("RPC_ServerSetUserExperience", Profiles.ServerSetUserExperience);
|
||||||
|
ZRoutedRpc.instance.Register<string, int>("RPC_ServerSetUserLevel", Profiles.ServerSetUserLevel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Update()
|
void Update()
|
||||||
{
|
{
|
||||||
|
// Only run if we are actually in the game world
|
||||||
|
if (Player.m_localPlayer != null)
|
||||||
|
{
|
||||||
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
Experience.UpdateProgressBar(pid);
|
||||||
|
}
|
||||||
|
|
||||||
// Only run if we are actually in the game world
|
// Only run if we are actually in the game world
|
||||||
if (Player.m_localPlayer != null)
|
if (Player.m_localPlayer != null)
|
||||||
{
|
{
|
||||||
@@ -166,6 +174,12 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
{
|
{
|
||||||
Clock clk = new Clock();
|
Clock clk = new Clock();
|
||||||
clk.CreateOverlay();
|
clk.CreateOverlay();
|
||||||
|
if (initExpBar)
|
||||||
|
{
|
||||||
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
Experience.UpdateProgressBar(pid);
|
||||||
|
initExpBar = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Clock.m_textComponent != null)
|
if (Clock.m_textComponent != null)
|
||||||
@@ -173,7 +187,8 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
Clock clk = new Clock();
|
Clock clk = new Clock();
|
||||||
string timeString = clk.GetFormattedTime();
|
string timeString = clk.GetFormattedTime();
|
||||||
int day = EnvMan.instance.GetDay();
|
int day = EnvMan.instance.GetDay();
|
||||||
Clock.m_textComponent.text = $"<color=#32a852>{Localization.instance.Localize("$clock_string_day")} {day}</color> <color=#677074>|</color> <color=#d1954a>{timeString}</color>";
|
string pid = GetUserPlayerID(Player.m_localPlayer).ToString();
|
||||||
|
Clock.m_textComponent.text = $"<color=#32a852>{Localization.instance.Localize("$clock_string_rank")} {Profiles.playerLevel}</color> <color=#677074>|</color> <color=#32a852>{Localization.instance.Localize("$clock_string_day")} {day}</color> <color=#677074>|</color> <color=#d1954a>{timeString}</color>";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -236,7 +251,11 @@ public class MxValheimMod : BaseUnityPlugin
|
|||||||
}
|
}
|
||||||
// KillFeed Logic
|
// KillFeed Logic
|
||||||
/////////////////
|
/////////////////
|
||||||
|
}
|
||||||
|
|
||||||
|
public static long GetUserPlayerID(Player player)
|
||||||
|
{
|
||||||
|
return player.GetPlayerID();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void LoadLocalization()
|
public static void LoadLocalization()
|
||||||
|
|||||||
@@ -35,16 +35,25 @@
|
|||||||
<Reference Include="0Harmony">
|
<Reference Include="0Harmony">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\core\0Harmony.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\core\0Harmony.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
|
<Reference Include="assembly_googleanalytics">
|
||||||
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_googleanalytics.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="assembly_guiutils_publicized">
|
<Reference Include="assembly_guiutils_publicized">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\publicized_assemblies\assembly_guiutils_publicized.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\publicized_assemblies\assembly_guiutils_publicized.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
|
<Reference Include="assembly_utils, Version=0.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||||
|
<SpecificVersion>False</SpecificVersion>
|
||||||
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_utils.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="assembly_valheim_publicized">
|
<Reference Include="assembly_valheim_publicized">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\publicized_assemblies\assembly_valheim_publicized.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\publicized_assemblies\assembly_valheim_publicized.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="BepInEx">
|
<Reference Include="BepInEx">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\core\BepInEx.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\core\BepInEx.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="gui_framework, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null" />
|
<Reference Include="gui_framework">
|
||||||
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\gui_framework.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="Newtonsoft.Json">
|
<Reference Include="Newtonsoft.Json">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\plugins\Newtonsoft.Json.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\BepInEx\plugins\Newtonsoft.Json.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
@@ -59,7 +68,9 @@
|
|||||||
<Reference Include="System.Data" />
|
<Reference Include="System.Data" />
|
||||||
<Reference Include="System.Net.Http" />
|
<Reference Include="System.Net.Http" />
|
||||||
<Reference Include="System.Xml" />
|
<Reference Include="System.Xml" />
|
||||||
<Reference Include="Unity.TextMeshPro, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null" />
|
<Reference Include="Unity.TextMeshPro">
|
||||||
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\Unity.TextMeshPro.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="UnityEngine">
|
<Reference Include="UnityEngine">
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
@@ -74,7 +85,9 @@
|
|||||||
<SpecificVersion>False</SpecificVersion>
|
<SpecificVersion>False</SpecificVersion>
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.PhysicsModule.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.PhysicsModule.dll</HintPath>
|
||||||
</Reference>
|
</Reference>
|
||||||
<Reference Include="UnityEngine.TextRenderingModule, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null" />
|
<Reference Include="UnityEngine.TextRenderingModule">
|
||||||
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.TextRenderingModule.dll</HintPath>
|
||||||
|
</Reference>
|
||||||
<Reference Include="UnityEngine.UI, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
<Reference Include="UnityEngine.UI, Version=1.0.0.0, Culture=neutral, processorArchitecture=MSIL">
|
||||||
<SpecificVersion>False</SpecificVersion>
|
<SpecificVersion>False</SpecificVersion>
|
||||||
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.UI.dll</HintPath>
|
<HintPath>E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\UnityEngine.UI.dll</HintPath>
|
||||||
@@ -85,6 +98,10 @@
|
|||||||
</Reference>
|
</Reference>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<Compile Include="EventSystem\Experience.cs" />
|
||||||
|
<Compile Include="EventSystem\Patch.cs" />
|
||||||
|
<Compile Include="EventSystem\Profiles.cs" />
|
||||||
|
<Compile Include="EventSystem\Reward.cs" />
|
||||||
<Compile Include="KillFeed\NPCIcon.cs" />
|
<Compile Include="KillFeed\NPCIcon.cs" />
|
||||||
<Compile Include="KillFeed\Patch.cs" />
|
<Compile Include="KillFeed\Patch.cs" />
|
||||||
<Compile Include="MxValheim.cs" />
|
<Compile Include="MxValheim.cs" />
|
||||||
@@ -99,7 +116,9 @@
|
|||||||
<Compile Include="Properties\AssemblyInfo.cs" />
|
<Compile Include="Properties\AssemblyInfo.cs" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
|
<Analyzer Include="E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_googleanalytics.dll" />
|
||||||
<Analyzer Include="E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_guiutils.dll" />
|
<Analyzer Include="E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_guiutils.dll" />
|
||||||
|
<Analyzer Include="E:\SteamLibrary\steamapps\common\Valheim\Valheim_Data\Managed\assembly_utils.dll" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
<Import Project="$(MSBuildToolsPath)\Microsoft.CSharp.targets" />
|
||||||
<PropertyGroup>
|
<PropertyGroup>
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ namespace MxValheim.Patch.HUD
|
|||||||
{
|
{
|
||||||
public static GameObject m_canvasObject;
|
public static GameObject m_canvasObject;
|
||||||
public static Text m_textComponent;
|
public static Text m_textComponent;
|
||||||
|
public static RectTransform m_barFill;
|
||||||
|
|
||||||
// This replaces the missing GetTimeString() method
|
// This replaces the missing GetTimeString() method
|
||||||
public string GetFormattedTime()
|
public string GetFormattedTime()
|
||||||
@@ -29,68 +30,62 @@ namespace MxValheim.Patch.HUD
|
|||||||
|
|
||||||
public void CreateOverlay()
|
public void CreateOverlay()
|
||||||
{
|
{
|
||||||
// 1. Root Canvas
|
|
||||||
m_canvasObject = new GameObject("ModdedCanvas");
|
m_canvasObject = new GameObject("ModdedCanvas");
|
||||||
Canvas canvas = m_canvasObject.AddComponent<Canvas>();
|
Canvas canvas = m_canvasObject.AddComponent<Canvas>();
|
||||||
canvas.renderMode = RenderMode.ScreenSpaceOverlay;
|
canvas.renderMode = RenderMode.ScreenSpaceOverlay;
|
||||||
canvas.sortingOrder = 999;
|
canvas.sortingOrder = 999;
|
||||||
|
|
||||||
// 2. Background Panel
|
// --- CLOCK PANEL ---
|
||||||
GameObject panelObj = new GameObject("TextBackground");
|
GameObject panelObj = new GameObject("TextBackground");
|
||||||
panelObj.transform.SetParent(m_canvasObject.transform, false);
|
panelObj.transform.SetParent(m_canvasObject.transform, false);
|
||||||
|
panelObj.AddComponent<Image>().color = new Color(0, 0, 0, 0.5f);
|
||||||
|
|
||||||
Image panelImage = panelObj.AddComponent<Image>();
|
ContentSizeFitter fitter = panelObj.AddComponent<ContentSizeFitter>();
|
||||||
panelImage.color = new Color(0, 0, 0, 0.5f); // 50% transparent black
|
fitter.horizontalFit = ContentSizeFitter.FitMode.PreferredSize;
|
||||||
|
fitter.verticalFit = ContentSizeFitter.FitMode.PreferredSize;
|
||||||
|
|
||||||
|
HorizontalLayoutGroup layout = panelObj.AddComponent<HorizontalLayoutGroup>();
|
||||||
|
layout.padding = new RectOffset(15, 15, 5, 5);
|
||||||
|
|
||||||
RectTransform panelRect = panelObj.GetComponent<RectTransform>();
|
RectTransform panelRect = panelObj.GetComponent<RectTransform>();
|
||||||
panelRect.anchorMin = new Vector2(0.5f, 1.0f); // Top Center
|
panelRect.anchorMin = new Vector2(0.5f, 1.0f);
|
||||||
panelRect.anchorMax = new Vector2(0.5f, 1.0f);
|
panelRect.anchorMax = new Vector2(0.5f, 1.0f);
|
||||||
panelRect.pivot = new Vector2(0.5f, 1.0f);
|
panelRect.pivot = new Vector2(0.5f, 1.0f);
|
||||||
panelRect.anchoredPosition = new Vector2(0, -5); // 5px gap from top
|
panelRect.anchoredPosition = new Vector2(0, -5);
|
||||||
panelRect.sizeDelta = new Vector2(180, 35); // Width and Height of the bar
|
|
||||||
|
|
||||||
// 3. Text (Attached to Panel)
|
// --- TEXT ---
|
||||||
GameObject textObj = new GameObject("ModdedText");
|
GameObject textObj = new GameObject("ModdedText");
|
||||||
textObj.transform.SetParent(panelObj.transform, false);
|
textObj.transform.SetParent(panelObj.transform, false);
|
||||||
|
|
||||||
m_textComponent = textObj.AddComponent<Text>();
|
m_textComponent = textObj.AddComponent<Text>();
|
||||||
// Attempt to find Valheim's specific font, fallback to Arial if not found
|
m_textComponent.font = Resources.GetBuiltinResource<Font>("Arial.ttf");
|
||||||
Font valheimFont = null;
|
|
||||||
foreach (Font f in Resources.FindObjectsOfTypeAll<Font>())
|
|
||||||
{
|
|
||||||
if (f.name == "AveriaSerifLibre-Bold")
|
|
||||||
{
|
|
||||||
valheimFont = f;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Corrected fallback for older Unity versions used in Valheim
|
|
||||||
if (valheimFont == null)
|
|
||||||
{
|
|
||||||
valheimFont = Resources.GetBuiltinResource<Font>("Arial.ttf");
|
|
||||||
}
|
|
||||||
|
|
||||||
m_textComponent.font = valheimFont;
|
|
||||||
m_textComponent.fontSize = 18;
|
m_textComponent.fontSize = 18;
|
||||||
m_textComponent.color = Color.white;
|
m_textComponent.color = Color.white;
|
||||||
m_textComponent.alignment = TextAnchor.MiddleCenter;
|
|
||||||
m_textComponent.supportRichText = true;
|
|
||||||
|
|
||||||
// Add Shadow so it's visible in snow
|
// --- PROGRESS BAR CONTAINER ---
|
||||||
var shadow = textObj.AddComponent<Shadow>();
|
// This sits under the main rectangle
|
||||||
shadow.effectColor = Color.black;
|
GameObject barBgObj = new GameObject("BarBackground");
|
||||||
shadow.effectDistance = new Vector2(2, 2);
|
barBgObj.transform.SetParent(m_canvasObject.transform, false);
|
||||||
|
barBgObj.AddComponent<Image>().color = new Color(0.1f, 0.1f, 0.1f, 0.8f); // Dark background
|
||||||
|
|
||||||
// FIX: Correct way to make text fill the parent panel
|
RectTransform barBgRect = barBgObj.GetComponent<RectTransform>();
|
||||||
RectTransform textRect = textObj.GetComponent<RectTransform>();
|
barBgRect.anchorMin = new Vector2(0.5f, 1.0f);
|
||||||
textRect.anchorMin = Vector2.zero; // (0, 0)
|
barBgRect.anchorMax = new Vector2(0.5f, 1.0f);
|
||||||
textRect.anchorMax = Vector2.one; // (1, 1)
|
barBgRect.pivot = new Vector2(0.5f, 1.0f);
|
||||||
textRect.pivot = new Vector2(0.5f, 0.5f);
|
// Positioned below the clock panel (Clock is 35px high + 5px gap = -40)
|
||||||
|
barBgRect.anchoredPosition = new Vector2(0, -45);
|
||||||
|
barBgRect.sizeDelta = new Vector2(180, 8); // Match width of clock, small height
|
||||||
|
|
||||||
// Resetting these to zero ensures the text box matches the panel exactly
|
// --- PROGRESS BAR FILL ---
|
||||||
textRect.offsetMin = Vector2.zero;
|
GameObject fillObj = new GameObject("BarFill");
|
||||||
textRect.offsetMax = Vector2.zero;
|
fillObj.transform.SetParent(barBgObj.transform, false);
|
||||||
|
fillObj.AddComponent<Image>().color = Color.yellow; // Classic Valheim XP color
|
||||||
|
|
||||||
|
m_barFill = fillObj.GetComponent<RectTransform>();
|
||||||
|
m_barFill.anchorMin = new Vector2(0, 0);
|
||||||
|
m_barFill.anchorMax = new Vector2(0.5f, 1); // This X value (0.5) will be updated via code
|
||||||
|
m_barFill.pivot = new Vector2(0, 0.5f);
|
||||||
|
m_barFill.offsetMin = Vector2.zero;
|
||||||
|
m_barFill.offsetMax = Vector2.zero;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ Official **MxValheim Server** Mod.
|
|||||||
**This mod is created to be used Client/Server side. A lot of features are not working in solo mode.**
|
**This mod is created to be used Client/Server side. A lot of features are not working in solo mode.**
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
- Enhanced Event System with level and reward based on current biome.
|
||||||
- Display the day and the formated time in-game.
|
- Display the day and the formated time in-game.
|
||||||
- Use your client language with built-in localization.
|
- Use your client language with built-in localization.
|
||||||
- Kill Feed with custom UI showing player kill and death.
|
- Kill Feed with custom UI showing player kill and death.
|
||||||
|
|||||||
Reference in New Issue
Block a user