804 lines
46 KiB
C#
804 lines
46 KiB
C#
using Microsoft.Xna.Framework;
|
|
using Microsoft.Xna.Framework.Graphics;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Immutable;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
|
|
namespace Barotrauma
|
|
{
|
|
class RoundSummary
|
|
{
|
|
private const float jobColumnWidthPercentage = 0.11f;
|
|
private const float characterColumnWidthPercentage = 0.44f;
|
|
private const float statusColumnWidthPercentage = 0.45f;
|
|
|
|
private int jobColumnWidth, characterColumnWidth, statusColumnWidth;
|
|
|
|
private readonly List<Mission> selectedMissions;
|
|
private readonly Location startLocation, endLocation;
|
|
|
|
private readonly GameMode gameMode;
|
|
|
|
private readonly Dictionary<Identifier, float> initialFactionReputations = new Dictionary<Identifier, float>();
|
|
|
|
public GUILayoutGroup ButtonArea { get; private set; }
|
|
|
|
public GUIButton ContinueButton { get; private set; }
|
|
|
|
public GUIComponent Frame { get; private set; }
|
|
|
|
public RoundSummary(GameMode gameMode, IEnumerable<Mission> selectedMissions, Location startLocation, Location endLocation)
|
|
{
|
|
this.gameMode = gameMode;
|
|
this.selectedMissions = selectedMissions.ToList();
|
|
this.startLocation = startLocation;
|
|
this.endLocation = endLocation;
|
|
if (gameMode is CampaignMode campaignMode)
|
|
{
|
|
foreach (Faction faction in campaignMode.Factions)
|
|
{
|
|
initialFactionReputations.Add(faction.Prefab.Identifier, faction.Reputation.Value);
|
|
}
|
|
}
|
|
}
|
|
|
|
public GUIFrame CreateSummaryFrame(GameSession gameSession, string endMessage, List<TraitorMissionResult> traitorResults, CampaignMode.TransitionType transitionType = CampaignMode.TransitionType.None)
|
|
{
|
|
bool singleplayer = GameMain.NetworkMember == null;
|
|
bool gameOver =
|
|
gameSession.GameMode.IsSinglePlayer ?
|
|
gameSession.CrewManager.GetCharacters().All(c => c.IsDead || c.IsIncapacitated) :
|
|
gameSession.CrewManager.GetCharacters().All(c => c.IsDead || c.IsIncapacitated || c.IsBot);
|
|
|
|
if (!singleplayer)
|
|
{
|
|
SoundPlayer.OverrideMusicType = (gameOver ? "crewdead" : "endround").ToIdentifier();
|
|
SoundPlayer.OverrideMusicDuration = 18.0f;
|
|
}
|
|
|
|
GUIFrame background = new GUIFrame(new RectTransform(GUI.Canvas.RelativeSize, GUI.Canvas, Anchor.Center), style: "GUIBackgroundBlocker")
|
|
{
|
|
UserData = this
|
|
};
|
|
|
|
List<GUIComponent> rightPanels = new List<GUIComponent>();
|
|
|
|
int minWidth = 400, minHeight = 350;
|
|
int padding = GUI.IntScale(25.0f);
|
|
|
|
//crew panel -------------------------------------------------------------------------------
|
|
|
|
GUIFrame crewFrame = new GUIFrame(new RectTransform(new Vector2(0.35f, 0.45f), background.RectTransform, Anchor.TopCenter, minSize: new Point(minWidth, minHeight)));
|
|
GUIFrame crewFrameInner = new GUIFrame(new RectTransform(new Point(crewFrame.Rect.Width - padding * 2, crewFrame.Rect.Height - padding * 2), crewFrame.RectTransform, Anchor.Center), style: "InnerFrame");
|
|
|
|
var crewContent = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.95f), crewFrameInner.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true
|
|
};
|
|
|
|
var crewHeader = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), crewContent.RectTransform),
|
|
TextManager.Get("crew"), textAlignment: Alignment.TopLeft, font: GUIStyle.SubHeadingFont);
|
|
crewHeader.RectTransform.MinSize = new Point(0, GUI.IntScale(crewHeader.Rect.Height * 2.0f));
|
|
|
|
CreateCrewList(crewContent, gameSession.CrewManager.GetCharacterInfos().Where(c => c.TeamID != CharacterTeamType.Team2));
|
|
|
|
//another crew frame for the 2nd team in combat missions
|
|
if (gameSession.Missions.Any(m => m is CombatMission))
|
|
{
|
|
crewHeader.Text = CombatMission.GetTeamName(CharacterTeamType.Team1);
|
|
GUIFrame crewFrame2 = new GUIFrame(new RectTransform(new Vector2(0.35f, 0.45f), background.RectTransform, Anchor.TopCenter, minSize: new Point(minWidth, minHeight)));
|
|
rightPanels.Add(crewFrame2);
|
|
GUIFrame crewFrameInner2 = new GUIFrame(new RectTransform(new Point(crewFrame2.Rect.Width - padding * 2, crewFrame2.Rect.Height - padding * 2), crewFrame2.RectTransform, Anchor.Center), style: "InnerFrame");
|
|
var crewContent2 = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.95f), crewFrameInner2.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true
|
|
};
|
|
var crewHeader2 = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), crewContent2.RectTransform),
|
|
CombatMission.GetTeamName(CharacterTeamType.Team2), textAlignment: Alignment.TopLeft, font: GUIStyle.SubHeadingFont);
|
|
crewHeader2.RectTransform.MinSize = new Point(0, GUI.IntScale(crewHeader2.Rect.Height * 2.0f));
|
|
CreateCrewList(crewContent2, gameSession.CrewManager.GetCharacterInfos().Where(c => c.TeamID == CharacterTeamType.Team2));
|
|
}
|
|
|
|
//header -------------------------------------------------------------------------------
|
|
|
|
LocalizedString headerText = GetHeaderText(gameOver, transitionType);
|
|
GUITextBlock headerTextBlock = null;
|
|
if (!headerText.IsNullOrEmpty())
|
|
{
|
|
headerTextBlock = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.5f), crewFrame.RectTransform, Anchor.TopLeft, Pivot.BottomLeft),
|
|
headerText, textAlignment: Alignment.BottomLeft, font: GUIStyle.LargeFont, wrap: true);
|
|
}
|
|
|
|
//traitor panel -------------------------------------------------------------------------------
|
|
|
|
if (traitorResults != null && traitorResults.Any())
|
|
{
|
|
GUIFrame traitorframe = new GUIFrame(new RectTransform(crewFrame.RectTransform.RelativeSize, background.RectTransform, Anchor.TopCenter, minSize: crewFrame.RectTransform.MinSize));
|
|
rightPanels.Add(traitorframe);
|
|
GUIFrame traitorframeInner = new GUIFrame(new RectTransform(new Point(traitorframe.Rect.Width - padding * 2, traitorframe.Rect.Height - padding * 2), traitorframe.RectTransform, Anchor.Center), style: "InnerFrame");
|
|
|
|
var traitorContent = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.95f), traitorframeInner.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true
|
|
};
|
|
|
|
var traitorHeader = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), traitorContent.RectTransform),
|
|
TextManager.Get("traitors"), font: GUIStyle.SubHeadingFont);
|
|
traitorHeader.RectTransform.MinSize = new Point(0, GUI.IntScale(traitorHeader.Rect.Height * 2.0f));
|
|
|
|
GUIListBox listBox = CreateCrewList(traitorContent, traitorResults.SelectMany(tr => tr.Characters.Select(c => c.Info)));
|
|
|
|
foreach (var traitorResult in traitorResults)
|
|
{
|
|
var traitorMission = TraitorMissionPrefab.Prefabs.Find(t => t.Identifier == traitorResult.MissionIdentifier);
|
|
if (traitorMission == null) { continue; }
|
|
|
|
//spacing
|
|
new GUIFrame(new RectTransform(new Point(listBox.Content.Rect.Width, GUI.IntScale(25)), listBox.Content.RectTransform), style: null);
|
|
|
|
var traitorResultHorizontal = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.3f), listBox.Content.RectTransform), childAnchor: Anchor.CenterLeft, isHorizontal: true)
|
|
{
|
|
RelativeSpacing = 0.05f,
|
|
Stretch = true
|
|
};
|
|
|
|
new GUIImage(new RectTransform(new Point(traitorResultHorizontal.Rect.Height), traitorResultHorizontal.RectTransform), traitorMission.Icon, scaleToFit: true)
|
|
{
|
|
Color = traitorMission.IconColor
|
|
};
|
|
|
|
LocalizedString traitorMessage = TextManager.GetServerMessage(traitorResult.EndMessage);
|
|
if (!traitorMessage.IsNullOrEmpty())
|
|
{
|
|
var textContent = new GUILayoutGroup(new RectTransform(Vector2.One, traitorResultHorizontal.RectTransform))
|
|
{
|
|
RelativeSpacing = 0.025f
|
|
};
|
|
|
|
var traitorStatusText = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), textContent.RectTransform),
|
|
TextManager.Get(traitorResult.Success ? "missioncompleted" : "missionfailed"),
|
|
textColor: traitorResult.Success ? GUIStyle.Green : GUIStyle.Red, font: GUIStyle.SubHeadingFont);
|
|
|
|
var traitorMissionInfo = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), textContent.RectTransform),
|
|
traitorMessage, font: GUIStyle.SmallFont, wrap: true);
|
|
|
|
traitorResultHorizontal.Recalculate();
|
|
|
|
traitorStatusText.CalculateHeightFromText();
|
|
traitorMissionInfo.CalculateHeightFromText();
|
|
traitorStatusText.RectTransform.MinSize = new Point(0, traitorStatusText.Rect.Height);
|
|
traitorMissionInfo.RectTransform.MinSize = new Point(0, traitorMissionInfo.Rect.Height);
|
|
textContent.RectTransform.MaxSize = new Point(int.MaxValue, (int)((traitorStatusText.Rect.Height + traitorMissionInfo.Rect.Height) * 1.2f));
|
|
traitorResultHorizontal.RectTransform.MinSize = new Point(0, traitorStatusText.RectTransform.MinSize.Y + traitorMissionInfo.RectTransform.MinSize.Y);
|
|
}
|
|
}
|
|
}
|
|
|
|
//reputation panel -------------------------------------------------------------------------------
|
|
|
|
var campaignMode = gameMode as CampaignMode;
|
|
if (campaignMode != null)
|
|
{
|
|
GUIFrame reputationframe = new GUIFrame(new RectTransform(crewFrame.RectTransform.RelativeSize, background.RectTransform, Anchor.TopCenter, minSize: crewFrame.RectTransform.MinSize));
|
|
rightPanels.Add(reputationframe);
|
|
GUIFrame reputationframeInner = new GUIFrame(new RectTransform(new Point(reputationframe.Rect.Width - padding * 2, reputationframe.Rect.Height - padding * 2), reputationframe.RectTransform, Anchor.Center), style: "InnerFrame");
|
|
|
|
var reputationContent = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.95f), reputationframeInner.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true
|
|
};
|
|
|
|
var reputationHeader = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), reputationContent.RectTransform),
|
|
TextManager.Get("reputation"), textAlignment: Alignment.TopLeft, font: GUIStyle.SubHeadingFont);
|
|
reputationHeader.RectTransform.MinSize = new Point(0, GUI.IntScale(reputationHeader.Rect.Height * 2.0f));
|
|
|
|
CreateReputationInfoPanel(reputationContent, campaignMode);
|
|
}
|
|
|
|
//mission panel -------------------------------------------------------------------------------
|
|
|
|
GUIFrame missionframe = new GUIFrame(new RectTransform(new Vector2(0.39f, 0.3f), background.RectTransform, Anchor.TopCenter, minSize: new Point(minWidth, minHeight / 4)));
|
|
GUILayoutGroup missionFrameContent = new GUILayoutGroup(new RectTransform(new Point(missionframe.Rect.Width - padding * 2, missionframe.Rect.Height - padding * 2), missionframe.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true,
|
|
RelativeSpacing = 0.05f
|
|
};
|
|
GUIFrame missionframeInner = new GUIFrame(new RectTransform(new Vector2(1.0f, 0.9f), missionFrameContent.RectTransform, Anchor.Center), style: "InnerFrame");
|
|
|
|
var missionContent = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.93f), missionframeInner.RectTransform, Anchor.Center))
|
|
{
|
|
Stretch = true
|
|
};
|
|
|
|
List<Mission> missionsToDisplay = new List<Mission>(selectedMissions.Where(m => m.Prefab.ShowInMenus));
|
|
if (!selectedMissions.Any() && startLocation != null)
|
|
{
|
|
foreach (Mission mission in startLocation.SelectedMissions)
|
|
{
|
|
if (!mission.Prefab.ShowInMenus) { continue; }
|
|
if (mission.Locations[0] == mission.Locations[1] ||
|
|
mission.Locations.Contains(campaignMode?.Map.SelectedLocation))
|
|
{
|
|
missionsToDisplay.Add(mission);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (missionsToDisplay.Any())
|
|
{
|
|
var missionHeader = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionContent.RectTransform),
|
|
TextManager.Get(missionsToDisplay.Count > 1 ? "Missions" : "Mission"), textAlignment: Alignment.TopLeft, font: GUIStyle.SubHeadingFont);
|
|
missionHeader.RectTransform.MinSize = new Point(0, (int)(missionHeader.Rect.Height * 1.2f));
|
|
}
|
|
|
|
GUIListBox missionList = new GUIListBox(new RectTransform(Vector2.One, missionContent.RectTransform, Anchor.Center))
|
|
{
|
|
Padding = new Vector4(4, 10, 0, 0) * GUI.Scale
|
|
};
|
|
missionList.ContentBackground.Color = Color.Transparent;
|
|
|
|
ButtonArea = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.1f), missionFrameContent.RectTransform, Anchor.BottomCenter), isHorizontal: true, childAnchor: Anchor.BottomRight)
|
|
{
|
|
RelativeSpacing = 0.025f
|
|
};
|
|
|
|
missionFrameContent.Recalculate();
|
|
missionContent.Recalculate();
|
|
|
|
if (!string.IsNullOrWhiteSpace(endMessage))
|
|
{
|
|
var endText = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionList.Content.RectTransform),
|
|
TextManager.GetServerMessage(endMessage), wrap: true)
|
|
{
|
|
CanBeFocused = false
|
|
};
|
|
endText.RectTransform.MinSize = new Point(0, endText.Rect.Height);
|
|
var line = new GUIFrame(new RectTransform(new Vector2(0.5f, 0.1f), missionList.Content.RectTransform), style: "HorizontalLine");
|
|
line.RectTransform.NonScaledSize = new Point(line.Rect.Width, GUI.IntScale(5.0f));
|
|
}
|
|
|
|
foreach (Mission displayedMission in missionsToDisplay)
|
|
{
|
|
var missionContentHorizontal = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.8f), missionList.Content.RectTransform), childAnchor: Anchor.CenterLeft, isHorizontal: true)
|
|
{
|
|
RelativeSpacing = 0.025f,
|
|
Stretch = true
|
|
};
|
|
|
|
LocalizedString missionMessage =
|
|
selectedMissions.Contains(displayedMission) ?
|
|
displayedMission.Completed ? displayedMission.SuccessMessage : displayedMission.FailureMessage :
|
|
displayedMission.Description;
|
|
GUIImage missionIcon = new GUIImage(new RectTransform(new Point((int)(missionContentHorizontal.Rect.Height)), missionContentHorizontal.RectTransform), displayedMission.Prefab.Icon, scaleToFit: true)
|
|
{
|
|
Color = displayedMission.Prefab.IconColor,
|
|
HoverColor = displayedMission.Prefab.IconColor,
|
|
SelectedColor = displayedMission.Prefab.IconColor
|
|
};
|
|
missionIcon.RectTransform.MinSize = new Point((int)(missionContentHorizontal.Rect.Height * 0.9f));
|
|
if (selectedMissions.Contains(displayedMission))
|
|
{
|
|
new GUIImage(new RectTransform(Vector2.One, missionIcon.RectTransform), displayedMission.Completed ? "MissionCompletedIcon" : "MissionFailedIcon", scaleToFit: true);
|
|
}
|
|
|
|
var missionTextContent = new GUILayoutGroup(new RectTransform(new Vector2(0.8f, 1.0f), missionContentHorizontal.RectTransform))
|
|
{
|
|
AbsoluteSpacing = GUI.IntScale(5)
|
|
};
|
|
missionContentHorizontal.Recalculate();
|
|
var missionNameTextBlock = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionTextContent.RectTransform),
|
|
displayedMission.Name, font: GUIStyle.SubHeadingFont);
|
|
if (displayedMission.Difficulty.HasValue)
|
|
{
|
|
var groupSize = missionNameTextBlock.Rect.Size;
|
|
groupSize.X -= (int)(missionNameTextBlock.Padding.X + missionNameTextBlock.Padding.Z);
|
|
var indicatorGroup = new GUILayoutGroup(new RectTransform(groupSize, missionTextContent.RectTransform) { AbsoluteOffset = new Point((int)missionNameTextBlock.Padding.X, 0) },
|
|
isHorizontal: true, childAnchor: Anchor.CenterLeft)
|
|
{
|
|
AbsoluteSpacing = 1
|
|
};
|
|
var difficultyColor = displayedMission.GetDifficultyColor();
|
|
for (int i = 0; i < displayedMission.Difficulty; i++)
|
|
{
|
|
new GUIImage(new RectTransform(Vector2.One, indicatorGroup.RectTransform, scaleBasis: ScaleBasis.Smallest) { IsFixedSize = true }, "DifficultyIndicator", scaleToFit: true)
|
|
{
|
|
CanBeFocused = false,
|
|
Color = difficultyColor
|
|
};
|
|
}
|
|
}
|
|
var missionDescription = new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionTextContent.RectTransform),
|
|
RichString.Rich(missionMessage), wrap: true);
|
|
int reward = displayedMission.GetReward(Submarine.MainSub);
|
|
if (selectedMissions.Contains(displayedMission) && displayedMission.Completed)
|
|
{
|
|
RichString reputationText = displayedMission.GetReputationRewardText();
|
|
if (!reputationText.IsNullOrEmpty())
|
|
{
|
|
new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionTextContent.RectTransform), reputationText);
|
|
}
|
|
|
|
if (reward > 0)
|
|
{
|
|
new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionTextContent.RectTransform), RichString.Rich(displayedMission.GetMissionRewardText(Submarine.MainSub)));
|
|
if (GameMain.IsMultiplayer && Character.Controlled is { } controlled)
|
|
{
|
|
var (share, percentage, _) = Mission.GetRewardShare(controlled.Wallet.RewardDistribution, GameSession.GetSessionCrewCharacters(CharacterType.Player).Where(c => c != controlled), Option<int>.Some(reward));
|
|
if (share > 0)
|
|
{
|
|
string shareFormatted = string.Format(CultureInfo.InvariantCulture, "{0:N0}", share);
|
|
RichString yourShareString = RichString.Rich(TextManager.GetWithVariables("crewwallet.missionreward.get", ("[money]", $"{shareFormatted}"), ("[share]", $"{percentage}")));
|
|
new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionTextContent.RectTransform), yourShareString);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (displayedMission != missionsToDisplay.Last())
|
|
{
|
|
var spacing = new GUIFrame(new RectTransform(new Vector2(1.0f, 1.0f), missionList.Content.RectTransform) { MaxSize = new Point(int.MaxValue, GUI.IntScale(15)) }, style: null);
|
|
new GUIFrame(new RectTransform(new Vector2(0.8f, 1.0f), spacing.RectTransform, Anchor.Center) { RelativeOffset = new Vector2(0.1f, 0.0f) }, "HorizontalLine");
|
|
}
|
|
|
|
foreach (GUIComponent child in missionTextContent.Children)
|
|
{
|
|
child.RectTransform.IsFixedSize = true;
|
|
}
|
|
missionTextContent.RectTransform.MinSize = new Point(0, missionTextContent.Children.Sum(c => c.Rect.Height + missionTextContent.AbsoluteSpacing));
|
|
missionContentHorizontal.RectTransform.MinSize = new Point(0, (int)(missionTextContent.Rect.Height / missionTextContent.RectTransform.RelativeSize.Y));
|
|
}
|
|
|
|
if (!missionsToDisplay.Any())
|
|
{
|
|
var missionContentHorizontal = new GUILayoutGroup(new RectTransform(Vector2.One, missionList.Content.RectTransform), childAnchor: Anchor.TopLeft, isHorizontal: true)
|
|
{
|
|
RelativeSpacing = 0.025f,
|
|
Stretch = true
|
|
};
|
|
GUIImage missionIcon = new GUIImage(new RectTransform(new Point((int)(missionContentHorizontal.Rect.Height * 0.7f)), missionContentHorizontal.RectTransform), style: "NoMissionIcon", scaleToFit: true);
|
|
missionIcon.RectTransform.MinSize = new Point((int)(missionContentHorizontal.Rect.Height * 0.7f));
|
|
new GUITextBlock(new RectTransform(new Vector2(1.0f, 0.0f), missionContentHorizontal.RectTransform),
|
|
TextManager.Get("nomission"), font: GUIStyle.LargeFont);
|
|
}
|
|
|
|
/*missionContentHorizontal.Recalculate();
|
|
missionContent.Recalculate();
|
|
missionIcon.RectTransform.MinSize = new Point(0, missionContentHorizontal.Rect.Height);
|
|
missionTextContent.RectTransform.MaxSize = new Point(int.MaxValue, missionIcon.Rect.Width);*/
|
|
|
|
ContinueButton = new GUIButton(new RectTransform(new Vector2(0.25f, 1.0f), ButtonArea.RectTransform), TextManager.Get("Close"));
|
|
ButtonArea.RectTransform.NonScaledSize = new Point(ButtonArea.Rect.Width, ContinueButton.Rect.Height);
|
|
ButtonArea.RectTransform.IsFixedSize = true;
|
|
|
|
missionFrameContent.Recalculate();
|
|
|
|
// set layout -------------------------------------------------------------------
|
|
|
|
int panelSpacing = GUI.IntScale(20);
|
|
int totalHeight = crewFrame.Rect.Height + panelSpacing + missionframe.Rect.Height;
|
|
int totalWidth = crewFrame.Rect.Width;
|
|
|
|
crewFrame.RectTransform.AbsoluteOffset = new Point(0, (GameMain.GraphicsHeight - totalHeight) / 2);
|
|
missionframe.RectTransform.AbsoluteOffset = new Point(0, crewFrame.Rect.Bottom + panelSpacing);
|
|
|
|
if (rightPanels.Any())
|
|
{
|
|
totalWidth = crewFrame.Rect.Width * 2 + panelSpacing;
|
|
if (headerTextBlock != null)
|
|
{
|
|
headerTextBlock.RectTransform.MinSize = new Point(totalWidth, 0);
|
|
}
|
|
crewFrame.RectTransform.AbsoluteOffset = new Point(-(crewFrame.Rect.Width + panelSpacing) / 2, crewFrame.RectTransform.AbsoluteOffset.Y);
|
|
foreach (var rightPanel in rightPanels)
|
|
{
|
|
rightPanel.RectTransform.AbsoluteOffset = new Point((rightPanel.Rect.Width + panelSpacing) / 2, crewFrame.RectTransform.AbsoluteOffset.Y);
|
|
}
|
|
}
|
|
|
|
Frame = background;
|
|
return background;
|
|
}
|
|
|
|
public void CreateReputationInfoPanel(GUIComponent parent, CampaignMode campaignMode)
|
|
{
|
|
GUIListBox reputationList = new GUIListBox(new RectTransform(Vector2.One, parent.RectTransform))
|
|
{
|
|
Padding = new Vector4(4, 10, 0, 0) * GUI.Scale
|
|
};
|
|
reputationList.ContentBackground.Color = Color.Transparent;
|
|
|
|
foreach (Faction faction in campaignMode.Factions.OrderBy(f => f.Prefab.MenuOrder).ThenBy(f => f.Prefab.Name))
|
|
{
|
|
float initialReputation = faction.Reputation.Value;
|
|
if (!initialFactionReputations.TryGetValue(faction.Prefab.Identifier, out initialReputation))
|
|
{
|
|
DebugConsole.AddWarning($"Could not determine reputation change for faction \"{faction.Prefab.Name}\" (faction was not present at the start of the round).");
|
|
}
|
|
var factionFrame = CreateReputationElement(
|
|
reputationList.Content,
|
|
faction.Prefab.Name,
|
|
faction.Reputation.Value, faction.Reputation.NormalizedValue, initialReputation,
|
|
faction.Prefab.ShortDescription, faction.Prefab.Description,
|
|
faction.Prefab.Icon, faction.Prefab.BackgroundPortrait, faction.Prefab.IconColor);
|
|
CreatePathUnlockElement(factionFrame, faction, null);
|
|
}
|
|
|
|
float maxDescriptionHeight = 0.0f;
|
|
foreach (GUIComponent child in reputationList.Content.Children)
|
|
{
|
|
var descriptionElement = child.FindChild("description", recursive: true) as GUITextBlock;
|
|
float descriptionHeight = descriptionElement.TextSize.Y * 1.1f;
|
|
if (child.FindChild("unlockinfo") is GUIComponent unlockInfoComponent)
|
|
{
|
|
descriptionHeight += 1.25f * unlockInfoComponent.Rect.Height;
|
|
}
|
|
maxDescriptionHeight = Math.Max(maxDescriptionHeight, descriptionHeight);
|
|
}
|
|
foreach (GUIComponent child in reputationList.Content.Children)
|
|
{
|
|
var headerElement = child.FindChild("header", recursive: true) as GUITextBlock;
|
|
var descriptionElement = child.FindChild("description", recursive: true) as GUITextBlock;
|
|
descriptionElement.RectTransform.NonScaledSize = new Point(descriptionElement.Rect.Width, (int)maxDescriptionHeight);
|
|
descriptionElement.RectTransform.IsFixedSize = true;
|
|
child.RectTransform.NonScaledSize = new Point(child.Rect.Width, headerElement.Rect.Height + descriptionElement.RectTransform.Parent.Children.Sum(c => c.Rect.Height + ((GUILayoutGroup)descriptionElement.Parent).AbsoluteSpacing));
|
|
}
|
|
|
|
void CreatePathUnlockElement(GUIComponent reputationFrame, Faction faction, Location location)
|
|
{
|
|
if (GameMain.GameSession?.Campaign?.Map == null) { return; }
|
|
|
|
IEnumerable<LocationConnection> connectionsBetweenBiomes =
|
|
GameMain.GameSession.Campaign.Map.Connections.Where(c => c.Locations[0].Biome != c.Locations[1].Biome);
|
|
|
|
foreach (LocationConnection connection in connectionsBetweenBiomes)
|
|
{
|
|
if (!connection.Locked || (!connection.Locations[0].Discovered && !connection.Locations[1].Discovered)) { continue; }
|
|
|
|
//don't show the "reputation required to unlock" text if another connection between the biomes has already been unlocked
|
|
if (connectionsBetweenBiomes.Where(c => !c.Locked).Any(c =>
|
|
(c.Locations[0].Biome == connection.Locations[0].Biome && c.Locations[1].Biome == connection.Locations[1].Biome) ||
|
|
(c.Locations[1].Biome == connection.Locations[0].Biome && c.Locations[0].Biome == connection.Locations[1].Biome)))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
var gateLocation = connection.Locations[0].IsGateBetweenBiomes ? connection.Locations[0] : connection.Locations[1];
|
|
var unlockEvent = EventPrefab.GetUnlockPathEvent(gateLocation.LevelData.Biome.Identifier, gateLocation.Faction);
|
|
|
|
if (unlockEvent == null) { continue; }
|
|
if (unlockEvent.Faction.IsEmpty)
|
|
{
|
|
if (location == null || gateLocation != location) { continue; }
|
|
}
|
|
else
|
|
{
|
|
if (faction == null || faction.Prefab.Identifier != unlockEvent.Faction) { continue; }
|
|
}
|
|
|
|
if (unlockEvent != null)
|
|
{
|
|
Reputation unlockReputation = gateLocation.Reputation;
|
|
Faction unlockFaction = null;
|
|
if (!unlockEvent.Faction.IsEmpty)
|
|
{
|
|
unlockFaction = GameMain.GameSession.Campaign.Factions.Find(f => f.Prefab.Identifier == unlockEvent.Faction);
|
|
unlockReputation = unlockFaction?.Reputation;
|
|
}
|
|
float normalizedUnlockReputation = MathUtils.InverseLerp(unlockReputation.MinReputation, unlockReputation.MaxReputation, unlockEvent.UnlockPathReputation);
|
|
RichString unlockText = RichString.Rich(TextManager.GetWithVariables(
|
|
"lockedpathreputationrequirement",
|
|
("[reputation]", Reputation.GetFormattedReputationText(normalizedUnlockReputation, unlockEvent.UnlockPathReputation, addColorTags: true)),
|
|
("[biomename]", $"‖color:gui.orange‖{connection.LevelData.Biome.DisplayName}‖end‖")));
|
|
var unlockInfoPanel = new GUITextBlock(new RectTransform(new Vector2(0.8f, 0.0f), reputationFrame.RectTransform, Anchor.BottomCenter) { MinSize = new Point(0, GUI.IntScale(30)), AbsoluteOffset = new Point(0, GUI.IntScale(3)) },
|
|
unlockText, style: "GUIButtonRound", textAlignment: Alignment.Center, textColor: GUIStyle.TextColorNormal);
|
|
unlockInfoPanel.Color = Color.Lerp(unlockInfoPanel.Color, Color.Black, 0.8f);
|
|
unlockInfoPanel.UserData = "unlockinfo";
|
|
if (unlockInfoPanel.TextSize.X > unlockInfoPanel.Rect.Width * 0.7f)
|
|
{
|
|
unlockInfoPanel.Font = GUIStyle.SmallFont;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private LocalizedString GetHeaderText(bool gameOver, CampaignMode.TransitionType transitionType)
|
|
{
|
|
string locationName = Submarine.MainSub is { AtEndExit: true } ? endLocation?.Name : startLocation?.Name;
|
|
|
|
string textTag;
|
|
if (gameOver)
|
|
{
|
|
textTag = "RoundSummaryGameOver";
|
|
}
|
|
else
|
|
{
|
|
switch (transitionType)
|
|
{
|
|
case CampaignMode.TransitionType.LeaveLocation:
|
|
locationName = startLocation?.Name;
|
|
textTag = "RoundSummaryLeaving";
|
|
break;
|
|
case CampaignMode.TransitionType.ProgressToNextLocation:
|
|
locationName = endLocation?.Name;
|
|
textTag = "RoundSummaryProgress";
|
|
break;
|
|
case CampaignMode.TransitionType.ProgressToNextEmptyLocation:
|
|
locationName = endLocation?.Name;
|
|
textTag = "RoundSummaryProgressToEmptyLocation";
|
|
break;
|
|
case CampaignMode.TransitionType.ReturnToPreviousLocation:
|
|
locationName = startLocation?.Name;
|
|
textTag = "RoundSummaryReturn";
|
|
break;
|
|
case CampaignMode.TransitionType.ReturnToPreviousEmptyLocation:
|
|
locationName = startLocation?.Name;
|
|
textTag = "RoundSummaryReturnToEmptyLocation";
|
|
break;
|
|
default:
|
|
textTag = Submarine.MainSub.AtEndExit ? "RoundSummaryProgress" : "RoundSummaryReturn";
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (startLocation?.Biome != null && startLocation.Biome.IsEndBiome)
|
|
{
|
|
locationName ??= startLocation.Name;
|
|
}
|
|
|
|
if (textTag == null) { return ""; }
|
|
|
|
if (locationName == null)
|
|
{
|
|
DebugConsole.ThrowError($"Error while creating round summary: could not determine destination location. Start location: {startLocation?.Name ?? "null"}, end location: {endLocation?.Name ?? "null"}");
|
|
locationName = "[UNKNOWN]";
|
|
}
|
|
|
|
LocalizedString subName = string.Empty;
|
|
SubmarineInfo currentOrPending = SubmarineSelection.CurrentOrPendingSubmarine();
|
|
if (currentOrPending != null)
|
|
{
|
|
subName = currentOrPending.DisplayName;
|
|
}
|
|
|
|
return TextManager.GetWithVariables(textTag, ("[sub]", subName), ("[location]", locationName));
|
|
}
|
|
|
|
private GUIListBox CreateCrewList(GUIComponent parent, IEnumerable<CharacterInfo> characterInfos)
|
|
{
|
|
var headerFrame = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.0f), parent.RectTransform, Anchor.TopCenter, minSize: new Point(0, (int)(30 * GUI.Scale))) { }, isHorizontal: true)
|
|
{
|
|
AbsoluteSpacing = 2
|
|
};
|
|
GUIButton jobButton = new GUIButton(new RectTransform(new Vector2(0f, 1f), headerFrame.RectTransform), TextManager.Get("tabmenu.job"), style: "GUIButtonSmallFreeScale");
|
|
GUIButton characterButton = new GUIButton(new RectTransform(new Vector2(0f, 1f), headerFrame.RectTransform), TextManager.Get("name"), style: "GUIButtonSmallFreeScale");
|
|
GUIButton statusButton = new GUIButton(new RectTransform(new Vector2(0f, 1f), headerFrame.RectTransform), TextManager.Get("label.statuslabel"), style: "GUIButtonSmallFreeScale");
|
|
|
|
float sizeMultiplier = 1.0f;
|
|
//sizeMultiplier = (headerFrame.Rect.Width - headerFrame.AbsoluteSpacing * (headerFrame.CountChildren - 1)) / (float)headerFrame.Rect.Width;
|
|
|
|
jobButton.RectTransform.RelativeSize = new Vector2(jobColumnWidthPercentage * sizeMultiplier, 1f);
|
|
characterButton.RectTransform.RelativeSize = new Vector2(characterColumnWidthPercentage * sizeMultiplier, 1f);
|
|
statusButton.RectTransform.RelativeSize = new Vector2(statusColumnWidthPercentage * sizeMultiplier, 1f);
|
|
|
|
jobButton.TextBlock.Font = characterButton.TextBlock.Font = statusButton.TextBlock.Font = GUIStyle.HotkeyFont;
|
|
jobButton.CanBeFocused = characterButton.CanBeFocused = statusButton.CanBeFocused = false;
|
|
jobButton.TextBlock.ForceUpperCase = characterButton.TextBlock.ForceUpperCase = statusButton.ForceUpperCase = ForceUpperCase.Yes;
|
|
|
|
jobColumnWidth = jobButton.Rect.Width;
|
|
characterColumnWidth = characterButton.Rect.Width;
|
|
statusColumnWidth = statusButton.Rect.Width;
|
|
|
|
GUIListBox crewList = new GUIListBox(new RectTransform(Vector2.One, parent.RectTransform))
|
|
{
|
|
Padding = new Vector4(4, 10, 0, 0) * GUI.Scale,
|
|
AutoHideScrollBar = false
|
|
};
|
|
crewList.ContentBackground.Color = Color.Transparent;
|
|
|
|
headerFrame.RectTransform.RelativeSize -= new Vector2(crewList.ScrollBar.RectTransform.RelativeSize.X, 0.0f);
|
|
|
|
foreach (CharacterInfo characterInfo in characterInfos)
|
|
{
|
|
if (characterInfo == null) { continue; }
|
|
CreateCharacterElement(characterInfo, crewList);
|
|
}
|
|
|
|
return crewList;
|
|
}
|
|
|
|
private void CreateCharacterElement(CharacterInfo characterInfo, GUIListBox listBox)
|
|
{
|
|
GUIFrame frame = new GUIFrame(new RectTransform(new Point(listBox.Content.Rect.Width, GUI.IntScale(45)), listBox.Content.RectTransform), style: "ListBoxElement")
|
|
{
|
|
CanBeFocused = false,
|
|
UserData = characterInfo,
|
|
Color = (Character.Controlled?.Info == characterInfo) ? TabMenu.OwnCharacterBGColor : Color.Transparent
|
|
};
|
|
|
|
var paddedFrame = new GUILayoutGroup(new RectTransform(new Vector2(1.0f, 0.9f), frame.RectTransform, Anchor.Center), isHorizontal: true)
|
|
{
|
|
AbsoluteSpacing = 2,
|
|
Stretch = true
|
|
};
|
|
|
|
new GUICustomComponent(new RectTransform(new Point(jobColumnWidth, paddedFrame.Rect.Height), paddedFrame.RectTransform, Anchor.Center), onDraw: (sb, component) => characterInfo.DrawJobIcon(sb, component.Rect))
|
|
{
|
|
ToolTip = characterInfo.Job.Name ?? "",
|
|
HoverColor = Color.White,
|
|
SelectedColor = Color.White
|
|
};
|
|
|
|
GUITextBlock characterNameBlock = new GUITextBlock(new RectTransform(new Point(characterColumnWidth, paddedFrame.Rect.Height), paddedFrame.RectTransform),
|
|
ToolBox.LimitString(characterInfo.Name, GUIStyle.Font, characterColumnWidth), textAlignment: Alignment.Center, textColor: characterInfo.Job.Prefab.UIColor);
|
|
|
|
LocalizedString statusText = TextManager.Get("StatusOK");
|
|
Color statusColor = GUIStyle.Green;
|
|
|
|
Character character = characterInfo.Character;
|
|
if (character == null || character.IsDead)
|
|
{
|
|
if (character == null && characterInfo.IsNewHire && characterInfo.CauseOfDeath == null)
|
|
{
|
|
statusText = TextManager.Get("CampaignCrew.NewHire");
|
|
statusColor = GUIStyle.Blue;
|
|
}
|
|
else if (characterInfo.CauseOfDeath == null)
|
|
{
|
|
statusText = TextManager.Get("CauseOfDeathDescription.Unknown");
|
|
statusColor = Color.DarkRed;
|
|
}
|
|
else if (characterInfo.CauseOfDeath.Type == CauseOfDeathType.Affliction && characterInfo.CauseOfDeath.Affliction == null)
|
|
{
|
|
string errorMsg = "Character \"[name]\" had an invalid cause of death (the type of the cause of death was Affliction, but affliction was not specified).";
|
|
DebugConsole.ThrowError(errorMsg.Replace("[name]", characterInfo.Name));
|
|
GameAnalyticsManager.AddErrorEventOnce("RoundSummary:InvalidCauseOfDeath", GameAnalyticsManager.ErrorSeverity.Error, errorMsg.Replace("[name]", characterInfo.SpeciesName.Value));
|
|
statusText = TextManager.Get("CauseOfDeathDescription.Unknown");
|
|
statusColor = GUIStyle.Red;
|
|
}
|
|
else
|
|
{
|
|
statusText = characterInfo.CauseOfDeath.Type == CauseOfDeathType.Affliction ?
|
|
characterInfo.CauseOfDeath.Affliction.CauseOfDeathDescription :
|
|
TextManager.Get("CauseOfDeathDescription." + characterInfo.CauseOfDeath.Type.ToString());
|
|
statusColor = Color.DarkRed;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (character.IsUnconscious)
|
|
{
|
|
statusText = TextManager.Get("Unconscious");
|
|
statusColor = Color.DarkOrange;
|
|
}
|
|
else if (character.Vitality / character.MaxVitality < 0.8f)
|
|
{
|
|
statusText = TextManager.Get("Injured");
|
|
statusColor = Color.DarkOrange;
|
|
}
|
|
}
|
|
|
|
GUITextBlock statusBlock = new GUITextBlock(new RectTransform(new Point(statusColumnWidth, paddedFrame.Rect.Height), paddedFrame.RectTransform),
|
|
ToolBox.LimitString(statusText.Value, GUIStyle.Font, characterColumnWidth), textAlignment: Alignment.Center, textColor: statusColor);
|
|
}
|
|
|
|
private GUIFrame CreateReputationElement(GUIComponent parent,
|
|
LocalizedString name, float reputation, float normalizedReputation, float initialReputation,
|
|
LocalizedString shortDescription, LocalizedString fullDescription, Sprite icon, Sprite backgroundPortrait, Color iconColor)
|
|
{
|
|
var factionFrame = new GUIFrame(new RectTransform(new Vector2(1.0f, 0.1f), parent.RectTransform), style: null);
|
|
|
|
if (backgroundPortrait != null)
|
|
{
|
|
new GUICustomComponent(new RectTransform(Vector2.One, factionFrame.RectTransform), onDraw: (sb, customComponent) =>
|
|
{
|
|
backgroundPortrait.Draw(sb, customComponent.Rect.Center.ToVector2(), customComponent.Color, backgroundPortrait.size / 2, scale: customComponent.Rect.Width / backgroundPortrait.size.X);
|
|
})
|
|
{
|
|
HideElementsOutsideFrame = true,
|
|
IgnoreLayoutGroups = true,
|
|
Color = iconColor * 0.2f
|
|
};
|
|
}
|
|
|
|
var factionInfoHorizontal = new GUILayoutGroup(new RectTransform(new Vector2(0.95f, 0.9f), factionFrame.RectTransform, Anchor.Center), childAnchor: Anchor.CenterLeft, isHorizontal: true)
|
|
{
|
|
AbsoluteSpacing = GUI.IntScale(5),
|
|
Stretch = true
|
|
};
|
|
|
|
var factionTextContent = new GUILayoutGroup(new RectTransform(Vector2.One, factionInfoHorizontal.RectTransform))
|
|
{
|
|
AbsoluteSpacing = GUI.IntScale(10),
|
|
Stretch = true
|
|
};
|
|
var factionIcon = new GUIImage(new RectTransform(Vector2.One * 0.7f, factionInfoHorizontal.RectTransform, scaleBasis: ScaleBasis.Smallest), icon, scaleToFit: true)
|
|
{
|
|
Color = iconColor
|
|
};
|
|
factionInfoHorizontal.Recalculate();
|
|
|
|
var header = new GUITextBlock(new RectTransform(new Point(factionTextContent.Rect.Width, GUI.IntScale(40)), factionTextContent.RectTransform),
|
|
name, font: GUIStyle.SubHeadingFont)
|
|
{
|
|
Padding = Vector4.Zero,
|
|
UserData = "header"
|
|
};
|
|
header.RectTransform.IsFixedSize = true;
|
|
|
|
var sliderHolder = new GUILayoutGroup(new RectTransform(new Point((int)(factionTextContent.Rect.Width * 0.8f), GUI.IntScale(20.0f)), factionTextContent.RectTransform),
|
|
childAnchor: Anchor.CenterLeft, isHorizontal: true)
|
|
{
|
|
RelativeSpacing = 0.05f,
|
|
Stretch = true
|
|
};
|
|
sliderHolder.RectTransform.IsFixedSize = true;
|
|
factionTextContent.Recalculate();
|
|
|
|
new GUICustomComponent(new RectTransform(new Vector2(0.8f, 1.0f), sliderHolder.RectTransform),
|
|
onDraw: (sb, customComponent) => DrawReputationBar(sb, customComponent.Rect, normalizedReputation));
|
|
|
|
LocalizedString reputationText = Reputation.GetFormattedReputationText(normalizedReputation, reputation, addColorTags: true);
|
|
int reputationChange = (int)Math.Round(reputation - initialReputation);
|
|
if (Math.Abs(reputationChange) > 0)
|
|
{
|
|
string changeText = $"{(reputationChange > 0 ? "+" : "") + reputationChange}";
|
|
string colorStr = XMLExtensions.ToStringHex(reputationChange > 0 ? GUIStyle.Green : GUIStyle.Red);
|
|
var richText = RichString.Rich($"{reputationText} (‖color:{colorStr}‖{changeText}‖color:end‖)");
|
|
new GUITextBlock(new RectTransform(new Vector2(0.5f, 1.0f), sliderHolder.RectTransform),
|
|
richText,
|
|
textAlignment: Alignment.CenterLeft, font: GUIStyle.SubHeadingFont);
|
|
}
|
|
else
|
|
{
|
|
new GUITextBlock(new RectTransform(new Vector2(0.5f, 1.0f), sliderHolder.RectTransform),
|
|
RichString.Rich(reputationText),
|
|
textAlignment: Alignment.CenterLeft, font: GUIStyle.SubHeadingFont);
|
|
}
|
|
|
|
//spacing
|
|
new GUIFrame(new RectTransform(new Vector2(1.0f, 0.0f), factionTextContent.RectTransform) { MinSize = new Point(0, GUI.IntScale(5)) }, style: null);
|
|
|
|
var factionDescription = new GUITextBlock(new RectTransform(new Vector2(0.8f, 0.6f), factionTextContent.RectTransform),
|
|
shortDescription, font: GUIStyle.SmallFont, wrap: true)
|
|
{
|
|
UserData = "description",
|
|
Padding = Vector4.Zero
|
|
};
|
|
if (shortDescription != fullDescription && !fullDescription.IsNullOrEmpty())
|
|
{
|
|
factionDescription.ToolTip = fullDescription;
|
|
}
|
|
|
|
//spacing
|
|
new GUIFrame(new RectTransform(new Vector2(1.0f, 0.0f), factionTextContent.RectTransform) { MinSize = new Point(0, GUI.IntScale(5)) }, style: null);
|
|
|
|
factionInfoHorizontal.Recalculate();
|
|
factionTextContent.Recalculate();
|
|
|
|
return factionFrame;
|
|
}
|
|
|
|
public static void DrawReputationBar(SpriteBatch sb, Rectangle rect, float normalizedReputation)
|
|
{
|
|
int segmentWidth = rect.Width / 5;
|
|
rect.Width = segmentWidth * 5;
|
|
for (int i = 0; i < 5; i++)
|
|
{
|
|
GUI.DrawRectangle(sb, new Rectangle(rect.X + (segmentWidth * i), rect.Y, segmentWidth, rect.Height), Reputation.GetReputationColor(i / 5.0f), isFilled: true);
|
|
GUI.DrawRectangle(sb, new Rectangle(rect.X + (segmentWidth * i), rect.Y, segmentWidth, rect.Height), GUIStyle.ColorInventoryBackground, isFilled: false);
|
|
}
|
|
GUI.DrawRectangle(sb, rect, GUIStyle.ColorInventoryBackground, isFilled: false);
|
|
|
|
GUI.Arrow.Draw(sb, new Vector2(rect.X + rect.Width * normalizedReputation, rect.Y), GUIStyle.ColorInventoryBackground, scale: GUI.Scale, spriteEffect: SpriteEffects.FlipVertically);
|
|
GUI.Arrow.Draw(sb, new Vector2(rect.X + rect.Width * normalizedReputation, rect.Y), GUIStyle.TextColorNormal, scale: GUI.Scale * 0.8f, spriteEffect: SpriteEffects.FlipVertically);
|
|
|
|
GUI.DrawString(sb, new Vector2(rect.X, rect.Bottom), "-100", GUIStyle.TextColorNormal, font: GUIStyle.SmallFont);
|
|
Vector2 textSize = GUIStyle.SmallFont.MeasureString("100");
|
|
GUI.DrawString(sb, new Vector2(rect.Right - textSize.X, rect.Bottom), "100", GUIStyle.TextColorNormal, font: GUIStyle.SmallFont);
|
|
}
|
|
}
|
|
}
|