]> Pileus Git - lackey/blob - src/view.c
Add event selection to day and week views
[lackey] / src / view.c
1 /*
2  * Copyright (C) 2012-2013 Andy Spencer <andy753421@gmail.com>
3  *
4  * This program is free software: you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation, either version 3 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
16  */
17
18 #define _XOPEN_SOURCE_EXTENDED
19
20 #include <string.h>
21 #include <ncurses.h>
22
23 #include "util.h"
24 #include "conf.h"
25 #include "date.h"
26 #include "cal.h"
27 #include "view.h"
28
29 /* Types */
30 typedef struct {
31         const char *name;
32         const char *title;
33         void      (*init)(WINDOW*);
34         void      (*size)(int,int);
35         void      (*draw)(void);
36         int       (*run)(int,mmask_t,int,int);
37         int         keys[8];
38         WINDOW     *win;
39 } view_t;
40
41 /* Macros */
42 #define VIEW(name, title, ...)                \
43         void name##_init(WINDOW *win);        \
44         void name##_size(int,int);            \
45         void name##_draw(void);               \
46         int  name##_run(int,mmask_t,int,int); \
47         view_t name##_view = {                \
48                 #name,                        \
49                 title,                        \
50                 name##_init,                  \
51                 name##_size,                  \
52                 name##_draw,                  \
53                 name##_run,                   \
54                 { __VA_ARGS__ }               \
55         }
56
57 /* Views */
58 VIEW(day,      "Day",      KEY_F(1), '1');
59 VIEW(week,     "Week",     KEY_F(2), '2');
60 VIEW(month,    "Month",    KEY_F(3), '3');
61 VIEW(year,     "Year",     KEY_F(4), '4');
62 VIEW(events,   "Events",   KEY_F(5), '5');
63 VIEW(todo,     "Todo",     KEY_F(6), '6');
64 VIEW(settings, "Settings", KEY_F(7), '7');
65 VIEW(help,     "Help",     KEY_F(8), '8');
66 VIEW(edit,     "Edit");
67
68 /* View data */
69 view_t  spacer = { "|", "|" };
70
71 view_t *views[] = {
72         &day_view, &week_view, &month_view, &year_view,
73         &events_view, &todo_view,
74         &settings_view, &help_view,
75         &edit_view
76 };
77
78 view_t *menu[] = {
79         &day_view, &week_view, &month_view, &year_view,
80         &spacer, &events_view, &todo_view,
81         &spacer, &settings_view, &help_view
82 };
83
84 /* Config data */
85 int COMPACT = 0;
86 int MORNING = 8;
87
88 /* Global data */
89 edit_t EDIT = EDIT_NONE;
90
91 /* Local data */
92 view_t *view   = &day_view;
93 view_t *active = &day_view;
94 view_t *popup  = NULL;
95
96 /* Local functions */
97 static void draw_header(void)
98 {
99         move(0, 0);
100         attron(COLOR_PAIR(COLOR_TITLE));
101         clrtoeol();
102
103         /* Draw menu */
104         for (int i = 0; i < N_ELEMENTS(menu); i++) {
105                 if (menu[i] == active)
106                         attron(A_BOLD);
107                 printw("%s ", menu[i]->title);
108                 if (menu[i] == active)
109                         attroff(A_BOLD);
110         }
111
112         /* Draw popup window */
113         if (popup) {
114                 printw("| ");
115                 attron(A_BOLD);
116                 printw("[%s]", popup->title);
117                 attroff(A_BOLD);
118         }
119
120         /* Draw date */
121         move(0, COLS-19);
122         printw("%04d-%02d-%02d %02d:%02d:%02d",
123                         NOW.year, NOW.month+1, NOW.day+1,
124                         NOW.hour, NOW.min,     NOW.sec);
125
126         attroff(COLOR_PAIR(COLOR_TITLE));
127         if (!COMPACT)
128                 mvhline(1, 0, ACS_HLINE, COLS);
129         refresh();
130 }
131
132 static int get_color(const char *cat)
133 {
134         return cat == NULL           ? 0           :
135                match(cat, "class") ? COLOR_CLASS :
136                match(cat, "ec")    ? COLOR_EC    :
137                match(cat, "work")  ? COLOR_WORK  : COLOR_OTHER ;
138 }
139
140 static int set_view(view_t *_active, view_t *_popup)
141 {
142         view = _popup ?: _active;
143         if (active != _active) {
144                 active = _active;
145                 set_string("view", 0, "active", active->name);
146                 view_draw();
147         }
148         if (popup != _popup) {
149                 popup = _popup;
150                 view_draw();
151         }
152         return 1;
153 }
154
155 /* Curses functions */
156 void wmvresize(WINDOW *win, int top, int left, int rows, int cols)
157 {
158         int y = getpary(win);
159         if (top < y)
160                 mvderwin(win, top, left);
161         wresize(win, rows, cols);
162         if (top > y)
163                 mvderwin(win, top, left);
164 }
165
166 void wshrink(WINDOW *win, int top)
167 {
168         int x    = getparx(win);
169         int y    = getpary(win);
170         int r    = getmaxy(win);
171         int c    = getmaxx(win);
172         int rows = r + (y - top);
173         if (top  <  y) mvderwin(win, top, x);
174         if (rows != r) wresize(win, rows, c);
175         if (top  >  y) mvderwin(win, top, x);
176 }
177
178 /* Helper functions */
179 void event_box(WINDOW *win, event_t *event, int y, int x, int h, int w)
180 {
181         int i, l = 0;
182         int s = y < 0 ? -y-1 : 0;
183
184         int color = get_color(event->cat);
185
186         if (color) wattron(win, COLOR_PAIR(color));
187
188         if (h >= 2) mvwhline_set(win, y,     x+1,   WACS_T_HLINE, w-2);
189         if (h <= 1) mvwadd_wch(win,   y,     x,     WACS_BULLET);
190         if (h >= 2) mvwadd_wch(win,   y,     x,     WACS_T_ULCORNER);
191         if (h >= 2) mvwadd_wch(win,   y,     x+w-1, WACS_T_URCORNER);
192         if (h >= 3) mvwvline_set(win, y+1+s, x,     WACS_T_VLINE, h-2-s);
193         if (h >= 3) mvwvline_set(win, y+1+s, x+w-1, WACS_T_VLINE, h-2-s);
194         if (h >= 2) mvwadd_wch(win,   y+h-1, x,     WACS_T_LLCORNER);
195         if (h >= 2) mvwadd_wch(win,   y+h-1, x+w-1, WACS_T_LRCORNER);
196         if (h >= 2) mvwhline_set(win, y+h-1, x+1,   WACS_T_HLINE, w-2);
197
198         for (i = 1; i < h-1; i++)
199                 mvwhline(win, y+i, x+1, ' ', w-2);
200
201         if (color) wattroff(win, COLOR_PAIR(color));
202
203         if (event == EVENT)     wattron(win, WA_BOLD | WA_REVERSE);
204         if (event == EVENT)     mvwhline(win, y+s, x, ' ', w);
205         if (l<h && event->name) mvwprintw(win, y+l++, x+1, "%.*s",   w-2, event->name);
206         if (event == EVENT)     wattroff(win, WA_REVERSE);
207         if (l<h && event->loc)  mvwprintw(win, y+l++, x+1, "@ %.*s", w-4, event->loc);
208         if (l<h && event->desc) mvwprintw(win, y+l++, x+1, "%.*s",   w-2, event->desc);
209         if (event == EVENT)     wattroff(win, WA_BOLD);
210 }
211
212 void event_line(WINDOW *win, event_t *event, int y, int x, int w, int flags)
213 {
214         int color = get_color(event->cat);
215
216         if (color) wattron(win, COLOR_PAIR(color));
217         mvwaddch(win, y, x++, ACS_BLOCK);
218         if (color) wattroff(win, COLOR_PAIR(color));
219
220         if (flags & SHOW_ACTIVE && event == EVENT)
221                 wattron(win, A_REVERSE | A_BOLD);
222         if (flags & SHOW_DETAILS) {
223                 if (all_day(&event->start, &event->end))
224                         mvwprintw(win, y, x+1, "[all day]   - ");
225                 else
226                         mvwprintw(win, y, x+1, "%2d:%02d-%2d:%02d - ",
227                                         event->start.hour, event->start.min,
228                                         event->end.hour,   event->end.min);
229                 x += 15;
230                 w -= 15;
231         }
232         if (event->name) {
233                 const char *label = event->name ?: event->desc;
234                 mvwprintw(win, y, x, "%-*.*s", w-1, w-1, label);
235                 x += MIN(strlen(label), w-1);
236         }
237         if (flags & SHOW_DETAILS && event->loc) {
238                 mvwprintw(win, y, x, " @ %s", event->loc);
239         }
240         if (flags & SHOW_ACTIVE && event == EVENT)
241                 wattroff(win, A_REVERSE | A_BOLD);
242 }
243
244 void todo_line(WINDOW *win, todo_t *todo, int y, int x, int w, int flags)
245 {
246         char perc[16];
247         char desc[LINES];
248         sprintf(perc, "%2d%%", todo->status);
249
250         int cat    = get_color(todo->cat);
251         int status = todo->status == NEW  ? COLOR_NEW  :
252                      todo->status == DONE ? COLOR_DONE : COLOR_WIP;
253
254         sprintf(desc, "%s", todo->name ?: todo->desc ?: "");
255         strsub(desc, '\n', ';');
256
257         /* Print category */
258         if (cat) wattron(win, COLOR_PAIR(cat));
259         mvwaddch(win, y, x, ACS_BLOCK);
260         if (cat) wattroff(win, COLOR_PAIR(cat));
261         x += 2;
262
263         /* Set background */
264         if (flags & SHOW_ACTIVE && todo == TODO)
265                 wattron(win, A_REVERSE | A_BOLD);
266         mvwhline(win, y, x, ' ', COLS-x);
267
268         /* Print time */
269         if (no_date(&todo->due))
270                 mvwprintw(win, y, x, "[no due date]");
271         else
272                 mvwprintw(win, y, x, "%04d-%02d-%02d %2d:%02d",
273                                 todo->due.year, todo->due.month+1, todo->due.day+1,
274                                 todo->due.hour, todo->due.min);
275         x += 18;
276
277         /* Print status */
278         if (status) wattron(win, COLOR_PAIR(status));
279         mvwprintw(win, y, x, "%s",
280                 todo->status == NEW    ? "new"  :
281                 todo->status == DONE   ? "done" : perc);
282         if (status) wattroff(win, COLOR_PAIR(status));
283         x += 6;
284
285         /* Print description */
286         mvwprintw(win, y, x, "%s", desc);
287
288         /* Reset flags */
289         if (flags & SHOW_ACTIVE && todo == TODO)
290                 wattroff(win, A_REVERSE | A_BOLD);
291 }
292
293 /* View init */
294 void view_init(void)
295 {
296         int hdr = COMPACT ? 1 : 2;
297         for (int i = 0; i < N_ELEMENTS(views); i++) {
298                 views[i]->win = newwin(LINES-hdr, COLS, hdr, 0);
299                 views[i]->init(views[i]->win);
300                 views[i]->size(LINES-hdr, COLS);
301         }
302 }
303
304 /* Config parser */
305 void view_config(const char *group, const char *name, const char *key, const char *value)
306 {
307         if (match(group, "view")) {
308                 if (match(key, "compact")) {
309                         COMPACT = get_bool(value);
310                 } else if (match(key, "morning")) {
311                         MORNING = get_number(value);
312                 } else if (match(key, "active")) {
313                         for (int i = 0; i < N_ELEMENTS(views); i++) {
314                                 if (match(value, views[i]->name)) {
315                                         get_string(value);
316                                         view = active = views[i];
317                                         break;
318                                 }
319                         }
320                 }
321         }
322 }
323
324 /* View draw */
325 void view_resize(void)
326 {
327         int hdr = COMPACT ? 1 : 2;
328         for (int i = 0; i < N_ELEMENTS(views); i++) {
329                 wresize(views[i]->win, LINES-hdr, COLS);
330                 mvwin(views[i]->win, hdr, 0);
331                 views[i]->size(LINES-hdr, COLS);
332         }
333 }
334
335 /* View draw */
336 void view_draw(void)
337 {
338         draw_header();
339         werase(view->win);
340         view->draw();
341         wrefresh(view->win);
342 }
343
344 /* View run */
345 int view_run(int key, mmask_t btn, int row, int col)
346 {
347         /* Refresh timestamp */
348         draw_header();
349
350         /* Check for mouse events on the menu */
351         if (key == KEY_MOUSE && row == 0) {
352                 int start = 1;
353                 for (int i = 0; i < N_ELEMENTS(menu); i++) {
354                         int end = start + strlen(menu[i]->name) - 1;
355                         if (start <= col && col <= end && menu[i]->draw)
356                                 return set_view(menu[i], NULL);
357                         start = end + 2;
358                 }
359         }
360
361         /* Look though menu for hotkeys */
362         for (int i = 0; i < N_ELEMENTS(menu); i++) {
363                 for (int j = 0; j < N_ELEMENTS(menu[i]->keys); j++)
364                         if (menu[i]->keys[j] == key)
365                                 return set_view(menu[i], NULL);
366         }
367
368         /* Shift windows with left/right keys */
369         int shift = key == KEY_RIGHT ? +1 :
370                     key == KEY_LEFT  ? -1 : 0;
371         if (shift) {
372                 int num = 0;
373                 for (int i = 0; i < N_ELEMENTS(menu); i++)
374                         if (menu[i] == active)
375                                 num = i;
376                 do  {
377                         num += shift;
378                         num += N_ELEMENTS(menu);
379                         num %= N_ELEMENTS(menu);
380                 } while (menu[num] == &spacer);
381                 return set_view(menu[num], NULL);
382         }
383
384         /* Handle other keys */
385         switch (key) {
386                 case 'c':
387                         COMPACT ^= 1;
388                         set_bool("view", 0, "compact", COMPACT);
389                         view_resize();
390                         view_draw();
391                         return 1;
392                 case '\033': // escape
393                         return set_view(active, NULL);
394                 case '?':    // help
395                         return set_view(active, &help_view);
396                 case 'e':    // edit
397                         return set_view(active, &edit_view);
398         }
399
400         /* Pass key to active view */
401         return view->run(key, btn, row, col);
402 }
403
404 /* View event */
405 void view_edit(edit_t mode)
406 {
407         EDIT = mode;
408         set_view(active, &edit_view);
409 }