beacon/src/forecast.rs

80 lines
2.9 KiB
Rust

use std::{collections::HashMap, f32::consts::PI};
use chrono::{DateTime, Duration, Timelike, Utc};
use crate::clock;
/// A forecast is used to divine specific weather patterns for a zone.
#[derive(Debug)]
pub struct Forecast {
pub zone_id: u32,
pub rates: Vec<Rate>,
}
/// A weather rate consists of the Weather ID and the likelihood in which it occurs (also called "target").
/// The weather algorithm that the game uses generates a random integer between 0 and 100, and whichever rate
/// first satisfies the "is larger than target" condition gets selected.
/// This means that rates are:
/// 1. Sorted ascending by rate
/// 2. The larger the rate, the less likely the weather is to occur, statistically
#[derive(Debug)]
pub struct Rate {
pub weather_id: u32,
pub rate: u32,
}
/// A hash map of zones, indexed by "rate ID", which is unique in our source data.
pub type ForecastSet = HashMap<u32, Forecast>;
impl Forecast {
pub fn weather_for_target(&self, target: u32) -> &Rate {
// TODO: Don't unwrap here!
self.rates.iter().find(|r| target < r.rate).unwrap()
}
pub fn weather_now(&self) -> &Rate {
let utc = Utc::now();
let target = calculate_target(utc);
self.weather_for_target(target)
}
/// Returns the weather `n` cycles before or after the current weather.
pub fn nth_weather(&self, n: i32) -> &Rate {
let last_weather_time = round_to_last_weather_time(&clock::get_current_eorzea_date());
let hours = n * 8;
let td = Duration::hours(hours.abs() as i64);
let new_date = if hours > 0 {
last_weather_time.checked_add_signed(td).unwrap()
} else {
last_weather_time.checked_sub_signed(td).unwrap()
};
let target = calculate_target(new_date);
self.weather_for_target(target)
}
}
/// Rounds to the last weather "start". These happen three times a day, at 0:00,
/// at 8:00, and at 16:00.
pub fn round_to_last_weather_time(date: &DateTime<Utc>) -> DateTime<Utc> {
let cur_hour = date.hour();
// This essentially performs division without rest.
let last_hour = (cur_hour / 8) * 8;
date.date_naive()
.and_hms_opt(last_hour, date.minute(), date.second())
.unwrap()
.and_utc()
}
/// Calculates the magic target number for a specific date.
/// It's important to note that this expects a human time, not an Eorzean time.
pub fn calculate_target(m: DateTime<Utc>) -> u32 {
let unix_time = m.timestamp().abs();
let ez_hour = unix_time / 175;
let inc = (ez_hour + 8 - (ez_hour % 8)) % 24;
let total_days = unix_time / 4200;
let calc_base: i32 = ((total_days * 100) + inc) as i32; // Needs to be i32 to assure correct shifting overflow!
let step1 = ((calc_base << 11) ^ calc_base) as u32; // Cast to u32 here to literally interpret the two's complement, I suppose
let step2 = (step1 >> 8) ^ step1;
step2 % 100
}