diff --git a/LICENSE.md b/LICENSE.md new file mode 100644 index 0000000..175443c --- /dev/null +++ b/LICENSE.md @@ -0,0 +1,595 @@ +GNU General Public License +========================== + +_Version 3, 29 June 2007_ +_Copyright © 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..14d175b --- /dev/null +++ b/README.md @@ -0,0 +1,41 @@ +# FFX + +## Installation + +per https: + +```sh +pip install https:////ffx.git@ +``` + +per git: + +```sh +pip install git+ssh://@//ffx.git@ +``` + +## Version history + +### 0.1.1 + +Bugfixes, TMBD identify shows + +### 0.1.2 + +Bugfixes + +### 0.1.3 + +Subtitle file imports + +### 0.2.0 + +Tests, Config-File + +### 0.2.1 + +Signature, Tags cleaning, Bugfixes, Refactoring + +### 0.2.2 + +CLI-Overrides diff --git a/bin/.ipynb_checkpoints/check-checkpoint.py b/bin/.ipynb_checkpoints/check-checkpoint.py deleted file mode 100644 index 9759724..0000000 --- a/bin/.ipynb_checkpoints/check-checkpoint.py +++ /dev/null @@ -1,32 +0,0 @@ -import os - -from ffx.pattern_controller import PatternController - -from ffx.model.show import Base -from sqlalchemy import create_engine, Column, Integer, String, ForeignKey -from sqlalchemy.orm import relationship, sessionmaker, Mapped, backref - -filename = 'Boruto.Naruto.Next.Generations.S01E256.GerEngSub.AAC.1080p.WebDL.x264-Tanuki.mkv' - - - -# Data 'input' variable -context = {} - -# Initialize DB -homeDir = os.path.expanduser("~") -ffxVarDir = os.path.join(homeDir, '.local', 'var', 'ffx') -if not os.path.exists(ffxVarDir): - os.makedirs(ffxVarDir) - -context['database_url'] = f"sqlite:///{os.path.join(ffxVarDir, 'ffx.db')}" -context['database_engine'] = create_engine(context['database_url']) -context['database_session'] = sessionmaker(bind=context['database_engine']) - -Base.metadata.create_all(context['database_engine']) - - -pc = PatternController(context) - - -print(pc.matchFilename(filename)) diff --git a/bin/.ipynb_checkpoints/ffx-checkpoint.py b/bin/.ipynb_checkpoints/ffx-checkpoint.py deleted file mode 100755 index 88fc807..0000000 --- a/bin/.ipynb_checkpoints/ffx-checkpoint.py +++ /dev/null @@ -1,554 +0,0 @@ -#! /usr/bin/python3 - -import os, sys, subprocess, json, click, time, re - -from textual.app import App, ComposeResult -from textual.screen import Screen -from textual.widgets import Header, Footer, Placeholder, Label - - -VERSION='0.1.0' - -DEFAULT_VIDEO_ENCODER = 'vp9' - -DEFAULT_QUALITY = 23 - -DEFAULT_AV1_PRESET = 5 - -DEFAULT_LABEL='output' -DEFAULT_FILE_SUFFIX = 'webm' - -DEFAULT_STEREO_BANDWIDTH = "128" -DEFAULT_AC3_BANDWIDTH = "256" -DEFAULT_DTS_BANDWIDTH = "320" - -DEFAULT_CROP_START = 60 -DEFAULT_CROP_LENGTH = 180 - -TEMP_FILE_NAME = "ffmpeg2pass-0.log" - - -MKVMERGE_METADATA_KEYS = ['BPS', - 'NUMBER_OF_FRAMES', - 'NUMBER_OF_BYTES', - '_STATISTICS_WRITING_APP', - '_STATISTICS_WRITING_DATE_UTC', - '_STATISTICS_TAGS'] - -FILE_EXTENSIONS = ['mkv', 'mp4', 'avi', 'flv', 'webm'] - - -COMMAND_TOKENS = ['ffmpeg', '-y', '-i'] -NULL_TOKENS = ['-f', 'null', '/dev/null'] - -STREAM_TYPE_VIDEO = 'video' -STREAM_TYPE_AUDIO = 'audio' -STREAM_TYPE_SUBTITLE = 'subtitle' - -STREAM_LAYOUT_6_1 = '6.1' -STREAM_LAYOUT_5_1 = '5.1(side)' -STREAM_LAYOUT_STEREO = 'stereo' -STREAM_LAYOUT_6CH = '6ch' - -SEASON_EPISODE_INDICATOR_MATCH = '([sS][0-9]+)([eE][0-9]+)' -SEASON_INDICATOR_MATCH = '([sS][0-9]+)' -EPISODE_INDICATOR_MATCH = '([eE][0-9]+)' - - -class DashboardScreen(Screen): - - def __init__(self): - super().__init__() - - context = self.app.getContext() - context['dashboard'] = 'dashboard' - - def compose(self) -> ComposeResult: - yield Header(show_clock=True) - yield Placeholder("Dashboard Screen") - yield Footer() - -class WarningScreen(Screen): - def __init__(self): - super().__init__() - context = self.app.getContext() - def compose(self) -> ComposeResult: - yield Label("Warning! This file is not compliant to the defined source schema!") - yield Footer() - - -class SettingsScreen(Screen): - def __init__(self): - super().__init__() - context = self.app.getContext() - def compose(self) -> ComposeResult: - yield Placeholder("Settings Screen") - yield Footer() - - -class HelpScreen(Screen): - def __init__(self): - super().__init__() - context = self.app.getContext() - def compose(self) -> ComposeResult: - yield Placeholder("Help Screen") - yield Footer() - - -class ModesApp(App): - - BINDINGS = [ - ("q", "quit()", "Quit"), - # ("d", "switch_mode('dashboard')", "Dashboard"), - # ("s", "switch_mode('settings')", "Settings"), - # ("h", "switch_mode('help')", "Help"), - ] - - MODES = { - "warning": WarningScreen, - "dashboard": DashboardScreen, - "settings": SettingsScreen, - "help": HelpScreen, - } - - - def __init__(self, context = {}): - super().__init__() - self.context = context - - def on_mount(self) -> None: - self.switch_mode("warning") - - def getContext(self): - return self.context - - -def executeProcess(commandSequence): - - process = subprocess.Popen(commandSequence, stdout=subprocess.PIPE, stderr=subprocess.PIPE) - - output, error = process.communicate() - - return output - - - -#[{'index': 0, 'codec_name': 'vp9', 'codec_long_name': 'Google VP9', 'profile': 'Profile 0', 'codec_type': 'video', 'codec_tag_string': '[0][0][0][0]', 'codec_tag': '0x0000', 'width': 1920, 'height': 1080, 'coded_width': 1920, 'coded_height': 1080, 'closed_captions': 0, 'film_grain': 0, 'has_b_frames': 0, 'sample_aspect_ratio': '1:1', 'display_aspect_ratio': '16:9', 'pix_fmt': 'yuv420p', 'level': -99, 'color_range': 'tv', 'chroma_location': 'left', 'field_order': 'progressive', 'refs': 1, 'r_frame_rate': '24000/1001', 'avg_frame_rate': '24000/1001', 'time_base': '1/1000', 'start_pts': 0, 'start_time': '0.000000', 'disposition': {'default': 1, 'dub': 0, 'original': 0, 'comment': 0, 'lyrics': 0, 'karaoke': 0, 'forced': 0, 'hearing_impaired': 0, 'visual_impaired': 0, 'clean_effects': 0, 'attached_pic': 0, 'timed_thumbnails': 0, 'non_diegetic': 0, 'captions': 0, 'descriptions': 0, 'metadata': 0, 'dependent': 0, 'still_image': 0}, 'tags': {'BPS': '7974017', 'NUMBER_OF_FRAMES': '34382', 'NUMBER_OF_BYTES': '1429358655', '_STATISTICS_WRITING_APP': "mkvmerge v63.0.0 ('Everything') 64-bit", '_STATISTICS_WRITING_DATE_UTC': '2023-10-07 13:59:46', '_STATISTICS_TAGS': 'BPS DURATION NUMBER_OF_FRAMES NUMBER_OF_BYTES', 'ENCODER': 'Lavc61.3.100 libvpx-vp9', 'DURATION': '00:23:54.016000000'}}] -#[{'index': 1, 'codec_name': 'opus', 'codec_long_name': 'Opus (Opus Interactive Audio Codec)', 'codec_type': 'audio', 'codec_tag_string': '[0][0][0][0]', 'codec_tag': '0x0000', 'sample_fmt': 'fltp', 'sample_rate': '48000', 'channels': 2, 'channel_layout': 'stereo', 'bits_per_sample': 0, 'initial_padding': 312, 'r_frame_rate': '0/0', 'avg_frame_rate': '0/0', 'time_base': '1/1000', 'start_pts': -7, 'start_time': '-0.007000', 'extradata_size': 19, 'disposition': {'default': 1, 'dub': 0, 'original': 0, 'comment': 0, 'lyrics': 0, 'karaoke': 0, 'forced': 0, 'hearing_impaired': 0, 'visual_impaired': 0, 'clean_effects': 0, 'attached_pic': 0, 'timed_thumbnails': 0, 'non_diegetic': 0, 'captions': 0, 'descriptions': 0, 'metadata': 0, 'dependent': 0, 'still_image': 0}, 'tags': {'language': 'jpn', 'title': 'Japanisch', 'BPS': '128000', 'NUMBER_OF_FRAMES': '61763', 'NUMBER_OF_BYTES': '22946145', '_STATISTICS_WRITING_APP': "mkvmerge v63.0.0 ('Everything') 64-bit", '_STATISTICS_WRITING_DATE_UTC': '2023-10-07 13:59:46', '_STATISTICS_TAGS': 'BPS DURATION NUMBER_OF_FRAMES NUMBER_OF_BYTES', 'ENCODER': 'Lavc61.3.100 libopus', 'DURATION': '00:23:54.141000000'}}] - -#[{'index': 2, 'codec_name': 'webvtt', 'codec_long_name': 'WebVTT subtitle', 'codec_type': 'subtitle', 'codec_tag_string': '[0][0][0][0]', 'codec_tag': '0x0000', 'r_frame_rate': '0/0', 'avg_frame_rate': '0/0', 'time_base': '1/1000', 'start_pts': -7, 'start_time': '-0.007000', 'duration_ts': 1434141, 'duration': '1434.141000', 'disposition': {'default': 1, 'dub': 0, 'original': 0, 'comment': 0, 'lyrics': 0, 'karaoke': 0, 'forced': 0, 'hearing_impaired': 0, 'visual_impaired': 0, 'clean_effects': 0, 'attached_pic': 0, 'timed_thumbnails': 0, 'non_diegetic': 0, 'captions': 0, 'descriptions': 0, 'metadata': 0, 'dependent': 0, 'still_image': 0}, 'tags': {'language': 'ger', 'title': 'Deutsch [Full]', 'BPS': '118', 'NUMBER_OF_FRAMES': '300', 'NUMBER_OF_BYTES': '21128', '_STATISTICS_WRITING_APP': "mkvmerge v63.0.0 ('Everything') 64-bit", '_STATISTICS_WRITING_DATE_UTC': '2023-10-07 13:59:46', '_STATISTICS_TAGS': 'BPS DURATION NUMBER_OF_FRAMES NUMBER_OF_BYTES', 'ENCODER': 'Lavc61.3.100 webvtt', 'DURATION': '00:23:54.010000000'}}, {'index': 3, 'codec_name': 'webvtt', 'codec_long_name': 'WebVTT subtitle', 'codec_type': 'subtitle', 'codec_tag_string': '[0][0][0][0]', 'codec_tag': '0x0000', 'r_frame_rate': '0/0', 'avg_frame_rate': '0/0', 'time_base': '1/1000', 'start_pts': -7, 'start_time': '-0.007000', 'duration_ts': 1434141, 'duration': '1434.141000', 'disposition': {'default': 0, 'dub': 0, 'original': 0, 'comment': 0, 'lyrics': 0, 'karaoke': 0, 'forced': 0, 'hearing_impaired': 0, 'visual_impaired': 0, 'clean_effects': 0, 'attached_pic': 0, 'timed_thumbnails': 0, 'non_diegetic': 0, 'captions': 0, 'descriptions': 0, 'metadata': 0, 'dependent': 0, 'still_image': 0}, 'tags': {'language': 'eng', 'title': 'Englisch [Full]', 'BPS': '101', 'NUMBER_OF_FRAMES': '276', 'NUMBER_OF_BYTES': '16980', '_STATISTICS_WRITING_APP': "mkvmerge v63.0.0 ('Everything') 64-bit", '_STATISTICS_WRITING_DATE_UTC': '2023-10-07 13:59:46', '_STATISTICS_TAGS': 'BPS DURATION NUMBER_OF_FRAMES NUMBER_OF_BYTES', 'ENCODER': 'Lavc61.3.100 webvtt', 'DURATION': '00:23:53.230000000'}}] - - - -def getStreamDescriptor(filename): - - ffprobeOutput = executeProcess(["ffprobe", - "-show_streams", - "-of", "json", - filename]) - - streamData = json.loads(ffprobeOutput)['streams'] - - descriptor = [] - - i = 0 - for d in [s for s in streamData if s['codec_type'] == STREAM_TYPE_VIDEO]: - descriptor.append({ - 'index': d['index'], - 'sub_index': i, - 'type': STREAM_TYPE_VIDEO, - 'codec': d['codec_name'] - }) - i += 1 - - i = 0 - for d in [s for s in streamData if s['codec_type'] == STREAM_TYPE_AUDIO]: - - streamDescriptor = { - 'index': d['index'], - 'sub_index': i, - 'type': STREAM_TYPE_AUDIO, - 'codec': d['codec_name'], - 'channels': d['channels'] - } - - if 'channel_layout' in d.keys(): - streamDescriptor['layout'] = d['channel_layout'] - elif d['channels'] == 6: - streamDescriptor['layout'] = STREAM_LAYOUT_6CH - else: - streamDescriptor['layout'] = 'undefined' - - descriptor.append(streamDescriptor) - i += 1 - - i = 0 - for d in [s for s in streamData if s['codec_type'] == STREAM_TYPE_SUBTITLE]: - descriptor.append({ - 'index': d['index'], - 'sub_index': i, - 'type': STREAM_TYPE_SUBTITLE, - 'codec': d['codec_name'] - }) - i += 1 - - return descriptor - - - -def generateAV1Tokens(q, p): - - return ['-c:v:0', 'libsvtav1', - '-svtav1-params', f"crf={q}:preset={p}:tune=0:enable-overlays=1:scd=1:scm=0", - '-pix_fmt', 'yuv420p10le'] - -def generateVP9Pass1Tokens(q): - - return ['-c:v:0', 'libvpx-vp9', - '-row-mt', '1', - '-crf', str(q), - '-pass', '1', - '-speed', '4', - '-frame-parallel', '0', - '-g', '9999', - '-aq-mode', '0'] - -def generateVP9Pass2Tokens(q): - - return ['-c:v:0', 'libvpx-vp9', - '-row-mt', '1', - '-crf', str(q), - '-pass', '2', - '-frame-parallel', '0', - '-g', '9999', - '-aq-mode', '0', - '-auto-alt-ref', '1', - '-lag-in-frames', '25'] - - -def generateCropTokens(start, length): - - return ['-ss', str(start), '-t', str(length)] - - -def generateDenoiseTokens(spatial=5, patch=7, research=7, hw=False): - filterName = 'nlmeans_opencl' if hw else 'nlmeans' - return ['-vf', f"{filterName}=s={spatial}:p={patch}:r={research}"] - - -def generateOutputTokens(f, suffix, q=None): - - if q is None: - return ['-f', 'webm', f"{f}.{suffix}"] - else: - return ['-f', 'webm', f"{f}_q{q}.{suffix}"] - - -# preset = DEFAULT_AV1_PRESET -# presetTokens = [p for p in sys.argv if p.startswith('p=')] -# if presetTokens: -# preset = int(presetTokens[0].split('=')[1]) - -# cropStart = '' -# cropLength = '' -# cropTokens = [c for c in sys.argv if c.startswith('crop')] -# if cropTokens: -# if '=' in cropTokens[0]: -# cropString = cropTokens[0].split('=')[1] -# cropStart, cropLength = cropString.split(',') -# else: -# cropStart = 60 -# cropLength = 180 -# -# denoiseTokens = [d for d in sys.argv if d.startswith('denoise')] -# - -# for aStream in audioStreams: -# if 'channel_layout' in aStream: -# print(f"audio stream: {aStream['channel_layout']}") #channel_layout -# else: -# print(f"unknown audio stream with {aStream['channels']} channels") #channel_layout - -def generateAudioTokens(context, index, layout): - - if layout == STREAM_LAYOUT_6_1: - return [f"-c:a:{index}", - 'libopus', - f"-filter:a:{index}", - 'channelmap=channel_layout=6.1', - f"-b:a:{index}", - context['bitrates']['dts']] - - elif layout == STREAM_LAYOUT_5_1: - return [f"-c:a:{index}", - 'libopus', - f"-filter:a:{index}", - "channelmap=FL-FL|FR-FR|FC-FC|LFE-LFE|SL-BL|SR-BR:5.1", - f"-b:a:{index}", - context['bitrates']['ac3']] - - elif layout == STREAM_LAYOUT_STEREO: - return [f"-c:a:{index}", - 'libopus', - f"-b:a:{index}", - context['bitrates']['stereo']] - - elif layout == STREAM_LAYOUT_6CH: - return [f"-c:a:{index}", - 'libopus', - f"-filter:a:{index}", - "channelmap=FL-FL|FR-FR|FC-FC|LFE-LFE|SL-BL|SR-BR:5.1", - f"-b:a:{index}", - context['bitrates']['ac3']] - else: - return [] - - -def generateClearTokens(streams): - clearTokens = [] - for s in streams: - for k in MKVMERGE_METADATA_KEYS: - clearTokens += [f"-metadata:s:{s['type'][0]}:{s['sub_index']}", f"{k}="] - return clearTokens - - -@click.group() -@click.pass_context -def ffx(ctx): - """FFX""" - ctx.obj = {} - pass - - -# Define a subcommand -@ffx.command() -def version(): - click.echo(VERSION) - - -# Another subcommand -@ffx.command() -def help(): - click.echo(f"ffx {VERSION}\n") - click.echo(f"Usage: ffx [input file] [output file] [vp9|av1] [q=[nn[,nn,...]]] [p=nn] [a=nnn[k]] [ac3=nnn[k]] [dts=nnn[k]] [crop]") - - -@click.argument('filename', nargs=1) -@ffx.command() -def streams(filename): - for d in getStreamDescriptor(filename): - click.echo(f"{d['codec']}{' (' + str(d['channels']) + ')' if d['type'] == 'audio' else ''}") - - - -@ffx.command() -@click.pass_context - -@click.argument('paths', nargs=-1) -@click.option('-l', '--label', type=str, default=DEFAULT_LABEL, help='Label to be used as filename prefix') - -@click.option('-v', '--video-encoder', type=str, default=DEFAULT_VIDEO_ENCODER, help='Target video encoder (vp9 or av1) default: vp9') - -@click.option('-q', '--quality', type=str, default=DEFAULT_QUALITY, help='Quality settings to be used with VP9 encoder (default: 23)') -@click.option('-p', '--preset', type=str, default=DEFAULT_QUALITY, help='Quality preset to be used with AV1 encoder (default: 5)') - -@click.option('-a', '--stereo-bitrate', type=int, default=DEFAULT_STEREO_BANDWIDTH, help='Bitrate in kbit/s to be used to encode stereo audio streams') -@click.option('-ac3', '--ac3-bitrate', type=int, default=DEFAULT_AC3_BANDWIDTH, help='Bitrate in kbit/s to be used to encode 5.1 audio streams') -@click.option('-dts', '--dts-bitrate', type=int, default=DEFAULT_DTS_BANDWIDTH, help='Bitrate in kbit/s to be used to encode 6.1 audio streams') - -@click.option('-ds', '--default-subtitle', type=int, help='Index of default subtitle stream') - -@click.option('-fa', '--forced-audio', type=int, help='Index of forced audio stream (including default audio stream tag)') -@click.option('-da', '--default-audio', type=int, help='Index of default audio stream') - - -@click.option("--crop", is_flag=False, flag_value="default", default="none") - -@click.option("-c", "--clear-metadata", is_flag=True, default=False) -@click.option("-d", "--denoise", is_flag=True, default=False) - -@click.option("-o", "--output-directory", type=str, default='') - - -def convert(ctx, paths, label, video_encoder, quality, preset, stereo_bitrate, ac3_bitrate, dts_bitrate, crop, clear_metadata, default_subtitle, forced_audio, default_audio, denoise, output_directory): - """Batch conversion of audiovideo files in format suitable for web playback, e.g. jellyfin - - Files found under PATHS will be converted according to parameters. - Filename extensions will be changed appropriately. - Suffices will we appended to filename in case of multiple created files - or if the filename has not changed.""" - - startTime = time.perf_counter() - - context = ctx.obj - - - click.echo(f"\nVideo encoder: {video_encoder}") - - qualityTokens = quality.split(',') - q_list = [q for q in qualityTokens if q.isnumeric()] - - click.echo(f"Qualities: {q_list}") - - - ctx.obj['bitrates'] = {} - ctx.obj['bitrates']['stereo'] = str(stereo_bitrate) if str(stereo_bitrate).endswith('k') else f"{stereo_bitrate}k" - ctx.obj['bitrates']['ac3'] = str(ac3_bitrate) if str(ac3_bitrate).endswith('k') else f"{ac3_bitrate}k" - ctx.obj['bitrates']['dts'] = str(dts_bitrate) if str(dts_bitrate).endswith('k') else f"{dts_bitrate}k" - - click.echo(f"Stereo bitrate: {ctx.obj['bitrates']['stereo']}") - click.echo(f"AC3 bitrate: {ctx.obj['bitrates']['ac3']}") - click.echo(f"DTS bitrate: {ctx.obj['bitrates']['dts']}") - - ctx.obj['perform_crop'] = (crop != 'none') - - if ctx.obj['perform_crop']: - - cropTokens = crop.split(',') - - if cropTokens and len(cropTokens) == 2: - - ctx.obj['crop_start'], ctx.obj['crop_length'] = crop.split(',') - else: - ctx.obj['crop_start'] = DEFAULT_CROP_START - ctx.obj['crop_length'] = DEFAULT_CROP_LENGTH - - click.echo(f"crop start={ctx.obj['crop_start']} length={ctx.obj['crop_length']}") - - - click.echo(f"\nRunning {len(paths) * len(q_list)} jobs") - - - se_match = re.compile(SEASON_EPISODE_INDICATOR_MATCH) - s_match = re.compile(SEASON_INDICATOR_MATCH) - e_match = re.compile(EPISODE_INDICATOR_MATCH) - - - for sourcePath in paths: - - - if not os.path.isfile(sourcePath): - click.echo(f"There is no file with path {sourcePath}, skipping ...") - continue - - sourceDirectory = os.path.dirname(sourcePath) - sourceFilename = os.path.basename(sourcePath) - sourcePathTokens = sourceFilename.split('.') - - if sourcePathTokens[-1] in FILE_EXTENSIONS: - sourceFileBasename = '.'.join(sourcePathTokens[:-1]) - sourceFilenameExtension = sourcePathTokens[-1] - else: - sourceFileBasename = sourceFilename - sourceFilenameExtension = '' - - #click.echo(f"dir={sourceDirectory} base={sourceFileBasename} ext={sourceFilenameExtension}") - - click.echo(f"\nProcessing file {sourcePath}") - - - se_result = se_match.search(sourceFilename) - s_result = s_match.search(sourceFilename) - e_result = e_match.search(sourceFilename) - - - - #streamDescriptor = getStreamDescriptor(sourcePath) - - #commandTokens = COMMAND_TOKENS + [sourcePath] - - - #for q in q_list: - - #click.echo(f"\nRunning job q={q}") - - #mappingVideoTokens = ['-map', 'v:0'] - #mappingTokens = mappingVideoTokens.copy() - #audioTokens = [] - - #audioIndex = 0 - #for audioStreamDescriptor in streamDescriptor: - - #if audioStreamDescriptor['type'] == STREAM_TYPE_AUDIO: - - #mappingTokens += ['-map', f"a:{audioIndex}"] - #audioTokens += generateAudioTokens(ctx.obj, audioIndex, audioStreamDescriptor['layout']) - #audioIndex += 1 - - - #for s in range(len([d for d in streamDescriptor if d['type'] == STREAM_TYPE_SUBTITLE])): - #mappingTokens += ['-map', f"s:{s}"] - - - - #if video_encoder == 'av1': - - #commandSequence = commandTokens + mappingTokens + audioTokens + generateAV1Tokens(q, preset) + audioTokens - - #if clear_metadata: - #commandSequence += generateClearTokens(streamDescriptor) - - #if performCrop: - #commandSequence += generateCropTokens(cropStart, cropLength) - - #commandSequence += generateOutputTokens(targetFilename, DEFAULT_FILE_SUFFIX, q) - - #click.echo(f"Command: {' '.join(commandSequence)}") - - #executeProcess(commandSequence) - - - #if video_encoder == 'vp9': - - #commandSequence1 = commandTokens + mappingVideoTokens + generateVP9Pass1Tokens(q) - - #if performCrop: - # commandSequence1 += generateCropTokens(cropStart, cropLength) - - #commandSequence1 += NULL_TOKENS - - #click.echo(f"Command 1: {' '.join(commandSequence1)}") - - #if os.path.exists(TEMP_FILE_NAME): - # os.remove(TEMP_FILE_NAME) - - #executeProcess(commandSequence1) - - - #commandSequence2 = commandTokens + mappingTokens - - #if denoise: - # commandSequence2 += generateDenoiseTokens() - - #commandSequence2 += generateVP9Pass2Tokens(q) + audioTokens - - #if clear_metadata: - # commandSequence2 += generateClearTokens(streamDescriptor) - - #if performCrop: - # commandSequence2 += generateCropTokens(cropStart, cropLength) - - #commandSequence2 += generateOutputTokens(targetFilename, DEFAULT_FILE_SUFFIX, q) - - #click.echo(f"Command 2: {' '.join(commandSequence2)}") - - #executeProcess(commandSequence2) - - - #app = ModesApp(ctx.obj) - #app.run() - - #click.confirm('Warning! This file is not compliant to the defined source schema! Do you want to continue?', abort=True) - - click.echo('\nDONE\n') - - endTime = time.perf_counter() - click.echo(f"Time elapsed {endTime - startTime}") - - - # click.echo(f"app result: {app.getContext()}") - - - -if __name__ == '__main__': - ffx() diff --git a/bin/ffx/.ipynb_checkpoints/file_properties-checkpoint.py b/bin/ffx/.ipynb_checkpoints/file_properties-checkpoint.py deleted file mode 100644 index 4e2521a..0000000 --- a/bin/ffx/.ipynb_checkpoints/file_properties-checkpoint.py +++ /dev/null @@ -1,66 +0,0 @@ -import os, re, click - -class FileProperties(): - - FILE_EXTENSIONS = ['mkv', 'mp4', 'avi', 'flv', 'webm'] - - SEASON_EPISODE_INDICATOR_MATCH = '[sS]([0-9]+)[eE]([0-9]+)' - EPISODE_INDICATOR_MATCH = '[eE]([0-9]+)' - - def ___init__(self, sourcePath, ): - - - # Separate basedir, basename and extension for current source file - self.__sourceDirectory = os.path.dirname(sourcePath) - self.__sourceFilename = os.path.basename(sourcePath) - sourcePathTokens = self.__sourceFilename.split('.') - - if sourcePathTokens[-1] in FilenameController.FILE_EXTENSIONS: - self.__sourceFileBasename = '.'.join(sourcePathTokens[:-1]) - self.__sourceFilenameExtension = sourcePathTokens[-1] - else: - self.__sourceFileBasename = self.__sourceFilename - self.__sourceFilenameExtension = '' - - - se_match = re.compile(FilenameController.SEASON_EPISODE_INDICATOR_MATCH) - e_match = re.compile(FilenameController.EPISODE_INDICATOR_MATCH) - - se_result = se_match.search(self.__sourceFilename) - e_result = e_match.search(self.__sourceFilename) - - self.__season = -1 - self.__episode = -1 - file_index = 0 - - if se_result is not None: - self.__season = int(se_result.group(1)) - self.__episode = int(se_result.group(2)) - elif e_result is not None: - self.__episode = int(e_result.group(1)) - else: - file_index += 1 - - - - matchingFileSubtitleDescriptors = sorted([d for d in availableFileSubtitleDescriptors if d['season'] == season and d['episode'] == episode], key=lambda d: d['stream']) if availableFileSubtitleDescriptors else [] - - print(f"season={season} episode={episode} file={file_index}") - - - # Assemble target filename tokens - targetFilenameTokens = [] - targetFilenameExtension = DEFAULT_FILE_EXTENSION - - if label: - targetFilenameTokens = [label] - - if season > -1 and episode > -1: - targetFilenameTokens += [f"S{season:0{season_digits}d}E{episode:0{episode_digits}d}"] - elif episode > -1: - targetFilenameTokens += [f"E{episode:0{episode_digits}d}"] - else: - targetFilenameTokens += [f"{file_index:0{index_digits}d}"] - - else: - targetFilenameTokens = [sourceFileBasename] diff --git a/bin/filtertest.py b/bin/filtertest.py deleted file mode 100644 index f68cf02..0000000 --- a/bin/filtertest.py +++ /dev/null @@ -1,9 +0,0 @@ -from ffx.filter.nlmeans_filter import NlmeansFilter -from ffx.filter.quality_filter import QualityFilter - -q = QualityFilter() -#q = QualityFilter('32,34') -# NlmeansFilter(researchWindow='5,7,9', strength='2.0,3.0,4.0') - -for cy in q.getChainYield(): - print(cy) diff --git a/ffx.egg-info/PKG-INFO b/ffx.egg-info/PKG-INFO new file mode 100644 index 0000000..2967e19 --- /dev/null +++ b/ffx.egg-info/PKG-INFO @@ -0,0 +1,57 @@ +Metadata-Version: 2.1 +Name: ffx +Version: 0.2.2 +Summary: FFX recoding and metadata managing tool +Home-page: https://gitea.maveno.de/Javanaut/ffx +Author: Javanaut +Author-email: javanaut@maveno.de +Project-URL: Bug Tracker, https://gitea.maveno.de/Javanaut/ffx/issues +Classifier: Operating System :: OS Independent +Requires-Python: >=3.6 +Description-Content-Type: text/markdown +Requires-Dist: requests +Requires-Dist: click +Requires-Dist: textual +Requires-Dist: sqlalchemy + +# FFX + +## Installation + +per https: + +```sh +pip install https:////ffx.git@ +``` + +per git: + +```sh +pip install git+ssh://@//ffx.git@ +``` + +## Version history + +### 0.1.1 + +Bugfixes, TMBD identify shows + +### 0.1.2 + +Bugfixes + +### 0.1.3 + +Subtitle file imports + +### 0.2.0 + +Tests, Config-File + +### 0.2.1 + +Signature, Tags cleaning, Bugfixes, Refactoring + +### 0.2.2 + +CLI-Overrides diff --git a/ffx.egg-info/SOURCES.txt b/ffx.egg-info/SOURCES.txt new file mode 100644 index 0000000..1917e97 --- /dev/null +++ b/ffx.egg-info/SOURCES.txt @@ -0,0 +1,121 @@ +README.md +pyproject.toml +setup.cfg +ffx.egg-info/PKG-INFO +ffx.egg-info/SOURCES.txt +ffx.egg-info/dependency_links.txt +ffx.egg-info/entry_points.txt +ffx.egg-info/requires.txt +ffx.egg-info/top_level.txt +src/ffx.py +src/ffx_tests.py +src/ffx/__init__.py +src/ffx/audio_layout.py +src/ffx/configuration_controller.py +src/ffx/constants.py +src/ffx/database.py +src/ffx/ffx_app.py +src/ffx/ffx_controller.py +src/ffx/file_properties.py +src/ffx/help_screen.py +src/ffx/helper.py +src/ffx/iso_language.py +src/ffx/media_controller.py +src/ffx/media_descriptor.py +src/ffx/media_details_screen.py +src/ffx/pattern_controller.py +src/ffx/pattern_delete_screen.py +src/ffx/pattern_details_screen.py +src/ffx/process.py +src/ffx/settings_screen.py +src/ffx/shifted_season_controller.py +src/ffx/shifted_season_delete_screen.py +src/ffx/shifted_season_details_screen.py +src/ffx/show_controller.py +src/ffx/show_delete_screen.py +src/ffx/show_descriptor.py +src/ffx/show_details_screen.py +src/ffx/shows_screen.py +src/ffx/tag_controller.py +src/ffx/tag_delete_screen.py +src/ffx/tag_details_screen.py +src/ffx/tmdb_controller.py +src/ffx/track_controller.py +src/ffx/track_delete_screen.py +src/ffx/track_descriptor.py +src/ffx/track_details_screen.py +src/ffx/track_disposition.py +src/ffx/track_type.py +src/ffx/video_encoder.py +src/ffx/filter/__init__.py +src/ffx/filter/filter.py +src/ffx/filter/nlmeans_filter.py +src/ffx/filter/preset_filter.py +src/ffx/filter/quality_filter.py +src/ffx/filter/scale_filter.py +src/ffx/model/__init__.py +src/ffx/model/media_tag.py +src/ffx/model/pattern.py +src/ffx/model/property.py +src/ffx/model/shifted_season.py +src/ffx/model/show.py +src/ffx/model/track.py +src/ffx/model/track_tag.py +src/ffx/model/conversions/__init__.py +src/ffx/model/conversions/conversion.py +src/ffx/model/conversions/conversion_2_3.py +src/ffx/model/conversions/conversion_3_4.py +src/ffx/test/_basename_combinator_1.py +src/ffx/test/basename_combinator.py +src/ffx/test/basename_combinator_0.py +src/ffx/test/basename_combinator_2.py +src/ffx/test/combinator.py +src/ffx/test/disposition_combinator_2.py +src/ffx/test/disposition_combinator_2_0.py +src/ffx/test/disposition_combinator_2_1.py +src/ffx/test/disposition_combinator_2_2.py +src/ffx/test/disposition_combinator_2_3 .py +src/ffx/test/disposition_combinator_3.py +src/ffx/test/disposition_combinator_3_0.py +src/ffx/test/disposition_combinator_3_1.py +src/ffx/test/disposition_combinator_3_2.py +src/ffx/test/disposition_combinator_3_3.py +src/ffx/test/disposition_combinator_3_4.py +src/ffx/test/helper.py +src/ffx/test/indicator_combinator.py +src/ffx/test/label_combinator.py +src/ffx/test/label_combinator_0.py +src/ffx/test/label_combinator_1.py +src/ffx/test/media_combinator.py +src/ffx/test/media_combinator_0.py +src/ffx/test/media_combinator_1.py +src/ffx/test/media_combinator_2.py +src/ffx/test/media_combinator_3.py +src/ffx/test/media_combinator_4.py +src/ffx/test/media_combinator_5.py +src/ffx/test/media_combinator_6.py +src/ffx/test/media_combinator_7.py +src/ffx/test/media_tag_combinator.py +src/ffx/test/media_tag_combinator_0.py +src/ffx/test/media_tag_combinator_1.py +src/ffx/test/media_tag_combinator_2.py +src/ffx/test/permutation_combinator_2.py +src/ffx/test/permutation_combinator_3.py +src/ffx/test/release_combinator.py +src/ffx/test/scenario.py +src/ffx/test/scenario_1.py +src/ffx/test/scenario_2.py +src/ffx/test/scenario_4.py +src/ffx/test/show_combinator.py +src/ffx/test/title_combinator.py +src/ffx/test/track_tag_combinator_2.py +src/ffx/test/track_tag_combinator_2_0.py +src/ffx/test/track_tag_combinator_2_1.py +src/ffx/test/track_tag_combinator_2_2.py +src/ffx/test/track_tag_combinator_2_3.py +src/ffx/test/track_tag_combinator_3.py +src/ffx/test/track_tag_combinator_3_0.py +src/ffx/test/track_tag_combinator_3_1.py +src/ffx/test/track_tag_combinator_3_2.py +src/ffx/test/track_tag_combinator_3_3.py +src/ffx/test/track_tag_combinator_3_4.py \ No newline at end of file diff --git a/ffx.egg-info/dependency_links.txt b/ffx.egg-info/dependency_links.txt new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/ffx.egg-info/dependency_links.txt @@ -0,0 +1 @@ + diff --git a/ffx.egg-info/entry_points.txt b/ffx.egg-info/entry_points.txt new file mode 100644 index 0000000..ef54e3c --- /dev/null +++ b/ffx.egg-info/entry_points.txt @@ -0,0 +1,2 @@ +[console_scripts] +osgw = ffx:ffx diff --git a/ffx.egg-info/requires.txt b/ffx.egg-info/requires.txt new file mode 100644 index 0000000..00d455c --- /dev/null +++ b/ffx.egg-info/requires.txt @@ -0,0 +1,4 @@ +requests +click +textual +sqlalchemy diff --git a/ffx.egg-info/top_level.txt b/ffx.egg-info/top_level.txt new file mode 100644 index 0000000..25e4d71 --- /dev/null +++ b/ffx.egg-info/top_level.txt @@ -0,0 +1,2 @@ +ansible +src diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000..577d963 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,37 @@ +[project] +name = "ffx" +description = "FFX recoding and metadata managing tool" +version = "0.2.2" +license = {file = "LICENSE.md"} +dependencies = [ + "requests", + "click", + "textual", + "sqlalchemy", +] +readme = {file = "README.md", content-type = "text/markdown"} +authors = [ + {name = "Marius", email = "javanaut@maveno.de"} +] +maintainers = [ + {name = "Marius", email = "javanaut@maveno.de"} +] +classifiers = [ + "Development Status :: 3 - Alpha", + "Programming Language :: Python" +] + +[project.urls] +Homepage = "https://gitea.maveno.de/Javanaut/ffx" +Repository = "https://gitea.maveno.de/Javanaut/ffx.git" +Issues = "https://gitea.maveno.de/Javanaut/ffx/issues" + +[build-system] +requires = [ + "setuptools", + "wheel" +] +build-backend = "setuptools.build_meta" + +[project.scripts] +ffx = "ffx.ffx:ffx" diff --git a/src/ffx.egg-info/PKG-INFO b/src/ffx.egg-info/PKG-INFO new file mode 100644 index 0000000..e94b45e --- /dev/null +++ b/src/ffx.egg-info/PKG-INFO @@ -0,0 +1,656 @@ +Metadata-Version: 2.1 +Name: ffx +Version: 0.2.2 +Summary: FFX recoding and metadata managing tool +Author-email: Marius +Maintainer-email: Marius +License: GNU General Public License + ========================== + + _Version 3, 29 June 2007_ + _Copyright © 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 + <>. + +Project-URL: Homepage, https://gitea.maveno.de/Javanaut/ffx +Project-URL: Repository, https://gitea.maveno.de/Javanaut/ffx.git +Project-URL: Issues, https://gitea.maveno.de/Javanaut/ffx/issues +Classifier: Development Status :: 3 - Alpha +Classifier: Programming Language :: Python +Description-Content-Type: text/markdown +License-File: LICENSE.md +Requires-Dist: rich +Requires-Dist: requests +Requires-Dist: click +Requires-Dist: textual +Requires-Dist: sqlalchemy + +# FFX + +## Installation + +per https: + +```sh +pip install https:////ffx.git@ +``` + +per git: + +```sh +pip install git+ssh://@//ffx.git@ +``` + +## Version history + +### 0.1.1 + +Bugfixes, TMBD identify shows + +### 0.1.2 + +Bugfixes + +### 0.1.3 + +Subtitle file imports + +### 0.2.0 + +Tests, Config-File + +### 0.2.1 + +Signature, Tags cleaning, Bugfixes, Refactoring + +### 0.2.2 + +CLI-Overrides diff --git a/src/ffx.egg-info/SOURCES.txt b/src/ffx.egg-info/SOURCES.txt new file mode 100644 index 0000000..f96b92c --- /dev/null +++ b/src/ffx.egg-info/SOURCES.txt @@ -0,0 +1,121 @@ +LICENSE.md +README.md +pyproject.toml +src/ffx/__init__.py +src/ffx/audio_layout.py +src/ffx/configuration_controller.py +src/ffx/constants.py +src/ffx/database.py +src/ffx/ffx.py +src/ffx/ffx_app.py +src/ffx/ffx_controller.py +src/ffx/ffx_tests.py +src/ffx/file_properties.py +src/ffx/help_screen.py +src/ffx/helper.py +src/ffx/iso_language.py +src/ffx/media_controller.py +src/ffx/media_descriptor.py +src/ffx/media_details_screen.py +src/ffx/pattern_controller.py +src/ffx/pattern_delete_screen.py +src/ffx/pattern_details_screen.py +src/ffx/process.py +src/ffx/settings_screen.py +src/ffx/shifted_season_controller.py +src/ffx/shifted_season_delete_screen.py +src/ffx/shifted_season_details_screen.py +src/ffx/show_controller.py +src/ffx/show_delete_screen.py +src/ffx/show_descriptor.py +src/ffx/show_details_screen.py +src/ffx/shows_screen.py +src/ffx/tag_controller.py +src/ffx/tag_delete_screen.py +src/ffx/tag_details_screen.py +src/ffx/tmdb_controller.py +src/ffx/track_controller.py +src/ffx/track_delete_screen.py +src/ffx/track_descriptor.py +src/ffx/track_details_screen.py +src/ffx/track_disposition.py +src/ffx/track_type.py +src/ffx/video_encoder.py +src/ffx.egg-info/PKG-INFO +src/ffx.egg-info/SOURCES.txt +src/ffx.egg-info/dependency_links.txt +src/ffx.egg-info/entry_points.txt +src/ffx.egg-info/requires.txt +src/ffx.egg-info/top_level.txt +src/ffx/filter/__init__.py +src/ffx/filter/filter.py +src/ffx/filter/nlmeans_filter.py +src/ffx/filter/preset_filter.py +src/ffx/filter/quality_filter.py +src/ffx/filter/scale_filter.py +src/ffx/model/__init__.py +src/ffx/model/media_tag.py +src/ffx/model/pattern.py +src/ffx/model/property.py +src/ffx/model/shifted_season.py +src/ffx/model/show.py +src/ffx/model/track.py +src/ffx/model/track_tag.py +src/ffx/model/conversions/__init__.py +src/ffx/model/conversions/conversion.py +src/ffx/model/conversions/conversion_2_3.py +src/ffx/model/conversions/conversion_3_4.py +src/ffx/test/_basename_combinator_1.py +src/ffx/test/basename_combinator.py +src/ffx/test/basename_combinator_0.py +src/ffx/test/basename_combinator_2.py +src/ffx/test/combinator.py +src/ffx/test/disposition_combinator_2.py +src/ffx/test/disposition_combinator_2_0.py +src/ffx/test/disposition_combinator_2_1.py +src/ffx/test/disposition_combinator_2_2.py +src/ffx/test/disposition_combinator_2_3 .py +src/ffx/test/disposition_combinator_3.py +src/ffx/test/disposition_combinator_3_0.py +src/ffx/test/disposition_combinator_3_1.py +src/ffx/test/disposition_combinator_3_2.py +src/ffx/test/disposition_combinator_3_3.py +src/ffx/test/disposition_combinator_3_4.py +src/ffx/test/helper.py +src/ffx/test/indicator_combinator.py +src/ffx/test/label_combinator.py +src/ffx/test/label_combinator_0.py +src/ffx/test/label_combinator_1.py +src/ffx/test/media_combinator.py +src/ffx/test/media_combinator_0.py +src/ffx/test/media_combinator_1.py +src/ffx/test/media_combinator_2.py +src/ffx/test/media_combinator_3.py +src/ffx/test/media_combinator_4.py +src/ffx/test/media_combinator_5.py +src/ffx/test/media_combinator_6.py +src/ffx/test/media_combinator_7.py +src/ffx/test/media_tag_combinator.py +src/ffx/test/media_tag_combinator_0.py +src/ffx/test/media_tag_combinator_1.py +src/ffx/test/media_tag_combinator_2.py +src/ffx/test/permutation_combinator_2.py +src/ffx/test/permutation_combinator_3.py +src/ffx/test/release_combinator.py +src/ffx/test/scenario.py +src/ffx/test/scenario_1.py +src/ffx/test/scenario_2.py +src/ffx/test/scenario_4.py +src/ffx/test/show_combinator.py +src/ffx/test/title_combinator.py +src/ffx/test/track_tag_combinator_2.py +src/ffx/test/track_tag_combinator_2_0.py +src/ffx/test/track_tag_combinator_2_1.py +src/ffx/test/track_tag_combinator_2_2.py +src/ffx/test/track_tag_combinator_2_3.py +src/ffx/test/track_tag_combinator_3.py +src/ffx/test/track_tag_combinator_3_0.py +src/ffx/test/track_tag_combinator_3_1.py +src/ffx/test/track_tag_combinator_3_2.py +src/ffx/test/track_tag_combinator_3_3.py +src/ffx/test/track_tag_combinator_3_4.py \ No newline at end of file diff --git a/src/ffx.egg-info/dependency_links.txt b/src/ffx.egg-info/dependency_links.txt new file mode 100644 index 0000000..8b13789 --- /dev/null +++ b/src/ffx.egg-info/dependency_links.txt @@ -0,0 +1 @@ + diff --git a/src/ffx.egg-info/entry_points.txt b/src/ffx.egg-info/entry_points.txt new file mode 100644 index 0000000..d4134ff --- /dev/null +++ b/src/ffx.egg-info/entry_points.txt @@ -0,0 +1,2 @@ +[console_scripts] +ffx = ffx.ffx:ffx diff --git a/src/ffx.egg-info/requires.txt b/src/ffx.egg-info/requires.txt new file mode 100644 index 0000000..bca8e96 --- /dev/null +++ b/src/ffx.egg-info/requires.txt @@ -0,0 +1,5 @@ +rich +requests +click +textual +sqlalchemy diff --git a/src/ffx.egg-info/top_level.txt b/src/ffx.egg-info/top_level.txt new file mode 100644 index 0000000..fb18c76 --- /dev/null +++ b/src/ffx.egg-info/top_level.txt @@ -0,0 +1 @@ +ffx diff --git a/bin/ffx/__init__.py b/src/ffx/__init__.py similarity index 100% rename from bin/ffx/__init__.py rename to src/ffx/__init__.py diff --git a/bin/ffx/audio_layout.py b/src/ffx/audio_layout.py similarity index 100% rename from bin/ffx/audio_layout.py rename to src/ffx/audio_layout.py diff --git a/bin/ffx/configuration_controller.py b/src/ffx/configuration_controller.py similarity index 100% rename from bin/ffx/configuration_controller.py rename to src/ffx/configuration_controller.py diff --git a/bin/ffx/constants.py b/src/ffx/constants.py similarity index 100% rename from bin/ffx/constants.py rename to src/ffx/constants.py diff --git a/bin/ffx/database.py b/src/ffx/database.py similarity index 100% rename from bin/ffx/database.py rename to src/ffx/database.py diff --git a/bin/ffx.py b/src/ffx/ffx.py similarity index 99% rename from bin/ffx.py rename to src/ffx/ffx.py index ea2f5c5..71070ec 100755 --- a/bin/ffx.py +++ b/src/ffx/ffx.py @@ -36,20 +36,6 @@ from ffx.constants import VERSION from ffx.shifted_season_controller import ShiftedSeasonController -# 0.1.1 -# Bugfixes, TMBD identify shows -# 0.1.2 -# Bugfixes -# 0.1.3 -# Subtitle file imports -# 0.2.0 -# Tests, Config-File -# 0.2.1 -# Signature, Tags cleaning, Bugfixes, Refactoring -# 0.2.2 -# CLI-Overrides - - @click.group() @click.pass_context @click.option('--database-file', type=str, default='', help='Path to database file') diff --git a/bin/ffx/ffx_app.py b/src/ffx/ffx_app.py similarity index 100% rename from bin/ffx/ffx_app.py rename to src/ffx/ffx_app.py diff --git a/bin/ffx/ffx_controller.py b/src/ffx/ffx_controller.py similarity index 100% rename from bin/ffx/ffx_controller.py rename to src/ffx/ffx_controller.py diff --git a/bin/ffx_tests.py b/src/ffx/ffx_tests.py similarity index 100% rename from bin/ffx_tests.py rename to src/ffx/ffx_tests.py diff --git a/bin/ffx/file_properties.py b/src/ffx/file_properties.py similarity index 100% rename from bin/ffx/file_properties.py rename to src/ffx/file_properties.py diff --git a/bin/ffx/filter/__init__.py b/src/ffx/filter/__init__.py similarity index 100% rename from bin/ffx/filter/__init__.py rename to src/ffx/filter/__init__.py diff --git a/bin/ffx/filter/filter.py b/src/ffx/filter/filter.py similarity index 100% rename from bin/ffx/filter/filter.py rename to src/ffx/filter/filter.py diff --git a/bin/ffx/filter/nlmeans_filter.py b/src/ffx/filter/nlmeans_filter.py similarity index 100% rename from bin/ffx/filter/nlmeans_filter.py rename to src/ffx/filter/nlmeans_filter.py diff --git a/bin/ffx/filter/preset_filter.py b/src/ffx/filter/preset_filter.py similarity index 100% rename from bin/ffx/filter/preset_filter.py rename to src/ffx/filter/preset_filter.py diff --git a/bin/ffx/filter/quality_filter.py b/src/ffx/filter/quality_filter.py similarity index 100% rename from bin/ffx/filter/quality_filter.py rename to src/ffx/filter/quality_filter.py diff --git a/bin/ffx/filter/scale_filter.py b/src/ffx/filter/scale_filter.py similarity index 100% rename from bin/ffx/filter/scale_filter.py rename to src/ffx/filter/scale_filter.py diff --git a/bin/ffx/help_screen.py b/src/ffx/help_screen.py similarity index 100% rename from bin/ffx/help_screen.py rename to src/ffx/help_screen.py diff --git a/bin/ffx/helper.py b/src/ffx/helper.py similarity index 100% rename from bin/ffx/helper.py rename to src/ffx/helper.py diff --git a/bin/ffx/iso_language.py b/src/ffx/iso_language.py similarity index 100% rename from bin/ffx/iso_language.py rename to src/ffx/iso_language.py diff --git a/bin/ffx/media_controller.py b/src/ffx/media_controller.py similarity index 100% rename from bin/ffx/media_controller.py rename to src/ffx/media_controller.py diff --git a/bin/ffx/media_descriptor.py b/src/ffx/media_descriptor.py similarity index 100% rename from bin/ffx/media_descriptor.py rename to src/ffx/media_descriptor.py diff --git a/bin/ffx/media_details_screen.py b/src/ffx/media_details_screen.py similarity index 99% rename from bin/ffx/media_details_screen.py rename to src/ffx/media_details_screen.py index 32605c3..702d967 100644 --- a/bin/ffx/media_details_screen.py +++ b/src/ffx/media_details_screen.py @@ -49,6 +49,16 @@ class MediaDetailsScreen(Screen): width: 100%; padding: 1; } + + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } Input { border: none; diff --git a/bin/ffx/model/__init__.py b/src/ffx/model/__init__.py similarity index 100% rename from bin/ffx/model/__init__.py rename to src/ffx/model/__init__.py diff --git a/bin/ffx/model/conversions/__init__.py b/src/ffx/model/conversions/__init__.py similarity index 100% rename from bin/ffx/model/conversions/__init__.py rename to src/ffx/model/conversions/__init__.py diff --git a/bin/ffx/model/conversions/conversion.py b/src/ffx/model/conversions/conversion.py similarity index 100% rename from bin/ffx/model/conversions/conversion.py rename to src/ffx/model/conversions/conversion.py diff --git a/bin/ffx/model/conversions/conversion_2_3.py b/src/ffx/model/conversions/conversion_2_3.py similarity index 100% rename from bin/ffx/model/conversions/conversion_2_3.py rename to src/ffx/model/conversions/conversion_2_3.py diff --git a/bin/ffx/model/conversions/conversion_3_4.py b/src/ffx/model/conversions/conversion_3_4.py similarity index 100% rename from bin/ffx/model/conversions/conversion_3_4.py rename to src/ffx/model/conversions/conversion_3_4.py diff --git a/bin/ffx/model/media_tag.py b/src/ffx/model/media_tag.py similarity index 100% rename from bin/ffx/model/media_tag.py rename to src/ffx/model/media_tag.py diff --git a/bin/ffx/model/pattern.py b/src/ffx/model/pattern.py similarity index 100% rename from bin/ffx/model/pattern.py rename to src/ffx/model/pattern.py diff --git a/bin/ffx/model/property.py b/src/ffx/model/property.py similarity index 100% rename from bin/ffx/model/property.py rename to src/ffx/model/property.py diff --git a/bin/ffx/model/shifted_season.py b/src/ffx/model/shifted_season.py similarity index 100% rename from bin/ffx/model/shifted_season.py rename to src/ffx/model/shifted_season.py diff --git a/bin/ffx/model/show.py b/src/ffx/model/show.py similarity index 100% rename from bin/ffx/model/show.py rename to src/ffx/model/show.py diff --git a/bin/ffx/model/track.py b/src/ffx/model/track.py similarity index 100% rename from bin/ffx/model/track.py rename to src/ffx/model/track.py diff --git a/bin/ffx/model/track_tag.py b/src/ffx/model/track_tag.py similarity index 100% rename from bin/ffx/model/track_tag.py rename to src/ffx/model/track_tag.py diff --git a/bin/ffx/pattern_controller.py b/src/ffx/pattern_controller.py similarity index 100% rename from bin/ffx/pattern_controller.py rename to src/ffx/pattern_controller.py diff --git a/bin/ffx/pattern_delete_screen.py b/src/ffx/pattern_delete_screen.py similarity index 100% rename from bin/ffx/pattern_delete_screen.py rename to src/ffx/pattern_delete_screen.py diff --git a/bin/ffx/pattern_details_screen.py b/src/ffx/pattern_details_screen.py similarity index 98% rename from bin/ffx/pattern_details_screen.py rename to src/ffx/pattern_details_screen.py index 9f92cc2..8ce7e6d 100644 --- a/bin/ffx/pattern_details_screen.py +++ b/src/ffx/pattern_details_screen.py @@ -59,6 +59,16 @@ class PatternDetailsScreen(Screen): min-height: 6; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #toplabel { height: 1; } diff --git a/bin/ffx/process.py b/src/ffx/process.py similarity index 100% rename from bin/ffx/process.py rename to src/ffx/process.py diff --git a/bin/ffx/settings_screen.py b/src/ffx/settings_screen.py similarity index 100% rename from bin/ffx/settings_screen.py rename to src/ffx/settings_screen.py diff --git a/bin/ffx/shifted_season_controller.py b/src/ffx/shifted_season_controller.py similarity index 100% rename from bin/ffx/shifted_season_controller.py rename to src/ffx/shifted_season_controller.py diff --git a/bin/ffx/shifted_season_delete_screen.py b/src/ffx/shifted_season_delete_screen.py similarity index 100% rename from bin/ffx/shifted_season_delete_screen.py rename to src/ffx/shifted_season_delete_screen.py diff --git a/bin/ffx/shifted_season_details_screen.py b/src/ffx/shifted_season_details_screen.py similarity index 97% rename from bin/ffx/shifted_season_details_screen.py rename to src/ffx/shifted_season_details_screen.py index 7cf0194..0dc822a 100644 --- a/bin/ffx/shifted_season_details_screen.py +++ b/src/ffx/shifted_season_details_screen.py @@ -42,6 +42,16 @@ class ShiftedSeasonDetailsScreen(Screen): min-height: 6; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #toplabel { height: 1; diff --git a/bin/ffx/show_controller.py b/src/ffx/show_controller.py similarity index 100% rename from bin/ffx/show_controller.py rename to src/ffx/show_controller.py diff --git a/bin/ffx/show_delete_screen.py b/src/ffx/show_delete_screen.py similarity index 100% rename from bin/ffx/show_delete_screen.py rename to src/ffx/show_delete_screen.py diff --git a/bin/ffx/show_descriptor.py b/src/ffx/show_descriptor.py similarity index 100% rename from bin/ffx/show_descriptor.py rename to src/ffx/show_descriptor.py diff --git a/bin/ffx/show_details_screen.py b/src/ffx/show_details_screen.py similarity index 98% rename from bin/ffx/show_details_screen.py rename to src/ffx/show_details_screen.py index 110ebb1..c5282ce 100644 --- a/bin/ffx/show_details_screen.py +++ b/src/ffx/show_details_screen.py @@ -48,11 +48,22 @@ class ShowDetailsScreen(Screen): Button { border: none; } + DataTable { column-span: 2; min-height: 8; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #toplabel { height: 1; } diff --git a/bin/ffx/shows_screen.py b/src/ffx/shows_screen.py similarity index 95% rename from bin/ffx/shows_screen.py rename to src/ffx/shows_screen.py index 524470f..da86175 100644 --- a/bin/ffx/shows_screen.py +++ b/src/ffx/shows_screen.py @@ -30,6 +30,16 @@ class ShowsScreen(Screen): padding: 1; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #top { height: 1; } diff --git a/bin/ffx/tag_controller.py b/src/ffx/tag_controller.py similarity index 100% rename from bin/ffx/tag_controller.py rename to src/ffx/tag_controller.py diff --git a/bin/ffx/tag_delete_screen.py b/src/ffx/tag_delete_screen.py similarity index 100% rename from bin/ffx/tag_delete_screen.py rename to src/ffx/tag_delete_screen.py diff --git a/bin/ffx/tag_details_screen.py b/src/ffx/tag_details_screen.py similarity index 92% rename from bin/ffx/tag_details_screen.py rename to src/ffx/tag_details_screen.py index b80c3fa..673735d 100644 --- a/bin/ffx/tag_details_screen.py +++ b/src/ffx/tag_details_screen.py @@ -30,10 +30,21 @@ class TagDetailsScreen(Screen): Select { border: none; } + DataTable { min-height: 6; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #toplabel { height: 1; } diff --git a/bin/ffx/test/_basename_combinator_1.py b/src/ffx/test/_basename_combinator_1.py similarity index 100% rename from bin/ffx/test/_basename_combinator_1.py rename to src/ffx/test/_basename_combinator_1.py diff --git a/bin/ffx/test/basename_combinator.py b/src/ffx/test/basename_combinator.py similarity index 100% rename from bin/ffx/test/basename_combinator.py rename to src/ffx/test/basename_combinator.py diff --git a/bin/ffx/test/basename_combinator_0.py b/src/ffx/test/basename_combinator_0.py similarity index 100% rename from bin/ffx/test/basename_combinator_0.py rename to src/ffx/test/basename_combinator_0.py diff --git a/bin/ffx/test/basename_combinator_2.py b/src/ffx/test/basename_combinator_2.py similarity index 100% rename from bin/ffx/test/basename_combinator_2.py rename to src/ffx/test/basename_combinator_2.py diff --git a/bin/ffx/test/combinator.py b/src/ffx/test/combinator.py similarity index 100% rename from bin/ffx/test/combinator.py rename to src/ffx/test/combinator.py diff --git a/bin/ffx/test/disposition_combinator_2.py b/src/ffx/test/disposition_combinator_2.py similarity index 100% rename from bin/ffx/test/disposition_combinator_2.py rename to src/ffx/test/disposition_combinator_2.py diff --git a/bin/ffx/test/disposition_combinator_2_0.py b/src/ffx/test/disposition_combinator_2_0.py similarity index 100% rename from bin/ffx/test/disposition_combinator_2_0.py rename to src/ffx/test/disposition_combinator_2_0.py diff --git a/bin/ffx/test/disposition_combinator_2_1.py b/src/ffx/test/disposition_combinator_2_1.py similarity index 100% rename from bin/ffx/test/disposition_combinator_2_1.py rename to src/ffx/test/disposition_combinator_2_1.py diff --git a/bin/ffx/test/disposition_combinator_2_2.py b/src/ffx/test/disposition_combinator_2_2.py similarity index 100% rename from bin/ffx/test/disposition_combinator_2_2.py rename to src/ffx/test/disposition_combinator_2_2.py diff --git a/bin/ffx/test/disposition_combinator_2_3 .py b/src/ffx/test/disposition_combinator_2_3 .py similarity index 100% rename from bin/ffx/test/disposition_combinator_2_3 .py rename to src/ffx/test/disposition_combinator_2_3 .py diff --git a/bin/ffx/test/disposition_combinator_3.py b/src/ffx/test/disposition_combinator_3.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3.py rename to src/ffx/test/disposition_combinator_3.py diff --git a/bin/ffx/test/disposition_combinator_3_0.py b/src/ffx/test/disposition_combinator_3_0.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3_0.py rename to src/ffx/test/disposition_combinator_3_0.py diff --git a/bin/ffx/test/disposition_combinator_3_1.py b/src/ffx/test/disposition_combinator_3_1.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3_1.py rename to src/ffx/test/disposition_combinator_3_1.py diff --git a/bin/ffx/test/disposition_combinator_3_2.py b/src/ffx/test/disposition_combinator_3_2.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3_2.py rename to src/ffx/test/disposition_combinator_3_2.py diff --git a/bin/ffx/test/disposition_combinator_3_3.py b/src/ffx/test/disposition_combinator_3_3.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3_3.py rename to src/ffx/test/disposition_combinator_3_3.py diff --git a/bin/ffx/test/disposition_combinator_3_4.py b/src/ffx/test/disposition_combinator_3_4.py similarity index 100% rename from bin/ffx/test/disposition_combinator_3_4.py rename to src/ffx/test/disposition_combinator_3_4.py diff --git a/bin/ffx/test/helper.py b/src/ffx/test/helper.py similarity index 100% rename from bin/ffx/test/helper.py rename to src/ffx/test/helper.py diff --git a/bin/ffx/test/indicator_combinator.py b/src/ffx/test/indicator_combinator.py similarity index 100% rename from bin/ffx/test/indicator_combinator.py rename to src/ffx/test/indicator_combinator.py diff --git a/bin/ffx/test/label_combinator.py b/src/ffx/test/label_combinator.py similarity index 100% rename from bin/ffx/test/label_combinator.py rename to src/ffx/test/label_combinator.py diff --git a/bin/ffx/test/label_combinator_0.py b/src/ffx/test/label_combinator_0.py similarity index 100% rename from bin/ffx/test/label_combinator_0.py rename to src/ffx/test/label_combinator_0.py diff --git a/bin/ffx/test/label_combinator_1.py b/src/ffx/test/label_combinator_1.py similarity index 100% rename from bin/ffx/test/label_combinator_1.py rename to src/ffx/test/label_combinator_1.py diff --git a/bin/ffx/test/media_combinator.py b/src/ffx/test/media_combinator.py similarity index 100% rename from bin/ffx/test/media_combinator.py rename to src/ffx/test/media_combinator.py diff --git a/bin/ffx/test/media_combinator_0.py b/src/ffx/test/media_combinator_0.py similarity index 100% rename from bin/ffx/test/media_combinator_0.py rename to src/ffx/test/media_combinator_0.py diff --git a/bin/ffx/test/media_combinator_1.py b/src/ffx/test/media_combinator_1.py similarity index 100% rename from bin/ffx/test/media_combinator_1.py rename to src/ffx/test/media_combinator_1.py diff --git a/bin/ffx/test/media_combinator_2.py b/src/ffx/test/media_combinator_2.py similarity index 100% rename from bin/ffx/test/media_combinator_2.py rename to src/ffx/test/media_combinator_2.py diff --git a/bin/ffx/test/media_combinator_3.py b/src/ffx/test/media_combinator_3.py similarity index 100% rename from bin/ffx/test/media_combinator_3.py rename to src/ffx/test/media_combinator_3.py diff --git a/bin/ffx/test/media_combinator_4.py b/src/ffx/test/media_combinator_4.py similarity index 100% rename from bin/ffx/test/media_combinator_4.py rename to src/ffx/test/media_combinator_4.py diff --git a/bin/ffx/test/media_combinator_5.py b/src/ffx/test/media_combinator_5.py similarity index 100% rename from bin/ffx/test/media_combinator_5.py rename to src/ffx/test/media_combinator_5.py diff --git a/bin/ffx/test/media_combinator_6.py b/src/ffx/test/media_combinator_6.py similarity index 100% rename from bin/ffx/test/media_combinator_6.py rename to src/ffx/test/media_combinator_6.py diff --git a/bin/ffx/test/media_combinator_7.py b/src/ffx/test/media_combinator_7.py similarity index 100% rename from bin/ffx/test/media_combinator_7.py rename to src/ffx/test/media_combinator_7.py diff --git a/bin/ffx/test/media_tag_combinator.py b/src/ffx/test/media_tag_combinator.py similarity index 100% rename from bin/ffx/test/media_tag_combinator.py rename to src/ffx/test/media_tag_combinator.py diff --git a/bin/ffx/test/media_tag_combinator_0.py b/src/ffx/test/media_tag_combinator_0.py similarity index 100% rename from bin/ffx/test/media_tag_combinator_0.py rename to src/ffx/test/media_tag_combinator_0.py diff --git a/bin/ffx/test/media_tag_combinator_1.py b/src/ffx/test/media_tag_combinator_1.py similarity index 100% rename from bin/ffx/test/media_tag_combinator_1.py rename to src/ffx/test/media_tag_combinator_1.py diff --git a/bin/ffx/test/media_tag_combinator_2.py b/src/ffx/test/media_tag_combinator_2.py similarity index 100% rename from bin/ffx/test/media_tag_combinator_2.py rename to src/ffx/test/media_tag_combinator_2.py diff --git a/bin/ffx/test/permutation_combinator_2.py b/src/ffx/test/permutation_combinator_2.py similarity index 100% rename from bin/ffx/test/permutation_combinator_2.py rename to src/ffx/test/permutation_combinator_2.py diff --git a/bin/ffx/test/permutation_combinator_3.py b/src/ffx/test/permutation_combinator_3.py similarity index 100% rename from bin/ffx/test/permutation_combinator_3.py rename to src/ffx/test/permutation_combinator_3.py diff --git a/bin/ffx/test/release_combinator.py b/src/ffx/test/release_combinator.py similarity index 100% rename from bin/ffx/test/release_combinator.py rename to src/ffx/test/release_combinator.py diff --git a/bin/ffx/test/scenario.py b/src/ffx/test/scenario.py similarity index 100% rename from bin/ffx/test/scenario.py rename to src/ffx/test/scenario.py diff --git a/bin/ffx/test/scenario_1.py b/src/ffx/test/scenario_1.py similarity index 100% rename from bin/ffx/test/scenario_1.py rename to src/ffx/test/scenario_1.py diff --git a/bin/ffx/test/scenario_2.py b/src/ffx/test/scenario_2.py similarity index 100% rename from bin/ffx/test/scenario_2.py rename to src/ffx/test/scenario_2.py diff --git a/bin/ffx/test/scenario_4.py b/src/ffx/test/scenario_4.py similarity index 100% rename from bin/ffx/test/scenario_4.py rename to src/ffx/test/scenario_4.py diff --git a/bin/ffx/test/show_combinator.py b/src/ffx/test/show_combinator.py similarity index 100% rename from bin/ffx/test/show_combinator.py rename to src/ffx/test/show_combinator.py diff --git a/bin/ffx/test/title_combinator.py b/src/ffx/test/title_combinator.py similarity index 100% rename from bin/ffx/test/title_combinator.py rename to src/ffx/test/title_combinator.py diff --git a/bin/ffx/test/track_tag_combinator_2.py b/src/ffx/test/track_tag_combinator_2.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_2.py rename to src/ffx/test/track_tag_combinator_2.py diff --git a/bin/ffx/test/track_tag_combinator_2_0.py b/src/ffx/test/track_tag_combinator_2_0.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_2_0.py rename to src/ffx/test/track_tag_combinator_2_0.py diff --git a/bin/ffx/test/track_tag_combinator_2_1.py b/src/ffx/test/track_tag_combinator_2_1.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_2_1.py rename to src/ffx/test/track_tag_combinator_2_1.py diff --git a/bin/ffx/test/track_tag_combinator_2_2.py b/src/ffx/test/track_tag_combinator_2_2.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_2_2.py rename to src/ffx/test/track_tag_combinator_2_2.py diff --git a/bin/ffx/test/track_tag_combinator_2_3.py b/src/ffx/test/track_tag_combinator_2_3.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_2_3.py rename to src/ffx/test/track_tag_combinator_2_3.py diff --git a/bin/ffx/test/track_tag_combinator_3.py b/src/ffx/test/track_tag_combinator_3.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3.py rename to src/ffx/test/track_tag_combinator_3.py diff --git a/bin/ffx/test/track_tag_combinator_3_0.py b/src/ffx/test/track_tag_combinator_3_0.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3_0.py rename to src/ffx/test/track_tag_combinator_3_0.py diff --git a/bin/ffx/test/track_tag_combinator_3_1.py b/src/ffx/test/track_tag_combinator_3_1.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3_1.py rename to src/ffx/test/track_tag_combinator_3_1.py diff --git a/bin/ffx/test/track_tag_combinator_3_2.py b/src/ffx/test/track_tag_combinator_3_2.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3_2.py rename to src/ffx/test/track_tag_combinator_3_2.py diff --git a/bin/ffx/test/track_tag_combinator_3_3.py b/src/ffx/test/track_tag_combinator_3_3.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3_3.py rename to src/ffx/test/track_tag_combinator_3_3.py diff --git a/bin/ffx/test/track_tag_combinator_3_4.py b/src/ffx/test/track_tag_combinator_3_4.py similarity index 100% rename from bin/ffx/test/track_tag_combinator_3_4.py rename to src/ffx/test/track_tag_combinator_3_4.py diff --git a/bin/ffx/tmdb_controller.py b/src/ffx/tmdb_controller.py similarity index 100% rename from bin/ffx/tmdb_controller.py rename to src/ffx/tmdb_controller.py diff --git a/bin/ffx/track_controller.py b/src/ffx/track_controller.py similarity index 100% rename from bin/ffx/track_controller.py rename to src/ffx/track_controller.py diff --git a/bin/ffx/track_delete_screen.py b/src/ffx/track_delete_screen.py similarity index 100% rename from bin/ffx/track_delete_screen.py rename to src/ffx/track_delete_screen.py diff --git a/bin/ffx/track_descriptor.py b/src/ffx/track_descriptor.py similarity index 100% rename from bin/ffx/track_descriptor.py rename to src/ffx/track_descriptor.py diff --git a/bin/ffx/track_details_screen.py b/src/ffx/track_details_screen.py similarity index 98% rename from bin/ffx/track_details_screen.py rename to src/ffx/track_details_screen.py index e64397f..9597c30 100644 --- a/bin/ffx/track_details_screen.py +++ b/src/ffx/track_details_screen.py @@ -54,10 +54,21 @@ class TrackDetailsScreen(Screen): Select { border: none; } + DataTable { min-height: 6; } + DataTable .datatable--cursor { + background: darkorange; + color: black; + } + + DataTable .datatable--header { + background: steelblue; + color: white; + } + #toplabel { height: 1; } diff --git a/bin/ffx/track_disposition.py b/src/ffx/track_disposition.py similarity index 100% rename from bin/ffx/track_disposition.py rename to src/ffx/track_disposition.py diff --git a/bin/ffx/track_type.py b/src/ffx/track_type.py similarity index 100% rename from bin/ffx/track_type.py rename to src/ffx/track_type.py diff --git a/bin/ffx/video_encoder.py b/src/ffx/video_encoder.py similarity index 100% rename from bin/ffx/video_encoder.py rename to src/ffx/video_encoder.py