fltk/src/Fl_Tabs.cxx

1144 lines
37 KiB
C++

//
// Tab widget for the Fast Light Tool Kit (FLTK).
//
// Copyright 1998-2023 by Bill Spitzak and others.
//
// This library is free software. Distribution and use rights are outlined in
// the file "COPYING" which should have been included with this file. If this
// file is missing or damaged, see the license at:
//
// https://www.fltk.org/COPYING.php
//
// Please see the following page on how to report bugs and issues:
//
// https://www.fltk.org/bugs.php
//
// This is the "file card tabs" interface to allow you to put lots and lots
// of buttons and switches in a panel, as popularized by many toolkits.
// Each child widget is a card, and its label() is printed on the card tab.
// Clicking the tab makes that card visible.
#include <FL/Fl.H>
#include <FL/Fl_Tabs.H>
#include <FL/fl_draw.H>
#include <FL/Fl_Tooltip.H>
#include <FL/Fl_Menu_Item.H>
#include <FL/Fl_Window.H>
#include <stdio.h>
#include <stdlib.h>
#define BORDER 2
#define EXTRASPACE 10
#define SELECTION_BORDER 5
#define EXTRAGAP 2
#define MARGIN 20
enum {LEFT, RIGHT, SELECTED};
static int fl_min(int a, int b) { return a < b ? a : b; }
/** Make sure that we redraw all tabs when new children are added. */
int Fl_Tabs::on_insert(Fl_Widget* candidate, int index) {
redraw_tabs();
return Fl_Group::on_insert(candidate, index);
}
/** Make sure that we redraw all tabs when children are moved. */
int Fl_Tabs::on_move(int a, int b) {
redraw_tabs();
return Fl_Group::on_move(a, b);
}
/** Make sure that we redraw all tabs when new children are removed. */
void Fl_Tabs::on_remove(int index) {
redraw_tabs();
if (child(index)->visible()) {
if (index+1<children())
value(child(index+1));
else if (index>0)
value(child(index-1));
}
if (children()==1)
damage(FL_DAMAGE_ALL);
Fl_Group::on_remove(index);
}
/** Make sure that we redraw all tabs when the widget size changes. */
void Fl_Tabs::resize(int X, int Y, int W, int H) {
redraw_tabs();
Fl_Group::resize(X, Y, W, H);
}
/** Calculate tab positions and widths.
This protected method calculates the horizontal display positions and
widths of all tabs. If the number of children \c 'nc' (see below) is \> 0
three internal arrays are allocated, otherwise the arrays are free'd
and the pointers are set to NULL. Note that the first array is larger
(nc+1).
- tab_pos[nc+1] : The left edges of each tab plus a fake left edge
for a tab past the right-hand one.
- tab_width[nc] : The width of each tab
- tab_flags[nc] : Flags, bit 0 is set if the tab is compressed
If needed, these arrays are (re)allocated.
These positions are actually of the left edge of the slope.
They are either separated by the correct distance
or by EXTRASPACE or by zero.
In OVERFLOW_COMPRESS mode, tab positions and widths are compressed to make
the entire tabs bar fit into the width of Fl_Tabs while keeping the selected
tab fully visible.
In other overflow modes, the tabs area may be dragged horizontally using
\ref tab_offset. The tab_pos array is not adjusted to the horizontal offset,
but starts at this->x() plus the box's left margin.
The protected variable `tab_count` is set to the currently allocated
size, i.e. the number of children (`nc`).
\returns Index of the selected item
\retval -1 If the number of children is 0 (zero).
\note Return values in 1.3 were not documented. Return values before Sep 2023
were documented as 1 based index and 0 if there were no children. This
was actually never the case. It always returned a 0 based index and
the (useless) value of also 0 if there were no children. The current
version return -1 if there are no children.
\note For this method to work, only on single child should be selected.
Calling the method \ref value() before calling \ref tab_positions()
will ensure that exactly one child is selected and return a pointer
to that child.
\see clear_tab_positions()
*/
int Fl_Tabs::tab_positions() {
const int nc = children();
if (nc != tab_count) {
clear_tab_positions();
if (nc) {
tab_pos = (int*)malloc((nc+1)*sizeof(int));
tab_width = (int*)malloc((nc)*sizeof(int));
tab_flags = (int*)malloc((nc)*sizeof(int));
}
tab_count = nc;
}
if (nc == 0) return -1;
int selected = 0;
Fl_Widget*const* a = array();
int i;
char prev_draw_shortcut = fl_draw_shortcut;
fl_draw_shortcut = 1;
int l = tab_pos[0] = Fl::box_dx(box());
for (i=0; i<nc; i++) {
Fl_Widget* o = *a++;
if (o->visible()) selected = i;
int wt = 0; int ht = 0;
Fl_Labeltype ot = o->labeltype();
Fl_Align oa = o->align();
if (ot == FL_NO_LABEL) {
o->labeltype(FL_NORMAL_LABEL);
}
o->align(tab_align());
o->measure_label(wt,ht);
o->labeltype(ot);
o->align(oa);
if (o->when() & FL_WHEN_CLOSED)
wt += labelsize()/2 + EXTRAGAP;
tab_width[i] = wt + EXTRASPACE;
tab_pos[i+1] = tab_pos[i] + tab_width[i] + BORDER;
tab_flags[i] = 0;
}
fl_draw_shortcut = prev_draw_shortcut;
if (overflow_type == OVERFLOW_COMPRESS) {
int r = w() - Fl::box_dw(box());;
if ( (nc > 1) && (tab_pos[nc] > r) ) {
int wdt = r - l;
// extreme case: the selected tab is wider than Fl_Tabs itself
int available = wdt - tab_width[selected];
if (available <= 8*nc) {
// if the current tab is so huge that it doesn't fit Fl_Tabs, we make
// shrink all other tabs to 8 pixels and give the selected tab the rest
for (i = 0; i < nc; i++) {
if (i < selected) {
tab_pos[i] = l + 8*i;
tab_flags[i] |= 1;
} else if (i>selected) {
tab_pos[i] = r - (nc-i)*8;
tab_flags[i] |= 1;
} else {
tab_pos[i] = l + 8*i;;
tab_flags[i] &= ~1;
}
tab_pos[nc] = r;
}
} else {
// This method tries to keep as many visible tabs to the left and right
// of the selected tab. All other tabs are compressed until they are
// no smaller than 8 pixels.
// Overlap to the left and right of the selection is proportional
// to the left and right total tabs widths.
// The dynamic of this method is really nice to watch: start FLUID and
// edit test/tabs. Select any tab and change the label to make the tab
// wider and smaller. All other tabs will move nicely to make room for
// the bigger label. Even if two tabs are each wider than Fl_Tabs.
int overflow = tab_pos[nc] - r;
int left_total = tab_pos[selected] - l;
int right_total = tab_pos[nc] - tab_pos[selected+1];
int left_overflow = left_total+right_total ? overflow * left_total / (left_total+right_total) : overflow;
int right_overflow = overflow - left_overflow;
// now clip the left tabs until we compensated overflow on the left
int xdelta = 0; // accumulate the tab x correction
for (i=0; i<selected; i++) { // do this for all tabs on the left of selected
int tw = tab_width[i]; // get the current width of this tab
if (left_overflow > 0) { // do we still need to compensate?
tw -= left_overflow; // try to compensate everything
if (tw < 8) tw = 8; // but keep a minimum width of 8
int wdelta = tab_width[i] - tw; // how many pixels did we actually take?
left_overflow -= wdelta; // remove that and keep the remaining overflow
xdelta += wdelta; // accumulate amount of pixel shift
if (wdelta > 16) tab_flags[i] |= 1; // remove the close button if we overlap too much
}
tab_pos[i+1] -= xdelta; // fix the overlap by moving the tab on the right
}
// and clip the right tabs until we compensated overflow on the right
xdelta = 0;
for (i=nc-1; i>selected; i--) {
int tw = tab_width[i];
if (right_overflow > 0) {
tw -= right_overflow;
if (tw < 8) tw = 8;
int wdelta = tab_width[i] - tw;
right_overflow -= wdelta;
xdelta += wdelta;
// with the close button on the left, overlapping gets more confusing,
// so remove the button sooner
if (wdelta > 4) tab_flags[i] |= 1;
}
tab_pos[i] -= overflow - xdelta;
}
tab_pos[nc] = r;
}
}
}
return selected;
}
/**
Return space (height) in pixels usable for tabs.
The calculated height is the largest space between all children
and the upper and lower widget boundaries, respectively. If the
space at the bottom is larger than at the top, the value will be
negative and the tabs should be placed at the bottom.
\returns Vertical space that can be used for the tabs.
\retval > 0 To put the tabs at the top of the widget.
\retval < 0 To put the tabs on the bottom.
\retval Full height, if children() == 0.
*/
int Fl_Tabs::tab_height() {
if (children() == 0) return h();
int H = h();
int H2 = y();
Fl_Widget*const* a = array();
for (int i=children(); i--;) {
Fl_Widget* o = *a++;
if (o->y() < y()+H) H = o->y()-y();
if (o->y()+o->h() > H2) H2 = o->y()+o->h();
}
H2 = y()+h()-H2;
if (H2 > H) return (H2 <= 0) ? 0 : -H2;
else return (H <= 0) ? 0 : H;
}
/** Return a pointer to the child widget with a tab at the given coordinates.
The Fl_Tabs::which() method returns a pointer to the child widget of the
Fl_Tabs container that corresponds to the tab at the given event coordinates.
If the event coordinates are outside the area of the tabs or if the Fl_Tabs
container has no children, the method returns NULL.
\param event_x, event_y event coordinates
\returns pointer to the selected child widget, or NULL
*/
Fl_Widget *Fl_Tabs::which(int event_x, int event_y) {
if (children() == 0) return 0;
int H = tab_height();
if (H < 0) {
if (event_y > y()+h() || event_y < y()+h()+H) return 0;
} else {
if (event_y > y()+H || event_y < y()) return 0;
}
if (event_x < x()) return 0;
Fl_Widget *ret = 0L;
const int nc = children();
tab_positions();
for (int i=0; i<nc; i++) {
if (event_x < x()+tab_pos[i+1]+tab_offset) {
ret = child(i);
break;
}
}
return ret;
}
/** Check whether the coordinates fall within the "close" button area of the tab.
The Fl_Tabs::hit_close() method checks whether the given event coordinates
fall within the area of the "close" button on the tab of the specified
child widget. This method should be called after the Fl_Tabs::which() method,
which updates a lookup table used to determine the width of each tab.
\param o check the tab of this widget
\param event_x, event_y event coordinates
\return 1 if we hit the close button, and 0 otherwise
*/
int Fl_Tabs::hit_close(Fl_Widget *o, int event_x, int event_y) {
(void)event_y;
for (int i=0; i<children(); i++) {
if (child(i)==o) {
// never hit the "close" button on a compressed tab unless it's the active one
if (tab_flags[i] & 1)
return 0;
// did we hit the area of teh "x"?
int tab_x = tab_pos[i] + tab_offset + x();
return ( (event_x >= tab_x)
&& (event_x < tab_x + (labelsize()+EXTRASPACE+EXTRAGAP)/2) );
}
}
return 0;
}
/** Determine if the coordinates are in the area of the overflow menu button.
\param event_x, event_y event coordinates
\return 1 if we hit the overflow menu button, and 0 otherwise
*/
int Fl_Tabs::hit_overflow_menu(int event_x, int event_y) {
if (!has_overflow_menu)
return 0;
int H = tab_height();
if (event_x < x()+w()-abs(H))
return 0;
if (H >= 0) {
if (event_y > y()+H)
return 0;
} else {
if (event_y < y()+h()+H)
return 0;
}
return 1;
}
/** Determine if the coordinates are within the tabs area.
\param event_x, event_y event coordinates
\return 1 if we hit the tabs area, and 0 otherwise
*/
int Fl_Tabs::hit_tabs_area(int event_x, int event_y) {
int H = tab_height();
if (H >= 0) {
if (event_y > y()+H)
return 0;
} else {
if (event_y < y()+h()+H)
return 0;
}
if (has_overflow_menu && event_x > x()+w()-abs(H))
return 0;
return 1;
}
/**
Check if the tabs overflow and sets the has_overflow_menu flag accordingly.
*/
void Fl_Tabs::check_overflow_menu() {
int nc = children();
int H = tab_height(); if (H < 0) H = -H;
if (tab_pos[nc] > w()-H) {
has_overflow_menu = 1;
} else {
has_overflow_menu = 0;
}
}
/**
This is called when the user clicks the overflow pulldown menu button.
This method creates a menu item array that contains the titles of all
tabs in the Fl_Tabs group. Visible and invisible tabs are separated
by dividers to indicate their state.
The menu is then presented until the user selects an item or cancels.
The chosen tab is then selected and made visible.
The menu item array is the deleted.
*/
void Fl_Tabs::handle_overflow_menu() {
int nc = children();
int H = tab_height(); if (H < 0) H = -H;
int i, fv=-1, lv=nc; // first and last visible tab
if (nc <= 0) return;
// count visible children
for (i = 0; i < nc; i++) {
if (tab_pos[i]+tab_offset < 0) fv = i;
if (tab_pos[i]+tab_width[i]+tab_offset <= w()-H) lv = i;
}
// create a menu with all children
Fl_Menu_Item* overflow_menu = new Fl_Menu_Item[nc+1];
memset(overflow_menu, 0, sizeof(Fl_Menu_Item)*(nc+1));
for (i = 0; i < nc; i++) {
overflow_menu[i].label(child(i)->label());
overflow_menu[i].user_data(child(i));
overflow_menu[i].labelfont(labelfont());
overflow_menu[i].labelsize(labelsize());
if ( (i == fv) || (i == lv) )
overflow_menu[i].flags |= FL_MENU_DIVIDER;
if (child(i)->visible())
overflow_menu[i].labelfont_ |= FL_BOLD;
}
// show the menu and handle the selection
const Fl_Menu_Item *m = overflow_menu->popup(x()+w()-H, (tab_height()>0)?(y()+H):(y()+h()));
if (m)
value((Fl_Widget*)m->user_data());
// delete the menu until we need it next time
if (overflow_menu) {
delete[] overflow_menu;
overflow_menu = NULL;
}
}
/**
Draw square button-like graphics with a down arrow in the top or bottom right corner.
*/
void Fl_Tabs::draw_overflow_menu_button() {
int H = tab_height();
int X, Y;
if (H > 0) {
X = x() + w() - H;
Y = y();
} else {
H = -H;
X = x() + w() - H;
Y = y() + h() - H - 1;
}
draw_box(box(), X, Y, H, H, color());
Fl_Rect r(X, Y, H, H);
Fl_Color arrow_color = active_r() ? labelcolor() : fl_inactive(labelcolor());
fl_draw_arrow(r, FL_ARROW_CHOICE, FL_ORIENT_NONE, arrow_color);
}
/**
Redraw all tabs (and only the tabs).
This method sets the Fl_Tab's damage flags so the tab area is redrawn.
*/
void Fl_Tabs::redraw_tabs() {
int H = tab_height();
if (H >= 0) {
damage(FL_DAMAGE_EXPOSE, x(), y(), w(), H + SELECTION_BORDER);
} else {
H = -H;
damage(FL_DAMAGE_EXPOSE, x(), y() + h() - H - SELECTION_BORDER, w(), H + SELECTION_BORDER);
}
}
/**
Handle all events in the tabs area and forward the rest to the selected child.
\param[in] event handle this event
\return 1 if the event was handled
*/
int Fl_Tabs::handle(int event) {
static int initial_x = 0;
static int initial_tab_offset = 0;
static int forward_motion_to_group = 0;
Fl_Widget *o;
int i;
switch (event) {
case FL_MOUSEWHEEL:
if ( ( (overflow_type == OVERFLOW_DRAG) || (overflow_type == OVERFLOW_PULLDOWN) )
&& hit_tabs_area(Fl::event_x(), Fl::event_y()) ) {
int original_tab_offset = tab_offset;
tab_offset -= 2 * Fl::event_dx();
if (tab_offset > 0)
tab_offset = 0;
int m = 0;
if (overflow_type == OVERFLOW_PULLDOWN) m = abs(tab_height());
int dw = tab_pos[children()] + tab_offset - w();
if (dw < -m)
tab_offset -= dw+m;
if (tab_offset != original_tab_offset)
redraw_tabs();
return 1;
}
return Fl_Group::handle(event);
case FL_PUSH:
initial_x = Fl::event_x();
initial_tab_offset = tab_offset;
forward_motion_to_group = 0;
if (hit_overflow_menu(Fl::event_x(), Fl::event_y())) {
handle_overflow_menu();
return 1;
}
if (!hit_tabs_area(Fl::event_x(), Fl::event_y())) {
forward_motion_to_group = 1;
}
/* FALLTHROUGH */
case FL_DRAG:
case FL_RELEASE:
if (forward_motion_to_group) {
return Fl_Group::handle(event);
}
o = which(Fl::event_x(), Fl::event_y());
if ( (overflow_type == OVERFLOW_DRAG) || (overflow_type == OVERFLOW_PULLDOWN) ) {
if (tab_pos[children()] < w() && tab_offset == 0) {
// fall through
} else if (!Fl::event_is_click()) {
tab_offset = initial_tab_offset + Fl::event_x() - initial_x;
int m = 0;
if (overflow_type == OVERFLOW_PULLDOWN) m = abs(tab_height());
if (tab_offset > 0) {
initial_tab_offset -= tab_offset;
tab_offset = 0;
} else {
int dw = tab_pos[children()] + tab_offset - w();
if (dw < -m) {
initial_tab_offset -= dw+m;
tab_offset -= dw+m;
}
}
redraw_tabs();
return 1;
}
}
if (event == FL_RELEASE) {
push(0);
if (o && Fl::visible_focus() && Fl::focus()!=this) {
Fl::focus(this);
redraw_tabs();
}
if (o && (o->when() & FL_WHEN_CLOSED) && hit_close(o, Fl::event_x(), Fl::event_y())) {
o->do_callback(FL_REASON_CLOSED);
return 1; // o may be deleted at this point
}
if (o && // Released on a tab and..
(value(o) || // tab changed value or..
(when()&(FL_WHEN_NOT_CHANGED)) // ..no change but WHEN_NOT_CHANGED set,
) // handles FL_WHEN_RELEASE_ALWAYS too.
) {
Fl_Widget_Tracker wp(o);
set_changed();
do_callback(FL_REASON_SELECTED);
if (wp.deleted()) return 1;
}
Fl_Tooltip::current(o);
} else {
push(o);
}
return 1;
case FL_MOVE: {
int ret = Fl_Group::handle(event);
Fl_Widget *tooltip_widget = Fl_Tooltip::current();
Fl_Widget *n; // initialized later
int H = tab_height();
if ( (H >= 0) && (Fl::event_y() > y()+H) )
return ret;
else if ( (H < 0) && (Fl::event_y() < y()+h()+H) )
return ret;
else {
n = which(Fl::event_x(), Fl::event_y());
if (!n) n = this;
}
if (n != tooltip_widget)
Fl_Tooltip::enter(n);
return ret; }
case FL_FOCUS:
case FL_UNFOCUS:
if (!Fl::visible_focus()) return Fl_Group::handle(event);
if (Fl::event() == FL_RELEASE ||
Fl::event() == FL_SHORTCUT ||
Fl::event() == FL_KEYBOARD ||
Fl::event() == FL_FOCUS ||
Fl::event() == FL_UNFOCUS) {
redraw_tabs();
if (Fl::event() == FL_FOCUS) return Fl_Group::handle(event);
if (Fl::event() == FL_UNFOCUS) return 0;
else return 1;
} else return Fl_Group::handle(event);
case FL_KEYBOARD:
switch (Fl::event_key()) {
case FL_Left:
if (!children()) return 0;
if (child(0)->visible()) return 0;
for (i = 1; i < children(); i ++)
if (child(i)->visible()) break;
value(child(i - 1));
set_changed();
do_callback(FL_REASON_SELECTED);
return 1;
case FL_Right:
if (!children()) return 0;
if (child(children() - 1)->visible()) return 0;
for (i = 0; i < children()-1; i++)
if (child(i)->visible()) break;
value(child(i + 1));
set_changed();
do_callback(FL_REASON_SELECTED);
return 1;
case FL_Down:
redraw();
return Fl_Group::handle(FL_FOCUS);
default:
break;
}
return Fl_Group::handle(event);
case FL_SHORTCUT:
for (i = 0; i < children(); ++i) {
Fl_Widget *c = child(i);
if (c->test_shortcut(c->label())) {
char sc = !c->visible();
value(c);
if (sc) {
set_changed();
do_callback(FL_REASON_SELECTED);
} else {
do_callback(FL_REASON_RESELECTED);
}
return 1;
}
}
return Fl_Group::handle(event);
case FL_SHOW:
value(); // update visibilities and fall through
default:
return Fl_Group::handle(event);
}
}
/**
This is called by the tab widget's handle() method to set the
tab group widget the user last FL_PUSH'ed on. Set back to zero
on FL_RELEASE.
As of this writing, the value is mainly used by draw_tab()
to determine whether or not to draw a 'down' box for the tab
when it's clicked, and to turn it off if the user drags off it.
\see push().
*/
int Fl_Tabs::push(Fl_Widget *o) {
if (push_ == o) return 0;
if ( (push_ && !push_->visible()) || (o && !o->visible()) )
redraw_tabs();
push_ = o;
return 1;
}
/**
Gets the currently visible widget/tab.
The Fl_Tabs::value() method returns a pointer to the currently visible child
widget of the Fl_Tabs container. The visible child is the first child that
is currently being displayed, or the last child if none of the children are
being displayed.
If child widgets have been added, moved, or deleted, this method ensures that
only one tab is visible at a time.
\return a pointer to the currently visible child
*/
Fl_Widget* Fl_Tabs::value() {
Fl_Widget* v = 0;
Fl_Widget*const* a = array();
for (int i=children(); i--;) {
Fl_Widget* o = *a++;
if (v) o->hide();
else if (o->visible()) v = o;
else if (!i) {o->show(); v = o;}
}
return v;
}
/** Sets the widget to become the current visible widget/tab.
The Fl_Tabs::value() method allows you to set a particular child widget of
the Fl_Tabs container to be the currently visible widget. If the specified
widget is a child of the Fl_Tabs container, it will be made visible and all
other children will be hidden. The method returns 1 if the value was changed,
and 0 if the specified value was already set.
\param[in] newvalue a poiner to a child widget
\return 1 if a different tab was chosen
\return 0 if there was no change (new value already set)
*/
int Fl_Tabs::value(Fl_Widget *newvalue) {
Fl_Widget*const* a = array();
int ret = 0;
int selected = -1;
for (int i=children(); i--;) {
Fl_Widget* o = *a++;
if (o == newvalue) {
if (!o->visible()) ret = 1;
o->show();
selected = children()-i-1;
} else {
o->hide();
}
}
// make sure that the selected tab is visible
if ( (selected >= 0)
&& (ret == 1)
&& ( (overflow_type == OVERFLOW_DRAG)
|| (overflow_type == OVERFLOW_PULLDOWN) ) ) {
int m = MARGIN;
if ( (selected == 0) || (selected == children()-1) ) m = 0;
int mr = m;
if (overflow_type == OVERFLOW_PULLDOWN) mr += abs(tab_height());
tab_positions();
if (tab_pos[selected]+tab_width[selected]+tab_offset+mr > w()) {
tab_offset = w() - tab_pos[selected] - tab_width[selected] - mr;
} else if (tab_pos[selected]+tab_offset-m < 0) {
tab_offset = -tab_pos[selected]+m;
}
}
redraw_tabs();
return ret;
}
/**
Draw the tabs area, the optional pulldown button, and all children.
*/
void Fl_Tabs::draw() {
//
// FL_DAMAGE_CHILD : this is set if any of the children asked for a redraw
// Fl_Tabs forwards this by calling `update_child(v)`
// FL_DAMAGE_EXPOSE : this is set if some setting in a widget changed
// Fl_Tabs uses this to indicate that the tabs area needs a full redraw
// FL_DAMAGE_SCROLL : this is used as a custom flag in various widgets
// Fl_Tabs honors this flag for back compatibly as FL_DAMAGE_EXPOSE
// FL_DAMAGE_ALL : just redraw everything
// Anatomy of tabs on top:
// +------+ +---+ <<-- selected tabs start at y()
// | text | +------+ | V | <-- other tabs are offset down by BORDER
// | | | text | +---+ <-- the pulldown button width equals H
// ++ +-----------------+ <-- tab_height() to tab_height + Fl::box_dx(box())
// +-------------------------+ <-- tab_height + SELECTION_BORDER
// | |
// ↑____↑ this area within the SELECTION_BORDER is called "stem"
//
// tab_height() calculates the distance from y() to the "highest" child.
// Note that the SELECTION_BORDER bleeds into the child area!
// Note that the clear area under the selected tab also bleeds into the child.
// Note that children have FL_NO_BOX and Fl_Tabs must draw the background.
//
// Horizontally, we start tabs at x() + Fl::box_dx()
// When uncompressed, the space between tabs is EXTRASPACE
// EXTRAGAP is the space between the close cross and the label
// MARGIN is the minimal distance to the left and right edge of Fl_Tabs for
// the selected tabs if the overflow mode allows scrolling
if (children() == 0) {
fl_rectf(x(), y(), w(), h(), color());
if (align() & FL_ALIGN_INSIDE)
draw_label();
clear_damage();
return;
}
Fl_Widget *selected_child = value(); // return the first visible child and hide all others
tab_positions();
int selected = find(selected_child); // find that child in the list and return 0..children()-1
if (selected == children()) selected = -1; // if anything fails, selected is -1 and
int H = tab_height();
Fl_Color selected_tab_color = selected_child ? selected_child->color() : color();
bool tabs_at_top = (H > 0);
bool colored_selection_border = (selection_color() != selected_tab_color);
int tabs_y, tabs_h;
int child_area_y, child_area_h;
int clipped_child_area_y, clipped_child_area_h;
int selection_border_y, selection_border_h;
selection_border_h = colored_selection_border ? SELECTION_BORDER : Fl::box_dx(box());
if (tabs_at_top) {
tabs_h = H;
tabs_y = y();
selection_border_y = y() + tabs_h;
child_area_y = y() + tabs_h;
child_area_h = h() - tabs_h;
clipped_child_area_y = y() + tabs_h + selection_border_h;
clipped_child_area_h = h() - tabs_h - selection_border_h;
} else {
tabs_h = -H;
tabs_y = y() + h() - tabs_h;
selection_border_y = tabs_y - selection_border_h;
child_area_y = y();
child_area_h = h() - tabs_h;
clipped_child_area_y = y();
clipped_child_area_h = h() - tabs_h - selection_border_h;
}
// ---- draw the tabs and the selection border
if (damage() & (FL_DAMAGE_ALL|FL_DAMAGE_EXPOSE|FL_DAMAGE_SCROLL))
{
// -- draw tabs background
if (parent()) {
Fl_Widget *p = parent();
fl_push_clip(x(), tabs_y, w(), tabs_h);
if (Fl_Window *win = p->as_window()) {
fl_draw_box(p->box(), 0, 0, p->w(), p->h(), p->color());
win->draw_backdrop();
} else {
fl_draw_box(p->box(), p->x(), p->y(), p->w(), p->h(), p->color());
}
fl_pop_clip();
} else {
fl_rectf(x(), tabs_y, w(), tabs_h, color());
}
// -- draw selection border
fl_push_clip(x(), selection_border_y, w(), selection_border_h);
if (colored_selection_border) {
draw_box(box(), x(), y(), w(), h(), selected_tab_color);
draw_box(box(), x(), selection_border_y, w(), selection_border_h, selection_color());
} else {
draw_box(box(), x(), child_area_y, w(), child_area_h, selected_tab_color);
}
// draw the stem, the area that reaches from the tab into the selection border
if (selected != -1) {
int stem_x = x() + tab_pos[selected] + tab_offset;
int stem_w = fl_min(tab_pos[selected+1] - tab_pos[selected], tab_width[selected]);
if (colored_selection_border) {
if (tabs_at_top)
fl_rectf(stem_x, selection_border_y, stem_w, selection_border_h/2, selection_color());
else
fl_rectf(stem_x, selection_border_y+selection_border_h-selection_border_h/2, stem_w, selection_border_h/2, selection_color());
} else {
fl_rectf(stem_x, child_area_y-tabs_h, stem_w, child_area_h+2*tabs_h, selection_color());
}
}
fl_pop_clip();
// -- draw all tabs
fl_push_clip(x(), tabs_y, w(), tabs_h);
int i, clip_left, clip_right;
int safe_selected = selected == -1 ? children() : selected;
// draw all tabs from the leftmost up to the selected one, stacking them
// visually as needed. The clipping assures that no tabs shine through gaps
// between tabs.
clip_left = x();
for (i=0; i<safe_selected; i++) {
clip_right = (i<tab_count-1) ? x()+(tab_offset+tab_pos[i+1]+tab_width[i+1]/2) : x() + w();
fl_push_clip(clip_left, tabs_y, clip_right-clip_left, tabs_h);
draw_tab(x()+tab_pos[i], x()+tab_pos[i+1],
tab_width[i], H, child(i), tab_flags[i], LEFT);
fl_pop_clip();
}
// draw all tabs from the rightmost back to the selected one, also visually stacking them
clip_right = x() + w();
for (i=children()-1; i > safe_selected; i--) {
clip_left = (i>0) ? (tab_offset+tab_pos[i]-tab_width[i-1]/2) : x();
fl_push_clip(clip_left, tabs_y, clip_right-clip_left, tabs_h);
draw_tab(x()+tab_pos[i], x()+tab_pos[i+1],
tab_width[i], H, child(i), tab_flags[i], RIGHT);
fl_pop_clip();
}
// if there is a selected tab, draw it last over all other tabs
if (selected > -1)
draw_tab(x()+tab_pos[selected], x()+tab_pos[selected+1],
tab_width[selected], H, selected_child, tab_flags[selected], SELECTED);
fl_pop_clip();
// -- draw the overflow menu button
if (overflow_type == OVERFLOW_PULLDOWN)
check_overflow_menu();
if (has_overflow_menu)
draw_overflow_menu_button();
}
// ---- draw the child area
if (damage() & (FL_DAMAGE_ALL|FL_DAMAGE_CHILD)) {
// clip to area below selection border
fl_push_clip(x(), clipped_child_area_y, w(), clipped_child_area_h);
if (damage() & (FL_DAMAGE_ALL)) {
// draw the box and background around the child
if (colored_selection_border)
draw_box(box(), x(), y(), w(), h(), selected_tab_color);
else
draw_box(box(), x(), child_area_y, w(), child_area_h, selected_tab_color);
// force draw the selected child
if (selected_child)
draw_child(*selected_child);
} else if (damage() & (FL_DAMAGE_CHILD)) {
// draw the selected child
if (selected_child)
update_child(*selected_child);
}
// stop clipping
fl_pop_clip();
}
clear_damage();
}
/**
Draw a tab in the top or bottom tabs area.
Tabs can be selected, or on the left or right side of the selected tab. If
overlapping, left tabs are drawn bottom to top using clipping. The selected
tab is then the topmost, followed by the right side tabs drawn top to bottom.
Tabs with the FL_WHEN_CLOSE bit set will draw a cross on their left side
only if they are not compressed/overlapping.
\param[in] x1 horizontal position of the left visible edge of the tab
\param[in] x2 horizontal position of the following tab
\param[in] W, H width and height of the tab
\param[in] o the child widget that corresponds to this tab
\param[in] flags if bit 1 is set, this tab is overlapped by another tab
\param[in] what can be LEFT, SELECTED, or RIGHT to indicate if the tab is to
the left side or the right side of the selected tab, or the selected tab itself
*/
void Fl_Tabs::draw_tab(int x1, int x2, int W, int H, Fl_Widget* o, int flags, int what) {
x1 += tab_offset;
x2 += tab_offset;
int sel = (what == SELECTED);
int dh = Fl::box_dh(box());
int wc = 0; // width of "close" button if drawn, or 0
char prev_draw_shortcut = fl_draw_shortcut;
fl_draw_shortcut = 1;
Fl_Boxtype bt = (o == push_ && !sel) ? fl_down(box()) : box();
Fl_Color bc = sel ? selection_color() : o->selection_color();
// Save the label color and label type
Fl_Color oc = o->labelcolor();
Fl_Labeltype ot = o->labeltype();
// Set a labeltype that really draws a label
if (ot == FL_NO_LABEL)
o->labeltype(FL_NORMAL_LABEL);
// compute offsets to make selected tab look bigger
int yofs = sel ? 0 : BORDER;
if ((x2 < x1+W) && what == RIGHT) x1 = x2 - W;
if (H >= 0) {
H += dh;
draw_box(bt, x1, y() + yofs, W, H + 10 - yofs, bc);
// Draw the label using the current color...
o->labelcolor(sel ? labelcolor() : o->labelcolor());
// Draw the "close" button if requested
if ( (o->when() & FL_WHEN_CLOSED) && !(flags & 1) ) {
int sz = labelsize()/2, sy = (H - sz)/2;
fl_draw_symbol("@3+", x1 + EXTRASPACE/2, y() + yofs/2 + sy, sz, sz, o->labelcolor());
wc = sz + EXTRAGAP;
}
// Draw the label text
o->draw_label(x1 + wc, y() + yofs, W - wc, H - yofs, tab_align());
// Draw the focus box
if (Fl::focus() == this && o->visible())
draw_focus(bt, x1, y(), W, H, bc);
} else {
H = -H;
H += dh;
draw_box(bt, x1, y() + h() - H - 10, W, H + 10 - yofs, bc);
// Draw the label using the current color...
o->labelcolor(sel ? labelcolor() : o->labelcolor());
// Draw the "close" button if requested
if ( (o->when() & FL_WHEN_CLOSED) && (x1+W < x2) ) {
int sz = labelsize()/2, sy = (H - sz)/2;
fl_draw_symbol("@3+", x1 + EXTRASPACE/2, y() + h() - H -yofs/2 + sy, sz, sz, o->labelcolor());
wc = sz + EXTRAGAP;
}
// Draw the label text
o->draw_label(x1 + wc, y() + h() - H, W - wc, H - yofs, tab_align());
// Draw the focus box
if (Fl::focus() == this && o->visible())
draw_focus(bt, x1, y()+h()-H+1, W, H, bc);
}
fl_draw_shortcut = prev_draw_shortcut;
// Restore the original label color and label type
o->labelcolor(oc);
o->labeltype(ot);
}
/**
Creates a new Fl_Tabs widget using the given position, size,
and label string. The default boxtype is FL_THIN_UP_BOX.
Use add(Fl_Widget*) to add each child, which are usually
Fl_Group widgets. The children should be sized to stay
away from the top or bottom edge of the Fl_Tabs widget,
which is where the tabs will be drawn.
All children of Fl_Tabs should have the same size and exactly fit on top of
each other. They should only leave space above or below where the tabs will
go, but not on the sides. If the first child of Fl_Tabs is set to
"resizable()", the riders will not resize when the tabs are resized.
The destructor <I>also deletes all the children</I>. This
allows a whole tree to be deleted at once, without having to
keep a pointer to all the children in the user code. A kludge
has been done so the Fl_Tabs and all of its children
can be automatic (local) variables, but you must declare the
Fl_Tabs widget <I>first</I> so that it is destroyed last.
*/
Fl_Tabs::Fl_Tabs(int X, int Y, int W, int H, const char *L) :
Fl_Group(X,Y,W,H,L)
{
box(FL_THIN_UP_BOX);
push_ = 0;
overflow_type = OVERFLOW_COMPRESS;
tab_offset = 0;
tab_pos = 0;
tab_width = 0;
tab_flags = NULL;
tab_count = 0;
tab_align_ = FL_ALIGN_CENTER;
has_overflow_menu = 0;
}
/**
Delete allocated resources and destroy all children.
*/
Fl_Tabs::~Fl_Tabs() {
clear_tab_positions();
}
/**
Returns the position and size available to be used by its children.
If there isn't any child yet the \p tabh parameter will be used to
calculate the return values. This assumes that the children's labelsize
is the same as the Fl_Tabs' labelsize and adds a small border.
If there are already children, the values of child(0) are returned, and
\p tabh is ignored.
\note Children should always use the same positions and sizes.
\p tabh can be one of
\li 0: calculate label size, tabs on top
\li -1: calculate label size, tabs on bottom
\li > 0: use given \p tabh value, tabs on top (height = tabh)
\li < -1: use given \p tabh value, tabs on bottom (height = -tabh)
\param[in] tabh position and optional height of tabs (see above)
\param[out] rx,ry,rw,rh (x,y,w,h) of client area for children
\since FLTK 1.3.0
*/
void Fl_Tabs::client_area(int &rx, int &ry, int &rw, int &rh, int tabh) {
if (children()) { // use existing values
rx = child(0)->x();
ry = child(0)->y();
rw = child(0)->w();
rh = child(0)->h();
} else { // calculate values
int y_offset;
int label_height = fl_height(labelfont(), labelsize()) + BORDER*2;
if (tabh == 0) // use default (at top)
y_offset = label_height;
else if (tabh == -1) // use default (at bottom)
y_offset = -label_height;
else
y_offset = tabh; // user given value
rx = x();
rw = w();
if (y_offset >= 0) { // labels at top
ry = y() + y_offset;
rh = h() - y_offset;
} else { // labels at bottom
ry = y();
rh = h() + y_offset;
}
}
}
/**
Clear internal array of tab positions and widths.
\see tab_positions().
*/
void Fl_Tabs::clear_tab_positions() {
if (tab_pos) {
free(tab_pos);
tab_pos = 0;
}
if (tab_width){
free(tab_width);
tab_width = 0;
}
if (tab_flags){
free(tab_flags);
tab_flags = NULL;
}
}
/** Set a method to handle an overflowing tab bar.
The Fl_Tabs widget allows you to specify how to handle the situation where
there are more tabs than can be displayed at once. The available options are:
- \c OVERFLOW_COMPRESS: Tabs will be compressed and overlaid on top of each other.
- \c OVERFLOW_CLIP: Only the first tabs that fit will be displayed.
- \c OVERFLOW_PULLDOWN: Tabs that do not fit will be placed in a pull-down menu.
- \c OVERFLOW_DRAG: The tab bar can be dragged horizontally to reveal additional tabs.
You can set the desired behavior using the overflow() method.
\param ov overflow type
\see OVERFLOW_COMPRESS, OVERFLOW_CLIP, OVERFLOW_PULLDOWN, OVERFLOW_DRAG
*/
void Fl_Tabs::handle_overflow(int ov) {
overflow_type = ov;
tab_offset = 0;
has_overflow_menu = 0;
damage(FL_DAMAGE_EXPOSE);
redraw();
}