#include <LiquidCrystal.h>

enum States
{
  RESET,
  WAITING_FOR_PRE_STAGE,
  WAITING_FOR_STAGE,
  WAITING_FOR_START_BUTTON,
  WAITING_FOR_COUNTDOWN_1,
  WAITING_FOR_COUNTDOWN_2,
  WAITING_FOR_COUNTDOWN_3,
  WAITING_FOR_COUNTDOWN_4,
  FALSE_START,
  WAITING_FOR_GO,
  WAITING_FOR_FINISH,
  FINISHED,
  WAITING_FOR_RESET_BUTTON
};

const float FEET_PER_MILE = 5280;
const float TRACK_LEN = 1320;

const byte RS = 52;
const byte EN = 49;
const byte D4 = 53;
const byte D5 = 50;
const byte D6 = 51;
const byte D7 = 48;
const byte LED_PRE_STAGE_PIN =  8;
const byte LED_STAGE_PIN = 7;
const byte LED_Y1_PIN = 6;
const byte LED_Y2_PIN = 5;
const byte LED_Y3_PIN = 4;
const byte LED_START_PIN = 3;
const byte LED_RED_LIGHT_PIN = 2;
const byte PRE_STAGE_SENSOR_PIN = A0;
const byte STAGE_SENSOR_PIN = A1;
const byte FINISH_SENSOR_PIN = A2;
const byte START_BUTTON_PIN = 46;
const byte RESET_BUTTON_PIN = 47;

const uint8_t BACKSLASH[8] =
{
  0b00000,
  0b10000,
  0b01000,
  0b00100,
  0b00010,
  0b00001,
  0b00000,
  0b00000,
};

States presentState = -1;  // Purposefully initialised to an invalid state.
States nextState = RESET;

unsigned long timestamp;
int preStageSensorValue;
int stageSensorValue;
int finishSensorValue;
bool startButtonActive;
bool resetButtonActive;

unsigned long timerStartTime;
unsigned long timerElapsedTime;
bool countdown1Active;
bool countdown2Active;
bool countdown3Active;
bool countdown4Active;

unsigned long raceStartTime;
unsigned long raceEndTime;
unsigned long reactionTime;

LiquidCrystal lcd(RS, EN, D4, D5, D6, D7);

void setup()
{
  pinMode(LED_BUILTIN, OUTPUT);

  pinMode(LED_PRE_STAGE_PIN, OUTPUT);
  pinMode(LED_STAGE_PIN, OUTPUT);
  pinMode(LED_Y1_PIN, OUTPUT);
  pinMode(LED_Y2_PIN, OUTPUT);
  pinMode(LED_Y3_PIN, OUTPUT);
  pinMode(LED_START_PIN, OUTPUT);
  pinMode(LED_RED_LIGHT_PIN, OUTPUT);

  pinMode(PRE_STAGE_SENSOR_PIN, INPUT);
  pinMode(STAGE_SENSOR_PIN, INPUT);
  pinMode(FINISH_SENSOR_PIN, INPUT);
  pinMode(START_BUTTON_PIN, INPUT_PULLUP);
  pinMode(RESET_BUTTON_PIN, INPUT_PULLUP);

  Serial.begin(115200);
  Serial.println();
  Serial.println(F("Setup"));
  Serial.println();

  lcd.createChar(1, BACKSLASH);
  lcd.begin(16, 2);
  lcd.setCursor(0, 1);
  lcd.print(F("System Ready"));
  delay(2000);
  lcd.clear();
}

void loop()
{
  if (presentState != nextState)
  {
    presentState = nextState;
    OutputLogic();
  }
  InputLogic();
  NextStateLogic();
  Animation();
  BlinkLed();
}

void InputLogic()
{
  timestamp = millis();
  preStageSensorValue = analogRead(PRE_STAGE_SENSOR_PIN);
  stageSensorValue = analogRead(STAGE_SENSOR_PIN);
  finishSensorValue = analogRead(FINISH_SENSOR_PIN);
  startButtonActive = digitalRead(START_BUTTON_PIN) == LOW;
  resetButtonActive = digitalRead(RESET_BUTTON_PIN) == LOW;
  timerElapsedTime = timestamp - timerStartTime;
  countdown1Active = timerElapsedTime > 500;
  countdown2Active = timerElapsedTime > 1000;
  countdown3Active = timerElapsedTime > 1500;
  countdown4Active = timerElapsedTime > 2000;
}

void NextStateLogic()
{
  switch (presentState)
  {
    case RESET:
      nextState = WAITING_FOR_PRE_STAGE;
      break;
    case WAITING_FOR_PRE_STAGE:
      if (preStageSensorValue < 500)
      {
        nextState = WAITING_FOR_STAGE;
      }
      break;
    case WAITING_FOR_STAGE:
      if (stageSensorValue < 500)
      {
        nextState = WAITING_FOR_START_BUTTON;
      }
      else if (preStageSensorValue > 500)
      {
        nextState = WAITING_FOR_PRE_STAGE;
      }
      break;
    case WAITING_FOR_START_BUTTON:
      if (stageSensorValue > 500)
      {
        nextState = WAITING_FOR_STAGE;
      }
      else if (startButtonActive)
      {
        timerStartTime = timestamp;
        nextState = WAITING_FOR_COUNTDOWN_1;
      }
      break;
    case WAITING_FOR_COUNTDOWN_1:
      if (countdown1Active)
      {
        nextState = WAITING_FOR_COUNTDOWN_2;
      }
      break;
    case WAITING_FOR_COUNTDOWN_2:
      if (countdown2Active)
      {
        nextState = WAITING_FOR_COUNTDOWN_3;
      }
      break;
    case WAITING_FOR_COUNTDOWN_3:
      if (countdown3Active)
      {
        nextState = WAITING_FOR_COUNTDOWN_4;
      }
      break;
    case WAITING_FOR_COUNTDOWN_4:
      if (countdown4Active)
      {
        // Check for false start.
        if (stageSensorValue > 500)
        {
          nextState = FALSE_START;
        }
        else
        {
          raceStartTime = timestamp;
          nextState = WAITING_FOR_GO;
        }
      }
      break;
    case FALSE_START:
      nextState = WAITING_FOR_RESET_BUTTON;
      break;
    case WAITING_FOR_GO:
      if (stageSensorValue > 500)
      {
        reactionTime = timestamp - raceStartTime;
        nextState = WAITING_FOR_FINISH;
      }
      break;
    case WAITING_FOR_FINISH:
      if (finishSensorValue < 500)
      {
        raceEndTime = timestamp;
        nextState = FINISHED;
      }
      break;
    case FINISHED:
      nextState = WAITING_FOR_RESET_BUTTON;
      break;
    case WAITING_FOR_RESET_BUTTON:
      if (resetButtonActive)
      {
        nextState = RESET;
      }
      break;
  }
}

void OutputLogic()
{
  Serial.print(F("timestamp="));
  Serial.print(timestamp);
  Serial.print(F(";  State="));
  Serial.println(presentState);
  switch (presentState)
  {
    case RESET:
      digitalWrite(LED_PRE_STAGE_PIN, LOW);
      digitalWrite(LED_STAGE_PIN, LOW);
      digitalWrite(LED_START_PIN, LOW);
      digitalWrite(LED_RED_LIGHT_PIN, LOW);
      digitalWrite(LED_Y1_PIN, LOW);
      digitalWrite(LED_Y2_PIN, LOW);
      digitalWrite(LED_Y3_PIN, LOW);
      lcd.clear();
      lcd.print(F("Please Pre Stage"));
      break;
    case WAITING_FOR_PRE_STAGE:
      digitalWrite(LED_PRE_STAGE_PIN, LOW);
      digitalWrite(LED_STAGE_PIN, LOW);
      digitalWrite(LED_START_PIN, LOW);
      digitalWrite(LED_RED_LIGHT_PIN, LOW);
      lcd.setCursor(0, 0);
      lcd.print(F("Please Pre Stage"));
      break;
    case WAITING_FOR_STAGE:
      digitalWrite(LED_PRE_STAGE_PIN, HIGH);
      digitalWrite(LED_STAGE_PIN, LOW);
      lcd.setCursor(0, 0);
      lcd.print(F("Please Stage    "));
      break;
    case WAITING_FOR_START_BUTTON:
      digitalWrite(LED_STAGE_PIN, HIGH);
      lcd.setCursor(0, 0);
      lcd.print(F("Vehicle Ready   "));
      break;
    case WAITING_FOR_COUNTDOWN_1:
      break;
    case WAITING_FOR_COUNTDOWN_2:
      digitalWrite(LED_Y1_PIN, HIGH);
      break;
    case WAITING_FOR_COUNTDOWN_3:
      digitalWrite(LED_Y1_PIN, LOW);
      digitalWrite(LED_Y2_PIN, HIGH);
      break;
    case WAITING_FOR_COUNTDOWN_4:
      digitalWrite(LED_Y2_PIN, LOW);
      digitalWrite(LED_Y3_PIN, HIGH);
      break;
    case FALSE_START:
      digitalWrite(LED_Y3_PIN, LOW);
      digitalWrite(LED_PRE_STAGE_PIN, LOW);
      digitalWrite(LED_STAGE_PIN, LOW);
      digitalWrite(LED_RED_LIGHT_PIN, HIGH);
      lcd.clear();
      lcd.print(F("!!False Start!!"));
      break;
    case WAITING_FOR_GO:
      digitalWrite(LED_Y3_PIN, LOW);
      digitalWrite(LED_START_PIN, HIGH);
      lcd.clear();
      lcd.print(F("       GO        "));
      lcd.print(raceStartTime);
      break;
    case WAITING_FOR_FINISH:
      break;
    case FINISHED:
      lcd.clear();
      lcd.print(F("E"));
      float etSeconds = float(raceEndTime - raceStartTime) / 1000.0f;
      lcd.print(etSeconds, 2);
      float rtSeconds = reactionTime / 1000.0f;
      lcd.setCursor(10, 0);
      lcd.print("R");
      lcd.print(rtSeconds, 3);
      lcd.setCursor(0, 1);
      lcd.print(F("MPH:"));
      float fps = TRACK_LEN / etSeconds;
      Serial.print(F("fps:")); Serial.println(fps, 4);
      float mph = (fps / FEET_PER_MILE) / 0.00028;
      Serial.print(F("MPH:")); Serial.println(mph, 4);
      lcd.print(mph, 4);

      Serial.print(F("Reaction time:"));
      Serial.println(reactionTime / 1000.0f, 3);
      break;
    case WAITING_FOR_RESET_BUTTON:
      break;
  }
}

void Animation()
{
  if (presentState == WAITING_FOR_FINISH)
  {
    {
      static byte flip = 0;
      const unsigned long INTERVAL = 50;
      static unsigned long previous = 0;
      if (timestamp - previous >= INTERVAL)
      {
        switch (flip)
        {
          case 0: lcd.setCursor(0, 0); lcd.print(F("|")); lcd.setCursor(15, 0); lcd.print(F("|")); break;
          case 1: lcd.setCursor(0, 0); lcd.print(F("/")); lcd.setCursor(15, 0); lcd.print(F("/")); break;
          case 2: lcd.setCursor(0, 0); lcd.print(F("-")); lcd.setCursor(15, 0); lcd.print(F("-")); break;
          case 3: lcd.setCursor(0, 0); lcd.print(F("\x01")); lcd.setCursor(15, 0); lcd.print(F("\x01")); break;
          case 4: lcd.setCursor(0, 0); lcd.print(F("|")); lcd.setCursor(15, 0); lcd.print(F("|")); break;
        }
        flip = ++flip % 5;
        previous += INTERVAL;
      }
    }
    {
      const unsigned long INTERVAL = 100;
      static unsigned long previous = 0;
      if (timestamp - previous >= INTERVAL)
      {
        lcd.setCursor(0, 1);
        lcd.print(float(timestamp - raceStartTime) / 1000.0f, 1);
        previous += INTERVAL;
      }
    }
  }
}

void BlinkLed()
{
  const unsigned long INTERVAL = 500;
  static unsigned long previous = 0;
  if (timestamp - previous >= INTERVAL)
  {
    digitalWrite(LED_BUILTIN, !digitalRead(LED_BUILTIN));
    previous += INTERVAL;
  }
}
mega:SCL
mega:SDA
mega:AREF
mega:GND.1
mega:13
mega:12
mega:11
mega:10
mega:9
mega:8
mega:7
mega:6
mega:5
mega:4
mega:3
mega:2
mega:1
mega:0
mega:14
mega:15
mega:16
mega:17
mega:18
mega:19
mega:20
mega:21
mega:5V.1
mega:5V.2
mega:22
mega:23
mega:24
mega:25
mega:26
mega:27
mega:28
mega:29
mega:30
mega:31
mega:32
mega:33
mega:34
mega:35
mega:36
mega:37
mega:38
mega:39
mega:40
mega:41
mega:42
mega:43
mega:44
mega:45
mega:46
mega:47
mega:48
mega:49
mega:50
mega:51
mega:52
mega:53
mega:GND.4
mega:GND.5
mega:IOREF
mega:RESET
mega:3.3V
mega:5V
mega:GND.2
mega:GND.3
mega:VIN
mega:A0
mega:A1
mega:A2
mega:A3
mega:A4
mega:A5
mega:A6
mega:A7
mega:A8
mega:A9
mega:A10
mega:A11
mega:A12
mega:A13
mega:A14
mega:A15
led1:A
led1:C
led2:A
led2:C
led5:A
led5:C
led6:A
led6:C
led7:A
led7:C
led8:A
led8:C
led9:A
led9:C
r1:1
r1:2
r2:1
r2:2
r3:1
r3:2
r4:1
r4:2
r5:1
r5:2
r6:1
r6:2
r7:1
r7:2
lcd1:VSS
lcd1:VDD
lcd1:V0
lcd1:RS
lcd1:RW
lcd1:E
lcd1:D0
lcd1:D1
lcd1:D2
lcd1:D3
lcd1:D4
lcd1:D5
lcd1:D6
lcd1:D7
lcd1:A
lcd1:K
r15:1
r15:2
btn1:1.l
btn1:2.l
btn1:1.r
btn1:2.r
btn2:1.l
btn2:2.l
btn2:1.r
btn2:2.r
pot1:VCC
pot1:SIG
pot1:GND
pot2:VCC
pot2:SIG
pot2:GND
pot3:VCC
pot3:SIG
pot3:GND