2007-09-05 23:51:01 +04:00
|
|
|
/*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Application for setting system date, time and local timezone
|
|
|
|
* Part of Equinox Desktop Environment (EDE).
|
|
|
|
* Copyright (c) 2000-2007 EDE Authors.
|
|
|
|
*
|
|
|
|
* This program is licenced under terms of the
|
|
|
|
* GNU General Public Licence version 2 or newer.
|
|
|
|
* See COPYING for details.
|
|
|
|
*/
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// This is the Calendar widget class inspired by the Calendar originally
|
|
|
|
// developed by Alexey Parshin for SPTK and later imported into efltk.
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
#include "EDE_Calendar.h"
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
#include <FL/Fl.H>
|
|
|
|
#include <FL/fl_draw.H>
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
#include <edelib/Nls.h>
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// TODO: replace this with a simple loop when operator++ is implemented in edelib::Date
|
|
|
|
long date_distance(edelib::Date da1, edelib::Date da2) {
|
|
|
|
if (da1 > da2) {
|
|
|
|
edelib::Date tmp = da2;
|
|
|
|
da2=da1;
|
|
|
|
da1=tmp;
|
|
|
|
}
|
|
|
|
int d1=da1.day(), m1=da1.month(), y1=da1.year(), d2=da2.day(), m2=da2.month(), y2=da2.year();
|
|
|
|
long result=0;
|
|
|
|
|
|
|
|
while (d1!=d2 || m1!=m2 || y1!=y2) {
|
|
|
|
result++;
|
|
|
|
d1++;
|
|
|
|
if (!da1.is_valid(y1,m1,d1)) {
|
|
|
|
d1=1;
|
|
|
|
m1++;
|
|
|
|
if (m1>12) {
|
|
|
|
m1=1;
|
|
|
|
y1++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Constants
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
static const char *switchLabels[4] = {
|
2007-09-05 23:51:01 +04:00
|
|
|
"@-1<<","@-1<","@-1>","@-1>>"
|
2006-08-20 22:43:09 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
static const int monthChanges[4] = {
|
2007-09-05 23:51:01 +04:00
|
|
|
-12,-1,1,12
|
2006-08-20 22:43:09 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
// TODO: read this from locale
|
|
|
|
const bool weekStartsOnMonday = false;
|
|
|
|
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Callback function for fwd / back buttons
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
void EDE_Calendar::cbSwitchButtonClicked(Fl_Widget *button, long month_change) {
|
|
|
|
EDE_Calendar* thecalendar = (EDE_Calendar*)button->parent();
|
|
|
|
edelib::Date d = thecalendar->active_date();
|
|
|
|
int year = d.year();
|
|
|
|
int month = d.month()+month_change;
|
|
|
|
int day = d.day();
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Fix month
|
|
|
|
while (month<1) { year--; month+=12; }
|
|
|
|
while (month>12) { year++; month-=12; }
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Switch to closest valid day
|
|
|
|
while (day>27 && !d.is_valid(year, month, day)) day--;
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
if (d.is_valid(year, month, day)) {
|
|
|
|
d.set(year, month, day);
|
|
|
|
thecalendar->active_date(d);
|
|
|
|
}
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// ctor
|
2006-08-20 22:43:09 +04:00
|
|
|
|
|
|
|
EDE_Calendar::EDE_Calendar(int x,int y,int w,int h,const char *lbl)
|
2007-09-05 23:51:01 +04:00
|
|
|
: Fl_Group(x,y,w,h,lbl) {
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
// Calendar contents, correct size and position is set by layout()
|
|
|
|
|
|
|
|
// Header box
|
|
|
|
m_monthNameBox = new Fl_Box(x,y,w,16);
|
|
|
|
m_monthNameBox->box(FL_NO_BOX);
|
|
|
|
|
|
|
|
// Weekday headers
|
|
|
|
for (i = 0; i < 7; i++) {
|
|
|
|
m_dayNameBoxes[i] = new Fl_Box(x+i*16,y+16,16,16);
|
|
|
|
m_dayNameBoxes[i]->box(FL_FLAT_BOX);
|
|
|
|
m_dayNameBoxes[i]->color(fl_color_average(color(), FL_GREEN, 0.8));
|
|
|
|
|
|
|
|
// get first two letters of day name
|
|
|
|
edelib::Date d;
|
|
|
|
d.set(1900,1,7+i); // 1.1.1900 was Monday
|
|
|
|
char tmp[3];
|
|
|
|
snprintf(tmp,3,"%s", d.day_name());
|
|
|
|
m_dayNameBoxes[i]->copy_label(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fillers (parts of calendar without day buttons)
|
|
|
|
for (int i=0; i<3; i++) {
|
|
|
|
m_filler[i] = new Fl_Box(x,y,16,16);
|
|
|
|
m_filler[i]->box(FL_FLAT_BOX);
|
|
|
|
m_filler[i]->color(fl_color_average(color(), FL_BLACK, 0.95)); // very mild grayish
|
|
|
|
}
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Day buttons
|
|
|
|
for (i = 0; i < 31; i++) {
|
|
|
|
m_dayButtons[i] = new Fl_Box(0,0,16,16);
|
|
|
|
char tmp[3];
|
|
|
|
snprintf(tmp,3, "%d", (i+1));
|
|
|
|
m_dayButtons[i]->copy_label(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Switch buttons
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
Fl_Repeat_Button* o;
|
|
|
|
m_switchButtons[i] = o = new Fl_Repeat_Button(x,y,16,16,switchLabels[i]);
|
|
|
|
o->callback(EDE_Calendar::cbSwitchButtonClicked, (long)monthChanges[i]);
|
|
|
|
o->labelcolor(fl_darker(FL_BACKGROUND_COLOR));
|
|
|
|
o->selection_color(fl_lighter(FL_BACKGROUND_COLOR));
|
|
|
|
}
|
|
|
|
|
|
|
|
end();
|
|
|
|
|
|
|
|
reset(); // this will eventually call layout()
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Number of rows and columns
|
|
|
|
|
|
|
|
#define CAL_ROWS 9
|
|
|
|
#define CAL_COLS 7
|
|
|
|
|
|
|
|
|
|
|
|
// Calculate positions and sizes of various boxes and buttons
|
|
|
|
// NOTE: This method is costly! Avoid calling it unless calendar is actually resized
|
|
|
|
void EDE_Calendar::layout(int X, int Y, int W, int H) {
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
// Leave some room for edges
|
|
|
|
int x_ = X+2;
|
|
|
|
int y_ = Y+2;
|
|
|
|
int w_ = W-4;
|
|
|
|
int h_ = H-4;
|
|
|
|
|
|
|
|
// Precalculate button grid
|
|
|
|
// By doing this we try to avoid floating point math while
|
|
|
|
// having coords without holes that add up to a desired width
|
|
|
|
int bx[CAL_COLS], bw[CAL_COLS], by[CAL_ROWS], bh[CAL_ROWS];
|
|
|
|
bx[0] = x_; by[0] = y_;
|
|
|
|
for (i=1; i<CAL_COLS; i++) {
|
|
|
|
bx[i] = x_ + (w_*i)/CAL_COLS;
|
|
|
|
bw[i-1] = bx[i]-bx[i-1];
|
|
|
|
}
|
|
|
|
bw[CAL_COLS-1] = x_ + w_ - bx[CAL_COLS-1];
|
|
|
|
for (i=1; i<CAL_ROWS; i++) {
|
|
|
|
by[i] = y_ + (h_*i)/CAL_ROWS;
|
|
|
|
bh[i-1] = by[i]-by[i-1];
|
|
|
|
}
|
|
|
|
bh[CAL_ROWS-1] = y_ + h_ - by[CAL_ROWS-1];
|
|
|
|
|
|
|
|
|
|
|
|
// Title
|
|
|
|
m_monthNameBox->resize( x_,y_,w_, bh[0] );
|
|
|
|
|
|
|
|
// Labels
|
|
|
|
for (i=0; i<7; i++)
|
|
|
|
m_dayNameBoxes[i]->resize( bx[i], by[1], bw[i], bh[1] );
|
|
|
|
|
|
|
|
// Find first day of week
|
|
|
|
edelib::Date d=active_date_;
|
|
|
|
d.set(d.year(), d.month(), 1);
|
|
|
|
uint day_of_week = d.day_of_week()-1;
|
|
|
|
|
|
|
|
// First filler
|
|
|
|
if (day_of_week>0) {
|
|
|
|
int k=0;
|
|
|
|
for (i=0; i<day_of_week; i++) k += bw[i];
|
|
|
|
m_filler[0]->resize( x_, by[2], k, bh[2] );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Days
|
|
|
|
int row=2;
|
|
|
|
for (i=0; i<d.days_in_month(); i++) {
|
|
|
|
m_dayButtons[i]->resize( bx[day_of_week], by[row], bw[day_of_week], bh[row] );
|
|
|
|
day_of_week++;
|
|
|
|
if (day_of_week==7) { day_of_week=0; row++; }
|
|
|
|
}
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Second filler
|
|
|
|
if (day_of_week<6) {
|
|
|
|
int k=0;
|
|
|
|
for (i=day_of_week; i<7; i++) k += bw[i];
|
|
|
|
m_filler[1]->resize( bx[day_of_week], by[row], k, bh[row] );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Third filler
|
|
|
|
if (row<7)
|
|
|
|
m_filler[2]->resize( x_, by[7], w_, bh[7] );
|
|
|
|
|
|
|
|
// Switch buttons
|
|
|
|
m_switchButtons[0]->resize ( x_, by[8], bw[0], bh[8] );
|
|
|
|
m_switchButtons[1]->resize ( bx[1], by[8], bw[1], bh[8] );
|
|
|
|
m_switchButtons[2]->resize ( bx[5], by[8], bw[5], bh[8] );
|
|
|
|
m_switchButtons[3]->resize ( bx[6], by[8], bw[6], bh[8] );
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Overloaded resize (used to call layout)
|
2006-08-20 22:43:09 +04:00
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
void EDE_Calendar::resize(int X, int Y, int W, int H) {
|
|
|
|
// avoid unnecessary resizing
|
|
|
|
static int oldw=0, oldh=0;
|
|
|
|
if (W==oldw && H==oldh) return Fl_Group::resize(X,Y,W,H);
|
|
|
|
oldw=W; oldh=H;
|
|
|
|
|
|
|
|
layout(X,Y,W,H);
|
|
|
|
Fl_Widget::resize(X,Y,W,H); // avoid Fl_Group resizing because we already resized everything
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
// Set visual appearance & colors of day buttons and tooltip
|
|
|
|
|
|
|
|
void EDE_Calendar::update_calendar() {
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
// Find first day of week
|
|
|
|
edelib::Date d=active_date_;
|
|
|
|
d.set(d.year(), d.month(), 1);
|
|
|
|
int day_of_week = d.day_of_week()-1;
|
|
|
|
|
|
|
|
// Show/hide first filler
|
|
|
|
if (day_of_week>0)
|
|
|
|
m_filler[0]->show();
|
|
|
|
else
|
|
|
|
m_filler[0]->hide();
|
|
|
|
|
|
|
|
// Days
|
|
|
|
int row=2;
|
|
|
|
for (i=0; i<d.days_in_month(); i++) {
|
|
|
|
Fl_Box* btn = m_dayButtons[i]; // shortcut
|
|
|
|
btn->show();
|
|
|
|
|
|
|
|
// Set button color
|
|
|
|
Fl_Color daycolor = color(); // base color is the color of calendar
|
|
|
|
|
|
|
|
if (day_of_week==0) // Special color for sunday
|
|
|
|
daycolor = fl_color_average(daycolor, FL_BLUE, 0.8);
|
|
|
|
|
|
|
|
if (i==(uint)today_date_.day()-1 && d.month()==today_date_.month() && d.year()==today_date_.year())
|
|
|
|
btn->color(fl_color_average(daycolor, FL_RED, 0.5)); // today
|
|
|
|
else if (i==(uint)active_date_.day()-1)
|
|
|
|
btn->color(fl_lighter(daycolor));
|
|
|
|
else
|
|
|
|
btn->color(daycolor);
|
|
|
|
|
|
|
|
// Set downbox for active day
|
|
|
|
if (i==(uint)active_date_.day()-1)
|
|
|
|
btn->box(FL_DOWN_BOX);
|
|
|
|
else
|
|
|
|
btn->box(FL_FLAT_BOX);
|
|
|
|
|
|
|
|
day_of_week++;
|
|
|
|
if (day_of_week==7) { day_of_week=0; row++; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hide remaining buttons
|
|
|
|
for (i=d.days_in_month(); i<31; i++)
|
|
|
|
m_dayButtons[i]->hide();
|
|
|
|
|
|
|
|
// Show/hide second filler
|
|
|
|
if (day_of_week<6)
|
|
|
|
m_filler[1]->show();
|
|
|
|
else
|
|
|
|
m_filler[1]->hide();
|
|
|
|
|
|
|
|
// Show/hide third filler
|
|
|
|
if (row<7)
|
|
|
|
m_filler[2]->show();
|
|
|
|
else
|
|
|
|
m_filler[2]->hide();
|
|
|
|
|
|
|
|
// Set title
|
|
|
|
static char title[30]; // No month name should be larger than 24 chars, even when localized
|
|
|
|
// and we can't show it anyway cause the box is too small
|
|
|
|
snprintf (title, 30, "%s, %d", d.month_name(), d.year());
|
|
|
|
m_monthNameBox->copy_label(title);
|
|
|
|
|
|
|
|
|
|
|
|
// Calculate tooltip (distance between today and active date)
|
|
|
|
static char tooltip_str[1024];
|
|
|
|
tooltip_str[0] = '\0';
|
|
|
|
|
|
|
|
if (today_date_ != active_date_) {
|
|
|
|
long dist = date_distance(today_date_, active_date_);
|
|
|
|
long weeks = dist/7;
|
|
|
|
int wdays = dist%7;
|
|
|
|
int months=0;
|
|
|
|
int mdays=0;
|
|
|
|
int years=0;
|
|
|
|
int ymonths=0;
|
|
|
|
|
|
|
|
// Find lower date, first part of tooltip
|
|
|
|
edelib::Date d1,d2;
|
|
|
|
if (today_date_ < active_date_) {
|
|
|
|
d1=today_date_;
|
|
|
|
d2=active_date_;
|
|
|
|
snprintf(tooltip_str, 1023, _("In %ld days"), dist);
|
|
|
|
} else {
|
|
|
|
d2=today_date_;
|
|
|
|
d1=active_date_;
|
|
|
|
snprintf(tooltip_str, 1023, _("%ld days ago"), dist);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If necessary, calculate distance in m/d and y/m/d format
|
|
|
|
if (dist>30) {
|
|
|
|
months = d2.month() - d1.month() + (d2.year() - d1.year())*12;
|
|
|
|
mdays = d2.day() - d1.day();
|
|
|
|
if (mdays<1) {
|
|
|
|
mdays += d2.days_in_month();
|
|
|
|
months--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (months>11) {
|
|
|
|
years = months/12;
|
|
|
|
ymonths = months%12;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append those strings using snprintf
|
|
|
|
if (weeks) {
|
|
|
|
char* tmp = strdup(tooltip_str);
|
|
|
|
snprintf(tooltip_str, 1023, _("%s\n%ld weeks and %d days"), tmp, weeks, wdays);
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (months) {
|
|
|
|
char* tmp = strdup(tooltip_str);
|
|
|
|
snprintf(tooltip_str, 1023, _("%s\n%d months and %d days"), tmp, months, mdays);
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (years) {
|
|
|
|
char* tmp = strdup(tooltip_str);
|
|
|
|
snprintf(tooltip_str, 1023, _("%s\n%d years, %d months and %d days"), tmp, years, ymonths, mdays);
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tooltip(tooltip_str);
|
|
|
|
|
|
|
|
layout(x(),y(),w(),h()); // relayout buttons if neccessary
|
|
|
|
redraw();
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-05 23:51:01 +04:00
|
|
|
int EDE_Calendar::handle(int e) {
|
|
|
|
// Forward events to switch buttons
|
|
|
|
for (int i=0; i<4; i++)
|
|
|
|
if (Fl::event_inside(m_switchButtons[i])) return m_switchButtons[i]->handle(e);
|
|
|
|
|
|
|
|
// Accept focus
|
|
|
|
if (e==FL_FOCUS) return 1;
|
|
|
|
|
|
|
|
if (e==FL_PUSH || e==FL_ENTER) return 1;
|
|
|
|
|
|
|
|
// Click on date to set active, doubleclick to set as today
|
|
|
|
if (e==FL_RELEASE) {
|
|
|
|
for (int i=0; i<31; i++)
|
|
|
|
if (Fl::event_inside(m_dayButtons[i])) {
|
|
|
|
edelib::Date d = active_date_;
|
|
|
|
d.set(d.year(), d.month(), i+1);
|
|
|
|
if (Fl::event_clicks() == 1)
|
|
|
|
today_date(d);
|
|
|
|
else
|
|
|
|
active_date(d);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
take_focus();
|
|
|
|
|
|
|
|
} else if (e==FL_KEYBOARD) {
|
|
|
|
int k=Fl::event_key();
|
|
|
|
|
|
|
|
// arrow navigation
|
|
|
|
if (k==FL_Up || k==FL_Down || k==FL_Left || k==FL_Right) {
|
|
|
|
edelib::Date ad = active_date_;
|
|
|
|
int d = ad.day();
|
|
|
|
int m = ad.month();
|
|
|
|
int y = ad.year();
|
|
|
|
|
|
|
|
if (k==FL_Up) d -= 7;
|
|
|
|
else if (k==FL_Down) d += 7;
|
|
|
|
else if (k==FL_Left) d--;
|
|
|
|
else if (k==FL_Right) d++;
|
|
|
|
|
|
|
|
if (d < 1) {
|
|
|
|
m--;
|
|
|
|
if (m<1) {
|
|
|
|
y--;
|
|
|
|
m+=12;
|
|
|
|
}
|
|
|
|
d += ad.days_in_month(y,m);
|
|
|
|
}
|
|
|
|
if (d > ad.days_in_month(y,m)) {
|
|
|
|
d -= ad.days_in_month(y,m);
|
|
|
|
m++;
|
|
|
|
if (m>12) {
|
|
|
|
y++;
|
|
|
|
m-=12;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ad.set(y,m,d);
|
|
|
|
active_date(ad);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return to today with Home
|
|
|
|
if (k==FL_Home) {
|
|
|
|
active_date(today_date());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set active day as today with Space
|
|
|
|
if (k==' ') {
|
|
|
|
today_date(active_date());
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allow moving focus with Tab key
|
|
|
|
if (k==FL_Tab) return Fl_Group::handle(e);
|
|
|
|
}
|
|
|
|
return Fl_Group::handle(e);
|
2006-08-20 22:43:09 +04:00
|
|
|
}
|