diff options
Diffstat (limited to 'OpenSim/Region')
-rw-r--r-- | OpenSim/Region/CoreModules/Avatar/Gods/GodsModule.cs | 150 | ||||
-rw-r--r-- | OpenSim/Region/Framework/Interfaces/IGodsModule.cs | 58 | ||||
-rw-r--r-- | OpenSim/Region/Framework/Scenes/Scene.cs | 127 | ||||
-rw-r--r-- | OpenSim/Region/Framework/Scenes/SceneGraph.cs | 2 |
4 files changed, 218 insertions, 119 deletions
diff --git a/OpenSim/Region/CoreModules/Avatar/Gods/GodsModule.cs b/OpenSim/Region/CoreModules/Avatar/Gods/GodsModule.cs new file mode 100644 index 0000000..dfd6c06 --- /dev/null +++ b/OpenSim/Region/CoreModules/Avatar/Gods/GodsModule.cs | |||
@@ -0,0 +1,150 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSim Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System.Collections.Generic; | ||
29 | using Nini.Config; | ||
30 | using OpenMetaverse; | ||
31 | using OpenSim.Framework; | ||
32 | using OpenSim.Region.Framework.Scenes; | ||
33 | using OpenSim.Region.Framework.Interfaces; | ||
34 | |||
35 | namespace OpenSim.Region.CoreModules.Avatar.Gods | ||
36 | { | ||
37 | public class GodsModules : IRegionModule, IGodsModule | ||
38 | { | ||
39 | protected Scene m_scene; | ||
40 | protected IDialogModule m_dialogModule; | ||
41 | |||
42 | public void Initialise(Scene scene, IConfigSource source) | ||
43 | { | ||
44 | m_scene = scene; | ||
45 | m_dialogModule = m_scene.RequestModuleInterface<IDialogModule>(); | ||
46 | m_scene.RegisterModuleInterface<IGodsModule>(this); | ||
47 | } | ||
48 | |||
49 | public void PostInitialise() {} | ||
50 | public void Close() {} | ||
51 | public string Name { get { return "Gods Module"; } } | ||
52 | public bool IsSharedModule { get { return false; } } | ||
53 | |||
54 | public void RequestGodlikePowers( | ||
55 | UUID agentID, UUID sessionID, UUID token, bool godLike, IClientAPI controllingClient) | ||
56 | { | ||
57 | ScenePresence sp = m_scene.GetScenePresence(agentID); | ||
58 | |||
59 | if (sp != null) | ||
60 | { | ||
61 | if (godLike == false) | ||
62 | { | ||
63 | sp.GrantGodlikePowers(agentID, sessionID, token, godLike); | ||
64 | return; | ||
65 | } | ||
66 | |||
67 | // First check that this is the sim owner | ||
68 | if (m_scene.Permissions.IsGod(agentID)) | ||
69 | { | ||
70 | // Next we check for spoofing..... | ||
71 | UUID testSessionID = sp.ControllingClient.SessionId; | ||
72 | if (sessionID == testSessionID) | ||
73 | { | ||
74 | if (sessionID == controllingClient.SessionId) | ||
75 | { | ||
76 | //m_log.Info("godlike: " + godLike.ToString()); | ||
77 | sp.GrantGodlikePowers(agentID, testSessionID, token, godLike); | ||
78 | } | ||
79 | } | ||
80 | } | ||
81 | else | ||
82 | { | ||
83 | m_dialogModule.SendAlertToUser(agentID, "Request for god powers denied"); | ||
84 | } | ||
85 | } | ||
86 | } | ||
87 | |||
88 | /// <summary> | ||
89 | /// Kicks User specified from the simulator. This logs them off of the grid | ||
90 | /// If the client gets the UUID: 44e87126e7944ded05b37c42da3d5cdb it assumes | ||
91 | /// that you're kicking it even if the avatar's UUID isn't the UUID that the | ||
92 | /// agent is assigned | ||
93 | /// </summary> | ||
94 | /// <param name="godID">The person doing the kicking</param> | ||
95 | /// <param name="sessionID">The session of the person doing the kicking</param> | ||
96 | /// <param name="agentID">the person that is being kicked</param> | ||
97 | /// <param name="kickflags">This isn't used apparently</param> | ||
98 | /// <param name="reason">The message to send to the user after it's been turned into a field</param> | ||
99 | public void KickUser(UUID godID, UUID sessionID, UUID agentID, uint kickflags, byte[] reason) | ||
100 | { | ||
101 | // For some reason the client sends this seemingly hard coded UUID for kicking everyone. Dun-know. | ||
102 | UUID kickUserID = new UUID("44e87126e7944ded05b37c42da3d5cdb"); | ||
103 | |||
104 | ScenePresence sp = m_scene.GetScenePresence(agentID); | ||
105 | |||
106 | if (sp != null || agentID == kickUserID) | ||
107 | { | ||
108 | if (m_scene.Permissions.IsGod(godID)) | ||
109 | { | ||
110 | if (agentID == kickUserID) | ||
111 | { | ||
112 | m_scene.ClientManager.ForEachClient( | ||
113 | delegate(IClientAPI controller) | ||
114 | { | ||
115 | if (controller.AgentId != godID) | ||
116 | controller.Kick(Utils.BytesToString(reason)); | ||
117 | } | ||
118 | ); | ||
119 | |||
120 | // This is a bit crude. It seems the client will be null before it actually stops the thread | ||
121 | // The thread will kill itself eventually :/ | ||
122 | // Is there another way to make sure *all* clients get this 'inter region' message? | ||
123 | m_scene.ForEachScenePresence( | ||
124 | delegate(ScenePresence p) | ||
125 | { | ||
126 | if (p.UUID != godID && !p.IsChildAgent) | ||
127 | { | ||
128 | // Possibly this should really be p.Close() though that method doesn't send a close | ||
129 | // to the client | ||
130 | p.ControllingClient.Close(true); | ||
131 | } | ||
132 | } | ||
133 | ); | ||
134 | } | ||
135 | else | ||
136 | { | ||
137 | m_scene.SceneGraph.removeUserCount(!sp.IsChildAgent); | ||
138 | |||
139 | sp.ControllingClient.Kick(Utils.BytesToString(reason)); | ||
140 | sp.ControllingClient.Close(true); | ||
141 | } | ||
142 | } | ||
143 | else | ||
144 | { | ||
145 | m_dialogModule.SendAlertToUser(godID, "Kick request denied"); | ||
146 | } | ||
147 | } | ||
148 | } | ||
149 | } | ||
150 | } \ No newline at end of file | ||
diff --git a/OpenSim/Region/Framework/Interfaces/IGodsModule.cs b/OpenSim/Region/Framework/Interfaces/IGodsModule.cs new file mode 100644 index 0000000..8e94750 --- /dev/null +++ b/OpenSim/Region/Framework/Interfaces/IGodsModule.cs | |||
@@ -0,0 +1,58 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSim Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using OpenMetaverse; | ||
29 | using OpenSim.Framework; | ||
30 | |||
31 | namespace OpenSim.Region.Framework.Interfaces | ||
32 | { | ||
33 | /// <summary> | ||
34 | /// This interface provides god related methods | ||
35 | /// </summary> | ||
36 | public interface IGodsModule | ||
37 | { | ||
38 | /// <summary> | ||
39 | /// Handle a request for admin rights | ||
40 | /// </summary> | ||
41 | /// <param name="agentID"></param> | ||
42 | /// <param name="sessionID"></param> | ||
43 | /// <param name="token"></param> | ||
44 | /// <param name="godLike"></param> | ||
45 | /// <param name="controllingClient"></param> | ||
46 | void RequestGodlikePowers(UUID agentID, UUID sessionID, UUID token, bool godLike, IClientAPI controllingClient); | ||
47 | |||
48 | /// <summary> | ||
49 | /// Kicks User specified from the simulator. This logs them off of the grid. | ||
50 | /// </summary> | ||
51 | /// <param name="godID">The person doing the kicking</param> | ||
52 | /// <param name="sessionID">The session of the person doing the kicking</param> | ||
53 | /// <param name="agentID">the person that is being kicked</param> | ||
54 | /// <param name="kickflags">This isn't used apparently</param> | ||
55 | /// <param name="reason">The message to send to the user after it's been turned into a field</param> | ||
56 | void KickUser(UUID godID, UUID sessionID, UUID agentID, uint kickflags, byte[] reason); | ||
57 | } | ||
58 | } | ||
diff --git a/OpenSim/Region/Framework/Scenes/Scene.cs b/OpenSim/Region/Framework/Scenes/Scene.cs index 2687422..c9e13ac 100644 --- a/OpenSim/Region/Framework/Scenes/Scene.cs +++ b/OpenSim/Region/Framework/Scenes/Scene.cs | |||
@@ -1931,9 +1931,7 @@ namespace OpenSim.Region.Framework.Scenes | |||
1931 | client.OnObjectDuplicate += m_sceneGraph.DuplicateObject; | 1931 | client.OnObjectDuplicate += m_sceneGraph.DuplicateObject; |
1932 | client.OnObjectDuplicateOnRay += doObjectDuplicateOnRay; | 1932 | client.OnObjectDuplicateOnRay += doObjectDuplicateOnRay; |
1933 | client.OnUpdatePrimFlags += m_sceneGraph.UpdatePrimFlags; | 1933 | client.OnUpdatePrimFlags += m_sceneGraph.UpdatePrimFlags; |
1934 | client.OnRequestObjectPropertiesFamily += m_sceneGraph.RequestObjectPropertiesFamily; | 1934 | client.OnRequestObjectPropertiesFamily += m_sceneGraph.RequestObjectPropertiesFamily; |
1935 | client.OnRequestGodlikePowers += handleRequestGodlikePowers; | ||
1936 | client.OnGodKickUser += HandleGodlikeKickUser; | ||
1937 | client.OnObjectPermissions += HandleObjectPermissionsUpdate; | 1935 | client.OnObjectPermissions += HandleObjectPermissionsUpdate; |
1938 | client.OnCreateNewInventoryItem += CreateNewInventoryItem; | 1936 | client.OnCreateNewInventoryItem += CreateNewInventoryItem; |
1939 | client.OnCreateNewInventoryFolder += HandleCreateInventoryFolder; | 1937 | client.OnCreateNewInventoryFolder += HandleCreateInventoryFolder; |
@@ -1970,9 +1968,12 @@ namespace OpenSim.Region.Framework.Scenes | |||
1970 | client.OnSetScriptRunning += SetScriptRunning; | 1968 | client.OnSetScriptRunning += SetScriptRunning; |
1971 | client.OnRegionHandleRequest += RegionHandleRequest; | 1969 | client.OnRegionHandleRequest += RegionHandleRequest; |
1972 | client.OnUnackedTerrain += TerrainUnAcked; | 1970 | client.OnUnackedTerrain += TerrainUnAcked; |
1973 | |||
1974 | client.OnObjectOwner += ObjectOwner; | 1971 | client.OnObjectOwner += ObjectOwner; |
1975 | 1972 | ||
1973 | IGodsModule godsModule = RequestModuleInterface<IGodsModule>(); | ||
1974 | client.OnGodKickUser += godsModule.KickUser; | ||
1975 | client.OnRequestGodlikePowers += godsModule.RequestGodlikePowers; | ||
1976 | |||
1976 | if (StatsReporter != null) | 1977 | if (StatsReporter != null) |
1977 | client.OnNetworkStatsUpdate += StatsReporter.AddPacketsFromClientStats; | 1978 | client.OnNetworkStatsUpdate += StatsReporter.AddPacketsFromClientStats; |
1978 | 1979 | ||
@@ -2658,7 +2659,7 @@ namespace OpenSim.Region.Framework.Scenes | |||
2658 | /// <param name="maxY"></param> | 2659 | /// <param name="maxY"></param> |
2659 | public void RequestMapBlocks(IClientAPI remoteClient, int minX, int minY, int maxX, int maxY) | 2660 | public void RequestMapBlocks(IClientAPI remoteClient, int minX, int minY, int maxX, int maxY) |
2660 | { | 2661 | { |
2661 | m_log.InfoFormat("[MAPBLOCK]: {0}-{1}, {2}-{3}", minX, minY, maxX, maxY); | 2662 | m_log.DebugFormat("[MAPBLOCK]: {0}-{1}, {2}-{3}", minX, minY, maxX, maxY); |
2662 | m_sceneGridService.RequestMapBlocks(remoteClient, minX, minY, maxX, maxY); | 2663 | m_sceneGridService.RequestMapBlocks(remoteClient, minX, minY, maxX, maxY); |
2663 | } | 2664 | } |
2664 | 2665 | ||
@@ -2817,116 +2818,6 @@ namespace OpenSim.Region.Framework.Scenes | |||
2817 | 2818 | ||
2818 | #endregion | 2819 | #endregion |
2819 | 2820 | ||
2820 | #region Alert Methods | ||
2821 | |||
2822 | /// <summary> | ||
2823 | /// Handle a request for admin rights | ||
2824 | /// </summary> | ||
2825 | /// <param name="agentID"></param> | ||
2826 | /// <param name="sessionID"></param> | ||
2827 | /// <param name="token"></param> | ||
2828 | /// <param name="controllingClient"></param> | ||
2829 | public void handleRequestGodlikePowers(UUID agentID, UUID sessionID, UUID token, bool godLike, | ||
2830 | IClientAPI controllingClient) | ||
2831 | { | ||
2832 | ScenePresence sp = null; | ||
2833 | |||
2834 | lock (m_scenePresences) | ||
2835 | { | ||
2836 | // User needs to be logged into this sim | ||
2837 | m_scenePresences.TryGetValue(agentID, out sp); | ||
2838 | } | ||
2839 | |||
2840 | if (sp != null) | ||
2841 | { | ||
2842 | if (godLike == false) | ||
2843 | { | ||
2844 | sp.GrantGodlikePowers(agentID, sessionID, token, godLike); | ||
2845 | return; | ||
2846 | } | ||
2847 | |||
2848 | // First check that this is the sim owner | ||
2849 | if (Permissions.IsGod(agentID)) | ||
2850 | { | ||
2851 | // Next we check for spoofing..... | ||
2852 | UUID testSessionID = sp.ControllingClient.SessionId; | ||
2853 | if (sessionID == testSessionID) | ||
2854 | { | ||
2855 | if (sessionID == controllingClient.SessionId) | ||
2856 | { | ||
2857 | //m_log.Info("godlike: " + godLike.ToString()); | ||
2858 | sp.GrantGodlikePowers(agentID, testSessionID, token, godLike); | ||
2859 | } | ||
2860 | } | ||
2861 | } | ||
2862 | else | ||
2863 | { | ||
2864 | m_dialogModule.SendAlertToUser(agentID, "Request for god powers denied"); | ||
2865 | } | ||
2866 | } | ||
2867 | } | ||
2868 | |||
2869 | /// <summary> | ||
2870 | /// Kicks User specified from the simulator. This logs them off of the grid | ||
2871 | /// If the client gets the UUID: 44e87126e7944ded05b37c42da3d5cdb it assumes | ||
2872 | /// that you're kicking it even if the avatar's UUID isn't the UUID that the | ||
2873 | /// agent is assigned | ||
2874 | /// </summary> | ||
2875 | /// <param name="godID">The person doing the kicking</param> | ||
2876 | /// <param name="sessionID">The session of the person doing the kicking</param> | ||
2877 | /// <param name="agentID">the person that is being kicked</param> | ||
2878 | /// <param name="kickflags">This isn't used apparently</param> | ||
2879 | /// <param name="reason">The message to send to the user after it's been turned into a field</param> | ||
2880 | public void HandleGodlikeKickUser(UUID godID, UUID sessionID, UUID agentID, uint kickflags, byte[] reason) | ||
2881 | { | ||
2882 | // For some reason the client sends this seemingly hard coded UUID for kicking everyone. Dun-know. | ||
2883 | UUID kickUserID = new UUID("44e87126e7944ded05b37c42da3d5cdb"); | ||
2884 | lock (m_scenePresences) | ||
2885 | { | ||
2886 | if (m_scenePresences.ContainsKey(agentID) || agentID == kickUserID) | ||
2887 | { | ||
2888 | if (Permissions.IsGod(godID)) | ||
2889 | { | ||
2890 | if (agentID == kickUserID) | ||
2891 | { | ||
2892 | ClientManager.ForEachClient(delegate(IClientAPI controller) | ||
2893 | { | ||
2894 | if (controller.AgentId != godID) | ||
2895 | controller.Kick(Utils.BytesToString(reason)); | ||
2896 | } | ||
2897 | ); | ||
2898 | |||
2899 | // This is a bit crude. It seems the client will be null before it actually stops the thread | ||
2900 | // The thread will kill itself eventually :/ | ||
2901 | // Is there another way to make sure *all* clients get this 'inter region' message? | ||
2902 | ClientManager.ForEachClient(delegate(IClientAPI controller) | ||
2903 | { | ||
2904 | ScenePresence p = GetScenePresence(controller.AgentId); | ||
2905 | bool childagent = p != null && p.IsChildAgent; | ||
2906 | if (controller.AgentId != godID && !childagent) | ||
2907 | // Do we really want to kick the initiator of this madness? | ||
2908 | { | ||
2909 | controller.Close(true); | ||
2910 | } | ||
2911 | } | ||
2912 | ); | ||
2913 | } | ||
2914 | else | ||
2915 | { | ||
2916 | m_sceneGraph.removeUserCount(!m_scenePresences[agentID].IsChildAgent); | ||
2917 | |||
2918 | m_scenePresences[agentID].ControllingClient.Kick(Utils.BytesToString(reason)); | ||
2919 | m_scenePresences[agentID].ControllingClient.Close(true); | ||
2920 | } | ||
2921 | } | ||
2922 | else | ||
2923 | { | ||
2924 | m_dialogModule.SendAlertToUser(godID, "Kick request denied"); | ||
2925 | } | ||
2926 | } | ||
2927 | } | ||
2928 | } | ||
2929 | |||
2930 | public void HandleObjectPermissionsUpdate(IClientAPI controller, UUID agentID, UUID sessionID, byte field, uint localId, uint mask, byte set) | 2821 | public void HandleObjectPermissionsUpdate(IClientAPI controller, UUID agentID, UUID sessionID, byte field, uint localId, uint mask, byte set) |
2931 | { | 2822 | { |
2932 | // Check for spoofing.. since this is permissions we're talking about here! | 2823 | // Check for spoofing.. since this is permissions we're talking about here! |
@@ -2944,8 +2835,6 @@ namespace OpenSim.Region.Framework.Scenes | |||
2944 | } | 2835 | } |
2945 | } | 2836 | } |
2946 | 2837 | ||
2947 | #endregion | ||
2948 | |||
2949 | /// <summary> | 2838 | /// <summary> |
2950 | /// Causes all clients to get a full object update on all of the objects in the scene. | 2839 | /// Causes all clients to get a full object update on all of the objects in the scene. |
2951 | /// </summary> | 2840 | /// </summary> |
@@ -3077,7 +2966,9 @@ namespace OpenSim.Region.Framework.Scenes | |||
3077 | } | 2966 | } |
3078 | else if ((parcel.landData.Flags & (uint)Parcel.ParcelFlags.AllowGroupScripts) != 0) | 2967 | else if ((parcel.landData.Flags & (uint)Parcel.ParcelFlags.AllowGroupScripts) != 0) |
3079 | { | 2968 | { |
3080 | if (part.OwnerID == parcel.landData.OwnerID || (parcel.landData.IsGroupOwned && part.GroupID == parcel.landData.GroupID) || Permissions.IsGod(part.OwnerID)) | 2969 | if (part.OwnerID == parcel.landData.OwnerID |
2970 | || (parcel.landData.IsGroupOwned && part.GroupID == parcel.landData.GroupID) | ||
2971 | || Permissions.IsGod(part.OwnerID)) | ||
3081 | { | 2972 | { |
3082 | return true; | 2973 | return true; |
3083 | } | 2974 | } |
diff --git a/OpenSim/Region/Framework/Scenes/SceneGraph.cs b/OpenSim/Region/Framework/Scenes/SceneGraph.cs index 69f5205..d6aca31 100644 --- a/OpenSim/Region/Framework/Scenes/SceneGraph.cs +++ b/OpenSim/Region/Framework/Scenes/SceneGraph.cs | |||
@@ -696,7 +696,7 @@ namespace OpenSim.Region.Framework.Scenes | |||
696 | } | 696 | } |
697 | } | 697 | } |
698 | 698 | ||
699 | protected internal void removeUserCount(bool TypeRCTF) | 699 | public void removeUserCount(bool TypeRCTF) |
700 | { | 700 | { |
701 | if (TypeRCTF) | 701 | if (TypeRCTF) |
702 | { | 702 | { |