| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412 |
- // $Id: Flu_Wrap_Group.cpp,v 1.8 2004/01/27 21:44:24 jbryan Exp $
- /***************************************************************
- * FLU - FLTK Utility Widgets
- * Copyright (C) 2002 Ohio Supercomputer Center, Ohio State University
- *
- * This file and its content is protected by a software license.
- * You should have received a copy of this license with this file.
- * If not, please contact the Ohio Supercomputer Center immediately:
- * Attn: Jason Bryan Re: FLU 1224 Kinnear Rd, Columbus, Ohio 43212
- *
- ***************************************************************/
- #include "FLU/Flu_Wrap_Group.H"
- #include <string.h>
- #include <stdio.h>
- #define MAX( x, y ) ( (x)>(y) ? (x) : (y) )
- #define SCROLL_SIZE 15
- Flu_Wrap_Group :: Scrollbar :: Scrollbar( int x, int y, int w, int h, const char *l )
- : Fl_Scrollbar( x, y, w, h, l )
- {
- }
- int Flu_Wrap_Group :: Scrollbar :: handle( int event )
- {
- if( event == FL_MOUSEWHEEL )
- {
- handle_drag( clamp( value() + linesize() * Fl::e_dy ) );
- return 1;
- }
- else
- return Fl_Scrollbar::handle( event );
- }
- Flu_Wrap_Group :: Flu_Wrap_Group( int x, int y, int w, int h, const char *l )
- : Fl_Group( x, y, w, h, l ), scrollbar( x+w-SCROLL_SIZE, y, SCROLL_SIZE, h ), group( x, y, w-SCROLL_SIZE, h )
- {
- offset( 0, 0 );
- spacing( 0, 0 );
- _type = FL_VERTICAL;
- scrollTo = NULL;
- Fl_Group::add( &scrollbar );
- scrollbar.callback( _scrollCB, this );
- scrollbar.linesize( 10 );
- scrollbar.range( 0, 100 );
- scrollbar.show();
- Fl_Group::add( &group );
- Fl_Group::resizable( group );
- Fl_Group::end();
- group.begin();
- }
- void Flu_Wrap_Group :: resize( int x, int y, int w, int h )
- {
- group.resizable( NULL );
- Fl_Group::resize( x, y, w, h );
- if( type() == FL_VERTICAL )
- {
- scrollbar.resize( x+w-SCROLL_SIZE-Fl::box_dx(box()), y+Fl::box_dy(box()), SCROLL_SIZE, h-Fl::box_dh(box()) );
- group.resize( x, y, w-SCROLL_SIZE-Fl::box_dx(box()), h );
- }
- else
- {
- scrollbar.resize( x+Fl::box_dx(box()), y+h-SCROLL_SIZE-Fl::box_dy(box()), w-Fl::box_dw(box()), SCROLL_SIZE );
- group.resize( x, y, w, h-SCROLL_SIZE-Fl::box_dh(box()) );
- }
- Fl_Group::init_sizes();
- redraw();
- }
- void Flu_Wrap_Group :: scroll_to( const Fl_Widget *w )
- {
- scrollTo = w;
- redraw();
- }
- void Flu_Wrap_Group :: scroll_to_beginning()
- {
- ((Fl_Valuator*)&scrollbar)->value( scrollbar.minimum() );
- }
- void Flu_Wrap_Group :: scroll_to_end()
- {
- ((Fl_Valuator*)&scrollbar)->value( scrollbar.maximum() );
- }
- void Flu_Wrap_Group :: type( int t )
- {
- _type = t;
- resize( x(), y(), w(), h() );
- }
- Fl_Widget* Flu_Wrap_Group :: next( Fl_Widget* w )
- {
- for( int i = 0; i < group.children()-1; i++ )
- {
- if( w == group.child(i) )
- return group.child(i+1);
- }
- return NULL;
- }
- Fl_Widget* Flu_Wrap_Group :: previous( Fl_Widget* w )
- {
- for( int i = 1; i < group.children(); i++ )
- {
- if( w == group.child(i) )
- return group.child(i-1);
- }
- return NULL;
- }
- Fl_Widget* Flu_Wrap_Group :: above( Fl_Widget* w )
- {
- for( int i = 0; i < group.children(); i++ )
- {
- if( w == group.child(i) )
- {
- int measure[2];
- measure[0] = w->x() + w->w()/2;
- measure[1] = w->y() - _spacing[1];
- int index = layout( scrollbar.visible(), false, measure );
- if( index >= 0 )
- return group.child(index);
- else
- return group.child(0);
- }
- }
- return NULL;
- }
- Fl_Widget* Flu_Wrap_Group :: below( Fl_Widget* w )
- {
- for( int i = 0; i < group.children(); i++ )
- {
- if( w == group.child(i) )
- {
- int measure[2];
- measure[0] = w->x() + w->w()/2;
- measure[1] = w->y() + w->h() + _spacing[1];
- int index = layout( scrollbar.visible(), false, measure );
- if( index >= 0 )
- return group.child(index);
- else
- return group.child(group.children()-1);
- }
- }
- return NULL;
- }
- Fl_Widget* Flu_Wrap_Group :: left( Fl_Widget* w )
- {
- for( int i = 0; i < group.children(); i++ )
- {
- if( w == group.child(i) )
- {
- int measure[2];
- measure[0] = w->x() - _spacing[0];
- measure[1] = w->y() + w->h()/2;
- int index = layout( scrollbar.visible(), false, measure );
- if( index >= 0 )
- return group.child(index);
- else
- return group.child(0);
- }
- }
- return NULL;
- }
- Fl_Widget* Flu_Wrap_Group :: right( Fl_Widget* w )
- {
- for( int i = 0; i < group.children(); i++ )
- {
- if( w == group.child(i) )
- {
- int measure[2];
- measure[0] = w->x() + w->w() + _spacing[0] + 1;
- measure[1] = w->y() + w->h()/2;
- int index = layout( scrollbar.visible(), false, measure );
- if( index >= 0 )
- return group.child(index);
- else
- return group.child(group.children()-1);
- }
- }
- return NULL;
- }
- int Flu_Wrap_Group :: layout( bool sbVisible, bool doScrollTo, int *measure )
- {
- int xx = x()+Fl::box_dx(box()), yy = y()+Fl::box_dy(box()),
- ww = w()-Fl::box_dw(box()), hh = h()-Fl::box_dh(box());
- if( type() == FL_VERTICAL )
- {
- int i, X, Y, maxH, H, col, row, maxW, scrollY;
- Fl_Widget *c;
- scrollbar.type( FL_VERTICAL );
- BEGIN_H:
- X = xx+_offset[0];
- Y = yy+_offset[1] - (sbVisible ? scrollbar.value() : 0);
- maxH = _offset[1];
- H = 0;
- col = 0;
- row = 0;
- scrollY = 0;
- maxW = xx + ww - (sbVisible ? scrollbar.w() : 0);
- for( i = 0; i < group.children(); i++ )
- {
- c = group.child(i);
- if( !c->visible() )
- continue;
- H = MAX( H, c->h() );
- if( col == 0 )
- maxH += H + _spacing[1];
- if( ( X + c->w() ) > maxW )
- {
- Y += H + _spacing[1];
- scrollY += H + _spacing[1];
- if( i == group.children()-1 )
- maxH += H + _spacing[1];
- if( measure )
- {
- if( xx+_offset[0] <= measure[0] && measure[0] <= xx+c->w()+_offset[0]+_spacing[0] &&
- Y <= measure[1] && measure[1] <= Y+c->h()+_spacing[1] )
- return i;
- }
- else
- c->position( xx+_offset[0], Y );
- col = 0;
- row++;
- H = 0;
- X = xx+c->w() + _offset[0] + _spacing[0];
- }
- else
- {
- if( measure )
- {
- if( X <= measure[0] && measure[0] <= X+c->w()+_spacing[0] &&
- Y <= measure[1] && measure[1] <= Y+c->h()+_spacing[1] )
- return i;
- }
- else
- c->position( X, Y );
- X += c->w() + _spacing[0];
- col++;
- }
- if( doScrollTo && (c == scrollTo) )
- {
- if( scrollY > scrollbar.maximum() )
- scrollY = (int)scrollbar.maximum();
- ((Fl_Valuator*)&scrollbar)->value( scrollY );
- scrollTo = NULL;
- goto BEGIN_H;
- }
- // if we exceed the height and the scrollbar is not visible,
- // then it will soon become visible so we don't need to process anymore
- if( !measure && !sbVisible && maxH > hh )
- return 1;
- }
- if( measure )
- return -1;
- else if( maxH > hh )
- {
- scrollbar.range( 0, maxH-hh );
- scrollbar.slider_size( MAX( float(scrollbar.h()-(maxH-hh))/float(scrollbar.h()), 0.08f ) );
- return 1;
- }
- else
- return 0;
- }
- else
- {
- int i, X, Y, W, maxW, maxH, col, row, scrollX;
- Fl_Widget *c;
- scrollbar.type( FL_HORIZONTAL );
- BEGIN_W:
- X = xx+_offset[0] - (sbVisible ? scrollbar.value() : 0);
- Y = yy+_offset[1];
- maxW = _offset[0];
- W = 0;
- col = 0;
- row = 0;
- scrollX = 0;
- maxH = yy + hh - (sbVisible ? scrollbar.h() : 0);
- for( i = 0; i < group.children(); i++ )
- {
- c = group.child(i);
- if( !c->visible() )
- continue;
- W = MAX( W, c->w() );
- if( row == 0 )
- maxW += W + _spacing[0];
- if( ( Y + c->h() ) > maxH )
- {
- X += W + _spacing[0];
- scrollX += W + _spacing[0];
- if( i == group.children()-1 )
- maxW += W + _spacing[0];
- if( measure )
- {
- if( X <= measure[0] && measure[0] <= X+c->w()+_spacing[0] &&
- yy+_offset[1] <= measure[1] && measure[1] <= yy+c->h()+_offset[1]+_spacing[1] )
- return i;
- }
- else
- c->position( X, yy+_offset[1] );
- row = 0;
- col++;
- W = 0;
- Y = yy+c->h() + _offset[1] + _spacing[1];
- }
- else
- {
- if( measure )
- {
- if( X <= measure[0] && measure[0] <= X+c->w()+_spacing[0] &&
- Y <= measure[1] && measure[1] <= Y+c->h()+_spacing[1] )
- return i;
- }
- else
- c->position( X, Y );
- Y += c->h() + _spacing[1];
- row++;
- }
- if( doScrollTo && (c == scrollTo) )
- {
- if( scrollX > scrollbar.maximum() )
- scrollX = (int)scrollbar.maximum();
- ((Fl_Valuator*)&scrollbar)->value( scrollX );
- scrollTo = NULL;
- goto BEGIN_W;
- }
- // if we exceed the width and the scrollbar is not visible,
- // then it will soon become visible so we don't need to process anymore
- if( !measure && !sbVisible && maxW > ww )
- return 1;
- }
- if( measure )
- return -1;
- else if( maxW > ww )
- {
- scrollbar.range( 0, maxW-ww );
- scrollbar.slider_size( MAX( float(scrollbar.w()-(maxW-ww))/float(scrollbar.w()), 0.08f ) );
- return 1;
- }
- else
- return 0;
- }
- }
- void Flu_Wrap_Group :: draw()
- {
- // we first try to fit all children assuming no scrollbar. if they do not all fit,
- // we have to turn the scrollbar on and try again
- if( layout( false, false ) )
- {
- scrollbar.show();
- layout( true, false );
- }
- else
- scrollbar.hide();
- // hack to look right when resizing smaller
- if( scrollbar.value() > scrollbar.maximum() )
- {
- ((Fl_Valuator*)&scrollbar)->value( scrollbar.maximum() );
- layout( scrollbar.visible(), scrollTo!=NULL );
- }
- else if( scrollTo )
- layout( scrollbar.visible(), true );
- scrollTo = NULL;
- if( damage() & ~FL_DAMAGE_CHILD)
- {
- draw_box();
- draw_label();
- }
- fl_push_clip( x()+Fl::box_dx(box()), y()+Fl::box_dy(box()),
- w()-Fl::box_dw(box()), h()-Fl::box_dh(box()) );
- draw_children();
- fl_pop_clip();
- }
|