#include <Adafruit_NeoPixel.h>

#define PIN 6

enum XY_matrix_config { SERPENTINE = 1, ROWMAJOR = 2, FLIPMAJOR = 4, FLIPMINOR = 8 };
#define kMatrixWidth   16
#define kMatrixHeight  16
#define XY_MATRIX      (ROWMAJOR)
#define NUM_LEDS       ((kMatrixWidth) * (kMatrixHeight))

Adafruit_NeoPixel strip = Adafruit_NeoPixel(NUM_LEDS, PIN, NEO_GRBW + NEO_KHZ800);

// Parameter 1 = number of pixels in strip
// Parameter 2 = Arduino pin number (most are valid)
// Parameter 3 = pixel type flags, add together as needed:
// NEO_KHZ800  800 KHz bitstream (most NeoPixel products w/WS2812 LEDs)
// NEO_KHZ400  400 KHz (classic 'v1' (not v2) FLORA pixels, WS2811 drivers)
// NEO_GRB     Pixels are wired for GRB bitstream (most NeoPixel products)
// NEO_RGB     Pixels are wired for RGB bitstream (v1 FLORA pixels, not v2)

// IMPORTANT: To reduce NeoPixel burnout risk, add 1000 uF capacitor across
// pixel power leads, add 300 - 500 Ohm resistor on first pixel's data input
// and minimize distance between Arduino and first pixel.  Avoid connecting
// on a live circuit...if you must, connect GND first.

String VERSION = "v1.0";

int pos = 0;

char cmd = 0;                    // a character to encode the command (L=light command)
String cmd_value = "";           // value associated with the command (L5-10:255-255-0-10)
boolean cmd_complete = false;  // whether the string is complete

void setup() {
  // put your setup code here, to run once:
  strip.begin();
  strip.show(); // Initialize all pixels to 'off'

  Serial.begin(115200);
  while (!Serial);
  manageStatus();
}

void loop() {
     /*
     * cmd, cmd_value and cmd_complete have been processed by the serialeEvent function
     * At this point we can check the cmd that is requested and process it with the cmd_value
     */

    // Do something only if command is complete
    if (cmd_complete) {

      if (cmd == '?') {   // Print arduino status
        manageStatus();
      }

      if (cmd == 'L') {   // Led command
        manageLedCommand(cmd_value);
      }

      // reset command variables
      cmd = 0;
      cmd_value = "";
      cmd_complete = false;
    }

}


/*
  SerialEvent occurs whenever a new data comes in the
 hardware serial RX.  This routine is run between each
 time loop() runs, so using delay inside loop can delay
 response.  Multiple bytes of data may be available.
 */
void serialEvent() {

  // Set triplet "cmd", "cmd_value" and "cmd_complete" variables

  while (Serial.available()) {
    // get the new byte:
    char inChar = (char)Serial.read();
    if (pos == 0){
      cmd = inChar;
      pos++;
    } else {
          cmd_value += inChar;
    }

    // if the incoming character is a newline, set a flag
    // so the main loop can do something about it:
    if (inChar == '\n') {
      cmd_complete = true;
      pos = 0;
    }
  }
}

/*
 * This function is managing the arduino status
 */
void manageStatus() {
  Serial.print("Baio Led Matrix Controller ");
  Serial.println(VERSION);
}

/*
 * This function is managing the led commands
 * Takes as input a command string with
 * L[u]-[v]:[r]-[g]-[b]-[w]
 * u, v : coordinates of pixels
 * r, g, b, w : color values 0 to 255
 *
 */
void manageLedCommand(String cmd_value) {

      char* c = strcpy(new char[cmd_value.length() + 1], cmd_value.c_str()); // copy command value to c

      // first parameter : coordinates
      char *coords = strtok(c, ":");
      if (coords == NULL) {
        print_Led_Error(cmd_value);
        return;
      }

      // second parameter : color
      char *color = strtok(NULL, ":");
      if (color == NULL) {
        print_Led_Error(cmd_value);
        return;
      }

      // parse coordinates
      char *p = strtok(coords, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int u = String(p).toInt();
      p = strtok(NULL, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int v = String(p).toInt();

      // parse color
      p = strtok(color, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int r = String(p).toInt();
      p = strtok(NULL, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int g = String(p).toInt();
      p = strtok(NULL, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int b = String(p).toInt();
      p = strtok(NULL, "-");
      if (p == NULL) {
        print_Led_Error(cmd_value);
        return;
      }
      int w = String(p).toInt();

      strip.setPixelColor(XY(u,v), r, g, b);
      strip.show();

      String msg = ">Set Led ";
      msg += u;
      msg += "-";
      msg += v;
      msg += ":";
      msg += r;
      msg += "-";
      msg += g;
      msg += "-";
      msg += b;
      msg += "-";
      msg += w;
      msg += " -> ok";
      Serial.println(msg);

      free(c);
}

void print_Led_Error(String cmd_value) {
    String msg = ">Error in command. Expected format is Lu-v:r-g-b-w here is an example L5-7:255-255-0-10 instead it was ";
    msg += cmd_value;
    Serial.println(msg);
}

uint16_t XY(uint8_t x, uint8_t y) {
  uint8_t major, minor, sz_major, sz_minor;
  if (x >= kMatrixWidth || y >= kMatrixHeight)
    return NUM_LEDS;
  if (XY_MATRIX & ROWMAJOR)
    major = x, minor = y, sz_major = kMatrixWidth,  sz_minor = kMatrixHeight;
  else
    major = y, minor = x, sz_major = kMatrixHeight, sz_minor = kMatrixWidth;
  if (XY_MATRIX & FLIPMINOR)
    minor = sz_minor - 1 - minor;
  if ((XY_MATRIX & FLIPMAJOR) ^ (minor & 1 && (XY_MATRIX & SERPENTINE)))
    major = sz_major - 1 - major;
  return (uint16_t) minor * sz_major + major;
}