#include <gtk/gtk.h>
-struct task {
- int id;
- char *description;
- char *status;
+#include "tw.h"
+
+static struct task **tasks;
+static GtkTextView *w_note;
+static GtkEntry *w_description;
+static GtkEntry *w_project;
+static GtkTreeView *w_treeview;
+static GtkWidget *w_tasksave_btn;
+static GtkComboBox *w_status;
+
+enum {
+ COL_ID,
+ COL_DESCRIPTION,
+ COL_PROJECT,
+ COL_UUID,
+ COL_PRIORITY
};
-static char *task_exec(char *opts)
+static struct task *get_selected_task(GtkTreeView *treeview)
{
- FILE *f;
- int ret, s;
- char *str, *tmp, *cmd, buf[1024];
+ GtkTreePath *path;
+ GtkTreeViewColumn *cols;
+ struct task **tasks_cur;
+ GtkTreeIter iter;
+ GtkTreeModel *model;
+ GValue value = {0,};
+ const char *uuid;
- str = NULL;
+ printf("get_selected_task\n");
- cmd = malloc(strlen("task rc.json.array=on ") + strlen(opts) + 1);
- strcpy(cmd, "task rc.json.array=on ");
- strcat(cmd, opts);
+ gtk_tree_view_get_cursor(treeview, &path, &cols);
- f = popen(cmd, "r");
+ if (path) {
+ model = gtk_tree_view_get_model(GTK_TREE_VIEW(treeview));
+ gtk_tree_model_get_iter(model, &iter, path);
+ gtk_tree_model_get_value(model, &iter, COL_UUID, &value);
- if (!f) {
- perror("popen");
- goto exit_free;
- }
+ uuid = g_value_get_string(&value);
- str = malloc(1);
- str[0] = '\0';
- while ((s = fread(buf, 1, 1024, f))) {
- tmp = malloc(strlen(str) + s + 1);
- memcpy(tmp, str, strlen(str));
- memcpy(tmp + strlen(str), buf, s);
- tmp[strlen(str) + s] = '\0';
- free(str);
- str = tmp;
+ for (tasks_cur = tasks; *tasks_cur; tasks_cur++)
+ if (!strcmp((*tasks_cur)->uuid, uuid))
+ return *tasks_cur;
+
+ gtk_tree_path_free(path);
}
- ret = pclose(f);
+ return NULL;
+}
+
+static void clear_task_panel()
+{
+ GtkTextBuffer *buf;
+
+ gtk_widget_set_sensitive(w_tasksave_btn, 0);
- if (ret == -1) {
- printf("pclose fails\n");
- perror("pclose");
+ buf = gtk_text_view_get_buffer(w_note);
+ gtk_text_buffer_set_text(buf, "", 0);
+ gtk_widget_set_sensitive(GTK_WIDGET(w_note), 0);
+
+ gtk_entry_set_text(w_description, "");
+ gtk_widget_set_sensitive(GTK_WIDGET(w_description), 0);
+
+ gtk_entry_set_text(w_project, "");
+ gtk_widget_set_sensitive(GTK_WIDGET(w_project), 0);
+}
+
+static void refresh()
+{
+ GtkTreeModel *model;
+ struct task **tasks_cur;
+ struct task *task;
+ int i;
+ GtkTreeIter iter;
+ int status;
+
+ clear_task_panel();
+
+ status = gtk_combo_box_get_active(w_status);
+ printf("status: %d\n", status);
+
+ switch (status) {
+ case 0:
+ tasks = tw_get_all_tasks("pending");
+ break;
+ case 1:
+ tasks = tw_get_all_tasks("completed");
+ break;
+ default:
+ tasks = tw_get_all_tasks("pending");
}
- exit_free:
- free(cmd);
+ model = gtk_tree_view_get_model(GTK_TREE_VIEW(w_treeview));
+ gtk_list_store_clear(GTK_LIST_STORE(model));
+ for (tasks_cur = tasks, i = 0; *tasks_cur; tasks_cur++, i++) {
+ task = (*tasks_cur);
+
+ gtk_list_store_append(GTK_LIST_STORE(model), &iter);
+
+ if (task->project)
+ gtk_list_store_set(GTK_LIST_STORE(model),
+ &iter,
+ COL_PROJECT, task->project,
+ -1);
- return str;
+ gtk_list_store_set(GTK_LIST_STORE(model),
+ &iter,
+ COL_ID, (*tasks_cur)->id,
+ COL_DESCRIPTION, (*tasks_cur)->description,
+ COL_UUID, (*tasks_cur)->uuid,
+ COL_PRIORITY, (*tasks_cur)->priority,
+ -1);
+ }
}
-static struct json_object *task_exec_json(char *opts)
+static int tasksave_clicked_cbk(GtkButton *btn, gpointer data)
{
- struct json_object *o;
- char *str;
+ struct task *task;
+ GtkTextBuffer *buf;
+ char *txt;
+ GtkTextIter sIter, eIter;
+ const char *ctxt;
+
+ task = get_selected_task(GTK_TREE_VIEW(w_treeview));
+
+ printf("tasksave_clicked_cbk %d\n", task->id);
- str = task_exec(opts);
+ if (task->note) {
+ buf = gtk_text_view_get_buffer(w_note);
- if (str) {
- o = json_tokener_parse(str);
- free(str);
- return o;
+ gtk_text_buffer_get_iter_at_offset(buf, &sIter, 0);
+ gtk_text_buffer_get_iter_at_offset(buf, &eIter, -1);
+ txt = gtk_text_buffer_get_text(buf, &sIter, &eIter, TRUE);
+
+ txt = escape(txt);
+
+ printf("%s\n", txt);
}
- return NULL;
+ ctxt = gtk_entry_get_text(w_description);
+ if (!task->description || strcmp(ctxt, task->description))
+ tw_modify_description(task->uuid, ctxt);
+
+ ctxt = gtk_entry_get_text(w_project);
+ if (!task->project || strcmp(ctxt, task->project))
+ tw_modify_project(task->uuid, ctxt);
+
+ refresh();
+
+ return FALSE;
}
-static struct task **get_all_tasks()
+int refresh_clicked_cbk(GtkButton *btn, gpointer data)
{
- int i, n;
- struct json_object *jtasks, *jtask, *json;
- struct task **tasks;
-
- jtasks = task_exec_json("export");
+ printf("refresh_clicked_cbk\n");
+ refresh();
- if (!jtasks)
- return NULL;
+ return FALSE;
+}
- n = json_object_array_length(jtasks);
+int newtask_clicked_cbk(GtkButton *btn, gpointer data)
+{
+ gint result;
+ static GtkDialog *diag;
+ GtkBuilder *builder;
+ GtkEntry *entry;
+ const char *ctxt;
- tasks = malloc((n + 1) * sizeof(struct task *));
+ printf("newtask_clicked_cbk\n");
- for (i = 0; i < n; i++) {
- jtask = json_object_array_get_idx(jtasks, i);
+ builder = gtk_builder_new();
+ gtk_builder_add_from_file
+ (builder,
+ PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "gtask.glade",
+ NULL);
+ diag = GTK_DIALOG(gtk_builder_get_object(builder, "diag_tasknew"));
+ gtk_builder_connect_signals(builder, NULL);
- tasks[i] = malloc(sizeof(struct task));
+ result = gtk_dialog_run(diag);
- json = json_object_object_get(jtask, "id");
- tasks[i]->id = json_object_get_int(json);
+ if (result == GTK_RESPONSE_ACCEPT) {
+ printf("ok\n");
+ entry = GTK_ENTRY(gtk_builder_get_object
+ (builder, "diag_tasknew_description"));
+ ctxt = gtk_entry_get_text(entry);
- json = json_object_object_get(jtask, "description");
- tasks[i]->description = strdup(json_object_get_string(json));
+ printf("%s\n", ctxt);
- json = json_object_object_get(jtask, "status");
- tasks[i]->status = strdup(json_object_get_string(json));
+ tw_add(ctxt);
+ refresh();
+ } else {
+ printf("cancel\n");
}
- tasks[n] = NULL;
+ g_object_unref(G_OBJECT(builder));
+
+ gtk_widget_destroy(GTK_WIDGET(diag));
+
+ return FALSE;
+}
- json_object_put(jtasks);
+static int status_changed_cbk(GtkComboBox *w, gpointer data)
+{
+ printf("status_changed_cbk\n");
+ refresh();
- return tasks;
+ return FALSE;
+}
+
+static int cursor_changed_cbk(GtkTreeView *treeview, gpointer data)
+{
+ struct task *task;
+ GtkTextBuffer *buf;
+
+ printf("cursor_changed_cbk\n");
+
+ task = get_selected_task(treeview);
+
+ if (task) {
+
+ buf = gtk_text_view_get_buffer(w_note);
+ if (task->note)
+ gtk_text_buffer_set_text(buf,
+ task->note,
+ strlen(task->note));
+ else
+ gtk_text_buffer_set_text(buf, "", 0);
+ gtk_widget_set_sensitive(GTK_WIDGET(w_note), 1);
+
+ gtk_entry_set_text(w_description, task->description);
+ gtk_widget_set_sensitive(GTK_WIDGET(w_description), 1);
+
+ if (task->project)
+ gtk_entry_set_text(w_project, task->project);
+ else
+ gtk_entry_set_text(w_project, "");
+ gtk_widget_set_sensitive(GTK_WIDGET(w_project), 1);
+
+ gtk_widget_set_sensitive(w_tasksave_btn, 1);
+ } else {
+ printf("clear task widgets\n");
+ clear_task_panel();
+ }
+
+ return FALSE;
}
int main(int argc, char **argv)
{
GtkWidget *window;
- GtkWidget *treeview;
+ GtkWidget *btn;
GtkBuilder *builder;
- GtkTreeIter iter;
- int i;
- GtkTreeModel *model;
- struct task **tasks, **tasks_cur;
gtk_init(NULL, NULL);
builder = gtk_builder_new();
PACKAGE_DATA_DIR G_DIR_SEPARATOR_S "gtask.glade",
NULL);
window = GTK_WIDGET(gtk_builder_get_object(builder, "window"));
- printf("%p\n", window);
- treeview = GTK_WIDGET(gtk_builder_get_object(builder, "treeview"));
+ w_treeview = GTK_TREE_VIEW(gtk_builder_get_object(builder, "treeview"));
- model = gtk_tree_view_get_model(GTK_TREE_VIEW(treeview));
+ w_note = GTK_TEXT_VIEW(gtk_builder_get_object(builder, "tasknote"));
- tasks = get_all_tasks();
+ w_description = GTK_ENTRY(gtk_builder_get_object(builder,
+ "taskdescription"));
+ w_project = GTK_ENTRY(gtk_builder_get_object(builder, "taskproject"));
+ w_status = GTK_COMBO_BOX(gtk_builder_get_object(builder, "status"));
- for (tasks_cur = tasks, i = 0; *tasks_cur; tasks_cur++, i++) {
- gtk_list_store_append(GTK_LIST_STORE(model), &iter);
- gtk_list_store_set(GTK_LIST_STORE(model),
- &iter,
- 0, (*tasks_cur)->id,
- 1, (*tasks_cur)->description,
- -1);
- }
+ refresh();
+
+ gtk_builder_connect_signals(builder, NULL);
+
+ g_signal_connect(w_treeview,
+ "cursor-changed", (GCallback)cursor_changed_cbk,
+ tasks);
+ g_signal_connect(w_status,
+ "changed", (GCallback)status_changed_cbk,
+ tasks);
+ btn = GTK_WIDGET(gtk_builder_get_object(builder, "tasksave"));
+ g_signal_connect(btn,
+ "clicked", (GCallback)tasksave_clicked_cbk, tasks);
+ gtk_widget_set_sensitive(btn, 0);
+ w_tasksave_btn = btn;
+
+ /*btn = GTK_WIDGET(gtk_builder_get_object(builder, "refresh"));
+ g_signal_connect(btn,
+ "clicked", (GCallback)refresh_clicked_cbk, tasks);*/
g_object_unref(G_OBJECT(builder));