buncha shit
This commit is contained in:
parent
1d6231789c
commit
675455a990
275
Partypacker/INIFile.cs
Normal file
275
Partypacker/INIFile.cs
Normal file
|
@ -0,0 +1,275 @@
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.IO;
|
||||||
|
|
||||||
|
namespace Partypacker
|
||||||
|
{
|
||||||
|
public class INIFile
|
||||||
|
{
|
||||||
|
private string _File;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Call the constructor creates a new object of the INIFile class to work with INI files.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="file">Name of INI file, which you want to access.</param>
|
||||||
|
/// <param name="createFile">Specifies whether the INI file should be created if it does not exist.</param>
|
||||||
|
public INIFile(string file, bool createFile = false)
|
||||||
|
{
|
||||||
|
if (createFile == true && File.Exists(file) == false)
|
||||||
|
{
|
||||||
|
FileInfo fileInfo = new FileInfo(file);
|
||||||
|
FileStream fileStream = fileInfo.Create();
|
||||||
|
fileStream.Close();
|
||||||
|
}
|
||||||
|
_File = file;
|
||||||
|
}
|
||||||
|
|
||||||
|
#region Public Methods
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Removes all comments and empty lines from a complete section and returns the sections.
|
||||||
|
/// This method is not case-sensitive.
|
||||||
|
/// The return value does not contain any spaces at the beginning or at the end of a line.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="section">Name of the requested section.</param>
|
||||||
|
/// <param name="includeComments">Specifies whether comments should also be returned.</param>
|
||||||
|
/// <returns>Returns the whole section.</returns>
|
||||||
|
public List<string> GetSection(string section, bool includeComments = false)
|
||||||
|
{
|
||||||
|
section = CheckSection(section);
|
||||||
|
|
||||||
|
List<string> completeSection = new List<string>();
|
||||||
|
bool sectionStart = false;
|
||||||
|
|
||||||
|
string[] fileArray = File.ReadAllLines(_File);
|
||||||
|
|
||||||
|
foreach (var item in fileArray)
|
||||||
|
{
|
||||||
|
if (item.Length <= 0) continue;
|
||||||
|
|
||||||
|
// Beginning of section.
|
||||||
|
if (item.Replace(" ", "").ToLower() == section)
|
||||||
|
{
|
||||||
|
sectionStart = true;
|
||||||
|
}
|
||||||
|
// Beginning of next section.
|
||||||
|
if (sectionStart == true && item.Replace(" ", "").ToLower() != section && item.Replace(" ", "").Substring(0, 1) == "[" && item.Replace(" ", "").Substring(item.Length - 1, 1) == "]")
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (sectionStart == true)
|
||||||
|
{
|
||||||
|
// Add the entry to the List<string> completeSection, if it is not a comment or an empty entry.
|
||||||
|
if (includeComments == false
|
||||||
|
&& item.Replace(" ", "").Substring(0, 1) != ";" && !string.IsNullOrWhiteSpace(item))
|
||||||
|
{
|
||||||
|
completeSection.Add(ReplaceSpacesAtStartAndEnd(item));
|
||||||
|
}
|
||||||
|
if (includeComments == true && !string.IsNullOrWhiteSpace(item))
|
||||||
|
{
|
||||||
|
completeSection.Add(ReplaceSpacesAtStartAndEnd(item));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return completeSection;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The method returns a value for the associated key.
|
||||||
|
/// This method is not case-sensitive.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="section">Name of the requested section.</param>
|
||||||
|
/// <param name="key">Name of the requested key.</param>
|
||||||
|
/// <param name="convertValueToLower">If "true" is passed, the value will be returned in lowercase letters.</param>
|
||||||
|
/// <returns>Returns the value for the specified key in the specified section, if available, otherwise null.</returns>
|
||||||
|
public string GetValue(string section, string key, bool convertValueToLower = false)
|
||||||
|
{
|
||||||
|
section = CheckSection(section);
|
||||||
|
key = key.ToLower();
|
||||||
|
|
||||||
|
List<string> completeSection = GetSection(section);
|
||||||
|
|
||||||
|
foreach (var item in completeSection)
|
||||||
|
{
|
||||||
|
// Continue if entry is no key.
|
||||||
|
if (!item.Contains("=") && item.Contains("[") && item.Contains("]")) continue;
|
||||||
|
|
||||||
|
string[] keyAndValue = item.Split(new string[] { "=" }, StringSplitOptions.RemoveEmptyEntries);
|
||||||
|
if (keyAndValue[0].ToLower() == key && keyAndValue.Count() > 1)
|
||||||
|
{
|
||||||
|
if (convertValueToLower == true)
|
||||||
|
{
|
||||||
|
keyAndValue[1] = keyAndValue[1].ToLower();
|
||||||
|
}
|
||||||
|
return keyAndValue[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Set or add a value of the associated key in the specified section.
|
||||||
|
/// This method is not case-sensitive.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="section">Name of the section.</param>
|
||||||
|
/// <param name="key">Name of the key.</param>
|
||||||
|
/// <param name="value">Value to save.</param>
|
||||||
|
/// <param name="convertValueToLower">If "true" is passed, the value will be saved in lowercase letters.</param>
|
||||||
|
public void SetValue(string section, string key, string value, bool convertValueToLower = false)
|
||||||
|
{
|
||||||
|
section = CheckSection(section, false);
|
||||||
|
string sectionToLower = section.ToLower();
|
||||||
|
|
||||||
|
bool sectionFound = false;
|
||||||
|
|
||||||
|
List<string> iniFileContent = new List<string>();
|
||||||
|
|
||||||
|
string[] fileLines = File.ReadAllLines(_File);
|
||||||
|
|
||||||
|
// Creates a new INI file if none exists.
|
||||||
|
if (fileLines.Length <= 0)
|
||||||
|
{
|
||||||
|
iniFileContent = AddSection(iniFileContent, section, key, value, convertValueToLower);
|
||||||
|
WriteFile(iniFileContent);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < fileLines.Length; i++)
|
||||||
|
{
|
||||||
|
// Possibility 1: The desired section has not (yet) been found.
|
||||||
|
if (fileLines[i].Replace(" ", "").ToLower() != sectionToLower)
|
||||||
|
{
|
||||||
|
iniFileContent.Add(fileLines[i]);
|
||||||
|
// If a section does not exist, the section will be created.
|
||||||
|
if (i == fileLines.Length - 1 && fileLines[i].Replace(" ", "").ToLower() != sectionToLower && sectionFound == false)
|
||||||
|
{
|
||||||
|
iniFileContent.Add(null);
|
||||||
|
iniFileContent = AddSection(iniFileContent, section, key, value, convertValueToLower);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Possibility 2 -> Desired section was found.
|
||||||
|
sectionFound = true;
|
||||||
|
|
||||||
|
// Get the complete section in which the target key may be.
|
||||||
|
List<string> targetSection = GetSection(sectionToLower, true);
|
||||||
|
|
||||||
|
for (int x = 0; x < targetSection.Count; x++)
|
||||||
|
{
|
||||||
|
string[] targetKey = targetSection[x].Split(new string[] { "=" }, StringSplitOptions.None);
|
||||||
|
// When the target key is found.
|
||||||
|
if (targetKey[0].ToLower() == key.ToLower())
|
||||||
|
{
|
||||||
|
if (convertValueToLower == true)
|
||||||
|
{
|
||||||
|
iniFileContent.Add(key + "=" + value.ToLower());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
iniFileContent.Add(key + "=" + value);
|
||||||
|
}
|
||||||
|
i = i + x;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
iniFileContent.Add(targetSection[x]);
|
||||||
|
// If the target key is not found, it will be created.
|
||||||
|
if (x == targetSection.Count - 1 && targetKey[0].ToLower() != key.ToLower())
|
||||||
|
{
|
||||||
|
if (convertValueToLower == true)
|
||||||
|
{
|
||||||
|
iniFileContent.Add(key + "=" + value.ToLower());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
iniFileContent.Add(key + "=" + value);
|
||||||
|
}
|
||||||
|
i = i + x;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WriteFile(iniFileContent);
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region Private Methods
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Ensures that a section is always in the following format: [section].
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="section">Section to be checked for correct format.</param>
|
||||||
|
/// <param name="convertToLower">Specifies whether the section should be vonverted in lower case letters.</param>
|
||||||
|
/// <returns>Returns section in this form: [section].</returns>
|
||||||
|
private string CheckSection(string section, bool convertToLower = true)
|
||||||
|
{
|
||||||
|
if (convertToLower == true)
|
||||||
|
{
|
||||||
|
section = section.ToLower();
|
||||||
|
}
|
||||||
|
if (!section.StartsWith("[") && !section.EndsWith("]"))
|
||||||
|
{
|
||||||
|
section = "[" + section + "]";
|
||||||
|
}
|
||||||
|
return section;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Removes leading and trailing spaces from sections, keys and values.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="item">String to be trimmed.</param>
|
||||||
|
/// <returns>Returns the trimmed string.</returns>
|
||||||
|
private string ReplaceSpacesAtStartAndEnd(string item)
|
||||||
|
{
|
||||||
|
// If the string has a key and a value.
|
||||||
|
if (item.Contains("=") && !item.Contains("[") && !item.Contains("]"))
|
||||||
|
{
|
||||||
|
string[] keyAndValue = item.Split(new string[] { "=" }, StringSplitOptions.None);
|
||||||
|
return keyAndValue[0].Trim() + "=" + keyAndValue[1].Trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
return item.Trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adds a new section with key value pair.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="iniFileContent">List iniFileContent from SetValue.</param>
|
||||||
|
/// <param name="section">Section to be created.</param>
|
||||||
|
/// <param name="key">Key to be added.</param>
|
||||||
|
/// <param name="value">Value to be added.</param>
|
||||||
|
/// <param name="convertValueToLower">Specifies whether the key and value should be saved in lower case letters.</param>
|
||||||
|
/// <returns>Returns the new created section with key value pair.</returns>
|
||||||
|
private List<string> AddSection(List<string> iniFileContent, string section, string key, string value, bool convertValueToLower)
|
||||||
|
{
|
||||||
|
if (convertValueToLower == true)
|
||||||
|
{
|
||||||
|
value = value.ToLower();
|
||||||
|
}
|
||||||
|
|
||||||
|
iniFileContent.Add(section);
|
||||||
|
iniFileContent.Add($"{key}={value}");
|
||||||
|
return iniFileContent;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void WriteFile(List<string> content)
|
||||||
|
{
|
||||||
|
StreamWriter writer = new StreamWriter(_File);
|
||||||
|
foreach (var item in content)
|
||||||
|
{
|
||||||
|
writer.WriteLine(item);
|
||||||
|
}
|
||||||
|
writer.Close();
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
||||||
|
}
|
|
@ -31,9 +31,14 @@
|
||||||
<ColumnDefinition />
|
<ColumnDefinition />
|
||||||
<ColumnDefinition Width="10" />
|
<ColumnDefinition Width="10" />
|
||||||
</Grid.ColumnDefinitions>
|
</Grid.ColumnDefinitions>
|
||||||
<TextBox Grid.Row="1" Grid.Column="1" adonisExtensions:WatermarkExtension.Watermark="Proxy Port (default: 6969)" TextChanged="OnChangePort" />
|
<TextBox PreviewTextInput="TextBox_PreviewTextInput" Grid.Row="2" Grid.Column="1" adonisExtensions:WatermarkExtension.Watermark="Proxy Port (default: 6969)" TextChanged="OnChangePort" Margin="0,10,0,0" Grid.RowSpan="2" />
|
||||||
<Button Grid.Row="3" Grid.Column="1" Click="OnLaunch">Launch</Button>
|
<Button x:Name="LaunchButton" Grid.Row="4" Grid.Column="1" Click="OnLaunch" IsEnabled="False" Margin="0,10,0,0" Grid.RowSpan="2">Launch</Button>
|
||||||
<Button Grid.Row="5" Grid.Column="1" Click="OnDashboard">Open Dashboard</Button>
|
<Button x:Name="LoginButton" Grid.Row="7" Grid.Column="1" Click="OnLoginUsingDiscord">Log in using Discord</Button>
|
||||||
<Button Grid.Row="7" Grid.Column="1" Click="OnLoginUsingDiscord">Log in using Discord</Button>
|
<Ellipse x:Name="PFPContainer" MouseLeftButtonDown="OnDashboard" RenderOptions.BitmapScalingMode="HighQuality" Margin="0,8,195,211" Grid.Column="1" Stretch="Uniform" Width="{Binding Source.PixelWidth, RelativeSource={RelativeSource Self}}" Height="{Binding Source.PixelHeight, RelativeSource={RelativeSource Self}}" Visibility="Hidden">
|
||||||
|
<Ellipse.Fill>
|
||||||
|
<ImageBrush x:Name="ProfilePictureImage" ImageSource="https://cdn.discordapp.com/avatars/419224403415662592/b9486673bd2a484314a1e718c9d9c57b.webp" Stretch="UniformToFill"/>
|
||||||
|
</Ellipse.Fill>
|
||||||
|
</Ellipse>
|
||||||
|
<TextBlock x:Name="UsernameTextBlock" HorizontalAlignment="Left" Margin="43,16,0,0" TextWrapping="Wrap" Text="AveryMadness (@averymadness)" VerticalAlignment="Top" Grid.Column="1" Visibility="Hidden"/>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Window>
|
</Window>
|
||||||
|
|
|
@ -1,12 +1,16 @@
|
||||||
using Newtonsoft.Json;
|
using Newtonsoft.Json;
|
||||||
|
using Newtonsoft.Json.Linq;
|
||||||
using Partypacker.Core;
|
using Partypacker.Core;
|
||||||
using Partypacker.Net;
|
using Partypacker.Net;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using System.Text.RegularExpressions;
|
||||||
using System.Web;
|
using System.Web;
|
||||||
using System.Windows;
|
using System.Windows;
|
||||||
using System.Windows.Controls;
|
using System.Windows.Controls;
|
||||||
|
using System.Windows.Input;
|
||||||
|
using System.Windows.Media;
|
||||||
using WatsonWebserver;
|
using WatsonWebserver;
|
||||||
|
|
||||||
namespace Partypacker
|
namespace Partypacker
|
||||||
|
@ -22,6 +26,7 @@ namespace Partypacker
|
||||||
static string Token;
|
static string Token;
|
||||||
static UserDetailObject UserDetails;
|
static UserDetailObject UserDetails;
|
||||||
static Server sv;
|
static Server sv;
|
||||||
|
static INIFile settings = new("settings.ini", true);
|
||||||
|
|
||||||
public MainWindow()
|
public MainWindow()
|
||||||
{
|
{
|
||||||
|
@ -37,6 +42,11 @@ namespace Partypacker
|
||||||
}
|
}
|
||||||
|
|
||||||
DiscordAuthURL = DiscordURL.Value;
|
DiscordAuthURL = DiscordURL.Value;
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(settings.GetValue("Launcher", "token")))
|
||||||
|
{
|
||||||
|
AutoLogin();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OnApplicationExit(object sender, ExitEventArgs e) => Proxx?.StopProxy();
|
void OnApplicationExit(object sender, ExitEventArgs e) => Proxx?.StopProxy();
|
||||||
|
@ -47,9 +57,59 @@ namespace Partypacker
|
||||||
Port = P;
|
Port = P;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OnDashboard(object sender, RoutedEventArgs e) => Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = PartypackServer.DashboardURL });
|
void OnDashboard(object sender, MouseButtonEventArgs e) => Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = PartypackServer.DashboardURL + "/profile"});
|
||||||
|
|
||||||
private static async Task DefaultRoute(HttpContext ctx)
|
private void UpdateUserUI()
|
||||||
|
{
|
||||||
|
this.Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
UsernameTextBlock.Text = @$"{UserDetails.GlobalName} (@{UserDetails.Username})";
|
||||||
|
ProfilePictureImage.ImageSource = (ImageSource)new ImageSourceConverter().ConvertFromString(UserDetails.Avatar);
|
||||||
|
UsernameTextBlock.Visibility = Visibility.Visible;
|
||||||
|
PFPContainer.Visibility = Visibility.Visible;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void AutoLogin()
|
||||||
|
{
|
||||||
|
Token = settings.GetValue("Launcher", "token");
|
||||||
|
UserDetails = JsonConvert.DeserializeObject<UserDetailObject>(Encoding.UTF8.GetString(Convert.FromHexString(HttpUtility.UrlDecode(settings.GetValue("Launcher", "user")))));
|
||||||
|
UpdateUserUI();
|
||||||
|
Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
LaunchButton.IsEnabled = true;
|
||||||
|
});
|
||||||
|
ConvertLoginToLogout();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ConvertLoginToLogout()
|
||||||
|
{
|
||||||
|
Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
LoginButton.Content = "Log Out";
|
||||||
|
LoginButton.Click -= OnLoginUsingDiscord;
|
||||||
|
LoginButton.Click += OnLogout;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void OnLogout(object sender, RoutedEventArgs e)
|
||||||
|
{
|
||||||
|
this.Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
UsernameTextBlock.Visibility = Visibility.Hidden;
|
||||||
|
PFPContainer.Visibility = Visibility.Hidden;
|
||||||
|
UserDetails = null;
|
||||||
|
Token = "";
|
||||||
|
settings.SetValue("Launcher", "user", "");
|
||||||
|
settings.SetValue("Launcher", "token", "");
|
||||||
|
|
||||||
|
LoginButton.Content = "Log in using Discord";
|
||||||
|
LoginButton.Click += OnLoginUsingDiscord;
|
||||||
|
LoginButton.Click -= OnLogout;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task DefaultRoute(HttpContext ctx)
|
||||||
{
|
{
|
||||||
string _Token = ctx.Request.Query.Elements["token"];
|
string _Token = ctx.Request.Query.Elements["token"];
|
||||||
string _UserDetails = ctx.Request.Query.Elements["user"];
|
string _UserDetails = ctx.Request.Query.Elements["user"];
|
||||||
|
@ -62,21 +122,34 @@ namespace Partypacker
|
||||||
|
|
||||||
Token = _Token;
|
Token = _Token;
|
||||||
UserDetails = JsonConvert.DeserializeObject<UserDetailObject>(Encoding.UTF8.GetString(Convert.FromHexString(HttpUtility.UrlDecode(_UserDetails))));
|
UserDetails = JsonConvert.DeserializeObject<UserDetailObject>(Encoding.UTF8.GetString(Convert.FromHexString(HttpUtility.UrlDecode(_UserDetails))));
|
||||||
|
settings.SetValue("Launcher", "user", HttpUtility.UrlDecode(_UserDetails));
|
||||||
|
settings.SetValue("Launcher", "token", Token);
|
||||||
|
UpdateUserUI();
|
||||||
|
Dispatcher.Invoke(() =>
|
||||||
|
{
|
||||||
|
LaunchButton.IsEnabled = true;
|
||||||
|
});
|
||||||
|
ConvertLoginToLogout();
|
||||||
|
|
||||||
await ctx.Response.Send("All done! You can close this tab now.");
|
await ctx.Response.Send("All done! You can close this tab now.");
|
||||||
sv.Stop();
|
sv.Stop();
|
||||||
|
sv = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
void OnLoginUsingDiscord(object sender, RoutedEventArgs e)
|
void OnLoginUsingDiscord(object sender, RoutedEventArgs e)
|
||||||
{
|
{
|
||||||
Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = $"{DiscordAuthURL}&state={HttpUtility.UrlEncode(Convert.ToBase64String(Encoding.UTF8.GetBytes(JsonConvert.SerializeObject(new
|
Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = $@"{DiscordAuthURL}&state={HttpUtility.UrlEncode(Convert.ToBase64String(Encoding.UTF8.GetBytes(JsonConvert.SerializeObject(new
|
||||||
{
|
{
|
||||||
Client = "PartypackerDesktop"
|
Client = "PartypackerDesktop"
|
||||||
}))))}" });
|
|
||||||
|
|
||||||
|
}))))}"});
|
||||||
|
|
||||||
|
if (sv == null)
|
||||||
|
{
|
||||||
sv = new Server("127.0.0.1", 14968, false, DefaultRoute);
|
sv = new Server("127.0.0.1", 14968, false, DefaultRoute);
|
||||||
sv.Start();
|
sv.Start();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void OnLaunch(object sender, RoutedEventArgs e)
|
void OnLaunch(object sender, RoutedEventArgs e)
|
||||||
{
|
{
|
||||||
|
@ -87,5 +160,11 @@ namespace Partypacker
|
||||||
|
|
||||||
//Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = "com.epicgames.launcher://apps/fn%3A4fe75bbc5a674f4f9b356b5c90567da5%3AFortnite?action=launch&silent=true" });
|
//Process.Start(new ProcessStartInfo { UseShellExecute = true, FileName = "com.epicgames.launcher://apps/fn%3A4fe75bbc5a674f4f9b356b5c90567da5%3AFortnite?action=launch&silent=true" });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void TextBox_PreviewTextInput(object sender, TextCompositionEventArgs e)
|
||||||
|
{
|
||||||
|
Regex regex = new Regex("[^0-9]+");
|
||||||
|
e.Handled = regex.IsMatch(e.Text);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
6
Partypacker/global.json
Normal file
6
Partypacker/global.json
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
{
|
||||||
|
"sdk": {
|
||||||
|
"version": "7.0.405",
|
||||||
|
"rollForward": "latestFeature"
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user