// The display also uses hardware SPI, plus #9 & #10
#define TFT_CS 15
#define TFT_DC 2
#define TFT_MOSI 23
#define TFT_SCLK 18

//#include <mcp_can.h>
//#include "vesc_can_bus_arduino.h"
#include "Digital-7100-L.h"
#include "Digital-7100-S.h"
#include <SPI.h>
#include <TFT_eSPI.h>       // Hardware-specific library

TFT_eSPI tft = TFT_eSPI();
//TFT_eSprite img = TFT_eSprite(&tft);  // Create a sprite object for entire screen
TFT_eSprite erpmSprite = TFT_eSprite(&tft);  // Create a sprite object for ERPM bar graph
TFT_eSprite voltageSprite = TFT_eSprite(&tft);  // Create a sprite object for Voltage bar graph
//TFT_eSprite mphSprite = TFT_eSprite(&tft);  // Create a sprite object for Voltage bar graph
TFT_eSprite mphSprite2 = TFT_eSprite(&tft);  // Create a sprite object for Voltage bar graph


//CAN can;             // get torque sensor data, throttle for now

bool print_realtime_data;
long last_print_data;
unsigned long last_refresh;
float dutyCycleMin0;
float speed, inpVoltage, dutyCycleNow, erpm, avgInputCurrent, avgMotorCurrent, tempFET, tempMotor;

// Define your min and max values for each variable you wish to simulate
const float ERPM_MIN = 0;
const float ERPM_MAX = 20000;
const float VOLTAGE_MIN = 42;
const float VOLTAGE_MAX = 58.8;
const float DUTY_CYCLE_MIN = 0;
const float DUTY_CYCLE_MAX = 1.0;  // Assuming duty cycle is between 0 and 1. Adjust if needed.
const float INPUT_CURRENT_MIN = 0;
const float INPUT_CURRENT_MAX = 100;  // Arbitrary max. Adjust as needed.
const float MOTOR_CURRENT_MIN = 0;
const float MOTOR_CURRENT_MAX = 200;  // Arbitrary max. Adjust as needed.
const float TEMP_FET_MIN = 0;
const float TEMP_FET_MAX = 80;  // Example temperature range in Celsius. Adjust if needed.
const float TEMP_MOTOR_MIN = 0;
const float TEMP_MOTOR_MAX = 80;  // Example temperature range in Celsius. Adjust if needed.

unsigned long lastSimulationUpdate = 0;
const unsigned long SIMULATION_INTERVAL = 100; // Adjust this to determine how fast values change

#define AA_FONT_LARGE Digital7100L
#define AA_FONT_SMALL Digital7100S

#define CAN0_INT 32                              // Set INT

//for img Sprite, (lower half text)
#define IWIDTH  320 //240 if 0 deg
#define IHEIGHT 240 //320 if 0 deg

//X IS HORZ
//Y IS VERT
// Constants for the ERPM bar graph
#define ERPM_BAR_X 5             // Starting X-coordinate of the bar //RELATIVE TO THE SPRITE WINDOW
#define ERPM_BAR_Y 0            // Starting Y-coordinate of the bar //RELATIVE TO THE SPRITE WINDOW
#define ERPM_BAR_WIDTH 300    // Maximum width of the bar graph when at max ERPM
#define ERPM_BAR_HEIGHT 30        // Height of the bar graph
#define ERPM_BAR_WIDTH_SPRITE 315    // Maximum width of the bar graph when at max ERPM
#define ERPM_BAR_HEIGHT_SPRITE 50        // Height of the bar graph
#define ERPM_NUM_PARTITIONS 10    // For values: 0, 5000, 10000, 15000, 20000
#define ERPM_INTERVAL 2000  // The ERPM value interval for each partition
#define ERPM_INTERVAL2 2  // The ERPM value interval for each partition

// Constants for the Input Voltage bar graph
#define VOLTAGE_BAR_X 5              // Starting X-coordinate of the bar //RELATIVE TO THE SPRITE WINDOW
#define VOLTAGE_BAR_Y 0              // Starting Y-coordinate of the bar //RELATIVE TO THE SPRITE WINDOW
#define VOLTAGE_BAR_WIDTH 300     // Maximum width of the bar graph when at max Voltage
#define VOLTAGE_BAR_HEIGHT 30         // Height of the bar graph
#define VOLTAGE_BAR_WIDTH_SPRITE 315     // Maximum width of the bar graph when at max Voltage
#define VOLTAGE_BAR_HEIGHT_SPRITE 50         // Height of the bar graph
//#define VOLTAGE_NUM_PARTITIONS 8.4     // For example values: 0, 10, 20, ..., 80 (adjust as needed)
#define VOLTAGE_MIN 42       // Define your minimum voltage
#define VOLTAGE_MAX 58.8     // Define your maximum voltage
#define VOLTAGE_INTERVAL 2 // The interval between each partition

#define MPH_WIDTH_SPRITE 200
#define MPH_HEIGHT_SPRITE 80

//#define TFT_CS    15     // TFT CS  pin is connected to  pin D2
//#define TFT_RST   2     // TFT RST pin is connected to  pin D3
//#define TFT_DC    4     // TFT DC  pin is connected to  pin D4
//#define TFT_SCLK  14
//#define TFT_MOSI  13

struct LimitedValue {
    float value;
    int decimalPlaces;
};

void setup()
{
    print_realtime_data = true;
    //Serial.begin(115200); // Wired comms from USB port

    //initialize();
    tft.init();
    tft.setRotation(1); //0 = 0, 1 = 90, 2 = 180, 3 = 270

    tft.fillScreen(TFT_BLACK);
    //img.setColorDepth(8); // Optionally set depth to 8 to halve RAM use
    //img.createSprite(IWIDTH, IHEIGHT);

    erpmSprite.createSprite(ERPM_BAR_WIDTH_SPRITE, ERPM_BAR_HEIGHT_SPRITE);
    voltageSprite.createSprite(VOLTAGE_BAR_WIDTH_SPRITE, VOLTAGE_BAR_HEIGHT_SPRITE);
    mphSprite2.createSprite(MPH_WIDTH_SPRITE, MPH_HEIGHT_SPRITE);
    //mphSprite2.createSprite(200,80);

    //img.fillSprite(TFT_BLACK);

    inpVoltage = VOLTAGE_MIN; //because non-zero starting

    //mphSprite.loadFont(AA_FONT_LARGE); // Load another different font into the sprite instance
    //mphSprite.setTextColor(TFT_WHITE, TFT_BLACK);
    mphSprite2.setTextColor(TFT_WHITE, TFT_BLACK);

    drawERPMBar(erpm);
    drawVoltageBar(inpVoltage);
    //drawMPH(erpm);

    //voltageSprite.pushSprite(VOLTAGE_BAR_X, VOLTAGE_BAR_Y);

    ////img.pushSprite(0, 0); // push all sprite data to visible screen

}

void loop()
{
    //img.setTextSize(2);
    //img.fillSprite(TFT_BLACK);
    //img.setTextColor(TFT_GREEN, TFT_BLACK);

    //spin(); // REAL TIME DATA
    simulateData(); // SIMULATION DATA
    
    if (print_realtime_data == true)
    {
        //Serial.println("true");
        if (millis() - last_print_data > 200)
        {
            /*Serial.println(erpm); 
            Serial.println(inpVoltage);
            Serial.println(dutyCycleNow);
            Serial.println(avgInputCurrent);
            Serial.println(avgMotorCurrent);
            Serial.println(tempFET);
            Serial.println(tempMotor);*/

            last_refresh = millis();

            if (dutyCycleNow < 0)
              {
                dutyCycleMin0 = 0.00;
              } else 
              {
                dutyCycleMin0 = dutyCycleNow;
              }         
            
            //updateERPMBar(erpm);//  OLD
            //updateInputVoltageBar(inpVoltage); // OLD
            
            
            //img.drawString(String("ERPM: ") + erpm, 0, 42, 1); //32

            // For IPVT
            LimitedValue limitedValue = limitToThreeSignificantDigits(inpVoltage);
            String formattedValue = formatFloat(limitedValue.value, limitedValue.decimalPlaces);
            //img.drawString(String("IPVT: ") + formattedValue, 0, 98, 1); //64

            // For DUTY
            float dutyPercentage = dutyCycleMin0 * 100;
            LimitedValue limitedDuty = limitToThreeSignificantDigits(dutyPercentage);
            String formattedDuty = formatFloat(limitedDuty.value, limitedDuty.decimalPlaces) + "%";  // Add the percentage symbol
            //img.drawString(String("DUTY: ") + formattedDuty, 0, 118, 1);  // Adjust y-coordinate as needed

            // For IPCR
            LimitedValue limitedIPCR = limitToThreeSignificantDigits(avgInputCurrent);
            String formattedIPCR = formatFloat(limitedIPCR.value, limitedIPCR.decimalPlaces);
            //img.drawString(String("IPCR: ") + formattedIPCR, 0, 138, 1);  // Adjust y-coordinate as needed

            // For MTCR
            LimitedValue limitedMTCR = limitToThreeSignificantDigits(avgMotorCurrent);
            String formattedMTCR = formatFloat(limitedMTCR.value, limitedMTCR.decimalPlaces);
            //img.drawString(String("MTCR: ") + formattedMTCR, 0, 158, 1);  // Adjust y-coordinate as needed

            // For TMPF
            LimitedValue limitedTMPF = limitToThreeSignificantDigits(tempFET);
            String formattedTMPF = formatFloat(limitedTMPF.value, limitedTMPF.decimalPlaces);
            //img.drawString(String("TMPF: ") + formattedTMPF, 0, 178, 1);  // Adjust y-coordinate as needed

            // For TMPM
            LimitedValue limitedTMPM = limitToThreeSignificantDigits(tempMotor);
            String formattedTMPM = formatFloat(limitedTMPM.value, limitedTMPM.decimalPlaces);
            //img.drawString(String("TMPM: ") + formattedTMPM, 0, 198, 1);  // Adjust y-coordinate as needed

            // For MPH
            speed = erpmToMph(erpm,3,80,11,15);
            LimitedValue limitedMPH = limitToThreeSignificantDigits(speed);
            String formattedMPH = formatFloat(limitedMPH.value, limitedMPH.decimalPlaces);
            //img.drawString(String("MPH:  ") + formattedMPH, 0, 218, 1);  // Adjust y-coordinate as needed
            
            //img.pushSprite(0, 80); // push all sprite data to visible screen

            drawERPMBar(erpm);
            drawVoltageBar(inpVoltage);
            //drawMPH(erpm);

            float speed = erpmToMph(erpm,3,80,11,15);
            drawFormattedFloat(mphSprite2, speed);

            erpmSprite.pushSprite(0, 0);
            voltageSprite.pushSprite(0, 60);
            //mphSprite.pushSprite(150,150);
            mphSprite2.pushSprite(176,172);

            //Serial.println(millis() - last_refresh);
            //Serial.println(millis() - last_print_data);

            last_print_data = millis();
        }
    }
}


float erpmToMph(float erpm, int polePairs, float axleTeeth, float clutchTeeth, float wheelDiameterInches) {
  // Calculate the drive wheel circumference in feet
  float wheelCircumferenceFeet = (PI * wheelDiameterInches) / 12.0; // Convert inches to feet

  // Calculate the gear ratio
  float gearRatio = axleTeeth / clutchTeeth;

  // Convert eRPM to mechanical RPM (Wheel RPM)
  float wheelRPM = erpm / polePairs / gearRatio;

  // Calculate feet per minute
  float feetPerMinute = wheelCircumferenceFeet * wheelRPM;

  // Calculate feet per hour
  float feetPerHour = feetPerMinute * 60.0;

  // Convert feet per hour to miles per hour
  float mph = feetPerHour / 5280.0;

  return mph;
}

void simulateData() {
    if (millis() - lastSimulationUpdate > SIMULATION_INTERVAL) {
        // Change the ERPM value
        erpm += 200;
        if (erpm > ERPM_MAX) erpm = ERPM_MIN;

        // Change the Input Voltage
        inpVoltage += 0.1;
        if (inpVoltage > VOLTAGE_MAX) inpVoltage = VOLTAGE_MIN;

        // Change the Duty Cycle
        dutyCycleNow += 0.01;
        if (dutyCycleNow > DUTY_CYCLE_MAX) dutyCycleNow = DUTY_CYCLE_MIN;

        // Change the Average Input Current
        avgInputCurrent += .1;
        if (avgInputCurrent > INPUT_CURRENT_MAX) avgInputCurrent = INPUT_CURRENT_MIN;

        // Change the Average Motor Current
        avgMotorCurrent += .1;
        if (avgMotorCurrent > MOTOR_CURRENT_MAX) avgMotorCurrent = MOTOR_CURRENT_MIN;

        // Change the FET Temperature
        tempFET += .1;
        if (tempFET > TEMP_FET_MAX) tempFET = TEMP_FET_MIN;

        // Change the Motor Temperature
        tempMotor += .1;
        if (tempMotor > TEMP_MOTOR_MAX) tempMotor = TEMP_MOTOR_MIN;

        lastSimulationUpdate = millis();
    }
}

LimitedValue limitToThreeSignificantDigits(float value) {
    LimitedValue result;
    if (value >= 100) {
        result.value = round(value);
        result.decimalPlaces = 0;
    } else if (value >= 10) {
        result.value = round(value * 10) / 10.0;
        result.decimalPlaces = 1;
    } else {
        result.value = round(value * 100) / 100.0;
        result.decimalPlaces = 2;
    }
    return result;
}

String formatFloat(float value, int decimalPlaces) {
    return String(value, decimalPlaces);
}

void drawERPMGradientBar(int x, int y, int width, int height, float value, float minValue, float maxValue, uint16_t startColor, uint16_t middleColor, uint16_t endColor, float firstTransition, float secondTransition) {
    int fillWidth = (int)(((value - minValue) / (maxValue - minValue)) * width);
    for (int i = x; i < x + fillWidth; i++) {
        float currentRatio = (i - x) / (float)width;
        uint16_t currentColor;
        if (currentRatio < firstTransition) {
            float localRatio = currentRatio / firstTransition;
            currentColor = blendColor(startColor, middleColor, localRatio);
        } else if (currentRatio < secondTransition) {
            float localRatio = (currentRatio - firstTransition) / (secondTransition - firstTransition);
            currentColor = blendColor(middleColor, endColor, localRatio);
        } else {
            currentColor = endColor;
        }
        erpmSprite.drawFastVLine(i, y, height, currentColor);
    }
}

void drawVoltageGradientBar(int x, int y, int width, int height, float value, float minValue, float maxValue, uint16_t startColor, uint16_t middleColor, uint16_t endColor, float firstTransition, float secondTransition) {
    int fillWidth = (int)(((value - minValue) / (maxValue - minValue)) * width);
    for (int i = x; i < x + fillWidth; i++) {
        float currentRatio = (i - x) / (float)width;
        uint16_t currentColor;
        if (currentRatio < firstTransition) {
            float localRatio = currentRatio / firstTransition;
            currentColor = blendColor(startColor, middleColor, localRatio);
        } else if (currentRatio < secondTransition) {
            float localRatio = (currentRatio - firstTransition) / (secondTransition - firstTransition);
            currentColor = blendColor(middleColor, endColor, localRatio);
        } else {
            currentColor = endColor;
        }
        voltageSprite.drawFastVLine(i, y, height, currentColor);
    }
}

void drawERPMBar(float erpm) {
    erpmSprite.fillSprite(TFT_BLACK);  // Clear the previous bar graph
    
    // Draw the empty bar as background
    erpmSprite.drawRect(ERPM_BAR_X, ERPM_BAR_Y, ERPM_BAR_WIDTH, ERPM_BAR_HEIGHT, TFT_WHITE);

    // Draw the gradient bar BEFORE
    drawERPMGradientBar(ERPM_BAR_X+1, ERPM_BAR_Y+1, ERPM_BAR_WIDTH-1, ERPM_BAR_HEIGHT-2, erpm, ERPM_MIN, ERPM_MAX, TFT_GREEN, TFT_YELLOW, TFT_RED, 0.7f, 0.9f);

    
    for(int i = 0; i <= ERPM_NUM_PARTITIONS; i++) {
        int partitionX = ERPM_BAR_X + (i * (ERPM_BAR_WIDTH / ERPM_NUM_PARTITIONS));
        erpmSprite.drawFastVLine(partitionX, ERPM_BAR_Y, ERPM_BAR_HEIGHT, TFT_WHITE);  // Vertical line for partition
        erpmSprite.setTextColor(TFT_WHITE, TFT_BLACK);
        erpmSprite.drawString(String(i * ERPM_INTERVAL2), partitionX - 5, ERPM_BAR_Y + ERPM_BAR_HEIGHT + 2, 2);  // Adjust positioning as needed
    }
    
    // Draw the gradient bar AFTER
    //drawERPMGradientBar(ERPM_BAR_X+1, ERPM_BAR_Y+1, ERPM_BAR_WIDTH-1, ERPM_BAR_HEIGHT-2, erpm, ERPM_MIN, ERPM_MAX, TFT_GREEN, TFT_YELLOW, TFT_RED, 0.7f, 0.9f);
}

void drawVoltageBar(float voltage) {
    voltageSprite.fillSprite(TFT_BLACK);  // Clear the previous bar graph

    // Draw the empty bar as background
    voltageSprite.drawRect(VOLTAGE_BAR_X, VOLTAGE_BAR_Y, VOLTAGE_BAR_WIDTH, VOLTAGE_BAR_HEIGHT, TFT_WHITE);

    // Draw the gradient bar BEFORE
    drawVoltageGradientBar(VOLTAGE_BAR_X+1, VOLTAGE_BAR_Y+1, VOLTAGE_BAR_WIDTH-1, VOLTAGE_BAR_HEIGHT-2, voltage, VOLTAGE_MIN, VOLTAGE_MAX, TFT_RED, TFT_YELLOW, TFT_GREEN, 0.4f, 0.7f);

    const int VOLTAGE_NUM_PARTITIONS = (VOLTAGE_MAX - VOLTAGE_MIN) / VOLTAGE_INTERVAL;
    for(int i = 0; i <= VOLTAGE_NUM_PARTITIONS; i++) {
        int partitionX = VOLTAGE_BAR_X + (i * (VOLTAGE_BAR_WIDTH / VOLTAGE_NUM_PARTITIONS));
        voltageSprite.drawFastVLine(partitionX, VOLTAGE_BAR_Y, VOLTAGE_BAR_HEIGHT, TFT_WHITE);  // Vertical line for partition
        voltageSprite.setTextColor(TFT_WHITE, TFT_BLACK);
        int valueForPartition = VOLTAGE_MIN + (i * VOLTAGE_INTERVAL);
        voltageSprite.drawString(String(valueForPartition), partitionX - 4, VOLTAGE_BAR_Y + VOLTAGE_BAR_HEIGHT + 2, 2);  // Adjust positioning as needed
    }

    // Draw the gradient bar AFTER
    //drawVoltageGradientBar(VOLTAGE_BAR_X+1, VOLTAGE_BAR_Y+1, VOLTAGE_BAR_WIDTH-1, VOLTAGE_BAR_HEIGHT-2, voltage, VOLTAGE_MIN, VOLTAGE_MAX, TFT_RED, TFT_YELLOW, TFT_GREEN, 0.4f, 0.7f);
}

/* //Old version
void drawMPH(float erpm) {
    mphSprite.fillSprite(TFT_BLACK);  // Clear the previous bar graph
    
    // box around digits
    //mphSprite.drawRect(0, 0, 120, 100, TFT_WHITE);

    mphSprite.setTextColor(TFT_WHITE, TFT_BLACK);

    speed = erpmToMph(erpm,3,80,11,15);
    //LimitedValue limitedMPH = limitToThreeSignificantDigits(speed);
    //String formattedMPH = formatFloat(limitedMPH.value, limitedMPH.decimalPlaces);

    mphSprite.setTextDatum(TR_DATUM); // Top Right datum
    mphSprite.drawFloat(speed, 1, 150, 0); // decimals, x, y
    
    //mphSprite.drawString(String(formattedMPH), 0, 0, 4);
}
*/

// Blend function (as provided earlier)
uint16_t blendColor(uint16_t colorStart, uint16_t colorEnd, float ratio) {
    if (ratio > 1.0f) ratio = 1.0f;
    if (ratio < 0.0f) ratio = 0.0f;

    uint8_t rStart = (colorStart & 0xF800) >> 8;
    uint8_t gStart = (colorStart & 0x07E0) >> 3;
    uint8_t bStart = (colorStart & 0x001F) << 3;

    uint8_t rEnd = (colorEnd & 0xF800) >> 8;
    uint8_t gEnd = (colorEnd & 0x07E0) >> 3;
    uint8_t bEnd = (colorEnd & 0x001F) << 3;

    uint8_t rBlend = (1.0f - ratio) * rStart + ratio * rEnd;
    uint8_t gBlend = (1.0f - ratio) * gStart + ratio * gEnd;
    uint8_t bBlend = (1.0f - ratio) * bStart + ratio * bEnd;

    return ((rBlend >> 3) << 11) | ((gBlend >> 2) << 5) | (bBlend >> 3);
}

void drawFormattedFloat(TFT_eSprite &sprite, float value) {
    sprite.fillScreen(TFT_BLACK);
    int intValue = (int)value;  // Get the integer part
    int decimalValue = (int)((value - intValue) * 10);  // Get the decimal part, assuming one decimal place

    // Convert integer and decimal parts to strings
    String intStr = String(intValue);
    String decimalStr = String(decimalValue);

    //int intWidth = sprite.textWidth(intStr);  // Get the width of the integer part
    sprite.loadFont(AA_FONT_LARGE); // Load another different font into the sprite instance
    sprite.setTextDatum(TR_DATUM); // Top Right datum
    sprite.drawString(intStr, 100, 0); //relative to sprite window, TR DATUM is top right and prints left of x,y here

    // decimal part and draw it
    sprite.loadFont(AA_FONT_SMALL); // Load another different font into the sprite instance
    sprite.setTextDatum(TL_DATUM); // Top Left datum
    sprite.drawString("." + decimalStr, 102, 30);  // Assuming font 1 for the decimal part
}
esp:VIN
esp:GND.2
esp:D13
esp:D12
esp:D14
esp:D27
esp:D26
esp:D25
esp:D33
esp:D32
esp:D35
esp:D34
esp:VN
esp:VP
esp:EN
esp:3V3
esp:GND.1
esp:D15
esp:D2
esp:D4
esp:RX2
esp:TX2
esp:D5
esp:D18
esp:D19
esp:D21
esp:RX0
esp:TX0
esp:D22
esp:D23
lcd1:VCC
lcd1:GND
lcd1:CS
lcd1:RST
lcd1:D/C
lcd1:MOSI
lcd1:SCK
lcd1:LED
lcd1:MISO
lcd1:SCL
lcd1:SDA