2018-09-10 18:57:16 +00:00
|
|
|
/**
|
|
|
|
* @file Arduboy2.cpp
|
|
|
|
* \brief
|
|
|
|
* The Arduboy2Base and Arduboy2 classes and support objects and definitions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "Arduboy2.h"
|
|
|
|
|
|
|
|
//========================================
|
|
|
|
//========== class Arduboy2Base ==========
|
|
|
|
//========================================
|
|
|
|
|
2021-07-28 21:28:54 +00:00
|
|
|
uint8_t Arduboy2Base::sBuffer[];
|
2018-09-10 18:57:16 +00:00
|
|
|
|
2021-07-28 21:28:54 +00:00
|
|
|
uint8_t Arduboy2Base::currentButtonState = 0;
|
|
|
|
uint8_t Arduboy2Base::previousButtonState = 0;
|
2018-09-10 18:57:16 +00:00
|
|
|
// frame management
|
2021-07-28 21:28:54 +00:00
|
|
|
uint16_t Arduboy2Base::frameCount = 0;
|
|
|
|
uint8_t Arduboy2Base::eachFrameMillis = 16;
|
|
|
|
uint8_t Arduboy2Base::thisFrameStart;
|
|
|
|
uint8_t Arduboy2Base::lastFrameDurationMs;
|
|
|
|
bool Arduboy2Base::justRendered = false;
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
// functions called here should be public so users can create their
|
|
|
|
// own init functions if they need different behavior than `begin`
|
|
|
|
// provides by default
|
|
|
|
void Arduboy2Base::begin()
|
|
|
|
{
|
|
|
|
boot(); // raw hardware
|
|
|
|
|
2019-07-22 20:29:34 +00:00
|
|
|
//using CLEAR_BUFFER so a sketch can be optimized when using CLEAR_BUFFER exclusivly
|
|
|
|
display(CLEAR_BUFFER); //sBuffer is global, so cleared automatically.
|
|
|
|
|
2018-09-10 18:57:16 +00:00
|
|
|
flashlight(); // light the RGB LED and screen if UP button is being held.
|
|
|
|
|
|
|
|
// check for and handle buttons held during start up for system control
|
|
|
|
systemButtons();
|
|
|
|
|
|
|
|
audio.begin();
|
|
|
|
|
|
|
|
bootLogo();
|
|
|
|
// alternative logo functions. Work the same as bootLogo() but may reduce
|
|
|
|
// memory size if the sketch uses the same bitmap drawing function
|
|
|
|
// bootLogoCompressed();
|
|
|
|
// bootLogoSpritesSelfMasked();
|
|
|
|
// bootLogoSpritesOverwrite();
|
|
|
|
// bootLogoSpritesBSelfMasked();
|
|
|
|
// bootLogoSpritesBOverwrite();
|
|
|
|
|
|
|
|
waitNoButtons(); // wait for all buttons to be released
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::flashlight()
|
|
|
|
{
|
|
|
|
if (!pressed(UP_BUTTON)) {
|
|
|
|
return;
|
|
|
|
}
|
2020-06-26 23:22:31 +00:00
|
|
|
#ifdef GU12864_800B
|
|
|
|
allPixelsOn(true);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
sendLCDCommand(OLED_ALL_PIXELS_ON); // smaller than allPixelsOn()
|
2020-06-26 23:22:31 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(RGB_ON, RGB_ON, RGB_ON);
|
|
|
|
|
|
|
|
// prevent the bootloader magic number from being overwritten by timer 0
|
|
|
|
// when a timer variable overlaps the magic number location, for when
|
|
|
|
// flashlight mode is used for upload problem recovery
|
|
|
|
power_timer0_disable();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
idle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::systemButtons()
|
|
|
|
{
|
|
|
|
while (pressed(B_BUTTON)) {
|
|
|
|
digitalWriteRGB(BLUE_LED, RGB_ON); // turn on blue LED
|
|
|
|
sysCtrlSound(UP_BUTTON + B_BUTTON, GREEN_LED, 0xff);
|
|
|
|
sysCtrlSound(DOWN_BUTTON + B_BUTTON, RED_LED, 0);
|
2020-09-15 15:17:03 +00:00
|
|
|
delayByte(200);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
digitalWriteRGB(BLUE_LED, RGB_OFF); // turn off blue LED
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::sysCtrlSound(uint8_t buttons, uint8_t led, uint8_t eeVal)
|
|
|
|
{
|
|
|
|
if (pressed(buttons)) {
|
|
|
|
digitalWriteRGB(BLUE_LED, RGB_OFF); // turn off blue LED
|
2020-09-15 15:17:03 +00:00
|
|
|
delayByte(200);
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(led, RGB_ON); // turn on "acknowledge" LED
|
2020-09-15 15:17:03 +00:00
|
|
|
EEPROM.update(eepromAudioOnOff, eeVal);
|
2018-09-10 18:57:16 +00:00
|
|
|
delayShort(500);
|
|
|
|
digitalWriteRGB(led, RGB_OFF); // turn off "acknowledge" LED
|
|
|
|
|
|
|
|
while (pressed(buttons)) { } // Wait for button release
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogo()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoBitmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoBitmap(int16_t y)
|
|
|
|
{
|
|
|
|
drawBitmap(20 - (64 - WIDTH / 2), y, arduboy_logo, 88, 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogoCompressed()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoCompressed);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoCompressed(int16_t y)
|
|
|
|
{
|
|
|
|
drawCompressed(20 - (64 - WIDTH / 2), y, arduboy_logo_compressed);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogoSpritesSelfMasked()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoSpritesSelfMasked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoSpritesSelfMasked(int16_t y)
|
|
|
|
{
|
|
|
|
Sprites::drawSelfMasked(20 - (64 - WIDTH / 2), y, arduboy_logo_sprite, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogoSpritesOverwrite()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoSpritesOverwrite);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoSpritesOverwrite(int16_t y)
|
|
|
|
{
|
|
|
|
Sprites::drawOverwrite(20 - (64 - WIDTH / 2), y, arduboy_logo_sprite, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogoSpritesBSelfMasked()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoSpritesBSelfMasked);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoSpritesBSelfMasked(int16_t y)
|
|
|
|
{
|
|
|
|
SpritesB::drawSelfMasked(20 - (64 - WIDTH / 2), y, arduboy_logo_sprite, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::bootLogoSpritesBOverwrite()
|
|
|
|
{
|
|
|
|
bootLogoShell(drawLogoSpritesBOverwrite);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLogoSpritesBOverwrite(int16_t y)
|
|
|
|
{
|
|
|
|
SpritesB::drawOverwrite(20 - (64 - WIDTH / 2), y, arduboy_logo_sprite, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// bootLogoText() should be kept in sync with bootLogoShell()
|
|
|
|
// if changes are made to one, equivalent changes should be made to the other
|
|
|
|
void Arduboy2Base::bootLogoShell(void (*drawLogo)(int16_t))
|
|
|
|
{
|
|
|
|
bool showLEDs = readShowBootLogoLEDsFlag();
|
|
|
|
|
|
|
|
if (!readShowBootLogoFlag()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_ON, RGB_OFF, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(RED_LED, RGB_ON);
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
for (int16_t y = -15; y <= 24; y++) {
|
2018-09-10 18:57:16 +00:00
|
|
|
if (pressed(RIGHT_BUTTON)) {
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_OFF); // all LEDs off
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs && y == 4) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_ON, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(RED_LED, RGB_OFF); // red LED off
|
|
|
|
digitalWriteRGB(GREEN_LED, RGB_ON); // green LED on
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Using display(CLEAR_BUFFER) instead of clear() may save code space.
|
|
|
|
// The extra time it takes to repaint the previous logo isn't an issue.
|
|
|
|
display(CLEAR_BUFFER);
|
|
|
|
(*drawLogo)(y); // call the function that actually draws the logo
|
|
|
|
display();
|
2020-09-15 15:17:03 +00:00
|
|
|
delayByte(15);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_ON);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(GREEN_LED, RGB_OFF); // green LED off
|
|
|
|
digitalWriteRGB(BLUE_LED, RGB_ON); // blue LED on
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
delayShort(400);
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(BLUE_LED, RGB_OFF);
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
bootLogoExtra();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Virtual function overridden by derived class
|
|
|
|
void Arduboy2Base::bootLogoExtra() { }
|
|
|
|
|
|
|
|
// wait for all buttons to be released
|
|
|
|
void Arduboy2Base::waitNoButtons() {
|
|
|
|
do {
|
2020-09-15 15:17:03 +00:00
|
|
|
delayByte(50); // simple button debounce
|
2018-09-10 18:57:16 +00:00
|
|
|
} while (buttonsState());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Frame management */
|
|
|
|
|
|
|
|
void Arduboy2Base::setFrameRate(uint8_t rate)
|
|
|
|
{
|
|
|
|
eachFrameMillis = 1000 / rate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::setFrameDuration(uint8_t duration)
|
|
|
|
{
|
|
|
|
eachFrameMillis = duration;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::everyXFrames(uint8_t frames)
|
|
|
|
{
|
|
|
|
return frameCount % frames == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::nextFrame()
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t now = *((uint8_t*)(&timer0_millis));
|
2018-09-10 18:57:16 +00:00
|
|
|
uint8_t frameDurationMs = now - thisFrameStart;
|
|
|
|
|
|
|
|
if (justRendered) {
|
|
|
|
lastFrameDurationMs = frameDurationMs;
|
|
|
|
justRendered = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (frameDurationMs < eachFrameMillis) {
|
|
|
|
// Only idle if at least a full millisecond remains, since idle() may
|
|
|
|
// sleep the processor until the next millisecond timer interrupt.
|
|
|
|
if (eachFrameMillis > ++frameDurationMs) {
|
|
|
|
idle();
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// pre-render
|
|
|
|
justRendered = true;
|
|
|
|
thisFrameStart = now;
|
2020-09-15 15:17:03 +00:00
|
|
|
#if defined __AVR_ARCH__
|
|
|
|
uint16_t* ptr = &frameCount;
|
|
|
|
asm volatile
|
|
|
|
(
|
|
|
|
"ld r24, z \n"
|
|
|
|
"ldd r25, z+1 \n"
|
|
|
|
"adiw r24, 1 \n"
|
|
|
|
"st z, r24 \n"
|
|
|
|
"std z+1, r25 \n"
|
|
|
|
|
|
|
|
: [ptr] "+z" (ptr)
|
|
|
|
:
|
|
|
|
: "r24", "r25"
|
|
|
|
);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
frameCount++;
|
2020-09-15 15:17:03 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::nextFrameDEV()
|
|
|
|
{
|
|
|
|
bool ret = nextFrame();
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
if (lastFrameDurationMs > eachFrameMillis)
|
|
|
|
TXLED1;
|
|
|
|
else
|
|
|
|
TXLED0;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Arduboy2Base::cpuLoad()
|
|
|
|
{
|
|
|
|
return lastFrameDurationMs*100 / eachFrameMillis;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::initRandomSeed()
|
|
|
|
{
|
|
|
|
randomSeed(generateRandomSeed());
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Graphics */
|
|
|
|
|
|
|
|
void Arduboy2Base::clear()
|
|
|
|
{
|
|
|
|
fillScreen(BLACK);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawPixel(int16_t x, int16_t y, uint8_t color)
|
|
|
|
{
|
|
|
|
#ifdef PIXEL_SAFE_MODE
|
|
|
|
if (x < 0 || x > (WIDTH-1) || y < 0 || y > (HEIGHT-1))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
uint16_t row_offset;
|
|
|
|
uint8_t bit;
|
2020-09-15 15:17:03 +00:00
|
|
|
#if defined __AVR_ARCH__
|
2018-09-10 18:57:16 +00:00
|
|
|
asm volatile
|
|
|
|
(
|
2019-07-22 20:29:34 +00:00
|
|
|
// bit = 1 << (y & 7)
|
|
|
|
"ldi %[bit], 1 \n" //bit = 1;
|
|
|
|
"sbrc %[y], 1 \n" //if (y & _BV(1)) bit = 4;
|
|
|
|
"ldi %[bit], 4 \n"
|
|
|
|
"sbrc %[y], 0 \n" //if (y & _BV(0)) bit = bit << 1;
|
|
|
|
"lsl %[bit] \n"
|
|
|
|
"sbrc %[y], 2 \n" //if (y & _BV(2)) bit = (bit << 4) | (bit >> 4);
|
|
|
|
"swap %[bit] \n"
|
|
|
|
//row_offset = y / 8 * WIDTH + x;
|
|
|
|
"andi %A[y], 0xf8 \n" //row_offset = (y & 0xF8) * WIDTH / 8
|
|
|
|
"mul %[width_offset], %A[y] \n"
|
2018-09-10 18:57:16 +00:00
|
|
|
"movw %[row_offset], r0 \n"
|
|
|
|
"clr __zero_reg__ \n"
|
2019-07-22 20:29:34 +00:00
|
|
|
"add %A[row_offset], %[x] \n" //row_offset += x
|
2020-09-15 15:17:03 +00:00
|
|
|
#if WIDTH != 128
|
2019-07-22 20:29:34 +00:00
|
|
|
"adc %B[row_offset], __zero_reg__ \n" // only non 128 width can overflow
|
2020-09-15 15:17:03 +00:00
|
|
|
#endif
|
|
|
|
"subi r26, lo8(-(%[buf])) \n"
|
|
|
|
"sbci r27, hi8(-(%[buf])) \n"
|
|
|
|
"ld r0, X \n"
|
|
|
|
"or r0, %[bit] \n"
|
|
|
|
"sbrs %[col], 0 \n"
|
|
|
|
"eor r0, %[bit] \n"
|
|
|
|
"st X, r0 \n"
|
2019-07-22 20:29:34 +00:00
|
|
|
: [row_offset] "=&x" (row_offset), // upper register (ANDI)
|
|
|
|
[bit] "=&d" (bit), // upper register (LDI)
|
|
|
|
[y] "+d" (y) // upper register (ANDI), must be writable
|
|
|
|
: [width_offset] "r" ((uint8_t)(WIDTH/8)),
|
2020-09-15 15:17:03 +00:00
|
|
|
[x] "r" ((uint8_t)x),
|
|
|
|
[buf] "" (&sBuffer),
|
|
|
|
[col] "r" (color)
|
2018-09-10 18:57:16 +00:00
|
|
|
:
|
|
|
|
);
|
2020-09-15 15:17:03 +00:00
|
|
|
#else
|
2019-07-22 20:29:34 +00:00
|
|
|
bit = 1 << (y & 7);
|
|
|
|
row_offset = (y & 0xF8) * WIDTH / 8 + x;
|
|
|
|
uint8_t data = sBuffer[row_offset] | bit;
|
|
|
|
if (!color) data ^= bit;
|
|
|
|
sBuffer[row_offset] = data;
|
2020-09-15 15:17:03 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Arduboy2Base::getPixel(uint8_t x, uint8_t y)
|
|
|
|
{
|
|
|
|
uint8_t row = y / 8;
|
|
|
|
uint8_t bit_position = y % 8;
|
|
|
|
return (sBuffer[(row*WIDTH) + x] & _BV(bit_position)) >> bit_position;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawCircle(int16_t x0, int16_t y0, uint8_t r, uint8_t color)
|
|
|
|
{
|
|
|
|
int16_t f = 1 - r;
|
|
|
|
int16_t ddF_x = 1;
|
|
|
|
int16_t ddF_y = -2 * r;
|
|
|
|
int16_t x = 0;
|
|
|
|
int16_t y = r;
|
|
|
|
|
|
|
|
drawPixel(x0, y0+r, color);
|
|
|
|
drawPixel(x0, y0-r, color);
|
|
|
|
drawPixel(x0+r, y0, color);
|
|
|
|
drawPixel(x0-r, y0, color);
|
|
|
|
|
|
|
|
while (x<y)
|
|
|
|
{
|
|
|
|
if (f >= 0)
|
|
|
|
{
|
|
|
|
y--;
|
|
|
|
ddF_y += 2;
|
|
|
|
f += ddF_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
x++;
|
|
|
|
ddF_x += 2;
|
|
|
|
f += ddF_x;
|
|
|
|
|
|
|
|
drawPixel(x0 + x, y0 + y, color);
|
|
|
|
drawPixel(x0 - x, y0 + y, color);
|
|
|
|
drawPixel(x0 + x, y0 - y, color);
|
|
|
|
drawPixel(x0 - x, y0 - y, color);
|
|
|
|
drawPixel(x0 + y, y0 + x, color);
|
|
|
|
drawPixel(x0 - y, y0 + x, color);
|
|
|
|
drawPixel(x0 + y, y0 - x, color);
|
|
|
|
drawPixel(x0 - y, y0 - x, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawCircleHelper
|
|
|
|
(int16_t x0, int16_t y0, uint8_t r, uint8_t corners, uint8_t color)
|
|
|
|
{
|
|
|
|
int16_t f = 1 - r;
|
|
|
|
int16_t ddF_x = 1;
|
|
|
|
int16_t ddF_y = -2 * r;
|
|
|
|
int16_t x = 0;
|
|
|
|
int16_t y = r;
|
|
|
|
|
|
|
|
while (x<y)
|
|
|
|
{
|
|
|
|
if (f >= 0)
|
|
|
|
{
|
|
|
|
y--;
|
|
|
|
ddF_y += 2;
|
|
|
|
f += ddF_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
x++;
|
|
|
|
ddF_x += 2;
|
|
|
|
f += ddF_x;
|
|
|
|
|
|
|
|
if (corners & 0x4) // lower right
|
|
|
|
{
|
|
|
|
drawPixel(x0 + x, y0 + y, color);
|
|
|
|
drawPixel(x0 + y, y0 + x, color);
|
|
|
|
}
|
|
|
|
if (corners & 0x2) // upper right
|
|
|
|
{
|
|
|
|
drawPixel(x0 + x, y0 - y, color);
|
|
|
|
drawPixel(x0 + y, y0 - x, color);
|
|
|
|
}
|
|
|
|
if (corners & 0x8) // lower left
|
|
|
|
{
|
|
|
|
drawPixel(x0 - y, y0 + x, color);
|
|
|
|
drawPixel(x0 - x, y0 + y, color);
|
|
|
|
}
|
|
|
|
if (corners & 0x1) // upper left
|
|
|
|
{
|
|
|
|
drawPixel(x0 - y, y0 - x, color);
|
|
|
|
drawPixel(x0 - x, y0 - y, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillCircle(int16_t x0, int16_t y0, uint8_t r, uint8_t color)
|
|
|
|
{
|
|
|
|
drawFastVLine(x0, y0-r, 2*r+1, color);
|
|
|
|
fillCircleHelper(x0, y0, r, 3, 0, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillCircleHelper
|
|
|
|
(int16_t x0, int16_t y0, uint8_t r, uint8_t sides, int16_t delta,
|
|
|
|
uint8_t color)
|
|
|
|
{
|
|
|
|
int16_t f = 1 - r;
|
|
|
|
int16_t ddF_x = 1;
|
|
|
|
int16_t ddF_y = -2 * r;
|
|
|
|
int16_t x = 0;
|
|
|
|
int16_t y = r;
|
|
|
|
|
|
|
|
while (x < y)
|
|
|
|
{
|
|
|
|
if (f >= 0)
|
|
|
|
{
|
|
|
|
y--;
|
|
|
|
ddF_y += 2;
|
|
|
|
f += ddF_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
x++;
|
|
|
|
ddF_x += 2;
|
|
|
|
f += ddF_x;
|
|
|
|
|
|
|
|
if (sides & 0x1) // right side
|
|
|
|
{
|
|
|
|
drawFastVLine(x0+x, y0-y, 2*y+1+delta, color);
|
|
|
|
drawFastVLine(x0+y, y0-x, 2*x+1+delta, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sides & 0x2) // left side
|
|
|
|
{
|
|
|
|
drawFastVLine(x0-x, y0-y, 2*y+1+delta, color);
|
|
|
|
drawFastVLine(x0-y, y0-x, 2*x+1+delta, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawLine
|
|
|
|
(int16_t x0, int16_t y0, int16_t x1, int16_t y1, uint8_t color)
|
|
|
|
{
|
|
|
|
// bresenham's algorithm - thx wikpedia
|
|
|
|
bool steep = abs(y1 - y0) > abs(x1 - x0);
|
|
|
|
if (steep) {
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(x0, y0);
|
|
|
|
swapInt16(x1, y1);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (x0 > x1) {
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(x0, x1);
|
|
|
|
swapInt16(y0, y1);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int16_t dx, dy;
|
|
|
|
dx = x1 - x0;
|
|
|
|
dy = abs(y1 - y0);
|
|
|
|
|
|
|
|
int16_t err = dx / 2;
|
|
|
|
int8_t ystep;
|
|
|
|
|
|
|
|
if (y0 < y1)
|
|
|
|
{
|
|
|
|
ystep = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ystep = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; x0 <= x1; x0++)
|
|
|
|
{
|
|
|
|
if (steep)
|
|
|
|
{
|
|
|
|
drawPixel(y0, x0, color);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
drawPixel(x0, y0, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
err -= dy;
|
|
|
|
if (err < 0)
|
|
|
|
{
|
|
|
|
y0 += ystep;
|
|
|
|
err += dx;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawRect
|
|
|
|
(int16_t x, int16_t y, uint8_t w, uint8_t h, uint8_t color)
|
|
|
|
{
|
|
|
|
drawFastHLine(x, y, w, color);
|
|
|
|
drawFastHLine(x, y+h-1, w, color);
|
|
|
|
drawFastVLine(x, y, h, color);
|
|
|
|
drawFastVLine(x+w-1, y, h, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawFastVLine
|
|
|
|
(int16_t x, int16_t y, uint8_t h, uint8_t color)
|
|
|
|
{
|
|
|
|
int end = y+h;
|
|
|
|
for (int a = max(0,y); a < min(end,HEIGHT); a++)
|
|
|
|
{
|
|
|
|
drawPixel(x,a,color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawFastHLine
|
|
|
|
(int16_t x, int16_t y, uint8_t w, uint8_t color)
|
|
|
|
{
|
|
|
|
int16_t xEnd; // last x point + 1
|
|
|
|
|
|
|
|
// Do y bounds checks
|
|
|
|
if (y < 0 || y >= HEIGHT)
|
|
|
|
return;
|
|
|
|
|
|
|
|
xEnd = x + w;
|
|
|
|
|
|
|
|
// Check if the entire line is not on the display
|
|
|
|
if (xEnd <= 0 || x >= WIDTH)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Don't start before the left edge
|
|
|
|
if (x < 0)
|
|
|
|
x = 0;
|
|
|
|
|
|
|
|
// Don't end past the right edge
|
|
|
|
if (xEnd > WIDTH)
|
|
|
|
xEnd = WIDTH;
|
|
|
|
|
|
|
|
// calculate actual width (even if unchanged)
|
|
|
|
w = xEnd - x;
|
|
|
|
|
|
|
|
// buffer pointer plus row offset + x offset
|
|
|
|
register uint8_t *pBuf = sBuffer + ((y / 8) * WIDTH) + x;
|
|
|
|
|
|
|
|
// pixel mask
|
|
|
|
register uint8_t mask = 1 << (y & 7);
|
|
|
|
|
|
|
|
switch (color)
|
|
|
|
{
|
|
|
|
case WHITE:
|
|
|
|
while (w--)
|
|
|
|
{
|
|
|
|
*pBuf++ |= mask;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BLACK:
|
|
|
|
mask = ~mask;
|
|
|
|
while (w--)
|
|
|
|
{
|
|
|
|
*pBuf++ &= mask;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillRect
|
|
|
|
(int16_t x, int16_t y, uint8_t w, uint8_t h, uint8_t color)
|
|
|
|
{
|
|
|
|
// stupidest version - update in subclasses if desired!
|
|
|
|
for (int16_t i=x; i<x+w; i++)
|
|
|
|
{
|
|
|
|
drawFastVLine(i, y, h, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillScreen(uint8_t color)
|
|
|
|
{
|
|
|
|
// C version:
|
|
|
|
//
|
|
|
|
// if (color != BLACK)
|
|
|
|
// {
|
|
|
|
// color = 0xFF; // all pixels on
|
|
|
|
// }
|
|
|
|
// for (int16_t i = 0; i < WIDTH * HEIGTH / 8; i++)
|
|
|
|
// {
|
|
|
|
// sBuffer[i] = color;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// This asm version is semi hard coded for 128x64, 96x96 and 128x96 resolutions
|
|
|
|
|
|
|
|
// local variable for screen buffer pointer,
|
|
|
|
// which can be declared a read-write operand
|
|
|
|
uint8_t* bPtr = sBuffer;
|
|
|
|
|
|
|
|
asm volatile
|
|
|
|
(
|
|
|
|
// if value is zero, skip assigning to 0xff
|
|
|
|
"cpse %[color], __zero_reg__\n"
|
|
|
|
"ldi %[color], 0xFF\n"
|
|
|
|
// counter = WIDTH * HEIGHT / 8 / 8
|
|
|
|
"ldi r24, %[cnt]\n"
|
2019-07-22 20:29:34 +00:00
|
|
|
"1:\n"
|
2018-09-10 18:57:16 +00:00
|
|
|
// (4x/8x) store color into screen buffer,
|
|
|
|
// then increment buffer position
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
#if defined(OLED_96X96) || defined(OLED_128X96) || defined(OLED_128X128) || defined(OLED_128X96_ON_128X128) || defined(OLED_96X96_ON_128X128)
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
"st Z+, %[color]\n"
|
|
|
|
#endif
|
|
|
|
// decrease counter
|
|
|
|
"subi r24, 1\n"
|
|
|
|
// repeat for 256, 144 or 192 loops depending on screen resolution
|
2019-07-22 20:29:34 +00:00
|
|
|
"brcc 1b\n"
|
2018-09-10 18:57:16 +00:00
|
|
|
: [color] "+d" (color),
|
|
|
|
"+z" (bPtr)
|
|
|
|
#if defined(OLED_96X96) || defined(OLED_128X96) || defined(OLED_128X128) || defined(OLED_128X96_ON_128X128) || defined(OLED_96X96_ON_128X128)
|
|
|
|
: [cnt] "M" (WIDTH * HEIGHT / 8 / 8 - 1)
|
|
|
|
#else
|
|
|
|
: [cnt] "M" (WIDTH * HEIGHT / 8 / 4 - 1)
|
|
|
|
#endif
|
|
|
|
: "r24"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawRoundRect
|
|
|
|
(int16_t x, int16_t y, uint8_t w, uint8_t h, uint8_t r, uint8_t color)
|
|
|
|
{
|
|
|
|
// smarter version
|
|
|
|
drawFastHLine(x+r, y, w-2*r, color); // Top
|
|
|
|
drawFastHLine(x+r, y+h-1, w-2*r, color); // Bottom
|
|
|
|
drawFastVLine(x, y+r, h-2*r, color); // Left
|
|
|
|
drawFastVLine(x+w-1, y+r, h-2*r, color); // Right
|
|
|
|
// draw four corners
|
|
|
|
drawCircleHelper(x+r, y+r, r, 1, color);
|
|
|
|
drawCircleHelper(x+w-r-1, y+r, r, 2, color);
|
|
|
|
drawCircleHelper(x+w-r-1, y+h-r-1, r, 4, color);
|
|
|
|
drawCircleHelper(x+r, y+h-r-1, r, 8, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillRoundRect
|
|
|
|
(int16_t x, int16_t y, uint8_t w, uint8_t h, uint8_t r, uint8_t color)
|
|
|
|
{
|
|
|
|
// smarter version
|
|
|
|
fillRect(x+r, y, w-2*r, h, color);
|
|
|
|
|
|
|
|
// draw four corners
|
|
|
|
fillCircleHelper(x+w-r-1, y+r, r, 1, h-2*r-1, color);
|
|
|
|
fillCircleHelper(x+r, y+r, r, 2, h-2*r-1, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawTriangle
|
|
|
|
(int16_t x0, int16_t y0, int16_t x1, int16_t y1, int16_t x2, int16_t y2, uint8_t color)
|
|
|
|
{
|
|
|
|
drawLine(x0, y0, x1, y1, color);
|
|
|
|
drawLine(x1, y1, x2, y2, color);
|
|
|
|
drawLine(x2, y2, x0, y0, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::fillTriangle
|
|
|
|
(int16_t x0, int16_t y0, int16_t x1, int16_t y1, int16_t x2, int16_t y2, uint8_t color)
|
|
|
|
{
|
|
|
|
|
|
|
|
int16_t a, b, y, last;
|
|
|
|
// Sort coordinates by Y order (y2 >= y1 >= y0)
|
|
|
|
if (y0 > y1)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(y0, y1); swapInt16(x0, x1);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
if (y1 > y2)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(y2, y1); swapInt16(x2, x1);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
if (y0 > y1)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(y0, y1); swapInt16(x0, x1);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(y0 == y2)
|
|
|
|
{ // Handle awkward all-on-same-line case as its own thing
|
|
|
|
a = b = x0;
|
|
|
|
if(x1 < a)
|
|
|
|
{
|
|
|
|
a = x1;
|
|
|
|
}
|
|
|
|
else if(x1 > b)
|
|
|
|
{
|
|
|
|
b = x1;
|
|
|
|
}
|
|
|
|
if(x2 < a)
|
|
|
|
{
|
|
|
|
a = x2;
|
|
|
|
}
|
|
|
|
else if(x2 > b)
|
|
|
|
{
|
|
|
|
b = x2;
|
|
|
|
}
|
|
|
|
drawFastHLine(a, y0, b-a+1, color);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t dx01 = x1 - x0,
|
|
|
|
dy01 = y1 - y0,
|
|
|
|
dx02 = x2 - x0,
|
|
|
|
dy02 = y2 - y0,
|
|
|
|
dx12 = x2 - x1,
|
|
|
|
dy12 = y2 - y1,
|
|
|
|
sa = 0,
|
|
|
|
sb = 0;
|
|
|
|
|
|
|
|
// For upper part of triangle, find scanline crossings for segments
|
|
|
|
// 0-1 and 0-2. If y1=y2 (flat-bottomed triangle), the scanline y1
|
|
|
|
// is included here (and second loop will be skipped, avoiding a /0
|
|
|
|
// error there), otherwise scanline y1 is skipped here and handled
|
|
|
|
// in the second loop...which also avoids a /0 error here if y0=y1
|
|
|
|
// (flat-topped triangle).
|
|
|
|
if (y1 == y2)
|
|
|
|
{
|
|
|
|
last = y1; // Include y1 scanline
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
last = y1-1; // Skip it
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for(y = y0; y <= last; y++)
|
|
|
|
{
|
|
|
|
a = x0 + sa / dy01;
|
|
|
|
b = x0 + sb / dy02;
|
|
|
|
sa += dx01;
|
|
|
|
sb += dx02;
|
|
|
|
|
|
|
|
if(a > b)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(a,b);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
drawFastHLine(a, y, b-a+1, color);
|
|
|
|
}
|
|
|
|
|
|
|
|
// For lower part of triangle, find scanline crossings for segments
|
|
|
|
// 0-2 and 1-2. This loop is skipped if y1=y2.
|
|
|
|
sa = dx12 * (y - y1);
|
|
|
|
sb = dx02 * (y - y0);
|
|
|
|
|
|
|
|
for(; y <= y2; y++)
|
|
|
|
{
|
|
|
|
a = x1 + sa / dy12;
|
|
|
|
b = x0 + sb / dy02;
|
|
|
|
sa += dx12;
|
|
|
|
sb += dx02;
|
|
|
|
|
|
|
|
if(a > b)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
swapInt16(a,b);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
drawFastHLine(a, y, b-a+1, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::drawBitmap
|
|
|
|
(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t w, uint8_t h,
|
|
|
|
uint8_t color)
|
|
|
|
{
|
|
|
|
// no need to draw at all if we're offscreen
|
2020-09-15 15:17:03 +00:00
|
|
|
if (x + w <= 0 || x > WIDTH - 1 || y + h <= 0 || y > HEIGHT - 1)
|
2018-09-10 18:57:16 +00:00
|
|
|
return;
|
|
|
|
|
2019-07-22 20:29:34 +00:00
|
|
|
int8_t yOffset = y & 7;
|
2019-07-23 12:51:05 +00:00
|
|
|
int8_t sRow = y >> 3;
|
2019-07-22 20:29:34 +00:00
|
|
|
uint8_t rows = h >> 3;
|
2021-07-28 20:51:37 +00:00
|
|
|
if (h % 8 != 0) rows++;
|
2018-09-10 18:57:16 +00:00
|
|
|
for (int a = 0; a < rows; a++) {
|
|
|
|
int bRow = sRow + a;
|
|
|
|
if (bRow > (HEIGHT/8)-1) break;
|
|
|
|
if (bRow > -2) {
|
|
|
|
for (int iCol = 0; iCol<w; iCol++) {
|
|
|
|
if (iCol + x > (WIDTH-1)) break;
|
|
|
|
if (iCol + x >= 0) {
|
2019-07-22 20:29:34 +00:00
|
|
|
uint16_t data = pgm_read_byte(bitmap+(a*w)+iCol) << yOffset;
|
2018-09-10 18:57:16 +00:00
|
|
|
if (bRow >= 0) {
|
|
|
|
if (color == WHITE)
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[(bRow*WIDTH) + x + iCol] |= data;
|
2018-09-10 18:57:16 +00:00
|
|
|
else if (color == BLACK)
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[(bRow*WIDTH) + x + iCol] &= ~data;
|
2018-09-10 18:57:16 +00:00
|
|
|
else
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[(bRow*WIDTH) + x + iCol] ^= data;
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
if (yOffset && bRow<(HEIGHT/8)-1 && bRow > -2) {
|
|
|
|
if (color == WHITE)
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[((bRow+1)*WIDTH) + x + iCol] |= (data >> 8);
|
2018-09-10 18:57:16 +00:00
|
|
|
else if (color == BLACK)
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[((bRow+1)*WIDTH) + x + iCol] &= ~(data >> 8);
|
2018-09-10 18:57:16 +00:00
|
|
|
else
|
2019-07-22 20:29:34 +00:00
|
|
|
sBuffer[((bRow+1)*WIDTH) + x + iCol] ^= (data >> 8);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Arduboy2Base::drawSlowXYBitmap
|
|
|
|
(int16_t x, int16_t y, const uint8_t *bitmap, uint8_t w, uint8_t h, uint8_t color)
|
|
|
|
{
|
|
|
|
// no need to draw at all of we're offscreen
|
2020-09-15 15:17:03 +00:00
|
|
|
if (x + w <= 0 || x > WIDTH - 1 || y + h <= 0 || y > HEIGHT - 1)
|
2018-09-10 18:57:16 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
int16_t xi, yi, byteWidth = (w + 7) / 8;
|
|
|
|
for(yi = 0; yi < h; yi++) {
|
|
|
|
for(xi = 0; xi < w; xi++ ) {
|
|
|
|
if(pgm_read_byte(bitmap + yi * byteWidth + xi / 8) & (128 >> (xi & 7))) {
|
|
|
|
drawPixel(x + xi, y + yi, color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Helper for drawCompressed()
|
2020-09-15 15:17:03 +00:00
|
|
|
struct Arduboy2Base::BitStreamReader
|
2018-09-10 18:57:16 +00:00
|
|
|
{
|
|
|
|
const uint8_t *source;
|
|
|
|
uint16_t sourceIndex;
|
|
|
|
uint8_t bitBuffer;
|
|
|
|
uint8_t byteBuffer;
|
|
|
|
|
|
|
|
BitStreamReader(const uint8_t *source)
|
|
|
|
: source(source), sourceIndex(), bitBuffer(), byteBuffer()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t readBits(uint8_t bitCount)
|
|
|
|
{
|
|
|
|
uint16_t result = 0;
|
|
|
|
for (uint8_t i = 0; i < bitCount; i++)
|
|
|
|
{
|
|
|
|
if (this->bitBuffer == 0)
|
|
|
|
{
|
|
|
|
this->bitBuffer = 0x1;
|
|
|
|
this->byteBuffer = pgm_read_byte(&this->source[this->sourceIndex]);
|
|
|
|
++this->sourceIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((this->byteBuffer & this->bitBuffer) != 0)
|
2019-07-22 20:29:34 +00:00
|
|
|
result |= (1 << i);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
this->bitBuffer += this->bitBuffer;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void Arduboy2Base::drawCompressed(int16_t sx, int16_t sy, const uint8_t *bitmap, uint8_t color)
|
|
|
|
{
|
|
|
|
// set up decompress state
|
|
|
|
BitStreamReader cs = BitStreamReader(bitmap);
|
|
|
|
|
|
|
|
// read header
|
|
|
|
int width = (int)cs.readBits(8) + 1;
|
|
|
|
int height = (int)cs.readBits(8) + 1;
|
|
|
|
uint8_t spanColour = (uint8_t)cs.readBits(1); // starting colour
|
|
|
|
|
|
|
|
// no need to draw at all if we're offscreen
|
2020-09-15 15:17:03 +00:00
|
|
|
if ((sx + width <= 0) || (sx > WIDTH - 1) || (sy + height <= 0) || (sy > HEIGHT - 1))
|
2018-09-10 18:57:16 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// sy = sy - (frame * height);
|
|
|
|
int yOffset = abs(sy) % 8;
|
|
|
|
int startRow = sy / 8;
|
|
|
|
if (sy < 0) {
|
|
|
|
startRow--;
|
|
|
|
yOffset = 8 - yOffset;
|
|
|
|
}
|
|
|
|
int rows = height / 8;
|
|
|
|
if ((height % 8) != 0)
|
|
|
|
++rows;
|
|
|
|
|
|
|
|
int rowOffset = 0; // +(frame*rows);
|
|
|
|
int columnOffset = 0;
|
|
|
|
|
|
|
|
uint8_t byte = 0x00;
|
|
|
|
uint8_t bit = 0x01;
|
|
|
|
while (rowOffset < rows) // + (frame*rows))
|
|
|
|
{
|
|
|
|
uint16_t bitLength = 1;
|
|
|
|
while (cs.readBits(1) == 0)
|
|
|
|
bitLength += 2;
|
|
|
|
|
|
|
|
uint16_t len = cs.readBits(bitLength) + 1; // span length
|
|
|
|
|
|
|
|
// draw the span
|
|
|
|
for (uint16_t i = 0; i < len; ++i)
|
|
|
|
{
|
|
|
|
if (spanColour != 0)
|
|
|
|
byte |= bit;
|
|
|
|
bit <<= 1;
|
|
|
|
|
|
|
|
if (bit == 0) // reached end of byte
|
|
|
|
{
|
|
|
|
// draw
|
|
|
|
int bRow = startRow + rowOffset;
|
|
|
|
|
|
|
|
//if (byte) // possible optimisation
|
|
|
|
if ((bRow <= (HEIGHT / 8) - 1) && (bRow > -2) &&
|
|
|
|
(columnOffset + sx <= (WIDTH - 1)) && (columnOffset + sx >= 0))
|
|
|
|
{
|
|
|
|
int16_t offset = (bRow * WIDTH) + sx + columnOffset;
|
|
|
|
if (bRow >= 0)
|
|
|
|
{
|
|
|
|
int16_t index = offset;
|
|
|
|
uint8_t value = byte << yOffset;
|
|
|
|
|
|
|
|
if (color != 0)
|
|
|
|
sBuffer[index] |= value;
|
|
|
|
else
|
|
|
|
sBuffer[index] &= ~value;
|
|
|
|
}
|
|
|
|
if ((yOffset != 0) && (bRow < (HEIGHT / 8) - 1))
|
|
|
|
{
|
|
|
|
int16_t index = offset + WIDTH;
|
|
|
|
uint8_t value = byte >> (8 - yOffset);
|
|
|
|
|
|
|
|
if (color != 0)
|
|
|
|
sBuffer[index] |= value;
|
|
|
|
else
|
|
|
|
sBuffer[index] &= ~value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// iterate
|
|
|
|
++columnOffset;
|
|
|
|
if (columnOffset >= width)
|
|
|
|
{
|
|
|
|
columnOffset = 0;
|
|
|
|
++rowOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset byte
|
|
|
|
byte = 0x00;
|
|
|
|
bit = 0x01;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
spanColour ^= 0x01; // toggle colour bit (bit 0) for next span
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::display()
|
|
|
|
{
|
|
|
|
paintScreen(sBuffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::display(bool clear)
|
|
|
|
{
|
|
|
|
paintScreen(sBuffer, clear);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t* Arduboy2Base::getBuffer()
|
|
|
|
{
|
|
|
|
return sBuffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::pressed(uint8_t buttons)
|
|
|
|
{
|
|
|
|
return (buttonsState() & buttons) == buttons;
|
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
bool Arduboy2Base::anyPressed(uint8_t buttons)
|
|
|
|
{
|
|
|
|
return (buttonsState() & buttons) != 0;
|
|
|
|
}
|
|
|
|
|
2018-09-10 18:57:16 +00:00
|
|
|
bool Arduboy2Base::notPressed(uint8_t buttons)
|
|
|
|
{
|
|
|
|
return (buttonsState() & buttons) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::pollButtons()
|
|
|
|
{
|
|
|
|
previousButtonState = currentButtonState;
|
|
|
|
currentButtonState = buttonsState();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::justPressed(uint8_t button)
|
|
|
|
{
|
|
|
|
return (!(previousButtonState & button) && (currentButtonState & button));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::justReleased(uint8_t button)
|
|
|
|
{
|
|
|
|
return ((previousButtonState & button) && !(currentButtonState & button));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::collide(Point point, Rect rect)
|
|
|
|
{
|
|
|
|
return ((point.x >= rect.x) && (point.x < rect.x + rect.width) &&
|
|
|
|
(point.y >= rect.y) && (point.y < rect.y + rect.height));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::collide(Rect rect1, Rect rect2)
|
|
|
|
{
|
|
|
|
return !(rect2.x >= rect1.x + rect1.width ||
|
|
|
|
rect2.x + rect2.width <= rect1.x ||
|
|
|
|
rect2.y >= rect1.y + rect1.height ||
|
|
|
|
rect2.y + rect2.height <= rect1.y);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t Arduboy2Base::readUnitID()
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
return EEPROM.read(eepromUnitID) |
|
|
|
|
(((uint16_t)(EEPROM.read(eepromUnitID + 1))) << 8);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::writeUnitID(uint16_t id)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
EEPROM.update(eepromUnitID, (uint8_t)(id & 0xff));
|
|
|
|
EEPROM.update(eepromUnitID + 1, (uint8_t)(id >> 8));
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Arduboy2Base::readUnitName(char* name)
|
|
|
|
{
|
|
|
|
char val;
|
|
|
|
uint8_t dest;
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t src = eepromUnitName;
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
for (dest = 0; dest < ARDUBOY_UNIT_NAME_LEN; dest++)
|
|
|
|
{
|
|
|
|
val = EEPROM.read(src);
|
|
|
|
name[dest] = val;
|
|
|
|
src++;
|
|
|
|
if (val == 0x00 || (byte)val == 0xFF) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
name[dest] = 0x00;
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
void Arduboy2Base::writeUnitName(const char* name)
|
2018-09-10 18:57:16 +00:00
|
|
|
{
|
|
|
|
bool done = false;
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t dest = eepromUnitName;
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
for (uint8_t src = 0; src < ARDUBOY_UNIT_NAME_LEN; src++)
|
|
|
|
{
|
|
|
|
if (name[src] == 0x00) {
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
// write character or 0 pad if finished
|
|
|
|
EEPROM.update(dest, done ? 0x00 : name[src]);
|
|
|
|
dest++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::readShowBootLogoFlag()
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
return (EEPROM.read(eepromSysFlags) & sysFlagShowLogoMask);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::writeShowBootLogoFlag(bool val)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t flags = EEPROM.read(eepromSysFlags);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
bitWrite(flags, sysFlagShowLogoBit, val);
|
|
|
|
EEPROM.update(eepromSysFlags, flags);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::readShowUnitNameFlag()
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
return (EEPROM.read(eepromSysFlags) & sysFlagUnameMask);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::writeShowUnitNameFlag(bool val)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t flags = EEPROM.read(eepromSysFlags);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
bitWrite(flags, sysFlagUnameBit, val);
|
|
|
|
EEPROM.update(eepromSysFlags, flags);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2Base::readShowBootLogoLEDsFlag()
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
return (EEPROM.read(eepromSysFlags) & sysFlagShowLogoLEDsMask);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2Base::writeShowBootLogoLEDsFlag(bool val)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t flags = EEPROM.read(eepromSysFlags);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
bitWrite(flags, sysFlagShowLogoLEDsBit, val);
|
|
|
|
EEPROM.update(eepromSysFlags, flags);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
void Arduboy2Base::swapInt16(int16_t& a, int16_t& b)
|
2018-09-10 18:57:16 +00:00
|
|
|
{
|
|
|
|
int16_t temp = a;
|
|
|
|
a = b;
|
|
|
|
b = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//====================================
|
|
|
|
//========== class Arduboy2 ==========
|
|
|
|
//====================================
|
|
|
|
|
2021-07-28 21:28:54 +00:00
|
|
|
int16_t Arduboy2::cursor_x = 0;
|
|
|
|
int16_t Arduboy2::cursor_y = 0;
|
|
|
|
uint8_t Arduboy2::textColor = 1;
|
|
|
|
uint8_t Arduboy2::textBackground = 0;
|
|
|
|
uint8_t Arduboy2::textSize = 1;
|
|
|
|
bool Arduboy2::textWrap = false;
|
|
|
|
//bool Arduboy2::textRaw = false;
|
|
|
|
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
// bootLogoText() should be kept in sync with bootLogoShell()
|
|
|
|
// if changes are made to one, equivalent changes should be made to the other
|
|
|
|
void Arduboy2::bootLogoText()
|
|
|
|
{
|
|
|
|
bool showLEDs = readShowBootLogoLEDsFlag();
|
|
|
|
|
|
|
|
if (!readShowBootLogoFlag()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_ON, RGB_OFF, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(RED_LED, RGB_ON);
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (int8_t y = -16; y <= 24; y++) {
|
|
|
|
if (pressed(RIGHT_BUTTON)) {
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_OFF); // all LEDs off
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs && y == 4) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_ON, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(RED_LED, RGB_OFF); // red LED off
|
|
|
|
digitalWriteRGB(GREEN_LED, RGB_ON); // green LED on
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Using display(CLEAR_BUFFER) instead of clear() may save code space.
|
|
|
|
// The extra time it takes to repaint the previous logo isn't an issue.
|
|
|
|
display(CLEAR_BUFFER);
|
|
|
|
cursor_x = 23 - (64 - WIDTH / 2);
|
|
|
|
cursor_y = y;
|
|
|
|
textSize = 2;
|
|
|
|
print(F("ARDUBOY"));
|
|
|
|
textSize = 1;
|
|
|
|
display();
|
2020-09-15 15:17:03 +00:00
|
|
|
delayByte(11);
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (showLEDs) {
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_ON);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(GREEN_LED, RGB_OFF); // green LED off
|
|
|
|
digitalWriteRGB(BLUE_LED, RGB_ON); // blue LED on
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
}
|
|
|
|
delayShort(400);
|
2018-09-16 16:34:36 +00:00
|
|
|
#if defined(LCD_ST7565)
|
|
|
|
digitalWriteRGB(RGB_OFF, RGB_OFF, RGB_OFF);
|
|
|
|
#else
|
2018-09-10 18:57:16 +00:00
|
|
|
digitalWriteRGB(BLUE_LED, RGB_OFF);
|
2018-09-16 16:34:36 +00:00
|
|
|
#endif
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
bootLogoExtra();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::bootLogoExtra()
|
|
|
|
{
|
|
|
|
uint8_t c;
|
|
|
|
|
|
|
|
if (!readShowUnitNameFlag())
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
c = EEPROM.read(eepromUnitName);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
if (c != 0xFF && c != 0x00)
|
|
|
|
{
|
2020-09-15 15:17:03 +00:00
|
|
|
uint8_t i = eepromUnitName;
|
2018-09-10 18:57:16 +00:00
|
|
|
cursor_x = 50 - (64 - WIDTH / 2);
|
|
|
|
cursor_y = 56;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
write(c);
|
|
|
|
c = EEPROM.read(++i);
|
|
|
|
}
|
2020-09-15 15:17:03 +00:00
|
|
|
while (i < eepromUnitName + ARDUBOY_UNIT_NAME_LEN);
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
display();
|
|
|
|
delayShort(1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t Arduboy2::write(uint8_t c)
|
|
|
|
{
|
|
|
|
if (c == '\n')
|
|
|
|
{
|
|
|
|
cursor_y += textSize * 8;
|
|
|
|
cursor_x = 0;
|
|
|
|
}
|
|
|
|
else if (c == '\r')
|
|
|
|
{
|
|
|
|
// skip em
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
drawChar(cursor_x, cursor_y, c, textColor, textBackground, textSize);
|
|
|
|
cursor_x += textSize * 6;
|
|
|
|
if (textWrap && (cursor_x > (WIDTH - textSize * 6)))
|
|
|
|
{
|
|
|
|
// calling ourselves recursively for 'newline' is
|
|
|
|
// 12 bytes smaller than doing the same math here
|
|
|
|
write('\n');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::drawChar
|
|
|
|
(int16_t x, int16_t y, unsigned char c, uint8_t color, uint8_t bg, uint8_t size)
|
|
|
|
{
|
|
|
|
uint8_t line;
|
|
|
|
bool draw_background = bg != color;
|
2020-09-15 15:17:03 +00:00
|
|
|
const uint8_t* bitmap = font5x7 + c * 5;
|
2018-09-10 18:57:16 +00:00
|
|
|
|
|
|
|
if ((x >= WIDTH) || // Clip right
|
|
|
|
(y >= HEIGHT) || // Clip bottom
|
|
|
|
((x + 5 * size - 1) < 0) || // Clip left
|
|
|
|
((y + 8 * size - 1) < 0) // Clip top
|
|
|
|
)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < 6; i++ )
|
|
|
|
{
|
|
|
|
line = pgm_read_byte(bitmap++);
|
|
|
|
if (i == 5) {
|
|
|
|
line = 0x0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint8_t j = 0; j < 8; j++)
|
|
|
|
{
|
|
|
|
uint8_t draw_color = (line & 0x1) ? color : bg;
|
|
|
|
|
|
|
|
if (draw_color || draw_background) {
|
|
|
|
for (uint8_t a = 0; a < size; a++ ) {
|
|
|
|
for (uint8_t b = 0; b < size; b++ ) {
|
|
|
|
drawPixel(x + (i * size) + a, y + (j * size) + b, draw_color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
line >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setCursor(int16_t x, int16_t y)
|
|
|
|
{
|
|
|
|
cursor_x = x;
|
|
|
|
cursor_y = y;
|
|
|
|
}
|
|
|
|
|
2020-09-15 15:17:03 +00:00
|
|
|
void Arduboy2::setCursorX(int16_t x)
|
|
|
|
{
|
|
|
|
cursor_x = x;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setCursorY(int16_t y)
|
|
|
|
{
|
|
|
|
cursor_y = y;
|
|
|
|
}
|
|
|
|
|
2018-09-10 18:57:16 +00:00
|
|
|
int16_t Arduboy2::getCursorX()
|
|
|
|
{
|
|
|
|
return cursor_x;
|
|
|
|
}
|
|
|
|
|
|
|
|
int16_t Arduboy2::getCursorY()
|
|
|
|
{
|
|
|
|
return cursor_y;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setTextColor(uint8_t color)
|
|
|
|
{
|
|
|
|
textColor = color;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Arduboy2::getTextColor()
|
|
|
|
{
|
|
|
|
return textColor;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setTextBackground(uint8_t bg)
|
|
|
|
{
|
|
|
|
textBackground = bg;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Arduboy2::getTextBackground()
|
|
|
|
{
|
|
|
|
return textBackground;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setTextSize(uint8_t s)
|
|
|
|
{
|
|
|
|
// size must always be 1 or higher
|
|
|
|
textSize = max(1, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t Arduboy2::getTextSize()
|
|
|
|
{
|
|
|
|
return textSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::setTextWrap(bool w)
|
|
|
|
{
|
|
|
|
textWrap = w;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Arduboy2::getTextWrap()
|
|
|
|
{
|
|
|
|
return textWrap;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Arduboy2::clear()
|
|
|
|
{
|
|
|
|
Arduboy2Base::clear();
|
|
|
|
cursor_x = cursor_y = 0;
|
|
|
|
}
|
|
|
|
|