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