//
//  Temperature.app
// 
//  Copyright (c) 2000-2002 Per Liden
// 
//  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 of the License, 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; if not, write to the Free Software
//  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1307, 
//  USA.
//

#include <X11/Xlib.h>
#include <iostream>
#include <fstream>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <math.h>
#include <time.h>
#include "Xpm.h"
#include "Temperature.h"

#include "pixmaps/main.xpm"
#include "pixmaps/redlight.xpm"

volatile static ChildStatus childStatus;

static void catchBrokenPipe(int sig) 
{
   exit(0);
}

static void catchChildExit(int sig) 
{
   int status;
   waitpid(-1, &status, 0);

   if (WIFEXITED(status)) {
      if (WEXITSTATUS(status)) {
         childStatus = ChildError;
      } else {
         childStatus = ChildDone;
      }
   } else {
      childStatus = ChildError;
   }

   if (childStatus == ChildError) {
      std::cerr << APPNAME << ": could not fetch temperature (wget failed), try option -V for more information" << std::endl;
   }
}

Temperature::Temperature(int argc, char** argv) 
{
   XClassHint classHint;
   XSizeHints sizeHints;
   XWMHints   wmHints;
   Atom       deleteWindow;
   Xpm*       image;
   char*      displayName = NULL;

   mInstanceName = INSTANCENAME;
   mStationId = 0;
   mTemperature[0] = 0;
   mTime[0] = 0;
   mTimeDiff = 0.0;
   mFahrenheit = false;
   mShowTime = false;
   mTime12HourFormat = false;
   mVerbose = false;

   // Parse command line
   if (argc>1) {
      for (int i=1; i<argc; i++) {
         // Display
         if (!strcmp(argv[i], "-d")) {
            checkArgument(argv, argc, i);
            displayName = argv[i+1];
            i++;
         }

         // Station id
         else if (!strcmp(argv[i], "-s")) {
            checkArgument(argv, argc, i);
            mStationId = argv[i+1];
            i++;
         }

         // Fahrenheit
         else if (!strcmp(argv[i], "-f")) {
            mFahrenheit = true;
         }

         // Time
         else if (!strcmp(argv[i], "-t")) {
            mShowTime = true;
            checkArgument(argv, argc, i);
            if (!strcmp(argv[i+1], "12")) {
               mTime12HourFormat = true;
            } else if (strcmp(argv[i+1], "24")) {
               std::cerr << APPNAME << ": unknown time format, use 12 or 24" << std::endl;
               exit(0);
            }
            i++;
         }

         // Verbose
         else if (!strcmp(argv[i], "-V")) {
            mVerbose = true;
         }

         // Instance name
         else if (!strcmp(argv[i], "-n")) {
            checkArgument(argv, argc, i);
            mInstanceName = argv[i+1];
            i++;
         }

         // Version
         else if (!strcmp(argv[i], "-v")) {
            std::cerr << APPNAME << " version " << VERSION << std::endl;
            exit(0);
         }

         // Help
         else if (!strcmp(argv[i], "-h") || !strcmp(argv[i], "--help")) {
            showHelp();
            exit(0);
         }

         // Unknown option
         else {
            std::cerr << APPNAME << ": invalid option '" << argv[i] << "'" << std::endl;
            tryHelp(argv[0]);
            exit(0);
         }
      }
   }

   if (mStationId == 0) {
      std::cerr << APPNAME << ": you must supply a station id using -s <id>" << std::endl;
      tryHelp(argv[0]);
      exit(0);
   }

   // Open display
   if ((mDisplay = XOpenDisplay(displayName)) == NULL) {
      std::cerr << APPNAME << ": could not open display " << displayName << std::endl;
      exit(0);
   }
 
   // Get root window
   mRoot = RootWindow(mDisplay, DefaultScreen(mDisplay));

   // Create windows
   mAppWin = XCreateSimpleWindow(mDisplay, mRoot, 1, 1, 64, 64, 0, 0, 0);
   mIconWin = XCreateSimpleWindow(mDisplay, mAppWin, 0, 0, 64, 64, 0, 0, 0);
        
   // Set classhint
   classHint.res_name =  mInstanceName;
   classHint.res_class = CLASSNAME;
   XSetClassHint(mDisplay, mAppWin, &classHint);

   // Create delete atom
   deleteWindow = XInternAtom(mDisplay, "WM_DELETE_WINDOW", False);
   XSetWMProtocols(mDisplay, mAppWin, &deleteWindow, 1);
   XSetWMProtocols(mDisplay, mIconWin, &deleteWindow, 1);

   // Set windowname
   XStoreName(mDisplay, mAppWin, APPNAME);
   XSetIconName(mDisplay, mAppWin, APPNAME);

   // Set sizehints
   sizeHints.flags= USPosition;
   sizeHints.x = 0;
   sizeHints.y = 0;
   XSetWMNormalHints(mDisplay, mAppWin, &sizeHints);

   // Set wmhints
   wmHints.initial_state = WithdrawnState;
   wmHints.icon_window = mIconWin;
   wmHints.icon_x = 0;
   wmHints.icon_y = 0;
   wmHints.window_group = mAppWin;
   wmHints.flags = StateHint | IconWindowHint | IconPositionHint | WindowGroupHint;
   XSetWMHints(mDisplay, mAppWin, &wmHints);

   // Set command
   XSetCommand(mDisplay, mAppWin, argv, argc);

   // Set background image
   image = new Xpm(mDisplay, mRoot, main_xpm);
   image->setWindowPixmapShaped(mIconWin);
   delete image;

   // Create status led
   mStatusLed = XCreateSimpleWindow(mDisplay, mIconWin, LED_X, LED_Y, 3, 2, 0, 0, 0);
   image = new Xpm(mDisplay, mRoot, redlight_xpm);
   image->setWindowPixmap(mStatusLed);
   delete image;

   XMapWindow(mDisplay, mIconWin);
   XMapWindow(mDisplay, mAppWin);
   XSync(mDisplay, False);

   // Catch broker pipe signal
   signal(SIGPIPE, catchBrokenPipe);

   // Catch child exit signal
   signal(SIGCHLD, catchChildExit);
}

void Temperature::tryHelp(char* appname)
{
   std::cerr << std::endl << "Try `" << appname << " --help' for more information" << std::endl;
}

void Temperature::showHelp() 
{
   std::cerr << APPNAME << " Copyright (c) 2000-2002 by Per Liden (per@fukt.bth.se)" << std::endl << std::endl
        << "options:" << std::endl
        << " -s <id>         set station id (ICAO Location Indicator)" << std::endl
        << " -t 12|24        display time of temperature observation (12 or 24 hour format)" << std::endl
        << " -f              display degrees in Fahrenheit" << std::endl
        << " -V              display verbose messages from wget" << std::endl
        << " -n <name>       set client instance name" << std::endl
        << " -d <disp>       set display" << std::endl
        << " -v              print version and exit" << std::endl
        << " -h, --help      display this help text and exit" << std::endl 
        << std::endl
        << "You must supply the ICAO Location Indicator (a 4-character string)" << std::endl
        << "of a weather station near you. You can search for a station on" << std::endl
        << "this site: http://www.nws.noaa.gov/oso/siteloc.shtml" << std::endl;
}

void Temperature::checkArgument(char** argv, int argc, int index)
{
   if (argc-1 < index+1) {
      std::cerr << APPNAME << ": option '" << argv[index] << "' requires an argument" << std::endl;
      tryHelp(argv[0]);
      exit(0);
   }
}

void Temperature::showLed(bool show)
{
   if (show) {
      XMapWindow(mDisplay, mStatusLed);
   } else {
      XUnmapWindow(mDisplay, mStatusLed);
   }
   XSync(mDisplay, False);
}

void Temperature::calcTimeDiff()
{
   struct tm* t;
   double localTime;
   double universalTime;
   time_t currentTime;

   currentTime = time(0);
   t = gmtime(&currentTime);
   universalTime = (double)t->tm_hour + (double)t->tm_min / 60.0 + (double)t->tm_sec / 3600.0;

   currentTime = time(0);
   t = localtime(&currentTime);
   localTime = (double)t->tm_hour + (double)t->tm_min / 60.0 + (double)t->tm_sec / 3600.0;

   mTimeDiff = universalTime - localTime;
   if (mTimeDiff > 24.0) {
      mTimeDiff -= 24.0;
   } else if (mTimeDiff < 0.0) {
      mTimeDiff += 24.0;
   }
}

void Temperature::setTime(char* utcTime)
{
   char unit[3];
   int hour = 0;
   int min = 0;   

   strncpy(unit, &utcTime[0], 2);
   hour = atoi(unit);
   strncpy(unit, &utcTime[2], 2);
   min = atoi(unit);

   double time = ((double)hour + (double)min / 60.0) - mTimeDiff;
   if (time < 0.0) {
      time += 24.0;
   } else if (time > 24.0) {
      time -= 24.0;
   }

   hour = (int)time;
   min = (int)((time - (double)hour) * 60.0 + 0.5);
   if (min >= 60){
      min = 0;
      if (++hour >= 24) {
         hour = 0;
      }
   }

   if (mTime12HourFormat) {
      if (hour >= 0 && hour <= 11) {
         mTimeAMPM = "AM";         
      } else {
         mTimeAMPM = "PM";
      }

      if (hour == 0) {
         hour = 12;
      } else if (hour > 12) {
         hour -= 12;
      }
   }
   sprintf(mTime, "%d:%.2d", hour, min);
}

bool Temperature::updateTemperture(ifstream& file)
{
   const int MAX_LINE = 1024;
   char buffer[MAX_LINE];

   if (mShowTime) {
      // Find time of observation
      char* start;
      char time[5];
      file.getline(buffer, MAX_LINE - 1);
      file.getline(buffer, MAX_LINE - 1);
      if ((start = strstr(buffer, "UTC")) == 0) {
         return false;
      }
      strncpy(time, start - 5, 4);
      setTime(time);
   }
   
   // Find temperature
   while (!file.eof()) {
      file >> buffer;
      if (!strcmp(buffer, "Temperature:")) {
         file >> buffer;
         if (buffer && strlen(buffer) < 5) {
            char* unit;
            if (mFahrenheit) {
               strcpy(mTemperature, buffer);
               unit = " �F";
            } else {
               sprintf(mTemperature, "%d", (int)rint((atoi(buffer) - 32) / 1.8));
               unit = " �C";
            }
            
            Xpm* image = new Xpm(mDisplay, mRoot, main_xpm);
            if (mShowTime) {
               if (mTime12HourFormat) {
                  image->drawComposedString(TIME_POS, TIME_FONT, mTime, AMPM_FONT, mTimeAMPM);
               } else {
                  image->drawString(TIME_POS, TIME_FONT, mTime);
               }
               image->drawComposedString(TEMP_WITH_TIME_POS, TEMP_FONT, mTemperature, UNIT_FONT, unit);
            } else {
               image->drawComposedString(TEMP_POS, TEMP_FONT, mTemperature, UNIT_FONT, unit);
            }
            image->setWindowPixmap(mIconWin);
            delete image;
            XSync(mDisplay, False);
            return true;
         }
      }
   }

   std::cerr << APPNAME << ": could not fetch temperature (unknown file format)" << std::endl;
   return false;
}

void Temperature::run() 
{
   if (mShowTime) {
      calcTimeDiff();
   }

   int counter = 0;
   while(1) {
      if (counter <= 0) {
         char tmpFile[sizeof(TMP_FILE)] = TMP_FILE;
         int fd = mkstemp(tmpFile);
         if (fd == -1) {
            std::cerr << APPNAME << ": could not create temporary file " << tmpFile << ": " << strerror(errno) << std::endl;
            exit(1);
         }
         close(fd);

         counter = UPDATE_INTERVAL;
         childStatus = ChildRunning;
         signal(SIGCHLD, catchChildExit);
         showLed(true);
         int pid = fork();
         if (pid == 0) {
            const char* verbose = (mVerbose ? "--verbose" : "--quiet");
            char* URL = new char[strlen(METAR_URL) + strlen(mStationId) + 1];
            sprintf(URL, METAR_URL, mStationId);
            execlp("wget", "wget", "--cache=off", "--tries=0", verbose, "-O", tmpFile, URL, 0);
            std::cerr << APPNAME << ": could not fetch temperature (wget not found in $PATH)" << std::endl;
            remove(tmpFile);
            exit(0);
         } else if (pid == -1) {
            std::cerr << APPNAME << ": could not fetch temperature (fork() failed)" << std::endl;
         } else {
            bool toggle = true;
            while (childStatus == ChildRunning) {
               showLed(toggle);
               toggle ^= true;
               sleep(1);
            }
            showLed(true);
            if (childStatus == ChildDone) {
               ifstream file(tmpFile);
               if (file) {
                  if (updateTemperture(file)) {
                     showLed(false);
                  }
                  file.close();
               }
            }
            remove(tmpFile);
         }
      } else {
         counter--;
         sleep(1);
         XSync(mDisplay, False);
      }
   }
}