using System;
using System.IO;
using System.Linq;
using Microsoft.Xna.Framework;
using Omegasis.NightOwl.Framework;
using StardewModdingAPI;
using StardewModdingAPI.Events;
using StardewValley;
/*TODO:
Issues:
-Mail can't be wiped without destroying all mail.
-Lighting transition does not work if it is raining.
-set the weather to clear if you are stayig up late.
-transition still doesnt work. However atleast it is dark now.
-Known glitched
*/
namespace Omegasis.NightOwl
{
/// The mod entry point.
public class NightOwl : Mod
{
/*********
** Properties
*********/
/// The mod configuration.
private ModConfig Config;
/****
** Context
****/
/// Whether the player stayed up all night.
private bool IsUpLate;
/// Whether the player should be reset to their pre-collapse details for the morning transition on the next update.
private bool ShouldResetPlayerAfterCollapseNow;
/// Whether the player just started a new day.
private bool JustStartedNewDay;
/// Whether the player just collapsed for the morning transition.
private bool JustCollapsed;
/****
** Pre-collapse state
****/
/// The player's location name before they collapsed.
private string PreCollapseMap;
/// The player's tile position before they collapsed.
private Point PreCollapseTile;
/// The player's money before they collapsed.
private int PreCollapseMoney;
/// The player's stamina before they collapsed.
private float PreCollapseStamina;
/// The player's health before they collapsed.
private int PreCollapseHealth;
/*********
** Public methods
*********/
/// The mod entry point, called after the mod is first loaded.
/// Provides simplified APIs for writing mods.
public override void Entry(IModHelper helper)
{
this.Config = helper.ReadConfig();
TimeEvents.TimeOfDayChanged += this.TimeEvents_TimeOfDayChanged;
TimeEvents.AfterDayStarted += this.TimeEvents_AfterDayStarted;
SaveEvents.AfterLoad += this.SaveEvents_AfterLoad;
GameEvents.FourthUpdateTick += this.GameEvents_FourthUpdateTick;
}
/*********
** Private methods
*********/
/// The method invoked every fourth game update (roughly 15 times per second).
/// The event sender.
/// The event data.
public void GameEvents_FourthUpdateTick(object sender, EventArgs e)
{
try
{
// reset position after collapse
if (Context.IsWorldReady && this.JustStartedNewDay && this.Config.KeepPositionAfterCollapse)
{
if (this.PreCollapseMap != null)
Game1.warpFarmer(this.PreCollapseMap, this.PreCollapseTile.X, this.PreCollapseTile.Y, false);
this.PreCollapseMap = null;
this.JustStartedNewDay = false;
this.JustCollapsed = false;
}
}
catch (Exception ex)
{
this.Monitor.Log(ex.ToString(), LogLevel.Error);
this.WriteErrorLog();
}
}
/// The method invoked after the player loads a save.
/// The event sender.
/// The event data.
public void SaveEvents_AfterLoad(object sender, EventArgs e)
{
this.IsUpLate = false;
this.JustStartedNewDay = false;
this.JustCollapsed = false;
}
/// The method invoked when a new day starts.
/// The event sender.
/// The event data.
public void TimeEvents_AfterDayStarted(object sender, EventArgs e)
{
try
{
// reset data
this.IsUpLate = false;
Game1.farmerShouldPassOut = false;
// transition to the next day
if (this.ShouldResetPlayerAfterCollapseNow)
{
this.ShouldResetPlayerAfterCollapseNow = false;
if (this.Config.KeepStaminaAfterCollapse)
Game1.player.stamina = this.PreCollapseStamina;
if (this.Config.KeepHealthAfterCollapse)
Game1.player.health = this.PreCollapseHealth;
if (this.Config.KeepMoneyAfterCollapse)
Game1.player.money = this.PreCollapseMoney;
if (this.Config.KeepPositionAfterCollapse)
Game1.warpFarmer(this.PreCollapseMap, this.PreCollapseTile.X, this.PreCollapseTile.Y, false);
}
// delete annoying charge messages (if only I could do this with mail IRL)
if (this.Config.SkipCollapseMail)
{
string[] validMail = Game1.mailbox
.Where(p => !p.Contains("passedOut"))
.ToArray();
Game1.mailbox.Clear();
foreach (string mail in validMail)
Game1.mailbox.Enqueue(mail);
}
this.JustStartedNewDay = true;
}
catch (Exception ex)
{
this.Monitor.Log(ex.ToString(), LogLevel.Error);
this.WriteErrorLog();
}
}
/// The method invoked when changes.
/// The event sender.
/// The event data.
private void TimeEvents_TimeOfDayChanged(object sender, EventArgsIntChanged e)
{
if (!Context.IsWorldReady)
return;
try
{
// transition morning light more realistically
if (this.Config.MorningLightTransition && Game1.timeOfDay > 400 && Game1.timeOfDay < 600)
{
float colorMod = (1300 - Game1.timeOfDay) / 1000f;
Game1.outdoorLight = Game1.ambientLight * colorMod;
}
// transition to next morning
if (this.Config.StayUp && Game1.timeOfDay == 2550)
{
Game1.isRaining = false; // remove rain, otherwise lighting gets screwy
Game1.updateWeatherIcon();
Game1.timeOfDay = 150; //change it from 1:50 am late, to 1:50 am early
}
// collapse player at 6am to save & reset
if (Game1.timeOfDay == 550)
this.IsUpLate = true;
if (this.IsUpLate && Game1.timeOfDay == 600 && !this.JustCollapsed)
{
this.JustCollapsed = true;
this.ShouldResetPlayerAfterCollapseNow = true;
this.PreCollapseTile = new Point(Game1.player.getTileX(), Game1.player.getTileY());
this.PreCollapseMap = Game1.player.currentLocation.name;
this.PreCollapseStamina = Game1.player.stamina;
this.PreCollapseHealth = Game1.player.health;
this.PreCollapseMoney = Game1.player.money;
if (Game1.currentMinigame != null)
Game1.currentMinigame = null;
Game1.farmerShouldPassOut = true;
}
}
catch (Exception ex)
{
this.Monitor.Log(ex.ToString(), LogLevel.Error);
this.WriteErrorLog();
}
}
/// Write the current mod state to the error log file.
private void WriteErrorLog()
{
var state = new
{
this.Config,
this.IsUpLate,
this.ShouldResetPlayerAfterCollapseNow,
this.JustStartedNewDay,
this.JustCollapsed,
this.PreCollapseMap,
this.PreCollapseTile,
this.PreCollapseMoney,
this.PreCollapseStamina,
this.PreCollapseHealth
};
string path = Path.Combine(this.Helper.DirectoryPath, "Error_Logs", "Mod_State.json");
this.Helper.WriteJsonFile(path, state);
}
}
}