/* $Id: mousebutton.c,v 1.12 2005/09/15 15:03:17 cegger Exp $ ****************************************************************************** This LibGIC module is a very simple parser for PtrButtonPress Events. Copyright (C) 1999 Andrew Apted [andrew@ggi-project.org] Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ****************************************************************************** */ #include #include #include #include #include #include #include /* for snprintf() */ typedef struct mousebutton_data { int button; } MouseButtonData; static int mbutton_check(gic_handle_t hand, gic_recognizer *ctrl,gii_event *event, gic_feature *feature,int recnum); static int mbutton_get_name(gic_handle_t hand, gic_recognizer *ctrl, char *string,size_t maxlen); static int mbutton_write_pvtdata(gic_handle_t hand, gic_recognizer *ctrl, char *string,int maxlen); static int mbutton_read_pvtdata(gic_handle_t hand, gic_recognizer *ctrl, const char *string); static void mbutton_free_pvtdata(gic_handle_t hand, gic_recognizer *ctrl); static int mbutton_train(gic_handle_t hand, gic_recognizer **ctrl, gii_event *event); static int mbutton_check_conflict(gic_handle_t hand, gic_recognizer *ctrl, gic_recognizer *ctrl2); static int mbutton_get_opposite(gic_handle_t hand, gic_recognizer *recognizer, gic_recognizer **opposite); static gic_recognizerdriver mycontrols = { "MouseButton", mbutton_check, mbutton_get_name, mbutton_write_pvtdata, mbutton_read_pvtdata, mbutton_free_pvtdata, mbutton_train, mbutton_check_conflict, mbutton_get_opposite }; static int getbutton(gic_handle_t hand, gic_recognizer *ctrl, gii_event *event,gic_feature *feature,int recnum) { MouseButtonData *but = ctrl->privdata; DPRINT_LIBS("MouseButton: Button%s %d [want %d].\n", (event->any.type == evPtrButtonPress) ? "Press" : "Release", event->pbutton.button, but->button); if (event->pbutton.button != (unsigned)but->button) { return 0; } gicFeatureActivate(hand, feature,(event->any.type == evPtrButtonPress) ? GIC_STATE_MAX : GIC_STATE_MIN , 0, recnum); return 1; } static int mbutton_check(gic_handle_t hand, gic_recognizer *ctrl,gii_event *event, gic_feature *feature,int recnum) { DPRINT_LIBS("MouseButton: Check with %p,%p.\n",ctrl,event); if ((event->any.type == evPtrButtonPress) || (event->any.type == evPtrButtonRelease)) { return getbutton(hand, ctrl, event, feature, recnum); } return 0; } static int mbutton_register(gic_handle_t hand, gic_recognizer **ctrl,MouseButtonData *but,gic_state state) { gic_recognizer *new_rec, *rl; MouseButtonData *new_but; for (rl=*ctrl; rl != NULL; rl=rl->next) { MouseButtonData *cur = (MouseButtonData *) rl->privdata; if (cur->button == but->button) { /* Duplicate entry -- just update the confidence * value. */ if (state > rl->confidence) { rl->confidence = state; /* !!! MOVE UP */ } return 1; } } new_rec = malloc(sizeof(gic_recognizer)); if (new_rec == NULL) return GGI_ENOMEM; new_but = malloc(sizeof(MouseButtonData)); if (new_but == NULL) { free(new_rec); return GGI_ENOMEM; } memcpy(new_but, but, sizeof(MouseButtonData)); new_rec->driver = &mycontrols; new_rec->confidence = state; new_rec->privdata = new_but; gicRecognizerTrainAdd(hand, ctrl, new_rec); return 1; } static struct trainingstate { int got_button; /* -1 when no current button */ } trainingstate; static int mbutton_train(gic_handle_t hand, gic_recognizer **ctrl, gii_event *event) { int rc; DPRINT_LIBS("MouseButton: Training with %p,%p.\n",ctrl,event); rc=0; if (event == NULL) { trainingstate.got_button = -1; DPRINT_LIBS("MouseButton: Initialized training state.\n"); return 0; } DPRINT_LIBS("MouseButton: Analyzing event ...\n"); if (event->any.type == evPtrButtonPress) { trainingstate.got_button = event->pbutton.button; DPRINT_LIBS("MouseButton: Remembered button %d ...\n", trainingstate.got_button); return 0; } if ((event->any.type == evPtrButtonRelease) && (event->pbutton.button = trainingstate.got_button)) { MouseButtonData but; but.button = trainingstate.got_button; rc+=mbutton_register(hand, ctrl, &but, GIC_STATE_MAX); DPRINT_LIBS("MouseButton: Registered button %d ...\n", trainingstate.got_button); return rc; } return rc; } static int mbutton_write_pvtdata(gic_handle_t hand, gic_recognizer *ctrl, char *string,int maxlen) { MouseButtonData *but = ctrl->privdata; if (maxlen < 20) { *string='\0'; return GGI_ENOSPACE; } sprintf(string, "%d", but->button); return 0; } static int mbutton_read_pvtdata(gic_handle_t hand, gic_recognizer *ctrl, const char *string) { MouseButtonData *but; but = ctrl->privdata = malloc(sizeof(MouseButtonData)); if (but == NULL) { return GGI_ENOMEM; } sscanf(string, "%i", &but->button); return 0; } static void mbutton_free_pvtdata(gic_handle_t hand, gic_recognizer *ctrl) { MouseButtonData *but = ctrl->privdata; free(but); ctrl->privdata = NULL; return; } static int mbutton_get_name(gic_handle_t hand, gic_recognizer *ctrl, char *string,size_t maxlen) { MouseButtonData *but = ctrl->privdata; char namebuf[40]; if (maxlen>20) maxlen=20; if (maxlen<=1) { /* The user must be Joking ... */ } else if (maxlen<=5) { snprintf(namebuf, 40, "M%.*s%d", (int)(maxlen-2), "But",but->button); } else if (maxlen<=9) { snprintf(namebuf, 40, "%.*sBut%d", (int)(maxlen-4), "Mouse", but->button); } else { snprintf(namebuf, 40, "Mouse%.*s%d", (int)(maxlen-6), "Button", but->button); } ggstrlcpy(string, namebuf, maxlen); return 0; } static int mbutton_check_conflict(gic_handle_t hand, gic_recognizer *ctrl, gic_recognizer *ctrl2) { MouseButtonData *but1 = ctrl ->privdata; MouseButtonData *but2 = ctrl2->privdata; if (ctrl == ctrl2) { return GIC_C_ISSAME; } if (ctrl->driver != ctrl2->driver) { return GIC_C_NOCONFLICT; } if (but1->button == but2->button) { return GIC_C_ISSAME; } return GIC_C_NOCONFLICT; } static int mbutton_get_opposite(gic_handle_t hand, gic_recognizer *recognizer, gic_recognizer **opposite) { return GGI_ENOMATCH; /* FIXME */ } EXPORTFUNC gic_recognizerdriver *GICdl_mousebutton(void); gic_recognizerdriver *GICdl_mousebutton(void) { return &mycontrols; }