8dcd89f1320cdcc4561e6f1c1995fa5f6d83026a
[ptask.git] / src / main.c
1 /*
2  * Copyright (C) 2010-2012 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
23 #include <json/json.h>
24
25 #include <gtk/gtk.h>
26
27 #include "note.h"
28 #include "tw.h"
29
30 static struct task **tasks;
31 static GtkTextView *w_note;
32 static GtkEntry *w_description;
33 static GtkEntry *w_project;
34 static GtkTreeView *w_treeview;
35 static GtkWidget *w_tasksave_btn;
36 static GtkWidget *w_taskdone_btn;
37 static GtkComboBox *w_status;
38 static GtkComboBox *w_priority;
39
40 enum {
41         COL_ID,
42         COL_DESCRIPTION,
43         COL_PROJECT,
44         COL_UUID,
45         COL_PRIORITY
46 };
47
48 static struct task *get_selected_task(GtkTreeView *treeview)
49 {
50         GtkTreePath *path;
51         GtkTreeViewColumn *cols;
52         struct task **tasks_cur;
53         GtkTreeIter iter;
54         GtkTreeModel *model;
55         GValue value = {0,};
56         const char *uuid;
57
58         printf("get_selected_task\n");
59
60         gtk_tree_view_get_cursor(treeview, &path, &cols);
61
62         if (path) {
63                 model = gtk_tree_view_get_model(GTK_TREE_VIEW(treeview));
64                 gtk_tree_model_get_iter(model, &iter, path);
65                 gtk_tree_model_get_value(model, &iter, COL_UUID, &value);
66
67                 uuid = g_value_get_string(&value);
68
69                 for (tasks_cur = tasks; *tasks_cur; tasks_cur++)
70                         if (!strcmp((*tasks_cur)->uuid, uuid))
71                                 return *tasks_cur;
72
73                 gtk_tree_path_free(path);
74         }
75
76         return NULL;
77 }
78
79 static void clear_task_panel()
80 {
81         GtkTextBuffer *buf;
82
83         gtk_widget_set_sensitive(w_tasksave_btn, 0);
84         gtk_widget_set_sensitive(w_taskdone_btn, 0);
85
86         buf = gtk_text_view_get_buffer(w_note);
87         gtk_text_buffer_set_text(buf, "", 0);
88         gtk_widget_set_sensitive(GTK_WIDGET(w_note), 0);
89
90         gtk_entry_set_text(w_description, "");
91         gtk_widget_set_sensitive(GTK_WIDGET(w_description), 0);
92
93         gtk_entry_set_text(w_project, "");
94         gtk_widget_set_sensitive(GTK_WIDGET(w_project), 0);
95
96         gtk_combo_box_set_active(w_priority, 0);
97         gtk_widget_set_sensitive(GTK_WIDGET(w_priority), 0);
98 }
99
100 static void refresh()
101 {
102         GtkTreeModel *model;
103         struct task **tasks_cur;
104         struct task *task;
105         int i;
106         GtkTreeIter iter;
107         int status;
108         const char *project;
109
110         printf("refresh\n");
111         clear_task_panel();
112
113         status = gtk_combo_box_get_active(w_status);
114         printf("status: %d\n", status);
115
116         if (tasks)
117                 tw_task_list_free(tasks);
118
119         switch (status) {
120         case 0:
121                 tasks = tw_get_all_tasks("pending");
122                 break;
123         case 1:
124                 tasks = tw_get_all_tasks("completed");
125                 break;
126         default:
127                 tasks = tw_get_all_tasks("pending");
128         }
129
130         model = gtk_tree_view_get_model(GTK_TREE_VIEW(w_treeview));
131         gtk_list_store_clear(GTK_LIST_STORE(model));
132         for (tasks_cur = tasks, i = 0; *tasks_cur; tasks_cur++, i++) {
133                 task = (*tasks_cur);
134
135                 gtk_list_store_append(GTK_LIST_STORE(model), &iter);
136
137                 if (task->project)
138                         project = task->project;
139                 else
140                         project = "";
141
142                 gtk_list_store_set(GTK_LIST_STORE(model),
143                                    &iter,
144                                    COL_ID, (*tasks_cur)->id,
145                                    COL_DESCRIPTION, (*tasks_cur)->description,
146                                    COL_PROJECT, project,
147                                    COL_UUID, (*tasks_cur)->uuid,
148                                    COL_PRIORITY, (*tasks_cur)->priority,
149                                    -1);
150         }
151         printf("refresh done\n");
152 }
153
154 int taskdone_clicked_cbk(GtkButton *btn, gpointer data)
155 {
156         struct task *task;
157
158         task = get_selected_task(GTK_TREE_VIEW(w_treeview));
159         tw_done(task->uuid);
160         refresh();
161
162         return FALSE;
163 }
164
165 static int tasksave_clicked_cbk(GtkButton *btn, gpointer data)
166 {
167         struct task *task;
168         GtkTextBuffer *buf;
169         char *txt, *pri;
170         GtkTextIter sIter, eIter;
171         const char *ctxt;
172         int priority;
173
174         task = get_selected_task(GTK_TREE_VIEW(w_treeview));
175
176         printf("tasksave_clicked_cbk %d\n", task->id);
177
178         buf = gtk_text_view_get_buffer(w_note);
179
180         gtk_text_buffer_get_iter_at_offset(buf, &sIter, 0);
181         gtk_text_buffer_get_iter_at_offset(buf, &eIter, -1);
182         txt = gtk_text_buffer_get_text(buf, &sIter, &eIter, TRUE);
183
184         printf("note=%s\n", txt);
185
186         if (!task->note || strcmp(txt, task->note))
187                 note_put(task->uuid, txt);
188
189         ctxt = gtk_entry_get_text(w_description);
190         if (!task->description || strcmp(ctxt, task->description))
191                 tw_modify_description(task->uuid, ctxt);
192
193         ctxt = gtk_entry_get_text(w_project);
194         if (!task->project || strcmp(ctxt, task->project))
195                 tw_modify_project(task->uuid, ctxt);
196
197         priority = gtk_combo_box_get_active(w_priority);
198         printf("priority: %d\n", priority);
199
200         switch (priority) {
201         case 3:
202                 pri = "H";
203                 break;
204         case 2:
205                 pri = "M";
206                 break;
207         case 1:
208                 pri = "L";
209                 break;
210         default:
211                 pri = "";
212         }
213
214         if (strcmp(task->priority, pri))
215                 tw_modify_priority(task->uuid, pri);
216
217         refresh();
218
219         return FALSE;
220 }
221
222 int refresh_clicked_cbk(GtkButton *btn, gpointer data)
223 {
224         printf("refresh_clicked_cbk\n");
225         refresh();
226
227         return FALSE;
228 }
229
230 int newtask_clicked_cbk(GtkButton *btn, gpointer data)
231 {
232         gint result;
233         static GtkDialog *diag;
234         GtkBuilder *builder;
235         GtkEntry *entry;
236         const char *ctxt;
237
238         printf("newtask_clicked_cbk\n");
239
240         builder = gtk_builder_new();
241         gtk_builder_add_from_file
242                 (builder,
243                  PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "gtask.glade",
244                  NULL);
245         diag = GTK_DIALOG(gtk_builder_get_object(builder, "diag_tasknew"));
246         gtk_builder_connect_signals(builder, NULL);
247
248         result = gtk_dialog_run(diag);
249
250         if (result == GTK_RESPONSE_ACCEPT) {
251                 printf("ok\n");
252                 entry = GTK_ENTRY(gtk_builder_get_object
253                                   (builder, "diag_tasknew_description"));
254                 ctxt = gtk_entry_get_text(entry);
255
256                 printf("%s\n", ctxt);
257
258                 tw_add(ctxt);
259                 refresh();
260         } else {
261                 printf("cancel\n");
262         }
263
264         g_object_unref(G_OBJECT(builder));
265
266         gtk_widget_destroy(GTK_WIDGET(diag));
267
268         return FALSE;
269 }
270
271 static int status_changed_cbk(GtkComboBox *w, gpointer data)
272 {
273         printf("status_changed_cbk\n");
274         refresh();
275
276         return FALSE;
277 }
278
279 static int priority_to_int(const char *str)
280 {
281         switch (*str) {
282         case 'H':
283                 return 3;
284         case 'M':
285                 return 2;
286         case 'L':
287                 return 1;
288         default:
289                 return 0;
290         }
291 }
292
293 static int cursor_changed_cbk(GtkTreeView *treeview, gpointer data)
294 {
295         struct task *task;
296         GtkTextBuffer *buf;
297         int priority;
298
299         printf("cursor_changed_cbk\n");
300
301         task = get_selected_task(treeview);
302
303         if (task) {
304
305                 buf = gtk_text_view_get_buffer(w_note);
306                 if (task->note)
307                         gtk_text_buffer_set_text(buf,
308                                                  task->note,
309                                                  strlen(task->note));
310                 else
311                         gtk_text_buffer_set_text(buf, "", 0);
312                 gtk_widget_set_sensitive(GTK_WIDGET(w_note), 1);
313
314                 gtk_entry_set_text(w_description, task->description);
315                 gtk_widget_set_sensitive(GTK_WIDGET(w_description), 1);
316
317                 if (task->project)
318                         gtk_entry_set_text(w_project, task->project);
319                 else
320                         gtk_entry_set_text(w_project, "");
321                 gtk_widget_set_sensitive(GTK_WIDGET(w_project), 1);
322
323                 gtk_widget_set_sensitive(w_tasksave_btn, 1);
324                 gtk_widget_set_sensitive(w_taskdone_btn, 1);
325
326                 gtk_widget_set_sensitive(GTK_WIDGET(w_priority), 1);
327                 priority = priority_to_int(task->priority);
328                 gtk_combo_box_set_active(w_priority, priority);
329         } else {
330                 printf("clear task widgets\n");
331                 clear_task_panel();
332                 printf("clear task widgets done\n");
333         }
334
335         return FALSE;
336 }
337
338 static gint priority_cmp(GtkTreeModel *model,
339                          GtkTreeIter *a,
340                          GtkTreeIter *b,
341                          gpointer user_data)
342 {
343         GValue v1 = {0,}, v2 = {0,};
344         const char *str1, *str2;
345         int i1, i2;
346
347         gtk_tree_model_get_value(model, a, COL_PRIORITY, &v1);
348         str1 = g_value_get_string(&v1);
349         i1 = priority_to_int(str1);
350
351         gtk_tree_model_get_value(model, b, COL_PRIORITY, &v2);
352         str2 = g_value_get_string(&v2);
353         i2 = priority_to_int(str2);
354
355         if (i1 < i2)
356                 return -1;
357         else if (i1 > i2)
358                 return 1;
359         else
360                 return 0;
361 }
362
363 int main(int argc, char **argv)
364 {
365         GtkWidget *window;
366         GtkWidget *btn;
367         GtkBuilder *builder;
368         GtkTreeModel *model;
369
370         gtk_init(NULL, NULL);
371         builder = gtk_builder_new();
372         gtk_builder_add_from_file
373                 (builder,
374                  PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "gtask.glade",
375                  NULL);
376         window = GTK_WIDGET(gtk_builder_get_object(builder, "window"));
377
378         w_treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview"));
379         model = gtk_tree_view_get_model(GTK_TREE_VIEW(w_treeview));
380         gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(model),
381                                         COL_PRIORITY,
382                                         priority_cmp,
383                                         NULL,
384                                         NULL);
385
386         w_note = GTK_TEXT_VIEW(gtk_builder_get_object(builder, "tasknote"));
387
388         w_description = GTK_ENTRY(gtk_builder_get_object(builder,
389                                                          "taskdescription"));
390         w_project = GTK_ENTRY(gtk_builder_get_object(builder, "taskproject"));
391         w_status = GTK_COMBO_BOX(gtk_builder_get_object(builder, "status"));
392         w_priority = GTK_COMBO_BOX(gtk_builder_get_object(builder,
393                                                           "taskpriority"));
394
395         refresh();
396
397         gtk_builder_connect_signals(builder, NULL);
398
399         g_signal_connect(w_treeview,
400                          "cursor-changed", (GCallback)cursor_changed_cbk,
401                          tasks);
402         g_signal_connect(w_status,
403                          "changed", (GCallback)status_changed_cbk,
404                          tasks);
405
406         btn = GTK_WIDGET(gtk_builder_get_object(builder, "tasksave"));
407         g_signal_connect(btn,
408                          "clicked", (GCallback)tasksave_clicked_cbk, tasks);
409         gtk_widget_set_sensitive(btn, 0);
410         w_tasksave_btn = btn;
411
412         w_taskdone_btn = GTK_WIDGET(gtk_builder_get_object(builder,
413                                                            "taskdone"));
414         gtk_widget_set_sensitive(w_taskdone_btn, 0);
415
416         g_object_unref(G_OBJECT(builder));
417
418         gtk_widget_show_all(window);
419
420         gtk_main();
421
422         exit(EXIT_SUCCESS);
423 }