summaryrefslogtreecommitdiff
path: root/src/modules
diff options
context:
space:
mode:
Diffstat (limited to 'src/modules')
-rw-r--r--src/modules/m_auditorium.cpp108
1 files changed, 59 insertions, 49 deletions
diff --git a/src/modules/m_auditorium.cpp b/src/modules/m_auditorium.cpp
index 4dec57df9..0e5c92310 100644
--- a/src/modules/m_auditorium.cpp
+++ b/src/modules/m_auditorium.cpp
@@ -25,15 +25,10 @@ class AuditoriumMode : public ModeHandler
ModeAction OnModeChange(User* source, User* dest, Channel* channel, std::string &parameter, bool adding)
{
- if (channel->IsModeSet('u') != adding)
- {
- channel->SetMode('u', adding);
- return MODEACTION_ALLOW;
- }
- else
- {
+ if (channel->IsModeSet(this) == adding)
return MODEACTION_DENY;
- }
+ channel->SetMode(this, adding);
+ return MODEACTION_ALLOW;
}
};
@@ -41,20 +36,22 @@ class ModuleAuditorium : public Module
{
private:
AuditoriumMode aum;
- bool ShowOps;
- bool OperOverride;
+ bool OpsVisible;
+ bool OpsCanSee;
+ bool OperCanSee;
public:
- ModuleAuditorium()
- : aum(this)
+ ModuleAuditorium() : aum(this)
{
- if (!ServerInstance->Modes->AddMode(&aum))
- throw ModuleException("Could not add new modes!");
+ }
+
+ void init()
+ {
+ ServerInstance->Modules->AddService(aum);
OnRehash(NULL);
Implementation eventlist[] = { I_OnUserJoin, I_OnUserPart, I_OnUserKick, I_OnBuildNeighborList, I_OnNamesListItem, I_OnRehash };
ServerInstance->Modules->Attach(eventlist, this, 6);
-
}
~ModuleAuditorium()
@@ -63,9 +60,10 @@ class ModuleAuditorium : public Module
void OnRehash(User* user)
{
- ConfigReader conf;
- ShowOps = conf.ReadFlag("auditorium", "showops", 0);
- OperOverride = conf.ReadFlag("auditorium", "operoverride", 0);
+ ConfigTag* tag = ServerInstance->Config->ConfValue("auditorium");
+ OpsVisible = tag->getBool("opvisible");
+ OpsCanSee = tag->getBool("opcansee");
+ OperCanSee = tag->getBool("opercansee", true);
}
Version GetVersion()
@@ -73,55 +71,64 @@ class ModuleAuditorium : public Module
return Version("Allows for auditorium channels (+u) where nobody can see others joining and parting or the nick list", VF_VENDOR);
}
- void OnNamesListItem(User* issuer, Membership* memb, std::string &prefixes, std::string &nick)
+ /* Can they be seen by everyone? */
+ bool IsVisible(Membership* memb)
{
- if (!memb->chan->IsModeSet('u'))
- return;
+ if (!memb->chan->IsModeSet(&aum))
+ return true;
+
+ ModResult res = ServerInstance->OnCheckExemption(memb->user, memb->chan, "auditorium-vis");
+ return res.check(OpsVisible && memb->getRank() >= OP_VALUE);
+ }
+
+ /* Can they see this specific membership? */
+ bool CanSee(User* issuer, Membership* memb)
+ {
+ // If user is oper and operoverride is on, don't touch the list
+ if (OperCanSee && issuer->HasPrivPermission("channels/auspex"))
+ return true;
+
+ // You can always see yourself
+ if (issuer == memb->user)
+ return true;
+
+ // Can you see the list by permission?
+ ModResult res = ServerInstance->OnCheckExemption(issuer,memb->chan,"auditorium-see");
+ if (res.check(OpsCanSee && memb->chan->GetPrefixValue(issuer) >= OP_VALUE))
+ return true;
- /* Some module hid this from being displayed, dont bother */
+ return false;
+ }
+
+ void OnNamesListItem(User* issuer, Membership* memb, std::string &prefixes, std::string &nick)
+ {
+ // Some module already hid this from being displayed, don't bother
if (nick.empty())
return;
- /* If user is oper and operoverride is on, don't touch the list */
- if (OperOverride && issuer->HasPrivPermission("channels/auspex"))
+ if (IsVisible(memb))
return;
- if (ShowOps && (issuer != memb->user) && (memb->getRank() < OP_VALUE))
- {
- /* Showops is set, hide all non-ops from the user, except themselves */
- nick.clear();
+ if (CanSee(issuer, memb))
return;
- }
- if (!ShowOps && (issuer != memb->user))
- {
- /* ShowOps is not set, hide everyone except the user whos requesting NAMES */
- nick.clear();
- return;
- }
+ nick.clear();
}
+ /** Build CUList for showing this join/part/kick */
void BuildExcept(Membership* memb, CUList& excepts)
{
- if (!memb->chan->IsModeSet('u'))
- return;
- if (ShowOps && memb->getRank() >= OP_VALUE)
+ if (IsVisible(memb))
return;
const UserMembList* users = memb->chan->GetUsers();
for(UserMembCIter i = users->begin(); i != users->end(); i++)
{
- if (i->first == memb->user || !IS_LOCAL(i->first))
- continue;
- if (ShowOps && i->second->getRank() >= OP_VALUE)
- continue;
- if (OperOverride && i->first->HasPrivPermission("channels/auspex"))
- continue;
- // This is a different user in the channel, local, and not op/oper
- // so, hide the join from them
- excepts.insert(i->first);
+ if (IS_LOCAL(i->first) && !CanSee(i->first, memb))
+ excepts.insert(i->first);
}
}
+
void OnUserJoin(Membership* memb, bool sync, bool created, CUList& excepts)
{
BuildExcept(memb, excepts);
@@ -143,8 +150,11 @@ class ModuleAuditorium : public Module
while (i != include.end())
{
Channel* c = *i++;
- if (c->IsModeSet('u'))
- include.erase(c);
+ Membership* memb = c->GetUser(source); // will be non-null
+ if (IsVisible(memb))
+ continue;
+ // TODO this doesn't take can-see into account
+ include.erase(c);
}
}
};