using System; using System.Collections.Generic; using Microsoft.Xna.Framework; using Microsoft.Xna.Framework.Graphics; using Microsoft.Xna.Framework.Input; using Newtonsoft.Json; using PyTK.CustomElementHandler; using StardustCore.Animations; using StardewValley; using StardewValley.Objects; using Netcode; using StardewModdingAPI; using Revitalize.Framework.Utilities; namespace Revitalize.Framework.Objects { // TODO: // -Multiple Lights // -Events when walking over? // -Inventories /// A custom object template. public class CustomObject : StardewValley.Object,ISaveElement { public string id { get { return this.info.id; } } public BasicItemInformation info; private GameLocation _location; [JsonIgnore] public GameLocation location { get { if (this._location == null) { this._location = Game1.getLocationFromName(this.info.locationName); return this._location; } return this._location; } set { this._location = value; if (this._location == null) this.info.locationName = ""; else { this.info.locationName = this._location.Name; } } } public Guid guid; /// The animation manager. public AnimationManager animationManager => this.info.animationManager; /// The display texture for this object. [JsonIgnore] public Texture2D displayTexture => this.animationManager.getTexture(); public string ItemInfo { get { return Revitalize.ModCore.Serializer.ToJSONString(this.info); } set { this.info = (BasicItemInformation)Revitalize.ModCore.Serializer.DeserializeFromJSONString(value, typeof(BasicItemInformation)); } } protected Netcode.NetString netItemInfo; /// Empty constructor. public CustomObject() { this.guid = Guid.NewGuid(); //this.InitNetFields(); } /// Construct an instance. public CustomObject(BasicItemInformation info, int Stack = 1) : base() { this.info = info; this.initializeBasics(); this.guid = Guid.NewGuid(); this.Stack = Stack; } /// Construct an instance. public CustomObject(BasicItemInformation info, Vector2 TileLocation, int Stack = 1) : base() { this.info = info; this.initializeBasics(); this.guid = Guid.NewGuid(); this.Stack = Stack; } /// Sets some basic information up. public virtual void initializeBasics() { this.name = this.info.name; this.displayName = this.getDisplayNameFromStringsFile(this.id); this.Edibility = this.info.edibility; this.Category = -9; //For crafting. this.displayName = this.info.name; this.setOutdoors.Value = this.info.canBeSetOutdoors; this.setIndoors.Value = this.info.canBeSetIndoors; this.isLamp.Value = false; this.Fragility = 0; this.updateDrawPosition(0, 0); this.bigCraftable.Value = this.info.bigCraftable; this.Price = this.info.price; ModCore.CustomObjects.Add(this.guid, this); this.NetFields.AddField(new NetGuid(this.guid)); //this.initNetFields(); //this.InitNetFields(); //if (this.info.ignoreBoundingBox) // this.boundingBox.Value = new Rectangle(int.MinValue, int.MinValue, 0, 0); } public override bool isPassable() { return this.info.ignoreBoundingBox; //|| Revitalize.ModCore.playerInfo.sittingInfo.SittingObject == this; } public override Rectangle getBoundingBox(Vector2 tileLocation) { //Revitalize.ModCore.log(System.Environment.StackTrace); return this.info.ignoreBoundingBox ? new Rectangle(int.MinValue, int.MinValue, 0, 0) : base.getBoundingBox(tileLocation); } /// Checks for interaction with the object. public override bool checkForAction(Farmer who, bool justCheckingForActivity = false) { MouseState mState = Mouse.GetState(); KeyboardState keyboardState = Game1.GetKeyboardState(); if (mState.RightButton == ButtonState.Pressed && (keyboardState.IsKeyDown(Keys.LeftShift) || !keyboardState.IsKeyDown(Keys.RightShift))) { //ModCore.log("Right clicked!"); return this.rightClicked(who); } if (mState.RightButton == ButtonState.Pressed && (keyboardState.IsKeyDown(Keys.LeftShift) || keyboardState.IsKeyDown(Keys.RightShift))) return this.shiftRightClicked(who); return base.checkForAction(who, justCheckingForActivity); if (justCheckingForActivity) return true; ModCore.log("Left clicked!"); return this.clicked(who); } /* public override ICustomObject recreate(Dictionary additionalSaveData, object replacement) { CustomObjectData data = CustomObjectData.collection[additionalSaveData["id"]]; BasicItemInformation info = Revitalize.ModCore.Serializer.DeserializeFromJSONString(additionalSaveData["ItemInfo"]); return new CustomObject(data, info); } */ /* public override Dictionary getAdditionalSaveData() { Dictionary serializedInfo = new Dictionary(); serializedInfo.Add("id", this.ItemInfo); serializedInfo.Add("ItemInfo", Revitalize.ModCore.Serializer.ToJSONString(this.info)); return serializedInfo; } */ /* public override void rebuild(Dictionary additionalSaveData, object replacement) { string id = additionalSaveData["id"]; this.ItemInfo = id; this.info = Revitalize.ModCore.Serializer.DeserializeFromJSONString(additionalSaveData["ItemInfo"]); base.rebuild(additionalSaveData, replacement); ModCore.log("Rebuilt custom object."); } */ /// What happens when the player right clicks the object. public virtual bool rightClicked(Farmer who) { // Game1.showRedMessage("YOOO"); //do some stuff when the right button is down // rotate(); if (this.heldObject.Value != null) { // Game1.player.addItemByMenuIfNecessary(this.heldObject); // this.heldObject = null; } else { // this.heldObject = Game1.player.ActiveObject; // Game1.player.removeItemFromInventory(heldObject); } return true; } /// What happens when the player shift-right clicks this object. public virtual bool shiftRightClicked(Farmer who) { //ModCore.log("Shift right clicked!"); return true; } /// What happens when the player left clicks the object. public override bool clicked(Farmer who) { //ModCore.log("Clicky click!"); return this.removeAndAddToPlayersInventory(); //return base.clicked(who); } /// What happens when a player uses a tool on this object. public override bool performToolAction(Tool t, GameLocation location) { if (t.GetType() == typeof(StardewValley.Tools.Axe) || t.GetType() == typeof(StardewValley.Tools.Pickaxe)) { Game1.createItemDebris(this, Game1.player.getStandingPosition(), Game1.player.getDirection()); this.location = null; this.updateDrawPosition(0, 0); Game1.player.currentLocation.removeObject(this.TileLocation, false); this.updateDrawPosition(0, 0); return false; } return false; //return base.performToolAction(t, location); } /// Remove the object from the world and add it to the player's inventory if possible. public virtual bool removeAndAddToPlayersInventory() { if (Game1.player.isInventoryFull()) { Game1.showRedMessage("Inventory full."); return false; } this.location = null; Game1.player.currentLocation.removeObject(this.TileLocation, false); Game1.player.addItemToInventory(this); this.updateDrawPosition(0, 0); return true; } /// Gets the category color for the object. public override Color getCategoryColor() { return this.info.categoryColor; //return data.categoryColor; } /// Gets the category name for the object. public override string getCategoryName() { return this.info.categoryName; } /// Gets the description for the object. public override string getDescription() { string text = this.info.description; SpriteFont smallFont = Game1.smallFont; int width = Game1.tileSize * 4 + Game1.tileSize / 4; return Game1.parseText(text, smallFont, width); } /// Places an object down. public override bool placementAction(GameLocation location, int x, int y, Farmer who = null) { if (this.info.canBeSetIndoors == false && location.IsOutdoors == false) return false; if (this.info.canBeSetOutdoors == false && location.IsOutdoors) return false; this.updateDrawPosition(x, y); this.location = location; return base.placementAction(location, x, y, who); } public override bool canBePlacedHere(GameLocation l, Vector2 tile) { if (this.info.ignoreBoundingBox && l.isObjectAtTile((int)tile.X, (int)tile.Y) == false) return true; return base.canBePlacedHere(l, tile); } public virtual void rotate() { if (this.info.facingDirection == Enums.Direction.Down) this.info.facingDirection = Enums.Direction.Right; else if (this.info.facingDirection == Enums.Direction.Right) this.info.facingDirection = Enums.Direction.Up; else if (this.info.facingDirection == Enums.Direction.Up) this.info.facingDirection = Enums.Direction.Left; else if (this.info.facingDirection == Enums.Direction.Left) this.info.facingDirection = Enums.Direction.Down; if (this.info.animationManager.animations.ContainsKey(this.generateRotationalAnimationKey())) { this.info.animationManager.setAnimation(this.generateRotationalAnimationKey()); } else { //Revitalize.ModCore.log("Animation does not exist...." + generateRotationalAnimationKey()); } } public string generateRotationalAnimationKey() { return (this.info.animationManager.currentAnimationName.Split('_')[0]) + "_" + (int)this.info.facingDirection; } public string generateDefaultRotationalAnimationKey() { return ("Default" + "_" + (int)this.info.facingDirection); } /// Updates a visual draw position. public virtual void updateDrawPosition(int x, int y) { this.info.drawPosition = new Vector2((int)(x / Game1.tileSize), (int)(y / Game1.tileSize)); //this.info.drawPosition = new Vector2((float)this.boundingBox.X, (float)(this.boundingBox.Y - (this.animationManager.currentAnimation.sourceRectangle.Height * Game1.pixelZoom - this.boundingBox.Height))); } /// Gets a clone of the game object. public override Item getOne() { return new CustomObject(this.info); } /// What happens when the object is drawn at a tile location. public override void draw(SpriteBatch spriteBatch, int x, int y, float alpha = 1f) { if (this.info == null) { ModCore.log("Info is null:"); ModCore.log("Need request for guid: " + this.guid); MultiplayerUtilities.SendRequestForSpecificGUID(this.guid); return; } if (x <= -1) { spriteBatch.Draw(this.info.animationManager.getTexture(), Game1.GlobalToLocal(Game1.viewport, this.info.drawPosition), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * alpha, 0f, Vector2.Zero, (float)Game1.pixelZoom, this.flipped ? SpriteEffects.FlipHorizontally : SpriteEffects.None, Math.Max(0f, (float)(this.TileLocation.Y * Game1.tileSize) / 10000f)); } else { //The actual planter box being drawn. if (this.animationManager == null) { if (this.animationManager.getExtendedTexture() == null) ModCore.ModMonitor.Log("Tex Extended is null???"); spriteBatch.Draw(this.displayTexture, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)(x * Game1.tileSize), y * Game1.tileSize)), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * alpha, 0f, Vector2.Zero, (float)Game1.pixelZoom, this.flipped ? SpriteEffects.FlipHorizontally : SpriteEffects.None, Math.Max(0f, (float)(this.TileLocation.Y * Game1.tileSize) / 10000f)); // Log.AsyncG("ANIMATION IS NULL?!?!?!?!"); } else { //Log.AsyncC("Animation Manager is working!"); int addedDepth = 0; if (this.info.ignoreBoundingBox) addedDepth++; this.animationManager.draw(spriteBatch, this.displayTexture, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)(x * Game1.tileSize), y * Game1.tileSize)), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * alpha, 0f, Vector2.Zero, (float)Game1.pixelZoom, this.flipped ? SpriteEffects.FlipHorizontally : SpriteEffects.None, Math.Max(0f, (float)((this.TileLocation.Y + addedDepth) * Game1.tileSize) / 10000f)); try { this.animationManager.tickAnimation(); // Log.AsyncC("Tick animation"); } catch (Exception err) { ModCore.ModMonitor.Log(err.ToString()); } } // spriteBatch.Draw(Game1.mouseCursors, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)((double)tileLocation.X * (double)Game1.tileSize + (((double)tileLocation.X * 11.0 + (double)tileLocation.Y * 7.0) % 10.0 - 5.0)) + (float)(Game1.tileSize / 2), (float)((double)tileLocation.Y * (double)Game1.tileSize + (((double)tileLocation.Y * 11.0 + (double)tileLocation.X * 7.0) % 10.0 - 5.0)) + (float)(Game1.tileSize / 2))), new Rectangle?(new Rectangle((int)((double)tileLocation.X * 51.0 + (double)tileLocation.Y * 77.0) % 3 * 16, 128 + this.whichForageCrop * 16, 16, 16)), Color.White, 0.0f, new Vector2(8f, 8f), (float)Game1.pixelZoom, SpriteEffects.None, (float)(((double)tileLocation.Y * (double)Game1.tileSize + (double)(Game1.tileSize / 2) + (((double)tileLocation.Y * 11.0 + (double)tileLocation.X * 7.0) % 10.0 - 5.0)) / 10000.0)); } } /// Draw the game object at a non-tile spot. Aka like debris. public override void draw(SpriteBatch spriteBatch, int xNonTile, int yNonTile, float layerDepth, float alpha = 1f) { if(this.info == null) { ModCore.log("Info is null:"); ModCore.log("Need request for guid: " + this.guid); MultiplayerUtilities.SendRequestForSpecificGUID(this.guid); return; } //The actual planter box being drawn. if (this.animationManager == null) { if (this.animationManager.getExtendedTexture() == null) ModCore.ModMonitor.Log("Tex Extended is null???"); spriteBatch.Draw(this.displayTexture, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)(xNonTile), yNonTile)), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * alpha, 0f, Vector2.Zero, (float)Game1.pixelZoom, this.Flipped ? SpriteEffects.FlipHorizontally : SpriteEffects.None, Math.Max(0f, layerDepth)); // Log.AsyncG("ANIMATION IS NULL?!?!?!?!"); } else { //Log.AsyncC("Animation Manager is working!"); int addedDepth = 0; if (this.info.ignoreBoundingBox) addedDepth++; this.animationManager.draw(spriteBatch, this.displayTexture, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)(xNonTile), yNonTile)), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * alpha, 0f, Vector2.Zero, (float)Game1.pixelZoom, this.Flipped ? SpriteEffects.FlipHorizontally : SpriteEffects.None, Math.Max(0f, layerDepth)); try { this.animationManager.tickAnimation(); // Log.AsyncC("Tick animation"); } catch (Exception err) { ModCore.ModMonitor.Log(err.ToString()); } } // spriteBatch.Draw(Game1.mouseCursors, Game1.GlobalToLocal(Game1.viewport, new Vector2((float)((double)tileLocation.X * (double)Game1.tileSize + (((double)tileLocation.X * 11.0 + (double)tileLocation.Y * 7.0) % 10.0 - 5.0)) + (float)(Game1.tileSize / 2), (float)((double)tileLocation.Y * (double)Game1.tileSize + (((double)tileLocation.Y * 11.0 + (double)tileLocation.X * 7.0) % 10.0 - 5.0)) + (float)(Game1.tileSize / 2))), new Rectangle?(new Rectangle((int)((double)tileLocation.X * 51.0 + (double)tileLocation.Y * 77.0) % 3 * 16, 128 + this.whichForageCrop * 16, 16, 16)), Color.White, 0.0f, new Vector2(8f, 8f), (float)Game1.pixelZoom, SpriteEffects.None, (float)(((double)tileLocation.Y * (double)Game1.tileSize + (double)(Game1.tileSize / 2) + (((double)tileLocation.Y * 11.0 + (double)tileLocation.X * 7.0) % 10.0 - 5.0)) / 10000.0)); } /// What happens when the object is drawn in a menu. public override void drawInMenu(SpriteBatch spriteBatch, Vector2 location, float scaleSize, float transparency, float layerDepth, bool drawStackNumber, Color c, bool drawShadow) { if (this.info == null) { ModCore.log("Info is null:"); ModCore.log("Need request for guid: " + this.guid); MultiplayerUtilities.SendRequestForSpecificGUID(this.guid); return; } if (drawStackNumber && this.maximumStackSize() > 1 && ((double)scaleSize > 0.3 && this.Stack != int.MaxValue) && this.Stack > 1) Utility.drawTinyDigits(this.Stack, spriteBatch, location + new Vector2((float)(Game1.tileSize - Utility.getWidthOfTinyDigitString(this.Stack, 3f * scaleSize)) + 3f * scaleSize, (float)((double)Game1.tileSize - 18.0 * (double)scaleSize + 2.0)), 3f * scaleSize, 1f, Color.White); if (drawStackNumber && this.Quality > 0) { float num = this.Quality < 4 ? 0.0f : (float)((Math.Cos((double)Game1.currentGameTime.TotalGameTime.Milliseconds * Math.PI / 512.0) + 1.0) * 0.0500000007450581); spriteBatch.Draw(Game1.mouseCursors, location + new Vector2(12f, (float)(Game1.tileSize - 12) + num), new Microsoft.Xna.Framework.Rectangle?(this.Quality < 4 ? new Microsoft.Xna.Framework.Rectangle(338 + (this.Quality - 1) * 8, 400, 8, 8) : new Microsoft.Xna.Framework.Rectangle(346, 392, 8, 8)), Color.White * transparency, 0.0f, new Vector2(4f, 4f), (float)(3.0 * (double)scaleSize * (1.0 + (double)num)), SpriteEffects.None, layerDepth); } spriteBatch.Draw(this.displayTexture, location + new Vector2((float)(Game1.tileSize / 2), (float)(Game1.tileSize)), new Rectangle?(this.animationManager.currentAnimation.sourceRectangle), this.info.drawColor * transparency, 0f, new Vector2((float)(this.animationManager.currentAnimation.sourceRectangle.Width / 2), (float)(this.animationManager.currentAnimation.sourceRectangle.Height)), scaleSize * 4f, SpriteEffects.None, layerDepth); } /// What happens when the object is drawn when held by a player. public override void drawWhenHeld(SpriteBatch spriteBatch, Vector2 objectPosition, StardewValley.Farmer f) { if (this.info == null) { ModCore.log("Info is null:"); ModCore.log("Need request for guid: " + this.guid); MultiplayerUtilities.SendRequestForSpecificGUID(this.guid); if (ModCore.CustomObjects.ContainsKey(this.guid)) { if (ModCore.CustomObjects[this.guid].info != null) { this.info = ModCore.CustomObjects[this.guid].info; } } else { foreach(var v in ModCore.CustomObjects) { if (v.Value == this) { this.info = v.Value.info; } } } return; } else { ModCore.log("Drawing custom object: " + this.guid); } if (this.animationManager == null) Revitalize.ModCore.log("Animation Manager Null"); if (this.displayTexture == null) Revitalize.ModCore.log("Display texture is null"); if (f.ActiveObject.bigCraftable.Value) { spriteBatch.Draw(this.displayTexture, objectPosition, this.animationManager.currentAnimation.sourceRectangle, this.info.drawColor, 0f, Vector2.Zero, (float)Game1.pixelZoom, SpriteEffects.None, Math.Max(0f, (float)(f.getStandingY() + 2) / 10000f)); return; } spriteBatch.Draw(this.displayTexture, objectPosition, this.animationManager.currentAnimation.sourceRectangle, this.info.drawColor, 0f, Vector2.Zero, (float)Game1.pixelZoom, SpriteEffects.None, Math.Max(0f, (float)(f.getStandingY() + 2) / 10000f)); if (f.ActiveObject != null && f.ActiveObject.Name.Contains("=")) { spriteBatch.Draw(this.displayTexture, objectPosition + new Vector2((float)(Game1.tileSize / 2), (float)(Game1.tileSize / 2)), this.animationManager.currentAnimation.sourceRectangle, Color.White, 0f, new Vector2((float)(Game1.tileSize / 2), (float)(Game1.tileSize / 2)), (float)Game1.pixelZoom + Math.Abs(Game1.starCropShimmerPause) / 8f, SpriteEffects.None, Math.Max(0f, (float)(f.getStandingY() + 2) / 10000f)); if (Math.Abs(Game1.starCropShimmerPause) <= 0.05f && Game1.random.NextDouble() < 0.97) { return; } Game1.starCropShimmerPause += 0.04f; if (Game1.starCropShimmerPause >= 0.8f) { Game1.starCropShimmerPause = -0.8f; } } //base.drawWhenHeld(spriteBatch, objectPosition, f); } public override void drawPlacementBounds(SpriteBatch spriteBatch, GameLocation location) { //Do nothing because this shouldn't be placeable anywhere. } /* public virtual bool InitNetFields() { if (Game1.IsMultiplayer == false) { //ModCore.log("Not multiplayer!"); return false; } if (MultiplayerUtilities.HasLoadedIn == false) { return false; } ModCore.log("Initialize net fields!"); //this.initNetFields(); this.syncObject = new PySync(this); ///this.NetFields.AddField(this.syncObject); //this.netItemInfo = new Netcode.NetString(this.ItemInfo); //this.NetFields.AddField(this.netItemInfo); //this.netInfo = new NetBasicItemInformation(this.info); //this.NetFields.AddField(this.netInfo); return true; } */ /// /// Syncs all of the info to all players. /// /// public virtual void replaceAfterLoad() { if (string.IsNullOrEmpty(this.info.locationName) == false) { //ModCore.log("Replace an object!"); this.location.removeObject(this.TileLocation, false); this.placementAction(this.location, (int)this.TileLocation.X * Game1.tileSize, (int)this.TileLocation.Y * Game1.tileSize); //ModCore.log("Do I ingnore BB? " + this.info.ignoreBoundingBox); //ModCore.log("This is my BB: " + this.boundingBox.Value); //this.InitNetFields(); } } public string getDisplayNameFromStringsFile(string objectID) { //Load in a file that has all object names referenced here or something. return this.info.name; } public object getReplacement() { return new StardewValley.Object(this.TileLocation, 14, false); } public Dictionary getAdditionalSaveData() { Dictionary saveData = new Dictionary(); saveData.Add("Greeting", "Hello"); saveData.Add("ItemInfo", ModCore.Serializer.ToJSONString(this.info)); return saveData; } public void rebuild(Dictionary additionalSaveData, object replacement) { this.DisplayName = additionalSaveData["Greeting"]; this.info =ModCore.Serializer.DeserializeFromJSONString(additionalSaveData["ItemInfo"]); this.initializeBasics(); ModCore.log("Rebuilt custom object and it's info from save!"); } } }