ws4kp/server/scripts/modules/hourly.mjs

210 lines
7.5 KiB
JavaScript
Raw Normal View History

2020-10-21 01:04:51 +00:00
// hourly forecast list
2022-11-22 22:19:10 +00:00
import STATUS from './status.mjs';
import { DateTime, Interval, Duration } from '../vendor/auto/luxon.mjs';
import { json } from './utils/fetch.mjs';
2022-12-06 22:25:28 +00:00
import { celsiusToFahrenheit, kilometersToMiles } from './utils/units.mjs';
2022-11-22 22:19:10 +00:00
import { getHourlyIcon } from './icons.mjs';
import { directionToNSEW } from './utils/calc.mjs';
2022-11-22 22:29:10 +00:00
import WeatherDisplay from './weatherdisplay.mjs';
2022-12-06 22:14:56 +00:00
import { registerDisplay } from './navigation.mjs';
import getSun from './almanac.mjs';
2020-10-21 01:04:51 +00:00
class Hourly extends WeatherDisplay {
constructor(navId, elemId, defaultActive) {
// special height and width for scrolling
2022-11-22 03:50:22 +00:00
super(navId, elemId, 'Hourly Forecast', defaultActive);
2020-10-21 01:04:51 +00:00
// set up the timing
this.timing.baseDelay = 20;
// 24 hours = 6 pages
const pages = 4; // first page is already displayed, last page doesn't happen
2022-07-29 21:12:42 +00:00
const timingStep = 75 * 4;
2020-10-29 21:44:28 +00:00
this.timing.delay = [150 + timingStep];
2020-10-21 01:04:51 +00:00
// add additional pages
2020-10-29 21:44:28 +00:00
for (let i = 0; i < pages; i += 1) this.timing.delay.push(timingStep);
2020-10-21 01:04:51 +00:00
// add the final 3 second delay
this.timing.delay.push(150);
}
async getData(weatherParameters) {
// super checks for enabled
2022-12-07 21:36:02 +00:00
const superResponse = super.getData(weatherParameters);
2020-10-21 01:04:51 +00:00
let forecast;
try {
// get the forecast
2022-11-22 22:19:10 +00:00
forecast = await json(weatherParameters.forecastGridData);
2020-10-21 01:04:51 +00:00
} catch (e) {
console.error('Get hourly forecast failed');
console.error(e.status, e.responseJSON);
2022-12-08 20:41:15 +00:00
if (this.enabled) this.setStatus(STATUS.failed);
// return undefined to other subscribers
this.getDataCallback(undefined);
return;
2020-10-21 01:04:51 +00:00
}
2020-10-29 21:44:28 +00:00
this.data = await Hourly.parseForecast(forecast.properties);
2022-12-07 21:36:02 +00:00
this.getDataCallback();
if (!superResponse) return;
2020-10-21 01:04:51 +00:00
this.setStatus(STATUS.loaded);
this.drawLongCanvas();
}
// extract specific values from forecast and format as an array
2020-10-29 21:44:28 +00:00
static async parseForecast(data) {
const temperature = Hourly.expand(data.temperature.values);
const apparentTemperature = Hourly.expand(data.apparentTemperature.values);
const windSpeed = Hourly.expand(data.windSpeed.values);
const windDirection = Hourly.expand(data.windDirection.values);
const skyCover = Hourly.expand(data.skyCover.values); // cloud icon
const weather = Hourly.expand(data.weather.values); // fog icon
const iceAccumulation = Hourly.expand(data.iceAccumulation.values); // ice icon
const probabilityOfPrecipitation = Hourly.expand(data.probabilityOfPrecipitation.values); // rain icon
const snowfallAmount = Hourly.expand(data.snowfallAmount.values); // snow icon
const icons = await Hourly.determineIcon(skyCover, weather, iceAccumulation, probabilityOfPrecipitation, snowfallAmount, windSpeed);
2020-10-21 01:04:51 +00:00
2022-12-06 22:25:28 +00:00
return temperature.map((val, idx) => ({
temperature: celsiusToFahrenheit(temperature[idx]),
apparentTemperature: celsiusToFahrenheit(apparentTemperature[idx]),
windSpeed: kilometersToMiles(windSpeed[idx]),
windDirection: directionToNSEW(windDirection[idx]),
2022-12-07 21:36:02 +00:00
probabilityOfPrecipitation: probabilityOfPrecipitation[idx],
skyCover: skyCover[idx],
2022-12-06 22:25:28 +00:00
icon: icons[idx],
}));
2020-10-21 01:04:51 +00:00
}
// given forecast paramaters determine a suitable icon
2020-10-29 21:44:28 +00:00
static async determineIcon(skyCover, weather, iceAccumulation, probabilityOfPrecipitation, snowfallAmount, windSpeed) {
2022-11-22 22:19:10 +00:00
const startOfHour = DateTime.local().startOf('hour');
2022-12-06 22:14:56 +00:00
const sunTimes = (await getSun()).sun;
2022-11-22 22:19:10 +00:00
const overnight = Interval.fromDateTimes(DateTime.fromJSDate(sunTimes[0].sunset), DateTime.fromJSDate(sunTimes[1].sunrise));
const tomorrowOvernight = DateTime.fromJSDate(sunTimes[1].sunset);
2020-10-21 01:04:51 +00:00
return skyCover.map((val, idx) => {
2020-10-29 21:44:28 +00:00
const hour = startOfHour.plus({ hours: idx });
2020-10-21 01:04:51 +00:00
const isNight = overnight.contains(hour) || (hour > tomorrowOvernight);
2022-11-22 22:19:10 +00:00
return getHourlyIcon(skyCover[idx], weather[idx], iceAccumulation[idx], probabilityOfPrecipitation[idx], snowfallAmount[idx], windSpeed[idx], isNight);
2020-10-21 01:04:51 +00:00
});
}
// expand a set of values with durations to an hour-by-hour array
2020-10-29 21:44:28 +00:00
static expand(data) {
2022-11-22 22:19:10 +00:00
const startOfHour = DateTime.utc().startOf('hour').toMillis();
2020-10-21 01:04:51 +00:00
const result = []; // resulting expanded values
2020-10-29 21:44:28 +00:00
data.forEach((item) => {
2020-10-21 01:04:51 +00:00
let startTime = Date.parse(item.validTime.substr(0, item.validTime.indexOf('/')));
2022-11-22 22:19:10 +00:00
const duration = Duration.fromISO(item.validTime.substr(item.validTime.indexOf('/') + 1)).shiftTo('milliseconds').values.milliseconds;
2020-10-21 01:04:51 +00:00
const endTime = startTime + duration;
// loop through duration at one hour intervals
do {
// test for timestamp greater than now
if (startTime >= startOfHour && result.length < 24) {
result.push(item.value); // push data array
} // timestamp is after now
// increment start time by 1 hour
2020-10-29 21:44:28 +00:00
startTime += 3600000;
2020-10-21 01:04:51 +00:00
} while (startTime < endTime && result.length < 24);
}); // for each value
return result;
}
2020-10-29 21:44:28 +00:00
async drawLongCanvas() {
2022-07-29 21:12:42 +00:00
// get the list element and populate
const list = this.elem.querySelector('.hourly-lines');
list.innerHTML = '';
2020-10-21 01:04:51 +00:00
2022-11-22 22:19:10 +00:00
const startingHour = DateTime.local();
2020-10-21 01:04:51 +00:00
2022-07-29 21:12:42 +00:00
const lines = this.data.map((data, index) => {
2022-08-03 02:39:27 +00:00
const fillValues = {};
2020-10-21 01:04:51 +00:00
// hour
2020-10-29 21:44:28 +00:00
const hour = startingHour.plus({ hours: index });
const formattedHour = hour.toLocaleString({ weekday: 'short', hour: 'numeric' });
2022-08-03 02:39:27 +00:00
fillValues.hour = formattedHour;
2020-10-21 01:04:51 +00:00
// temperatures, convert to strings with no decimal
const temperature = Math.round(data.temperature).toString().padStart(3);
const feelsLike = Math.round(data.apparentTemperature).toString().padStart(3);
2022-08-03 02:39:27 +00:00
fillValues.temp = temperature;
2020-10-21 01:04:51 +00:00
// only plot apparent temperature if there is a difference
2022-08-03 02:39:27 +00:00
// if (temperature !== feelsLike) line.querySelector('.like').innerHTML = feelsLike;
if (temperature !== feelsLike) fillValues.like = feelsLike;
2020-10-21 01:04:51 +00:00
// wind
let wind = 'Calm';
if (data.windSpeed > 0) {
const windSpeed = Math.round(data.windSpeed).toString();
wind = data.windDirection + (Array(6 - data.windDirection.length - windSpeed.length).join(' ')) + windSpeed;
}
2022-08-03 02:39:27 +00:00
fillValues.wind = wind;
2020-10-21 01:04:51 +00:00
2022-07-29 21:12:42 +00:00
// image
2022-08-03 02:39:27 +00:00
fillValues.icon = { type: 'img', src: data.icon };
2020-10-21 01:04:51 +00:00
2022-08-03 02:39:27 +00:00
return this.fillTemplate('hourly-row', fillValues);
2022-07-29 21:12:42 +00:00
});
2020-10-21 01:04:51 +00:00
2022-07-29 21:12:42 +00:00
list.append(...lines);
}
2020-10-21 01:04:51 +00:00
2022-07-29 21:12:42 +00:00
drawCanvas() {
super.drawCanvas();
2020-10-21 01:04:51 +00:00
this.finishDraw();
}
2022-07-29 21:12:42 +00:00
showCanvas() {
// special to hourly to draw the remainder of the canvas
this.drawCanvas();
2020-10-21 01:04:51 +00:00
super.showCanvas();
}
// screen index change callback just runs the base count callback
screenIndexChange() {
this.baseCountChange(this.navBaseCount);
}
// base count change callback
baseCountChange(count) {
// calculate scroll offset and don't go past end
2022-07-29 21:12:42 +00:00
let offsetY = Math.min(this.elem.querySelector('.hourly-lines').getBoundingClientRect().height - 289, (count - 150));
2020-10-21 01:04:51 +00:00
// don't let offset go negative
if (offsetY < 0) offsetY = 0;
// copy the scrolled portion of the canvas
2022-07-29 21:12:42 +00:00
this.elem.querySelector('.main').scrollTo(0, offsetY);
2020-10-21 01:04:51 +00:00
}
2020-10-29 21:44:28 +00:00
static getTravelCitiesDayName(cities) {
2020-10-21 01:04:51 +00:00
// effectively returns early on the first found date
return cities.reduce((dayName, city) => {
if (city && dayName === '') {
// today or tomorrow
2020-10-29 21:44:28 +00:00
const day = DateTime.local().plus({ days: (city.today) ? 0 : 1 });
2020-10-21 01:04:51 +00:00
// return the day
2020-10-29 21:44:28 +00:00
return day.toLocaleString({ weekday: 'long' });
2020-10-21 01:04:51 +00:00
}
return dayName;
}, '');
}
2022-12-07 21:36:02 +00:00
// make data available outside this class
// promise allows for data to be requested before it is available
async getCurrentData() {
return new Promise((resolve) => {
if (this.data) resolve(this.data);
// data not available, put it into the data callback queue
this.getDataCallbacks.push(() => resolve(this.data));
});
}
2020-10-29 21:44:28 +00:00
}
2022-11-22 22:19:10 +00:00
2022-12-06 22:14:56 +00:00
// register display
2022-12-07 21:36:02 +00:00
const display = new Hourly(2, 'hourly', false);
registerDisplay(display);
export default display.getCurrentData.bind(display);