mirror of
https://github.com/weechat/weechat.git
synced 2026-06-26 04:46:37 +02:00
966 lines
32 KiB
C
966 lines
32 KiB
C
/*
|
|
* Copyright (c) 2003-2010 by FlashCode <flashcode@flashtux.org>
|
|
* See README for License detail, AUTHORS for developers list.
|
|
*
|
|
* This program 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.
|
|
*
|
|
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
* irc-channel.c: channel and private chat management for IRC plugin
|
|
*/
|
|
|
|
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <time.h>
|
|
|
|
#include "../weechat-plugin.h"
|
|
#include "irc.h"
|
|
#include "irc-channel.h"
|
|
#include "irc-buffer.h"
|
|
#include "irc-color.h"
|
|
#include "irc-command.h"
|
|
#include "irc-config.h"
|
|
#include "irc-nick.h"
|
|
#include "irc-server.h"
|
|
#include "irc-input.h"
|
|
|
|
|
|
/*
|
|
* irc_channel_valid: check if a channel pointer exists for a server
|
|
* return 1 if channel exists
|
|
* 0 if channel is not found
|
|
*/
|
|
|
|
int
|
|
irc_channel_valid (struct t_irc_server *server, struct t_irc_channel *channel)
|
|
{
|
|
struct t_irc_channel *ptr_channel;
|
|
|
|
if (!server)
|
|
return 0;
|
|
|
|
for (ptr_channel = server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if (ptr_channel == channel)
|
|
return 1;
|
|
}
|
|
|
|
/* channel not found */
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_move_near_server: move new channel/pv buffer near server
|
|
*/
|
|
|
|
void
|
|
irc_channel_move_near_server (struct t_irc_server *server, int channel_type,
|
|
struct t_gui_buffer *buffer)
|
|
{
|
|
int number, number_channel, number_last_channel, number_last_private;
|
|
int number_found;
|
|
char str_number[32];
|
|
struct t_irc_channel *ptr_channel;
|
|
|
|
number = weechat_buffer_get_integer (buffer, "number");
|
|
number_last_channel = 0;
|
|
number_last_private = 0;
|
|
number_found = 0;
|
|
|
|
if (server->channels)
|
|
{
|
|
/* search last channel/pv number for server */
|
|
for (ptr_channel = server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if (ptr_channel->buffer)
|
|
{
|
|
number_channel = weechat_buffer_get_integer (ptr_channel->buffer,
|
|
"number");
|
|
switch (ptr_channel->type)
|
|
{
|
|
case IRC_CHANNEL_TYPE_CHANNEL:
|
|
if (number_channel > number_last_channel)
|
|
number_last_channel = number_channel;
|
|
break;
|
|
case IRC_CHANNEL_TYPE_PRIVATE:
|
|
if (number_channel > number_last_private)
|
|
number_last_private = number_channel;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* use last channel/pv number + 1 */
|
|
switch (channel_type)
|
|
{
|
|
case IRC_CHANNEL_TYPE_CHANNEL:
|
|
if (number_last_channel > 0)
|
|
number_found = number_last_channel + 1;
|
|
break;
|
|
case IRC_CHANNEL_TYPE_PRIVATE:
|
|
if (number_last_private > 0)
|
|
number_found = number_last_private + 1;
|
|
else if (number_last_channel > 0)
|
|
number_found = number_last_channel + 1;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (weechat_config_integer (irc_config_look_server_buffer) ==
|
|
IRC_CONFIG_LOOK_SERVER_BUFFER_INDEPENDENT)
|
|
{
|
|
number_found = weechat_buffer_get_integer (server->buffer, "number") + 1;
|
|
}
|
|
}
|
|
|
|
/* switch to number found */
|
|
if ((number_found >= 1) && (number_found != number))
|
|
{
|
|
snprintf (str_number, sizeof (str_number), "%d", number_found);
|
|
weechat_buffer_set (buffer, "number", str_number);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_new: allocate a new channel for a server and add it to channels
|
|
* list
|
|
*/
|
|
|
|
struct t_irc_channel *
|
|
irc_channel_new (struct t_irc_server *server, int channel_type,
|
|
const char *channel_name, int switch_to_channel,
|
|
int auto_switch)
|
|
{
|
|
struct t_irc_channel *new_channel;
|
|
struct t_gui_buffer *new_buffer;
|
|
int buffer_created;
|
|
char *buffer_name;
|
|
|
|
/* alloc memory for new channel */
|
|
if ((new_channel = malloc (sizeof (*new_channel))) == NULL)
|
|
{
|
|
weechat_printf (NULL,
|
|
_("%s%s: cannot allocate new channel"),
|
|
weechat_prefix ("error"), IRC_PLUGIN_NAME);
|
|
return NULL;
|
|
}
|
|
|
|
/* create buffer for channel (or use existing one) */
|
|
buffer_created = 0;
|
|
buffer_name = irc_buffer_build_name (server->name, channel_name);
|
|
new_buffer = weechat_buffer_search (IRC_PLUGIN_NAME, buffer_name);
|
|
if (new_buffer)
|
|
weechat_nicklist_remove_all (new_buffer);
|
|
else
|
|
{
|
|
new_buffer = weechat_buffer_new (buffer_name,
|
|
&irc_input_data_cb, NULL,
|
|
&irc_buffer_close_cb, NULL);
|
|
if (!new_buffer)
|
|
{
|
|
free (new_channel);
|
|
return NULL;
|
|
}
|
|
if (((channel_type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
&& weechat_config_boolean (irc_config_look_open_channel_near_server))
|
|
|| ((channel_type == IRC_CHANNEL_TYPE_PRIVATE)
|
|
&& weechat_config_boolean (irc_config_look_open_pv_near_server)))
|
|
{
|
|
irc_channel_move_near_server (server, channel_type, new_buffer);
|
|
}
|
|
buffer_created = 1;
|
|
}
|
|
|
|
if (buffer_created)
|
|
{
|
|
weechat_buffer_set (new_buffer, "short_name", channel_name);
|
|
weechat_buffer_set (new_buffer, "localvar_set_type",
|
|
(channel_type == IRC_CHANNEL_TYPE_CHANNEL) ? "channel" : "private");
|
|
weechat_buffer_set (new_buffer, "localvar_set_nick", server->nick);
|
|
weechat_buffer_set (new_buffer, "localvar_set_server", server->name);
|
|
weechat_buffer_set (new_buffer, "localvar_set_channel", channel_name);
|
|
if (server->is_away && server->away_message)
|
|
weechat_buffer_set (new_buffer, "localvar_set_away", server->away_message);
|
|
weechat_hook_signal_send ("logger_backlog",
|
|
WEECHAT_HOOK_SIGNAL_POINTER, new_buffer);
|
|
if (weechat_config_boolean (irc_config_network_send_unknown_commands))
|
|
weechat_buffer_set (new_buffer, "input_get_unknown_commands", "1");
|
|
if (channel_type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
{
|
|
weechat_buffer_set (new_buffer, "nicklist", "1");
|
|
weechat_buffer_set (new_buffer, "nicklist_display_groups", "0");
|
|
}
|
|
|
|
/* set highlights settings on channel buffer */
|
|
weechat_buffer_set(new_buffer, "highlight_words_add", "$nick");
|
|
if (weechat_config_string (irc_config_look_highlight_tags)
|
|
&& weechat_config_string (irc_config_look_highlight_tags)[0])
|
|
{
|
|
weechat_buffer_set (new_buffer, "highlight_tags",
|
|
weechat_config_string (irc_config_look_highlight_tags));
|
|
}
|
|
}
|
|
|
|
if (channel_type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
{
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_CHANOWNER,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_CHANADMIN,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_CHANADMIN2,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_OP,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_HALFOP,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_VOICE,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_CHANUSER,
|
|
"weechat.color.nicklist_group", 1);
|
|
weechat_nicklist_add_group (new_buffer, NULL, IRC_NICK_GROUP_NORMAL,
|
|
"weechat.color.nicklist_group", 1);
|
|
}
|
|
|
|
/* initialize new channel */
|
|
new_channel->type = channel_type;
|
|
new_channel->name = strdup (channel_name);
|
|
new_channel->topic = NULL;
|
|
new_channel->modes = NULL;
|
|
new_channel->limit = 0;
|
|
new_channel->key = NULL;
|
|
new_channel->checking_away = 0;
|
|
new_channel->away_message = NULL;
|
|
new_channel->has_quit_server = 0;
|
|
new_channel->cycle = 0;
|
|
new_channel->display_creation_date = 0;
|
|
new_channel->nick_completion_reset = 0;
|
|
new_channel->pv_remote_nick_color = NULL;
|
|
new_channel->hook_autorejoin = NULL;
|
|
new_channel->nicks_count = 0;
|
|
new_channel->nicks = NULL;
|
|
new_channel->last_nick = NULL;
|
|
new_channel->nicks_speaking[0] = NULL;
|
|
new_channel->nicks_speaking[1] = NULL;
|
|
new_channel->nicks_speaking_time = NULL;
|
|
new_channel->last_nick_speaking_time = NULL;
|
|
new_channel->buffer = new_buffer;
|
|
new_channel->buffer_as_string = NULL;
|
|
|
|
/* add new channel to channels list */
|
|
new_channel->prev_channel = server->last_channel;
|
|
new_channel->next_channel = NULL;
|
|
if (server->channels)
|
|
(server->last_channel)->next_channel = new_channel;
|
|
else
|
|
server->channels = new_channel;
|
|
server->last_channel = new_channel;
|
|
|
|
if (switch_to_channel)
|
|
{
|
|
weechat_buffer_set (new_buffer, "display",
|
|
(auto_switch) ? "auto" : "1");
|
|
}
|
|
|
|
weechat_hook_signal_send ((channel_type == IRC_CHANNEL_TYPE_CHANNEL) ?
|
|
"irc_channel_opened" : "irc_pv_opened",
|
|
WEECHAT_HOOK_SIGNAL_POINTER, new_buffer);
|
|
|
|
/* all is ok, return address of new channel */
|
|
return new_channel;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_set_topic: set topic for a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_set_topic (struct t_irc_channel *channel, const char *topic)
|
|
{
|
|
if (channel->topic)
|
|
free (channel->topic);
|
|
|
|
channel->topic = (topic) ? strdup (topic) : NULL;
|
|
weechat_buffer_set (channel->buffer, "title",
|
|
(channel->topic) ? channel->topic : "");
|
|
}
|
|
|
|
/*
|
|
* irc_channel_search: returns pointer on a channel with name
|
|
*/
|
|
|
|
struct t_irc_channel *
|
|
irc_channel_search (struct t_irc_server *server, const char *channel_name)
|
|
{
|
|
struct t_irc_channel *ptr_channel;
|
|
|
|
if (!server || !channel_name)
|
|
return NULL;
|
|
|
|
for (ptr_channel = server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if (weechat_strcasecmp (ptr_channel->name, channel_name) == 0)
|
|
return ptr_channel;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_is_channel: returns 1 if string is channel
|
|
*/
|
|
|
|
int
|
|
irc_channel_is_channel (const char *string)
|
|
{
|
|
char first_char[2];
|
|
|
|
if (!string)
|
|
return 0;
|
|
|
|
first_char[0] = string[0];
|
|
first_char[1] = '\0';
|
|
return (strpbrk (first_char, IRC_CHANNEL_PREFIX)) ? 1 : 0;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_remove_away: remove away for all nicks on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_remove_away (struct t_irc_server *server,
|
|
struct t_irc_channel *channel)
|
|
{
|
|
struct t_irc_nick *ptr_nick;
|
|
|
|
if (channel->type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
{
|
|
for (ptr_nick = channel->nicks; ptr_nick; ptr_nick = ptr_nick->next_nick)
|
|
{
|
|
irc_nick_set (server, channel, ptr_nick, 0, IRC_NICK_AWAY);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_check_away: check for away on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_check_away (struct t_irc_server *server,
|
|
struct t_irc_channel *channel, int force)
|
|
{
|
|
if (channel->type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
{
|
|
if (force
|
|
|| (weechat_config_integer (irc_config_network_away_check_max_nicks) == 0)
|
|
|| (channel->nicks_count <= weechat_config_integer (irc_config_network_away_check_max_nicks)))
|
|
{
|
|
channel->checking_away++;
|
|
irc_server_sendf (server, IRC_SERVER_OUTQUEUE_PRIO_LOW,
|
|
"WHO %s", channel->name);
|
|
}
|
|
else
|
|
irc_channel_remove_away (server, channel);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_set_away: set/unset away status for a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_set_away (struct t_irc_server *server,
|
|
struct t_irc_channel *channel, const char *nick_name,
|
|
int is_away)
|
|
{
|
|
struct t_irc_nick *ptr_nick;
|
|
|
|
if (channel->type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
{
|
|
ptr_nick = irc_nick_search (channel, nick_name);
|
|
if (ptr_nick)
|
|
irc_nick_set_away (server, channel, ptr_nick, is_away);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_add_to_list: add a nick speaking on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_add_to_list (struct t_irc_channel *channel,
|
|
const char *nick_name,
|
|
int highlight)
|
|
{
|
|
int size, to_remove, i;
|
|
struct t_weelist_item *ptr_item;
|
|
|
|
/* create list if it does not exist */
|
|
if (!channel->nicks_speaking[highlight])
|
|
channel->nicks_speaking[highlight] = weechat_list_new ();
|
|
|
|
/* remove item if it was already in list */
|
|
ptr_item = weechat_list_casesearch (channel->nicks_speaking[highlight],
|
|
nick_name);
|
|
if (ptr_item)
|
|
weechat_list_remove (channel->nicks_speaking[highlight], ptr_item);
|
|
|
|
/* add nick in list */
|
|
weechat_list_add (channel->nicks_speaking[highlight], nick_name,
|
|
WEECHAT_LIST_POS_END, NULL);
|
|
|
|
/* reduce list size if it's too big */
|
|
size = weechat_list_size (channel->nicks_speaking[highlight]);
|
|
if (size > IRC_CHANNEL_NICKS_SPEAKING_LIMIT)
|
|
{
|
|
to_remove = size - IRC_CHANNEL_NICKS_SPEAKING_LIMIT;
|
|
for (i = 0; i < to_remove; i++)
|
|
{
|
|
weechat_list_remove (channel->nicks_speaking[highlight],
|
|
weechat_list_get (channel->nicks_speaking[highlight], 0));
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_add: add a nick speaking on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_add (struct t_irc_channel *channel,
|
|
const char *nick_name, int highlight)
|
|
{
|
|
if (highlight < 0)
|
|
highlight = 0;
|
|
if (highlight > 1)
|
|
highlight = 1;
|
|
if (highlight)
|
|
irc_channel_nick_speaking_add_to_list (channel, nick_name, 1);
|
|
|
|
irc_channel_nick_speaking_add_to_list (channel, nick_name, 0);
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_rename: rename a nick speaking on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_rename (struct t_irc_channel *channel,
|
|
const char *old_nick,
|
|
const char *new_nick)
|
|
{
|
|
struct t_weelist_item *ptr_item;
|
|
int i;
|
|
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
if (channel->nicks_speaking[i])
|
|
{
|
|
ptr_item = weechat_list_search (channel->nicks_speaking[i], old_nick);
|
|
if (ptr_item)
|
|
weechat_list_set (ptr_item, new_nick);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_search: search a nick speaking time on a
|
|
* channel
|
|
*/
|
|
|
|
struct t_irc_channel_speaking *
|
|
irc_channel_nick_speaking_time_search (struct t_irc_channel *channel,
|
|
const char *nick_name,
|
|
int check_time)
|
|
{
|
|
struct t_irc_channel_speaking *ptr_nick;
|
|
time_t time_limit;
|
|
|
|
time_limit = time (NULL) -
|
|
(weechat_config_integer (irc_config_look_smart_filter_delay) * 60);
|
|
|
|
for (ptr_nick = channel->nicks_speaking_time; ptr_nick;
|
|
ptr_nick = ptr_nick->next_nick)
|
|
{
|
|
if (strcmp (ptr_nick->nick, nick_name) == 0)
|
|
{
|
|
if (check_time && (ptr_nick->time_last_message < time_limit))
|
|
return NULL;
|
|
return ptr_nick;
|
|
}
|
|
}
|
|
|
|
/* nick speaking time not found */
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_free: free a nick speaking on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_time_free (struct t_irc_channel *channel,
|
|
struct t_irc_channel_speaking *nick_speaking)
|
|
{
|
|
/* free data */
|
|
if (nick_speaking->nick)
|
|
free (nick_speaking->nick);
|
|
|
|
/* remove nick from list */
|
|
if (nick_speaking->prev_nick)
|
|
(nick_speaking->prev_nick)->next_nick = nick_speaking->next_nick;
|
|
if (nick_speaking->next_nick)
|
|
(nick_speaking->next_nick)->prev_nick = nick_speaking->prev_nick;
|
|
if (channel->nicks_speaking_time == nick_speaking)
|
|
channel->nicks_speaking_time = nick_speaking->next_nick;
|
|
if (channel->last_nick_speaking_time == nick_speaking)
|
|
channel->last_nick_speaking_time = nick_speaking->prev_nick;
|
|
|
|
free (nick_speaking);
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_free_all: free all nick speaking on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_time_free_all (struct t_irc_channel *channel)
|
|
{
|
|
while (channel->nicks_speaking_time)
|
|
{
|
|
irc_channel_nick_speaking_time_free (channel,
|
|
channel->nicks_speaking_time);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_remove_old: remove old nicks speaking
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_time_remove_old (struct t_irc_channel *channel)
|
|
{
|
|
time_t time_limit;
|
|
|
|
time_limit = time (NULL) -
|
|
(weechat_config_integer (irc_config_look_smart_filter_delay) * 60);
|
|
|
|
while (channel->last_nick_speaking_time)
|
|
{
|
|
if (channel->last_nick_speaking_time->time_last_message >= time_limit)
|
|
break;
|
|
|
|
irc_channel_nick_speaking_time_free (channel,
|
|
channel->last_nick_speaking_time);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_add: add a nick speaking time on a channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_time_add (struct t_irc_channel *channel,
|
|
const char *nick_name,
|
|
time_t time_last_message)
|
|
{
|
|
struct t_irc_channel_speaking *ptr_nick, *new_nick;
|
|
|
|
ptr_nick = irc_channel_nick_speaking_time_search (channel, nick_name, 0);
|
|
if (ptr_nick)
|
|
irc_channel_nick_speaking_time_free (channel, ptr_nick);
|
|
|
|
new_nick = malloc (sizeof (*new_nick));
|
|
if (new_nick)
|
|
{
|
|
new_nick->nick = strdup (nick_name);
|
|
new_nick->time_last_message = time_last_message;
|
|
|
|
/* insert nick at beginning of list */
|
|
new_nick->prev_nick = NULL;
|
|
new_nick->next_nick = channel->nicks_speaking_time;
|
|
if (channel->nicks_speaking_time)
|
|
channel->nicks_speaking_time->prev_nick = new_nick;
|
|
else
|
|
channel->last_nick_speaking_time = new_nick;
|
|
channel->nicks_speaking_time = new_nick;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_nick_speaking_time_rename: rename a nick speaking time on a
|
|
* channel
|
|
*/
|
|
|
|
void
|
|
irc_channel_nick_speaking_time_rename (struct t_irc_channel *channel,
|
|
const char *old_nick,
|
|
const char *new_nick)
|
|
{
|
|
struct t_irc_channel_speaking *ptr_nick;
|
|
|
|
if (channel->nicks_speaking_time)
|
|
{
|
|
ptr_nick = irc_channel_nick_speaking_time_search (channel, old_nick, 0);
|
|
if (ptr_nick)
|
|
{
|
|
free (ptr_nick->nick);
|
|
ptr_nick->nick = strdup (new_nick);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_rejoin: rejoin a channel (for example after kick)
|
|
*/
|
|
|
|
void
|
|
irc_channel_rejoin (struct t_irc_server *server, struct t_irc_channel *channel)
|
|
{
|
|
char join_args[256];
|
|
|
|
snprintf (join_args, sizeof (join_args), "%s%s%s",
|
|
channel->name,
|
|
(channel->key) ? " " : "",
|
|
(channel->key) ? channel->key : "");
|
|
|
|
irc_command_join_server (server, join_args);
|
|
}
|
|
|
|
/*
|
|
* irc_channel_autorejoin_cb: callback for autorejoin on a channel
|
|
*/
|
|
|
|
int
|
|
irc_channel_autorejoin_cb (void *data, int remaining_calls)
|
|
{
|
|
struct t_irc_server *ptr_server, *ptr_server_found;
|
|
struct t_irc_channel *ptr_channel_arg, *ptr_channel;
|
|
|
|
/* make C compiler happy */
|
|
(void) remaining_calls;
|
|
|
|
ptr_channel_arg = (struct t_irc_channel *)data;
|
|
|
|
ptr_server_found = NULL;
|
|
for (ptr_server = irc_servers; ptr_server;
|
|
ptr_server = ptr_server->next_server)
|
|
{
|
|
for (ptr_channel = ptr_server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if (ptr_channel == ptr_channel_arg)
|
|
{
|
|
ptr_server_found = ptr_server;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ptr_server_found && (ptr_channel_arg->hook_autorejoin))
|
|
{
|
|
irc_channel_rejoin (ptr_server_found, ptr_channel_arg);
|
|
ptr_channel_arg->hook_autorejoin = NULL;
|
|
}
|
|
|
|
return WEECHAT_RC_OK;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_display_nick_back_in_pv: display a message in pv buffer if nick
|
|
* is back and if private has flag
|
|
* "has_quit_server"
|
|
*/
|
|
|
|
void
|
|
irc_channel_display_nick_back_in_pv (struct t_irc_server *server,
|
|
struct t_irc_nick *nick,
|
|
const char *nickname)
|
|
{
|
|
struct t_irc_channel *ptr_channel;
|
|
|
|
if (!server || (!nick && !nickname))
|
|
return;
|
|
|
|
for (ptr_channel = server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if ((ptr_channel->type == IRC_CHANNEL_TYPE_PRIVATE)
|
|
&& ptr_channel->has_quit_server
|
|
&& (strcmp (ptr_channel->name, (nick) ? nick->name : nickname) == 0))
|
|
{
|
|
if (weechat_config_boolean (irc_config_look_display_pv_back))
|
|
{
|
|
weechat_printf (ptr_channel->buffer,
|
|
_("%s%s%s %s(%s%s%s)%s is back on server"),
|
|
weechat_prefix ("join"),
|
|
IRC_COLOR_NICK_IN_SERVER_MESSAGE(nick),
|
|
(nick) ? nick->name : nickname,
|
|
IRC_COLOR_CHAT_DELIMITERS,
|
|
IRC_COLOR_CHAT_HOST,
|
|
(nick) ? nick->host : "",
|
|
IRC_COLOR_CHAT_DELIMITERS,
|
|
IRC_COLOR_MESSAGE_JOIN);
|
|
}
|
|
ptr_channel->has_quit_server = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_free: free a channel and remove it from channels list
|
|
*/
|
|
|
|
void
|
|
irc_channel_free (struct t_irc_server *server, struct t_irc_channel *channel)
|
|
{
|
|
struct t_irc_channel *new_channels;
|
|
|
|
if (!server || !channel)
|
|
return;
|
|
|
|
/* remove channel from channels list */
|
|
if (server->last_channel == channel)
|
|
server->last_channel = channel->prev_channel;
|
|
if (channel->prev_channel)
|
|
{
|
|
(channel->prev_channel)->next_channel = channel->next_channel;
|
|
new_channels = server->channels;
|
|
}
|
|
else
|
|
new_channels = channel->next_channel;
|
|
|
|
if (channel->next_channel)
|
|
(channel->next_channel)->prev_channel = channel->prev_channel;
|
|
|
|
/* free data */
|
|
if (channel->name)
|
|
free (channel->name);
|
|
if (channel->topic)
|
|
free (channel->topic);
|
|
if (channel->modes)
|
|
free (channel->modes);
|
|
if (channel->key)
|
|
free (channel->key);
|
|
irc_nick_free_all (server, channel);
|
|
if (channel->away_message)
|
|
free (channel->away_message);
|
|
if (channel->pv_remote_nick_color)
|
|
free (channel->pv_remote_nick_color);
|
|
if (channel->hook_autorejoin)
|
|
weechat_unhook(channel->hook_autorejoin);
|
|
if (channel->nicks_speaking[0])
|
|
weechat_list_free (channel->nicks_speaking[0]);
|
|
if (channel->nicks_speaking[1])
|
|
weechat_list_free (channel->nicks_speaking[1]);
|
|
irc_channel_nick_speaking_time_free_all (channel);
|
|
if (channel->buffer_as_string)
|
|
free (channel->buffer_as_string);
|
|
|
|
free (channel);
|
|
|
|
server->channels = new_channels;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_free_all: free all allocated channels for a server
|
|
*/
|
|
|
|
void
|
|
irc_channel_free_all (struct t_irc_server *server)
|
|
{
|
|
while (server->channels)
|
|
{
|
|
irc_channel_free (server, server->channels);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* irc_channel_add_to_infolist: add a channel in an infolist
|
|
* return 1 if ok, 0 if error
|
|
*/
|
|
|
|
int
|
|
irc_channel_add_to_infolist (struct t_infolist *infolist,
|
|
struct t_irc_channel *channel)
|
|
{
|
|
struct t_infolist_item *ptr_item;
|
|
struct t_weelist_item *ptr_list_item;
|
|
struct t_irc_channel_speaking *ptr_nick;
|
|
char option_name[64];
|
|
int i, index;
|
|
|
|
if (!infolist || !channel)
|
|
return 0;
|
|
|
|
ptr_item = weechat_infolist_new_item (infolist);
|
|
if (!ptr_item)
|
|
return 0;
|
|
|
|
if (!weechat_infolist_new_var_pointer (ptr_item, "buffer", channel->buffer))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "buffer_name",
|
|
(channel->buffer) ?
|
|
weechat_buffer_get_string (channel->buffer, "name") : ""))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "buffer_short_name",
|
|
(channel->buffer) ?
|
|
weechat_buffer_get_string (channel->buffer, "short_name") : ""))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "type", channel->type))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "name", channel->name))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "topic", channel->topic))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "modes", channel->modes))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "limit", channel->limit))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "key", channel->key))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "pv_remote_nick_color", channel->pv_remote_nick_color))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "nicks_count", channel->nicks_count))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "checking_away", channel->checking_away))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_string (ptr_item, "away_message", channel->away_message))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "has_quit_server", channel->has_quit_server))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "cycle", channel->cycle))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "display_creation_date", channel->display_creation_date))
|
|
return 0;
|
|
if (!weechat_infolist_new_var_integer (ptr_item, "nick_completion_reset", channel->nick_completion_reset))
|
|
return 0;
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
if (channel->nicks_speaking[i])
|
|
{
|
|
index = 0;
|
|
for (ptr_list_item = weechat_list_get (channel->nicks_speaking[i], 0);
|
|
ptr_list_item;
|
|
ptr_list_item = weechat_list_next (ptr_list_item))
|
|
{
|
|
snprintf (option_name, sizeof (option_name),
|
|
"nick_speaking%d_%05d", i, index);
|
|
if (!weechat_infolist_new_var_string (ptr_item, option_name,
|
|
weechat_list_string (ptr_list_item)))
|
|
return 0;
|
|
index++;
|
|
}
|
|
}
|
|
}
|
|
if (channel->nicks_speaking_time)
|
|
{
|
|
i = 0;
|
|
for (ptr_nick = channel->last_nick_speaking_time; ptr_nick;
|
|
ptr_nick = ptr_nick->prev_nick)
|
|
{
|
|
snprintf (option_name, sizeof (option_name),
|
|
"nick_speaking_time_nick_%05d", i);
|
|
if (!weechat_infolist_new_var_string (ptr_item, option_name,
|
|
ptr_nick->nick))
|
|
return 0;
|
|
snprintf (option_name, sizeof (option_name),
|
|
"nick_speaking_time_time_%05d", i);
|
|
if (!weechat_infolist_new_var_time (ptr_item, option_name,
|
|
ptr_nick->time_last_message))
|
|
return 0;
|
|
i++;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* irc_channel_print_log: print channel infos in log (usually for crash dump)
|
|
*/
|
|
|
|
void
|
|
irc_channel_print_log (struct t_irc_channel *channel)
|
|
{
|
|
struct t_weelist_item *ptr_item;
|
|
struct t_irc_channel_speaking *ptr_nick_speaking;
|
|
int i, index;
|
|
struct t_irc_nick *ptr_nick;
|
|
|
|
weechat_log_printf ("");
|
|
weechat_log_printf (" => channel %s (addr:0x%lx)]", channel->name, channel);
|
|
weechat_log_printf (" type . . . . . . . . . . : %d", channel->type);
|
|
weechat_log_printf (" topic. . . . . . . . . . : '%s'", channel->topic);
|
|
weechat_log_printf (" modes. . . . . . . . . . : '%s'", channel->modes);
|
|
weechat_log_printf (" limit. . . . . . . . . . : %d", channel->limit);
|
|
weechat_log_printf (" key. . . . . . . . . . . : '%s'", channel->key);
|
|
weechat_log_printf (" checking_away. . . . . . : %d", channel->checking_away);
|
|
weechat_log_printf (" away_message . . . . . . : '%s'", channel->away_message);
|
|
weechat_log_printf (" has_quit_server. . . . . : %d", channel->has_quit_server);
|
|
weechat_log_printf (" cycle. . . . . . . . . . : %d", channel->cycle);
|
|
weechat_log_printf (" display_creation_date. . : %d", channel->display_creation_date);
|
|
weechat_log_printf (" nick_completion_reset. . : %d", channel->nick_completion_reset);
|
|
weechat_log_printf (" pv_remote_nick_color . . : '%s'", channel->pv_remote_nick_color);
|
|
weechat_log_printf (" hook_autorejoin. . . . . : 0x%lx", channel->hook_autorejoin);
|
|
weechat_log_printf (" nicks_count. . . . . . . : %d", channel->nicks_count);
|
|
weechat_log_printf (" nicks. . . . . . . . . . : 0x%lx", channel->nicks);
|
|
weechat_log_printf (" last_nick. . . . . . . . : 0x%lx", channel->last_nick);
|
|
weechat_log_printf (" nicks_speaking[0]. . . . : 0x%lx", channel->nicks_speaking[0]);
|
|
weechat_log_printf (" nicks_speaking[1]. . . . : 0x%lx", channel->nicks_speaking[1]);
|
|
weechat_log_printf (" nicks_speaking_time. . . : 0x%lx", channel->nicks_speaking_time);
|
|
weechat_log_printf (" last_nick_speaking_time. : 0x%lx", channel->last_nick_speaking_time);
|
|
weechat_log_printf (" buffer . . . . . . . . . : 0x%lx", channel->buffer);
|
|
weechat_log_printf (" buffer_as_string . . . . : '%s'", channel->buffer_as_string);
|
|
weechat_log_printf (" prev_channel . . . . . . : 0x%lx", channel->prev_channel);
|
|
weechat_log_printf (" next_channel . . . . . . : 0x%lx", channel->next_channel);
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
if (channel->nicks_speaking[i])
|
|
{
|
|
weechat_log_printf ("");
|
|
index = 0;
|
|
for (ptr_item = weechat_list_get (channel->nicks_speaking[i], 0);
|
|
ptr_item; ptr_item = weechat_list_next (ptr_item))
|
|
{
|
|
weechat_log_printf (" nick speaking[%d][%d]: '%s'",
|
|
i, index, weechat_list_string (ptr_item));
|
|
index++;
|
|
}
|
|
}
|
|
}
|
|
if (channel->nicks_speaking_time)
|
|
{
|
|
weechat_log_printf ("");
|
|
for (ptr_nick_speaking = channel->nicks_speaking_time;
|
|
ptr_nick_speaking;
|
|
ptr_nick_speaking = ptr_nick_speaking->next_nick)
|
|
{
|
|
weechat_log_printf (" nick speaking time: '%s', time: %ld",
|
|
ptr_nick_speaking->nick,
|
|
ptr_nick_speaking->time_last_message);
|
|
}
|
|
}
|
|
for (ptr_nick = channel->nicks; ptr_nick; ptr_nick = ptr_nick->next_nick)
|
|
{
|
|
irc_nick_print_log (ptr_nick);
|
|
}
|
|
}
|