template<class T> inline Print &operator <<(Print &obj, T arg) { obj.print(arg); return obj; }

#include <CTBot.h>
#include <CTBotDataStructures.h>
#include <CTBotDefines.h>
#include <CTBotInlineKeyboard.h>
#include <CTBotReplyKeyboard.h>
#include <CTBotSecureConnection.h>
#include <CTBotStatusPin.h>
#include <CTBotWifiSetup.h>
#include <Utilities.h>

#include <ArduinoJson.h>
#include <ArduinoJson.hpp>
#include <Adafruit_GFX.h>
#include <DHT.h>

// COMENTAR Y DESCOMENTAR DEPENDIENDO DEL ESCENARIO

// SI ESTAMOS EN WOKWI:
#include <Adafruit_SSD1306.h>
// SI ESTAMOS EN LA PLACA
//#include <Adafruit_SH110X.h>

using namespace std;


// Definiciones para el display
#define ANCHO_PANTALLA 128
#define ALTO_PANTALLA 64

#define OLED_RESET -1
#define DIRECCION_PANTALLA 0x3c


// COMENTAR Y DESCOMENTAR DEPENDIENDO DEL ESCENARIO

// SI ESTAMOS EN WOKWI:
Adafruit_SSD1306 display(ANCHO_PANTALLA, ALTO_PANTALLA, &Wire, OLED_RESET);

// SI ESTAMOS EN LA PLACA
//Adafruit_SH1106G display = Adafruit_SH1106G(ANCHO_PANTALLA, ALTO_PANTALLA, &Wire, OLED_RESET);



CTBot miBot;

const char* ssid = "Wokwi-GUEST";
const char* password = "";
const String token = "6397153222:AAEFctwbHBcBK7GIfcq6Sqe5V6_P-c3qpbE";
const int64_t id =  1679707339;

// Conexiones

#define ledBlue 2
volatile bool ledBlueState = LOW;

#define ledGreen 5
volatile int ledGreenValue = 0;
volatile int input_value = 0;

#define pinDHT 33
DHT dht(pinDHT, DHT22);

#define potenciometro 32
volatile int estadoPotenciometro;

const uint8_t PROGMEM gamma8[] = {
  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,
  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  1,  1,  1,  1,
  1,  1,  1,  1,  1,  1,  1,  1,  1,  2,  2,  2,  2,  2,  2,  2,
  2,  3,  3,  3,  3,  3,  3,  3,  4,  4,  4,  4,  4,  5,  5,  5,
  5,  6,  6,  6,  6,  7,  7,  7,  7,  8,  8,  8,  9,  9,  9, 10,
  10, 10, 11, 11, 11, 12, 12, 13, 13, 13, 14, 14, 15, 15, 16, 16,
  17, 17, 18, 18, 19, 19, 20, 20, 21, 21, 22, 22, 23, 24, 24, 25,
  25, 26, 27, 27, 28, 29, 29, 30, 31, 32, 32, 33, 34, 35, 35, 36,
  37, 38, 39, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 50,
  51, 52, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 66, 67, 68,
  69, 70, 72, 73, 74, 75, 77, 78, 79, 81, 82, 83, 85, 86, 87, 89,
  90, 92, 93, 95, 96, 98, 99, 101, 102, 104, 105, 107, 109, 110, 112, 114,
  115, 117, 119, 120, 122, 124, 126, 127, 129, 131, 133, 135, 137, 138, 140, 142,
  144, 146, 148, 150, 152, 154, 156, 158, 160, 162, 164, 167, 169, 171, 173, 175,
  177, 180, 182, 184, 186, 189, 191, 193, 196, 198, 200, 203, 205, 208, 210, 213,
  215, 218, 220, 223, 225, 228, 231, 233, 236, 239, 241, 244, 247, 249, 252, 255
};

// Variables del programa

const String menu = "Menú de opciones:\n"
              "1. /led_blue \n"
              "2. /led_green \n"
              "3. /tempYhum \n"
              "4. /mensaje \n"
              "5. /potenciometro \n"
              "6. /menu \n";

volatile float temperatura = 0;
volatile float humedad = 0;

// Funciones

bool esNumero(const String& cadena) {
    for (size_t i = 0; i < cadena.length(); i++) {
        if (!isdigit(cadena.charAt(i))) {
            return false;
        }
    }
    return true;
}

void setup() {
  Serial.begin(115200);
  Serial.println("Iniciando WIFI y Bot de Telegram: ");
  miBot.wifiConnect(ssid, password);
  miBot.setTelegramToken(token);

  if (miBot.testConnection()) {
    Serial.println("Conexion Establecida");
  }
  else {
    Serial.println("\n No resultó bien");
  }


  // DISPLAY 
  // Descomentar si es la placa
  //display.begin(DIRECCION_PANTALLA);

  if(!display.begin(SSD1306_EXTERNALVCC, DIRECCION_PANTALLA)){
    Serial.println("Fallo en la asignacion de SSD1306");
  }

  display.setTextColor(WHITE);

  // CONEXIONES 

  pinMode(ledBlue, OUTPUT);
  pinMode(ledGreen, OUTPUT);
  
  pinMode(potenciometro, INPUT);
  analogReadResolution(8);
  

  dht.begin();
  

}

void loop() {
  // Leo los datos para la primera iteracion, asi se muestra en el display
  if(temperatura == 0 && humedad == 0){
    temperatura = dht.readTemperature();
    humedad = dht.readHumidity();
  }

  estadoPotenciometro = analogRead(potenciometro);

  TBMessage msg;
  String mensaje = "";

  if (CTBotMessageText == miBot.getNewMessage(msg)) {

      // Siempre mostramos en pantalla el mensaje y el usuario
      Serial << "Mensaje: " << msg.sender.firstName << " - " << msg.text << "\n";
      
      String request = "";
      // Obtengo del mensaje la primera parte que es el comando
      // Y el mensaje que es el resto
      size_t posicionEspacio = msg.text.indexOf(" ");
      if(posicionEspacio != string::npos){
        request = msg.text.substring(0, posicionEspacio);
        mensaje = msg.text.substring(posicionEspacio +1, msg.text.length());
      } else {
        request = msg.text;
      }
      String response = "";

      if(esNumero(mensaje) && mensaje != ""){
        input_value = mensaje.toInt();
        mensaje = "";
      }

      
      if (request == "/start"){
        response = "Hola " + msg.sender.username + "! \n";
        response += menu;
        miBot.sendMessage(msg.sender.id, response);
      }

      if (request == "/menu"){
        miBot.sendMessage(msg.sender.id, menu);
      }

      if (request == "/tempYhum"){
        temperatura = dht.readTemperature();
        humedad = dht.readHumidity();
        response = "Temperatura: " + String(temperatura) +"\n";
        response += "Humedad: " + String(humedad);

        miBot.sendMessage(msg.sender.id, response);
      }

      if (request == "/led_blue"){
        String menu_led_blue = "";
        if(ledBlueState == LOW){
          response = "Estado --> Apagado \n"
                          "1- /led_blue_on para encenderlo \n";
          miBot.sendMessage(msg.sender.id, response);
        } else {
          response = "Estado --> Encendido \n"
                          "1- /led_blue_off para apagarlo \n";
          miBot.sendMessage(msg.sender.id, response);
        }
      }

      if(request == "/led_blue_on"){
        ledBlueState = HIGH;
        miBot.sendMessage(msg.sender.id, "led_blue encendido");

      }

      if(request == "/led_blue_off"){
        ledBlueState = LOW;
        miBot.sendMessage(msg.sender.id, "led_blue apagado");
      }

      if(request == "/led_green"){
        if( input_value == 0){
          response = "Estado --> Apagado \n"
                            "1. /led_green_on -> para encenderlo \n"
                            "2. /led_green_intensity <intensidad> -> para encenderlo con intensidad deseada \n";
          miBot.sendMessage(msg.sender.id, response);
                            
        } else {
          response = "Estado --> Encendido \n"
                      "1. /led_green_off -> para apagarlo \n"
                      "2. /led_green_intensity <intensidad> -> para aumentar intensidad \n";
          response += "Potencia Actual: " + String(input_value);
          miBot.sendMessage(msg.sender.id, response);
        }
      }

      if( request == "/led_green_on" && input_value == 0){
        input_value = 255;
        
        miBot.sendMessage(msg.sender.id, "Se encendio led green");
      }
      if( request == "/led_green_intensity" && input_value < 256 && input_value >= 0 ){
        response = "Led green encendido, potencia: ";
        response += String(input_value);
        miBot.sendMessage(msg.sender.id, response);
      } else {
        if( input_value < 0 || input_value > 255){
          response = "Debe ingresar un valor entre 0 y 256";
          miBot.sendMessage(msg.sender.id, response);
        }
      }
      if(request == "/led_green_off"){
        input_value = 0;
        miBot.sendMessage(msg.sender.id, "Se apago el led green");
      }


      if (request == "/mensaje"){
        response = "El mensaje " + mensaje + " ha sido mostrado.";
        miBot.sendMessage(msg.sender.id, response);
      }

      if (request == "/potenciometro"){
        response = "Potenciometro: " + String(estadoPotenciometro);
        miBot.sendMessage(msg.sender.id, response);
      }

    }
    digitalWrite(ledBlue, ledBlueState);
    ledGreenValue = pgm_read_byte(&gamma8[input_value]);
    analogWrite(ledGreen, ledGreenValue);


  display.clearDisplay();
  display.setCursor(0,0);
    display.println("Temp: " + String(temperatura) + "Hum: "+ String(humedad));
    display.print("LED BLUE: ");
    display.println(ledBlueState ? "ON " : "OFF");
    display.print("LED GREEN: ");
    display.println(ledGreenValue > 0 ? "ON - POTENCIA: " + String(input_value) : "OFF" );
    display.println("Potenciometro: " + String(estadoPotenciometro));
    display.println(mensaje == "" ? "No hay mensajes para mostrar" : mensaje);
  display.display();
}