1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-06-16 21:35:24 +00:00
profanity/src/ui/window.c

2023 lines
61 KiB
C
Raw Normal View History

/*
* window.c
2019-11-13 11:11:05 +00:00
* vim: expandtab:ts=4:sts=4:sw=4
*
2019-01-22 10:31:45 +00:00
* Copyright (C) 2012 - 2019 James Booth <boothj5@gmail.com>
2021-01-08 15:36:30 +00:00
* Copyright (C) 2019 - 2021 Michael Vetter <jubalh@iodoru.org>
*
* 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
2016-07-24 00:14:49 +00:00
* along with Profanity. If not, see <https://www.gnu.org/licenses/>.
*
* 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.
*
*/
2016-03-31 20:05:02 +00:00
#include "config.h"
#include <stdlib.h>
#include <string.h>
#include <time.h>
#include <assert.h>
2015-02-01 00:14:11 +00:00
#include <wchar.h>
#include <glib.h>
2016-07-24 14:43:51 +00:00
2016-03-31 20:05:02 +00:00
#ifdef HAVE_NCURSESW_NCURSES_H
2013-01-02 20:27:37 +00:00
#include <ncursesw/ncurses.h>
2016-03-31 20:05:02 +00:00
#elif HAVE_NCURSES_H
#include <ncurses.h>
#elif HAVE_CURSES_H
#include <curses.h>
#endif
#include "log.h"
#include "config/theme.h"
#include "config/preferences.h"
2014-10-07 16:06:02 +00:00
#include "ui/ui.h"
2013-02-02 20:55:58 +00:00
#include "ui/window.h"
#include "ui/screen.h"
#include "xmpp/xmpp.h"
#include "xmpp/roster_list.h"
2014-12-21 22:01:56 +00:00
#define CONS_WIN_TITLE "Profanity. Type /help for help information."
2020-07-07 12:18:57 +00:00
#define XML_WIN_TITLE "XML Console"
2020-07-07 12:18:57 +00:00
#define CEILING(X) (X - (int)(X) > 0 ? (int)(X + 1) : (int)(X))
2014-11-15 22:26:43 +00:00
static void
2020-07-07 12:18:57 +00:00
_win_printf(ProfWin* window, const char* show_char, int pad_indent, GDateTime* timestamp, int flags, theme_item_t theme_item, const char* const display_from, const char* const from_jid, const char* const message_id, const char* const message, ...);
static void _win_print_internal(ProfWin* window, const char* show_char, int pad_indent, GDateTime* time,
int flags, theme_item_t theme_item, const char* const from, const char* const message, DeliveryReceipt* receipt);
static void _win_print_wrapped(WINDOW* win, const char* const message, size_t indent, int pad_indent);
2014-07-20 22:35:48 +00:00
int
2014-11-15 22:26:43 +00:00
win_roster_cols(void)
{
int roster_win_percent = prefs_get_roster_size();
int cols = getmaxx(stdscr);
2020-07-07 12:18:57 +00:00
return CEILING((((double)cols) / 100) * roster_win_percent);
}
2014-07-20 22:35:48 +00:00
int
win_occpuants_cols(void)
{
int occupants_win_percent = prefs_get_occupants_size();
int cols = getmaxx(stdscr);
2020-07-07 12:18:57 +00:00
return CEILING((((double)cols) / 100) * occupants_win_percent);
}
2014-12-16 20:53:23 +00:00
static ProfLayout*
_win_create_simple_layout(void)
{
int cols = getmaxx(stdscr);
2020-07-07 12:18:57 +00:00
ProfLayoutSimple* layout = malloc(sizeof(ProfLayoutSimple));
2014-12-21 19:40:15 +00:00
layout->base.type = LAYOUT_SIMPLE;
layout->base.win = newpad(PAD_SIZE, cols);
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
layout->base.buffer = buffer_create();
layout->base.y_pos = 0;
layout->base.paged = 0;
scrollok(layout->base.win, TRUE);
return &layout->base;
2014-12-16 20:53:23 +00:00
}
static ProfLayout*
_win_create_split_layout(void)
{
int cols = getmaxx(stdscr);
2014-12-16 01:39:47 +00:00
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = malloc(sizeof(ProfLayoutSplit));
2014-12-21 19:40:15 +00:00
layout->base.type = LAYOUT_SPLIT;
layout->base.win = newpad(PAD_SIZE, cols);
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
layout->base.buffer = buffer_create();
layout->base.y_pos = 0;
layout->base.paged = 0;
scrollok(layout->base.win, TRUE);
2014-12-16 01:39:47 +00:00
layout->subwin = NULL;
layout->sub_y_pos = 0;
2014-12-16 22:14:15 +00:00
layout->memcheck = LAYOUT_SPLIT_MEMCHECK;
2014-12-16 01:39:47 +00:00
2014-12-21 19:40:15 +00:00
return &layout->base;
2014-12-16 20:53:23 +00:00
}
2014-12-16 01:39:47 +00:00
2014-12-16 20:53:23 +00:00
ProfWin*
win_create_console(void)
{
2020-07-07 12:18:57 +00:00
ProfConsoleWin* new_win = malloc(sizeof(ProfConsoleWin));
2014-12-21 19:40:15 +00:00
new_win->window.type = WIN_CONSOLE;
new_win->window.layout = _win_create_split_layout();
2014-12-16 20:53:23 +00:00
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
2020-07-07 12:18:57 +00:00
win_create_chat(const char* const barejid)
{
2020-07-07 12:18:57 +00:00
ProfChatWin* new_win = malloc(sizeof(ProfChatWin));
2014-12-21 19:40:15 +00:00
new_win->window.type = WIN_CHAT;
new_win->window.layout = _win_create_simple_layout();
new_win->barejid = strdup(barejid);
2015-01-10 18:17:10 +00:00
new_win->resource_override = NULL;
new_win->is_otr = FALSE;
2015-06-13 00:13:56 +00:00
new_win->otr_is_trusted = FALSE;
new_win->pgp_recv = FALSE;
new_win->pgp_send = FALSE;
new_win->is_omemo = FALSE;
new_win->is_ox = FALSE;
2014-12-16 01:39:47 +00:00
new_win->history_shown = FALSE;
2014-12-21 23:14:01 +00:00
new_win->unread = 0;
2015-01-11 20:20:17 +00:00
new_win->state = chat_state_new();
new_win->enctext = NULL;
new_win->incoming_char = NULL;
new_win->outgoing_char = NULL;
new_win->last_message = NULL;
new_win->last_msg_id = NULL;
new_win->has_attention = FALSE;
new_win->memcheck = PROFCHATWIN_MEMCHECK;
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
2020-07-07 12:18:57 +00:00
win_create_muc(const char* const roomjid)
{
2020-07-07 12:18:57 +00:00
ProfMucWin* new_win = malloc(sizeof(ProfMucWin));
int cols = getmaxx(stdscr);
2014-12-21 19:40:15 +00:00
new_win->window.type = WIN_MUC;
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = malloc(sizeof(ProfLayoutSplit));
2014-12-21 19:40:15 +00:00
layout->base.type = LAYOUT_SPLIT;
if (prefs_get_boolean(PREF_OCCUPANTS)) {
int subwin_cols = win_occpuants_cols();
2014-12-21 19:40:15 +00:00
layout->base.win = newpad(PAD_SIZE, cols - subwin_cols);
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
2020-07-07 12:18:57 +00:00
layout->subwin = newpad(PAD_SIZE, subwin_cols);
2014-12-16 01:39:47 +00:00
wbkgd(layout->subwin, theme_attrs(THEME_TEXT));
} else {
2014-12-21 19:40:15 +00:00
layout->base.win = newpad(PAD_SIZE, (cols));
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
2014-12-16 01:39:47 +00:00
layout->subwin = NULL;
}
layout->sub_y_pos = 0;
2014-12-16 22:14:15 +00:00
layout->memcheck = LAYOUT_SPLIT_MEMCHECK;
2014-12-21 19:40:15 +00:00
layout->base.buffer = buffer_create();
layout->base.y_pos = 0;
layout->base.paged = 0;
scrollok(layout->base.win, TRUE);
new_win->window.layout = (ProfLayout*)layout;
2014-12-16 01:39:47 +00:00
new_win->roomjid = strdup(roomjid);
2020-01-23 17:22:36 +00:00
new_win->room_name = NULL;
2014-12-21 23:14:01 +00:00
new_win->unread = 0;
new_win->unread_mentions = FALSE;
new_win->unread_triggers = FALSE;
if (prefs_get_boolean(PREF_OCCUPANTS_JID)) {
new_win->showjid = TRUE;
} else {
new_win->showjid = FALSE;
}
2021-03-05 12:29:36 +00:00
if (prefs_get_boolean(PREF_OCCUPANTS_OFFLINE)) {
new_win->showoffline = TRUE;
} else {
new_win->showoffline = FALSE;
}
new_win->enctext = NULL;
new_win->message_char = NULL;
new_win->is_omemo = FALSE;
new_win->last_message = NULL;
new_win->last_msg_id = NULL;
new_win->has_attention = FALSE;
2014-12-16 01:39:47 +00:00
new_win->memcheck = PROFMUCWIN_MEMCHECK;
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
2020-07-07 12:18:57 +00:00
win_create_config(const char* const roomjid, DataForm* form, ProfConfWinCallback submit, ProfConfWinCallback cancel, const void* userdata)
{
2020-07-07 12:18:57 +00:00
ProfConfWin* new_win = malloc(sizeof(ProfConfWin));
new_win->window.type = WIN_CONFIG;
2014-12-21 19:40:15 +00:00
new_win->window.layout = _win_create_simple_layout();
2014-12-21 22:01:56 +00:00
new_win->roomjid = strdup(roomjid);
2014-12-16 01:39:47 +00:00
new_win->form = form;
2018-05-30 17:25:06 +00:00
new_win->submit = submit;
new_win->cancel = cancel;
new_win->userdata = userdata;
new_win->memcheck = PROFCONFWIN_MEMCHECK;
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
2020-07-07 12:18:57 +00:00
win_create_private(const char* const fulljid)
{
2020-07-07 12:18:57 +00:00
ProfPrivateWin* new_win = malloc(sizeof(ProfPrivateWin));
2014-12-21 19:40:15 +00:00
new_win->window.type = WIN_PRIVATE;
new_win->window.layout = _win_create_simple_layout();
new_win->fulljid = strdup(fulljid);
2014-12-21 23:14:01 +00:00
new_win->unread = 0;
new_win->occupant_offline = FALSE;
new_win->room_left = FALSE;
new_win->memcheck = PROFPRIVATEWIN_MEMCHECK;
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
win_create_xmlconsole(void)
{
2020-07-07 12:18:57 +00:00
ProfXMLWin* new_win = malloc(sizeof(ProfXMLWin));
2014-12-21 19:40:15 +00:00
new_win->window.type = WIN_XML;
new_win->window.layout = _win_create_simple_layout();
2014-12-10 01:14:11 +00:00
2014-12-21 22:01:56 +00:00
new_win->memcheck = PROFXMLWIN_MEMCHECK;
2014-12-21 19:40:15 +00:00
return &new_win->window;
}
ProfWin*
2020-07-07 12:18:57 +00:00
win_create_plugin(const char* const plugin_name, const char* const tag)
{
2020-07-07 12:18:57 +00:00
ProfPluginWin* new_win = malloc(sizeof(ProfPluginWin));
2018-03-08 19:55:17 +00:00
new_win->window.type = WIN_PLUGIN;
new_win->window.layout = _win_create_simple_layout();
new_win->tag = strdup(tag);
2016-07-12 01:16:12 +00:00
new_win->plugin_name = strdup(plugin_name);
new_win->memcheck = PROFPLUGINWIN_MEMCHECK;
2018-03-08 19:55:17 +00:00
return &new_win->window;
}
2015-10-25 23:31:11 +00:00
char*
2020-07-07 12:18:57 +00:00
win_get_title(ProfWin* window)
2014-12-21 22:01:56 +00:00
{
if (window == NULL) {
return strdup(CONS_WIN_TITLE);
}
if (window->type == WIN_CONSOLE) {
return strdup(CONS_WIN_TITLE);
}
if (window->type == WIN_CHAT) {
2020-07-07 12:18:57 +00:00
ProfChatWin* chatwin = (ProfChatWin*)window;
2014-12-21 22:01:56 +00:00
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
2016-05-05 22:51:49 +00:00
jabber_conn_status_t conn_status = connection_get_status();
if (conn_status == JABBER_CONNECTED) {
PContact contact = roster_get_contact(chatwin->barejid);
if (contact) {
2020-07-07 12:18:57 +00:00
const char* name = p_contact_name_or_jid(contact);
return strdup(name);
} else {
return strdup(chatwin->barejid);
}
2014-12-21 22:39:00 +00:00
} else {
return strdup(chatwin->barejid);
}
2014-12-21 22:01:56 +00:00
}
if (window->type == WIN_MUC) {
2020-07-07 12:18:57 +00:00
ProfMucWin* mucwin = (ProfMucWin*)window;
2014-12-21 22:01:56 +00:00
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
2020-03-24 20:46:40 +00:00
gboolean show_titlebar_jid = prefs_get_boolean(PREF_TITLEBAR_MUC_TITLE_JID);
gboolean show_titlebar_name = prefs_get_boolean(PREF_TITLEBAR_MUC_TITLE_NAME);
2020-07-07 12:18:57 +00:00
GString* title = g_string_new("");
if (show_titlebar_name && mucwin->room_name) {
g_string_append(title, mucwin->room_name);
g_string_append(title, " ");
}
if (show_titlebar_jid) {
g_string_append(title, mucwin->roomjid);
2020-03-24 20:46:40 +00:00
}
2020-07-07 12:18:57 +00:00
char* title_str = title->str;
g_string_free(title, FALSE);
return title_str;
2014-12-21 22:01:56 +00:00
}
if (window->type == WIN_CONFIG) {
2020-07-07 12:18:57 +00:00
ProfConfWin* confwin = (ProfConfWin*)window;
2014-12-21 22:01:56 +00:00
assert(confwin->memcheck == PROFCONFWIN_MEMCHECK);
2020-07-07 12:18:57 +00:00
GString* title = g_string_new(confwin->roomjid);
2014-12-21 22:01:56 +00:00
g_string_append(title, " config");
if (confwin->form->modified) {
g_string_append(title, " *");
}
2020-07-07 12:18:57 +00:00
char* title_str = title->str;
2014-12-21 22:01:56 +00:00
g_string_free(title, FALSE);
return title_str;
}
if (window->type == WIN_PRIVATE) {
2020-07-07 12:18:57 +00:00
ProfPrivateWin* privatewin = (ProfPrivateWin*)window;
2014-12-21 22:01:56 +00:00
assert(privatewin->memcheck == PROFPRIVATEWIN_MEMCHECK);
return strdup(privatewin->fulljid);
}
if (window->type == WIN_XML) {
return strdup(XML_WIN_TITLE);
}
if (window->type == WIN_PLUGIN) {
2020-07-07 12:18:57 +00:00
ProfPluginWin* pluginwin = (ProfPluginWin*)window;
assert(pluginwin->memcheck == PROFPLUGINWIN_MEMCHECK);
return strdup(pluginwin->tag);
}
2014-12-21 22:01:56 +00:00
return NULL;
}
2018-03-09 21:11:59 +00:00
char*
2020-07-07 12:18:57 +00:00
win_get_tab_identifier(ProfWin* window)
2018-03-09 21:11:59 +00:00
{
assert(window != NULL);
switch (window->type) {
2020-07-07 12:18:57 +00:00
case WIN_CONSOLE:
{
return strdup("console");
}
case WIN_CHAT:
{
ProfChatWin* chatwin = (ProfChatWin*)window;
return strdup(chatwin->barejid);
}
case WIN_MUC:
{
ProfMucWin* mucwin = (ProfMucWin*)window;
return strdup(mucwin->roomjid);
}
case WIN_CONFIG:
{
ProfConfWin* confwin = (ProfConfWin*)window;
return strdup(confwin->roomjid);
}
case WIN_PRIVATE:
{
ProfPrivateWin* privwin = (ProfPrivateWin*)window;
return strdup(privwin->fulljid);
}
case WIN_PLUGIN:
{
ProfPluginWin* pluginwin = (ProfPluginWin*)window;
return strdup(pluginwin->tag);
}
case WIN_XML:
{
return strdup("xmlconsole");
}
default:
return strdup("UNKNOWN");
2018-03-09 21:11:59 +00:00
}
}
char*
win_get_last_sent_message(ProfWin* window)
{
char* last_message = NULL;
switch (window->type) {
case WIN_CHAT:
{
ProfChatWin* chatwin = (ProfChatWin*)window;
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
last_message = chatwin->last_message;
break;
}
case WIN_MUC:
{
ProfMucWin* mucwin = (ProfMucWin*)window;
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
last_message = mucwin->last_message;
}
default:
break;
}
if (last_message == NULL) {
return NULL;
}
return last_message;
}
2015-11-29 23:58:52 +00:00
char*
2020-07-07 12:18:57 +00:00
win_to_string(ProfWin* window)
2015-11-29 23:58:52 +00:00
{
assert(window != NULL);
switch (window->type) {
2020-07-07 12:18:57 +00:00
case WIN_CONSOLE:
{
ProfConsoleWin* conswin = (ProfConsoleWin*)window;
return cons_get_string(conswin);
}
case WIN_CHAT:
{
ProfChatWin* chatwin = (ProfChatWin*)window;
return chatwin_get_string(chatwin);
}
case WIN_MUC:
{
ProfMucWin* mucwin = (ProfMucWin*)window;
return mucwin_get_string(mucwin);
}
case WIN_CONFIG:
{
ProfConfWin* confwin = (ProfConfWin*)window;
return confwin_get_string(confwin);
}
case WIN_PRIVATE:
{
ProfPrivateWin* privwin = (ProfPrivateWin*)window;
return privwin_get_string(privwin);
}
case WIN_XML:
{
ProfXMLWin* xmlwin = (ProfXMLWin*)window;
return xmlwin_get_string(xmlwin);
}
case WIN_PLUGIN:
{
ProfPluginWin* pluginwin = (ProfPluginWin*)window;
GString* gstring = g_string_new("");
g_string_append_printf(gstring, "Plugin: %s", pluginwin->tag);
char* res = gstring->str;
g_string_free(gstring, FALSE);
return res;
}
default:
return NULL;
2015-11-29 23:58:52 +00:00
}
}
2014-10-09 20:27:16 +00:00
void
2020-07-07 12:18:57 +00:00
win_hide_subwin(ProfWin* window)
2014-10-09 20:27:16 +00:00
{
2014-12-16 01:39:47 +00:00
if (window->layout->type == LAYOUT_SPLIT) {
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2014-12-16 01:39:47 +00:00
if (layout->subwin) {
delwin(layout->subwin);
2014-12-10 00:54:46 +00:00
}
2014-12-16 01:39:47 +00:00
layout->subwin = NULL;
layout->sub_y_pos = 0;
int cols = getmaxx(stdscr);
2014-12-21 19:40:15 +00:00
wresize(layout->base.win, PAD_SIZE, cols);
2014-12-16 01:39:47 +00:00
win_redraw(window);
} else {
int cols = getmaxx(stdscr);
wresize(window->layout->win, PAD_SIZE, cols);
win_redraw(window);
2014-10-09 20:27:16 +00:00
}
}
2014-10-09 20:42:09 +00:00
void
2020-07-07 12:18:57 +00:00
win_show_subwin(ProfWin* window)
2014-10-09 20:42:09 +00:00
{
2014-12-10 00:54:46 +00:00
int cols = getmaxx(stdscr);
int subwin_cols = 0;
2014-10-09 20:42:09 +00:00
2014-12-20 22:44:01 +00:00
if (window->layout->type != LAYOUT_SPLIT) {
return;
}
if (window->type == WIN_MUC) {
subwin_cols = win_occpuants_cols();
} else if (window->type == WIN_CONSOLE) {
2014-12-10 00:54:46 +00:00
subwin_cols = win_roster_cols();
2014-10-09 20:42:09 +00:00
}
2014-12-20 22:44:01 +00:00
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2014-12-20 22:44:01 +00:00
layout->subwin = newpad(PAD_SIZE, subwin_cols);
wbkgd(layout->subwin, theme_attrs(THEME_TEXT));
2014-12-21 19:40:15 +00:00
wresize(layout->base.win, PAD_SIZE, cols - subwin_cols);
2014-12-20 22:44:01 +00:00
win_redraw(window);
2014-10-09 20:42:09 +00:00
}
void
2013-04-21 18:48:56 +00:00
win_free(ProfWin* window)
{
2014-12-16 01:39:47 +00:00
if (window->layout->type == LAYOUT_SPLIT) {
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2014-12-16 01:39:47 +00:00
if (layout->subwin) {
delwin(layout->subwin);
2014-12-10 00:54:46 +00:00
}
2014-12-21 19:40:15 +00:00
buffer_free(layout->base.buffer);
delwin(layout->base.win);
2014-12-16 01:39:47 +00:00
} else {
buffer_free(window->layout->buffer);
delwin(window->layout->win);
2014-10-07 16:06:02 +00:00
}
2015-02-06 21:39:53 +00:00
free(window->layout);
2014-12-10 00:54:46 +00:00
2016-03-07 18:24:04 +00:00
switch (window->type) {
case WIN_CHAT:
{
2020-07-07 12:18:57 +00:00
ProfChatWin* chatwin = (ProfChatWin*)window;
free(chatwin->barejid);
2015-01-10 18:17:10 +00:00
free(chatwin->resource_override);
free(chatwin->enctext);
free(chatwin->incoming_char);
free(chatwin->outgoing_char);
free(chatwin->last_message);
free(chatwin->last_msg_id);
chat_state_free(chatwin->state);
2016-03-07 18:24:04 +00:00
break;
2014-12-10 01:44:32 +00:00
}
2016-03-07 18:24:04 +00:00
case WIN_MUC:
{
2020-07-07 12:18:57 +00:00
ProfMucWin* mucwin = (ProfMucWin*)window;
free(mucwin->roomjid);
free(mucwin->room_name);
free(mucwin->enctext);
free(mucwin->message_char);
free(mucwin->last_message);
free(mucwin->last_msg_id);
2016-03-07 18:24:04 +00:00
break;
}
case WIN_CONFIG:
2016-03-07 18:24:04 +00:00
{
2020-07-07 12:18:57 +00:00
ProfConfWin* conf = (ProfConfWin*)window;
free(conf->roomjid);
form_destroy(conf->form);
2016-03-07 18:24:04 +00:00
break;
2014-12-10 01:14:11 +00:00
}
2016-03-07 18:24:04 +00:00
case WIN_PRIVATE:
{
2020-07-07 12:18:57 +00:00
ProfPrivateWin* privatewin = (ProfPrivateWin*)window;
free(privatewin->fulljid);
2016-03-07 18:24:04 +00:00
break;
}
case WIN_PLUGIN:
{
2020-07-07 12:18:57 +00:00
ProfPluginWin* pluginwin = (ProfPluginWin*)window;
2016-03-07 18:24:04 +00:00
free(pluginwin->tag);
2016-07-12 23:25:45 +00:00
free(pluginwin->plugin_name);
2016-03-07 18:24:04 +00:00
break;
}
default:
break;
}
free(window);
}
2013-04-20 22:39:17 +00:00
2015-01-15 00:14:12 +00:00
void
2020-07-07 12:18:57 +00:00
win_page_up(ProfWin* window)
2015-01-31 22:00:22 +00:00
{
int rows = getmaxy(stdscr);
int y = getcury(window->layout->win);
int page_space = rows - 4;
2020-07-07 12:18:57 +00:00
int* page_start = &(window->layout->y_pos);
2015-01-31 22:00:22 +00:00
*page_start -= page_space;
// went past beginning, show first page
if (*page_start < 0)
*page_start = 0;
window->layout->paged = 1;
win_update_virtual(window);
// switch off page if last line and space line visible
if ((y) - *page_start == page_space) {
window->layout->paged = 0;
}
}
void
2020-07-07 12:18:57 +00:00
win_page_down(ProfWin* window)
2015-01-31 22:00:22 +00:00
{
int rows = getmaxy(stdscr);
int y = getcury(window->layout->win);
int page_space = rows - 4;
2020-07-07 12:18:57 +00:00
int* page_start = &(window->layout->y_pos);
2015-01-31 22:00:22 +00:00
*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 - 1;
window->layout->paged = 1;
win_update_virtual(window);
// switch off page if last line and space line visible
if ((y) - *page_start == page_space) {
window->layout->paged = 0;
}
}
void
2020-07-07 12:18:57 +00:00
win_sub_page_down(ProfWin* window)
2015-01-31 22:00:22 +00:00
{
if (window->layout->type == LAYOUT_SPLIT) {
int rows = getmaxy(stdscr);
int page_space = rows - 4;
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* split_layout = (ProfLayoutSplit*)window->layout;
2015-01-31 22:00:22 +00:00
int sub_y = getcury(split_layout->subwin);
2020-07-07 12:18:57 +00:00
int* sub_y_pos = &(split_layout->sub_y_pos);
2015-01-31 22:00:22 +00:00
*sub_y_pos += page_space;
// only got half a screen, show full screen
2020-07-07 12:18:57 +00:00
if ((sub_y - (*sub_y_pos)) < page_space)
2015-01-31 22:00:22 +00:00
*sub_y_pos = sub_y - page_space;
// went past end, show full screen
else if (*sub_y_pos >= sub_y)
*sub_y_pos = sub_y - page_space - 1;
win_update_virtual(window);
}
}
void
2020-07-07 12:18:57 +00:00
win_sub_page_up(ProfWin* window)
2015-01-31 22:00:22 +00:00
{
if (window->layout->type == LAYOUT_SPLIT) {
int rows = getmaxy(stdscr);
int page_space = rows - 4;
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* split_layout = (ProfLayoutSplit*)window->layout;
int* sub_y_pos = &(split_layout->sub_y_pos);
2015-01-31 22:00:22 +00:00
*sub_y_pos -= page_space;
// went past beginning, show first page
if (*sub_y_pos < 0)
*sub_y_pos = 0;
win_update_virtual(window);
}
}
2015-06-15 16:54:47 +00:00
void
2020-07-07 12:18:57 +00:00
win_clear(ProfWin* window)
2015-06-15 16:54:47 +00:00
{
2016-11-07 00:26:15 +00:00
if (!prefs_get_boolean(PREF_CLEAR_PERSIST_HISTORY)) {
werase(window->layout->win);
buffer_free(window->layout->buffer);
window->layout->buffer = buffer_create();
2016-11-07 00:26:15 +00:00
return;
}
int y = getcury(window->layout->win);
2020-07-07 12:18:57 +00:00
int* page_start = &(window->layout->y_pos);
*page_start = y;
window->layout->paged = 1;
2015-06-15 16:54:47 +00:00
win_update_virtual(window);
}
2015-06-15 17:08:47 +00:00
void
2020-07-07 12:18:57 +00:00
win_resize(ProfWin* window)
2015-06-15 17:08:47 +00:00
{
int cols = getmaxx(stdscr);
if (window->layout->type == LAYOUT_SPLIT) {
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2015-06-15 17:08:47 +00:00
if (layout->subwin) {
int subwin_cols = 0;
2015-06-15 17:08:47 +00:00
if (window->type == WIN_CONSOLE) {
subwin_cols = win_roster_cols();
} else if (window->type == WIN_MUC) {
subwin_cols = win_occpuants_cols();
}
2016-01-10 03:11:05 +00:00
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
2015-06-15 17:08:47 +00:00
wresize(layout->base.win, PAD_SIZE, cols - subwin_cols);
2016-01-10 03:11:05 +00:00
wbkgd(layout->subwin, theme_attrs(THEME_TEXT));
2015-06-15 17:08:47 +00:00
wresize(layout->subwin, PAD_SIZE, subwin_cols);
if (window->type == WIN_CONSOLE) {
rosterwin_roster();
} else if (window->type == WIN_MUC) {
2020-07-07 12:18:57 +00:00
ProfMucWin* mucwin = (ProfMucWin*)window;
2015-06-15 17:08:47 +00:00
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
occupantswin_occupants(mucwin->roomjid);
}
} else {
2016-01-10 03:11:05 +00:00
wbkgd(layout->base.win, theme_attrs(THEME_TEXT));
2015-06-15 17:08:47 +00:00
wresize(layout->base.win, PAD_SIZE, cols);
}
} else {
2016-01-10 03:11:05 +00:00
wbkgd(window->layout->win, theme_attrs(THEME_TEXT));
2015-06-15 17:08:47 +00:00
wresize(window->layout->win, PAD_SIZE, cols);
}
win_redraw(window);
}
2013-10-06 16:46:22 +00:00
void
2020-07-07 12:18:57 +00:00
win_update_virtual(ProfWin* window)
2013-10-06 00:30:53 +00:00
{
int cols = getmaxx(stdscr);
2014-10-07 15:37:14 +00:00
int row_start = screen_mainwin_row_start();
int row_end = screen_mainwin_row_end();
2014-12-16 01:39:47 +00:00
if (window->layout->type == LAYOUT_SPLIT) {
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2014-12-16 01:39:47 +00:00
if (layout->subwin) {
int subwin_cols = 0;
2014-12-16 01:39:47 +00:00
if (window->type == WIN_MUC) {
subwin_cols = win_occpuants_cols();
} else {
subwin_cols = win_roster_cols();
}
2020-07-07 12:18:57 +00:00
pnoutrefresh(layout->base.win, layout->base.y_pos, 0, row_start, 0, row_end, (cols - subwin_cols) - 1);
pnoutrefresh(layout->subwin, layout->sub_y_pos, 0, row_start, (cols - subwin_cols), row_end, cols - 1);
2014-12-10 00:54:46 +00:00
} else {
2020-07-07 12:18:57 +00:00
pnoutrefresh(layout->base.win, layout->base.y_pos, 0, row_start, 0, row_end, cols - 1);
}
2014-12-16 01:39:47 +00:00
} else {
2020-07-07 12:18:57 +00:00
pnoutrefresh(window->layout->win, window->layout->y_pos, 0, row_start, 0, row_end, cols - 1);
2014-10-07 15:37:14 +00:00
}
2013-10-06 00:28:25 +00:00
}
2015-06-15 17:59:20 +00:00
void
2020-07-07 12:18:57 +00:00
win_refresh_without_subwin(ProfWin* window)
2015-06-15 17:59:20 +00:00
{
int cols = getmaxx(stdscr);
2015-06-15 17:59:20 +00:00
if ((window->type == WIN_MUC) || (window->type == WIN_CONSOLE)) {
int row_start = screen_mainwin_row_start();
int row_end = screen_mainwin_row_end();
2020-07-07 12:18:57 +00:00
pnoutrefresh(window->layout->win, window->layout->y_pos, 0, row_start, 0, row_end, cols - 1);
2015-06-15 17:59:20 +00:00
}
}
void
2020-07-07 12:18:57 +00:00
win_refresh_with_subwin(ProfWin* window)
2015-06-15 17:59:20 +00:00
{
int subwin_cols = 0;
int cols = getmaxx(stdscr);
int row_start = screen_mainwin_row_start();
int row_end = screen_mainwin_row_end();
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2020-04-20 14:09:40 +00:00
2015-06-15 17:59:20 +00:00
if (window->type == WIN_MUC) {
subwin_cols = win_occpuants_cols();
} else if (window->type == WIN_CONSOLE) {
subwin_cols = win_roster_cols();
} else {
// Other window types don't support subwindows, we shouldn't be here
return;
2015-06-15 17:59:20 +00:00
}
2020-04-20 14:09:40 +00:00
2020-07-07 12:18:57 +00:00
pnoutrefresh(layout->base.win, layout->base.y_pos, 0, row_start, 0, row_end, (cols - subwin_cols) - 1);
pnoutrefresh(layout->subwin, layout->sub_y_pos, 0, row_start, (cols - subwin_cols), row_end, cols - 1);
2015-06-15 17:59:20 +00:00
}
2013-10-06 23:59:17 +00:00
void
2020-07-07 12:18:57 +00:00
win_move_to_end(ProfWin* window)
2013-10-06 23:59:17 +00:00
{
2014-12-16 01:39:47 +00:00
window->layout->paged = 0;
2013-10-06 23:59:17 +00:00
int rows = getmaxy(stdscr);
2014-12-16 01:39:47 +00:00
int y = getcury(window->layout->win);
2013-10-06 23:59:17 +00:00
int size = rows - 3;
2014-12-16 01:39:47 +00:00
window->layout->y_pos = y - (size - 1);
if (window->layout->y_pos < 0) {
window->layout->y_pos = 0;
2013-10-06 23:59:17 +00:00
}
}
void
2020-07-07 12:18:57 +00:00
win_show_occupant(ProfWin* window, Occupant* occupant)
{
2020-07-07 12:18:57 +00:00
const char* presence_str = string_from_resource_presence(occupant->presence);
theme_item_t presence_colour = theme_main_presence_attrs(presence_str);
win_print(window, presence_colour, "-", "%s", occupant->nick);
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " is %s", presence_str);
if (occupant->status) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", \"%s\"", occupant->status);
}
2016-10-15 18:19:44 +00:00
win_appendln(window, presence_colour, "");
}
2013-10-06 22:22:46 +00:00
void
2020-07-07 12:18:57 +00:00
win_show_contact(ProfWin* window, PContact contact)
{
2020-07-07 12:18:57 +00:00
const char* barejid = p_contact_barejid(contact);
const char* name = p_contact_name(contact);
const char* presence = p_contact_presence(contact);
const char* status = p_contact_status(contact);
GDateTime* last_activity = p_contact_last_activity(contact);
theme_item_t presence_colour = theme_main_presence_attrs(presence);
2015-05-04 22:11:32 +00:00
if (name) {
win_print(window, presence_colour, "-", "%s", name);
} else {
win_print(window, presence_colour, "-", "%s", barejid);
}
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " is %s", presence);
2015-05-04 22:11:32 +00:00
if (last_activity) {
2020-07-07 12:18:57 +00:00
GDateTime* now = g_date_time_new_now_local();
GTimeSpan span = g_date_time_difference(now, last_activity);
2016-07-25 22:09:51 +00:00
g_date_time_unref(now);
int hours = span / G_TIME_SPAN_HOUR;
span = span - hours * G_TIME_SPAN_HOUR;
int minutes = span / G_TIME_SPAN_MINUTE;
span = span - minutes * G_TIME_SPAN_MINUTE;
int seconds = span / G_TIME_SPAN_SECOND;
if (hours > 0) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", idle %dh%dm%ds", hours, minutes, seconds);
} else {
win_append(window, presence_colour, ", idle %dm%ds", minutes, seconds);
}
}
2015-05-04 22:11:32 +00:00
if (status) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", \"%s\"", p_contact_status(contact));
}
2016-10-15 18:19:44 +00:00
win_appendln(window, presence_colour, "");
}
2013-10-06 00:28:25 +00:00
void
2020-07-07 12:18:57 +00:00
win_show_occupant_info(ProfWin* window, const char* const room, Occupant* occupant)
{
2020-07-07 12:18:57 +00:00
const char* presence_str = string_from_resource_presence(occupant->presence);
const char* occupant_affiliation = muc_occupant_affiliation_str(occupant);
const char* occupant_role = muc_occupant_role_str(occupant);
theme_item_t presence_colour = theme_main_presence_attrs(presence_str);
win_print(window, presence_colour, "!", "%s", occupant->nick);
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " is %s", presence_str);
if (occupant->status) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", \"%s\"", occupant->status);
}
win_newline(window);
2014-10-01 12:27:01 +00:00
if (occupant->jid) {
win_println(window, THEME_DEFAULT, "!", " Jid: %s", occupant->jid);
2014-10-01 12:27:01 +00:00
}
win_println(window, THEME_DEFAULT, "!", " Affiliation: %s", occupant_affiliation);
win_println(window, THEME_DEFAULT, "!", " Role: %s", occupant_role);
2020-07-07 12:18:57 +00:00
Jid* jidp = jid_create_from_bare_and_resource(room, occupant->nick);
EntityCapabilities* caps = caps_lookup(jidp->fulljid);
2014-11-01 01:48:36 +00:00
jid_destroy(jidp);
if (caps) {
// show identity
if (caps->identity) {
2020-07-07 12:18:57 +00:00
DiscoIdentity* identity = caps->identity;
win_print(window, THEME_DEFAULT, "!", " Identity: ");
if (identity->name) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->name);
if (identity->category || identity->type) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, " ");
}
}
if (identity->type) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->type);
if (identity->category) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, " ");
}
}
if (identity->category) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->category);
}
win_newline(window);
}
2015-05-04 22:11:32 +00:00
if (caps->software_version) {
2020-07-07 12:18:57 +00:00
SoftwareVersion* software_version = caps->software_version;
if (software_version->software) {
win_print(window, THEME_DEFAULT, "!", " Software: %s", software_version->software);
}
if (software_version->software_version) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, ", %s", software_version->software_version);
}
if (software_version->software || software_version->software_version) {
win_newline(window);
}
if (software_version->os) {
win_print(window, THEME_DEFAULT, "!", " OS: %s", software_version->os);
}
if (software_version->os_version) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, ", %s", software_version->os_version);
}
if (software_version->os || software_version->os_version) {
win_newline(window);
}
}
caps_destroy(caps);
}
win_println(window, THEME_DEFAULT, "-", "");
}
2014-06-03 20:14:21 +00:00
void
2020-07-07 12:18:57 +00:00
win_show_info(ProfWin* window, PContact contact)
2014-06-03 20:14:21 +00:00
{
2020-07-07 12:18:57 +00:00
const char* barejid = p_contact_barejid(contact);
const char* name = p_contact_name(contact);
const char* presence = p_contact_presence(contact);
const char* sub = p_contact_subscription(contact);
GDateTime* last_activity = p_contact_last_activity(contact);
2014-06-03 20:14:21 +00:00
theme_item_t presence_colour = theme_main_presence_attrs(presence);
win_println(window, THEME_DEFAULT, "-", "");
win_print(window, presence_colour, "-", "%s", barejid);
2015-05-04 22:11:32 +00:00
if (name) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " (%s)", name);
2014-06-03 20:14:21 +00:00
}
2016-10-15 18:19:44 +00:00
win_appendln(window, THEME_DEFAULT, ":");
2014-06-03 20:14:21 +00:00
2015-05-04 22:11:32 +00:00
if (sub) {
win_println(window, THEME_DEFAULT, "-", "Subscription: %s", sub);
2014-06-03 20:14:21 +00:00
}
2015-05-04 22:11:32 +00:00
if (last_activity) {
2020-07-07 12:18:57 +00:00
GDateTime* now = g_date_time_new_now_local();
2014-06-03 20:14:21 +00:00
GTimeSpan span = g_date_time_difference(now, last_activity);
int hours = span / G_TIME_SPAN_HOUR;
span = span - hours * G_TIME_SPAN_HOUR;
int minutes = span / G_TIME_SPAN_MINUTE;
span = span - minutes * G_TIME_SPAN_MINUTE;
int seconds = span / G_TIME_SPAN_SECOND;
if (hours > 0) {
2020-07-07 12:18:57 +00:00
win_println(window, THEME_DEFAULT, "-", "Last activity: %dh%dm%ds", hours, minutes, seconds);
} else {
win_println(window, THEME_DEFAULT, "-", "Last activity: %dm%ds", minutes, seconds);
}
2014-06-03 20:14:21 +00:00
g_date_time_unref(now);
}
2020-07-07 12:18:57 +00:00
GList* resources = p_contact_get_available_resources(contact);
GList* ordered_resources = NULL;
2015-05-04 22:11:32 +00:00
if (resources) {
win_println(window, THEME_DEFAULT, "-", "Resources:");
2014-06-03 20:14:21 +00:00
2015-02-17 09:52:37 +00:00
// sort in order of availability
2020-07-07 12:18:57 +00:00
GList* curr = resources;
2015-05-04 22:11:32 +00:00
while (curr) {
2020-07-07 12:18:57 +00:00
Resource* resource = curr->data;
2014-06-03 20:14:21 +00:00
ordered_resources = g_list_insert_sorted(ordered_resources,
2020-07-07 12:18:57 +00:00
resource, (GCompareFunc)resource_compare_availability);
2015-02-09 21:46:54 +00:00
curr = g_list_next(curr);
2014-06-03 20:14:21 +00:00
}
}
2015-02-09 21:46:54 +00:00
g_list_free(resources);
2014-06-03 20:14:21 +00:00
2020-07-07 12:18:57 +00:00
GList* curr = ordered_resources;
2015-05-04 22:11:32 +00:00
while (curr) {
2020-07-07 12:18:57 +00:00
Resource* resource = curr->data;
const char* resource_presence = string_from_resource_presence(resource->presence);
theme_item_t presence_colour = theme_main_presence_attrs(resource_presence);
win_print(window, presence_colour, "-", " %s (%d), %s", resource->name, resource->priority, resource_presence);
2015-05-04 22:11:32 +00:00
if (resource->status) {
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", \"%s\"", resource->status);
2014-06-03 20:14:21 +00:00
}
win_newline(window);
2014-06-03 20:14:21 +00:00
2020-07-07 12:18:57 +00:00
Jid* jidp = jid_create_from_bare_and_resource(barejid, resource->name);
EntityCapabilities* caps = caps_lookup(jidp->fulljid);
2014-11-01 01:48:36 +00:00
jid_destroy(jidp);
if (caps) {
// show identity
if (caps->identity) {
2020-07-07 12:18:57 +00:00
DiscoIdentity* identity = caps->identity;
win_print(window, THEME_DEFAULT, "-", " Identity: ");
if (identity->name) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->name);
if (identity->category || identity->type) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, " ");
2014-06-03 20:14:21 +00:00
}
}
if (identity->type) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->type);
if (identity->category) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, " ");
2014-06-03 20:14:21 +00:00
}
}
if (identity->category) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, "%s", identity->category);
2014-06-03 20:14:21 +00:00
}
win_newline(window);
}
2015-05-04 22:11:32 +00:00
if (caps->software_version) {
2020-07-07 12:18:57 +00:00
SoftwareVersion* software_version = caps->software_version;
if (software_version->software) {
win_print(window, THEME_DEFAULT, "-", " Software: %s", software_version->software);
}
if (software_version->software_version) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, ", %s", software_version->software_version);
}
if (software_version->software || software_version->software_version) {
win_newline(window);
}
if (software_version->os) {
win_print(window, THEME_DEFAULT, "-", " OS: %s", software_version->os);
}
if (software_version->os_version) {
2016-10-15 18:53:49 +00:00
win_append(window, THEME_DEFAULT, ", %s", software_version->os_version);
}
if (software_version->os || software_version->os_version) {
win_newline(window);
}
2014-06-03 20:14:21 +00:00
}
2014-09-22 00:36:45 +00:00
caps_destroy(caps);
2014-06-03 20:14:21 +00:00
}
2015-02-09 21:46:54 +00:00
curr = g_list_next(curr);
2014-06-03 20:14:21 +00:00
}
2015-02-09 21:46:54 +00:00
g_list_free(ordered_resources);
2014-06-03 20:14:21 +00:00
}
void
2020-07-07 12:18:57 +00:00
win_show_status_string(ProfWin* window, const char* const from,
const char* const show, const char* const status,
GDateTime* last_activity, const char* const pre,
const char* const default_show)
{
2014-11-16 20:40:19 +00:00
theme_item_t presence_colour;
2015-05-04 22:11:32 +00:00
if (show) {
presence_colour = theme_main_presence_attrs(show);
} else if (strcmp(default_show, "online") == 0) {
2014-11-16 20:40:19 +00:00
presence_colour = THEME_ONLINE;
} else {
2014-11-16 20:40:19 +00:00
presence_colour = THEME_OFFLINE;
}
win_print(window, presence_colour, "-", "%s %s", pre, from);
2015-05-04 22:11:32 +00:00
if (show)
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " is %s", show);
else
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, " is %s", default_show);
2015-05-04 22:11:32 +00:00
if (last_activity) {
2020-07-07 12:18:57 +00:00
gchar* date_fmt = NULL;
char* time_pref = prefs_get_string(PREF_TIME_LASTACTIVITY);
2015-09-29 22:30:23 +00:00
date_fmt = g_date_time_format(last_activity, time_pref);
g_free(time_pref);
2015-09-29 22:30:23 +00:00
assert(date_fmt != NULL);
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", last activity: %s", date_fmt);
2015-09-29 22:30:23 +00:00
g_free(date_fmt);
}
2015-05-04 22:11:32 +00:00
if (status)
2016-10-15 18:53:49 +00:00
win_append(window, presence_colour, ", \"%s\"", status);
2016-10-15 18:19:44 +00:00
win_appendln(window, presence_colour, "");
}
static void
2020-07-07 12:18:57 +00:00
_win_correct(ProfWin* window, const char* const message, const char* const id, const char* const replace_id, const char* const from_jid)
{
2020-07-07 12:18:57 +00:00
ProfBuffEntry* entry = buffer_get_entry_by_id(window->layout->buffer, replace_id);
if (!entry) {
log_debug("Replace ID %s could not be found in buffer. Message: %s", replace_id, message);
return;
}
2020-03-09 12:33:48 +00:00
if (g_strcmp0(entry->from_jid, from_jid) != 0) {
log_debug("Illicit LMC attempt from %s for message from %s with: %s", from_jid, entry->from_jid, message);
cons_show("Illicit LMC attempt from %s for message from %s", from_jid, entry->from_jid);
2020-03-09 12:19:25 +00:00
return;
}
/*TODO: set date?
if (entry->date) {
if (entry->date->timestamp) {
g_date_time_unref(entry->date->timestamp);
}
free(entry->date);
}
entry->date = buffer_date_new_now();
*/
free(entry->show_char);
entry->show_char = prefs_get_correction_char();
if (entry->message) {
free(entry->message);
}
entry->message = strdup(message);
if (entry->id) {
free(entry->id);
}
entry->id = strdup(id);
win_redraw(window);
}
void
2020-07-07 12:18:57 +00:00
win_print_incoming(ProfWin* window, const char* const display_name_from, ProfMessage* message)
{
int flags = NO_ME;
if (!message->trusted) {
flags |= UNTRUSTED;
}
2020-07-07 12:18:57 +00:00
switch (window->type) {
case WIN_CHAT:
{
2020-07-07 12:18:57 +00:00
char* enc_char;
ProfChatWin* chatwin = (ProfChatWin*)window;
if (chatwin->incoming_char) {
enc_char = strdup(chatwin->incoming_char);
} else if (message->enc == PROF_MSG_ENC_OTR) {
enc_char = prefs_get_otr_char();
} else if (message->enc == PROF_MSG_ENC_PGP) {
enc_char = prefs_get_pgp_char();
} else if (message->enc == PROF_MSG_ENC_OX) { // XEP-0373: OpenPGP for XMPP
enc_char = prefs_get_ox_char();
} else if (message->enc == PROF_MSG_ENC_OMEMO) {
enc_char = prefs_get_omemo_char();
} else {
enc_char = strdup("-");
}
2020-07-07 12:18:57 +00:00
if (prefs_get_boolean(PREF_CORRECTION_ALLOW) && message->replace_id) {
_win_correct(window, message->plain, message->id, message->replace_id, message->from_jid->barejid);
} else {
_win_printf(window, enc_char, 0, message->timestamp, flags, THEME_TEXT_THEM, display_name_from, message->from_jid->barejid, message->id, "%s", message->plain);
}
2020-07-07 12:18:57 +00:00
free(enc_char);
break;
}
case WIN_PRIVATE:
_win_printf(window, "-", 0, message->timestamp, flags, THEME_TEXT_THEM, display_name_from, message->from_jid->barejid, message->id, "%s", message->plain);
break;
default:
assert(FALSE);
break;
}
}
2016-10-15 19:07:33 +00:00
void
2020-07-07 12:18:57 +00:00
win_print_them(ProfWin* window, theme_item_t theme_item, const char* const show_char, int flags, const char* const them)
2016-10-15 21:12:07 +00:00
{
_win_printf(window, show_char, 0, NULL, flags | NO_ME | NO_EOL, theme_item, them, NULL, NULL, "");
2016-10-15 21:12:07 +00:00
}
void
2020-07-07 12:18:57 +00:00
win_println_incoming_muc_msg(ProfWin* window, char* show_char, int flags, const ProfMessage* const message)
2016-10-15 19:07:33 +00:00
{
if (prefs_get_boolean(PREF_CORRECTION_ALLOW) && message->replace_id) {
_win_correct(window, message->plain, message->id, message->replace_id, message->from_jid->fulljid);
} else {
_win_printf(window, show_char, 0, message->timestamp, flags | NO_ME, THEME_TEXT_THEM, message->from_jid->resourcepart, message->from_jid->fulljid, message->id, "%s", message->plain);
}
2016-10-15 19:07:33 +00:00
inp_nonblocking(TRUE);
}
2016-10-15 19:55:45 +00:00
void
2020-07-07 12:18:57 +00:00
win_print_outgoing_muc_msg(ProfWin* window, char* show_char, const char* const me, const char* const id, const char* const replace_id, const char* const message)
2016-10-15 19:55:45 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 19:55:45 +00:00
if (prefs_get_boolean(PREF_CORRECTION_ALLOW) && replace_id) {
2020-03-09 12:19:25 +00:00
_win_correct(window, message, id, replace_id, me);
} else {
2020-03-09 12:19:25 +00:00
_win_printf(window, show_char, 0, timestamp, 0, THEME_TEXT_ME, me, me, id, "%s", message);
}
2016-10-15 19:55:45 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
}
2014-07-16 21:03:46 +00:00
void
2020-07-07 12:18:57 +00:00
win_print_outgoing(ProfWin* window, const char* show_char, const char* const id, const char* const replace_id, const char* const message)
2014-07-16 21:03:46 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2014-07-16 21:03:46 +00:00
2020-07-07 12:18:57 +00:00
const char* myjid = connection_get_fulljid();
if (replace_id) {
2020-03-09 12:19:25 +00:00
_win_correct(window, message, id, replace_id, myjid);
} else {
_win_printf(window, show_char, 0, timestamp, 0, THEME_TEXT_ME, "me", myjid, id, "%s", message);
}
2016-10-15 21:12:07 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
}
void
2020-07-07 12:18:57 +00:00
win_print_history(ProfWin* window, const ProfMessage* const message)
2016-10-15 21:12:07 +00:00
{
g_date_time_ref(message->timestamp);
2016-10-15 21:12:07 +00:00
2020-07-07 12:18:57 +00:00
char* display_name;
int flags = 0;
2020-07-07 12:18:57 +00:00
const char* jid = connection_get_fulljid();
Jid* jidp = jid_create(jid);
if (g_strcmp0(jidp->barejid, message->from_jid->barejid) == 0) {
display_name = strdup("me");
} else {
display_name = roster_get_msg_display_name(message->from_jid->barejid, message->from_jid->resourcepart);
}
jid_destroy(jidp);
buffer_append(window->layout->buffer, "-", 0, message->timestamp, flags, THEME_TEXT_HISTORY, display_name, NULL, message->plain, NULL, NULL);
_win_print_internal(window, "-", 0, message->timestamp, flags, THEME_TEXT_HISTORY, display_name, message->plain, NULL);
free(display_name);
2016-10-15 21:12:07 +00:00
2016-10-11 23:14:55 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(message->timestamp);
2016-10-11 23:14:55 +00:00
}
void
2020-07-07 12:18:57 +00:00
win_print(ProfWin* window, theme_item_t theme_item, const char* show_char, const char* const message, ...)
2016-10-15 17:29:02 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 17:29:02 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 17:29:02 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, show_char, 0, timestamp, NO_EOL, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, show_char, 0, timestamp, NO_EOL, theme_item, "", fmt_msg->str, NULL);
2016-10-15 17:29:02 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
void
2020-07-07 12:18:57 +00:00
win_println(ProfWin* window, theme_item_t theme_item, const char* show_char, const char* const message, ...)
2016-10-11 23:14:55 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-11 23:14:55 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-11 23:14:55 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, show_char, 0, timestamp, 0, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, show_char, 0, timestamp, 0, theme_item, "", fmt_msg->str, NULL);
2016-10-11 23:14:55 +00:00
2015-11-01 18:56:34 +00:00
inp_nonblocking(TRUE);
2015-08-08 23:40:37 +00:00
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2016-10-15 20:05:26 +00:00
void
2020-07-07 12:18:57 +00:00
win_println_indent(ProfWin* window, int pad, const char* const message, ...)
2016-10-15 20:05:26 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 20:05:26 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 20:05:26 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, "-", pad, timestamp, 0, THEME_DEFAULT, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, "-", pad, timestamp, 0, THEME_DEFAULT, "", fmt_msg->str, NULL);
2016-10-15 20:05:26 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2016-10-15 18:53:49 +00:00
void
2020-07-07 12:18:57 +00:00
win_append(ProfWin* window, theme_item_t theme_item, const char* const message, ...)
2016-10-15 18:53:49 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 18:53:49 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 18:53:49 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, "-", 0, timestamp, NO_DATE | NO_EOL, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, "-", 0, timestamp, NO_DATE | NO_EOL, theme_item, "", fmt_msg->str, NULL);
2016-10-15 18:53:49 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2016-10-15 18:19:44 +00:00
void
2020-07-07 12:18:57 +00:00
win_appendln(ProfWin* window, theme_item_t theme_item, const char* const message, ...)
2016-10-15 18:19:44 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 18:19:44 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 18:19:44 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, "-", 0, timestamp, NO_DATE, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, "-", 0, timestamp, NO_DATE, theme_item, "", fmt_msg->str, NULL);
2016-10-15 18:19:44 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2016-10-15 18:53:49 +00:00
2016-10-15 21:12:07 +00:00
void
2020-07-07 12:18:57 +00:00
win_append_highlight(ProfWin* window, theme_item_t theme_item, const char* const message, ...)
2016-10-15 21:12:07 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 21:12:07 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 21:12:07 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, "-", 0, timestamp, NO_DATE | NO_ME | NO_EOL, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, "-", 0, timestamp, NO_DATE | NO_ME | NO_EOL, theme_item, "", fmt_msg->str, NULL);
2016-10-15 21:12:07 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
void
2020-07-07 12:18:57 +00:00
win_appendln_highlight(ProfWin* window, theme_item_t theme_item, const char* const message, ...)
2016-10-15 21:12:07 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* timestamp = g_date_time_new_now_local();
2016-10-15 21:12:07 +00:00
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 21:12:07 +00:00
g_string_vprintf(fmt_msg, message, arg);
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, "-", 0, timestamp, NO_DATE | NO_ME, theme_item, "", NULL, fmt_msg->str, NULL, NULL);
_win_print_internal(window, "-", 0, timestamp, NO_DATE | NO_ME, theme_item, "", fmt_msg->str, NULL);
2016-10-15 21:12:07 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2016-10-11 22:40:15 +00:00
void
2020-07-20 11:01:05 +00:00
win_print_http_transfer(ProfWin* window, const char* const message, char* url)
2016-10-11 22:40:15 +00:00
{
win_print_outgoing_with_receipt(window, "!", NULL, message, url, NULL);
2016-10-11 22:40:15 +00:00
}
2015-03-14 16:05:13 +00:00
void
2020-07-07 12:18:57 +00:00
win_print_outgoing_with_receipt(ProfWin* window, const char* show_char, const char* const from, const char* const message, char* id, const char* const replace_id)
2015-03-14 16:05:13 +00:00
{
2020-07-07 12:18:57 +00:00
GDateTime* time = g_date_time_new_now_local();
2015-03-14 16:05:13 +00:00
2020-07-07 12:18:57 +00:00
DeliveryReceipt* receipt = malloc(sizeof(struct delivery_receipt_t));
2015-03-14 16:05:13 +00:00
receipt->received = FALSE;
2020-07-07 12:18:57 +00:00
const char* myjid = connection_get_fulljid();
if (replace_id) {
2020-03-09 12:19:25 +00:00
_win_correct(window, message, id, replace_id, myjid);
free(receipt); // TODO: probably we should use this in _win_correct()
} else {
2020-03-09 12:19:25 +00:00
buffer_append(window->layout->buffer, show_char, 0, time, 0, THEME_TEXT_ME, from, myjid, message, receipt, id);
_win_print_internal(window, show_char, 0, time, 0, THEME_TEXT_ME, from, message, receipt);
}
2015-03-14 16:05:13 +00:00
// TODO: cross-reference.. this should be replaced by a real event-based system
2015-11-01 18:56:34 +00:00
inp_nonblocking(TRUE);
2015-08-09 22:08:36 +00:00
g_date_time_unref(time);
2015-03-14 16:05:13 +00:00
}
void
2020-07-07 12:18:57 +00:00
win_mark_received(ProfWin* window, const char* const id)
2015-03-14 16:05:13 +00:00
{
gboolean received = buffer_mark_received(window->layout->buffer, id);
if (received) {
win_redraw(window);
}
}
2016-04-11 18:13:18 +00:00
void
2020-07-07 12:18:57 +00:00
win_update_entry_message(ProfWin* window, const char* const id, const char* const message)
2016-04-11 18:13:18 +00:00
{
2020-07-07 12:18:57 +00:00
ProfBuffEntry* entry = buffer_get_entry_by_id(window->layout->buffer, id);
2016-04-11 18:13:18 +00:00
if (entry) {
free(entry->message);
entry->message = strdup(message);
win_redraw(window);
}
}
void
2020-07-07 12:18:57 +00:00
win_remove_entry_message(ProfWin* window, const char* const id)
{
buffer_remove_entry_by_id(window->layout->buffer, id);
win_redraw(window);
}
2014-07-20 22:57:31 +00:00
void
2020-07-07 12:18:57 +00:00
win_newline(ProfWin* window)
2014-07-20 22:57:31 +00:00
{
2016-10-15 18:19:44 +00:00
win_appendln(window, THEME_DEFAULT, "");
2014-07-20 22:57:31 +00:00
}
2016-10-15 21:12:07 +00:00
static void
2020-07-07 12:18:57 +00:00
_win_printf(ProfWin* window, const char* show_char, int pad_indent, GDateTime* timestamp, int flags, theme_item_t theme_item, const char* const display_from, const char* const from_jid, const char* const message_id, const char* const message, ...)
2016-10-15 21:12:07 +00:00
{
if (timestamp == NULL) {
timestamp = g_date_time_new_now_local();
} else {
g_date_time_ref(timestamp);
}
va_list arg;
va_start(arg, message);
2020-07-07 12:18:57 +00:00
GString* fmt_msg = g_string_new(NULL);
2016-10-15 21:12:07 +00:00
g_string_vprintf(fmt_msg, message, arg);
buffer_append(window->layout->buffer, show_char, pad_indent, timestamp, flags, theme_item, display_from, from_jid, fmt_msg->str, NULL, message_id);
2016-10-15 21:12:07 +00:00
2020-03-09 10:51:32 +00:00
_win_print_internal(window, show_char, pad_indent, timestamp, flags, theme_item, display_from, fmt_msg->str, NULL);
2016-10-15 21:12:07 +00:00
inp_nonblocking(TRUE);
g_date_time_unref(timestamp);
g_string_free(fmt_msg, TRUE);
va_end(arg);
}
2014-07-20 22:35:48 +00:00
static void
2020-07-07 12:18:57 +00:00
_win_print_internal(ProfWin* window, const char* show_char, int pad_indent, GDateTime* time,
int flags, theme_item_t theme_item, const char* const from, const char* const message, DeliveryReceipt* receipt)
2014-07-16 21:03:46 +00:00
{
2020-02-19 15:36:54 +00:00
// flags : 1st bit = 0/1 - me/not me. define: NO_ME
// 2nd bit = 0/1 - date/no date. define: NO_DATE
// 3rd bit = 0/1 - eol/no eol. define: NO_EOL
// 4th bit = 0/1 - color from/no color from. define: NO_COLOUR_FROM
// 5th bit = 0/1 - color date/no date. define: NO_COLOUR_DATE
// 6th bit = 0/1 - trusted/untrusted. define: UNTRUSTED
gboolean me_message = FALSE;
int offset = 0;
2014-11-16 20:40:19 +00:00
int colour = theme_attrs(THEME_ME);
2015-05-30 01:45:10 +00:00
size_t indent = 0;
2014-07-16 21:03:46 +00:00
2020-07-07 12:18:57 +00:00
char* time_pref = NULL;
switch (window->type) {
2020-07-07 12:18:57 +00:00
case WIN_CHAT:
time_pref = prefs_get_string(PREF_TIME_CHAT);
break;
case WIN_MUC:
time_pref = prefs_get_string(PREF_TIME_MUC);
break;
case WIN_CONFIG:
time_pref = prefs_get_string(PREF_TIME_CONFIG);
break;
case WIN_PRIVATE:
time_pref = prefs_get_string(PREF_TIME_PRIVATE);
break;
case WIN_XML:
time_pref = prefs_get_string(PREF_TIME_XMLCONSOLE);
break;
default:
time_pref = prefs_get_string(PREF_TIME_CONSOLE);
break;
}
gchar* date_fmt = NULL;
if (g_strcmp0(time_pref, "off") == 0 || time == NULL) {
date_fmt = g_strdup("");
} else {
date_fmt = g_date_time_format(time, time_pref);
}
g_free(time_pref);
assert(date_fmt != NULL);
2014-11-16 01:11:00 +00:00
2020-07-07 12:18:57 +00:00
if (strlen(date_fmt) != 0) {
indent = 3 + strlen(date_fmt);
}
2014-11-16 01:11:00 +00:00
if ((flags & NO_DATE) == 0) {
if (date_fmt && strlen(date_fmt)) {
2014-11-16 01:28:41 +00:00
if ((flags & NO_COLOUR_DATE) == 0) {
2016-01-21 00:50:55 +00:00
wbkgdset(window->layout->win, theme_attrs(THEME_TIME));
2014-12-16 01:39:47 +00:00
wattron(window->layout->win, theme_attrs(THEME_TIME));
2014-11-16 01:28:41 +00:00
}
wprintw(window->layout->win, "%s %s ", date_fmt, show_char);
2014-11-16 01:28:41 +00:00
if ((flags & NO_COLOUR_DATE) == 0) {
2014-12-16 01:39:47 +00:00
wattroff(window->layout->win, theme_attrs(THEME_TIME));
2014-11-16 01:28:41 +00:00
}
}
}
2013-10-06 18:11:51 +00:00
if (from && strlen(from) > 0) {
2014-07-16 21:56:58 +00:00
if (flags & NO_ME) {
2014-11-16 20:40:19 +00:00
colour = theme_attrs(THEME_THEM);
2014-07-16 21:03:46 +00:00
}
2020-07-07 12:18:57 +00:00
char* color_pref = prefs_get_string(PREF_COLOR_NICK);
if (color_pref != NULL && (strcmp(color_pref, "false") != 0)) {
if (flags & NO_ME || (!(flags & NO_ME) && prefs_get_boolean(PREF_COLOR_NICK_OWN))) {
colour = theme_hash_attrs(from);
}
}
g_free(color_pref);
2014-07-16 21:56:58 +00:00
if (flags & NO_COLOUR_FROM) {
2014-07-16 21:03:46 +00:00
colour = 0;
}
2015-03-14 16:05:13 +00:00
if (receipt && !receipt->received) {
2015-03-15 19:35:25 +00:00
colour = theme_attrs(THEME_RECEIPT_SENT);
2015-03-14 16:05:13 +00:00
}
2016-01-21 00:50:55 +00:00
wbkgdset(window->layout->win, colour);
2014-12-16 01:39:47 +00:00
wattron(window->layout->win, colour);
2014-07-16 21:03:46 +00:00
if (strncmp(message, "/me ", 4) == 0) {
2014-12-16 01:39:47 +00:00
wprintw(window->layout->win, "*%s ", from);
2014-07-16 21:03:46 +00:00
offset = 4;
me_message = TRUE;
2014-07-16 21:03:46 +00:00
} else {
2014-12-16 01:39:47 +00:00
wprintw(window->layout->win, "%s: ", from);
wattroff(window->layout->win, colour);
2014-07-16 21:03:46 +00:00
}
}
2014-07-16 21:03:46 +00:00
if (!me_message) {
2015-03-14 16:05:13 +00:00
if (receipt && !receipt->received) {
2016-01-21 00:50:55 +00:00
wbkgdset(window->layout->win, theme_attrs(THEME_RECEIPT_SENT));
2015-03-15 19:35:25 +00:00
wattron(window->layout->win, theme_attrs(THEME_RECEIPT_SENT));
} else if (flags & UNTRUSTED) {
wbkgdset(window->layout->win, theme_attrs(THEME_UNTRUSTED));
wattron(window->layout->win, theme_attrs(THEME_UNTRUSTED));
2015-03-14 16:05:13 +00:00
} else {
2016-01-21 00:50:55 +00:00
wbkgdset(window->layout->win, theme_attrs(theme_item));
2015-03-14 16:05:13 +00:00
wattron(window->layout->win, theme_attrs(theme_item));
}
}
2014-07-16 21:03:46 +00:00
if (prefs_get_boolean(PREF_WRAP)) {
2020-07-07 12:18:57 +00:00
_win_print_wrapped(window->layout->win, message + offset, indent, pad_indent);
2014-07-16 21:56:58 +00:00
} else {
2020-07-07 12:18:57 +00:00
wprintw(window->layout->win, "%s", message + offset);
}
if ((flags & NO_EOL) == 0) {
int curx = getcurx(window->layout->win);
if (curx != 0) {
wprintw(window->layout->win, "\n");
}
2014-07-16 21:03:46 +00:00
}
if (me_message) {
2014-12-16 01:39:47 +00:00
wattroff(window->layout->win, colour);
} else {
2015-03-14 16:05:13 +00:00
if (receipt && !receipt->received) {
2015-03-15 19:35:25 +00:00
wattroff(window->layout->win, theme_attrs(THEME_RECEIPT_SENT));
2015-03-14 16:05:13 +00:00
} else {
wattroff(window->layout->win, theme_attrs(theme_item));
}
2013-10-06 18:11:51 +00:00
}
2015-08-08 23:40:37 +00:00
g_free(date_fmt);
2013-10-06 18:11:51 +00:00
}
2014-11-08 21:03:28 +00:00
static void
2020-07-07 12:18:57 +00:00
_win_indent(WINDOW* win, int size)
2014-11-08 21:03:28 +00:00
{
for (int i = 0; i < size; i++) {
2014-11-16 16:27:11 +00:00
waddch(win, ' ');
}
}
static void
2020-07-07 12:18:57 +00:00
_win_print_wrapped(WINDOW* win, const char* const message, size_t indent, int pad_indent)
2014-11-16 16:27:11 +00:00
{
2015-07-29 21:48:03 +00:00
int starty = getcury(win);
2014-11-08 21:03:28 +00:00
int wordi = 0;
2020-07-07 12:18:57 +00:00
char* word = malloc(strlen(message) + 1);
2014-11-08 21:03:28 +00:00
2020-07-07 12:18:57 +00:00
gchar* curr_ch = g_utf8_offset_to_pointer(message, 0);
2015-02-01 00:14:11 +00:00
while (*curr_ch != '\0') {
2015-07-29 21:48:03 +00:00
// handle space
2015-02-01 00:14:11 +00:00
if (*curr_ch == ' ') {
2014-11-16 16:27:11 +00:00
waddch(win, ' ');
2015-02-01 00:14:11 +00:00
curr_ch = g_utf8_next_char(curr_ch);
2015-07-29 21:48:03 +00:00
2020-07-07 12:18:57 +00:00
// handle newline
2015-02-01 00:14:11 +00:00
} else if (*curr_ch == '\n') {
2014-11-16 01:11:00 +00:00
waddch(win, '\n');
2015-07-29 21:48:03 +00:00
_win_indent(win, indent + pad_indent);
2015-02-01 00:14:11 +00:00
curr_ch = g_utf8_next_char(curr_ch);
2015-07-29 21:48:03 +00:00
2020-07-07 12:18:57 +00:00
// handle word
2014-11-08 21:03:28 +00:00
} else {
wordi = 0;
int wordlen = 0;
2015-02-01 00:14:11 +00:00
while (*curr_ch != ' ' && *curr_ch != '\n' && *curr_ch != '\0') {
size_t ch_len = mbrlen(curr_ch, MB_CUR_MAX, NULL);
if ((ch_len == (size_t)-2) || (ch_len == (size_t)-1)) {
curr_ch++;
continue;
}
2015-02-01 00:14:11 +00:00
int offset = 0;
while (offset < ch_len) {
word[wordi++] = curr_ch[offset++];
}
curr_ch = g_utf8_next_char(curr_ch);
2014-11-08 21:03:28 +00:00
}
word[wordi] = '\0';
wordlen = utf8_display_len(word);
2014-11-08 21:16:16 +00:00
int curx = getcurx(win);
2019-07-22 12:27:14 +00:00
int cury;
2014-11-08 21:16:16 +00:00
int maxx = getmaxx(win);
// wrap required
if (curx + wordlen > maxx) {
int linelen = maxx - (indent + pad_indent);
// word larger than line
if (wordlen > linelen) {
2020-07-07 12:18:57 +00:00
gchar* word_ch = g_utf8_offset_to_pointer(word, 0);
while (*word_ch != '\0') {
curx = getcurx(win);
cury = getcury(win);
gboolean firstline = cury == starty;
if (firstline && curx < indent) {
_win_indent(win, indent);
}
if (!firstline && curx < (indent + pad_indent)) {
_win_indent(win, indent + pad_indent);
}
2020-07-07 12:18:57 +00:00
gchar copy[wordi + 1];
g_utf8_strncpy(copy, word_ch, 1);
waddstr(win, copy);
word_ch = g_utf8_next_char(word_ch);
}
2015-07-29 21:48:03 +00:00
2020-07-07 12:18:57 +00:00
// newline and print word
} else {
waddch(win, '\n');
curx = getcurx(win);
2015-07-29 21:48:03 +00:00
cury = getcury(win);
gboolean firstline = cury == starty;
2015-07-29 21:48:03 +00:00
if (firstline && curx < indent) {
2014-11-16 16:27:11 +00:00
_win_indent(win, indent);
}
2015-07-29 21:48:03 +00:00
if (!firstline && curx < (indent + pad_indent)) {
_win_indent(win, indent + pad_indent);
}
waddstr(win, word);
}
2020-07-07 12:18:57 +00:00
// no wrap required
} else {
curx = getcurx(win);
cury = getcury(win);
gboolean firstline = cury == starty;
2015-07-29 21:48:03 +00:00
if (firstline && curx < indent) {
2014-11-16 16:27:11 +00:00
_win_indent(win, indent);
}
2015-07-29 21:48:03 +00:00
if (!firstline && curx < (indent + pad_indent)) {
_win_indent(win, indent + pad_indent);
}
waddstr(win, word);
2014-11-08 21:16:16 +00:00
}
2014-11-08 21:03:28 +00:00
}
2015-07-29 21:48:03 +00:00
// consume first space of next line
int curx = getcurx(win);
int cury = getcury(win);
gboolean firstline = (cury == starty);
if (!firstline && curx == 0 && *curr_ch == ' ') {
curr_ch = g_utf8_next_char(curr_ch);
}
2014-11-08 21:03:28 +00:00
}
free(word);
}
void
2020-07-07 12:18:57 +00:00
win_print_trackbar(ProfWin* window)
{
int cols = getmaxx(window->layout->win);
wbkgdset(window->layout->win, theme_attrs(THEME_TRACKBAR));
wattron(window->layout->win, theme_attrs(THEME_TRACKBAR));
for (int i = 1; i <= cols; i++) {
wprintw(window->layout->win, "-");
}
wattroff(window->layout->win, theme_attrs(THEME_TRACKBAR));
}
2014-07-16 21:03:46 +00:00
void
2020-07-07 12:18:57 +00:00
win_redraw(ProfWin* window)
2014-07-16 21:03:46 +00:00
{
int size;
2014-12-16 01:39:47 +00:00
werase(window->layout->win);
size = buffer_size(window->layout->buffer);
2014-07-16 21:03:46 +00:00
for (int i = 0; i < size; i++) {
2020-07-07 12:18:57 +00:00
ProfBuffEntry* e = buffer_get_entry(window->layout->buffer, i);
if (e->display_from == NULL && e->message && e->message[0] == '-') {
2020-02-21 18:17:12 +00:00
// just an indicator to print the trackbar/separator not the actual message
win_print_trackbar(window);
} else {
// regular thing to print
_win_print_internal(window, e->show_char, e->pad_indent, e->time, e->flags, e->theme_item, e->display_from, e->message, e->receipt);
}
}
2014-07-18 21:28:49 +00:00
}
2014-12-16 01:39:47 +00:00
gboolean
2020-07-07 12:18:57 +00:00
win_has_active_subwin(ProfWin* window)
2014-12-16 01:39:47 +00:00
{
if (window->layout->type == LAYOUT_SPLIT) {
2020-07-07 12:18:57 +00:00
ProfLayoutSplit* layout = (ProfLayoutSplit*)window->layout;
2014-12-16 01:39:47 +00:00
return (layout->subwin != NULL);
} else {
return FALSE;
}
}
gboolean
2020-07-07 12:18:57 +00:00
win_notify_remind(ProfWin* window)
{
switch (window->type) {
case WIN_CHAT:
{
2020-07-07 12:18:57 +00:00
ProfChatWin* chatwin = (ProfChatWin*)window;
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
if (prefs_get_boolean(PREF_NOTIFY_CHAT) && chatwin->unread > 0) {
return TRUE;
} else {
return FALSE;
}
}
case WIN_MUC:
{
2020-07-07 12:18:57 +00:00
ProfMucWin* mucwin = (ProfMucWin*)window;
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
2016-02-07 01:15:40 +00:00
return prefs_do_room_notify_mention(mucwin->roomjid, mucwin->unread, mucwin->unread_mentions, mucwin->unread_triggers);
}
case WIN_PRIVATE:
{
2020-07-07 12:18:57 +00:00
ProfPrivateWin* privatewin = (ProfPrivateWin*)window;
assert(privatewin->memcheck == PROFPRIVATEWIN_MEMCHECK);
if (prefs_get_boolean(PREF_NOTIFY_CHAT) && privatewin->unread > 0) {
return TRUE;
} else {
return FALSE;
}
}
default:
return FALSE;
}
}
2014-12-21 23:14:01 +00:00
int
2020-07-07 12:18:57 +00:00
win_unread(ProfWin* window)
2014-12-21 23:14:01 +00:00
{
if (window->type == WIN_CHAT) {
2020-07-07 12:18:57 +00:00
ProfChatWin* chatwin = (ProfChatWin*)window;
2014-12-21 23:14:01 +00:00
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
return chatwin->unread;
} else if (window->type == WIN_MUC) {
2020-07-07 12:18:57 +00:00
ProfMucWin* mucwin = (ProfMucWin*)window;
2014-12-21 23:14:01 +00:00
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
return mucwin->unread;
} else if (window->type == WIN_PRIVATE) {
2020-07-07 12:18:57 +00:00
ProfPrivateWin* privatewin = (ProfPrivateWin*)window;
2014-12-21 23:14:01 +00:00
assert(privatewin->memcheck == PROFPRIVATEWIN_MEMCHECK);
return privatewin->unread;
} else {
return 0;
}
}
2021-05-29 18:16:21 +00:00
gboolean
2021-10-05 08:01:27 +00:00
win_has_attention(ProfWin* window)
2021-05-29 18:16:21 +00:00
{
if (window->type == WIN_CHAT) {
ProfChatWin* chatwin = (ProfChatWin*)window;
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
return chatwin->has_attention;
} else if (window->type == WIN_MUC) {
ProfMucWin* mucwin = (ProfMucWin*)window;
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
return mucwin->has_attention;
}
return FALSE;
}
2021-10-05 08:01:27 +00:00
gboolean
win_toggle_attention(ProfWin* window)
2021-05-29 18:16:21 +00:00
{
if (window->type == WIN_CHAT) {
ProfChatWin* chatwin = (ProfChatWin*)window;
assert(chatwin->memcheck == PROFCHATWIN_MEMCHECK);
chatwin->has_attention = !chatwin->has_attention;
return chatwin->has_attention;
} else if (window->type == WIN_MUC) {
ProfMucWin* mucwin = (ProfMucWin*)window;
assert(mucwin->memcheck == PROFMUCWIN_MEMCHECK);
mucwin->has_attention = !mucwin->has_attention;
return mucwin->has_attention;
}
return FALSE;
}
void
2020-07-07 12:18:57 +00:00
win_sub_print(WINDOW* win, char* msg, gboolean newline, gboolean wrap, int indent)
{
int maxx = getmaxx(win);
int curx = getcurx(win);
int cury = getcury(win);
2015-11-22 01:39:20 +00:00
if (wrap) {
_win_print_wrapped(win, msg, 1, indent);
} else {
waddnstr(win, msg, maxx - curx);
}
2015-11-21 21:03:53 +00:00
if (newline) {
2020-07-07 12:18:57 +00:00
wmove(win, cury + 1, 0);
2015-11-21 21:03:53 +00:00
}
}
void
2020-07-07 12:18:57 +00:00
win_sub_newline_lazy(WINDOW* win)
{
int curx;
if (win == NULL) {
return;
}
curx = getcurx(win);
if (curx > 0) {
int cury = getcury(win);
2020-07-07 12:18:57 +00:00
wmove(win, cury + 1, 0);
}
}
void
2020-07-07 12:18:57 +00:00
win_command_list_error(ProfWin* window, const char* const error)
{
assert(window != NULL);
win_println(window, THEME_ERROR, "!", "Error retrieving command list: %s", error);
}
2018-03-23 18:49:48 +00:00
void
2020-07-07 12:18:57 +00:00
win_command_exec_error(ProfWin* window, const char* const command, const char* const error, ...)
2018-03-23 18:49:48 +00:00
{
assert(window != NULL);
va_list arg;
va_start(arg, error);
2020-07-07 12:18:57 +00:00
GString* msg = g_string_new(NULL);
2018-03-23 18:49:48 +00:00
g_string_vprintf(msg, error, arg);
win_println(window, THEME_ERROR, "!", "Error executing command %s: %s", command, msg->str);
2018-03-23 18:49:48 +00:00
g_string_free(msg, TRUE);
va_end(arg);
}
void
2020-07-07 12:18:57 +00:00
win_handle_command_list(ProfWin* window, GSList* cmds)
{
assert(window != NULL);
if (cmds) {
win_println(window, THEME_DEFAULT, "!", "Ad hoc commands:");
2020-07-07 12:18:57 +00:00
GSList* curr_cmd = cmds;
while (curr_cmd) {
2020-07-07 12:18:57 +00:00
const char* cmd = curr_cmd->data;
win_println(window, THEME_DEFAULT, "!", " %s", cmd);
curr_cmd = g_slist_next(curr_cmd);
}
win_println(window, THEME_DEFAULT, "!", "");
} else {
win_println(window, THEME_DEFAULT, "!", "No commands found");
win_println(window, THEME_DEFAULT, "!", "");
}
}
2018-05-30 17:25:06 +00:00
void
2020-07-07 12:18:57 +00:00
win_handle_command_exec_status(ProfWin* window, const char* const command, const char* const value)
2018-05-30 17:25:06 +00:00
{
assert(window != NULL);
win_println(window, THEME_DEFAULT, "!", "%s %s", command, value);
2018-05-30 17:25:06 +00:00
}
void
2020-07-07 12:18:57 +00:00
win_handle_command_exec_result_note(ProfWin* window, const char* const type, const char* const value)
{
assert(window != NULL);
win_println(window, THEME_DEFAULT, "!", value);
}
void
2020-07-07 12:18:57 +00:00
win_insert_last_read_position_marker(ProfWin* window, char* id)
{
int size;
size = buffer_size(window->layout->buffer);
// TODO: this is somewhat costly. We should improve this later.
// check if we already have a separator present
for (int i = 0; i < size; i++) {
2020-07-07 12:18:57 +00:00
ProfBuffEntry* e = buffer_get_entry(window->layout->buffer, i);
// if yes, don't print a new one
if (e->id && (g_strcmp0(e->id, id) == 0)) {
return;
}
}
2020-07-07 12:18:57 +00:00
GDateTime* time = g_date_time_new_now_local();
2020-02-21 18:17:12 +00:00
// the trackbar/separator will actually be print in win_redraw().
// this only puts it in the buffer and win_redraw() will interpret it.
// so that we have the correct length even when resizing.
2020-03-09 11:47:43 +00:00
buffer_append(window->layout->buffer, " ", 0, time, 0, THEME_TEXT, NULL, NULL, "-", NULL, id);
win_redraw(window);
g_date_time_unref(time);
}