committed by
GitHub
19 changed files with 1185 additions and 26 deletions
@ -0,0 +1,325 @@ |
|||||
|
""" |
||||
|
The MIT License (MIT) |
||||
|
|
||||
|
Copyright (c) 2015-present Rapptz |
||||
|
|
||||
|
Permission is hereby granted, free of charge, to any person obtaining a |
||||
|
copy of this software and associated documentation files (the "Software"), |
||||
|
to deal in the Software without restriction, including without limitation |
||||
|
the rights to use, copy, modify, merge, publish, distribute, sublicense, |
||||
|
and/or sell copies of the Software, and to permit persons to whom the |
||||
|
Software is furnished to do so, subject to the following conditions: |
||||
|
|
||||
|
The above copyright notice and this permission notice shall be included in |
||||
|
all copies or substantial portions of the Software. |
||||
|
|
||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
||||
|
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
||||
|
DEALINGS IN THE SOFTWARE. |
||||
|
""" |
||||
|
|
||||
|
from __future__ import annotations |
||||
|
|
||||
|
from typing import TYPE_CHECKING, Optional |
||||
|
|
||||
|
from . import utils |
||||
|
from .mixins import Hashable |
||||
|
from .partial_emoji import PartialEmoji, _EmojiTag |
||||
|
from .user import User |
||||
|
from .utils import MISSING |
||||
|
from .asset import Asset, AssetMixin |
||||
|
|
||||
|
if TYPE_CHECKING: |
||||
|
import datetime |
||||
|
from typing import Dict, Any |
||||
|
|
||||
|
from .types.soundboard import ( |
||||
|
BaseSoundboardSound as BaseSoundboardSoundPayload, |
||||
|
SoundboardDefaultSound as SoundboardDefaultSoundPayload, |
||||
|
SoundboardSound as SoundboardSoundPayload, |
||||
|
) |
||||
|
from .state import ConnectionState |
||||
|
from .guild import Guild |
||||
|
from .message import EmojiInputType |
||||
|
|
||||
|
__all__ = ('BaseSoundboardSound', 'SoundboardDefaultSound', 'SoundboardSound') |
||||
|
|
||||
|
|
||||
|
class BaseSoundboardSound(Hashable, AssetMixin): |
||||
|
"""Represents a generic Discord soundboard sound. |
||||
|
|
||||
|
.. versionadded:: 2.5 |
||||
|
|
||||
|
.. container:: operations |
||||
|
|
||||
|
.. describe:: x == y |
||||
|
|
||||
|
Checks if two sounds are equal. |
||||
|
|
||||
|
.. describe:: x != y |
||||
|
|
||||
|
Checks if two sounds are not equal. |
||||
|
|
||||
|
.. describe:: hash(x) |
||||
|
|
||||
|
Returns the sound's hash. |
||||
|
|
||||
|
Attributes |
||||
|
------------ |
||||
|
id: :class:`int` |
||||
|
The ID of the sound. |
||||
|
volume: :class:`float` |
||||
|
The volume of the sound as floating point percentage (e.g. ``1.0`` for 100%). |
||||
|
""" |
||||
|
|
||||
|
__slots__ = ('_state', 'id', 'volume') |
||||
|
|
||||
|
def __init__(self, *, state: ConnectionState, data: BaseSoundboardSoundPayload): |
||||
|
self._state: ConnectionState = state |
||||
|
self.id: int = int(data['sound_id']) |
||||
|
self._update(data) |
||||
|
|
||||
|
def __eq__(self, other: object) -> bool: |
||||
|
if isinstance(other, self.__class__): |
||||
|
return self.id == other.id |
||||
|
return NotImplemented |
||||
|
|
||||
|
def __ne__(self, other: object) -> bool: |
||||
|
return not self.__eq__(other) |
||||
|
|
||||
|
def _update(self, data: BaseSoundboardSoundPayload): |
||||
|
self.volume: float = data['volume'] |
||||
|
|
||||
|
@property |
||||
|
def url(self) -> str: |
||||
|
""":class:`str`: Returns the URL of the sound.""" |
||||
|
return f'{Asset.BASE}/soundboard-sounds/{self.id}' |
||||
|
|
||||
|
|
||||
|
class SoundboardDefaultSound(BaseSoundboardSound): |
||||
|
"""Represents a Discord soundboard default sound. |
||||
|
|
||||
|
.. versionadded:: 2.5 |
||||
|
|
||||
|
.. container:: operations |
||||
|
|
||||
|
.. describe:: x == y |
||||
|
|
||||
|
Checks if two sounds are equal. |
||||
|
|
||||
|
.. describe:: x != y |
||||
|
|
||||
|
Checks if two sounds are not equal. |
||||
|
|
||||
|
.. describe:: hash(x) |
||||
|
|
||||
|
Returns the sound's hash. |
||||
|
|
||||
|
Attributes |
||||
|
------------ |
||||
|
id: :class:`int` |
||||
|
The ID of the sound. |
||||
|
volume: :class:`float` |
||||
|
The volume of the sound as floating point percentage (e.g. ``1.0`` for 100%). |
||||
|
name: :class:`str` |
||||
|
The name of the sound. |
||||
|
emoji: :class:`PartialEmoji` |
||||
|
The emoji of the sound. |
||||
|
""" |
||||
|
|
||||
|
__slots__ = ('name', 'emoji') |
||||
|
|
||||
|
def __init__(self, *, state: ConnectionState, data: SoundboardDefaultSoundPayload): |
||||
|
self.name: str = data['name'] |
||||
|
self.emoji: PartialEmoji = PartialEmoji(name=data['emoji_name']) |
||||
|
super().__init__(state=state, data=data) |
||||
|
|
||||
|
def __repr__(self) -> str: |
||||
|
attrs = [ |
||||
|
('id', self.id), |
||||
|
('name', self.name), |
||||
|
('volume', self.volume), |
||||
|
('emoji', self.emoji), |
||||
|
] |
||||
|
inner = ' '.join('%s=%r' % t for t in attrs) |
||||
|
return f"<{self.__class__.__name__} {inner}>" |
||||
|
|
||||
|
|
||||
|
class SoundboardSound(BaseSoundboardSound): |
||||
|
"""Represents a Discord soundboard sound. |
||||
|
|
||||
|
.. versionadded:: 2.5 |
||||
|
|
||||
|
.. container:: operations |
||||
|
|
||||
|
.. describe:: x == y |
||||
|
|
||||
|
Checks if two sounds are equal. |
||||
|
|
||||
|
.. describe:: x != y |
||||
|
|
||||
|
Checks if two sounds are not equal. |
||||
|
|
||||
|
.. describe:: hash(x) |
||||
|
|
||||
|
Returns the sound's hash. |
||||
|
|
||||
|
Attributes |
||||
|
------------ |
||||
|
id: :class:`int` |
||||
|
The ID of the sound. |
||||
|
volume: :class:`float` |
||||
|
The volume of the sound as floating point percentage (e.g. ``1.0`` for 100%). |
||||
|
name: :class:`str` |
||||
|
The name of the sound. |
||||
|
emoji: Optional[:class:`PartialEmoji`] |
||||
|
The emoji of the sound. ``None`` if no emoji is set. |
||||
|
guild: :class:`Guild` |
||||
|
The guild in which the sound is uploaded. |
||||
|
available: :class:`bool` |
||||
|
Whether this sound is available for use. |
||||
|
""" |
||||
|
|
||||
|
__slots__ = ('_state', 'name', 'emoji', '_user', 'available', '_user_id', 'guild') |
||||
|
|
||||
|
def __init__(self, *, guild: Guild, state: ConnectionState, data: SoundboardSoundPayload): |
||||
|
super().__init__(state=state, data=data) |
||||
|
self.guild = guild |
||||
|
self._user_id = utils._get_as_snowflake(data, 'user_id') |
||||
|
self._user = data.get('user') |
||||
|
|
||||
|
self._update(data) |
||||
|
|
||||
|
def __repr__(self) -> str: |
||||
|
attrs = [ |
||||
|
('id', self.id), |
||||
|
('name', self.name), |
||||
|
('volume', self.volume), |
||||
|
('emoji', self.emoji), |
||||
|
('user', self.user), |
||||
|
] |
||||
|
inner = ' '.join('%s=%r' % t for t in attrs) |
||||
|
return f"<{self.__class__.__name__} {inner}>" |
||||
|
|
||||
|
def _update(self, data: SoundboardSoundPayload): |
||||
|
super()._update(data) |
||||
|
|
||||
|
self.name: str = data['name'] |
||||
|
self.emoji: Optional[PartialEmoji] = None |
||||
|
|
||||
|
emoji_id = utils._get_as_snowflake(data, 'emoji_id') |
||||
|
emoji_name = data['emoji_name'] |
||||
|
if emoji_id is not None or emoji_name is not None: |
||||
|
self.emoji = PartialEmoji(id=emoji_id, name=emoji_name) # type: ignore # emoji_name cannot be None here |
||||
|
|
||||
|
self.available: bool = data['available'] |
||||
|
|
||||
|
@property |
||||
|
def created_at(self) -> datetime.datetime: |
||||
|
""":class:`datetime.datetime`: Returns the snowflake's creation time in UTC.""" |
||||
|
return utils.snowflake_time(self.id) |
||||
|
|
||||
|
@property |
||||
|
def user(self) -> Optional[User]: |
||||
|
"""Optional[:class:`User`]: The user who uploaded the sound.""" |
||||
|
if self._user is None: |
||||
|
if self._user_id is None: |
||||
|
return None |
||||
|
return self._state.get_user(self._user_id) |
||||
|
return User(state=self._state, data=self._user) |
||||
|
|
||||
|
async def edit( |
||||
|
self, |
||||
|
*, |
||||
|
name: str = MISSING, |
||||
|
volume: Optional[float] = MISSING, |
||||
|
emoji: Optional[EmojiInputType] = MISSING, |
||||
|
reason: Optional[str] = None, |
||||
|
): |
||||
|
"""|coro| |
||||
|
|
||||
|
Edits the soundboard sound. |
||||
|
|
||||
|
You must have :attr:`~Permissions.manage_expressions` to edit the sound. |
||||
|
If the sound was created by the client, you must have either :attr:`~Permissions.manage_expressions` |
||||
|
or :attr:`~Permissions.create_expressions`. |
||||
|
|
||||
|
Parameters |
||||
|
---------- |
||||
|
name: :class:`str` |
||||
|
The new name of the sound. Must be between 2 and 32 characters. |
||||
|
volume: Optional[:class:`float`] |
||||
|
The new volume of the sound. Must be between 0 and 1. |
||||
|
emoji: Optional[Union[:class:`Emoji`, :class:`PartialEmoji`, :class:`str`]] |
||||
|
The new emoji of the sound. |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for editing this sound. Shows up on the audit log. |
||||
|
|
||||
|
Raises |
||||
|
------- |
||||
|
Forbidden |
||||
|
You do not have permissions to edit the soundboard sound. |
||||
|
HTTPException |
||||
|
Editing the soundboard sound failed. |
||||
|
|
||||
|
Returns |
||||
|
------- |
||||
|
:class:`SoundboardSound` |
||||
|
The newly updated soundboard sound. |
||||
|
""" |
||||
|
|
||||
|
payload: Dict[str, Any] = {} |
||||
|
|
||||
|
if name is not MISSING: |
||||
|
payload['name'] = name |
||||
|
|
||||
|
if volume is not MISSING: |
||||
|
payload['volume'] = volume |
||||
|
|
||||
|
if emoji is not MISSING: |
||||
|
if emoji is None: |
||||
|
payload['emoji_id'] = None |
||||
|
payload['emoji_name'] = None |
||||
|
else: |
||||
|
if isinstance(emoji, _EmojiTag): |
||||
|
partial_emoji = emoji._to_partial() |
||||
|
elif isinstance(emoji, str): |
||||
|
partial_emoji = PartialEmoji.from_str(emoji) |
||||
|
else: |
||||
|
partial_emoji = None |
||||
|
|
||||
|
if partial_emoji is not None: |
||||
|
if partial_emoji.id is None: |
||||
|
payload['emoji_name'] = partial_emoji.name |
||||
|
else: |
||||
|
payload['emoji_id'] = partial_emoji.id |
||||
|
|
||||
|
data = await self._state.http.edit_soundboard_sound(self.guild.id, self.id, reason=reason, **payload) |
||||
|
return SoundboardSound(guild=self.guild, state=self._state, data=data) |
||||
|
|
||||
|
async def delete(self, *, reason: Optional[str] = None) -> None: |
||||
|
"""|coro| |
||||
|
|
||||
|
Deletes the soundboard sound. |
||||
|
|
||||
|
You must have :attr:`~Permissions.manage_expressions` to delete the sound. |
||||
|
If the sound was created by the client, you must have either :attr:`~Permissions.manage_expressions` |
||||
|
or :attr:`~Permissions.create_expressions`. |
||||
|
|
||||
|
Parameters |
||||
|
----------- |
||||
|
reason: Optional[:class:`str`] |
||||
|
The reason for deleting this sound. Shows up on the audit log. |
||||
|
|
||||
|
Raises |
||||
|
------- |
||||
|
Forbidden |
||||
|
You do not have permissions to delete the soundboard sound. |
||||
|
HTTPException |
||||
|
Deleting the soundboard sound failed. |
||||
|
""" |
||||
|
await self._state.http.delete_soundboard_sound(self.guild.id, self.id, reason=reason) |
@ -0,0 +1,49 @@ |
|||||
|
""" |
||||
|
The MIT License (MIT) |
||||
|
|
||||
|
Copyright (c) 2015-present Rapptz |
||||
|
|
||||
|
Permission is hereby granted, free of charge, to any person obtaining a |
||||
|
copy of this software and associated documentation files (the "Software"), |
||||
|
to deal in the Software without restriction, including without limitation |
||||
|
the rights to use, copy, modify, merge, publish, distribute, sublicense, |
||||
|
and/or sell copies of the Software, and to permit persons to whom the |
||||
|
Software is furnished to do so, subject to the following conditions: |
||||
|
|
||||
|
The above copyright notice and this permission notice shall be included in |
||||
|
all copies or substantial portions of the Software. |
||||
|
|
||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS |
||||
|
OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
||||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
||||
|
DEALINGS IN THE SOFTWARE. |
||||
|
""" |
||||
|
|
||||
|
from typing import TypedDict, Optional, Union |
||||
|
from typing_extensions import NotRequired |
||||
|
|
||||
|
from .snowflake import Snowflake |
||||
|
from .user import User |
||||
|
|
||||
|
|
||||
|
class BaseSoundboardSound(TypedDict): |
||||
|
sound_id: Union[Snowflake, str] # basic string number when it's a default sound |
||||
|
volume: float |
||||
|
|
||||
|
|
||||
|
class SoundboardSound(BaseSoundboardSound): |
||||
|
name: str |
||||
|
emoji_name: Optional[str] |
||||
|
emoji_id: Optional[Snowflake] |
||||
|
user_id: NotRequired[Snowflake] |
||||
|
available: bool |
||||
|
guild_id: NotRequired[Snowflake] |
||||
|
user: NotRequired[User] |
||||
|
|
||||
|
|
||||
|
class SoundboardDefaultSound(BaseSoundboardSound): |
||||
|
name: str |
||||
|
emoji_name: str |
Loading…
Reference in new issue