mirror of
https://github.com/weechat/weechat.git
synced 2026-06-25 20:36:38 +02:00
cf6aca1619
This pointer is the first argument received by callbacks, and the existing argument "data" is now automatically freed by WeeChat when the object containing the callback is removed. With this new pointer, the linked list of callbacks in scripts has been removed. This will improve speed of scripts (using a lot of hooks), reduce memory used by scripts and reduce time to unload scripts. Following functions are affected in the C API: * exec_on_files * config_new * config_new_section * config_new_option * hook_command * hook_command_run * hook_timer * hook_fd * hook_process * hook_process_hashtable * hook_connect * hook_print * hook_signal * hook_hsignal * hook_config * hook_completion * hook_modifier * hook_info * hook_info_hashtable * hook_infolist * hook_hdata * hook_focus * unhook_all_plugin * buffer_new * bar_item_new * upgrade_new * upgrade_read
279 lines
7.2 KiB
C
279 lines
7.2 KiB
C
/*
|
|
* irc-buffer.c - buffer functions for IRC plugin
|
|
*
|
|
* Copyright (C) 2003-2016 Sébastien Helleu <flashcode@flashtux.org>
|
|
*
|
|
* 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <limits.h>
|
|
|
|
#include "../weechat-plugin.h"
|
|
#include "irc.h"
|
|
#include "irc-buffer.h"
|
|
#include "irc-channel.h"
|
|
#include "irc-command.h"
|
|
#include "irc-config.h"
|
|
#include "irc-raw.h"
|
|
#include "irc-server.h"
|
|
|
|
|
|
/*
|
|
* Gets IRC server and channel pointers with a buffer pointer (buffer may be a
|
|
* server or a channel).
|
|
*/
|
|
|
|
void
|
|
irc_buffer_get_server_and_channel (struct t_gui_buffer *buffer,
|
|
struct t_irc_server **server,
|
|
struct t_irc_channel **channel)
|
|
{
|
|
struct t_irc_server *ptr_server;
|
|
struct t_irc_channel *ptr_channel;
|
|
|
|
if (server)
|
|
*server = NULL;
|
|
if (channel)
|
|
*channel = NULL;
|
|
|
|
if (!buffer)
|
|
return;
|
|
|
|
/* look for a server or channel using this buffer */
|
|
for (ptr_server = irc_servers; ptr_server;
|
|
ptr_server = ptr_server->next_server)
|
|
{
|
|
if (ptr_server->buffer == buffer)
|
|
{
|
|
if (server)
|
|
*server = ptr_server;
|
|
return;
|
|
}
|
|
|
|
for (ptr_channel = ptr_server->channels; ptr_channel;
|
|
ptr_channel = ptr_channel->next_channel)
|
|
{
|
|
if (ptr_channel->buffer == buffer)
|
|
{
|
|
if (server)
|
|
*server = ptr_server;
|
|
if (channel)
|
|
*channel = ptr_channel;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* no server or channel found */
|
|
}
|
|
|
|
/*
|
|
* Builds buffer name with a server and a channel.
|
|
*/
|
|
|
|
const char *
|
|
irc_buffer_build_name (const char *server, const char *channel)
|
|
{
|
|
static char buffer[128];
|
|
|
|
buffer[0] = '\0';
|
|
|
|
if (!server && !channel)
|
|
return buffer;
|
|
|
|
if (server && channel)
|
|
snprintf (buffer, sizeof (buffer), "%s.%s", server, channel);
|
|
else if (server)
|
|
snprintf (buffer, sizeof (buffer), "server.%s", server);
|
|
else
|
|
snprintf (buffer, sizeof (buffer), "%s", channel);
|
|
|
|
return buffer;
|
|
}
|
|
|
|
/*
|
|
* Callback called when a buffer is closed.
|
|
*/
|
|
|
|
int
|
|
irc_buffer_close_cb (const void *pointer, void *data,
|
|
struct t_gui_buffer *buffer)
|
|
{
|
|
struct t_irc_channel *next_channel;
|
|
|
|
IRC_BUFFER_GET_SERVER_CHANNEL(buffer);
|
|
|
|
/* make C compiler happy */
|
|
(void) pointer;
|
|
(void) data;
|
|
|
|
if (buffer == irc_raw_buffer)
|
|
{
|
|
irc_raw_buffer = NULL;
|
|
}
|
|
else
|
|
{
|
|
if (ptr_channel)
|
|
{
|
|
/* send PART for channel if its buffer is closed */
|
|
if ((ptr_channel->type == IRC_CHANNEL_TYPE_CHANNEL)
|
|
&& (ptr_channel->nicks))
|
|
{
|
|
irc_command_part_channel (ptr_server, ptr_channel->name, NULL);
|
|
}
|
|
irc_channel_free (ptr_server, ptr_channel);
|
|
}
|
|
else
|
|
{
|
|
if (ptr_server)
|
|
{
|
|
if (!ptr_server->disconnected)
|
|
{
|
|
/* send QUIT to server, then disconnect */
|
|
irc_command_quit_server (ptr_server, NULL);
|
|
irc_server_disconnect (ptr_server, 0, 0);
|
|
}
|
|
ptr_channel = ptr_server->channels;
|
|
while (ptr_channel)
|
|
{
|
|
next_channel = ptr_channel->next_channel;
|
|
if (ptr_channel->buffer != buffer)
|
|
weechat_buffer_close (ptr_channel->buffer);
|
|
ptr_channel = next_channel;
|
|
}
|
|
ptr_server->buffer = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
return WEECHAT_RC_OK;
|
|
}
|
|
|
|
/*
|
|
* Callback for comparing two nicks in nicklist (called when searching a nick in
|
|
* nicklist).
|
|
* The "casemapping" of server is used in comparison.
|
|
*
|
|
* Returns:
|
|
* -1: nick1 < nick2
|
|
* 0: nick1 == nick2
|
|
* 1: nick2 > nick2
|
|
*/
|
|
|
|
int
|
|
irc_buffer_nickcmp_cb (const void *pointer, void *data,
|
|
struct t_gui_buffer *buffer,
|
|
const char *nick1,
|
|
const char *nick2)
|
|
{
|
|
struct t_irc_server *server;
|
|
|
|
/* make C compiler happy */
|
|
(void) data;
|
|
|
|
if (pointer)
|
|
server = (struct t_irc_server *)pointer;
|
|
else
|
|
irc_buffer_get_server_and_channel (buffer, &server, NULL);
|
|
|
|
if (server)
|
|
{
|
|
return irc_server_strcasecmp (server, nick1, nick2);
|
|
}
|
|
else
|
|
{
|
|
/* default is RFC 1459 casemapping comparison */
|
|
return weechat_strcasecmp_range (nick1, nick2, 29);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Searches for the server buffer with the lowest number.
|
|
*
|
|
* Returns pointer to buffer found, NULL if not found.
|
|
*/
|
|
|
|
struct t_gui_buffer *
|
|
irc_buffer_search_server_lowest_number ()
|
|
{
|
|
struct t_gui_buffer *ptr_buffer;
|
|
struct t_irc_server *ptr_server;
|
|
int number, number_found;
|
|
|
|
ptr_buffer = NULL;
|
|
number_found = INT_MAX;
|
|
|
|
for (ptr_server = irc_servers; ptr_server;
|
|
ptr_server = ptr_server->next_server)
|
|
{
|
|
if (ptr_server->buffer)
|
|
{
|
|
number = weechat_buffer_get_integer (ptr_server->buffer, "number");
|
|
if (number < number_found)
|
|
{
|
|
number_found = number;
|
|
ptr_buffer = ptr_server->buffer;
|
|
}
|
|
}
|
|
}
|
|
return ptr_buffer;
|
|
}
|
|
|
|
/*
|
|
* Searches for the private buffer with the lowest number.
|
|
* If server is not NULL, searches only for this server.
|
|
*
|
|
* Returns pointer to buffer found, NULL if not found.
|
|
*/
|
|
|
|
struct t_gui_buffer *
|
|
irc_buffer_search_private_lowest_number (struct t_irc_server *server)
|
|
{
|
|
struct t_gui_buffer *ptr_buffer;
|
|
struct t_irc_server *ptr_server;
|
|
struct t_irc_channel *ptr_channel;
|
|
int number, number_found;
|
|
|
|
ptr_buffer = NULL;
|
|
number_found = INT_MAX;
|
|
|
|
for (ptr_server = (server) ? 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->type == IRC_CHANNEL_TYPE_PRIVATE)
|
|
&& ptr_channel->buffer)
|
|
{
|
|
number = weechat_buffer_get_integer (ptr_channel->buffer,
|
|
"number");
|
|
if (number < number_found)
|
|
{
|
|
number_found = number;
|
|
ptr_buffer = ptr_channel->buffer;
|
|
}
|
|
}
|
|
}
|
|
if (server)
|
|
break;
|
|
}
|
|
return ptr_buffer;
|
|
}
|