rockbox/apps/plugins/stopwatch.c

316 lines
9.5 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2004 Mike Holden
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "plugin.h"
PLUGIN_HEADER
#ifdef HAVE_LCD_BITMAP
#define TIMER_Y 1
#else
#define TIMER_Y 0
#endif
#define LAP_Y TIMER_Y+1
#define MAX_LAPS 64
/* variable button definitions */
#if CONFIG_KEYPAD == RECORDER_PAD
#define STOPWATCH_QUIT BUTTON_OFF
#define STOPWATCH_START_STOP BUTTON_PLAY
#define STOPWATCH_RESET_TIMER BUTTON_LEFT
#define STOPWATCH_LAP_TIMER BUTTON_ON
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == ARCHOS_AV300_PAD
#define STOPWATCH_QUIT BUTTON_OFF
#define STOPWATCH_START_STOP BUTTON_SELECT
#define STOPWATCH_RESET_TIMER BUTTON_LEFT
#define STOPWATCH_LAP_TIMER BUTTON_ON
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == ONDIO_PAD
#define STOPWATCH_QUIT BUTTON_OFF
#define STOPWATCH_START_STOP BUTTON_RIGHT
#define STOPWATCH_RESET_TIMER BUTTON_LEFT
#define STOPWATCH_LAP_TIMER BUTTON_MENU
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == PLAYER_PAD
#define STOPWATCH_QUIT BUTTON_MENU
#define STOPWATCH_START_STOP BUTTON_PLAY
#define STOPWATCH_RESET_TIMER BUTTON_STOP
#define STOPWATCH_LAP_TIMER BUTTON_ON
#define STOPWATCH_SCROLL_UP BUTTON_RIGHT
#define STOPWATCH_SCROLL_DOWN BUTTON_LEFT
#elif (CONFIG_KEYPAD == IRIVER_H100_PAD) || \
(CONFIG_KEYPAD == IRIVER_H300_PAD)
#define STOPWATCH_QUIT BUTTON_OFF
#define STOPWATCH_START_STOP BUTTON_SELECT
#define STOPWATCH_RESET_TIMER BUTTON_DOWN
#define STOPWATCH_LAP_TIMER BUTTON_ON
#define STOPWATCH_SCROLL_UP BUTTON_RIGHT
#define STOPWATCH_SCROLL_DOWN BUTTON_LEFT
#define STOPWATCH_RC_QUIT BUTTON_RC_STOP
#elif (CONFIG_KEYPAD == IPOD_4G_PAD) || \
(CONFIG_KEYPAD == IPOD_3G_PAD) || \
(CONFIG_KEYPAD == IPOD_1G2G_PAD)
#define STOPWATCH_QUIT BUTTON_MENU
#define STOPWATCH_START_STOP BUTTON_SELECT
#define STOPWATCH_RESET_TIMER BUTTON_LEFT
#define STOPWATCH_LAP_TIMER BUTTON_RIGHT
#define STOPWATCH_SCROLL_UP BUTTON_SCROLL_FWD
#define STOPWATCH_SCROLL_DOWN BUTTON_SCROLL_BACK
#elif CONFIG_KEYPAD == IRIVER_IFP7XX_PAD
#define STOPWATCH_QUIT BUTTON_PLAY
#define STOPWATCH_START_STOP BUTTON_MODE
#define STOPWATCH_RESET_TIMER BUTTON_EQ
#define STOPWATCH_LAP_TIMER BUTTON_SELECT
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == IAUDIO_X5M5_PAD
#define STOPWATCH_QUIT BUTTON_POWER
#define STOPWATCH_START_STOP BUTTON_PLAY
#define STOPWATCH_RESET_TIMER BUTTON_REC
#define STOPWATCH_LAP_TIMER BUTTON_SELECT
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == GIGABEAT_PAD
#define STOPWATCH_QUIT BUTTON_POWER
#define STOPWATCH_START_STOP BUTTON_SELECT
#define STOPWATCH_RESET_TIMER BUTTON_A
#define STOPWATCH_LAP_TIMER BUTTON_MENU
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif (CONFIG_KEYPAD == SANSA_E200_PAD) || \
(CONFIG_KEYPAD == SANSA_C200_PAD)
#define STOPWATCH_QUIT BUTTON_POWER
#define STOPWATCH_START_STOP BUTTON_RIGHT
#define STOPWATCH_RESET_TIMER BUTTON_LEFT
#define STOPWATCH_LAP_TIMER BUTTON_SELECT
#define STOPWATCH_SCROLL_UP BUTTON_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_DOWN
#elif CONFIG_KEYPAD == IRIVER_H10_PAD
#define STOPWATCH_QUIT BUTTON_POWER
#define STOPWATCH_START_STOP BUTTON_PLAY
#define STOPWATCH_RESET_TIMER BUTTON_REW
#define STOPWATCH_LAP_TIMER BUTTON_FF
#define STOPWATCH_SCROLL_UP BUTTON_SCROLL_UP
#define STOPWATCH_SCROLL_DOWN BUTTON_SCROLL_DOWN
#endif
static struct plugin_api* rb;
static int stopwatch = 0;
static long start_at = 0;
static int prev_total = 0;
static bool counting = false;
static int curr_lap = 0;
static int lap_scroll = 0;
static int lap_start;
static int lap_times[MAX_LAPS];
static void ticks_to_string(int ticks,int lap,int buflen, char * buf)
{
int hours, minutes, seconds, cs;
hours = ticks / (HZ * 3600);
ticks -= (HZ * hours * 3600);
minutes = ticks / (HZ * 60);
ticks -= (HZ * minutes * 60);
seconds = ticks / HZ;
ticks -= (HZ * seconds);
cs = ticks;
if (!lap)
{
rb->snprintf(buf, buflen,
"%2d:%02d:%02d.%02d",
hours, minutes, seconds, cs);
}
else
{
if (lap > 1)
{
int last_ticks, last_hours, last_minutes, last_seconds, last_cs;
last_ticks = lap_times[(lap-1)%MAX_LAPS] - lap_times[(lap-2)%MAX_LAPS];
last_hours = last_ticks / (HZ * 3600);
last_ticks -= (HZ * last_hours * 3600);
last_minutes = last_ticks / (HZ * 60);
last_ticks -= (HZ * last_minutes * 60);
last_seconds = last_ticks / HZ;
last_ticks -= (HZ * last_seconds);
last_cs = last_ticks;
rb->snprintf(buf, buflen,
"%2d %2d:%02d:%02d.%02d [%2d:%02d:%02d.%02d]",
lap, hours, minutes, seconds, cs, last_hours,
last_minutes, last_seconds, last_cs);
}
else
{
rb->snprintf(buf, buflen,
"%2d %2d:%02d:%02d.%02d",
lap, hours, minutes, seconds, cs);
}
}
}
enum plugin_status plugin_start(struct plugin_api* api, void* parameter)
{
char buf[32];
int button;
int lap;
int done = false;
bool update_lap = true;
int lines;
(void)parameter;
rb = api;
#ifdef HAVE_LCD_BITMAP
int h;
rb->lcd_setfont(FONT_UI);
rb->lcd_getstringsize("M", NULL, &h);
lines = (LCD_HEIGHT / h) - (LAP_Y);
#else
lines = 1;
#endif
rb->lcd_clear_display();
while (!done)
{
if (counting)
{
stopwatch = prev_total + *rb->current_tick - start_at;
}
else
{
stopwatch = prev_total;
}
ticks_to_string(stopwatch,0,32,buf);
rb->lcd_puts(0, TIMER_Y, buf);
if(update_lap)
{
lap_start = curr_lap - lap_scroll;
for (lap = lap_start; lap > lap_start - lines; lap--)
{
if (lap > 0)
{
ticks_to_string(lap_times[(lap-1)%MAX_LAPS],lap,32,buf);
rb->lcd_puts_scroll(0, LAP_Y + lap_start - lap, buf);
}
else
{
rb->lcd_puts(0, LAP_Y + lap_start - lap,
" ");
}
}
update_lap = false;
}
rb->lcd_update();
if (! counting)
{
button = rb->button_get(true);
}
else
{
button = rb->button_get_w_tmo(10);
/* Make sure that the jukebox isn't powered off
automatically */
rb->reset_poweroff_timer();
}
switch (button)
{
/* exit */
#ifdef STOPWATCH_RC_QUIT
case STOPWATCH_RC_QUIT:
#endif
case STOPWATCH_QUIT:
done = true;
break;
/* Stop/Start toggle */
case STOPWATCH_START_STOP:
counting = ! counting;
if (counting)
{
start_at = *rb->current_tick;
stopwatch = prev_total + *rb->current_tick - start_at;
}
else
{
prev_total += *rb->current_tick - start_at;
stopwatch = prev_total;
}
break;
/* Reset timer */
case STOPWATCH_RESET_TIMER:
if (!counting)
{
prev_total = 0;
curr_lap = 0;
update_lap = true;
}
break;
/* Lap timer */
case STOPWATCH_LAP_TIMER:
lap_times[curr_lap%MAX_LAPS] = stopwatch;
curr_lap++;
update_lap = true;
break;
/* Scroll Lap timer up */
case STOPWATCH_SCROLL_UP:
if (lap_scroll > 0)
{
lap_scroll --;
update_lap = true;
}
break;
/* Scroll Lap timer down */
case STOPWATCH_SCROLL_DOWN:
if ((lap_scroll < curr_lap - lines) &&
(lap_scroll < (MAX_LAPS - lines)) )
{
lap_scroll ++;
update_lap = true;
}
break;
default:
if (rb->default_event_handler(button) == SYS_USB_CONNECTED)
return PLUGIN_USB_CONNECTED;
break;
}
}
return PLUGIN_OK;
}