Blame examples/table-spreadsheet-with-keyboard-nav.cxx

Packit 328d5c
//
Packit 328d5c
// "$Id: table-spreadsheet-with-keyboard-nav.cxx 9838 2013-03-18 20:00:04Z greg.ercolano $"
Packit 328d5c
//
Packit 328d5c
//	Simple example of an interactive spreadsheet using Fl_Table.
Packit 328d5c
//	Uses Mr. Satan's technique of instancing an Fl_Input around.
Packit 328d5c
//	Modified to test Jean-Marc's mods for keyboard nav and mouse selection.
Packit 328d5c
//
Packit 328d5c
//      Fl_Table[1.00/LGPL] 04/18/03 Mister Satan      -- Initial implementation, submitted to erco for Fl_Table
Packit 328d5c
//      Fl_Table[1.10/LGPL] 05/17/03 Greg Ercolano     -- Small mods to follow changes to Fl_Table
Packit 328d5c
//      Fl_Table[1.20/LGPL] 02/22/04 Jean-Marc Lienher -- Keyboard nav and mouse selection
Packit 328d5c
//      Fl_Table[1.21/LGPL] 02/22/04 Greg Ercolano     -- Small reformatting mods, comments
Packit 328d5c
//         FLTK[1.3.0/LGPL] 10/26/10 Greg Ercolano     -- Moved from Fl_Table to FLTK 1.3.x, CMP compliance
Packit 328d5c
//
Packit 328d5c
// Copyright 1998-2010 by Bill Spitzak and others.
Packit 328d5c
//
Packit 328d5c
// This library is free software. Distribution and use rights are outlined in
Packit 328d5c
// the file "COPYING" which should have been included with this file.  If this
Packit 328d5c
// file is missing or damaged, see the license at:
Packit 328d5c
//
Packit 328d5c
//     http://www.fltk.org/COPYING.php
Packit 328d5c
//
Packit 328d5c
// Please report all bugs and problems on the following page:
Packit 328d5c
//
Packit 328d5c
//     http://www.fltk.org/str.php
Packit 328d5c
//
Packit 328d5c
#include <stdio.h>
Packit 328d5c
#include <stdlib.h>
Packit 328d5c
#include <FL/Fl.H>
Packit 328d5c
#include <FL/Fl_Double_Window.H>
Packit 328d5c
#include <FL/Fl_Table.H>
Packit 328d5c
#include <FL/Fl_Int_Input.H>
Packit 328d5c
#include <FL/Fl_Value_Slider.H>
Packit 328d5c
#include <FL/fl_draw.H>
Packit 328d5c
Packit 328d5c
const int MAX_COLS = 26;
Packit 328d5c
const int MAX_ROWS = 500;
Packit 328d5c
Packit 328d5c
class Spreadsheet : public Fl_Table {
Packit 328d5c
  Fl_Int_Input *input;					// single instance of Fl_Int_Input widget
Packit 328d5c
  int values[MAX_ROWS][MAX_COLS];			// array of data for cells
Packit 328d5c
  int row_edit, col_edit;				// row/col being modified
Packit 328d5c
  int s_left, s_top, s_right, s_bottom;			// kb nav + mouse selection
Packit 328d5c
Packit 328d5c
protected:
Packit 328d5c
  void draw_cell(TableContext context,int=0,int=0,int=0,int=0,int=0,int=0);
Packit 328d5c
  void event_callback2();				// table's event callback (instance)
Packit 328d5c
  static void event_callback(Fl_Widget*, void *v) {	// table's event callback (static)
Packit 328d5c
    ((Spreadsheet*)v)->event_callback2();
Packit 328d5c
  }
Packit 328d5c
  static void input_cb(Fl_Widget*, void* v) {		// input widget's callback
Packit 328d5c
    ((Spreadsheet*)v)->set_value_hide();
Packit 328d5c
  }
Packit 328d5c
Packit 328d5c
public:
Packit 328d5c
  Spreadsheet(int X,int Y,int W,int H,const char* L=0) : Fl_Table(X,Y,W,H,L) {
Packit 328d5c
    callback(&event_callback, (void*)this);
Packit 328d5c
    when(FL_WHEN_NOT_CHANGED|when());
Packit 328d5c
    // Create input widget that we'll use whenever user clicks on a cell
Packit 328d5c
    input = new Fl_Int_Input(W/2,H/2,0,0);
Packit 328d5c
    input->hide();
Packit 328d5c
    input->callback(input_cb, (void*)this);
Packit 328d5c
    input->when(FL_WHEN_ENTER_KEY_ALWAYS);		// callback triggered when user hits Enter
Packit 328d5c
    input->maximum_size(5);
Packit 328d5c
    row_edit = col_edit = 0;
Packit 328d5c
    s_left = s_top = s_right = s_bottom = 0;
Packit 328d5c
    for (int c = 0; c < MAX_COLS; c++)
Packit 328d5c
      for (int r = 0; r < MAX_ROWS; r++)
Packit 328d5c
	values[r][c] = (r + 2) * (c + 3);		// initialize cells
Packit 328d5c
    end();
Packit 328d5c
  }
Packit 328d5c
  ~Spreadsheet() { }
Packit 328d5c
Packit 328d5c
  // Apply value from input widget to values[row][col] array and hide (done editing)
Packit 328d5c
  void set_value_hide() {
Packit 328d5c
    values[row_edit][col_edit] = atoi(input->value());
Packit 328d5c
    input->hide();
Packit 328d5c
    window()->cursor(FL_CURSOR_DEFAULT);		// XXX: if we don't do this, cursor can disappear!
Packit 328d5c
  }
Packit 328d5c
  // Change number of rows
Packit 328d5c
  void rows(int val) {
Packit 328d5c
    Fl_Table::rows(val);
Packit 328d5c
  }
Packit 328d5c
  // Change number of columns
Packit 328d5c
  void cols(int val) {
Packit 328d5c
    Fl_Table::cols(val);
Packit 328d5c
  }
Packit 328d5c
  // Get number of rows
Packit 328d5c
  inline int rows() {
Packit 328d5c
    return Fl_Table::rows();
Packit 328d5c
  }
Packit 328d5c
  // Get number of columns
Packit 328d5c
  inline int cols() {
Packit 328d5c
    return Fl_Table::cols();
Packit 328d5c
  }
Packit 328d5c
  // Start editing a new cell: move the Fl_Int_Input widget to specified row/column
Packit 328d5c
  //    Preload the widget with the cell's current value,
Packit 328d5c
  //    and make the widget 'appear' at the cell's location.
Packit 328d5c
  //
Packit 328d5c
  void start_editing(int R, int C) {
Packit 328d5c
    row_edit = R;					// Now editing this row/col
Packit 328d5c
    col_edit = C;
Packit 328d5c
    int X,Y,W,H;
Packit 328d5c
    find_cell(CONTEXT_CELL, R,C, X,Y,W,H);		// Find X/Y/W/H of cell
Packit 328d5c
    input->resize(X,Y,W,H);				// Move Fl_Input widget there
Packit 328d5c
    char s[30]; sprintf(s, "%d", values[R][C]);		// Load input widget with cell's current value
Packit 328d5c
    input->value(s);
Packit 328d5c
    input->position(0,strlen(s));			// Select entire input field
Packit 328d5c
    input->show();					// Show the input widget, now that we've positioned it
Packit 328d5c
    input->take_focus();
Packit 328d5c
  }
Packit 328d5c
  // Tell the input widget it's done editing, and to 'hide'
Packit 328d5c
  void done_editing() {
Packit 328d5c
    if (input->visible()) {				// input widget visible, ie. edit in progress?
Packit 328d5c
      set_value_hide();					// Transfer its current contents to cell and hide
Packit 328d5c
    }
Packit 328d5c
  }
Packit 328d5c
  // Return the sum of all rows in this column
Packit 328d5c
  int sum_rows(int C) {
Packit 328d5c
    int sum = 0;
Packit 328d5c
    for (int r=0; r
Packit 328d5c
      sum += values[r][C];
Packit 328d5c
    return(sum);
Packit 328d5c
  }
Packit 328d5c
  // Return the sum of all cols in this row
Packit 328d5c
  int sum_cols(int R) {
Packit 328d5c
    int sum = 0;
Packit 328d5c
    for (int c=0; c
Packit 328d5c
      sum += values[R][c];
Packit 328d5c
    return(sum);
Packit 328d5c
  }
Packit 328d5c
  // Return the sum of all cells in table
Packit 328d5c
  int sum_all() {
Packit 328d5c
    int sum = 0;
Packit 328d5c
    for (int c=0; c
Packit 328d5c
      for (int r=0; r
Packit 328d5c
	sum += values[r][c];
Packit 328d5c
    return(sum);
Packit 328d5c
  }
Packit 328d5c
};
Packit 328d5c
Packit 328d5c
// Handle drawing all cells in table
Packit 328d5c
void Spreadsheet::draw_cell(TableContext context, int R,int C, int X,int Y,int W,int H) {
Packit 328d5c
  static char s[30]; 
Packit 328d5c
  switch ( context ) {
Packit 328d5c
    case CONTEXT_STARTPAGE:			// table about to redraw
Packit 328d5c
      // Get kb nav + mouse 'selection region' for use below
Packit 328d5c
      get_selection(s_top, s_left, s_bottom, s_right);
Packit 328d5c
      break;
Packit 328d5c
Packit 328d5c
    case CONTEXT_COL_HEADER:			// table wants us to draw a column heading (C is column)
Packit 328d5c
      fl_font(FL_HELVETICA | FL_BOLD, 14);	// set font for heading to bold
Packit 328d5c
      fl_push_clip(X,Y,W,H);			// clip region for text
Packit 328d5c
      {
Packit 328d5c
	fl_draw_box(FL_THIN_UP_BOX, X,Y,W,H, col_header_color());
Packit 328d5c
	fl_color(FL_BLACK);
Packit 328d5c
	if (C == cols()-1) {			// Last column? show 'TOTAL'
Packit 328d5c
	  fl_draw("TOTAL", X,Y,W,H, FL_ALIGN_CENTER);
Packit 328d5c
	} else {				// Not last column? show column letter
Packit 328d5c
	  sprintf(s, "%c", 'A' + C);
Packit 328d5c
	  fl_draw(s, X,Y,W,H, FL_ALIGN_CENTER);
Packit 328d5c
	}
Packit 328d5c
      }
Packit 328d5c
      fl_pop_clip();
Packit 328d5c
      return;
Packit 328d5c
Packit 328d5c
    case CONTEXT_ROW_HEADER:			// table wants us to draw a row heading (R is row)
Packit 328d5c
      fl_font(FL_HELVETICA | FL_BOLD, 14);	// set font for row heading to bold
Packit 328d5c
      fl_push_clip(X,Y,W,H);
Packit 328d5c
      {
Packit 328d5c
	fl_draw_box(FL_THIN_UP_BOX, X,Y,W,H, row_header_color());
Packit 328d5c
	fl_color(FL_BLACK);
Packit 328d5c
	if (R == rows()-1) {			// Last row? Show 'Total'
Packit 328d5c
	  fl_draw("TOTAL", X,Y,W,H, FL_ALIGN_CENTER);
Packit 328d5c
	} else {				// Not last row? show row#
Packit 328d5c
	  sprintf(s, "%d", R+1);
Packit 328d5c
	  fl_draw(s, X,Y,W,H, FL_ALIGN_CENTER);
Packit 328d5c
	}
Packit 328d5c
      }
Packit 328d5c
      fl_pop_clip();
Packit 328d5c
      return;
Packit 328d5c
Packit 328d5c
    case CONTEXT_CELL: {			// table wants us to draw a cell
Packit 328d5c
      if (R == row_edit && C == col_edit && input->visible()) {
Packit 328d5c
	return;					// dont draw for cell with input widget over it
Packit 328d5c
      }
Packit 328d5c
      // Background
Packit 328d5c
      // Keyboard nav and mouse selection highlighting
Packit 328d5c
      if (R >= s_top && R <= s_bottom && C >= s_left && C <= s_right) {
Packit 328d5c
	fl_draw_box(FL_THIN_UP_BOX, X,Y,W,H, FL_YELLOW);
Packit 328d5c
      } else if ( C < cols()-1 && R < rows()-1 ) {
Packit 328d5c
	fl_draw_box(FL_THIN_UP_BOX, X,Y,W,H, FL_WHITE);
Packit 328d5c
      } else {
Packit 328d5c
	fl_draw_box(FL_THIN_UP_BOX, X,Y,W,H, 0xbbddbb00);	// money green
Packit 328d5c
      }
Packit 328d5c
      // Text
Packit 328d5c
      fl_push_clip(X+3, Y+3, W-6, H-6);
Packit 328d5c
      {
Packit 328d5c
	fl_color(FL_BLACK); 
Packit 328d5c
	if (C == cols()-1 || R == rows()-1) {	// Last row or col? Show total
Packit 328d5c
	  fl_font(FL_HELVETICA | FL_BOLD, 14);	// ..in bold font
Packit 328d5c
	  if (C == cols()-1 && R == rows()-1) {	// Last row+col? Total all cells
Packit 328d5c
	    sprintf(s, "%d", sum_all());
Packit 328d5c
	  } else if (C == cols()-1) {		// Row subtotal
Packit 328d5c
	    sprintf(s, "%d", sum_cols(R));
Packit 328d5c
	  } else if (R == rows()-1) {		// Col subtotal
Packit 328d5c
	    sprintf(s, "%d", sum_rows(C));
Packit 328d5c
	  }
Packit 328d5c
	  fl_draw(s, X+3,Y+3,W-6,H-6, FL_ALIGN_RIGHT);
Packit 328d5c
	} else {				// Not last row or col? Show cell contents
Packit 328d5c
	  fl_font(FL_HELVETICA, 14);		// ..in regular font
Packit 328d5c
	  sprintf(s, "%d", values[R][C]);
Packit 328d5c
	  fl_draw(s, X+3,Y+3,W-6,H-6, FL_ALIGN_RIGHT);
Packit 328d5c
	}
Packit 328d5c
      }
Packit 328d5c
      fl_pop_clip();
Packit 328d5c
      return;
Packit 328d5c
    }
Packit 328d5c
Packit 328d5c
    case CONTEXT_RC_RESIZE: {			// table resizing rows or columns
Packit 328d5c
      if (!input->visible()) return;
Packit 328d5c
      find_cell(CONTEXT_TABLE, row_edit, col_edit, X, Y, W, H);
Packit 328d5c
      if (X==input->x() && Y==input->y() && W==input->w() && H==input->h()) {
Packit 328d5c
	return;					// no change? ignore
Packit 328d5c
      }
Packit 328d5c
      input->resize(X,Y,W,H);
Packit 328d5c
      return;
Packit 328d5c
    }
Packit 328d5c
Packit 328d5c
    default:
Packit 328d5c
      return;
Packit 328d5c
  }
Packit 328d5c
}
Packit 328d5c
Packit 328d5c
// Callback whenever someone clicks on different parts of the table
Packit 328d5c
void Spreadsheet::event_callback2() {
Packit 328d5c
  int R = callback_row();
Packit 328d5c
  int C = callback_col();
Packit 328d5c
  TableContext context = callback_context(); 
Packit 328d5c
Packit 328d5c
  switch ( context ) {
Packit 328d5c
    case CONTEXT_CELL: {				// A table event occurred on a cell
Packit 328d5c
      switch (Fl::event()) { 				// see what FLTK event caused it
Packit 328d5c
	case FL_PUSH:					// mouse click?
Packit 328d5c
	  done_editing();				// finish editing previous
Packit 328d5c
	  if (R != rows()-1 && C != cols()-1 )		// only edit cells not in total's columns
Packit 328d5c
	    start_editing(R,C);				// start new edit
Packit 328d5c
	  return;
Packit 328d5c
Packit 328d5c
	case FL_KEYBOARD:				// key press in table?
Packit 328d5c
	  if ( Fl::event_key() == FL_Escape ) exit(0);	// ESC closes app
Packit 328d5c
	  if (C == cols()-1 || R == rows()-1) return;	// no editing of totals column
Packit 328d5c
	  done_editing();				// finish any previous editing
Packit 328d5c
	  set_selection(R, C, R, C);			// select the current cell
Packit 328d5c
	  start_editing(R,C);				// start new edit
Packit 328d5c
	  if (Fl::event() == FL_KEYBOARD && Fl::e_text[0] != '\r') {
Packit 328d5c
	    input->handle(Fl::event());			// pass keypress to input widget
Packit 328d5c
	  }
Packit 328d5c
	  return;
Packit 328d5c
      }
Packit 328d5c
      return;
Packit 328d5c
    }
Packit 328d5c
Packit 328d5c
    case CONTEXT_TABLE:					// A table event occurred on dead zone in table
Packit 328d5c
    case CONTEXT_ROW_HEADER:				// A table event occurred on row/column header
Packit 328d5c
    case CONTEXT_COL_HEADER:
Packit 328d5c
      done_editing();					// done editing, hide
Packit 328d5c
      return;
Packit 328d5c
    
Packit 328d5c
    default:
Packit 328d5c
      return;
Packit 328d5c
  }
Packit 328d5c
}
Packit 328d5c
Packit 328d5c
// Change number of columns
Packit 328d5c
void setcols_cb(Fl_Widget* w, void* v) {
Packit 328d5c
  Spreadsheet* table = (Spreadsheet*)v;
Packit 328d5c
  Fl_Valuator* in = (Fl_Valuator*)w;
Packit 328d5c
  int cols = int(in->value()) + 1;
Packit 328d5c
  table->cols(cols);
Packit 328d5c
  table->redraw();
Packit 328d5c
}
Packit 328d5c
Packit 328d5c
// Change number of rows
Packit 328d5c
void setrows_cb(Fl_Widget* w, void* v) {
Packit 328d5c
  Spreadsheet* table = (Spreadsheet*)v;
Packit 328d5c
  Fl_Valuator* in = (Fl_Valuator*)w;
Packit 328d5c
  int rows = int(in->value()) + 1;
Packit 328d5c
  table->rows(rows);
Packit 328d5c
  table->redraw();
Packit 328d5c
}
Packit 328d5c
Packit 328d5c
int main() {
Packit 328d5c
  Fl::option(Fl::OPTION_ARROW_FOCUS, 1);		// we want arrow keys to navigate table's widgets
Packit 328d5c
  Fl_Double_Window *win = new Fl_Double_Window(922, 382, "Fl_Table Spreadsheet with Keyboard Navigation");
Packit 328d5c
  Spreadsheet* table = new Spreadsheet(20, 20, win->w()-80, win->h()-80);
Packit 328d5c
  // Table rows
Packit 328d5c
  table->row_header(1);
Packit 328d5c
  table->row_header_width(70);
Packit 328d5c
  table->row_resize(1);
Packit 328d5c
  table->rows(11);
Packit 328d5c
  table->row_height_all(25);
Packit 328d5c
  // Table cols
Packit 328d5c
  table->col_header(1);
Packit 328d5c
  table->col_header_height(25);
Packit 328d5c
  table->col_resize(1);
Packit 328d5c
  table->cols(11);
Packit 328d5c
  table->col_width_all(70);
Packit 328d5c
  table->set_selection(0,0,0,0);	// select top/left cell
Packit 328d5c
Packit 328d5c
  // Add children to window
Packit 328d5c
  win->begin();
Packit 328d5c
Packit 328d5c
  // Row slider
Packit 328d5c
  Fl_Value_Slider setrows(win->w()-40,20,20,win->h()-80, 0);
Packit 328d5c
  setrows.type(FL_VERT_NICE_SLIDER);
Packit 328d5c
  setrows.bounds(2,MAX_ROWS);
Packit 328d5c
  setrows.step(1);
Packit 328d5c
  setrows.value(table->rows()-1);
Packit 328d5c
  setrows.callback(setrows_cb, (void*)table);
Packit 328d5c
  setrows.when(FL_WHEN_CHANGED);
Packit 328d5c
  setrows.clear_visible_focus();
Packit 328d5c
Packit 328d5c
  // Column slider
Packit 328d5c
  Fl_Value_Slider setcols(20,win->h()-40,win->w()-80,20, 0);
Packit 328d5c
  setcols.type(FL_HOR_NICE_SLIDER);
Packit 328d5c
  setcols.bounds(2,MAX_COLS);
Packit 328d5c
  setcols.step(1);
Packit 328d5c
  setcols.value(table->cols()-1);
Packit 328d5c
  setcols.callback(setcols_cb, (void*)table);
Packit 328d5c
  setcols.when(FL_WHEN_CHANGED);
Packit 328d5c
  setcols.clear_visible_focus();
Packit 328d5c
Packit 328d5c
  win->end();
Packit 328d5c
  win->resizable(table);
Packit 328d5c
  win->show();
Packit 328d5c
Packit 328d5c
  return Fl::run();
Packit 328d5c
}
Packit 328d5c
Packit 328d5c
//
Packit 328d5c
// End of "$Id: table-spreadsheet-with-keyboard-nav.cxx 9838 2013-03-18 20:00:04Z greg.ercolano $".
Packit 328d5c
//