breakout71/src/pure_functions.ts

145 lines
3.7 KiB
TypeScript
Raw Normal View History

2025-04-23 10:56:50 +02:00
import { Ball, GameState } from "./types";
2025-04-15 16:47:04 +02:00
export function clamp(value: number, min: number, max: number) {
2025-04-09 11:28:32 +02:00
return Math.max(min, Math.min(value, max));
}
export function comboKeepingRate(level: number) {
2025-04-09 11:28:32 +02:00
return clamp(1 - (1 / (1 + level)) * 1.5, 0, 1);
2025-03-29 21:28:05 +01:00
}
2025-04-21 13:25:06 +02:00
export function ballTransparency(ball: Ball, gameState: GameState) {
if (!gameState.perks.transparency) return 0;
return clamp(
gameState.perks.transparency *
(1 - (ball.y / gameState.gameZoneHeight) * 1.2),
0,
1,
);
2025-04-21 13:16:16 +02:00
}
2025-04-21 13:25:06 +02:00
export function coinsBoostedCombo(gameState: GameState) {
let boost =
1 + gameState.perks.sturdy_bricks / 2 + gameState.perks.smaller_puck / 2;
if (gameState.perks.transparency) {
let min = 1;
gameState.balls.forEach((ball) => {
const bt = ballTransparency(ball, gameState);
if (bt < min) {
min = bt;
2025-04-21 13:16:16 +02:00
}
2025-04-21 13:25:06 +02:00
});
boost += (min * gameState.perks.transparency) / 2;
2025-04-21 13:16:16 +02:00
}
2025-04-23 20:51:44 +02:00
if(gameState.perks.minefield){
gameState.bricks.forEach(brick=>{
if(brick === 'black'){
boost+=0.1 * gameState.perks.minefield
}
})
}
2025-04-21 13:25:06 +02:00
return Math.ceil(Math.max(gameState.combo, gameState.lastCombo) * boost);
2025-04-21 13:16:16 +02:00
}
2025-03-31 20:08:17 +02:00
export function miniMarkDown(md: string) {
2025-04-09 11:28:32 +02:00
let html: { tagName: string; text: string }[] = [];
let lastNode: { tagName: string; text: string } | null = null;
2025-03-31 20:08:17 +02:00
2025-04-09 11:28:32 +02:00
md.split("\n").forEach((line) => {
const titlePrefix = line.match(/^#+ /)?.[0];
2025-03-31 20:08:17 +02:00
2025-04-09 11:28:32 +02:00
if (titlePrefix) {
if (lastNode) html.push(lastNode);
lastNode = {
tagName: "h" + (titlePrefix.length - 1),
text: line.slice(titlePrefix.length),
};
} else if (line.startsWith("- ")) {
if (lastNode?.tagName !== "ul") {
if (lastNode) html.push(lastNode);
lastNode = { tagName: "ul", text: "" };
}
lastNode.text += "<li>" + line.slice(2) + "</li>";
} else if (!line.trim()) {
if (lastNode) html.push(lastNode);
lastNode = null;
} else {
if (lastNode?.tagName !== "p") {
if (lastNode) html.push(lastNode);
lastNode = { tagName: "p", text: "" };
}
lastNode.text += line + " ";
2025-03-31 20:08:17 +02:00
}
2025-04-09 11:28:32 +02:00
});
if (lastNode) {
html.push(lastNode);
}
return html
.map(
(h) =>
"<" +
h.tagName +
">" +
h.text.replace(
/\bhttps?:\/\/[^\s<>]+/gi,
(a) => `<a href="${a}">${a}</a>`,
) +
"</" +
h.tagName +
">",
)
.join("\n");
}
2025-04-08 14:03:38 +02:00
export function firstWhere<Input, Output>(
2025-04-09 11:28:32 +02:00
arr: Input[],
mapper: (item: Input, index: number) => Output | undefined,
2025-04-08 14:03:38 +02:00
): Output | undefined {
2025-04-09 11:28:32 +02:00
for (let i = 0; i < arr.length; i++) {
const result = mapper(arr[i], i);
if (typeof result !== "undefined") return result;
}
2025-04-08 14:03:38 +02:00
}
2025-04-08 21:54:19 +02:00
2025-04-09 11:28:32 +02:00
export const wallBouncedBest = 3,
wallBouncedGood = 10,
levelTimeBest = 30,
levelTimeGood = 60,
catchRateBest = 95,
catchRateGood = 90,
missesBest = 3,
missesGood = 6;
2025-04-22 16:37:56 +02:00
export const MAX_LEVEL_SIZE = 21;
export const MIN_LEVEL_SIZE = 2;
export function automaticBackgroundColor(bricks: string[]) {
2025-04-23 10:56:50 +02:00
return bricks.filter((b) => b === "g").length >
2025-04-22 16:37:56 +02:00
bricks.filter((b) => b !== "_").length * 0.05
2025-04-23 10:56:50 +02:00
? "#115988"
2025-04-23 15:10:21 +02:00
: "#000000";
2025-04-22 16:37:56 +02:00
}
export function levelCodeToRawLevel(code: string) {
2025-04-23 10:56:50 +02:00
let [name, credit] = code.match(/\[([^\]]+)]/gi) || ["", ""];
2025-04-22 16:37:56 +02:00
2025-04-23 10:56:50 +02:00
let bricks = code.split(name)[1].split(credit)[0].replace(/\s/gi, "");
name = name.slice(1, -1);
credit = credit.slice(1, -1);
name ||= "Imported on " + new Date().toISOString().slice(0, 10);
credit ||= "";
const size = Math.sqrt(bricks.length);
if (
Math.floor(size) === size &&
size >= MIN_LEVEL_SIZE &&
size <= MAX_LEVEL_SIZE
)
return {
color: automaticBackgroundColor(bricks.split("")),
size,
bricks,
name,
credit,
};
}