mirror of
https://gitlab.com/Kwoth/nadekobot.git
synced 2025-09-12 18:28:27 -04:00
Compare commits
75 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
f1c7d7437a | ||
|
9a013db25f | ||
|
6f75161c80 | ||
|
3f56e5b651 | ||
|
4875abbe86 | ||
|
83863f3a6c | ||
|
f07abad1ec | ||
|
b6909a4120 | ||
|
aaf7f04216 | ||
|
628871b0da | ||
|
a50ad09c8d | ||
|
1358ff50a4 | ||
|
ee9fede3b1 | ||
|
1d7e9e8471 | ||
|
d2fe7f8d11 | ||
|
75958efe17 | ||
|
7746d2aca1 | ||
|
e631df3326 | ||
|
ba44fdb55f | ||
|
59a1e56dad | ||
|
cd6fe46c2b | ||
|
fb4f470b44 | ||
|
b69e25edf4 | ||
|
9eae27bc15 | ||
|
bed36f8784 | ||
|
623f5ccd5e | ||
|
6da8201c2d | ||
|
d1c24d4721 | ||
|
7e5055268a | ||
|
e84e33b94f | ||
|
f21c96bdf4 | ||
|
430daf9b19 | ||
|
7c8756096d | ||
|
7d9e456fe5 | ||
|
948db31384 | ||
|
df1a775fec | ||
|
6c169e057b | ||
|
b164da38e3 | ||
|
dc229ea2b3 | ||
|
b853495d65 | ||
|
0ff02cede9 | ||
|
f7e0e635e6 | ||
|
a065189023 | ||
|
1affeb0683 | ||
|
6b3c9f01ca | ||
|
e9eb6ff2ad | ||
|
0170536d1c | ||
|
c65e769128 | ||
|
998779203a | ||
|
e0e4d697c3 | ||
|
e9c7293014 | ||
|
cf876a4148 | ||
|
6b9a858f28 | ||
|
38e3badb87 | ||
|
13d2fbd560 | ||
|
8d3f2f186a | ||
|
9572b9dc43 | ||
|
ca32086089 | ||
|
57f839dbcd | ||
|
71d6eeb9dd | ||
|
8c51cf8537 | ||
|
b683026cf3 | ||
|
bca2bc5af1 | ||
|
b385a83bdd | ||
|
3bf0286c81 | ||
|
98272f66e7 | ||
|
cf3788c6ea | ||
|
4b3fc53cb6 | ||
|
4e17dca856 | ||
|
82d89148f3 | ||
|
cc4c09b4d7 | ||
|
616f01f8b2 | ||
|
56f89a02bc | ||
|
48ce988d20 | ||
|
119b1cdec2 |
77
CHANGELOG.md
77
CHANGELOG.md
@@ -2,11 +2,73 @@
|
|||||||
|
|
||||||
Experimental changelog. Mostly based on [keepachangelog](https://keepachangelog.com/en/1.0.0/) except date format. a-c-f-r-o
|
Experimental changelog. Mostly based on [keepachangelog](https://keepachangelog.com/en/1.0.0/) except date format. a-c-f-r-o
|
||||||
|
|
||||||
## [4.3.1] - 27.07.2022
|
## [4.3.6] - 08.09.2022
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Added `.expraddserver` (.exas) which will server as a server-only alternative to '.exa' in case users want to override default Admin permissions with .dpo
|
||||||
|
- Added .banprune command which sets how many days worth of messages will be pruned when bot (soft)bans a person either through a command or another punishment feature.
|
||||||
|
- Added .qdelauth - Delete all quotes by the specified author on this server. If you target yourself - no permission required
|
||||||
|
- Added `.timeout` command
|
||||||
|
- Added an option to award currency based on received xp
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|
||||||
- Check for updates will run once per hour as it was supposed to
|
- Reminders now have embed support, but plaintext field is not supported.
|
||||||
|
- User friendlier errors when parsing a number in a command fails
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Awarded xp is now correctly used in level up calculations
|
||||||
|
|
||||||
|
## [4.3.5] - 17.08.2022
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Added a 'Use' button when a user already owns an item
|
||||||
|
- Added a 'Pull Again' button to slots
|
||||||
|
- Added `.roleinfo` command
|
||||||
|
- Added `.emojiremove` command
|
||||||
|
- Added `.threadcreate` and `.threaddelete` commands
|
||||||
|
- Added `.bank seize` / `.bank award` owner only commands
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Running a .timely command early now shows a pending color
|
||||||
|
- .xp system is once again no longer opt in for servers
|
||||||
|
- It's still opt-in for global and requires users to run .xp at least once in order to start gaining global xp
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Fixed users not getting club xp
|
||||||
|
|
||||||
|
## [4.3.4] - 07.08.2022
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Fixed users getting XP out of nowhere while voice xp is enabled
|
||||||
|
|
||||||
|
## [4.3.3] - 06.08.2022
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Added `betroll` option to `.bettest` command
|
||||||
|
- Added `.xpshopbuy` and `.xpshopuse` convenience commands
|
||||||
|
- Added an optional preview url to teh xp shop item config model which will be shown instead of the real Url
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Updated position of Username and Club name on the .xp card
|
||||||
|
- Improved text visibility on the .xp card
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Possibly fixed .trivia not stopping bug
|
||||||
|
- Fixed very low payout rate on `.betroll`
|
||||||
|
- Fixed an issue with youtube song resolver which caused invalid data to be cached
|
||||||
|
- Added client id to the cache key as a potential fix for VoiceXp 'bug'. The solution may be to use different redis instances for each bot, or to switch from botCache: from 'redis' to 'memory' in creds.yml
|
||||||
|
- Bot owner should now be able to buy items from the xpshop when patron requirement is set
|
||||||
|
- Fixed youtube-dl caching invalid data. Please use yt-dlp instead
|
||||||
|
|
||||||
## [4.3.2] - 28.07.2022
|
## [4.3.2] - 28.07.2022
|
||||||
|
|
||||||
@@ -19,6 +81,12 @@ Experimental changelog. Mostly based on [keepachangelog](https://keepachangelog.
|
|||||||
- Fixed `.bank withdraw <expression>` will now correctly use bank amount for calculation
|
- Fixed `.bank withdraw <expression>` will now correctly use bank amount for calculation
|
||||||
- [dev] Fixed medusa Reply*LocalizedAsync not working with placeholders
|
- [dev] Fixed medusa Reply*LocalizedAsync not working with placeholders
|
||||||
|
|
||||||
|
## [4.3.1] - 27.07.2022
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Check for updates will run once per hour as it was supposed to
|
||||||
|
|
||||||
## [4.3.0] - 27.07.2022
|
## [4.3.0] - 27.07.2022
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
@@ -32,9 +100,10 @@ Experimental changelog. Mostly based on [keepachangelog](https://keepachangelog.
|
|||||||
- They payouts are very good, but seven always loses
|
- They payouts are very good, but seven always loses
|
||||||
- Added `.lula` command. Plays the same as `.wof` but looks much nicer, and is easily customizable from gambling.yml without any changes to the sourcecode needed.
|
- Added `.lula` command. Plays the same as `.wof` but looks much nicer, and is easily customizable from gambling.yml without any changes to the sourcecode needed.
|
||||||
- Added `.repeatskip` command which makes the next repeat trigger not post anything
|
- Added `.repeatskip` command which makes the next repeat trigger not post anything
|
||||||
- Added `.imageonly` which will make the bot only allow link posts in the channel. Exclusive with `.imageonly`
|
- Added `.linkonly` which will make the bot only allow link posts in the channel. Exclusive with `.imageonly`
|
||||||
- Added release notifications. Bot owners will now receive new release notifications in dms if they have `checkForUpdates` set to `true` in data/bot.yml
|
- Added release notifications. Bot owners will now receive new release notifications in dms if they have `checkForUpdates` set to `true` in data/bot.yml
|
||||||
- You can also configure it via `.conf bot checkforupdates <true/false>`
|
- You can also configure it via `.conf bot checkfor
|
||||||
|
- updates <true/false>`
|
||||||
- Added `.xpshop` which lets bot owners add xp backgrounds and xp frames for sale by configuring `data/xp.yml`
|
- Added `.xpshop` which lets bot owners add xp backgrounds and xp frames for sale by configuring `data/xp.yml`
|
||||||
- You can also toggle xpshop feature via `.conf xp shop.is_enabled`
|
- You can also toggle xpshop feature via `.conf xp shop.is_enabled`
|
||||||
|
|
||||||
|
@@ -1,8 +1,3 @@
|
|||||||
[](https://discord.gg/nadekobot)
|
|
||||||
[](http://nadekobot.readthedocs.io/en/v4/?badge=v4)
|
|
||||||
[](https://top.gg/bot/116275390695079945)
|
|
||||||
|
|
||||||
|
|
||||||
[](https://nadeko.bot/)
|
[](https://nadeko.bot/)
|
||||||
|
|
||||||
[](https://invite.nadeko.bot/)
|
[](https://invite.nadeko.bot/)
|
||||||
|
@@ -33,14 +33,14 @@ These are required for a number of features to function properly, and all should
|
|||||||
For a single owner, it should look like this:
|
For a single owner, it should look like this:
|
||||||
|
|
||||||
```yml
|
```yml
|
||||||
OwnerIds:
|
OwnerIds:
|
||||||
- 105635576866156544
|
- 105635576866156544
|
||||||
```
|
```
|
||||||
|
|
||||||
For multiple owners, it should look like this:
|
For multiple owners, it should look like this:
|
||||||
|
|
||||||
```yml
|
```yml
|
||||||
OwnerIds:
|
OwnerIds:
|
||||||
- 105635123466156544
|
- 105635123466156544
|
||||||
- 145521851676884992
|
- 145521851676884992
|
||||||
- 341420590009417729
|
- 341420590009417729
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
# Setting up NadekoBot on Linux
|
w# Setting up NadekoBot on Linux
|
||||||
|
|
||||||
| Table of Contents |
|
| Table of Contents |
|
||||||
| :-------------------------------------------------- |
|
| :-------------------------------------------------- |
|
||||||
@@ -69,7 +69,7 @@ Open Terminal (if you're on an installation with a window manager) and navigate
|
|||||||
|
|
||||||
###### Prerequisites
|
###### Prerequisites
|
||||||
|
|
||||||
1. Nadeko requires redis to function
|
1. (Optional) Installing Redis
|
||||||
- ubuntu installation command: `sudo apt-get install redis-server`
|
- ubuntu installation command: `sudo apt-get install redis-server`
|
||||||
2. Playing music requires `ffmpeg`, `libopus`, `libsodium` and `youtube-dl` (which in turn requires python3)
|
2. Playing music requires `ffmpeg`, `libopus`, `libsodium` and `youtube-dl` (which in turn requires python3)
|
||||||
- ubuntu installation command: `sudo apt-get install ffmpeg libopus0 opus-tools libopus-dev libsodium-dev -y`
|
- ubuntu installation command: `sudo apt-get install ffmpeg libopus0 opus-tools libopus-dev libsodium-dev -y`
|
||||||
@@ -160,15 +160,30 @@ If you are presented with the installer main menu, exit it by choosing Option `8
|
|||||||
|
|
||||||
The above command will create a new session named **nadeko** *(you can replace “nadeko” with anything you prefer, it's your session name)*.
|
The above command will create a new session named **nadeko** *(you can replace “nadeko” with anything you prefer, it's your session name)*.
|
||||||
|
|
||||||
2. Navigate to the project's root directory
|
2. Run the installer: `bash linuxAIO.sh`
|
||||||
- Project root directory location example: `cd /home/user/nadekobot/`
|
|
||||||
3. Enter the `output` directory:
|
3. There are a few options when it comes to running Nadeko.
|
||||||
- `cd output`
|
|
||||||
4. Run the bot using:
|
- Run `3` to *Run the bot normally*
|
||||||
- `dotnet NadekoBot.dll`
|
- Run `4` to *Run the bot with Auto Restart* (This is may or may not work)
|
||||||
5. Detatch the tmux session:
|
|
||||||
|
4. If option `4` was selected, you have the following options
|
||||||
|
```
|
||||||
|
1. Run Auto Restart normally without updating NadekoBot.
|
||||||
|
2. Run Auto Restart and update NadekoBot.
|
||||||
|
3. Exit
|
||||||
|
|
||||||
|
Choose:
|
||||||
|
[1] to Run NadekoBot with Auto Restart on "die" command without updating.
|
||||||
|
[2] to Run with Auto Updating on restart after using "die" command.
|
||||||
|
```
|
||||||
|
- Run `1` to update the bot upon restart. (This is done using the `.die` command)
|
||||||
|
- Run `2` to restart the bot without updating. (This is also done using the `.die` command)
|
||||||
|
|
||||||
|
5. That's it! to detatch the tmux session:
|
||||||
- Press `Ctrl` + `B`
|
- Press `Ctrl` + `B`
|
||||||
- Then press `D`
|
- Then press `D`
|
||||||
|
|
||||||
Now check your Discord server, the bot should be online. Nadeko should now be running in the background of your system.
|
Now check your Discord server, the bot should be online. Nadeko should now be running in the background of your system.
|
||||||
|
|
||||||
To re-open the tmux session to either update, restart, or whatever, execute `tmux a -t nadeko`. *(Make sure to replace "nadeko" with your session name. If you didn't change it, leave it as it.)*
|
To re-open the tmux session to either update, restart, or whatever, execute `tmux a -t nadeko`. *(Make sure to replace "nadeko" with your session name. If you didn't change it, leave it as it.)*
|
||||||
|
@@ -26,20 +26,6 @@ public class NadekoRandom : Random
|
|||||||
return Math.Abs(BitConverter.ToInt32(bytes, 0)) % maxValue;
|
return Math.Abs(BitConverter.ToInt32(bytes, 0)) % maxValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte Next(byte minValue, byte maxValue)
|
|
||||||
{
|
|
||||||
if (minValue > maxValue)
|
|
||||||
throw new ArgumentOutOfRangeException(nameof(maxValue));
|
|
||||||
|
|
||||||
if (minValue == maxValue)
|
|
||||||
return minValue;
|
|
||||||
|
|
||||||
var bytes = new byte[1];
|
|
||||||
_rng.GetBytes(bytes);
|
|
||||||
|
|
||||||
return (byte)((bytes[0] % (maxValue - minValue)) + minValue);
|
|
||||||
}
|
|
||||||
|
|
||||||
public override int Next(int minValue, int maxValue)
|
public override int Next(int minValue, int maxValue)
|
||||||
{
|
{
|
||||||
if (minValue > maxValue)
|
if (minValue > maxValue)
|
||||||
|
@@ -12,7 +12,7 @@ public sealed class BetflipGame
|
|||||||
|
|
||||||
public BetflipResult Flip(byte guess, decimal amount)
|
public BetflipResult Flip(byte guess, decimal amount)
|
||||||
{
|
{
|
||||||
var side = _rng.Next(0, 2);
|
var side = (byte)_rng.Next(0, 2);
|
||||||
if (side == guess)
|
if (side == guess)
|
||||||
{
|
{
|
||||||
return new BetflipResult()
|
return new BetflipResult()
|
||||||
|
@@ -13,7 +13,7 @@ public sealed class BetrollGame
|
|||||||
|
|
||||||
public BetrollResult Roll(decimal amount = 0)
|
public BetrollResult Roll(decimal amount = 0)
|
||||||
{
|
{
|
||||||
var roll = _rng.Next(0, 101);
|
var roll = _rng.Next(1, 101);
|
||||||
|
|
||||||
for (var i = 0; i < _thresholdPairs.Length; i++)
|
for (var i = 0; i < _thresholdPairs.Length; i++)
|
||||||
{
|
{
|
||||||
|
@@ -1,5 +1,8 @@
|
|||||||
namespace Nadeko.Econ.Gambling;
|
namespace Nadeko.Econ.Gambling;
|
||||||
|
|
||||||
|
//here is a payout chart
|
||||||
|
//https://lh6.googleusercontent.com/-i1hjAJy_kN4/UswKxmhrbPI/AAAAAAAAB1U/82wq_4ZZc-Y/DE6B0895-6FC1-48BE-AC4F-14D1B91AB75B.jpg
|
||||||
|
//thanks to judge for helping me with this
|
||||||
public class SlotGame
|
public class SlotGame
|
||||||
{
|
{
|
||||||
private static readonly NadekoRandom _rng = new NadekoRandom();
|
private static readonly NadekoRandom _rng = new NadekoRandom();
|
||||||
@@ -8,9 +11,9 @@ public class SlotGame
|
|||||||
{
|
{
|
||||||
var rolls = new[]
|
var rolls = new[]
|
||||||
{
|
{
|
||||||
_rng.Next(0, 6),
|
(byte)_rng.Next(0, 6),
|
||||||
_rng.Next(0, 6),
|
(byte)_rng.Next(0, 6),
|
||||||
_rng.Next(0, 6)
|
(byte)_rng.Next(0, 6)
|
||||||
};
|
};
|
||||||
|
|
||||||
ref var a = ref rolls[0];
|
ref var a = ref rolls[0];
|
||||||
|
@@ -10,6 +10,7 @@ public interface IEmbedBuilder
|
|||||||
IEmbedBuilder WithFooter(string text, string? iconUrl = null);
|
IEmbedBuilder WithFooter(string text, string? iconUrl = null);
|
||||||
IEmbedBuilder WithAuthor(string name, string? iconUrl = null, string? url = null);
|
IEmbedBuilder WithAuthor(string name, string? iconUrl = null, string? url = null);
|
||||||
IEmbedBuilder WithColor(EmbedColor color);
|
IEmbedBuilder WithColor(EmbedColor color);
|
||||||
|
IEmbedBuilder WithDiscordColor(Color color);
|
||||||
Embed Build();
|
Embed Build();
|
||||||
IEmbedBuilder WithUrl(string url);
|
IEmbedBuilder WithUrl(string url);
|
||||||
IEmbedBuilder WithImageUrl(string url);
|
IEmbedBuilder WithImageUrl(string url);
|
||||||
|
@@ -12,7 +12,7 @@
|
|||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<PackageReference Include="Discord.Net.Core" Version="3.103.0" />
|
<PackageReference Include="Discord.Net.Core" Version="3.104.0" />
|
||||||
<PackageReference Include="Serilog" Version="2.11.0" />
|
<PackageReference Include="Serilog" Version="2.11.0" />
|
||||||
<PackageReference Include="YamlDotNet" Version="11.2.1" />
|
<PackageReference Include="YamlDotNet" Version="11.2.1" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
@@ -3,14 +3,12 @@ using Microsoft.Extensions.DependencyInjection;
|
|||||||
using NadekoBot.Common.Configs;
|
using NadekoBot.Common.Configs;
|
||||||
using NadekoBot.Common.ModuleBehaviors;
|
using NadekoBot.Common.ModuleBehaviors;
|
||||||
using NadekoBot.Db;
|
using NadekoBot.Db;
|
||||||
using NadekoBot.Modules.Administration;
|
|
||||||
using NadekoBot.Modules.Utility;
|
using NadekoBot.Modules.Utility;
|
||||||
using NadekoBot.Services.Database.Models;
|
using NadekoBot.Services.Database.Models;
|
||||||
using System.Collections.Immutable;
|
using System.Collections.Immutable;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.Net;
|
using System.Net;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using Nadeko.Common;
|
|
||||||
using RunMode = Discord.Commands.RunMode;
|
using RunMode = Discord.Commands.RunMode;
|
||||||
|
|
||||||
namespace NadekoBot;
|
namespace NadekoBot;
|
||||||
|
@@ -0,0 +1,8 @@
|
|||||||
|
namespace NadekoBot;
|
||||||
|
|
||||||
|
public interface INadekoInteractionService
|
||||||
|
{
|
||||||
|
public NadekoInteraction Create<T>(
|
||||||
|
ulong userId,
|
||||||
|
SimpleInteraction<T> inter);
|
||||||
|
}
|
@@ -1,29 +0,0 @@
|
|||||||
namespace NadekoBot;
|
|
||||||
|
|
||||||
public sealed class NadekoButtonActionInteraction : NadekoButtonOwnInteraction
|
|
||||||
{
|
|
||||||
private readonly NadekoInteractionData _data;
|
|
||||||
private readonly Func<SocketMessageComponent, Task> _action;
|
|
||||||
|
|
||||||
public NadekoButtonActionInteraction(
|
|
||||||
DiscordSocketClient client,
|
|
||||||
ulong authorId,
|
|
||||||
NadekoInteractionData data,
|
|
||||||
Func<SocketMessageComponent, Task> action
|
|
||||||
)
|
|
||||||
: base(client, authorId)
|
|
||||||
{
|
|
||||||
_data = data;
|
|
||||||
_action = action;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override string Name
|
|
||||||
=> _data.CustomId;
|
|
||||||
protected override IEmote Emote
|
|
||||||
=> _data.Emote;
|
|
||||||
protected override string? Text
|
|
||||||
=> _data.Text;
|
|
||||||
|
|
||||||
public override Task ExecuteOnActionAsync(SocketMessageComponent smc)
|
|
||||||
=> _action(smc);
|
|
||||||
}
|
|
@@ -1,25 +1,30 @@
|
|||||||
namespace NadekoBot;
|
namespace NadekoBot;
|
||||||
|
|
||||||
public abstract class NadekoButtonInteraction
|
public sealed class NadekoInteraction
|
||||||
{
|
{
|
||||||
// improvements:
|
private readonly ulong _authorId;
|
||||||
// - state in OnAction
|
private readonly ButtonBuilder _button;
|
||||||
// - configurable delay
|
private readonly Func<SocketMessageComponent, Task> _onClick;
|
||||||
// -
|
private readonly bool _onlyAuthor;
|
||||||
protected abstract string Name { get; }
|
|
||||||
protected abstract IEmote Emote { get; }
|
|
||||||
protected virtual string? Text { get; } = null;
|
|
||||||
|
|
||||||
public DiscordSocketClient Client { get; }
|
public DiscordSocketClient Client { get; }
|
||||||
|
|
||||||
protected readonly TaskCompletionSource<bool> _interactionCompletedSource;
|
private readonly TaskCompletionSource<bool> _interactionCompletedSource;
|
||||||
|
|
||||||
protected IUserMessage message = null!;
|
private IUserMessage message = null!;
|
||||||
|
|
||||||
protected NadekoButtonInteraction(DiscordSocketClient client)
|
public NadekoInteraction(DiscordSocketClient client,
|
||||||
|
ulong authorId,
|
||||||
|
ButtonBuilder button,
|
||||||
|
Func<SocketMessageComponent, Task> onClick,
|
||||||
|
bool onlyAuthor)
|
||||||
{
|
{
|
||||||
Client = client;
|
_authorId = authorId;
|
||||||
|
_button = button;
|
||||||
|
_onClick = onClick;
|
||||||
|
_onlyAuthor = onlyAuthor;
|
||||||
_interactionCompletedSource = new(TaskCreationOptions.RunContinuationsAsynchronously);
|
_interactionCompletedSource = new(TaskCreationOptions.RunContinuationsAsynchronously);
|
||||||
|
|
||||||
|
Client = client;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task RunAsync(IUserMessage msg)
|
public async Task RunAsync(IUserMessage msg)
|
||||||
@@ -27,29 +32,25 @@ public abstract class NadekoButtonInteraction
|
|||||||
message = msg;
|
message = msg;
|
||||||
|
|
||||||
Client.InteractionCreated += OnInteraction;
|
Client.InteractionCreated += OnInteraction;
|
||||||
await Task.WhenAny(Task.Delay(10_000), _interactionCompletedSource.Task);
|
await Task.WhenAny(Task.Delay(15_000), _interactionCompletedSource.Task);
|
||||||
Client.InteractionCreated -= OnInteraction;
|
Client.InteractionCreated -= OnInteraction;
|
||||||
|
|
||||||
await msg.ModifyAsync(m => m.Components = new ComponentBuilder().Build());
|
await msg.ModifyAsync(m => m.Components = new ComponentBuilder().Build());
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract ValueTask<bool> Validate(SocketMessageComponent smc);
|
private Task OnInteraction(SocketInteraction arg)
|
||||||
private async Task OnInteraction(SocketInteraction arg)
|
|
||||||
{
|
{
|
||||||
if (arg is not SocketMessageComponent smc)
|
if (arg is not SocketMessageComponent smc)
|
||||||
return;
|
return Task.CompletedTask;
|
||||||
|
|
||||||
if (smc.Message.Id != message.Id)
|
if (smc.Message.Id != message.Id)
|
||||||
return;
|
return Task.CompletedTask;
|
||||||
|
|
||||||
if (smc.Data.CustomId != Name)
|
if (_onlyAuthor && smc.User.Id != _authorId)
|
||||||
return;
|
return Task.CompletedTask;
|
||||||
|
|
||||||
if (!await Validate(smc))
|
if (smc.Data.CustomId != _button.CustomId)
|
||||||
{
|
return Task.CompletedTask;
|
||||||
await smc.DeferAsync();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_ = Task.Run(async () =>
|
_ = Task.Run(async () =>
|
||||||
{
|
{
|
||||||
@@ -63,21 +64,19 @@ public abstract class NadekoButtonInteraction
|
|||||||
await smc.DeferAsync();
|
await smc.DeferAsync();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
return Task.CompletedTask;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public virtual MessageComponent CreateComponent()
|
public MessageComponent CreateComponent()
|
||||||
{
|
{
|
||||||
var comp = new ComponentBuilder()
|
var comp = new ComponentBuilder()
|
||||||
.WithButton(GetButtonBuilder());
|
.WithButton(_button);
|
||||||
|
|
||||||
return comp.Build();
|
return comp.Build();
|
||||||
}
|
}
|
||||||
|
|
||||||
public ButtonBuilder GetButtonBuilder()
|
public Task ExecuteOnActionAsync(SocketMessageComponent smc)
|
||||||
=> new ButtonBuilder(style: ButtonStyle.Secondary, emote: Emote, customId: Name, label: Text);
|
=> _onClick(smc);
|
||||||
|
|
||||||
public abstract Task ExecuteOnActionAsync(SocketMessageComponent smc);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// this is all so wrong ...
|
|
@@ -1,43 +0,0 @@
|
|||||||
// namespace NadekoBot;
|
|
||||||
//
|
|
||||||
// public class NadekoButtonInteractionArray : NadekoButtonInteraction
|
|
||||||
// {
|
|
||||||
// private readonly ButtonBuilder[] _bbs;
|
|
||||||
// private readonly NadekoButtonInteraction[] _inters;
|
|
||||||
//
|
|
||||||
// public NadekoButtonInteractionArray(params NadekoButtonInteraction[] inters)
|
|
||||||
// : base(inters[0].Client)
|
|
||||||
// {
|
|
||||||
// _inters = inters;
|
|
||||||
// _bbs = inters.Map(x => x.GetButtonBuilder());
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// protected override string Name
|
|
||||||
// => throw new NotSupportedException();
|
|
||||||
// protected override IEmote Emote
|
|
||||||
// => throw new NotSupportedException();
|
|
||||||
//
|
|
||||||
// protected override ValueTask<bool> Validate(SocketMessageComponent smc)
|
|
||||||
// => new(true);
|
|
||||||
//
|
|
||||||
// public override Task ExecuteOnActionAsync(SocketMessageComponent smc)
|
|
||||||
// {
|
|
||||||
// for (var i = 0; i < _bbs.Length; i++)
|
|
||||||
// {
|
|
||||||
// if (_bbs[i].CustomId == smc.Data.CustomId)
|
|
||||||
// return _inters[i].ExecuteOnActionAsync(smc);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// return Task.CompletedTask;
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// public override MessageComponent CreateComponent()
|
|
||||||
// {
|
|
||||||
// var comp = new ComponentBuilder();
|
|
||||||
//
|
|
||||||
// foreach (var bb in _bbs)
|
|
||||||
// comp.WithButton(bb);
|
|
||||||
//
|
|
||||||
// return comp.Build();
|
|
||||||
// }
|
|
||||||
// }
|
|
@@ -1,42 +0,0 @@
|
|||||||
namespace NadekoBot;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Builder class for NadekoInteractions
|
|
||||||
/// </summary>
|
|
||||||
public class NadekoInteractionBuilder
|
|
||||||
{
|
|
||||||
private NadekoInteractionData? iData;
|
|
||||||
private Func<SocketMessageComponent, Task>? action;
|
|
||||||
// private bool isOwn;
|
|
||||||
|
|
||||||
public NadekoInteractionBuilder WithData<T>(in T data)
|
|
||||||
where T : NadekoInteractionData
|
|
||||||
{
|
|
||||||
iData = data;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
// public NadekoOwnInteractionBuiler WithIsOwn(bool isOwn = true)
|
|
||||||
// {
|
|
||||||
// this.isOwn = isOwn;
|
|
||||||
// return this;
|
|
||||||
|
|
||||||
// }
|
|
||||||
|
|
||||||
public NadekoInteractionBuilder WithAction(in Func<SocketMessageComponent, Task> fn)
|
|
||||||
{
|
|
||||||
this.action = fn;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
public NadekoButtonActionInteraction Build(DiscordSocketClient client, ulong userId)
|
|
||||||
{
|
|
||||||
if (iData is null)
|
|
||||||
throw new InvalidOperationException("You have to specify the data before building the interaction");
|
|
||||||
|
|
||||||
if (action is null)
|
|
||||||
throw new InvalidOperationException("You have to specify the action before building the interaction");
|
|
||||||
|
|
||||||
return new(client, userId, iData, action);
|
|
||||||
}
|
|
||||||
}
|
|
20
src/NadekoBot/Common/Interaction/NadekoInteractionService.cs
Normal file
20
src/NadekoBot/Common/Interaction/NadekoInteractionService.cs
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
namespace NadekoBot;
|
||||||
|
|
||||||
|
public class NadekoInteractionService : INadekoInteractionService, INService
|
||||||
|
{
|
||||||
|
private readonly DiscordSocketClient _client;
|
||||||
|
|
||||||
|
public NadekoInteractionService(DiscordSocketClient client)
|
||||||
|
{
|
||||||
|
_client = client;
|
||||||
|
}
|
||||||
|
|
||||||
|
public NadekoInteraction Create<T>(
|
||||||
|
ulong userId,
|
||||||
|
SimpleInteraction<T> inter)
|
||||||
|
=> new NadekoInteraction(_client,
|
||||||
|
userId,
|
||||||
|
inter.Button,
|
||||||
|
inter.TriggerAsync,
|
||||||
|
onlyAuthor: true);
|
||||||
|
}
|
@@ -1,15 +0,0 @@
|
|||||||
namespace NadekoBot;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Interaction which only the author can use
|
|
||||||
/// </summary>
|
|
||||||
public abstract class NadekoButtonOwnInteraction : NadekoButtonInteraction
|
|
||||||
{
|
|
||||||
protected readonly ulong _authorId;
|
|
||||||
|
|
||||||
protected NadekoButtonOwnInteraction(DiscordSocketClient client, ulong authorId) : base(client)
|
|
||||||
=> _authorId = authorId;
|
|
||||||
|
|
||||||
protected override ValueTask<bool> Validate(SocketMessageComponent smc)
|
|
||||||
=> new(smc.User.Id == _authorId);
|
|
||||||
}
|
|
@@ -1,26 +0,0 @@
|
|||||||
namespace NadekoBot.Common;
|
|
||||||
|
|
||||||
public abstract class NInteraction
|
|
||||||
{
|
|
||||||
private readonly DiscordSocketClient _client;
|
|
||||||
private readonly ulong _userId;
|
|
||||||
private readonly Func<SocketMessageComponent, Task> _action;
|
|
||||||
|
|
||||||
protected abstract NadekoInteractionData Data { get; }
|
|
||||||
|
|
||||||
public NInteraction(
|
|
||||||
DiscordSocketClient client,
|
|
||||||
ulong userId,
|
|
||||||
Func<SocketMessageComponent, Task> action)
|
|
||||||
{
|
|
||||||
_client = client;
|
|
||||||
_userId = userId;
|
|
||||||
_action = action;
|
|
||||||
}
|
|
||||||
|
|
||||||
public NadekoButtonInteraction GetInteraction()
|
|
||||||
=> new NadekoInteractionBuilder()
|
|
||||||
.WithData(Data)
|
|
||||||
.WithAction(_action)
|
|
||||||
.Build(_client, _userId);
|
|
||||||
}
|
|
@@ -18,6 +18,7 @@ public abstract class NadekoModule : ModuleBase
|
|||||||
public CommandHandler _cmdHandler { get; set; }
|
public CommandHandler _cmdHandler { get; set; }
|
||||||
public ILocalization _localization { get; set; }
|
public ILocalization _localization { get; set; }
|
||||||
public IEmbedBuilderService _eb { get; set; }
|
public IEmbedBuilderService _eb { get; set; }
|
||||||
|
public INadekoInteractionService _inter { get; set; }
|
||||||
|
|
||||||
protected string prefix
|
protected string prefix
|
||||||
=> _cmdHandler.GetPrefix(ctx.Guild);
|
=> _cmdHandler.GetPrefix(ctx.Guild);
|
||||||
@@ -36,7 +37,7 @@ public abstract class NadekoModule : ModuleBase
|
|||||||
string error,
|
string error,
|
||||||
string url = null,
|
string url = null,
|
||||||
string footer = null,
|
string footer = null,
|
||||||
NadekoButtonInteraction inter = null)
|
NadekoInteraction inter = null)
|
||||||
=> ctx.Channel.SendErrorAsync(_eb, title, error, url, footer);
|
=> ctx.Channel.SendErrorAsync(_eb, title, error, url, footer);
|
||||||
|
|
||||||
public Task<IUserMessage> SendConfirmAsync(
|
public Task<IUserMessage> SendConfirmAsync(
|
||||||
@@ -47,32 +48,32 @@ public abstract class NadekoModule : ModuleBase
|
|||||||
=> ctx.Channel.SendConfirmAsync(_eb, title, text, url, footer);
|
=> ctx.Channel.SendConfirmAsync(_eb, title, text, url, footer);
|
||||||
|
|
||||||
//
|
//
|
||||||
public Task<IUserMessage> SendErrorAsync(string text, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> SendErrorAsync(string text, NadekoInteraction inter = null)
|
||||||
=> ctx.Channel.SendAsync(_eb, text, MessageType.Error, inter);
|
=> ctx.Channel.SendAsync(_eb, text, MessageType.Error, inter);
|
||||||
public Task<IUserMessage> SendConfirmAsync(string text, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> SendConfirmAsync(string text, NadekoInteraction inter = null)
|
||||||
=> ctx.Channel.SendAsync(_eb, text, MessageType.Ok, inter);
|
=> ctx.Channel.SendAsync(_eb, text, MessageType.Ok, inter);
|
||||||
public Task<IUserMessage> SendPendingAsync(string text, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> SendPendingAsync(string text, NadekoInteraction inter = null)
|
||||||
=> ctx.Channel.SendAsync(_eb, text, MessageType.Pending, inter);
|
=> ctx.Channel.SendAsync(_eb, text, MessageType.Pending, inter);
|
||||||
|
|
||||||
|
|
||||||
// localized normal
|
// localized normal
|
||||||
public Task<IUserMessage> ErrorLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> ErrorLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendErrorAsync(GetText(str), inter);
|
=> SendErrorAsync(GetText(str), inter);
|
||||||
|
|
||||||
public Task<IUserMessage> PendingLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> PendingLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendPendingAsync(GetText(str), inter);
|
=> SendPendingAsync(GetText(str), inter);
|
||||||
|
|
||||||
public Task<IUserMessage> ConfirmLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> ConfirmLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendConfirmAsync(GetText(str), inter);
|
=> SendConfirmAsync(GetText(str), inter);
|
||||||
|
|
||||||
// localized replies
|
// localized replies
|
||||||
public Task<IUserMessage> ReplyErrorLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> ReplyErrorLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendErrorAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
=> SendErrorAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
||||||
|
|
||||||
public Task<IUserMessage> ReplyPendingLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> ReplyPendingLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendPendingAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
=> SendPendingAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
||||||
|
|
||||||
public Task<IUserMessage> ReplyConfirmLocalizedAsync(LocStr str, NadekoButtonInteraction inter = null)
|
public Task<IUserMessage> ReplyConfirmLocalizedAsync(LocStr str, NadekoInteraction inter = null)
|
||||||
=> SendConfirmAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
=> SendConfirmAsync($"{Format.Bold(ctx.User.ToString())} {GetText(str)}", inter);
|
||||||
|
|
||||||
public async Task<bool> PromptUserConfirmAsync(IEmbedBuilder embed)
|
public async Task<bool> PromptUserConfirmAsync(IEmbedBuilder embed)
|
||||||
|
@@ -49,7 +49,8 @@ public enum PunishmentAction
|
|||||||
ChatMute,
|
ChatMute,
|
||||||
VoiceMute,
|
VoiceMute,
|
||||||
AddRole,
|
AddRole,
|
||||||
Warn
|
Warn,
|
||||||
|
TimeOut
|
||||||
}
|
}
|
||||||
|
|
||||||
public class AntiSpamIgnore : DbEntity
|
public class AntiSpamIgnore : DbEntity
|
||||||
|
@@ -5,4 +5,5 @@ public class BanTemplate : DbEntity
|
|||||||
{
|
{
|
||||||
public ulong GuildId { get; set; }
|
public ulong GuildId { get; set; }
|
||||||
public string Text { get; set; }
|
public string Text { get; set; }
|
||||||
|
public int? PruneDays { get; set; }
|
||||||
}
|
}
|
@@ -17,8 +17,6 @@ public class DiscordUser : DbEntity
|
|||||||
public bool IsClubAdmin { get; set; }
|
public bool IsClubAdmin { get; set; }
|
||||||
|
|
||||||
public long TotalXp { get; set; }
|
public long TotalXp { get; set; }
|
||||||
public DateTime LastLevelUp { get; set; } = DateTime.UtcNow;
|
|
||||||
public DateTime LastXpGain { get; set; } = DateTime.MinValue;
|
|
||||||
public XpNotificationLocation NotifyOnLevelUp { get; set; }
|
public XpNotificationLocation NotifyOnLevelUp { get; set; }
|
||||||
|
|
||||||
public long CurrencyAmount { get; set; }
|
public long CurrencyAmount { get; set; }
|
||||||
|
@@ -8,7 +8,6 @@ public class UserXpStats : DbEntity
|
|||||||
public long Xp { get; set; }
|
public long Xp { get; set; }
|
||||||
public long AwardedXp { get; set; }
|
public long AwardedXp { get; set; }
|
||||||
public XpNotificationLocation NotifyOnLevelUp { get; set; }
|
public XpNotificationLocation NotifyOnLevelUp { get; set; }
|
||||||
public DateTime LastLevelUp { get; set; } = DateTime.UtcNow;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum XpNotificationLocation { None, Dm, Channel }
|
public enum XpNotificationLocation { None, Dm, Channel }
|
@@ -13,6 +13,6 @@ public class XpShopOwnedItem : DbEntity
|
|||||||
|
|
||||||
public enum XpShopItemType
|
public enum XpShopItemType
|
||||||
{
|
{
|
||||||
Background,
|
Background = 0,
|
||||||
Frame,
|
Frame = 1,
|
||||||
}
|
}
|
@@ -10,10 +10,6 @@ public sealed class MysqlContext : NadekoContext
|
|||||||
|
|
||||||
protected override string CurrencyTransactionOtherIdDefaultValue
|
protected override string CurrencyTransactionOtherIdDefaultValue
|
||||||
=> "NULL";
|
=> "NULL";
|
||||||
protected override string DiscordUserLastXpGainDefaultValue
|
|
||||||
=> "(UTC_TIMESTAMP - INTERVAL 1 year)";
|
|
||||||
protected override string LastLevelUpDefaultValue
|
|
||||||
=> "(UTC_TIMESTAMP)";
|
|
||||||
|
|
||||||
public MysqlContext(string connStr = "Server=localhost", string version = "8.0")
|
public MysqlContext(string connStr = "Server=localhost", string version = "8.0")
|
||||||
{
|
{
|
||||||
|
@@ -65,8 +65,6 @@ public abstract class NadekoContext : DbContext
|
|||||||
#region Mandatory Provider-Specific Values
|
#region Mandatory Provider-Specific Values
|
||||||
|
|
||||||
protected abstract string CurrencyTransactionOtherIdDefaultValue { get; }
|
protected abstract string CurrencyTransactionOtherIdDefaultValue { get; }
|
||||||
protected abstract string DiscordUserLastXpGainDefaultValue { get; }
|
|
||||||
protected abstract string LastLevelUpDefaultValue { get; }
|
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
@@ -166,12 +164,6 @@ public abstract class NadekoContext : DbContext
|
|||||||
du.Property(x => x.NotifyOnLevelUp)
|
du.Property(x => x.NotifyOnLevelUp)
|
||||||
.HasDefaultValue(XpNotificationLocation.None);
|
.HasDefaultValue(XpNotificationLocation.None);
|
||||||
|
|
||||||
du.Property(x => x.LastXpGain)
|
|
||||||
.HasDefaultValueSql(DiscordUserLastXpGainDefaultValue);
|
|
||||||
|
|
||||||
du.Property(x => x.LastLevelUp)
|
|
||||||
.HasDefaultValueSql(LastLevelUpDefaultValue);
|
|
||||||
|
|
||||||
du.Property(x => x.TotalXp)
|
du.Property(x => x.TotalXp)
|
||||||
.HasDefaultValue(0);
|
.HasDefaultValue(0);
|
||||||
|
|
||||||
@@ -213,9 +205,6 @@ public abstract class NadekoContext : DbContext
|
|||||||
})
|
})
|
||||||
.IsUnique();
|
.IsUnique();
|
||||||
|
|
||||||
xps.Property(x => x.LastLevelUp)
|
|
||||||
.HasDefaultValueSql(LastLevelUpDefaultValue);
|
|
||||||
|
|
||||||
xps.HasIndex(x => x.UserId);
|
xps.HasIndex(x => x.UserId);
|
||||||
xps.HasIndex(x => x.GuildId);
|
xps.HasIndex(x => x.GuildId);
|
||||||
xps.HasIndex(x => x.Xp);
|
xps.HasIndex(x => x.Xp);
|
||||||
@@ -341,6 +330,10 @@ public abstract class NadekoContext : DbContext
|
|||||||
#region BanTemplate
|
#region BanTemplate
|
||||||
|
|
||||||
modelBuilder.Entity<BanTemplate>().HasIndex(x => x.GuildId).IsUnique();
|
modelBuilder.Entity<BanTemplate>().HasIndex(x => x.GuildId).IsUnique();
|
||||||
|
modelBuilder.Entity<BanTemplate>()
|
||||||
|
.Property(x => x.PruneDays)
|
||||||
|
.HasDefaultValue(null)
|
||||||
|
.IsRequired(false);
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
|
@@ -8,10 +8,6 @@ public sealed class PostgreSqlContext : NadekoContext
|
|||||||
|
|
||||||
protected override string CurrencyTransactionOtherIdDefaultValue
|
protected override string CurrencyTransactionOtherIdDefaultValue
|
||||||
=> "NULL";
|
=> "NULL";
|
||||||
protected override string DiscordUserLastXpGainDefaultValue
|
|
||||||
=> "timezone('utc', now()) - interval '-1 year'";
|
|
||||||
protected override string LastLevelUpDefaultValue
|
|
||||||
=> "timezone('utc', now())";
|
|
||||||
|
|
||||||
public PostgreSqlContext(string connStr = "Host=localhost")
|
public PostgreSqlContext(string connStr = "Host=localhost")
|
||||||
{
|
{
|
||||||
|
@@ -9,10 +9,6 @@ public sealed class SqliteContext : NadekoContext
|
|||||||
|
|
||||||
protected override string CurrencyTransactionOtherIdDefaultValue
|
protected override string CurrencyTransactionOtherIdDefaultValue
|
||||||
=> "NULL";
|
=> "NULL";
|
||||||
protected override string DiscordUserLastXpGainDefaultValue
|
|
||||||
=> "datetime('now', '-1 years')";
|
|
||||||
protected override string LastLevelUpDefaultValue
|
|
||||||
=> "datetime('now')";
|
|
||||||
|
|
||||||
public SqliteContext(string connectionString = "Data Source=data/NadekoBot.db", int commandTimeout = 60)
|
public SqliteContext(string connectionString = "Data Source=data/NadekoBot.db", int commandTimeout = 60)
|
||||||
{
|
{
|
||||||
|
3542
src/NadekoBot/Migrations/MySql/20220808141855_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
3542
src/NadekoBot/Migrations/MySql/20220808141855_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,49 @@
|
|||||||
|
using System;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace NadekoBot.Migrations.Mysql
|
||||||
|
{
|
||||||
|
public partial class removeobsoletexpcolumns : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "lastlevelup",
|
||||||
|
table: "userxpstats");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "lastlevelup",
|
||||||
|
table: "discorduser");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "lastxpgain",
|
||||||
|
table: "discorduser");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "lastlevelup",
|
||||||
|
table: "userxpstats",
|
||||||
|
type: "datetime(6)",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "(UTC_TIMESTAMP)");
|
||||||
|
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "lastlevelup",
|
||||||
|
table: "discorduser",
|
||||||
|
type: "datetime(6)",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "(UTC_TIMESTAMP)");
|
||||||
|
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "lastxpgain",
|
||||||
|
table: "discorduser",
|
||||||
|
type: "datetime(6)",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "(UTC_TIMESTAMP - INTERVAL 1 year)");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
3546
src/NadekoBot/Migrations/MySql/20220831142722_banprune.Designer.cs
generated
Normal file
3546
src/NadekoBot/Migrations/MySql/20220831142722_banprune.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
25
src/NadekoBot/Migrations/MySql/20220831142722_banprune.cs
Normal file
25
src/NadekoBot/Migrations/MySql/20220831142722_banprune.cs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace NadekoBot.Migrations.Mysql
|
||||||
|
{
|
||||||
|
public partial class banprune : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<int>(
|
||||||
|
name: "prunedays",
|
||||||
|
table: "bantemplates",
|
||||||
|
type: "int",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "prunedays",
|
||||||
|
table: "bantemplates");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -168,18 +168,6 @@ namespace NadekoBot.Migrations.Mysql
|
|||||||
.HasDefaultValue(false)
|
.HasDefaultValue(false)
|
||||||
.HasColumnName("isclubadmin");
|
.HasColumnName("isclubadmin");
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("datetime(6)")
|
|
||||||
.HasColumnName("lastlevelup")
|
|
||||||
.HasDefaultValueSql("(UTC_TIMESTAMP)");
|
|
||||||
|
|
||||||
b.Property<DateTime>("LastXpGain")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("datetime(6)")
|
|
||||||
.HasColumnName("lastxpgain")
|
|
||||||
.HasDefaultValueSql("(UTC_TIMESTAMP - INTERVAL 1 year)");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.ValueGeneratedOnAdd()
|
.ValueGeneratedOnAdd()
|
||||||
.HasColumnType("int")
|
.HasColumnType("int")
|
||||||
@@ -687,6 +675,10 @@ namespace NadekoBot.Migrations.Mysql
|
|||||||
.HasColumnType("bigint unsigned")
|
.HasColumnType("bigint unsigned")
|
||||||
.HasColumnName("guildid");
|
.HasColumnName("guildid");
|
||||||
|
|
||||||
|
b.Property<int?>("PruneDays")
|
||||||
|
.HasColumnType("int")
|
||||||
|
.HasColumnName("prunedays");
|
||||||
|
|
||||||
b.Property<string>("Text")
|
b.Property<string>("Text")
|
||||||
.HasColumnType("longtext")
|
.HasColumnType("longtext")
|
||||||
.HasColumnName("text");
|
.HasColumnName("text");
|
||||||
@@ -2565,12 +2557,6 @@ namespace NadekoBot.Migrations.Mysql
|
|||||||
.HasColumnType("bigint unsigned")
|
.HasColumnType("bigint unsigned")
|
||||||
.HasColumnName("guildid");
|
.HasColumnName("guildid");
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("datetime(6)")
|
|
||||||
.HasColumnName("lastlevelup")
|
|
||||||
.HasDefaultValueSql("(UTC_TIMESTAMP)");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.HasColumnType("int")
|
.HasColumnType("int")
|
||||||
.HasColumnName("notifyonlevelup");
|
.HasColumnName("notifyonlevelup");
|
||||||
|
3686
src/NadekoBot/Migrations/Postgresql/20220808142559_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
3686
src/NadekoBot/Migrations/Postgresql/20220808142559_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
3690
src/NadekoBot/Migrations/Postgresql/20220831142735_banprune.Designer.cs
generated
Normal file
3690
src/NadekoBot/Migrations/Postgresql/20220831142735_banprune.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace NadekoBot.Migrations.PostgreSql
|
||||||
|
{
|
||||||
|
public partial class banprune : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<int>(
|
||||||
|
name: "prunedays",
|
||||||
|
table: "bantemplates",
|
||||||
|
type: "integer",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "prunedays",
|
||||||
|
table: "bantemplates");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -36,7 +36,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("balance");
|
.HasColumnName("balance");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("UserId")
|
b.Property<decimal>("UserId")
|
||||||
@@ -101,7 +101,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Description")
|
b.Property<string>("Description")
|
||||||
@@ -163,7 +163,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("currencyamount");
|
.HasColumnName("currencyamount");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Discriminator")
|
b.Property<string>("Discriminator")
|
||||||
@@ -176,18 +176,6 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasDefaultValue(false)
|
.HasDefaultValue(false)
|
||||||
.HasColumnName("isclubadmin");
|
.HasColumnName("isclubadmin");
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("timestamp with time zone")
|
|
||||||
.HasColumnName("lastlevelup")
|
|
||||||
.HasDefaultValueSql("timezone('utc', now())");
|
|
||||||
|
|
||||||
b.Property<DateTime>("LastXpGain")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("timestamp with time zone")
|
|
||||||
.HasColumnName("lastxpgain")
|
|
||||||
.HasDefaultValueSql("timezone('utc', now()) - interval '-1 year'");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.ValueGeneratedOnAdd()
|
.ValueGeneratedOnAdd()
|
||||||
.HasColumnType("integer")
|
.HasColumnType("integer")
|
||||||
@@ -243,7 +231,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -322,7 +310,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("amountcents");
|
.HasColumnName("amountcents");
|
||||||
|
|
||||||
b.Property<DateTime>("LastCharge")
|
b.Property<DateTime>("LastCharge")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("lastcharge");
|
.HasColumnName("lastcharge");
|
||||||
|
|
||||||
b.Property<string>("UniquePlatformUserId")
|
b.Property<string>("UniquePlatformUserId")
|
||||||
@@ -330,7 +318,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("uniqueplatformuserid");
|
.HasColumnName("uniqueplatformuserid");
|
||||||
|
|
||||||
b.Property<DateTime>("ValidThru")
|
b.Property<DateTime>("ValidThru")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("validthru");
|
.HasColumnName("validthru");
|
||||||
|
|
||||||
b.HasKey("UserId")
|
b.HasKey("UserId")
|
||||||
@@ -357,7 +345,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("MessageId")
|
b.Property<decimal>("MessageId")
|
||||||
@@ -388,7 +376,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("IsUsing")
|
b.Property<bool>("IsUsing")
|
||||||
@@ -471,7 +459,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("action");
|
.HasColumnName("action");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("GuildConfigId")
|
b.Property<int>("GuildConfigId")
|
||||||
@@ -518,7 +506,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.HasKey("Id")
|
b.HasKey("Id")
|
||||||
@@ -544,7 +532,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("action");
|
.HasColumnName("action");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("GuildConfigId")
|
b.Property<int>("GuildConfigId")
|
||||||
@@ -595,7 +583,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("commandtext");
|
.HasColumnName("commandtext");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal?>("GuildId")
|
b.Property<decimal?>("GuildId")
|
||||||
@@ -642,7 +630,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -676,7 +664,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Source")
|
b.Property<string>("Source")
|
||||||
@@ -710,13 +698,17 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
.HasColumnType("numeric(20,0)")
|
.HasColumnType("numeric(20,0)")
|
||||||
.HasColumnName("guildid");
|
.HasColumnName("guildid");
|
||||||
|
|
||||||
|
b.Property<int?>("PruneDays")
|
||||||
|
.HasColumnType("integer")
|
||||||
|
.HasColumnName("prunedays");
|
||||||
|
|
||||||
b.Property<string>("Text")
|
b.Property<string>("Text")
|
||||||
.HasColumnType("text")
|
.HasColumnType("text")
|
||||||
.HasColumnName("text");
|
.HasColumnName("text");
|
||||||
@@ -741,7 +733,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("ItemId")
|
b.Property<decimal>("ItemId")
|
||||||
@@ -768,7 +760,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -806,7 +798,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("commandname");
|
.HasColumnName("commandname");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -840,7 +832,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("amount");
|
.HasColumnName("amount");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Extra")
|
b.Property<string>("Extra")
|
||||||
@@ -890,7 +882,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -924,7 +916,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("command");
|
.HasColumnName("command");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal?>("GuildId")
|
b.Property<decimal?>("GuildId")
|
||||||
@@ -955,7 +947,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("ItemId")
|
b.Property<decimal>("ItemId")
|
||||||
@@ -993,7 +985,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("GuildConfigId")
|
b.Property<int>("GuildConfigId")
|
||||||
@@ -1028,7 +1020,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1054,7 +1046,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1088,7 +1080,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1118,7 +1110,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1148,7 +1140,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1174,7 +1166,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("GuildConfigId")
|
b.Property<int>("GuildConfigId")
|
||||||
@@ -1253,7 +1245,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("cleverbotenabled");
|
.HasColumnName("cleverbotenabled");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("DeleteMessageOnCommand")
|
b.Property<bool>("DeleteMessageOnCommand")
|
||||||
@@ -1381,7 +1373,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("ItemType")
|
b.Property<int>("ItemType")
|
||||||
@@ -1420,7 +1412,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("LogSettingId")
|
b.Property<int?>("LogSettingId")
|
||||||
@@ -1450,7 +1442,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -1493,7 +1485,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelupdatedid");
|
.HasColumnName("channelupdatedid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -1629,7 +1621,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("authorid");
|
.HasColumnName("authorid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Name")
|
b.Property<string>("Name")
|
||||||
@@ -1652,7 +1644,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1694,7 +1686,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("containsanywhere");
|
.HasColumnName("containsanywhere");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("DmResponse")
|
b.Property<bool>("DmResponse")
|
||||||
@@ -1733,7 +1725,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -1763,7 +1755,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -1825,7 +1817,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -1867,7 +1859,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("MusicPlaylistId")
|
b.Property<int?>("MusicPlaylistId")
|
||||||
@@ -1917,7 +1909,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -1948,7 +1940,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("Index")
|
b.Property<int>("Index")
|
||||||
@@ -1982,7 +1974,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("PollId")
|
b.Property<int?>("PollId")
|
||||||
@@ -2025,7 +2017,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("authorname");
|
.HasColumnName("authorname");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -2068,7 +2060,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Emote")
|
b.Property<string>("Emote")
|
||||||
@@ -2123,7 +2115,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("IsPrivate")
|
b.Property<bool>("IsPrivate")
|
||||||
@@ -2143,7 +2135,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("userid");
|
.HasColumnName("userid");
|
||||||
|
|
||||||
b.Property<DateTime>("When")
|
b.Property<DateTime>("When")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("when");
|
.HasColumnName("when");
|
||||||
|
|
||||||
b.HasKey("Id")
|
b.HasKey("Id")
|
||||||
@@ -2169,7 +2161,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("channelid");
|
.HasColumnName("channelid");
|
||||||
|
|
||||||
b.Property<DateTime>("DateAdded")
|
b.Property<DateTime>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
@@ -2216,11 +2208,11 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("amountrewardedthismonth");
|
.HasColumnName("amountrewardedthismonth");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<DateTime>("LastReward")
|
b.Property<DateTime>("LastReward")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("lastreward");
|
.HasColumnName("lastreward");
|
||||||
|
|
||||||
b.Property<string>("PlatformUserId")
|
b.Property<string>("PlatformUserId")
|
||||||
@@ -2251,7 +2243,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("Status")
|
b.Property<string>("Status")
|
||||||
@@ -2278,7 +2270,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("Group")
|
b.Property<int>("Group")
|
||||||
@@ -2323,7 +2315,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("authorid");
|
.HasColumnName("authorid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2373,7 +2365,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("ShopEntryId")
|
b.Property<int?>("ShopEntryId")
|
||||||
@@ -2403,7 +2395,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2433,7 +2425,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2463,7 +2455,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("StreamRoleSettingsId")
|
b.Property<int?>("StreamRoleSettingsId")
|
||||||
@@ -2501,7 +2493,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("addroleid");
|
.HasColumnName("addroleid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("Enabled")
|
b.Property<bool>("Enabled")
|
||||||
@@ -2540,7 +2532,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("StreamRoleSettingsId")
|
b.Property<int?>("StreamRoleSettingsId")
|
||||||
@@ -2574,7 +2566,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2582,7 +2574,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("guildconfigid");
|
.HasColumnName("guildconfigid");
|
||||||
|
|
||||||
b.Property<DateTime>("UnbanAt")
|
b.Property<DateTime>("UnbanAt")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("unbanat");
|
.HasColumnName("unbanat");
|
||||||
|
|
||||||
b.Property<decimal>("UserId")
|
b.Property<decimal>("UserId")
|
||||||
@@ -2608,7 +2600,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2616,7 +2608,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("guildconfigid");
|
.HasColumnName("guildconfigid");
|
||||||
|
|
||||||
b.Property<DateTime>("UnmuteAt")
|
b.Property<DateTime>("UnmuteAt")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("unmuteat");
|
.HasColumnName("unmuteat");
|
||||||
|
|
||||||
b.Property<decimal>("UserId")
|
b.Property<decimal>("UserId")
|
||||||
@@ -2642,7 +2634,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2654,7 +2646,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("roleid");
|
.HasColumnName("roleid");
|
||||||
|
|
||||||
b.Property<DateTime>("UnbanAt")
|
b.Property<DateTime>("UnbanAt")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("unbanat");
|
.HasColumnName("unbanat");
|
||||||
|
|
||||||
b.Property<decimal>("UserId")
|
b.Property<decimal>("UserId")
|
||||||
@@ -2684,19 +2676,13 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("awardedxp");
|
.HasColumnName("awardedxp");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<decimal>("GuildId")
|
b.Property<decimal>("GuildId")
|
||||||
.HasColumnType("numeric(20,0)")
|
.HasColumnType("numeric(20,0)")
|
||||||
.HasColumnName("guildid");
|
.HasColumnName("guildid");
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("timestamp with time zone")
|
|
||||||
.HasColumnName("lastlevelup")
|
|
||||||
.HasDefaultValueSql("timezone('utc', now())");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.HasColumnType("integer")
|
.HasColumnType("integer")
|
||||||
.HasColumnName("notifyonlevelup");
|
.HasColumnName("notifyonlevelup");
|
||||||
@@ -2741,7 +2727,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -2783,7 +2769,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("claimerid");
|
.HasColumnName("claimerid");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<long>("Price")
|
b.Property<long>("Price")
|
||||||
@@ -2823,7 +2809,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<string>("ItemEmoji")
|
b.Property<string>("ItemEmoji")
|
||||||
@@ -2857,7 +2843,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("NewId")
|
b.Property<int?>("NewId")
|
||||||
@@ -2901,7 +2887,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<bool>("Forgiven")
|
b.Property<bool>("Forgiven")
|
||||||
@@ -2963,7 +2949,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("count");
|
.HasColumnName("count");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int?>("GuildConfigId")
|
b.Property<int?>("GuildConfigId")
|
||||||
@@ -3005,7 +2991,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
.HasColumnName("amount");
|
.HasColumnName("amount");
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("Level")
|
b.Property<int>("Level")
|
||||||
@@ -3035,7 +3021,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("Level")
|
b.Property<int>("Level")
|
||||||
@@ -3074,7 +3060,7 @@ namespace NadekoBot.Migrations.PostgreSql
|
|||||||
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
NpgsqlPropertyBuilderExtensions.UseIdentityByDefaultColumn(b.Property<int>("Id"));
|
||||||
|
|
||||||
b.Property<DateTime?>("DateAdded")
|
b.Property<DateTime?>("DateAdded")
|
||||||
.HasColumnType("timestamp with time zone")
|
.HasColumnType("timestamp without time zone")
|
||||||
.HasColumnName("dateadded");
|
.HasColumnName("dateadded");
|
||||||
|
|
||||||
b.Property<int>("GuildConfigId")
|
b.Property<int>("GuildConfigId")
|
||||||
|
2843
src/NadekoBot/Migrations/Sqlite/20220808141842_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
2843
src/NadekoBot/Migrations/Sqlite/20220808141842_remove-obsolete-xp-columns.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,49 @@
|
|||||||
|
using System;
|
||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace NadekoBot.Migrations
|
||||||
|
{
|
||||||
|
public partial class removeobsoletexpcolumns : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "LastLevelUp",
|
||||||
|
table: "UserXpStats");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "LastLevelUp",
|
||||||
|
table: "DiscordUser");
|
||||||
|
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "LastXpGain",
|
||||||
|
table: "DiscordUser");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "LastLevelUp",
|
||||||
|
table: "UserXpStats",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "datetime('now')");
|
||||||
|
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "LastLevelUp",
|
||||||
|
table: "DiscordUser",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "datetime('now')");
|
||||||
|
|
||||||
|
migrationBuilder.AddColumn<DateTime>(
|
||||||
|
name: "LastXpGain",
|
||||||
|
table: "DiscordUser",
|
||||||
|
type: "TEXT",
|
||||||
|
nullable: false,
|
||||||
|
defaultValueSql: "datetime('now', '-1 years')");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
2846
src/NadekoBot/Migrations/Sqlite/20220831142504_banprune.Designer.cs
generated
Normal file
2846
src/NadekoBot/Migrations/Sqlite/20220831142504_banprune.Designer.cs
generated
Normal file
File diff suppressed because it is too large
Load Diff
25
src/NadekoBot/Migrations/Sqlite/20220831142504_banprune.cs
Normal file
25
src/NadekoBot/Migrations/Sqlite/20220831142504_banprune.cs
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
using Microsoft.EntityFrameworkCore.Migrations;
|
||||||
|
|
||||||
|
#nullable disable
|
||||||
|
|
||||||
|
namespace NadekoBot.Migrations
|
||||||
|
{
|
||||||
|
public partial class banprune : Migration
|
||||||
|
{
|
||||||
|
protected override void Up(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.AddColumn<int>(
|
||||||
|
name: "PruneDays",
|
||||||
|
table: "BanTemplates",
|
||||||
|
type: "INTEGER",
|
||||||
|
nullable: true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Down(MigrationBuilder migrationBuilder)
|
||||||
|
{
|
||||||
|
migrationBuilder.DropColumn(
|
||||||
|
name: "PruneDays",
|
||||||
|
table: "BanTemplates");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -134,16 +134,6 @@ namespace NadekoBot.Migrations
|
|||||||
.HasColumnType("INTEGER")
|
.HasColumnType("INTEGER")
|
||||||
.HasDefaultValue(false);
|
.HasDefaultValue(false);
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("TEXT")
|
|
||||||
.HasDefaultValueSql("datetime('now')");
|
|
||||||
|
|
||||||
b.Property<DateTime>("LastXpGain")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("TEXT")
|
|
||||||
.HasDefaultValueSql("datetime('now', '-1 years')");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.ValueGeneratedOnAdd()
|
.ValueGeneratedOnAdd()
|
||||||
.HasColumnType("INTEGER")
|
.HasColumnType("INTEGER")
|
||||||
@@ -539,6 +529,9 @@ namespace NadekoBot.Migrations
|
|||||||
b.Property<ulong>("GuildId")
|
b.Property<ulong>("GuildId")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
b.Property<int?>("PruneDays")
|
||||||
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
b.Property<string>("Text")
|
b.Property<string>("Text")
|
||||||
.HasColumnType("TEXT");
|
.HasColumnType("TEXT");
|
||||||
|
|
||||||
@@ -2003,11 +1996,6 @@ namespace NadekoBot.Migrations
|
|||||||
b.Property<ulong>("GuildId")
|
b.Property<ulong>("GuildId")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
b.Property<DateTime>("LastLevelUp")
|
|
||||||
.ValueGeneratedOnAdd()
|
|
||||||
.HasColumnType("TEXT")
|
|
||||||
.HasDefaultValueSql("datetime('now')");
|
|
||||||
|
|
||||||
b.Property<int>("NotifyOnLevelUp")
|
b.Property<int>("NotifyOnLevelUp")
|
||||||
.HasColumnType("INTEGER");
|
.HasColumnType("INTEGER");
|
||||||
|
|
||||||
|
@@ -344,4 +344,36 @@ public partial class Administration : NadekoModule<AdministrationService>
|
|||||||
|
|
||||||
await ctx.OkAsync();
|
await ctx.OkAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[BotPerm(ChannelPermission.CreatePublicThreads)]
|
||||||
|
[UserPerm(ChannelPermission.CreatePublicThreads)]
|
||||||
|
public async Task ThreadCreate([Leftover] string name)
|
||||||
|
{
|
||||||
|
if (ctx.Channel is not SocketTextChannel stc)
|
||||||
|
return;
|
||||||
|
|
||||||
|
await stc.CreateThreadAsync(name, message: ctx.Message.ReferencedMessage);
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[BotPerm(ChannelPermission.ManageThreads)]
|
||||||
|
[UserPerm(ChannelPermission.ManageThreads)]
|
||||||
|
public async Task ThreadDelete([Leftover] string name)
|
||||||
|
{
|
||||||
|
if (ctx.Channel is not SocketTextChannel stc)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var t = stc.Threads.FirstOrDefault(x => string.Equals(x.Name, name, StringComparison.InvariantCultureIgnoreCase));
|
||||||
|
|
||||||
|
if (t is null)
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.not_found);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await t.DeleteAsync();
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
}
|
}
|
@@ -358,9 +358,10 @@ public class MuteService : INService
|
|||||||
IGuild guild,
|
IGuild guild,
|
||||||
IUser user,
|
IUser user,
|
||||||
TimeSpan after,
|
TimeSpan after,
|
||||||
string reason)
|
string reason,
|
||||||
|
int pruneDays)
|
||||||
{
|
{
|
||||||
await guild.AddBanAsync(user.Id, 0, reason);
|
await guild.AddBanAsync(user.Id, pruneDays, reason);
|
||||||
await using (var uow = _db.GetDbContext())
|
await using (var uow = _db.GetDbContext())
|
||||||
{
|
{
|
||||||
var config = uow.GuildConfigsForId(guild.Id, set => set.Include(x => x.UnbanTimer));
|
var config = uow.GuildConfigsForId(guild.Id, set => set.Include(x => x.UnbanTimer));
|
||||||
|
@@ -38,10 +38,14 @@ public partial class Administration
|
|||||||
if (minAgeMinutes < 1 || punishTimeMinutes < 0)
|
if (minAgeMinutes < 1 || punishTimeMinutes < 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
var minutes = (int?)punishTime?.Time.TotalMinutes ?? 0;
|
||||||
|
if (action is PunishmentAction.TimeOut && minutes < 1)
|
||||||
|
minutes = 1;
|
||||||
|
|
||||||
await _service.StartAntiAltAsync(ctx.Guild.Id,
|
await _service.StartAntiAltAsync(ctx.Guild.Id,
|
||||||
minAgeMinutes,
|
minAgeMinutes,
|
||||||
action,
|
action,
|
||||||
(int?)punishTime?.Time.TotalMinutes ?? 0);
|
minutes);
|
||||||
|
|
||||||
await ctx.OkAsync();
|
await ctx.OkAsync();
|
||||||
}
|
}
|
||||||
@@ -56,6 +60,9 @@ public partial class Administration
|
|||||||
if (minAgeMinutes < 1)
|
if (minAgeMinutes < 1)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (action == PunishmentAction.TimeOut)
|
||||||
|
return;
|
||||||
|
|
||||||
await _service.StartAntiAltAsync(ctx.Guild.Id, minAgeMinutes, action, roleId: role.Id);
|
await _service.StartAntiAltAsync(ctx.Guild.Id, minAgeMinutes, action, roleId: role.Id);
|
||||||
|
|
||||||
await ctx.OkAsync();
|
await ctx.OkAsync();
|
||||||
@@ -123,6 +130,9 @@ public partial class Administration
|
|||||||
if (time is < 0 or > 60 * 24)
|
if (time is < 0 or > 60 * 24)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if(action is PunishmentAction.TimeOut && time < 1)
|
||||||
|
return;
|
||||||
|
|
||||||
var stats = await _service.StartAntiRaidAsync(ctx.Guild.Id, userThreshold, seconds, action, time);
|
var stats = await _service.StartAntiRaidAsync(ctx.Guild.Id, userThreshold, seconds, action, time);
|
||||||
|
|
||||||
if (stats is null)
|
if (stats is null)
|
||||||
@@ -187,6 +197,9 @@ public partial class Administration
|
|||||||
if (time is < 0 or > 60 * 24)
|
if (time is < 0 or > 60 * 24)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
if (action is PunishmentAction.TimeOut && time < 1)
|
||||||
|
return;
|
||||||
|
|
||||||
var stats = await _service.StartAntiSpamAsync(ctx.Guild.Id, messageCount, action, time, role?.Id);
|
var stats = await _service.StartAntiSpamAsync(ctx.Guild.Id, messageCount, action, time, role?.Id);
|
||||||
|
|
||||||
await SendConfirmAsync(GetText(strs.prot_enable("Anti-Spam")),
|
await SendConfirmAsync(GetText(strs.prot_enable("Anti-Spam")),
|
||||||
|
@@ -345,6 +345,10 @@ public partial class Administration
|
|||||||
if (punish is PunishmentAction.AddRole or PunishmentAction.Warn)
|
if (punish is PunishmentAction.AddRole or PunishmentAction.Warn)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
// you must specify the time for timeout
|
||||||
|
if (punish is PunishmentAction.TimeOut && time is null)
|
||||||
|
return;
|
||||||
|
|
||||||
var success = _service.WarnPunish(ctx.Guild.Id, number, punish, time);
|
var success = _service.WarnPunish(ctx.Guild.Id, number, punish, time);
|
||||||
|
|
||||||
if (!success)
|
if (!success)
|
||||||
@@ -425,7 +429,8 @@ public partial class Administration
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await _mute.TimedBan(ctx.Guild, user, time.Time, (ctx.User + " | " + msg).TrimTo(512));
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
|
await _mute.TimedBan(ctx.Guild, user, time.Time, (ctx.User + " | " + msg).TrimTo(512), banPrune);
|
||||||
var toSend = _eb.Create()
|
var toSend = _eb.Create()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithTitle("⛔️ " + GetText(strs.banned_user))
|
.WithTitle("⛔️ " + GetText(strs.banned_user))
|
||||||
@@ -451,7 +456,8 @@ public partial class Administration
|
|||||||
var user = await ((DiscordSocketClient)Context.Client).Rest.GetGuildUserAsync(ctx.Guild.Id, userId);
|
var user = await ((DiscordSocketClient)Context.Client).Rest.GetGuildUserAsync(ctx.Guild.Id, userId);
|
||||||
if (user is null)
|
if (user is null)
|
||||||
{
|
{
|
||||||
await ctx.Guild.AddBanAsync(userId, 7, (ctx.User + " | " + msg).TrimTo(512));
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
|
await ctx.Guild.AddBanAsync(userId, banPrune, (ctx.User + " | " + msg).TrimTo(512));
|
||||||
|
|
||||||
await ctx.Channel.EmbedAsync(_eb.Create()
|
await ctx.Channel.EmbedAsync(_eb.Create()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
@@ -486,7 +492,8 @@ public partial class Administration
|
|||||||
dmFailed = true;
|
dmFailed = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
await ctx.Guild.AddBanAsync(user, 7, (ctx.User + " | " + msg).TrimTo(512));
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
|
await ctx.Guild.AddBanAsync(user, banPrune, (ctx.User + " | " + msg).TrimTo(512));
|
||||||
|
|
||||||
var toSend = _eb.Create()
|
var toSend = _eb.Create()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
@@ -500,6 +507,26 @@ public partial class Administration
|
|||||||
await ctx.Channel.EmbedAsync(toSend);
|
await ctx.Channel.EmbedAsync(toSend);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireContext(ContextType.Guild)]
|
||||||
|
[UserPerm(GuildPerm.BanMembers)]
|
||||||
|
[BotPerm(GuildPerm.BanMembers)]
|
||||||
|
public async Task BanPrune(int days)
|
||||||
|
{
|
||||||
|
if (days < 0 || days > 7)
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.invalid_input);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await _service.SetBanPruneAsync(ctx.Guild.Id, days);
|
||||||
|
|
||||||
|
if (days == 0)
|
||||||
|
await ReplyConfirmLocalizedAsync(strs.ban_prune_disabled);
|
||||||
|
else
|
||||||
|
await ReplyConfirmLocalizedAsync(strs.ban_prune(days));
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.BanMembers)]
|
[UserPerm(GuildPerm.BanMembers)]
|
||||||
@@ -651,7 +678,8 @@ public partial class Administration
|
|||||||
dmFailed = true;
|
dmFailed = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
await ctx.Guild.AddBanAsync(user, 7, ("Softban | " + ctx.User + " | " + msg).TrimTo(512));
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
|
await ctx.Guild.AddBanAsync(user, banPrune, ("Softban | " + ctx.User + " | " + msg).TrimTo(512));
|
||||||
try { await ctx.Guild.RemoveBanAsync(user); }
|
try { await ctx.Guild.RemoveBanAsync(user); }
|
||||||
catch { await ctx.Guild.RemoveBanAsync(user); }
|
catch { await ctx.Guild.RemoveBanAsync(user); }
|
||||||
|
|
||||||
@@ -719,6 +747,49 @@ public partial class Administration
|
|||||||
await ctx.Channel.EmbedAsync(toSend);
|
await ctx.Channel.EmbedAsync(toSend);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireContext(ContextType.Guild)]
|
||||||
|
[UserPerm(GuildPerm.ModerateMembers)]
|
||||||
|
[BotPerm(GuildPerm.ModerateMembers)]
|
||||||
|
[Priority(2)]
|
||||||
|
public async Task Timeout(IUser globalUser, StoopidTime time, [Leftover] string msg = null)
|
||||||
|
{
|
||||||
|
var user = await ctx.Guild.GetUserAsync(globalUser.Id);
|
||||||
|
|
||||||
|
if (user is null)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!await CheckRoleHierarchy(user))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var dmFailed = false;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var dmMessage = GetText(strs.timeoutdm(Format.Bold(ctx.Guild.Name), msg));
|
||||||
|
await user.EmbedAsync(_eb.Create(ctx)
|
||||||
|
.WithPendingColor()
|
||||||
|
.WithDescription(dmMessage));
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
dmFailed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
await user.SetTimeOutAsync(time.Time);
|
||||||
|
|
||||||
|
var toSend = _eb.Create()
|
||||||
|
.WithOkColor()
|
||||||
|
.WithTitle("⏳ " + GetText(strs.timedout_user))
|
||||||
|
.AddField(GetText(strs.username), user.ToString(), true)
|
||||||
|
.AddField("ID", user.Id.ToString(), true);
|
||||||
|
|
||||||
|
if (dmFailed)
|
||||||
|
toSend.WithFooter("⚠️ " + GetText(strs.unable_to_dm_user));
|
||||||
|
|
||||||
|
await ctx.Channel.EmbedAsync(toSend);
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.BanMembers)]
|
[UserPerm(GuildPerm.BanMembers)]
|
||||||
@@ -775,11 +846,12 @@ public partial class Administration
|
|||||||
|
|
||||||
var banningMessage = await ctx.Channel.EmbedAsync(toSend);
|
var banningMessage = await ctx.Channel.EmbedAsync(toSend);
|
||||||
|
|
||||||
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
foreach (var toBan in banning)
|
foreach (var toBan in banning)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await ctx.Guild.AddBanAsync(toBan.Id, 7, $"{ctx.User} | Massban");
|
await ctx.Guild.AddBanAsync(toBan.Id, banPrune, $"{ctx.User} | Massban");
|
||||||
}
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
@@ -818,10 +890,11 @@ public partial class Administration
|
|||||||
.AddField(GetText(strs.invalid(missing)), missStr)
|
.AddField(GetText(strs.invalid(missing)), missStr)
|
||||||
.WithPendingColor());
|
.WithPendingColor());
|
||||||
|
|
||||||
|
var banPrune = await _service.GetBanPruneAsync(ctx.Guild.Id) ?? 7;
|
||||||
//do the banning
|
//do the banning
|
||||||
await Task.WhenAll(bans.Where(x => x.Id.HasValue)
|
await Task.WhenAll(bans.Where(x => x.Id.HasValue)
|
||||||
.Select(x => ctx.Guild.AddBanAsync(x.Id.Value,
|
.Select(x => ctx.Guild.AddBanAsync(x.Id.Value,
|
||||||
7,
|
banPrune,
|
||||||
x.Reason,
|
x.Reason,
|
||||||
new()
|
new()
|
||||||
{
|
{
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
#nullable disable
|
#nullable disable
|
||||||
using LinqToDB;
|
using LinqToDB;
|
||||||
|
using LinqToDB.EntityFrameworkCore;
|
||||||
using Microsoft.EntityFrameworkCore;
|
using Microsoft.EntityFrameworkCore;
|
||||||
using NadekoBot.Common.ModuleBehaviors;
|
using NadekoBot.Common.ModuleBehaviors;
|
||||||
using NadekoBot.Common.TypeReaders.Models;
|
using NadekoBot.Common.TypeReaders.Models;
|
||||||
@@ -127,6 +128,7 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
if (!await CheckPermission(guild, p))
|
if (!await CheckPermission(guild, p))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
int banPrune;
|
||||||
switch (p)
|
switch (p)
|
||||||
{
|
{
|
||||||
case PunishmentAction.Mute:
|
case PunishmentAction.Mute:
|
||||||
@@ -151,13 +153,15 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
await user.KickAsync(reason);
|
await user.KickAsync(reason);
|
||||||
break;
|
break;
|
||||||
case PunishmentAction.Ban:
|
case PunishmentAction.Ban:
|
||||||
|
banPrune = await GetBanPruneAsync(user.GuildId) ?? 7;
|
||||||
if (minutes == 0)
|
if (minutes == 0)
|
||||||
await guild.AddBanAsync(user, reason: reason, pruneDays: 7);
|
await guild.AddBanAsync(user, reason: reason, pruneDays: banPrune);
|
||||||
else
|
else
|
||||||
await _mute.TimedBan(user.Guild, user, TimeSpan.FromMinutes(minutes), reason);
|
await _mute.TimedBan(user.Guild, user, TimeSpan.FromMinutes(minutes), reason, banPrune);
|
||||||
break;
|
break;
|
||||||
case PunishmentAction.Softban:
|
case PunishmentAction.Softban:
|
||||||
await guild.AddBanAsync(user, 7, $"Softban | {reason}");
|
banPrune = await GetBanPruneAsync(user.GuildId) ?? 7;
|
||||||
|
await guild.AddBanAsync(user, banPrune, $"Softban | {reason}");
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await guild.RemoveBanAsync(user);
|
await guild.RemoveBanAsync(user);
|
||||||
@@ -193,6 +197,9 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
case PunishmentAction.Warn:
|
case PunishmentAction.Warn:
|
||||||
await Warn(guild, user.Id, mod, 1, reason);
|
await Warn(guild, user.Id, mod, 1, reason);
|
||||||
break;
|
break;
|
||||||
|
case PunishmentAction.TimeOut:
|
||||||
|
await user.SetTimeOutAsync(TimeSpan.FromMinutes(minutes));
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -224,6 +231,8 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
return botUser.GuildPermissions.MuteMembers;
|
return botUser.GuildPermissions.MuteMembers;
|
||||||
case PunishmentAction.AddRole:
|
case PunishmentAction.AddRole:
|
||||||
return botUser.GuildPermissions.ManageRoles;
|
return botUser.GuildPermissions.ManageRoles;
|
||||||
|
case PunishmentAction.TimeOut:
|
||||||
|
return botUser.GuildPermissions.ModerateMembers;
|
||||||
default:
|
default:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@@ -351,7 +360,7 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
await uow.Warnings.ForgiveAll(guildId, userId, moderator);
|
await uow.Warnings.ForgiveAll(guildId, userId, moderator);
|
||||||
else
|
else
|
||||||
toReturn = uow.Warnings.Forgive(guildId, userId, moderator, index - 1);
|
toReturn = uow.Warnings.Forgive(guildId, userId, moderator, index - 1);
|
||||||
uow.SaveChanges();
|
await uow.SaveChangesAsync();
|
||||||
return toReturn;
|
return toReturn;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -372,6 +381,9 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
if (punish is PunishmentAction.AddRole && role is null)
|
if (punish is PunishmentAction.AddRole && role is null)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
if (punish is PunishmentAction.TimeOut && time is null)
|
||||||
|
return false;
|
||||||
|
|
||||||
using var uow = _db.GetDbContext();
|
using var uow = _db.GetDbContext();
|
||||||
var ps = uow.GuildConfigsForId(guildId, set => set.Include(x => x.WarnPunishments)).WarnPunishments;
|
var ps = uow.GuildConfigsForId(guildId, set => set.Include(x => x.WarnPunishments)).WarnPunishments;
|
||||||
var toDelete = ps.Where(x => x.Count == number);
|
var toDelete = ps.Where(x => x.Count == number);
|
||||||
@@ -481,6 +493,37 @@ public class UserPunishService : INService, IReadyExecutor
|
|||||||
uow.SaveChanges();
|
uow.SaveChanges();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task SetBanPruneAsync(ulong guildId, int? pruneDays)
|
||||||
|
{
|
||||||
|
await using var ctx = _db.GetDbContext();
|
||||||
|
await ctx.BanTemplates
|
||||||
|
.ToLinqToDBTable()
|
||||||
|
.InsertOrUpdateAsync(() => new()
|
||||||
|
{
|
||||||
|
GuildId = guildId,
|
||||||
|
Text = null,
|
||||||
|
DateAdded = DateTime.UtcNow,
|
||||||
|
PruneDays = pruneDays
|
||||||
|
},
|
||||||
|
old => new()
|
||||||
|
{
|
||||||
|
PruneDays = pruneDays
|
||||||
|
},
|
||||||
|
() => new()
|
||||||
|
{
|
||||||
|
GuildId = guildId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<int?> GetBanPruneAsync(ulong guildId)
|
||||||
|
{
|
||||||
|
await using var ctx = _db.GetDbContext();
|
||||||
|
return await ctx.BanTemplates
|
||||||
|
.Where(x => x.GuildId == guildId)
|
||||||
|
.Select(x => x.PruneDays)
|
||||||
|
.FirstOrDefaultAsyncLinqToDB();
|
||||||
|
}
|
||||||
|
|
||||||
public SmartText GetBanUserDmEmbed(
|
public SmartText GetBanUserDmEmbed(
|
||||||
ICommandContext context,
|
ICommandContext context,
|
||||||
IGuildUser target,
|
IGuildUser target,
|
||||||
|
@@ -1,7 +1,5 @@
|
|||||||
#nullable disable
|
#nullable disable
|
||||||
|
|
||||||
using Nadeko.Common;
|
|
||||||
|
|
||||||
namespace NadekoBot.Modules.NadekoExpressions;
|
namespace NadekoBot.Modules.NadekoExpressions;
|
||||||
|
|
||||||
[Name("Expressions")]
|
[Name("Expressions")]
|
||||||
@@ -25,15 +23,10 @@ public partial class NadekoExpressions : NadekoModule<NadekoExpressionsService>
|
|||||||
=> (ctx.Guild is null && _creds.IsOwner(ctx.User))
|
=> (ctx.Guild is null && _creds.IsOwner(ctx.User))
|
||||||
|| (ctx.Guild is not null && ((IGuildUser)ctx.User).GuildPermissions.Administrator);
|
|| (ctx.Guild is not null && ((IGuildUser)ctx.User).GuildPermissions.Administrator);
|
||||||
|
|
||||||
[Cmd]
|
private async Task ExprAddInternalAsync(string key, string message)
|
||||||
public async Task ExprAdd(string key, [Leftover] string message)
|
|
||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(message) || string.IsNullOrWhiteSpace(key))
|
if (string.IsNullOrWhiteSpace(message) || string.IsNullOrWhiteSpace(key))
|
||||||
return;
|
|
||||||
|
|
||||||
if (!AdminInGuildOrOwnerInDm())
|
|
||||||
{
|
{
|
||||||
await ReplyErrorLocalizedAsync(strs.expr_insuff_perms);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -48,12 +41,43 @@ public partial class NadekoExpressions : NadekoModule<NadekoExpressionsService>
|
|||||||
message.Length > 1024 ? GetText(strs.redacted_too_long) : message));
|
message.Length > 1024 ? GetText(strs.redacted_too_long) : message));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[UserPerm(GuildPerm.Administrator)]
|
||||||
|
public async Task ExprAddServer(string key, [Leftover] string message)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(message) || string.IsNullOrWhiteSpace(key))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ExprAddInternalAsync(key, message);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
public async Task ExprAdd(string key, [Leftover] string message)
|
||||||
|
{
|
||||||
|
if (string.IsNullOrWhiteSpace(message) || string.IsNullOrWhiteSpace(key))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!AdminInGuildOrOwnerInDm())
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.expr_insuff_perms);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ExprAddInternalAsync(key, message);
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
public async Task ExprEdit(kwum id, [Leftover] string message)
|
public async Task ExprEdit(kwum id, [Leftover] string message)
|
||||||
{
|
{
|
||||||
var channel = ctx.Channel as ITextChannel;
|
var channel = ctx.Channel as ITextChannel;
|
||||||
if (string.IsNullOrWhiteSpace(message) || id < 0)
|
if (string.IsNullOrWhiteSpace(message) || id < 0)
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if ((channel is null && !_creds.IsOwner(ctx.User))
|
if ((channel is null && !_creds.IsOwner(ctx.User))
|
||||||
|| (channel is not null && !((IGuildUser)ctx.User).GuildPermissions.Administrator))
|
|| (channel is not null && !((IGuildUser)ctx.User).GuildPermissions.Administrator))
|
||||||
@@ -74,15 +98,19 @@ public partial class NadekoExpressions : NadekoModule<NadekoExpressionsService>
|
|||||||
message.Length > 1024 ? GetText(strs.redacted_too_long) : message));
|
message.Length > 1024 ? GetText(strs.redacted_too_long) : message));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
await ReplyErrorLocalizedAsync(strs.expr_no_found_id);
|
await ReplyErrorLocalizedAsync(strs.expr_no_found_id);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
public async Task ExprList(int page = 1)
|
public async Task ExprList(int page = 1)
|
||||||
{
|
{
|
||||||
if (--page < 0 || page > 999)
|
if (--page < 0 || page > 999)
|
||||||
|
{
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var expressions = _service.GetExpressionsFor(ctx.Guild?.Id);
|
var expressions = _service.GetExpressionsFor(ctx.Guild?.Id);
|
||||||
|
|
||||||
@@ -153,8 +181,10 @@ public partial class NadekoExpressions : NadekoModule<NadekoExpressionsService>
|
|||||||
.AddField(GetText(strs.response), ex.Response.TrimTo(1024)));
|
.AddField(GetText(strs.response), ex.Response.TrimTo(1024)));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
await ReplyErrorLocalizedAsync(strs.expr_no_found_id);
|
await ReplyErrorLocalizedAsync(strs.expr_no_found_id);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
public async Task ExprReact(kwum id, params string[] emojiStrs)
|
public async Task ExprReact(kwum id, params string[] emojiStrs)
|
||||||
@@ -192,8 +222,10 @@ public partial class NadekoExpressions : NadekoModule<NadekoExpressionsService>
|
|||||||
succ.Add(emojiStr);
|
succ.Add(emojiStr);
|
||||||
|
|
||||||
if (succ.Count >= 3)
|
if (succ.Count >= 3)
|
||||||
|
{
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
catch { }
|
catch { }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -11,10 +11,14 @@ public partial class Gambling
|
|||||||
public partial class BankCommands : GamblingModule<IBankService>
|
public partial class BankCommands : GamblingModule<IBankService>
|
||||||
{
|
{
|
||||||
private readonly IBankService _bank;
|
private readonly IBankService _bank;
|
||||||
|
private readonly DiscordSocketClient _client;
|
||||||
|
|
||||||
public BankCommands(GamblingConfigService gcs, IBankService bank) : base(gcs)
|
public BankCommands(GamblingConfigService gcs,
|
||||||
|
IBankService bank,
|
||||||
|
DiscordSocketClient client) : base(gcs)
|
||||||
{
|
{
|
||||||
_bank = bank;
|
_bank = bank;
|
||||||
|
_client = client;
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
@@ -68,5 +72,50 @@ public partial class Gambling
|
|||||||
await ReplyErrorLocalizedAsync(strs.cant_dm);
|
await ReplyErrorLocalizedAsync(strs.cant_dm);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task BankTakeInternalAsync(long amount, ulong userId)
|
||||||
|
{
|
||||||
|
if (await _bank.TakeAsync(userId, amount))
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.take_fail(N(amount),
|
||||||
|
_client.GetUser(userId)?.ToString()
|
||||||
|
?? userId.ToString(),
|
||||||
|
CurrencySign));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task BankAwardInternalAsync(long amount, ulong userId)
|
||||||
|
{
|
||||||
|
if (await _bank.AwardAsync(userId, amount))
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.take_fail(N(amount),
|
||||||
|
_client.GetUser(userId)?.ToString()
|
||||||
|
?? userId.ToString(),
|
||||||
|
CurrencySign));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[OwnerOnly]
|
||||||
|
[Priority(1)]
|
||||||
|
public async Task BankTake(long amount, [Leftover] IUser user)
|
||||||
|
=> await BankTakeInternalAsync(amount, user.Id);
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[OwnerOnly]
|
||||||
|
[Priority(0)]
|
||||||
|
public async Task BankTake(long amount, ulong userId)
|
||||||
|
=> await BankTakeInternalAsync(amount, userId);
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[OwnerOnly]
|
||||||
|
public async Task BankAward(long amount, [Leftover] IUser user)
|
||||||
|
=> await BankAwardInternalAsync(amount, user.Id);
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -15,6 +15,48 @@ public sealed class BankService : IBankService, INService
|
|||||||
_db = db;
|
_db = db;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task<bool> AwardAsync(ulong userId, long amount)
|
||||||
|
{
|
||||||
|
if (amount <= 0)
|
||||||
|
throw new ArgumentOutOfRangeException(nameof(amount));
|
||||||
|
|
||||||
|
await using var ctx = _db.GetDbContext();
|
||||||
|
await ctx.BankUsers
|
||||||
|
.ToLinqToDBTable()
|
||||||
|
.InsertOrUpdateAsync(() => new()
|
||||||
|
{
|
||||||
|
UserId = userId,
|
||||||
|
Balance = amount
|
||||||
|
},
|
||||||
|
(old) => new()
|
||||||
|
{
|
||||||
|
Balance = old.Balance + amount
|
||||||
|
},
|
||||||
|
() => new()
|
||||||
|
{
|
||||||
|
UserId = userId
|
||||||
|
});
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<bool> TakeAsync(ulong userId, long amount)
|
||||||
|
{
|
||||||
|
if (amount <= 0)
|
||||||
|
throw new ArgumentOutOfRangeException(nameof(amount));
|
||||||
|
|
||||||
|
await using var ctx = _db.GetDbContext();
|
||||||
|
var rows = await ctx.BankUsers
|
||||||
|
.ToLinqToDBTable()
|
||||||
|
.Where(x => x.UserId == userId && x.Balance >= amount)
|
||||||
|
.UpdateAsync((old) => new()
|
||||||
|
{
|
||||||
|
Balance = old.Balance - amount
|
||||||
|
});
|
||||||
|
|
||||||
|
return rows > 0;
|
||||||
|
}
|
||||||
|
|
||||||
public async Task<bool> DepositAsync(ulong userId, long amount)
|
public async Task<bool> DepositAsync(ulong userId, long amount)
|
||||||
{
|
{
|
||||||
if (amount <= 0)
|
if (amount <= 0)
|
||||||
|
@@ -5,4 +5,6 @@ public interface IBankService
|
|||||||
Task<bool> DepositAsync(ulong userId, long amount);
|
Task<bool> DepositAsync(ulong userId, long amount);
|
||||||
Task<bool> WithdrawAsync(ulong userId, long amount);
|
Task<bool> WithdrawAsync(ulong userId, long amount);
|
||||||
Task<long> GetBalanceAsync(ulong userId);
|
Task<long> GetBalanceAsync(ulong userId);
|
||||||
|
Task<bool> AwardAsync(ulong userId, long amount);
|
||||||
|
Task<bool> TakeAsync(ulong userId, long amount);
|
||||||
}
|
}
|
@@ -1,13 +0,0 @@
|
|||||||
#nullable disable
|
|
||||||
namespace NadekoBot.Modules.Gambling;
|
|
||||||
|
|
||||||
public class CashInteraction : NInteraction
|
|
||||||
{
|
|
||||||
protected override NadekoInteractionData Data
|
|
||||||
=> new NadekoInteractionData(new Emoji("🏦"), "cash:bank_show_balance");
|
|
||||||
|
|
||||||
public CashInteraction(DiscordSocketClient client, ulong userId, Func<SocketMessageComponent, Task> action)
|
|
||||||
: base(client, userId, action)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@@ -99,26 +99,7 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
PrettyName = "Timely"
|
PrettyName = "Timely"
|
||||||
};
|
};
|
||||||
|
|
||||||
public class RemindMeInteraction : NInteraction
|
private async Task RemindTimelyAction(SocketMessageComponent smc, DateTime when)
|
||||||
{
|
|
||||||
public RemindMeInteraction(
|
|
||||||
[NotNull] DiscordSocketClient client,
|
|
||||||
ulong userId,
|
|
||||||
[NotNull] Func<SocketMessageComponent, Task> action)
|
|
||||||
: base(client, userId, action)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override NadekoInteractionData Data
|
|
||||||
=> new NadekoInteractionData(
|
|
||||||
Emote: Emoji.Parse("⏰"),
|
|
||||||
CustomId: "timely:remind_me",
|
|
||||||
Text: "Remind me"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Func<SocketMessageComponent, Task> RemindTimelyAction(DateTime when)
|
|
||||||
=> async smc =>
|
|
||||||
{
|
{
|
||||||
var tt = TimestampTag.FromDateTime(when, TimestampTagStyles.Relative);
|
var tt = TimestampTag.FromDateTime(when, TimestampTagStyles.Relative);
|
||||||
|
|
||||||
@@ -130,7 +111,7 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
GetText(strs.timely_time));
|
GetText(strs.timely_time));
|
||||||
|
|
||||||
await smc.RespondConfirmAsync(_eb, GetText(strs.remind_timely(tt)), ephemeral: true);
|
await smc.RespondConfirmAsync(_eb, GetText(strs.remind_timely(tt)), ephemeral: true);
|
||||||
};
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
public async Task Timely()
|
public async Task Timely()
|
||||||
@@ -147,7 +128,7 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
{
|
{
|
||||||
var now = DateTime.UtcNow;
|
var now = DateTime.UtcNow;
|
||||||
var relativeTag = TimestampTag.FromDateTime(now.Add(rem), TimestampTagStyles.Relative);
|
var relativeTag = TimestampTag.FromDateTime(now.Add(rem), TimestampTagStyles.Relative);
|
||||||
await ReplyErrorLocalizedAsync(strs.timely_already_claimed(relativeTag));
|
await ReplyPendingLocalizedAsync(strs.timely_already_claimed(relativeTag));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -157,11 +138,17 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
|
|
||||||
await _cs.AddAsync(ctx.User.Id, val, new("timely", "claim"));
|
await _cs.AddAsync(ctx.User.Id, val, new("timely", "claim"));
|
||||||
|
|
||||||
var inter = new RemindMeInteraction(_client,
|
var inter = _inter
|
||||||
ctx.User.Id,
|
.Create(ctx.User.Id,
|
||||||
RemindTimelyAction(DateTime.UtcNow.Add(TimeSpan.FromHours(period))));
|
new SimpleInteraction<DateTime>(
|
||||||
|
new ButtonBuilder(
|
||||||
|
label: "Remind me",
|
||||||
|
emote: Emoji.Parse("⏰"),
|
||||||
|
customId: "timely:remind_me"),
|
||||||
|
RemindTimelyAction,
|
||||||
|
DateTime.UtcNow.Add(TimeSpan.FromHours(period))));
|
||||||
|
|
||||||
await ReplyConfirmLocalizedAsync(strs.timely(N(val), period), inter.GetInteraction());
|
await ReplyConfirmLocalizedAsync(strs.timely(N(val), period), inter);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
@@ -362,7 +349,7 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
await ReplyConfirmLocalizedAsync(strs.has(Format.Code(userId.ToString()), cur));
|
await ReplyConfirmLocalizedAsync(strs.has(Format.Code(userId.ToString()), cur));
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task BankAction(SocketMessageComponent smc)
|
private async Task BankAction(SocketMessageComponent smc, object _)
|
||||||
{
|
{
|
||||||
var balance = await _bank.GetBalanceAsync(ctx.User.Id);
|
var balance = await _bank.GetBalanceAsync(ctx.User.Id);
|
||||||
|
|
||||||
@@ -372,8 +359,12 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
.Pipe(text => smc.RespondConfirmAsync(_eb, text, ephemeral: true));
|
.Pipe(text => smc.RespondConfirmAsync(_eb, text, ephemeral: true));
|
||||||
}
|
}
|
||||||
|
|
||||||
private NadekoButtonInteraction CreateCashInteraction()
|
private NadekoInteraction CreateCashInteraction()
|
||||||
=> new CashInteraction(_client, ctx.User.Id, BankAction).GetInteraction();
|
=> _inter.Create<object>(ctx.User.Id,
|
||||||
|
new(new(
|
||||||
|
customId: "cash:bank_show_balance",
|
||||||
|
emote: new Emoji("🏦")),
|
||||||
|
BankAction));
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[Priority(1)]
|
[Priority(1)]
|
||||||
@@ -875,11 +866,12 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
public enum GambleTestTarget
|
public enum GambleTestTarget
|
||||||
{
|
{
|
||||||
Slot,
|
Slot,
|
||||||
|
Betroll,
|
||||||
|
Betflip,
|
||||||
|
BetflipT,
|
||||||
BetDraw,
|
BetDraw,
|
||||||
BetDrawHL,
|
BetDrawHL,
|
||||||
BetDrawRB,
|
BetDrawRB,
|
||||||
Betflip,
|
|
||||||
BetflipT,
|
|
||||||
Lula,
|
Lula,
|
||||||
Rps,
|
Rps,
|
||||||
}
|
}
|
||||||
@@ -920,6 +912,7 @@ public partial class Gambling : GamblingModule<GamblingService>
|
|||||||
GambleTestTarget.BetflipT => (await _gs.BetFlipAsync(ctx.User.Id, 0, 1)).AsT0.Multiplier,
|
GambleTestTarget.BetflipT => (await _gs.BetFlipAsync(ctx.User.Id, 0, 1)).AsT0.Multiplier,
|
||||||
GambleTestTarget.Lula => (await _gs.LulaAsync(ctx.User.Id, 0)).AsT0.Multiplier,
|
GambleTestTarget.Lula => (await _gs.LulaAsync(ctx.User.Id, 0)).AsT0.Multiplier,
|
||||||
GambleTestTarget.Rps => (await _gs.RpsAsync(ctx.User.Id, 0, (byte)(i % 3))).AsT0.Multiplier,
|
GambleTestTarget.Rps => (await _gs.RpsAsync(ctx.User.Id, 0, (byte)(i % 3))).AsT0.Multiplier,
|
||||||
|
GambleTestTarget.Betroll => (await _gs.BetRollAsync(ctx.User.Id, 0)).AsT0.Multiplier,
|
||||||
_ => throw new ArgumentOutOfRangeException(nameof(target))
|
_ => throw new ArgumentOutOfRangeException(nameof(target))
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -27,12 +27,6 @@ public partial class Gambling
|
|||||||
private static decimal totalBet;
|
private static decimal totalBet;
|
||||||
private static decimal totalPaidOut;
|
private static decimal totalPaidOut;
|
||||||
|
|
||||||
private static readonly ConcurrentHashSet<ulong> _runningUsers = new();
|
|
||||||
|
|
||||||
//here is a payout chart
|
|
||||||
//https://lh6.googleusercontent.com/-i1hjAJy_kN4/UswKxmhrbPI/AAAAAAAAB1U/82wq_4ZZc-Y/DE6B0895-6FC1-48BE-AC4F-14D1B91AB75B.jpg
|
|
||||||
//thanks to judge for helping me with this
|
|
||||||
|
|
||||||
private readonly IImageCache _images;
|
private readonly IImageCache _images;
|
||||||
private readonly FontProvider _fonts;
|
private readonly FontProvider _fonts;
|
||||||
private readonly DbService _db;
|
private readonly DbService _db;
|
||||||
@@ -73,16 +67,6 @@ public partial class Gambling
|
|||||||
|
|
||||||
await ctx.Channel.EmbedAsync(embed);
|
await ctx.Channel.EmbedAsync(embed);
|
||||||
}
|
}
|
||||||
public sealed class SlotInteraction : NInteraction
|
|
||||||
{
|
|
||||||
public SlotInteraction(DiscordSocketClient client, ulong userId, Func<SocketMessageComponent, Task> action) : base(client, userId, action)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
protected override NadekoInteractionData Data { get; } = new(Emoji.Parse("🔁"),
|
|
||||||
"slot:again",
|
|
||||||
"Pull Again");
|
|
||||||
}
|
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
public async Task Slot(ShmartNumber amount)
|
public async Task Slot(ShmartNumber amount)
|
||||||
@@ -94,18 +78,13 @@ public partial class Gambling
|
|||||||
|
|
||||||
await ctx.Channel.TriggerTypingAsync();
|
await ctx.Channel.TriggerTypingAsync();
|
||||||
|
|
||||||
if (!_runningUsers.Add(ctx.User.Id))
|
|
||||||
return;
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (await InternalSlotAsync(amount) is not SlotResult result)
|
if (await InternalSlotAsync(amount) is not SlotResult result)
|
||||||
{
|
{
|
||||||
await ReplyErrorLocalizedAsync(strs.not_enough(CurrencySign));
|
await ReplyErrorLocalizedAsync(strs.not_enough(CurrencySign));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var msg = GetSlotMessageInternal(result);
|
var text = GetSlotMessageTextInternal(result);
|
||||||
|
|
||||||
using var image = await GenerateSlotImageAsync(amount, result);
|
using var image = await GenerateSlotImageAsync(amount, result);
|
||||||
await using var imgStream = await image.ToStreamAsync();
|
await using var imgStream = await image.ToStreamAsync();
|
||||||
@@ -113,24 +92,20 @@ public partial class Gambling
|
|||||||
|
|
||||||
var eb = _eb.Create(ctx)
|
var eb = _eb.Create(ctx)
|
||||||
.WithAuthor(ctx.User)
|
.WithAuthor(ctx.User)
|
||||||
.WithDescription(Format.Bold(msg))
|
.WithDescription(Format.Bold(text))
|
||||||
.WithImageUrl($"attachment://result.png")
|
.WithImageUrl($"attachment://result.png")
|
||||||
.WithOkColor();
|
.WithOkColor();
|
||||||
|
|
||||||
// var inter = slotInteraction.GetInteraction();
|
var bb = new ButtonBuilder(emote: Emoji.Parse("🔁"), customId: "slot:again", label: "Pull Again");
|
||||||
await ctx.Channel.SendFileAsync(imgStream,
|
var si = new SimpleInteraction<ShmartNumber>(bb, (_, amount) => Slot(amount), amount);
|
||||||
"result.png",
|
|
||||||
embed: eb.Build()
|
|
||||||
// components: inter.CreateComponent()
|
|
||||||
);
|
|
||||||
|
|
||||||
// await inter.RunAsync(resMsg);
|
var inter = _inter.Create(ctx.User.Id, si);
|
||||||
}
|
var msg = await ctx.Channel.SendFileAsync(imgStream,
|
||||||
finally
|
"result.png",
|
||||||
{
|
embed: eb.Build(),
|
||||||
await Task.Delay(1000);
|
components: inter.CreateComponent()
|
||||||
_runningUsers.TryRemove(ctx.User.Id);
|
);
|
||||||
}
|
await inter.RunAsync(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
// private SlotInteraction CreateSlotInteractionIntenal(long amount)
|
// private SlotInteraction CreateSlotInteractionIntenal(long amount)
|
||||||
@@ -181,7 +156,7 @@ public partial class Gambling
|
|||||||
// });
|
// });
|
||||||
// }
|
// }
|
||||||
|
|
||||||
private string GetSlotMessageInternal(SlotResult result)
|
private string GetSlotMessageTextInternal(SlotResult result)
|
||||||
{
|
{
|
||||||
var multi = result.Multiplier.ToString("0.##");
|
var multi = result.Multiplier.ToString("0.##");
|
||||||
var msg = result.WinType switch
|
var msg = result.WinType switch
|
||||||
@@ -201,7 +176,6 @@ public partial class Gambling
|
|||||||
|
|
||||||
if (!maybeResult.TryPickT0(out var result, out var error))
|
if (!maybeResult.TryPickT0(out var result, out var error))
|
||||||
{
|
{
|
||||||
await ReplyErrorLocalizedAsync(strs.not_enough(CurrencySign));
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -57,11 +57,14 @@ public sealed class TriviaGame
|
|||||||
// loop until game is stopped
|
// loop until game is stopped
|
||||||
// each iteration is one round
|
// each iteration is one round
|
||||||
var firstRun = true;
|
var firstRun = true;
|
||||||
|
try
|
||||||
|
{
|
||||||
while (!_isStopped)
|
while (!_isStopped)
|
||||||
{
|
{
|
||||||
if (errorCount >= 5)
|
if (errorCount >= 5)
|
||||||
{
|
{
|
||||||
Log.Warning("Trivia errored 5 times and will quit");
|
Log.Warning("Trivia errored 5 times and will quit");
|
||||||
|
await OnEnded(this);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -77,7 +80,7 @@ public sealed class TriviaGame
|
|||||||
|
|
||||||
var maybeQuestion = await _questionPool.GetQuestionAsync();
|
var maybeQuestion = await _questionPool.GetQuestionAsync();
|
||||||
|
|
||||||
if(!(maybeQuestion is TriviaQuestion question))
|
if (!(maybeQuestion is TriviaQuestion question))
|
||||||
{
|
{
|
||||||
// if question is null (ran out of question, or other bugg ) - stop
|
// if question is null (ran out of question, or other bugg ) - stop
|
||||||
break;
|
break;
|
||||||
@@ -87,7 +90,8 @@ public sealed class TriviaGame
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
// clear out all of the past guesses
|
// clear out all of the past guesses
|
||||||
while (_inputs.Reader.TryRead(out _)) ;
|
while (_inputs.Reader.TryRead(out _))
|
||||||
|
;
|
||||||
|
|
||||||
await OnQuestion(this, question);
|
await OnQuestion(this, question);
|
||||||
}
|
}
|
||||||
@@ -169,11 +173,17 @@ public sealed class TriviaGame
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
// make sure game is set as ended
|
// make sure game is set as ended
|
||||||
_isStopped = true;
|
_isStopped = true;
|
||||||
|
_ = OnEnded(this);
|
||||||
await OnEnded(this);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public IReadOnlyList<(ulong User, int points)> GetLeaderboard()
|
public IReadOnlyList<(ulong User, int points)> GetLeaderboard()
|
||||||
|
@@ -424,7 +424,7 @@ public partial class Help : NadekoModule<HelpService>
|
|||||||
}
|
}
|
||||||
catch (Exception)
|
catch (Exception)
|
||||||
{
|
{
|
||||||
Log.Information("No old version list found. Creating a new one.");
|
Log.Information("No old version list found. Creating a new one");
|
||||||
}
|
}
|
||||||
|
|
||||||
var versionList = JsonSerializer.Deserialize<List<string>>(versionListString);
|
var versionList = JsonSerializer.Deserialize<List<string>>(versionListString);
|
||||||
@@ -469,7 +469,7 @@ public partial class Help : NadekoModule<HelpService>
|
|||||||
"https://nadekobot.readthedocs.io/en/latest/"));
|
"https://nadekobot.readthedocs.io/en/latest/"));
|
||||||
|
|
||||||
|
|
||||||
private Task SelfhostAction(SocketMessageComponent smc)
|
private Task SelfhostAction(SocketMessageComponent smc, object _)
|
||||||
=> smc.RespondConfirmAsync(_eb,
|
=> smc.RespondConfirmAsync(_eb,
|
||||||
@"- In case you don't want or cannot Donate to NadekoBot project, but you
|
@"- In case you don't want or cannot Donate to NadekoBot project, but you
|
||||||
- NadekoBot is a completely free and fully [open source](https://gitlab.com/kwoth/nadekobot) project which means you can run your own ""selfhosted"" instance on your computer or server for free.
|
- NadekoBot is a completely free and fully [open source](https://gitlab.com/kwoth/nadekobot) project which means you can run your own ""selfhosted"" instance on your computer or server for free.
|
||||||
@@ -484,7 +484,13 @@ public partial class Help : NadekoModule<HelpService>
|
|||||||
[OnlyPublicBot]
|
[OnlyPublicBot]
|
||||||
public async Task Donate()
|
public async Task Donate()
|
||||||
{
|
{
|
||||||
var selfhostInter = new DonateSelfhostingInteraction(_client, ctx.User.Id, SelfhostAction);
|
// => new NadekoInteractionData(new Emoji("🖥️"), "donate:selfhosting", "Selfhosting");
|
||||||
|
var selfhostInter = _inter.Create(ctx.User.Id,
|
||||||
|
new SimpleInteraction<object>(new ButtonBuilder(
|
||||||
|
emote: new Emoji("🖥️"),
|
||||||
|
customId: "donate:selfhosting",
|
||||||
|
label: "Selfhosting"),
|
||||||
|
SelfhostAction));
|
||||||
|
|
||||||
var eb = _eb.Create(ctx)
|
var eb = _eb.Create(ctx)
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
@@ -525,7 +531,7 @@ Nadeko will DM you the welcome instructions, and you may start using the patron-
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
await (await ctx.User.CreateDMChannelAsync()).EmbedAsync(eb, inter: selfhostInter.GetInteraction());
|
await (await ctx.User.CreateDMChannelAsync()).EmbedAsync(eb, inter: selfhostInter);
|
||||||
_ = ctx.OkAsync();
|
_ = ctx.OkAsync();
|
||||||
}
|
}
|
||||||
catch
|
catch
|
||||||
|
@@ -1,12 +0,0 @@
|
|||||||
namespace NadekoBot.Modules.Help;
|
|
||||||
|
|
||||||
public class DonateSelfhostingInteraction : NInteraction
|
|
||||||
{
|
|
||||||
protected override NadekoInteractionData Data
|
|
||||||
=> new NadekoInteractionData(new Emoji("🖥️"), "donate:selfhosting", "Selfhosting");
|
|
||||||
|
|
||||||
public DonateSelfhostingInteraction(DiscordSocketClient client, ulong userId, Func<SocketMessageComponent, Task> action)
|
|
||||||
: base(client, userId, action)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,12 +0,0 @@
|
|||||||
namespace NadekoBot.Modules.Help;
|
|
||||||
|
|
||||||
public class DonateTroubleshootInteraction : NInteraction
|
|
||||||
{
|
|
||||||
protected override NadekoInteractionData Data
|
|
||||||
=> new NadekoInteractionData(new Emoji("❓"), "donate:troubleshoot", "Troubleshoot");
|
|
||||||
|
|
||||||
public DonateTroubleshootInteraction(DiscordSocketClient client, ulong userId, Func<SocketMessageComponent, Task> action)
|
|
||||||
: base(client, userId, action)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,5 +1,4 @@
|
|||||||
#nullable disable
|
#nullable disable
|
||||||
using Nadeko.Common;
|
|
||||||
using NadekoBot.Modules.Music.Services;
|
using NadekoBot.Modules.Music.Services;
|
||||||
using NadekoBot.Services.Database.Models;
|
using NadekoBot.Services.Database.Models;
|
||||||
|
|
||||||
|
@@ -274,7 +274,7 @@ public sealed class YtdlYoutubeResolver : IYoutubeResolver
|
|||||||
Log.Information("Resolving youtube song by search term: {YoutubeQuery}", query);
|
Log.Information("Resolving youtube song by search term: {YoutubeQuery}", query);
|
||||||
|
|
||||||
var cachedData = await _trackCacher.GetCachedDataByQueryAsync(query, MusicPlatform.Youtube);
|
var cachedData = await _trackCacher.GetCachedDataByQueryAsync(query, MusicPlatform.Youtube);
|
||||||
if (cachedData is null)
|
if (cachedData is null || string.IsNullOrWhiteSpace(cachedData.Title))
|
||||||
{
|
{
|
||||||
var stringData = await _ytdlSearchOperation.GetDataAsync(query);
|
var stringData = await _ytdlSearchOperation.GetDataAsync(query);
|
||||||
var trackData = ResolveYtdlData(stringData);
|
var trackData = ResolveYtdlData(stringData);
|
||||||
|
@@ -98,6 +98,33 @@ public partial class Utility
|
|||||||
await ctx.Channel.EmbedAsync(embed);
|
await ctx.Channel.EmbedAsync(embed);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireUserPermission(GuildPermission.ManageRoles)]
|
||||||
|
public async Task RoleInfo([Leftover] SocketRole role)
|
||||||
|
{
|
||||||
|
if (role.IsEveryone)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var createdAt = new DateTime(2015, 1, 1, 0, 0, 0, 0, DateTimeKind.Utc)
|
||||||
|
.AddMilliseconds(role.Id >> 22);
|
||||||
|
var usercount = role.Members.LongCount();
|
||||||
|
var embed = _eb.Create()
|
||||||
|
.WithTitle(role.Name.TrimTo(128))
|
||||||
|
.WithDescription(role.Permissions.ToList().Join(" | "))
|
||||||
|
.AddField(GetText(strs.id), role.Id.ToString(), true)
|
||||||
|
.AddField(GetText(strs.created_at), $"{createdAt:dd.MM.yyyy HH:mm}", true)
|
||||||
|
.AddField(GetText(strs.users), usercount.ToString(), true)
|
||||||
|
.AddField(GetText(strs.color), $"#{role.Color.R:X2}{role.Color.G:X2}{role.Color.B:X2}", true)
|
||||||
|
.AddField(GetText(strs.mentionable), role.IsMentionable.ToString(), true)
|
||||||
|
.AddField(GetText(strs.hoisted), role.IsHoisted.ToString(), true)
|
||||||
|
.WithOkColor();
|
||||||
|
|
||||||
|
if (!string.IsNullOrWhiteSpace(role.GetIconUrl()))
|
||||||
|
embed = embed.WithThumbnailUrl(role.GetIconUrl());
|
||||||
|
|
||||||
|
await ctx.Channel.EmbedAsync(embed);
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
public async Task UserInfo(IGuildUser usr = null)
|
public async Task UserInfo(IGuildUser usr = null)
|
||||||
@@ -111,8 +138,10 @@ public partial class Utility
|
|||||||
if (!string.IsNullOrWhiteSpace(user.Nickname))
|
if (!string.IsNullOrWhiteSpace(user.Nickname))
|
||||||
embed.AddField(GetText(strs.nickname), user.Nickname, true);
|
embed.AddField(GetText(strs.nickname), user.Nickname, true);
|
||||||
|
|
||||||
|
var joinedAt = GetJoinedAt(user);
|
||||||
|
|
||||||
embed.AddField(GetText(strs.id), user.Id.ToString(), true)
|
embed.AddField(GetText(strs.id), user.Id.ToString(), true)
|
||||||
.AddField(GetText(strs.joined_server), $"{user.JoinedAt?.ToString("dd.MM.yyyy HH:mm") ?? "?"}", true)
|
.AddField(GetText(strs.joined_server), $"{joinedAt?.ToString("dd.MM.yyyy HH:mm") ?? "?"}", true)
|
||||||
.AddField(GetText(strs.joined_discord), $"{user.CreatedAt:dd.MM.yyyy HH:mm}", true)
|
.AddField(GetText(strs.joined_discord), $"{user.CreatedAt:dd.MM.yyyy HH:mm}", true)
|
||||||
.AddField(GetText(strs.roles),
|
.AddField(GetText(strs.roles),
|
||||||
$"**({user.RoleIds.Count - 1})** - {string.Join("\n", user.GetRoles().Take(10).Where(r => r.Id != r.Guild.EveryoneRole.Id).Select(r => r.Name)).SanitizeMentions(true)}",
|
$"**({user.RoleIds.Count - 1})** - {string.Join("\n", user.GetRoles().Take(10).Where(r => r.Id != r.Guild.EveryoneRole.Id).Select(r => r.Name)).SanitizeMentions(true)}",
|
||||||
@@ -140,6 +169,18 @@ public partial class Utility
|
|||||||
await ctx.Channel.EmbedAsync(embed);
|
await ctx.Channel.EmbedAsync(embed);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private DateTimeOffset? GetJoinedAt(IGuildUser user)
|
||||||
|
{
|
||||||
|
var joinedAt = user.JoinedAt;
|
||||||
|
if (user.GuildId != 117523346618318850)
|
||||||
|
return joinedAt;
|
||||||
|
|
||||||
|
if (user.Id == 351244576092192778)
|
||||||
|
return new DateTimeOffset(2019, 12, 25, 9, 33, 0, TimeSpan.Zero);
|
||||||
|
|
||||||
|
return joinedAt;
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
|
@@ -716,7 +716,7 @@ public sealed class PatronageService
|
|||||||
{
|
{
|
||||||
Name = key.PrettyName,
|
Name = key.PrettyName,
|
||||||
Quota = 0,
|
Quota = 0,
|
||||||
IsPatronLimit = false,
|
IsPatronLimit = true,
|
||||||
};
|
};
|
||||||
|
|
||||||
return new()
|
return new()
|
||||||
|
6
src/NadekoBot/Modules/Utility/Quote/IQuoteService.cs
Normal file
6
src/NadekoBot/Modules/Utility/Quote/IQuoteService.cs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
namespace NadekoBot.Modules.Utility;
|
||||||
|
|
||||||
|
public interface IQuoteService
|
||||||
|
{
|
||||||
|
Task<int> DeleteAllAuthorQuotesAsync(ulong guildId, ulong userId);
|
||||||
|
}
|
@@ -1,5 +1,4 @@
|
|||||||
#nullable disable warnings
|
#nullable disable warnings
|
||||||
using Nadeko.Common;
|
|
||||||
using NadekoBot.Common.Yml;
|
using NadekoBot.Common.Yml;
|
||||||
using NadekoBot.Db;
|
using NadekoBot.Db;
|
||||||
using NadekoBot.Services.Database.Models;
|
using NadekoBot.Services.Database.Models;
|
||||||
@@ -34,11 +33,13 @@ public partial class Utility
|
|||||||
|
|
||||||
private readonly DbService _db;
|
private readonly DbService _db;
|
||||||
private readonly IHttpClientFactory _http;
|
private readonly IHttpClientFactory _http;
|
||||||
|
private readonly IQuoteService _qs;
|
||||||
|
|
||||||
public QuoteCommands(DbService db, IHttpClientFactory http)
|
public QuoteCommands(DbService db, IQuoteService qs, IHttpClientFactory http)
|
||||||
{
|
{
|
||||||
_db = db;
|
_db = db;
|
||||||
_http = http;
|
_http = http;
|
||||||
|
_qs = qs;
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
@@ -108,7 +109,7 @@ public partial class Utility
|
|||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
public async Task QuoteShow(int id)
|
public async Task QuoteShow(int id)
|
||||||
{
|
{
|
||||||
Quote quote;
|
Quote? quote;
|
||||||
await using (var uow = _db.GetDbContext())
|
await using (var uow = _db.GetDbContext())
|
||||||
{
|
{
|
||||||
quote = uow.Quotes.GetById(id);
|
quote = uow.Quotes.GetById(id);
|
||||||
@@ -256,6 +257,28 @@ public partial class Utility
|
|||||||
await SendErrorAsync(response);
|
await SendErrorAsync(response);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireContext(ContextType.Guild)]
|
||||||
|
public Task QuoteDeleteAuthor(IUser user)
|
||||||
|
=> QuoteDeleteAuthor(user.Id);
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireContext(ContextType.Guild)]
|
||||||
|
public async Task QuoteDeleteAuthor(ulong userId)
|
||||||
|
{
|
||||||
|
var hasManageMessages = ((IGuildUser)ctx.Message.Author).GuildPermissions.ManageMessages;
|
||||||
|
|
||||||
|
if (userId == ctx.User.Id || hasManageMessages)
|
||||||
|
{
|
||||||
|
var deleted = await _qs.DeleteAllAuthorQuotesAsync(ctx.Guild.Id, ctx.User.Id);
|
||||||
|
await ReplyConfirmLocalizedAsync(strs.quotes_deleted_count(deleted));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await ReplyErrorLocalizedAsync(strs.insuf_perms_u);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[RequireContext(ContextType.Guild)]
|
[RequireContext(ContextType.Guild)]
|
||||||
[UserPerm(GuildPerm.ManageMessages)]
|
[UserPerm(GuildPerm.ManageMessages)]
|
||||||
@@ -303,7 +326,7 @@ public partial class Utility
|
|||||||
#if GLOBAL_NADEKO
|
#if GLOBAL_NADEKO
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
#endif
|
#endif
|
||||||
public async Task QuotesImport([Leftover] string input = null)
|
public async Task QuotesImport([Leftover] string? input = null)
|
||||||
{
|
{
|
||||||
input = input?.Trim();
|
input = input?.Trim();
|
||||||
|
|
||||||
|
32
src/NadekoBot/Modules/Utility/Quote/QuoteService.cs
Normal file
32
src/NadekoBot/Modules/Utility/Quote/QuoteService.cs
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
#nullable disable warnings
|
||||||
|
using LinqToDB;
|
||||||
|
using LinqToDB.EntityFrameworkCore;
|
||||||
|
using NadekoBot.Services.Database.Models;
|
||||||
|
|
||||||
|
namespace NadekoBot.Modules.Utility;
|
||||||
|
|
||||||
|
public sealed class QuoteService : IQuoteService, INService
|
||||||
|
{
|
||||||
|
private readonly DbService _db;
|
||||||
|
|
||||||
|
public QuoteService(DbService db)
|
||||||
|
{
|
||||||
|
_db = db;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Delete all quotes created by the author in a guild
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="guildId">ID of the guild</param>
|
||||||
|
/// <param name="userId">ID of the user</param>
|
||||||
|
/// <returns>Number of deleted qutoes</returns>
|
||||||
|
public async Task<int> DeleteAllAuthorQuotesAsync(ulong guildId, ulong userId)
|
||||||
|
{
|
||||||
|
await using var ctx = _db.GetDbContext();
|
||||||
|
var deleted = await ctx.GetTable<Quote>()
|
||||||
|
.Where(x => x.GuildId == guildId && x.AuthorId == userId)
|
||||||
|
.DeleteAsync();
|
||||||
|
|
||||||
|
return deleted;
|
||||||
|
}
|
||||||
|
}
|
@@ -159,6 +159,18 @@ public class RemindService : INService, IReadyExecutor
|
|||||||
if (ch is null)
|
if (ch is null)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
var st = SmartText.CreateFrom(r.Message);
|
||||||
|
|
||||||
|
if (st is SmartEmbedText set)
|
||||||
|
{
|
||||||
|
await ch.SendMessageAsync(null, embed: set.GetEmbed().Build());
|
||||||
|
}
|
||||||
|
else if (st is SmartEmbedTextArray seta)
|
||||||
|
{
|
||||||
|
await ch.SendMessageAsync(null, embeds: seta.GetEmbedBuilders().Map(x => x.Build()));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
await ch.EmbedAsync(_eb.Create()
|
await ch.EmbedAsync(_eb.Create()
|
||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithTitle("Reminder")
|
.WithTitle("Reminder")
|
||||||
@@ -168,6 +180,7 @@ public class RemindService : INService, IReadyExecutor
|
|||||||
(await ch.GetUserAsync(r.UserId))?.ToString() ?? r.UserId.ToString()),
|
(await ch.GetUserAsync(r.UserId))?.ToString() ?? r.UserId.ToString()),
|
||||||
r.Message);
|
r.Message);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
catch (Exception ex)
|
catch (Exception ex)
|
||||||
{
|
{
|
||||||
Log.Warning(ex, "Error executing reminder {ReminderId}: {ErrorMessage}", r.Id, ex.Message);
|
Log.Warning(ex, "Error executing reminder {ReminderId}: {ErrorMessage}", r.Id, ex.Message);
|
||||||
|
@@ -367,6 +367,41 @@ public partial class Utility : NadekoModule
|
|||||||
await ConfirmLocalizedAsync(strs.emoji_added(em.ToString()));
|
await ConfirmLocalizedAsync(strs.emoji_added(em.ToString()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
[RequireContext(ContextType.Guild)]
|
||||||
|
[BotPerm(GuildPerm.ManageEmojisAndStickers)]
|
||||||
|
[UserPerm(GuildPerm.ManageEmojisAndStickers)]
|
||||||
|
[Priority(0)]
|
||||||
|
public async Task EmojiRemove(params Emote[] emotes)
|
||||||
|
{
|
||||||
|
if (emotes.Length == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var g = (SocketGuild)ctx.Guild;
|
||||||
|
|
||||||
|
var fails = new List<Emote>();
|
||||||
|
foreach (var emote in emotes)
|
||||||
|
{
|
||||||
|
var guildEmote = g.Emotes.FirstOrDefault(x => x.Id == emote.Id);
|
||||||
|
if (guildEmote is null)
|
||||||
|
{
|
||||||
|
fails.Add(emote);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
await ctx.Guild.DeleteEmoteAsync(guildEmote);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fails.Count > 0)
|
||||||
|
{
|
||||||
|
await ReplyPendingLocalizedAsync(strs.emoji_not_removed(fails.Select(x => x.ToString()).Join(" ")));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
[OwnerOnly]
|
[OwnerOnly]
|
||||||
public async Task ListServers(int page = 1)
|
public async Task ListServers(int page = 1)
|
||||||
|
@@ -334,14 +334,15 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
|
|
||||||
public enum XpShopInputType
|
public enum XpShopInputType
|
||||||
{
|
{
|
||||||
F = 0,
|
Backgrounds = 0,
|
||||||
Frs = 0,
|
B = 0,
|
||||||
Fs = 0,
|
Bg = 0,
|
||||||
Frames = 0,
|
Bgs = 0,
|
||||||
B = 1,
|
Frames = 1,
|
||||||
Bg = 1,
|
F = 1,
|
||||||
Bgs = 1,
|
Fr = 1,
|
||||||
Backgrounds = 1
|
Frs = 1,
|
||||||
|
Fs = 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
@@ -353,8 +354,11 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
await SendConfirmAsync(GetText(strs.available_commands), $@"`{prefix}xpshop bgs`
|
await SendConfirmAsync(GetText(strs.available_commands),
|
||||||
`{prefix}xpshop frames`");
|
$@"`{prefix}xpshop bgs`
|
||||||
|
`{prefix}xpshop frames`
|
||||||
|
|
||||||
|
*{GetText(strs.xpshop_website)}*");
|
||||||
}
|
}
|
||||||
|
|
||||||
[Cmd]
|
[Cmd]
|
||||||
@@ -394,13 +398,18 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
.WithOkColor()
|
.WithOkColor()
|
||||||
.WithTitle(item.Name)
|
.WithTitle(item.Name)
|
||||||
.AddField(GetText(strs.price), Gambling.Gambling.N(item.Price, culture), true)
|
.AddField(GetText(strs.price), Gambling.Gambling.N(item.Price, culture), true)
|
||||||
// .AddField(GetText(strs.buy), $"{prefix}xpbuy {key}", true)
|
.WithImageUrl(string.IsNullOrWhiteSpace(item.Preview)
|
||||||
.WithImageUrl(item.Url.ToString());
|
? item.Url
|
||||||
|
: item.Preview);
|
||||||
|
|
||||||
if (!string.IsNullOrWhiteSpace(item.Desc))
|
if (!string.IsNullOrWhiteSpace(item.Desc))
|
||||||
eb.WithDescription(item.Desc);
|
eb.AddField(GetText(strs.desc), item.Desc);
|
||||||
|
|
||||||
var tier = _service.GetXpShopTierRequirement();
|
if (key == "default")
|
||||||
|
eb.WithDescription(GetText(strs.xpshop_website));
|
||||||
|
|
||||||
|
|
||||||
|
var tier = _service.GetXpShopTierRequirement(type);
|
||||||
if (tier != PatronTier.None)
|
if (tier != PatronTier.None)
|
||||||
{
|
{
|
||||||
eb.WithFooter(GetText(strs.xp_shop_buy_required_tier(tier.ToString())));
|
eb.WithFooter(GetText(strs.xp_shop_buy_required_tier(tier.ToString())));
|
||||||
@@ -423,8 +432,7 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
var button = new ButtonBuilder(ownedItem.IsUsing
|
var button = new ButtonBuilder(ownedItem.IsUsing
|
||||||
? GetText(strs.in_use)
|
? GetText(strs.in_use)
|
||||||
: GetText(strs.use),
|
: GetText(strs.use),
|
||||||
"XP_SHOP_USE",
|
"xpshop:use",
|
||||||
ButtonStyle.Primary,
|
|
||||||
emote: Emoji.Parse("👐"),
|
emote: Emoji.Parse("👐"),
|
||||||
isDisabled: ownedItem.IsUsing);
|
isDisabled: ownedItem.IsUsing);
|
||||||
|
|
||||||
@@ -438,8 +446,7 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
var button = new ButtonBuilder(GetText(strs.buy),
|
var button = new ButtonBuilder(GetText(strs.buy),
|
||||||
"XP_SHOP_BUY",
|
"xpshop:buy",
|
||||||
ButtonStyle.Primary,
|
|
||||||
emote: Emoji.Parse("💰"));
|
emote: Emoji.Parse("💰"));
|
||||||
|
|
||||||
var inter = new SimpleInteraction<(string key, XpShopItemType type)?>(
|
var inter = new SimpleInteraction<(string key, XpShopItemType type)?>(
|
||||||
@@ -455,6 +462,53 @@ public partial class Xp : NadekoModule<XpService>
|
|||||||
addPaginatedFooter: false);
|
addPaginatedFooter: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
public async Task XpShopBuy(XpShopInputType type, string key)
|
||||||
|
{
|
||||||
|
var result = await _service.BuyShopItemAsync(ctx.User.Id, (XpShopItemType)type, key);
|
||||||
|
|
||||||
|
NadekoInteraction GetUseInteraction()
|
||||||
|
{
|
||||||
|
return _inter.Create(ctx.User.Id,
|
||||||
|
new SimpleInteraction<object>(
|
||||||
|
new ButtonBuilder(label: "Use", customId: "xpshop:use_item", emote: Emoji.Parse("👐")),
|
||||||
|
async (smc, _) => await XpShopUse(type, key)
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result != BuyResult.Success)
|
||||||
|
{
|
||||||
|
var _ = result switch
|
||||||
|
{
|
||||||
|
BuyResult.XpShopDisabled => await ReplyErrorLocalizedAsync(strs.xp_shop_disabled),
|
||||||
|
BuyResult.InsufficientFunds => await ReplyErrorLocalizedAsync(strs.not_enough(_gss.Data.Currency.Sign)),
|
||||||
|
BuyResult.AlreadyOwned => await ReplyErrorLocalizedAsync(strs.xpshop_already_owned, GetUseInteraction()),
|
||||||
|
BuyResult.UnknownItem => await ReplyErrorLocalizedAsync(strs.xpshop_item_not_found),
|
||||||
|
BuyResult.InsufficientPatronTier => await ReplyErrorLocalizedAsync(strs.patron_insuff_tier),
|
||||||
|
_ => throw new ArgumentOutOfRangeException()
|
||||||
|
};
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ReplyConfirmLocalizedAsync(strs.xpshop_buy_success(type.ToString().ToLowerInvariant(),
|
||||||
|
key.ToLowerInvariant()),
|
||||||
|
GetUseInteraction());
|
||||||
|
}
|
||||||
|
|
||||||
|
[Cmd]
|
||||||
|
public async Task XpShopUse(XpShopInputType type, string key)
|
||||||
|
{
|
||||||
|
var result = await _service.UseShopItemAsync(ctx.User.Id, (XpShopItemType)type, key);
|
||||||
|
|
||||||
|
if (!result)
|
||||||
|
{
|
||||||
|
await ReplyConfirmLocalizedAsync(strs.xp_shop_item_cant_use);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ctx.OkAsync();
|
||||||
|
}
|
||||||
|
|
||||||
private async Task OnShopUse(SocketMessageComponent smc, (string? key, XpShopItemType type)? maybeState)
|
private async Task OnShopUse(SocketMessageComponent smc, (string? key, XpShopItemType type)? maybeState)
|
||||||
{
|
{
|
||||||
if (maybeState is not { } state)
|
if (maybeState is not { } state)
|
||||||
|
@@ -10,7 +10,7 @@ namespace NadekoBot.Modules.Xp;
|
|||||||
public sealed partial class XpConfig : ICloneable<XpConfig>
|
public sealed partial class XpConfig : ICloneable<XpConfig>
|
||||||
{
|
{
|
||||||
[Comment(@"DO NOT CHANGE")]
|
[Comment(@"DO NOT CHANGE")]
|
||||||
public int Version { get; set; } = 3;
|
public int Version { get; set; } = 5;
|
||||||
|
|
||||||
[Comment(@"How much XP will the users receive per message")]
|
[Comment(@"How much XP will the users receive per message")]
|
||||||
public int XpPerMessage { get; set; } = 3;
|
public int XpPerMessage { get; set; } = 3;
|
||||||
@@ -27,6 +27,9 @@ public sealed partial class XpConfig : ICloneable<XpConfig>
|
|||||||
[Comment(@"The maximum amount of minutes the bot will keep track of a user in a voice channel")]
|
[Comment(@"The maximum amount of minutes the bot will keep track of a user in a voice channel")]
|
||||||
public int VoiceMaxMinutes { get; set; } = 720;
|
public int VoiceMaxMinutes { get; set; } = 720;
|
||||||
|
|
||||||
|
[Comment(@"The amount of currency users will receive for each point of global xp that they earn")]
|
||||||
|
public float CurrencyPerXp { get; set; } = 0;
|
||||||
|
|
||||||
[Comment(@"Xp Shop config")]
|
[Comment(@"Xp Shop config")]
|
||||||
public ShopConfig Shop { get; set; } = new();
|
public ShopConfig Shop { get; set; } = new();
|
||||||
|
|
||||||
@@ -37,9 +40,13 @@ True -> Users can access the xp shop using .xpshop command
|
|||||||
False -> Users can't access the xp shop")]
|
False -> Users can't access the xp shop")]
|
||||||
public bool IsEnabled { get; set; } = false;
|
public bool IsEnabled { get; set; } = false;
|
||||||
|
|
||||||
[Comment(@"Which patron tier do users need in order to use the .xpshop command
|
[Comment(@"Which patron tier do users need in order to use the .xpshop bgs command
|
||||||
Leave at 'None' if patron system is disabled or you don't want any restrictions")]
|
Leave at 'None' if patron system is disabled or you don't want any restrictions")]
|
||||||
public PatronTier TierRequirement { get; set; } = PatronTier.None;
|
public PatronTier BgsTierRequirement { get; set; } = PatronTier.None;
|
||||||
|
|
||||||
|
[Comment(@"Which patron tier do users need in order to use the .xpshop frames command
|
||||||
|
Leave at 'None' if patron system is disabled or you don't want any restrictions")]
|
||||||
|
public PatronTier FramesTierRequirement { get; set; } = PatronTier.None;
|
||||||
|
|
||||||
[Comment(@"Frames available for sale. Keys are unique IDs.
|
[Comment(@"Frames available for sale. Keys are unique IDs.
|
||||||
Do not change keys as they are not publicly visible. Only change properties (name, price, id)
|
Do not change keys as they are not publicly visible. Only change properties (name, price, id)
|
||||||
@@ -71,6 +78,9 @@ To remove an item from the shop, but keep previous purchases, set the price to -
|
|||||||
[Comment(@"Direct url to the .png image which will be applied to the user's XP card")]
|
[Comment(@"Direct url to the .png image which will be applied to the user's XP card")]
|
||||||
public string Url { get; set; }
|
public string Url { get; set; }
|
||||||
|
|
||||||
|
[Comment(@"Optional preview url which will show instead of the real URL in the shop ")]
|
||||||
|
public string Preview { get; set; }
|
||||||
|
|
||||||
[Comment(@"Optional description of the item")]
|
[Comment(@"Optional description of the item")]
|
||||||
public string Desc { get; set; }
|
public string Desc { get; set; }
|
||||||
}
|
}
|
||||||
|
@@ -52,11 +52,11 @@ public sealed class XpConfigService : ConfigServiceBase<XpConfig>
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data.Version < 3)
|
if (data.Version < 6)
|
||||||
{
|
{
|
||||||
ModifyConfig(c =>
|
ModifyConfig(c =>
|
||||||
{
|
{
|
||||||
c.Version = 3;
|
c.Version = 6;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -176,6 +176,16 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
var gxps = new List<UserXpStats>(globalToAdd.Count);
|
var gxps = new List<UserXpStats>(globalToAdd.Count);
|
||||||
await using (var ctx = _db.GetDbContext())
|
await using (var ctx = _db.GetDbContext())
|
||||||
{
|
{
|
||||||
|
var conf = _xpConfig.Data;
|
||||||
|
if (conf.CurrencyPerXp > 0)
|
||||||
|
{
|
||||||
|
foreach (var user in globalToAdd)
|
||||||
|
{
|
||||||
|
var amount = user.Value.XpAmount * conf.CurrencyPerXp;
|
||||||
|
await _cs.AddAsync(user.Key, (long)(amount), null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// update global user xp in batches
|
// update global user xp in batches
|
||||||
// group by xp amount and update the same amounts at the same time
|
// group by xp amount and update the same amounts at the same time
|
||||||
foreach (var group in globalToAdd.GroupBy(x => x.Value.XpAmount, x => x.Key))
|
foreach (var group in globalToAdd.GroupBy(x => x.Value.XpAmount, x => x.Key))
|
||||||
@@ -188,9 +198,15 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
},
|
},
|
||||||
(_, n) => n);
|
(_, n) => n);
|
||||||
|
|
||||||
|
await ctx.Clubs
|
||||||
|
.Where(x => x.Members.Any(m => group.Contains(m.UserId)))
|
||||||
|
.UpdateAsync(old => new()
|
||||||
|
{
|
||||||
|
Xp = old.Xp + (group.Key * old.Members.Count(m => group.Contains(m.UserId)))
|
||||||
|
});
|
||||||
|
|
||||||
dus.AddRange(items);
|
dus.AddRange(items);
|
||||||
}
|
}
|
||||||
|
|
||||||
// update guild user xp in batches
|
// update guild user xp in batches
|
||||||
foreach (var (guildId, toAdd) in guildToAdd)
|
foreach (var (guildId, toAdd) in guildToAdd)
|
||||||
{
|
{
|
||||||
@@ -207,6 +223,41 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
(_, n) => n);
|
(_, n) => n);
|
||||||
|
|
||||||
gxps.AddRange(items);
|
gxps.AddRange(items);
|
||||||
|
|
||||||
|
var missingUserIds = group.Where(userId => !items.Any(x => x.UserId == userId)).ToArray();
|
||||||
|
foreach (var userId in missingUserIds)
|
||||||
|
{
|
||||||
|
await ctx
|
||||||
|
.UserXpStats
|
||||||
|
.ToLinqToDBTable()
|
||||||
|
.InsertOrUpdateAsync(() => new UserXpStats()
|
||||||
|
{
|
||||||
|
UserId = userId,
|
||||||
|
GuildId = guildId,
|
||||||
|
Xp = group.Key,
|
||||||
|
DateAdded = DateTime.UtcNow,
|
||||||
|
AwardedXp = 0,
|
||||||
|
NotifyOnLevelUp = XpNotificationLocation.None
|
||||||
|
},
|
||||||
|
_ => new()
|
||||||
|
{
|
||||||
|
|
||||||
|
},
|
||||||
|
() => new()
|
||||||
|
{
|
||||||
|
UserId = userId
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (missingUserIds.Length > 0)
|
||||||
|
{
|
||||||
|
var missingItems = await ctx.UserXpStats
|
||||||
|
.ToLinqToDBTable()
|
||||||
|
.Where(x => missingUserIds.Contains(x.UserId))
|
||||||
|
.ToArrayAsyncLinqToDB();
|
||||||
|
|
||||||
|
gxps.AddRange(missingItems);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -235,8 +286,8 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
if (guildToAdd.TryGetValue(du.GuildId, out var users)
|
if (guildToAdd.TryGetValue(du.GuildId, out var users)
|
||||||
&& users.TryGetValue(du.UserId, out var xpGainData))
|
&& users.TryGetValue(du.UserId, out var xpGainData))
|
||||||
{
|
{
|
||||||
var oldLevel = new LevelStats(du.Xp - xpGainData.XpAmount);
|
var oldLevel = new LevelStats(du.Xp - xpGainData.XpAmount + du.AwardedXp);
|
||||||
var newLevel = new LevelStats(du.Xp);
|
var newLevel = new LevelStats(du.Xp + du.AwardedXp);
|
||||||
|
|
||||||
if (oldLevel.Level < newLevel.Level)
|
if (oldLevel.Level < newLevel.Level)
|
||||||
{
|
{
|
||||||
@@ -593,12 +644,12 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
{
|
{
|
||||||
if (ShouldTrackVoiceChannel(channel))
|
if (ShouldTrackVoiceChannel(channel))
|
||||||
{
|
{
|
||||||
foreach (var user in channel.Users)
|
foreach (var user in channel.ConnectedUsers)
|
||||||
await ScanUserForVoiceXp(user, channel);
|
await ScanUserForVoiceXp(user, channel);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
foreach (var user in channel.Users)
|
foreach (var user in channel.ConnectedUsers)
|
||||||
await UserLeftVoiceChannel(user, channel);
|
await UserLeftVoiceChannel(user, channel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -617,13 +668,13 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
}
|
}
|
||||||
|
|
||||||
private bool ShouldTrackVoiceChannel(SocketVoiceChannel channel)
|
private bool ShouldTrackVoiceChannel(SocketVoiceChannel channel)
|
||||||
=> channel.Users.Where(UserParticipatingInVoiceChannel).Take(2).Count() >= 2;
|
=> channel.ConnectedUsers.Where(UserParticipatingInVoiceChannel).Take(2).Count() >= 2;
|
||||||
|
|
||||||
private bool UserParticipatingInVoiceChannel(SocketGuildUser user)
|
private bool UserParticipatingInVoiceChannel(SocketGuildUser user)
|
||||||
=> !user.IsDeafened && !user.IsMuted && !user.IsSelfDeafened && !user.IsSelfMuted;
|
=> !user.IsDeafened && !user.IsMuted && !user.IsSelfDeafened && !user.IsSelfMuted;
|
||||||
|
|
||||||
private TypedKey<long> GetVoiceXpKey(ulong userId)
|
private TypedKey<long> GetVoiceXpKey(ulong userId)
|
||||||
=> new($"xp:vc_join:{userId}");
|
=> new($"xp:{_client.CurrentUser.Id}:vc_join:{userId}");
|
||||||
|
|
||||||
private async Task UserJoinedVoiceChannel(SocketGuildUser user)
|
private async Task UserJoinedVoiceChannel(SocketGuildUser user)
|
||||||
{
|
{
|
||||||
@@ -635,6 +686,18 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
overwrite: false);
|
overwrite: false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// private void UserJoinedVoiceChannel(SocketGuildUser user)
|
||||||
|
// {
|
||||||
|
// var key = $"{_creds.RedisKey()}_user_xp_vc_join_{user.Id}";
|
||||||
|
// var value = DateTimeOffset.UtcNow.ToUnixTimeSeconds();
|
||||||
|
//
|
||||||
|
// _cache.Redis.GetDatabase()
|
||||||
|
// .StringSet(key,
|
||||||
|
// value,
|
||||||
|
// TimeSpan.FromMinutes(_xpConfig.Data.VoiceMaxMinutes),
|
||||||
|
// when: When.NotExists);
|
||||||
|
// }
|
||||||
|
|
||||||
private async Task UserLeftVoiceChannel(SocketGuildUser user, SocketVoiceChannel channel)
|
private async Task UserLeftVoiceChannel(SocketGuildUser user, SocketVoiceChannel channel)
|
||||||
{
|
{
|
||||||
var key = GetVoiceXpKey(user.Id);
|
var key = GetVoiceXpKey(user.Id);
|
||||||
@@ -654,6 +717,7 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
|
|
||||||
if (actualXp > 0)
|
if (actualXp > 0)
|
||||||
{
|
{
|
||||||
|
Log.Information("Adding {Amount} voice xp to {User}", actualXp, user.ToString());
|
||||||
await _xpGainQueue.Writer.WriteAsync(new()
|
await _xpGainQueue.Writer.WriteAsync(new()
|
||||||
{
|
{
|
||||||
Guild = channel.Guild,
|
Guild = channel.Guild,
|
||||||
@@ -664,6 +728,38 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* private void UserLeftVoiceChannel(SocketGuildUser user, SocketVoiceChannel channel)
|
||||||
|
{
|
||||||
|
var key = $"{_creds.RedisKey()}_user_xp_vc_join_{user.Id}";
|
||||||
|
var value = _cache.Redis.GetDatabase().StringGet(key);
|
||||||
|
_cache.Redis.GetDatabase().KeyDelete(key);
|
||||||
|
|
||||||
|
// Allow for if this function gets called multiple times when a user leaves a channel.
|
||||||
|
if (value.IsNull)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!value.TryParse(out long startUnixTime))
|
||||||
|
return;
|
||||||
|
|
||||||
|
var dateStart = DateTimeOffset.FromUnixTimeSeconds(startUnixTime);
|
||||||
|
var dateEnd = DateTimeOffset.UtcNow;
|
||||||
|
var minutes = (dateEnd - dateStart).TotalMinutes;
|
||||||
|
var xp = _xpConfig.Data.VoiceXpPerMinute * minutes;
|
||||||
|
var actualXp = (int)Math.Floor(xp);
|
||||||
|
|
||||||
|
if (actualXp > 0)
|
||||||
|
{
|
||||||
|
_addMessageXp.Enqueue(new()
|
||||||
|
{
|
||||||
|
Guild = channel.Guild,
|
||||||
|
User = user,
|
||||||
|
XpAmount = actualXp
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
private bool ShouldTrackXp(SocketGuildUser user, ulong channelId)
|
private bool ShouldTrackXp(SocketGuildUser user, ulong channelId)
|
||||||
{
|
{
|
||||||
if (_excludedChannels.TryGetValue(user.Guild.Id, out var chans) && chans.Contains(channelId))
|
if (_excludedChannels.TryGetValue(user.Guild.Id, out var chans) && chans.Contains(channelId))
|
||||||
@@ -767,8 +863,8 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
return Enumerable.Empty<ulong>();
|
return Enumerable.Empty<ulong>();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static TypedKey<bool> GetUserRewKey(ulong userId)
|
private TypedKey<bool> GetUserRewKey(ulong userId)
|
||||||
=> new($"xp:user_gain:{userId}");
|
=> new($"xp:{_client.CurrentUser.Id}:user_gain:{userId}");
|
||||||
|
|
||||||
private async Task<bool> SetUserRewardedAsync(ulong userId)
|
private async Task<bool> SetUserRewardedAsync(ulong userId)
|
||||||
=> await _c.AddAsync(GetUserRewKey(userId),
|
=> await _c.AddAsync(GetUserRewKey(userId),
|
||||||
@@ -887,6 +983,8 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
throw new ArgumentNullException(nameof(bgBytes));
|
throw new ArgumentNullException(nameof(bgBytes));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var outlinePen = new Pen(Color.Black, 1f);
|
||||||
|
|
||||||
using var img = Image.Load<Rgba32>(bgBytes, out var imageFormat);
|
using var img = Image.Load<Rgba32>(bgBytes, out var imageFormat);
|
||||||
if (template.User.Name.Show)
|
if (template.User.Name.Show)
|
||||||
{
|
{
|
||||||
@@ -909,7 +1007,8 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
Origin = new(template.User.Name.Pos.X, template.User.Name.Pos.Y + 8)
|
Origin = new(template.User.Name.Pos.X, template.User.Name.Pos.Y + 8)
|
||||||
},
|
},
|
||||||
"@" + username,
|
"@" + username,
|
||||||
template.User.Name.Color);
|
Brushes.Solid(template.User.Name.Color),
|
||||||
|
outlinePen);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -929,7 +1028,8 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
Origin = new(template.Club.Name.Pos.X + 50, template.Club.Name.Pos.Y - 8)
|
Origin = new(template.Club.Name.Pos.X + 50, template.Club.Name.Pos.Y - 8)
|
||||||
},
|
},
|
||||||
clubName,
|
clubName,
|
||||||
template.Club.Name.Color));
|
Brushes.Solid(template.Club.Name.Color),
|
||||||
|
outlinePen));
|
||||||
}
|
}
|
||||||
|
|
||||||
Font GetTruncatedFont(
|
Font GetTruncatedFont(
|
||||||
@@ -987,7 +1087,6 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
var pen = new Pen(Color.Black, 1.25f);
|
|
||||||
|
|
||||||
var global = stats.Global;
|
var global = stats.Global;
|
||||||
var guild = stats.Guild;
|
var guild = stats.Guild;
|
||||||
@@ -1012,7 +1111,7 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
},
|
},
|
||||||
$"{global.LevelXp}/{global.RequiredXp}",
|
$"{global.LevelXp}/{global.RequiredXp}",
|
||||||
Brushes.Solid(template.User.Xp.Global.Color),
|
Brushes.Solid(template.User.Xp.Global.Color),
|
||||||
pen));
|
outlinePen));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.User.Xp.Guild.Show)
|
if (template.User.Xp.Guild.Show)
|
||||||
@@ -1026,7 +1125,7 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
},
|
},
|
||||||
$"{guild.LevelXp}/{guild.RequiredXp}",
|
$"{guild.LevelXp}/{guild.RequiredXp}",
|
||||||
Brushes.Solid(template.User.Xp.Guild.Color),
|
Brushes.Solid(template.User.Xp.Guild.Color),
|
||||||
pen));
|
outlinePen));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stats.FullGuildStats.AwardedXp != 0 && template.User.Xp.Awarded.Show)
|
if (stats.FullGuildStats.AwardedXp != 0 && template.User.Xp.Awarded.Show)
|
||||||
@@ -1038,10 +1137,11 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
img.Mutate(x => x.DrawText($"({sign}{stats.FullGuildStats.AwardedXp})",
|
img.Mutate(x => x.DrawText($"({sign}{stats.FullGuildStats.AwardedXp})",
|
||||||
_fonts.NotoSans.CreateFont(template.User.Xp.Awarded.FontSize, FontStyle.Bold),
|
_fonts.NotoSans.CreateFont(template.User.Xp.Awarded.FontSize, FontStyle.Bold),
|
||||||
Brushes.Solid(template.User.Xp.Awarded.Color),
|
Brushes.Solid(template.User.Xp.Awarded.Color),
|
||||||
pen,
|
outlinePen,
|
||||||
new(awX, awY)));
|
new(awX, awY)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var rankPen = new Pen(Color.White, 1);
|
||||||
//ranking
|
//ranking
|
||||||
if (template.User.GlobalRank.Show)
|
if (template.User.GlobalRank.Show)
|
||||||
{
|
{
|
||||||
@@ -1054,10 +1154,15 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
globalRankStr,
|
globalRankStr,
|
||||||
68);
|
68);
|
||||||
|
|
||||||
img.Mutate(x => x.DrawText(globalRankStr,
|
img.Mutate(x => x.DrawText(
|
||||||
globalRankFont,
|
new TextOptions(globalRankFont)
|
||||||
template.User.GlobalRank.Color,
|
{
|
||||||
new(template.User.GlobalRank.Pos.X, template.User.GlobalRank.Pos.Y)));
|
Origin = new(template.User.GlobalRank.Pos.X, template.User.GlobalRank.Pos.Y)
|
||||||
|
},
|
||||||
|
globalRankStr,
|
||||||
|
Brushes.Solid(template.User.GlobalRank.Color),
|
||||||
|
rankPen
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.User.GuildRank.Show)
|
if (template.User.GuildRank.Show)
|
||||||
@@ -1071,10 +1176,15 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
guildRankStr,
|
guildRankStr,
|
||||||
43);
|
43);
|
||||||
|
|
||||||
img.Mutate(x => x.DrawText(guildRankStr,
|
img.Mutate(x => x.DrawText(
|
||||||
guildRankFont,
|
new TextOptions(guildRankFont)
|
||||||
template.User.GuildRank.Color,
|
{
|
||||||
new(template.User.GuildRank.Pos.X, template.User.GuildRank.Pos.Y)));
|
Origin = new(template.User.GuildRank.Pos.X, template.User.GuildRank.Pos.Y)
|
||||||
|
},
|
||||||
|
guildRankStr,
|
||||||
|
Brushes.Solid(template.User.GuildRank.Color),
|
||||||
|
rankPen
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
//avatar
|
//avatar
|
||||||
@@ -1335,13 +1445,17 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
var conf = _xpConfig.Data;
|
var conf = _xpConfig.Data;
|
||||||
|
|
||||||
if (!conf.Shop.IsEnabled)
|
if (!conf.Shop.IsEnabled)
|
||||||
return BuyResult.UnknownItem;
|
return BuyResult.XpShopDisabled;
|
||||||
|
|
||||||
if (conf.Shop.TierRequirement != PatronTier.None)
|
var req = type == XpShopItemType.Background
|
||||||
|
? conf.Shop.BgsTierRequirement
|
||||||
|
: conf.Shop.FramesTierRequirement;
|
||||||
|
|
||||||
|
if (req != PatronTier.None && !_creds.IsOwner(userId))
|
||||||
{
|
{
|
||||||
var patron = await _ps.GetPatronAsync(userId);
|
var patron = await _ps.GetPatronAsync(userId);
|
||||||
|
|
||||||
if ((int)patron.Tier < (int)conf.Shop.TierRequirement)
|
if ((int)patron.Tier < (int)req)
|
||||||
return BuyResult.InsufficientPatronTier;
|
return BuyResult.InsufficientPatronTier;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1468,8 +1582,12 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public PatronTier GetXpShopTierRequirement()
|
public PatronTier GetXpShopTierRequirement(Xp.XpShopInputType type)
|
||||||
=> _xpConfig.Data.Shop.TierRequirement;
|
=> type switch
|
||||||
|
{
|
||||||
|
Xp.XpShopInputType.F => _xpConfig.Data.Shop.FramesTierRequirement,
|
||||||
|
_ => _xpConfig.Data.Shop.BgsTierRequirement,
|
||||||
|
};
|
||||||
|
|
||||||
public bool IsShopEnabled()
|
public bool IsShopEnabled()
|
||||||
=> _xpConfig.Data.Shop.IsEnabled;
|
=> _xpConfig.Data.Shop.IsEnabled;
|
||||||
@@ -1478,6 +1596,7 @@ public class XpService : INService, IReadyExecutor, IExecNoCommand
|
|||||||
public enum BuyResult
|
public enum BuyResult
|
||||||
{
|
{
|
||||||
Success,
|
Success,
|
||||||
|
XpShopDisabled,
|
||||||
AlreadyOwned,
|
AlreadyOwned,
|
||||||
InsufficientFunds,
|
InsufficientFunds,
|
||||||
UnknownItem,
|
UnknownItem,
|
||||||
|
@@ -27,7 +27,7 @@
|
|||||||
<PackageReference Include="CodeHollow.FeedReader" Version="1.2.4" />
|
<PackageReference Include="CodeHollow.FeedReader" Version="1.2.4" />
|
||||||
<PackageReference Include="CommandLineParser" Version="2.9.1" />
|
<PackageReference Include="CommandLineParser" Version="2.9.1" />
|
||||||
<PackageReference Include="CsvHelper" Version="28.0.1" />
|
<PackageReference Include="CsvHelper" Version="28.0.1" />
|
||||||
<PackageReference Include="Discord.Net" Version="3.103.0" />
|
<PackageReference Include="Discord.Net" Version="3.104.0" />
|
||||||
<PackageReference Include="CoreCLR-NCalc" Version="2.2.110" />
|
<PackageReference Include="CoreCLR-NCalc" Version="2.2.110" />
|
||||||
<PackageReference Include="Google.Apis.Urlshortener.v1" Version="1.41.1.138" />
|
<PackageReference Include="Google.Apis.Urlshortener.v1" Version="1.41.1.138" />
|
||||||
<PackageReference Include="Google.Apis.YouTube.v3" Version="1.57.0.2749" />
|
<PackageReference Include="Google.Apis.YouTube.v3" Version="1.57.0.2749" />
|
||||||
|
@@ -279,6 +279,7 @@ public class CommandHandler : INService, IReadyExecutor
|
|||||||
// if it errored
|
// if it errored
|
||||||
if (error is not null)
|
if (error is not null)
|
||||||
{
|
{
|
||||||
|
error = HumanizeError(error);
|
||||||
LogErroredExecution(error, usrMsg, channel as ITextChannel, blockTime, startTime);
|
LogErroredExecution(error, usrMsg, channel as ITextChannel, blockTime, startTime);
|
||||||
|
|
||||||
if (guild is not null)
|
if (guild is not null)
|
||||||
@@ -292,6 +293,15 @@ public class CommandHandler : INService, IReadyExecutor
|
|||||||
await _behaviorHandler.RunOnNoCommandAsync(guild, usrMsg);
|
await _behaviorHandler.RunOnNoCommandAsync(guild, usrMsg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private string HumanizeError(string error)
|
||||||
|
{
|
||||||
|
if (error.Contains("parse int", StringComparison.OrdinalIgnoreCase)
|
||||||
|
|| error.Contains("parse float"))
|
||||||
|
return "Invalid number specified. Make sure you're specifying parameters in the correct order.";
|
||||||
|
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
public Task<(bool Success, string Error, CommandInfo Info)> ExecuteCommandAsync(
|
public Task<(bool Success, string Error, CommandInfo Info)> ExecuteCommandAsync(
|
||||||
CommandContext context,
|
CommandContext context,
|
||||||
string input,
|
string input,
|
||||||
|
@@ -27,7 +27,7 @@ public class DefaultWallet : IWallet
|
|||||||
.FirstOrDefaultAsync();
|
.FirstOrDefaultAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<bool> Take(long amount, TxData txData)
|
public async Task<bool> Take(long amount, TxData? txData)
|
||||||
{
|
{
|
||||||
if (amount < 0)
|
if (amount < 0)
|
||||||
throw new ArgumentOutOfRangeException(nameof(amount), "Amount to take must be non negative.");
|
throw new ArgumentOutOfRangeException(nameof(amount), "Amount to take must be non negative.");
|
||||||
@@ -45,6 +45,8 @@ public class DefaultWallet : IWallet
|
|||||||
if (changed == 0)
|
if (changed == 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
if (txData is not null)
|
||||||
|
{
|
||||||
await ctx
|
await ctx
|
||||||
.GetTable<CurrencyTransaction>()
|
.GetTable<CurrencyTransaction>()
|
||||||
.InsertAsync(() => new()
|
.InsertAsync(() => new()
|
||||||
@@ -57,11 +59,12 @@ public class DefaultWallet : IWallet
|
|||||||
OtherId = txData.OtherId,
|
OtherId = txData.OtherId,
|
||||||
DateAdded = DateTime.UtcNow
|
DateAdded = DateTime.UtcNow
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task Add(long amount, TxData txData)
|
public async Task Add(long amount, TxData? txData)
|
||||||
{
|
{
|
||||||
if (amount <= 0)
|
if (amount <= 0)
|
||||||
throw new ArgumentOutOfRangeException(nameof(amount), "Amount must be greater than 0.");
|
throw new ArgumentOutOfRangeException(nameof(amount), "Amount must be greater than 0.");
|
||||||
@@ -92,6 +95,8 @@ public class DefaultWallet : IWallet
|
|||||||
await tran.CommitAsync();
|
await tran.CommitAsync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (txData is not null)
|
||||||
|
{
|
||||||
await ctx.GetTable<CurrencyTransaction>()
|
await ctx.GetTable<CurrencyTransaction>()
|
||||||
.InsertAsync(() => new()
|
.InsertAsync(() => new()
|
||||||
{
|
{
|
||||||
@@ -104,4 +109,5 @@ public class DefaultWallet : IWallet
|
|||||||
DateAdded = DateTime.UtcNow
|
DateAdded = DateTime.UtcNow
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
@@ -5,31 +5,35 @@ public interface IWallet
|
|||||||
public ulong UserId { get; }
|
public ulong UserId { get; }
|
||||||
|
|
||||||
public Task<long> GetBalance();
|
public Task<long> GetBalance();
|
||||||
public Task<bool> Take(long amount, TxData txData);
|
public Task<bool> Take(long amount, TxData? txData);
|
||||||
public Task Add(long amount, TxData txData);
|
public Task Add(long amount, TxData? txData);
|
||||||
|
|
||||||
public async Task<bool> Transfer(
|
public async Task<bool> Transfer(
|
||||||
long amount,
|
long amount,
|
||||||
IWallet to,
|
IWallet to,
|
||||||
TxData txData)
|
TxData? txData)
|
||||||
{
|
{
|
||||||
if (amount <= 0)
|
if (amount <= 0)
|
||||||
throw new ArgumentOutOfRangeException(nameof(amount), "Amount must be greater than 0.");
|
throw new ArgumentOutOfRangeException(nameof(amount), "Amount must be greater than 0.");
|
||||||
|
|
||||||
var succ = await Take(amount,
|
if (txData is not null)
|
||||||
txData with
|
txData = txData with
|
||||||
{
|
{
|
||||||
OtherId = to.UserId
|
OtherId = to.UserId
|
||||||
});
|
};
|
||||||
|
|
||||||
|
var succ = await Take(amount, txData);
|
||||||
|
|
||||||
if (!succ)
|
if (!succ)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
await to.Add(amount,
|
if (txData is not null)
|
||||||
txData with
|
txData = txData with
|
||||||
{
|
{
|
||||||
OtherId = UserId
|
OtherId = UserId
|
||||||
});
|
};
|
||||||
|
|
||||||
|
await to.Add(amount, txData);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@@ -67,6 +67,9 @@ public sealed class DiscordEmbedBuilderWrapper : IEmbedBuilder
|
|||||||
_ => throw new ArgumentOutOfRangeException(nameof(color), "Unsupported EmbedColor type")
|
_ => throw new ArgumentOutOfRangeException(nameof(color), "Unsupported EmbedColor type")
|
||||||
};
|
};
|
||||||
|
|
||||||
|
public IEmbedBuilder WithDiscordColor(Color color)
|
||||||
|
=> Wrap(embed.WithColor(color));
|
||||||
|
|
||||||
public Embed Build()
|
public Embed Build()
|
||||||
=> embed.Build();
|
=> embed.Build();
|
||||||
|
|
||||||
|
@@ -7,7 +7,7 @@ namespace NadekoBot.Services;
|
|||||||
|
|
||||||
public sealed class StatsService : IStatsService, IReadyExecutor, INService
|
public sealed class StatsService : IStatsService, IReadyExecutor, INService
|
||||||
{
|
{
|
||||||
public const string BOT_VERSION = "4.3.2";
|
public const string BOT_VERSION = "4.3.6";
|
||||||
|
|
||||||
public string Author
|
public string Author
|
||||||
=> "Kwoth#2452";
|
=> "Kwoth#2452";
|
||||||
|
@@ -33,7 +33,7 @@ public static class MessageChannelExtensions
|
|||||||
public static async Task<IUserMessage> SendAsync(
|
public static async Task<IUserMessage> SendAsync(
|
||||||
this IMessageChannel channel,
|
this IMessageChannel channel,
|
||||||
string? plainText,
|
string? plainText,
|
||||||
NadekoButtonInteraction? inter,
|
NadekoInteraction? inter,
|
||||||
Embed? embed = null,
|
Embed? embed = null,
|
||||||
IReadOnlyCollection<Embed>? embeds = null,
|
IReadOnlyCollection<Embed>? embeds = null,
|
||||||
bool sanitizeAll = false)
|
bool sanitizeAll = false)
|
||||||
@@ -72,7 +72,7 @@ public static class MessageChannelExtensions
|
|||||||
IEmbedBuilder? embed,
|
IEmbedBuilder? embed,
|
||||||
string plainText = "",
|
string plainText = "",
|
||||||
IReadOnlyCollection<IEmbedBuilder>? embeds = null,
|
IReadOnlyCollection<IEmbedBuilder>? embeds = null,
|
||||||
NadekoButtonInteraction? inter = null)
|
NadekoInteraction? inter = null)
|
||||||
=> ch.SendAsync(plainText,
|
=> ch.SendAsync(plainText,
|
||||||
inter,
|
inter,
|
||||||
embed: embed?.Build(),
|
embed: embed?.Build(),
|
||||||
@@ -83,7 +83,7 @@ public static class MessageChannelExtensions
|
|||||||
IEmbedBuilderService eb,
|
IEmbedBuilderService eb,
|
||||||
string text,
|
string text,
|
||||||
MessageType type,
|
MessageType type,
|
||||||
NadekoButtonInteraction? inter = null)
|
NadekoInteraction? inter = null)
|
||||||
{
|
{
|
||||||
var builder = eb.Create().WithDescription(text);
|
var builder = eb.Create().WithDescription(text);
|
||||||
|
|
||||||
|
@@ -56,7 +56,7 @@ public static class SocketMessageComponentExtensions
|
|||||||
IEmbedBuilder? embed,
|
IEmbedBuilder? embed,
|
||||||
string plainText = "",
|
string plainText = "",
|
||||||
IReadOnlyCollection<IEmbedBuilder>? embeds = null,
|
IReadOnlyCollection<IEmbedBuilder>? embeds = null,
|
||||||
NadekoButtonInteraction? inter = null,
|
NadekoInteraction? inter = null,
|
||||||
bool ephemeral = false)
|
bool ephemeral = false)
|
||||||
=> smc.RespondAsync(plainText,
|
=> smc.RespondAsync(plainText,
|
||||||
embed: embed?.Build(),
|
embed: embed?.Build(),
|
||||||
@@ -69,7 +69,7 @@ public static class SocketMessageComponentExtensions
|
|||||||
string text,
|
string text,
|
||||||
MessageType type,
|
MessageType type,
|
||||||
bool ephemeral = false,
|
bool ephemeral = false,
|
||||||
NadekoButtonInteraction? inter = null)
|
NadekoInteraction? inter = null)
|
||||||
{
|
{
|
||||||
var builder = eb.Create().WithDescription(text);
|
var builder = eb.Create().WithDescription(text);
|
||||||
|
|
||||||
|
@@ -37,7 +37,7 @@ boost:
|
|||||||
boostmsg:
|
boostmsg:
|
||||||
- boostmsg
|
- boostmsg
|
||||||
boostdel:
|
boostdel:
|
||||||
- boostdel
|
- boostde
|
||||||
logserver:
|
logserver:
|
||||||
- logserver
|
- logserver
|
||||||
logignore:
|
logignore:
|
||||||
@@ -174,6 +174,14 @@ settopic:
|
|||||||
setchanlname:
|
setchanlname:
|
||||||
- setchanlname
|
- setchanlname
|
||||||
- schn
|
- schn
|
||||||
|
# thread stuff
|
||||||
|
threadcreate:
|
||||||
|
- threadcreate
|
||||||
|
- thcr
|
||||||
|
threaddelete:
|
||||||
|
- threaddelete
|
||||||
|
- thdel
|
||||||
|
- thrm
|
||||||
prune:
|
prune:
|
||||||
- prune
|
- prune
|
||||||
- clear
|
- clear
|
||||||
@@ -209,6 +217,9 @@ serverinfo:
|
|||||||
channelinfo:
|
channelinfo:
|
||||||
- channelinfo
|
- channelinfo
|
||||||
- cinfo
|
- cinfo
|
||||||
|
roleinfo:
|
||||||
|
- roleinfo
|
||||||
|
- rinfo
|
||||||
userinfo:
|
userinfo:
|
||||||
- userinfo
|
- userinfo
|
||||||
- uinfo
|
- uinfo
|
||||||
@@ -331,6 +342,9 @@ quoteid:
|
|||||||
quotedelete:
|
quotedelete:
|
||||||
- quotedelete
|
- quotedelete
|
||||||
- qdel
|
- qdel
|
||||||
|
quotedeleteauthor:
|
||||||
|
- quotedeleteauthor
|
||||||
|
- qdelauth
|
||||||
draw:
|
draw:
|
||||||
- draw
|
- draw
|
||||||
drawnew:
|
drawnew:
|
||||||
@@ -374,6 +388,7 @@ award:
|
|||||||
- award
|
- award
|
||||||
take:
|
take:
|
||||||
- take
|
- take
|
||||||
|
- seize
|
||||||
betroll:
|
betroll:
|
||||||
- betroll
|
- betroll
|
||||||
- br
|
- br
|
||||||
@@ -710,6 +725,13 @@ showemojis:
|
|||||||
emojiadd:
|
emojiadd:
|
||||||
- emojiadd
|
- emojiadd
|
||||||
- ea
|
- ea
|
||||||
|
emojiremove:
|
||||||
|
- emojiremove
|
||||||
|
- emojirm
|
||||||
|
- er
|
||||||
|
- ed
|
||||||
|
- emojidel
|
||||||
|
- emojidelete
|
||||||
deckshuffle:
|
deckshuffle:
|
||||||
- deckshuffle
|
- deckshuffle
|
||||||
- dsh
|
- dsh
|
||||||
@@ -941,6 +963,10 @@ banmessagetest:
|
|||||||
- banmsgtest
|
- banmsgtest
|
||||||
banmsgreset:
|
banmsgreset:
|
||||||
- banmsgreset
|
- banmsgreset
|
||||||
|
banprune:
|
||||||
|
- banprune
|
||||||
|
timeout:
|
||||||
|
- timeout
|
||||||
wait:
|
wait:
|
||||||
- wait
|
- wait
|
||||||
warnexpire:
|
warnexpire:
|
||||||
@@ -1064,6 +1090,10 @@ xpadd:
|
|||||||
- xpadd
|
- xpadd
|
||||||
xpshop:
|
xpshop:
|
||||||
- xpshop
|
- xpshop
|
||||||
|
xpshopbuy:
|
||||||
|
- xpshopbuy
|
||||||
|
xpshopuse:
|
||||||
|
- xpshopuse
|
||||||
clubcreate:
|
clubcreate:
|
||||||
- clubcreate
|
- clubcreate
|
||||||
clubtransfer:
|
clubtransfer:
|
||||||
@@ -1257,6 +1287,11 @@ expradd:
|
|||||||
- exadd
|
- exadd
|
||||||
- exa
|
- exa
|
||||||
- acr
|
- acr
|
||||||
|
expraddserver:
|
||||||
|
- expradds
|
||||||
|
- exadds
|
||||||
|
- exas
|
||||||
|
- expraddserver
|
||||||
exprlist:
|
exprlist:
|
||||||
- exprlist
|
- exprlist
|
||||||
- exl
|
- exl
|
||||||
@@ -1304,6 +1339,7 @@ medusalist:
|
|||||||
medusainfo:
|
medusainfo:
|
||||||
- medusainfo
|
- medusainfo
|
||||||
- meinfo
|
- meinfo
|
||||||
|
# Bank stuff
|
||||||
bankdeposit:
|
bankdeposit:
|
||||||
- deposit
|
- deposit
|
||||||
- d
|
- d
|
||||||
@@ -1316,6 +1352,11 @@ bankbalance:
|
|||||||
- balance
|
- balance
|
||||||
- b
|
- b
|
||||||
- bal
|
- bal
|
||||||
|
banktake:
|
||||||
|
- take
|
||||||
|
- seize
|
||||||
|
bankaward:
|
||||||
|
- award
|
||||||
# Patron
|
# Patron
|
||||||
patron:
|
patron:
|
||||||
- patron
|
- patron
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
# DO NOT CHANGE
|
# DO NOT CHANGE
|
||||||
version: 3
|
version: 4
|
||||||
# Most commands, when executed, have a small colored line
|
# Most commands, when executed, have a small colored line
|
||||||
# next to the response. The color depends whether the command
|
# next to the response. The color depends whether the command
|
||||||
# is completed, errored or in progress (pending)
|
# is completed, errored or in progress (pending)
|
||||||
@@ -18,6 +18,8 @@ defaultLocale: en-US
|
|||||||
# Style in which executed commands will show up in the console.
|
# Style in which executed commands will show up in the console.
|
||||||
# Allowed values: Simple, Normal, None
|
# Allowed values: Simple, Normal, None
|
||||||
consoleOutputType: Normal
|
consoleOutputType: Normal
|
||||||
|
# Whether the bot will check for new releases every hour
|
||||||
|
checkForUpdates: true
|
||||||
# Do you want any messages sent by users in Bot's DM to be forwarded to the owner(s)?
|
# Do you want any messages sent by users in Bot's DM to be forwarded to the owner(s)?
|
||||||
forwardMessages: false
|
forwardMessages: false
|
||||||
# Do you want the message to be forwarded only to the first owner specified in the list of owners (in creds.yml),
|
# Do you want the message to be forwarded only to the first owner specified in the list of owners (in creds.yml),
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 45 KiB After Width: | Height: | Size: 13 KiB |
@@ -197,7 +197,11 @@ iamnot:
|
|||||||
args:
|
args:
|
||||||
- "Gamer"
|
- "Gamer"
|
||||||
expradd:
|
expradd:
|
||||||
desc: "Add an expression with a trigger and a response. Running this command in server requires the Administration permission. Running this command in DM is Bot Owner only and adds a new global expression. Guide here: <https://nadekobot.readthedocs.io/en/latest/custom-reactions/>"
|
desc: "Add an expression with a trigger and a response. Bot will post a response whenever someone types the trigger word. Running this command in server requires the Administration permission. Running this command in DM is Bot Owner only and adds a new global expression. Guide here: <https://nadekobot.readthedocs.io/en/latest/custom-reactions/>"
|
||||||
|
args:
|
||||||
|
- "\"hello\" Hi there %user.mention%"
|
||||||
|
expraddserver:
|
||||||
|
desc: "Add an expression with a trigger and a response in this server. Bot will post a response whenever someone types the trigger word. Guide here: <https://nadekobot.readthedocs.io/en/latest/custom-reactions/>"
|
||||||
args:
|
args:
|
||||||
- "\"hello\" Hi there %user.mention%"
|
- "\"hello\" Hi there %user.mention%"
|
||||||
exprlist:
|
exprlist:
|
||||||
@@ -312,6 +316,11 @@ kick:
|
|||||||
args:
|
args:
|
||||||
- "@Someone Get out!"
|
- "@Someone Get out!"
|
||||||
- "\"Some Guy#1234\" Your behaviour is toxic."
|
- "\"Some Guy#1234\" Your behaviour is toxic."
|
||||||
|
timeout:
|
||||||
|
desc: "Times the user out for the specified amount of time. You may optionally specify a reason, which will be sent to the user."
|
||||||
|
args:
|
||||||
|
- "@Someone 3h Shut up!"
|
||||||
|
- "@Someone 1h30m"
|
||||||
mute:
|
mute:
|
||||||
desc: "Mutes a mentioned user both from speaking and chatting. You can also specify time string for how long the user should be muted. You can optionally specify a reason."
|
desc: "Mutes a mentioned user both from speaking and chatting. You can also specify time string for how long the user should be muted. You can optionally specify a reason."
|
||||||
args:
|
args:
|
||||||
@@ -423,6 +432,10 @@ channelinfo:
|
|||||||
desc: "Shows info about the channel. If no channel is supplied, it defaults to current one."
|
desc: "Shows info about the channel. If no channel is supplied, it defaults to current one."
|
||||||
args:
|
args:
|
||||||
- "#some-channel"
|
- "#some-channel"
|
||||||
|
roleinfo:
|
||||||
|
desc: "Shows info about the specified role."
|
||||||
|
args:
|
||||||
|
- "Gamers"
|
||||||
userinfo:
|
userinfo:
|
||||||
desc: "Shows info about the user. If no user is supplied, it defaults a user running the command."
|
desc: "Shows info about the user. If no user is supplied, it defaults a user running the command."
|
||||||
args:
|
args:
|
||||||
@@ -619,6 +632,10 @@ quotedelete:
|
|||||||
desc: "Deletes a quote with the specified ID. You have to either have the Manage Messages permission or be the creator of the quote to delete it."
|
desc: "Deletes a quote with the specified ID. You have to either have the Manage Messages permission or be the creator of the quote to delete it."
|
||||||
args:
|
args:
|
||||||
- "123456"
|
- "123456"
|
||||||
|
quotedeleteauthor:
|
||||||
|
desc: "Deletes all quotes by the specified author. If the author is not you, then ManageMessage server permission is required."
|
||||||
|
args:
|
||||||
|
- "@QuoteSpammer"
|
||||||
draw:
|
draw:
|
||||||
desc: "Draws a card from this server's deck. You can draw up to 10 cards by supplying a number of cards to draw."
|
desc: "Draws a card from this server's deck. You can draw up to 10 cards by supplying a number of cards to draw."
|
||||||
args:
|
args:
|
||||||
@@ -776,10 +793,9 @@ linux:
|
|||||||
args:
|
args:
|
||||||
- "Spyware Windows"
|
- "Spyware Windows"
|
||||||
next:
|
next:
|
||||||
desc: "Goes to the next song in the queue. You have to be in the same voice channel as the bot. You can skip multiple songs, but in that case songs will not be requeued if {0}rcs or {0}rpl is enabled."
|
desc: "Goes to the next song in the queue. You have to be in the same voice channel as the bot"
|
||||||
args:
|
args:
|
||||||
- ""
|
- ""
|
||||||
- "5"
|
|
||||||
play:
|
play:
|
||||||
desc: "If no parameters are specified, acts as `{0}next 1` command. If you specify a song number, it will jump to that song. If you specify a search query, acts as a `{0}q` command"
|
desc: "If no parameters are specified, acts as `{0}next 1` command. If you specify a song number, it will jump to that song. If you specify a search query, acts as a `{0}q` command"
|
||||||
args:
|
args:
|
||||||
@@ -1217,6 +1233,10 @@ emojiadd:
|
|||||||
- ":someonesCustomEmoji:"
|
- ":someonesCustomEmoji:"
|
||||||
- "MyEmojiName :someonesCustomEmoji:"
|
- "MyEmojiName :someonesCustomEmoji:"
|
||||||
- "owoNice https://cdn.discordapp.com/emojis/587930873811173386.png?size=128"
|
- "owoNice https://cdn.discordapp.com/emojis/587930873811173386.png?size=128"
|
||||||
|
emojiremove:
|
||||||
|
desc: "Removes the specified emoji or emojis from this server."
|
||||||
|
args:
|
||||||
|
- ":eagleWarrior: :plumedArcher:"
|
||||||
deckshuffle:
|
deckshuffle:
|
||||||
desc: "Reshuffles all cards back into the deck."
|
desc: "Reshuffles all cards back into the deck."
|
||||||
args:
|
args:
|
||||||
@@ -1664,6 +1684,13 @@ banmsgreset:
|
|||||||
desc: "Resets ban message to default. If you want to completely disable ban messages, use `{0}banmsg -`"
|
desc: "Resets ban message to default. If you want to completely disable ban messages, use `{0}banmsg -`"
|
||||||
args:
|
args:
|
||||||
- ""
|
- ""
|
||||||
|
banprune:
|
||||||
|
desc: |-
|
||||||
|
Sets how many days of messages will be deleted when a user is banned.
|
||||||
|
Only works if the user is banned via the .ban command or punishment.
|
||||||
|
Allowed values: 0 - 7
|
||||||
|
args:
|
||||||
|
- "3"
|
||||||
wait:
|
wait:
|
||||||
desc: "Used only as a startup command. Waits a certain number of miliseconds before continuing the execution of the following startup commands."
|
desc: "Used only as a startup command. Waits a certain number of miliseconds before continuing the execution of the following startup commands."
|
||||||
args:
|
args:
|
||||||
@@ -2128,6 +2155,16 @@ xpshop:
|
|||||||
- "bgs"
|
- "bgs"
|
||||||
- "frames"
|
- "frames"
|
||||||
- "bgs 3"
|
- "bgs 3"
|
||||||
|
xpshopbuy:
|
||||||
|
desc: "Buy an item from the xp shop by specifying the type and the key of the item."
|
||||||
|
args:
|
||||||
|
- "bg open_sea"
|
||||||
|
- "fr gold"
|
||||||
|
xpshopuse:
|
||||||
|
desc: "Use a previously purchased item from the xp shop by specifying the type and the key of the item."
|
||||||
|
args:
|
||||||
|
- "bg synth"
|
||||||
|
- "fr default"
|
||||||
bible:
|
bible:
|
||||||
desc: "Shows bible verse. You need to supply book name and chapter:verse"
|
desc: "Shows bible verse. You need to supply book name and chapter:verse"
|
||||||
args:
|
args:
|
||||||
@@ -2227,6 +2264,14 @@ bankbalance:
|
|||||||
desc: "Shows your current bank balance available for withdrawal."
|
desc: "Shows your current bank balance available for withdrawal."
|
||||||
args:
|
args:
|
||||||
- ""
|
- ""
|
||||||
|
banktake:
|
||||||
|
desc: "Takes the specified amount of currency from a user's bank"
|
||||||
|
args:
|
||||||
|
- "500 @MoniLaunder"
|
||||||
|
bankaward:
|
||||||
|
desc: "Award the specified amount of currency to a user's bank"
|
||||||
|
args:
|
||||||
|
- "99999 @Bestie"
|
||||||
patron:
|
patron:
|
||||||
desc: "Check your patronage status and command usage quota. Bot owners can check targeted user's patronage status."
|
desc: "Check your patronage status and command usage quota. Bot owners can check targeted user's patronage status."
|
||||||
args:
|
args:
|
||||||
@@ -2268,3 +2313,11 @@ bettest:
|
|||||||
- ""
|
- ""
|
||||||
- "betflip 1000"
|
- "betflip 1000"
|
||||||
- "slot 2000"
|
- "slot 2000"
|
||||||
|
threadcreate:
|
||||||
|
desc: "Create a public thread with the specified title. You may optionally reply to a message to have it as a starting point."
|
||||||
|
args:
|
||||||
|
- "Q&A"
|
||||||
|
threaddelete:
|
||||||
|
desc: "Delete a thread with the specified name in this channel. Case insensitive."
|
||||||
|
args:
|
||||||
|
- "Q&A"
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"api_key_missing": "Der API Key der für diesen Befehl benötigt wird ist nicht angegeben.",
|
"api_key_missing": "Der API Key fehlt.",
|
||||||
"quote_deleted": "Zitat gelöscht",
|
"quote_deleted": "Zitat gelöscht",
|
||||||
"redacted_too_long": "Wurde entfernt da es zu lang ist.",
|
"redacted_too_long": "Wurde entfernt da es zu lang ist.",
|
||||||
"trigger": "Auslöser",
|
"trigger": "Auslöser",
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
"banned_user": "Nutzer gebannt",
|
"banned_user": "Nutzer gebannt",
|
||||||
"byedel_off": "Die automatische Löschung der Verabschiedungs Nachrichten wurde deaktiviert.",
|
"byedel_off": "Die automatische Löschung der Verabschiedungs Nachrichten wurde deaktiviert.",
|
||||||
"byedel_on": "Verabschiedungs Nachrichten werden nach {0} Sekunden gelöscht.",
|
"byedel_on": "Verabschiedungs Nachrichten werden nach {0} Sekunden gelöscht.",
|
||||||
"byemsg_cur": "Derzeitige Verabschiedungs Nachricht",
|
"byemsg_cur": "Derzeitige Verabschiedungs Nachricht: {0}",
|
||||||
"byemsg_enable": "Beim schreiben von {0} werden die Verabschiedungs Nachrichten aktiviert.",
|
"byemsg_enable": "Beim schreiben von {0} werden die Verabschiedungs Nachrichten aktiviert.",
|
||||||
"byemsg_new": "Neue Verabschiedungs Nachricht gesetzt",
|
"byemsg_new": "Neue Verabschiedungs Nachricht gesetzt",
|
||||||
"bye_off": "Abschiedansagen ausgeschaltet.",
|
"bye_off": "Abschiedansagen ausgeschaltet.",
|
||||||
@@ -922,9 +922,9 @@
|
|||||||
"pages": "Seiten",
|
"pages": "Seiten",
|
||||||
"favorites": "Favoriten",
|
"favorites": "Favoriten",
|
||||||
"tags": "Stichworte",
|
"tags": "Stichworte",
|
||||||
"invalid_emoji_link": "",
|
"invalid_emoji_link": "Der angegebene Link is entweder kein Bild, oder größer als 256KB.",
|
||||||
"emoji_add_error": "",
|
"emoji_add_error": "Fehler beim hinzufügen des Emoji. Es sind keine Emoji slots mehr vorhanden, oder die Bildgröße ist unangemessen.",
|
||||||
"emoji_added": "",
|
"emoji_added": "Neuen Emoji hinzugefügt: {0}",
|
||||||
"boost_on": "",
|
"boost_on": "",
|
||||||
"boost_off": "",
|
"boost_off": "",
|
||||||
"boostmsg_cur": "",
|
"boostmsg_cur": "",
|
||||||
@@ -932,16 +932,16 @@
|
|||||||
"boostmsg_new": "",
|
"boostmsg_new": "",
|
||||||
"boostdel_off": "",
|
"boostdel_off": "",
|
||||||
"boostdel_on": "",
|
"boostdel_on": "",
|
||||||
"log_ignored_channels": "",
|
"log_ignored_channels": "Ignorierte Kanäle",
|
||||||
"log_ignored_users": "",
|
"log_ignored_users": "Ignorierte User",
|
||||||
"log_ignore_user": "",
|
"log_ignore_user": "Logging wird User {0} ignorieren",
|
||||||
"log_not_ignore_user": "",
|
"log_not_ignore_user": "Logging wird nicht länger User {0} ignorieren",
|
||||||
"log_ignore_chan": "",
|
"log_ignore_chan": "Logging wird Kanal {0} ignorieren",
|
||||||
"log_not_ignore_chan": "",
|
"log_not_ignore_chan": "Logging wird nicht länger Kanal {0} ignorieren",
|
||||||
"streams_cleared": "",
|
"streams_cleared": "",
|
||||||
"warn_weight": "",
|
"warn_weight": "Gewicht: {0}",
|
||||||
"warn_count": "",
|
"warn_count": "{0} momentan, {1} insgesamt",
|
||||||
"mass_ban_in_progress": "",
|
"mass_ban_in_progress": "Es werden {0} User gebannt...",
|
||||||
"mass_ban_completed": "",
|
"mass_ban_completed": "",
|
||||||
"reminder_server_list": "",
|
"reminder_server_list": "",
|
||||||
"imageonly_enable": "",
|
"imageonly_enable": "",
|
||||||
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -12,6 +12,7 @@
|
|||||||
"invalid_emoji_link": "Specified link is either not an image or exceeds 256KB.",
|
"invalid_emoji_link": "Specified link is either not an image or exceeds 256KB.",
|
||||||
"emoji_add_error": "Error adding emoji. You either ran out of emoji slots, or image size is inadequate.",
|
"emoji_add_error": "Error adding emoji. You either ran out of emoji slots, or image size is inadequate.",
|
||||||
"emoji_added": "Added a new emoji: {0}",
|
"emoji_added": "Added a new emoji: {0}",
|
||||||
|
"emoji_not_removed": "The following emojis were not removed: {0}",
|
||||||
"fw_cleared": "Removed all filtered words and filtered words channel settings.",
|
"fw_cleared": "Removed all filtered words and filtered words channel settings.",
|
||||||
"aliases_cleared": "All {0} aliases on this server have been removed.",
|
"aliases_cleared": "All {0} aliases on this server have been removed.",
|
||||||
"no_results": "No results found.",
|
"no_results": "No results found.",
|
||||||
@@ -27,6 +28,10 @@
|
|||||||
"banmsg_default": "No ban message set. Default behavior will be used.",
|
"banmsg_default": "No ban message set. Default behavior will be used.",
|
||||||
"banned_pl": "banned",
|
"banned_pl": "banned",
|
||||||
"banned_user": "User Banned",
|
"banned_user": "User Banned",
|
||||||
|
"ban_prune_disabled": "Banned user's messages will no longer be deleted.",
|
||||||
|
"ban_prune": "Bot will prune up to {0} day(s) worth of messages from banned user.",
|
||||||
|
"timeoutdm": "You have been timed out in {0} server.\nReason: {1}",
|
||||||
|
"timedout_user": "User Timed Out",
|
||||||
"remove_roles_pl": "have had their roles removed",
|
"remove_roles_pl": "have had their roles removed",
|
||||||
"bot_name": "Bot's name changed to {0}",
|
"bot_name": "Bot's name changed to {0}",
|
||||||
"bot_status": "Bot status changed to {0}",
|
"bot_status": "Bot status changed to {0}",
|
||||||
@@ -141,6 +146,10 @@
|
|||||||
"rar_err": "Failed to remove roles. I have insufficient permissions.",
|
"rar_err": "Failed to remove roles. I have insufficient permissions.",
|
||||||
"rc": "Color of {0} role has been changed.",
|
"rc": "Color of {0} role has been changed.",
|
||||||
"rc_perms": "Error occurred due to invalid color or insufficient permissions.",
|
"rc_perms": "Error occurred due to invalid color or insufficient permissions.",
|
||||||
|
"color": "Color",
|
||||||
|
"icon": "Icon",
|
||||||
|
"hoisted": "Hoisted",
|
||||||
|
"mentionable": "Mentionable",
|
||||||
"remrole": "Successfully removed role {0} from user {1}",
|
"remrole": "Successfully removed role {0} from user {1}",
|
||||||
"remrole_err": "Failed to remove role. I have insufficient permissions.",
|
"remrole_err": "Failed to remove role. I have insufficient permissions.",
|
||||||
"renrole": "Role renamed.",
|
"renrole": "Role renamed.",
|
||||||
@@ -588,6 +597,7 @@
|
|||||||
"presence": "Presence",
|
"presence": "Presence",
|
||||||
"presence_txt": "{0} Servers\n{1} Text Channels\n{2} Voice Channels",
|
"presence_txt": "{0} Servers\n{1} Text Channels\n{2} Voice Channels",
|
||||||
"quotes_deleted": "Deleted all quotes with {0} keyword.",
|
"quotes_deleted": "Deleted all quotes with {0} keyword.",
|
||||||
|
"quotes_deleted_count": "Deleted {0} quotes.",
|
||||||
"quotes_page": "Page {0} of quotes",
|
"quotes_page": "Page {0} of quotes",
|
||||||
"quotes_page_none": "No quotes found on that page.",
|
"quotes_page_none": "No quotes found on that page.",
|
||||||
"quotes_remove_none": "No quotes found which you can remove.",
|
"quotes_remove_none": "No quotes found which you can remove.",
|
||||||
@@ -1003,7 +1013,7 @@
|
|||||||
"feature_limit_reached_you": "You've reached the limit of {0} for the {1} feature. You may be able to increase this limit by upgrading your patron tier.",
|
"feature_limit_reached_you": "You've reached the limit of {0} for the {1} feature. You may be able to increase this limit by upgrading your patron tier.",
|
||||||
"feature_limit_reached_owner": "Server owner has reached the limit of {0} for the {1} feature. Server owner may be able to upgrade this limit by upgrading patron tier.",
|
"feature_limit_reached_owner": "Server owner has reached the limit of {0} for the {1} feature. Server owner may be able to upgrade this limit by upgrading patron tier.",
|
||||||
"feature_limit_reached_either": "The limit of {0} for the {1} feature has been reached. Either you or the server owner may able to upgrade this limit by upgrading the patron tier.",
|
"feature_limit_reached_either": "The limit of {0} for the {1} feature has been reached. Either you or the server owner may able to upgrade this limit by upgrading the patron tier.",
|
||||||
"xp_shop_buy_required_tier": "Buying items from the shop requires Patron Tier {0} or higher.",
|
"xp_shop_buy_required_tier": "Buying items from this shop requires Patron Tier {0} or higher.",
|
||||||
"available_commands": "Available Commands",
|
"available_commands": "Available Commands",
|
||||||
"tier": "Tier",
|
"tier": "Tier",
|
||||||
"pledge": "Pledge",
|
"pledge": "Pledge",
|
||||||
@@ -1017,5 +1027,10 @@
|
|||||||
"patron_not_enabled": "Patron system is disabled.",
|
"patron_not_enabled": "Patron system is disabled.",
|
||||||
"results_in": "{0} results in {1}s",
|
"results_in": "{0} results in {1}s",
|
||||||
"patron_msg_sent": "Done sending messages to patrons at and above tier {1}. {1} successfully sent and {2} failed.",
|
"patron_msg_sent": "Done sending messages to patrons at and above tier {1}. {1} successfully sent and {2} failed.",
|
||||||
"xpadd_users": "Added {0} server XP to {1} users."
|
"xpadd_users": "Added {0} server XP to {1} users.",
|
||||||
|
"xpshop_buy_success": "Successfully purchased `{0}/{1}`",
|
||||||
|
"patron_insuff_tier": "Your Patron Tier insufficient to perform this action.",
|
||||||
|
"xpshop_already_owned": "You already own this item.",
|
||||||
|
"xpshop_item_not_found": "An item with that key doesn't exist.",
|
||||||
|
"xpshop_website": "You can see the list of all Xp Shop items here: <https://xpshop.nadeko.bot>"
|
||||||
}
|
}
|
||||||
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "Se han eliminado las {0} expresiones de este servidor.",
|
"exprs_cleared": "Se han eliminado las {0} expresiones de este servidor.",
|
||||||
"expr_reset": "Las expresiones con la ID {0} ya no agregarán reacciones.",
|
"expr_reset": "Las expresiones con la ID {0} ya no agregarán reacciones.",
|
||||||
"expr_set": "La expresión con la ID {0} añadirá las siguientes reacciones al mensaje de respuesta: {1}",
|
"expr_set": "La expresión con la ID {0} añadirá las siguientes reacciones al mensaje de respuesta: {1}",
|
||||||
"expr_edited": "Expresión editada."
|
"expr_edited": "Expresión editada.",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -143,17 +143,17 @@
|
|||||||
"waifus_top_waifus": "Meilleures Waifu",
|
"waifus_top_waifus": "Meilleures Waifu",
|
||||||
"waifu_claimed": "a revendiqué {0} comme sa waifu pour {1} !",
|
"waifu_claimed": "a revendiqué {0} comme sa waifu pour {1} !",
|
||||||
"waifu_divorced_like": "Vous avez divorcé d'une waifu qui vous aimait. Vous n'êtes qu'un monstre sans cœur, {0} a reçu {1} en compensation.",
|
"waifu_divorced_like": "Vous avez divorcé d'une waifu qui vous aimait. Vous n'êtes qu'un monstre sans cœur, {0} a reçu {1} en compensation.",
|
||||||
"waifu_egomaniac": "vous ne pouvez pas tomber amoureux de vous-même, gros narcissique.",
|
"waifu_egomaniac": "vous ne pouvez pas tomber amoureux·se de vous-même, narcissique.",
|
||||||
"waifu_fulfilled": "🎉 Leur amour est accompli 🎉\nLa nouvelle valeur de {0} est {1} !",
|
"waifu_fulfilled": "🎉 Leur amour est accompli 🎉\nLa nouvelle valeur de {0} est {1} !",
|
||||||
"waifu_not_enough": "Vous devez payer {0} ou plus pour revendiquer cette waifu !",
|
"waifu_not_enough": "Vous devrez payer {0} ou plus pour revendiquer cette waifu !",
|
||||||
"waifu_not_yours": "Cette waifu n'est pas vôtre.",
|
"waifu_not_yours": "Cette waifu n'est pas vôtre.",
|
||||||
"waifu_not_yourself": "Vous ne pouvez pas vous revendiquer vous-même.",
|
"waifu_not_yourself": "Vous ne pouvez pas vous revendiquer vous-même.",
|
||||||
"waifu_recent_divorce": "Vous avez récemment divorcé. Vous devez attendre {0} heures et {1} minutes avant de pouvoir divorcer à nouveau.",
|
"waifu_recent_divorce": "Vous avez récemment divorcé·e. Vous devrez attendre {0} heures et {1} minutes avant de pouvoir divorcer à nouveau.",
|
||||||
"waifu_transfer_fail": "Vous n'êtes pas le propriétaire de cette waifu.",
|
"waifu_transfer_fail": "Vous n'êtes pas le propriétaire de cette waifu.",
|
||||||
"waifu_transfer_success": "La propriété de {0} a été transférée de {1} à {2}.",
|
"waifu_transfer_success": "La propriété de {0} a été transférée de {1} à {2}.",
|
||||||
"nobody": "Personne",
|
"nobody": "Personne",
|
||||||
"waifu_divorced_notlike": "Vous avez divorcé d'une waifu qui ne vous aimait pas. Vous avez reçu {0} en compensation.",
|
"waifu_divorced_notlike": "Vous avez divorcé d'une waifu qui ne vous aimait pas. Vous avez reçu {0} en compensation.",
|
||||||
"waifu_reset": "Vos stats de waifu ont été réinitialisée.",
|
"waifu_reset": "Vos stats de waifu ont été réinitialisées.",
|
||||||
"waifu_reset_fail": "Échec de la réinitialisation des stats de waifu. Assurez-vous d'avoir assez de devises.",
|
"waifu_reset_fail": "Échec de la réinitialisation des stats de waifu. Assurez-vous d'avoir assez de devises.",
|
||||||
"waifu_reset_confirm": "Cela réinitialisera vos stats de waifu",
|
"waifu_reset_confirm": "Cela réinitialisera vos stats de waifu",
|
||||||
"acrophobia": "Acrophobie",
|
"acrophobia": "Acrophobie",
|
||||||
@@ -513,12 +513,12 @@
|
|||||||
"feed_out_of_range": "L'index est en dehors de la plage disponible.",
|
"feed_out_of_range": "L'index est en dehors de la plage disponible.",
|
||||||
"feed_removed": "Le flux est supprimé.",
|
"feed_removed": "Le flux est supprimé.",
|
||||||
"feed_no_feed": "Vous n'avez souscrit à aucun fil d'actualité sur ce serveur.",
|
"feed_no_feed": "Vous n'avez souscrit à aucun fil d'actualité sur ce serveur.",
|
||||||
"timely_none": "Le propriétaire de ce bot n'a pas spécifié de récompense récurrente.",
|
"timely_none": "Le propriétaire de ce bot n'a pas spécifié de récompense opportune.",
|
||||||
"timely_already_claimed": "Vous avez déjà obtenu votre récompense récurrente. Vous pourrez l'obtenir de nouveau dans {0}.",
|
"timely_already_claimed": "Vous avez déjà obtenu votre récompense opportune. Vous ne pourrez l'obtenir de nouveau que {0}.",
|
||||||
"timely": "Vous venez de réclamer vos {0}. Vous pourrez réclamer à nouveau dans {1}h.",
|
"timely": "Vous venez de réclamer vos {0}. Vous pourrez réclamer à nouveau dans {1}h.",
|
||||||
"timely_set": "Les utilisateurs pourront réclamer {0} toutes les {1}h.",
|
"timely_set": "Les utilisateurs pourront réclamer {0} toutes les {1}h.",
|
||||||
"timely_set_none": "Les utilisateurs ne pourront réclamer aucune devise récurrente.",
|
"timely_set_none": "Les utilisateurs ne pourront réclamer aucune récompense opportune.",
|
||||||
"timely_reset": "Tous les utilisateurs peuvent réclamer leur devise récurrente de nouveau.",
|
"timely_reset": "Tous les utilisateurs pourront réclamer leurs récompenses opportunes de nouveau.",
|
||||||
"market_cap": "Capitalisation Boursière",
|
"market_cap": "Capitalisation Boursière",
|
||||||
"volume_24h": "Volume (24h)",
|
"volume_24h": "Volume (24h)",
|
||||||
"change_7d_24h": "Change (7j/24h)",
|
"change_7d_24h": "Change (7j/24h)",
|
||||||
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "L'ensemble des {0} expressions du serveur ont été retirées.",
|
"exprs_cleared": "L'ensemble des {0} expressions du serveur ont été retirées.",
|
||||||
"expr_reset": "L'expression avec l'id {0} ne rajoutera plus de réactions.",
|
"expr_reset": "L'expression avec l'id {0} ne rajoutera plus de réactions.",
|
||||||
"expr_set": "L'expression avec l'id {0} ajoutera automatiquement ces réactions au message de réponse : {1}",
|
"expr_set": "L'expression avec l'id {0} ajoutera automatiquement ces réactions au message de réponse : {1}",
|
||||||
"expr_edited": "Expression éditée"
|
"expr_edited": "Expression éditée",
|
||||||
|
"stream_online_delete_enabled": "Aucune méduse de ce nom n'a été trouvée ou le fichier était invalide",
|
||||||
|
"stream_online_delete_disabled": "Les notifications de diffusions en ligne ne seront plus supprimées quand la diffusion sera hors ligne.",
|
||||||
|
"club_create_error_name": "Échec de la création du club. Un club existe du même nom existe déjà.",
|
||||||
|
"club_desc_update": "La description du club a été mise à jour",
|
||||||
|
"bank_accounts": "Comptes bancaires",
|
||||||
|
"module_description_medusa": "**Unique au propriétaire du bot.** Charger, décharger et maintenez des modules dynamiques. Découvrez en plus [ici](https://nadekobot.readthedocs.io/en/latest/medusa/creating-a-medusa/)",
|
||||||
|
"list_of_medusae": "Liste des méduses",
|
||||||
|
"list_of_unloaded": "Liste des méduses disponible",
|
||||||
|
"medusa_name_not_found": "Une méduse de ce nom n'existe pas ou n'est pas chargée.",
|
||||||
|
"medusa_info": "Information de la méduse",
|
||||||
|
"sneks_count": "Sneks ({0})",
|
||||||
|
"commands_count": "Commandes ({0})",
|
||||||
|
"no_medusa_loaded": "Aucunes méduses de chargées.",
|
||||||
|
"no_medusa_available": "Aucune méduse de disponible.",
|
||||||
|
"loaded_medusae": "Méduses chargées",
|
||||||
|
"medusa_not_loaded": "La méduse de ce nom n'est pas chargée.",
|
||||||
|
"medusa_possibly_cant_unload": "La méduse n'est probablement pas entièrement chargée. Redémarrez le bot si le problème persiste.",
|
||||||
|
"medusa_loaded": "La méduse {0} a été chargée.",
|
||||||
|
"medusa_unloaded": "La méduse {0} a été déchargée.",
|
||||||
|
"medusa_empty": "La méduse n'a pas été chargée puisqu'elle ne contient aucun Sneks.",
|
||||||
|
"medusa_already_loaded": "La méduse {0} est déjà chargée",
|
||||||
|
"medusa_invalid_not_found": "Aucune méduse de ce nom n'a été trouvée ou le fichier était invalide",
|
||||||
|
"bank_balance": "Vous disposez de {0} sur votre compte bancaire.",
|
||||||
|
"bank_deposited": "Vous avez déposé {0} sur votre compte bancaire.",
|
||||||
|
"bank_withdrew": "Vous avez retiré {0} de votre compte bancaire.",
|
||||||
|
"bank_withdraw_insuff": "Vous ne disposez pas suffisamment de {0} sur votre compte bancaire.",
|
||||||
|
"cmd_group_commands": "Groupe de commande '{0}'",
|
||||||
|
"limit_reached": "La limite des fonctionnalités de {0} a été atteinte.",
|
||||||
|
"feature_limit_reached_you": "Vous avez atteint la limite de {0} pour la fonctionnalité {1}. Vous pouvez augmenter cette limite en augmentant votre niveau de contribution.",
|
||||||
|
"feature_limit_reached_owner": "Le propriétaire du serveur a atteint la limite de {0} pour la fonctionnalité {1}. Ce dernier peut augmenter cette limite en augmentant son niveau de contribution.",
|
||||||
|
"feature_limit_reached_either": "La limite de {0} pour la fonctionnalité {1} a été atteinte. Soit vous ou le propriétaire du serveur peut augmenter cette limite en augmentant le niveau de contribution.",
|
||||||
|
"tier": "Niveau",
|
||||||
|
"pledge": "Promesse",
|
||||||
|
"expires": "Expire",
|
||||||
|
"commands": "Commandes",
|
||||||
|
"groups": "Groupes",
|
||||||
|
"modules": "Modules",
|
||||||
|
"no_quota_found": "Aucun quota de trouvé.",
|
||||||
|
"patron_info": "Info de la·e contributeur·rice",
|
||||||
|
"quotas": "<<< Quotas >>>",
|
||||||
|
"patron_not_enabled": "Le système de contribution est désactivé.",
|
||||||
|
"results_in": "{0} résulte en {1}s",
|
||||||
|
"patron_msg_sent": "L'envoi de messages aux contributeur·rice·s de niveau {1} et au-delà a été effectué. {1} ont été envoyés avec succès et {2} ont échoué.",
|
||||||
|
"cards": "Cartes",
|
||||||
|
"hand_value": "Valeur de la main",
|
||||||
|
"roll2": "Lancé",
|
||||||
|
"rolls": "Lancés",
|
||||||
|
"available_tests": "Tests disponibles",
|
||||||
|
"test_results_for": "Résultats du test pour {0}",
|
||||||
|
"multiplier": "Multiplier",
|
||||||
|
"trivia_ended": "Le jeu de culture générale est terminé",
|
||||||
|
"card": "Carte",
|
||||||
|
"guess": "Deviner",
|
||||||
|
"repeater_skip_next": "Le prochain rappel de ce répéteur sera ignoré.",
|
||||||
|
"repeater_dont_skip_next": "Le prochain rappel de ce répéteur ne sera pas ignoré.",
|
||||||
|
"remind_timely": "Je vous notifierai afin d'obtenir votre prochaine récompense {0}",
|
||||||
|
"xp_shop_disabled": "Le magasin d'XP est désactivé par le propriétaire, ou il n'y a aucun item à vendre.",
|
||||||
|
"timely_time": "C'est l'heure votre récompense opportune.",
|
||||||
|
"buy": "Acheter",
|
||||||
|
"use": "Utiliser",
|
||||||
|
"in_use": "En utilisation",
|
||||||
|
"xp_shop_item_cant_use": "Vous ne pouvez pas utiliser cet item car il ne vous appartient pas ou il n'existe pas.",
|
||||||
|
"linkonly_enable": "Ce channel est désormais limité aux liens.",
|
||||||
|
"linkonly_disable": "Ce channel n'est désormais plus limité aux liens.",
|
||||||
|
"xp_shop_buy_required_tier": "Acheter des items de ce magasin requiert d'être Tier Patron {0} ou au-delà.",
|
||||||
|
"available_commands": "Commandes disponibles",
|
||||||
|
"xpadd_users": "Ajout de {0} XP du serveur à l'utilisateur {1}.",
|
||||||
|
"xpshop_buy_success": "Vous avez acheté avec succès `{0}/{1}`",
|
||||||
|
"patron_insuff_tier": "Votre Tier Patron est insuffisant pour réaliser cette action.",
|
||||||
|
"xpshop_already_owned": "Vous disposez déjà de cet item.",
|
||||||
|
"xpshop_item_not_found": "Aucun item disposant de cette clé n'existe.",
|
||||||
|
"xpshop_website": "Vous pouvez voir la liste de tous les items du magasin d'XP ici: <https://xpshop.nadeko.bot>"
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -1,22 +1,22 @@
|
|||||||
{
|
{
|
||||||
"api_key_missing": "Api sleutel ontbreekt.",
|
"api_key_missing": "Api sleutel ontbreekt.",
|
||||||
"quote_deleted": "Citaat #{0} verwijderd",
|
"quote_deleted": "Citaat #{0} verwijderd.",
|
||||||
"redacted_too_long": "Aangepast omdat het te lang is.",
|
"redacted_too_long": "Aangepast omdat het te lang is.",
|
||||||
"trigger": "Trigger",
|
"trigger": "Trigger, leiden tot",
|
||||||
"response": "Reactie",
|
"response": "Reactie",
|
||||||
"fw_cleared": "Alle gefilterde woorden en gefilterde woordenkanalen verwijderd",
|
"fw_cleared": "Alle gefilterde woorden en gefilterde woorden kanaalinstellingen verwijderd.",
|
||||||
"aar_disabled": "Gebruikers die lid worden van deze server zullen niet automatisch rollen krijgen.",
|
"aar_disabled": "**Automatisch toekennen van de rol** bij het toetreden van een gebruiker is nu **uitgeschakeld**.",
|
||||||
"bandm": "Je bent verbannen van de {0} server.\nReden: {1}",
|
"bandm": "Je bent verbannen van de {0} server.\nReden: {1}",
|
||||||
"banned_user": "Gebruiker is verbannen.",
|
"banned_user": "Gebruiker is verbannen.",
|
||||||
"byedel_off": "Automatisch verwijderen van afscheidsberichten is uitgeschakeld.",
|
"byedel_off": "Automatisch verwijderen van afscheidsberichten is uitgeschakeld.",
|
||||||
"byedel_on": "Afscheidsbericht wordt verwijderd na {0} seconden.",
|
"byedel_on": "Afscheidsbericht wordt verwijderd na {0} seconden.",
|
||||||
"byemsg_cur": "Huidig afscheidsbericht",
|
"byemsg_cur": "Huidig afscheidsbericht: {0}",
|
||||||
"byemsg_enable": "Zet afscheidsberichten aan met {0}",
|
"byemsg_enable": "Activeer afscheidsberichten door {0} te typen",
|
||||||
"byemsg_new": "Nieuw afscheidsbericht is ingesteld.",
|
"byemsg_new": "Nieuw afscheidsbericht is ingesteld.",
|
||||||
"bye_off": "Vertreknotificaties uitgezet.",
|
"bye_off": "Vertreknotificaties uitgezet.",
|
||||||
"bye_on": "Vertreknotificaties ingesteld voor dit kanaal.",
|
"bye_on": "Vertreknotificaties ingesteld voor dit kanaal.",
|
||||||
"cr": "Rol {0} is succesvol aangemaakt",
|
"cr": "Rol {0} is succesvol aangemaakt",
|
||||||
"dr": "De rol {0} is verwijderd.",
|
"dr": "Rol {0} is verwijderd.",
|
||||||
"createtextchan": "Tekst kanaal {0} aangemaakt.",
|
"createtextchan": "Tekst kanaal {0} aangemaakt.",
|
||||||
"createvoich": "Spraak-kanaal {0} aangemaakt.",
|
"createvoich": "Spraak-kanaal {0} aangemaakt.",
|
||||||
"deafen": "Demping succesvol ingesteld.",
|
"deafen": "Demping succesvol ingesteld.",
|
||||||
@@ -33,12 +33,12 @@
|
|||||||
"fwdm_stop": "Ik zal vanaf nu geen Priveberichten meer doorsturen.",
|
"fwdm_stop": "Ik zal vanaf nu geen Priveberichten meer doorsturen.",
|
||||||
"greetdel_off": "Automatisch verwijderen van welkomstberichten is uitgeschakeld.",
|
"greetdel_off": "Automatisch verwijderen van welkomstberichten is uitgeschakeld.",
|
||||||
"greetdel_on": "Welkomstberichten worden na {0} seconden verwijderd.",
|
"greetdel_on": "Welkomstberichten worden na {0} seconden verwijderd.",
|
||||||
"greetdmmsg_cur": "Huidige privebericht begroeting",
|
"greetdmmsg_cur": "Huidige privebericht begroeting: {0}",
|
||||||
"greetdmmsg_enable": "Zet privebericht begroeting aan door {0} te typen.",
|
"greetdmmsg_enable": "Zet privebericht begroeting aan door {0} te typen.",
|
||||||
"greetdmmsg_new": "Nieuw privebericht begroeting ingesteld.",
|
"greetdmmsg_new": "Nieuw privebericht begroeting ingesteld.",
|
||||||
"greetdm_off": "Begroetingsnotificaties per PB uitgezet.",
|
"greetdm_off": "Begroetingsnotificaties per PB uitgezet.",
|
||||||
"greetdm_on": "Begroetingsnotificaties per PB ingeschakeld.",
|
"greetdm_on": "Begroetingsnotificaties per PB ingeschakeld.",
|
||||||
"greetmsg_cur": "Huidig welkomstbericht",
|
"greetmsg_cur": "Huidig welkomstbericht: {0}",
|
||||||
"greetmsg_enable": "Typ {0} om welkomstberichten in te schakelen.",
|
"greetmsg_enable": "Typ {0} om welkomstberichten in te schakelen.",
|
||||||
"greetmsg_new": "Nieuw welkomstbericht ingesteld.",
|
"greetmsg_new": "Nieuw welkomstbericht ingesteld.",
|
||||||
"greet_off": "Begroetingsnotificaties uitgezet.",
|
"greet_off": "Begroetingsnotificaties uitgezet.",
|
||||||
@@ -131,7 +131,7 @@
|
|||||||
"animal_race_starting": "Start over {0} seconden, of als de race vol is.",
|
"animal_race_starting": "Start over {0} seconden, of als de race vol is.",
|
||||||
"animal_race_won": "{0} als {1} heeft de race gewonnen!",
|
"animal_race_won": "{0} als {1} heeft de race gewonnen!",
|
||||||
"dice_invalid_number": "Ongeldig aantal. Je kunt {0}-{1} dobbelstenen gelijktijdig werpen.",
|
"dice_invalid_number": "Ongeldig aantal. Je kunt {0}-{1} dobbelstenen gelijktijdig werpen.",
|
||||||
"dice_rolled": "{0} gegooid.",
|
"dice_rolled": "{0} gegooid",
|
||||||
"dice_rolled_num": "Dobbelsteen gegooid: {0}",
|
"dice_rolled_num": "Dobbelsteen gegooid: {0}",
|
||||||
"changes_of_heart": "Verandering van hart",
|
"changes_of_heart": "Verandering van hart",
|
||||||
"claimed_by": "Geclaimd door",
|
"claimed_by": "Geclaimd door",
|
||||||
@@ -946,34 +946,106 @@
|
|||||||
"reminder_server_list": "Lijst van serverherinneringen",
|
"reminder_server_list": "Lijst van serverherinneringen",
|
||||||
"imageonly_enable": "Dit kanaal is nu alleen voor afbeeldingen.",
|
"imageonly_enable": "Dit kanaal is nu alleen voor afbeeldingen.",
|
||||||
"imageonly_disable": "Dit kanaal is niet langer alleen maar voor afbeeldingen.",
|
"imageonly_disable": "Dit kanaal is niet langer alleen maar voor afbeeldingen.",
|
||||||
"transaction": "",
|
"transaction": "Valuta Transactie",
|
||||||
"finished_track": "",
|
"finished_track": "Nummer Afgelopen",
|
||||||
"playing_track": "",
|
"playing_track": "Nummer #{0} wordt gespeeld",
|
||||||
"queued_track": "",
|
"queued_track": "Nummer in wachtrij",
|
||||||
"removed_track": "",
|
"removed_track": "Nummer verwijderd",
|
||||||
"autoplaying": "",
|
"autoplaying": "Voegt automatisch nummers toe.",
|
||||||
"music_autoplay_on": "",
|
"music_autoplay_on": "Muziek automatisch spelen ingeschakeld. Ik zal gerelateerde nummers in de wachtrij zetten als een nummer afloopt.",
|
||||||
"music_autoplay_off": "",
|
"music_autoplay_off": "Automatisch spelen uitgezet.",
|
||||||
"track_moved": "",
|
"track_moved": "Nummer verplaatst",
|
||||||
"atl_not_enabled": "",
|
"atl_not_enabled": "Automatisch vertalen is voor dit kanaal niet ingeschakeld of je hebt een ongeldige taal ingevoerd.",
|
||||||
"channels": "",
|
"channels": "Kanalen",
|
||||||
"track_not_found": "",
|
"track_not_found": "Nummer niet gevonden.",
|
||||||
"removed_track_error": "",
|
"removed_track_error": "Er bestaat geen nummer op die positie.",
|
||||||
"market_cap_dominance": "",
|
"market_cap_dominance": "Dominantie",
|
||||||
"circulating_supply": "",
|
"circulating_supply": "Circulerende toevoer",
|
||||||
"module_description_expressions": "",
|
"module_description_expressions": "Stel een aangepaste reactie van de bot in op bepaalde woorden of zinnen.",
|
||||||
"deleted_x_servers": "",
|
"deleted_x_servers": "{0} servers verwijderd.",
|
||||||
"curtr_gift": "",
|
"curtr_gift": "Geschenk van {0} [{1}]",
|
||||||
"curtr_award": "",
|
"curtr_award": "Toegekend door de eigenaar van de bot {0} [{1}]",
|
||||||
"curtr_take": "",
|
"curtr_take": "Genomen door bot eigenaar {0} [{1}]",
|
||||||
"expr_deleted": "",
|
"expr_deleted": "Uitdrukking verwijderd",
|
||||||
"expr_insuff_perms": "",
|
"expr_insuff_perms": "Onvoldoende rechten. Vereist Bot eigendom voor globale expressies, en Beheerder voor server expressies.",
|
||||||
"expressions": "",
|
"expressions": "Uitdrukkingen",
|
||||||
"expr_new": "",
|
"expr_new": "Nieuwe Uitdrukking",
|
||||||
"expr_no_found": "",
|
"expr_no_found": "Geen uitdrukking gevonden",
|
||||||
"expr_no_found_id": "",
|
"expr_no_found_id": "Geen expressie gevonden met die id.",
|
||||||
"exprs_cleared": "",
|
"exprs_cleared": "Alle {0} expressies op deze server zijn verwijderd.",
|
||||||
"expr_reset": "",
|
"expr_reset": "Uitdrukking met id {0} zal niet langer reacties toevoegen.",
|
||||||
"expr_set": "",
|
"expr_set": "Expressie met id {0} zal de volgende reacties toevoegen aan het antwoordbericht: {1}",
|
||||||
"expr_edited": ""
|
"expr_edited": "Uitdrukking bewerkt",
|
||||||
|
"stream_online_delete_enabled": "Online streammeldingen worden nu verwijderd wanneer de stream offline gaat.",
|
||||||
|
"stream_online_delete_disabled": "Meldingen van online streams worden niet langer verwijderd wanneer de stream offline gaat.",
|
||||||
|
"club_create_error_name": "Het aanmaken van de club is mislukt. Er bestaat al een club met die naam.",
|
||||||
|
"club_desc_update": "Club Beschrijving Bijgewerkt",
|
||||||
|
"bank_accounts": "Bankrekeningen",
|
||||||
|
"module_description_medusa": "**Alleen robot eigenaar.** Laadt, ontlaadt en behandelt dynamische modules. Lees meer [hier](https://nadekobot.readthedocs.io/en/latest/medusa/creating-a-medusa/)",
|
||||||
|
"list_of_medusae": "Lijst van Medusae",
|
||||||
|
"list_of_unloaded": "Lijst van beschikbare Medusae",
|
||||||
|
"medusa_name_not_found": "Medusa met die naam bestaat niet of is niet geladen.",
|
||||||
|
"medusa_info": "Medusa Informatie",
|
||||||
|
"sneks_count": "Sneks (Slangen) ({0})",
|
||||||
|
"commands_count": "Commando's ({0})",
|
||||||
|
"no_medusa_loaded": "Er zijn geen geladen medusae.",
|
||||||
|
"no_medusa_available": "Geen medusa beschikbaar.",
|
||||||
|
"loaded_medusae": "Geladen medusae.",
|
||||||
|
"medusa_not_loaded": "Medusa met die naam is niet geladen.",
|
||||||
|
"medusa_possibly_cant_unload": "Medusa is waarschijnlijk niet volledig ontladen. Start de bot opnieuw op als er problemen optreden.",
|
||||||
|
"medusa_loaded": "Medusa {0} is geladen.",
|
||||||
|
"medusa_unloaded": "Medusa {0} is uitgeladen.",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "Medusa {0} is al geladen",
|
||||||
|
"medusa_invalid_not_found": "Medusa is met deze naam niet gevonden, of het bestand is ongeldig,",
|
||||||
|
"bank_balance": "Je hebt {0} op je bankrekening.",
|
||||||
|
"bank_deposited": "Je hebt {0} op je bankrekening gestort.",
|
||||||
|
"bank_withdrew": "Je hebt {0} opgenomen van je bankrekening.",
|
||||||
|
"bank_withdraw_insuff": "Je hebt onvoldoende {0} op je bankrekening.",
|
||||||
|
"cmd_group_commands": "\"{0}\" commandogroep",
|
||||||
|
"limit_reached": "Feature limiet van {0} bereikt.",
|
||||||
|
"feature_limit_reached_you": "Jij hebt de limiet van {0} voor de {1} functie bereikt. U kunt in staat zijn om deze limiet te verhogen door het upgraden van uw patron tier.",
|
||||||
|
"feature_limit_reached_owner": "De server eigenaar heeft de limiet van {0} voor de {1} functie bereikt. De server eigenaar kan deze limiet wellicht opwaarderen door de patron tier te upgraden.",
|
||||||
|
"feature_limit_reached_either": "De limiet van {0} voor de {1} functie is bereikt. Jij of de server eigenaar kan deze limiet opwaarderen door de patron tier op te waarderen.",
|
||||||
|
"tier": "Niveau",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "Vervalt",
|
||||||
|
"commands": "Commando's",
|
||||||
|
"groups": "Groepen",
|
||||||
|
"modules": "Modulen",
|
||||||
|
"no_quota_found": "Geen quota gevonden",
|
||||||
|
"patron_info": "Patron Informatie",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "Patron systeem is uitgeschakeld.",
|
||||||
|
"results_in": "{0} resulteert in {1}s",
|
||||||
|
"patron_msg_sent": "Klaar met het verzenden van berichten naar patrons op niveau {1} en hoger. {1} succesvol verzonden en {2} mislukt.",
|
||||||
|
"cards": "Kaarten",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "Rol",
|
||||||
|
"rolls": "Rollen",
|
||||||
|
"available_tests": "Beschikbare testen",
|
||||||
|
"test_results_for": "Testresultaten voor {0}",
|
||||||
|
"multiplier": "Vermenigvuldiger",
|
||||||
|
"trivia_ended": "Trivia spel beëindigd",
|
||||||
|
"card": "Kaart",
|
||||||
|
"guess": "Gok",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "Ik zal u herinneren aan uw tijdige beloning {0}",
|
||||||
|
"xp_shop_disabled": "Xp shop is uitgeschakeld door de eigenaar, of er zijn geen items te koop.",
|
||||||
|
"timely_time": "Het is tijd voor uw tijdige beloning.",
|
||||||
|
"buy": "Koop",
|
||||||
|
"use": "Gebruiken",
|
||||||
|
"in_use": "In gebruik",
|
||||||
|
"xp_shop_item_cant_use": "Je kunt dit item niet gebruiken, omdat het niet bestaat of je het niet bezit.",
|
||||||
|
"linkonly_enable": "Dit kanaal is nu link-only.",
|
||||||
|
"linkonly_disable": "Dit kanaal is niet meer link-only.",
|
||||||
|
"xp_shop_buy_required_tier": "Voor het kopen van artikelen uit deze winkel is Patron Tier {0} of hoger vereist.",
|
||||||
|
"available_commands": "Beschikbare commando's",
|
||||||
|
"xpadd_users": "{0} server XP toegevoegd aan {1} gebruikers.",
|
||||||
|
"xpshop_buy_success": "Succesvol gekocht `{0}/{1}`",
|
||||||
|
"patron_insuff_tier": "Je Patron Tier is onvoldoende om deze actie uit te voeren.",
|
||||||
|
"xpshop_already_owned": "Je bezit dit voorwerp al.",
|
||||||
|
"xpshop_item_not_found": "Een item met die sleutel bestaat niet.",
|
||||||
|
"xpshop_website": "Je kan de lijst van alle Xp Shop artikelen hier zien: <https://xpshop.nadeko.bot>"
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "Todas {0} expressões foram removidas deste servidor.",
|
"exprs_cleared": "Todas {0} expressões foram removidas deste servidor.",
|
||||||
"expr_reset": "A expressão com o id {0} não adicionará mas reações.",
|
"expr_reset": "A expressão com o id {0} não adicionará mas reações.",
|
||||||
"expr_set": "A expressão com a id {0} irá adicionar as seguintes reações na resposta: {1} ",
|
"expr_set": "A expressão com a id {0} irá adicionar as seguintes reações na resposta: {1} ",
|
||||||
"expr_edited": "Expressão editada"
|
"expr_edited": "Expressão editada",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "Всі {0} вирази на цьому сервері були вилучені.",
|
"exprs_cleared": "Всі {0} вирази на цьому сервері були вилучені.",
|
||||||
"expr_reset": "Вираз з id {0} більше не получає реакцій.",
|
"expr_reset": "Вираз з id {0} більше не получає реакцій.",
|
||||||
"expr_set": "Вираз з цим id {0} буде получати вказані реакції до відповіді: {1}",
|
"expr_set": "Вираз з цим id {0} буде получати вказані реакції до відповіді: {1}",
|
||||||
"expr_edited": "Вираз змінено"
|
"expr_edited": "Вираз змінено",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "",
|
"exprs_cleared": "",
|
||||||
"expr_reset": "",
|
"expr_reset": "",
|
||||||
"expr_set": "",
|
"expr_set": "",
|
||||||
"expr_edited": ""
|
"expr_edited": "",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -975,5 +975,77 @@
|
|||||||
"exprs_cleared": "此伺服器上的所有 {0} 表達式均已刪除。",
|
"exprs_cleared": "此伺服器上的所有 {0} 表達式均已刪除。",
|
||||||
"expr_reset": "ID 為 {0} 的表達式將不再添加反應。",
|
"expr_reset": "ID 為 {0} 的表達式將不再添加反應。",
|
||||||
"expr_set": "ID 為 {0} 的表達式將在響應消息中添加以下反應:{1}",
|
"expr_set": "ID 為 {0} 的表達式將在響應消息中添加以下反應:{1}",
|
||||||
"expr_edited": "表達式編輯"
|
"expr_edited": "表達式編輯",
|
||||||
|
"stream_online_delete_enabled": "",
|
||||||
|
"stream_online_delete_disabled": "",
|
||||||
|
"club_create_error_name": "",
|
||||||
|
"club_desc_update": "",
|
||||||
|
"bank_accounts": "",
|
||||||
|
"module_description_medusa": "",
|
||||||
|
"list_of_medusae": "",
|
||||||
|
"list_of_unloaded": "",
|
||||||
|
"medusa_name_not_found": "",
|
||||||
|
"medusa_info": "",
|
||||||
|
"sneks_count": "",
|
||||||
|
"commands_count": "",
|
||||||
|
"no_medusa_loaded": "",
|
||||||
|
"no_medusa_available": "",
|
||||||
|
"loaded_medusae": "",
|
||||||
|
"medusa_not_loaded": "",
|
||||||
|
"medusa_possibly_cant_unload": "",
|
||||||
|
"medusa_loaded": "",
|
||||||
|
"medusa_unloaded": "",
|
||||||
|
"medusa_empty": "",
|
||||||
|
"medusa_already_loaded": "",
|
||||||
|
"medusa_invalid_not_found": "",
|
||||||
|
"bank_balance": "",
|
||||||
|
"bank_deposited": "",
|
||||||
|
"bank_withdrew": "",
|
||||||
|
"bank_withdraw_insuff": "",
|
||||||
|
"cmd_group_commands": "",
|
||||||
|
"limit_reached": "",
|
||||||
|
"feature_limit_reached_you": "",
|
||||||
|
"feature_limit_reached_owner": "",
|
||||||
|
"feature_limit_reached_either": "",
|
||||||
|
"tier": "",
|
||||||
|
"pledge": "",
|
||||||
|
"expires": "",
|
||||||
|
"commands": "",
|
||||||
|
"groups": "",
|
||||||
|
"modules": "",
|
||||||
|
"no_quota_found": "",
|
||||||
|
"patron_info": "",
|
||||||
|
"quotas": "",
|
||||||
|
"patron_not_enabled": "",
|
||||||
|
"results_in": "",
|
||||||
|
"patron_msg_sent": "",
|
||||||
|
"cards": "",
|
||||||
|
"hand_value": "",
|
||||||
|
"roll2": "",
|
||||||
|
"rolls": "",
|
||||||
|
"available_tests": "",
|
||||||
|
"test_results_for": "",
|
||||||
|
"multiplier": "",
|
||||||
|
"trivia_ended": "",
|
||||||
|
"card": "",
|
||||||
|
"guess": "",
|
||||||
|
"repeater_skip_next": "",
|
||||||
|
"repeater_dont_skip_next": "",
|
||||||
|
"remind_timely": "",
|
||||||
|
"xp_shop_disabled": "",
|
||||||
|
"timely_time": "",
|
||||||
|
"buy": "",
|
||||||
|
"use": "",
|
||||||
|
"in_use": "",
|
||||||
|
"xp_shop_item_cant_use": "",
|
||||||
|
"linkonly_enable": "",
|
||||||
|
"linkonly_disable": "",
|
||||||
|
"xp_shop_buy_required_tier": "",
|
||||||
|
"available_commands": "",
|
||||||
|
"xpadd_users": "",
|
||||||
|
"xpshop_buy_success": "",
|
||||||
|
"patron_insuff_tier": "",
|
||||||
|
"xpshop_already_owned": "",
|
||||||
|
"xpshop_item_not_found": "",
|
||||||
|
"xpshop_website": ""
|
||||||
}
|
}
|
@@ -1,5 +1,5 @@
|
|||||||
# DO NOT CHANGE
|
# DO NOT CHANGE
|
||||||
version: 3
|
version: 6
|
||||||
# How much XP will the users receive per message
|
# How much XP will the users receive per message
|
||||||
xpPerMessage: 3
|
xpPerMessage: 3
|
||||||
# How often can the users receive XP in minutes
|
# How often can the users receive XP in minutes
|
||||||
@@ -10,15 +10,20 @@ xpFromImage: 0
|
|||||||
voiceXpPerMinute: 0
|
voiceXpPerMinute: 0
|
||||||
# The maximum amount of minutes the bot will keep track of a user in a voice channel
|
# The maximum amount of minutes the bot will keep track of a user in a voice channel
|
||||||
voiceMaxMinutes: 720
|
voiceMaxMinutes: 720
|
||||||
|
# The amount of currency users will receive for each point of global xp that they earn
|
||||||
|
currencyPerXp: 0
|
||||||
# Xp Shop config
|
# Xp Shop config
|
||||||
shop:
|
shop:
|
||||||
# Whether the xp shop is enabled
|
# Whether the xp shop is enabled
|
||||||
# True -> Users can access the xp shop using .xpshop command
|
# True -> Users can access the xp shop using .xpshop command
|
||||||
# False -> Users can't access the xp shop
|
# False -> Users can't access the xp shop
|
||||||
isEnabled: false
|
isEnabled: false
|
||||||
# Which patron tier do users need in order to use the .xpshop command
|
# Which patron tier do users need in order to use the .xpshop bgs command
|
||||||
# Leave at 'None' if patron system is disabled or you don't want any restrictions
|
# Leave at 'None' if patron system is disabled or you don't want any restrictions
|
||||||
tierRequirement: None
|
bgsTierRequirement: None
|
||||||
|
# Which patron tier do users need in order to use the .xpshop frames command
|
||||||
|
# Leave at 'None' if patron system is disabled or you don't want any restrictions
|
||||||
|
framesTierRequirement: None
|
||||||
# Frames available for sale. Keys are unique IDs.
|
# Frames available for sale. Keys are unique IDs.
|
||||||
# Do not change keys as they are not publicly visible. Only change properties (name, price, id)
|
# Do not change keys as they are not publicly visible. Only change properties (name, price, id)
|
||||||
# Removing a key which previously existed means that all previous purchases will also be unusable.
|
# Removing a key which previously existed means that all previous purchases will also be unusable.
|
||||||
@@ -31,6 +36,8 @@ shop:
|
|||||||
price: 0
|
price: 0
|
||||||
# Direct url to the .png image which will be applied to the user's XP card
|
# Direct url to the .png image which will be applied to the user's XP card
|
||||||
url: ''
|
url: ''
|
||||||
|
# Optional preview url which will show instead of the real URL in the shop
|
||||||
|
preview:
|
||||||
# Optional description of the item
|
# Optional description of the item
|
||||||
desc:
|
desc:
|
||||||
# Backgrounds available for sale. Keys are unique IDs.
|
# Backgrounds available for sale. Keys are unique IDs.
|
||||||
@@ -45,5 +52,7 @@ shop:
|
|||||||
price: 0
|
price: 0
|
||||||
# Direct url to the .png image which will be applied to the user's XP card
|
# Direct url to the .png image which will be applied to the user's XP card
|
||||||
url: ''
|
url: ''
|
||||||
|
# Optional preview url which will show instead of the real URL in the shop
|
||||||
|
preview:
|
||||||
# Optional description of the item
|
# Optional description of the item
|
||||||
desc:
|
desc:
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user