pimoroni-pico/examples/pico_trackball_display/pico_trackball_display.cpp

176 lines
5.7 KiB
C++

#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;
}