2013-08-20 18:08:21 -04:00
|
|
|
/*
|
|
|
|
* windows.c
|
|
|
|
*
|
2014-03-08 20:18:19 -05:00
|
|
|
* Copyright (C) 2012 - 2014 James Booth <boothj5@gmail.com>
|
2013-08-20 18:08:21 -04:00
|
|
|
*
|
|
|
|
* 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/>.
|
|
|
|
*
|
2014-08-24 15:57:39 -04:00
|
|
|
* In addition, as a special exception, the copyright holders give permission to
|
|
|
|
* link the code of portions of this program with the OpenSSL library under
|
|
|
|
* certain conditions as described in each individual source file, and
|
|
|
|
* distribute linked combinations including the two.
|
|
|
|
*
|
|
|
|
* You must obey the GNU General Public License in all respects for all of the
|
|
|
|
* code used other than OpenSSL. If you modify file(s) with this exception, you
|
|
|
|
* may extend this exception to your version of the file(s), but you are not
|
|
|
|
* obligated to do so. If you do not wish to do so, delete this exception
|
|
|
|
* statement from your version. If you delete this exception statement from all
|
|
|
|
* source files in the program, then also delete it here.
|
|
|
|
*
|
2013-08-20 18:08:21 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
2013-08-27 17:02:23 -04:00
|
|
|
#include "common.h"
|
2013-12-14 10:34:17 -05:00
|
|
|
#include "roster_list.h"
|
2013-08-20 18:08:21 -04:00
|
|
|
#include "config/theme.h"
|
|
|
|
#include "ui/ui.h"
|
2014-04-07 16:12:30 -04:00
|
|
|
#include "ui/statusbar.h"
|
2013-08-20 18:08:21 -04:00
|
|
|
#include "ui/window.h"
|
|
|
|
#include "ui/windows.h"
|
|
|
|
|
2013-08-26 12:29:20 -04:00
|
|
|
#define CONS_WIN_TITLE "_cons"
|
2013-08-20 18:08:21 -04:00
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
static GHashTable *windows;
|
2013-08-20 18:08:21 -04:00
|
|
|
static int current;
|
|
|
|
static int max_cols;
|
|
|
|
|
|
|
|
void
|
|
|
|
wins_init(void)
|
|
|
|
{
|
|
|
|
windows = g_hash_table_new_full(g_direct_hash, g_direct_equal, NULL,
|
2013-08-27 18:38:25 -04:00
|
|
|
(GDestroyNotify)win_free);
|
2013-08-20 18:08:21 -04:00
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
max_cols = getmaxx(stdscr);
|
2013-08-20 18:08:21 -04:00
|
|
|
int cols = getmaxx(stdscr);
|
|
|
|
ProfWin *console = win_create(CONS_WIN_TITLE, cols, WIN_CONSOLE);
|
2013-08-27 18:38:25 -04:00
|
|
|
g_hash_table_insert(windows, GINT_TO_POINTER(1), console);
|
2013-08-20 18:08:21 -04:00
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
current = 1;
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_get_console(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
return g_hash_table_lookup(windows, GINT_TO_POINTER(1));
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_get_current(void)
|
|
|
|
{
|
2014-09-16 17:23:59 -04:00
|
|
|
if (windows != NULL) {
|
|
|
|
return g_hash_table_lookup(windows, GINT_TO_POINTER(current));
|
|
|
|
} else {
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2013-08-29 19:33:46 -04:00
|
|
|
GList *
|
|
|
|
wins_get_nums(void)
|
|
|
|
{
|
|
|
|
return g_hash_table_get_keys(windows);
|
|
|
|
}
|
|
|
|
|
2013-08-20 18:08:21 -04:00
|
|
|
void
|
|
|
|
wins_set_current_by_num(int i)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
if (g_hash_table_lookup(windows, GINT_TO_POINTER(i)) != NULL) {
|
2013-08-20 18:08:21 -04:00
|
|
|
current = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_get_by_num(int i)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
return g_hash_table_lookup(windows, GINT_TO_POINTER(i));
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2013-09-25 19:25:04 -04:00
|
|
|
ProfWin *
|
|
|
|
wins_get_next(void)
|
|
|
|
{
|
|
|
|
// get and sort win nums
|
|
|
|
GList *keys = g_hash_table_get_keys(windows);
|
|
|
|
keys = g_list_sort(keys, cmp_win_num);
|
|
|
|
GList *curr = keys;
|
|
|
|
|
|
|
|
// find our place in the list
|
|
|
|
while (curr != NULL) {
|
|
|
|
if (current == GPOINTER_TO_INT(curr->data)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
curr = g_list_next(curr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is a next window return it
|
|
|
|
curr = g_list_next(curr);
|
|
|
|
if (curr != NULL) {
|
2014-07-07 16:29:51 -04:00
|
|
|
int next = GPOINTER_TO_INT(curr->data);
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2014-07-07 16:29:51 -04:00
|
|
|
return wins_get_by_num(next);
|
2013-09-25 19:25:04 -04:00
|
|
|
// otherwise return the first window (console)
|
|
|
|
} else {
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2013-09-25 19:25:04 -04:00
|
|
|
return wins_get_console();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_get_previous(void)
|
|
|
|
{
|
|
|
|
// get and sort win nums
|
|
|
|
GList *keys = g_hash_table_get_keys(windows);
|
|
|
|
keys = g_list_sort(keys, cmp_win_num);
|
|
|
|
GList *curr = keys;
|
|
|
|
|
|
|
|
// find our place in the list
|
|
|
|
while (curr != NULL) {
|
|
|
|
if (current == GPOINTER_TO_INT(curr->data)) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
curr = g_list_next(curr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// if there is a previous window return it
|
|
|
|
curr = g_list_previous(curr);
|
|
|
|
if (curr != NULL) {
|
2014-07-07 16:29:51 -04:00
|
|
|
int previous = GPOINTER_TO_INT(curr->data);
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2014-07-07 16:29:51 -04:00
|
|
|
return wins_get_by_num(previous);
|
2013-09-25 19:25:04 -04:00
|
|
|
// otherwise return the last window
|
|
|
|
} else {
|
|
|
|
int new_num = GPOINTER_TO_INT(g_list_last(keys)->data);
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2013-09-25 19:25:04 -04:00
|
|
|
return wins_get_by_num(new_num);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-20 18:08:21 -04:00
|
|
|
ProfWin *
|
|
|
|
wins_get_by_recipient(const char * const recipient)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
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) {
|
2014-06-15 16:36:47 -04:00
|
|
|
g_list_free(values);
|
2013-08-27 18:38:25 -04:00
|
|
|
return window;
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2014-06-15 16:04:07 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
wins_get_num(ProfWin *window)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
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) {
|
2014-06-15 16:36:47 -04:00
|
|
|
g_list_free(keys);
|
2013-08-27 18:38:25 -04:00
|
|
|
return GPOINTER_TO_INT(num_p);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2014-06-15 16:36:47 -04:00
|
|
|
g_list_free(keys);
|
2013-08-27 18:38:25 -04:00
|
|
|
return -1;
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
{
|
2014-04-06 16:06:50 -04:00
|
|
|
// console cannot be closed
|
2013-08-27 18:38:25 -04:00
|
|
|
if (i != 1) {
|
2014-04-06 16:06:50 -04:00
|
|
|
|
|
|
|
// go to console if closing current window
|
2013-08-20 18:08:21 -04:00
|
|
|
if (i == current) {
|
2013-08-27 18:38:25 -04:00
|
|
|
current = 1;
|
2014-04-06 16:06:50 -04:00
|
|
|
ProfWin *window = wins_get_current();
|
|
|
|
win_update_virtual(window);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-04-06 16:06:50 -04:00
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
g_hash_table_remove(windows, GINT_TO_POINTER(i));
|
2013-08-29 19:21:41 -04:00
|
|
|
status_bar_inactive(i);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
wins_clear_current(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
ProfWin *window = wins_get_current();
|
|
|
|
werase(window->win);
|
2014-04-06 16:06:50 -04:00
|
|
|
win_update_virtual(window);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
wins_is_current(ProfWin *window)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
ProfWin *current_window = wins_get_current();
|
|
|
|
|
|
|
|
if (g_strcmp0(current_window->from, window->from) == 0) {
|
2013-08-20 18:08:21 -04:00
|
|
|
return TRUE;
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_new(const char * const from, win_type_t type)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
2014-06-15 16:23:21 -04:00
|
|
|
g_list_free(keys);
|
2013-08-27 18:38:25 -04:00
|
|
|
return new;
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
wins_get_total_unread(void)
|
|
|
|
{
|
|
|
|
int result = 0;
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-06-17 17:01:01 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
wins_resize_all(void)
|
|
|
|
{
|
|
|
|
int rows, cols;
|
|
|
|
getmaxyx(stdscr, rows, cols);
|
|
|
|
|
2014-06-29 09:04:23 -04:00
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
while (curr != NULL) {
|
2014-10-07 12:06:02 -04:00
|
|
|
ProfWin *window = curr->data;
|
|
|
|
if (window->type == WIN_MUC) {
|
|
|
|
wresize(window->win, PAD_SIZE, (cols/OCCUPANT_WIN_SIZE) * (OCCUPANT_WIN_SIZE-1));
|
|
|
|
} else {
|
|
|
|
wresize(window->win, PAD_SIZE, cols);
|
|
|
|
}
|
|
|
|
win_redraw(window);
|
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-06-29 09:04:23 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
ProfWin *current_win = wins_get_current();
|
2014-10-07 12:06:02 -04:00
|
|
|
if (current_win->type == WIN_MUC) {
|
|
|
|
pnoutrefresh(current_win->win, current_win->y_pos, 0, 1, 0, rows-3, ((cols/OCCUPANT_WIN_SIZE) * (OCCUPANT_WIN_SIZE-1)) -1);
|
|
|
|
pnoutrefresh(current_win->subwin, 0, 0, 1, (cols/OCCUPANT_WIN_SIZE) * (OCCUPANT_WIN_SIZE-1), rows-3, cols-1);
|
|
|
|
} else {
|
|
|
|
pnoutrefresh(current_win->win, current_win->y_pos, 0, 1, 0, rows-3, cols-1);
|
|
|
|
}
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
wins_duck_exists(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
|
|
|
|
while (curr != NULL) {
|
|
|
|
ProfWin *window = curr->data;
|
2014-06-17 18:14:54 -04:00
|
|
|
if (window->type == WIN_DUCK) {
|
|
|
|
g_list_free(values);
|
2013-08-27 18:38:25 -04:00
|
|
|
return TRUE;
|
2014-06-17 18:14:54 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-04-15 08:16:32 -04:00
|
|
|
gboolean
|
|
|
|
wins_xmlconsole_exists(void)
|
|
|
|
{
|
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
|
|
|
|
while (curr != NULL) {
|
|
|
|
ProfWin *window = curr->data;
|
2014-06-15 12:43:16 -04:00
|
|
|
if (window->type == WIN_XML) {
|
|
|
|
g_list_free(values);
|
2014-04-15 08:16:32 -04:00
|
|
|
return TRUE;
|
2014-06-15 12:43:16 -04:00
|
|
|
}
|
2014-04-15 08:16:32 -04:00
|
|
|
curr = g_list_next(curr);
|
|
|
|
}
|
|
|
|
|
2014-06-15 12:43:16 -04:00
|
|
|
g_list_free(values);
|
2014-04-15 08:16:32 -04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
ProfWin *
|
|
|
|
wins_get_xmlconsole(void)
|
|
|
|
{
|
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
|
|
|
|
while (curr != NULL) {
|
|
|
|
ProfWin *window = curr->data;
|
2014-06-17 18:14:54 -04:00
|
|
|
if (window->type == WIN_XML) {
|
|
|
|
g_list_free(values);
|
2014-04-15 08:16:32 -04:00
|
|
|
return window;
|
2014-06-17 18:14:54 -04:00
|
|
|
}
|
2014-04-15 08:16:32 -04:00
|
|
|
curr = g_list_next(curr);
|
|
|
|
}
|
|
|
|
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(values);
|
2014-04-15 08:16:32 -04:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-08-20 18:08:21 -04:00
|
|
|
GSList *
|
|
|
|
wins_get_chat_recipients(void)
|
|
|
|
{
|
|
|
|
GSList *result = NULL;
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-06-17 17:27:14 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
GSList *
|
|
|
|
wins_get_prune_recipients(void)
|
|
|
|
{
|
|
|
|
GSList *result = NULL;
|
2013-08-27 18:38:25 -04:00
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
|
|
|
|
while (curr != NULL) {
|
|
|
|
ProfWin *window = curr->data;
|
2014-09-06 19:49:42 -04:00
|
|
|
if (window->unread == 0 &&
|
|
|
|
window->type != WIN_MUC &&
|
|
|
|
window->type != WIN_MUC_CONFIG &&
|
|
|
|
window->type != WIN_XML &&
|
|
|
|
window->type != WIN_CONSOLE) {
|
2013-08-27 18:38:25 -04:00
|
|
|
result = g_slist_append(result, window->from);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
wins_lost_connection(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
GList *values = g_hash_table_get_values(windows);
|
|
|
|
GList *curr = values;
|
|
|
|
|
|
|
|
while (curr != NULL) {
|
|
|
|
ProfWin *window = curr->data;
|
|
|
|
if (window->type != WIN_CONSOLE) {
|
2014-06-21 04:16:50 -04:00
|
|
|
win_save_print(window, '-', NULL, 0, COLOUR_ERROR, "", "Lost connection.");
|
2013-08-20 18:08:21 -04:00
|
|
|
|
|
|
|
// if current win, set current_win_dirty
|
|
|
|
if (wins_is_current(window)) {
|
2014-04-06 16:06:50 -04:00
|
|
|
win_update_virtual(window);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(values);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2014-04-24 16:50:59 -04:00
|
|
|
gboolean
|
|
|
|
wins_swap(int source_win, int target_win)
|
|
|
|
{
|
|
|
|
ProfWin *source = g_hash_table_lookup(windows, GINT_TO_POINTER(source_win));
|
|
|
|
|
|
|
|
if (source != NULL) {
|
|
|
|
ProfWin *target = g_hash_table_lookup(windows, GINT_TO_POINTER(target_win));
|
|
|
|
|
2014-04-24 18:06:42 -04:00
|
|
|
// target window empty
|
2014-04-24 16:50:59 -04:00
|
|
|
if (target == NULL) {
|
|
|
|
g_hash_table_steal(windows, GINT_TO_POINTER(source_win));
|
|
|
|
status_bar_inactive(source_win);
|
|
|
|
g_hash_table_insert(windows, GINT_TO_POINTER(target_win), source);
|
|
|
|
if (source->unread > 0) {
|
|
|
|
status_bar_new(target_win);
|
|
|
|
} else {
|
|
|
|
status_bar_active(target_win);
|
|
|
|
}
|
|
|
|
if ((wins_get_current_num() == source_win) || (wins_get_current_num() == target_win)) {
|
|
|
|
ui_switch_win(1);
|
|
|
|
}
|
|
|
|
return TRUE;
|
2014-04-24 18:06:42 -04:00
|
|
|
|
|
|
|
// target window occupied
|
2014-04-24 16:50:59 -04:00
|
|
|
} else {
|
2014-04-24 18:06:42 -04:00
|
|
|
g_hash_table_steal(windows, GINT_TO_POINTER(source_win));
|
|
|
|
g_hash_table_steal(windows, GINT_TO_POINTER(target_win));
|
|
|
|
g_hash_table_insert(windows, GINT_TO_POINTER(source_win), target);
|
|
|
|
g_hash_table_insert(windows, GINT_TO_POINTER(target_win), source);
|
|
|
|
if (source->unread > 0) {
|
|
|
|
status_bar_new(target_win);
|
|
|
|
} else {
|
|
|
|
status_bar_active(target_win);
|
|
|
|
}
|
|
|
|
if (target->unread > 0) {
|
|
|
|
status_bar_new(source_win);
|
|
|
|
} else {
|
|
|
|
status_bar_active(source_win);
|
|
|
|
}
|
|
|
|
if ((wins_get_current_num() == source_win) || (wins_get_current_num() == target_win)) {
|
|
|
|
ui_switch_win(1);
|
|
|
|
}
|
|
|
|
return TRUE;
|
2014-04-24 16:50:59 -04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-20 18:08:21 -04:00
|
|
|
gboolean
|
|
|
|
wins_tidy(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
|
|
|
|
2013-08-28 16:45:46 -04:00
|
|
|
// found gap (next available before last window)
|
|
|
|
if (cmp_win_num(GINT_TO_POINTER(next_available), GINT_TO_POINTER(last_num)) < 0) {
|
2013-08-27 18:38:25 -04:00
|
|
|
tidy_required = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tidy_required) {
|
2013-08-29 19:49:38 -04:00
|
|
|
status_bar_set_all_inactive();
|
2013-08-27 18:38:25 -04:00
|
|
|
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) {
|
2013-08-28 18:09:54 -04:00
|
|
|
ProfWin *window = g_hash_table_lookup(windows, curr->data);
|
2013-08-27 18:38:25 -04:00
|
|
|
if (num == 10) {
|
2013-08-28 18:09:54 -04:00
|
|
|
g_hash_table_insert(new_windows, GINT_TO_POINTER(0), window);
|
2013-08-27 18:38:25 -04:00
|
|
|
if (window->unread > 0) {
|
|
|
|
status_bar_new(0);
|
|
|
|
} else {
|
|
|
|
status_bar_active(0);
|
|
|
|
}
|
|
|
|
} else {
|
2013-08-28 18:09:54 -04:00
|
|
|
g_hash_table_insert(new_windows, GINT_TO_POINTER(num), window);
|
2013-08-27 18:38:25 -04:00
|
|
|
if (window->unread > 0) {
|
|
|
|
status_bar_new(num);
|
|
|
|
} else {
|
|
|
|
status_bar_active(num);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
num++;
|
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
windows = new_windows;
|
2013-10-02 19:02:30 -04:00
|
|
|
current = 1;
|
|
|
|
ui_switch_win(1);
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2013-08-27 18:38:25 -04:00
|
|
|
return TRUE;
|
|
|
|
} else {
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2013-08-27 18:38:25 -04:00
|
|
|
return FALSE;
|
|
|
|
}
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
GSList *
|
|
|
|
wins_create_summary(void)
|
|
|
|
{
|
|
|
|
GSList *result = NULL;
|
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
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;
|
2014-09-06 19:49:42 -04:00
|
|
|
GString *muc_config_string;
|
2013-08-27 18:38:25 -04:00
|
|
|
GString *duck_string;
|
2014-04-15 08:16:32 -04:00
|
|
|
GString *xml_string;
|
2013-08-27 18:38:25 -04:00
|
|
|
|
|
|
|
switch (window->type)
|
|
|
|
{
|
|
|
|
case WIN_CONSOLE:
|
|
|
|
result = g_slist_append(result, strdup("1: Console"));
|
|
|
|
break;
|
|
|
|
case WIN_CHAT:
|
|
|
|
chat_string = g_string_new("");
|
|
|
|
|
2013-08-31 10:07:05 -04:00
|
|
|
PContact contact = roster_get_contact(window->from);
|
|
|
|
if (contact == NULL) {
|
|
|
|
g_string_printf(chat_string, "%d: Chat %s", ui_index, window->from);
|
|
|
|
} else {
|
|
|
|
const char *display_name = p_contact_name_or_jid(contact);
|
|
|
|
g_string_printf(chat_string, "%d: Chat %s", ui_index, display_name);
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
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);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2014-09-06 19:49:42 -04:00
|
|
|
case WIN_MUC_CONFIG:
|
|
|
|
muc_config_string = g_string_new("");
|
|
|
|
g_string_printf(muc_config_string, "%d: %s", ui_index, window->from);
|
2014-09-16 19:49:06 -04:00
|
|
|
if ((window->form != NULL) && (window->form->modified)) {
|
|
|
|
g_string_append(muc_config_string, " *");
|
|
|
|
}
|
2014-09-06 19:49:42 -04:00
|
|
|
result = g_slist_append(result, strdup(muc_config_string->str));
|
|
|
|
g_string_free(muc_config_string, TRUE);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
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;
|
|
|
|
|
2014-04-15 08:16:32 -04:00
|
|
|
case WIN_XML:
|
|
|
|
xml_string = g_string_new("");
|
|
|
|
g_string_printf(xml_string, "%d: XML console", ui_index);
|
|
|
|
result = g_slist_append(result, strdup(xml_string->str));
|
|
|
|
g_string_free(xml_string, TRUE);
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2013-08-27 18:38:25 -04:00
|
|
|
default:
|
|
|
|
break;
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
2013-08-27 18:38:25 -04:00
|
|
|
curr = g_list_next(curr);
|
2013-08-20 18:08:21 -04:00
|
|
|
}
|
|
|
|
|
2014-06-17 18:14:54 -04:00
|
|
|
g_list_free(keys);
|
2013-08-20 18:08:21 -04:00
|
|
|
return result;
|
|
|
|
}
|
2013-08-26 12:47:29 -04:00
|
|
|
|
|
|
|
void
|
|
|
|
wins_destroy(void)
|
|
|
|
{
|
2013-08-27 18:38:25 -04:00
|
|
|
g_hash_table_destroy(windows);
|
2013-08-26 12:47:29 -04:00
|
|
|
}
|