2024-01-13 17:24:24 +00:00
|
|
|
import { Biome } from "./enums/biome";
|
2023-04-20 20:46:05 +01:00
|
|
|
import { getPokemonMessage } from "../messages";
|
2024-03-01 01:08:50 +00:00
|
|
|
import Pokemon from "../field/pokemon";
|
2023-04-17 05:46:50 +01:00
|
|
|
import { Type } from "./type";
|
|
|
|
import Move, { AttackMove } from "./move";
|
2023-04-20 20:46:05 +01:00
|
|
|
import * as Utils from "../utils";
|
2023-04-27 19:30:03 +01:00
|
|
|
import BattleScene from "../battle-scene";
|
2024-03-19 01:22:27 +00:00
|
|
|
import { SuppressWeatherEffectAbAttr } from "./ability";
|
2024-03-10 02:57:33 +00:00
|
|
|
import { TerrainType } from "./terrain";
|
2023-04-17 01:41:52 +01:00
|
|
|
|
|
|
|
export enum WeatherType {
|
|
|
|
NONE,
|
|
|
|
SUNNY,
|
|
|
|
RAIN,
|
|
|
|
SANDSTORM,
|
|
|
|
HAIL,
|
|
|
|
FOG,
|
|
|
|
HEAVY_RAIN,
|
|
|
|
HARSH_SUN,
|
|
|
|
STRONG_WINDS
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Weather {
|
|
|
|
public weatherType: WeatherType;
|
|
|
|
public turnsLeft: integer;
|
|
|
|
|
|
|
|
constructor(weatherType: WeatherType, turnsLeft?: integer) {
|
|
|
|
this.weatherType = weatherType;
|
2024-04-06 04:45:31 +01:00
|
|
|
this.turnsLeft = !this.isImmutable() ? turnsLeft || 0 : 0;
|
2023-04-17 01:41:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
lapse(): boolean {
|
2024-04-06 04:45:31 +01:00
|
|
|
if (this.isImmutable())
|
|
|
|
return true;
|
2023-04-17 01:41:52 +01:00
|
|
|
if (this.turnsLeft)
|
|
|
|
return !!--this.turnsLeft;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2023-04-17 05:46:50 +01:00
|
|
|
|
|
|
|
isImmutable(): boolean {
|
|
|
|
switch (this.weatherType) {
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
case WeatherType.STRONG_WINDS:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
isDamaging(): boolean {
|
|
|
|
switch (this.weatherType) {
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
isTypeDamageImmune(type: Type): boolean {
|
|
|
|
switch (this.weatherType) {
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
return type === Type.GROUND || type === Type.ROCK || type === Type.STEEL;
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return type === Type.ICE;
|
|
|
|
}
|
2023-04-19 19:07:38 +01:00
|
|
|
|
|
|
|
return false;
|
2023-04-17 05:46:50 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
getAttackTypeMultiplier(attackType: Type): number {
|
|
|
|
switch (this.weatherType) {
|
|
|
|
case WeatherType.SUNNY:
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
if (attackType === Type.FIRE)
|
|
|
|
return 1.5;
|
|
|
|
if (attackType === Type.WATER)
|
|
|
|
return 0.5;
|
|
|
|
break;
|
|
|
|
case WeatherType.RAIN:
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
if (attackType === Type.FIRE)
|
|
|
|
return 0.5;
|
|
|
|
if (attackType === Type.WATER)
|
|
|
|
return 1.5;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
isMoveWeatherCancelled(move: Move): boolean {
|
|
|
|
switch (this.weatherType) {
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
return move instanceof AttackMove && move.type === Type.WATER;
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
return move instanceof AttackMove && move.type === Type.FIRE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2023-04-27 19:30:03 +01:00
|
|
|
|
|
|
|
isEffectSuppressed(scene: BattleScene): boolean {
|
2024-03-12 00:55:41 +00:00
|
|
|
const field = scene.getField(true);
|
2023-04-27 19:30:03 +01:00
|
|
|
|
2023-05-18 16:11:06 +01:00
|
|
|
for (let pokemon of field) {
|
|
|
|
const suppressWeatherEffectAbAttr = pokemon.getAbility().getAttrs(SuppressWeatherEffectAbAttr).find(() => true) as SuppressWeatherEffectAbAttr;
|
2023-05-03 03:27:04 +01:00
|
|
|
if (suppressWeatherEffectAbAttr && (!this.isImmutable() || suppressWeatherEffectAbAttr.affectsImmutable))
|
|
|
|
return true;
|
|
|
|
}
|
2023-04-27 19:30:03 +01:00
|
|
|
|
2023-05-03 03:27:04 +01:00
|
|
|
return false;
|
2023-04-27 19:30:03 +01:00
|
|
|
}
|
2023-04-17 05:46:50 +01:00
|
|
|
}
|
|
|
|
|
2024-03-10 02:57:33 +00:00
|
|
|
export function getWeatherStartMessage(weatherType: WeatherType): string {
|
2023-04-17 05:46:50 +01:00
|
|
|
switch (weatherType) {
|
|
|
|
case WeatherType.SUNNY:
|
|
|
|
return 'The sunlight got bright!';
|
|
|
|
case WeatherType.RAIN:
|
|
|
|
return 'A downpour started!';
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
return 'A sandstorm brewed!';
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return 'It started to hail!';
|
|
|
|
case WeatherType.FOG:
|
|
|
|
return 'A thick fog emerged!'
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
return 'A heavy downpour started!'
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
return 'The sunlight got hot!'
|
|
|
|
case WeatherType.STRONG_WINDS:
|
|
|
|
return 'A heavy wind began!';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-03-10 02:57:33 +00:00
|
|
|
export function getWeatherLapseMessage(weatherType: WeatherType): string {
|
2023-04-17 05:46:50 +01:00
|
|
|
switch (weatherType) {
|
|
|
|
case WeatherType.SUNNY:
|
|
|
|
return 'The sunlight is strong.';
|
|
|
|
case WeatherType.RAIN:
|
|
|
|
return 'The downpour continues.';
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
return 'The sandstorm rages.';
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return 'Hail continues to fall.';
|
|
|
|
case WeatherType.FOG:
|
|
|
|
return 'The fog continues.';
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
return 'The heavy downpour continues.'
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
return 'The sun is scorching hot.'
|
|
|
|
case WeatherType.STRONG_WINDS:
|
|
|
|
return 'The wind blows intensely.';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-03-10 02:57:33 +00:00
|
|
|
export function getWeatherDamageMessage(weatherType: WeatherType, pokemon: Pokemon): string {
|
2023-04-17 05:46:50 +01:00
|
|
|
switch (weatherType) {
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
return getPokemonMessage(pokemon, ' is buffeted\nby the sandstorm!');
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return getPokemonMessage(pokemon, ' is pelted\nby the hail!');
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-03-10 02:57:33 +00:00
|
|
|
export function getWeatherClearMessage(weatherType: WeatherType): string {
|
2023-04-17 05:46:50 +01:00
|
|
|
switch (weatherType) {
|
|
|
|
case WeatherType.SUNNY:
|
|
|
|
return 'The sunlight faded.';
|
|
|
|
case WeatherType.RAIN:
|
|
|
|
return 'The rain stopped.';
|
|
|
|
case WeatherType.SANDSTORM:
|
|
|
|
return 'The sandstorm subsided.';
|
|
|
|
case WeatherType.HAIL:
|
|
|
|
return 'The hail stopped.';
|
|
|
|
case WeatherType.FOG:
|
|
|
|
return 'The fog disappeared.'
|
|
|
|
case WeatherType.HEAVY_RAIN:
|
|
|
|
return 'The heavy rain stopped.'
|
|
|
|
case WeatherType.HARSH_SUN:
|
|
|
|
return 'The harsh sunlight faded.'
|
|
|
|
case WeatherType.STRONG_WINDS:
|
|
|
|
return 'The heavy wind stopped.';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2023-04-17 01:41:52 +01:00
|
|
|
}
|
|
|
|
|
2024-03-10 02:57:33 +00:00
|
|
|
export function getTerrainStartMessage(terrainType: TerrainType): string {
|
2024-03-18 22:03:13 +00:00
|
|
|
switch (terrainType) {
|
|
|
|
case TerrainType.MISTY:
|
|
|
|
return 'Mist swirled around the battlefield!';
|
|
|
|
case TerrainType.ELECTRIC:
|
|
|
|
return 'An electric current ran across the battlefield!';
|
|
|
|
case TerrainType.GRASSY:
|
|
|
|
return 'Grass grew to cover the battlefield!';
|
|
|
|
case TerrainType.PSYCHIC:
|
|
|
|
return 'The battlefield got weird!';
|
|
|
|
}
|
2024-03-10 02:57:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function getTerrainClearMessage(terrainType: TerrainType): string {
|
2024-03-18 22:03:13 +00:00
|
|
|
switch (terrainType) {
|
|
|
|
case TerrainType.MISTY:
|
|
|
|
return 'The mist disappeared from the battlefield.';
|
|
|
|
case TerrainType.ELECTRIC:
|
|
|
|
return 'The electricity disappeared from the battlefield.';
|
|
|
|
case TerrainType.GRASSY:
|
|
|
|
return 'The grass disappeared from the battlefield.';
|
|
|
|
case TerrainType.PSYCHIC:
|
|
|
|
return 'The weirdness disappeared from the battlefield!';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTerrainBlockMessage(pokemon: Pokemon, terrainType: TerrainType): string {
|
|
|
|
if (terrainType === TerrainType.MISTY)
|
|
|
|
return getPokemonMessage(pokemon, ` surrounds itself with a protective mist!`);
|
|
|
|
return getPokemonMessage(pokemon, ` is protected by the ${Utils.toReadableString(TerrainType[terrainType])} Terrain!`);
|
2024-03-10 02:57:33 +00:00
|
|
|
}
|
|
|
|
|
2023-04-17 01:41:52 +01:00
|
|
|
interface WeatherPoolEntry {
|
|
|
|
weatherType: WeatherType;
|
|
|
|
weight: integer;
|
|
|
|
}
|
|
|
|
|
2023-12-30 02:04:40 +00:00
|
|
|
export function getRandomWeatherType(arena: any /* Importing from arena causes a circular dependency */): WeatherType {
|
2023-04-17 01:41:52 +01:00
|
|
|
let weatherPool: WeatherPoolEntry[] = [];
|
2023-12-30 02:04:40 +00:00
|
|
|
const hasSun = arena.getTimeOfDay() < 2;
|
|
|
|
switch (arena.biomeType) {
|
2023-04-17 01:41:52 +01:00
|
|
|
case Biome.GRASS:
|
|
|
|
weatherPool = [
|
2023-12-30 02:04:40 +00:00
|
|
|
{ weatherType: WeatherType.NONE, weight: 7 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 3 });
|
2023-04-17 01:41:52 +01:00
|
|
|
break;
|
|
|
|
case Biome.TALL_GRASS:
|
|
|
|
weatherPool = [
|
2024-03-28 02:40:10 +00:00
|
|
|
{ weatherType: WeatherType.NONE, weight: 8 },
|
2023-04-17 05:46:50 +01:00
|
|
|
{ weatherType: WeatherType.RAIN, weight: 5 },
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 2 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 8 });
|
2023-04-17 01:41:52 +01:00
|
|
|
break;
|
|
|
|
case Biome.FOREST:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 8 },
|
2023-10-28 22:59:49 +01:00
|
|
|
{ weatherType: WeatherType.RAIN, weight: 5 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.SEA:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 3 },
|
2023-10-28 22:59:49 +01:00
|
|
|
{ weatherType: WeatherType.RAIN, weight: 12 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.SWAMP:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 2 },
|
|
|
|
{ weatherType: WeatherType.RAIN, weight: 5 },
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 8 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.BEACH:
|
|
|
|
weatherPool = [
|
2023-12-14 00:24:10 +00:00
|
|
|
{ weatherType: WeatherType.NONE, weight: 8 },
|
|
|
|
{ weatherType: WeatherType.RAIN, weight: 3 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 5 });
|
2023-04-17 01:41:52 +01:00
|
|
|
break;
|
|
|
|
case Biome.LAKE:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 10 },
|
|
|
|
{ weatherType: WeatherType.RAIN, weight: 5 },
|
2023-10-28 22:59:49 +01:00
|
|
|
{ weatherType: WeatherType.FOG, weight: 3 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.SEABED:
|
|
|
|
weatherPool = [
|
2023-10-28 22:59:49 +01:00
|
|
|
{ weatherType: WeatherType.RAIN, weight: 1 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.MOUNTAIN:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.STRONG_WINDS, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
2023-05-11 15:31:39 +01:00
|
|
|
case Biome.BADLANDS:
|
2023-04-17 01:41:52 +01:00
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 8 },
|
2023-10-28 22:59:49 +01:00
|
|
|
{ weatherType: WeatherType.SANDSTORM, weight: 2 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 5 });
|
2023-04-17 01:41:52 +01:00
|
|
|
break;
|
|
|
|
case Biome.DESERT:
|
|
|
|
weatherPool = [
|
2023-12-30 02:04:40 +00:00
|
|
|
{ weatherType: WeatherType.SANDSTORM, weight: 2 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 2 });
|
2023-04-17 01:41:52 +01:00
|
|
|
break;
|
|
|
|
case Biome.ICE_CAVE:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.HAIL, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.MEADOW:
|
|
|
|
weatherPool = [
|
2023-12-30 02:04:40 +00:00
|
|
|
{ weatherType: WeatherType.NONE, weight: 2 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
2023-12-30 02:04:40 +00:00
|
|
|
if (hasSun)
|
|
|
|
weatherPool.push({ weatherType: WeatherType.SUNNY, weight: 2 });
|
2023-04-17 01:41:52 +01:00
|
|
|
case Biome.VOLCANO:
|
|
|
|
weatherPool = [
|
2023-12-30 02:04:40 +00:00
|
|
|
{ weatherType: hasSun ? WeatherType.SUNNY : WeatherType.NONE, weight: 1 }
|
2023-04-17 01:41:52 +01:00
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.GRAVEYARD:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.RUINS:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 4 },
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.WASTELAND:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 4 },
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
case Biome.ABYSS:
|
|
|
|
weatherPool = [
|
|
|
|
{ weatherType: WeatherType.NONE, weight: 4 },
|
|
|
|
{ weatherType: WeatherType.FOG, weight: 1 }
|
|
|
|
];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (weatherPool.length > 1) {
|
|
|
|
let totalWeight = 0;
|
|
|
|
weatherPool.forEach(w => totalWeight += w.weight);
|
|
|
|
|
2023-10-18 23:01:15 +01:00
|
|
|
const rand = Utils.randSeedInt(totalWeight);
|
2023-04-17 01:41:52 +01:00
|
|
|
let w = 0;
|
|
|
|
for (let weather of weatherPool) {
|
|
|
|
w += weather.weight;
|
|
|
|
if (rand < w)
|
2023-04-17 05:46:50 +01:00
|
|
|
return weather.weatherType;
|
2023-04-17 01:41:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return weatherPool.length
|
2023-04-17 05:46:50 +01:00
|
|
|
? weatherPool[0].weatherType
|
|
|
|
: WeatherType.NONE;
|
2023-04-17 01:41:52 +01:00
|
|
|
}
|