mirror of
https://github.com/mm201/pkmn-classic-framework.git
synced 2026-03-22 09:54:09 -05:00
592 lines
25 KiB
C#
592 lines
25 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Text;
|
|
using System.IO;
|
|
using System.Net.Sockets;
|
|
using System.Threading;
|
|
using MySql.Data.MySqlClient;
|
|
using System.Configuration;
|
|
using PkmnFoundations.Data;
|
|
using PkmnFoundations.Support;
|
|
using System.Data;
|
|
using System.Net.Security;
|
|
using System.Security.Cryptography.X509Certificates;
|
|
|
|
namespace PkmnFoundations
|
|
{
|
|
public class BvCrawler5
|
|
{
|
|
public static void Main(string[] args)
|
|
{
|
|
m_upload_dir = ConfigurationManager.AppSettings["pkmnFoundationsBoxUpload5Dir"];
|
|
|
|
Console.WriteLine("Pokémon BW/BW2 Battle Video Crawler by mm201");
|
|
int pid = 330241374; // White 1 Jenny (?)
|
|
Directory.CreateDirectory(String.Format("{0}", m_upload_dir));
|
|
DateTime last_top30 = DateTime.MinValue;
|
|
DateTime last_top_link = DateTime.MinValue;
|
|
DateTime last_top_subway = DateTime.MinValue;
|
|
DateTime last_retry_all = DateTime.MinValue;
|
|
|
|
m_session_key = new byte[]{
|
|
0x66, 0x87, 0xF1, 0xB5, 0x96, 0x47, 0x4D, 0xFB,
|
|
0x0E, 0x0B, 0x19, 0xBD, 0xBD, 0x69, 0x5E, 0x71,
|
|
0x03, 0x39, 0xED, 0xB2, 0x38, 0xA7, 0xD5, 0x5A,
|
|
0x19, 0x80, 0x09, 0xD4, 0xAA, 0x7F, 0xAE, 0xD3,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
|
};
|
|
|
|
m_session_time = DateTime.MinValue;
|
|
|
|
while (true)
|
|
{
|
|
ulong videoId;
|
|
try
|
|
{
|
|
using (MySqlConnection db = CreateConnection())
|
|
{
|
|
db.Open();
|
|
videoId = DequeueVideo(db);
|
|
db.Close();
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
// haven't touched the server, sleep short
|
|
LogError(ex);
|
|
Thread.Sleep(1000 * 1);
|
|
continue;
|
|
}
|
|
|
|
if (videoId == 0)
|
|
{
|
|
try
|
|
{
|
|
if (last_retry_all < DateTime.Now.AddHours(-6))
|
|
{
|
|
last_retry_all = DateTime.Now;
|
|
RetryAll();
|
|
continue;
|
|
}
|
|
if (last_top30 < DateTime.Now.AddMinutes(-60))
|
|
{
|
|
last_top30 = DateTime.Now;
|
|
QueueSpecial(pid, SearchSpecial.Latest30);
|
|
continue;
|
|
}
|
|
if (last_top_link < DateTime.Now.AddMinutes(-120))
|
|
{
|
|
last_top_link = DateTime.Now;
|
|
QueueSpecial(pid, SearchSpecial.TopLinkBattles);
|
|
continue;
|
|
}
|
|
if (last_top_subway < DateTime.Now.AddMinutes(-120))
|
|
{
|
|
last_top_subway = DateTime.Now;
|
|
QueueSpecial(pid, SearchSpecial.TopSubwayBattles);
|
|
continue;
|
|
}
|
|
else if (RunSearch(pid))
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
Thread.Sleep(1000 * 30);
|
|
continue;
|
|
}
|
|
|
|
Console.WriteLine("Nothing to do. Idling 1 minute.");
|
|
Thread.Sleep(1000 * 60);
|
|
continue;
|
|
}
|
|
|
|
String formatted = FormatVideoId(videoId);
|
|
String filename = String.Format("{0}\\{1}.bin", m_upload_dir, formatted);
|
|
|
|
if (File.Exists(filename))
|
|
{
|
|
Console.WriteLine("Skipped video {0}. Already present on disk.", formatted);
|
|
Thread.Sleep(1000 * 1);
|
|
continue;
|
|
}
|
|
|
|
byte[] data;
|
|
try
|
|
{
|
|
data = GetBattleVideo(pid, videoId);
|
|
|
|
using (FileStream file = File.Create(filename))
|
|
{
|
|
file.Write(data, 0, data.Length);
|
|
file.Close();
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
LogError(ex);
|
|
Thread.Sleep(1000 * 30);
|
|
continue;
|
|
}
|
|
|
|
Console.WriteLine("Successfully saved battle video {0}.", formatted);
|
|
Thread.Sleep(1000 * 30);
|
|
}
|
|
}
|
|
|
|
private static String m_upload_dir;
|
|
private static byte[] m_session_key;
|
|
private static String m_session_str;
|
|
private static DateTime m_session_time;
|
|
|
|
public static String FormatVideoId(ulong videoId)
|
|
{
|
|
String number = videoId.ToString("D12");
|
|
String[] split = new String[3];
|
|
split[0] = number.Substring(0, number.Length - 10);
|
|
split[1] = number.Substring(number.Length - 10, 5);
|
|
split[2] = number.Substring(number.Length - 5, 5);
|
|
return String.Join("-", split);
|
|
}
|
|
|
|
public static void WriteBase64shit(Stream s)
|
|
{
|
|
// First 32 bytes shared by all:
|
|
// 6687F1B596474DFB0E0B19BDBD695E710339EDB238A7D55A198009D4AA7FAED3
|
|
//
|
|
// The last 32 bytes seem to be random but verifiable:
|
|
// White 1: A194196DAC7CAA4B75A9038E0FF2C71E64E61C40CFA0340178FBB9B6B72C4A63
|
|
// White 1: F8A9E7EA602F169A146429A49AB7BF2D26BBB178AA2BCF1DA259310A263D41ED
|
|
// White 1: BEAB158B25170200BEEBD9C456CD5BE00A19A3C4F69F4749DED6427AD173834E
|
|
// Black 2: C307898C422687861BCA69B25C9FC007B6516A098A05E027BB49764EF2E8B5B1
|
|
// Black 2: BFAA5408C6474EDDE340CF57D3405379E61A78B331191637AA6CE2818ECCE42B
|
|
//
|
|
// I'm guessing it's Hash(Secret + Random)
|
|
// The bytes don't vary with a session but are changed when you reset your DS.
|
|
|
|
if (m_session_str == null || m_session_time.AddMinutes(60) < DateTime.Now)
|
|
{
|
|
m_session_time = DateTime.Now;
|
|
Random rnd = new Random();
|
|
byte[] data = new byte[32];
|
|
|
|
rnd.NextBytes(data);
|
|
|
|
// hack in
|
|
data = new byte[]{
|
|
0xBE, 0xAB, 0x15, 0x8B, 0x25, 0x17, 0x02, 0x00,
|
|
0xBE, 0xEB, 0xD9, 0xC4, 0x56, 0xCD, 0x5B, 0xE0,
|
|
0x0A, 0x19, 0xA3, 0xC4, 0xF6, 0x9F, 0x47, 0x49,
|
|
0xDE, 0xD6, 0x42, 0x7A, 0xD1, 0x73, 0x83, 0x4E
|
|
};
|
|
|
|
Array.Copy(data, 0, m_session_key, 32, 32);
|
|
m_session_str = Convert.ToBase64String(m_session_key);
|
|
}
|
|
|
|
StreamWriter w = new StreamWriter(s);
|
|
w.Write(m_session_str);
|
|
w.Flush();
|
|
}
|
|
|
|
public static byte[] GetBattleVideo(int pid, ulong videoId)
|
|
{
|
|
String formatted = FormatVideoId(videoId);
|
|
Console.WriteLine("Attempting to retrieve battle video {0} from server.", formatted);
|
|
|
|
byte[] data = new byte[0x14c];
|
|
MemoryStream request = new MemoryStream(data);
|
|
request.Write(new byte[4], 0, 4); // length goes here, see end
|
|
request.Write(new byte[] { 0xf2, 0x55, 0x00, 0x00 }, 0, 4); // request type, sanity 0000
|
|
request.Write(BitConverter.GetBytes(pid), 0, 4); // pid, hopefully this doesn't ban me
|
|
request.Write(new byte[] { 0x17, 0x02 }, 0, 2);
|
|
|
|
WriteBase64shit(request);
|
|
|
|
request.Write(new byte[0xda], 0, 0xda);
|
|
|
|
request.Write(BitConverter.GetBytes(videoId), 0, 8);
|
|
request.Write(new byte[] { 0x64, 0x00, 0x00, 0x00 }, 0, 4);
|
|
request.Flush();
|
|
PutLength(data);
|
|
|
|
byte[] response = Conversation(data);
|
|
|
|
if (response.Length < 9) throw new InvalidDataException("Battle video was not retrieved.");
|
|
|
|
Console.WriteLine("Successfully retrieved {0} byte response for battle video {1}.", response.Length, formatted);
|
|
return response;
|
|
}
|
|
|
|
public static bool RunSearch(int pid)
|
|
{
|
|
Random rand = new Random();
|
|
|
|
SearchMetagames[] metagames = (SearchMetagames[])Enum.GetValues(typeof(SearchMetagames));
|
|
int metaCount = metagames.Length - 1; // remove None from the list
|
|
int metaIndex = rand.Next(1, metaCount + 1);
|
|
SearchMetagames metagame = metagames[metaIndex];
|
|
|
|
ushort species = (ushort)rand.Next(0, 649);
|
|
byte country = 0xff;
|
|
byte region = 0xff;
|
|
|
|
using (MySqlConnection db = CreateConnection())
|
|
{
|
|
db.Open();
|
|
|
|
if ((long)db.ExecuteScalar(
|
|
"SELECT Count(*) FROM BattleVideoSearchHistory5 WHERE Metagame = @metagame " +
|
|
"AND Species = @species AND Country = @country AND Region = @region AND Special = 0",
|
|
new MySqlParameter("@metagame", (int)metagame),
|
|
new MySqlParameter("@species", (int)species),
|
|
new MySqlParameter("@country", (int)country),
|
|
new MySqlParameter("@region", (int)region))
|
|
== 0)
|
|
{
|
|
// exact match
|
|
QueueSearch(pid, SearchSpecial.None, species, metagame, country, region);
|
|
return true;
|
|
}
|
|
|
|
DataTable dt;
|
|
dt = db.ExecuteDataTable("SELECT DISTINCT Metagame, Species, Country, Region " +
|
|
"FROM BattleVideoSearchHistory5 WHERE Metagame = @metagame AND Special = 0 ORDER BY Species",
|
|
new MySqlParameter("@metagame", (int)metagame));
|
|
if (dt.Rows.Count < 649)
|
|
{
|
|
int prevSpecies = 1;
|
|
foreach (DataRow row in dt.Rows)
|
|
{
|
|
if ((int)row["Species"] != prevSpecies)
|
|
{
|
|
QueueSearch(pid, SearchSpecial.None, (ushort)(prevSpecies), metagame, country, region);
|
|
return true;
|
|
}
|
|
prevSpecies++;
|
|
}
|
|
}
|
|
|
|
dt = db.ExecuteDataTable("SELECT DISTINCT Metagame, Species, Country, Region " +
|
|
"FROM BattleVideoSearchHistory5 WHERE Species = @species AND Special = 0 ORDER BY Metagame",
|
|
new MySqlParameter("@species", (int)species));
|
|
if (dt.Rows.Count < metaCount)
|
|
{
|
|
int prevMeta = 1;
|
|
foreach (DataRow row in dt.Rows)
|
|
{
|
|
if ((int)row["Metagame"] != (int)metagames[prevMeta])
|
|
{
|
|
QueueSearch(pid, SearchSpecial.None, species, metagames[prevMeta], country, region);
|
|
return true;
|
|
}
|
|
prevMeta++;
|
|
}
|
|
}
|
|
|
|
dt = db.ExecuteDataTable("SELECT DISTINCT Metagame, Species, Country, Region " +
|
|
"FROM BattleVideoSearchHistory5 WHERE Special = 0 ORDER BY Metagame, Species");
|
|
if (dt.Rows.Count < 649 * metaCount)
|
|
{
|
|
int prevSpecies = 1;
|
|
int prevMeta = 1;
|
|
foreach (DataRow row in dt.Rows)
|
|
{
|
|
if ((int)row["Species"] != prevSpecies || (int)row["Metagame"] != (int)metagames[prevMeta])
|
|
{
|
|
QueueSearch(pid, SearchSpecial.None, (ushort)(prevSpecies), metagames[prevMeta], country, region);
|
|
return true;
|
|
}
|
|
prevSpecies++;
|
|
if (prevSpecies > 649)
|
|
{
|
|
prevSpecies = 1;
|
|
prevMeta++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public static void QueueSpecial(int pid, SearchSpecial special)
|
|
{
|
|
QueueSearch(pid, special, 0x0000, SearchMetagames.None, 0x00, 0x00);
|
|
}
|
|
|
|
public static void QueueSearch(int pid, SearchSpecial special, ushort species, SearchMetagames meta, byte country, byte region)
|
|
{
|
|
switch (special)
|
|
{
|
|
case SearchSpecial.Latest30:
|
|
Console.WriteLine("Searching for latest 30 videos.");
|
|
break;
|
|
case SearchSpecial.TopLinkBattles:
|
|
Console.WriteLine("Searching for top link battles.");
|
|
break;
|
|
case SearchSpecial.TopSubwayBattles:
|
|
Console.WriteLine("Searching for top Subway battles.");
|
|
break;
|
|
default:
|
|
{
|
|
Console.Write("Searching for ");
|
|
if (species != 0xffff)
|
|
Console.Write("species {0}, ", species);
|
|
if (meta != SearchMetagames.None)
|
|
Console.Write("{0}, ", meta);
|
|
if (country != 0xff)
|
|
Console.Write("country {0}, ", region);
|
|
if (region != 0xff)
|
|
Console.Write("region {0}", region);
|
|
} break;
|
|
}
|
|
|
|
byte[] data = new byte[0x15c];
|
|
MemoryStream request = new MemoryStream(data);
|
|
request.Write(new byte[4], 0, 4); // length goes here, see end
|
|
request.Write(new byte[] { 0xf1, 0x55, 0x00, 0x00 }, 0, 4); // request type, sanity 0000
|
|
request.Write(BitConverter.GetBytes(pid), 0, 4); // pid, hopefully this doesn't ban me
|
|
request.Write(new byte[] { 0x14, 0x02 }, 0, 2);
|
|
|
|
WriteBase64shit(request);
|
|
|
|
request.Write(new byte[0xda], 0, 0xda);
|
|
|
|
request.Write(BitConverter.GetBytes((uint)special), 0, 4);
|
|
request.Write(BitConverter.GetBytes(species), 0, 2);
|
|
request.Write(BitConverter.GetBytes((uint)meta), 0, 4);
|
|
request.WriteByte(country);
|
|
request.WriteByte(region);
|
|
|
|
request.Write(new byte[] {
|
|
0x64, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00
|
|
}, 0, 16);
|
|
request.Flush();
|
|
PutLength(data);
|
|
|
|
byte[] response = Conversation(data);
|
|
|
|
if (special != SearchSpecial.Latest30)
|
|
{
|
|
using (MySqlConnection db = CreateConnection())
|
|
{
|
|
db.Open();
|
|
db.ExecuteNonQuery("INSERT INTO BattleVideoSearchHistory5 (Metagame, Species, " +
|
|
"Country, Region, Special) VALUES (@metagame, @species, @country, @region, @special)",
|
|
new MySqlParameter("@metagame", (int)meta),
|
|
new MySqlParameter("@species", (int)species),
|
|
new MySqlParameter("@country", (int)country),
|
|
new MySqlParameter("@region", (int)region),
|
|
new MySqlParameter("@special", (int)special));
|
|
db.Close();
|
|
}
|
|
}
|
|
|
|
QueueSearchResults(response);
|
|
}
|
|
|
|
public static void QueueSearchResults(byte[] data)
|
|
{
|
|
if (data.Length % 208 != 12) throw new InvalidDataException("Search results blob should be 12 bytes + 208 per result.");
|
|
|
|
int count = data.Length / 208;
|
|
Console.WriteLine("{0} results found.", count);
|
|
|
|
if (count == 0)
|
|
{
|
|
// Nothing found. Sleep as to not spam the server with lots of empty searches
|
|
Thread.Sleep(1000 * 15);
|
|
return;
|
|
}
|
|
|
|
// 12 bytes of header plus 208 bytes per search result.
|
|
using (MySqlConnection db = CreateConnection())
|
|
{
|
|
db.Open();
|
|
for (int x = 0; x < count; x++)
|
|
{
|
|
ulong videoId = BitConverter.ToUInt64(data, 208 + x * 208);
|
|
QueueVideoId(db, videoId);
|
|
}
|
|
db.Close();
|
|
}
|
|
}
|
|
|
|
public static void QueueVideoId(MySqlConnection db, ulong id)
|
|
{
|
|
String formatted = FormatVideoId(id);
|
|
|
|
using (MySqlTransaction tran = db.BeginTransaction())
|
|
{
|
|
long count = (long)tran.ExecuteScalar("SELECT Count(*) FROM BattleVideoCrawlQueue5 WHERE SerialNumber = @serial_number", new MySqlParameter("@serial_number", id));
|
|
if (count > 0)
|
|
{
|
|
tran.Rollback();
|
|
Console.WriteLine("Skipped video {0}. Already present in database.", formatted);
|
|
return;
|
|
}
|
|
tran.ExecuteNonQuery("INSERT INTO BattleVideoCrawlQueue5 (SerialNumber, `Timestamp`) VALUES (@serial_number, NOW())", new MySqlParameter("@serial_number", id));
|
|
tran.Commit();
|
|
Console.WriteLine("Queued video {0}.", formatted);
|
|
}
|
|
}
|
|
|
|
public static ulong DequeueVideo(MySqlConnection db)
|
|
{
|
|
using (MySqlTransaction tran = db.BeginTransaction())
|
|
{
|
|
object o = tran.ExecuteScalar("SELECT SerialNumber FROM BattleVideoCrawlQueue5 WHERE Complete = 0 ORDER BY `Timestamp` LIMIT 1");
|
|
if (o == null || o == DBNull.Value)
|
|
{
|
|
tran.Rollback();
|
|
return 0;
|
|
}
|
|
ulong id = (ulong)o;
|
|
tran.ExecuteNonQuery("UPDATE BattleVideoCrawlQueue5 SET Complete = 1 WHERE SerialNumber = @serial_number", new MySqlParameter("@serial_number", id));
|
|
tran.Commit();
|
|
return id;
|
|
}
|
|
}
|
|
|
|
public static void PutLength(byte[] data)
|
|
{
|
|
// places the actual length in the first 4 bytes.
|
|
byte[] length = BitConverter.GetBytes(data.Length);
|
|
Array.Copy(length, data, 4);
|
|
}
|
|
|
|
public static byte[] Conversation(byte[] request)
|
|
{
|
|
MemoryStream response = new MemoryStream();
|
|
|
|
using (TcpClient client = new TcpClient("pkgdsprod.nintendo.co.jp", 12401))
|
|
{
|
|
SslStream sslClient = new SslStream(client.GetStream(), false, delegate(object s, X509Certificate certificate, X509Chain chain, SslPolicyErrors sslPolicyErrors) { return true; });
|
|
sslClient.AuthenticateAsClient("pkgdsprod.nintendo.co.jp");
|
|
|
|
sslClient.Write(request, 0, request.Length);
|
|
sslClient.CopyTo(response);
|
|
sslClient.Close();
|
|
}
|
|
response.Flush();
|
|
byte[] dataResponse = response.ToArray();
|
|
|
|
int length = BitConverter.ToInt32(dataResponse, 0);
|
|
AssertHelper.Equals(length, dataResponse.Length);
|
|
|
|
return dataResponse;
|
|
}
|
|
|
|
public static void RetryAll()
|
|
{
|
|
String path = String.Format("{0}", m_upload_dir);
|
|
|
|
using (MySqlConnection db = CreateConnection())
|
|
{
|
|
db.Open();
|
|
|
|
DataTable SerialNumbers = db.ExecuteDataTable("SELECT SerialNumber FROM BattleVideoCrawlQueue5 WHERE Complete = 1");
|
|
SerialNumbers.PrimaryKey = new DataColumn[] { SerialNumbers.Columns["SerialNumber"] };
|
|
IEnumerable<String> filenames = Directory.EnumerateFiles(path);
|
|
|
|
foreach (String s in filenames)
|
|
{
|
|
int slash = s.LastIndexOf(Path.DirectorySeparatorChar) + 1;
|
|
int dot = s.LastIndexOf(".");
|
|
if (dot < 0) dot = s.Length;
|
|
if (dot < slash) dot = s.Length;
|
|
|
|
ulong SerialNumber;
|
|
UInt64.TryParse(s.Substring(slash, dot - slash).Replace("-", ""),
|
|
out SerialNumber);
|
|
|
|
if (SerialNumber == 0) continue;
|
|
DataRow row = SerialNumbers.Rows.Find(SerialNumber);
|
|
if (row == null) continue; // video in the folder but not database. todo: insert.
|
|
|
|
SerialNumbers.Rows.Remove(row);
|
|
}
|
|
|
|
StringBuilder toRecheck = new StringBuilder();
|
|
bool hasRows = false;
|
|
|
|
foreach (DataRow row in SerialNumbers.Rows)
|
|
{
|
|
ulong SerialNumber = (ulong)row["SerialNumber"];
|
|
if (hasRows) toRecheck.Append(',');
|
|
toRecheck.Append(SerialNumber.ToString());
|
|
hasRows = true;
|
|
|
|
Console.WriteLine("Battle video {0} in database but not in directory. Requeueing.", FormatVideoId(SerialNumber));
|
|
}
|
|
if (hasRows)
|
|
{
|
|
db.ExecuteNonQuery("UPDATE BattleVideoCrawlQueue5 SET Complete = 0 " +
|
|
"WHERE SerialNumber IN (" + toRecheck.ToString() + ")");
|
|
}
|
|
|
|
db.Clone();
|
|
}
|
|
}
|
|
|
|
public static MySqlConnection CreateConnection()
|
|
{
|
|
return new MySqlConnection(ConfigurationManager.ConnectionStrings["pkmnFoundationsConnectionString"].ConnectionString);
|
|
}
|
|
|
|
public static void LogError(Exception ex)
|
|
{
|
|
Console.WriteLine(ex.Message);
|
|
Console.WriteLine(ex.StackTrace);
|
|
}
|
|
|
|
public enum SearchSpecial : uint
|
|
{
|
|
None = 0x00000000,
|
|
|
|
// if special, all other fields are 0
|
|
Latest30 = 0x00000001, // 01 00 00 00 00 00 00 00 00 00 00 00
|
|
TopLinkBattles = 0x00000003, // 03 00 00 00 00 00 00 00 00 00 00 00
|
|
TopSubwayBattles = 0x00000002, // 02 00 00 00 00 00 00 00 00 00 00 00
|
|
}
|
|
|
|
public enum SearchMetagames : uint
|
|
{
|
|
None = 0x00000000,
|
|
|
|
// Spcial (4), Species (2), Meta (4), Country (1), Region (1)
|
|
ColosseumSingleNoLauncher = 0x00bf0018, // 00 00 00 00 ff ff 18 00 bf 00 ff ff
|
|
ColosseumSingleLauncher = 0x00bf0098, // 00 00 00 00 ff ff 98 00 bf 00 ff ff
|
|
ColosseumDoubleNoLauncher = 0x00bf0019, // 00 00 00 00 ff ff 19 00 bf 00 ff ff
|
|
ColosseumDoubleLauncher = 0x00bf0099, // 00 00 00 00 ff ff 99 00 bf 00 ff ff
|
|
ColosseumTripleNoLauncher = 0x00bf001a, // 00 00 00 00 ff ff 1a 00 bf 00 ff ff
|
|
ColosseumTripleLauncher = 0x00bf009a, // 00 00 00 00 ff ff 9a 00 bf 00 ff ff
|
|
ColosseumRotationNoLauncher = 0x00bf001b, // 00 00 00 00 ff ff 1b 00 bf 00 ff ff
|
|
ColosseumRotationLauncher = 0x00bf009b, // 00 00 00 00 ff ff 9b 00 bf 00 ff ff
|
|
ColosseumMultiNoLauncher = 0x00bf001c, // 00 00 00 00 ff ff 1c 00 bf 00 ff ff
|
|
ColosseumMultiLauncher = 0x00bf009c, // 00 00 00 00 ff ff 9c 00 bf 00 ff ff
|
|
|
|
BattleSubwaySingle = 0x003f0000, // 00 00 00 00 ff ff 00 00 3f 00 ff ff
|
|
BattleSubwayDouble = 0x003f0001, // 00 00 00 00 ff ff 01 00 3f 00 ff ff
|
|
BattleSubwayMulti = 0x003f0004, // 00 00 00 00 ff ff 04 00 3f 00 ff ff
|
|
|
|
RandomMatchupSingle = 0x003f0028, // 00 00 00 00 ff ff 28 00 3f 00 ff ff
|
|
RandomMatchupDouble = 0x003f0029, // 00 00 00 00 ff ff 29 00 3f 00 ff ff
|
|
RandomMatchupTriple = 0x003f002a, // 00 00 00 00 ff ff 2a 00 3f 00 ff ff
|
|
RandomMatchupRotation = 0x003f002b, // 00 00 00 00 ff ff 2b 00 3f 00 ff ff
|
|
RandomMatchupLauncher = 0x00bf00aa, // 00 00 00 00 ff ff aa 00 bf 00 ff ff
|
|
|
|
BattleCompetition = 0x00380038, // 00 00 00 00 ff ff 38 00 38 00 ff ff
|
|
}
|
|
}
|
|
}
|