dockapps/wmsun/wmsun.c
Doug Torrance a835dae3aa wmsun: Fix need to restart for Daylight Savings change.
Based on patch by Martin Stigge <martin@stigge.org> to fix Debian
bug #414489 [1].

[1] https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=414489
2015-06-14 23:23:24 +01:00

377 lines
7.9 KiB
C

/*
*
* wmSun-1.03 (C) 1999 Mike Henderson (mghenderson@lanl.gov)
*
* - Shows Sun Rise/Set Times....
*
*
*
*
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA
*
* Things TODO:
* - clean up code!
* - support for 8-bit displays.
* - more detailed documentation.
* - eclipses?
* - add buttons to play will date and lat lon...
* Could be something like this;
* First click brings up buttons to change date.
* Second click brings up buttons to change lat/lon.
* Third goes back to display
* Set time delay to go back to display if user doesnt do it...
*/
/*
* Includes
*/
#define _POSIX_C_SOURCE 1
#include <stdio.h>
#include <math.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <X11/X.h>
#include <X11/xpm.h>
#include "wmgeneral/wmgeneral.h"
#include "wmSun_master.xpm"
#include "wmSun_mask.xbm"
/*
* Delay between refreshes (in microseconds)
*/
#define DELAY 1000000L
#define WMSUN_VERSION "1.03"
#define DegPerRad 57.29577951308232087680
#define RadPerDeg 0.01745329251994329576
void ParseCMDLine(int argc, char *argv[]);
void pressEvent(XButtonEvent *xev);
void SunRise(int year, int month, int day, double LocalHour, double *UTRise,
double *UTSet);
int ToggleWindow = 0;
int nMAX = 1;
int Flag = 1;
int UseUserTimeDiff = 0;
int UseUserDate = 0;
long UserDate;
double Glat, Glon, SinGlat, CosGlat, TimeZone, UserTimeDiff;
int TwelveHour = 0;
int xDigit[11] = {8, 18, 27, 37, 46, 55, 64, 74, 83, 92, 102};
/*
* main
*/
int main(int argc, char *argv[]) {
struct tm *GMTTime, *LocalTime;
XEvent event;
int n;
int Year, Month;
int DayOfMonth;
long CurrentLocalTime, CurrentGMTTime, date;
double UT, val, LTRise, LTSet, LocalHour, hour24();
int H, M;
struct timeval timeout;
fd_set xfdset;
/*
* Parse any command line arguments.
*/
Glat = Glon = 0.0;
ParseCMDLine(argc, argv);
Glat *= RadPerDeg; SinGlat = sin( Glat ); CosGlat = cos( Glat );
openXwindow(argc, argv, wmSun_master, (char *)wmSun_mask_bits, wmSun_mask_width, wmSun_mask_height);
/*
* Loop until we die
*/
n = 32000;
while(1) {
if (Flag) {
n = 32000;
Flag = 0;
}
/*
* The Moon Ephemeris calculations are somewhat costly (the Moon is one of the most
* difficult objects to compute position for). So only process every nMAXth cycle of this
* loop. We run outer loop it faster to catch expose events, button presses, etc...
*
*/
if (n>nMAX){
struct tm result;
n = 0;
nMAX = 60;
CurrentGMTTime = time(CurrentTime);
GMTTime = gmtime_r(&CurrentGMTTime, &result);
DayOfMonth = GMTTime->tm_mday;
UT = GMTTime->tm_hour + GMTTime->tm_min/60.0 + GMTTime->tm_sec/3600.0;
Year = GMTTime->tm_year+1900;
Month = GMTTime->tm_mon+1;
CurrentLocalTime = CurrentGMTTime;
LocalTime = localtime_r(&CurrentLocalTime, &result);
Flag = 0;
if (UseUserDate){
date = UserDate;
Year = date/10000;
date -= Year*10000;
Month = date/100;
date -= Month*100;
DayOfMonth = date;
date = UserDate;
} else {
date = Year*10000 + Month*100 + DayOfMonth;
}
LocalHour = LocalTime->tm_hour + LocalTime->tm_min/60.0 + LocalTime->tm_sec/3600.0;
TimeZone = (UseUserTimeDiff) ? UserTimeDiff : UT - LocalHour;
/*
* Clear Plotting area
*/
copyXPMArea(65, 5, 54, 54, 5, 5);
/*
* Compute Sun Rise/Set Times in Local Time
*/
SunRise(Year, Month, DayOfMonth, LocalHour, &LTRise, &LTSet);
if (LTRise > 0.0){
val = LTRise;
H = (int)val; val = (val-H)*60.0;
if (TwelveHour) {
H = H % 12;
if (H == 0)
H = 12;
}
M = (int)val;
copyXPMArea(xDigit[H/10], 73, 7, 9, 17, 13);
copyXPMArea(xDigit[H%10], 73, 7, 9, 17+7, 13);
copyXPMArea(xDigit[10], 75, 3, 6, 17+15, 15);
copyXPMArea(xDigit[M/10], 73, 7, 9, 17+19, 13);
copyXPMArea(xDigit[M%10], 73, 7, 9, 17+26, 13);
} else {
copyXPMArea(10, 84, 28, 7, 19, 15);
}
if (LTSet > 0.0){
val = LTSet;
H = (int)val; val = (val-H)*60.0;
if (TwelveHour) {
H = H % 12;
if (H == 0)
H = 12;
}
M = (int)val;
copyXPMArea(xDigit[H/10], 73, 7, 9, 17, 40);
copyXPMArea(xDigit[H%10], 73, 7, 9, 17+7, 40);
copyXPMArea(xDigit[10], 75, 3, 6, 17+15, 42);
copyXPMArea(xDigit[M/10], 73, 7, 9, 17+19, 40);
copyXPMArea(xDigit[M%10], 73, 7, 9, 17+26, 40);
} else {
copyXPMArea(10, 84, 28, 7, 19, 40);
}
} else {
/*
* Update the counter.
*/
++n;
}
/*
* Add X display to file descriptor set for polling.
*/
FD_ZERO(&xfdset);
FD_SET(ConnectionNumber(display), &xfdset);
/*
* Process any pending X events.
*/
while(XPending(display)){
XNextEvent(display, &event);
switch(event.type){
case Expose:
RedrawWindow();
break;
case ButtonPress:
pressEvent(&event.xbutton);
break;
case ButtonRelease:
break;
}
}
/*
* Redraw and wait for next update
*/
RedrawWindow();
timeout.tv_sec = DELAY / 1000000L;
timeout.tv_usec = DELAY % 1000000L;
select(ConnectionNumber(display) + 1, &xfdset, NULL, NULL, &timeout);
}
}
/*
* ParseCMDLine()
*/
void ParseCMDLine(int argc, char *argv[]) {
int i;
for (i = 1; i < argc; i++) {
if (!strcmp(argv[i], "-display")){
++i;
} else if (!strcmp(argv[i], "-geometry")){
++i;
} else if (!strcmp(argv[i], "-lat")){
Glat = atof(argv[++i]);
} else if (!strcmp(argv[i], "-lon")){
Glon = atof(argv[++i]);
} else if (!strcmp(argv[i], "-td")){
UseUserTimeDiff = 1;
UserTimeDiff = atof(argv[++i]);
} else if (!strcmp(argv[i], "-date")){
UseUserDate = 1;
UserDate = atoi(argv[++i]);
} else if (!strcmp(argv[i], "-12")){
TwelveHour = 1;
} else {
printf("\nwmSun version: %s\n", WMSUN_VERSION);
printf("\nusage: wmsun [-display <Display>] [-lat <Latitude>] [-lon <Longitude>] [-h]\n\n");
printf("\t-display <Display>\tUse alternate X display.\n");
printf("\t-geometry <Geometry>\tSet window geometry.\n");
printf("\t-lat <Latitude>\t\tObservers Latitude. Positive to the west.\n");
printf("\t-lon <Longitude>\tObservers Longitude.\n");
printf("\t-td <Delta Time>\tUser defined difference between UT an LT (hrs).\n");
printf("\t-12\t\t\tUse 12-hour clock.\n");
printf("\t-h\t\t\tDisplay help screen.\n\n");
exit(1);
}
}
}
/*
* This routine handles button presses. Clicking in the window
* toggles the display.
*
*/
void pressEvent(XButtonEvent *xev){
(void) xev;
++ToggleWindow;
if (ToggleWindow > 4) ToggleWindow = 0;
Flag = 1;
return;
}