1
0
mirror of https://github.com/weechat/weechat.git synced 2026-06-24 11:56:38 +02:00
Files
weechat/src/plugins/irc/irc-input.c
T

595 lines
19 KiB
C

/*
* SPDX-FileCopyrightText: 2003-2026 Sébastien Helleu <flashcode@flashtux.org>
*
* SPDX-License-Identifier: GPL-3.0-or-later
*
* This file is part of WeeChat, the extensible chat client.
*
* WeeChat 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.
*
* WeeChat 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 WeeChat. If not, see <https://www.gnu.org/licenses/>.
*/
/* Input data management for IRC buffers */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "../weechat-plugin.h"
#include "irc.h"
#include "irc-buffer.h"
#include "irc-server.h"
#include "irc-channel.h"
#include "irc-message.h"
#include "irc-nick.h"
#include "irc-color.h"
#include "irc-config.h"
#include "irc-list.h"
#include "irc-msgbuffer.h"
#include "irc-protocol.h"
#include "irc-raw.h"
/*
* Display user message.
*
* If ctcp_type == "action", then message is displayed as an action
* (like command /me), for example:
*
* * | nick is testing
*
* If target is a channel or a nick, the message is displayed like this:
*
* nick | test
*
* If target is a channel with STATUSMSG (for example "@#test"), the message
* is displayed with the target, like this (message, action, notice):
*
* Msg(nick) -> @#test: test message for ops
* Action -> @#test: nick is testing
* Notice(nick) -> @#test: test notice for ops
*
* If decode_colors is 1, colors are stripped if the option
* irc.network.colors_send is off.
*/
void
irc_input_user_message_display (struct t_irc_server *server,
time_t date,
int date_usec,
struct t_hashtable *tags,
const char *target,
const char *address,
const char *command,
const char *ctcp_type,
const char *text,
int decode_colors)
{
struct t_irc_channel *ptr_channel;
struct t_gui_buffer *ptr_buffer;
struct t_irc_nick *ptr_nick;
struct t_irc_protocol_ctxt ctxt;
const char *ptr_target;
char *text2, *text_decoded, str_tags[256], *str_color;
const char *ptr_text;
int is_notice, is_action, is_channel, display_target;
if (!server || !target)
return;
memset (&ctxt, 0, sizeof (ctxt));
ctxt.server = server;
ctxt.date = date;
ctxt.date_usec = date_usec;
ctxt.tags = tags;
ctxt.address = (char *)address;
ctxt.command = (char *)command;
is_notice = (weechat_strcasecmp (command, "notice") == 0);
is_action = (ctcp_type && (weechat_strcasecmp (ctcp_type, "action") == 0));
is_channel = 0;
ptr_channel = NULL;
display_target = is_notice || (ctcp_type && !is_action);
ptr_target = target;
if (irc_server_prefix_char_statusmsg (server, target[0])
&& irc_channel_is_channel (server, target + 1))
{
ptr_channel = irc_channel_search (server, target + 1);
is_channel = 1;
display_target = 1;
ptr_target = target + 1;
}
else
{
ptr_channel = irc_channel_search (server, target);
if (irc_channel_is_channel (server, target))
is_channel = 1;
}
if (ptr_channel)
{
ptr_buffer = ptr_channel->buffer;
}
else
{
ptr_buffer = irc_msgbuffer_get_target_buffer (
server, ptr_target, command, NULL, NULL);
if (!ptr_buffer)
ptr_buffer = server->buffer;
}
if (ptr_buffer == server->buffer)
display_target = 1;
text2 = irc_message_hide_password (server, target, text);
text_decoded = (decode_colors) ?
irc_color_decode (
(text2) ? text2 : text,
weechat_config_boolean (irc_config_network_colors_send)) : NULL;
ptr_nick = NULL;
if (ptr_channel && (ptr_channel->type == IRC_CHANNEL_TYPE_CHANNEL))
ptr_nick = irc_nick_search (server, ptr_channel, server->nick);
ctxt.nick = (ptr_nick) ? ptr_nick->name : server->nick;
ctxt.nick_is_me = (irc_server_strcasecmp (server, ctxt.nick, server->nick) == 0);
if (is_action)
{
snprintf (str_tags, sizeof (str_tags),
"irc_action,self_msg,notify_none,no_highlight");
}
else
{
if (display_target)
{
snprintf (str_tags, sizeof (str_tags),
"%sself_msg,notify_none,no_highlight",
(ctcp_type) ? "irc_ctcp," : "");
}
else
{
str_color = irc_color_for_tags (
weechat_config_color (
weechat_config_get ("weechat.color.chat_nick_self")));
snprintf (str_tags, sizeof (str_tags),
"%sself_msg,notify_none,no_highlight,prefix_nick_%s",
(ctcp_type) ? "irc_ctcp," : "",
(str_color) ? str_color : "default");
free (str_color);
}
}
ptr_text = (text_decoded) ? text_decoded : ((text2) ? text2 : text);
if (is_action)
{
if (display_target)
{
weechat_printf_datetime_tags (
ptr_buffer,
date,
date_usec,
irc_protocol_tags (&ctxt, str_tags),
"%s%s -> %s%s%s: %s%s%s%s%s%s",
weechat_prefix ("network"),
/* TRANSLATORS: "Action" is an IRC CTCP "ACTION" sent with /me or /action */
_("Action"),
(is_channel) ?
IRC_COLOR_CHAT_CHANNEL : irc_nick_color_for_msg (server, 0, NULL, target),
target,
IRC_COLOR_RESET,
irc_nick_mode_for_display (server, ptr_nick, 0),
IRC_COLOR_CHAT_NICK_SELF,
server->nick,
(ptr_text && ptr_text[0]) ? IRC_COLOR_RESET : "",
(ptr_text && ptr_text[0]) ? " " : "",
IRC_COLOR_MSG(ptr_text));
}
else
{
weechat_printf_datetime_tags (
ptr_buffer,
date,
date_usec,
irc_protocol_tags (&ctxt, str_tags),
"%s%s%s%s%s%s%s",
weechat_prefix ("action"),
irc_nick_mode_for_display (server, ptr_nick, 0),
IRC_COLOR_CHAT_NICK_SELF,
server->nick,
IRC_COLOR_RESET,
(ptr_text && ptr_text[0]) ? " " : "",
IRC_COLOR_MSG(ptr_text));
}
}
else if (ctcp_type)
{
weechat_printf_datetime_tags (
ptr_buffer,
date,
date_usec,
irc_protocol_tags (&ctxt, str_tags),
_("%sCTCP query to %s%s%s: %s%s%s%s%s"),
weechat_prefix ("network"),
(is_channel) ?
IRC_COLOR_CHAT_CHANNEL : irc_nick_color_for_msg (server, 0, NULL, target),
target,
IRC_COLOR_RESET,
IRC_COLOR_CHAT_CHANNEL,
ctcp_type,
IRC_COLOR_RESET,
(ptr_text && ptr_text[0]) ? " " : "",
IRC_COLOR_MSG(ptr_text));
}
else if (display_target)
{
weechat_printf_datetime_tags (
ptr_buffer,
date,
date_usec,
irc_protocol_tags (&ctxt, str_tags),
"%s%s%s%s%s(%s%s%s%s)%s -> %s%s%s: %s",
weechat_prefix ("network"),
(is_notice) ? IRC_COLOR_NOTICE : "",
(is_notice) ?
/* TRANSLATORS: "Notice" is command name in IRC protocol (translation is frequently the same word) */
_("Notice") :
"Msg",
(is_notice) ? IRC_COLOR_RESET : "",
IRC_COLOR_CHAT_DELIMITERS,
irc_nick_mode_for_display (server, ptr_nick, 0),
IRC_COLOR_CHAT_NICK_SELF,
server->nick,
IRC_COLOR_CHAT_DELIMITERS,
IRC_COLOR_RESET,
(is_channel) ?
IRC_COLOR_CHAT_CHANNEL : irc_nick_color_for_msg (server, 0, NULL, target),
target,
IRC_COLOR_RESET,
IRC_COLOR_MSG(ptr_text));
}
else
{
weechat_printf_datetime_tags (
ptr_buffer,
date,
date_usec,
irc_protocol_tags (&ctxt, str_tags),
"%s%s",
irc_nick_as_prefix (
server,
(ptr_nick) ? ptr_nick : NULL,
(ptr_nick) ? NULL : server->nick,
IRC_COLOR_CHAT_NICK_SELF),
(ptr_text) ? ptr_text : "");
}
free (text2);
free (text_decoded);
}
/*
* Send a PRIVMSG message, and split it if message size is > 512 bytes (by default).
*
* Warning: this function makes temporary changes in "message".
*/
void
irc_input_send_user_message (struct t_gui_buffer *buffer, int flags,
const char *tags, char *message)
{
int i, action, list_size;
struct t_arraylist *list_messages;
IRC_BUFFER_GET_SERVER_CHANNEL(buffer);
if (!ptr_server || !ptr_channel || !message || !message[0])
return;
if (!ptr_server->is_connected)
{
weechat_printf (buffer,
_("%s%s: you are not connected to server"),
weechat_prefix ("error"), IRC_PLUGIN_NAME);
return;
}
list_messages = irc_server_sendf (ptr_server,
flags | IRC_SERVER_SEND_RETURN_LIST
| IRC_SERVER_SEND_MULTILINE,
tags,
"PRIVMSG %s :%s",
ptr_channel->name, message);
if (list_messages)
{
/* display only if capability "echo-message" is NOT enabled */
if (!weechat_hashtable_has_key (ptr_server->cap_list, "echo-message"))
{
action = ((strncmp (message, "\001ACTION ", 8) == 0)
|| (strncmp (message, "\001ACTION\001", 8) == 0));
list_size = weechat_arraylist_size (list_messages);
for (i = 0; i < list_size; i++)
{
irc_input_user_message_display (
ptr_server,
0, /* date */
0, /* date_usec */
NULL, /* tags */
ptr_channel->name,
NULL, /* address */
"privmsg",
(action) ? "action" : NULL,
(const char *)weechat_arraylist_get (list_messages, i),
1); /* decode_colors */
}
}
weechat_arraylist_free (list_messages);
}
}
/*
* Input data in a buffer.
*/
int
irc_input_data (struct t_gui_buffer *buffer, const char *input_data, int flags,
int force_user_message)
{
const char *ptr_data;
char *data_with_colors, *msg;
IRC_BUFFER_GET_SERVER_CHANNEL(buffer);
if (buffer == irc_raw_buffer)
{
if (weechat_strcmp (input_data, "q") == 0)
weechat_buffer_close (buffer);
else
irc_raw_filter_options (input_data);
}
else if (weechat_strcmp (
weechat_buffer_get_string (buffer,
"localvar_type"), "list") == 0)
{
irc_list_buffer_input_data (buffer, input_data);
}
else
{
/*
* if send unknown commands is enabled and that input data is a
* command, then send this command to IRC server
*/
if (!force_user_message
&& weechat_config_boolean (irc_config_network_send_unknown_commands)
&& !weechat_string_input_for_buffer (input_data))
{
if (ptr_server)
{
irc_server_sendf (ptr_server,
flags | IRC_SERVER_SEND_MULTILINE,
NULL,
"%s", weechat_utf8_next_char (input_data));
}
return WEECHAT_RC_OK;
}
if (ptr_channel)
{
ptr_data = input_data;
if (!force_user_message)
{
ptr_data = weechat_string_input_for_buffer (input_data);
if (!ptr_data)
ptr_data = input_data;
}
data_with_colors = irc_color_encode (
ptr_data,
weechat_config_boolean (irc_config_network_colors_send));
msg = strdup ((data_with_colors) ? data_with_colors : ptr_data);
if (msg)
{
irc_input_send_user_message (buffer, flags, NULL, msg);
free (msg);
}
free (data_with_colors);
}
else
{
weechat_printf (buffer,
_("%s%s: this buffer is not a channel!"),
weechat_prefix ("error"), IRC_PLUGIN_NAME);
}
}
return WEECHAT_RC_OK;
}
/*
* Callback for input data in a buffer.
*/
int
irc_input_data_cb (const void *pointer, void *data,
struct t_gui_buffer *buffer,
const char *input_data)
{
/* make C compiler happy */
(void) pointer;
(void) data;
return irc_input_data (buffer, input_data,
IRC_SERVER_SEND_OUTQ_PRIO_HIGH, 0);
}
/*
* Callback for signal "irc_input_send" signal.
*
* This signal can be used by other plugins/scripts, it simulates input or
* command from user on an IRC buffer (it is used for example by Relay plugin).
*
* Format of signal_data (string) is "server;channel;options;tags;text"
* server: server name (required)
* channel: channel name (optional)
* options: comma-separated list of options (optional):
* "priority_high": send with high priority (default)
* "priority_low": send with low priority
* "user_message": force user message (don't execute a command)
* tags: tags for irc_server_sendf() (optional)
* text: text or command (required).
*/
int
irc_input_send_cb (const void *pointer, void *data,
const char *signal,
const char *type_data, void *signal_data)
{
const char *ptr_string, *ptr_message;
char *pos_semicol1, *pos_semicol2, *pos_semicol3, *pos_semicol4;
char *server, *channel, *options, *tags, *data_with_colors, **list_options;
int i, num_options, flags, force_user_message;
struct t_irc_server *ptr_server;
struct t_irc_channel *ptr_channel;
struct t_gui_buffer *ptr_buffer;
/* make C compiler happy */
(void) pointer;
(void) data;
(void) signal;
(void) type_data;
ptr_string = (const char *)signal_data;
server = NULL;
channel = NULL;
options = NULL;
flags = IRC_SERVER_SEND_OUTQ_PRIO_HIGH;
force_user_message = 0;
tags = NULL;
ptr_message = NULL;
ptr_server = NULL;
ptr_channel = NULL;
pos_semicol1 = strchr (ptr_string, ';');
if (pos_semicol1)
{
if (pos_semicol1 > ptr_string + 1)
{
server = weechat_strndup (ptr_string, pos_semicol1 - ptr_string);
}
pos_semicol2 = strchr (pos_semicol1 + 1, ';');
if (pos_semicol2)
{
if (pos_semicol2 > pos_semicol1 + 1)
{
channel = weechat_strndup (pos_semicol1 + 1,
pos_semicol2 - pos_semicol1 - 1);
}
pos_semicol3 = strchr (pos_semicol2 + 1, ';');
if (pos_semicol3)
{
if (pos_semicol3 > pos_semicol2 + 1)
{
options = weechat_strndup (pos_semicol2 + 1,
pos_semicol3 - pos_semicol2 - 1);
}
pos_semicol4 = strchr (pos_semicol3 + 1, ';');
if (pos_semicol4)
{
if (pos_semicol4 > pos_semicol3 + 1)
{
tags = weechat_strndup (pos_semicol3 + 1,
pos_semicol4 - pos_semicol3 - 1);
}
ptr_message = pos_semicol4 + 1;
}
}
}
}
if (options && options[0])
{
list_options = weechat_string_split (
options,
",",
NULL,
WEECHAT_STRING_SPLIT_STRIP_LEFT
| WEECHAT_STRING_SPLIT_STRIP_RIGHT
| WEECHAT_STRING_SPLIT_COLLAPSE_SEPS,
0,
&num_options);
if (list_options)
{
for (i = 0; i < num_options; i++)
{
if (strcmp (list_options[i], "priority_high") == 0)
flags = IRC_SERVER_SEND_OUTQ_PRIO_HIGH;
else if (strcmp (list_options[i], "priority_low") == 0)
flags = IRC_SERVER_SEND_OUTQ_PRIO_LOW;
else if (strcmp (list_options[i], "user_message") == 0)
force_user_message = 1;
}
weechat_string_free_split (list_options);
}
}
if (server && ptr_message)
{
ptr_server = irc_server_search (server);
if (ptr_server)
{
ptr_buffer = ptr_server->buffer;
if (channel)
{
ptr_channel = irc_channel_search (ptr_server, channel);
if (ptr_channel)
ptr_buffer = ptr_channel->buffer;
}
/* set tags to use by default */
irc_server_set_send_default_tags (tags);
/* send text to buffer, or execute command */
if (force_user_message
|| weechat_string_input_for_buffer (ptr_message))
{
/* text as input */
irc_input_data (ptr_buffer, ptr_message, flags, 1);
}
else
{
/* command */
data_with_colors = irc_color_encode (
ptr_message,
weechat_config_boolean (irc_config_network_colors_send));
weechat_command (
ptr_buffer,
(data_with_colors) ? data_with_colors : ptr_message);
free (data_with_colors);
}
/* reset tags to use by default */
irc_server_set_send_default_tags (NULL);
}
}
free (server);
free (channel);
free (options);
free (tags);
return WEECHAT_RC_OK;
}