Stardew_Valley_Mods/GeneralMods/Revitalize/Framework/Objects/CustomObject.cs

599 lines
27 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using Microsoft.Xna.Framework;
using Microsoft.Xna.Framework.Graphics;
using Microsoft.Xna.Framework.Input;
2019-01-10 18:31:23 +08:00
using Newtonsoft.Json;
using PyTK.CustomElementHandler;
using StardustCore.Animations;
using StardewValley;
using StardewValley.Objects;
2019-08-24 05:46:07 +08:00
using Netcode;
using StardewModdingAPI;
using Revitalize.Framework.Utilities;
namespace Revitalize.Framework.Objects
{
// TODO:
// -Multiple Lights
// -Events when walking over?
// -Inventories
/// <summary>A custom object template.</summary>
public class CustomObject : StardewValley.Object,ISaveElement
{
public string id
{
get
{
return this.info.id;
}
}
2019-01-10 18:31:23 +08:00
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;
/// <summary>The animation manager.</summary>
public AnimationManager animationManager => this.info.animationManager;
/// <summary>The display texture for this object.</summary>
2019-01-10 18:31:23 +08:00
[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));
}
}
2019-01-10 18:31:23 +08:00
protected Netcode.NetString netItemInfo;
2019-01-10 18:31:23 +08:00
/// <summary>Empty constructor.</summary>
public CustomObject()
{
this.guid = Guid.NewGuid();
//this.InitNetFields();
}
/// <summary>Construct an instance.</summary>
public CustomObject(BasicItemInformation info, int Stack = 1)
: base()
{
this.info = info;
this.initializeBasics();
this.guid = Guid.NewGuid();
this.Stack = Stack;
}
/// <summary>Construct an instance.</summary>
public CustomObject(BasicItemInformation info, Vector2 TileLocation, int Stack = 1)
: base()
{
this.info = info;
this.initializeBasics();
this.guid = Guid.NewGuid();
this.Stack = Stack;
}
/// <summary>Sets some basic information up.</summary>
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);
}
/// <summary>Checks for interaction with the object.</summary>
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);
2019-05-16 13:27:38 +08:00
return base.checkForAction(who, justCheckingForActivity);
if (justCheckingForActivity)
return true;
ModCore.log("Left clicked!");
return this.clicked(who);
}
/*
public override ICustomObject recreate(Dictionary<string, string> additionalSaveData, object replacement)
{
CustomObjectData data = CustomObjectData.collection[additionalSaveData["id"]];
BasicItemInformation info = Revitalize.ModCore.Serializer.DeserializeFromJSONString<BasicItemInformation>(additionalSaveData["ItemInfo"]);
return new CustomObject(data, info);
}
*/
/*
public override Dictionary<string, string> getAdditionalSaveData()
{
Dictionary<string, string> serializedInfo = new Dictionary<string, string>();
serializedInfo.Add("id", this.ItemInfo);
serializedInfo.Add("ItemInfo", Revitalize.ModCore.Serializer.ToJSONString(this.info));
return serializedInfo;
}
*/
/*
public override void rebuild(Dictionary<string, string> additionalSaveData, object replacement)
{
string id = additionalSaveData["id"];
this.ItemInfo = id;
this.info = Revitalize.ModCore.Serializer.DeserializeFromJSONString<BasicItemInformation>(additionalSaveData["ItemInfo"]);
base.rebuild(additionalSaveData, replacement);
ModCore.log("Rebuilt custom object.");
}
*/
/// <summary>What happens when the player right clicks the object.</summary>
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;
}
/// <summary>What happens when the player shift-right clicks this object.</summary>
public virtual bool shiftRightClicked(Farmer who)
{
//ModCore.log("Shift right clicked!");
return true;
}
/// <summary>What happens when the player left clicks the object.</summary>
public override bool clicked(Farmer who)
{
//ModCore.log("Clicky click!");
return this.removeAndAddToPlayersInventory();
//return base.clicked(who);
}
/// <summary>What happens when a player uses a tool on this object.</summary>
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);
}
/// <summary>Remove the object from the world and add it to the player's inventory if possible.</summary>
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;
}
/// <summary>Gets the category color for the object.</summary>
public override Color getCategoryColor()
{
return this.info.categoryColor;
//return data.categoryColor;
}
/// <summary>Gets the category name for the object.</summary>
public override string getCategoryName()
{
return this.info.categoryName;
}
/// <summary>Gets the description for the object.</summary>
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);
}
/// <summary>Places an object down.</summary>
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);
}
/// <summary>Updates a visual draw position.</summary>
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)));
}
/// <summary>Gets a clone of the game object.</summary>
public override Item getOne()
{
return new CustomObject(this.info);
}
/// <summary>What happens when the object is drawn at a tile location.</summary>
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));
}
}
/// <summary>Draw the game object at a non-tile spot. Aka like debris.</summary>
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));
}
/// <summary>What happens when the object is drawn in a menu.</summary>
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);
}
/// <summary>What happens when the object is drawn when held by a player.</summary>
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;
}
*/
/// <summary>
/// Syncs all of the info to all players.
/// </summary>
/// <param name="syncData"></param>
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;
}
2019-08-24 05:46:07 +08:00
public object getReplacement()
{
return new StardewValley.Object(this.TileLocation, 14, false);
}
public Dictionary<string, string> getAdditionalSaveData()
{
Dictionary<string, string> saveData = new Dictionary<string, string>();
saveData.Add("Greeting", "Hello");
saveData.Add("ItemInfo", ModCore.Serializer.ToJSONString(this.info));
return saveData;
}
2019-08-24 05:46:07 +08:00
public void rebuild(Dictionary<string, string> additionalSaveData, object replacement)
2019-08-24 05:46:07 +08:00
{
this.DisplayName = additionalSaveData["Greeting"];
this.info =ModCore.Serializer.DeserializeFromJSONString<BasicItemInformation>(additionalSaveData["ItemInfo"]);
this.initializeBasics();
ModCore.log("Rebuilt custom object and it's info from save!");
2019-08-24 05:46:07 +08:00
}
}
}