#include <string.h>
#include <math.h>

// **************************************************************************
// Demonstrate the Pimoroni Trackball module
// Assumes that a Pico Display Pack (a 1.14inch IPS LCD screen with four
// useful buttons) is installed, and that the Trackball I2C module has
// sda, scl and int on GPIO 20, 21 and 22
// Displays a set of coloured circles and a small cursor.
// Use the trackball to move the cursor around. The LED of the trackball
// will become the colour of whatever circle the cursosr is under.
// Pressing in on the trackball whilst over a circle will cause it to
// appear "pressed". Doing so on a "pressed" circle will make it "unpressed".
// **************************************************************************

// To use PicoExplorer rather than PicoDisplay, uncomment the following line
#define USE_PICO_EXPLORER 1
// This:
// - Includes pico_explorer.hpp rather than display.hpp
// - Replaces all PicoDisplay references with PicoExplorer
#ifdef USE_PICO_EXPLORER
#include "pico_explorer.hpp"
#else
#include "pico_display.hpp"
#endif
#include "breakout_trackball.hpp"

#include "drivers/st7789/st7789.hpp"
#include "libraries/pico_graphics/pico_graphics.hpp"

using namespace pimoroni;

struct TrackballColour {
  uint8_t r;
  uint8_t g;
  uint8_t b;
  uint8_t w;
};

#ifdef USE_PICO_EXPLORER
const uint16_t screen_width = PicoExplorer::WIDTH;
const uint16_t screen_height = PicoExplorer::HEIGHT;
#else
const uint16_t screen_width = PicoDisplay::WIDTH;
const uint16_t screen_height = PicoDisplay::HEIGHT;
#endif

ST7789 st7789(screen_width, screen_height, ROTATE_0, false, get_spi_pins(BG_SPI_FRONT));
PicoGraphics_PenRGB332 graphics(st7789.width, st7789.height, nullptr);

const Point screen_centre(screen_width / 2, screen_height / 2);
const uint16_t circle_radius = std::min(screen_centre.x, screen_centre.y) / 4;
const float ring_radius_mult = 0.7f;
const uint16_t circle_border = 4;
const uint16_t cursor_radius = 8;
const uint16_t cursor_border = 4;


BreakoutTrackball trackball;

const uint8_t NUM_CIRCLES = 6;
TrackballColour colour_circles[NUM_CIRCLES] = {
  { 255, 0, 0, 0 },
  { 255, 255, 0, 0 },
  { 0, 255, 0, 0 },
  { 0, 255, 255, 0 },
  { 0, 0, 255, 0 },
  { 255, 0, 255, 0 },
};
bool circle_states[NUM_CIRCLES] = { false, false, false, false, false, false };
bool centre_circle_state = false;


int main() {
  int16_t x = screen_centre.x;
  int16_t y = screen_centre.y;

  trackball.init();

  Point positions[NUM_CIRCLES];
  for(uint8_t i = 0; i < NUM_CIRCLES; i++) {
    float angle_rad = (i * 2 * M_PI) / NUM_CIRCLES;
    Point pos(screen_centre.x + (sinf(angle_rad) * screen_centre.x * ring_radius_mult),
              screen_centre.y + (cosf(angle_rad) * screen_centre.y * ring_radius_mult));

    positions[i] = pos;
  }

  Pen WHITE = graphics.create_pen(255, 255, 255);
  Pen BLACK = graphics.create_pen(0, 0, 0);
  Pen LIGHT_GREY = graphics.create_pen(212, 212, 212);
  Pen MID_GREY = graphics.create_pen(128, 128, 128);

  while(true) {
    Trackball::State state = trackball.read();
    x = std::min(std::max(x - state.left + state.right, 0), (int)screen_width);
    y = std::min(std::max(y - state.up + state.down, 0), (int)screen_height);
    Point cursor_pos(x, y);

    graphics.set_pen(BLACK);
    graphics.clear();

    //Draw a set of circles in a ring around the screen centre
    for(uint8_t i = 0; i < NUM_CIRCLES; i++) {
      TrackballColour col = colour_circles[i];

      if(circle_states[i]) {
        graphics.set_pen(graphics.create_pen(col.r, col.g, col.b));
        graphics.circle(positions[i], circle_radius + circle_border);
        graphics.set_pen(graphics.create_pen(col.r >> 1, col.g >> 1, col.b >> 1));
        graphics.circle(positions[i], circle_radius);
      }
      else {
        graphics.set_pen(graphics.create_pen(col.r >> 1, col.g >> 1, col.b >> 1));
        graphics.circle(positions[i], circle_radius + circle_border);
        graphics.set_pen(graphics.create_pen(col.r, col.g, col.b));
        graphics.circle(positions[i], circle_radius);
      }
    }

    //Draw a centre circle
    if(centre_circle_state) {
      graphics.set_pen(WHITE);
      graphics.circle(screen_centre, circle_radius + circle_border);
      graphics.set_pen(MID_GREY);
      graphics.circle(screen_centre, circle_radius);
    }
    else {
      graphics.set_pen(MID_GREY);
      graphics.circle(screen_centre, circle_radius + circle_border);
      graphics.set_pen(WHITE);
      graphics.circle(screen_centre, circle_radius);
    }

    //Draw the cursor
    graphics.set_pen(BLACK);
    graphics.circle(cursor_pos, cursor_radius + cursor_border);
    graphics.set_pen(LIGHT_GREY);
    graphics.circle(cursor_pos, cursor_radius);

    int16_t x_diff = cursor_pos.x - screen_centre.x;
    int16_t y_diff = cursor_pos.y - screen_centre.y;
    if((x_diff * x_diff) + (y_diff * y_diff) < circle_radius * circle_radius) {
      trackball.set_rgbw(0, 0, 0, 255);
      if(state.sw_changed && state.sw_pressed)
        centre_circle_state = !centre_circle_state;
    }
    else {
      bool colour_set = false;
      for(uint8_t i = 0; i < NUM_CIRCLES; i++) {
        uint8_t inv_i = NUM_CIRCLES - i - 1;

        TrackballColour col = colour_circles[inv_i];
        x_diff = cursor_pos.x - positions[inv_i].x;
        y_diff = cursor_pos.y - positions[inv_i].y;
        if((x_diff * x_diff) + (y_diff * y_diff) < circle_radius * circle_radius) {
          trackball.set_rgbw(col.r, col.g, col.b, col.w);
          colour_set = true;
          if(state.sw_changed && state.sw_pressed)
            circle_states[inv_i] = !circle_states[inv_i];
          break;
        }
      }

      if(!colour_set) {
        trackball.set_rgbw(0, 0, 0, 0);
      }
    }

    // update screen
    st7789.update(&graphics);
  }

  return 0;
}