added --debug
[ptask.git] / src / main.c
1 /*
2  * Copyright (C) 2012-2013 jeanfi@gmail.com
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * 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, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301 USA
18  */
19 #include <stdio.h>
20 #include <stdlib.h>
21 #include <string.h>
22 #include <getopt.h>
23 #include <sys/stat.h>
24
25 #include <json/json.h>
26
27 #include <glib/gi18n.h>
28 #include <gtk/gtk.h>
29
30 #include <config.h>
31
32 #include "log.h"
33 #include "note.h"
34 #include "tw.h"
35
36 static const char *program_name;
37 static struct task **tasks;
38 static GtkTextView *w_note;
39 static GtkEntry *w_description;
40 static GtkEntry *w_project;
41 static GtkTreeView *w_treeview;
42 static GtkWidget *w_tasksave_btn;
43 static GtkWidget *w_taskdone_btn;
44 static GtkComboBox *w_status;
45 static GtkComboBox *w_priority;
46
47 enum {
48         COL_ID,
49         COL_DESCRIPTION,
50         COL_PROJECT,
51         COL_UUID,
52         COL_PRIORITY
53 };
54
55 static struct option long_options[] = {
56         {"version", no_argument, 0, 'v'},
57         {"help", no_argument, 0, 'h'},
58         {"debug", required_argument, 0, 'd'},
59         {0, 0, 0, 0}
60 };
61
62 static void print_version()
63 {
64         printf("ptask %s\n", VERSION);
65         printf(_("Copyright (C) %s jeanfi@gmail.com\n"
66                  "License GPLv2: GNU GPL version 2 or later "
67                  "<http://www.gnu.org/licenses/old-licenses/gpl-2.0.html>\n"
68                  "This is free software: you are free to change and "
69                  " redistribute it.\n"
70                  "There is NO WARRANTY, to the extent permitted by law.\n"),
71                "2012-2013");
72 }
73
74 static void print_help()
75 {
76         printf(_("Usage: %s [OPTION]...\n"), program_name);
77
78         puts(_("Ptask is a task management UI based on taskwarrior."));
79
80         puts("");
81         puts(_("Options:"));
82         puts(_("  -h, --help          display this help and exit\n"
83                "  -v, --version       display version information and exit"));
84
85         puts("");
86
87         puts(_("  -d, --debug=LEVEL   "
88                "set the debug level, integer between 0 and 3"));
89
90         puts("");
91
92         printf(_("Report bugs to: %s\n"), PACKAGE_BUGREPORT);
93         puts("");
94         printf(_("%s home page: <%s>\n"), PACKAGE_NAME, PACKAGE_URL);
95 }
96
97 static struct task *get_selected_task(GtkTreeView *treeview)
98 {
99         GtkTreePath *path;
100         GtkTreeViewColumn *cols;
101         struct task **tasks_cur;
102         GtkTreeIter iter;
103         GtkTreeModel *model;
104         GValue value = {0,};
105         const char *uuid;
106
107         log_debug("get_selected_task");
108
109         gtk_tree_view_get_cursor(treeview, &path, &cols);
110
111         if (path) {
112                 model = gtk_tree_view_get_model(GTK_TREE_VIEW(treeview));
113                 gtk_tree_model_get_iter(model, &iter, path);
114                 gtk_tree_model_get_value(model, &iter, COL_UUID, &value);
115
116                 uuid = g_value_get_string(&value);
117
118                 for (tasks_cur = tasks; *tasks_cur; tasks_cur++)
119                         if (!strcmp((*tasks_cur)->uuid, uuid))
120                                 return *tasks_cur;
121
122                 gtk_tree_path_free(path);
123         }
124
125         return NULL;
126 }
127
128 static void clear_task_panel()
129 {
130         GtkTextBuffer *buf;
131
132         gtk_widget_set_sensitive(w_tasksave_btn, 0);
133         gtk_widget_set_sensitive(w_taskdone_btn, 0);
134
135         buf = gtk_text_view_get_buffer(w_note);
136         gtk_text_buffer_set_text(buf, "", 0);
137         gtk_widget_set_sensitive(GTK_WIDGET(w_note), 0);
138
139         gtk_entry_set_text(w_description, "");
140         gtk_widget_set_sensitive(GTK_WIDGET(w_description), 0);
141
142         gtk_entry_set_text(w_project, "");
143         gtk_widget_set_sensitive(GTK_WIDGET(w_project), 0);
144
145         gtk_combo_box_set_active(w_priority, 0);
146         gtk_widget_set_sensitive(GTK_WIDGET(w_priority), 0);
147 }
148
149 static void refresh()
150 {
151         GtkWidget *dialog;
152         GtkTreeModel *model;
153         struct task **tasks_cur;
154         struct task *task;
155         int i;
156         GtkTreeIter iter;
157         int status;
158         const char *project;
159
160         log_debug("refresh");
161         clear_task_panel();
162
163         status = gtk_combo_box_get_active(w_status);
164         log_debug("status: %d", status);
165
166         if (tasks)
167                 tw_task_list_free(tasks);
168
169         switch (status) {
170         case 0:
171                 tasks = tw_get_all_tasks("pending");
172                 break;
173         case 1:
174                 tasks = tw_get_all_tasks("completed");
175                 break;
176         default:
177                 tasks = tw_get_all_tasks("pending");
178         }
179
180         model = gtk_tree_view_get_model(GTK_TREE_VIEW(w_treeview));
181         gtk_list_store_clear(GTK_LIST_STORE(model));
182
183         if (tasks) {
184                 for (tasks_cur = tasks, i = 0; *tasks_cur; tasks_cur++, i++) {
185                         task = (*tasks_cur);
186
187                         gtk_list_store_append(GTK_LIST_STORE(model), &iter);
188
189                         if (task->project)
190                                 project = task->project;
191                         else
192                                 project = "";
193
194                         gtk_list_store_set(GTK_LIST_STORE(model),
195                                            &iter,
196                                            COL_ID, (*tasks_cur)->id,
197                                            COL_DESCRIPTION,
198                                            (*tasks_cur)->description,
199                                            COL_PROJECT, project,
200                                            COL_UUID, (*tasks_cur)->uuid,
201                                            COL_PRIORITY, (*tasks_cur)->priority,
202                                            -1);
203                 }
204         } else {
205                 dialog = gtk_message_dialog_new(NULL,
206                                                 GTK_DIALOG_DESTROY_WITH_PARENT,
207                                                 GTK_MESSAGE_ERROR,
208                                                 GTK_BUTTONS_CLOSE,
209                                                 _("Error loading tasks, verify "
210                                                   "that taskwarrior is "
211                                                   "correctly installed, and its"
212                                                   " configuration file exist."
213                                                   ));
214                 gtk_dialog_run(GTK_DIALOG(dialog));
215                 gtk_widget_destroy(dialog);
216         }
217         log_debug("refresh done");
218 }
219
220 int taskdone_clicked_cbk(GtkButton *btn, gpointer data)
221 {
222         struct task *task;
223
224         task = get_selected_task(GTK_TREE_VIEW(w_treeview));
225         tw_done(task->uuid);
226         refresh();
227
228         return FALSE;
229 }
230
231 static int tasksave_clicked_cbk(GtkButton *btn, gpointer data)
232 {
233         struct task *task;
234         GtkTextBuffer *buf;
235         char *txt, *pri;
236         GtkTextIter sIter, eIter;
237         const char *ctxt;
238         int priority;
239
240         task = get_selected_task(GTK_TREE_VIEW(w_treeview));
241
242         log_debug("tasksave_clicked_cbk %d", task->id);
243
244         buf = gtk_text_view_get_buffer(w_note);
245
246         gtk_text_buffer_get_iter_at_offset(buf, &sIter, 0);
247         gtk_text_buffer_get_iter_at_offset(buf, &eIter, -1);
248         txt = gtk_text_buffer_get_text(buf, &sIter, &eIter, TRUE);
249
250         log_debug("note=%s", txt);
251
252         if (!task->note || strcmp(txt, task->note))
253                 note_put(task->uuid, txt);
254
255         ctxt = gtk_entry_get_text(w_description);
256         if (!task->description || strcmp(ctxt, task->description))
257                 tw_modify_description(task->uuid, ctxt);
258
259         ctxt = gtk_entry_get_text(w_project);
260         if (!task->project || strcmp(ctxt, task->project))
261                 tw_modify_project(task->uuid, ctxt);
262
263         priority = gtk_combo_box_get_active(w_priority);
264         log_debug("priority: %d", priority);
265
266         switch (priority) {
267         case 3:
268                 pri = "H";
269                 break;
270         case 2:
271                 pri = "M";
272                 break;
273         case 1:
274                 pri = "L";
275                 break;
276         default:
277                 pri = "";
278         }
279
280         if (strcmp(task->priority, pri))
281                 tw_modify_priority(task->uuid, pri);
282
283         refresh();
284
285         return FALSE;
286 }
287
288 int refresh_clicked_cbk(GtkButton *btn, gpointer data)
289 {
290         log_debug("refresh_clicked_cbk");
291         refresh();
292
293         return FALSE;
294 }
295
296 static gboolean delete_event_cbk(GtkWidget *w, GdkEvent *evt, gpointer data)
297 {
298         gtk_widget_destroy(w);
299         gtk_main_quit();
300
301         return FALSE;
302 }
303
304
305 int newtask_clicked_cbk(GtkButton *btn, gpointer data)
306 {
307         gint result;
308         static GtkDialog *diag;
309         GtkBuilder *builder;
310         GtkEntry *entry;
311         const char *ctxt;
312
313         log_debug("newtask_clicked_cbk");
314
315         builder = gtk_builder_new();
316         gtk_builder_add_from_file
317                 (builder,
318                  PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "ptask.glade",
319                  NULL);
320         diag = GTK_DIALOG(gtk_builder_get_object(builder, "diag_tasknew"));
321         gtk_builder_connect_signals(builder, NULL);
322
323         result = gtk_dialog_run(diag);
324
325         if (result == GTK_RESPONSE_ACCEPT) {
326                 log_debug("ok");
327                 entry = GTK_ENTRY(gtk_builder_get_object
328                                   (builder, "diag_tasknew_description"));
329                 ctxt = gtk_entry_get_text(entry);
330
331                 log_debug("%s", ctxt);
332
333                 tw_add(ctxt);
334                 refresh();
335         } else {
336                 log_debug("cancel");
337         }
338
339         g_object_unref(G_OBJECT(builder));
340
341         gtk_widget_destroy(GTK_WIDGET(diag));
342
343         return FALSE;
344 }
345
346 static int status_changed_cbk(GtkComboBox *w, gpointer data)
347 {
348         log_debug("status_changed_cbk");
349         refresh();
350
351         return FALSE;
352 }
353
354 static int priority_to_int(const char *str)
355 {
356         switch (*str) {
357         case 'H':
358                 return 3;
359         case 'M':
360                 return 2;
361         case 'L':
362                 return 1;
363         default:
364                 return 0;
365         }
366 }
367
368 static int cursor_changed_cbk(GtkTreeView *treeview, gpointer data)
369 {
370         struct task *task;
371         GtkTextBuffer *buf;
372         int priority;
373
374         log_debug("cursor_changed_cbk");
375
376         task = get_selected_task(treeview);
377
378         if (task) {
379
380                 buf = gtk_text_view_get_buffer(w_note);
381                 if (task->note)
382                         gtk_text_buffer_set_text(buf,
383                                                  task->note,
384                                                  strlen(task->note));
385                 else
386                         gtk_text_buffer_set_text(buf, "", 0);
387                 gtk_widget_set_sensitive(GTK_WIDGET(w_note), 1);
388
389                 gtk_entry_set_text(w_description, task->description);
390                 gtk_widget_set_sensitive(GTK_WIDGET(w_description), 1);
391
392                 if (task->project)
393                         gtk_entry_set_text(w_project, task->project);
394                 else
395                         gtk_entry_set_text(w_project, "");
396                 gtk_widget_set_sensitive(GTK_WIDGET(w_project), 1);
397
398                 gtk_widget_set_sensitive(w_tasksave_btn, 1);
399                 gtk_widget_set_sensitive(w_taskdone_btn, 1);
400
401                 gtk_widget_set_sensitive(GTK_WIDGET(w_priority), 1);
402                 priority = priority_to_int(task->priority);
403                 gtk_combo_box_set_active(w_priority, priority);
404         } else {
405                 log_debug("clear task widgets");
406                 clear_task_panel();
407                 log_debug("clear task widgets done");
408         }
409
410         return FALSE;
411 }
412
413 static gint priority_cmp(GtkTreeModel *model,
414                          GtkTreeIter *a,
415                          GtkTreeIter *b,
416                          gpointer user_data)
417 {
418         GValue v1 = {0,}, v2 = {0,};
419         const char *str1, *str2;
420         int i1, i2;
421
422         gtk_tree_model_get_value(model, a, COL_PRIORITY, &v1);
423         str1 = g_value_get_string(&v1);
424         i1 = priority_to_int(str1);
425
426         gtk_tree_model_get_value(model, b, COL_PRIORITY, &v2);
427         str2 = g_value_get_string(&v2);
428         i2 = priority_to_int(str2);
429
430         if (i1 < i2)
431                 return -1;
432         else if (i1 > i2)
433                 return 1;
434         else
435                 return 0;
436 }
437
438 static void log_init()
439 {
440         char *home, *path, *dir;
441
442         home = getenv("HOME");
443
444         if (!home)
445                 return ;
446
447         dir = malloc(strlen(home)+1+strlen(".ptask")+1);
448         sprintf(dir, "%s/%s", home, ".ptask");
449         mkdir(dir, 0777);
450
451         path = malloc(strlen(dir)+1+strlen("log")+1);
452         sprintf(path, "%s/%s", dir, "log");
453
454         log_open(path);
455
456         free(dir);
457         free(path);
458 }
459
460 int main(int argc, char **argv)
461 {
462         GtkWidget *window, *btn;
463         GtkBuilder *builder;
464         GtkTreeModel *model;
465         int optc, cmdok, opti;
466
467         program_name = argv[0];
468
469         setlocale(LC_ALL, "");
470
471 #if ENABLE_NLS
472         bindtextdomain(PACKAGE, LOCALEDIR);
473         textdomain(PACKAGE);
474 #endif
475
476         cmdok = 1;
477         while ((optc = getopt_long(argc, argv, "vhd:", long_options,
478                                    &opti)) != -1) {
479                 switch (optc) {
480                 case 'h':
481                         print_help();
482                         exit(EXIT_SUCCESS);
483                 case 'v':
484                         print_version();
485                         exit(EXIT_SUCCESS);
486                 case 'd':
487                         log_level = atoi(optarg);
488                         log_info(_("Enables debug mode."));
489                         break;
490                 default:
491                         cmdok = 0;
492                         break;
493                 }
494         }
495
496         if (!cmdok || optind != argc) {
497                 fprintf(stderr, _("Try `%s --help' for more information.\n"),
498                         program_name);
499                 exit(EXIT_FAILURE);
500         }
501
502         log_init();
503
504         gtk_init(NULL, NULL);
505         builder = gtk_builder_new();
506         gtk_builder_add_from_file
507                 (builder,
508                  PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "ptask.glade",
509                  NULL);
510         window = GTK_WIDGET(gtk_builder_get_object(builder, "window"));
511
512         g_signal_connect(window, "delete_event",
513                          G_CALLBACK(delete_event_cbk), NULL);
514
515         w_treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview"));
516         model = gtk_tree_view_get_model(GTK_TREE_VIEW(w_treeview));
517         gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(model),
518                                         COL_PRIORITY,
519                                         priority_cmp,
520                                         NULL,
521                                         NULL);
522
523         w_note = GTK_TEXT_VIEW(gtk_builder_get_object(builder, "tasknote"));
524
525         w_description = GTK_ENTRY(gtk_builder_get_object(builder,
526                                                          "taskdescription"));
527         w_project = GTK_ENTRY(gtk_builder_get_object(builder, "taskproject"));
528         w_status = GTK_COMBO_BOX(gtk_builder_get_object(builder, "status"));
529         w_priority = GTK_COMBO_BOX(gtk_builder_get_object(builder,
530                                                           "taskpriority"));
531
532         refresh();
533
534         gtk_builder_connect_signals(builder, NULL);
535
536         g_signal_connect(w_treeview,
537                          "cursor-changed", (GCallback)cursor_changed_cbk,
538                          tasks);
539         g_signal_connect(w_status,
540                          "changed", (GCallback)status_changed_cbk,
541                          tasks);
542
543         btn = GTK_WIDGET(gtk_builder_get_object(builder, "tasksave"));
544         g_signal_connect(btn,
545                          "clicked", (GCallback)tasksave_clicked_cbk, tasks);
546         gtk_widget_set_sensitive(btn, 0);
547         w_tasksave_btn = btn;
548
549         w_taskdone_btn = GTK_WIDGET(gtk_builder_get_object(builder,
550                                                            "taskdone"));
551         gtk_widget_set_sensitive(w_taskdone_btn, 0);
552
553         g_object_unref(G_OBJECT(builder));
554
555         gtk_widget_show_all(window);
556
557         gtk_main();
558
559         exit(EXIT_SUCCESS);
560 }