]> git.netwichtig.de Git - user/henk/code/inspircd.git/blob - src/modules/m_namedmodes.cpp
4bf8960e15058321147394a4375a2520f2ccd880
[user/henk/code/inspircd.git] / src / modules / m_namedmodes.cpp
1 /*       +------------------------------------+
2  *       | Inspire Internet Relay Chat Daemon |
3  *       +------------------------------------+
4  *
5  *  InspIRCd: (C) 2002-2009 InspIRCd Development Team
6  * See: http://wiki.inspircd.org/Credits
7  *
8  * This program is free but copyrighted software; see
9  *            the file COPYING for details.
10  *
11  * ---------------------------------------------------
12  */
13
14 #include "inspircd.h"
15
16 class ModuleNamedModes : public Module
17 {
18  public:
19         ModuleNamedModes()
20         {
21                 Implementation eventlist[] = { I_OnPreMode, I_On005Numeric };
22                 ServerInstance->Modules->Attach(eventlist, this, 2);
23         }
24
25         Version GetVersion()
26         {
27                 return Version("Provides the ability to manipulate modes via long names.",VF_VENDOR);
28         }
29
30         void On005Numeric(std::string& line)
31         {
32                 std::string::size_type pos = line.find(" CHANMODES=");
33                 if (pos != std::string::npos)
34                 {
35                         pos += 11;
36                         while (line[pos] > 'A' && line[pos] < 'Z')
37                                 pos++;
38                         line.insert(pos, 1, 'Z');
39                 }
40         }
41
42         void DisplayList(User* user, Channel* channel)
43         {
44                 for(char letter = 'A'; letter <= 'z'; letter++)
45                 {
46                         ModeHandler* mh = ServerInstance->Modes->FindMode(letter, MODETYPE_CHANNEL);
47                         if (!mh || mh->IsListMode())
48                                 continue;
49                         if (!channel->IsModeSet(letter))
50                                 continue;
51                         std::string item = mh->name;
52                         if (mh->GetNumParams(true))
53                                 item += "=" + channel->GetModeParameter(letter);
54                         user->WriteNumeric(961, "%s %s %s", user->nick.c_str(), channel->name.c_str(), item.c_str());
55                 }
56                 user->WriteNumeric(960, "%s %s :End of mode list", user->nick.c_str(), channel->name.c_str());
57         }
58
59         ModResult OnPreMode(User* source, User* dest, Channel* channel, const std::vector<std::string>& parameters)
60         {
61                 if (!channel)
62                         return MOD_RES_PASSTHRU;
63                 if (parameters[1].find('Z') == std::string::npos)
64                         return MOD_RES_PASSTHRU;
65                 if (parameters.size() <= 2)
66                 {
67                         DisplayList(source, channel);
68                         return MOD_RES_DENY;
69                 }
70
71                 std::vector<std::string> newparms;
72                 newparms.push_back(parameters[0]);
73                 newparms.push_back(parameters[1]);
74
75                 std::string modelist = newparms[1];
76                 bool adding = true;
77                 unsigned int param_at = 2;
78                 for(unsigned int i = 0; i < modelist.length(); i++)
79                 {
80                         unsigned char modechar = modelist[i];
81                         if (modechar == '+' || modechar == '-')
82                         {
83                                 adding = (modechar == '+');
84                                 continue;
85                         }
86                         ModeHandler *mh = ServerInstance->Modes->FindMode(modechar, MODETYPE_CHANNEL);
87                         if (modechar == 'Z')
88                         {
89                                 modechar = 0;
90                                 std::string name, value;
91                                 if (param_at < parameters.size())
92                                         name = parameters[param_at++];
93                                 std::string::size_type eq = name.find('=');
94                                 if (eq != std::string::npos)
95                                 {
96                                         value = name.substr(eq + 1);
97                                         name = name.substr(0, eq);
98                                 }
99                                 for(char letter = 'A'; modechar == 0 && letter <= 'z'; letter++)
100                                 {
101                                         mh = ServerInstance->Modes->FindMode(letter, MODETYPE_CHANNEL);
102                                         if (mh && mh->name == name)
103                                         {
104                                                 if (mh->GetNumParams(adding))
105                                                 {
106                                                         if (!value.empty())
107                                                         {
108                                                                 newparms.push_back(value);
109                                                                 modechar = letter;
110                                                                 break;
111                                                         }
112                                                 }
113                                                 else
114                                                 {
115                                                         modechar = letter;
116                                                         break;
117                                                 }
118                                         }
119                                 }
120                                 if (modechar)
121                                         modelist[i] = modechar;
122                                 else
123                                         modelist.erase(i, 1);
124                         }
125                         else if (mh && mh->GetNumParams(adding) && param_at < parameters.size())
126                         {
127                                 newparms.push_back(parameters[param_at++]);
128                         }
129                 }
130                 newparms[1] = modelist;
131                 ServerInstance->Modes->Process(newparms, source, false);
132                 return MOD_RES_DENY;
133         }
134 };
135
136 MODULE_INIT(ModuleNamedModes)