/** * \file DropDown.C * Copyright 2001 the LyX Team * Read the file COPYING * * \author John Levon */ #include #include "DropDown.h" #include extern "C" void C_DropDownCompletedCB(FL_OBJECT * ob, long) { DropDown * d = static_cast(ob->form->u_vdata); d->completed(); } extern "C" int C_DropDownPeekEventCB(FL_FORM * form, void *xev) { DropDown * d = static_cast(form->u_vdata); return d->peek(static_cast(xev)); } DropDown::DropDown(LyXView * lv, FL_OBJECT * ob) : lv_(lv) { form_ = fl_bgn_form(FL_NO_BOX, ob->w, 100); fl_add_box(FL_UP_BOX, 0, 0, ob->w, 100, ""); browser_ = fl_add_browser(FL_SELECT_BROWSER, 0, 0, ob->w, 100, ""); form_->u_vdata = this; fl_set_browser_dblclick_callback(browser_, C_DropDownCompletedCB, 0); fl_register_raw_callback(form_, KeyPressMask|ButtonPressMask, C_DropDownPeekEventCB); fl_end_form(); } DropDown::~DropDown() { if (form_->visible) fl_hide_form(form_); fl_free_form(form_); } void DropDown::select(std::vector const & choices, int x, int y, int w) { fl_set_form_geometry(form_, x, y, w, 100); fl_clear_browser(browser_); for (std::vector::const_iterator cit = choices.begin(); cit != choices.end(); ++cit) { fl_add_browser_line(browser_, cit->c_str()); } fl_show_form(form_, FL_PLACE_POSITION, FL_NOBORDER, ""); XGrabPointer(fl_get_display(), form_->window, false, ButtonPressMask | ButtonReleaseMask | PointerMotionMask, GrabModeAsync, GrabModeAsync, 0, 0, 0); XFlush(fl_get_display()); } void DropDown::line_up() { if (fl_get_browser(browser_) > 1) fl_select_browser_line(browser_, fl_get_browser(browser_) - 1); if (fl_get_browser(browser_) >= fl_get_browser_topline(browser_) + fl_get_browser_screenlines(browser_)) fl_set_browser_topline(browser_, fl_get_browser(browser_) - fl_get_browser_screenlines(browser_) + 1); if (fl_get_browser(browser_) < fl_get_browser_topline(browser_)) fl_set_browser_topline(browser_, fl_get_browser(browser_)); } void DropDown::line_down() { if (fl_get_browser(browser_) < fl_get_browser_maxline(browser_)) fl_select_browser_line(browser_, fl_get_browser(browser_) + 1); if (fl_get_browser(browser_) >= fl_get_browser_topline(browser_) + fl_get_browser_screenlines(browser_)) fl_set_browser_topline(browser_, fl_get_browser(browser_) - fl_get_browser_screenlines(browser_) + 1); if (fl_get_browser(browser_) < fl_get_browser_topline(browser_)) fl_set_browser_topline(browser_, fl_get_browser(browser_)); } int DropDown::peek(XEvent * xev) { int x,y; unsigned int keymask; fl_get_mouse(&x, &y, &keymask); if (xev->type == ButtonPress) { if (!(x >= form_->x && x <= (form_->x + form_->w) && y >= form_->y && y << (form_->y + form_->h))) { fl_hide_form(form_); return 1; } XUngrabPointer(fl_get_display(), CurrentTime); } else if (xev->type == KeyPress) { char s_r[10]; s_r[9] = '\0'; KeySym keysym_return; XLookupString(&xev->xkey, s_r, 10, &keysym_return, 0); switch (keysym_return) { case XK_Down: line_down(); return 1; case XK_Up: line_up(); return 1; case XK_Return: completed(); return 1; case XK_Escape: fl_select_browser_line(browser_, 0); completed(); return 1; default: // FIXME: if someone has a got a way to // convince the event to fall back to the // minibuffer, I'm glad to hear it. // fl_XPutBackEvent() doesn't work. fl_select_browser_line(browser_, 0); completed(); return 1; } } return 0; } void DropDown::completed() { XUngrabPointer(fl_get_display(), CurrentTime); string selection; int i = fl_get_browser(browser_); if (i < 1) selection = ""; else selection = fl_get_browser_line(browser_, i); fl_hide_form(form_); result.emit(selection); }