2022-07-28 18:41:34 -04:00
|
|
|
using Avalonia.Collections;
|
|
|
|
using Avalonia.Controls;
|
|
|
|
using Avalonia.Threading;
|
|
|
|
using LibHac.Common;
|
|
|
|
using LibHac.Fs;
|
|
|
|
using LibHac.Fs.Fsa;
|
|
|
|
using LibHac.FsSystem;
|
|
|
|
using LibHac.Tools.Fs;
|
|
|
|
using LibHac.Tools.FsSystem;
|
|
|
|
using LibHac.Tools.FsSystem.NcaUtils;
|
2022-11-25 06:41:34 -05:00
|
|
|
using LibHac.Tools.FsSystem.Save;
|
2022-07-28 18:41:34 -04:00
|
|
|
using Ryujinx.Ava.Common.Locale;
|
|
|
|
using Ryujinx.Ava.Ui.Controls;
|
|
|
|
using Ryujinx.Ava.Ui.Models;
|
|
|
|
using Ryujinx.Common.Configuration;
|
|
|
|
using Ryujinx.Common.Utilities;
|
|
|
|
using Ryujinx.HLE.FileSystem;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2022-11-25 06:41:34 -05:00
|
|
|
using System.Collections.ObjectModel;
|
|
|
|
using System.ComponentModel;
|
2022-07-28 18:41:34 -04:00
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
2022-11-25 06:41:34 -05:00
|
|
|
using System.Reactive.Linq;
|
2022-07-28 18:41:34 -04:00
|
|
|
using System.Text;
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
using Path = System.IO.Path;
|
|
|
|
|
|
|
|
namespace Ryujinx.Ava.Ui.Windows
|
|
|
|
{
|
|
|
|
public partial class DownloadableContentManagerWindow : StyleableWindow
|
|
|
|
{
|
|
|
|
private readonly List<DownloadableContentContainer> _downloadableContentContainerList;
|
2022-11-25 06:41:34 -05:00
|
|
|
private readonly string _downloadableContentJsonPath;
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
private VirtualFileSystem _virtualFileSystem { get; }
|
|
|
|
private AvaloniaList<DownloadableContentModel> _downloadableContents { get; set; }
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
private ulong TitleId { get; }
|
|
|
|
private string TitleName { get; }
|
2022-07-28 18:41:34 -04:00
|
|
|
|
|
|
|
public DownloadableContentManagerWindow()
|
|
|
|
{
|
|
|
|
DataContext = this;
|
|
|
|
|
|
|
|
InitializeComponent();
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
Title = $"Ryujinx {Program.Version} - {LocaleManager.Instance["DlcWindowTitle"]} - {TitleName} ({TitleId:X16})";
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public DownloadableContentManagerWindow(VirtualFileSystem virtualFileSystem, ulong titleId, string titleName)
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
_virtualFileSystem = virtualFileSystem;
|
|
|
|
_downloadableContents = new AvaloniaList<DownloadableContentModel>();
|
|
|
|
TitleId = titleId;
|
|
|
|
TitleName = titleName;
|
2022-07-28 18:41:34 -04:00
|
|
|
|
|
|
|
_downloadableContentJsonPath = Path.Combine(AppDataManager.GamesDirPath, titleId.ToString("x16"), "dlc.json");
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
_downloadableContentContainerList = JsonHelper.DeserializeFromFile<List<DownloadableContentContainer>>(_downloadableContentJsonPath);
|
|
|
|
}
|
|
|
|
catch
|
|
|
|
{
|
|
|
|
_downloadableContentContainerList = new List<DownloadableContentContainer>();
|
|
|
|
}
|
|
|
|
|
|
|
|
DataContext = this;
|
|
|
|
|
|
|
|
InitializeComponent();
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
RemoveButton.IsEnabled = false;
|
|
|
|
|
|
|
|
DlcDataGrid.SelectionChanged += DlcDataGrid_SelectionChanged;
|
|
|
|
|
|
|
|
Title = $"Ryujinx {Program.Version} - {LocaleManager.Instance["DlcWindowTitle"]} - {TitleName} ({TitleId:X16})";
|
2022-07-28 18:41:34 -04:00
|
|
|
|
|
|
|
LoadDownloadableContents();
|
2022-11-25 06:41:34 -05:00
|
|
|
PrintHeading();
|
|
|
|
}
|
|
|
|
|
|
|
|
private void DlcDataGrid_SelectionChanged(object sender, SelectionChangedEventArgs e)
|
|
|
|
{
|
|
|
|
RemoveButton.IsEnabled = (DlcDataGrid.SelectedItems.Count > 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void PrintHeading()
|
|
|
|
{
|
|
|
|
Heading.Text = string.Format(LocaleManager.Instance["DlcWindowHeading"], _downloadableContents.Count, TitleName, TitleId.ToString("X16"));
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
private void LoadDownloadableContents()
|
|
|
|
{
|
|
|
|
foreach (DownloadableContentContainer downloadableContentContainer in _downloadableContentContainerList)
|
|
|
|
{
|
|
|
|
if (File.Exists(downloadableContentContainer.ContainerPath))
|
|
|
|
{
|
|
|
|
using FileStream containerFile = File.OpenRead(downloadableContentContainer.ContainerPath);
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
PartitionFileSystem pfs = new(containerFile.AsStorage());
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
_virtualFileSystem.ImportTickets(pfs);
|
2022-07-28 18:41:34 -04:00
|
|
|
|
|
|
|
foreach (DownloadableContentNca downloadableContentNca in downloadableContentContainer.DownloadableContentNcaList)
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
using UniqueRef<IFile> ncaFile = new();
|
2022-07-28 18:41:34 -04:00
|
|
|
|
|
|
|
pfs.OpenFile(ref ncaFile.Ref(), downloadableContentNca.FullPath.ToU8Span(), OpenMode.Read).ThrowIfFailure();
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
Nca nca = TryOpenNca(ncaFile.Get.AsStorage(), downloadableContentContainer.ContainerPath);
|
2022-07-28 18:41:34 -04:00
|
|
|
if (nca != null)
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
_downloadableContents.Add(new DownloadableContentModel(nca.Header.TitleId.ToString("X16"),
|
|
|
|
downloadableContentContainer.ContainerPath,
|
|
|
|
downloadableContentNca.FullPath,
|
|
|
|
downloadableContentNca.Enabled));
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: Save the list again to remove leftovers.
|
|
|
|
Save();
|
|
|
|
}
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
private Nca TryOpenNca(IStorage ncaStorage, string containerPath)
|
2022-07-28 18:41:34 -04:00
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
return new Nca(_virtualFileSystem.KeySet, ncaStorage);
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
catch (Exception ex)
|
|
|
|
{
|
|
|
|
Dispatcher.UIThread.InvokeAsync(async () =>
|
|
|
|
{
|
|
|
|
await ContentDialogHelper.CreateErrorDialog(string.Format(LocaleManager.Instance["DialogDlcLoadNcaErrorMessage"], ex.Message, containerPath));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
private async Task AddDownloadableContent(string path)
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
if (!File.Exists(path) || _downloadableContents.FirstOrDefault(x => x.ContainerPath == path) != null)
|
2022-07-28 18:41:34 -04:00
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
using FileStream containerFile = File.OpenRead(path);
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
PartitionFileSystem partitionFileSystem = new(containerFile.AsStorage());
|
|
|
|
bool containsDownloadableContent = false;
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
_virtualFileSystem.ImportTickets(partitionFileSystem);
|
|
|
|
|
|
|
|
foreach (DirectoryEntryEx fileEntry in partitionFileSystem.EnumerateEntries("/", "*.nca"))
|
|
|
|
{
|
|
|
|
using var ncaFile = new UniqueRef<IFile>();
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
partitionFileSystem.OpenFile(ref ncaFile.Ref(), fileEntry.FullPath.ToU8Span(), OpenMode.Read).ThrowIfFailure();
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
Nca nca = TryOpenNca(ncaFile.Get.AsStorage(), path);
|
|
|
|
if (nca == null)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
if (nca.Header.ContentType == NcaContentType.PublicData)
|
|
|
|
{
|
|
|
|
if ((nca.Header.TitleId & 0xFFFFFFFFFFFFE000) != TitleId)
|
2022-07-28 18:41:34 -04:00
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
break;
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
_downloadableContents.Add(new DownloadableContentModel(nca.Header.TitleId.ToString("X16"), path, fileEntry.FullPath, true));
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
containsDownloadableContent = true;
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
2022-11-25 06:41:34 -05:00
|
|
|
}
|
2022-07-28 18:41:34 -04:00
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
if (!containsDownloadableContent)
|
|
|
|
{
|
|
|
|
await ContentDialogHelper.CreateErrorDialog(LocaleManager.Instance["DialogDlcNoDlcErrorMessage"]);
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void RemoveDownloadableContents(bool removeSelectedOnly = false)
|
|
|
|
{
|
|
|
|
if (removeSelectedOnly)
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
AvaloniaList<DownloadableContentModel> removedItems = new();
|
|
|
|
|
|
|
|
foreach (var item in DlcDataGrid.SelectedItems)
|
|
|
|
{
|
|
|
|
removedItems.Add(item as DownloadableContentModel);
|
|
|
|
}
|
|
|
|
|
|
|
|
DlcDataGrid.SelectedItems.Clear();
|
|
|
|
|
|
|
|
foreach (var item in removedItems)
|
|
|
|
{
|
|
|
|
_downloadableContents.RemoveAll(_downloadableContents.Where(x => x.TitleId == item.TitleId).ToList());
|
|
|
|
}
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-11-25 06:41:34 -05:00
|
|
|
_downloadableContents.Clear();
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
2022-11-25 06:41:34 -05:00
|
|
|
|
|
|
|
PrintHeading();
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveSelected()
|
|
|
|
{
|
|
|
|
RemoveDownloadableContents(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveAll()
|
|
|
|
{
|
|
|
|
RemoveDownloadableContents();
|
|
|
|
}
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
public void EnableAll()
|
|
|
|
{
|
|
|
|
foreach(var item in _downloadableContents)
|
|
|
|
{
|
|
|
|
item.Enabled = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void DisableAll()
|
|
|
|
{
|
|
|
|
foreach (var item in _downloadableContents)
|
|
|
|
{
|
|
|
|
item.Enabled = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-28 18:41:34 -04:00
|
|
|
public async void Add()
|
|
|
|
{
|
|
|
|
OpenFileDialog dialog = new OpenFileDialog()
|
|
|
|
{
|
|
|
|
Title = LocaleManager.Instance["SelectDlcDialogTitle"],
|
|
|
|
AllowMultiple = true
|
|
|
|
};
|
|
|
|
|
|
|
|
dialog.Filters.Add(new FileDialogFilter
|
|
|
|
{
|
|
|
|
Name = "NSP",
|
|
|
|
Extensions = { "nsp" }
|
|
|
|
});
|
|
|
|
|
|
|
|
string[] files = await dialog.ShowAsync(this);
|
|
|
|
|
|
|
|
if (files != null)
|
|
|
|
{
|
|
|
|
foreach (string file in files)
|
|
|
|
{
|
|
|
|
await AddDownloadableContent(file);
|
|
|
|
}
|
|
|
|
}
|
2022-11-25 06:41:34 -05:00
|
|
|
|
|
|
|
PrintHeading();
|
2022-07-28 18:41:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public void Save()
|
|
|
|
{
|
|
|
|
_downloadableContentContainerList.Clear();
|
|
|
|
|
|
|
|
DownloadableContentContainer container = default;
|
|
|
|
|
2022-11-25 06:41:34 -05:00
|
|
|
foreach (DownloadableContentModel downloadableContent in _downloadableContents)
|
2022-07-28 18:41:34 -04:00
|
|
|
{
|
|
|
|
if (container.ContainerPath != downloadableContent.ContainerPath)
|
|
|
|
{
|
|
|
|
if (!string.IsNullOrWhiteSpace(container.ContainerPath))
|
|
|
|
{
|
|
|
|
_downloadableContentContainerList.Add(container);
|
|
|
|
}
|
|
|
|
|
|
|
|
container = new DownloadableContentContainer
|
|
|
|
{
|
|
|
|
ContainerPath = downloadableContent.ContainerPath,
|
|
|
|
DownloadableContentNcaList = new List<DownloadableContentNca>()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
container.DownloadableContentNcaList.Add(new DownloadableContentNca
|
|
|
|
{
|
|
|
|
Enabled = downloadableContent.Enabled,
|
|
|
|
TitleId = Convert.ToUInt64(downloadableContent.TitleId, 16),
|
|
|
|
FullPath = downloadableContent.FullPath
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(container.ContainerPath))
|
|
|
|
{
|
|
|
|
_downloadableContentContainerList.Add(container);
|
|
|
|
}
|
|
|
|
|
|
|
|
using (FileStream downloadableContentJsonStream = File.Create(_downloadableContentJsonPath, 4096, FileOptions.WriteThrough))
|
|
|
|
{
|
|
|
|
downloadableContentJsonStream.Write(Encoding.UTF8.GetBytes(JsonHelper.Serialize(_downloadableContentContainerList, true)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SaveAndClose()
|
|
|
|
{
|
|
|
|
Save();
|
|
|
|
Close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|