1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-11-03 19:37:16 -05:00
profanity/windows.c

614 lines
16 KiB
C
Raw Normal View History

2012-02-20 15:07:38 -05:00
/*
* windows.c
*
* Copyright (C) 2012 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 <string.h>
#include <stdlib.h>
2012-05-03 20:00:01 -04:00
2012-02-05 18:08:15 -05:00
#include <ncurses.h>
2012-05-09 19:30:03 -04:00
#include <glib.h>
2012-05-03 20:00:01 -04:00
2012-05-23 20:11:43 -04:00
#include "ui.h"
2012-02-08 19:48:17 -05:00
#include "util.h"
2012-05-03 20:00:01 -04:00
#include "contact.h"
#include "preferences.h"
2012-02-05 18:08:15 -05:00
2012-02-29 19:34:54 -05:00
#define CONS_WIN_TITLE "_cons"
2012-03-04 19:06:24 -05:00
#define PAD_SIZE 200
2012-02-29 20:02:10 -05:00
#define NUM_WINS 10
2012-02-29 19:34:54 -05:00
2012-02-29 19:40:51 -05:00
// holds console at index 0 and chat wins 1 through to 9
2012-02-29 20:02:10 -05:00
static struct prof_win _wins[NUM_WINS];
2012-02-29 19:40:51 -05:00
// the window currently being displayed
2012-02-29 19:34:54 -05:00
static int _curr_prof_win = 0;
2012-02-29 19:40:51 -05:00
// shortcut pointer to console window
2012-02-29 19:34:54 -05:00
static WINDOW * _cons_win = NULL;
2012-02-05 18:08:15 -05:00
// current window state
static int dirty;
// max columns for main windows, never resize below
static int max_cols = 0;
2012-02-12 17:09:07 -05:00
static void _create_windows(void);
2012-03-08 20:06:55 -05:00
static int _find_prof_win_index(const char * const contact);
static int _new_prof_win(const char * const contact);
static void _current_window_refresh(void);
static void _win_switch_if_active(const int i);
2012-02-29 19:34:54 -05:00
static void _win_show_time(WINDOW *win);
2012-03-08 20:06:55 -05:00
static void _win_show_user(WINDOW *win, const char * const user, const int colour);
static void _win_show_message(WINDOW *win, const char * const message);
static void _show_status_string(WINDOW *win, const char * const from,
const char * const show, const char * const status, const char * const pre,
const char * const default_show);
static void _cons_show_incoming_message(const char * const short_from,
const int win_index);
2012-03-10 15:46:30 -05:00
static void _win_handle_switch(const int * const ch);
static void _win_handle_page(const int * const ch);
2012-04-22 17:35:54 -04:00
static void _win_resize_all(void);
2012-02-05 18:29:09 -05:00
2012-02-06 19:08:59 -05:00
void gui_init(void)
2012-02-05 18:29:09 -05:00
{
initscr();
cbreak();
keypad(stdscr, TRUE);
2012-02-18 16:02:51 -05:00
if (has_colors()) {
2012-04-04 04:51:30 -04:00
use_default_colors();
2012-02-18 16:02:51 -05:00
start_color();
2012-04-04 04:51:30 -04:00
init_pair(1, COLOR_WHITE, -1);
init_pair(2, COLOR_GREEN, -1);
2012-02-18 16:02:51 -05:00
init_pair(3, COLOR_WHITE, COLOR_BLUE);
init_pair(4, COLOR_CYAN, COLOR_BLUE);
2012-04-04 04:51:30 -04:00
init_pair(5, COLOR_CYAN, -1);
init_pair(6, COLOR_RED, -1);
init_pair(7, COLOR_MAGENTA, -1);
init_pair(8, COLOR_YELLOW, -1);
2012-02-18 16:02:51 -05:00
}
2012-02-05 18:29:09 -05:00
refresh();
create_title_bar();
2012-02-12 17:20:21 -05:00
create_status_bar();
create_input_window();
2012-02-12 17:09:07 -05:00
_create_windows();
dirty = TRUE;
2012-02-06 19:08:59 -05:00
}
void gui_refresh(void)
{
title_bar_refresh();
status_bar_refresh();
if (dirty) {
_current_window_refresh();
dirty = FALSE;
}
inp_put_back();
}
2012-02-06 19:08:59 -05:00
void gui_close(void)
{
endwin();
}
2012-04-17 18:28:21 -04:00
void gui_resize(const int ch, const char * const input, const int size)
{
title_bar_resize();
2012-04-21 20:25:19 -04:00
status_bar_resize();
2012-04-22 17:35:54 -04:00
_win_resize_all();
2012-04-22 15:59:36 -04:00
inp_win_resize(input, size);
2012-04-17 18:28:21 -04:00
dirty = TRUE;
}
2012-02-29 19:40:51 -05:00
int win_close_win(void)
2012-02-08 18:12:34 -05:00
{
2012-02-29 19:40:51 -05:00
if (win_in_chat()) {
// reset the chat win to unused
strcpy(_wins[_curr_prof_win].from, "");
wclear(_wins[_curr_prof_win].win);
2012-02-08 18:12:34 -05:00
2012-02-29 19:40:51 -05:00
// set it as inactive in the status bar
status_bar_inactive(_curr_prof_win);
// go back to console window
_curr_prof_win = 0;
title_bar_title();
dirty = TRUE;
2012-02-08 18:12:34 -05:00
2012-02-29 19:40:51 -05:00
// success
return 1;
} else {
// didn't close anything
return 0;
}
2012-02-08 18:12:34 -05:00
}
2012-02-12 17:09:07 -05:00
int win_in_chat(void)
2012-02-08 17:46:35 -05:00
{
2012-02-29 19:34:54 -05:00
return ((_curr_prof_win != 0) &&
(strcmp(_wins[_curr_prof_win].from, "") != 0));
2012-02-08 17:46:35 -05:00
}
char *win_get_recipient(void)
2012-02-08 17:46:35 -05:00
{
2012-02-29 19:34:54 -05:00
struct prof_win current = _wins[_curr_prof_win];
char *recipient = (char *) malloc(sizeof(char) * (strlen(current.from) + 1));
strcpy(recipient, current.from);
return recipient;
2012-02-08 17:46:35 -05:00
}
2012-03-08 20:06:55 -05:00
void win_show_incomming_msg(const char * const from, const char * const message)
2012-02-26 13:26:38 -05:00
{
char from_cpy[strlen(from) + 1];
strcpy(from_cpy, from);
char *short_from = strtok(from_cpy, "/");
2012-02-29 19:57:35 -05:00
int win_index = _find_prof_win_index(short_from);
2012-02-29 20:02:10 -05:00
if (win_index == NUM_WINS)
2012-02-29 19:57:35 -05:00
win_index = _new_prof_win(short_from);
2012-02-29 19:34:54 -05:00
WINDOW *win = _wins[win_index].win;
2012-02-26 13:26:38 -05:00
_win_show_time(win);
2012-02-26 13:35:40 -05:00
_win_show_user(win, short_from, 1);
2012-02-29 19:34:54 -05:00
_win_show_message(win, message);
if (win_index == _curr_prof_win) {
status_bar_active(win_index);
dirty = TRUE;
} else {
status_bar_new(win_index);
_cons_show_incoming_message(short_from, win_index);
}
2012-04-23 20:11:39 -04:00
if (prefs_get_beep())
2012-04-23 20:24:54 -04:00
beep();
2012-02-06 19:08:59 -05:00
}
2012-03-08 20:06:55 -05:00
void win_show_outgoing_msg(const char * const from, const char * const to,
const char * const message)
2012-02-06 19:37:42 -05:00
{
2012-02-29 19:57:35 -05:00
int win_index = _find_prof_win_index(to);
2012-02-29 20:02:10 -05:00
if (win_index == NUM_WINS)
2012-02-29 19:57:35 -05:00
win_index = _new_prof_win(to);
2012-02-29 19:34:54 -05:00
WINDOW *win = _wins[win_index].win;
2012-02-26 13:26:38 -05:00
_win_show_time(win);
2012-02-26 13:35:40 -05:00
_win_show_user(win, from, 0);
2012-02-29 19:34:54 -05:00
_win_show_message(win, message);
2012-02-26 13:35:40 -05:00
2012-02-29 19:34:54 -05:00
status_bar_active(win_index);
if (win_index == _curr_prof_win) {
dirty = TRUE;
} else {
status_bar_new(win_index);
}
2012-02-06 19:37:42 -05:00
}
2012-03-08 20:06:55 -05:00
void win_contact_online(const char * const from, const char * const show,
const char * const status)
{
2012-02-29 20:37:09 -05:00
_show_status_string(_cons_win, from, show, status, "++", "online");
int win_index = _find_prof_win_index(from);
if (win_index != NUM_WINS) {
WINDOW *win = _wins[win_index].win;
_show_status_string(win, from, show, status, "++", "online");
}
if (win_index == _curr_prof_win)
dirty = TRUE;
}
2012-03-08 20:06:55 -05:00
void win_contact_offline(const char * const from, const char * const show,
const char * const status)
{
2012-02-29 20:37:09 -05:00
_show_status_string(_cons_win, from, show, status, "--", "offline");
int win_index = _find_prof_win_index(from);
if (win_index != NUM_WINS) {
WINDOW *win = _wins[win_index].win;
_show_status_string(win, from, show, status, "--", "offline");
}
if (win_index == _curr_prof_win)
dirty = TRUE;
}
void win_disconnected(void)
{
int i;
// show message in all active chats
for (i = 1; i < NUM_WINS; i++) {
if (strcmp(_wins[i].from, "") != 0) {
WINDOW *win = _wins[_curr_prof_win].win;
_win_show_time(win);
wattron(win, COLOR_PAIR(6));
wprintw(win, "%s\n", "Lost connection.");
wattroff(win, COLOR_PAIR(6));
// if current win, set dirty
if (i == _curr_prof_win) {
dirty = TRUE;
}
}
}
}
2012-02-12 17:23:51 -05:00
void cons_help(void)
{
2012-03-10 19:39:13 -05:00
cons_show("");
2012-05-28 18:40:11 -04:00
cons_show("Basic Commands:");
2012-03-10 19:39:13 -05:00
cons_show("");
2012-03-08 18:03:26 -05:00
cons_show("/help : This help.");
cons_show("/connect user@host : Login to jabber.");
cons_show("/msg user@host mesg : Send mesg to user.");
2012-05-28 18:40:11 -04:00
cons_show("/close : Close a chat window.");
cons_show("/who : Find out who is online.");
cons_show("/ros : List all contacts.");
2012-05-28 18:40:11 -04:00
cons_show("/quit : Quit Profanity.");
cons_show("");
cons_show("Settings:");
cons_show("");
2012-04-23 20:24:54 -04:00
cons_show("/beep <on/off> : Enable/disable sound notification");
2012-04-23 20:39:23 -04:00
cons_show("/flash <on/off> : Enable/disable screen flash notification");
2012-05-28 18:40:11 -04:00
cons_show("");
cons_show("Status changes (msg is optional):");
cons_show("");
cons_show("/away <msg> : Set status to away.");
cons_show("/online <msg> : Set status to online.");
cons_show("/dnd <msg> : Set status to dnd (do not disturb).");
cons_show("/chat <msg> : Set status to chat (available for chat).");
cons_show("/xa <msg> : Set status to xa (extended away).");
2012-03-10 19:39:13 -05:00
cons_show("");
cons_show("Keys:");
cons_show("");
2012-03-08 18:03:26 -05:00
cons_show("F1 : This console window.");
2012-03-10 19:39:13 -05:00
cons_show("F2-F10 : Chat windows.");
2012-03-08 18:03:26 -05:00
cons_show("UP, DOWN : Navigate input history.");
cons_show("LEFT, RIGHT : Edit current input.");
2012-03-10 19:39:13 -05:00
cons_show("TAB : Autocomplete recipient.");
2012-03-08 18:03:26 -05:00
cons_show("PAGE UP, PAGE DOWN : Page the chat window.");
2012-05-28 18:40:11 -04:00
cons_show("");
if (_curr_prof_win == 0)
dirty = TRUE;
2012-02-08 21:47:25 -05:00
}
2012-05-09 19:30:03 -04:00
void cons_show_online_contacts(GSList *list)
2012-03-08 18:03:26 -05:00
{
_win_show_time(_cons_win);
wprintw(_cons_win, "Online contacts:\n");
2012-05-09 19:30:03 -04:00
GSList *curr = list;
while(curr) {
2012-05-09 19:30:03 -04:00
PContact contact = curr->data;
_win_show_time(_cons_win);
wattron(_cons_win, COLOR_PAIR(2));
2012-05-03 20:00:01 -04:00
wprintw(_cons_win, "%s", p_contact_name(contact));
if (p_contact_show(contact))
wprintw(_cons_win, " is %s", p_contact_show(contact));
if (p_contact_status(contact))
wprintw(_cons_win, ", \"%s\"", p_contact_status(contact));
2012-03-09 18:07:53 -05:00
wprintw(_cons_win, "\n");
wattroff(_cons_win, COLOR_PAIR(2));
2012-05-09 19:30:03 -04:00
curr = g_slist_next(curr);
2012-03-08 18:03:26 -05:00
}
}
2012-03-08 20:06:55 -05:00
void cons_bad_show(const char * const msg)
2012-02-18 19:43:35 -05:00
{
2012-02-29 19:34:54 -05:00
_win_show_time(_cons_win);
wattron(_cons_win, COLOR_PAIR(6));
wprintw(_cons_win, "%s\n", msg);
wattroff(_cons_win, COLOR_PAIR(6));
if (_curr_prof_win == 0)
dirty = TRUE;
2012-02-18 19:43:35 -05:00
}
2012-03-08 20:06:55 -05:00
void cons_show(const char * const msg)
2012-02-18 19:43:35 -05:00
{
2012-02-29 19:34:54 -05:00
_win_show_time(_cons_win);
wprintw(_cons_win, "%s\n", msg);
if (_curr_prof_win == 0)
dirty = TRUE;
2012-02-18 19:43:35 -05:00
}
2012-03-08 20:06:55 -05:00
void cons_bad_command(const char * const cmd)
{
2012-02-29 19:34:54 -05:00
_win_show_time(_cons_win);
wprintw(_cons_win, "Unknown command: %s\n", cmd);
if (_curr_prof_win == 0)
dirty = TRUE;
2012-02-05 18:29:09 -05:00
}
2012-03-10 15:46:30 -05:00
void win_handle_special_keys(const int * const ch)
2012-02-26 21:01:19 -05:00
{
2012-03-10 15:46:30 -05:00
_win_handle_switch(ch);
_win_handle_page(ch);
2012-02-26 21:01:19 -05:00
}
2012-03-04 19:06:24 -05:00
void win_page_off(void)
{
int rows, cols;
getmaxyx(stdscr, rows, cols);
_wins[_curr_prof_win].paged = 0;
int y, x;
getyx(_wins[_curr_prof_win].win, y, x);
2012-03-04 19:06:24 -05:00
int size = rows - 3;
2012-03-04 19:06:24 -05:00
_wins[_curr_prof_win].y_pos = y - (size - 1);
if (_wins[_curr_prof_win].y_pos < 0)
_wins[_curr_prof_win].y_pos = 0;
dirty = TRUE;
2012-03-04 19:06:24 -05:00
}
2012-02-12 17:09:07 -05:00
static void _create_windows(void)
{
int rows, cols;
getmaxyx(stdscr, rows, cols);
max_cols = cols;
// create the console window in 0
struct prof_win cons;
2012-02-29 19:34:54 -05:00
strcpy(cons.from, CONS_WIN_TITLE);
2012-03-04 19:06:24 -05:00
cons.win = newpad(PAD_SIZE, cols);
cons.y_pos = 0;
cons.paged = 0;
scrollok(cons.win, TRUE);
2012-02-08 19:48:17 -05:00
2012-02-12 17:09:07 -05:00
_wins[0] = cons;
2012-02-29 19:34:54 -05:00
_cons_win = _wins[0].win;
wattrset(_cons_win, A_BOLD);
_win_show_time(_cons_win);
wprintw(_cons_win, "Welcome to Profanity.\n");
2012-03-04 19:06:24 -05:00
prefresh(_cons_win, 0, 0, 1, 0, rows-3, cols-1);
dirty = TRUE;
// create the chat windows
int i;
2012-02-29 20:02:10 -05:00
for (i = 1; i < NUM_WINS; i++) {
struct prof_win chat;
strcpy(chat.from, "");
chat.win = newpad(PAD_SIZE, cols);
chat.y_pos = 0;
chat.paged = 0;
2012-02-29 18:40:30 -05:00
wattrset(chat.win, A_BOLD);
scrollok(chat.win, TRUE);
2012-02-12 17:09:07 -05:00
_wins[i] = chat;
}
}
2012-02-12 19:53:10 -05:00
2012-03-08 20:06:55 -05:00
static int _find_prof_win_index(const char * const contact)
2012-02-12 19:53:10 -05:00
{
// find the chat window for recipient
int i;
2012-02-29 20:02:10 -05:00
for (i = 1; i < NUM_WINS; i++)
2012-02-12 19:53:10 -05:00
if (strcmp(_wins[i].from, contact) == 0)
break;
2012-02-29 20:02:10 -05:00
return i;
2012-02-29 19:57:35 -05:00
}
2012-02-12 19:53:10 -05:00
2012-03-08 20:06:55 -05:00
static int _new_prof_win(const char * const contact)
2012-02-29 19:57:35 -05:00
{
int i;
// find the first unused one
2012-02-29 20:02:10 -05:00
for (i = 1; i < NUM_WINS; i++)
2012-02-29 19:57:35 -05:00
if (strcmp(_wins[i].from, "") == 0)
break;
// set it up
strcpy(_wins[i].from, contact);
wclear(_wins[i].win);
2012-02-12 20:05:11 -05:00
2012-02-18 16:23:26 -05:00
return i;
2012-02-12 19:53:10 -05:00
}
2012-03-08 20:06:55 -05:00
static void _win_switch_if_active(const int i)
2012-02-26 21:01:19 -05:00
{
win_page_off();
2012-02-26 21:01:19 -05:00
if (strcmp(_wins[i].from, "") != 0) {
2012-02-29 19:34:54 -05:00
_curr_prof_win = i;
win_page_off();
2012-02-26 21:01:19 -05:00
if (i == 0) {
2012-02-26 21:01:19 -05:00
title_bar_title();
} else {
2012-02-26 21:01:19 -05:00
title_bar_show(_wins[i].from);
status_bar_active(i);
}
2012-02-26 21:01:19 -05:00
}
dirty = TRUE;
2012-02-26 21:01:19 -05:00
}
2012-02-29 19:34:54 -05:00
static void _win_show_time(WINDOW *win)
2012-02-26 13:35:40 -05:00
{
char tstmp[80];
get_time(tstmp);
2012-02-29 19:34:54 -05:00
wprintw(win, "%s - ", tstmp);
2012-02-26 13:35:40 -05:00
}
2012-03-08 20:06:55 -05:00
static void _win_show_user(WINDOW *win, const char * const user, const int colour)
2012-02-26 13:35:40 -05:00
{
if (colour)
2012-02-29 19:34:54 -05:00
wattron(win, COLOR_PAIR(2));
wprintw(win, "%s: ", user);
2012-02-26 13:35:40 -05:00
if (colour)
2012-02-29 19:34:54 -05:00
wattroff(win, COLOR_PAIR(2));
2012-02-26 13:35:40 -05:00
}
2012-03-08 20:06:55 -05:00
static void _win_show_message(WINDOW *win, const char * const message)
2012-02-29 18:46:25 -05:00
{
wattroff(win, A_BOLD);
wprintw(win, "%s\n", message);
wattron(win, A_BOLD);
}
2012-04-22 17:35:54 -04:00
static void _current_window_refresh(void)
2012-02-12 19:59:04 -05:00
{
2012-03-04 19:06:24 -05:00
int rows, cols;
getmaxyx(stdscr, rows, cols);
WINDOW *current = _wins[_curr_prof_win].win;
prefresh(current, _wins[_curr_prof_win].y_pos, 0, 1, 0, rows-3, cols-1);
2012-02-12 19:59:04 -05:00
}
2012-04-22 17:35:54 -04:00
void _win_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;
int i;
for (i = 0; i < NUM_WINS; i++) {
wresize(_wins[i].win, PAD_SIZE, cols);
}
}
2012-04-22 17:35:54 -04:00
WINDOW *current = _wins[_curr_prof_win].win;
prefresh(current, _wins[_curr_prof_win].y_pos, 0, 1, 0, rows-3, cols-1);
}
2012-03-08 20:06:55 -05:00
static void _show_status_string(WINDOW *win, const char * const from,
const char * const show, const char * const status, const char * const pre,
const char * const default_show)
{
2012-02-29 20:37:09 -05:00
_win_show_time(win);
if (strcmp(default_show, "online") == 0) {
2012-02-29 19:34:54 -05:00
wattron(win, COLOR_PAIR(2));
2012-02-29 20:37:09 -05:00
} else {
2012-02-29 19:34:54 -05:00
wattron(win, COLOR_PAIR(5));
wattroff(win, A_BOLD);
}
2012-02-29 20:37:09 -05:00
wprintw(win, "%s %s", pre, from);
if (show != NULL)
2012-02-29 20:37:09 -05:00
wprintw(win, " is %s", show);
else
2012-02-29 20:37:09 -05:00
wprintw(win, " is %s", default_show);
if (status != NULL)
2012-02-29 20:37:09 -05:00
wprintw(win, ", \"%s\"", status);
2012-02-29 20:37:09 -05:00
wprintw(win, "\n");
2012-02-29 20:37:09 -05:00
if (strcmp(default_show, "online") == 0) {
wattroff(win, COLOR_PAIR(2));
} else {
wattroff(win, COLOR_PAIR(5));
wattron(win, A_BOLD);
}
}
2012-03-08 20:06:55 -05:00
static void _cons_show_incoming_message(const char * const short_from, const int win_index)
{
_win_show_time(_cons_win);
wattron(_cons_win, COLOR_PAIR(8));
wprintw(_cons_win, "<< incoming from %s (%d)\n", short_from, win_index + 1);
wattroff(_cons_win, COLOR_PAIR(8));
}
2012-03-10 15:46:30 -05:00
static void _win_handle_switch(const int * const ch)
{
if (*ch == KEY_F(1)) {
_win_switch_if_active(0);
} else if (*ch == KEY_F(2)) {
_win_switch_if_active(1);
} else if (*ch == KEY_F(3)) {
_win_switch_if_active(2);
} else if (*ch == KEY_F(4)) {
_win_switch_if_active(3);
} else if (*ch == KEY_F(5)) {
_win_switch_if_active(4);
} else if (*ch == KEY_F(6)) {
_win_switch_if_active(5);
} else if (*ch == KEY_F(7)) {
_win_switch_if_active(6);
} else if (*ch == KEY_F(8)) {
_win_switch_if_active(7);
} else if (*ch == KEY_F(9)) {
_win_switch_if_active(8);
} else if (*ch == KEY_F(10)) {
_win_switch_if_active(9);
}
}
static void _win_handle_page(const int * const ch)
{
int rows, cols, y, x;
getmaxyx(stdscr, rows, cols);
getyx(_wins[_curr_prof_win].win, y, x);
int page_space = rows - 4;
int *page_start = &_wins[_curr_prof_win].y_pos;
// page up
if (*ch == KEY_PPAGE) {
*page_start -= page_space;
// went past beginning, show first page
if (*page_start < 0)
*page_start = 0;
_wins[_curr_prof_win].paged = 1;
dirty = TRUE;
// page down
} else if (*ch == KEY_NPAGE) {
*page_start += page_space;
// only got half a screen, show full screen
if ((y - (*page_start)) < page_space)
*page_start = y - page_space;
// went past end, show full screen
else if (*page_start >= y)
*page_start = y - page_space;
_wins[_curr_prof_win].paged = 1;
dirty = TRUE;
}
}