summaryrefslogtreecommitdiff
path: root/src/core/nickmatch-cache.c
blob: 5eb4be6cb166b8462f20bd3ad377fd4cd7c4e9e1 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
/*
 nickmatch-cache.c : irssi

    Copyright (C) 2001 Timo Sirainen

    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 2 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, write to the Free Software
    51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
*/

#include "module.h"
#include "signals.h"

#include "channels.h"
#include "nicklist.h"

#include "nickmatch-cache.h"

static GSList *lists;

NICKMATCH_REC *nickmatch_init(NICKMATCH_REBUILD_FUNC func)
{
	NICKMATCH_REC *rec;

	rec = g_new0(NICKMATCH_REC, 1);
	rec->func = func;

	lists = g_slist_append(lists, rec);
        return rec;
}

void nickmatch_deinit(NICKMATCH_REC *rec)
{
	lists = g_slist_remove(lists, rec);

        g_hash_table_destroy(rec->nicks);
        g_free(rec);
}

static void nickmatch_check_channel(CHANNEL_REC *channel, NICKMATCH_REC *rec)
{
	GSList *nicks, *tmp;

	nicks = nicklist_getnicks(channel);
	for (tmp = nicks; tmp != NULL; tmp = tmp->next) {
		NICK_REC *nick = tmp->data;

		rec->func(rec->nicks, channel, nick);
	}
        g_slist_free(nicks);
}

void nickmatch_rebuild(NICKMATCH_REC *rec)
{
	if (rec->nicks != NULL)
		g_hash_table_destroy(rec->nicks);

	rec->nicks = g_hash_table_new((GHashFunc) g_direct_hash,
				      (GCompareFunc) g_direct_equal);

	g_slist_foreach(channels, (GFunc) nickmatch_check_channel, rec);
}

static void sig_nick_new(CHANNEL_REC *channel, NICK_REC *nick)
{
	GSList *tmp;

        g_return_if_fail(channel != NULL);
	g_return_if_fail(nick != NULL);

	for (tmp = lists; tmp != NULL; tmp = tmp->next) {
		NICKMATCH_REC *rec = tmp->data;

		rec->func(rec->nicks, channel, nick);
	}
}

static void sig_nick_remove(CHANNEL_REC *channel, NICK_REC *nick)
{
	GSList *tmp;

        g_return_if_fail(channel != NULL);
	g_return_if_fail(nick != NULL);

	for (tmp = lists; tmp != NULL; tmp = tmp->next) {
		NICKMATCH_REC *rec = tmp->data;

                g_hash_table_remove(rec->nicks, nick);
	}
}

void nickmatch_cache_init(void)
{
	lists = NULL;
        signal_add("nicklist new", (SIGNAL_FUNC) sig_nick_new);
        signal_add("nicklist changed", (SIGNAL_FUNC) sig_nick_new);
        signal_add("nicklist host changed", (SIGNAL_FUNC) sig_nick_new);
        signal_add("nicklist remove", (SIGNAL_FUNC) sig_nick_remove);
}

void nickmatch_cache_deinit(void)
{
	g_slist_foreach(lists, (GFunc) nickmatch_deinit, NULL);
        g_slist_free(lists);

	signal_remove("nicklist new", (SIGNAL_FUNC) sig_nick_new);
        signal_remove("nicklist changed", (SIGNAL_FUNC) sig_nick_new);
        signal_remove("nicklist host changed", (SIGNAL_FUNC) sig_nick_new);
        signal_remove("nicklist remove", (SIGNAL_FUNC) sig_nick_remove);
}