diff --git a/.gitmodules b/.gitmodules
index 9ff1a297..1fa69c2d 100644
--- a/.gitmodules
+++ b/.gitmodules
@@ -1,3 +1,3 @@
[submodule "discord.net"]
path = discord.net
- url = git://github.com/rogueexception/discord.net.git
+ url = git://github.com/kwoth/discord.net.git
diff --git a/NadekoBot.sln b/NadekoBot.sln
index 155512c4..eb9860d0 100644
--- a/NadekoBot.sln
+++ b/NadekoBot.sln
@@ -22,51 +22,95 @@ Global
EndGlobalSection
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
+ Debug|x64 = Debug|x64
FullDebug|Any CPU = FullDebug|Any CPU
+ FullDebug|x64 = FullDebug|x64
NadekoRelease|Any CPU = NadekoRelease|Any CPU
+ NadekoRelease|x64 = NadekoRelease|x64
Release|Any CPU = Release|Any CPU
+ Release|x64 = Release|x64
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Debug|x64.ActiveCfg = Debug|x64
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Debug|x64.Build.0 = Debug|x64
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.FullDebug|Any CPU.ActiveCfg = Debug|Any CPU
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.FullDebug|Any CPU.Build.0 = Debug|Any CPU
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.FullDebug|x64.ActiveCfg = Debug|x64
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.FullDebug|x64.Build.0 = Debug|x64
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.NadekoRelease|Any CPU.ActiveCfg = NadekoRelease|Any CPU
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.NadekoRelease|Any CPU.Build.0 = NadekoRelease|Any CPU
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.NadekoRelease|x64.ActiveCfg = NadekoRelease|x64
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.NadekoRelease|x64.Build.0 = NadekoRelease|x64
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Release|Any CPU.ActiveCfg = Release|Any CPU
{27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Release|Any CPU.Build.0 = Release|Any CPU
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Release|x64.ActiveCfg = Release|x64
+ {27A886F5-CDDA-4F4A-81EE-6DAFCCE9DE46}.Release|x64.Build.0 = Release|x64
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Debug|x64.ActiveCfg = Debug|x64
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Debug|x64.Build.0 = Debug|x64
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.FullDebug|Any CPU.ActiveCfg = Debug|Any CPU
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.FullDebug|Any CPU.Build.0 = Debug|Any CPU
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.FullDebug|x64.ActiveCfg = Debug|x64
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.FullDebug|x64.Build.0 = Debug|x64
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.NadekoRelease|Any CPU.ActiveCfg = Release|Any CPU
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.NadekoRelease|Any CPU.Build.0 = Release|Any CPU
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.NadekoRelease|x64.ActiveCfg = Release|x64
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.NadekoRelease|x64.Build.0 = Release|x64
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Release|Any CPU.ActiveCfg = Release|Any CPU
{7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Release|Any CPU.Build.0 = Release|Any CPU
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Release|x64.ActiveCfg = Release|x64
+ {7BFEF748-B934-4621-9B11-6302E3A9F6B3}.Release|x64.Build.0 = Release|x64
{8D71A857-879A-4A10-859E-5FF824ED6688}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{8D71A857-879A-4A10-859E-5FF824ED6688}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.Debug|x64.ActiveCfg = Debug|x64
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.Debug|x64.Build.0 = Debug|x64
{8D71A857-879A-4A10-859E-5FF824ED6688}.FullDebug|Any CPU.ActiveCfg = Debug|Any CPU
{8D71A857-879A-4A10-859E-5FF824ED6688}.FullDebug|Any CPU.Build.0 = Debug|Any CPU
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.FullDebug|x64.ActiveCfg = Debug|x64
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.FullDebug|x64.Build.0 = Debug|x64
{8D71A857-879A-4A10-859E-5FF824ED6688}.NadekoRelease|Any CPU.ActiveCfg = Release|Any CPU
{8D71A857-879A-4A10-859E-5FF824ED6688}.NadekoRelease|Any CPU.Build.0 = Release|Any CPU
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.NadekoRelease|x64.ActiveCfg = Release|x64
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.NadekoRelease|x64.Build.0 = Release|x64
{8D71A857-879A-4A10-859E-5FF824ED6688}.Release|Any CPU.ActiveCfg = Release|Any CPU
{8D71A857-879A-4A10-859E-5FF824ED6688}.Release|Any CPU.Build.0 = Release|Any CPU
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.Release|x64.ActiveCfg = Release|x64
+ {8D71A857-879A-4A10-859E-5FF824ED6688}.Release|x64.Build.0 = Release|x64
{3091164F-66AE-4543-A63D-167C1116241D}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{3091164F-66AE-4543-A63D-167C1116241D}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {3091164F-66AE-4543-A63D-167C1116241D}.Debug|x64.ActiveCfg = Debug|x64
+ {3091164F-66AE-4543-A63D-167C1116241D}.Debug|x64.Build.0 = Debug|x64
{3091164F-66AE-4543-A63D-167C1116241D}.FullDebug|Any CPU.ActiveCfg = Debug|Any CPU
{3091164F-66AE-4543-A63D-167C1116241D}.FullDebug|Any CPU.Build.0 = Debug|Any CPU
+ {3091164F-66AE-4543-A63D-167C1116241D}.FullDebug|x64.ActiveCfg = Debug|x64
+ {3091164F-66AE-4543-A63D-167C1116241D}.FullDebug|x64.Build.0 = Debug|x64
{3091164F-66AE-4543-A63D-167C1116241D}.NadekoRelease|Any CPU.ActiveCfg = Release|Any CPU
{3091164F-66AE-4543-A63D-167C1116241D}.NadekoRelease|Any CPU.Build.0 = Release|Any CPU
+ {3091164F-66AE-4543-A63D-167C1116241D}.NadekoRelease|x64.ActiveCfg = Release|x64
+ {3091164F-66AE-4543-A63D-167C1116241D}.NadekoRelease|x64.Build.0 = Release|x64
{3091164F-66AE-4543-A63D-167C1116241D}.Release|Any CPU.ActiveCfg = Release|Any CPU
{3091164F-66AE-4543-A63D-167C1116241D}.Release|Any CPU.Build.0 = Release|Any CPU
+ {3091164F-66AE-4543-A63D-167C1116241D}.Release|x64.ActiveCfg = Release|x64
+ {3091164F-66AE-4543-A63D-167C1116241D}.Release|x64.Build.0 = Release|x64
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.Debug|x64.ActiveCfg = Debug|x64
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.Debug|x64.Build.0 = Debug|x64
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.FullDebug|Any CPU.ActiveCfg = Debug|Any CPU
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.FullDebug|Any CPU.Build.0 = Debug|Any CPU
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.FullDebug|x64.ActiveCfg = Debug|x64
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.FullDebug|x64.Build.0 = Debug|x64
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.NadekoRelease|Any CPU.ActiveCfg = Release|Any CPU
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.NadekoRelease|Any CPU.Build.0 = Release|Any CPU
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.NadekoRelease|x64.ActiveCfg = Release|x64
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.NadekoRelease|x64.Build.0 = Release|x64
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.Release|Any CPU.ActiveCfg = Release|Any CPU
{1B5603B4-6F8F-4289-B945-7BAAE523D740}.Release|Any CPU.Build.0 = Release|Any CPU
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.Release|x64.ActiveCfg = Release|x64
+ {1B5603B4-6F8F-4289-B945-7BAAE523D740}.Release|x64.Build.0 = Release|x64
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
diff --git a/NadekoBot/Classes/DBHandler.cs b/NadekoBot/Classes/DBHandler.cs
index 9fb664f9..3f2963cc 100644
--- a/NadekoBot/Classes/DBHandler.cs
+++ b/NadekoBot/Classes/DBHandler.cs
@@ -161,9 +161,7 @@ namespace NadekoBot.Classes
using (var conn = new SQLiteConnection(FilePath))
{
foreach (var o in ocol)
- {
- conn.InsertOrReplace(o, typeof(T));
- }
+ conn.InsertOrReplace(o);
}
}
diff --git a/NadekoBot/Classes/Extensions.cs b/NadekoBot/Classes/Extensions.cs
index 1b162209..a4071ded 100644
--- a/NadekoBot/Classes/Extensions.cs
+++ b/NadekoBot/Classes/Extensions.cs
@@ -46,10 +46,10 @@ namespace NadekoBot.Extensions
if (num == 0)
return string.Empty;
if (num <= 3)
- return string.Join("", str.Select(c => '.'));
+ return string.Concat(str.Select(c => '.'));
if (str.Length < num)
return str;
- return string.Join("", str.Take(num - 3)) + (hideDots ? "" : "...");
+ return string.Concat(str.Take(num - 3)) + (hideDots ? "" : "...");
}
///
/// Removes trailing S or ES (if specified) on the given string if the num is 1
@@ -237,7 +237,7 @@ namespace NadekoBot.Extensions
public static string Matrix(this string s)
=>
- string.Join("", s.Select(c => c.ToString() + " ̵̢̬̜͉̞̭̖̰͋̉̎ͬ̔̇̌̀".TrimTo(rng.Next(0, 12), true)));
+ string.Concat(s.Select(c => c.ToString() + " ̵̢̬̜͉̞̭̖̰͋̉̎ͬ̔̇̌̀".TrimTo(rng.Next(0, 12), true)));
//.Replace("`", "");
public static void ForEach(this IEnumerable source, Action action)
diff --git a/NadekoBot/Classes/SearchHelper.cs b/NadekoBot/Classes/SearchHelper.cs
index 77c46151..43bfbe37 100644
--- a/NadekoBot/Classes/SearchHelper.cs
+++ b/NadekoBot/Classes/SearchHelper.cs
@@ -384,7 +384,7 @@ namespace NadekoBot.Classes
{
var i = 0;
return "```xl\n" + string.Join("\n", items.GroupBy(item => (i++) / cols)
- .Select(ig => string.Join("", ig.Select(el => howToPrint(el)))))
+ .Select(ig => string.Concat(ig.Select(el => howToPrint(el)))))
+ $"\n```";
}
}
diff --git a/NadekoBot/Classes/ServerSpecificConfig.cs b/NadekoBot/Classes/ServerSpecificConfig.cs
index 4296b648..ff42f0fd 100644
--- a/NadekoBot/Classes/ServerSpecificConfig.cs
+++ b/NadekoBot/Classes/ServerSpecificConfig.cs
@@ -100,6 +100,21 @@ namespace NadekoBot.Classes
}
}
+ [JsonIgnore]
+ private ObservableCollection logserverIgnoreChannels;
+ public ObservableCollection LogserverIgnoreChannels {
+ get { return logserverIgnoreChannels; }
+ set {
+ logserverIgnoreChannels = value;
+ if (value != null)
+ logserverIgnoreChannels.CollectionChanged += (s, e) =>
+ {
+ if (!SpecificConfigurations.Instantiated) return;
+ OnPropertyChanged();
+ };
+ }
+ }
+
[JsonProperty("LogPresenceChannel")]
private ulong? logPresenceChannel = null;
[JsonIgnore]
@@ -142,6 +157,8 @@ namespace NadekoBot.Classes
}
}
+
+
[JsonIgnore]
private ulong autoAssignedRole = 0;
public ulong AutoAssignedRole {
@@ -179,6 +196,19 @@ namespace NadekoBot.Classes
}
}
+ [JsonIgnore]
+ private bool exclusiveSelfAssignedRoles = false;
+ public bool ExclusiveSelfAssignedRoles
+ {
+ get { return exclusiveSelfAssignedRoles; }
+ set
+ {
+ exclusiveSelfAssignedRoles = value;
+ if (!SpecificConfigurations.Instantiated) return;
+ OnPropertyChanged();
+ }
+ }
+
[JsonIgnore]
private ObservableCollection observingStreams;
@@ -212,6 +242,7 @@ namespace NadekoBot.Classes
ObservingStreams = new ObservableCollection();
GenerateCurrencyChannels = new ObservableConcurrentDictionary();
VoiceChannelLog = new ObservableConcurrentDictionary();
+ LogserverIgnoreChannels = new ObservableCollection();
}
public event PropertyChangedEventHandler PropertyChanged = delegate { SpecificConfigurations.Default.Save(); };
diff --git a/NadekoBot/Modules/Administration/AdministrationModule.cs b/NadekoBot/Modules/Administration/AdministrationModule.cs
index 9b2e6c4e..ab6dafae 100644
--- a/NadekoBot/Modules/Administration/AdministrationModule.cs
+++ b/NadekoBot/Modules/Administration/AdministrationModule.cs
@@ -633,22 +633,13 @@ namespace NadekoBot.Modules.Administration
.Parameter("num", ParameterType.Optional)
.Do(async e =>
{
+ Message[] msgs;
if (string.IsNullOrWhiteSpace(e.GetArg("user_or_num"))) // if nothing is set, clear nadeko's messages, no permissions required
{
- await Task.Run(async () =>
- {
- var msgs = (await e.Channel.DownloadMessages(100).ConfigureAwait(false)).Where(m => m.User.Id == e.Server.CurrentUser.Id);
- foreach (var m in msgs)
- {
- try
- {
- await m.Delete().ConfigureAwait(false);
- }
- catch { }
- await Task.Delay(100).ConfigureAwait(false);
- }
-
- }).ConfigureAwait(false);
+ msgs = (await e.Channel.DownloadMessages(100).ConfigureAwait(false)).Where(m => m.User.Id == e.Server.CurrentUser.Id).ToArray();
+ if (!msgs.Any())
+ return;
+ await e.Channel.DeleteMessages(msgs).ConfigureAwait(false);
return;
}
if (!e.User.GetPermissions(e.Channel).ManageMessages)
@@ -665,11 +656,7 @@ namespace NadekoBot.Modules.Administration
if (val <= 0)
return;
val++;
- foreach (var msg in await e.Channel.DownloadMessages(val).ConfigureAwait(false))
- {
- await msg.Delete().ConfigureAwait(false);
- await Task.Delay(100).ConfigureAwait(false);
- }
+ await e.Channel.DeleteMessages((await e.Channel.DownloadMessages(val).ConfigureAwait(false)).ToArray()).ConfigureAwait(false);
return;
}
//else if first argument is user
@@ -679,20 +666,10 @@ namespace NadekoBot.Modules.Administration
val = 100;
if (!int.TryParse(e.GetArg("num"), out val))
val = 100;
- await Task.Run(async () =>
- {
- var msgs = (await e.Channel.DownloadMessages(100).ConfigureAwait(false)).Where(m => m.User.Id == usr.Id).Take(val);
- foreach (var m in msgs)
- {
- try
- {
- await m.Delete().ConfigureAwait(false);
- }
- catch { }
- await Task.Delay(100).ConfigureAwait(false);
- }
-
- }).ConfigureAwait(false);
+ msgs = (await e.Channel.DownloadMessages(100).ConfigureAwait(false)).Where(m => m.User.Id == usr.Id).Take(val).ToArray();
+ if (!msgs.Any())
+ return;
+ await e.Channel.DeleteMessages(msgs).ConfigureAwait(false);
});
cgb.CreateCommand(Prefix + "die")
@@ -761,7 +738,7 @@ namespace NadekoBot.Modules.Administration
if (string.IsNullOrWhiteSpace(msg))
return;
- var ids = e.GetArg("ids").Split('-');
+ var ids = e.GetArg("ids").Split('|');
if (ids.Length != 2)
return;
var sid = ulong.Parse(ids[0]);
diff --git a/NadekoBot/Modules/Administration/Commands/LogCommand.cs b/NadekoBot/Modules/Administration/Commands/LogCommand.cs
index 8f51c9e5..3bbce6d2 100644
--- a/NadekoBot/Modules/Administration/Commands/LogCommand.cs
+++ b/NadekoBot/Modules/Administration/Commands/LogCommand.cs
@@ -51,8 +51,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
try
{
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || config.LogserverIgnoreChannels.Contains(e.After.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -72,8 +73,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
try
{
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || config.LogserverIgnoreChannels.Contains(e.Channel.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -87,8 +89,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
try
{
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || config.LogserverIgnoreChannels.Contains(e.Channel.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -164,8 +167,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
if (e.Server == null || e.Channel.IsPrivate || e.User.Id == NadekoBot.Client.CurrentUser.Id)
return;
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null || e.Channel.Id == chId)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || e.Channel.Id == chId || config.LogserverIgnoreChannels.Contains(e.Channel.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -192,8 +196,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
if (e.Server == null || e.Channel.IsPrivate || e.User?.Id == NadekoBot.Client.CurrentUser.Id)
return;
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null || e.Channel.Id == chId)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || e.Channel.Id == chId || config.LogserverIgnoreChannels.Contains(e.Channel.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -219,8 +224,9 @@ namespace NadekoBot.Modules.Administration.Commands
{
if (e.Server == null || e.Channel.IsPrivate || e.User?.Id == NadekoBot.Client.CurrentUser.Id)
return;
- var chId = SpecificConfigurations.Default.Of(e.Server.Id).LogServerChannel;
- if (chId == null || e.Channel.Id == chId)
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ var chId = config.LogServerChannel;
+ if (chId == null || e.Channel.Id == chId || config.LogserverIgnoreChannels.Contains(e.Channel.Id))
return;
Channel ch;
if ((ch = e.Server.TextChannels.Where(tc => tc.Id == chId).FirstOrDefault()) == null)
@@ -370,6 +376,25 @@ $@"🕔`{prettyCurrentTime}` **Message** 📝 `#{e.Channel.Name}`
await e.Channel.SendMessage($"❗**NO LONGER LOGGING IN {ch.Mention} CHANNEL**❗").ConfigureAwait(false);
});
+
+ cgb.CreateCommand(Prefix + "logignore")
+ .Description($"Toggles whether the {Prefix}logserver command ignores this channel. Useful if you have hidden admin channel and public log channel.")
+ .AddCheck(SimpleCheckers.OwnerOnly())
+ .AddCheck(SimpleCheckers.ManageServer())
+ .Do(async e =>
+ {
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ if (config.LogserverIgnoreChannels.Remove(e.Channel.Id))
+ {
+ await e.Channel.SendMessage($"`{Prefix}logserver will stop ignoring this channel.`");
+ }
+ else
+ {
+ config.LogserverIgnoreChannels.Add(e.Channel.Id);
+ await e.Channel.SendMessage($"`{Prefix}logserver will ignore this channel.`");
+ }
+ });
+
cgb.CreateCommand(Module.Prefix + "userpresence")
.Description("Starts logging to this channel when someone from the server goes online/offline/idle.")
.AddCheck(SimpleCheckers.ManageServer())
diff --git a/NadekoBot/Modules/Administration/Commands/SelfAssignedRolesCommand.cs b/NadekoBot/Modules/Administration/Commands/SelfAssignedRolesCommand.cs
index 723b0f06..2ef54772 100644
--- a/NadekoBot/Modules/Administration/Commands/SelfAssignedRolesCommand.cs
+++ b/NadekoBot/Modules/Administration/Commands/SelfAssignedRolesCommand.cs
@@ -1,6 +1,8 @@
using Discord.Commands;
+using Discord.Net;
using NadekoBot.Classes;
using NadekoBot.Modules.Permissions.Classes;
+using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
@@ -74,7 +76,7 @@ namespace NadekoBot.Modules.Administration.Commands
var config = SpecificConfigurations.Default.Of(e.Server.Id);
var msg = new StringBuilder($"There are `{config.ListOfSelfAssignableRoles.Count}` self assignable roles:\n");
var toRemove = new HashSet();
- foreach (var roleId in config.ListOfSelfAssignableRoles)
+ foreach (var roleId in config.ListOfSelfAssignableRoles.OrderBy(r=>r.ToString()))
{
var role = e.Server.GetRole(roleId);
if (role == null)
@@ -94,6 +96,19 @@ namespace NadekoBot.Modules.Administration.Commands
await e.Channel.SendMessage(msg.ToString()).ConfigureAwait(false);
});
+
+
+ cgb.CreateCommand(Module.Prefix + "togglexclsar").Alias(Module.Prefix +"tesar")
+ .Description("toggle whether the self-assigned roles should be exclusive")
+ .AddCheck(SimpleCheckers.CanManageRoles)
+ .Do(async e =>
+ {
+ var config = SpecificConfigurations.Default.Of(e.Server.Id);
+ config.ExclusiveSelfAssignedRoles = !config.ExclusiveSelfAssignedRoles;
+ string exl = config.ExclusiveSelfAssignedRoles ? "exclusive" : "not exclusive";
+ await e.Channel.SendMessage("Self assigned roles are now " + exl);
+ });
+
cgb.CreateCommand(Module.Prefix + "iam")
.Description("Adds a role to you that you choose. " +
"Role must be on a list of self-assignable roles." +
@@ -121,11 +136,20 @@ namespace NadekoBot.Modules.Administration.Commands
await e.Channel.SendMessage($":anger:You already have {role.Name} role.").ConfigureAwait(false);
return;
}
+ var sameRoles = e.User.Roles.Where(r => config.ListOfSelfAssignableRoles.Contains(r.Id));
+ if (config.ExclusiveSelfAssignedRoles && sameRoles.Any())
+ {
+ await e.Channel.SendMessage($":anger:You already have {sameRoles.FirstOrDefault().Name} role.").ConfigureAwait(false);
+ return;
+ }
try
{
await e.User.AddRoles(role).ConfigureAwait(false);
}
- catch
+ catch(HttpException ex) when (ex.StatusCode == System.Net.HttpStatusCode.InternalServerError)
+ {
+ }
+ catch (Exception)
{
await e.Channel.SendMessage($":anger:`I am unable to add that role to you. I can't add roles to owners or other roles higher than my role in the role hierarchy.`").ConfigureAwait(false);
}
diff --git a/NadekoBot/Modules/ClashOfClans/ClashOfClansModule.cs b/NadekoBot/Modules/ClashOfClans/ClashOfClansModule.cs
index 532f3cbb..9c383509 100644
--- a/NadekoBot/Modules/ClashOfClans/ClashOfClansModule.cs
+++ b/NadekoBot/Modules/ClashOfClans/ClashOfClansModule.cs
@@ -27,8 +27,7 @@ namespace NadekoBot.Modules.ClashOfClans
cgb.CreateCommand(Prefix + "createwar")
.Alias(Prefix + "cw")
- .Description(
- $"Creates a new war by specifying a size (>10 and multiple of 5) and enemy clan name. |{Prefix}cw 15 The Enemy Clan")
+ .Description($"Creates a new war by specifying a size (>10 and multiple of 5) and enemy clan name. |{Prefix}cw 15 The Enemy Clan")
.Parameter("size")
.Parameter("enemy_clan", ParameterType.Unparsed)
.Do(async e =>
diff --git a/NadekoBot/Modules/Conversations/Conversations.cs b/NadekoBot/Modules/Conversations/Conversations.cs
index cdc6eb26..328be562 100644
--- a/NadekoBot/Modules/Conversations/Conversations.cs
+++ b/NadekoBot/Modules/Conversations/Conversations.cs
@@ -9,13 +9,14 @@ using System;
using System.Diagnostics;
using System.IO;
using System.Linq;
+using System.Text;
using System.Threading.Tasks;
namespace NadekoBot.Modules.Conversations
{
internal class Conversations : DiscordModule
{
- private const string firestr = "🔥 ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็็็้้้้้็็ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็็็้้้้้็็็็็้้้้ 🔥";
+ private const string firestr = "🔥 ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็ด้้้้้็็็็็้้้้้็็็็็้้้้้้้้็็็็็้้้้้็็็็็้้้้ 🔥";
public Conversations()
{
commands.Add(new RipCommand(this));
@@ -153,22 +154,23 @@ namespace NadekoBot.Modules.Conversations
.Parameter("times", ParameterType.Optional)
.Do(async e =>
{
- var count = 1;
- int.TryParse(e.Args[0], out count);
- if (count == 0)
+ int count;
+ if (string.IsNullOrWhiteSpace(e.Args[0]))
count = 1;
+ else
+ int.TryParse(e.Args[0], out count);
if (count < 1 || count > 12)
{
- await e.Channel.SendMessage("Number must be between 0 and 12").ConfigureAwait(false);
+ await e.Channel.SendMessage("Number must be between 1 and 12").ConfigureAwait(false);
return;
}
- var str = "";
+ var str = new StringBuilder();
for (var i = 0; i < count; i++)
{
- str += firestr;
+ str.Append(firestr);
}
- await e.Channel.SendMessage(str).ConfigureAwait(false);
+ await e.Channel.SendMessage(str.ToString()).ConfigureAwait(false);
});
cgb.CreateCommand("dump")
diff --git a/NadekoBot/Modules/DiscordCommand.cs b/NadekoBot/Modules/DiscordCommand.cs
index 726c813b..05f9a42a 100644
--- a/NadekoBot/Modules/DiscordCommand.cs
+++ b/NadekoBot/Modules/DiscordCommand.cs
@@ -7,7 +7,7 @@ namespace NadekoBot.Classes
/// Base DiscordCommand Class.
/// Inherit this class to create your own command.
///
- internal abstract class DiscordCommand
+ public abstract class DiscordCommand
{
///
diff --git a/NadekoBot/Modules/DiscordModule.cs b/NadekoBot/Modules/DiscordModule.cs
index 48427732..5e131233 100644
--- a/NadekoBot/Modules/DiscordModule.cs
+++ b/NadekoBot/Modules/DiscordModule.cs
@@ -3,7 +3,7 @@ using System.Collections.Generic;
using NadekoBot.Classes;
namespace NadekoBot.Modules {
- internal abstract class DiscordModule : IModule {
+ public abstract class DiscordModule : IModule {
protected readonly HashSet commands = new HashSet();
public abstract string Prefix { get; }
diff --git a/NadekoBot/Modules/Gambling/DiceRollCommand.cs b/NadekoBot/Modules/Gambling/DiceRollCommand.cs
index 5e1a7833..68a665fe 100644
--- a/NadekoBot/Modules/Gambling/DiceRollCommand.cs
+++ b/NadekoBot/Modules/Gambling/DiceRollCommand.cs
@@ -24,6 +24,13 @@ namespace NadekoBot.Modules.Gambling
" If you split 2 numbers with letter d (xdy) it will roll x dice from 1 to y. | $roll or $roll 7 or $roll 3d5")
.Parameter("num", ParameterType.Optional)
.Do(RollFunc());
+
+ cgb.CreateCommand(Module.Prefix + "rolluo")
+ .Description("Rolls 0-100. If you supply a number [x] it rolls up to 30 normal dice (unordered)." +
+ " If you split 2 numbers with letter d (xdy) it will roll x dice from 1 to y. | $roll or $roll 7 or $roll 3d5")
+ .Parameter("num", ParameterType.Optional)
+ .Do(RollFunc(false));
+
cgb.CreateCommand(Module.Prefix + "nroll")
.Description("Rolls in a given range. | `$nroll 5` (rolls 0-5) or `$nroll 5-15`")
.Parameter("range", ParameterType.Required)
@@ -40,7 +47,7 @@ namespace NadekoBot.Modules.Gambling
Regex dndRegex = new Regex(@"(?\d+)d(?\d+)", RegexOptions.Compiled);
- private Func RollFunc()
+ private Func RollFunc(bool ordered = true)
{
var r = new Random();
return async e =>
@@ -73,7 +80,7 @@ namespace NadekoBot.Modules.Gambling
arr[i] = r.Next(1, n2 + 1);
}
var elemCnt = 0;
- await e.Channel.SendMessage($"`Rolled {n1} {(n1 == 1 ? "die" : "dice")} 1-{n2}.`\n`Result:` " + string.Join(", ", arr.OrderBy(x => x).Select(x => elemCnt++ % 2 == 0 ? $"**{x}**" : x.ToString()))).ConfigureAwait(false);
+ await e.Channel.SendMessage($"`Rolled {n1} {(n1 == 1 ? "die" : "dice")} 1-{n2}.`\n`Result:` " + string.Join(", ", (ordered ? arr.OrderBy(x => x).AsEnumerable() : arr).Select(x => elemCnt++ % 2 == 0 ? $"**{x}**" : x.ToString()))).ConfigureAwait(false);
}
return;
}
@@ -92,17 +99,23 @@ namespace NadekoBot.Modules.Gambling
{
var randomNumber = r.Next(1, 7);
var toInsert = dices.Count;
- if (randomNumber == 6 || dices.Count == 0)
- toInsert = 0;
- else if (randomNumber != 1)
- for (var j = 0; j < dices.Count; j++)
- {
- if (values[j] < randomNumber)
+ if (ordered)
+ {
+ if (randomNumber == 6 || dices.Count == 0)
+ toInsert = 0;
+ else if (randomNumber != 1)
+ for (var j = 0; j < dices.Count; j++)
{
- toInsert = j;
- break;
+ if (values[j] < randomNumber)
+ {
+ toInsert = j;
+ break;
+ }
}
- }
+ }
+ else {
+ toInsert = dices.Count;
+ }
dices.Insert(toInsert, GetDice(randomNumber));
values.Insert(toInsert, randomNumber);
}
diff --git a/NadekoBot/Modules/Gambling/GamblingModule.cs b/NadekoBot/Modules/Gambling/GamblingModule.cs
index 96567ced..579e1ed6 100644
--- a/NadekoBot/Modules/Gambling/GamblingModule.cs
+++ b/NadekoBot/Modules/Gambling/GamblingModule.cs
@@ -68,7 +68,7 @@ namespace NadekoBot.Modules.Gambling
{
var amountStr = e.GetArg("amount")?.Trim();
long amount;
- if (!long.TryParse(amountStr, out amount) || amount < 0)
+ if (!long.TryParse(amountStr, out amount) || amount <= 0)
return;
var mentionedUser = e.Message.MentionedUsers.FirstOrDefault(u =>
diff --git a/NadekoBot/Modules/Gambling/Helpers/Cards.cs b/NadekoBot/Modules/Gambling/Helpers/Cards.cs
index 76dc73a5..8408535e 100644
--- a/NadekoBot/Modules/Gambling/Helpers/Cards.cs
+++ b/NadekoBot/Modules/Gambling/Helpers/Cards.cs
@@ -146,7 +146,7 @@ namespace NadekoBot.Modules.Gambling.Helpers
var orderedPool = cardPool.OrderBy(x => r.Next());
cardPool = cardPool as List ?? orderedPool.ToList();
}
- public override string ToString() => string.Join("", cardPool.Select(c => c.ToString())) + Environment.NewLine;
+ public override string ToString() => string.Concat(cardPool.Select(c => c.ToString())) + Environment.NewLine;
private static void InitHandValues()
{
@@ -226,7 +226,7 @@ namespace NadekoBot.Modules.Gambling.Helpers
{
return kvp.Key;
}
- return "High card " + cards.Max().GetName();
+ return "High card " + (cards.FirstOrDefault(c => c.Number == 1)?.GetName() ?? cards.Max().GetName());
}
}
}
diff --git a/NadekoBot/Modules/Games/Commands/PlantPick.cs b/NadekoBot/Modules/Games/Commands/PlantPick.cs
index 6f01c170..6af69857 100644
--- a/NadekoBot/Modules/Games/Commands/PlantPick.cs
+++ b/NadekoBot/Modules/Games/Commands/PlantPick.cs
@@ -1,9 +1,11 @@
using Discord;
using Discord.Commands;
using NadekoBot.Classes;
+using NadekoBot.Extensions;
using NadekoBot.Modules.Permissions.Classes;
using System;
using System.Collections.Concurrent;
+using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Security.Cryptography;
@@ -43,22 +45,19 @@ namespace NadekoBot.Modules.Games.Commands
if (config.GenerateCurrencyChannels.TryGetValue(e.Channel.Id, out cd))
if (!plantpickCooldowns.TryGetValue(e.Channel.Id, out lastSpawned) || (lastSpawned + new TimeSpan(0, cd, 0)) < now)
{
- var rnd = Math.Abs(GetRandomNumber());
- if ((rnd % 50) == 0)
+ var rnd = Math.Abs(rng.Next(0,101));
+ if (rnd == 0)
{
- var msg = await e.Channel.SendFile(GetRandomCurrencyImagePath());
- var msg2 = await e.Channel.SendMessage($"❗ A random {NadekoBot.Config.CurrencyName} appeared! Pick it up by typing `>pick`");
- plantedFlowerChannels.AddOrUpdate(e.Channel.Id, msg, (u, m) => { m.Delete().GetAwaiter().GetResult(); return msg; });
+ var msgs = new[] { await e.Channel.SendFile(GetRandomCurrencyImagePath()), await e.Channel.SendMessage($"❗ A random {NadekoBot.Config.CurrencyName} appeared! Pick it up by typing `>pick`") };
+ plantedFlowerChannels.AddOrUpdate(e.Channel.Id, msgs, (u, m) => { m.ForEach(async msgToDelete => { try { await msgToDelete.Delete(); } catch { } }); return msgs; });
plantpickCooldowns.AddOrUpdate(e.Channel.Id, now, (i, d) => now);
- await Task.Delay(5000);
- await msg2.Delete();
}
}
}
catch { }
}
//channelid/messageid pair
- ConcurrentDictionary plantedFlowerChannels = new ConcurrentDictionary();
+ ConcurrentDictionary> plantedFlowerChannels = new ConcurrentDictionary>();
private object locker = new object();
@@ -68,22 +67,24 @@ namespace NadekoBot.Modules.Games.Commands
.Description("Picks a flower planted in this channel.")
.Do(async e =>
{
- Message msg;
+ IEnumerable msgs;
await e.Message.Delete().ConfigureAwait(false);
- if (!plantedFlowerChannels.TryRemove(e.Channel.Id, out msg))
+ if (!plantedFlowerChannels.TryRemove(e.Channel.Id, out msgs))
return;
- await msg.Delete().ConfigureAwait(false);
+ foreach(var msgToDelete in msgs)
+ await msgToDelete.Delete().ConfigureAwait(false);
+
await FlowersHandler.AddFlowersAsync(e.User, "Picked a flower.", 1, true).ConfigureAwait(false);
- msg = await e.Channel.SendMessage($"**{e.User.Name}** picked a {NadekoBot.Config.CurrencyName}!").ConfigureAwait(false);
+ var msg = await e.Channel.SendMessage($"**{e.User.Name}** picked a {NadekoBot.Config.CurrencyName}!").ConfigureAwait(false);
await Task.Delay(10000).ConfigureAwait(false);
await msg.Delete().ConfigureAwait(false);
});
cgb.CreateCommand(Module.Prefix + "plant")
.Description("Spend a flower to plant it in this channel. (If bot is restarted or crashes, flower will be lost)")
- .Do(async e =>
+ .Do(e =>
{
lock (locker)
{
@@ -92,7 +93,7 @@ namespace NadekoBot.Modules.Games.Commands
e.Channel.SendMessage($"There is already a {NadekoBot.Config.CurrencyName} in this channel.");
return;
}
- var removed = FlowersHandler.RemoveFlowers(e.User, "Planted a flower.", 1).GetAwaiter().GetResult();
+ var removed = FlowersHandler.RemoveFlowers(e.User, "Planted a flower.", 1, true).GetAwaiter().GetResult();
if (!removed)
{
e.Channel.SendMessage($"You don't have any {NadekoBot.Config.CurrencyName}s.").Wait();
@@ -101,17 +102,14 @@ namespace NadekoBot.Modules.Games.Commands
var file = GetRandomCurrencyImagePath();
Message msg;
- //todo send message after, not in lock
if (file == null)
msg = e.Channel.SendMessage(NadekoBot.Config.CurrencySign).GetAwaiter().GetResult();
else
msg = e.Channel.SendFile(file).GetAwaiter().GetResult();
- plantedFlowerChannels.TryAdd(e.Channel.Id, msg);
+ var vowelFirst = new[] { 'a', 'e', 'i', 'o', 'u' }.Contains(NadekoBot.Config.CurrencyName[0]);
+ var msg2 = e.Channel.SendMessage($"Oh how Nice! **{e.User.Name}** planted {(vowelFirst ? "an" : "a")} {NadekoBot.Config.CurrencyName}. Pick it using {Module.Prefix}pick").GetAwaiter().GetResult();
+ plantedFlowerChannels.TryAdd(e.Channel.Id, new[] { msg, msg2 });
}
- var vowelFirst = new[] { 'a', 'e', 'i', 'o', 'u' }.Contains(NadekoBot.Config.CurrencyName[0]);
- var msg2 = await e.Channel.SendMessage($"Oh how Nice! **{e.User.Name}** planted {(vowelFirst ? "an" : "a")} {NadekoBot.Config.CurrencyName}. Pick it using {Module.Prefix}pick");
- await Task.Delay(20000).ConfigureAwait(false);
- await msg2.Delete().ConfigureAwait(false);
});
cgb.CreateCommand(Prefix + "gencurrency")
diff --git a/NadekoBot/Modules/Games/Commands/SpeedTyping.cs b/NadekoBot/Modules/Games/Commands/SpeedTyping.cs
index 8a807208..c224de1d 100644
--- a/NadekoBot/Modules/Games/Commands/SpeedTyping.cs
+++ b/NadekoBot/Modules/Games/Commands/SpeedTyping.cs
@@ -190,8 +190,6 @@ namespace NadekoBot.Modules.Games.Commands
await e.Channel.SendMessage("Added new article for typing game.").ConfigureAwait(false);
});
-
- //todo add user submissions
}
}
}
diff --git a/NadekoBot/Modules/Help/Commands/HelpCommand.cs b/NadekoBot/Modules/Help/Commands/HelpCommand.cs
index 2b4ad3b8..4d3fa441 100644
--- a/NadekoBot/Modules/Help/Commands/HelpCommand.cs
+++ b/NadekoBot/Modules/Help/Commands/HelpCommand.cs
@@ -5,6 +5,7 @@ using NadekoBot.Modules.Permissions.Classes;
using System;
using System.IO;
using System.Linq;
+using System.Text.RegularExpressions;
using System.Threading.Tasks;
namespace NadekoBot.Classes.Help.Commands
@@ -24,8 +25,13 @@ namespace NadekoBot.Classes.Help.Commands
var com = NadekoBot.Client.GetService().AllCommands
.FirstOrDefault(c => c.Text.ToLowerInvariant().Equals(comToFind) ||
c.Aliases.Select(a => a.ToLowerInvariant()).Contains(comToFind));
+
+ var str = "";
+ var alias = com.Aliases.FirstOrDefault();
+ if (alias != null)
+ str = $" / `{ com.Aliases.FirstOrDefault()}`";
if (com != null)
- await e.Channel.SendMessage($"**__Help for `{com.Text}`__ / __`{("" + com.Aliases.FirstOrDefault() + "" ?? "")}`__**\n**Desc:** {com.Description.Replace("|", "\n**Usage:**")}").ConfigureAwait(false);
+ await e.Channel.SendMessage($@"**__Help for:__ `{com.Text}`**" + str + $"\n**Desc:** {new Regex(@"\|").Replace(com.Description, "\n**Usage:**",1)}").ConfigureAwait(false);
}).ConfigureAwait(false);
};
public static string HelpString {
@@ -43,7 +49,7 @@ namespace NadekoBot.Classes.Help.Commands
{
string helpstr =
$@"######For more information and how to setup your own NadekoBot, go to: **http://github.com/Kwoth/NadekoBot/**
-######You can donate on paypal: `nadekodiscordbot@gmail.com` or Bitcoin `17MZz1JAqME39akMLrVT4XBPffQJ2n1EPa`
+######You can donate on paypal: `nadekodiscordbot@gmail.com`
#NadekoBot List Of Commands
Version: `{NadekoStats.Instance.BotVersion}`";
diff --git a/NadekoBot/Modules/Help/HelpModule.cs b/NadekoBot/Modules/Help/HelpModule.cs
index 511779c2..20e8279c 100644
--- a/NadekoBot/Modules/Help/HelpModule.cs
+++ b/NadekoBot/Modules/Help/HelpModule.cs
@@ -46,14 +46,15 @@ namespace NadekoBot.Modules.Help
if (string.IsNullOrWhiteSpace(module))
return;
var cmds = NadekoBot.Client.GetService().AllCommands
- .Where(c => c.Category.ToLower() == module);
+ .Where(c => c.Category.ToLower() == module)
+ .OrderBy(c=>c.Text)
+ .AsEnumerable();
var cmdsArray = cmds as Command[] ?? cmds.ToArray();
if (!cmdsArray.Any())
{
await e.Channel.SendMessage("That module does not exist.").ConfigureAwait(false);
return;
}
- var i = 0;
if (module != "customreactions" && module != "conversations")
{
await e.Channel.SendMessage("`List Of Commands:`\n" + SearchHelper.ShowInPrettyCode(cmdsArray,
diff --git a/NadekoBot/Modules/Music/Classes/MusicControls.cs b/NadekoBot/Modules/Music/Classes/MusicControls.cs
index 41e8c7c1..bcd661f8 100644
--- a/NadekoBot/Modules/Music/Classes/MusicControls.cs
+++ b/NadekoBot/Modules/Music/Classes/MusicControls.cs
@@ -2,7 +2,9 @@
using Discord.Audio;
using NadekoBot.Extensions;
using System;
+using System.Collections.Concurrent;
using System.Collections.Generic;
+using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace NadekoBot.Modules.Music.Classes
@@ -20,22 +22,18 @@ namespace NadekoBot.Modules.Music.Classes
{
Resolving,
Queued,
- Buffering, //not using it atm
Playing,
Completed
}
public class MusicPlayer
{
- public static int MaximumPlaylistSize => 50;
-
private IAudioClient audioClient { get; set; }
private readonly List playlist = new List();
public IReadOnlyCollection Playlist => playlist;
- private readonly object playlistLock = new object();
- public Song CurrentSong { get; set; } = default(Song);
+ public Song CurrentSong { get; private set; }
private CancellationTokenSource SongCancelSource { get; set; }
private CancellationToken cancelToken { get; set; }
@@ -54,6 +52,8 @@ namespace NadekoBot.Modules.Music.Classes
public bool Autoplay { get; set; } = false;
public uint MaxQueueSize { get; set; } = 0;
+ private ConcurrentQueue actionQueue { get; set; } = new ConcurrentQueue();
+
public MusicPlayer(Channel startingVoiceChannel, float? defaultVolume)
{
if (startingVoiceChannel == null)
@@ -68,85 +68,110 @@ namespace NadekoBot.Modules.Music.Classes
Task.Run(async () =>
{
- while (!Destroyed)
+ try
{
- try
- {
- if (audioClient?.State != ConnectionState.Connected)
- audioClient = await PlaybackVoiceChannel.JoinAudio().ConfigureAwait(false);
- }
- catch
- {
- await Task.Delay(1000).ConfigureAwait(false);
- continue;
- }
- CurrentSong = GetNextSong();
- var curSong = CurrentSong;
- if (curSong != null)
+ while (!Destroyed)
{
try
{
- OnStarted(this, curSong);
- await curSong.Play(audioClient, cancelToken).ConfigureAwait(false);
+ Action action;
+ if (actionQueue.TryDequeue(out action))
+ {
+ action();
+ }
}
- catch (OperationCanceledException)
+ finally
{
- Console.WriteLine("Song canceled");
+ await Task.Delay(100).ConfigureAwait(false);
}
- catch (Exception ex)
- {
- Console.WriteLine($"Exception in PlaySong: {ex}");
- }
- OnCompleted(this, curSong);
- curSong = CurrentSong; //to check if its null now
- if (curSong != null)
- if (RepeatSong)
- playlist.Insert(0, curSong);
- else if (RepeatPlaylist)
- playlist.Insert(playlist.Count, curSong);
- SongCancelSource = new CancellationTokenSource();
- cancelToken = SongCancelSource.Token;
}
- await Task.Delay(1000).ConfigureAwait(false);
}
- });
+ catch (Exception ex)
+ {
+ Console.WriteLine("Action queue crashed");
+ Console.WriteLine(ex);
+ }
+ }).ConfigureAwait(false);
+
+ var t = new Thread(new ThreadStart(async () =>
+ {
+ try
+ {
+ while (!Destroyed)
+ {
+ try
+ {
+ if (audioClient?.State != ConnectionState.Connected)
+ {
+ audioClient = await PlaybackVoiceChannel.JoinAudio();
+ continue;
+ }
+
+ CurrentSong = GetNextSong();
+ RemoveSongAt(0);
+
+ if (CurrentSong == null)
+ continue;
+
+ try
+ {
+ OnStarted(this, CurrentSong);
+ await CurrentSong.Play(audioClient, cancelToken);
+ }
+ catch (OperationCanceledException)
+ {
+ Console.WriteLine("Song canceled");
+ SongCancelSource = new CancellationTokenSource();
+ cancelToken = SongCancelSource.Token;
+ }
+ OnCompleted(this, CurrentSong);
+
+ if (RepeatPlaylist)
+ AddSong(CurrentSong, CurrentSong.QueuerName);
+
+ if (RepeatSong)
+ AddSong(CurrentSong, 0);
+
+ }
+ finally
+ {
+ await Task.Delay(300).ConfigureAwait(false);
+ CurrentSong = null;
+ }
+ }
+ }
+ catch (Exception ex) {
+ Console.WriteLine("Music thread crashed.");
+ Console.WriteLine(ex);
+ }
+ }));
+
+ t.Start();
}
public void Next()
{
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
- if (!SongCancelSource.IsCancellationRequested)
- {
- Paused = false;
- SongCancelSource.Cancel();
- }
- }
+ Paused = false;
+ SongCancelSource.Cancel();
+ });
}
public void Stop()
{
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
- playlist.Clear();
- CurrentSong = null;
RepeatPlaylist = false;
RepeatSong = false;
+ playlist.Clear();
if (!SongCancelSource.IsCancellationRequested)
SongCancelSource.Cancel();
- }
+ });
}
public void TogglePause() => Paused = !Paused;
- public void Shuffle()
- {
- lock (playlistLock)
- {
- playlist.Shuffle();
- }
- }
-
public int SetVolume(int volume)
{
if (volume < 0)
@@ -158,16 +183,15 @@ namespace NadekoBot.Modules.Music.Classes
return volume;
}
- private Song GetNextSong()
+ private Song GetNextSong() =>
+ playlist.FirstOrDefault();
+
+ public void Shuffle()
{
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
- if (playlist.Count == 0)
- return null;
- var toReturn = playlist[0];
- playlist.RemoveAt(0);
- return toReturn;
- }
+ playlist.Shuffle();
+ });
}
public void AddSong(Song s, string username)
@@ -175,42 +199,64 @@ namespace NadekoBot.Modules.Music.Classes
if (s == null)
throw new ArgumentNullException(nameof(s));
ThrowIfQueueFull();
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
s.MusicPlayer = this;
s.QueuerName = username.TrimTo(10);
playlist.Add(s);
- }
+ });
}
public void AddSong(Song s, int index)
{
if (s == null)
throw new ArgumentNullException(nameof(s));
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
playlist.Insert(index, s);
- }
+ });
}
public void RemoveSong(Song s)
{
if (s == null)
throw new ArgumentNullException(nameof(s));
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
playlist.Remove(s);
- }
+ });
}
public void RemoveSongAt(int index)
{
- lock (playlistLock)
+ actionQueue.Enqueue(() =>
{
if (index < 0 || index >= playlist.Count)
- throw new ArgumentException("Invalid index");
+ return;
playlist.RemoveAt(index);
- }
+ });
+ }
+
+ internal void ClearQueue()
+ {
+ actionQueue.Enqueue(() =>
+ {
+ playlist.Clear();
+ });
+ }
+
+ public void Destroy()
+ {
+ actionQueue.Enqueue(() =>
+ {
+ RepeatPlaylist = false;
+ RepeatSong = false;
+ Destroyed = true;
+ playlist.Clear();
+ if (!SongCancelSource.IsCancellationRequested)
+ SongCancelSource.Cancel();
+ audioClient.Disconnect();
+ });
}
internal Task MoveToVoiceChannel(Channel voiceChannel)
@@ -221,27 +267,6 @@ namespace NadekoBot.Modules.Music.Classes
return PlaybackVoiceChannel.JoinAudio();
}
- internal void ClearQueue()
- {
- lock (playlistLock)
- {
- playlist.Clear();
- }
- }
-
- public void Destroy()
- {
- lock (playlistLock)
- {
- playlist.Clear();
- Destroyed = true;
- CurrentSong = null;
- if (!SongCancelSource.IsCancellationRequested)
- SongCancelSource.Cancel();
- audioClient.Disconnect();
- }
- }
-
internal bool ToggleRepeatSong() => this.RepeatSong = !this.RepeatSong;
internal bool ToggleRepeatPlaylist() => this.RepeatPlaylist = !this.RepeatPlaylist;
diff --git a/NadekoBot/Modules/Music/Classes/Song.cs b/NadekoBot/Modules/Music/Classes/Song.cs
index e718302e..025863e0 100644
--- a/NadekoBot/Modules/Music/Classes/Song.cs
+++ b/NadekoBot/Modules/Music/Classes/Song.cs
@@ -31,7 +31,7 @@ namespace NadekoBot.Modules.Music.Classes
public SongInfo SongInfo { get; }
public string QueuerName { get; set; }
- private PoopyBuffer songBuffer { get; } = new PoopyBuffer(NadekoBot.Config.BufferSize);
+ private PoopyBuffer songBuffer { get; set; }
private bool prebufferingComplete { get; set; } = false;
public MusicPlayer MusicPlayer { get; set; }
@@ -137,6 +137,9 @@ namespace NadekoBot.Modules.Music.Classes
internal async Task Play(IAudioClient voiceClient, CancellationToken cancelToken)
{
+ // initialize the buffer here because if this song was playing before (requeued), we must delete old buffer data
+ songBuffer = new PoopyBuffer(NadekoBot.Config.BufferSize);
+
var bufferTask = BufferSong(cancelToken).ConfigureAwait(false);
var bufferAttempts = 0;
const int waitPerAttempt = 500;
@@ -145,7 +148,6 @@ namespace NadekoBot.Modules.Music.Classes
{
await Task.Delay(waitPerAttempt, cancelToken).ConfigureAwait(false);
}
- cancelToken.ThrowIfCancellationRequested();
Console.WriteLine($"Prebuffering done? in {waitPerAttempt * bufferAttempts}");
const int blockSize = 3840;
var attempt = 0;
diff --git a/NadekoBot/Modules/Music/MusicModule.cs b/NadekoBot/Modules/Music/MusicModule.cs
index 8c04a58b..ba04d598 100644
--- a/NadekoBot/Modules/Music/MusicModule.cs
+++ b/NadekoBot/Modules/Music/MusicModule.cs
@@ -31,17 +31,17 @@ namespace NadekoBot.Modules.Music
{
var client = NadekoBot.Client;
- manager.CreateCommands(Prefix, cgb =>
+ manager.CreateCommands("", cgb =>
{
cgb.AddCheck(PermissionChecker.Instance);
commands.ForEach(cmd => cmd.Init(cgb));
- cgb.CreateCommand("next")
- .Alias("n")
- .Alias("skip")
- .Description("Goes to the next song in the queue. You have to be in the same voice channel as the bot. | `!m n`")
+ cgb.CreateCommand(Prefix + "next")
+ .Alias(Prefix + "n")
+ .Alias(Prefix + "skip")
+ .Description($"Goes to the next song in the queue. You have to be in the same voice channel as the bot. | `{Prefix}n`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -50,9 +50,9 @@ namespace NadekoBot.Modules.Music
musicPlayer.Next();
});
- cgb.CreateCommand("stop")
- .Alias("s")
- .Description("Stops the music and clears the playlist. Stays in the channel. | `!m s`")
+ cgb.CreateCommand(Prefix + "stop")
+ .Alias(Prefix + "s")
+ .Description($"Stops the music and clears the playlist. Stays in the channel. | `{Prefix}s`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -64,10 +64,10 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("destroy")
- .Alias("d")
+ cgb.CreateCommand(Prefix + "destroy")
+ .Alias(Prefix + "d")
.Description("Completely stops the music and unbinds the bot from the channel. " +
- "(may cause weird behaviour) | `!m d`")
+ $"(may cause weird behaviour) | `{Prefix}d`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -76,9 +76,9 @@ namespace NadekoBot.Modules.Music
musicPlayer.Destroy();
});
- cgb.CreateCommand("pause")
- .Alias("p")
- .Description("Pauses or Unpauses the song. | `!m p`")
+ cgb.CreateCommand(Prefix + "pause")
+ .Alias(Prefix + "p")
+ .Description($"Pauses or Unpauses the song. | `{Prefix}p`")
.Do(async e =>
{
MusicPlayer musicPlayer;
@@ -92,11 +92,11 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage("🎵`Music Player unpaused.`").ConfigureAwait(false);
});
- cgb.CreateCommand("queue")
- .Alias("q")
- .Alias("yq")
+ cgb.CreateCommand(Prefix + "queue")
+ .Alias(Prefix + "q")
+ .Alias(Prefix + "yq")
.Description("Queue a song using keywords or a link. Bot will join your voice channel." +
- "**You must be in a voice channel**. | `!m q Dream Of Venice`")
+ $"**You must be in a voice channel**. | `{Prefix}q Dream Of Venice`")
.Parameter("query", ParameterType.Unparsed)
.Do(async e =>
{
@@ -108,10 +108,10 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("soundcloudqueue")
- .Alias("sq")
+ cgb.CreateCommand(Prefix + "soundcloudqueue")
+ .Alias(Prefix + "sq")
.Description("Queue a soundcloud song using keywords. Bot will join your voice channel." +
- "**You must be in a voice channel**. | `!m sq Dream Of Venice`")
+ $"**You must be in a voice channel**. | `{Prefix}sq Dream Of Venice`")
.Parameter("query", ParameterType.Unparsed)
.Do(async e =>
{
@@ -123,9 +123,9 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("listqueue")
- .Alias("lq")
- .Description("Lists 15 currently queued songs per page. Default page is 1. | `!m lq` or `!m lq 2`")
+ cgb.CreateCommand(Prefix + "listqueue")
+ .Alias(Prefix + "lq")
+ .Description($"Lists 15 currently queued songs per page. Default page is 1. | `{Prefix}lq` or `{Prefix}lq 2`")
.Parameter("page", ParameterType.Optional)
.Do(async e =>
{
@@ -151,7 +151,7 @@ namespace NadekoBot.Modules.Music
else if (musicPlayer.RepeatPlaylist)
toSend += "🔁";
toSend += $" **{musicPlayer.Playlist.Count}** `tracks currently queued. Showing page {page}` ";
- if (musicPlayer.Playlist.Count >= MusicPlayer.MaximumPlaylistSize)
+ if (musicPlayer.MaxQueueSize != 0 && musicPlayer.Playlist.Count >= musicPlayer.MaxQueueSize)
toSend += "**Song queue is full!**\n";
else
toSend += "\n";
@@ -161,9 +161,9 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage(toSend + string.Join("\n", musicPlayer.Playlist.Skip(startAt).Take(15).Select(v => $"`{number++}.` {v.PrettyName}"))).ConfigureAwait(false);
});
- cgb.CreateCommand("nowplaying")
- .Alias("np")
- .Description("Shows the song currently playing. | `!m np`")
+ cgb.CreateCommand(Prefix + "nowplaying")
+ .Alias(Prefix + "np")
+ .Description($"Shows the song currently playing. | `{Prefix}np`")
.Do(async e =>
{
MusicPlayer musicPlayer;
@@ -176,9 +176,9 @@ namespace NadekoBot.Modules.Music
$"{currentSong.PrettyCurrentTime()}").ConfigureAwait(false);
});
- cgb.CreateCommand("volume")
- .Alias("vol")
- .Description("Sets the music volume 0-100% | `!m vol 50`")
+ cgb.CreateCommand(Prefix + "volume")
+ .Alias(Prefix + "vol")
+ .Description($"Sets the music volume 0-100% | `{Prefix}vol 50`")
.Parameter("val", ParameterType.Required)
.Do(async e =>
{
@@ -198,10 +198,10 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎵 `Volume set to {volume}%`").ConfigureAwait(false);
});
- cgb.CreateCommand("defvol")
- .Alias("dv")
+ cgb.CreateCommand(Prefix + "defvol")
+ .Alias(Prefix + "dv")
.Description("Sets the default music volume when music playback is started (0-100)." +
- " Persists through restarts. | `!m dv 80`")
+ $" Persists through restarts. | `{Prefix}dv 80`")
.Parameter("val", ParameterType.Required)
.Do(async e =>
{
@@ -217,9 +217,9 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎵 `Default volume set to {volume}%`").ConfigureAwait(false);
});
- cgb.CreateCommand("mute")
- .Alias("min")
- .Description("Sets the music volume to 0% | `!m min`")
+ cgb.CreateCommand(Prefix + "mute")
+ .Alias(Prefix + "min")
+ .Description($"Sets the music volume to 0% | `{Prefix}min`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -230,8 +230,8 @@ namespace NadekoBot.Modules.Music
musicPlayer.SetVolume(0);
});
- cgb.CreateCommand("max")
- .Description("Sets the music volume to 100%. | `!m max`")
+ cgb.CreateCommand(Prefix + "max")
+ .Description($"Sets the music volume to 100%. | `{Prefix}max`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -242,8 +242,8 @@ namespace NadekoBot.Modules.Music
musicPlayer.SetVolume(100);
});
- cgb.CreateCommand("half")
- .Description("Sets the music volume to 50%. | `!m half`")
+ cgb.CreateCommand(Prefix + "half")
+ .Description($"Sets the music volume to 50%. | `{Prefix}half`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -254,9 +254,9 @@ namespace NadekoBot.Modules.Music
musicPlayer.SetVolume(50);
});
- cgb.CreateCommand("shuffle")
- .Alias("sh")
- .Description("Shuffles the current playlist. | `!m sh`")
+ cgb.CreateCommand(Prefix + "shuffle")
+ .Alias(Prefix + "sh")
+ .Description($"Shuffles the current playlist. | `{Prefix}sh`")
.Do(async e =>
{
MusicPlayer musicPlayer;
@@ -274,9 +274,9 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage("🎵 `Songs shuffled.`").ConfigureAwait(false);
});
- cgb.CreateCommand("playlist")
- .Alias("pl")
- .Description("Queues up to 500 songs from a youtube playlist specified by a link, or keywords. | `!m pl playlist link or name`")
+ cgb.CreateCommand(Prefix + "playlist")
+ .Alias(Prefix + "pl")
+ .Description($"Queues up to 500 songs from a youtube playlist specified by a link, or keywords. | `{Prefix}pl playlist link or name`")
.Parameter("playlist", ParameterType.Unparsed)
.Do(async e =>
{
@@ -300,7 +300,6 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎵 `Failed to find any songs.`").ConfigureAwait(false);
return;
}
- //todo TEMPORARY SOLUTION, USE RESOLVE QUEUE IN THE FUTURE
var idArray = ids as string[] ?? ids.ToArray();
var count = idArray.Length;
var msg =
@@ -318,9 +317,9 @@ namespace NadekoBot.Modules.Music
await msg.Edit("🎵 `Playlist queue complete.`").ConfigureAwait(false);
});
- cgb.CreateCommand("soundcloudpl")
- .Alias("scpl")
- .Description("Queue a soundcloud playlist using a link. | `!m scpl https://soundcloud.com/saratology/sets/symphony`")
+ cgb.CreateCommand(Prefix + "soundcloudpl")
+ .Alias(Prefix + "scpl")
+ .Description($"Queue a soundcloud playlist using a link. | `{Prefix}scpl https://soundcloud.com/saratology/sets/symphony`")
.Parameter("pl", ParameterType.Unparsed)
.Do(async e =>
{
@@ -353,9 +352,9 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("localplaylst")
- .Alias("lopl")
- .Description("Queues all songs from a directory. **Bot Owner Only!** | `!m lopl C:/music/classical`")
+ cgb.CreateCommand(Prefix + "localplaylst")
+ .Alias(Prefix + "lopl")
+ .Description($"Queues all songs from a directory. **Bot Owner Only!** | `{Prefix}lopl C:/music/classical`")
.Parameter("directory", ParameterType.Unparsed)
.AddCheck(SimpleCheckers.OwnerOnly())
.Do(async e =>
@@ -384,8 +383,8 @@ namespace NadekoBot.Modules.Music
catch { }
});
- cgb.CreateCommand("radio").Alias("ra")
- .Description("Queues a radio stream from a link. It can be a direct mp3 radio stream, .m3u, .pls .asx or .xspf (Usage Video: ) | `!m ra radio link here`")
+ cgb.CreateCommand(Prefix + "radio").Alias(Prefix + "ra")
+ .Description($"Queues a radio stream from a link. It can be a direct mp3 radio stream, .m3u, .pls .asx or .xspf (Usage Video: ) | `{Prefix}ra radio link here`")
.Parameter("radio_link", ParameterType.Required)
.Do(async e =>
{
@@ -402,9 +401,9 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("local")
- .Alias("lo")
- .Description("Queues a local file by specifying a full path. **Bot Owner Only!** | `!m lo C:/music/mysong.mp3`")
+ cgb.CreateCommand(Prefix + "local")
+ .Alias(Prefix + "lo")
+ .Description($"Queues a local file by specifying a full path. **Bot Owner Only!** | `{Prefix}lo C:/music/mysong.mp3`")
.Parameter("path", ParameterType.Unparsed)
.AddCheck(SimpleCheckers.OwnerOnly())
.Do(async e =>
@@ -415,9 +414,9 @@ namespace NadekoBot.Modules.Music
await QueueSong(e.User, e.Channel, e.User.VoiceChannel, e.GetArg("path"), musicType: MusicType.Local).ConfigureAwait(false);
});
- cgb.CreateCommand("move")
- .Alias("mv")
- .Description("Moves the bot to your voice channel. (works only if music is already playing) | `!m mv`")
+ cgb.CreateCommand(Prefix + "move")
+ .Alias(Prefix + "mv")
+ .Description($"Moves the bot to your voice channel. (works only if music is already playing) | `{Prefix}mv`")
.Do(e =>
{
MusicPlayer musicPlayer;
@@ -427,9 +426,9 @@ namespace NadekoBot.Modules.Music
musicPlayer.MoveToVoiceChannel(voiceChannel);
});
- cgb.CreateCommand("remove")
- .Alias("rm")
- .Description("Remove a song by its # in the queue, or 'all' to remove whole queue. | `!m rm 5`")
+ cgb.CreateCommand(Prefix + "remove")
+ .Alias(Prefix + "rm")
+ .Description($"Remove a song by its # in the queue, or 'all' to remove whole queue. | `{Prefix}rm 5`")
.Parameter("num", ParameterType.Required)
.Do(async e =>
{
@@ -460,8 +459,8 @@ namespace NadekoBot.Modules.Music
});
//var msRegex = new Regex(@"(?\d+)>(?\d+)", RegexOptions.Compiled);
- cgb.CreateCommand("movesong")
- .Alias("ms")
+ cgb.CreateCommand(Prefix + "movesong")
+ .Alias(Prefix + "ms")
.Description($"Moves a song from one position to another. | `{Prefix} ms` 5>3")
.Parameter("fromto")
.Do(async e =>
@@ -496,9 +495,9 @@ namespace NadekoBot.Modules.Music
});
- cgb.CreateCommand("setmaxqueue")
- .Alias("smq")
- .Description($"Sets a maximum queue size. Supply 0 or no argument to have no limit. | `{Prefix} smq` 50 or `{Prefix} smq`")
+ cgb.CreateCommand(Prefix + "setmaxqueue")
+ .Alias(Prefix + "smq")
+ .Description($"Sets a maximum queue size. Supply 0 or no argument to have no limit. | `{Prefix}smq` 50 or `{Prefix}smq`")
.Parameter("size", ParameterType.Unparsed)
.Do(async e =>
{
@@ -519,8 +518,8 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎵 `Max queue set to {(size == 0 ? ("unlimited") : size + " tracks")}`");
});
- cgb.CreateCommand("cleanup")
- .Description("Cleans up hanging voice connections. **Bot Owner Only!** | `!m cleanup`")
+ cgb.CreateCommand(Prefix + "cleanup")
+ .Description($"Cleans up hanging voice connections. **Bot Owner Only!** | `{Prefix}cleanup`")
.AddCheck(SimpleCheckers.OwnerOnly())
.Do(e =>
{
@@ -537,9 +536,9 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("reptcursong")
- .Alias("rcs")
- .Description("Toggles repeat of current song. | `!m rcs`")
+ cgb.CreateCommand(Prefix + "reptcursong")
+ .Alias(Prefix + "rcs")
+ .Description($"Toggles repeat of current song. | `{Prefix}rcs`")
.Do(async e =>
{
MusicPlayer musicPlayer;
@@ -555,9 +554,9 @@ namespace NadekoBot.Modules.Music
.ConfigureAwait(false);
});
- cgb.CreateCommand("rpeatplaylst")
- .Alias("rpl")
- .Description("Toggles repeat of all songs in the queue (every song that finishes is added to the end of the queue). | `!m rpl`")
+ cgb.CreateCommand(Prefix + "rpeatplaylst")
+ .Alias(Prefix + "rpl")
+ .Description($"Toggles repeat of all songs in the queue (every song that finishes is added to the end of the queue). | `{Prefix}rpl`")
.Do(async e =>
{
MusicPlayer musicPlayer;
@@ -567,8 +566,8 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎵🔁`Repeat playlist {(currentValue ? "enabled" : "disabled")}`").ConfigureAwait(false);
});
- cgb.CreateCommand("save")
- .Description("Saves a playlist under a certain name. Name must be no longer than 20 characters and mustn't contain dashes. | `!m save classical1`")
+ cgb.CreateCommand(Prefix + "save")
+ .Description($"Saves a playlist under a certain name. Name must be no longer than 20 characters and mustn't contain dashes. | `{Prefix}save classical1`")
.Parameter("name", ParameterType.Unparsed)
.Do(async e =>
{
@@ -620,8 +619,8 @@ namespace NadekoBot.Modules.Music
});
- cgb.CreateCommand("load")
- .Description("Loads a playlist under a certain name. | `!m load classical-1`")
+ cgb.CreateCommand(Prefix + "load")
+ .Description($"Loads a playlist under a certain name. | `{Prefix}load classical-1`")
.Parameter("name", ParameterType.Unparsed)
.Do(async e =>
{
@@ -679,9 +678,9 @@ namespace NadekoBot.Modules.Music
}
});
- cgb.CreateCommand("playlists")
- .Alias("pls")
- .Description("Lists all playlists. Paginated. 20 per page. Default page is 0. |`!m pls 1`")
+ cgb.CreateCommand(Prefix + "playlists")
+ .Alias(Prefix + "pls")
+ .Description($"Lists all playlists. Paginated. 20 per page. Default page is 0. |`{Prefix}pls 1`")
.Parameter("num", ParameterType.Optional)
.Do(e =>
{
@@ -696,9 +695,9 @@ namespace NadekoBot.Modules.Music
e.Channel.SendMessage($"```js\n--- List of saved playlists ---\n\n" + string.Join("\n", result.Select(r => $"'{r.Name}-{r.Id}' by {r.Creator} ({r.SongCnt} songs)")) + $"\n\n --- Page {num} ---```").ConfigureAwait(false);
});
- cgb.CreateCommand("deleteplaylist")
- .Alias("delpls")
- .Description("Deletes a saved playlist. Only if you made it or if you are the bot owner. | `!m delpls animu-5`")
+ cgb.CreateCommand(Prefix + "deleteplaylist")
+ .Alias(Prefix + "delpls")
+ .Description($"Deletes a saved playlist. Only if you made it or if you are the bot owner. | `{Prefix}delpls animu-5`")
.Parameter("pl", ParameterType.Required)
.Do(async e =>
{
@@ -713,7 +712,7 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage("`Ok.` :ok:").ConfigureAwait(false);
});
- cgb.CreateCommand("goto")
+ cgb.CreateCommand(Prefix + "goto")
.Description("Goes to a specific time in seconds in a song.")
.Parameter("time")
.Do(async e =>
@@ -750,8 +749,8 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"`Skipped to {minutes}:{seconds}`").ConfigureAwait(false);
});
- cgb.CreateCommand("getlink")
- .Alias("gl")
+ cgb.CreateCommand(Prefix + "getlink")
+ .Alias(Prefix + "gl")
.Description("Shows a link to the currently playing song.")
.Do(async e =>
{
@@ -764,8 +763,8 @@ namespace NadekoBot.Modules.Music
await e.Channel.SendMessage($"🎶`Current song:` <{curSong.SongInfo.Query}>").ConfigureAwait(false);
});
- cgb.CreateCommand("autoplay")
- .Alias("ap")
+ cgb.CreateCommand(Prefix + "autoplay")
+ .Alias(Prefix + "ap")
.Description("Toggles autoplay - When the song is finished, automatically queue a related youtube song. (Works only for youtube songs and when queue is empty)")
.Do(async e =>
{
diff --git a/NadekoBot/Modules/Permissions/Classes/PermissionChecker.cs b/NadekoBot/Modules/Permissions/Classes/PermissionChecker.cs
index 551ad193..412ce69a 100644
--- a/NadekoBot/Modules/Permissions/Classes/PermissionChecker.cs
+++ b/NadekoBot/Modules/Permissions/Classes/PermissionChecker.cs
@@ -4,6 +4,7 @@ using Discord.Commands.Permissions;
using NadekoBot.Classes.JSONModels;
using System;
using System.Collections.Concurrent;
+using System.Collections.Generic;
using System.Threading.Tasks;
namespace NadekoBot.Modules.Permissions.Classes
@@ -13,7 +14,10 @@ namespace NadekoBot.Modules.Permissions.Classes
{
public static PermissionChecker Instance { get; } = new PermissionChecker();
- private ConcurrentDictionary timeBlackList { get; } = new ConcurrentDictionary();
+ //key - sid:command
+ //value - userid
+ private ConcurrentDictionary commandCooldowns = new ConcurrentDictionary();
+ private HashSet timeBlackList { get; } = new HashSet();
static PermissionChecker() { }
private PermissionChecker()
@@ -22,8 +26,8 @@ namespace NadekoBot.Modules.Permissions.Classes
{
while (true)
{
- //blacklist is cleared every 1.75 seconds. That is the most time anyone will be blocked
- await Task.Delay(1750).ConfigureAwait(false);
+ //blacklist is cleared every 1.00 seconds. That is the most time anyone will be blocked
+ await Task.Delay(1000).ConfigureAwait(false);
timeBlackList.Clear();
}
});
@@ -46,18 +50,26 @@ namespace NadekoBot.Modules.Permissions.Classes
return false;
}
- if (timeBlackList.ContainsKey(user))
+ if (timeBlackList.Contains(user.Id))
return false;
- timeBlackList.TryAdd(user, DateTime.Now);
-
if (!channel.IsPrivate && !channel.Server.CurrentUser.GetPermissions(channel).SendMessages)
{
return false;
}
- //{
- // user.SendMessage($"I ignored your command in {channel.Server.Name}/#{channel.Name} because i don't have permissions to write to it. Please use `;acm channel_name 0` in that server instead of muting me.").GetAwaiter().GetResult();
- //}
+
+ timeBlackList.Add(user.Id);
+
+ ServerPermissions perms;
+ PermissionsHandler.PermissionsDict.TryGetValue(user.Server.Id, out perms);
+
+ AddUserCooldown(user.Server.Id, user.Id, command.Text.ToLower());
+ if (commandCooldowns.Keys.Contains(user.Server.Id+":"+command.Text.ToLower()))
+ {
+ if(perms?.Verbose == true)
+ error = $"{user.Mention} You have a cooldown on that command.";
+ return false;
+ }
try
{
@@ -75,8 +87,6 @@ namespace NadekoBot.Modules.Permissions.Classes
catch { }
if (user.Server.Owner.Id == user.Id || (role != null && user.HasRole(role)))
return true;
- ServerPermissions perms;
- PermissionsHandler.PermissionsDict.TryGetValue(user.Server.Id, out perms);
throw new Exception($"You don't have the necessary role (**{(perms?.PermissionsControllerRole ?? "Nadeko")}**) to change permissions.");
}
@@ -128,8 +138,7 @@ namespace NadekoBot.Modules.Permissions.Classes
Console.WriteLine($"Exception in canrun: {ex}");
try
{
- ServerPermissions perms;
- if (PermissionsHandler.PermissionsDict.TryGetValue(user.Server.Id, out perms) && perms.Verbose)
+ if (perms != null && perms.Verbose)
//if verbose - print errors
error = ex.Message;
}
@@ -140,5 +149,26 @@ namespace NadekoBot.Modules.Permissions.Classes
return false;
}
}
+
+ public void AddUserCooldown(ulong serverId, ulong userId, string commandName) {
+ commandCooldowns.TryAdd(commandName, userId);
+ var tosave = serverId + ":" + commandName;
+ Task.Run(async () =>
+ {
+ ServerPermissions perms;
+ PermissionsHandler.PermissionsDict.TryGetValue(serverId, out perms);
+ int cd;
+ if (!perms.CommandCooldowns.TryGetValue(commandName,out cd)) {
+ return;
+ }
+ if (commandCooldowns.TryAdd(tosave, userId))
+ {
+ await Task.Delay(cd * 1000);
+ ulong throwaway;
+ commandCooldowns.TryRemove(tosave, out throwaway);
+ }
+
+ });
+ }
}
}
diff --git a/NadekoBot/Modules/Permissions/Classes/PermissionHelper.cs b/NadekoBot/Modules/Permissions/Classes/PermissionHelper.cs
index aef62ac8..c972953e 100644
--- a/NadekoBot/Modules/Permissions/Classes/PermissionHelper.cs
+++ b/NadekoBot/Modules/Permissions/Classes/PermissionHelper.cs
@@ -55,9 +55,11 @@ namespace NadekoBot.Modules.Permissions.Classes
if (string.IsNullOrWhiteSpace(commandText))
throw new ArgumentNullException(nameof(commandText));
+ var normalizedCmdTxt = commandText.Trim().ToUpperInvariant();
+
foreach (var com in NadekoBot.Client.GetService().AllCommands)
{
- if (com.Text.ToLower().Equals(commandText.Trim().ToLower()))
+ if (com.Text.ToUpperInvariant().Equals(normalizedCmdTxt) || com.Aliases.Select(c=>c.ToUpperInvariant()).Contains(normalizedCmdTxt))
return com.Text;
}
throw new NullReferenceException("That command does not exist.");
diff --git a/NadekoBot/Modules/Permissions/Classes/PermissionsHandler.cs b/NadekoBot/Modules/Permissions/Classes/PermissionsHandler.cs
index 93fa8686..c5d2e64c 100644
--- a/NadekoBot/Modules/Permissions/Classes/PermissionsHandler.cs
+++ b/NadekoBot/Modules/Permissions/Classes/PermissionsHandler.cs
@@ -424,6 +424,21 @@ namespace NadekoBot.Modules.Permissions.Classes
Task.Run(() => WriteServerToJson(serverPerms));
}
+ public static void SetCommandCooldown(Server server, string commandName, int value)
+ {
+ var serverPerms = PermissionsDict.GetOrAdd(server.Id,
+ new ServerPermissions(server.Id, server.Name));
+ if (value == 0) {
+ int throwaway;
+ serverPerms.CommandCooldowns.TryRemove(commandName, out throwaway);
+ }
+ else {
+ serverPerms.CommandCooldowns.AddOrUpdate(commandName, value, (str, v) => value);
+ }
+
+ Task.Run(() => WriteServerToJson(serverPerms));
+ }
+
public static void AddFilteredWord(Server server, string word)
{
var serverPerms = PermissionsDict.GetOrAdd(server.Id,
@@ -537,6 +552,10 @@ namespace NadekoBot.Modules.Permissions.Classes
public Dictionary UserPermissions { get; set; }
public Dictionary ChannelPermissions { get; set; }
public Dictionary RolePermissions { get; set; }
+ ///
+ /// Dictionary of command names with their respective cooldowns
+ ///
+ public ConcurrentDictionary CommandCooldowns { get; set; }
public ServerPermissions(ulong id, string name)
{
@@ -549,6 +568,7 @@ namespace NadekoBot.Modules.Permissions.Classes
UserPermissions = new Dictionary();
ChannelPermissions = new Dictionary();
RolePermissions = new Dictionary();
+ CommandCooldowns = new ConcurrentDictionary();
Words = new HashSet();
}
}
diff --git a/NadekoBot/Modules/Permissions/PermissionsModule.cs b/NadekoBot/Modules/Permissions/PermissionsModule.cs
index e210c1e2..4c13b9ff 100644
--- a/NadekoBot/Modules/Permissions/PermissionsModule.cs
+++ b/NadekoBot/Modules/Permissions/PermissionsModule.cs
@@ -1,5 +1,6 @@
using Discord.Commands;
using Discord.Modules;
+using NadekoBot.Classes;
using NadekoBot.Classes.JSONModels;
using NadekoBot.Extensions;
using NadekoBot.Modules.Games.Commands;
@@ -776,6 +777,57 @@ namespace NadekoBot.Modules.Permissions
await e.Channel.SendMessage($"`Sucessfully blacklisted server {server.Name}`").ConfigureAwait(false);
}).ConfigureAwait(false);
});
+
+ cgb.CreateCommand(Prefix + "cmdcooldown")
+ .Alias(Prefix+ "cmdcd")
+ .Description($"Sets a cooldown per user for a command. Set 0 to clear. | `{Prefix}cmdcd \"some cmd\" 5`")
+ .Parameter("command", ParameterType.Required)
+ .Parameter("secs",ParameterType.Required)
+ .AddCheck(SimpleCheckers.ManageMessages())
+ .Do(async e =>
+ {
+ try
+ {
+ var command = PermissionHelper.ValidateCommand(e.GetArg("command"));
+ var secsStr = e.GetArg("secs").Trim();
+ int secs;
+ if (!int.TryParse(secsStr, out secs) || secs < 0 || secs > 3600)
+ throw new ArgumentOutOfRangeException("secs", "Invalid second parameter. (Must be a number between 0 and 3600)");
+
+
+ PermissionsHandler.SetCommandCooldown(e.Server, command, secs);
+ if(secs == 0)
+ await e.Channel.SendMessage($"Command **{command}** has no coooldown now.").ConfigureAwait(false);
+ else
+ await e.Channel.SendMessage($"Command **{command}** now has a **{secs} {(secs==1 ? "second" : "seconds")}** cooldown.").ConfigureAwait(false);
+ }
+ catch (ArgumentException exArg)
+ {
+ await e.Channel.SendMessage(exArg.Message).ConfigureAwait(false);
+ }
+ catch (Exception ex)
+ {
+ await e.Channel.SendMessage("Something went terribly wrong - " + ex.Message).ConfigureAwait(false);
+ }
+ });
+
+ cgb.CreateCommand(Prefix + "allcmdcooldowns")
+ .Alias(Prefix + "acmdcds")
+ .Description("Shows a list of all commands and their respective cooldowns.")
+ .Do(async e =>
+ {
+ ServerPermissions perms;
+ PermissionsHandler.PermissionsDict.TryGetValue(e.Server.Id, out perms);
+ if (perms == null)
+ return;
+
+ if (!perms.CommandCooldowns.Any())
+ {
+ await e.Channel.SendMessage("`No command cooldowns set.`").ConfigureAwait(false);
+ return;
+ }
+ await e.Channel.SendMessage(SearchHelper.ShowInPrettyCode(perms.CommandCooldowns.Select(c=>c.Key+ ": "+c.Value+" secs"),s=>$"{s,-30}",2)).ConfigureAwait(false);
+ });
});
}
}
diff --git a/NadekoBot/Modules/Pokemon/PokemonModule.cs b/NadekoBot/Modules/Pokemon/PokemonModule.cs
index 9d9c4a29..ae3ae0d9 100644
--- a/NadekoBot/Modules/Pokemon/PokemonModule.cs
+++ b/NadekoBot/Modules/Pokemon/PokemonModule.cs
@@ -210,7 +210,7 @@ namespace NadekoBot.Modules.Pokemon
});
cgb.CreateCommand(Prefix + "heal")
- .Description($"Heals someone. Revives those that fainted. Costs a {NadekoBot.Config.CurrencyName} |{Prefix}revive @someone")
+ .Description($"Heals someone. Revives those who fainted. Costs a {NadekoBot.Config.CurrencyName} | {Prefix}heal @someone")
.Parameter("target", ParameterType.Unparsed)
.Do(async e =>
{
@@ -242,7 +242,7 @@ namespace NadekoBot.Modules.Pokemon
return;
}
var target = (usr.Id == e.User.Id) ? "yourself" : usr.Name;
- FlowersHandler.RemoveFlowers(e.User, $"Poke-Heal {target}", amount);
+ await FlowersHandler.RemoveFlowers(e.User, $"Poke-Heal {target}", amount).ConfigureAwait(false);
//healing
targetStats.Hp = targetStats.MaxHp;
if (HP < 0)
@@ -309,7 +309,7 @@ namespace NadekoBot.Modules.Pokemon
await e.Channel.SendMessage($"{e.User.Mention} you don't have enough {NadekoBot.Config.CurrencyName}s! \nYou still need {amount - pts} {NadekoBot.Config.CurrencySign} to be able to do this!").ConfigureAwait(false);
return;
}
- FlowersHandler.RemoveFlowers(e.User, $"set usertype to {targetTypeStr}", amount);
+ await FlowersHandler.RemoveFlowers(e.User, $"set usertype to {targetTypeStr}", amount).ConfigureAwait(false);
//Actually changing the type here
var preTypes = DbHandler.Instance.GetAllRows();
Dictionary Dict = preTypes.ToDictionary(x => x.UserId, y => y.Id.Value);
diff --git a/NadekoBot/Modules/Searches/Commands/EvalCommand.cs b/NadekoBot/Modules/Searches/Commands/EvalCommand.cs
index 0e815f71..ed469517 100644
--- a/NadekoBot/Modules/Searches/Commands/EvalCommand.cs
+++ b/NadekoBot/Modules/Searches/Commands/EvalCommand.cs
@@ -49,11 +49,11 @@ namespace NadekoBot.Modules.Searches.Commands
string result = parser.Parse(expression).ToString();
return result;
}
- catch (OverflowException e)
+ catch (OverflowException)
{
return $"Overflow error on {expression}";
}
- catch (FormatException e)
+ catch (FormatException)
{
return $"\"{expression}\" was not formatted correctly";
}
diff --git a/NadekoBot/Modules/Searches/Commands/MemegenCommands.cs b/NadekoBot/Modules/Searches/Commands/MemegenCommands.cs
index ebbdbf34..5eff7c78 100644
--- a/NadekoBot/Modules/Searches/Commands/MemegenCommands.cs
+++ b/NadekoBot/Modules/Searches/Commands/MemegenCommands.cs
@@ -25,7 +25,7 @@ namespace NadekoBot.Modules.Searches.Commands
string.Join("\n", JsonConvert.DeserializeObject>(await SearchHelper.GetResponseStringAsync("http://memegen.link/templates/"))
.Select(kvp => Path.GetFileName(kvp.Value))
.GroupBy(item => (i++) / 4)
- .Select(ig => string.Join("", ig.Select(el => $"{el,-17}"))))
+ .Select(ig => string.Concat(ig.Select(el => $"{el,-17}"))))
+ $"\n```").ConfigureAwait(false);
});
diff --git a/NadekoBot/Modules/Searches/SearchesModule.cs b/NadekoBot/Modules/Searches/SearchesModule.cs
index 466ff12c..1a2ed20b 100644
--- a/NadekoBot/Modules/Searches/SearchesModule.cs
+++ b/NadekoBot/Modules/Searches/SearchesModule.cs
@@ -15,6 +15,7 @@ using System.Drawing;
using System.IO;
using System.Linq;
using System.Net.Http;
+using System.Web;
namespace NadekoBot.Modules.Searches
{
@@ -219,6 +220,19 @@ $@"🌍 **Weather for** 【{obj["target"]}】
.ConfigureAwait(false);
});
+ cgb.CreateCommand(Prefix + "google")
+ .Alias(Prefix + "g")
+ .Description("Get a google search link for some terms.")
+ .Parameter("terms", ParameterType.Unparsed)
+ .Do(async e =>
+ {
+ var terms = e.GetArg("terms")?.Trim();
+ if (string.IsNullOrWhiteSpace(terms))
+ return;
+ await e.Channel.SendMessage($"https://google.com/search?q={ HttpUtility.UrlEncode(terms).Replace(' ', '+') }")
+ .ConfigureAwait(false);
+ });
+
cgb.CreateCommand(Prefix + "hs")
.Description("Searches for a Hearthstone card and shows its image. Takes a while to complete. |~hs Ysera")
.Parameter("name", ParameterType.Unparsed)
diff --git a/NadekoBot/Modules/Translator/Helpers/GoogleTranslator.cs b/NadekoBot/Modules/Translator/Helpers/GoogleTranslator.cs
index 1743af45..b1a10a8c 100644
--- a/NadekoBot/Modules/Translator/Helpers/GoogleTranslator.cs
+++ b/NadekoBot/Modules/Translator/Helpers/GoogleTranslator.cs
@@ -60,7 +60,7 @@ namespace NadekoBot.Modules.Translator.Helpers
text = await http.GetStringAsync(url).ConfigureAwait(false);
}
- return JArray.Parse(text)[0][0][0].ToString();
+ return (string.Concat(JArray.Parse(text)[0].Select(x => x[0])));
}
#endregion
diff --git a/NadekoBot/Modules/Utility/UtilityModule.cs b/NadekoBot/Modules/Utility/UtilityModule.cs
index c85348aa..380e5acf 100644
--- a/NadekoBot/Modules/Utility/UtilityModule.cs
+++ b/NadekoBot/Modules/Utility/UtilityModule.cs
@@ -48,7 +48,7 @@ namespace NadekoBot.Modules.Utility
if (arr.Length == 0)
await e.Channel.SendMessage("Nobody. (not 100% sure)").ConfigureAwait(false);
else
- await e.Channel.SendMessage("```xl\n" + string.Join("\n", arr.GroupBy(item => (i++) / 3).Select(ig => string.Join("", ig.Select(el => $" {el,-35}")))) + "\n```").ConfigureAwait(false);
+ await e.Channel.SendMessage("```xl\n" + string.Join("\n", arr.GroupBy(item => (i++) / 3).Select(ig => string.Concat(ig.Select(el => $" {el,-35}")))) + "\n```").ConfigureAwait(false);
});
cgb.CreateCommand(Prefix + "inrole")
diff --git a/NadekoBot/NadekoBot.cs b/NadekoBot/NadekoBot.cs
index 9a4b804c..4a091e3e 100644
--- a/NadekoBot/NadekoBot.cs
+++ b/NadekoBot/NadekoBot.cs
@@ -41,6 +41,7 @@ namespace NadekoBot
public static LocalizedStrings Locale { get; set; } = new LocalizedStrings();
public static string BotMention { get; set; } = "";
public static bool Ready { get; set; } = false;
+ public static Action OnReady { get; set; } = delegate { };
private static List OwnerPrivateChannels { get; set; }
@@ -196,7 +197,7 @@ namespace NadekoBot
return;
}
#if NADEKO_RELEASE
- await Task.Delay(120000).ConfigureAwait(false);
+ await Task.Delay(90000).ConfigureAwait(false);
#else
await Task.Delay(1000).ConfigureAwait(false);
#endif
@@ -229,6 +230,7 @@ namespace NadekoBot
};
PermissionsHandler.Initialize();
NadekoBot.Ready = true;
+ NadekoBot.OnReady();
});
Console.WriteLine("Exiting...");
Console.ReadKey();
@@ -277,5 +279,3 @@ namespace NadekoBot
}
}
}
-
-//95520984584429568 meany
diff --git a/NadekoBot/NadekoBot.csproj b/NadekoBot/NadekoBot.csproj
index 09bc70ea..cd467b7f 100644
--- a/NadekoBot/NadekoBot.csproj
+++ b/NadekoBot/NadekoBot.csproj
@@ -40,10 +40,12 @@
full
false
bin\Debug\
- TRACE;DEBUG;__DEMO__,__DEMO_EXPERIMENTAL__
+ TRACE;DEBUG
prompt
4
true
+
+
AnyCPU
@@ -74,6 +76,46 @@
MinimumRecommendedRules.ruleset
true
+
+ true
+ bin\x64\Debug\
+ TRACE;DEBUG
+ full
+ x64
+ prompt
+ MinimumRecommendedRules.ruleset
+ true
+
+
+ bin\x64\Release\
+ TRACE
+ true
+ pdbonly
+ x64
+ prompt
+ MinimumRecommendedRules.ruleset
+ true
+
+
+ true
+ bin\x64\PRIVATE\
+ DEBUG;TRACE
+ full
+ x64
+ prompt
+ MinimumRecommendedRules.ruleset
+ true
+
+
+ bin\Release\
+ TRACE;NADEKO_RELEASE
+ true
+ pdbonly
+ x64
+ prompt
+ MinimumRecommendedRules.ruleset
+ true
+
..\packages\VideoLibrary.1.3.3\lib\portable-net45+win+wpa81+MonoAndroid10+xamarinios10+MonoTouch10\libvideo.dll
diff --git a/NadekoBot/_Models/JSONModels/Configuration.cs b/NadekoBot/_Models/JSONModels/Configuration.cs
index 93f9ebb9..295134fe 100644
--- a/NadekoBot/_Models/JSONModels/Configuration.cs
+++ b/NadekoBot/_Models/JSONModels/Configuration.cs
@@ -175,7 +175,7 @@ Nadeko Support Server: ";
public string Conversations { get; set; } = "<@{0}>";
public string ClashOfClans { get; set; } = ",";
public string Help { get; set; } = "-";
- public string Music { get; set; } = "!m";
+ public string Music { get; set; } = "!!";
public string Trello { get; set; } = "trello ";
public string Games { get; set; } = ">";
public string Gambling { get; set; } = "$";
diff --git a/NadekoBot/_Models/JSONModels/_JSONModels.cs b/NadekoBot/_Models/JSONModels/_JSONModels.cs
index ae7241bd..c60d32d8 100644
--- a/NadekoBot/_Models/JSONModels/_JSONModels.cs
+++ b/NadekoBot/_Models/JSONModels/_JSONModels.cs
@@ -7,7 +7,7 @@ namespace NadekoBot.Classes.JSONModels
public class Credentials
{
public string Token { get; set; } = "";
- public string ClientId { get; set; } = "116275390695079945";
+ public string ClientId { get; set; } = "170254782546575360";
public ulong BotId { get; set; } = 1231231231231;
public ulong[] OwnerIds { get; set; } = { 123123123123, 5675675679845 };
public string GoogleAPIKey { get; set; } = "";
diff --git a/NadekoBot/bin/Debug/credentials_example.json b/NadekoBot/bin/Debug/credentials_example.json
index 245b1141..e5c9f2dd 100644
--- a/NadekoBot/bin/Debug/credentials_example.json
+++ b/NadekoBot/bin/Debug/credentials_example.json
@@ -1,6 +1,6 @@
{
"Token": "",
- "ClientId": "116275390695079945",
+ "ClientId": "170254782546575360",
"BotId": 1231231231231,
"OwnerIds": [
123123123123,
diff --git a/NadekoBot/bin/Debug/data/config_example.json b/NadekoBot/bin/Debug/data/config_example.json
index 469aa4c0..1ae8a26e 100644
--- a/NadekoBot/bin/Debug/data/config_example.json
+++ b/NadekoBot/bin/Debug/data/config_example.json
@@ -83,7 +83,7 @@
"Conversations": "<@{0}>",
"ClashOfClans": ",",
"Help": "-",
- "Music": "!m",
+ "Music": "!!",
"Trello": "trello ",
"Games": ">",
"Gambling": "$",
diff --git a/commandlist.md b/commandlist.md
index 3a9cce39..50cb13cc 100644
--- a/commandlist.md
+++ b/commandlist.md
@@ -1,8 +1,8 @@
######For more information and how to setup your own NadekoBot, go to: **http://github.com/Kwoth/NadekoBot/**
-######You can donate on paypal: `nadekodiscordbot@gmail.com` or Bitcoin `17MZz1JAqME39akMLrVT4XBPffQJ2n1EPa`
+######You can donate on paypal: `nadekodiscordbot@gmail.com`
#NadekoBot List Of Commands
-Version: `NadekoBot v0.9.6036.32870`
+Version: `NadekoBot v0.9.6048.2992`
### Help
Command and aliases | Description | Usage
----------------|--------------|-------
@@ -25,6 +25,7 @@ Command and aliases | Description | Usage
`.greetpm` | Toggles whether the greet messages will be sent in a PM or in the text channel.
`.spmom` | Toggles whether mentions of other offline users on your server will send a pm to them.
`.logserver` | Toggles logging in this channel. Logs every message sent/deleted/edited on the server. **Bot Owner Only!**
+`.logignore` | Toggles whether the .logserver command ignores this channel. Useful if you have hidden admin channel and public log channel.
`.userpresence` | Starts logging to this channel when someone from the server goes online/offline/idle.
`.voicepresence` | Toggles logging to this channel whenever someone joins or leaves a voice channel you are in right now.
`.repeatinvoke`, `.repinv` | Immediately shows the repeat message and restarts the timer.
@@ -42,6 +43,7 @@ Command and aliases | Description | Usage
`.asar` | Adds a role, or list of roles separated by whitespace(use quotations for multiword roles) to the list of self-assignable roles. | .asar Gamer
`.rsar` | Removes a specified role from the list of self-assignable roles.
`.lsar` | Lists all self-assignable roles.
+`.togglexclsar`, `.tesar` | toggle whether the self-assigned roles should be exclusive
`.iam` | Adds a role to you that you choose. Role must be on a list of self-assignable roles. | .iam Gamer
`.iamnot`, `.iamn` | Removes a role to you that you choose. Role must be on a list of self-assignable roles. | .iamn Gamer
`.addcustreact`, `.acr` | Add a custom reaction. Guide here: **Bot Owner Only!** | .acr "hello" I love saying hello to %user%
@@ -86,7 +88,6 @@ Command and aliases | Description | Usage
`.donators` | List of lovely people who donated to keep this project alive.
`.donadd` | Add a donator to the database.
`.announce` | Sends a message to all servers' general channel bot is connected to.**Bot Owner Only!** | .announce Useless spam
-`.leave` | Leaves a server with a supplied ID. | `.leave 493243292839`
`.savechat` | Saves a number of messages to a text file and sends it to you. **Bot Owner Only** | `.chatsave 150`
### Utility
@@ -145,6 +146,8 @@ Command and aliases | Description | Usage
`;cbl` | Blacklists a mentioned channel (#general for example). | ;cbl #some_channel
`;cubl` | Unblacklists a mentioned channel (#general for example). | ;cubl #some_channel
`;sbl` | Blacklists a server by a name or id (#general for example). **BOT OWNER ONLY** | ;sbl [servername/serverid]
+`;cmdcooldown`, `;cmdcd` | Sets a cooldown per user for a command. Set 0 to clear. | `;cmdcd "some cmd" 5`
+`;allcmdcooldowns`, `;acmdcds` | Shows a list of all commands and their respective cooldowns.
### Conversations
Command and aliases | Description | Usage
@@ -157,7 +160,6 @@ Command and aliases | Description | Usage
`@BotName do you love me` | Replies with positive answer only to the bot owner.
`@BotName how are you`, `@BotName how are you?` | Replies positive only if bot owner is online.
`@BotName fire` | Shows a unicode fire message. Optional parameter [x] tells her how many times to repeat the fire. | @NadekoBot fire [x]
-`@BotName slm` | Shows the message where you were last mentioned in this channel (checks last 10k messages)
`@BotName dump` | Dumps all of the invites it can to dump.txt.** Owner Only.**
`@BotName ab` | Try to get 'abalabahaha'
@@ -167,13 +169,16 @@ Command and aliases | Description | Usage
`$draw` | Draws a card from the deck.If you supply number [x], she draws up to 5 cards from the deck. | $draw [x]
`$shuffle`, `$sh` | Reshuffles all cards back into the deck.
`$flip` | Flips coin(s) - heads or tails, and shows an image. | `$flip` or `$flip 3`
+`$betflip`, `$bf` | Bet to guess will the result be heads or tails. Guessing award you double flowers you've bet. | `$bf 5 heads` or `$bf 3 t`
`$roll` | Rolls 0-100. If you supply a number [x] it rolls up to 30 normal dice. If you split 2 numbers with letter d (xdy) it will roll x dice from 1 to y. | $roll or $roll 7 or $roll 3d5
+`$rolluo` | Rolls 0-100. If you supply a number [x] it rolls up to 30 normal dice (unordered). If you split 2 numbers with letter d (xdy) it will roll x dice from 1 to y. | $roll or $roll 7 or $roll 3d5
`$nroll` | Rolls in a given range. | `$nroll 5` (rolls 0-5) or `$nroll 5-15`
`$raffle` | Prints a name and ID of a random user from the online list from the (optional) role.
`$$$` | Check how much NadekoFlowers a person has. (Defaults to yourself) | `$$$` or `$$$ @Someone`
`$give` | Give someone a certain amount of NadekoFlowers
`$award` | Gives someone a certain amount of flowers. **Bot Owner Only!** | `$award 100 @person`
`$take` | Takes a certain amount of flowers from someone. **Bot Owner Only!**
+`$betroll`, `$br` | Bets a certain amount of NadekoFlowers and rolls a dice. Rolling over 66 yields x2 flowers, over 90 - x3 and 100 x10. | $br 5
`$leaderboard`, `$lb` |
### Games
@@ -199,39 +204,39 @@ Command and aliases | Description | Usage
### Music
Command and aliases | Description | Usage
----------------|--------------|-------
-`!m next`, `!m n`, `!m skip` | Goes to the next song in the queue. You have to be in the same voice channel as the bot. | `!m n`
-`!m stop`, `!m s` | Stops the music and clears the playlist. Stays in the channel. | `!m s`
-`!m destroy`, `!m d` | Completely stops the music and unbinds the bot from the channel. (may cause weird behaviour) | `!m d`
-`!m pause`, `!m p` | Pauses or Unpauses the song. | `!m p`
-`!m queue`, `!m q`, `!m yq` | Queue a song using keywords or a link. Bot will join your voice channel.**You must be in a voice channel**. | `!m q Dream Of Venice`
-`!m soundcloudqueue`, `!m sq` | Queue a soundcloud song using keywords. Bot will join your voice channel.**You must be in a voice channel**. | `!m sq Dream Of Venice`
-`!m listqueue`, `!m lq` | Lists 15 currently queued songs per page. Default page is 1. | `!m lq` or `!m lq 2`
-`!m nowplaying`, `!m np` | Shows the song currently playing. | `!m np`
-`!m volume`, `!m vol` | Sets the music volume 0-100% | `!m vol 50`
-`!m defvol`, `!m dv` | Sets the default music volume when music playback is started (0-100). Persists through restarts. | `!m dv 80`
-`!m mute`, `!m min` | Sets the music volume to 0% | `!m min`
-`!m max` | Sets the music volume to 100%. | `!m max`
-`!m half` | Sets the music volume to 50%. | `!m half`
-`!m shuffle`, `!m sh` | Shuffles the current playlist. | `!m sh`
-`!m playlist`, `!m pl` | Queues up to 500 songs from a youtube playlist specified by a link, or keywords. | `!m pl playlist link or name`
-`!m soundcloudpl`, `!m scpl` | Queue a soundcloud playlist using a link. | `!m scpl https://soundcloud.com/saratology/sets/symphony`
-`!m localplaylst`, `!m lopl` | Queues all songs from a directory. **Bot Owner Only!** | `!m lopl C:/music/classical`
-`!m radio`, `!m ra` | Queues a radio stream from a link. It can be a direct mp3 radio stream, .m3u, .pls .asx or .xspf | `!m ra radio link here`
-`!m local`, `!m lo` | Queues a local file by specifying a full path. **Bot Owner Only!** | `!m lo C:/music/mysong.mp3`
-`!m move`, `!m mv` | Moves the bot to your voice channel. (works only if music is already playing) | `!m mv`
-`!m remove`, `!m rm` | Remove a song by its # in the queue, or 'all' to remove whole queue. | `!m rm 5`
-`!m movesong`, `!m ms` | Moves a song from one position to another. | `!m ms` 5>3
-`!m setmaxqueue`, `!m smq` | Sets a maximum queue size. Supply 0 or no argument to have no limit. | `!m smq` 50 or `!m smq`
-`!m cleanup` | Cleans up hanging voice connections. **Bot Owner Only!** | `!m cleanup`
-`!m reptcursong`, `!m rcs` | Toggles repeat of current song. | `!m rcs`
-`!m rpeatplaylst`, `!m rpl` | Toggles repeat of all songs in the queue (every song that finishes is added to the end of the queue). | `!m rpl`
-`!m save` | Saves a playlist under a certain name. Name must be no longer than 20 characters and mustn't contain dashes. | `!m save classical1`
-`!m load` | Loads a playlist under a certain name. | `!m load classical-1`
-`!m playlists`, `!m pls` | Lists all playlists. Paginated. 20 per page. Default page is 0. | `!m pls 1`
-`!m deleteplaylist`, `!m delpls` | Deletes a saved playlist. Only if you made it or if you are the bot owner. | `!m delpls animu-5`
-`!m goto` | Goes to a specific time in seconds in a song.
-`!m getlink`, `!m gl` | Shows a link to the currently playing song.
-`!m autoplay`, `!m ap` | Toggles autoplay - When the song is finished, automatically queue a related youtube song. (Works only for youtube songs and when queue is empty)
+`!!next`, `!!n`, `!!skip` | Goes to the next song in the queue. You have to be in the same voice channel as the bot. | `!!n`
+`!!stop`, `!!s` | Stops the music and clears the playlist. Stays in the channel. | `!!s`
+`!!destroy`, `!!d` | Completely stops the music and unbinds the bot from the channel. (may cause weird behaviour) | `!!d`
+`!!pause`, `!!p` | Pauses or Unpauses the song. | `!!p`
+`!!queue`, `!!q`, `!!yq` | Queue a song using keywords or a link. Bot will join your voice channel.**You must be in a voice channel**. | `!!q Dream Of Venice`
+`!!soundcloudqueue`, `!!sq` | Queue a soundcloud song using keywords. Bot will join your voice channel.**You must be in a voice channel**. | `!!sq Dream Of Venice`
+`!!listqueue`, `!!lq` | Lists 15 currently queued songs per page. Default page is 1. | `!!lq` or `!!lq 2`
+`!!nowplaying`, `!!np` | Shows the song currently playing. | `!!np`
+`!!volume`, `!!vol` | Sets the music volume 0-100% | `!!vol 50`
+`!!defvol`, `!!dv` | Sets the default music volume when music playback is started (0-100). Persists through restarts. | `!!dv 80`
+`!!mute`, `!!min` | Sets the music volume to 0% | `!!min`
+`!!max` | Sets the music volume to 100%. | `!!max`
+`!!half` | Sets the music volume to 50%. | `!!half`
+`!!shuffle`, `!!sh` | Shuffles the current playlist. | `!!sh`
+`!!playlist`, `!!pl` | Queues up to 500 songs from a youtube playlist specified by a link, or keywords. | `!!pl playlist link or name`
+`!!soundcloudpl`, `!!scpl` | Queue a soundcloud playlist using a link. | `!!scpl https://soundcloud.com/saratology/sets/symphony`
+`!!localplaylst`, `!!lopl` | Queues all songs from a directory. **Bot Owner Only!** | `!!lopl C:/music/classical`
+`!!radio`, `!!ra` | Queues a radio stream from a link. It can be a direct mp3 radio stream, .m3u, .pls .asx or .xspf (Usage Video: ) | `!!ra radio link here`
+`!!local`, `!!lo` | Queues a local file by specifying a full path. **Bot Owner Only!** | `!!lo C:/music/mysong.mp3`
+`!!move`, `!!mv` | Moves the bot to your voice channel. (works only if music is already playing) | `!!mv`
+`!!remove`, `!!rm` | Remove a song by its # in the queue, or 'all' to remove whole queue. | `!!rm 5`
+`!!movesong`, `!!ms` | Moves a song from one position to another. | `!! ms` 5>3
+`!!setmaxqueue`, `!!smq` | Sets a maximum queue size. Supply 0 or no argument to have no limit. | `!!smq` 50 or `!!smq`
+`!!cleanup` | Cleans up hanging voice connections. **Bot Owner Only!** | `!!cleanup`
+`!!reptcursong`, `!!rcs` | Toggles repeat of current song. | `!!rcs`
+`!!rpeatplaylst`, `!!rpl` | Toggles repeat of all songs in the queue (every song that finishes is added to the end of the queue). | `!!rpl`
+`!!save` | Saves a playlist under a certain name. Name must be no longer than 20 characters and mustn't contain dashes. | `!!save classical1`
+`!!load` | Loads a playlist under a certain name. | `!!load classical-1`
+`!!playlists`, `!!pls` | Lists all playlists. Paginated. 20 per page. Default page is 0. | `!!pls 1`
+`!!deleteplaylist`, `!!delpls` | Deletes a saved playlist. Only if you made it or if you are the bot owner. | `!!delpls animu-5`
+`!!goto` | Goes to a specific time in seconds in a song.
+`!!getlink`, `!!gl` | Shows a link to the currently playing song.
+`!!autoplay`, `!!ap` | Toggles autoplay - When the song is finished, automatically queue a related youtube song. (Works only for youtube songs and when queue is empty)
### Searches
Command and aliases | Description | Usage
@@ -266,6 +271,7 @@ Command and aliases | Description | Usage
`~i` | Pulls the first image found using a search parameter. Use ~ir for different results. | ~i cute kitten
`~ir` | Pulls a random image using a search parameter. | ~ir cute kitten
`~lmgtfy` | Google something for an idiot.
+`~google`, `~g` | Get a google search link for some terms.
`~hs` | Searches for a Hearthstone card and shows its image. Takes a while to complete. | ~hs Ysera
`~ud` | Searches Urban Dictionary for a word. | ~ud Pineapple
`~#` | Searches Tagdef.com for a hashtag. | ~# ff
@@ -313,7 +319,7 @@ Command and aliases | Description | Usage
----------------|--------------|-------
`>attack` | Attacks a target with the given move. Use `>movelist` to see a list of moves your type can use. | `>attack "vine whip" @someguy`
`>movelist`, `>ml` | Lists the moves you are able to use
-`>heal` | Heals someone. Revives those that fainted. Costs a NadekoFlower | >revive @someone
+`>heal` | Heals someone. Revives those who fainted. Costs a NadekoFlower | >heal @someone
`>type` | Get the poketype of the target. | >type @someone
`>settype` | Set your poketype. Costs a NadekoFlower. | >settype fire
@@ -331,16 +337,17 @@ Command and aliases | Description | Usage
`moveto` | Custom reaction. | moveto
`comeatmebro` | Custom reaction. | comeatmebro
`e` | Custom reaction. | e
-`@BotName insult`, `<@!119777021319577610> insult` | Custom reaction. | %mention% insult
-`@BotName praise`, `<@!119777021319577610> praise` | Custom reaction. | %mention% praise
-`@BotName pat`, `<@!119777021319577610> pat` | Custom reaction. | %mention% pat
-`@BotName cry`, `<@!119777021319577610> cry` | Custom reaction. | %mention% cry
-`@BotName are you real?`, `<@!119777021319577610> are you real?` | Custom reaction. | %mention% are you real?
-`@BotName are you there?`, `<@!119777021319577610> are you there?` | Custom reaction. | %mention% are you there?
-`@BotName draw`, `<@!119777021319577610> draw` | Custom reaction. | %mention% draw
-`@BotName bb`, `<@!119777021319577610> bb` | Custom reaction. | %mention% bb
-`@BotName call`, `<@!119777021319577610> call` | Custom reaction. | %mention% call
-`@BotName disguise`, `<@!119777021319577610> disguise` | Custom reaction. | %mention% disguise
+`@BotName insult`, `<@!116275390695079945> insult` | Custom reaction. | %mention% insult
+`@BotName praise`, `<@!116275390695079945> praise` | Custom reaction. | %mention% praise
+`@BotName pat`, `<@!116275390695079945> pat` | Custom reaction. | %mention% pat
+`@BotName cry`, `<@!116275390695079945> cry` | Custom reaction. | %mention% cry
+`@BotName are you real?`, `<@!116275390695079945> are you real?` | Custom reaction. | %mention% are you real?
+`@BotName are you there?`, `<@!116275390695079945> are you there?` | Custom reaction. | %mention% are you there?
+`@BotName draw`, `<@!116275390695079945> draw` | Custom reaction. | %mention% draw
+`@BotName bb`, `<@!116275390695079945> bb` | Custom reaction. | %mention% bb
+`@BotName call`, `<@!116275390695079945> call` | Custom reaction. | %mention% call
+`@BotName disguise`, `<@!116275390695079945> disguise` | Custom reaction. | %mention% disguise
+`~hentai` | Custom reaction. | ~hentai
### Trello
Command and aliases | Description | Usage
diff --git a/discord.net b/discord.net
index 80f9d6f2..3e519b5e 160000
--- a/discord.net
+++ b/discord.net
@@ -1 +1 @@
-Subproject commit 80f9d6f2de25355a245bf93d4019d16e3fd033ca
+Subproject commit 3e519b5e0b33175e5a5ca247322b7082de484e15