2024-02-25 02:16:19 +00:00
|
|
|
export const MissingTextureKey = '__MISSING';
|
|
|
|
|
2023-10-18 22:01:15 +00:00
|
|
|
export function toReadableString(str: string): string {
|
|
|
|
return str.replace(/\_/g, ' ').split(' ').map(s => `${s.slice(0, 1)}${s.slice(1).toLowerCase()}`).join(' ');
|
|
|
|
}
|
|
|
|
|
2024-01-03 02:31:59 +00:00
|
|
|
export function randomString(length: integer, seeded: boolean = false) {
|
2023-10-18 22:01:15 +00:00
|
|
|
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
|
|
|
|
let result = '';
|
|
|
|
|
|
|
|
for (let i = 0; i < length; i++) {
|
2024-01-03 02:31:59 +00:00
|
|
|
const randomIndex = seeded ? randSeedInt(characters.length) : Math.floor(Math.random() * characters.length);
|
2023-10-18 22:01:15 +00:00
|
|
|
result += characters[randomIndex];
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function shiftCharCodes(str: string, shiftCount: integer) {
|
|
|
|
if (!shiftCount)
|
|
|
|
shiftCount = 0;
|
|
|
|
|
|
|
|
let newStr = '';
|
|
|
|
|
|
|
|
for (let i = 0; i < str.length; i++) {
|
|
|
|
let charCode = str.charCodeAt(i);
|
|
|
|
let newCharCode = charCode + shiftCount;
|
|
|
|
newStr += String.fromCharCode(newCharCode);
|
|
|
|
}
|
|
|
|
|
|
|
|
return newStr;
|
2023-03-28 18:54:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function clampInt(value: integer, min: integer, max: integer): integer {
|
|
|
|
return Math.min(Math.max(value, min), max);
|
|
|
|
}
|
|
|
|
|
2024-01-05 04:57:21 +00:00
|
|
|
export function randGauss(stdev: number, mean: number = 0): number {
|
2024-03-19 23:52:27 +00:00
|
|
|
if (!stdev)
|
|
|
|
return 0;
|
2024-01-05 04:57:21 +00:00
|
|
|
const u = 1 - Math.random();
|
|
|
|
const v = Math.random();
|
|
|
|
const z = Math.sqrt(-2.0 * Math.log(u)) * Math.cos(2.0 * Math.PI * v);
|
|
|
|
return z * stdev + mean;
|
2023-03-28 18:54:52 +00:00
|
|
|
}
|
|
|
|
|
2024-01-05 04:57:21 +00:00
|
|
|
export function randSeedGauss(stdev: number, mean: number = 0): number {
|
2024-03-19 23:52:27 +00:00
|
|
|
if (!stdev)
|
|
|
|
return 0;
|
2024-01-05 04:57:21 +00:00
|
|
|
const u = 1 - Phaser.Math.RND.realInRange(0, 1);
|
|
|
|
const v = Phaser.Math.RND.realInRange(0, 1);
|
|
|
|
const z = Math.sqrt(-2.0 * Math.log(u)) * Math.cos(2.0 * Math.PI * v);
|
|
|
|
return z * stdev + mean;
|
2023-10-27 01:42:53 +00:00
|
|
|
}
|
|
|
|
|
2023-03-28 18:54:52 +00:00
|
|
|
export function padInt(value: integer, length: integer, padWith?: string): string {
|
|
|
|
if (!padWith)
|
|
|
|
padWith = '0';
|
|
|
|
let valueStr = value.toString();
|
|
|
|
while (valueStr.length < length)
|
|
|
|
valueStr = `${padWith}${valueStr}`;
|
|
|
|
return valueStr;
|
|
|
|
}
|
|
|
|
|
2024-01-03 02:31:59 +00:00
|
|
|
export function randInt(range: integer, min: integer = 0): integer {
|
2023-05-18 15:11:06 +00:00
|
|
|
if (range === 1)
|
|
|
|
return min;
|
2023-03-28 18:54:52 +00:00
|
|
|
return Math.floor(Math.random() * range) + min;
|
|
|
|
}
|
|
|
|
|
2024-01-03 02:31:59 +00:00
|
|
|
export function randSeedInt(range: integer, min: integer = 0): integer {
|
2024-01-08 04:17:24 +00:00
|
|
|
if (range <= 1)
|
2023-10-18 22:01:15 +00:00
|
|
|
return min;
|
|
|
|
return Phaser.Math.RND.integerInRange(min, (range - 1) + min);
|
|
|
|
}
|
|
|
|
|
2023-10-04 21:24:28 +00:00
|
|
|
export function randIntRange(min: integer, max: integer): integer {
|
|
|
|
return randInt(max - min, min);
|
|
|
|
}
|
|
|
|
|
2024-03-21 18:53:35 +00:00
|
|
|
export function randItem<T>(items: T[]): T {
|
|
|
|
return items.length === 1
|
|
|
|
? items[0]
|
|
|
|
: items[randInt(items.length)];
|
|
|
|
}
|
|
|
|
|
2024-02-17 05:40:03 +00:00
|
|
|
export function randSeedItem<T>(items: T[]): T {
|
|
|
|
return items.length === 1
|
|
|
|
? items[0]
|
|
|
|
: Phaser.Math.RND.pick(items);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function randSeedWeightedItem<T>(items: T[]): T {
|
|
|
|
return items.length === 1
|
|
|
|
? items[0]
|
|
|
|
: Phaser.Math.RND.weightedPick(items);
|
|
|
|
}
|
|
|
|
|
2024-04-02 04:48:13 +00:00
|
|
|
export function randSeedEasedWeightedItem<T>(items: T[], easingFunction: string = 'Sine.easeIn'): T {
|
|
|
|
if (!items.length)
|
|
|
|
return null;
|
|
|
|
if (items.length === 1)
|
|
|
|
return items[0];
|
|
|
|
const value = Phaser.Math.RND.realInRange(0, 1);
|
|
|
|
const easedValue = Phaser.Tweens.Builders.GetEaseFunction(easingFunction)(value);
|
|
|
|
return items[Math.floor(easedValue * items.length)];
|
|
|
|
}
|
|
|
|
|
2023-12-20 04:51:48 +00:00
|
|
|
export function getSunday(date: Date): Date {
|
2024-03-15 23:40:13 +00:00
|
|
|
const day = date.getDay();
|
|
|
|
const diff = date.getDate() - day;
|
2023-12-20 04:51:48 +00:00
|
|
|
const newDate = new Date(date.setDate(diff));
|
2024-03-15 23:40:13 +00:00
|
|
|
return new Date(Date.UTC(newDate.getUTCFullYear(), newDate.getUTCMonth(), newDate.getUTCDate()));
|
2023-12-20 04:51:48 +00:00
|
|
|
}
|
|
|
|
|
2023-06-05 01:47:43 +00:00
|
|
|
export function getFrameMs(frameCount: integer): integer {
|
|
|
|
return Math.floor((1 / 60) * 1000 * frameCount);
|
|
|
|
}
|
|
|
|
|
2023-12-30 02:04:40 +00:00
|
|
|
export function getCurrentTime(): number {
|
|
|
|
const date = new Date();
|
|
|
|
return (((date.getHours() * 60 + date.getMinutes()) / 1440) + 0.675) % 1;
|
|
|
|
}
|
|
|
|
|
2024-01-12 01:27:50 +00:00
|
|
|
const secondsInHour = 3600;
|
|
|
|
|
|
|
|
export function getPlayTimeString(totalSeconds: integer): string {
|
|
|
|
const days = `${Math.floor(totalSeconds / (secondsInHour * 24))}`;
|
|
|
|
const hours = `${Math.floor(totalSeconds % (secondsInHour * 24) / secondsInHour)}`;
|
|
|
|
const minutes = `${Math.floor(totalSeconds % secondsInHour / 60)}`;
|
|
|
|
const seconds = `${Math.floor(totalSeconds % 60)}`;
|
|
|
|
|
|
|
|
return `${days.padStart(2, '0')}:${hours.padStart(2, '0')}:${minutes.padStart(2, '0')}:${seconds.padStart(2, '0')}`;
|
|
|
|
}
|
|
|
|
|
2023-03-28 18:54:52 +00:00
|
|
|
export function binToDec(input: string): integer {
|
2023-11-13 04:47:04 +00:00
|
|
|
let place: integer[] = [];
|
|
|
|
let binary: string[] = [];
|
2023-03-28 18:54:52 +00:00
|
|
|
|
|
|
|
let decimalNum = 0;
|
|
|
|
|
|
|
|
for (let i = 0; i < input.length; i++) {
|
|
|
|
binary.push(input[i]);
|
|
|
|
place.push(Math.pow(2, i));
|
|
|
|
decimalNum += place[i] * parseInt(binary[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return decimalNum;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function decToBin(input: integer): string {
|
|
|
|
let bin = '';
|
|
|
|
let intNum = input;
|
|
|
|
while (intNum > 0) {
|
|
|
|
bin = intNum % 2 ? `1${bin}` : `0${bin}`;
|
|
|
|
intNum = Math.floor(intNum * 0.5);
|
|
|
|
}
|
|
|
|
|
|
|
|
return bin;
|
|
|
|
}
|
|
|
|
|
2024-03-29 04:03:54 +00:00
|
|
|
export function getIvsFromId(id: integer): integer[] {
|
|
|
|
return [
|
|
|
|
binToDec(decToBin(id).substring(0, 5)),
|
|
|
|
binToDec(decToBin(id).substring(5, 10)),
|
|
|
|
binToDec(decToBin(id).substring(10, 15)),
|
|
|
|
binToDec(decToBin(id).substring(15, 20)),
|
|
|
|
binToDec(decToBin(id).substring(20, 25)),
|
|
|
|
binToDec(decToBin(id).substring(25, 30))
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2024-03-15 19:13:32 +00:00
|
|
|
export function formatLargeNumber(count: integer, threshold: integer): string {
|
|
|
|
if (count < threshold)
|
|
|
|
return count.toString();
|
|
|
|
let ret = count.toString();
|
2024-02-29 20:25:15 +00:00
|
|
|
let suffix = '';
|
|
|
|
switch (Math.ceil(ret.length / 3) - 1) {
|
|
|
|
case 1:
|
|
|
|
suffix = 'K';
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
suffix = 'M';
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
suffix = 'B';
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return '?';
|
|
|
|
}
|
|
|
|
const digits = ((ret.length + 2) % 3) + 1;
|
|
|
|
const decimalNumber = parseInt(ret.slice(digits, digits + (3 - digits)));
|
|
|
|
return `${ret.slice(0, digits)}${decimalNumber ? `.${decimalNumber}` : ''}${suffix}`;
|
|
|
|
}
|
|
|
|
|
2024-03-15 19:13:32 +00:00
|
|
|
export function formatStat(stat: integer, forHp: boolean = false): string {
|
|
|
|
return formatLargeNumber(stat, forHp ? 100000 : 1000000);
|
|
|
|
}
|
|
|
|
|
2023-04-11 17:00:04 +00:00
|
|
|
export function getEnumKeys(enumType): string[] {
|
|
|
|
return Object.values(enumType).filter(v => isNaN(parseInt(v.toString()))).map(v => v.toString());
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getEnumValues(enumType): integer[] {
|
2023-03-28 18:54:52 +00:00
|
|
|
return Object.values(enumType).filter(v => !isNaN(parseInt(v.toString()))).map(v => parseInt(v.toString()));
|
|
|
|
}
|
|
|
|
|
2023-10-31 18:09:33 +00:00
|
|
|
export function executeIf<T>(condition: boolean, promiseFunc: () => Promise<T>): Promise<T> {
|
|
|
|
return condition ? promiseFunc() : new Promise<T>(resolve => resolve(null));
|
|
|
|
}
|
|
|
|
|
2023-12-30 23:41:25 +00:00
|
|
|
export const sessionIdKey = 'pokerogue_sessionId';
|
|
|
|
export const isLocal = window.location.hostname === 'localhost';
|
|
|
|
export const serverUrl = isLocal ? 'http://localhost:8001' : '';
|
2024-04-24 23:08:02 +00:00
|
|
|
export const apiUrl = isLocal ? serverUrl : 'https://api.pokerogue.net';
|
|
|
|
export const fallbackApiUrl = isLocal ? serverUrl : 'api';
|
2023-12-30 23:41:25 +00:00
|
|
|
|
|
|
|
export function setCookie(cName: string, cValue: string): void {
|
2024-04-10 04:29:03 +00:00
|
|
|
const expiration = new Date();
|
|
|
|
expiration.setTime(new Date().getTime() + 3600000 * 24 * 7);
|
|
|
|
document.cookie = `${cName}=${cValue};SameSite=Strict;path=/;expires=${expiration.toUTCString()}`;
|
2023-12-30 23:41:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getCookie(cName: string): string {
|
|
|
|
const name = `${cName}=`;
|
|
|
|
const ca = document.cookie.split(';');
|
|
|
|
for (let i = 0; i < ca.length; i++) {
|
|
|
|
let c = ca[i];
|
|
|
|
while (c.charAt(0) === ' ')
|
|
|
|
c = c.substring(1);
|
|
|
|
if (c.indexOf(name) === 0)
|
|
|
|
return c.substring(name.length, c.length);
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2024-04-24 23:08:02 +00:00
|
|
|
export function apiFetch(path: string, authed: boolean = false, fallback: boolean = false): Promise<Response> {
|
2023-12-30 23:41:25 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
2024-04-19 21:35:49 +00:00
|
|
|
const request = {};
|
|
|
|
if (authed) {
|
|
|
|
const sId = getCookie(sessionIdKey);
|
|
|
|
if (sId)
|
|
|
|
request['headers'] = { 'Authorization': sId };
|
|
|
|
}
|
2024-04-24 23:08:02 +00:00
|
|
|
fetch(`${!fallback ? apiUrl : fallbackApiUrl}/${path}`, request)
|
|
|
|
.then(response => {
|
|
|
|
if (!response.ok && response.status === 404 && !fallback)
|
|
|
|
return apiFetch(path, authed, true).then(res => resolve(res));
|
|
|
|
resolve(response);
|
|
|
|
})
|
|
|
|
.catch(err => {
|
|
|
|
if (fallback)
|
|
|
|
reject(err);
|
|
|
|
else
|
|
|
|
apiFetch(path, authed, true).then(res => resolve(res));
|
|
|
|
});
|
2023-12-30 23:41:25 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-04-24 23:08:02 +00:00
|
|
|
export function apiPost(path: string, data?: any, contentType: string = 'application/json', authed: boolean = false, fallback: boolean = false): Promise<Response> {
|
2023-12-30 23:41:25 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const headers = {
|
|
|
|
'Accept': contentType,
|
|
|
|
'Content-Type': contentType,
|
|
|
|
};
|
2024-04-24 23:08:02 +00:00
|
|
|
if (authed) {
|
|
|
|
const sId = getCookie(sessionIdKey);
|
|
|
|
if (sId)
|
|
|
|
headers['Authorization'] = sId;
|
|
|
|
}
|
|
|
|
fetch(`${!fallback ? apiUrl : fallbackApiUrl}/${path}`, { method: 'POST', headers: headers, body: data })
|
2023-12-30 23:41:25 +00:00
|
|
|
.then(response => resolve(response))
|
2024-04-24 23:08:02 +00:00
|
|
|
.catch(err => {
|
|
|
|
if (fallback)
|
|
|
|
reject(err);
|
|
|
|
else
|
|
|
|
apiPost(path, data, contentType, authed, true).then(res => resolve(res));
|
|
|
|
});
|
2023-12-30 23:41:25 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-04-12 23:09:15 +00:00
|
|
|
export class BooleanHolder {
|
|
|
|
public value: boolean;
|
|
|
|
|
|
|
|
constructor(value: boolean) {
|
|
|
|
this.value = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-31 03:02:35 +00:00
|
|
|
export class NumberHolder {
|
|
|
|
public value: number;
|
|
|
|
|
|
|
|
constructor(value: number) {
|
|
|
|
this.value = value;
|
|
|
|
}
|
|
|
|
}
|
2023-04-12 15:30:47 +00:00
|
|
|
|
2023-11-12 05:31:40 +00:00
|
|
|
export class IntegerHolder extends NumberHolder {
|
2023-03-28 18:54:52 +00:00
|
|
|
constructor(value: integer) {
|
2023-11-12 05:31:40 +00:00
|
|
|
super(value);
|
2023-03-28 18:54:52 +00:00
|
|
|
}
|
2023-04-12 15:30:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export class FixedInt extends IntegerHolder {
|
|
|
|
constructor(value: integer) {
|
|
|
|
super(value);
|
|
|
|
}
|
2023-10-26 03:15:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function fixedInt(value: integer): integer {
|
|
|
|
return new FixedInt(value) as unknown as integer;
|
2023-11-24 04:52:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function rgbToHsv(r: integer, g: integer, b: integer) {
|
|
|
|
let v = Math.max(r, g, b);
|
|
|
|
let c = v - Math.min(r, g, b);
|
|
|
|
let h = c && ((v === r) ? (g - b) / c : ((v === g) ? 2 + (b - r) / c : 4 + (r - g) / c));
|
|
|
|
return [ 60 * (h < 0 ? h + 6 : h), v && c / v, v];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compare color difference in RGB
|
|
|
|
* @param {Array} rgb1 First RGB color in array
|
|
|
|
* @param {Array} rgb2 Second RGB color in array
|
|
|
|
*/
|
|
|
|
export function deltaRgb(rgb1: integer[], rgb2: integer[]): integer {
|
|
|
|
const [ r1, g1, b1 ] = rgb1;
|
|
|
|
const [ r2, g2, b2 ] = rgb2;
|
|
|
|
const drp2 = Math.pow(r1 - r2, 2);
|
|
|
|
const dgp2 = Math.pow(g1 - g2, 2);
|
|
|
|
const dbp2 = Math.pow(b1 - b2, 2);
|
|
|
|
const t = (r1 + r2) / 2;
|
|
|
|
|
|
|
|
return Math.ceil(Math.sqrt(2 * drp2 + 4 * dgp2 + 3 * dbp2 + t * (drp2 - dbp2) / 256));
|
2024-04-13 22:59:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function rgbHexToRgba(hex: string) {
|
|
|
|
const color = hex.match(/^([\da-f]{2})([\da-f]{2})([\da-f]{2})$/i);
|
|
|
|
return {
|
|
|
|
r: parseInt(color[1], 16),
|
|
|
|
g: parseInt(color[2], 16),
|
|
|
|
b: parseInt(color[3], 16),
|
|
|
|
a: 255
|
|
|
|
};
|
2024-04-19 02:52:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function rgbaToInt(rgba: integer[]): integer {
|
|
|
|
return (rgba[0] << 24) + (rgba[1] << 16) + (rgba[2] << 8) + rgba[3];
|
2023-03-28 18:54:52 +00:00
|
|
|
}
|