mirror of
https://github.com/openfrontio/OpenFrontIO.git
synced 2026-06-21 14:00:54 +00:00
283 lines
8.9 KiB
TypeScript
283 lines
8.9 KiB
TypeScript
import { Gold, Player, PlayerInfo, PlayerType, TerrainType, TerraNullius, Tick, Tile, Unit, UnitInfo, UnitType } from "../game/Game";
|
|
import { GameID } from "../Schemas";
|
|
import { assertNever, distSort, manhattanDist, simpleHash, within } from "../Util";
|
|
import { Config, Theme } from "./Config";
|
|
import { pastelTheme } from "./PastelTheme";
|
|
|
|
|
|
|
|
export class DefaultConfig implements Config {
|
|
|
|
maxUnitCost(): number {
|
|
return 99_999_999
|
|
}
|
|
|
|
cityPopulationIncrease(): number {
|
|
return 250_000
|
|
}
|
|
|
|
falloutDefenseModifier(): number {
|
|
return 2
|
|
}
|
|
|
|
defensePostRange(): number {
|
|
return 30
|
|
}
|
|
defensePostDefenseBonus(): number {
|
|
return 3
|
|
}
|
|
spawnNPCs(): boolean {
|
|
return true
|
|
}
|
|
tradeShipGold(src: Unit, dst: Unit): Gold {
|
|
const dist = manhattanDist(src.tile().cell(), dst.tile().cell())
|
|
return 10000 + 100 * Math.pow(dist, 1.1)
|
|
}
|
|
tradeShipSpawnRate(): number {
|
|
return 500
|
|
}
|
|
|
|
unitInfo(type: UnitType): UnitInfo {
|
|
const fn = () => {
|
|
switch (type) {
|
|
case UnitType.TransportShip:
|
|
return {
|
|
cost: () => 0,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.Destroyer:
|
|
return {
|
|
cost: (p: Player) => (p.units(UnitType.Destroyer).length + 1) * 250_000,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.Battleship:
|
|
return {
|
|
cost: (p: Player) => (p.units(UnitType.Battleship).length + 1) * 500_000,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.Shell:
|
|
return {
|
|
cost: (p: Player) => 0,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.Port:
|
|
return {
|
|
cost: (p: Player) => Math.pow(2, p.units(UnitType.Port).length) * 250_000,
|
|
territoryBound: true
|
|
}
|
|
case UnitType.AtomBomb:
|
|
return {
|
|
cost: () => 1_000_000,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.HydrogenBomb:
|
|
return {
|
|
cost: () => 5_000_000,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.TradeShip:
|
|
return {
|
|
cost: () => 0,
|
|
territoryBound: false
|
|
}
|
|
case UnitType.MissileSilo:
|
|
return {
|
|
cost: () => 1_000_000,
|
|
territoryBound: true
|
|
}
|
|
case UnitType.DefensePost:
|
|
return {
|
|
cost: (p: Player) => Math.pow(2, p.units(UnitType.Port).length) * 100_000,
|
|
territoryBound: true
|
|
}
|
|
case UnitType.City:
|
|
return {
|
|
cost: (p: Player) => Math.pow(2, p.units(UnitType.Port).length) * 250_000,
|
|
territoryBound: true
|
|
}
|
|
default:
|
|
assertNever(type)
|
|
}
|
|
}
|
|
const ui = fn()
|
|
const oldCost = ui.cost
|
|
ui.cost = (p: Player) => Math.min(this.maxUnitCost(), oldCost(p))
|
|
return ui
|
|
}
|
|
defaultDonationAmount(sender: Player): number {
|
|
return Math.floor(sender.troops() / 3)
|
|
}
|
|
donateCooldown(): Tick {
|
|
return 10 * 10
|
|
}
|
|
emojiMessageDuration(): Tick {
|
|
return 5 * 10
|
|
}
|
|
emojiMessageCooldown(): Tick {
|
|
return 15 * 10
|
|
}
|
|
targetDuration(): Tick {
|
|
return 10 * 10
|
|
}
|
|
targetCooldown(): Tick {
|
|
return 15 * 10
|
|
}
|
|
allianceRequestCooldown(): Tick {
|
|
return 30 * 10
|
|
}
|
|
allianceDuration(): Tick {
|
|
return 600 * 10
|
|
}
|
|
percentageTilesOwnedToWin(): number {
|
|
return 95
|
|
}
|
|
boatMaxNumber(): number {
|
|
return 3
|
|
}
|
|
boatMaxDistance(): number {
|
|
return 500
|
|
}
|
|
numSpawnPhaseTurns(): number {
|
|
return 100
|
|
}
|
|
numBots(): number {
|
|
return 400
|
|
}
|
|
turnIntervalMs(): number {
|
|
return 100
|
|
}
|
|
gameCreationRate(): number {
|
|
return 60 * 1000
|
|
}
|
|
lobbyLifetime(): number {
|
|
return 120 * 1000
|
|
}
|
|
theme(): Theme { return pastelTheme; }
|
|
|
|
attackLogic(attackTroops: number, attacker: Player, defender: Player | TerraNullius, tileToConquer: Tile): { attackerTroopLoss: number; defenderTroopLoss: number; tilesPerTickUsed: number } {
|
|
let mag = 0
|
|
let speed = 0
|
|
switch (tileToConquer.terrain()) {
|
|
case TerrainType.Plains:
|
|
mag = 50
|
|
speed = 10
|
|
break
|
|
case TerrainType.Highland:
|
|
mag = 100
|
|
speed = 20
|
|
break
|
|
case TerrainType.Mountain:
|
|
mag = 150
|
|
speed = 30
|
|
break
|
|
}
|
|
mag *= tileToConquer.defenseBonus(attacker)
|
|
speed *= tileToConquer.defenseBonus(attacker)
|
|
if (tileToConquer.hasFallout()) {
|
|
mag *= this.falloutDefenseModifier()
|
|
speed *= this.falloutDefenseModifier()
|
|
}
|
|
|
|
if (attacker.isPlayer() && defender.isPlayer()) {
|
|
if (attacker.type() == PlayerType.Human && defender.type() == PlayerType.Bot) {
|
|
mag *= .8
|
|
}
|
|
if (attacker.type() == PlayerType.FakeHuman && defender.type() == PlayerType.Bot) {
|
|
mag *= .8
|
|
}
|
|
}
|
|
|
|
if (defender.isPlayer()) {
|
|
return {
|
|
attackerTroopLoss: within(defender.troops() / attacker.troops(), .5, 2) * mag,
|
|
defenderTroopLoss: defender.troops() / defender.numTilesOwned(),
|
|
tilesPerTickUsed: within(defender.troops() / (attackTroops * 5), .2, 1.5) * speed
|
|
}
|
|
} else {
|
|
return {
|
|
attackerTroopLoss: attacker.type() == PlayerType.Bot ? mag / 10 : mag / 5,
|
|
defenderTroopLoss: 0,
|
|
tilesPerTickUsed: within(this.startManpower(attacker.info()) / (attackTroops * 5), .2, 3) * Math.max(10, speed / 1.5)
|
|
}
|
|
}
|
|
}
|
|
|
|
attackTilesPerTick(attacker: Player, defender: Player | TerraNullius, numAdjacentTilesWithEnemy: number): number {
|
|
if (defender.isPlayer()) {
|
|
return within(attacker.troops() / defender.troops() * 2, .01, .5) * numAdjacentTilesWithEnemy * 3
|
|
} else {
|
|
return numAdjacentTilesWithEnemy * 2
|
|
}
|
|
}
|
|
|
|
boatAttackAmount(attacker: Player, defender: Player | TerraNullius): number {
|
|
return Math.floor(attacker.troops() / 5)
|
|
}
|
|
|
|
attackAmount(attacker: Player, defender: Player | TerraNullius) {
|
|
if (attacker.type() == PlayerType.Bot) {
|
|
return attacker.troops() / 20
|
|
} else {
|
|
return attacker.troops() / 5
|
|
}
|
|
}
|
|
|
|
startManpower(playerInfo: PlayerInfo): number {
|
|
if (playerInfo.playerType == PlayerType.Bot) {
|
|
return 10000
|
|
}
|
|
if (playerInfo.playerType == PlayerType.FakeHuman) {
|
|
return 2000 // start troops * strength * difficulty
|
|
}
|
|
return 25000
|
|
}
|
|
|
|
maxPopulation(player: Player): number {
|
|
let maxPop = Math.sqrt(player.numTilesOwned()) * 3000 + 50000
|
|
if (player.type() == PlayerType.Bot) {
|
|
return maxPop
|
|
}
|
|
return maxPop * 2 + player.units(UnitType.City).length * this.cityPopulationIncrease()
|
|
}
|
|
|
|
populationIncreaseRate(player: Player): number {
|
|
let max = this.maxPopulation(player)
|
|
|
|
let toAdd = 10 + (player.population() + Math.sqrt(player.population() * player.numTilesOwned())) / 100
|
|
|
|
const ratio = 1 - (player.population() / max)
|
|
toAdd *= ratio
|
|
toAdd *= .5
|
|
// console.log(`to add ${toAdd}`)
|
|
|
|
if (player.type() == PlayerType.FakeHuman) {
|
|
toAdd *= 1.0
|
|
}
|
|
if (player.type() == PlayerType.Bot) {
|
|
toAdd *= .7
|
|
}
|
|
|
|
return Math.min(player.troops() + toAdd, max) - player.troops()
|
|
}
|
|
|
|
goldAdditionRate(player: Player): number {
|
|
return Math.sqrt(player.workers() * player.numTilesOwned()) / 200
|
|
}
|
|
|
|
troopAdjustmentRate(player: Player): number {
|
|
const maxDiff = this.maxPopulation(player) / 1000
|
|
const target = player.population() * player.targetTroopRatio()
|
|
const diff = target - player.troops()
|
|
if (Math.abs(diff) < maxDiff) {
|
|
return diff
|
|
}
|
|
const adjustment = maxDiff * Math.sign(diff)
|
|
// Can ramp down troops much faster
|
|
if (adjustment < 0) {
|
|
return adjustment * 5
|
|
}
|
|
return adjustment
|
|
}
|
|
}
|
|
|
|
export const defaultConfig = new DefaultConfig() |