public static void GameLoop_SaveLoaded(object sender, StardewModdingAPI.Events.SaveLoadedEventArgs e) { Monitor.Log($"total apps: {ModEntry.apps.Count}"); PhoneUtils.OrderApps(); PhoneUtils.RefreshPhoneLayout(); Helper.Events.Display.RenderedWorld += PhoneVisuals.Display_RenderedWorld; if (ModEntry.npcAdventureModApi != null) { Monitor.Log("Testing NpcAdventureModApi..."); try { Monitor.Log($"Can recruit: {ModEntry.npcAdventureModApi.CanRecruitCompanions()}"); Monitor.Log($"Possible companions: {ModEntry.npcAdventureModApi.GetPossibleCompanions().Count()}"); Monitor.Log($"Can recruit Abigail: {ModEntry.npcAdventureModApi.IsPossibleCompanion("Abigail")}"); Monitor.Log($"Recruit Abigail: {ModEntry.npcAdventureModApi.IsPossibleCompanion("Abigail") && ModEntry.npcAdventureModApi.RecruitCompanion(Game1.player, Game1.getCharacterFromName("Abigail"))}"); } catch (Exception ex) { Monitor.Log($"Error testing NpcAdventureModApi: {ex}", LogLevel.Warn); } Monitor.Log("Testing NpcAdventureModApi finished"); } }
public static void GameLoop_GameLaunched(object sender, StardewModdingAPI.Events.GameLaunchedEventArgs e) { foreach (IContentPack contentPack in Helper.ContentPacks.GetOwned()) { Monitor.Log($"Reading content pack: {contentPack.Manifest.Name} {contentPack.Manifest.Version} from {contentPack.DirectoryPath}"); try { MobilePhonePackJSON json = contentPack.ReadJsonFile <MobilePhonePackJSON>("content.json") ?? null; if (json != null) { if (json.apps != null && json.apps.Any()) { foreach (AppJSON app in json.apps) { Texture2D tex = contentPack.LoadAsset <Texture2D>(app.iconPath); if (tex == null) { continue; } ModEntry.apps.Add(app.id, new MobileApp(app.name, app.keyPress, app.closePhone, tex)); Monitor.Log($"Added app {app.name} from {contentPack.DirectoryPath}"); } } else if (json.iconPath != null) { Texture2D icon = contentPack.LoadAsset <Texture2D>(json.iconPath); if (icon == null) { continue; } ModEntry.apps.Add(json.id, new MobileApp(json.name, json.keyPress, json.closePhone, icon)); Monitor.Log($"Added app {json.name} from {contentPack.DirectoryPath}"); } if (json.invites != null && json.invites.Any()) { foreach (EventInvite invite in json.invites) { MobilePhoneCall.eventInvites.Add(invite); Monitor.Log($"Added event invite {invite.name} from {contentPack.DirectoryPath}"); } } } } catch (Exception ex) { Monitor.Log($"error reading content.json file in content pack {contentPack.Manifest.Name}.\r\n{ex}", LogLevel.Error); } if (Directory.Exists(Path.Combine(contentPack.DirectoryPath, "assets", "events"))) { Monitor.Log($"Adding events"); string[] events = Directory.GetFiles(Path.Combine(contentPack.DirectoryPath, "assets", "events"), "*.json"); Monitor.Log($"CP has {events.Length} events"); foreach (string eventFile in events) { try { string eventPath = Path.Combine("assets", "events", Path.GetFileName(eventFile)); Monitor.Log($"Adding events {Path.GetFileName(eventFile)} from {contentPack.DirectoryPath}"); Reminiscence r = contentPack.ReadJsonFile <Reminiscence>(eventPath); var key = Path.GetFileName(eventFile).Replace(".json", ""); MobilePhoneCall.contentPackReminiscences.TryAdd(key, new Reminiscence()); MobilePhoneCall.contentPackReminiscences[key].events.AddRange(r.events); Monitor.Log($"Added event {Path.GetFileName(eventFile)} from {contentPack.DirectoryPath}"); } catch { } } } if (Directory.Exists(Path.Combine(contentPack.DirectoryPath, "assets", "skins"))) { Monitor.Log($"Adding skins"); string[] skins = Directory.GetFiles(Path.Combine(contentPack.DirectoryPath, "assets", "skins"), "*_landscape.png"); Monitor.Log($"CP has {skins.Length} skins"); foreach (string skinFile in skins) { try { string skinPath = Path.Combine("assets", "skins", Path.GetFileName(skinFile)); Monitor.Log($"Adding skin {Path.GetFileName(skinFile).Replace("_landscape.png", "")} from {contentPack.DirectoryPath}"); Texture2D skin = contentPack.LoadAsset <Texture2D>(skinPath.Replace("_landscape.png", ".png")); Texture2D skinl = contentPack.LoadAsset <Texture2D>(skinPath); ThemeApp.skinList.Add(contentPack.Manifest.UniqueID + ":" + Path.GetFileName(skinFile).Replace("_landscape.png", "")); ThemeApp.skinDict.Add(contentPack.Manifest.UniqueID + ":" + Path.GetFileName(skinFile).Replace("_landscape.png", ""), new Texture2D[] { skin, skinl }); Monitor.Log($"Added skin {Path.GetFileName(skinFile).Replace("_landscape.png", "")} from {contentPack.DirectoryPath}"); } catch { } } } if (Directory.Exists(Path.Combine(contentPack.DirectoryPath, "assets", "backgrounds"))) { Monitor.Log($"Adding backgrounds"); string[] backgrounds = Directory.GetFiles(Path.Combine(contentPack.DirectoryPath, "assets", "backgrounds"), "*_landscape.png"); Monitor.Log($"CP has {backgrounds.Length} backgrounds"); foreach (string backFile in backgrounds) { try { string backPath = Path.Combine("assets", "backgrounds", Path.GetFileName(backFile)); Monitor.Log($"Adding background {Path.GetFileName(backFile).Replace("_landscape.png", "")} from {contentPack.DirectoryPath}"); Texture2D back = contentPack.LoadAsset <Texture2D>(backPath.Replace("_landscape.png", ".png")); Texture2D backl = contentPack.LoadAsset <Texture2D>(backPath); ThemeApp.backgroundDict.Add(contentPack.Manifest.UniqueID + ":" + Path.GetFileName(backFile).Replace("_landscape.png", ""), new Texture2D[] { back, backl }); ThemeApp.backgroundList.Add(contentPack.Manifest.UniqueID + ":" + Path.GetFileName(backFile).Replace("_landscape.png", "")); Monitor.Log($"Added background {Path.GetFileName(backFile).Replace("_landscape.png", "")} from {contentPack.DirectoryPath}"); } catch { } } } if (Directory.Exists(Path.Combine(contentPack.DirectoryPath, "assets", "ringtones"))) { Monitor.Log($"Adding ringtones"); string[] rings = Directory.GetFiles(Path.Combine(contentPack.DirectoryPath, "assets", "ringtones"), "*.wav"); Monitor.Log($"CP has {rings.Length} ringtones"); foreach (string path in rings) { try { object ring; try { var type = Type.GetType("System.Media.SoundPlayer, System"); ring = Activator.CreateInstance(type, new object[] { path }); } catch { ring = SoundEffect.FromStream(new FileStream(path, FileMode.Open)); } if (ring != null) { ThemeApp.ringDict.Add(string.Concat(contentPack.Manifest.UniqueID, ":", Path.GetFileName(path).Replace(".wav", "")), ring); ThemeApp.ringList.Add(string.Concat(contentPack.Manifest.UniqueID, ":", Path.GetFileName(path).Replace(".wav", ""))); Monitor.Log($"loaded ring {path}"); } else { Monitor.Log($"Couldn't load ring {path}"); } } catch (Exception ex) { Monitor.Log($"Couldn't load ring {path}:\r\n{ex}", LogLevel.Error); } } } } ModEntry.listHeight = Config.IconMarginY + (int)Math.Ceiling(ModEntry.apps.Count / (float)ModEntry.gridWidth) * (Config.IconHeight + Config.IconMarginY); PhoneVisuals.CreatePhoneTextures(); PhoneUtils.RefreshPhoneLayout(); if (Helper.ModRegistry.IsLoaded("purrplingcat.npcadventure")) { INpcAdventureModApi api = Helper.ModRegistry.GetApi <INpcAdventureModApi>("purrplingcat.npcadventure"); if (api != null) { Monitor.Log("Loaded NpcAdventureModApi successfully"); ModEntry.npcAdventureModApi = api; } } if (Helper.ModRegistry.IsLoaded("tlitookilakin.HDPortraits")) { IHDPortraitsAPI api = Helper.ModRegistry.GetApi <IHDPortraitsAPI>("tlitookilakin.HDPortraits"); if (api != null) { Monitor.Log("Loaded HD Portraits api successfully"); ModEntry.iHDPortraitsAPI = api; } } }
public static void Display_WindowResized(object sender, StardewModdingAPI.Events.WindowResizedEventArgs e) { PhoneUtils.RefreshPhoneLayout(); }
public static void Display_RenderedWorld(object sender, StardewModdingAPI.Events.RenderedWorldEventArgs e) { Point mousePos = Game1.getMousePosition(); Point lastMousePos = ModEntry.lastMousePosition; ModEntry.lastMousePosition = mousePos; if (!ModEntry.phoneOpen) { ModEntry.appRunning = false; ModEntry.runningApp = null; if (Config.ShowPhoneIcon && Game1.displayHUD && !Game1.eventUp && Game1.currentBillboard == 0 && Game1.gameMode == 3 && !Game1.freezeControls && !Game1.panMode && !Game1.HostPaused && !Game1.game1.takingMapScreenshot) { if (ModEntry.clickingPhoneIcon) { if (Helper.Input.IsSuppressed(SButton.MouseLeft) && lastMousePos != mousePos) { ModEntry.draggingPhoneIcon = true; Config.PhoneIconOffsetX += mousePos.X - lastMousePos.X; Config.PhoneIconOffsetY += mousePos.Y - lastMousePos.Y; ModEntry.phoneIconPosition = PhoneUtils.GetPhoneIconPosition(); } else if (!Helper.Input.IsSuppressed(SButton.MouseLeft)) { if (!ModEntry.draggingPhoneIcon) { PhoneUtils.TogglePhone(true); } else { Helper.WriteConfig(Config); } ModEntry.clickingPhoneIcon = false; ModEntry.draggingPhoneIcon = false; } } e.SpriteBatch.Draw(ModEntry.iconTexture, ModEntry.phoneIconPosition, Color.White); } return; } else { ModEntry.clickingPhoneIcon = false; ModEntry.draggingPhoneIcon = false; } if (Game1.game1.takingMapScreenshot) { return; } if (ModEntry.draggingPhone) { if (Helper.Input.IsSuppressed(SButton.MouseLeft)) { if (mousePos != lastMousePos) { int x = mousePos.X - lastMousePos.X; int y = mousePos.Y - lastMousePos.Y; if (ModEntry.phoneRotated) { Config.PhoneRotatedOffsetX += x; Config.PhoneRotatedOffsetY += y; } else { Config.PhoneOffsetX += x; Config.PhoneOffsetY += y; } PhoneUtils.RefreshPhoneLayout(); } } else { ModEntry.context.Helper.WriteConfig(Config); ModEntry.draggingPhone = false; Monitor.Log($"released dragging phone"); } } else if (Helper.Input.IsSuppressed(SButton.MouseLeft) && !ModEntry.movingAppIcon) { int dy = mousePos.Y - lastMousePos.Y; if (Math.Abs(dy) > 0 && ModEntry.screenRect.Contains(mousePos)) { ModEntry.clickingApp = -1; ModEntry.draggingIcons = true; } if (ModEntry.draggingIcons) { ModEntry.yOffset = (int)Math.Max(Math.Min(0, ModEntry.yOffset + dy), -1 * Math.Max(0, ModEntry.listHeight - PhoneUtils.GetScreenSize().Y)); } } e.SpriteBatch.Draw(ModEntry.phoneRotated ? ModEntry.backgroundRotatedTexture : ModEntry.backgroundTexture, ModEntry.phoneRect, Color.White); e.SpriteBatch.Draw(ModEntry.phoneRotated ? ModEntry.phoneRotatedTexture : ModEntry.phoneTexture, ModEntry.phoneRect, Color.White); Vector2 screenPos = PhoneUtils.GetScreenPosition(); Vector2 screenSize = PhoneUtils.GetScreenSize(); if (ModEntry.callingNPC != null) { Rectangle destRect; Rectangle portraitSource = Game1.getSourceRectForStandardTileSheet(ModEntry.callingNPC.Portrait, 0, 64, 64); float scale; if ((float)portraitSource.Height * screenSize.X / (float)portraitSource.Width > screenSize.Y - Config.AppHeaderHeight) { scale = (screenSize.Y - Config.AppHeaderHeight) / (float)portraitSource.Height; destRect = new Rectangle((int)(screenPos.X + screenSize.X / 2f - portraitSource.Width * scale / 2f), (int)screenPos.Y, (int)(portraitSource.Width * scale), (int)screenSize.Y - Config.AppHeaderHeight); } else { scale = screenSize.X / (float)portraitSource.Width; //Monitor.Log($"{screenSize} {portraitSource.Width},{portraitSource.Height} {scale}"); destRect = new Rectangle((int)(screenPos.X), (int)(screenPos.Y + (screenSize.Y - Config.AppHeaderHeight) / 2f - portraitSource.Height * scale / 2f), (int)screenSize.X, (int)(portraitSource.Height * scale)); } e.SpriteBatch.Draw(ModEntry.callingNPC.Portrait, destRect, new Rectangle?(portraitSource), Color.White); SpriteText.drawStringHorizontallyCenteredAt(e.SpriteBatch, ModEntry.callingNPC.getName(), destRect.X + destRect.Width / 2, destRect.Bottom + 16, 999999, -1, 999999, 1f, 0.88f, false, -1, 99999); if (!ModEntry.inCall) { Rectangle answerRect = new Rectangle((int)(screenPos.X), ModEntry.screenRect.Bottom - Config.AppHeaderHeight, (int)(screenSize.X / 2f), Config.AppHeaderHeight); Rectangle declineRect = new Rectangle((int)(screenPos.X + screenSize.X / 2f), ModEntry.screenRect.Bottom - Config.AppHeaderHeight, (int)(screenSize.X / 2f), Config.AppHeaderHeight); e.SpriteBatch.Draw(ModEntry.answerTexture, answerRect, Color.White); e.SpriteBatch.Draw(ModEntry.declineTexture, declineRect, Color.White); float textScale = Config.CallTextScale; string ans = Helper.Translation.Get("answer"); Vector2 ansSize = Game1.dialogueFont.MeasureString(ans) * textScale; string dec = Helper.Translation.Get("decline"); Vector2 decSize = Game1.dialogueFont.MeasureString(dec) * textScale; e.SpriteBatch.DrawString(Game1.dialogueFont, ans, new Vector2(answerRect.X + answerRect.Width / 2f - ansSize.X / 2f, answerRect.Top + answerRect.Height / 2f - ansSize.Y / 2f), Config.CallTextColor, 0f, Vector2.Zero, textScale, SpriteEffects.None, 1f); e.SpriteBatch.DrawString(Game1.dialogueFont, dec, new Vector2(declineRect.X + declineRect.Width / 2f - decSize.X / 2f, declineRect.Top + declineRect.Height / 2f - decSize.Y / 2f), Config.CallTextColor, 0f, Vector2.Zero, textScale, SpriteEffects.None, 1f); if (ModEntry.clicking && !Helper.Input.IsSuppressed(SButton.MouseLeft)) { if (answerRect.Contains(mousePos)) { MobilePhoneApp.CallNPC(ModEntry.callingNPC); ModEntry.currentCallRings = 0; } else if (declineRect.Contains(mousePos)) { ModEntry.currentCallRings = 0; ModEntry.callingNPC = null; } } } else { Rectangle endRect = new Rectangle((int)(screenPos.X + screenSize.X / 4), ModEntry.screenRect.Bottom - Config.AppHeaderHeight, (int)(screenSize.X / 2), Config.AppHeaderHeight); e.SpriteBatch.Draw(ModEntry.declineTexture, endRect, Color.White); float textScale = Config.CallTextScale; string ends = Helper.Translation.Get("end-call"); Vector2 endsSize = Game1.dialogueFont.MeasureString(ends) * textScale; e.SpriteBatch.DrawString(Game1.dialogueFont, ends, new Vector2(endRect.X + endRect.Width / 2f - endsSize.X / 2f, endRect.Top + endRect.Height / 2f - endsSize.Y / 2f), Config.CallTextColor, 0f, Vector2.Zero, textScale, SpriteEffects.None, 1f); if (ModEntry.clicking && !Helper.Input.IsSuppressed(SButton.MouseLeft)) { if (endRect.Contains(mousePos)) { MobilePhoneCall.EndCall(); if (Game1.activeClickableMenu is DialogueBox) { Game1.activeClickableMenu = null; } } } } if ((ModEntry.clicking || ModEntry.clickingPhoneIcon) && !Helper.Input.IsSuppressed(SButton.MouseLeft)) { ModEntry.clickingApp = -1; ModEntry.switchingApp = -1; ModEntry.movingAppIconOffset = new Point(0, 0); ModEntry.clickingTicks = 0; ModEntry.clicking = false; ModEntry.movingAppIcon = false; ModEntry.clickingPhoneIcon = false; ModEntry.draggingPhone = false; ModEntry.draggingIcons = false; } return; } if (ModEntry.appRunning) { return; } if (ModEntry.runningApp == Helper.ModRegistry.ModID && Game1.activeClickableMenu == null) { MobilePhoneApp.OpenPhoneBook(); return; } List <string> keys = new List <string>(ModEntry.appOrder); if ((ModEntry.clicking || ModEntry.clickingPhoneIcon) && !Helper.Input.IsSuppressed(SButton.MouseLeft)) { Monitor.Log($"released click"); if (ModEntry.movingAppIcon && ModEntry.switchingApp != -1) { Monitor.Log($"switching app: {ModEntry.switchingApp} clicking app {ModEntry.clickingApp}"); Game1.playSound("stoneStep"); ModEntry.appOrder[ModEntry.clickingApp] = keys[ModEntry.switchingApp]; ModEntry.appOrder[ModEntry.switchingApp] = keys[ModEntry.clickingApp]; keys = new List <string>(ModEntry.appOrder); Config.AppList = keys.ToArray(); Helper.WriteConfig(Config); } else if (ModEntry.draggingIcons) { ModEntry.draggingIcons = false; } else if (!ModEntry.movingAppIcon) { for (int i = 0; i < keys.Count; i++) { Vector2 pos = PhoneUtils.GetAppPos(i); Rectangle r = new Rectangle((int)pos.X, (int)pos.Y, Config.IconWidth, Config.IconHeight); if (r.Contains(mousePos)) { if (ModEntry.apps[keys[i]].keyPress != null) { Monitor.Log($"pressing key {ModEntry.apps[keys[i]].keyPress}"); PhoneInput.PressKey(ModEntry.apps[keys[i]]); } else { Monitor.Log($"starting app {ModEntry.apps[keys[i]].name}"); ModEntry.apps[keys[i]].action.Invoke(); } } } } ModEntry.clickingApp = -1; ModEntry.switchingApp = -1; ModEntry.movingAppIconOffset = new Point(0, 0); ModEntry.clickingTicks = 0; ModEntry.clicking = false; ModEntry.movingAppIcon = false; ModEntry.clickingPhoneIcon = false; ModEntry.draggingPhone = false; ModEntry.draggingIcons = false; } else if (ModEntry.clicking && ModEntry.clickingApp != -1 && !ModEntry.movingAppIcon) { if (lastMousePos == mousePos) { if (ModEntry.clickingTicks > Config.TicksToMoveAppIcon) { Game1.playSound("pickUpItem"); ModEntry.movingAppIconOffset = new Point(5, 5); ModEntry.movingAppIcon = true; } else { ModEntry.clickingTicks++; } } else { ModEntry.clickingTicks = 0; ModEntry.clickingApp = -1; } } else if (ModEntry.movingAppIcon) { ModEntry.movingAppIconOffset = new Point(ModEntry.movingAppIconOffset.X + mousePos.X - lastMousePos.X, ModEntry.movingAppIconOffset.Y + mousePos.Y - lastMousePos.Y); Vector2 currentPos = PhoneUtils.GetAppPos(ModEntry.clickingApp); if (ModEntry.screenRect.Contains(Utility.Vector2ToPoint(currentPos))) { bool alreadySwitched = false; if (ModEntry.switchingApp != -1) { Vector2 pos = PhoneUtils.GetAppPos(ModEntry.switchingApp, true); if (Vector2.Distance(currentPos, pos) < (Config.IconWidth + Config.IconHeight) / 4f) { alreadySwitched = true; } } if (!alreadySwitched) { for (int i = 0; i < keys.Count; i++) { Vector2 pos = PhoneUtils.GetAppPos(i); if (i != ModEntry.clickingApp && Vector2.Distance(currentPos, pos) < (Config.IconWidth + Config.IconHeight) / 4f) { ModEntry.switchingApp = i; Monitor.Log($"new switching app: {ModEntry.switchingApp} clicking app {ModEntry.clickingApp}"); break; } else { ModEntry.switchingApp = -1; } } } } } string appHover = null; bool hover = false; if (mousePos == lastMousePos && !Helper.Input.IsSuppressed(SButton.MouseLeft)) { hover = true; } else { ModEntry.ticksSinceMoved = 0; } int screenBottom = (int)(screenPos.Y + screenSize.Y); for (int i = keys.Count - 1; i >= 0; i--) { MobileApp app = ModEntry.apps[keys[i]]; Vector2 appPos = PhoneUtils.GetAppPos(i); Rectangle r = new Rectangle(0, 0, Config.IconWidth, Config.IconHeight); Rectangle sourceRect = r; if (!ModEntry.movingAppIcon || i != ModEntry.clickingApp) { if (appPos.Y < screenPos.Y - r.Height * 2 || appPos.Y >= screenBottom) { continue; } if (appPos.Y < screenPos.Y) { int cutTop = (int)screenPos.Y - (int)appPos.Y; sourceRect = new Rectangle(r.X, r.Y + cutTop, r.Width, r.Height - cutTop); appPos = new Vector2(appPos.X, screenPos.Y); } else if (appPos.Y > screenBottom - r.Height) { int cutBottom = screenBottom - r.Height - (int)appPos.Y; sourceRect = new Rectangle(r.X, r.Y, r.Width, r.Height + cutBottom); } } e.SpriteBatch.Draw(app.icon, new Rectangle((int)appPos.X, (int)appPos.Y, sourceRect.Width, sourceRect.Height), sourceRect, Color.White, 0, Vector2.Zero, SpriteEffects.None, i == ModEntry.clickingApp && ModEntry.movingAppIcon ? 1f : 0.5f); Rectangle rect = new Rectangle((int)appPos.X, (int)appPos.Y, Config.IconWidth, sourceRect.Height); if (hover && !Helper.Input.IsSuppressed(SButton.MouseLeft) && rect.Contains(mousePos)) { ModEntry.ticksSinceMoved++; if (ModEntry.ticksSinceMoved > Config.ToolTipDelayTicks) { appHover = app.name; } } } if (appHover != null) { e.SpriteBatch.DrawString(Game1.dialogueFont, appHover, new Vector2(mousePos.X, mousePos.Y) - Game1.dialogueFont.MeasureString(appHover) + new Vector2(-2, 2), Color.Black); e.SpriteBatch.DrawString(Game1.dialogueFont, appHover, new Vector2(mousePos.X, mousePos.Y) - Game1.dialogueFont.MeasureString(appHover), Color.White); } }