2022-03-04 03:39:48 +00:00
|
|
|
import projInfo from "data/projInfo.json";
|
2022-03-11 20:02:41 +00:00
|
|
|
import Decimal from "util/bignum";
|
2022-01-14 04:25:47 +00:00
|
|
|
import { createNanoEvents } from "nanoevents";
|
2022-01-25 04:23:30 +00:00
|
|
|
import { App, Ref } from "vue";
|
|
|
|
import { GenericLayer } from "./layers";
|
2022-01-14 04:25:47 +00:00
|
|
|
import player from "./player";
|
|
|
|
import settings, { Settings } from "./settings";
|
|
|
|
import state from "./state";
|
|
|
|
|
|
|
|
export interface GlobalEvents {
|
|
|
|
addLayer: (layer: GenericLayer, saveData: Record<string, unknown>) => void;
|
|
|
|
removeLayer: (layer: GenericLayer) => void;
|
2022-03-11 20:02:41 +00:00
|
|
|
update: (diff: number, trueDiff: number) => void;
|
2022-01-14 04:25:47 +00:00
|
|
|
loadSettings: (settings: Partial<Settings>) => void;
|
|
|
|
setupVue: (vue: App) => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const globalBus = createNanoEvents<GlobalEvents>();
|
|
|
|
|
|
|
|
let intervalID: number | null = null;
|
|
|
|
|
2022-01-25 04:23:30 +00:00
|
|
|
// Not imported immediately due to dependency cycles
|
|
|
|
// This gets set during startGameLoop(), and will only be used in the update function
|
|
|
|
let hasWon: null | Ref<boolean> = null;
|
|
|
|
|
2022-01-14 04:25:47 +00:00
|
|
|
function update() {
|
|
|
|
const now = Date.now();
|
2022-03-11 20:02:41 +00:00
|
|
|
let diff = (now - player.time) / 1e3;
|
2022-01-14 04:25:47 +00:00
|
|
|
player.time = now;
|
|
|
|
const trueDiff = diff;
|
|
|
|
|
|
|
|
state.lastTenTicks.push(trueDiff);
|
|
|
|
if (state.lastTenTicks.length > 10) {
|
|
|
|
state.lastTenTicks = state.lastTenTicks.slice(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Stop here if the game is paused on the win screen
|
2022-01-25 04:23:30 +00:00
|
|
|
if (hasWon?.value && !player.keepGoing) {
|
2022-01-14 04:25:47 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Stop here if the player had a NaN value
|
|
|
|
if (state.hasNaN) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-11 20:02:41 +00:00
|
|
|
diff = Math.max(diff, 0);
|
2022-01-14 04:25:47 +00:00
|
|
|
|
2022-02-27 19:49:34 +00:00
|
|
|
if (player.devSpeed === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-01-14 04:25:47 +00:00
|
|
|
// Add offline time if any
|
|
|
|
if (player.offlineTime != undefined) {
|
2022-03-03 02:12:56 +00:00
|
|
|
if (Decimal.gt(player.offlineTime, projInfo.offlineLimit * 3600)) {
|
2022-03-11 20:02:41 +00:00
|
|
|
player.offlineTime = projInfo.offlineLimit * 3600;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
2022-01-25 04:25:34 +00:00
|
|
|
if (Decimal.gt(player.offlineTime, 0) && player.devSpeed !== 0) {
|
2022-03-11 20:02:41 +00:00
|
|
|
const offlineDiff = Math.max(player.offlineTime / 10, diff);
|
|
|
|
player.offlineTime = player.offlineTime - offlineDiff;
|
|
|
|
diff += offlineDiff;
|
2022-01-14 04:25:47 +00:00
|
|
|
} else if (player.devSpeed === 0) {
|
2022-03-11 20:02:41 +00:00
|
|
|
player.offlineTime += diff;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
2022-01-25 04:25:34 +00:00
|
|
|
if (!player.offlineProd || Decimal.lt(player.offlineTime, 0)) {
|
2022-01-14 04:25:47 +00:00
|
|
|
player.offlineTime = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cap at max tick length
|
2022-03-11 20:02:41 +00:00
|
|
|
diff = Math.min(diff, projInfo.maxTickLength);
|
2022-01-14 04:25:47 +00:00
|
|
|
|
|
|
|
// Apply dev speed
|
|
|
|
if (player.devSpeed != undefined) {
|
2022-03-11 20:02:41 +00:00
|
|
|
diff *= player.devSpeed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Number.isFinite(diff)) {
|
|
|
|
diff = 1e308;
|
2022-01-14 04:25:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Update
|
2022-03-11 20:02:41 +00:00
|
|
|
if (Decimal.eq(diff, 0)) {
|
2022-01-14 04:25:47 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-03-11 20:02:41 +00:00
|
|
|
|
|
|
|
player.timePlayed += diff;
|
|
|
|
if (!Number.isFinite(player.timePlayed)) {
|
|
|
|
player.timePlayed = 1e308;
|
|
|
|
}
|
2022-01-14 04:25:47 +00:00
|
|
|
globalBus.emit("update", diff, trueDiff);
|
|
|
|
|
|
|
|
if (settings.unthrottled) {
|
|
|
|
requestAnimationFrame(update);
|
|
|
|
if (intervalID != null) {
|
|
|
|
clearInterval(intervalID);
|
|
|
|
intervalID = null;
|
|
|
|
}
|
|
|
|
} else if (intervalID == null) {
|
|
|
|
intervalID = setInterval(update, 50);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-25 04:23:30 +00:00
|
|
|
export async function startGameLoop() {
|
2022-03-04 03:39:48 +00:00
|
|
|
hasWon = (await import("data/projEntry")).hasWon;
|
2022-01-14 04:25:47 +00:00
|
|
|
if (settings.unthrottled) {
|
|
|
|
requestAnimationFrame(update);
|
|
|
|
} else {
|
|
|
|
intervalID = setInterval(update, 50);
|
|
|
|
}
|
|
|
|
}
|