PKHeX/PKHeX.Core/Saves/Substructures/Battle Videos/BVRequestUtil.cs
Kurt 88830e0d00
Update from .NET Framework 4.6 to .NET 7 (#3729)
Updates from net46->net7, dropping support for mono in favor of using the latest runtime (along with the performance/API improvements). Releases will be posted as 64bit only for now.

Refactors a good amount of internal API methods to be more performant and more customizable for future updates & fixes.

Adds functionality for Batch Editor commands to `>`, `<` and <=/>=

TID/SID properties renamed to TID16/SID16 for clarity; other properties exposed for Gen7 / display variants.

Main window has a new layout to account for DPI scaling (8 point grid)

Fixed: Tatsugiri and Paldean Tauros now output Showdown form names as Showdown expects
Changed: Gen9 species now interact based on the confirmed National Dex IDs (closes #3724)
Fixed: Pokedex set all no longer clears species with unavailable non-base forms (closes #3720)
Changed: Hyper Training suggestions now apply for level 50 in SV. (closes #3714)
Fixed: B2/W2 hatched egg met locations exclusive to specific versions are now explicitly checked (closes #3691)
Added: Properties for ribbon/mark count (closes #3659)
Fixed: Traded SV eggs are now checked correctly (closes #3692)
2023-01-21 20:02:33 -08:00

109 lines
3.1 KiB
C#

using System;
using System.Diagnostics;
using static System.Buffers.Binary.BinaryPrimitives;
namespace PKHeX.Core;
public static class BVRequestUtil
{
public static string GetSMBattleVideoURL(string code)
{
code = code.Replace("-", string.Empty);
Debug.Assert(code.Length == 16);
var video_id = StrToU64(code, out bool valid);
if (!valid)
return string.Empty;
return $"https://ctr-bnda-live.s3.amazonaws.com/10.CTR_BNDA_datastore/ds/1/data/{video_id:D11}-00001"; // Sun datastore
}
public static ulong StrToU64(ReadOnlySpan<char> input, out bool valid)
{
var chk = Pull(input[..4]) >> 4; // first four chars are checksum bits
var result = Pull(input[4..]); // next 12 chars are the 70 value bits
Span<byte> temp = stackalloc byte[8];
WriteUInt64LittleEndian(temp, result);
var actual = Checksums.CRC16_CCITT(temp);
valid = chk == actual;
return result;
static ulong Pull(ReadOnlySpan<char> input)
{
ulong val = 0;
foreach (char c in input)
{
if (c == '-')
continue;
val <<= 5;
val |= Get5BitFromChar(c) & 0b11111;
}
return val;
}
}
public static string U64ToStr(ulong input, bool insertDash)
{
Span<byte> temp = stackalloc byte[8];
WriteUInt64LittleEndian(temp, input);
uint chk = Checksums.CRC16_CCITT(temp);
Span<char> buff = stackalloc char[16];
int ctr = 15;
Push(buff, ref ctr, 12, input); // store value bits
Push(buff, ref ctr, 04, chk << 4); // store checksum bits
return !insertDash ? new string(buff) : GetStringWithDashesEvery(buff, 4);
static void Push(Span<char> buff, ref int ctr, int bit5Chunks, ulong value)
{
for (int i = 0; i < bit5Chunks; i++)
{
buff[ctr--] = Set5BitToChar((char)(value & 0x1F));
value >>= 5;
}
}
}
private static string GetStringWithDashesEvery(ReadOnlySpan<char> buff, int spacer)
{
Span<char> buff2 = stackalloc char[buff.Length + ((buff.Length / spacer) - 1)];
for (int i = 0, ctr = 0; i < buff.Length; i++)
{
buff2[ctr++] = buff[i];
if (i % spacer == 3 && ctr < buff2.Length)
buff2[ctr++] = '-'; // add dash between every chunk of size {spacer}
}
return new string(buff2);
}
private static char Set5BitToChar(char c)
{
var shift = c > 9 ? '7' : '0';
c += shift;
return MapToChar(c);
}
private static uint Get5BitFromChar(char c)
{
c = MapFromChar(c);
var shift = c >= 'A' ? '7' : '0';
return (uint)(c - shift);
}
private static char MapToChar(char c) => c switch
{
'0' => 'W',
'1' => 'X',
'I' => 'Y',
'O' => 'Z',
_ => c,
};
private static char MapFromChar(char c) => c switch
{
'W' => '0',
'X' => '1',
'Y' => 'I',
'Z' => 'O',
_ => c,
};
}