From 26b9f179707c17229a50292fd05993ecb5903d52 Mon Sep 17 00:00:00 2001 From: csd4ni3l Date: Sat, 24 May 2025 10:33:23 +0200 Subject: [PATCH] Initial commit --- .gitignore | 182 +++++++ .python-version | 1 + CREDITS | 17 + LICENSE | 674 +++++++++++++++++++++++++ README.md | 19 + assets/fonts/ProtestStrike-Regular.ttf | Bin 0 -> 83412 bytes assets/graphics/button.png | Bin 0 -> 280 bytes assets/graphics/button_hovered.png | Bin 0 -> 291 bytes assets/graphics/download.png | Bin 0 -> 741 bytes assets/graphics/files.png | Bin 0 -> 541 bytes assets/graphics/loop.png | Bin 0 -> 596 bytes assets/graphics/music.png | Bin 0 -> 9454 bytes assets/graphics/no_loop.png | Bin 0 -> 547 bytes assets/graphics/no_shuffle.png | Bin 0 -> 591 bytes assets/graphics/pause.png | Bin 0 -> 244 bytes assets/graphics/playlist.png | Bin 0 -> 468 bytes assets/graphics/plus.png | Bin 0 -> 400 bytes assets/graphics/reload.png | Bin 0 -> 1083 bytes assets/graphics/resume.png | Bin 0 -> 348 bytes assets/graphics/settings.png | Bin 0 -> 1114 bytes assets/graphics/shuffle.png | Bin 0 -> 630 bytes assets/graphics/stop.png | Bin 0 -> 256 bytes menus/add_music.py | 68 +++ menus/downloader.py | 137 +++++ menus/main.py | 464 +++++++++++++++++ menus/new_tab.py | 77 +++ menus/settings.py | 298 +++++++++++ pyproject.toml | 14 + requirements.txt | 6 + run.py | 98 ++++ utils/constants.py | 72 +++ utils/preload.py | 21 + utils/utils.py | 168 ++++++ uv.lock | 326 ++++++++++++ 34 files changed, 2642 insertions(+) create mode 100644 .gitignore create mode 100644 .python-version create mode 100644 CREDITS create mode 100644 LICENSE create mode 100644 README.md create mode 100644 assets/fonts/ProtestStrike-Regular.ttf create mode 100644 assets/graphics/button.png create mode 100644 assets/graphics/button_hovered.png create mode 100644 assets/graphics/download.png create mode 100644 assets/graphics/files.png create mode 100644 assets/graphics/loop.png create mode 100644 assets/graphics/music.png create mode 100644 assets/graphics/no_loop.png create mode 100644 assets/graphics/no_shuffle.png create mode 100644 assets/graphics/pause.png create mode 100644 assets/graphics/playlist.png create mode 100644 assets/graphics/plus.png create mode 100644 assets/graphics/reload.png create mode 100644 assets/graphics/resume.png create mode 100644 assets/graphics/settings.png create mode 100644 assets/graphics/shuffle.png create mode 100644 assets/graphics/stop.png create mode 100644 menus/add_music.py create mode 100644 menus/downloader.py create mode 100644 menus/main.py create mode 100644 menus/new_tab.py create mode 100644 menus/settings.py create mode 100644 pyproject.toml create mode 100644 requirements.txt create mode 100644 run.py create mode 100644 utils/constants.py create mode 100644 utils/preload.py create mode 100644 utils/utils.py create mode 100644 uv.lock diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..81c71f9 --- /dev/null +++ b/.gitignore @@ -0,0 +1,182 @@ +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# UV +# Similar to Pipfile.lock, it is generally recommended to include uv.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +#uv.lock + +# poetry +# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control +#poetry.lock + +# pdm +# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. +#pdm.lock +# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it +# in version control. +# https://pdm.fming.dev/latest/usage/project/#working-with-version-control +.pdm.toml +.pdm-python +.pdm-build/ + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# PyCharm +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. +#.idea/ + +# Ruff stuff: +.ruff_cache/ + +# PyPI configuration file +.pypirc + +.vscode + +test*.py +.zed/ +logs/ +logs +settings.json diff --git a/.python-version b/.python-version new file mode 100644 index 0000000..2c07333 --- /dev/null +++ b/.python-version @@ -0,0 +1 @@ +3.11 diff --git a/CREDITS b/CREDITS new file mode 100644 index 0000000..4425a0a --- /dev/null +++ b/CREDITS @@ -0,0 +1,17 @@ +Some icons used in this project are from Font Awesome Free by Fonticons, Inc. +Licensed under the Creative Commons Attribution 4.0 International License (CC BY 4.0): https://creativecommons.org/licenses/by/4.0/ +Icons were modified (repainted to white and exported to PNG). + +Huge Thanks to Python for being the programming language used in this application. +https://www.python.org/ + +Huge thanks to Arcade and Pyglet for being the graphical engines used in this application. +https://arcade.academy/ +https://pyglet.readthedocs.io/en/latest/ + +Thanks to the following other libraries used in this application: +pypresence - https://github.com/qwertyquerty/pypresence - Used for Discord Rich Presence +pydub - https://github.com/jiaaro/pydub - Used for audio dBFS and normalization +yt-dlp - https://github.com/yt-dlp/yt-dlp - Used for downloading music +thefuzz and dependencies - https://github.com/seatgeek/thefuzz - Used for fuzzy search +mutagen - https://github.com/quodlibet/mutagen - Used for audio metadata extraction and modification diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..f288702 --- /dev/null +++ b/LICENSE @@ -0,0 +1,674 @@ + GNU GENERAL PUBLIC LICENSE + Version 3, 29 June 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU General Public License is a free, copyleft license for +software and other kinds of works. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +the GNU General Public License is intended to guarantee your freedom to +share and change all versions of a program--to make sure it remains free +software for all its users. We, the Free Software Foundation, use the +GNU General Public License for most of our software; it applies also to +any other work released this way by its authors. You can apply it to +your programs, too. + + When we speak of free software, we are referring to freedom, not +price. Our General Public Licenses are designed to make sure that you +have the freedom to distribute copies of free software (and charge for +them if you wish), that you receive source code or can get it if you +want it, that you can change the software or use pieces of it in new +free programs, and that you know you can do these things. + + To protect your rights, we need to prevent others from denying you +these rights or asking you to surrender the rights. Therefore, you have +certain responsibilities if you distribute copies of the software, or if +you modify it: responsibilities to respect the freedom of others. + + For example, if you distribute copies of such a program, whether +gratis or for a fee, you must pass on to the recipients the same +freedoms that you received. You must make sure that they, too, receive +or can get the source code. And you must show them these terms so they +know their rights. + + Developers that use the GNU GPL protect your rights with two steps: +(1) assert copyright on the software, and (2) offer you this License +giving you legal permission to copy, distribute and/or modify it. + + For the developers' and authors' protection, the GPL clearly explains +that there is no warranty for this free software. For both users' and +authors' sake, the GPL requires that modified versions be marked as +changed, so that their problems will not be attributed erroneously to +authors of previous versions. + + Some devices are designed to deny users access to install or run +modified versions of the software inside them, although the manufacturer +can do so. This is fundamentally incompatible with the aim of +protecting users' freedom to change the software. The systematic +pattern of such abuse occurs in the area of products for individuals to +use, which is precisely where it is most unacceptable. Therefore, we +have designed this version of the GPL to prohibit the practice for those +products. If such problems arise substantially in other domains, we +stand ready to extend this provision to those domains in future versions +of the GPL, as needed to protect the freedom of users. + + Finally, every program is threatened constantly by software patents. +States should not allow patents to restrict development and use of +software on general-purpose computers, but in those that do, we wish to +avoid the special danger that patents applied to a free program could +make it effectively proprietary. To prevent this, the GPL assures that +patents cannot be used to render the program non-free. + + The precise terms and conditions for copying, distribution and +modification follow. + + TERMS AND CONDITIONS + + 0. Definitions. + + "This License" refers to version 3 of the GNU General Public License. + + "Copyright" also means copyright-like laws that apply to other kinds of +works, such as semiconductor masks. + + "The Program" refers to any copyrightable work licensed under this +License. Each licensee is addressed as "you". "Licensees" and +"recipients" may be individuals or organizations. + + To "modify" a work means to copy from or adapt all or part of the work +in a fashion requiring copyright permission, other than the making of an +exact copy. The resulting work is called a "modified version" of the +earlier work or a work "based on" the earlier work. + + A "covered work" means either the unmodified Program or a work based +on the Program. + + To "propagate" a work means to do anything with it that, without +permission, would make you directly or secondarily liable for +infringement under applicable copyright law, except executing it on a +computer or modifying a private copy. Propagation includes copying, +distribution (with or without modification), making available to the +public, and in some countries other activities as well. + + To "convey" a work means any kind of propagation that enables other +parties to make or receive copies. Mere interaction with a user through +a computer network, with no transfer of a copy, is not conveying. + + An interactive user interface displays "Appropriate Legal Notices" +to the extent that it includes a convenient and prominently visible +feature that (1) displays an appropriate copyright notice, and (2) +tells the user that there is no warranty for the work (except to the +extent that warranties are provided), that licensees may convey the +work under this License, and how to view a copy of this License. If +the interface presents a list of user commands or options, such as a +menu, a prominent item in the list meets this criterion. + + 1. Source Code. + + The "source code" for a work means the preferred form of the work +for making modifications to it. "Object code" means any non-source +form of a work. + + A "Standard Interface" means an interface that either is an official +standard defined by a recognized standards body, or, in the case of +interfaces specified for a particular programming language, one that +is widely used among developers working in that language. + + The "System Libraries" of an executable work include anything, other +than the work as a whole, that (a) is included in the normal form of +packaging a Major Component, but which is not part of that Major +Component, and (b) serves only to enable use of the work with that +Major Component, or to implement a Standard Interface for which an +implementation is available to the public in source code form. A +"Major Component", in this context, means a major essential component +(kernel, window system, and so on) of the specific operating system +(if any) on which the executable work runs, or a compiler used to +produce the work, or an object code interpreter used to run it. + + The "Corresponding Source" for a work in object code form means all +the source code needed to generate, install, and (for an executable +work) run the object code and to modify the work, including scripts to +control those activities. However, it does not include the work's +System Libraries, or general-purpose tools or generally available free +programs which are used unmodified in performing those activities but +which are not part of the work. For example, Corresponding Source +includes interface definition files associated with source files for +the work, and the source code for shared libraries and dynamically +linked subprograms that the work is specifically designed to require, +such as by intimate data communication or control flow between those +subprograms and other parts of the work. + + The Corresponding Source need not include anything that users +can regenerate automatically from other parts of the Corresponding +Source. + + The Corresponding Source for a work in source code form is that +same work. + + 2. Basic Permissions. + + All rights granted under this License are granted for the term of +copyright on the Program, and are irrevocable provided the stated +conditions are met. This License explicitly affirms your unlimited +permission to run the unmodified Program. The output from running a +covered work is covered by this License only if the output, given its +content, constitutes a covered work. This License acknowledges your +rights of fair use or other equivalent, as provided by copyright law. + + You may make, run and propagate covered works that you do not +convey, without conditions so long as your license otherwise remains +in force. You may convey covered works to others for the sole purpose +of having them make modifications exclusively for you, or provide you +with facilities for running those works, provided that you comply with +the terms of this License in conveying all material for which you do +not control copyright. Those thus making or running the covered works +for you must do so exclusively on your behalf, under your direction +and control, on terms that prohibit them from making any copies of +your copyrighted material outside their relationship with you. + + Conveying under any other circumstances is permitted solely under +the conditions stated below. Sublicensing is not allowed; section 10 +makes it unnecessary. + + 3. Protecting Users' Legal Rights From Anti-Circumvention Law. + + No covered work shall be deemed part of an effective technological +measure under any applicable law fulfilling obligations under article +11 of the WIPO copyright treaty adopted on 20 December 1996, or +similar laws prohibiting or restricting circumvention of such +measures. + + When you convey a covered work, you waive any legal power to forbid +circumvention of technological measures to the extent such circumvention +is effected by exercising rights under this License with respect to +the covered work, and you disclaim any intention to limit operation or +modification of the work as a means of enforcing, against the work's +users, your or third parties' legal rights to forbid circumvention of +technological measures. + + 4. Conveying Verbatim Copies. + + You may convey verbatim copies of the Program's source code as you +receive it, in any medium, provided that you conspicuously and +appropriately publish on each copy an appropriate copyright notice; +keep intact all notices stating that this License and any +non-permissive terms added in accord with section 7 apply to the code; +keep intact all notices of the absence of any warranty; and give all +recipients a copy of this License along with the Program. + + You may charge any price or no price for each copy that you convey, +and you may offer support or warranty protection for a fee. + + 5. Conveying Modified Source Versions. + + You may convey a work based on the Program, or the modifications to +produce it from the Program, in the form of source code under the +terms of section 4, provided that you also meet all of these conditions: + + a) The work must carry prominent notices stating that you modified + it, and giving a relevant date. + + b) The work must carry prominent notices stating that it is + released under this License and any conditions added under section + 7. This requirement modifies the requirement in section 4 to + "keep intact all notices". + + c) You must license the entire work, as a whole, under this + License to anyone who comes into possession of a copy. This + License will therefore apply, along with any applicable section 7 + additional terms, to the whole of the work, and all its parts, + regardless of how they are packaged. This License gives no + permission to license the work in any other way, but it does not + invalidate such permission if you have separately received it. + + d) If the work has interactive user interfaces, each must display + Appropriate Legal Notices; however, if the Program has interactive + interfaces that do not display Appropriate Legal Notices, your + work need not make them do so. + + A compilation of a covered work with other separate and independent +works, which are not by their nature extensions of the covered work, +and which are not combined with it such as to form a larger program, +in or on a volume of a storage or distribution medium, is called an +"aggregate" if the compilation and its resulting copyright are not +used to limit the access or legal rights of the compilation's users +beyond what the individual works permit. Inclusion of a covered work +in an aggregate does not cause this License to apply to the other +parts of the aggregate. + + 6. Conveying Non-Source Forms. + + You may convey a covered work in object code form under the terms +of sections 4 and 5, provided that you also convey the +machine-readable Corresponding Source under the terms of this License, +in one of these ways: + + a) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by the + Corresponding Source fixed on a durable physical medium + customarily used for software interchange. + + b) Convey the object code in, or embodied in, a physical product + (including a physical distribution medium), accompanied by a + written offer, valid for at least three years and valid for as + long as you offer spare parts or customer support for that product + model, to give anyone who possesses the object code either (1) a + copy of the Corresponding Source for all the software in the + product that is covered by this License, on a durable physical + medium customarily used for software interchange, for a price no + more than your reasonable cost of physically performing this + conveying of source, or (2) access to copy the + Corresponding Source from a network server at no charge. + + c) Convey individual copies of the object code with a copy of the + written offer to provide the Corresponding Source. This + alternative is allowed only occasionally and noncommercially, and + only if you received the object code with such an offer, in accord + with subsection 6b. + + d) Convey the object code by offering access from a designated + place (gratis or for a charge), and offer equivalent access to the + Corresponding Source in the same way through the same place at no + further charge. You need not require recipients to copy the + Corresponding Source along with the object code. If the place to + copy the object code is a network server, the Corresponding Source + may be on a different server (operated by you or a third party) + that supports equivalent copying facilities, provided you maintain + clear directions next to the object code saying where to find the + Corresponding Source. Regardless of what server hosts the + Corresponding Source, you remain obligated to ensure that it is + available for as long as needed to satisfy these requirements. + + e) Convey the object code using peer-to-peer transmission, provided + you inform other peers where the object code and Corresponding + Source of the work are being offered to the general public at no + charge under subsection 6d. + + A separable portion of the object code, whose source code is excluded +from the Corresponding Source as a System Library, need not be +included in conveying the object code work. + + A "User Product" is either (1) a "consumer product", which means any +tangible personal property which is normally used for personal, family, +or household purposes, or (2) anything designed or sold for incorporation +into a dwelling. In determining whether a product is a consumer product, +doubtful cases shall be resolved in favor of coverage. For a particular +product received by a particular user, "normally used" refers to a +typical or common use of that class of product, regardless of the status +of the particular user or of the way in which the particular user +actually uses, or expects or is expected to use, the product. A product +is a consumer product regardless of whether the product has substantial +commercial, industrial or non-consumer uses, unless such uses represent +the only significant mode of use of the product. + + "Installation Information" for a User Product means any methods, +procedures, authorization keys, or other information required to install +and execute modified versions of a covered work in that User Product from +a modified version of its Corresponding Source. The information must +suffice to ensure that the continued functioning of the modified object +code is in no case prevented or interfered with solely because +modification has been made. + + If you convey an object code work under this section in, or with, or +specifically for use in, a User Product, and the conveying occurs as +part of a transaction in which the right of possession and use of the +User Product is transferred to the recipient in perpetuity or for a +fixed term (regardless of how the transaction is characterized), the +Corresponding Source conveyed under this section must be accompanied +by the Installation Information. But this requirement does not apply +if neither you nor any third party retains the ability to install +modified object code on the User Product (for example, the work has +been installed in ROM). + + The requirement to provide Installation Information does not include a +requirement to continue to provide support service, warranty, or updates +for a work that has been modified or installed by the recipient, or for +the User Product in which it has been modified or installed. Access to a +network may be denied when the modification itself materially and +adversely affects the operation of the network or violates the rules and +protocols for communication across the network. + + Corresponding Source conveyed, and Installation Information provided, +in accord with this section must be in a format that is publicly +documented (and with an implementation available to the public in +source code form), and must require no special password or key for +unpacking, reading or copying. + + 7. Additional Terms. + + "Additional permissions" are terms that supplement the terms of this +License by making exceptions from one or more of its conditions. +Additional permissions that are applicable to the entire Program shall +be treated as though they were included in this License, to the extent +that they are valid under applicable law. If additional permissions +apply only to part of the Program, that part may be used separately +under those permissions, but the entire Program remains governed by +this License without regard to the additional permissions. + + When you convey a copy of a covered work, you may at your option +remove any additional permissions from that copy, or from any part of +it. (Additional permissions may be written to require their own +removal in certain cases when you modify the work.) You may place +additional permissions on material, added by you to a covered work, +for which you have or can give appropriate copyright permission. + + Notwithstanding any other provision of this License, for material you +add to a covered work, you may (if authorized by the copyright holders of +that material) supplement the terms of this License with terms: + + a) Disclaiming warranty or limiting liability differently from the + terms of sections 15 and 16 of this License; or + + b) Requiring preservation of specified reasonable legal notices or + author attributions in that material or in the Appropriate Legal + Notices displayed by works containing it; or + + c) Prohibiting misrepresentation of the origin of that material, or + requiring that modified versions of such material be marked in + reasonable ways as different from the original version; or + + d) Limiting the use for publicity purposes of names of licensors or + authors of the material; or + + e) Declining to grant rights under trademark law for use of some + trade names, trademarks, or service marks; or + + f) Requiring indemnification of licensors and authors of that + material by anyone who conveys the material (or modified versions of + it) with contractual assumptions of liability to the recipient, for + any liability that these contractual assumptions directly impose on + those licensors and authors. + + All other non-permissive additional terms are considered "further +restrictions" within the meaning of section 10. If the Program as you +received it, or any part of it, contains a notice stating that it is +governed by this License along with a term that is a further +restriction, you may remove that term. If a license document contains +a further restriction but permits relicensing or conveying under this +License, you may add to a covered work material governed by the terms +of that license document, provided that the further restriction does +not survive such relicensing or conveying. + + If you add terms to a covered work in accord with this section, you +must place, in the relevant source files, a statement of the +additional terms that apply to those files, or a notice indicating +where to find the applicable terms. + + Additional terms, permissive or non-permissive, may be stated in the +form of a separately written license, or stated as exceptions; +the above requirements apply either way. + + 8. Termination. + + You may not propagate or modify a covered work except as expressly +provided under this License. Any attempt otherwise to propagate or +modify it is void, and will automatically terminate your rights under +this License (including any patent licenses granted under the third +paragraph of section 11). + + However, if you cease all violation of this License, then your +license from a particular copyright holder is reinstated (a) +provisionally, unless and until the copyright holder explicitly and +finally terminates your license, and (b) permanently, if the copyright +holder fails to notify you of the violation by some reasonable means +prior to 60 days after the cessation. + + Moreover, your license from a particular copyright holder is +reinstated permanently if the copyright holder notifies you of the +violation by some reasonable means, this is the first time you have +received notice of violation of this License (for any work) from that +copyright holder, and you cure the violation prior to 30 days after +your receipt of the notice. + + Termination of your rights under this section does not terminate the +licenses of parties who have received copies or rights from you under +this License. If your rights have been terminated and not permanently +reinstated, you do not qualify to receive new licenses for the same +material under section 10. + + 9. Acceptance Not Required for Having Copies. + + You are not required to accept this License in order to receive or +run a copy of the Program. Ancillary propagation of a covered work +occurring solely as a consequence of using peer-to-peer transmission +to receive a copy likewise does not require acceptance. However, +nothing other than this License grants you permission to propagate or +modify any covered work. These actions infringe copyright if you do +not accept this License. Therefore, by modifying or propagating a +covered work, you indicate your acceptance of this License to do so. + + 10. Automatic Licensing of Downstream Recipients. + + Each time you convey a covered work, the recipient automatically +receives a license from the original licensors, to run, modify and +propagate that work, subject to this License. You are not responsible +for enforcing compliance by third parties with this License. + + An "entity transaction" is a transaction transferring control of an +organization, or substantially all assets of one, or subdividing an +organization, or merging organizations. If propagation of a covered +work results from an entity transaction, each party to that +transaction who receives a copy of the work also receives whatever +licenses to the work the party's predecessor in interest had or could +give under the previous paragraph, plus a right to possession of the +Corresponding Source of the work from the predecessor in interest, if +the predecessor has it or can get it with reasonable efforts. + + You may not impose any further restrictions on the exercise of the +rights granted or affirmed under this License. For example, you may +not impose a license fee, royalty, or other charge for exercise of +rights granted under this License, and you may not initiate litigation +(including a cross-claim or counterclaim in a lawsuit) alleging that +any patent claim is infringed by making, using, selling, offering for +sale, or importing the Program or any portion of it. + + 11. Patents. + + A "contributor" is a copyright holder who authorizes use under this +License of the Program or a work on which the Program is based. The +work thus licensed is called the contributor's "contributor version". + + A contributor's "essential patent claims" are all patent claims +owned or controlled by the contributor, whether already acquired or +hereafter acquired, that would be infringed by some manner, permitted +by this License, of making, using, or selling its contributor version, +but do not include claims that would be infringed only as a +consequence of further modification of the contributor version. For +purposes of this definition, "control" includes the right to grant +patent sublicenses in a manner consistent with the requirements of +this License. + + Each contributor grants you a non-exclusive, worldwide, royalty-free +patent license under the contributor's essential patent claims, to +make, use, sell, offer for sale, import and otherwise run, modify and +propagate the contents of its contributor version. + + In the following three paragraphs, a "patent license" is any express +agreement or commitment, however denominated, not to enforce a patent +(such as an express permission to practice a patent or covenant not to +sue for patent infringement). To "grant" such a patent license to a +party means to make such an agreement or commitment not to enforce a +patent against the party. + + If you convey a covered work, knowingly relying on a patent license, +and the Corresponding Source of the work is not available for anyone +to copy, free of charge and under the terms of this License, through a +publicly available network server or other readily accessible means, +then you must either (1) cause the Corresponding Source to be so +available, or (2) arrange to deprive yourself of the benefit of the +patent license for this particular work, or (3) arrange, in a manner +consistent with the requirements of this License, to extend the patent +license to downstream recipients. "Knowingly relying" means you have +actual knowledge that, but for the patent license, your conveying the +covered work in a country, or your recipient's use of the covered work +in a country, would infringe one or more identifiable patents in that +country that you have reason to believe are valid. + + If, pursuant to or in connection with a single transaction or +arrangement, you convey, or propagate by procuring conveyance of, a +covered work, and grant a patent license to some of the parties +receiving the covered work authorizing them to use, propagate, modify +or convey a specific copy of the covered work, then the patent license +you grant is automatically extended to all recipients of the covered +work and works based on it. + + A patent license is "discriminatory" if it does not include within +the scope of its coverage, prohibits the exercise of, or is +conditioned on the non-exercise of one or more of the rights that are +specifically granted under this License. You may not convey a covered +work if you are a party to an arrangement with a third party that is +in the business of distributing software, under which you make payment +to the third party based on the extent of your activity of conveying +the work, and under which the third party grants, to any of the +parties who would receive the covered work from you, a discriminatory +patent license (a) in connection with copies of the covered work +conveyed by you (or copies made from those copies), or (b) primarily +for and in connection with specific products or compilations that +contain the covered work, unless you entered into that arrangement, +or that patent license was granted, prior to 28 March 2007. + + Nothing in this License shall be construed as excluding or limiting +any implied license or other defenses to infringement that may +otherwise be available to you under applicable patent law. + + 12. No Surrender of Others' Freedom. + + If conditions are imposed on you (whether by court order, agreement or +otherwise) that contradict the conditions of this License, they do not +excuse you from the conditions of this License. If you cannot convey a +covered work so as to satisfy simultaneously your obligations under this +License and any other pertinent obligations, then as a consequence you may +not convey it at all. For example, if you agree to terms that obligate you +to collect a royalty for further conveying from those to whom you convey +the Program, the only way you could satisfy both those terms and this +License would be to refrain entirely from conveying the Program. + + 13. Use with the GNU Affero General Public License. + + Notwithstanding any other provision of this License, you have +permission to link or combine any covered work with a work licensed +under version 3 of the GNU Affero General Public License into a single +combined work, and to convey the resulting work. The terms of this +License will continue to apply to the part which is the covered work, +but the special requirements of the GNU Affero General Public License, +section 13, concerning interaction through a network will apply to the +combination as such. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU General Public License from time to time. Such new versions will +be similar in spirit to the present version, but may differ in detail to +address new problems or concerns. + + Each version is given a distinguishing version number. If the +Program specifies that a certain numbered version of the GNU General +Public License "or any later version" applies to it, you have the +option of following the terms and conditions either of that numbered +version or of any later version published by the Free Software +Foundation. If the Program does not specify a version number of the +GNU General Public License, you may choose any version ever published +by the Free Software Foundation. + + If the Program specifies that a proxy can decide which future +versions of the GNU General Public License can be used, that proxy's +public statement of acceptance of a version permanently authorizes you +to choose that version for the Program. + + Later license versions may give you additional or different +permissions. However, no additional obligations are imposed on any +author or copyright holder as a result of your choosing to follow a +later version. + + 15. Disclaimer of Warranty. + + THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY +APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT +HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY +OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, +THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR +PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM +IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF +ALL NECESSARY SERVICING, REPAIR OR CORRECTION. + + 16. Limitation of Liability. + + IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING +WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS +THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY +GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE +USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF +DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD +PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), +EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF +SUCH DAMAGES. + + 17. Interpretation of Sections 15 and 16. + + If the disclaimer of warranty and limitation of liability provided +above cannot be given local legal effect according to their terms, +reviewing courts shall apply local law that most closely approximates +an absolute waiver of all civil liability in connection with the +Program, unless a warranty or assumption of liability accompanies a +copy of the Program in return for a fee. + + END OF TERMS AND CONDITIONS + + How to Apply These Terms to Your New Programs + + If you develop a new program, and you want it to be of the greatest +possible use to the public, the best way to achieve this is to make it +free software which everyone can redistribute and change under these terms. + + To do so, attach the following notices to the program. It is safest +to attach them to the start of each source file to most effectively +state the exclusion of warranty; and each file should have at least +the "copyright" line and a pointer to where the full notice is found. + + + Copyright (C) + + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If the program does terminal interaction, make it output a short +notice like this when it starts in an interactive mode: + + Copyright (C) + This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. + This is free software, and you are welcome to redistribute it + under certain conditions; type `show c' for details. + +The hypothetical commands `show w' and `show c' should show the appropriate +parts of the General Public License. Of course, your program's commands +might be different; for a GUI interface, you would use an "about box". + + You should also get your employer (if you work as a programmer) or school, +if any, to sign a "copyright disclaimer" for the program, if necessary. +For more information on this, and how to apply and follow the GNU GPL, see +. + + The GNU General Public License does not permit incorporating your program +into proprietary programs. If your program is a subroutine library, you +may consider it more useful to permit linking proprietary applications with +the library. If this is what you want to do, use the GNU Lesser General +Public License instead of this License. But first, please read +. diff --git a/README.md b/README.md new file mode 100644 index 0000000..c3c9bbf --- /dev/null +++ b/README.md @@ -0,0 +1,19 @@ +A simple and fast music player in Arcade. + +Note: FFmpeg is required due to both yt-dlp and to support most formats + +Features: +- Music playing including looping, pausing/resuming, shuffling and skipping to next track +- Music metadata fixing from title +- Audio normalization using pydub +- Tab based music selection(directories) +- Custom playlists +- Fast search using just text, and instant best result playback using enter +- Discord RPC + +Features TBD: +- Improved UI looks +- More keyboard shortcuts +- Vim keybindings(focusing with up down arrow keys) +- Replace paths with own file manager +- And much more! diff --git a/assets/fonts/ProtestStrike-Regular.ttf b/assets/fonts/ProtestStrike-Regular.ttf new file mode 100644 index 0000000000000000000000000000000000000000..3a88f0c7d02486e4771519267ac8bf2862de2d1f GIT binary patch literal 83412 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 Y4P9o`cJWPMOdW>A9Y3%a6W@{kU-tH#9RL6T literal 0 HcmV?d00001 diff --git a/assets/graphics/button.png b/assets/graphics/button.png new file mode 100644 index 0000000000000000000000000000000000000000..92bebf98fb533a044035366d7a49e2007434b8f1 GIT binary patch literal 280 zcmeAS@N?(olHy`uVBq!ia0vp^4j|0I1|(Ny7TyC=_dQ)4Ln`LHoo&e1pdfI7FRsr1 z`pFiBO=d-J)vUg(D%>q5ZS?up-T%TfZ-zZTeRwk4{F%;QjxrzTlm9qx{rkLmr!3Or zWy2oK+}wIuFiSLuEih!zbx3B2YB5CudNgcUbDmCx}xG;9?CvY)!qPh&^1#P8Hw2(UB-)3&0Y32r+W^SNq<_4N(ZlG!A z2Abx-F^~b9z%k*ye!YJz6`cw3+fi3W&U`i}%&HIpK@*)cuhGI9pDYHGGop^ zL^h$!(B`Kx+$W|rp-x1sZ6lXq$}C~D)prGn2Yn)sw)(yx#E99iXSj;A~SrEIa0r=j~sossUm*BzS_pnY*O&qQ4@Jv3h@K(q3$ILrus!)GGF{v z^=ra>yDjwuIA7vjzKCDIQ{W}=iSSwJ3UCLwOYB@1PJs z+N45fd0C}fun(N=b)oA8`v9B+dZg_GsaPfSpDscC!wF?hT56ZjN&ZE47l0Q{tEj+M zOU($kX{l*7CD{jVED)aCE5JdsdP=bm+yYLA|CMMH2AbQ1f#>goY+yb5v20DHN Xrynq$q+EB~00000NkvXXu0mjfL}x&i literal 0 HcmV?d00001 diff --git a/assets/graphics/files.png b/assets/graphics/files.png new file mode 100644 index 0000000000000000000000000000000000000000..c00e88facc0b637431306a954c0d384dbe85525c GIT binary patch literal 541 zcmV+&0^U}r?_?L=+FpI6XA3k6&Mifnu$%lHU3`T|L#f>>D#id~F? zX^zFtny^fS7~R>C{oodRxh&`G&g{(GHIfLK$hUw|`Bw%efK#9ebO04BI#yc@jHjZF zoda{g1+e%VPbCL#flc7iNSXjzgd;xX23U=N6T%Y@unZg-NiAT-x0)Z)G?HF{if<)9 zq+=BLO7^eTfgmXj_@*@Ao6>-9N&~(r4fv)s;G2E}m%srqPjWsm1&fN+b_iU3sh0F0 zX()9-}{}J**Ts?5;?cJq`ss-lA5FH^M(__jRum| zlIkmx$XV_U{{i2DLtk+o}@)dpWGiNv^e;>uO3M?4X=41 zX+8xs$v`aJ7-)G${#Ljzh+^T!K*uq#mjP}^abMxmCe~LZ{Q_pK=W-D^Eoc`EsUqnC zn6$1<5e82wssguG|GG@LG0+Br2;MFe4ixflI7&;vK4B`h-^0K)Fk?QKZD1O>v-pIsN`P23CL@;5SguXkgRnpN&ej^(an_dKnUOLDI96y5G{kjou0mocml#`j{4; z#=xYcFOq(GM*bpcS@Ivtn+fo|^uzFc26_d8Um|$b@FqM@YEi*(3G4tT_C;S1HGnSR zAvKHcfF5uT97o{4Zc$tkt`fcz_-3e*(b#{A3!*Nt0Sxklr7>VkY!klQ8TE2^;MoPf iZ5c$nku3+kF@iMbPeT1#_T`l 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 literal 0 HcmV?d00001 diff --git a/assets/graphics/no_shuffle.png b/assets/graphics/no_shuffle.png new file mode 100644 index 0000000000000000000000000000000000000000..c7291969accd99be74a965ec5c300500d4a6653e GIT binary patch literal 591 zcmV-V0c0>zMQNc>3?wTT8h?&!Bk0nV3-JetB8V02Ug($IDHu16 zJ}zd)j5qUfXH4_LCCr?A&Ux?onwyR)|DwYJu~hIbB(!h>^Cqzl90A9`=B%e`Fo$)R zvo}$<^YTtw0#zse-cyO$BAlpODs?@4wuZ!Z@cdal7uX0fQSW8)m_c&2hQyQf_lLS#pns6t z`}gEc2$Y37JbAxF_qMvHeo^m+@@x{uUei6PLO?yx3d~pQ}-j;j|!snv_%xM zNNiv-rDCw-vh*rdD+Fui3O>8`9bP0l+XkK=R95! literal 0 HcmV?d00001 diff --git a/assets/graphics/playlist.png b/assets/graphics/playlist.png new file mode 100644 index 0000000000000000000000000000000000000000..90df3f29cfed0678f9da5516927e471fafedc1e3 GIT binary patch literal 468 zcmV;_0W1EAP)4;LhnGo`izeenRaxeA_3051UU+UFI34*=UsI`uC|%DN@FV3$Sv4HlPd z3X*lid7*A?!XLOHIU~6_a&`zdB&Q^=#gS3Mp%J?Q%!cwXk%<)XeL}XyY{qBI=2Lj@ zdkblI!EXs*nP42VnV->_Q<(1-adjluDjqK4DarM)sor5ta>AV=mTR<$7&kP0jTYP& zb0vUh{1xmHz$XkhvM_+`$X*M10NP|9Kq#jR08iML(>3%v?qPyH5L2X;UkS#h<#eASHYKOCyGBDbnq4Dv zIjxNcpiTAx_;gx($;PMCO89hIyN&yF+MJyB>GY*>IjxNcpv@1D!XF_tbt>He0000< KMNUMnLSTY8nZmgM literal 0 HcmV?d00001 diff --git a/assets/graphics/plus.png b/assets/graphics/plus.png new file mode 100644 index 0000000000000000000000000000000000000000..35b327930cb9287c6383c6735abf3412202b5b7a GIT binary patch literal 400 zcmV;B0dM|^P)HAWE(f=uB%D48_`lAX_2Uw~Xa^ z(n{7AU(dT9X-1CBj6U$xwZOf&sG4VSUJNXY=TBfSOExprO3W;^3#pmSRQZvx>c4@v zioLlaTq&9Xqno`P7%Re+qIE%$BD^+OaZxocs-~Bs385{^2o1)6R~=2V zGT;lCl_6#K6azcprRf{D#1WVR`|?EBJf0*3rxjeT{?whzpBOy;)u+HKS6BZ7EC?TP z-j&rAIl*N?G0FNUqQ*tlxTqQzRpX*+TvW{vRN+d|cfyQQk^ uaAiLVFyZ&_n}aG?E!rA`tol~zjp!M4g4wkwsHMOF0000W=U literal 0 HcmV?d00001 diff --git a/assets/graphics/reload.png b/assets/graphics/reload.png new file mode 100644 index 0000000000000000000000000000000000000000..83b07e0e56a04714de7cd6b82227993e3e1ff2d1 GIT binary patch literal 1083 zcmV-B1jPG^P)5Zfr~o)mo)46>%fBJ{RIjOBHJs*F_Og+nR`q zii;vhG5)%%vASW|S^er=>JD`%ecn{@QT1R0rHeSIj+Dq7P$%ZK|1~wDuHxsanOAyI0USTUUb!r0G|V!fVY6nWsSRu zLe~Nx0_Rn?eRLf~VduTne9-DX&wshILV)#ucU zD#n^$R;qi{zZ$sM0v9S1GHCF}GTv4%t)Y0fT;{^)Qrm#CHSjFKAB)`OLEt`Mx^kL} znF8(v9tO@d8lm9Ltg$#$&Z?Uu-g;4-Kh)+7JP#}@sCyLXGYb3x-Xt&9I5OUY%6u zYKG5~>Tb19rB9a*^*v?DfL^m?K(9#+=mTB^P6E~}lY}E@+YAD`3A_I^@CqXkna`6~N;KbqBH!Vhwl?xR-E7Zi|=rFDabm)$D^913JLnz&{aVWZ9OCna2mD zU!e17YQU$!J+%Ag?*PUN>dpX9=2bK`U;`no(iYo*cPLIIi~(Qg)+F(CXT$hEpsMQy*g9g#c_elbZvl=IcJWm;`mF-?0LPLsikKm|*6E@K-Ci+CaiVoOupM{=_>8dF z4il2Q(}c6@YC>{%18`gNTjVY&zb9?KE!V8tg*xg+_4~U16~Rx5%t|+`a-lR-9tLhC zCybp1jsVvsGAl_9_#>yT!@^K`JQ;YLris{cW(n)!dctjHcO+9WWVkp>nJ{E0il%Nv zJh*rdD+Fui3O>8`9KQ|k)H~3BEUE-gsd`xRez@g;ggo@;Wkk6w<8uxcDrocjwGbB_56Lt8=fIx%Bf?IBAt`VcuqQ3La+LkzjI5^`Bw=R$_mcs zU0J*LhIefGr#ZDJ*dwohIa6Pqkr`@uJmd0%D95Z=|AIZaAL3aucJNWjE%Q>Krx`q5{an^LB{Ts5!p(`@ literal 0 HcmV?d00001 diff --git a/assets/graphics/settings.png b/assets/graphics/settings.png new file mode 100644 index 0000000000000000000000000000000000000000..729e79901a89337009cf94983364a08bfae369ea GIT binary patch literal 1114 zcmV-g1f~0lP)Ff< z6$C-jf`5acl4UGXRH&kge|~XU=Vl;=l~w6JX;2C15?14z=y!P7pszdUGK>q@B6ab6oOph|Hj*$rPb82{$ASMRa#adRx-Cq*8^; zp*XH_QK$>}9*YuTtOHkp&w$;)ng4w|?hx<^@Dnf$>;)c*$SeWR0XM@pqYg9ySAhMV zi2261z&_xg;70WOJIBT)4=@0n58nvyz>~mlKx>0Zr9}yN8km9V9T;WflLzDYZtx@6 zfo;GvwFh^jVj1WJmVzBP&gLNxR;IrNR+9kqcs5YPqS-VSYIg!d_ii$xDGt#aF~j~6Tn)^ zL)Ze$0gpP!E&|VEL7@ssrR)^&mm`rYq$dwhuB>%Ii5T)wzZP$ERDRP@E)Kst$|Z8~ z(UIjTsa@?SP9EwhSaek0?I@RLZ87OVk^`N8zM^bfR0< z$Y(`tDKQ``kn&<)mSI~|OT0zW8D)F(QIsXUq);ZlgiRTIVF^o8QJP)Z2g0(vtHjl zvys{^<4beUK%C$d=mO7SF&a1r^njf}U>)d_GcB*5zyYuev>F6%0Z)N9g>Q&cN8$nM zSI65sb*1K>SJcA>aboK?y@H#-D#5Pj51Ky;Otf|LQkWXq> z;CsclV|6v*|1DMdBm}CCI_M!QD(_VI-Z9tM7xi9@PtsCTVheh>JVBhP31sF+U$(Ly zW=4wO_F@wqAnhvxdGY5{>5p&95Se zu)H$$!;~Z%&jW9PTfi`Z4GZk>f6oIy)Ys}0^?ai8c(v4B^`P(_XD90Btf%5-@+mmB zQ=kL9Sj1HPH#y9HEdNdZA__JF9);wh*rdD+Fui3O>8`9?$IEHw1zP+}Q_mF`=>%;$- zE(c5&TDl~ae;bRo@RDU|LMpQv>$KE7{5&SkFM0br=7v?fS6bVZZ?TNZ0S#IyleDNs+Vn06^wA2 y+n{dg{lBZLD(HPyY3-`0E1SK(?08(JAF?hQAxvX bool | None: + self.status_label.text = self.yt_dl_logger.buffer + + if "WARNING" in self.yt_dl_logger.buffer: + self.status_label.update_font(font_color=arcade.color.YELLOW) + elif "ERROR" in self.yt_dl_logger.buffer: + self.status_label.update_font(font_color=arcade.color.RED) + else: + self.status_label.update_font(font_color=arcade.color.LIGHT_GREEN) + + def download(self): + if not self.tab_selector.value: + return + + url = self.url_name_input.text + + if not "http" in url: + url = f"ytsearch1:{url}" + + path = os.path.expanduser(self.tab_selector.value) + + try: + info = self.yt_dl.extract_info(url, download=True) + except yt_dlp.DownloadError as e: + message = "".join(e.msg.strip().split("] ")[1:]) if e.msg else "Unknown yt-dlp error." + self.yt_dl_logger.buffer = f"ERROR: {message}" + return + + if info: + entry = info['entries'][0] if 'entries' in info else info + title = entry.get('title', 'Unknown') + uploader = entry.get('uploader', 'Unknown') + + if " - " in title: + artist, track_title = title.split(" - ", 1) + else: + artist = uploader + track_title = title + title = f"{artist} - {track_title}" + + try: + audio = EasyID3("downloaded_music.mp3") + audio["artist"] = artist + audio["title"] = track_title + audio.save() + except Exception as meta_err: + self.yt_dl_logger.buffer = f"ERROR: Tried to override metadata based on title, but failed: {meta_err}" + return + + if self.settings_dict.get("normalize_audio", True): + try: + audio = AudioSegment.from_file("downloaded_music.mp3") + + if audio.dBFS < self.settings_dict.get("normalized_volume", -8): + change = self.settings_dict.get("normalized_volume", -8) - audio.dBFS + audio = audio.apply_gain(change) + + audio.export("downloaded_music.mp3", format="mp3") + + except Exception as e: + self.yt_dl_logger.buffer = f"ERROR: Could not normalize volume due to an error: {e}" + return + try: + output_filename = os.path.join(path, f"{title}.mp3") + os.replace("downloaded_music.mp3", output_filename) + + except Exception as e: + self.yt_dl_logger.buffer = f"ERROR: Could not move file due to an error: {e}" + return + else: + self.yt_dl_logger.buffer = f"ERROR: Info unavailable. This maybe due to being unable to download it due to DRM or other issues" + return + + self.yt_dl_logger.buffer = f"Successfully downloaded {title} to {path}" + + def main_exit(self): + from menus.main import Main + self.window.show_view(Main(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) diff --git a/menus/main.py b/menus/main.py new file mode 100644 index 0000000..7c8ed32 --- /dev/null +++ b/menus/main.py @@ -0,0 +1,464 @@ +import random, asyncio, pypresence, time, copy, json, os, logging +import arcade, arcade.gui, pyglet + +from utils.preload import * +from utils.constants import button_style, slider_style, audio_extensions, discord_presence_id +from utils.utils import FakePyPresence, UIFocusTextureButton, extract_metadata, truncate_end + +from arcade.gui.property import bind +from thefuzz import process, fuzz +from pydub import AudioSegment + +from arcade.gui.experimental.scroll_area import UIScrollArea, UIScrollBar +from arcade.gui.experimental.focus import UIFocusGroup + +class Main(arcade.gui.UIView): + def __init__(self, pypresence_client: None | FakePyPresence | pypresence.Presence=None, current_mode: str | None=None, current_music_name: str | None=None, + current_length: int | None=None, current_music_player: pyglet.media.Player | None=None, queue: list | None=None, + loaded_sounds: dict | None=None, shuffle: bool=False): + super().__init__() + self.pypresence_client = pypresence_client + + with open("settings.json", "r") as file: + self.settings_dict = json.load(file) + + if self.settings_dict.get('discord_rpc', True): + if self.pypresence_client == None: # Game has started + try: + asyncio.get_event_loop() + except: + asyncio.set_event_loop(asyncio.new_event_loop()) + + try: + self.pypresence_client = pypresence.Presence(discord_presence_id) + self.pypresence_client.connect() + self.pypresence_client.start_time = time.time() + except: + self.pypresence_client = FakePyPresence() + self.pypresence_client.start_time = time.time() + + elif isinstance(self.pypresence_client, FakePyPresence): # the user has enabled RPC in the settings in this session. + # get start time from old object + start_time = copy.deepcopy(self.pypresence_client.start_time) + try: + self.pypresence_client = pypresence.Presence(discord_presence_id) + self.pypresence_client.connect() + self.pypresence_client.start_time = start_time + except: + self.pypresence_client = FakePyPresence() + self.pypresence_client.start_time = start_time + + else: + self.pypresence_client = pypresence_client + else: # game has started, but the user has disabled RPC in the settings. + self.pypresence_client = FakePyPresence() + self.pypresence_client.start_time = time.time() + + self.tab_options = self.settings_dict.get("tab_options", ["~/Music", "~/Downloads"]) + self.tab_content = {} + self.playlist_content = {} + self.tab_buttons = {} + self.music_buttons = {} + + self.current_music_name = current_music_name + self.current_length = current_length if current_length else 0 + self.current_music_player = current_music_player + self.current_mode = current_mode or "files" + self.current_playlist = None + self.time_to_seek = None + self.current_tab = self.tab_options[0] + self.queue = queue if queue else [] + self.loaded_sounds = loaded_sounds if loaded_sounds else {} + self.shuffle = shuffle + self.search_term = "" + self.highest_score_file = "" + self.volume = self.settings_dict.get("default_volume", 100) + + def on_show_view(self): + super().on_show_view() + + self.anchor = self.add_widget(arcade.gui.UIAnchorLayout(size_hint=(1, 1))) + + self.ui_box = self.anchor.add(arcade.gui.UIBoxLayout(size_hint=(1, 1), space_between=10)) + + # Tabs + + self.load_content() + if self.current_mode == "playlist" and not self.current_playlist: + self.current_playlist = list(self.playlist_content.keys())[0] if self.playlist_content else None + self.load_tabs() + + # Scrollable Sounds + self.scroll_box = self.ui_box.add(arcade.gui.UIBoxLayout(size_hint=(0.95, 0.8), space_between=15, vertical=False)) + + self.scroll_area = UIScrollArea(size_hint=(0.9, 1)) # center on screen + self.scroll_area.scroll_speed = -50 + self.scroll_box.add(self.scroll_area) + + self.scrollbar = UIScrollBar(self.scroll_area) + self.scrollbar.size_hint = (0.02, 1) + self.scroll_box.add(self.scrollbar) + + self.music_box = arcade.gui.UIBoxLayout(space_between=2) + self.scroll_area.add(self.music_box) + + # Utility + + self.settings_box = self.anchor.add(arcade.gui.UIBoxLayout(space_between=10), anchor_x="right", anchor_y="center", align_x=-10) + + self.new_tab_button = self.settings_box.add(UIFocusTextureButton(texture=plus_icon, texture_hovered=plus_icon, texture_pressed=plus_icon, style=button_style)) + self.new_tab_button.on_click = lambda event: self.new_tab() + + self.downloader_button = self.settings_box.add(UIFocusTextureButton(texture=download_icon, texture_hovered=download_icon, texture_pressed=download_icon, style=button_style)) + self.downloader_button.on_click = lambda event: self.downloader() + + self.reload_button = self.settings_box.add(UIFocusTextureButton(texture=reload_icon, texture_hovered=reload_icon, texture_pressed=reload_icon, style=button_style)) + self.reload_button.on_click = lambda event: self.reload() + + mode_icon = playlist_icon if self.current_mode == "files" else files_icon + + self.mode_button = self.settings_box.add(UIFocusTextureButton(texture=mode_icon, texture_hovered=mode_icon, texture_pressed=mode_icon, style=button_style)) + self.mode_button.on_click = lambda event: self.change_mode() + + self.settings_button = self.settings_box.add(UIFocusTextureButton(texture=settings_icon, style=button_style)) + self.settings_button.on_click = lambda event: self.settings() + + # Controls + + self.control_box = self.ui_box.add(arcade.gui.UIBoxLayout(size_hint=(0.95, 0.1), space_between=10, vertical=False)) + self.current_music_label = self.control_box.add(arcade.gui.UILabel(text=truncate_end(self.current_music_name, int(self.window.width / 30)) if self.current_music_name else "No songs playing", font_name="Protest Strike", font_size=16)) + self.time_label = self.control_box.add(arcade.gui.UILabel(text="00:00", font_name="Protest Strike", font_size=16)) + + self.progressbar = self.control_box.add(arcade.gui.UISlider(style=slider_style, width=self.window.width / 3, height=35)) + self.progressbar.on_change = self.on_progress_change + + self.pause_start_button = self.control_box.add(UIFocusTextureButton(texture=pause_icon if not self.current_music_player or self.current_music_player.playing else resume_icon)) + self.pause_start_button.on_click = lambda event: self.pause_start() + + self.skip_button = self.control_box.add(UIFocusTextureButton(texture=stop_icon)) + self.skip_button.on_click = lambda event: self.skip_sound() + + self.loop_button = self.control_box.add(UIFocusTextureButton(texture=no_loop_icon if not self.current_music_player or self.current_music_player.loop else loop_icon)) + self.loop_button.on_click = lambda event: self.loop_sound() + + self.shuffle_button = self.control_box.add(UIFocusTextureButton(texture=shuffle_icon)) + self.shuffle_button.on_click = lambda event: self.shuffle_sound() + + if self.current_music_player: + self.progressbar.max_value = self.current_length + self.volume = int(self.current_music_player.volume * 100) + + self.volume_label = self.control_box.add(arcade.gui.UILabel(text=f"{self.volume}%", font_name="Protest Strike", font_size=16)) + self.volume_slider = self.control_box.add(arcade.gui.UISlider(style=slider_style, width=self.window.width / 10, height=35, value=self.volume, max_value=100)) + self.volume_slider.on_change = self.on_volume_slider_change + + if self.current_mode == "files": + self.show_content(os.path.expanduser(self.current_tab)) + elif self.current_mode == "playlist": + self.show_content(self.current_playlist) + + arcade.schedule(self.update_presence, 2.5) + + self.update_presence(None) + + def update_buttons(self): + if self.current_mode == "files": + self.mode_button.texture = playlist_icon + self.mode_button.texture_hovered = playlist_icon + self.mode_button.texture_pressed = playlist_icon + + elif self.current_mode == "playlist": + self.mode_button.texture = files_icon + self.mode_button.texture_hovered = files_icon + self.mode_button.texture_pressed = files_icon + + self.shuffle_button.texture = no_shuffle_icon if self.shuffle else shuffle_icon + self.shuffle_button.texture_hovered = no_shuffle_icon if self.shuffle else shuffle_icon + self.shuffle_button.texture_pressed = no_shuffle_icon if self.shuffle else shuffle_icon + + if self.current_music_player: + self.pause_start_button.texture = pause_icon if self.current_music_player.playing else resume_icon + self.pause_start_button.texture_hovered = pause_icon if self.current_music_player.playing else resume_icon + self.pause_start_button.texture_pressed = pause_icon if self.current_music_player.playing else resume_icon + + self.loop_button.texture = no_loop_icon if self.current_music_player.loop else loop_icon + self.loop_button.texture_hovered = no_loop_icon if self.current_music_player.loop else loop_icon + self.loop_button.texture_pressed = no_loop_icon if self.current_music_player.loop else loop_icon + else: + self.pause_start_button.texture = pause_icon + self.pause_start_button.texture_hovered = pause_icon + self.pause_start_button.texture_pressed = pause_icon + + self.loop_button.texture = loop_icon + self.loop_button.texture_hovered = loop_icon + self.loop_button.texture_pressed = loop_icon + + def change_mode(self): + self.current_mode = "playlist" if self.current_mode == "files" else "files" + + self.current_playlist = list(self.playlist_content.keys())[0] if self.playlist_content else None + + self.highest_score_file = "" + self.search_term = "" + + self.reload() + + def skip_sound(self): + if not self.current_music_player is None: + if self.current_music_player.loop: + self.current_music_player.seek(0) + return + + if self.settings_dict.get("music_mode", "Streaming") == "Streaming": + del self.loaded_sounds[self.current_music_name] + + self.current_length = 0 + self.current_music_name = None + self.current_music_player.delete() + self.current_music_player = None + self.progressbar.value = 0 + self.current_music_label.text = "No songs playing" + self.time_label.text = "00:00" + + self.update_buttons() + + def pause_start(self): + if self.current_music_player is not None: + self.current_music_player._set_playing(not self.current_music_player.playing) + self.update_buttons() + + def loop_sound(self): + if not self.current_music_player is None: + self.current_music_player.loop = not self.current_music_player.loop + self.update_buttons() + + def shuffle_sound(self): + if not self.current_music_player is None: + self.shuffle = not self.shuffle + self.update_buttons() + + def show_content(self, tab): + self.music_box.clear() + self.music_buttons.clear() + + if self.current_mode == "files": + self.current_tab = tab + if not self.search_term == "": + matches = process.extract(self.search_term, self.tab_content[self.current_tab], limit=5, processor=lambda text: text.lower(), scorer=fuzz.partial_token_sort_ratio) + self.highest_score_file = f"{self.current_tab}/{matches[0][0]}" + for match in matches: + music_filename = match[0] + self.music_buttons[music_filename] = self.music_box.add(arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=music_filename, style=button_style, width=self.window.width * 0.85, height=self.window.height / 30)) + self.music_buttons[music_filename].on_click = lambda event, tab=tab, music_filename=music_filename: self.queue.append(f"{tab}/{music_filename}") + + else: + self.highest_score_file = "" + for music_filename in self.tab_content[tab]: + self.music_buttons[music_filename] = self.music_box.add(arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=music_filename, style=button_style, width=self.window.width * 0.85, height=self.window.height / 30)) + self.music_buttons[music_filename].on_click = lambda event, tab=tab, music_filename=music_filename: self.queue.append(f"{tab}/{music_filename}") + + elif self.current_mode == "playlist": + self.current_playlist = tab + + if self.current_playlist: + if not self.search_term == "": + matches = process.extract(self.search_term, self.playlist_content[tab], limit=5, processor=lambda text: text.lower(), scorer=fuzz.partial_token_sort_ratio) + self.highest_score_file = matches[0][0] + for match in matches: + music_filename = match[0] + self.music_buttons[music_filename] = self.music_box.add(arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=music_filename, style=button_style, width=self.window.width * 0.85, height=self.window.height / 30)) + self.music_buttons[music_filename].on_click = lambda event, tab=tab, music_filename=music_filename: self.queue.append(music_filename) + + else: + self.highest_score_file = "" + for music_filename in self.playlist_content[tab]: + self.music_buttons[music_filename] = self.music_box.add(arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=music_filename, style=button_style, width=self.window.width * 0.85, height=self.window.height / 30)) + self.music_buttons[music_filename].on_click = lambda event, tab=tab, music_filename=music_filename: self.queue.append(music_filename) + + self.music_buttons["add_music"] = self.music_box.add(arcade.gui.UITextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text="Add Music", style=button_style, width=self.window.width * 0.85, height=self.window.height / 30)) + self.music_buttons["add_music"].on_click = lambda event: self.add_music() + + self.update_buttons() + + def load_content(self): + self.tab_content.clear() + self.playlist_content.clear() + + for tab in self.tab_options: + self.tab_content[os.path.expanduser(tab)] = [] + for filename in os.listdir(os.path.expanduser(tab)): + if filename.split(".")[-1] in audio_extensions: + self.tab_content[os.path.expanduser(tab)].append(filename) + + for playlist, content in self.settings_dict.get("playlists", {}).items(): + self.playlist_content[playlist] = content + + def load_tabs(self): + self.tab_box = self.ui_box.add(arcade.gui.UIBoxLayout(size_hint=(0.95, 0.1), space_between=10, vertical=False)) + + if self.current_mode == "files": + for tab in self.tab_options: + self.tab_buttons[os.path.expanduser(tab)] = self.tab_box.add(UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=os.path.basename(os.path.normpath(os.path.expanduser(tab))), style=button_style, width=self.window.width / 10, height=self.window.height / 15)) + self.tab_buttons[os.path.expanduser(tab)].on_click = lambda event, tab=os.path.expanduser(tab): self.show_content(tab) + elif self.current_mode == "playlist": + for playlist in self.playlist_content: + self.tab_buttons[playlist] = self.tab_box.add(UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=playlist, style=button_style, width=self.window.width / 10, height=self.window.height / 15)) + self.tab_buttons[playlist].on_click = lambda event, playlist=playlist: self.show_content(playlist) + + def on_progress_change(self, event): + if not self.current_music_player is None: + scale = self.progressbar.value / self.progressbar.max_value + + self.time_to_seek = self.current_length * scale + + def on_volume_slider_change(self, event): + self.volume = int(self.volume_slider.value) + self.volume_label.text = f"{self.volume}%" + + if not self.current_music_player is None: + self.current_music_player.volume = self.volume / 100 + + def on_update(self, delta_time): + if self.current_music_player is None or self.current_music_player.time == 0: + if len(self.queue) > 0: + music_path = self.queue.pop(0) + + artist, title = extract_metadata(music_path) + + music_name = f"{artist} - {title}" + + if self.settings_dict.get("normalize_audio", True): + try: + audio = AudioSegment.from_file(music_path) + + if int(audio.dBFS) != self.settings_dict.get("normalized_volume", -8): + change = self.settings_dict.get("normalized_volume", -8) - audio.dBFS + audio = audio.apply_gain(change) + + audio.export(music_path, format="mp3") + except Exception as e: + logging.error(f"Couldn't normalize volume for {music_path}: {e}") + + if not music_name in self.loaded_sounds: + self.loaded_sounds[music_name] = arcade.Sound(music_path, streaming=self.settings_dict.get("music_mode", "Stream") == "Stream") + + self.volume = self.settings_dict.get("default_volume", 100) + self.volume_label.text = f"{self.volume}%" + self.volume_slider.value = self.volume + + self.current_music_player = self.loaded_sounds[music_name].play() + self.current_music_player.volume = self.volume / 100 + self.current_length = self.loaded_sounds[music_name].get_length() + + self.current_music_name = music_name + self.current_music_label.text = truncate_end(music_name, int(self.window.width / 25)) + self.time_label.text = "00:00" + self.progressbar.max_value = self.current_length + self.progressbar.value = 0 + + else: + if self.current_music_player is not None: + self.skip_sound() # reset properties + + if self.shuffle: + self.queue.append(f"{self.current_tab}/{random.choice(self.tab_content[self.current_tab])}") + + if not self.current_music_player is None: + if self.time_to_seek is not None: + self.current_music_player.seek(self.time_to_seek) + self.progressbar.value = self.time_to_seek + self.time_to_seek = None + else: + self.progressbar.value = self.current_music_player.time + mins, secs = divmod(self.current_music_player.time, 60) + self.time_label.text = f"{int(mins):02d}:{int(secs):02d}" + + def on_key_press(self, symbol: int, modifiers: int) -> bool | None: + if symbol == arcade.key.SPACE: + self.pause_start() + elif symbol == arcade.key.DELETE: + self.skip_sound() + elif symbol == arcade.key.RIGHT and self.current_music_player: + self.current_music_player.seek(self.current_music_player.time + 5) + elif symbol == arcade.key.LEFT and self.current_music_player: + self.current_music_player.seek(self.current_music_player.time - 5) + elif symbol == arcade.key.UP and self.current_music_player: + self.current_music_player.pitch += 0.1 # type: ignore + elif symbol == arcade.key.DOWN and self.current_music_player: + self.current_music_player.pitch -= 0.1 # type: ignore + elif symbol == arcade.key.BACKSPACE: + self.search_term = self.search_term[:-1] + if self.current_mode == "files": + self.show_content(self.current_tab) + elif self.current_mode == "playlist": + self.show_content(self.current_playlist) + elif symbol == arcade.key.ENTER and self.highest_score_file: + self.queue.append(self.highest_score_file) + self.highest_score_file = "" + self.search_term = "" + if self.current_mode == "files": + self.show_content(self.current_tab) + elif self.current_mode == "playlist": + self.show_content(self.current_playlist) + elif symbol == arcade.key.ESCAPE: + self.highest_score_file = "" + self.search_term = "" + if self.current_mode == "files": + self.show_content(self.current_tab) + elif self.current_mode == "playlist": + self.show_content(self.current_playlist) + + def on_text(self, text): + if not text.isprintable() or text == " ": + return + + self.search_term += text + + if self.current_mode == "files": + self.show_content(self.current_tab) + elif self.current_mode == "playlist": + self.show_content(self.current_playlist) + + def settings(self): + from menus.settings import Settings + arcade.unschedule(self.update_presence) + self.ui.clear() + self.window.show_view(Settings(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) + + def new_tab(self): + from menus.new_tab import NewTab + arcade.unschedule(self.update_presence) + self.ui.clear() + self.window.show_view(NewTab(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) + + def add_music(self): + from menus.add_music import AddMusic + arcade.unschedule(self.update_presence) + self.ui.clear() + self.window.show_view(AddMusic(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) + + def downloader(self): + from menus.downloader import Downloader + arcade.unschedule(self.update_presence) + self.ui.clear() + self.window.show_view(Downloader(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) + + def reload(self): + self.ui.clear() + self.on_show_view() + self.update_buttons() + + def update_presence(self, _): + if self.current_music_label.text != "No songs playing" and self.current_music_player: + details = f"Listening to {self.current_music_name}" + + if self.current_music_player.playing: + mins, secs = divmod(self.current_length, 60) + state = f"{self.time_label.text} / {int(mins):02d}:{int(secs):02d}" + else: + state = "Paused" + else: + details = "" + state = "No songs playing" + + self.pypresence_client.update(state=state, details=details, start=self.pypresence_client.start_time) diff --git a/menus/new_tab.py b/menus/new_tab.py new file mode 100644 index 0000000..4fec857 --- /dev/null +++ b/menus/new_tab.py @@ -0,0 +1,77 @@ +import arcade, arcade.gui, os, json + +from utils.constants import button_style +from utils.preload import button_texture, button_hovered_texture +from utils.utils import UIFocusTextureButton + +from arcade.gui.experimental.focus import UIFocusGroup + +class NewTab(arcade.gui.UIView): + def __init__(self, pypresence_client, current_mode, current_music_name, current_length, current_music_player, queue, loaded_sounds, shuffle): + super().__init__() + + self.current_mode = current_mode + self.current_music_name = current_music_name + self.current_length = current_length + self.current_music_player = current_music_player + self.queue = queue + self.loaded_sounds = loaded_sounds + self.shuffle = shuffle + + with open("settings.json", "r") as file: + self.settings_dict = json.load(file) + + self.tab_options = self.settings_dict.get("tab_options", ["~/Music", "~/Downloads"]) + self.playlists = self.settings_dict.get("playlists", {}) + + self.pypresence_client = pypresence_client + self.pypresence_client.update(state="Adding new tab", start=self.pypresence_client.start_time) + + def on_show_view(self): + super().on_show_view() + + self.anchor = self.add_widget(UIFocusGroup(size_hint=(1, 1))) + self.box = self.anchor.add(arcade.gui.UIBoxLayout(space_between=10), anchor_x="center", anchor_y="center") + + if self.current_mode == "files": + self.new_tab_label = self.box.add(arcade.gui.UILabel(text="New Tab Path:", font_name="Protest Strike", font_size=32)) + self.new_tab_input = self.box.add(arcade.gui.UIInputText(font_name="Protest Strike", font_size=32, width=self.window.width / 2, height=self.window.height / 10)) + self.new_tab_button = self.box.add(UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='Add new tab', style=button_style, width=self.window.width / 2, height=self.window.height / 10)) + self.new_tab_button.on_click = lambda event: self.add_tab() + elif self.current_mode == "playlist": + self.new_tab_label = self.box.add(arcade.gui.UILabel(text="New Playlist Name:", font_name="Protest Strike", font_size=32)) + self.new_tab_input = self.box.add(arcade.gui.UIInputText(font_name="Protest Strike", font_size=32, width=self.window.width / 2, height=self.window.height / 10)) + self.new_tab_button = self.box.add(UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='Add new Playlist', style=button_style, width=self.window.width / 2, height=self.window.height / 10)) + self.new_tab_button.on_click = lambda event: self.add_tab() + + self.back_button = self.anchor.add(UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='<--', style=button_style, width=100, height=50), anchor_x="left", anchor_y="top", align_x=5, align_y=-5) + self.back_button.on_click = lambda event: self.main_exit() + + self.anchor.detect_focusable_widgets() + + def add_tab(self): + if self.current_mode == "files": + tab_path = self.new_tab_input.text + + if not tab_path: + return + + if not os.path.exists(os.path.expanduser(tab_path)) or not os.path.isdir(os.path.expanduser(tab_path)) or not os.path.isabs(os.path.expanduser(tab_path)): + return + + if tab_path in self.tab_options or os.path.expanduser(tab_path) in self.tab_options: + return + + self.tab_options.append(tab_path) + self.settings_dict["tab_options"] = self.tab_options + + elif self.current_mode == "playlist": + self.playlists[self.new_tab_input.text] = [] + self.settings_dict["playlists"] = self.playlists + + with open("settings.json", "w") as file: + file.write(json.dumps(self.settings_dict)) + + def main_exit(self): + from menus.main import Main + self.window.show_view(Main(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) diff --git a/menus/settings.py b/menus/settings.py new file mode 100644 index 0000000..b25dd2a --- /dev/null +++ b/menus/settings.py @@ -0,0 +1,298 @@ +import copy, pypresence, json + +import arcade, arcade.gui + +from utils.constants import button_style, dropdown_style, slider_style, settings, discord_presence_id, settings_start_category +from utils.utils import FakePyPresence, UIFocusTextureButton +from utils.preload import button_texture, button_hovered_texture + +from arcade.gui.experimental.focus import UIFocusGroup + +class Settings(arcade.gui.UIView): + def __init__(self, pypresence_client, current_mode, current_music_name, current_length, current_music_player, queue, loaded_sounds, shuffle): + super().__init__() + + self.current_mode = current_mode + self.current_music_name = current_music_name + self.current_length = current_length + self.current_music_player = current_music_player + self.queue = queue + self.loaded_sounds = loaded_sounds + self.shuffle = shuffle + + with open("settings.json", "r") as file: + self.settings_dict = json.load(file) + + self.pypresence_client = pypresence_client + self.pypresence_client.update(state='In Settings', details='Modifying Settings', start=self.pypresence_client.start_time) + + self.slider_labels = {} + self.sliders = {} + + self.on_radiobuttons = {} + self.off_radiobuttons = {} + + self.current_category = settings_start_category + + self.modified_settings = {} + + def create_layouts(self): + self.anchor = self.add_widget(UIFocusGroup(size_hint=(1, 1))) + + self.box = arcade.gui.UIBoxLayout(space_between=50, align="center", vertical=False) + self.anchor.add(self.box, anchor_x="center", anchor_y="top", align_x=10, align_y=-75) + + self.top_box = arcade.gui.UIBoxLayout(space_between=self.window.width / 160, vertical=False) + self.anchor.add(self.top_box, anchor_x="left", anchor_y="top", align_x=10, align_y=-10) + + self.key_layout = self.box.add(arcade.gui.UIBoxLayout(space_between=20, align='left')) + self.value_layout = self.box.add(arcade.gui.UIBoxLayout(space_between=13, align='left')) + + def on_show_view(self): + super().on_show_view() + + self.create_layouts() + + self.ui.push_handlers(self) + + self.back_button = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='<--', style=button_style, width=100, height=50) + self.back_button.on_click = lambda event: self.main_exit() + self.top_box.add(self.back_button) + + self.display_categories() + + self.display_category(settings_start_category) + + self.anchor.detect_focusable_widgets() + + def display_categories(self): + for category in settings: + category_button = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text=category, style=button_style, width=self.window.width / 10, height=50) + + if not category == "Credits": + category_button.on_click = lambda event, category=category: self.display_category(category) + else: + category_button.on_click = lambda event: self.credits() + + self.top_box.add(category_button) + + self.anchor.detect_focusable_widgets() + + def display_category(self, category): + if hasattr(self, 'apply_button'): + self.anchor.remove(self.apply_button) + del self.apply_button + + if hasattr(self, 'credits_label'): + self.anchor.remove(self.credits_label) + del self.credits_label + + self.current_category = category + + self.key_layout.clear() + self.value_layout.clear() + + for setting in settings[category]: + label = arcade.gui.UILabel(text=setting, font_name="Protest Strike", font_size=28, text_color=arcade.color.WHITE ) + self.key_layout.add(label) + + setting_dict = settings[category][setting] + + if setting_dict['type'] == "option": + dropdown = arcade.gui.UIDropdown(options=setting_dict['options'], width=200, height=50, default=self.settings_dict.get(setting_dict["config_key"], setting_dict["options"][0]), active_style=dropdown_style, dropdown_style=dropdown_style, primary_style=dropdown_style) + dropdown.on_change = lambda _, setting=setting, dropdown=dropdown: self.update(setting, dropdown.value, "option") + self.value_layout.add(dropdown) + + elif setting_dict['type'] == "bool": + button_layout = self.value_layout.add(arcade.gui.UIBoxLayout(space_between=50, vertical=False)) + + on_radiobutton = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text="ON", style=button_style, width=150, height=50) + self.on_radiobuttons[setting] = on_radiobutton + on_radiobutton.on_click = lambda _, setting=setting: self.update(setting, True, "bool") + button_layout.add(on_radiobutton) + + off_radiobutton = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text="OFF", style=button_style, width=150, height=50) + self.off_radiobuttons[setting] = off_radiobutton + off_radiobutton.on_click = lambda _, setting=setting: self.update(setting, False, "bool") + button_layout.add(off_radiobutton) + + if self.settings_dict.get(setting_dict["config_key"], setting_dict["default"]): + self.set_highlighted_style(on_radiobutton) + self.set_normal_style(off_radiobutton) + else: + self.set_highlighted_style(off_radiobutton) + self.set_normal_style(on_radiobutton) + + elif setting_dict['type'] == "slider": + if setting == "FPS Limit": + if self.settings_dict.get(setting_dict["config_key"]) == 0: + label_text = "FPS Limit: Disabled" + else: + label_text = f"FPS Limit: {self.settings_dict.get(setting_dict['config_key'], setting_dict['default'])}" + else: + label_text = f"{setting}: {int(self.settings_dict.get(setting_dict['config_key'], setting_dict['default']))}" + + label.text = label_text + + self.slider_labels[setting] = label + + slider = arcade.gui.UISlider(width=400, height=50, value=self.settings_dict.get(setting_dict["config_key"], setting_dict["default"]), min_value=setting_dict['min'], max_value=setting_dict['max'], style=slider_style) + slider.on_change = lambda _, setting=setting, slider=slider: self.update(setting, slider.value, "slider") + + self.sliders[setting] = slider + self.value_layout.add(slider) + + self.apply_button = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='Apply', style=button_style, width=200, height=100) + self.apply_button.on_click = lambda event: self.apply_settings() + self.anchor.add(self.apply_button, anchor_x="right", anchor_y="bottom", align_x=-10, align_y=10) + + self.anchor.detect_focusable_widgets() + + def apply_settings(self): + for config_key, value in self.modified_settings.items(): + self.settings_dict[config_key] = value + + if self.settings_dict['window_mode'] == "Fullscreen": + self.window.set_fullscreen(True) + else: + self.window.set_fullscreen(False) + width, height = map(int, self.settings_dict['resolution'].split('x')) + self.window.set_size(width, height) + + if self.settings_dict['vsync']: + self.window.set_vsync(True) + display_mode = self.window.display.get_default_screen().get_mode() + refresh_rate = display_mode.rate + self.window.set_update_rate(1 / refresh_rate) + self.window.set_draw_rate(1 / refresh_rate) + + elif not self.settings_dict['fps_limit'] == 0: + self.window.set_vsync(False) + self.window.set_update_rate(1 / self.settings_dict['fps_limit']) + self.window.set_draw_rate(1 / self.settings_dict['fps_limit']) + + else: + self.window.set_vsync(False) + self.window.set_update_rate(1 / 99999999) + self.window.set_draw_rate(1 / 99999999) + + if self.settings_dict['discord_rpc']: + if isinstance(self.pypresence_client, FakePyPresence): # the user has enabled RPC in the settings in this session. + start_time = copy.deepcopy(self.pypresence_client.start_time) + self.pypresence_client.close() + del self.pypresence_client + try: + self.pypresence_client = pypresence.Presence(discord_presence_id) + self.pypresence_client.connect() + self.pypresence_client.update(state='In Settings', details='Modifying Settings', start=start_time) + self.pypresence_client.start_time = start_time + except: + self.pypresence_client = FakePyPresence() + self.pypresence_client.start_time = start_time + else: + if not isinstance(self.pypresence_client, FakePyPresence): + start_time = copy.deepcopy(self.pypresence_client.start_time) + self.pypresence_client.update() + self.pypresence_client.close() + del self.pypresence_client + self.pypresence_client = FakePyPresence() + self.pypresence_client.start_time = start_time + + self.ui_cleanup() + + self.ui = arcade.gui.UIManager() + self.ui.enable() + + self.create_layouts() + + self.back_button = UIFocusTextureButton(texture=button_texture, texture_hovered=button_hovered_texture, text='<--', style=button_style, width=100, height=50) + self.back_button.on_click = lambda event: self.main_exit() + self.top_box.add(self.back_button) + + self.display_categories() + + self.display_category(self.current_category) + + with open("settings.json", "w") as file: + file.write(json.dumps(self.settings_dict, indent=4)) + + def update(self, setting=None, button_state=None, setting_type="bool"): + setting_dict = settings[self.current_category][setting] + config_key = settings[self.current_category][setting]["config_key"] + + if setting_type == "option": + self.modified_settings[config_key] = button_state + + elif setting_type == "bool": + self.modified_settings[config_key] = button_state + + if button_state: + self.set_highlighted_style(self.on_radiobuttons[setting]) + self.set_normal_style(self.off_radiobuttons[setting]) + else: + self.set_highlighted_style(self.off_radiobuttons[setting]) + self.set_normal_style(self.on_radiobuttons[setting]) + + elif setting_type == "slider": + new_value = int(button_state) + + self.modified_settings[config_key] = new_value + self.sliders[setting].value = new_value + + if setting == "FPS Limit": + if new_value == 0: + label_text = "FPS Limit: Disabled" + else: + label_text = f"FPS Limit: {str(new_value).rjust(8)}" + else: + label_text = f"{setting}: {str(new_value).rjust(8)}" + + self.slider_labels[setting].text = label_text + + def credits(self): + if hasattr(self, 'apply_button'): + self.anchor.remove(self.apply_button) + del self.apply_button + + if hasattr(self, 'credits_label'): + self.anchor.remove(self.credits_label) + del self.credits_label + + self.key_layout.clear() + self.value_layout.clear() + + with open('CREDITS', 'r') as file: + text = file.read() + + if self.window.width == 3840: + font_size = 30 + elif self.window.width == 2560: + font_size = 20 + elif self.window.width == 1920: + font_size = 17 + elif self.window.width >= 1440: + font_size = 14 + else: + font_size = 12 + + self.credits_label = arcade.gui.UILabel(text=text, text_color=arcade.color.WHITE, font_name="Protest Strike", font_size=font_size, align="center", multiline=True) + + self.key_layout.add(self.credits_label) + + self.anchor.detect_focusable_widgets() + + def set_highlighted_style(self, element): + element.texture = button_hovered_texture + element.texture_hovered = button_texture + + def set_normal_style(self, element): + element.texture_hovered = button_hovered_texture + element.texture = button_texture + + def main_exit(self): + from menus.main import Main + self.window.show_view(Main(self.pypresence_client, self.current_mode, self.current_music_name, self.current_length, self.current_music_player, self.queue, self.loaded_sounds, self.shuffle)) + + def ui_cleanup(self): + self.ui.clear() + del self.ui diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..e80080d --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,14 @@ +[project] +name = "MusicPlayer" +version = "0.1.0" +description = "A Music Player in Python, made with arcade and pyglet" +readme = "README.md" +requires-python = ">=3.11" +dependencies = [ + "arcade==3.2.0", + "mutagen>=1.47.0", + "pydub>=0.25.1", + "pypresence>=4.3.0", + "thefuzz>=0.22.1", + "yt-dlp>=2025.4.30", +] diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..13b6a79 --- /dev/null +++ b/requirements.txt @@ -0,0 +1,6 @@ +Pillow +arcade==3.2.0 +pypresence +mutagen +yt-dlp +thefuzz diff --git a/run.py b/run.py new file mode 100644 index 0000000..997ff09 --- /dev/null +++ b/run.py @@ -0,0 +1,98 @@ +import pyglet + +pyglet.options.debug_gl = False + +import logging, datetime, os, json, sys, arcade + +from utils.utils import get_closest_resolution, print_debug_info, on_exception, ErrorView +from utils.constants import log_dir, menu_background_color +from menus.main import Main +from arcade.experimental.controller_window import ControllerWindow + +sys.excepthook = on_exception + +pyglet.resource.path.append(os.getcwd()) +pyglet.font.add_directory('./assets/fonts') + +if not log_dir in os.listdir(): + os.makedirs(log_dir) + +while len(os.listdir(log_dir)) >= 5: + files = [(file, os.path.getctime(os.path.join(log_dir, file))) for file in os.listdir(log_dir)] + oldest_file = sorted(files, key=lambda x: x[1])[0][0] + os.remove(os.path.join(log_dir, oldest_file)) + +timestamp = datetime.datetime.now().strftime("%Y-%m-%d_%H-%M-%S") +log_filename = f"debug_{timestamp}.log" +logging.basicConfig(filename=f'{os.path.join(log_dir, log_filename)}', format='%(asctime)s %(name)s %(levelname)s: %(message)s', level=logging.DEBUG) + +for logger_name_to_disable in ['arcade', "numba"]: + logging.getLogger(logger_name_to_disable).propagate = False + logging.getLogger(logger_name_to_disable).disabled = True + +if os.path.exists('settings.json'): + with open('settings.json', 'r') as settings_file: + settings = json.load(settings_file) + + resolution = list(map(int, settings['resolution'].split('x'))) + + if not settings.get("anti_aliasing", "4x MSAA") == "None": + antialiasing = int(settings.get("anti_aliasing", "4x MSAA").split('x')[0]) + else: + antialiasing = 0 + + fullscreen = settings['window_mode'] == 'Fullscreen' + style = arcade.Window.WINDOW_STYLE_BORDERLESS if settings['window_mode'] == 'borderless' else arcade.Window.WINDOW_STYLE_DEFAULT + vsync = settings['vsync'] + fps_limit = settings['fps_limit'] +else: + resolution = get_closest_resolution() + antialiasing = 4 + fullscreen = False + style = arcade.Window.WINDOW_STYLE_DEFAULT + vsync = True + fps_limit = 0 + + settings = { + "resolution": f"{resolution[0]}x{resolution[1]}", + "antialiasing": "4x MSAA", + "window_mode": "Windowed", + "vsync": True, + "fps_limit": 60, + "discord_rpc": True + } + + with open("settings.json", "w") as file: + file.write(json.dumps(settings)) + +window = ControllerWindow(width=resolution[0], height=resolution[1], title='Music Player', samples=antialiasing, antialiasing=antialiasing > 0, fullscreen=fullscreen, vsync=vsync, resizable=False, style=style) + +if vsync: + window.set_vsync(True) + display_mode = window.display.get_default_screen().get_mode() + refresh_rate = display_mode.rate + window.set_update_rate(1 / refresh_rate) + window.set_draw_rate(1 / refresh_rate) +elif not fps_limit == 0: + window.set_update_rate(1 / fps_limit) + window.set_draw_rate(1 / fps_limit) +else: + window.set_update_rate(1 / 99999999) + window.set_draw_rate(1 / 99999999) + +arcade.set_background_color(menu_background_color) + +print_debug_info() + +if pyglet.media.codecs.have_ffmpeg(): + menu = Main() +else: + menu = ErrorView("FFmpeg has not been found but is required for this application.", "FFmpeg lib not found.") + +window.show_view(menu) + +logging.debug('Game started.') + +arcade.run() + +logging.info('Exited with error code 0.') diff --git a/utils/constants.py b/utils/constants.py new file mode 100644 index 0000000..9995237 --- /dev/null +++ b/utils/constants.py @@ -0,0 +1,72 @@ +import arcade.color +from arcade.types import Color +from arcade.gui.widgets.buttons import UITextureButtonStyle, UIFlatButtonStyle +from arcade.gui.widgets.slider import UISliderStyle + +menu_background_color = (17, 17, 17) +log_dir = 'logs' +discord_presence_id = 1368277020332523530 + +audio_extensions = [ + "3g2", "3gp", "aac", "ac3", "aiff", "alac", "amr", "ape", "au", "caf", + "dts", "flac", "gsm", "m4a", "mka", "mlp", "mmf", "mp2", "mp3", + "oga", "ogg", "opus", "ra", "rm", "sln", "tta", "vorbis", "voc", "vox", + "wav", "webm", "wma", "wv" +] + +yt_dlp_parameters = { + "final_ext": "mp3", + "format": "bestaudio/best", + "outtmpl": {"pl_thumbnail": "", "default": "downloaded_music.mp3"}, + "postprocessors": [ + { + "key": "FFmpegExtractAudio", + "nopostoverwrites": False, + "preferredcodec": "mp3", + "preferredquality": "5" + }, + { + "add_chapters": True, + "add_infojson": "if_exists", + "add_metadata": True, + "key": "FFmpegMetadata" + }, + { "already_have_thumbnail": False, "key": "EmbedThumbnail" } + ], + "writethumbnail": True +} + +button_style = {'normal': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK), 'hover': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK), + 'press': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK), 'disabled': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK)} +big_button_style = {'normal': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, font_size=26), 'hover': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, font_size=26), + 'press': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, font_size=26), 'disabled': UITextureButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, font_size=26)} + +dropdown_style = {'normal': UIFlatButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, bg=Color(128, 128, 128)), 'hover': UIFlatButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, bg=Color(49, 154, 54)), + 'press': UIFlatButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, bg=Color(128, 128, 128)), 'disabled': UIFlatButtonStyle(font_name="Protest Strike", font_color=arcade.color.BLACK, bg=Color(128, 128, 128))} + +slider_default_style = UISliderStyle(bg=Color(128, 128, 128), unfilled_track=Color(128, 128, 128), filled_track=Color(49, 154, 54)) +slider_hover_style = UISliderStyle(bg=Color(49, 154, 54), unfilled_track=Color(128, 128, 128), filled_track=Color(49, 154, 54)) + +slider_style = {'normal': slider_default_style, 'hover': slider_hover_style, 'press': slider_hover_style, 'disabled': slider_default_style} + +settings = { + "Music": { + "Default Volume": {"type": "slider", "min": 0, "max": 100, "config_key": "default_volume", "default": 100}, + "Audio Mode": {"type": "option", "options": ["Stream", "Preload"], "config_key": "audio_mode", "default": "Stream"}, + "Normalize Audio": {"type": "bool", "config_key": "normalize_audio", "default": True}, + "Normalized dBFS": {"type": "slider", "min": -30, "max": 0, "config_key": "normalized_volume", "default": -8}, + }, + "Graphics": { + "Window Mode": {"type": "option", "options": ["Windowed", "Fullscreen", "Borderless"], "config_key": "window_mode", "default": "Windowed"}, + "Resolution": {"type": "option", "options": ["1366x768", "1440x900", "1600x900", "1920x1080", "2560x1440", "3840x2160"], "config_key": "resolution"}, + "Anti-Aliasing": {"type": "option", "options": ["None", "2x MSAA", "4x MSAA", "8x MSAA", "16x MSAA"], "config_key": "anti_aliasing", "default": "4x MSAA"}, + "VSync": {"type": "bool", "config_key": "vsync", "default": True}, + "FPS Limit": {"type": "slider", "min": 0, "max": 480, "config_key": "fps_limit", "default": 60}, + }, + "Miscellaneous": { + "Discord RPC": {"type": "bool", "config_key": "discord_rpc", "default": True}, + }, + "Credits": {} +} + +settings_start_category = "Music" diff --git a/utils/preload.py b/utils/preload.py new file mode 100644 index 0000000..08842e0 --- /dev/null +++ b/utils/preload.py @@ -0,0 +1,21 @@ +import arcade.gui, arcade + +button_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button.png")) +button_hovered_texture = arcade.gui.NinePatchTexture(64 // 4, 64 // 4, 64 // 4, 64 // 4, arcade.load_texture("assets/graphics/button_hovered.png")) + +pause_icon = arcade.load_texture("assets/graphics/pause.png") +resume_icon = arcade.load_texture("assets/graphics/resume.png") + +stop_icon = arcade.load_texture("assets/graphics/stop.png") +loop_icon = arcade.load_texture("assets/graphics/loop.png") +no_loop_icon = arcade.load_texture("assets/graphics/no_loop.png") + +shuffle_icon = arcade.load_texture("assets/graphics/shuffle.png") +no_shuffle_icon = arcade.load_texture("assets/graphics/no_shuffle.png") + +settings_icon = arcade.load_texture("assets/graphics/settings.png") +reload_icon = arcade.load_texture("assets/graphics/reload.png") +download_icon = arcade.load_texture("assets/graphics/download.png") +plus_icon = arcade.load_texture("assets/graphics/plus.png") +playlist_icon = arcade.load_texture("assets/graphics/playlist.png") +files_icon = arcade.load_texture("assets/graphics/files.png") diff --git a/utils/utils.py b/utils/utils.py new file mode 100644 index 0000000..28512da --- /dev/null +++ b/utils/utils.py @@ -0,0 +1,168 @@ +import logging, arcade, arcade.gui, sys, traceback, os, re + +from mutagen.easyid3 import EasyID3 + +from utils.constants import menu_background_color +import pyglet + +def dump_platform(): + import platform + logging.debug(f'Platform: {platform.platform()}') + logging.debug(f'Release: {platform.release()}') + logging.debug(f'Machine: {platform.machine()}') + logging.debug(f'Architecture: {platform.architecture()}') + +def dump_gl(): + from pyglet.gl import gl_info as info + logging.debug(f'gl_info.get_version(): {info.get_version()}') + logging.debug(f'gl_info.get_vendor(): {info.get_vendor()}') + logging.debug(f'gl_info.get_renderer(): {info.get_renderer()}') + +def print_debug_info(): + logging.debug('########################## DEBUG INFO ##########################') + logging.debug('') + dump_platform() + dump_gl() + logging.debug('') + logging.debug(f'Number of screens: {len(pyglet.display.get_display().get_screens())}') + logging.debug('') + for n, screen in enumerate(pyglet.display.get_display().get_screens()): + logging.debug(f"Screen #{n+1}:") + logging.debug(f'DPI: {screen.get_dpi()}') + logging.debug(f'Scale: {screen.get_scale()}') + logging.debug(f'Size: {screen.width}, {screen.height}') + logging.debug(f'Position: {screen.x}, {screen.y}') + logging.debug('') + logging.debug('########################## DEBUG INFO ##########################') + logging.debug('') + +class ErrorView(arcade.gui.UIView): + def __init__(self, message: str, title: str): + super().__init__() + + self.message = message + self.title = title + + def exit(self): + logging.fatal('Exited with error code 1.') + sys.exit(1) + + def on_show_view(self): + super().on_show_view() + + self.window.set_caption('Music Player - Error') + self.window.set_mouse_visible(True) + self.window.set_exclusive_mouse(False) + arcade.set_background_color(menu_background_color) + + msgbox = arcade.gui.UIMessageBox(width=self.window.width / 2, height=self.window.height / 2, message_text=self.message, title=self.title) + msgbox.on_action = lambda event: self.exit() + self.add_widget(msgbox) + +def on_exception(*exc_info): + logging.error(f"Unhandled exception:\n{''.join(traceback.format_exception(exc_info[1], limit=None))}") + +def get_closest_resolution(): + allowed_resolutions = [(1366, 768), (1440, 900), (1600,900), (1920,1080), (2560,1440), (3840,2160)] + screen_width, screen_height = arcade.get_screens()[0].width, arcade.get_screens()[0].height + if (screen_width, screen_height) in allowed_resolutions: + if not allowed_resolutions.index((screen_width, screen_height)) == 0: + closest_resolution = allowed_resolutions[allowed_resolutions.index((screen_width, screen_height))-1] + else: + closest_resolution = (screen_width, screen_height) + else: + target_width, target_height = screen_width // 2, screen_height // 2 + + closest_resolution = min( + allowed_resolutions, + key=lambda res: abs(res[0] - target_width) + abs(res[1] - target_height) + ) + return closest_resolution + +class FakePyPresence(): + def __init__(self): + ... + def update(self, *args, **kwargs): + ... + def close(self, *args, **kwargs): + ... + +class UIFocusTextureButton(arcade.gui.UITextureButton): + def __init__(self, *args, **kwargs): + super().__init__(*args, **kwargs) + arcade.gui.bind(self, "hovered", self.on_hover) + + def on_hover(self): + if self.hovered: + self.resize(width=self.width * 1.1, height=self.height * 1.1) + else: + self.resize(width=self.width / 1.1, height=self.height / 1.1) + +class BufferLogger: + def __init__(self): + self.buffer = "No errors." + + def debug(self, msg): + self._log(msg) + + def info(self, msg): + self._log(msg) + + def warning(self, msg): + self._log(f"WARNING: {msg}") + + def error(self, msg): + self._log(f"ERROR: {msg}") + + def _log(self, msg): + self.buffer = msg + +def truncate_end(text: str, max_length: int) -> str: + if len(text) <= max_length: + return text + if max_length <= 3: + return text + return text[:max_length - 3] + '...' + +def extract_metadata(filename): + artist = "Unknown" + title = "" + + basename = os.path.basename(filename) + name_only = os.path.splitext(basename)[0] + + name_only = re.sub(r'\s*\[[a-zA-Z0-9\-_]{5,}\]$', '', name_only) + + try: + audio = EasyID3(filename) + + artist = str(audio["artist"][0]) + title = str(audio["title"][0]) + + artist_title_match = re.search(r'^.+\s*-\s*.+$', title) # check for Artist - Title titles, so Artist doesnt appear twice + + if artist_title_match: + title = title.split("- ")[1] + + if artist != "Unknown" and title: + return artist, title + except: + pass + + if artist == "Unknown" or not title: + match = re.search(r'^(.*?)\s+-\s+(.*?)$', name_only) + if match: + filename_artist, filename_title = match.groups() + + if artist == "Unknown": + artist = filename_artist + + if not title: + title = filename_title + + return artist, title + + if not title: + title = name_only + + return artist, title diff --git a/uv.lock b/uv.lock new file mode 100644 index 0000000..c2bd518 --- /dev/null +++ b/uv.lock @@ -0,0 +1,326 @@ +version = 1 +revision = 2 +requires-python = ">=3.11" + +[[package]] +name = "arcade" +version = "3.2.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "pillow" }, + { name = "pyglet" }, + { name = "pymunk" }, + { name = "pytiled-parser" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/e5/39/87eaffdfc50ec9d4b4573652ef8b80cca0592e5ccafb5fc5bc8612b1445d/arcade-3.2.0.tar.gz", hash = "sha256:1c2c56181560665f6542157b9ab316b9551274a9ee8468bae017ed5b8fee18fd", size = 41941030, upload_time = "2025-05-09T20:16:20.112Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/40/9a/ac86f5cbccfe5455a28308fcf2d7179af8d9c3087ad4eb45706c2a7b089b/arcade-3.2.0-py3-none-any.whl", hash = "sha256:7bb47cf643b43272e4300d8a5ca5f1b1e9e131b0f3f1d3fad013cb29528d3062", size = 42635264, upload_time = "2025-05-09T20:16:15.98Z" }, +] + +[[package]] +name = "attrs" +version = "25.3.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/5a/b0/1367933a8532ee6ff8d63537de4f1177af4bff9f3e829baf7331f595bb24/attrs-25.3.0.tar.gz", hash = "sha256:75d7cefc7fb576747b2c81b4442d4d4a1ce0900973527c011d1030fd3bf4af1b", size = 812032, upload_time = "2025-03-13T11:10:22.779Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/77/06/bb80f5f86020c4551da315d78b3ab75e8228f89f0162f2c3a819e407941a/attrs-25.3.0-py3-none-any.whl", hash = "sha256:427318ce031701fea540783410126f03899a97ffc6f61596ad581ac2e40e3bc3", size = 63815, upload_time = "2025-03-13T11:10:21.14Z" }, +] + +[[package]] +name = "cffi" +version = "1.17.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "pycparser" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/fc/97/c783634659c2920c3fc70419e3af40972dbaf758daa229a7d6ea6135c90d/cffi-1.17.1.tar.gz", hash = "sha256:1c39c6016c32bc48dd54561950ebd6836e1670f2ae46128f67cf49e789c52824", size = 516621, upload_time = "2024-09-04T20:45:21.852Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6b/f4/927e3a8899e52a27fa57a48607ff7dc91a9ebe97399b357b85a0c7892e00/cffi-1.17.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:a45e3c6913c5b87b3ff120dcdc03f6131fa0065027d0ed7ee6190736a74cd401", size = 182264, upload_time = "2024-09-04T20:43:51.124Z" }, + { url = "https://files.pythonhosted.org/packages/6c/f5/6c3a8efe5f503175aaddcbea6ad0d2c96dad6f5abb205750d1b3df44ef29/cffi-1.17.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:30c5e0cb5ae493c04c8b42916e52ca38079f1b235c2f8ae5f4527b963c401caf", size = 178651, upload_time = "2024-09-04T20:43:52.872Z" }, + { url = "https://files.pythonhosted.org/packages/94/dd/a3f0118e688d1b1a57553da23b16bdade96d2f9bcda4d32e7d2838047ff7/cffi-1.17.1-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f75c7ab1f9e4aca5414ed4d8e5c0e303a34f4421f8a0d47a4d019ceff0ab6af4", size = 445259, upload_time = "2024-09-04T20:43:56.123Z" }, + { url = "https://files.pythonhosted.org/packages/2e/ea/70ce63780f096e16ce8588efe039d3c4f91deb1dc01e9c73a287939c79a6/cffi-1.17.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a1ed2dd2972641495a3ec98445e09766f077aee98a1c896dcb4ad0d303628e41", size = 469200, upload_time = "2024-09-04T20:43:57.891Z" }, + { url = "https://files.pythonhosted.org/packages/1c/a0/a4fa9f4f781bda074c3ddd57a572b060fa0df7655d2a4247bbe277200146/cffi-1.17.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:46bf43160c1a35f7ec506d254e5c890f3c03648a4dbac12d624e4490a7046cd1", size = 477235, upload_time = "2024-09-04T20:44:00.18Z" }, + { url = "https://files.pythonhosted.org/packages/62/12/ce8710b5b8affbcdd5c6e367217c242524ad17a02fe5beec3ee339f69f85/cffi-1.17.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a24ed04c8ffd54b0729c07cee15a81d964e6fee0e3d4d342a27b020d22959dc6", size = 459721, upload_time = "2024-09-04T20:44:01.585Z" }, + { url = "https://files.pythonhosted.org/packages/ff/6b/d45873c5e0242196f042d555526f92aa9e0c32355a1be1ff8c27f077fd37/cffi-1.17.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:610faea79c43e44c71e1ec53a554553fa22321b65fae24889706c0a84d4ad86d", size = 467242, upload_time = "2024-09-04T20:44:03.467Z" }, + { url = "https://files.pythonhosted.org/packages/1a/52/d9a0e523a572fbccf2955f5abe883cfa8bcc570d7faeee06336fbd50c9fc/cffi-1.17.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:a9b15d491f3ad5d692e11f6b71f7857e7835eb677955c00cc0aefcd0669adaf6", size = 477999, upload_time = "2024-09-04T20:44:05.023Z" }, + { url = "https://files.pythonhosted.org/packages/44/74/f2a2460684a1a2d00ca799ad880d54652841a780c4c97b87754f660c7603/cffi-1.17.1-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:de2ea4b5833625383e464549fec1bc395c1bdeeb5f25c4a3a82b5a8c756ec22f", size = 454242, upload_time = "2024-09-04T20:44:06.444Z" }, + { url = "https://files.pythonhosted.org/packages/f8/4a/34599cac7dfcd888ff54e801afe06a19c17787dfd94495ab0c8d35fe99fb/cffi-1.17.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:fc48c783f9c87e60831201f2cce7f3b2e4846bf4d8728eabe54d60700b318a0b", size = 478604, upload_time = "2024-09-04T20:44:08.206Z" }, + { url = "https://files.pythonhosted.org/packages/34/33/e1b8a1ba29025adbdcda5fb3a36f94c03d771c1b7b12f726ff7fef2ebe36/cffi-1.17.1-cp311-cp311-win32.whl", hash = "sha256:85a950a4ac9c359340d5963966e3e0a94a676bd6245a4b55bc43949eee26a655", size = 171727, upload_time = "2024-09-04T20:44:09.481Z" }, + { url = "https://files.pythonhosted.org/packages/3d/97/50228be003bb2802627d28ec0627837ac0bf35c90cf769812056f235b2d1/cffi-1.17.1-cp311-cp311-win_amd64.whl", hash = "sha256:caaf0640ef5f5517f49bc275eca1406b0ffa6aa184892812030f04c2abf589a0", size = 181400, upload_time = "2024-09-04T20:44:10.873Z" }, + { url = "https://files.pythonhosted.org/packages/5a/84/e94227139ee5fb4d600a7a4927f322e1d4aea6fdc50bd3fca8493caba23f/cffi-1.17.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:805b4371bf7197c329fcb3ead37e710d1bca9da5d583f5073b799d5c5bd1eee4", size = 183178, upload_time = "2024-09-04T20:44:12.232Z" }, + { url = "https://files.pythonhosted.org/packages/da/ee/fb72c2b48656111c4ef27f0f91da355e130a923473bf5ee75c5643d00cca/cffi-1.17.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:733e99bc2df47476e3848417c5a4540522f234dfd4ef3ab7fafdf555b082ec0c", size = 178840, upload_time = "2024-09-04T20:44:13.739Z" }, + { url = "https://files.pythonhosted.org/packages/cc/b6/db007700f67d151abadf508cbfd6a1884f57eab90b1bb985c4c8c02b0f28/cffi-1.17.1-cp312-cp312-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1257bdabf294dceb59f5e70c64a3e2f462c30c7ad68092d01bbbfb1c16b1ba36", size = 454803, upload_time = "2024-09-04T20:44:15.231Z" }, + { url = "https://files.pythonhosted.org/packages/1a/df/f8d151540d8c200eb1c6fba8cd0dfd40904f1b0682ea705c36e6c2e97ab3/cffi-1.17.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da95af8214998d77a98cc14e3a3bd00aa191526343078b530ceb0bd710fb48a5", size = 478850, upload_time = "2024-09-04T20:44:17.188Z" }, + { url = "https://files.pythonhosted.org/packages/28/c0/b31116332a547fd2677ae5b78a2ef662dfc8023d67f41b2a83f7c2aa78b1/cffi-1.17.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d63afe322132c194cf832bfec0dc69a99fb9bb6bbd550f161a49e9e855cc78ff", size = 485729, upload_time = "2024-09-04T20:44:18.688Z" }, + { url = "https://files.pythonhosted.org/packages/91/2b/9a1ddfa5c7f13cab007a2c9cc295b70fbbda7cb10a286aa6810338e60ea1/cffi-1.17.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f79fc4fc25f1c8698ff97788206bb3c2598949bfe0fef03d299eb1b5356ada99", size = 471256, upload_time = "2024-09-04T20:44:20.248Z" }, + { url = "https://files.pythonhosted.org/packages/b2/d5/da47df7004cb17e4955df6a43d14b3b4ae77737dff8bf7f8f333196717bf/cffi-1.17.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b62ce867176a75d03a665bad002af8e6d54644fad99a3c70905c543130e39d93", size = 479424, upload_time = "2024-09-04T20:44:21.673Z" }, + { url = "https://files.pythonhosted.org/packages/0b/ac/2a28bcf513e93a219c8a4e8e125534f4f6db03e3179ba1c45e949b76212c/cffi-1.17.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:386c8bf53c502fff58903061338ce4f4950cbdcb23e2902d86c0f722b786bbe3", size = 484568, upload_time = "2024-09-04T20:44:23.245Z" }, + { url = "https://files.pythonhosted.org/packages/d4/38/ca8a4f639065f14ae0f1d9751e70447a261f1a30fa7547a828ae08142465/cffi-1.17.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:4ceb10419a9adf4460ea14cfd6bc43d08701f0835e979bf821052f1805850fe8", size = 488736, upload_time = "2024-09-04T20:44:24.757Z" }, + { url = "https://files.pythonhosted.org/packages/86/c5/28b2d6f799ec0bdecf44dced2ec5ed43e0eb63097b0f58c293583b406582/cffi-1.17.1-cp312-cp312-win32.whl", hash = "sha256:a08d7e755f8ed21095a310a693525137cfe756ce62d066e53f502a83dc550f65", size = 172448, upload_time = "2024-09-04T20:44:26.208Z" }, + { url = "https://files.pythonhosted.org/packages/50/b9/db34c4755a7bd1cb2d1603ac3863f22bcecbd1ba29e5ee841a4bc510b294/cffi-1.17.1-cp312-cp312-win_amd64.whl", hash = "sha256:51392eae71afec0d0c8fb1a53b204dbb3bcabcb3c9b807eedf3e1e6ccf2de903", size = 181976, upload_time = "2024-09-04T20:44:27.578Z" }, + { url = "https://files.pythonhosted.org/packages/8d/f8/dd6c246b148639254dad4d6803eb6a54e8c85c6e11ec9df2cffa87571dbe/cffi-1.17.1-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:f3a2b4222ce6b60e2e8b337bb9596923045681d71e5a082783484d845390938e", size = 182989, upload_time = "2024-09-04T20:44:28.956Z" }, + { url = "https://files.pythonhosted.org/packages/8b/f1/672d303ddf17c24fc83afd712316fda78dc6fce1cd53011b839483e1ecc8/cffi-1.17.1-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:0984a4925a435b1da406122d4d7968dd861c1385afe3b45ba82b750f229811e2", size = 178802, upload_time = "2024-09-04T20:44:30.289Z" }, + { url = "https://files.pythonhosted.org/packages/0e/2d/eab2e858a91fdff70533cab61dcff4a1f55ec60425832ddfdc9cd36bc8af/cffi-1.17.1-cp313-cp313-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d01b12eeeb4427d3110de311e1774046ad344f5b1a7403101878976ecd7a10f3", size = 454792, upload_time = "2024-09-04T20:44:32.01Z" }, + { url = "https://files.pythonhosted.org/packages/75/b2/fbaec7c4455c604e29388d55599b99ebcc250a60050610fadde58932b7ee/cffi-1.17.1-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:706510fe141c86a69c8ddc029c7910003a17353970cff3b904ff0686a5927683", size = 478893, upload_time = "2024-09-04T20:44:33.606Z" }, + { url = "https://files.pythonhosted.org/packages/4f/b7/6e4a2162178bf1935c336d4da8a9352cccab4d3a5d7914065490f08c0690/cffi-1.17.1-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:de55b766c7aa2e2a3092c51e0483d700341182f08e67c63630d5b6f200bb28e5", size = 485810, upload_time = "2024-09-04T20:44:35.191Z" }, + { url = "https://files.pythonhosted.org/packages/c7/8a/1d0e4a9c26e54746dc08c2c6c037889124d4f59dffd853a659fa545f1b40/cffi-1.17.1-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c59d6e989d07460165cc5ad3c61f9fd8f1b4796eacbd81cee78957842b834af4", size = 471200, upload_time = "2024-09-04T20:44:36.743Z" }, + { url = "https://files.pythonhosted.org/packages/26/9f/1aab65a6c0db35f43c4d1b4f580e8df53914310afc10ae0397d29d697af4/cffi-1.17.1-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd398dbc6773384a17fe0d3e7eeb8d1a21c2200473ee6806bb5e6a8e62bb73dd", size = 479447, upload_time = "2024-09-04T20:44:38.492Z" }, + { url = "https://files.pythonhosted.org/packages/5f/e4/fb8b3dd8dc0e98edf1135ff067ae070bb32ef9d509d6cb0f538cd6f7483f/cffi-1.17.1-cp313-cp313-musllinux_1_1_aarch64.whl", hash = "sha256:3edc8d958eb099c634dace3c7e16560ae474aa3803a5df240542b305d14e14ed", size = 484358, upload_time = "2024-09-04T20:44:40.046Z" }, + { url = "https://files.pythonhosted.org/packages/f1/47/d7145bf2dc04684935d57d67dff9d6d795b2ba2796806bb109864be3a151/cffi-1.17.1-cp313-cp313-musllinux_1_1_x86_64.whl", hash = "sha256:72e72408cad3d5419375fc87d289076ee319835bdfa2caad331e377589aebba9", size = 488469, upload_time = "2024-09-04T20:44:41.616Z" }, + { url = "https://files.pythonhosted.org/packages/bf/ee/f94057fa6426481d663b88637a9a10e859e492c73d0384514a17d78ee205/cffi-1.17.1-cp313-cp313-win32.whl", hash = "sha256:e03eab0a8677fa80d646b5ddece1cbeaf556c313dcfac435ba11f107ba117b5d", size = 172475, upload_time = "2024-09-04T20:44:43.733Z" }, + { url = "https://files.pythonhosted.org/packages/7c/fc/6a8cb64e5f0324877d503c854da15d76c1e50eb722e320b15345c4d0c6de/cffi-1.17.1-cp313-cp313-win_amd64.whl", hash = "sha256:f6a16c31041f09ead72d69f583767292f750d24913dadacf5756b966aacb3f1a", size = 182009, upload_time = "2024-09-04T20:44:45.309Z" }, +] + +[[package]] +name = "musicplayer" +version = "0.1.0" +source = { virtual = "." } +dependencies = [ + { name = "arcade" }, + { name = "mutagen" }, + { name = "pydub" }, + { name = "pypresence" }, + { name = "thefuzz" }, + { name = "yt-dlp" }, +] + +[package.metadata] +requires-dist = [ + { name = "arcade", specifier = "==3.2.0" }, + { name = "mutagen", specifier = ">=1.47.0" }, + { name = "pydub", specifier = ">=0.25.1" }, + { name = "pypresence", specifier = ">=4.3.0" }, + { name = "thefuzz", specifier = ">=0.22.1" }, + { name = "yt-dlp", specifier = ">=2025.4.30" }, +] + +[[package]] +name = "mutagen" +version = "1.47.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/81/e6/64bc71b74eef4b68e61eb921dcf72dabd9e4ec4af1e11891bbd312ccbb77/mutagen-1.47.0.tar.gz", hash = "sha256:719fadef0a978c31b4cf3c956261b3c58b6948b32023078a2117b1de09f0fc99", size = 1274186, upload_time = "2023-09-03T16:33:33.411Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/b0/7a/620f945b96be1f6ee357d211d5bf74ab1b7fe72a9f1525aafbfe3aee6875/mutagen-1.47.0-py3-none-any.whl", hash = "sha256:edd96f50c5907a9539d8e5bba7245f62c9f520aef333d13392a79a4f70aca719", size = 194391, upload_time = "2023-09-03T16:33:29.955Z" }, +] + +[[package]] +name = "pillow" +version = "11.0.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/a5/26/0d95c04c868f6bdb0c447e3ee2de5564411845e36a858cfd63766bc7b563/pillow-11.0.0.tar.gz", hash = "sha256:72bacbaf24ac003fea9bff9837d1eedb6088758d41e100c1552930151f677739", size = 46737780, upload_time = "2024-10-15T14:24:29.672Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/f0/eb/f7e21b113dd48a9c97d364e0915b3988c6a0b6207652f5a92372871b7aa4/pillow-11.0.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:1c1d72714f429a521d8d2d018badc42414c3077eb187a59579f28e4270b4b0fc", size = 3154705, upload_time = "2024-10-15T14:22:15.419Z" }, + { url = "https://files.pythonhosted.org/packages/25/b3/2b54a1d541accebe6bd8b1358b34ceb2c509f51cb7dcda8687362490da5b/pillow-11.0.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:499c3a1b0d6fc8213519e193796eb1a86a1be4b1877d678b30f83fd979811d1a", size = 2979222, upload_time = "2024-10-15T14:22:17.681Z" }, + { url = "https://files.pythonhosted.org/packages/20/12/1a41eddad8265c5c19dda8fb6c269ce15ee25e0b9f8f26286e6202df6693/pillow-11.0.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c8b2351c85d855293a299038e1f89db92a2f35e8d2f783489c6f0b2b5f3fe8a3", size = 4190220, upload_time = "2024-10-15T14:22:19.826Z" }, + { url = "https://files.pythonhosted.org/packages/a9/9b/8a8c4d07d77447b7457164b861d18f5a31ae6418ef5c07f6f878fa09039a/pillow-11.0.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6f4dba50cfa56f910241eb7f883c20f1e7b1d8f7d91c750cd0b318bad443f4d5", size = 4291399, upload_time = "2024-10-15T14:22:22.129Z" }, + { url = "https://files.pythonhosted.org/packages/fc/e4/130c5fab4a54d3991129800dd2801feeb4b118d7630148cd67f0e6269d4c/pillow-11.0.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:5ddbfd761ee00c12ee1be86c9c0683ecf5bb14c9772ddbd782085779a63dd55b", size = 4202709, upload_time = "2024-10-15T14:22:23.953Z" }, + { url = "https://files.pythonhosted.org/packages/39/63/b3fc299528d7df1f678b0666002b37affe6b8751225c3d9c12cf530e73ed/pillow-11.0.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:45c566eb10b8967d71bf1ab8e4a525e5a93519e29ea071459ce517f6b903d7fa", size = 4372556, upload_time = "2024-10-15T14:22:25.706Z" }, + { url = "https://files.pythonhosted.org/packages/c6/a6/694122c55b855b586c26c694937d36bb8d3b09c735ff41b2f315c6e66a10/pillow-11.0.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:b4fd7bd29610a83a8c9b564d457cf5bd92b4e11e79a4ee4716a63c959699b306", size = 4287187, upload_time = "2024-10-15T14:22:27.362Z" }, + { url = "https://files.pythonhosted.org/packages/ba/a9/f9d763e2671a8acd53d29b1e284ca298bc10a595527f6be30233cdb9659d/pillow-11.0.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:cb929ca942d0ec4fac404cbf520ee6cac37bf35be479b970c4ffadf2b6a1cad9", size = 4418468, upload_time = "2024-10-15T14:22:29.093Z" }, + { url = "https://files.pythonhosted.org/packages/6e/0e/b5cbad2621377f11313a94aeb44ca55a9639adabcaaa073597a1925f8c26/pillow-11.0.0-cp311-cp311-win32.whl", hash = "sha256:006bcdd307cc47ba43e924099a038cbf9591062e6c50e570819743f5607404f5", size = 2249249, upload_time = "2024-10-15T14:22:31.268Z" }, + { url = "https://files.pythonhosted.org/packages/dc/83/1470c220a4ff06cd75fc609068f6605e567ea51df70557555c2ab6516b2c/pillow-11.0.0-cp311-cp311-win_amd64.whl", hash = "sha256:52a2d8323a465f84faaba5236567d212c3668f2ab53e1c74c15583cf507a0291", size = 2566769, upload_time = "2024-10-15T14:22:32.974Z" }, + { url = "https://files.pythonhosted.org/packages/52/98/def78c3a23acee2bcdb2e52005fb2810ed54305602ec1bfcfab2bda6f49f/pillow-11.0.0-cp311-cp311-win_arm64.whl", hash = "sha256:16095692a253047fe3ec028e951fa4221a1f3ed3d80c397e83541a3037ff67c9", size = 2254611, upload_time = "2024-10-15T14:22:35.496Z" }, + { url = "https://files.pythonhosted.org/packages/1c/a3/26e606ff0b2daaf120543e537311fa3ae2eb6bf061490e4fea51771540be/pillow-11.0.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:d2c0a187a92a1cb5ef2c8ed5412dd8d4334272617f532d4ad4de31e0495bd923", size = 3147642, upload_time = "2024-10-15T14:22:37.736Z" }, + { url = "https://files.pythonhosted.org/packages/4f/d5/1caabedd8863526a6cfa44ee7a833bd97f945dc1d56824d6d76e11731939/pillow-11.0.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:084a07ef0821cfe4858fe86652fffac8e187b6ae677e9906e192aafcc1b69903", size = 2978999, upload_time = "2024-10-15T14:22:39.654Z" }, + { url = "https://files.pythonhosted.org/packages/d9/ff/5a45000826a1aa1ac6874b3ec5a856474821a1b59d838c4f6ce2ee518fe9/pillow-11.0.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8069c5179902dcdce0be9bfc8235347fdbac249d23bd90514b7a47a72d9fecf4", size = 4196794, upload_time = "2024-10-15T14:22:41.598Z" }, + { url = "https://files.pythonhosted.org/packages/9d/21/84c9f287d17180f26263b5f5c8fb201de0f88b1afddf8a2597a5c9fe787f/pillow-11.0.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f02541ef64077f22bf4924f225c0fd1248c168f86e4b7abdedd87d6ebaceab0f", size = 4300762, upload_time = "2024-10-15T14:22:45.952Z" }, + { url = "https://files.pythonhosted.org/packages/84/39/63fb87cd07cc541438b448b1fed467c4d687ad18aa786a7f8e67b255d1aa/pillow-11.0.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:fcb4621042ac4b7865c179bb972ed0da0218a076dc1820ffc48b1d74c1e37fe9", size = 4210468, upload_time = "2024-10-15T14:22:47.789Z" }, + { url = "https://files.pythonhosted.org/packages/7f/42/6e0f2c2d5c60f499aa29be14f860dd4539de322cd8fb84ee01553493fb4d/pillow-11.0.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:00177a63030d612148e659b55ba99527803288cea7c75fb05766ab7981a8c1b7", size = 4381824, upload_time = "2024-10-15T14:22:49.668Z" }, + { url = "https://files.pythonhosted.org/packages/31/69/1ef0fb9d2f8d2d114db982b78ca4eeb9db9a29f7477821e160b8c1253f67/pillow-11.0.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:8853a3bf12afddfdf15f57c4b02d7ded92c7a75a5d7331d19f4f9572a89c17e6", size = 4296436, upload_time = "2024-10-15T14:22:51.911Z" }, + { url = "https://files.pythonhosted.org/packages/44/ea/dad2818c675c44f6012289a7c4f46068c548768bc6c7f4e8c4ae5bbbc811/pillow-11.0.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:3107c66e43bda25359d5ef446f59c497de2b5ed4c7fdba0894f8d6cf3822dafc", size = 4429714, upload_time = "2024-10-15T14:22:53.967Z" }, + { url = "https://files.pythonhosted.org/packages/af/3a/da80224a6eb15bba7a0dcb2346e2b686bb9bf98378c0b4353cd88e62b171/pillow-11.0.0-cp312-cp312-win32.whl", hash = "sha256:86510e3f5eca0ab87429dd77fafc04693195eec7fd6a137c389c3eeb4cfb77c6", size = 2249631, upload_time = "2024-10-15T14:22:56.404Z" }, + { url = "https://files.pythonhosted.org/packages/57/97/73f756c338c1d86bb802ee88c3cab015ad7ce4b838f8a24f16b676b1ac7c/pillow-11.0.0-cp312-cp312-win_amd64.whl", hash = "sha256:8ec4a89295cd6cd4d1058a5e6aec6bf51e0eaaf9714774e1bfac7cfc9051db47", size = 2567533, upload_time = "2024-10-15T14:22:58.087Z" }, + { url = "https://files.pythonhosted.org/packages/0b/30/2b61876e2722374558b871dfbfcbe4e406626d63f4f6ed92e9c8e24cac37/pillow-11.0.0-cp312-cp312-win_arm64.whl", hash = "sha256:27a7860107500d813fcd203b4ea19b04babe79448268403172782754870dac25", size = 2254890, upload_time = "2024-10-15T14:22:59.918Z" }, + { url = "https://files.pythonhosted.org/packages/63/24/e2e15e392d00fcf4215907465d8ec2a2f23bcec1481a8ebe4ae760459995/pillow-11.0.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:bcd1fb5bb7b07f64c15618c89efcc2cfa3e95f0e3bcdbaf4642509de1942a699", size = 3147300, upload_time = "2024-10-15T14:23:01.855Z" }, + { url = "https://files.pythonhosted.org/packages/43/72/92ad4afaa2afc233dc44184adff289c2e77e8cd916b3ddb72ac69495bda3/pillow-11.0.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:0e038b0745997c7dcaae350d35859c9715c71e92ffb7e0f4a8e8a16732150f38", size = 2978742, upload_time = "2024-10-15T14:23:03.749Z" }, + { url = "https://files.pythonhosted.org/packages/9e/da/c8d69c5bc85d72a8523fe862f05ababdc52c0a755cfe3d362656bb86552b/pillow-11.0.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0ae08bd8ffc41aebf578c2af2f9d8749d91f448b3bfd41d7d9ff573d74f2a6b2", size = 4194349, upload_time = "2024-10-15T14:23:06.055Z" }, + { url = "https://files.pythonhosted.org/packages/cd/e8/686d0caeed6b998351d57796496a70185376ed9c8ec7d99e1d19ad591fc6/pillow-11.0.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d69bfd8ec3219ae71bcde1f942b728903cad25fafe3100ba2258b973bd2bc1b2", size = 4298714, upload_time = "2024-10-15T14:23:07.919Z" }, + { url = "https://files.pythonhosted.org/packages/ec/da/430015cec620d622f06854be67fd2f6721f52fc17fca8ac34b32e2d60739/pillow-11.0.0-cp313-cp313-manylinux_2_28_aarch64.whl", hash = "sha256:61b887f9ddba63ddf62fd02a3ba7add935d053b6dd7d58998c630e6dbade8527", size = 4208514, upload_time = "2024-10-15T14:23:10.19Z" }, + { url = "https://files.pythonhosted.org/packages/44/ae/7e4f6662a9b1cb5f92b9cc9cab8321c381ffbee309210940e57432a4063a/pillow-11.0.0-cp313-cp313-manylinux_2_28_x86_64.whl", hash = "sha256:c6a660307ca9d4867caa8d9ca2c2658ab685de83792d1876274991adec7b93fa", size = 4380055, upload_time = "2024-10-15T14:23:12.08Z" }, + { url = "https://files.pythonhosted.org/packages/74/d5/1a807779ac8a0eeed57f2b92a3c32ea1b696e6140c15bd42eaf908a261cd/pillow-11.0.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:73e3a0200cdda995c7e43dd47436c1548f87a30bb27fb871f352a22ab8dcf45f", size = 4296751, upload_time = "2024-10-15T14:23:13.836Z" }, + { url = "https://files.pythonhosted.org/packages/38/8c/5fa3385163ee7080bc13026d59656267daaaaf3c728c233d530e2c2757c8/pillow-11.0.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:fba162b8872d30fea8c52b258a542c5dfd7b235fb5cb352240c8d63b414013eb", size = 4430378, upload_time = "2024-10-15T14:23:15.735Z" }, + { url = "https://files.pythonhosted.org/packages/ca/1d/ad9c14811133977ff87035bf426875b93097fb50af747793f013979facdb/pillow-11.0.0-cp313-cp313-win32.whl", hash = "sha256:f1b82c27e89fffc6da125d5eb0ca6e68017faf5efc078128cfaa42cf5cb38798", size = 2249588, upload_time = "2024-10-15T14:23:17.905Z" }, + { url = "https://files.pythonhosted.org/packages/fb/01/3755ba287dac715e6afdb333cb1f6d69740a7475220b4637b5ce3d78cec2/pillow-11.0.0-cp313-cp313-win_amd64.whl", hash = "sha256:8ba470552b48e5835f1d23ecb936bb7f71d206f9dfeee64245f30c3270b994de", size = 2567509, upload_time = "2024-10-15T14:23:19.643Z" }, + { url = "https://files.pythonhosted.org/packages/c0/98/2c7d727079b6be1aba82d195767d35fcc2d32204c7a5820f822df5330152/pillow-11.0.0-cp313-cp313-win_arm64.whl", hash = "sha256:846e193e103b41e984ac921b335df59195356ce3f71dcfd155aa79c603873b84", size = 2254791, upload_time = "2024-10-15T14:23:21.601Z" }, + { url = "https://files.pythonhosted.org/packages/eb/38/998b04cc6f474e78b563716b20eecf42a2fa16a84589d23c8898e64b0ffd/pillow-11.0.0-cp313-cp313t-macosx_10_13_x86_64.whl", hash = "sha256:4ad70c4214f67d7466bea6a08061eba35c01b1b89eaa098040a35272a8efb22b", size = 3150854, upload_time = "2024-10-15T14:23:23.91Z" }, + { url = "https://files.pythonhosted.org/packages/13/8e/be23a96292113c6cb26b2aa3c8b3681ec62b44ed5c2bd0b258bd59503d3c/pillow-11.0.0-cp313-cp313t-macosx_11_0_arm64.whl", hash = "sha256:6ec0d5af64f2e3d64a165f490d96368bb5dea8b8f9ad04487f9ab60dc4bb6003", size = 2982369, upload_time = "2024-10-15T14:23:27.184Z" }, + { url = "https://files.pythonhosted.org/packages/97/8a/3db4eaabb7a2ae8203cd3a332a005e4aba00067fc514aaaf3e9721be31f1/pillow-11.0.0-cp313-cp313t-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c809a70e43c7977c4a42aefd62f0131823ebf7dd73556fa5d5950f5b354087e2", size = 4333703, upload_time = "2024-10-15T14:23:28.979Z" }, + { url = "https://files.pythonhosted.org/packages/28/ac/629ffc84ff67b9228fe87a97272ab125bbd4dc462745f35f192d37b822f1/pillow-11.0.0-cp313-cp313t-manylinux_2_28_x86_64.whl", hash = "sha256:4b60c9520f7207aaf2e1d94de026682fc227806c6e1f55bba7606d1c94dd623a", size = 4412550, upload_time = "2024-10-15T14:23:30.846Z" }, + { url = "https://files.pythonhosted.org/packages/d6/07/a505921d36bb2df6868806eaf56ef58699c16c388e378b0dcdb6e5b2fb36/pillow-11.0.0-cp313-cp313t-musllinux_1_2_x86_64.whl", hash = "sha256:1e2688958a840c822279fda0086fec1fdab2f95bf2b717b66871c4ad9859d7e8", size = 4461038, upload_time = "2024-10-15T14:23:32.687Z" }, + { url = "https://files.pythonhosted.org/packages/d6/b9/fb620dd47fc7cc9678af8f8bd8c772034ca4977237049287e99dda360b66/pillow-11.0.0-cp313-cp313t-win32.whl", hash = "sha256:607bbe123c74e272e381a8d1957083a9463401f7bd01287f50521ecb05a313f8", size = 2253197, upload_time = "2024-10-15T14:23:35.309Z" }, + { url = "https://files.pythonhosted.org/packages/df/86/25dde85c06c89d7fc5db17940f07aae0a56ac69aa9ccb5eb0f09798862a8/pillow-11.0.0-cp313-cp313t-win_amd64.whl", hash = "sha256:5c39ed17edea3bc69c743a8dd3e9853b7509625c2462532e62baa0732163a904", size = 2572169, upload_time = "2024-10-15T14:23:37.33Z" }, + { url = "https://files.pythonhosted.org/packages/51/85/9c33f2517add612e17f3381aee7c4072779130c634921a756c97bc29fb49/pillow-11.0.0-cp313-cp313t-win_arm64.whl", hash = "sha256:75acbbeb05b86bc53cbe7b7e6fe00fbcf82ad7c684b3ad82e3d711da9ba287d3", size = 2256828, upload_time = "2024-10-15T14:23:39.826Z" }, +] + +[[package]] +name = "pycparser" +version = "2.22" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/1d/b2/31537cf4b1ca988837256c910a668b553fceb8f069bedc4b1c826024b52c/pycparser-2.22.tar.gz", hash = "sha256:491c8be9c040f5390f5bf44a5b07752bd07f56edf992381b05c701439eec10f6", size = 172736, upload_time = "2024-03-30T13:22:22.564Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/13/a3/a812df4e2dd5696d1f351d58b8fe16a405b234ad2886a0dab9183fb78109/pycparser-2.22-py3-none-any.whl", hash = "sha256:c3702b6d3dd8c7abc1afa565d7e63d53a1d0bd86cdc24edd75470f4de499cfcc", size = 117552, upload_time = "2024-03-30T13:22:20.476Z" }, +] + +[[package]] +name = "pydub" +version = "0.25.1" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/fe/9a/e6bca0eed82db26562c73b5076539a4a08d3cffd19c3cc5913a3e61145fd/pydub-0.25.1.tar.gz", hash = "sha256:980a33ce9949cab2a569606b65674d748ecbca4f0796887fd6f46173a7b0d30f", size = 38326, upload_time = "2021-03-10T02:09:54.659Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a6/53/d78dc063216e62fc55f6b2eebb447f6a4b0a59f55c8406376f76bf959b08/pydub-0.25.1-py2.py3-none-any.whl", hash = "sha256:65617e33033874b59d87db603aa1ed450633288aefead953b30bded59cb599a6", size = 32327, upload_time = "2021-03-10T02:09:53.503Z" }, +] + +[[package]] +name = "pyglet" +version = "2.1.6" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f7/bc/0533ccb30566ee59b540d700dbbf916dafa89132a4d582d0fd1fe158243d/pyglet-2.1.6.tar.gz", hash = "sha256:18483880b1411b39692eaf7756819285797b1aaf9ef63d40eb9f9b5d01c63416", size = 6546705, upload_time = "2025-04-27T01:12:30.995Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/a3/ad/e16f9b56c4a935934341e385753d0d0a2a83b7d320e52906b44f32698feb/pyglet-2.1.6-py3-none-any.whl", hash = "sha256:52ef9e75f3969b6a28bfa5c223e50ff03a05c2baa67bfe00d2a9eec4e831a7c5", size = 983998, upload_time = "2025-04-27T01:12:26.307Z" }, +] + +[[package]] +name = "pymunk" +version = "6.9.0" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "cffi" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/ac/08/1513c868bc2a6bfa22d47acded27f5525c1db10bf1db4fdfa39160991616/pymunk-6.9.0.tar.gz", hash = "sha256:765f7c561a859a1b565bc517a47cc3992d6258e860f9174c533033c218af63c3", size = 3104088, upload_time = "2024-10-13T09:02:40.008Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/6f/ba/34524aac6c57990aa9561c4a949543794e5f7128a0b01537ed061bdaed08/pymunk-6.9.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:536cf3ef9a3add0ea04d83a4c01fe090ff137fb591c3b6fff6e69102384ec5d5", size = 364338, upload_time = "2024-10-13T08:58:08.889Z" }, + { url = "https://files.pythonhosted.org/packages/19/9a/0d4931e3114495c31b600a17f27d5541f2ee35883e7c693199e1ccdf1ab0/pymunk-6.9.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:0e474bb748ded01d96d6eac8e282446baef324b67e0280213b495b1f936c06e7", size = 346937, upload_time = "2024-10-13T08:58:10.604Z" }, + { url = "https://files.pythonhosted.org/packages/61/d0/acd6a6cd8266ac0333792ac3ae36558a58859ca806e0add8f5ea01627b24/pymunk-6.9.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f54bd14512ca5fed0e77f964b1de4e7da1a31386dbf125e33482874d69bb6537", size = 1065273, upload_time = "2024-10-13T08:58:13.012Z" }, + { url = "https://files.pythonhosted.org/packages/5d/d3/2e5763d2eea69e8953782da83fe81a0235650339c22a4f8c65ecdd07cec0/pymunk-6.9.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:cc8d6fe79f77f3ed6e2f33682d355eedb6864684120b845a3501fdf2d3efdcb6", size = 988611, upload_time = "2024-10-13T08:58:15.262Z" }, + { url = "https://files.pythonhosted.org/packages/ac/db/ff2cfa5b87d3e60992b2264a03ffedc738de64d0107b4ce96c623f9098e7/pymunk-6.9.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:70ee413899672c2d7d2ffbecabee133dba49a109867b520d77c829c0d9b3fe92", size = 974971, upload_time = "2024-10-13T08:58:17.706Z" }, + { url = "https://files.pythonhosted.org/packages/ff/44/8fd8677048aa864d91915702522c70c5aaadedfd7cd95000b75d7aabeffd/pymunk-6.9.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:d163dcba2e5814bc5f1274e0ee6ec2a7e06bed8bf0050f30f22b604634bf7dbc", size = 1037097, upload_time = "2024-10-13T08:58:20.264Z" }, + { url = "https://files.pythonhosted.org/packages/19/fc/e6b8bf53255f2012dbdf4a2b063b6c02f8c13ce13b21fdfd84dda64fea80/pymunk-6.9.0-cp311-cp311-win32.whl", hash = "sha256:5d3ae7df3d39afe5b11633496cd464b198d5c62bec69f767f3b61f9fe7f09b98", size = 315321, upload_time = "2024-10-13T08:58:22.475Z" }, + { url = "https://files.pythonhosted.org/packages/bc/3c/925a0193bbcca7203f46fc531f4f0703885c102c1e2c118c8db35816aee3/pymunk-6.9.0-cp311-cp311-win_amd64.whl", hash = "sha256:2db4797ecec3668d51bc112a37192ee1836e236bbacdf5ed12f5a994cf1bae33", size = 366711, upload_time = "2024-10-13T08:58:24.796Z" }, + { url = "https://files.pythonhosted.org/packages/93/96/d8505f4e9661c0e5343db5492895b90b2ada6ec4547fdc7a2df50eb0cdf2/pymunk-6.9.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:02bb0fbbbce2b12c18a033e2cec747e6c4b0db93d2cb9a20f45e569b571ba184", size = 364703, upload_time = "2024-10-13T08:58:27.144Z" }, + { url = "https://files.pythonhosted.org/packages/54/3e/610a2f2b0c6c14038168f6f862148cb245aef867b01906ce18704acafe1c/pymunk-6.9.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:6aae4f93ac686d5e2ec60b01faa1b3722a8ab630464d0c127e16462e7bef6292", size = 347056, upload_time = "2024-10-13T08:58:29.39Z" }, + { url = "https://files.pythonhosted.org/packages/4a/dd/4e12fb3671a6c4f2c0604420f0f15b5402b05c4964bba001088a3d92e3b9/pymunk-6.9.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d7734d13e490e84665b1f03e616270b248d5279ed34e03859267f67868f1b94c", size = 1071014, upload_time = "2024-10-13T08:58:32.274Z" }, + { url = "https://files.pythonhosted.org/packages/91/f8/0618a9204aff896da8b2a9df44179390b178bf00b189851affd4809b1f03/pymunk-6.9.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b05dbfa58d366dea860f7259ca48483922a83620ab6a19effaa74e85a4251966", size = 990358, upload_time = "2024-10-13T08:58:35.295Z" }, + { url = "https://files.pythonhosted.org/packages/af/67/ea2ff4a26b66acad394e4f28e4e316fbe306d34909eca401baae211ca182/pymunk-6.9.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:cb9520c52c043de4b2b1f83979f0d097929f6ff13c8a4059d9d211b98ae25887", size = 976300, upload_time = "2024-10-13T08:58:37.339Z" }, + { url = "https://files.pythonhosted.org/packages/91/d9/a69b268712dceacf227cfff74401e2292b53050383661d456605a1928a84/pymunk-6.9.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:da0e153d321073cd07a48380cfc1b7bd8d40bf4ee1b14a7ede33d90a69ee0452", size = 1042511, upload_time = "2024-10-13T08:58:40.044Z" }, + { url = "https://files.pythonhosted.org/packages/f0/40/21c2a08b027d99f351b75daa36f8a2e2385daba45098078d225811275ff8/pymunk-6.9.0-cp312-cp312-win32.whl", hash = "sha256:8325c9092345764876b1c3855126cb14450dc83dc5b141ff54983a7c77fbae52", size = 315339, upload_time = "2024-10-13T09:01:37.995Z" }, + { url = "https://files.pythonhosted.org/packages/78/b4/0a18c632f96924f969924cc5903689afcaf474d4c472305805dab391b247/pymunk-6.9.0-cp312-cp312-win_amd64.whl", hash = "sha256:13246a79b599c44d174f5619596c62b656d8539797f28bdb2797c4b700c90a33", size = 366671, upload_time = "2024-10-13T09:01:39.965Z" }, + { url = "https://files.pythonhosted.org/packages/b6/5a/c76904d21f3fdb0b713b3a8056622733a0b773f7e55ef974fa4546068cbd/pymunk-6.9.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:5c59e5cf904e148dd0d35cffb7bafe146835042de9280672cafecc3a41caf7a3", size = 364703, upload_time = "2024-10-13T09:01:42.628Z" }, + { url = "https://files.pythonhosted.org/packages/63/b2/378d54b79812da5312b10de272c27aa0ac621498e059aa50eb4eec33ab52/pymunk-6.9.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:4cbc2d37f69d85fedc1097af64edc8f4c43973a13429d51004883cbb9342875e", size = 347058, upload_time = "2024-10-13T09:01:44.529Z" }, + { url = "https://files.pythonhosted.org/packages/ba/a8/c7ea141a1d0e3f5b08ad653f0b5a4ebc0e5854f92bc7049a2a921fbe0d65/pymunk-6.9.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fd64ef76e9e47fda929a2961fe98759ac46b5a7b6126d1ba3e6f04493da6519b", size = 1070851, upload_time = "2024-10-13T09:01:46.638Z" }, + { url = "https://files.pythonhosted.org/packages/10/a2/f40bcc9be90c2af1fe8cf4ba4281385b48d9f5667f03f6834c49aba600fd/pymunk-6.9.0-cp313-cp313-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c568c7402acd5d9a55e3965565ae0a596e4603ba8a7b7b7f0952efadd0e69524", size = 990371, upload_time = "2024-10-13T09:01:49.622Z" }, + { url = "https://files.pythonhosted.org/packages/01/ae/ff7fdf1c8d32ba89d1ccada39b5f7ed66e35420b8d31bdc9af6d5d20ea2f/pymunk-6.9.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:f6cbe3d06e468be11a615d4facecc4a870bf58c1a27c365e655b5a85685ec942", size = 976294, upload_time = "2024-10-13T09:01:52.196Z" }, + { url = "https://files.pythonhosted.org/packages/c6/90/64ef000011f0c930b42354f0d91a07b4bc7f70819ec5b6034b84198bf53f/pymunk-6.9.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:65a9a93a51dbaf1c77efa4d2425549888a1eda9f5c9cd9a5a89b7ca66310968a", size = 1042493, upload_time = "2024-10-13T09:01:55.665Z" }, + { url = "https://files.pythonhosted.org/packages/e5/fb/6516bd5fe565ea51a88308869632dfc896ca6b05b2579b016ffa8047a8ec/pymunk-6.9.0-cp313-cp313-win32.whl", hash = "sha256:a78b37bb360e715657c76caedaf40cdaaf6dab354d497eda481a976cc5cab3d7", size = 315341, upload_time = "2024-10-13T09:01:58.049Z" }, + { url = "https://files.pythonhosted.org/packages/e0/7c/1542df7ffbff70a4523ccb02c9241c9fe4dc24c77b747e2c16fb94891156/pymunk-6.9.0-cp313-cp313-win_amd64.whl", hash = "sha256:d6419e1531df80ff0bb6f1f8215e044f57415514386b7b212dc148919ca629ed", size = 366673, upload_time = "2024-10-13T09:01:59.733Z" }, +] + +[[package]] +name = "pypresence" +version = "4.3.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f4/2e/d110f862720b5e3ba1b0b719657385fc4151929befa2c6981f48360aa480/pypresence-4.3.0.tar.gz", hash = "sha256:a6191a3af33a9667f2a4ef0185577c86b962ee70aa82643c472768a6fed1fbf3", size = 10696, upload_time = "2023-07-08T00:33:53.49Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/1e/40/1d30b30e18f81eb71365681223971a9822a89b3d6ee5269dd2aa955bc228/pypresence-4.3.0-py2.py3-none-any.whl", hash = "sha256:af878c6d49315084f1b108aec86b31915080614d9421d6dd3a44737aba9ff13f", size = 11778, upload_time = "2023-07-08T00:33:52.018Z" }, +] + +[[package]] +name = "pytiled-parser" +version = "2.2.9" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "attrs" }, + { name = "typing-extensions" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/79/62/0d8a2220ee0747522f3b73e4f38bea7c78aefdf707afb86decf26f799fc5/pytiled_parser-2.2.9.tar.gz", hash = "sha256:225269fdd37afcbcd3b76ea3e2cab6b1e742387027106055990db43fd7451ebd", size = 45958, upload_time = "2025-01-23T18:43:30.538Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/d7/f7/6b6c51b50ed8681a31146e5e7ac325b78fe776ff48b1ec8f56d7e4995d72/pytiled_parser-2.2.9-py2.py3-none-any.whl", hash = "sha256:37f73d31950bf4d02ee3bda59f3d6123c55194dc8d8e876821dd2080af5f1f91", size = 44452, upload_time = "2025-01-23T18:43:28.207Z" }, +] + +[[package]] +name = "rapidfuzz" +version = "3.13.0" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/ed/f6/6895abc3a3d056b9698da3199b04c0e56226d530ae44a470edabf8b664f0/rapidfuzz-3.13.0.tar.gz", hash = "sha256:d2eaf3839e52cbcc0accbe9817a67b4b0fcf70aaeb229cfddc1c28061f9ce5d8", size = 57904226, upload_time = "2025-04-03T20:38:51.226Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/87/17/9be9eff5a3c7dfc831c2511262082c6786dca2ce21aa8194eef1cb71d67a/rapidfuzz-3.13.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:d395a5cad0c09c7f096433e5fd4224d83b53298d53499945a9b0e5a971a84f3a", size = 1999453, upload_time = "2025-04-03T20:35:40.804Z" }, + { url = "https://files.pythonhosted.org/packages/75/67/62e57896ecbabe363f027d24cc769d55dd49019e576533ec10e492fcd8a2/rapidfuzz-3.13.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:b7b3eda607a019169f7187328a8d1648fb9a90265087f6903d7ee3a8eee01805", size = 1450881, upload_time = "2025-04-03T20:35:42.734Z" }, + { url = "https://files.pythonhosted.org/packages/96/5c/691c5304857f3476a7b3df99e91efc32428cbe7d25d234e967cc08346c13/rapidfuzz-3.13.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:98e0bfa602e1942d542de077baf15d658bd9d5dcfe9b762aff791724c1c38b70", size = 1422990, upload_time = "2025-04-03T20:35:45.158Z" }, + { url = "https://files.pythonhosted.org/packages/46/81/7a7e78f977496ee2d613154b86b203d373376bcaae5de7bde92f3ad5a192/rapidfuzz-3.13.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bef86df6d59667d9655905b02770a0c776d2853971c0773767d5ef8077acd624", size = 5342309, upload_time = "2025-04-03T20:35:46.952Z" }, + { url = "https://files.pythonhosted.org/packages/51/44/12fdd12a76b190fe94bf38d252bb28ddf0ab7a366b943e792803502901a2/rapidfuzz-3.13.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fedd316c165beed6307bf754dee54d3faca2c47e1f3bcbd67595001dfa11e969", size = 1656881, upload_time = "2025-04-03T20:35:49.954Z" }, + { url = "https://files.pythonhosted.org/packages/27/ae/0d933e660c06fcfb087a0d2492f98322f9348a28b2cc3791a5dbadf6e6fb/rapidfuzz-3.13.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5158da7f2ec02a930be13bac53bb5903527c073c90ee37804090614cab83c29e", size = 1608494, upload_time = "2025-04-03T20:35:51.646Z" }, + { url = "https://files.pythonhosted.org/packages/3d/2c/4b2f8aafdf9400e5599b6ed2f14bc26ca75f5a923571926ccbc998d4246a/rapidfuzz-3.13.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3b6f913ee4618ddb6d6f3e387b76e8ec2fc5efee313a128809fbd44e65c2bbb2", size = 3072160, upload_time = "2025-04-03T20:35:53.472Z" }, + { url = "https://files.pythonhosted.org/packages/60/7d/030d68d9a653c301114101c3003b31ce01cf2c3224034cd26105224cd249/rapidfuzz-3.13.0-cp311-cp311-musllinux_1_2_aarch64.whl", hash = "sha256:d25fdbce6459ccbbbf23b4b044f56fbd1158b97ac50994eaae2a1c0baae78301", size = 2491549, upload_time = "2025-04-03T20:35:55.391Z" }, + { url = "https://files.pythonhosted.org/packages/8e/cd/7040ba538fc6a8ddc8816a05ecf46af9988b46c148ddd7f74fb0fb73d012/rapidfuzz-3.13.0-cp311-cp311-musllinux_1_2_i686.whl", hash = "sha256:25343ccc589a4579fbde832e6a1e27258bfdd7f2eb0f28cb836d6694ab8591fc", size = 7584142, upload_time = "2025-04-03T20:35:57.71Z" }, + { url = "https://files.pythonhosted.org/packages/c1/96/85f7536fbceb0aa92c04a1c37a3fc4fcd4e80649e9ed0fb585382df82edc/rapidfuzz-3.13.0-cp311-cp311-musllinux_1_2_ppc64le.whl", hash = "sha256:a9ad1f37894e3ffb76bbab76256e8a8b789657183870be11aa64e306bb5228fd", size = 2896234, upload_time = "2025-04-03T20:35:59.969Z" }, + { url = "https://files.pythonhosted.org/packages/55/fd/460e78438e7019f2462fe9d4ecc880577ba340df7974c8a4cfe8d8d029df/rapidfuzz-3.13.0-cp311-cp311-musllinux_1_2_s390x.whl", hash = "sha256:5dc71ef23845bb6b62d194c39a97bb30ff171389c9812d83030c1199f319098c", size = 3437420, upload_time = "2025-04-03T20:36:01.91Z" }, + { url = "https://files.pythonhosted.org/packages/cc/df/c3c308a106a0993befd140a414c5ea78789d201cf1dfffb8fd9749718d4f/rapidfuzz-3.13.0-cp311-cp311-musllinux_1_2_x86_64.whl", hash = "sha256:b7f4c65facdb94f44be759bbd9b6dda1fa54d0d6169cdf1a209a5ab97d311a75", size = 4410860, upload_time = "2025-04-03T20:36:04.352Z" }, + { url = "https://files.pythonhosted.org/packages/75/ee/9d4ece247f9b26936cdeaae600e494af587ce9bf8ddc47d88435f05cfd05/rapidfuzz-3.13.0-cp311-cp311-win32.whl", hash = "sha256:b5104b62711565e0ff6deab2a8f5dbf1fbe333c5155abe26d2cfd6f1849b6c87", size = 1843161, upload_time = "2025-04-03T20:36:06.802Z" }, + { url = "https://files.pythonhosted.org/packages/c9/5a/d00e1f63564050a20279015acb29ecaf41646adfacc6ce2e1e450f7f2633/rapidfuzz-3.13.0-cp311-cp311-win_amd64.whl", hash = "sha256:9093cdeb926deb32a4887ebe6910f57fbcdbc9fbfa52252c10b56ef2efb0289f", size = 1629962, upload_time = "2025-04-03T20:36:09.133Z" }, + { url = "https://files.pythonhosted.org/packages/3b/74/0a3de18bc2576b794f41ccd07720b623e840fda219ab57091897f2320fdd/rapidfuzz-3.13.0-cp311-cp311-win_arm64.whl", hash = "sha256:f70f646751b6aa9d05be1fb40372f006cc89d6aad54e9d79ae97bd1f5fce5203", size = 866631, upload_time = "2025-04-03T20:36:11.022Z" }, + { url = "https://files.pythonhosted.org/packages/13/4b/a326f57a4efed8f5505b25102797a58e37ee11d94afd9d9422cb7c76117e/rapidfuzz-3.13.0-cp312-cp312-macosx_10_13_x86_64.whl", hash = "sha256:4a1a6a906ba62f2556372282b1ef37b26bca67e3d2ea957277cfcefc6275cca7", size = 1989501, upload_time = "2025-04-03T20:36:13.43Z" }, + { url = "https://files.pythonhosted.org/packages/b7/53/1f7eb7ee83a06c400089ec7cb841cbd581c2edd7a4b21eb2f31030b88daa/rapidfuzz-3.13.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:2fd0975e015b05c79a97f38883a11236f5a24cca83aa992bd2558ceaa5652b26", size = 1445379, upload_time = "2025-04-03T20:36:16.439Z" }, + { url = "https://files.pythonhosted.org/packages/07/09/de8069a4599cc8e6d194e5fa1782c561151dea7d5e2741767137e2a8c1f0/rapidfuzz-3.13.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5d4e13593d298c50c4f94ce453f757b4b398af3fa0fd2fde693c3e51195b7f69", size = 1405986, upload_time = "2025-04-03T20:36:18.447Z" }, + { url = "https://files.pythonhosted.org/packages/5d/77/d9a90b39c16eca20d70fec4ca377fbe9ea4c0d358c6e4736ab0e0e78aaf6/rapidfuzz-3.13.0-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ed6f416bda1c9133000009d84d9409823eb2358df0950231cc936e4bf784eb97", size = 5310809, upload_time = "2025-04-03T20:36:20.324Z" }, + { url = "https://files.pythonhosted.org/packages/1e/7d/14da291b0d0f22262d19522afaf63bccf39fc027c981233fb2137a57b71f/rapidfuzz-3.13.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1dc82b6ed01acb536b94a43996a94471a218f4d89f3fdd9185ab496de4b2a981", size = 1629394, upload_time = "2025-04-03T20:36:22.256Z" }, + { url = "https://files.pythonhosted.org/packages/b7/e4/79ed7e4fa58f37c0f8b7c0a62361f7089b221fe85738ae2dbcfb815e985a/rapidfuzz-3.13.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:e9d824de871daa6e443b39ff495a884931970d567eb0dfa213d234337343835f", size = 1600544, upload_time = "2025-04-03T20:36:24.207Z" }, + { url = "https://files.pythonhosted.org/packages/4e/20/e62b4d13ba851b0f36370060025de50a264d625f6b4c32899085ed51f980/rapidfuzz-3.13.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2d18228a2390375cf45726ce1af9d36ff3dc1f11dce9775eae1f1b13ac6ec50f", size = 3052796, upload_time = "2025-04-03T20:36:26.279Z" }, + { url = "https://files.pythonhosted.org/packages/cd/8d/55fdf4387dec10aa177fe3df8dbb0d5022224d95f48664a21d6b62a5299d/rapidfuzz-3.13.0-cp312-cp312-musllinux_1_2_aarch64.whl", hash = "sha256:9f5fe634c9482ec5d4a6692afb8c45d370ae86755e5f57aa6c50bfe4ca2bdd87", size = 2464016, upload_time = "2025-04-03T20:36:28.525Z" }, + { url = "https://files.pythonhosted.org/packages/9b/be/0872f6a56c0f473165d3b47d4170fa75263dc5f46985755aa9bf2bbcdea1/rapidfuzz-3.13.0-cp312-cp312-musllinux_1_2_i686.whl", hash = "sha256:694eb531889f71022b2be86f625a4209c4049e74be9ca836919b9e395d5e33b3", size = 7556725, upload_time = "2025-04-03T20:36:30.629Z" }, + { url = "https://files.pythonhosted.org/packages/5d/f3/6c0750e484d885a14840c7a150926f425d524982aca989cdda0bb3bdfa57/rapidfuzz-3.13.0-cp312-cp312-musllinux_1_2_ppc64le.whl", hash = "sha256:11b47b40650e06147dee5e51a9c9ad73bb7b86968b6f7d30e503b9f8dd1292db", size = 2859052, upload_time = "2025-04-03T20:36:32.836Z" }, + { url = "https://files.pythonhosted.org/packages/6f/98/5a3a14701b5eb330f444f7883c9840b43fb29c575e292e09c90a270a6e07/rapidfuzz-3.13.0-cp312-cp312-musllinux_1_2_s390x.whl", hash = "sha256:98b8107ff14f5af0243f27d236bcc6e1ef8e7e3b3c25df114e91e3a99572da73", size = 3390219, upload_time = "2025-04-03T20:36:35.062Z" }, + { url = "https://files.pythonhosted.org/packages/e9/7d/f4642eaaeb474b19974332f2a58471803448be843033e5740965775760a5/rapidfuzz-3.13.0-cp312-cp312-musllinux_1_2_x86_64.whl", hash = "sha256:b836f486dba0aceb2551e838ff3f514a38ee72b015364f739e526d720fdb823a", size = 4377924, upload_time = "2025-04-03T20:36:37.363Z" }, + { url = "https://files.pythonhosted.org/packages/8e/83/fa33f61796731891c3e045d0cbca4436a5c436a170e7f04d42c2423652c3/rapidfuzz-3.13.0-cp312-cp312-win32.whl", hash = "sha256:4671ee300d1818d7bdfd8fa0608580d7778ba701817216f0c17fb29e6b972514", size = 1823915, upload_time = "2025-04-03T20:36:39.451Z" }, + { url = "https://files.pythonhosted.org/packages/03/25/5ee7ab6841ca668567d0897905eebc79c76f6297b73bf05957be887e9c74/rapidfuzz-3.13.0-cp312-cp312-win_amd64.whl", hash = "sha256:6e2065f68fb1d0bf65adc289c1bdc45ba7e464e406b319d67bb54441a1b9da9e", size = 1616985, upload_time = "2025-04-03T20:36:41.631Z" }, + { url = "https://files.pythonhosted.org/packages/76/5e/3f0fb88db396cb692aefd631e4805854e02120a2382723b90dcae720bcc6/rapidfuzz-3.13.0-cp312-cp312-win_arm64.whl", hash = "sha256:65cc97c2fc2c2fe23586599686f3b1ceeedeca8e598cfcc1b7e56dc8ca7e2aa7", size = 860116, upload_time = "2025-04-03T20:36:43.915Z" }, + { url = "https://files.pythonhosted.org/packages/0a/76/606e71e4227790750f1646f3c5c873e18d6cfeb6f9a77b2b8c4dec8f0f66/rapidfuzz-3.13.0-cp313-cp313-macosx_10_13_x86_64.whl", hash = "sha256:09e908064d3684c541d312bd4c7b05acb99a2c764f6231bd507d4b4b65226c23", size = 1982282, upload_time = "2025-04-03T20:36:46.149Z" }, + { url = "https://files.pythonhosted.org/packages/0a/f5/d0b48c6b902607a59fd5932a54e3518dae8223814db8349b0176e6e9444b/rapidfuzz-3.13.0-cp313-cp313-macosx_11_0_arm64.whl", hash = "sha256:57c390336cb50d5d3bfb0cfe1467478a15733703af61f6dffb14b1cd312a6fae", size = 1439274, upload_time = "2025-04-03T20:36:48.323Z" }, + { url = "https://files.pythonhosted.org/packages/59/cf/c3ac8c80d8ced6c1f99b5d9674d397ce5d0e9d0939d788d67c010e19c65f/rapidfuzz-3.13.0-cp313-cp313-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0da54aa8547b3c2c188db3d1c7eb4d1bb6dd80baa8cdaeaec3d1da3346ec9caa", size = 1399854, upload_time = "2025-04-03T20:36:50.294Z" }, + { url = "https://files.pythonhosted.org/packages/09/5d/ca8698e452b349c8313faf07bfa84e7d1c2d2edf7ccc67bcfc49bee1259a/rapidfuzz-3.13.0-cp313-cp313-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:df8e8c21e67afb9d7fbe18f42c6111fe155e801ab103c81109a61312927cc611", size = 5308962, upload_time = "2025-04-03T20:36:52.421Z" }, + { url = "https://files.pythonhosted.org/packages/66/0a/bebada332854e78e68f3d6c05226b23faca79d71362509dbcf7b002e33b7/rapidfuzz-3.13.0-cp313-cp313-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:461fd13250a2adf8e90ca9a0e1e166515cbcaa5e9c3b1f37545cbbeff9e77f6b", size = 1625016, upload_time = "2025-04-03T20:36:54.639Z" }, + { url = "https://files.pythonhosted.org/packages/de/0c/9e58d4887b86d7121d1c519f7050d1be5eb189d8a8075f5417df6492b4f5/rapidfuzz-3.13.0-cp313-cp313-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:c2b3dd5d206a12deca16870acc0d6e5036abeb70e3cad6549c294eff15591527", size = 1600414, upload_time = "2025-04-03T20:36:56.669Z" }, + { url = "https://files.pythonhosted.org/packages/9b/df/6096bc669c1311568840bdcbb5a893edc972d1c8d2b4b4325c21d54da5b1/rapidfuzz-3.13.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1343d745fbf4688e412d8f398c6e6d6f269db99a54456873f232ba2e7aeb4939", size = 3053179, upload_time = "2025-04-03T20:36:59.366Z" }, + { url = "https://files.pythonhosted.org/packages/f9/46/5179c583b75fce3e65a5cd79a3561bd19abd54518cb7c483a89b284bf2b9/rapidfuzz-3.13.0-cp313-cp313-musllinux_1_2_aarch64.whl", hash = "sha256:b1b065f370d54551dcc785c6f9eeb5bd517ae14c983d2784c064b3aa525896df", size = 2456856, upload_time = "2025-04-03T20:37:01.708Z" }, + { url = "https://files.pythonhosted.org/packages/6b/64/e9804212e3286d027ac35bbb66603c9456c2bce23f823b67d2f5cabc05c1/rapidfuzz-3.13.0-cp313-cp313-musllinux_1_2_i686.whl", hash = "sha256:11b125d8edd67e767b2295eac6eb9afe0b1cdc82ea3d4b9257da4b8e06077798", size = 7567107, upload_time = "2025-04-03T20:37:04.521Z" }, + { url = "https://files.pythonhosted.org/packages/8a/f2/7d69e7bf4daec62769b11757ffc31f69afb3ce248947aadbb109fefd9f65/rapidfuzz-3.13.0-cp313-cp313-musllinux_1_2_ppc64le.whl", hash = "sha256:c33f9c841630b2bb7e69a3fb5c84a854075bb812c47620978bddc591f764da3d", size = 2854192, upload_time = "2025-04-03T20:37:06.905Z" }, + { url = "https://files.pythonhosted.org/packages/05/21/ab4ad7d7d0f653e6fe2e4ccf11d0245092bef94cdff587a21e534e57bda8/rapidfuzz-3.13.0-cp313-cp313-musllinux_1_2_s390x.whl", hash = "sha256:ae4574cb66cf1e85d32bb7e9ec45af5409c5b3970b7ceb8dea90168024127566", size = 3398876, upload_time = "2025-04-03T20:37:09.692Z" }, + { url = "https://files.pythonhosted.org/packages/0f/a8/45bba94c2489cb1ee0130dcb46e1df4fa2c2b25269e21ffd15240a80322b/rapidfuzz-3.13.0-cp313-cp313-musllinux_1_2_x86_64.whl", hash = "sha256:e05752418b24bbd411841b256344c26f57da1148c5509e34ea39c7eb5099ab72", size = 4377077, upload_time = "2025-04-03T20:37:11.929Z" }, + { url = "https://files.pythonhosted.org/packages/0c/f3/5e0c6ae452cbb74e5436d3445467447e8c32f3021f48f93f15934b8cffc2/rapidfuzz-3.13.0-cp313-cp313-win32.whl", hash = "sha256:0e1d08cb884805a543f2de1f6744069495ef527e279e05370dd7c83416af83f8", size = 1822066, upload_time = "2025-04-03T20:37:14.425Z" }, + { url = "https://files.pythonhosted.org/packages/96/e3/a98c25c4f74051df4dcf2f393176b8663bfd93c7afc6692c84e96de147a2/rapidfuzz-3.13.0-cp313-cp313-win_amd64.whl", hash = "sha256:9a7c6232be5f809cd39da30ee5d24e6cadd919831e6020ec6c2391f4c3bc9264", size = 1615100, upload_time = "2025-04-03T20:37:16.611Z" }, + { url = "https://files.pythonhosted.org/packages/60/b1/05cd5e697c00cd46d7791915f571b38c8531f714832eff2c5e34537c49ee/rapidfuzz-3.13.0-cp313-cp313-win_arm64.whl", hash = "sha256:3f32f15bacd1838c929b35c84b43618481e1b3d7a61b5ed2db0291b70ae88b53", size = 858976, upload_time = "2025-04-03T20:37:19.336Z" }, + { url = "https://files.pythonhosted.org/packages/88/df/6060c5a9c879b302bd47a73fc012d0db37abf6544c57591bcbc3459673bd/rapidfuzz-3.13.0-pp311-pypy311_pp73-macosx_10_15_x86_64.whl", hash = "sha256:1ba007f4d35a45ee68656b2eb83b8715e11d0f90e5b9f02d615a8a321ff00c27", size = 1905935, upload_time = "2025-04-03T20:38:18.07Z" }, + { url = "https://files.pythonhosted.org/packages/a2/6c/a0b819b829e20525ef1bd58fc776fb8d07a0c38d819e63ba2b7c311a2ed4/rapidfuzz-3.13.0-pp311-pypy311_pp73-macosx_11_0_arm64.whl", hash = "sha256:d7a217310429b43be95b3b8ad7f8fc41aba341109dc91e978cd7c703f928c58f", size = 1383714, upload_time = "2025-04-03T20:38:20.628Z" }, + { url = "https://files.pythonhosted.org/packages/6a/c1/3da3466cc8a9bfb9cd345ad221fac311143b6a9664b5af4adb95b5e6ce01/rapidfuzz-3.13.0-pp311-pypy311_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:558bf526bcd777de32b7885790a95a9548ffdcce68f704a81207be4a286c1095", size = 1367329, upload_time = "2025-04-03T20:38:23.01Z" }, + { url = "https://files.pythonhosted.org/packages/da/f0/9f2a9043bfc4e66da256b15d728c5fc2d865edf0028824337f5edac36783/rapidfuzz-3.13.0-pp311-pypy311_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:202a87760f5145140d56153b193a797ae9338f7939eb16652dd7ff96f8faf64c", size = 5251057, upload_time = "2025-04-03T20:38:25.52Z" }, + { url = "https://files.pythonhosted.org/packages/6a/ff/af2cb1d8acf9777d52487af5c6b34ce9d13381a753f991d95ecaca813407/rapidfuzz-3.13.0-pp311-pypy311_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cfcccc08f671646ccb1e413c773bb92e7bba789e3a1796fd49d23c12539fe2e4", size = 2992401, upload_time = "2025-04-03T20:38:28.196Z" }, + { url = "https://files.pythonhosted.org/packages/c1/c5/c243b05a15a27b946180db0d1e4c999bef3f4221505dff9748f1f6c917be/rapidfuzz-3.13.0-pp311-pypy311_pp73-win_amd64.whl", hash = "sha256:1f219f1e3c3194d7a7de222f54450ce12bc907862ff9a8962d83061c1f923c86", size = 1553782, upload_time = "2025-04-03T20:38:30.778Z" }, +] + +[[package]] +name = "thefuzz" +version = "0.22.1" +source = { registry = "https://pypi.org/simple" } +dependencies = [ + { name = "rapidfuzz" }, +] +sdist = { url = "https://files.pythonhosted.org/packages/81/4b/d3eb25831590d6d7d38c2f2e3561d3ba41d490dc89cd91d9e65e7c812508/thefuzz-0.22.1.tar.gz", hash = "sha256:7138039a7ecf540da323792d8592ef9902b1d79eb78c147d4f20664de79f3680", size = 19993, upload_time = "2024-01-19T19:18:23.135Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/82/4f/1695e70ceb3604f19eda9908e289c687ea81c4fecef4d90a9d1d0f2f7ae9/thefuzz-0.22.1-py3-none-any.whl", hash = "sha256:59729b33556850b90e1093c4cf9e618af6f2e4c985df193fdf3c5b5cf02ca481", size = 8245, upload_time = "2024-01-19T19:18:20.362Z" }, +] + +[[package]] +name = "typing-extensions" +version = "4.13.2" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/f6/37/23083fcd6e35492953e8d2aaaa68b860eb422b34627b13f2ce3eb6106061/typing_extensions-4.13.2.tar.gz", hash = "sha256:e6c81219bd689f51865d9e372991c540bda33a0379d5573cddb9a3a23f7caaef", size = 106967, upload_time = "2025-04-10T14:19:05.416Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/8b/54/b1ae86c0973cc6f0210b53d508ca3641fb6d0c56823f288d108bc7ab3cc8/typing_extensions-4.13.2-py3-none-any.whl", hash = "sha256:a439e7c04b49fec3e5d3e2beaa21755cadbbdc391694e28ccdd36ca4a1408f8c", size = 45806, upload_time = "2025-04-10T14:19:03.967Z" }, +] + +[[package]] +name = "yt-dlp" +version = "2025.4.30" +source = { registry = "https://pypi.org/simple" } +sdist = { url = "https://files.pythonhosted.org/packages/75/ca/1d1a33dec2107463f59bc4b448fcf43718d86a36b6150e8a0cfd1a96a893/yt_dlp-2025.4.30.tar.gz", hash = "sha256:d01367d0c3ae94e35cb1e2eccb7a7c70e181c4ca448f4ee2374f26489d263603", size = 2981094, upload_time = "2025-04-30T23:31:17.679Z" } +wheels = [ + { url = "https://files.pythonhosted.org/packages/ea/18/bbccb18661a853c3db947152439229b9bf686fe0ab2cc7848ab3a24f2ad2/yt_dlp-2025.4.30-py3-none-any.whl", hash = "sha256:53cd82bf13f12a1fe9c564b0004e001156b153c9247fa3cef14d1400ab359150", size = 3239476, upload_time = "2025-04-30T23:31:14.997Z" }, +]