Twatch/src/main.c

393 lines
11 KiB
C

#include <pebble.h>
#include "twatch.h"
#define FORCE12H 0
static Window *window;
static Layer *s_simple_bg_layer, *s_date_layer, *s_hands_layer;
static TextLayer *s_bt_label, *s_day_label, *s_mon_label;
static GColor colors[KEY_LAST];
static GPath *s_tic_path;
static GPath *s_second_arrow, *s_minute_arrow, *s_hour_arrow;
static char s_mon_buffer[4], s_day_buffer[6];
static TextLayer *s_hour_label[4];
static char s_hour[4][4];
static bool seconds = true;
static bool bluetooth = true;
static bool bt_connected = true;
static void bg_update_proc(Layer *layer, GContext *ctx) {
// Fill background
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_FACE]);
graphics_fill_rect(ctx, layer_get_bounds(layer), 0, GCornerNone);
// Draw tics
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_TIC]);
for (int i = 0; i < 12; i += 1) {
switch (i) {
case 0:
case 3:
case 6:
case 9:
break;
default:
gpath_rotate_to(s_tic_path, TRIG_MAX_ANGLE * i / 12);
gpath_draw_filled(ctx, s_tic_path);
break;
}
}
// Draw numbers
for (int i = 0; i < 4; i += 1) {
int hour = 3 * (i+1);
if (!FORCE12H && clock_is_24h_style()) {
time_t epoch;
struct tm *now;
time(&epoch);
now = localtime(&epoch);
if ((now->tm_hour > 12) || (now->tm_hour == 0)) {
hour += 12;
}
}
snprintf(s_hour[i], 3, "%d", hour);
text_layer_set_text(s_hour_label[i], s_hour[i]);
text_layer_set_text_color(s_hour_label[i], colors[KEY_COLOR_NUM]);
}
}
static void hands_update_proc(Layer *layer, GContext *ctx) {
GRect bounds = layer_get_bounds(layer);
time_t now = time(NULL);
struct tm *t = localtime(&now);
graphics_context_set_stroke_color(ctx, colors[KEY_COLOR_FACE]);
// minute hand
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_MIN]);
gpath_rotate_to(s_minute_arrow, TRIG_MAX_ANGLE * t->tm_min / 60);
gpath_draw_filled(ctx, s_minute_arrow);
gpath_draw_outline(ctx, s_minute_arrow);
// hour hand
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_HR]);
gpath_rotate_to(s_hour_arrow, (TRIG_MAX_ANGLE * (((t->tm_hour % 12) * 6) + (t->tm_min / 10))) / (12 * 6));
gpath_draw_filled(ctx, s_hour_arrow);
gpath_draw_outline(ctx, s_hour_arrow);
if (seconds) {
// second hand
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_SEC]);
gpath_rotate_to(s_second_arrow, TRIG_MAX_ANGLE * t->tm_sec / 60);
gpath_draw_filled(ctx, s_second_arrow);
//gpath_draw_outline(ctx, s_second_arrow);
}
// dot in the middle
graphics_context_set_fill_color(ctx, colors[KEY_COLOR_TIC]);
graphics_fill_circle(ctx, GPoint(bounds.size.w / 2, bounds.size.h / 2), 2);
}
static void date_update_proc(Layer *layer, GContext *ctx) {
time_t now = time(NULL);
struct tm *t = localtime(&now);
char *b = s_day_buffer;
APP_LOG(APP_LOG_LEVEL_DEBUG, "blueteething: %d", bluetooth);
if (bluetooth) {
if (bt_connected) {
text_layer_set_text(s_bt_label, "");
} else {
text_layer_set_text(s_bt_label, "");
}
text_layer_set_text_color(s_bt_label, colors[KEY_COLOR_NUM]);
}
strftime(s_mon_buffer, sizeof(s_mon_buffer), "%b", t);
text_layer_set_text(s_mon_label, s_mon_buffer);
text_layer_set_text_color(s_mon_label, colors[KEY_COLOR_MON]);
strftime(s_day_buffer, sizeof(s_day_buffer), "%d", t);
if (b[0] == '0') {
b += 1;
}
text_layer_set_text(s_day_label, b);
text_layer_set_text_color(s_day_label, colors[KEY_COLOR_DAY]);
}
static void handle_second_tick(struct tm *tick_time, TimeUnits units_changed) {
layer_mark_dirty(window_get_root_layer(window));
}
#define NUM_HEIGHT 50
#define NUM_WIDTH 54
#define BASELINE_OFFSET -7
#define NUM_FONT RESOURCE_ID_FONT_48
static void window_load(Window *window) {
Layer *window_layer = window_get_root_layer(window);
GRect bounds = layer_get_bounds(window_layer);
int xmax = bounds.size.w;
int ymax = bounds.size.h;
s_simple_bg_layer = layer_create(bounds);
layer_set_update_proc(s_simple_bg_layer, bg_update_proc);
layer_add_child(window_layer, s_simple_bg_layer);
for (int i = 0; i < 4; i += 1) {
int x, y;
GTextAlignment align;
switch (i) {
case 0:
x = (xmax - NUM_WIDTH) - 2; y = (ymax - NUM_HEIGHT) / 2 + BASELINE_OFFSET;
align = GTextAlignmentRight;
break;
case 1:
x = (xmax - NUM_WIDTH) / 2; y = ymax + BASELINE_OFFSET - NUM_HEIGHT;
align = GTextAlignmentCenter;
break;
case 2:
x = 2; y = (ymax - NUM_HEIGHT) / 2 + BASELINE_OFFSET;
align = GTextAlignmentLeft;
break;
case 3:
x = (xmax - NUM_WIDTH) / 2; y = BASELINE_OFFSET;
align = GTextAlignmentCenter;
break;
}
s_hour_label[i] = text_layer_create(GRect(x, y, NUM_WIDTH, NUM_HEIGHT));
text_layer_set_text_alignment(s_hour_label[i], align);
text_layer_set_background_color(s_hour_label[i], GColorClear);
text_layer_set_font(s_hour_label[i], fonts_load_custom_font(resource_get_handle(NUM_FONT)));
layer_add_child(s_simple_bg_layer, text_layer_get_layer(s_hour_label[i]));
}
s_date_layer = layer_create(bounds);
layer_set_update_proc(s_date_layer, date_update_proc);
layer_add_child(window_layer, s_date_layer);
// Missing phone
#ifdef PBL_RECT
s_bt_label = text_layer_create(GRect(20, 95, 52, 52));
#else
s_bt_label = text_layer_create(GRect(xmax/2 - 44, ymax/2 + 5, 52, 52));
#endif
text_layer_set_text_alignment(s_bt_label, GTextAlignmentLeft);
text_layer_set_text(s_bt_label, "");
text_layer_set_background_color(s_bt_label, GColorClear);
text_layer_set_font(s_bt_label, fonts_load_custom_font(resource_get_handle(RESOURCE_ID_SYMBOLS_52)));
layer_add_child(s_date_layer, text_layer_get_layer(s_bt_label));
// Month
#ifdef PBL_RECT
s_mon_label = text_layer_create(GRect(116, 145, 27, 18));
text_layer_set_font(s_mon_label, fonts_get_system_font(FONT_KEY_GOTHIC_18));
#else
s_mon_label = text_layer_create(GRect(124, 140, 27, 14));
text_layer_set_font(s_mon_label, fonts_get_system_font(FONT_KEY_GOTHIC_14));
#endif
text_layer_set_text_alignment(s_mon_label, GTextAlignmentRight);
text_layer_set_text(s_mon_label, s_mon_buffer);
text_layer_set_background_color(s_mon_label, GColorClear);
layer_add_child(s_date_layer, text_layer_get_layer(s_mon_label));
// Day
#ifdef PBL_RECT
s_day_label = text_layer_create(GRect(121, 122, 20, 24));
text_layer_set_font(s_day_label, fonts_get_system_font(FONT_KEY_GOTHIC_24_BOLD));
#else
s_day_label = text_layer_create(GRect(135, 124, 20, 18));
text_layer_set_font(s_day_label, fonts_get_system_font(FONT_KEY_GOTHIC_18_BOLD));
#endif
text_layer_set_text_alignment(s_day_label, GTextAlignmentRight);
text_layer_set_text(s_day_label, s_day_buffer);
text_layer_set_background_color(s_day_label, GColorClear);
layer_add_child(s_date_layer, text_layer_get_layer(s_day_label));
s_hands_layer = layer_create(bounds);
layer_set_update_proc(s_hands_layer, hands_update_proc);
layer_add_child(window_layer, s_hands_layer);
}
static void window_unload(Window *window) {
layer_destroy(s_simple_bg_layer);
layer_destroy(s_date_layer);
text_layer_destroy(s_day_label);
text_layer_destroy(s_mon_label);
layer_destroy(s_hands_layer);
}
static void bt_handler(bool connected) {
bt_connected = connected;
if (bluetooth && (! connected)) {
vibes_long_pulse();
}
layer_mark_dirty(s_date_layer);
}
static GColor color_of_int(int color) {
#ifdef PBL_COLOR
return GColorFromHEX(color);
#else
// XXX: This is surely going to turn out to be naïve
if (color & 0x808080) {
return GColorWhite;
} else {
return GColorBlack;
}
#endif
}
static void tick_subscribe() {
if (seconds) {
tick_timer_service_subscribe(SECOND_UNIT, handle_second_tick);
} else {
tick_timer_service_subscribe(MINUTE_UNIT, handle_second_tick);
}
}
static void restore(void) {
int i;
for (i = 0; i < KEY_LAST; i += 1) {
if (! persist_exists(i)) {
continue;
}
switch(i) {
case KEY_COLOR_FACE:
case KEY_COLOR_TIC:
case KEY_COLOR_NUM:
case KEY_COLOR_DAY:
case KEY_COLOR_MON:
case KEY_COLOR_SEC:
case KEY_COLOR_MIN:
case KEY_COLOR_HR:
colors[i] = color_of_int(persist_read_int(i));
break;
case KEY_SECONDS:
seconds = persist_read_bool(i);
tick_subscribe();
case KEY_BLUETOOTH:
bluetooth = persist_read_bool(i);
break;
}
}
}
static void in_received_handler(DictionaryIterator *rec, void *context) {
int i;
for (i = 0; i < KEY_LAST; i += 1) {
Tuple *cur = dict_find(rec, i);
if (! cur) {
APP_LOG(APP_LOG_LEVEL_DEBUG, "Holy crap! Key %i isn't around!", i);
continue;
}
switch (i) {
case KEY_COLOR_FACE:
case KEY_COLOR_TIC:
case KEY_COLOR_NUM:
case KEY_COLOR_DAY:
case KEY_COLOR_MON:
case KEY_COLOR_SEC:
case KEY_COLOR_MIN:
case KEY_COLOR_HR:
persist_write_int(i, cur->value->int32);
break;
case KEY_SECONDS:
persist_write_bool(i, !!cur->value->int32);
break;
case KEY_BLUETOOTH:
persist_write_bool(i, !!cur->value->int32);
break;
}
}
restore();
}
static void in_dropped_handler(AppMessageResult reason, void *context) {
// XXX: I don't understand what we could possibly do here
}
static void init() {
int i;
for (i = 0; i < KEY_LAST; i += 1) {
colors[i] = GColorWhite;
}
colors[KEY_COLOR_FACE] = GColorBlack;
colors[KEY_COLOR_SEC] = COLOR_FALLBACK(GColorWindsorTan, GColorWhite);
restore();
window = window_create();
window_set_window_handlers(window, (WindowHandlers) {
.load = window_load,
.unload = window_unload,
});
window_stack_push(window, true);
s_day_buffer[0] = '\0';
s_mon_buffer[0] = '\0';
// init paths
s_tic_path = gpath_create(&TIC_POINTS);
s_second_arrow = gpath_create(&SECOND_HAND_POINTS);
s_minute_arrow = gpath_create(&MINUTE_HAND_POINTS);
s_hour_arrow = gpath_create(&HOUR_HAND_POINTS);
Layer *window_layer = window_get_root_layer(window);
GRect bounds = layer_get_bounds(window_layer);
GPoint center = grect_center_point(&bounds);
gpath_move_to(s_tic_path, center);
gpath_move_to(s_second_arrow, center);
gpath_move_to(s_minute_arrow, center);
gpath_move_to(s_hour_arrow, center);
tick_subscribe();
#ifdef PBL_ROUND
bt_connected = true;
#else
bluetooth_connection_service_subscribe(bt_handler);
bt_connected = bluetooth_connection_service_peek();
#endif
app_message_register_inbox_received(in_received_handler);
app_message_register_inbox_dropped(in_dropped_handler);
app_message_open(256, 64);
}
static void deinit() {
gpath_destroy(s_tic_path);
gpath_destroy(s_second_arrow);
gpath_destroy(s_minute_arrow);
gpath_destroy(s_hour_arrow);
// XXX: text destroy?
tick_timer_service_unsubscribe();
window_destroy(window);
}
int main() {
init();
app_event_loop();
deinit();
}