167d767e0f
git-svn-id: file:///fltk/svn/fltk/branches/branch-1.3@6614 ea41ed52-d2ee-0310-a9c1-e6b18d33e121
217 lines
7.3 KiB
C++
217 lines
7.3 KiB
C++
//
|
|
// "$Id$"
|
|
//
|
|
// An input/chooser widget.
|
|
// ______________ ____
|
|
// | || __ |
|
|
// | input area || \/ |
|
|
// |______________||____|
|
|
//
|
|
// Copyright 1998-2009 by Bill Spitzak and others.
|
|
// Copyright 2004 by Greg Ercolano.
|
|
//
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU Library General Public
|
|
// License as published by the Free Software Foundation; either
|
|
// version 2 of the License, or (at your option) any later version.
|
|
//
|
|
// This library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// Library General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Library General Public
|
|
// License along with this library; if not, write to the Free Software
|
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
// USA.
|
|
//
|
|
// Please report all bugs and problems on the following page:
|
|
//
|
|
// http://www.fltk.org/str.php
|
|
//
|
|
|
|
/* \file
|
|
Fl_Input_Choice widget . */
|
|
|
|
|
|
|
|
#ifndef Fl_Input_Choice_H
|
|
#define Fl_Input_Choice_H
|
|
|
|
#include <FL/Fl.H>
|
|
#include <FL/Fl_Group.H>
|
|
#include <FL/Fl_Input.H>
|
|
#include <FL/Fl_Menu_Button.H>
|
|
#include <FL/fl_draw.H>
|
|
#include <string.h>
|
|
|
|
/**
|
|
A combination of the input widget and a menu button.
|
|
The user can either type into the input area, or use the
|
|
menu button chooser on the right, which loads the input area
|
|
with predefined text. Normally it is drawn with an inset box
|
|
and a white background.
|
|
<P>
|
|
The application can directly access both the input and menu
|
|
widgets directly, using the menubutton()
|
|
and input() accessor methods.
|
|
*/
|
|
class Fl_Input_Choice : public Fl_Group {
|
|
// Private class to handle slightly 'special' behavior of menu button
|
|
class InputMenuButton : public Fl_Menu_Button {
|
|
void draw() {
|
|
draw_box(FL_UP_BOX, color());
|
|
fl_color(active_r() ? labelcolor() : fl_inactive(labelcolor()));
|
|
int xc = x()+w()/2, yc=y()+h()/2;
|
|
fl_polygon(xc-5,yc-3,xc+5,yc-3,xc,yc+3);
|
|
if (Fl::focus() == this) draw_focus();
|
|
}
|
|
public:
|
|
InputMenuButton(int x,int y,int w,int h,const char*l=0) :
|
|
Fl_Menu_Button(x,y,w,h,l) { box(FL_UP_BOX); }
|
|
};
|
|
|
|
Fl_Input *inp_;
|
|
InputMenuButton *menu_;
|
|
|
|
static void menu_cb(Fl_Widget*, void *data) {
|
|
Fl_Input_Choice *o=(Fl_Input_Choice *)data;
|
|
const Fl_Menu_Item *item = o->menubutton()->mvalue();
|
|
if (item && item->flags & (FL_SUBMENU|FL_SUBMENU_POINTER)) return; // ignore submenus
|
|
if (!strcmp(o->inp_->value(), o->menu_->text()))
|
|
{
|
|
o->Fl_Widget::clear_changed();
|
|
if (o->when() & FL_WHEN_NOT_CHANGED)
|
|
o->do_callback();
|
|
}
|
|
else
|
|
{
|
|
o->inp_->value(o->menu_->text());
|
|
o->inp_->set_changed();
|
|
o->Fl_Widget::set_changed();
|
|
if (o->when() & (FL_WHEN_CHANGED|FL_WHEN_RELEASE))
|
|
o->do_callback();
|
|
}
|
|
|
|
if (o->callback() != default_callback)
|
|
{
|
|
o->Fl_Widget::clear_changed();
|
|
o->inp_->clear_changed();
|
|
}
|
|
}
|
|
|
|
static void inp_cb(Fl_Widget*, void *data) {
|
|
Fl_Input_Choice *o=(Fl_Input_Choice *)data;
|
|
if (o->inp_->changed()) {
|
|
o->Fl_Widget::set_changed();
|
|
if (o->when() & (FL_WHEN_CHANGED|FL_WHEN_RELEASE))
|
|
o->do_callback();
|
|
} else {
|
|
o->Fl_Widget::clear_changed();
|
|
if (o->when() & FL_WHEN_NOT_CHANGED)
|
|
o->do_callback();
|
|
}
|
|
|
|
if (o->callback() != default_callback)
|
|
o->Fl_Widget::clear_changed();
|
|
}
|
|
|
|
// Custom resize behavior -- input stretches, menu button doesn't
|
|
inline int inp_x() { return(x() + Fl::box_dx(box())); }
|
|
inline int inp_y() { return(y() + Fl::box_dy(box())); }
|
|
inline int inp_w() { return(w() - Fl::box_dw(box()) - 20); }
|
|
inline int inp_h() { return(h() - Fl::box_dh(box())); }
|
|
|
|
inline int menu_x() { return(x() + w() - 20 - Fl::box_dx(box())); }
|
|
inline int menu_y() { return(y() + Fl::box_dy(box())); }
|
|
inline int menu_w() { return(20); }
|
|
inline int menu_h() { return(h() - Fl::box_dh(box())); }
|
|
|
|
public:
|
|
/**
|
|
Creates a new Fl_Input_Choice widget using the given position, size,
|
|
and label string.
|
|
<P> Inherited destructor Destroys the widget and any value associated with it.
|
|
*/
|
|
Fl_Input_Choice (int x,int y,int w,int h,const char*l=0) : Fl_Group(x,y,w,h,l) {
|
|
Fl_Group::box(FL_DOWN_BOX);
|
|
align(FL_ALIGN_LEFT); // default like Fl_Input
|
|
inp_ = new Fl_Input(inp_x(), inp_y(),
|
|
inp_w(), inp_h());
|
|
inp_->callback(inp_cb, (void*)this);
|
|
inp_->box(FL_FLAT_BOX); // cosmetic
|
|
inp_->when(FL_WHEN_CHANGED|FL_WHEN_NOT_CHANGED);
|
|
menu_ = new InputMenuButton(menu_x(), menu_y(),
|
|
menu_w(), menu_h());
|
|
menu_->callback(menu_cb, (void*)this);
|
|
menu_->box(FL_FLAT_BOX); // cosmetic
|
|
end();
|
|
}
|
|
|
|
/** Adds an item to the menu.*/
|
|
void add(const char *s) { menu_->add(s); }
|
|
int changed() const { return inp_->changed() | Fl_Widget::changed();}
|
|
void clear_changed() {
|
|
inp_->clear_changed();
|
|
Fl_Widget::clear_changed();
|
|
}
|
|
void set_changed() {
|
|
inp_->set_changed();
|
|
// no need to call Fl_Widget::set_changed()
|
|
}
|
|
/** Removes all items from the menu. */
|
|
void clear() { menu_->clear(); }
|
|
/** Gets the box type of the menu button */
|
|
Fl_Boxtype down_box() const { return (menu_->down_box()); }
|
|
/** Sets the box type of the menu button */
|
|
void down_box(Fl_Boxtype b) { menu_->down_box(b); }
|
|
/** Gets the Fl_Menu_Item array used for the menu. */
|
|
const Fl_Menu_Item *menu() { return (menu_->menu()); }
|
|
/** Sets the Fl_Menu_Item array used for the menu. */
|
|
void menu(const Fl_Menu_Item *m) { menu_->menu(m); }
|
|
void resize(int X, int Y, int W, int H) {
|
|
Fl_Group::resize(X,Y,W,H);
|
|
inp_->resize(inp_x(), inp_y(), inp_w(), inp_h());
|
|
menu_->resize(menu_x(), menu_y(), menu_w(), menu_h());
|
|
}
|
|
/** Gets the encapsulated input text color attributes */
|
|
Fl_Color textcolor() const { return (inp_->textcolor());}
|
|
/** Sets the encapsulated input text color attributes */
|
|
void textcolor(Fl_Color c) { inp_->textcolor(c);}
|
|
/** Gets the encapsulated input text font attributes */
|
|
Fl_Font textfont() const { return (inp_->textfont());}
|
|
/** Sets the encapsulated input text font attributes */
|
|
void textfont(Fl_Font f) { inp_->textfont(f);}
|
|
/** Gets the encapsulated input size attributes */
|
|
Fl_Fontsize textsize() const { return (inp_->textsize()); }
|
|
/** Sets the encapsulated input size attributes */
|
|
void textsize(Fl_Fontsize s) { inp_->textsize(s); }
|
|
/** See void Fl_Input_Choice::value(const char *s) */
|
|
const char* value() const { return (inp_->value()); }
|
|
/**
|
|
Sets or returns the input widget's current contents. The
|
|
second form sets the contents using the index into the menu
|
|
which you can set as an integer. Setting the value effectively
|
|
'chooses' this menu item, and sets it as the new input text,
|
|
deleting the previous text.
|
|
*/
|
|
void value(const char *val) { inp_->value(val); }
|
|
/** See void Fl_Input_Choice::value(const char *s) */
|
|
void value(int val) {
|
|
menu_->value(val);
|
|
inp_->value(menu_->text(val));
|
|
}
|
|
/** Returns a reference to the internal Fl_Menu_Button widget. */
|
|
Fl_Menu_Button *menubutton() { return menu_; }
|
|
/**
|
|
Returns a reference to the internal Fl_Input widget.</p>
|
|
*/
|
|
Fl_Input *input() { return inp_; }
|
|
};
|
|
|
|
#endif // !Fl_Input_Choice_H
|
|
|
|
//
|
|
// End of "$Id$".
|
|
//
|