2013-11-10 08:33:44 +04:00
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// Demonstrate Fl_Tree custom item draw callback. - erco 11/09/2013
|
2013-11-10 08:33:44 +04:00
|
|
|
//
|
|
|
|
// Copyright 2013 Greg Ercolano.
|
2016-11-19 04:20:53 +03:00
|
|
|
// Copyright 1998-2016 by Bill Spitzak and others.
|
2013-11-10 08:33:44 +04:00
|
|
|
//
|
|
|
|
// 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:
|
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// https://www.fltk.org/COPYING.php
|
2013-11-10 08:33:44 +04:00
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// Please see the following page on how to report bugs and issues:
|
2013-11-10 08:33:44 +04:00
|
|
|
//
|
2020-07-01 19:03:10 +03:00
|
|
|
// https://www.fltk.org/bugs.php
|
2013-11-10 08:33:44 +04:00
|
|
|
//
|
|
|
|
#include <stdio.h>
|
2020-07-01 19:03:10 +03:00
|
|
|
#include <time.h> /* ctime.. */
|
2013-11-10 08:33:44 +04:00
|
|
|
#include <FL/Fl.H>
|
|
|
|
#include <FL/Fl_Double_Window.H>
|
|
|
|
#include <FL/Fl_Tree.H>
|
|
|
|
|
2014-01-21 01:23:24 +04:00
|
|
|
#ifndef MAX
|
|
|
|
#define MAX(a,b) ((a)>(b))?(a):(b)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// DERIVE CUSTOM CLASS FROM Fl_Tree_Item TO IMPLEMENT SHOWING THE TIME OF DAY
|
|
|
|
// This demonstrates that item content can be dynamic and highly customized.
|
|
|
|
//
|
|
|
|
class MyTimeItem : public Fl_Tree_Item {
|
|
|
|
const char *time_format;
|
|
|
|
protected:
|
|
|
|
// Remove trailing crlf
|
|
|
|
const char* StripCrlf(char *s)
|
|
|
|
{ char *ss = strchr(s, '\n'); if (ss) *ss = 0; return s; }
|
|
|
|
const struct tm* GetTimeStruct() {
|
|
|
|
time_t t = time(NULL);
|
|
|
|
if ( strcmp(time_format, "Local") == 0 ) return localtime(&t);
|
|
|
|
if ( strcmp(time_format, "GMT" ) == 0 ) return gmtime(&t);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
public:
|
|
|
|
MyTimeItem(Fl_Tree *tree, const char *time_format) : Fl_Tree_Item(tree) {
|
|
|
|
label(time_format);
|
|
|
|
this->time_format = time_format;
|
|
|
|
}
|
|
|
|
// Handle custom drawing of the item
|
|
|
|
// Fl_Tree has already handled drawing everything to the left
|
|
|
|
// of the label area, including any 'user icon', collapse buttons,
|
|
|
|
// connector lines, etc.
|
|
|
|
//
|
|
|
|
// All we're responsible for is drawing the 'label' area of the item
|
|
|
|
// and it's background. Fl_Tree gives us a hint as to what the
|
|
|
|
// foreground and background colors should be via the fg/bg parameters,
|
|
|
|
// and whether we're supposed to render anything or not.
|
|
|
|
//
|
|
|
|
// The only other thing we must do is return the maximum X position
|
|
|
|
// of scrollable content, i.e. the right most X position of content
|
|
|
|
// that we want the user to be able to use the horizontal scrollbar
|
|
|
|
// to reach.
|
|
|
|
//
|
|
|
|
int draw_item_content(int render) {
|
|
|
|
Fl_Color fg = drawfgcolor();
|
|
|
|
Fl_Color bg = drawbgcolor();
|
|
|
|
// Show the date and time as two small strings
|
|
|
|
// one on top of the other in a single item.
|
|
|
|
//
|
|
|
|
// Our item's label dimensions
|
|
|
|
int X = label_x(), Y = label_y(),
|
2020-07-01 19:03:10 +03:00
|
|
|
W = label_w(), H = label_h();
|
2014-01-21 01:23:24 +04:00
|
|
|
// Render background
|
|
|
|
if ( render ) {
|
2020-07-01 19:03:10 +03:00
|
|
|
if ( is_selected() ) { // Selected? Use selectbox() style
|
2014-01-21 01:23:24 +04:00
|
|
|
fl_draw_box(prefs().selectbox(),X,Y,W,H,bg);
|
2020-07-01 19:03:10 +03:00
|
|
|
} else { // Not Selected? use plain filled rectangle
|
2014-01-21 01:23:24 +04:00
|
|
|
fl_color(bg); fl_rectf(X,Y,W,H);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Render the label
|
|
|
|
if ( render ) {
|
|
|
|
fl_color(fg);
|
|
|
|
if ( label() ) fl_draw(label(), X,Y,W,H, FL_ALIGN_LEFT);
|
|
|
|
}
|
|
|
|
int lw=0, lh=0;
|
|
|
|
if ( label() ) {
|
|
|
|
lw=0; lh=0; fl_measure(label(), lw, lh);
|
|
|
|
}
|
|
|
|
X += lw + 8;
|
|
|
|
// Draw some red/grn/blu boxes
|
|
|
|
if ( render ) {
|
|
|
|
fl_color(FL_RED); fl_rectf(X+0, Y+2, 10, H-4);
|
|
|
|
fl_color(FL_GREEN); fl_rectf(X+10, Y+2, 10, H-4);
|
|
|
|
fl_color(FL_BLUE); fl_rectf(X+20, Y+2, 10, H-4);
|
|
|
|
}
|
|
|
|
X += 35;
|
|
|
|
// Render the date and time, one over the other
|
2020-07-01 19:03:10 +03:00
|
|
|
fl_font(labelfont(), 8); // small font
|
2019-04-16 20:56:33 +03:00
|
|
|
fl_color(fg);
|
2014-01-21 01:23:24 +04:00
|
|
|
const struct tm *tm = GetTimeStruct();
|
|
|
|
char s[80];
|
|
|
|
sprintf(s, "Date: %02d/%02d/%02d", tm->tm_mon+1, tm->tm_mday, tm->tm_year % 100);
|
2020-07-01 19:03:10 +03:00
|
|
|
lw=0, lh=0; fl_measure(s, lw, lh); // get box around text (including white space)
|
2014-01-21 01:23:24 +04:00
|
|
|
if ( render ) fl_draw(s, X,Y+4,W,H, FL_ALIGN_LEFT|FL_ALIGN_TOP);
|
|
|
|
sprintf(s, "Time: %02d:%02d:%02d", tm->tm_hour, tm->tm_min, tm->tm_sec);
|
|
|
|
if ( render ) fl_draw(s, X,Y+H/2,W,H/2, FL_ALIGN_LEFT|FL_ALIGN_TOP);
|
|
|
|
int lw2=0, lh2=0; fl_measure(s, lw2, lh2);
|
|
|
|
X += MAX(lw, lw2);
|
2020-07-01 19:03:10 +03:00
|
|
|
return X; // return right most edge of what we've rendered
|
2013-11-10 08:33:44 +04:00
|
|
|
}
|
2014-01-21 01:23:24 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// TIMER TO HANDLE DYNAMIC CONTENT IN THE TREE
|
|
|
|
void Timer_CB(void *data) {
|
|
|
|
Fl_Tree *tree = (Fl_Tree*)data;
|
2020-07-01 19:03:10 +03:00
|
|
|
tree->redraw(); // keeps time updated
|
2014-01-21 01:23:24 +04:00
|
|
|
Fl::repeat_timeout(0.2, Timer_CB, data);
|
2013-11-10 08:33:44 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[]) {
|
|
|
|
Fl::scheme("gtk+");
|
2018-03-21 17:57:15 +03:00
|
|
|
Fl_Double_Window *win = new Fl_Double_Window(350, 400, "Tree Custom Draw Items");
|
2013-11-10 08:33:44 +04:00
|
|
|
win->begin();
|
|
|
|
{
|
|
|
|
// Create the tree
|
|
|
|
Fl_Tree *tree = new Fl_Tree(0, 0, win->w(), win->h());
|
2020-07-01 19:03:10 +03:00
|
|
|
tree->showroot(0); // don't show root of tree
|
|
|
|
tree->selectmode(FL_TREE_SELECT_MULTI); // multiselect
|
2013-11-10 08:33:44 +04:00
|
|
|
|
|
|
|
// Add some items
|
|
|
|
tree->add("Flintstones/Fred");
|
|
|
|
tree->add("Flintstones/Wilma");
|
|
|
|
tree->add("Flintstones/Pebbles");
|
2014-01-21 01:23:24 +04:00
|
|
|
{
|
|
|
|
MyTimeItem *myitem;
|
2020-07-01 19:03:10 +03:00
|
|
|
myitem = new MyTimeItem(tree, "Local"); // create custom item
|
2014-01-21 01:23:24 +04:00
|
|
|
myitem->labelsize(20);
|
|
|
|
tree->add("Time Add Item/Local", myitem);
|
|
|
|
|
2020-07-01 19:03:10 +03:00
|
|
|
myitem = new MyTimeItem(tree, "GMT"); // create custom item
|
2014-01-21 01:23:24 +04:00
|
|
|
myitem->labelsize(20);
|
|
|
|
tree->add("Time Add Item/GMT", myitem);
|
|
|
|
}
|
|
|
|
// 'Replace' approach
|
|
|
|
{
|
|
|
|
Fl_Tree_Item *item;
|
|
|
|
MyTimeItem *myitem;
|
|
|
|
item = tree->add("Time Replace Item/Local Time");
|
|
|
|
// Replace the 'Local' item with our own
|
2020-07-01 19:03:10 +03:00
|
|
|
myitem = new MyTimeItem(tree, "Local"); // create custom item
|
2014-01-21 01:23:24 +04:00
|
|
|
myitem->labelsize(20);
|
2020-07-01 19:03:10 +03:00
|
|
|
item->replace(myitem); // replace normal item with custom
|
2014-01-21 01:23:24 +04:00
|
|
|
|
|
|
|
item = tree->add("Time Replace Item/GMT Time");
|
|
|
|
// Replace the 'GMT' item with our own
|
2020-07-01 19:03:10 +03:00
|
|
|
myitem = new MyTimeItem(tree, "GMT"); // create custom item
|
2014-01-21 01:23:24 +04:00
|
|
|
myitem->labelsize(20);
|
2020-07-01 19:03:10 +03:00
|
|
|
item->replace(myitem); // replace normal item with custom
|
2014-01-21 01:23:24 +04:00
|
|
|
}
|
2013-11-10 08:33:44 +04:00
|
|
|
tree->add("Superjail/Warden");
|
|
|
|
tree->add("Superjail/Jared");
|
|
|
|
tree->add("Superjail/Alice");
|
|
|
|
tree->add("Superjail/Jailbot");
|
|
|
|
|
2014-01-21 01:23:24 +04:00
|
|
|
tree->show_self();
|
|
|
|
|
2013-11-10 08:33:44 +04:00
|
|
|
// Start with some items closed
|
|
|
|
tree->close("Superjail");
|
2014-01-21 01:23:24 +04:00
|
|
|
|
|
|
|
// Set up a timer to keep time in tree updated
|
|
|
|
Fl::add_timeout(0.2, Timer_CB, (void*)tree);
|
2013-11-10 08:33:44 +04:00
|
|
|
}
|
|
|
|
win->end();
|
|
|
|
win->resizable(win);
|
|
|
|
win->show(argc, argv);
|
|
|
|
return(Fl::run());
|
|
|
|
}
|