dockapps/wmWeather/Src/wmWeather.c
2014-10-05 19:18:49 +01:00

1220 lines
30 KiB
C

/*
*
* wmWeather-1.31 (C) 1999 Mike Henderson (mghenderson@lanl.gov)
*
* - Shows Local Weather conditions
*
*
*
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program (see the file COPYING); if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
*
*
*
* ToDo:
*
* - Add a GTK popup window to display data in a nicer way. Currently just use
* xmessage...
*
* - Add "current conditions" graphic (as background?). I.e. one of those little
* cartoons that show clouds or sun with rain or snow, etc. on it...
*
* - Scrolling line to display "sundry" parameters.
*
* - Or maybe auto-switch between panels at some user-defined rate?
*
*
* Changes:
*
* Version 1.31 - released May 4, 1999.
* fixed some conversion bugs in wind speed..
*
* Version 1.30 - released April 13, 1999.
* Fixed a bug whereby the App would crash when trying to gain input
* focus under non-WindowMaker WMs (focus is now grabbed by
* `PointerRoot' not `iconwin').
*
* Added StationID and `time-of-last-update' labels. To do this I needed
* to shrink the fonts down and scrunch them together a bit more.
*
* Added new command line option to change their color;
* -tc <color>
*
* Added code to properly decode wind speed when in MPS.
*
* Fixed bug in beaufort wind speed calcs.
*
* Version 1.29 - released March 17, 1999.
* Reorganized wmgeneral.c and renamed it xutils.c (wmgeneral.h
* -> xutils.h as well ). Also moved it into the same directory as wmWeather.
* Now, the openXwindow is split into 2 parts. You first need to call
* initXwindow(argc, argv). This allows us to check the display depth
* before we commit to a particular pixmap (this will be useful in my
* other DockApps to dynamically set appropriate pixmaps based on depth).
* Got rid of alot of the other routines that I never use.
*
* Added 4 more command line option to set the colors of the text:
*
* -bc <color> for setting the BackGround color.
* -lc <color> for setting the Label color.
* -dc <color> for setting the Data color.
* -wgc <color> for setting the Wind Gust color.
*
* Also cleaned up the pixmap to minimize the number of colors used.
*
* Changed metric toggle to work with a key press (any key).
*
* Added double click support. Now double clicking does the following:
*
* Double Mouse Left: pops up the fully decoded METAR file
* in xmessage.
*
* Double Mouse Middle: Currently undefined.
*
* Double Mouse Right: Forces a new update (i.e. download.)
*
*
* Version 1.28 - released March 9, 1999.
* Changed -celsius (-c) option to -metric (-m). Naming makes more
* sense that way...
*
* Added -W option to display WindChill instead of DewPoint.
* Since Windchill is not always available, we only show it if its
* available. If its not, we paste up DewPoint as default.
*
* Also added -mps option to display wind speed in units of
* meters/second (when in -metric mode).
*
* Version 1.27 - released March 8, 1999.
* fixed bug in speed calculation when wind is gusting.
*
* Version 1.26 - released February 24, 1999.
* Added -delay option.
*
* Version 1.25 - released February 16, 1999.
* Added Wind speeds on the 'Beaufort scale'
* Thanks to Paul Martin <pm@zetnet.net> for this addition.
*
* Version 1.24 - released February 12, 1999.
* Added --passive-ftp option to wget.
*
* Version 1.23 - released February 2, 1999.
* Few more bug fixes...
* Added support for different Pressure units...
*
*
* Version 1.22 - released February 1, 1999.
* Fixed minor bug in direction abbreviations. Added a bit more to man
* page.
*
* Version 1.21 - released January 29, 1999.
* Fixed a problem in the perl script. Made the file paths absolute.
*
* Version 1.2 - released January 29, 1999.
* Added Wind speed line. Ended up decoding the Raw METAR line.
* Fixed a few bugs...
* Changed location of files from /tmp to ~/.wmWeatherReports
* Changed units of pressure and wind speed to mmHg and km/h
* when Metric is set. (Really should change the flag to -metric).
*
* Version 1.1 - released January 25, 1999.
* Bug fixes...
* Added command line switch to display Temp's in deg. C
*
*
* Version 1.0 - released January 19, 1999.
*
*
*/
/*
* Includes
*/
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <time.h>
#include <X11/X.h>
#include <X11/xpm.h>
#include "xutils.h"
#include "wmWeather_master.xpm"
#include "wmWeather_mask.xbm"
/*
* Delay between refreshes (in microseconds)
*/
#define DELAY 10000L
#define WMWEATHER_VERSION "1.31"
#define DEFAULT_UPDATEDELAY 900L
void ParseCMDLine(int argc, char *argv[]);
void ButtonPressEvent(XButtonEvent *);
void KeyPressEvent(XKeyEvent *);
char *StringToUpper(char *);
char StationID[10];
int UpToDate = 0;
int ShowWindChill = 0;
int WindChillAvail = 1;
int Metric = 0;
int Beaufort = 0;
int ForceUpdate = 1;
int ForceDownload = 1;
int PressureUnits = 0;
int MetersPerSecond = 0;
double PressureConv = 1.0;
long UpdateDelay;
int GotFirstClick1, GotDoubleClick1;
int GotFirstClick2, GotDoubleClick2;
int GotFirstClick3, GotDoubleClick3;
int DblClkDelay;
/*
* In a more readable form the Compass directions are:
* static char *CompassDirection[] = { "N", "NNE", "NE", "ENE", "E", "ESE", "SE", "SSE",
* "S", "SSW", "SW", "WSW", "W", "WNW", "NW", "NNW"}
* We convert to digits in the sequence `NWSE' so we dont have to put all these
* combinations into the pixmap.
*/
static char *CompassDirection[] = { "0", "003", "03", "303", "3", "323", "23", "223",
"2", "221", "21", "121", "1", "101", "01", "001"};
char LabelColor[30] = "#79bdbf";
char WindGustColor[30] = "#ff0000";
char DataColor[30] = "#ffbf50";
char BackColor[30] = "#181818";
char StationTimeColor[30] = "#c5a6ff";
/*
* main
*/
int main(int argc, char *argv[]) {
struct tm *tTime;
XEvent event;
int n, s, m, dt1, dt2, dt3, yd;
int i, j, len;
char dir[5];
int Year, Month, Day;
int Hours, Mins, Secs;
int UpdateLTHour = 0.0, UpdateLTMin = 0.0, UpdateUTHour, UpdateUTMin;
long CurrentLocalTime;
double UpdateUT, UpdateLT, UT, LT, DT, hour24();
double jd(), CurrentJD;
char command[1024], Line[512], FileName[10];
int Tens, q, digit, chr;
double Pressure, Temperature, sgn, Humidity, DewPoint, WindChill, val;
double Direction, Speed;
FILE *fp;
/*
* Parse any command line arguments.
*/
ParseCMDLine(argc, argv);
/*
* Do the window opening in 2 stages. After the initXwindow() call,
* we know what the Depth of the Display is. We can then pick an appropriate
* XPM to use. I.e. may want to use one that has fewer colors to make the App work
* better on a low-color 8-bit display.
*/
initXwindow(argc, argv);
openXwindow(argc, argv, wmWeather_master, wmWeather_mask_bits, wmWeather_mask_width,
wmWeather_mask_height, BackColor, LabelColor, WindGustColor, DataColor, StationTimeColor);
/*
* Loop until we die
*/
n = 32000;
s = 32000;
m = 32000;
dt1 = 32000;
dt2 = 32000;
dt3 = 32000;
DblClkDelay = 32000;
UpToDate = 0;
while(1) {
/*
* Keep track of # of seconds
*/
if (m > 100){
m = 0;
++dt1;
++dt2;
++dt3;
} else {
/*
* Increment counter
*/
++m;
}
/*
* Double Click Delays
* Keep track of click events. If Delay too long, set GotFirstClick's to False.
*/
if (DblClkDelay > 15) {
DblClkDelay = 0;
GotFirstClick1 = 0; GotDoubleClick1 = 0;
GotFirstClick2 = 0; GotDoubleClick2 = 0;
GotFirstClick3 = 0; GotDoubleClick3 = 0;
} else {
++DblClkDelay;
}
/*
* Process any pending X events.
*/
while(XPending(display)){
XNextEvent(display, &event);
switch(event.type){
case Expose:
RedrawWindow();
break;
case ButtonPress:
ButtonPressEvent(&event.xbutton);
break;
case KeyPress:
KeyPressEvent(&event.xkey);
break;
case ButtonRelease:
break;
case EnterNotify:
XSetInputFocus(display, PointerRoot, RevertToParent, CurrentTime);
break;
case LeaveNotify:
XSetInputFocus(display, PointerRoot, RevertToParent, CurrentTime);
break;
}
}
/*
* Check the Current Conditions file every (approx.) several seconds.
* Can significantly reduce this frequency later. But its
* easier to debug this way...
* Do this before trying to download again! The file may be there and it
* may be Up-To-Date!
*/
if ((dt2 > 5)||(ForceUpdate)){
dt2 = 0;
/*
* Compute Current Julian Date
*/
CurrentLocalTime = time(CurrentTime);
tTime = gmtime(&CurrentLocalTime);
Year = tTime->tm_year+1900;
Month = tTime->tm_mon+1;
Day = tTime->tm_mday;
Hours = tTime->tm_hour;
Mins = tTime->tm_min;
Secs = tTime->tm_sec;
UT = (double)Hours + (double)Mins/60.0 + (double)Secs/3600.0;
CurrentJD = jd(Year, Month, Day, UT);
CurrentLocalTime = time(CurrentTime);
tTime = localtime(&CurrentLocalTime);
Year = tTime->tm_year+1900;
Month = tTime->tm_mon+1;
Day = tTime->tm_mday;
Hours = tTime->tm_hour;
Mins = tTime->tm_min;
Secs = tTime->tm_sec;
LT = (double)Hours + (double)Mins/60.0 + (double)Secs/3600.0;
DT = UT - LT;
if (DT > 24.0) DT -= 24.0;
if (DT < 0.00) DT += 24.0;
/*
* Read in weather data
*/
sprintf(FileName, "%s/.wmWeatherReports/%s.dat", getenv("HOME"), StationID);
if ((fp = fopen(FileName, "r")) != NULL){
fgets(Line, 512, fp);
fgets(Line, 512, fp);
fgets(Line, 512, fp);
fscanf(fp, "%d:%d", &UpdateUTHour, &UpdateUTMin);
if (UpdateUTHour != 99){
UpdateUT = UpdateUTHour + UpdateUTMin/60.0;
UpdateLT = UpdateUT - DT;
if (UpdateLT < 0.0) UpdateLT += 24.0;
if (UpdateLT > 24.0) UpdateLT -= 24.0;
UpdateLTHour = (int)UpdateLT;
UpdateLTMin = (int)((UpdateLT - (double)UpdateLTHour)*60.0 + 0.5);
if (UpdateLTMin >= 60){
++UpdateLTHour;
if (UpdateLTHour >= 24) UpdateLTHour = 0;
UpdateLTMin = 0;
}
} else {
UpdateLTHour = 99;
UpdateLTMin = 99;
}
fscanf(fp, "%lf", &Temperature);
if (Metric) Temperature = (Temperature-32.0)*5.0/9.0;
fscanf(fp, "%lf", &DewPoint);
if (Metric) DewPoint = (DewPoint-32.0)*5.0/9.0;
fscanf(fp, "%lf", &WindChill);
/*
* If WindChill is not available, revert to DewPoint
*/
WindChillAvail = (WindChill < -900.0) ? 0 : 1;
if (Metric) WindChill = (WindChill-32.0)*5.0/9.0;
fscanf(fp, "%lf", &Pressure); Pressure += 0.005;
Pressure *= PressureConv;
fscanf(fp, "%lf", &Humidity);
fscanf(fp, "%lf", &Direction);
fscanf(fp, "%lf", &Speed);
if (Metric){
if (MetersPerSecond) Speed *= 0.4473;
else if (!Beaufort) Speed *= 1.609;
}
fclose(fp);
} else {
Temperature = -9999.0;
DewPoint = -9999.0;
WindChill = -9999.0;
Humidity = -9999.0;
Pressure = -9999.0;
Direction = -9999.0;
Speed = -9999.0;
}
}
/*
* Draw window.
*/
if ( (dt3 > 5) || ForceUpdate){
dt3 = 0;
/*
* Clear window.
*/
copyXPMArea(5, 69, 54, 54, 5, 5);
/*
* Paste up Station ID and time of last update.
*/
q = 0;
chr = (int)StationID[0] - 65; copyXPMArea(chr*5+2, 128, 5, 6, 7+q, 6); q+= 5;
chr = (int)StationID[1] - 65; copyXPMArea(chr*5+2, 128, 5, 6, 7+q, 6); q+= 5;
chr = (int)StationID[2] - 65; copyXPMArea(chr*5+2, 128, 5, 6, 7+q, 6); q+= 5;
chr = (int)StationID[3] - 65; copyXPMArea(chr*5+2, 128, 5, 6, 7+q, 6); q+= 5;
if (UpdateLTHour != 99){
q = 0;
Tens = (int)(UpdateLTHour);
copyXPMArea(Tens/10*5+2, 135, 5, 6, 36+q, 6); q+= 5;
copyXPMArea(Tens%10*5+2, 135, 5, 6, 36+q, 6); q+= 5;
copyXPMArea(53, 135, 1, 6, 36+q, 6); q+= 2;
Tens = (int)(UpdateLTMin);
copyXPMArea(Tens/10*5+2, 135, 5, 6, 36+q, 6); q+= 5;
copyXPMArea(Tens%10*5+2, 135, 5, 6, 36+q, 6); q+= 5;
}
/*
* Paste up Temperature.
*/
if ((Temperature > -999.0)&&(Temperature < 1000.0)){
sgn = (Temperature < 0.0) ? -1.0 : 1.0;
Temperature *= sgn;
Temperature = (double)((int)(Temperature + 0.5));
q = 0;
if (Temperature >= 100.0){
if (sgn < 0.0) { copyXPMArea(0*5+66, 35, 5, 6, 25, 15); q += 5; }
digit = (int)(Temperature/100.0);
copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 15); q+= 5;
Tens = (int)(Temperature-digit*100.0);
copyXPMArea(Tens/10*5+66, 57, 5, 6, 25+q, 15); q+= 5;
copyXPMArea(Tens%10*5+66, 57, 5, 6, 25+q, 15); q+= 5;
} else {
if (sgn < 0.0) { copyXPMArea(0*5+66, 35, 5, 6, 25, 15); q += 5; }
Tens = (int)(Temperature);
if (Tens >= 10) { copyXPMArea(Tens/10*5+66, 57, 5, 6, 25+q, 15); q+= 5; }
copyXPMArea(Tens%10*5+66, 57, 5, 6, 25+q, 15); q+= 5;
}
if (Metric){
copyXPMArea(72, 34, 3, 3, 25+q, 14); q += 4;
copyXPMArea(81, 35, 4, 6, 25+q, 15);
} else {
copyXPMArea(72, 34, 8, 7, 25+q, 14);
}
}
/*
* Paste up DewPoint or WindChill Temperature.
*/
if (ShowWindChill && WindChillAvail){
val = WindChill;
copyXPMArea(66, 87, 17, 8, 5, 24);
} else {
val = DewPoint;
copyXPMArea(5, 87, 17, 8, 5, 24);
}
if ((val > -999.0)&&(val < 1000.0)){
sgn = (val < 0.0) ? -1.0 : 1.0;
val *= sgn;
val = (double)((int)(val + 0.5));
q = 0;
if (val >= 100.0){
if (sgn < 0.0) { copyXPMArea(0*5+66, 57, 5, 6, 25, 24); q += 5; }
digit = (int)(val/100.0);
copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 24); q+= 5;
Tens = (int)(val-digit*100.0);
copyXPMArea(Tens/10*5+66, 57, 5, 6, 25+q, 24); q+= 5;
copyXPMArea(Tens%10*5+66, 57, 5, 6, 25+q, 24); q+= 5;
} else {
if (sgn < 0.0) { copyXPMArea(0*5+66, 35, 5, 6, 25, 24); q += 5; }
Tens = (int)(val);
if (Tens >= 10) { copyXPMArea(Tens/10*5+66, 57, 5, 6, 25+q, 24); q+= 5; }
copyXPMArea(Tens%10*5+66, 57, 5, 6, 25+q, 24); q+= 5;
}
if (Metric){
copyXPMArea(72, 34, 3, 3, 25+q, 23); q += 4;
copyXPMArea(81, 35, 4, 6, 25+q, 24);
} else {
copyXPMArea(72, 34, 8, 7, 25+q, 23);
}
}
/*
* Paste up Pressure.
*/
q = 0; s= 0;
if ((Pressure > 0.0)&&(Pressure <= 10000.0)){
val = Pressure;
digit = (int)(val/1000.0);
if (digit > 0) { copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5; }
val -= (double)digit*1000;
digit = (int)(val/100.0);
if ((digit > 0)||(Pressure > 999.0)) { copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5; }
val -= (double)digit*100;
digit = (int)(val/10.0);
if ((digit > 0)||(Pressure > 99.0)) { copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5; }
val -= (double)digit*10;
digit = (int)val;
copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5;
val -= (double)digit;
if ((PressureUnits != 2)||(!Metric)){
copyXPMArea(10*5+66+1, 57, 4, 6, 25+q, 33); q += 4;
val *= 10; digit = (int)val;
copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5;
val -= (double)digit;
if (!Metric){
val *= 10; digit = (int)val;
copyXPMArea(digit*5+66, 57, 5, 6, 25+q, 33); q += 5;
/* val -= (double)digit; */
}
}
}
/*
* Paste up Humidity.
*/
q = 0;
if ((Humidity > -999.0)&&(Humidity <= 100.0)){
if (Humidity == 100.0){
copyXPMArea(1*5+66, 57, 5, 6, 25+q, 42); q += 5;
copyXPMArea(0*5+66, 57, 5, 6, 25+q, 42); q += 5;
copyXPMArea(0*5+66, 57, 5, 6, 25+q, 42); q += 5;
} else {
Tens = (int)(Humidity);
if (Tens >= 10) { copyXPMArea(Tens/10*5+66, 57, 5, 6, 25+q, 42); q += 5; }
copyXPMArea(Tens%10*5+66, 57, 5, 6, 25+q, 42); q += 5;
}
copyXPMArea(121, 57, 5, 6, 25+q, 42);
}
/*
* Paste up Wind Info.
*/
if ((Direction == 0.0)&&(Speed == 0.0)){
/*
* Just write out `Calm' if both values are 0
*/
copyXPMArea(66, 4, 23, 7, 25, 51);
} else {
/*
* If the Direction < 0 this means that there was a "Wind direction
* Variability Group" found in the METAR code. I.e. Direction is variable.
* I think value should be the average. In any case flag it by making it a
* different color.
*
* Likewise, if Speed < 0, this means that a "Gusty" modifier was found
* in the RAW METAR code, and the value is only the average of the low
* and high values given. Again, flag it in a different color...
*
*/
if ((Direction >= -360.0)&&(Direction <= 360.0)){
sgn = (Direction < 0.0) ? -1.0 : 1.0;
yd = (sgn < 0.0) ? 50 : 43;
Direction *= sgn;
q = 0;
i = (int)(Direction/360.0*16.0 + 0.5);
if (i>15) i = 0;
strcpy(dir, CompassDirection[i]);
len = strlen(dir);
for (j=0; j<len; ++j){
digit = (int)dir[j] - 48;
copyXPMArea(digit*5+66, yd, 5, 6, 25+q, 51); q+= 5;
}
q += 2;
} else if (Direction > 0.0){
/*
* In this case, the wind direction is variable with speed < 6 Knots.
* A numerical direction is not given in these cases. Just write out 'VRB'.
*/
q = 0;
copyXPMArea(4*5+66, 43, 5, 6, 25+q, 51); q+= 5;
copyXPMArea(5*5+66, 43, 5, 6, 25+q, 51); q+= 5;
copyXPMArea(6*5+66, 43, 5, 6, 25+q, 51); q+= 9;
}
if (Metric && Beaufort) {
int beau = 0;
int spd;
sgn = (Speed < 0.0) ? -1.0 : 1.0;
spd = (int)(sgn * (int)Speed);
if (spd > 1) { beau = 1; }
if (spd > 3) { beau = 2; }
if (spd > 4) { beau = 3; }
if (spd > 10) { beau = 4; }
if (spd > 16) { beau = 5; }
if (spd > 21) { beau = 6; }
if (spd > 27) { beau = 7; }
if (spd > 33) { beau = 8; }
if (spd > 40) { beau = 9; }
if (spd > 47) { beau = 10; }
if (spd > 55) { beau = 11; }
if (spd > 63) { beau = 12; }
if (spd > 71) { beau = 13; }
Speed = sgn * (double) beau;
q++; copyXPMArea(76, 35, 4, 6, 25+q, 51); q+= 6;
}
if ((Speed > -999.0)&&(Speed < 1000.0)){
sgn = (Speed < 0.0) ? -1.0 : 1.0;
yd = (sgn < 0.0) ? 64 : 57;
Speed *= sgn;
if (Speed >= 100.0){
digit = (int)(Speed/100.0);
copyXPMArea(digit*5+66, yd, 5, 6, 25+q, 51); q+= 5;
Tens = (int)(Speed-digit*100.0);
copyXPMArea(Tens/10*5+66, yd, 5, 6, 25+q, 51); q+= 5;
copyXPMArea(Tens%10*5+66, yd, 5, 6, 25+q, 51); q+= 5;
} else {
Tens = (int)(Speed);
if (Tens >= 10) { copyXPMArea(Tens/10*5+66, yd, 5, 6, 25+q, 51); q+= 5; }
copyXPMArea(Tens%10*5+66, yd, 5, 6, 25+q, 51); q+= 5;
}
}
}
/*
* Make changes visible
*/
RedrawWindow();
}
/*
* Reset "force update" flag
*/
ForceUpdate = 0;
/*
* Check every 5 min if the values are not up to date...
*/
/*
* We still need to add a flashing LED to warn about
* times that are out of date. Also need to determine if it is uptodate...
*/
UpToDate = 0;
if (((!UpToDate)&&(dt1 > UpdateDelay)) || ForceDownload){
dt1 = 0;
/*
* Execute Perl script to grab the Latest METAR Report
*/
sprintf(command, "GrabWeather %s &", StationID);
system(command);
ForceDownload = 0;
ForceUpdate = 1;
}
/*
* Wait for next update
*/
usleep(DELAY);
}
}
/*
* ParseCMDLine()
*/
void ParseCMDLine(int argc, char *argv[]) {
int i;
void print_usage();
StationID[0] = '\0';
PressureUnits = 0;
MetersPerSecond = 0;
ShowWindChill = 0;
UpdateDelay = DEFAULT_UPDATEDELAY;
for (i = 1; i < argc; i++) {
if (!strcmp(argv[i], "-display")){
++i;
} else if (!strcmp(argv[i], "-bc")){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No color found\n");
print_usage();
exit(-1);
}
strcpy(BackColor, argv[++i]);
} else if (!strcmp(argv[i], "-tc")){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No color found\n");
print_usage();
exit(-1);
}
strcpy(StationTimeColor, argv[++i]);
} else if (!strcmp(argv[i], "-lc")){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No color found\n");
print_usage();
exit(-1);
}
strcpy(LabelColor, argv[++i]);
} else if (!strcmp(argv[i], "-wgc")){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No color found\n");
print_usage();
exit(-1);
}
strcpy(WindGustColor, argv[++i]);
} else if (!strcmp(argv[i], "-dc")){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No color found\n");
print_usage();
exit(-1);
}
strcpy(DataColor, argv[++i]);
} else if (!strcmp(argv[i], "-beaufort")){
Beaufort = 1;
} else if (!strcmp(argv[i], "-mps")){
MetersPerSecond = 1;
} else if (!strcmp(argv[i], "-W")){
ShowWindChill = 1;
} else if ((!strcmp(argv[i], "-metric"))||(!strcmp(argv[i], "-m"))){
Metric = 1;
} else if (!strcmp(argv[i], "-kPa")){
PressureUnits = 1;
PressureConv = 3.38639;
} else if (!strcmp(argv[i], "-hPa")){
PressureUnits = 2;
PressureConv = 33.8639;
} else if (!strcmp(argv[i], "-mmHg")){
PressureUnits = 3;
PressureConv = 25.4;
} else if ((!strcmp(argv[i], "-station"))||(!strcmp(argv[i], "-s"))){
if ((i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr, "wmWeather: No METAR station ID found\n");
print_usage();
exit(-1);
}
strcpy(StationID, StringToUpper(argv[++i]));
} else if (!strcmp(argv[i], "-delay")) {
if( (i+1 >= argc)||(argv[i+1][0] == '-')) {
fprintf(stderr,"You must give a time with the -delay option.\n");
print_usage();
exit(-1);
} else if(sscanf(argv[i+1], "%ld", &UpdateDelay) != 1) {
fprintf(stderr,"Dont understand the delay time you have entered (%s).\n", argv[i+1]);
print_usage();
exit(-1);
}
/*
* Convert Time to seconds
*/
UpdateDelay *= 60;
++i;
} else {
print_usage();
exit(-11);
}
}
if (StationID[0] == '\0') {
fprintf(stderr, "\nwmWeather: You must specify a METAR station code\n\n");
print_usage();
exit(1);
}
if ((Metric)&&(PressureUnits == 0)){
PressureUnits = 3;
PressureConv = 25.4;
}
if (!Metric){
PressureConv = 1.0;
}
if (Beaufort && MetersPerSecond){
fprintf(stderr, "\nwmWeather: You cant use both -beaufort and -mps together.\n\n");
print_usage();
exit(1);
}
}
void print_usage(){
printf("\nwmWeather version: %s\n", WMWEATHER_VERSION);
printf("\nusage: wmWeather -s <StationID> [-display <Display>] [-h] [-metric] [-kPa] [-hPa] [-mmHg]\n");
printf(" [-beaufort] [-mps] [-W] [-delay <Time in Minutes>] [-bc <color>]\n");
printf(" [-lc <color>] [-dc <color>] [-wgc <color>] [-tc <color>]\n\n");
printf("\t-display <Display>\t\tUse alternate X display.\n\n");
printf("\t-h\t\t\t\tDisplay help screen.\n");
printf("\n\t-station <METAR StationID>\n");
printf("\t-s <METAR StationID>\tThe 4-character METAR Station ID.\n\n");
printf("\t-W\t\t\t\tIf available, display WindChill instead\n");
printf("\t\t\t\t\tof DewPoint Temperature.\n");
printf("\t-metric\n");
printf("\t-m\t\t\t\tDisplay variables in metric units.\n\n");
printf("\t-kPa\t\t\t\tWhen toggled to metric display, show pressure\n");
printf("\t\t\t\t\tin units of kPa.\n\n");
printf("\t-hPa\t\t\t\tWhen toggled to metric display, show pressure\n");
printf("\t\t\t\t\tin units of hPa.\n\n");
printf("\t-mmHg\t\t\t\tWhen toggled to metric display, show pressure\n");
printf("\t\t\t\t\tin units of mmHg. (This is the default for metric).\n\n");
printf("\t-beaufort\t\t\tWhen toggled to metric display, show windspeed\n");
printf("\t\t\t\t\ton the Beaufort scale. (default is km/h.)\n\n");
printf("\t-mps\t\t\t\tWhen toggled to metric display, show windspeed\n");
printf("\t\t\t\t\tin units of meters/second. (default is km/h.)\n\n");
printf("\t-bc <color>\t\t\tBackground color. (#7e9e69 is a greenish LCD color).\n\n");
printf("\t-lc <color>\t\t\tLabel color.\n\n");
printf("\t-dc <color>\t\t\tData color.\n\n");
printf("\t-wgc <color>\t\t\tGusty-wind/variable-direction color.\n\n");
printf("\t-tc <color>\t\t\tStation ID and Time color.\n\n");
printf("\t-delay <Time in Minutes>\tOverride time (in minutes) between updates (default\n");
printf("\t\t\t\t\tis %ld minutes). (Times are approximate.)\n", DEFAULT_UPDATEDELAY/60);
printf("\n\nTo find out more about the METAR/TAF system and to find the \n");
printf("METAR code for your location, look at:\n\n");
printf(" http://www.nws.noaa.gov/oso/oso1/oso12/metar.htm \n\n");
printf("for NOAA's ""National Weather Service METAR/TAF Information"" page.\n");
printf("To locate your site ID go to NOAA's ""Meteorological Station Information\nLookup"" page at:\n\n");
printf(" http://www.nws.noaa.gov/oso/siteloc.shtml\n\n");
}
/*
* Compute the Julian Day number for the given date.
* Julian Date is the number of days since noon of Jan 1 4713 B.C.
*/
double jd(ny, nm, nd, UT)
int ny, nm, nd;
double UT;
{
double A, B, C, D, JD, day;
day = nd + UT/24.0;
if ((nm == 1) || (nm == 2)){
ny = ny - 1;
nm = nm + 12;
}
if (((double)ny+nm/12.0+day/365.25)>=(1582.0+10.0/12.0+15.0/365.25)){
A = ((int)(ny / 100.0));
B = 2.0 - A + (int)(A/4.0);
}
else{
B = 0.0;
}
if (ny < 0.0){
C = (int)((365.25*(double)ny) - 0.75);
}
else{
C = (int)(365.25*(double)ny);
}
D = (int)(30.6001*(double)(nm+1));
JD = B + C + D + day + 1720994.5;
return(JD);
}
/*
* This routine handles button presses.
*
* - Left Mouse single click toggles Deg F/C for temperatures.
* - Some other click event should display the full METAR report -- lots of
* juicy stuff in there... Should bring up a separate window...
*
*
*/
void ButtonPressEvent(XButtonEvent *xev){
char Command[80];
/*
* Process single clicks.
*/
DblClkDelay = 0;
if ((xev->button == Button1) && (xev->type == ButtonPress)){
if (GotFirstClick1) GotDoubleClick1 = 1;
else GotFirstClick1 = 1;
} else if ((xev->button == Button2) && (xev->type == ButtonPress)){
if (GotFirstClick2) GotDoubleClick2 = 1;
else GotFirstClick2 = 1;
} else if ((xev->button == Button3) && (xev->type == ButtonPress)){
if (GotFirstClick3) GotDoubleClick3 = 1;
else GotFirstClick3 = 1;
}
/*
* We got a double click on Mouse Button1 (i.e. the left one)
*/
if (GotDoubleClick1) {
GotFirstClick1 = 0;
GotDoubleClick1 = 0;
sprintf(Command, "xmessage -center -file %s/.wmWeatherReports/%s.TXT &", getenv("HOME"), StationID);
system(Command);
}
/*
* We got a double click on Mouse Button2 (i.e. the left one)
*/
if (GotDoubleClick2) {
GotFirstClick2 = 0;
GotDoubleClick2 = 0;
}
/*
* We got a double click on Mouse Button3 (i.e. the left one)
*/
if (GotDoubleClick3) {
GotFirstClick3 = 0;
GotDoubleClick3 = 0;
ForceDownload = 1;
ForceUpdate = 1;
}
return;
}
/*
* This routine handles key presses.
*
*/
void KeyPressEvent(XKeyEvent *xev){
Metric = !Metric;
if (Metric){
switch(PressureUnits){
case 0: PressureConv = 25.4;
break;
case 1: PressureConv = 3.38639;
break;
case 2: PressureConv = 33.8639;
break;
case 3: PressureConv = 25.4;
break;
}
} else if (!Metric){
PressureConv = 1.0;
}
ForceUpdate = 1;
return;
}
char *StringToUpper(char *String) {
int i;
for (i = 0; i < strlen(String); i++)
String[i] = toupper(String[i]);
return String;
}