#define USE_LittleFS
#ifdef USE_LittleFS
#include <FS.h>
#define MYFS LittleFS
#include <LittleFS.h>
#include <ESP8266WiFi.h>
#include <ESPAsyncTCP.h>
#include <ESP8266mDNS.h>
#endif
#include <ESPAsyncWebServer.h>
#include <ArduinoJson.h>
#include <ezOutput.h>

AsyncWebServer server(80);
AsyncWebSocket ws("/ws");
AsyncEventSource events("/events");
ezOutput led(LED_BUILTIN);

const char *ssid = "SSID";
const char *password = "SSID_PASSWORD";
const char *hostName = "mastled";
const char *http_username = "admin";
const char *http_password = "admin";

int previousState;

void informeWSClients()
{
  const uint8_t size = JSON_OBJECT_SIZE(1);
  StaticJsonDocument<size> json;
  json["status"] = led.getState() == LOW ? "on" : "off";

  char buffer[17];
  size_t len = serializeJson(json, buffer);

  ws.textAll(buffer, len);
}

String processor(const String &var)
{
  return String(var == "STATE" && led.getState() == LOW ? "on" : "off");
}
void onRootRequest(AsyncWebServerRequest *request)
{
  request->send(MYFS, "/index.html", "text/html", false, processor);
}
void initWebSocket()
{
  server.addHandler(&ws);
}
void handleWebSocketMessage(void *arg, uint8_t *data, size_t len)
{
  AwsFrameInfo *info = (AwsFrameInfo *)arg;
  if (info->final && info->index == 0 && info->len == len && info->opcode == WS_TEXT)
  {

    const uint8_t size = JSON_OBJECT_SIZE(1);
    StaticJsonDocument<size> json;
    DeserializationError err = deserializeJson(json, data);
    if (err)
    {
      Serial.print(F("deserializeJson() failed with code "));
      Serial.println(err.c_str());
      return;
    }

    const char *action = json["action"];
    if (strcmp(action, "toggle") == 0)
    {
      Serial.printf("WebSocket client toogle recu\n");
      led.toggle();
      previousState = led.getState();
      informeWSClients();
    }
  }
}
void onWsEvent(AsyncWebSocket *server, AsyncWebSocketClient *client, AwsEventType type, void *arg, uint8_t *data, size_t len)
{
  switch (type)
  {
  case WS_EVT_CONNECT:
    Serial.printf("ws[%s][%u] connect\n", server->url(), client->id());
    Serial.printf("WebSocket client #%u connected from %s\n", client->id(), client->remoteIP().toString().c_str());
    //client->printf("Hello Client %s :)", client->remoteIP().toString().c_str());
    // ws.text(client->id(), led.getState() == LOW ? "on" : "off"); //onboard_led.on ? "on" : "off"
    // client->ping();
    break;
  case WS_EVT_DISCONNECT:
    Serial.printf("ws[%s][%u] disconnect\n", server->url(), client->id());
    Serial.printf("WebSocket client #%u disconnected\n", client->id());
    break;
  case WS_EVT_ERROR:
    Serial.printf("ws[%s][%u] error(%u): %s\n", server->url(), client->id(), *((uint16_t *)arg), (char *)data);
    break;
  case WS_EVT_PONG:
    Serial.printf("ws[%s][%u] pong[%u]: %s\n", server->url(), client->id(), len, (len) ? (char *)data : "");
    break;
  case WS_EVT_DATA:
    Serial.printf("data de %s :\n", client->remoteIP().toString().c_str());
    handleWebSocketMessage(arg, data, len);
    break;
  }
}

void setup()
{
  Serial.begin(115200);
  // Serial.setDebugOutput(true);
  WiFi.mode(WIFI_AP_STA);
  WiFi.softAP(hostName);
  WiFi.begin(ssid, password);
  if (WiFi.waitForConnectResult() != WL_CONNECTED)
  {
    Serial.printf("STA: Failed!\n");
    WiFi.disconnect(false);
    delay(1000);
    WiFi.begin(ssid, password);
  }

  Serial.print(F("*CONNECTED* IP:"));
  Serial.println(WiFi.localIP());
  MDNS.addService("http", "tcp", 80);

  //FS
  if (MYFS.begin())
  {
    Serial.print(F("FS mounted\n"));
  }
  else
  {
    Serial.print(F("FS mount failed\n"));
  }

  ws.onEvent(onWsEvent);
  initWebSocket();

  events.onConnect([](AsyncEventSourceClient *client) {
    //client->send("hello!", NULL, millis(), 1000);
    informeWSClients();
  });
  server.addHandler(&events);
  server.on("/", onRootRequest);
  server.serveStatic("/", MYFS, "/").setDefaultFile("index.html");
  server.onNotFound([](AsyncWebServerRequest *request) {
    Serial.printf("NOT_FOUND: ");
    if (request->method() == HTTP_GET)
      Serial.printf("GET");
    else if (request->method() == HTTP_POST)
      Serial.printf("POST");
    else if (request->method() == HTTP_DELETE)
      Serial.printf("DELETE");
    else if (request->method() == HTTP_PUT)
      Serial.printf("PUT");
    else if (request->method() == HTTP_PATCH)
      Serial.printf("PATCH");
    else if (request->method() == HTTP_HEAD)
      Serial.printf("HEAD");
    else if (request->method() == HTTP_OPTIONS)
      Serial.printf("OPTIONS");
    else
      Serial.printf("UNKNOWN");
    Serial.printf(" http://%s%s\n", request->host().c_str(), request->url().c_str());

    if (request->contentLength())
    {
      Serial.printf("_CONTENT_TYPE: %s\n", request->contentType().c_str());
      Serial.printf("_CONTENT_LENGTH: %u\n", request->contentLength());
    }

    int headers = request->headers();
    int i;
    for (i = 0; i < headers; i++)
    {
      AsyncWebHeader *h = request->getHeader(i);
      Serial.printf("_HEADER[%s]: %s\n", h->name().c_str(), h->value().c_str());
    }

    int params = request->params();
    for (i = 0; i < params; i++)
    {
      AsyncWebParameter *p = request->getParam(i);
      if (p->isFile())
      {
        Serial.printf("_FILE[%s]: %s, size: %u\n", p->name().c_str(), p->value().c_str(), p->size());
      }
      else if (p->isPost())
      {
        Serial.printf("_POST[%s]: %s\n", p->name().c_str(), p->value().c_str());
      }
      else
      {
        Serial.printf("_GET[%s]: %s\n", p->name().c_str(), p->value().c_str());
      }
    }

    request->send(404);
  });

  server.begin();
  led.blink(1000, 1000);
  previousState = led.getState();
}

void loop()
{
  led.loop();
  if (previousState != led.getState())
  {
    informeWSClients();
    previousState = led.getState();
  }
  if (millis() % 120000 < 50) // coupe la com après 2 mins
  {
    led.blink(1500, 1500);
    ws.cleanupClients();
  }
}