3 * This file is part of LyX, the document processor.
4 * Licence details can be found in the file COPYING.
6 * \author Angus Leeming
8 * Full author contact details are available in file CREDITS
11 /* Tooltips for xforms. xforms 0.89 supports them directly, but 0.88 needs
12 * a bit of jiggery pokery. This class wraps it all up in a neat interface.
13 * Based on code originally in Toolbar_pimpl.C that appears to have been
14 * written by Matthias Ettrich and Jean-Marc Lasgouttes.
20 #pragma implementation
24 #include "xforms_helpers.h" // formatted
26 #include "support/lstrings.h"
27 #include "support/LAssert.h"
28 #include FORMS_H_LOCATION
30 #include <boost/bind.hpp>
32 bool Tooltips::enabled_ = true;
34 boost::signal0<void> Tooltips::toggled;
37 #if FL_VERSION > 0 || FL_REVISION >= 89
41 toggled.connect(boost::bind(&Tooltips::set, this));
45 void Tooltips::toggleEnabled()
54 if (tooltipsMap.empty())
57 TooltipsMap::const_iterator it = tooltipsMap.begin();
58 TooltipsMap::const_iterator end = tooltipsMap.end();
59 for (; it != end; ++it) {
60 FL_OBJECT * const ob = it->first;
61 char const * const c_str = enabled_ ? it->second.c_str() : 0;
62 fl_set_object_helper(ob, c_str);
67 void Tooltips::init(FL_OBJECT * ob, string const & tip)
69 lyx::Assert(ob && ob->form);
71 // Store the tooltip string
72 string const str = formatted(trim(tip), 400);
73 tooltipsMap[ob] = str;
76 char const * const c_str = enabled_ ? str.c_str() : 0;
77 fl_set_object_helper(ob, c_str);
81 #else // if FL_REVISION < 89
85 int TooltipHandler(FL_OBJECT *ob, int event);
87 void TooltipTimerCB(FL_OBJECT * timer, long data);
93 static int C_TooltipHandler(FL_OBJECT * ob, int event,
94 FL_Coord, FL_Coord, int, void *)
96 return TooltipHandler(ob, event);
100 static void C_TooltipTimerCB(FL_OBJECT * ob, long data)
102 TooltipTimerCB(ob, data);
111 toggled.connect(boost::bind(&Tooltips::set, this));
115 void Tooltips::toggleEnabled()
117 enabled_ = !enabled_;
126 void Tooltips::init(FL_OBJECT * ob, string const & tip)
128 lyx::Assert(ob && ob->form);
131 TooltipsMap::const_iterator it = tooltipsMap.find(ob);
132 if (it != tooltipsMap.end())
135 string const str = trim(tip);
139 // Store the tooltip string
140 tooltipsMap[ob] = formatted(str, 400);
142 if (!tooltip_timer_) {
143 if (fl_current_form && ob->form != fl_current_form)
146 bool const open_form = !fl_current_form;
148 fl_addto_form(ob->form);
150 tooltip_timer_ = fl_add_timer(FL_HIDDEN_TIMER, 0, 0, 0, 0, "");
156 fl_set_object_posthandler(ob, C_TooltipHandler);
157 ob->u_cdata = reinterpret_cast<char *>(tooltip_timer_);
158 tooltip_timer_->u_vdata = this;
162 string const Tooltips::get(FL_OBJECT * ob) const
164 TooltipsMap::const_iterator it = tooltipsMap.find(ob);
165 if (it == tooltipsMap.end())
173 void TooltipTimerCB(FL_OBJECT * timer, long data)
175 FL_OBJECT * ob = reinterpret_cast<FL_OBJECT*>(data);
176 lyx::Assert(ob && ob->form && timer && timer->u_vdata);
177 FL_FORM * form = ob->form;
178 Tooltips * tooltip = static_cast<Tooltips *>(timer->u_vdata);
180 string const help = tooltip->get(ob);
184 fl_show_oneliner(help.c_str(),
185 form->x + ob->x, form->y + ob->y + ob->h);
189 // post_handler for tooltip help
190 int TooltipHandler(FL_OBJECT * ob, int event)
192 if (!Tooltips::enabled())
196 FL_OBJECT * timer = reinterpret_cast<FL_OBJECT *>(ob->u_cdata);
199 // We do not test for empty help here, since this can never happen
200 if (event == FL_ENTER) {
201 fl_set_object_callback(timer,
203 reinterpret_cast<long>(ob));
204 fl_set_timer(timer, 1);
206 else if (event != FL_MOTION) {
207 fl_set_timer(timer, 0);
215 #endif // FL_REVISION >= 89