2 * roleplay commands for charybdis.
4 * adds NPC, NPCA, and SCENE which allow users to send messages from 'fake'
5 * nicknames. in the case of NPC and NPCA, the nickname will be underlined
6 * to clearly show that it is fake. SCENE is a special case and not underlined.
7 * these commands only work on channels set +N
9 * also adds oper commands FSAY and FACTION, which are like NPC and NPCA
10 * except without the underline.
12 * all of these messages have the hostmask npc.fakeuser.invalid, and their ident
13 * is the nickname of the user running the commands.
25 #include "inline/stringops.h"
31 static int m_scene(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
32 static int m_fsay(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
33 static int m_faction(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
34 static int m_npc(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
35 static int m_npca(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
36 static int m_displaymsg(struct Client
*source_p
, const char *channel
, int underline
, int action
, const char *nick
, const char *text
);
37 static int me_roleplay(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[]);
38 static unsigned int mymode
;
43 /* initalize the +N cmode */
44 mymode
= cflag_add('N', chm_simple
);
54 /* orphan the +N cmode on modunload */
59 struct Message scene_msgtab
= {
60 "SCENE", 0, 0, 0, MFLG_SLOW
,
61 {mg_unreg
, {m_scene
, 3}, mg_ignore
, mg_ignore
, mg_ignore
, {m_scene
, 3}}
64 /* this serves as an alias for people who are used to inspircd/unreal m_roleplay */
65 struct Message ambiance_msgtab
= {
66 "AMBIANCE", 0, 0, 0, MFLG_SLOW
,
67 {mg_unreg
, {m_scene
, 3}, mg_ignore
, mg_ignore
, mg_ignore
, {m_scene
, 3}}
70 struct Message fsay_msgtab
= {
71 "FSAY", 0, 0, 0, MFLG_SLOW
,
72 {mg_unreg
, {m_npc
, 4}, mg_ignore
, mg_ignore
, mg_ignore
, {m_fsay
, 4}}
75 struct Message faction_msgtab
= {
76 "FACTION", 0, 0, 0, MFLG_SLOW
,
77 {mg_unreg
, {m_npca
, 4}, mg_ignore
, mg_ignore
, mg_ignore
, {m_faction
, 4}}
80 struct Message npc_msgtab
= {
81 "NPC", 0, 0, 0, MFLG_SLOW
,
82 {mg_unreg
, {m_npc
, 4}, mg_ignore
, mg_ignore
, mg_ignore
, {m_npc
, 4}}
85 struct Message npca_msgtab
= {
86 "NPCA", 0, 0, 0, MFLG_SLOW
,
87 {mg_unreg
, {m_npca
, 4}, mg_ignore
, mg_ignore
, mg_ignore
, {m_npca
, 4}}
90 struct Message roleplay_msgtab
= {
91 "ROLEPLAY", 0, 0, 0, MFLG_SLOW
,
92 {mg_ignore
, mg_ignore
, mg_ignore
, mg_ignore
, {me_roleplay
, 4}, mg_ignore
}
95 mapi_clist_av1 roleplay_clist
[] = { &scene_msgtab
, &ambiance_msgtab
, &fsay_msgtab
, &faction_msgtab
, &npc_msgtab
, &npca_msgtab
, &roleplay_msgtab
, NULL
};
97 DECLARE_MODULE_AV1(roleplay
, _modinit
, _moddeinit
, roleplay_clist
, NULL
, NULL
, "$m_roleplay$");
100 m_scene(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
102 m_displaymsg(source_p
, parv
[1], 0, 0, "=Scene=", parv
[2]);
107 m_fsay(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
109 m_displaymsg(source_p
, parv
[1], 0, 0, parv
[2], parv
[3]);
114 m_faction(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
116 m_displaymsg(source_p
, parv
[1], 0, 1, parv
[2], parv
[3]);
121 m_npc(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
123 m_displaymsg(source_p
, parv
[1], 1, 0, parv
[2], parv
[3]);
128 m_npca(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
130 m_displaymsg(source_p
, parv
[1], 1, 1, parv
[2], parv
[3]);
135 m_displaymsg(struct Client
*source_p
, const char *channel
, int underline
, int action
, const char *nick
, const char *text
)
137 struct Channel
*chptr
;
138 struct membership
*msptr
;
139 char nick2
[NICKLEN
+1];
140 char *nick3
= rb_strdup(nick
);
143 if(!IsFloodDone(source_p
))
144 flood_endgrace(source_p
);
146 if((chptr
= find_channel(channel
)) == NULL
)
148 sendto_one(source_p
, form_str(ERR_NOSUCHCHANNEL
), channel
);
152 if(!(msptr
= find_channel_membership(chptr
, source_p
)))
154 sendto_one_numeric(source_p
, ERR_NOTONCHANNEL
,
155 form_str(ERR_NOTONCHANNEL
), chptr
->chname
);
159 if(!(chptr
->mode
.mode
& chmode_flags
['N']))
161 sendto_one_numeric(source_p
, 573, "%s :Roleplay commands are not enabled on this channel.", chptr
->chname
);
165 if(!can_send(chptr
, source_p
, msptr
))
167 sendto_one_numeric(source_p
, 573, "%s :Cannot send to channel.", chptr
->chname
);
171 /* enforce flood stuff on roleplay commands */
172 if(flood_attack_channel(0, source_p
, chptr
, chptr
->chname
))
175 /* enforce target change on roleplay commands */
176 if(!is_chanop_voiced(msptr
) && !IsOper(source_p
) && !add_channel_target(source_p
, chptr
))
178 sendto_one(source_p
, form_str(ERR_TARGCHANGE
),
179 me
.name
, source_p
->name
, chptr
->chname
);
184 rb_snprintf(nick2
, sizeof(nick2
), "\x1F%s\x1F", strip_unprintable(nick3
));
186 rb_snprintf(nick2
, sizeof(nick2
), "%s", strip_unprintable(nick3
));
188 /* don't allow nicks to be empty after stripping
189 * this prevents nastiness like fake factions, etc. */
190 if(EmptyString(nick3
))
192 sendto_one_numeric(source_p
, 573, "%s :No visible non-stripped characters in nick.", chptr
->chname
);
197 rb_snprintf(text2
, sizeof(text2
), "\1ACTION %s\1", text
);
199 rb_snprintf(text2
, sizeof(text2
), "%s", text
);
201 sendto_channel_local(ALL_MEMBERS
, chptr
, ":%s!%s@npc.fakeuser.invalid PRIVMSG %s :%s", nick2
, source_p
->name
, channel
, text2
);
202 sendto_match_servs(source_p
, "*", CAP_ENCAP
, NOCAPS
, "ENCAP * ROLEPLAY %s %s :%s",
203 channel
, nick2
, text2
);
208 me_roleplay(struct Client
*client_p
, struct Client
*source_p
, int parc
, const char *parv
[])
210 struct Channel
*chptr
;
212 /* Don't segfault if we get ROLEPLAY with an invalid channel.
213 * This shouldn't happen but it's best to be on the safe side. */
214 if((chptr
= find_channel(parv
[1])) == NULL
)
217 sendto_channel_local(ALL_MEMBERS
, chptr
, ":%s!%s@npc.fakeuser.invalid PRIVMSG %s :%s", parv
[2], source_p
->name
, parv
[1], parv
[3]);