2013-07-05 15:46:34 +00:00
|
|
|
/*
|
|
|
|
* This file is subject to the terms of the GFX License. If a copy of
|
|
|
|
* the license was not distributed with this file, you can obtain one at:
|
|
|
|
*
|
2013-07-21 20:20:37 +00:00
|
|
|
* http://ugfx.org/license.html
|
2013-07-05 15:46:34 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @file src/gwin/radio.c
|
|
|
|
* @brief GWIN sub-system radio button code.
|
|
|
|
*
|
|
|
|
* @defgroup RadioButton RadioButton
|
|
|
|
* @ingroup GWIN
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "gfx.h"
|
|
|
|
|
|
|
|
#if GFX_USE_GWIN && GWIN_NEED_RADIO
|
|
|
|
|
2014-02-18 14:36:52 +00:00
|
|
|
#include "src/gwin/class_gwin.h"
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
// Our pressed state
|
|
|
|
#define GRADIO_FLG_PRESSED (GWIN_FIRST_CONTROL_FLAG<<0)
|
|
|
|
|
|
|
|
// Send the button event
|
2013-07-07 09:40:37 +00:00
|
|
|
static void SendRadioEvent(GWidgetObject *gw) {
|
2013-07-05 15:46:34 +00:00
|
|
|
GSourceListener * psl;
|
|
|
|
GEvent * pe;
|
|
|
|
#define pbe ((GEventGWinRadio *)pe)
|
|
|
|
|
|
|
|
// Trigger a GWIN Button Event
|
|
|
|
psl = 0;
|
|
|
|
while ((psl = geventGetSourceListener(GWIDGET_SOURCE, psl))) {
|
|
|
|
if (!(pe = geventGetEventBuffer(psl)))
|
|
|
|
continue;
|
|
|
|
pbe->type = GEVENT_GWIN_RADIO;
|
|
|
|
pbe->radio = (GHandle)gw;
|
|
|
|
pbe->group = ((GRadioObject *)gw)->group;
|
|
|
|
geventSendEvent(psl);
|
|
|
|
}
|
|
|
|
|
|
|
|
#undef pbe
|
|
|
|
}
|
|
|
|
|
|
|
|
#if GINPUT_NEED_MOUSE
|
|
|
|
// A mouse down has occurred over the button
|
|
|
|
static void MouseDown(GWidgetObject *gw, coord_t x, coord_t y) {
|
|
|
|
(void) x; (void) y;
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
gwinRadioPress((GHandle)gw);
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if GINPUT_NEED_TOGGLE
|
|
|
|
// A toggle on has occurred
|
|
|
|
static void ToggleOn(GWidgetObject *gw, uint16_t role) {
|
|
|
|
(void) role;
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
gwinRadioPress((GHandle)gw);
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void ToggleAssign(GWidgetObject *gw, uint16_t role, uint16_t instance) {
|
|
|
|
(void) role;
|
|
|
|
((GRadioObject *)gw)->toggle = instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint16_t ToggleGet(GWidgetObject *gw, uint16_t role) {
|
|
|
|
(void) role;
|
|
|
|
return ((GRadioObject *)gw)->toggle;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// The radio button VMT table
|
|
|
|
static const gwidgetVMT radioVMT = {
|
|
|
|
{
|
|
|
|
"Radio", // The classname
|
|
|
|
sizeof(GRadioObject), // The object size
|
|
|
|
_gwidgetDestroy, // The destroy routine
|
|
|
|
_gwidgetRedraw, // The redraw routine
|
|
|
|
0, // The after-clear routine
|
|
|
|
},
|
|
|
|
gwinRadioDraw_Radio, // The default drawing routine
|
|
|
|
#if GINPUT_NEED_MOUSE
|
|
|
|
{
|
|
|
|
MouseDown, // Process mouse down events
|
|
|
|
0, // Process mouse up events (NOT USED)
|
|
|
|
0, // Process mouse move events (NOT USED)
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
#if GINPUT_NEED_TOGGLE
|
|
|
|
{
|
|
|
|
1, // 1 toggle role
|
|
|
|
ToggleAssign, // Assign Toggles
|
|
|
|
ToggleGet, // Get Toggles
|
|
|
|
0, // Process toggle off events (NOT USED)
|
|
|
|
ToggleOn, // Process toggle on events
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
#if GINPUT_NEED_DIAL
|
|
|
|
{
|
|
|
|
0, // No dial roles
|
|
|
|
0, // Assign Dials (NOT USED)
|
|
|
|
0, // Get Dials (NOT USED)
|
|
|
|
0, // Process dial move events (NOT USED)
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2013-10-24 08:36:11 +00:00
|
|
|
GHandle gwinGRadioCreate(GDisplay *g, GRadioObject *gw, const GWidgetInit *pInit, uint16_t group) {
|
|
|
|
if (!(gw = (GRadioObject *)_gwidgetCreate(g, &gw->w, pInit, &radioVMT)))
|
2013-07-05 15:46:34 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
#if GINPUT_NEED_TOGGLE
|
|
|
|
gw->toggle = GWIDGET_NO_INSTANCE;
|
|
|
|
#endif
|
|
|
|
gw->group = group;
|
|
|
|
gwinSetVisible((GHandle)gw, pInit->g.show);
|
|
|
|
return (GHandle)gw;
|
|
|
|
}
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
void gwinRadioPress(GHandle gh) {
|
2013-07-05 15:46:34 +00:00
|
|
|
GHandle gx;
|
|
|
|
|
|
|
|
if (gh->vmt != (gwinVMT *)&radioVMT || (gh->flags & GRADIO_FLG_PRESSED))
|
|
|
|
return;
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
if ((gx = gwinRadioGetActive(((GRadioObject *)gh)->group))) {
|
2013-07-05 15:46:34 +00:00
|
|
|
gx->flags &= ~GRADIO_FLG_PRESSED;
|
|
|
|
_gwidgetRedraw(gx);
|
|
|
|
}
|
|
|
|
gh->flags |= GRADIO_FLG_PRESSED;
|
|
|
|
_gwidgetRedraw(gh);
|
2013-07-07 09:40:37 +00:00
|
|
|
SendRadioEvent((GWidgetObject *)gh);
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
bool_t gwinRadioIsPressed(GHandle gh) {
|
2013-07-05 15:46:34 +00:00
|
|
|
if (gh->vmt != (gwinVMT *)&radioVMT)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return (gh->flags & GRADIO_FLG_PRESSED) ? TRUE : FALSE;
|
|
|
|
}
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
GHandle gwinRadioGetActive(uint16_t group) {
|
2013-07-05 15:46:34 +00:00
|
|
|
const gfxQueueASyncItem * qi;
|
|
|
|
GHandle gh;
|
|
|
|
|
|
|
|
for(qi = gfxQueueASyncPeek(&_GWINList); qi; qi = gfxQueueASyncNext(qi)) {
|
|
|
|
gh = QItem2GWindow(qi);
|
|
|
|
if (gh->vmt == (gwinVMT *)&radioVMT && ((GRadioObject *)gh)->group == group && (gh->flags & GRADIO_FLG_PRESSED))
|
|
|
|
return gh;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*----------------------------------------------------------
|
|
|
|
* Custom Draw Routines
|
|
|
|
*----------------------------------------------------------*/
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
static const GColorSet *getDrawColors(GWidgetObject *gw) {
|
|
|
|
if (!(gw->g.flags & GWIN_FLG_ENABLED)) return &gw->pstyle->disabled;
|
|
|
|
if ((gw->g.flags & GRADIO_FLG_PRESSED)) return &gw->pstyle->pressed;
|
|
|
|
return &gw->pstyle->enabled;
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void gwinRadioDraw_Radio(GWidgetObject *gw, void *param) {
|
2013-07-07 09:40:37 +00:00
|
|
|
#define gcw ((GRadioObject *)gw)
|
|
|
|
coord_t ld, df;
|
|
|
|
const GColorSet * pcol;
|
|
|
|
(void) param;
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
if (gw->g.vmt != (gwinVMT *)&radioVMT) return;
|
|
|
|
pcol = getDrawColors(gw);
|
|
|
|
|
|
|
|
ld = gw->g.width < gw->g.height ? gw->g.width : gw->g.height;
|
|
|
|
|
|
|
|
#if GDISP_NEED_CIRCLE
|
2013-07-07 09:40:37 +00:00
|
|
|
df = (ld-1)/2;
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillArea(gw->g.display, gw->g.x, gw->g.y, ld, ld, gw->pstyle->background);
|
|
|
|
gdispGDrawCircle(gw->g.display, gw->g.x+df, gw->g.y+df, df, pcol->edge);
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
if (gw->g.flags & GRADIO_FLG_PRESSED)
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillCircle(gw->g.display, gw->g.x+df, gw->g.y+df, df <= 2 ? 1 : (df-2), pcol->fill);
|
2013-07-05 15:46:34 +00:00
|
|
|
#else
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillArea(gw->g.display, gw->g.x+1, gw->g.y+1, ld, ld-2, gw->pstyle->background);
|
|
|
|
gdispGDrawBox(gw->g.display, gw->g.x, gw->g.y, ld, ld, pcol->edge);
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
df = ld < 4 ? 1 : 2;
|
|
|
|
if (gw->g.flags & GRADIO_FLG_PRESSED)
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillArea(gw->g.display, gw->g.x+df, gw->g.y+df, ld-2*df, ld-2*df, pcol->fill);
|
2013-07-05 15:46:34 +00:00
|
|
|
#endif
|
|
|
|
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillStringBox(gw->g.display, gw->g.x+ld+1, gw->g.y, gw->g.width-ld-1, gw->g.height, gw->text, gw->g.font, pcol->text, gw->pstyle->background, justifyLeft);
|
2013-07-05 15:46:34 +00:00
|
|
|
#undef gcw
|
|
|
|
}
|
|
|
|
|
|
|
|
void gwinRadioDraw_Button(GWidgetObject *gw, void *param) {
|
2013-07-07 09:40:37 +00:00
|
|
|
const GColorSet * pcol;
|
|
|
|
(void) param;
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
if (gw->g.vmt != (gwinVMT *)&radioVMT) return;
|
|
|
|
pcol = getDrawColors(gw);
|
|
|
|
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillStringBox(gw->g.display, gw->g.x, gw->g.y, gw->g.width-1, gw->g.height-1, gw->text, gw->g.font, pcol->text, pcol->fill, justifyCenter);
|
|
|
|
gdispGDrawLine(gw->g.display, gw->g.x+gw->g.width-1, gw->g.y, gw->g.x+gw->g.width-1, gw->g.y+gw->g.height-1, pcol->edge);
|
|
|
|
gdispGDrawLine(gw->g.display, gw->g.x, gw->g.y+gw->g.height-1, gw->g.x+gw->g.width-2, gw->g.y+gw->g.height-1, pcol->edge);
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void gwinRadioDraw_Tab(GWidgetObject *gw, void *param) {
|
2013-07-07 09:40:37 +00:00
|
|
|
const GColorSet * pcol;
|
|
|
|
(void) param;
|
2013-07-05 15:46:34 +00:00
|
|
|
|
|
|
|
if (gw->g.vmt != (gwinVMT *)&radioVMT) return;
|
|
|
|
pcol = getDrawColors(gw);
|
|
|
|
|
2013-07-07 09:40:37 +00:00
|
|
|
if ((gw->g.flags & GRADIO_FLG_PRESSED)) {
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGDrawBox(gw->g.display, gw->g.x, gw->g.y, gw->g.width, gw->g.height, pcol->edge);
|
|
|
|
gdispGFillStringBox(gw->g.display, gw->g.x+1, gw->g.y+1, gw->g.width-2, gw->g.height-1, gw->text, gw->g.font, pcol->text, pcol->fill, justifyCenter);
|
2013-07-07 09:40:37 +00:00
|
|
|
} else {
|
2013-10-24 08:36:11 +00:00
|
|
|
gdispGFillStringBox(gw->g.display, gw->g.x, gw->g.y, gw->g.width-1, gw->g.height-1, gw->text, gw->g.font, pcol->text, pcol->fill, justifyCenter);
|
|
|
|
gdispGDrawLine(gw->g.display, gw->g.x+gw->g.width-1, gw->g.y, gw->g.x+gw->g.width-1, gw->g.y+gw->g.height-1, pcol->edge);
|
|
|
|
gdispGDrawLine(gw->g.display, gw->g.x, gw->g.y+gw->g.height-1, gw->g.x+gw->g.width-2, gw->g.y+gw->g.height-1, pcol->edge);
|
2013-07-07 09:40:37 +00:00
|
|
|
}
|
2013-07-05 15:46:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* GFX_USE_GWIN && GWIN_NEED_BUTTON */
|
|
|
|
/** @} */
|
|
|
|
|