about summary refs log tree commit diff stats
path: root/src/ui/windows.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/ui/windows.c')
-rw-r--r--src/ui/windows.c475
1 files changed, 475 insertions, 0 deletions
diff --git a/src/ui/windows.c b/src/ui/windows.c
new file mode 100644
index 00000000..e956068c
--- /dev/null
+++ b/src/ui/windows.c
@@ -0,0 +1,475 @@
+/*
+ * windows.c
+ *
+ * Copyright (C) 2012, 2013 James Booth <boothj5@gmail.com>
+ *
+ * This file is part of Profanity.
+ *
+ * Profanity is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * Profanity is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with Profanity.  If not, see <http://www.gnu.org/licenses/>.
+ *
+ */
+
+#include "config.h"
+
+#include <string.h>
+
+#include <glib.h>
+
+#ifdef HAVE_NCURSESW_NCURSES_H
+#include <ncursesw/ncurses.h>
+#elif HAVE_NCURSES_H
+#include <ncurses.h>
+#endif
+
+#include "common.h"
+#include "config/theme.h"
+#include "ui/ui.h"
+#include "ui/window.h"
+#include "ui/windows.h"
+
+#define CONS_WIN_TITLE "_cons"
+
+static GHashTable *windows;
+static int current;
+static int max_cols;
+
+void
+wins_init(void)
+{
+    windows = g_hash_table_new_full(g_direct_hash, g_direct_equal, NULL,
+        (GDestroyNotify)win_free);
+
+    max_cols = getmaxx(stdscr);
+    int cols = getmaxx(stdscr);
+    ProfWin *console = win_create(CONS_WIN_TITLE, cols, WIN_CONSOLE);
+    g_hash_table_insert(windows, GINT_TO_POINTER(1), console);
+
+    current = 1;
+}
+
+ProfWin *
+wins_get_console(void)
+{
+    return g_hash_table_lookup(windows, GINT_TO_POINTER(1));
+}
+
+ProfWin *
+wins_get_current(void)
+{
+    return g_hash_table_lookup(windows, GINT_TO_POINTER(current));
+}
+
+GList *
+wins_get_nums(void)
+{
+    return g_hash_table_get_keys(windows);
+}
+
+void
+wins_set_current_by_num(int i)
+{
+    if (g_hash_table_lookup(windows, GINT_TO_POINTER(i)) != NULL) {
+        current = i;
+    }
+}
+
+ProfWin *
+wins_get_by_num(int i)
+{
+    return g_hash_table_lookup(windows, GINT_TO_POINTER(i));
+}
+
+ProfWin *
+wins_get_by_recipient(const char * const recipient)
+{
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        if (g_strcmp0(window->from, recipient) == 0) {
+            return window;
+        }
+        curr = g_list_next(curr);
+    }
+
+    return NULL;
+}
+
+int
+wins_get_num(ProfWin *window)
+{
+    GList *keys = g_hash_table_get_keys(windows);
+    GList *curr = keys;
+
+    while (curr != NULL) {
+        gconstpointer num_p = curr->data;
+        ProfWin *curr_win = g_hash_table_lookup(windows, num_p);
+        if (g_strcmp0(curr_win->from, window->from) == 0) {
+            return GPOINTER_TO_INT(num_p);
+        }
+        curr = g_list_next(curr);
+    }
+
+    return -1;
+}
+
+int
+wins_get_current_num(void)
+{
+    return current;
+}
+
+void
+wins_close_current(void)
+{
+    wins_close_by_num(current);
+}
+
+void
+wins_close_by_num(int i)
+{
+    if (i != 1) {
+        if (i == current) {
+            current = 1;
+            wins_refresh_current();
+        }
+        g_hash_table_remove(windows, GINT_TO_POINTER(i));
+        status_bar_inactive(i);
+    }
+}
+
+void
+wins_refresh_current(void)
+{
+    ProfWin *window = wins_get_current();
+    int rows, cols;
+    getmaxyx(stdscr, rows, cols);
+    prefresh(window->win, window->y_pos, 0, 1, 0, rows-3, cols-1);
+}
+
+void
+wins_clear_current(void)
+{
+    ProfWin *window = wins_get_current();
+    werase(window->win);
+    wins_refresh_current();
+}
+
+gboolean
+wins_is_current(ProfWin *window)
+{
+    ProfWin *current_window = wins_get_current();
+
+    if (g_strcmp0(current_window->from, window->from) == 0) {
+        return TRUE;
+    } else {
+        return FALSE;
+    }
+}
+
+ProfWin *
+wins_new(const char * const from, win_type_t type)
+{
+    GList *keys = g_hash_table_get_keys(windows);
+    int result = get_next_available_win_num(keys);
+    int cols = getmaxx(stdscr);
+    ProfWin *new = win_create(from, cols, type);
+    g_hash_table_insert(windows, GINT_TO_POINTER(result), new);
+    return new;
+}
+
+int
+wins_get_total_unread(void)
+{
+    int result = 0;
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        result += window->unread;
+        curr = g_list_next(curr);
+    }
+    return result;
+}
+
+void
+wins_resize_all(void)
+{
+    int rows, cols;
+    getmaxyx(stdscr, rows, cols);
+
+    // only make the pads bigger, to avoid data loss on cropping
+    if (cols > max_cols) {
+        max_cols = cols;
+
+        GList *values = g_hash_table_get_values(windows);
+        GList *curr = values;
+
+        while (curr != NULL) {
+            ProfWin *window = curr->data;
+            wresize(window->win, PAD_SIZE, cols);
+            curr = g_list_next(curr);
+        }
+    }
+
+    ProfWin *current_win = wins_get_current();
+
+    prefresh(current_win->win, current_win->y_pos, 0, 1, 0, rows-3, cols-1);
+}
+
+void
+wins_refresh_console(void)
+{
+    if (current == 0) {
+        wins_refresh_current();
+    }
+}
+
+gboolean
+wins_duck_exists(void)
+{
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        if (window->type == WIN_DUCK)
+            return TRUE;
+        curr = g_list_next(curr);
+    }
+
+    return FALSE;
+}
+
+GSList *
+wins_get_chat_recipients(void)
+{
+    GSList *result = NULL;
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        if (window->type == WIN_CHAT) {
+            result = g_slist_append(result, window->from);
+        }
+        curr = g_list_next(curr);
+    }
+    return result;
+}
+
+GSList *
+wins_get_prune_recipients(void)
+{
+    GSList *result = NULL;
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        if (window->unread == 0 && window->type != WIN_MUC && window->type != WIN_CONSOLE) {
+            result = g_slist_append(result, window->from);
+        }
+        curr = g_list_next(curr);
+    }
+    return result;
+}
+
+void
+wins_lost_connection(void)
+{
+    GList *values = g_hash_table_get_values(windows);
+    GList *curr = values;
+
+    while (curr != NULL) {
+        ProfWin *window = curr->data;
+        if (window->type != WIN_CONSOLE) {
+            win_print_time(window, '-');
+            wattron(window->win, COLOUR_ERROR);
+            wprintw(window->win, "%s\n", "Lost connection.");
+            wattroff(window->win, COLOUR_ERROR);
+
+            // if current win, set current_win_dirty
+            if (wins_is_current(window)) {
+                wins_refresh_current();
+            }
+        }
+        curr = g_list_next(curr);
+    }
+}
+
+gboolean
+wins_tidy(void)
+{
+    gboolean tidy_required = FALSE;
+    // check for gaps
+    GList *keys = g_hash_table_get_keys(windows);
+    keys = g_list_sort(keys, cmp_win_num);
+
+    // get last used
+    GList *last = g_list_last(keys);
+    int last_num = GPOINTER_TO_INT(last->data);
+
+    // find first free num TODO - Will sort again
+    int next_available = get_next_available_win_num(keys);
+
+    // found gap (next available before last window)
+    if (cmp_win_num(GINT_TO_POINTER(next_available), GINT_TO_POINTER(last_num)) < 0) {
+        tidy_required = TRUE;
+    }
+
+    if (tidy_required) {
+        status_bar_set_all_inactive();
+        GHashTable *new_windows = g_hash_table_new_full(g_direct_hash,
+            g_direct_equal, NULL, (GDestroyNotify)win_free);
+
+        int num = 1;
+        GList *curr = keys;
+        while (curr != NULL) {
+            ProfWin *window = g_hash_table_lookup(windows, curr->data);
+            if (num == 10) {
+                g_hash_table_insert(new_windows, GINT_TO_POINTER(0), window);
+                if (window->unread > 0) {
+                    status_bar_new(0);
+                } else {
+                    status_bar_active(0);
+                }
+            } else {
+                g_hash_table_insert(new_windows, GINT_TO_POINTER(num), window);
+                if (window->unread > 0) {
+                    status_bar_new(num);
+                } else {
+                    status_bar_active(num);
+                }
+            }
+            num++;
+            curr = g_list_next(curr);
+        }
+
+        windows = new_windows;
+        return TRUE;
+    } else {
+        return FALSE;
+    }
+}
+
+GSList *
+wins_create_summary(void)
+{
+    GSList *result = NULL;
+
+    GList *keys = g_hash_table_get_keys(windows);
+    keys = g_list_sort(keys, cmp_win_num);
+    GList *curr = keys;
+
+    while (curr != NULL) {
+        ProfWin *window = g_hash_table_lookup(windows, curr->data);
+        int ui_index = GPOINTER_TO_INT(curr->data);
+
+        GString *chat_string;
+        GString *priv_string;
+        GString *muc_string;
+        GString *duck_string;
+
+        switch (window->type)
+        {
+            case WIN_CONSOLE:
+                result = g_slist_append(result, strdup("1: Console"));
+                break;
+            case WIN_CHAT:
+                chat_string = g_string_new("");
+                g_string_printf(chat_string, "%d: Chat %s", ui_index, window->from);
+                PContact contact = roster_get_contact(window->from);
+
+                if (contact != NULL) {
+                    if (p_contact_name(contact) != NULL) {
+                        GString *chat_name = g_string_new("");
+                        g_string_printf(chat_name, " (%s)", p_contact_name(contact));
+                        g_string_append(chat_string, chat_name->str);
+                        g_string_free(chat_name, TRUE);
+                    }
+                    GString *chat_presence = g_string_new("");
+                    g_string_printf(chat_presence, " - %s", p_contact_presence(contact));
+                    g_string_append(chat_string, chat_presence->str);
+                    g_string_free(chat_presence, TRUE);
+                }
+
+                if (window->unread > 0) {
+                    GString *chat_unread = g_string_new("");
+                    g_string_printf(chat_unread, ", %d unread", window->unread);
+                    g_string_append(chat_string, chat_unread->str);
+                    g_string_free(chat_unread, TRUE);
+                }
+
+                result = g_slist_append(result, strdup(chat_string->str));
+                g_string_free(chat_string, TRUE);
+
+                break;
+
+            case WIN_PRIVATE:
+                priv_string = g_string_new("");
+                g_string_printf(priv_string, "%d: Private %s", ui_index, window->from);
+
+                if (window->unread > 0) {
+                    GString *priv_unread = g_string_new("");
+                    g_string_printf(priv_unread, ", %d unread", window->unread);
+                    g_string_append(priv_string, priv_unread->str);
+                    g_string_free(priv_unread, TRUE);
+                }
+
+                result = g_slist_append(result, strdup(priv_string->str));
+                g_string_free(priv_string, TRUE);
+
+                break;
+
+            case WIN_MUC:
+                muc_string = g_string_new("");
+                g_string_printf(muc_string, "%d: Room %s", ui_index, window->from);
+
+                if (window->unread > 0) {
+                    GString *muc_unread = g_string_new("");
+                    g_string_printf(muc_unread, ", %d unread", window->unread);
+                    g_string_append(muc_string, muc_unread->str);
+                    g_string_free(muc_unread, TRUE);
+                }
+
+                result = g_slist_append(result, strdup(muc_string->str));
+                g_string_free(muc_string, TRUE);
+
+                break;
+
+            case WIN_DUCK:
+                duck_string = g_string_new("");
+                g_string_printf(duck_string, "%d: DuckDuckGo search", ui_index);
+                result = g_slist_append(result, strdup(duck_string->str));
+                g_string_free(duck_string, TRUE);
+
+                break;
+
+            default:
+                break;
+        }
+        curr = g_list_next(curr);
+    }
+
+    return result;
+}
+
+void
+wins_destroy(void)
+{
+    g_hash_table_destroy(windows);
+}