mirror of
https://github.com/UpsilonNumworks/Upsilon.git
synced 2026-01-19 16:57:31 +01:00
188 lines
6.9 KiB
C++
188 lines
6.9 KiB
C++
extern "C" {
|
|
#include "modkandinsky.h"
|
|
#include <py/runtime.h>
|
|
}
|
|
#include <escher/palette.h>
|
|
#include <kandinsky.h>
|
|
#include <ion.h>
|
|
#include "port.h"
|
|
#include <py/obj.h>
|
|
|
|
|
|
static mp_obj_t TupleForKDColor(KDColor c) {
|
|
mp_obj_tuple_t * t = static_cast<mp_obj_tuple_t *>(MP_OBJ_TO_PTR(mp_obj_new_tuple(3, NULL)));
|
|
t->items[0] = MP_OBJ_NEW_SMALL_INT(c.red());
|
|
t->items[1] = MP_OBJ_NEW_SMALL_INT(c.green());
|
|
t->items[2] = MP_OBJ_NEW_SMALL_INT(c.blue());
|
|
return MP_OBJ_FROM_PTR(t);
|
|
}
|
|
|
|
/* KDIonContext::sharedContext needs to be set to the wanted Rect before
|
|
* calling kandinsky_get_pixel, kandinsky_set_pixel and kandinsky_draw_string.
|
|
* We do this here with displaySandbox(), which pushes the SandboxController on
|
|
* the stackViewController and forces the window to redraw itself.
|
|
* KDIonContext::sharedContext is set to the frame of the last object drawn. */
|
|
|
|
mp_obj_t modkandinsky_color(size_t n_args, const mp_obj_t *args) {
|
|
mp_obj_t color;
|
|
if (n_args == 1) {
|
|
color = args[0];
|
|
} else if (n_args == 2) {
|
|
mp_raise_TypeError("color takes 1 or 3 arguments");
|
|
return mp_const_none;
|
|
} else {
|
|
assert(n_args == 3);
|
|
color = mp_obj_new_tuple(n_args, args);
|
|
}
|
|
return TupleForKDColor(MicroPython::Color::Parse(color));
|
|
}
|
|
|
|
/* Calling ExecutionEnvironment::displaySandbox() hides the console and switches
|
|
* to another mode. So it's a good idea to retrieve and handle input parameters
|
|
* before calling displaySandbox, otherwise error messages (such as TypeError)
|
|
* won't be visible until the user comes back to the console screen. */
|
|
|
|
mp_obj_t modkandinsky_get_pixel(mp_obj_t x, mp_obj_t y) {
|
|
KDPoint point(mp_obj_get_int(x), mp_obj_get_int(y));
|
|
KDColor c;
|
|
KDIonContext::sharedContext()->getPixel(point, &c);
|
|
return TupleForKDColor(c);
|
|
}
|
|
|
|
mp_obj_t modkandinsky_set_pixel(mp_obj_t x, mp_obj_t y, mp_obj_t input) {
|
|
KDPoint point(mp_obj_get_int(x), mp_obj_get_int(y));
|
|
KDColor kdColor = MicroPython::Color::Parse(input);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->setPixel(point, kdColor);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_draw_string(size_t n_args, const mp_obj_t * args) {
|
|
const char * text = mp_obj_str_get_str(args[0]);
|
|
KDPoint point(mp_obj_get_int(args[1]), mp_obj_get_int(args[2]));
|
|
KDColor textColor = (n_args >= 4) ? MicroPython::Color::Parse(args[3]) : Palette::PrimaryText;
|
|
KDColor backgroundColor = (n_args >= 5) ? MicroPython::Color::Parse(args[4]) : Palette::HomeBackground;
|
|
bool bigFont = (n_args >= 6) ? !mp_obj_is_true(args[5]) : true;
|
|
bool isItalic = (n_args >= 7) ? mp_obj_is_true(args[6]) : false;
|
|
const KDFont * font = KDFont::LargeFont;
|
|
if (bigFont && !isItalic) {
|
|
font = KDFont::LargeFont;
|
|
} else if (!bigFont && !isItalic) {
|
|
font = KDFont::SmallFont;
|
|
} else if (bigFont && isItalic) {
|
|
font = KDFont::ItalicLargeFont;
|
|
} else if (!bigFont && isItalic) {
|
|
font = KDFont::ItalicSmallFont;
|
|
}
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->drawString(text, point, font, textColor, backgroundColor);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_draw_line(size_t n_args, const mp_obj_t * args) {
|
|
mp_int_t x1 = mp_obj_get_int(args[0]);
|
|
mp_int_t y1 = mp_obj_get_int(args[1]);
|
|
mp_int_t x2 = mp_obj_get_int(args[2]);
|
|
mp_int_t y2 = mp_obj_get_int(args[3]);
|
|
KDPoint p1 = KDPoint(x1, y1);
|
|
KDPoint p2 = KDPoint(x2, y2);
|
|
KDColor color = MicroPython::Color::Parse(args[4]);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->drawLine(p1, p2, color);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_draw_circle(size_t n_args, const mp_obj_t * args) {
|
|
mp_int_t cx = mp_obj_get_int(args[0]);
|
|
mp_int_t cy = mp_obj_get_int(args[1]);
|
|
mp_int_t r = mp_obj_get_int(args[2]);
|
|
if(r<0)
|
|
{
|
|
r = -r;
|
|
}
|
|
KDPoint center = KDPoint(cx, cy);
|
|
KDColor color = MicroPython::Color::Parse(args[3]);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->drawCircle(center, r, color);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_fill_rect(size_t n_args, const mp_obj_t * args) {
|
|
mp_int_t x = mp_obj_get_int(args[0]);
|
|
mp_int_t y = mp_obj_get_int(args[1]);
|
|
mp_int_t width = mp_obj_get_int(args[2]);
|
|
mp_int_t height = mp_obj_get_int(args[3]);
|
|
if (width < 0) {
|
|
width = -width;
|
|
x = x - width;
|
|
}
|
|
if (height < 0) {
|
|
height = -height;
|
|
y = y - height;
|
|
}
|
|
KDRect rect(x, y, width, height);
|
|
KDColor color = MicroPython::Color::Parse(args[4]);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->fillRect(rect, color);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_fill_circle(size_t n_args, const mp_obj_t * args) {
|
|
mp_int_t cx = mp_obj_get_int(args[0]);
|
|
mp_int_t cy = mp_obj_get_int(args[1]);
|
|
mp_int_t r = mp_obj_get_int(args[2]);
|
|
if(r<0)
|
|
{
|
|
r = -r;
|
|
}
|
|
KDPoint center = KDPoint(cx, cy);
|
|
KDColor color = MicroPython::Color::Parse(args[3]);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->fillCircle(center, r, color);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_fill_polygon(size_t n_args, const mp_obj_t * args) {
|
|
size_t itemLength;
|
|
mp_obj_t * items;
|
|
|
|
mp_obj_get_array(args[0], &itemLength, &items);
|
|
|
|
KDCoordinate pointsX[itemLength];
|
|
KDCoordinate pointsY[itemLength];
|
|
|
|
if (itemLength < 3) {
|
|
mp_raise_ValueError("polygon must have at least 3 points");
|
|
}
|
|
|
|
for(int i=0; i<itemLength; i++) {
|
|
mp_obj_t * coordinates;
|
|
mp_obj_get_array_fixed_n(items[i], 2, &coordinates);
|
|
|
|
pointsX[i] = mp_obj_get_int(coordinates[0]);
|
|
pointsY[i] = mp_obj_get_int(coordinates[1]);
|
|
}
|
|
|
|
KDColor color = MicroPython::Color::Parse(args[1]);
|
|
MicroPython::ExecutionEnvironment::currentExecutionEnvironment()->displaySandbox();
|
|
KDIonContext::sharedContext()->fillPolygon(pointsX, pointsY, itemLength, color);
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_wait_vblank() {
|
|
micropython_port_interrupt_if_needed();
|
|
Ion::Display::waitForVBlank();
|
|
return mp_const_none;
|
|
}
|
|
|
|
mp_obj_t modkandinsky_get_palette() {
|
|
mp_obj_t modkandinsky_palette_table = mp_obj_new_dict(0);
|
|
mp_obj_dict_store(modkandinsky_palette_table, MP_ROM_QSTR(MP_QSTR_PrimaryText), TupleForKDColor(Palette::PrimaryText));
|
|
mp_obj_dict_store(modkandinsky_palette_table, MP_ROM_QSTR(MP_QSTR_SecondaryText), TupleForKDColor(Palette::SecondaryText));
|
|
mp_obj_dict_store(modkandinsky_palette_table, MP_ROM_QSTR(MP_QSTR_AccentText), TupleForKDColor(Palette::AccentText));
|
|
mp_obj_dict_store(modkandinsky_palette_table, MP_ROM_QSTR(MP_QSTR_Toolbar), TupleForKDColor(Palette::Toolbar));
|
|
mp_obj_dict_store(modkandinsky_palette_table, MP_ROM_QSTR(MP_QSTR_HomeBackground), TupleForKDColor(Palette::HomeBackground));
|
|
|
|
return modkandinsky_palette_table;
|
|
}
|