commit 09124a9e8c33408e7c69d8df98ee9cae92c46433 Author: Valeria Fadeeva Date: Mon Nov 13 13:14:34 2023 +0500 cleanup diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..0ad25db --- /dev/null +++ b/LICENSE @@ -0,0 +1,661 @@ + GNU AFFERO GENERAL PUBLIC LICENSE + Version 3, 19 November 2007 + + Copyright (C) 2007 Free Software Foundation, Inc. + Everyone is permitted to copy and distribute verbatim copies + of this license document, but changing it is not allowed. + + Preamble + + The GNU Affero General Public License is a free, copyleft license for +software and other kinds of works, specifically designed to ensure +cooperation with the community in the case of network server software. + + The licenses for most software and other practical works are designed +to take away your freedom to share and change the works. By contrast, +our General Public Licenses are 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. + + 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. + + Developers that use our General Public Licenses protect your rights +with two steps: (1) assert copyright on the software, and (2) offer +you this License which gives you legal permission to copy, distribute +and/or modify the software. + + A secondary benefit of defending all users' freedom is that +improvements made in alternate versions of the program, if they +receive widespread use, become available for other developers to +incorporate. Many developers of free software are heartened and +encouraged by the resulting cooperation. However, in the case of +software used on network servers, this result may fail to come about. +The GNU General Public License permits making a modified version and +letting the public access it on a server without ever releasing its +source code to the public. + + The GNU Affero General Public License is designed specifically to +ensure that, in such cases, the modified source code becomes available +to the community. It requires the operator of a network server to +provide the source code of the modified version running there to the +users of that server. Therefore, public use of a modified version, on +a publicly accessible server, gives the public access to the source +code of the modified version. + + An older license, called the Affero General Public License and +published by Affero, was designed to accomplish similar goals. This is +a different license, not a version of the Affero GPL, but Affero has +released a new version of the Affero GPL which permits relicensing under +this license. + + 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 Affero 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. Remote Network Interaction; Use with the GNU General Public License. + + Notwithstanding any other provision of this License, if you modify the +Program, your modified version must prominently offer all users +interacting with it remotely through a computer network (if your version +supports such interaction) an opportunity to receive the Corresponding +Source of your version by providing access to the Corresponding Source +from a network server at no charge, through some standard or customary +means of facilitating copying of software. This Corresponding Source +shall include the Corresponding Source for any work covered by version 3 +of the GNU General Public License that is incorporated pursuant to the +following paragraph. + + 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 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 work with which it is combined will remain governed by version +3 of the GNU General Public License. + + 14. Revised Versions of this License. + + The Free Software Foundation may publish revised and/or new versions of +the GNU Affero 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 Affero 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 Affero 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 Affero 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 Affero 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 Affero General Public License for more details. + + You should have received a copy of the GNU Affero General Public License + along with this program. If not, see . + +Also add information on how to contact you by electronic and paper mail. + + If your software can interact with users remotely through a computer +network, you should also make sure that it provides a way for users to +get its source. For example, if your program is a web application, its +interface could display a "Source" link that leads users to an archive +of the code. There are many ways you could offer source, and different +solutions will be better for different programs; see section 13 for the +specific requirements. + + 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 AGPL, see +. diff --git a/README.md b/README.md new file mode 100644 index 0000000..f04673d --- /dev/null +++ b/README.md @@ -0,0 +1,11 @@ +# melawy-plasma-plasmoid-presentwindows +Show desktop mosaic plasmoid + +### Donate +[Tinkoff](https://www.tinkoff.ru/rm/fadeeva.valeriya96/9bLRi79066) + +[YooMoney](https://yoomoney.ru/to/4100115921160758) + +[Qiwi](https://qiwi.com/n/VALERIAFADEEVA) + +Etherium 0x981FBf878fe451BDB83BEaF68078394d4B13213f diff --git a/cleanup.sh b/cleanup.sh new file mode 100755 index 0000000..cb4c124 --- /dev/null +++ b/cleanup.sh @@ -0,0 +1,19 @@ +#!/bin/bash + +mv .git/config config + +rm -rf .git + +git init -b main + +mv config .git/config + +git add --all . + +git commit -m "cleanup" + +git push origin main --force + +echo "################################################################" +echo "################### cleanup Done ######################" +echo "################################################################" diff --git a/com.github.zren.presentwindows/contents/config/config.qml b/com.github.zren.presentwindows/contents/config/config.qml new file mode 100644 index 0000000..a303bc5 --- /dev/null +++ b/com.github.zren.presentwindows/contents/config/config.qml @@ -0,0 +1,10 @@ +import QtQuick 2.0 +import org.kde.plasma.configuration 2.0 + +ConfigModel { + ConfigCategory { + name: i18n("General") + icon: "configure" + source: "config/ConfigGeneral.qml" + } +} diff --git a/com.github.zren.presentwindows/contents/config/main.xml b/com.github.zren.presentwindows/contents/config/main.xml new file mode 100644 index 0000000..f792db3 --- /dev/null +++ b/com.github.zren.presentwindows/contents/config/main.xml @@ -0,0 +1,22 @@ + + + + + + + ExposeAll + + + presentwindows-24px + + + + 1 + + + + + diff --git a/com.github.zren.presentwindows/contents/icons/presentwindows-16px.svg b/com.github.zren.presentwindows/contents/icons/presentwindows-16px.svg new file mode 100644 index 0000000..9a9f6db --- /dev/null +++ b/com.github.zren.presentwindows/contents/icons/presentwindows-16px.svg @@ -0,0 +1,18 @@ + + + + + + + + diff --git a/com.github.zren.presentwindows/contents/icons/presentwindows-22px.svg b/com.github.zren.presentwindows/contents/icons/presentwindows-22px.svg new file mode 100644 index 0000000..b4557a1 --- /dev/null +++ b/com.github.zren.presentwindows/contents/icons/presentwindows-22px.svg @@ -0,0 +1,18 @@ + + + + + + + + diff --git a/com.github.zren.presentwindows/contents/icons/presentwindows-24px.svg b/com.github.zren.presentwindows/contents/icons/presentwindows-24px.svg new file mode 100644 index 0000000..c4a4c2c --- /dev/null +++ b/com.github.zren.presentwindows/contents/icons/presentwindows-24px.svg @@ -0,0 +1,18 @@ + + + + + + + + diff --git a/com.github.zren.presentwindows/contents/icons/unity7selectedworkspace.svg b/com.github.zren.presentwindows/contents/icons/unity7selectedworkspace.svg new file mode 100644 index 0000000..18fc31e --- /dev/null +++ b/com.github.zren.presentwindows/contents/icons/unity7selectedworkspace.svg @@ -0,0 +1,147 @@ + + + + + + + + + + + + + + + + + image/svg+xml + + + + + + + + + + + + + + + + diff --git a/com.github.zren.presentwindows/contents/locale/nl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo b/com.github.zren.presentwindows/contents/locale/nl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo new file mode 100644 index 0000000..edd0a22 Binary files /dev/null and b/com.github.zren.presentwindows/contents/locale/nl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo differ diff --git a/com.github.zren.presentwindows/contents/locale/nl_NL/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo b/com.github.zren.presentwindows/contents/locale/nl_NL/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo new file mode 100644 index 0000000..42f883d Binary files /dev/null and b/com.github.zren.presentwindows/contents/locale/nl_NL/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo differ diff --git a/com.github.zren.presentwindows/contents/locale/pl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo b/com.github.zren.presentwindows/contents/locale/pl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo new file mode 100644 index 0000000..ca732f0 Binary files /dev/null and b/com.github.zren.presentwindows/contents/locale/pl/LC_MESSAGES/plasma_applet_com.github.zren.presentwindows.mo differ diff --git a/com.github.zren.presentwindows/contents/ui/Main.qml b/com.github.zren.presentwindows/contents/ui/Main.qml new file mode 100644 index 0000000..03878e5 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/Main.qml @@ -0,0 +1,136 @@ +import QtQuick 2.0 +import QtQuick.Layouts 1.1 +import org.kde.plasma.plasmoid 2.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +import "lib" as Lib + +Item { + id: widget + + UnityThemeDetector { + id: unityThemeDetector + } + + property bool disableLatteParabolicIcon: true // Don't hide the representation in Latte (https://github.com/psifidotos/Latte-Dock/issues/983) + + Plasmoid.onActivated: widget.activate() + + Plasmoid.preferredRepresentation: Plasmoid.fullRepresentation + Plasmoid.fullRepresentation: Item { + id: panelItem + + readonly property bool inPanel: (plasmoid.location == PlasmaCore.Types.TopEdge + || plasmoid.location == PlasmaCore.Types.RightEdge + || plasmoid.location == PlasmaCore.Types.BottomEdge + || plasmoid.location == PlasmaCore.Types.LeftEdge) + + Layout.minimumWidth: { + switch (plasmoid.formFactor) { + case PlasmaCore.Types.Vertical: + return 0; + case PlasmaCore.Types.Horizontal: + return height; + default: + return PlasmaCore.Units.gridUnit * 3; + } + } + + Layout.minimumHeight: { + switch (plasmoid.formFactor) { + case PlasmaCore.Types.Vertical: + return width; + case PlasmaCore.Types.Horizontal: + return 0; + default: + return PlasmaCore.Units.gridUnit * 3; + } + } + + Layout.maximumWidth: inPanel ? PlasmaCore.Units.iconSizeHints.panel : -1 + Layout.maximumHeight: inPanel ? PlasmaCore.Units.iconSizeHints.panel : -1 + + Lib.AppletIcon { + id: icon + anchors.fill: parent + visible: !unityThemeDetector.useUnityTheme + + source: plasmoid.configuration.icon + active: mouseArea.containsMouse + } + Loader { + anchors.fill: parent + active: unityThemeDetector.useUnityTheme + visible: active + source: "Unity7Workspaces.qml" + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + onClicked: widget.activate() + } + } + + PlasmaCore.DataSource { + id: executable + engine: "executable" + connectedSources: [] + onNewData: disconnectSource(sourceName) + + function exec(cmd) { + executable.connectSource(cmd) + } + } + + function action_showOverview() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "Overview"') + } + + function action_exposeAll() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "ExposeAll"') + } + + function action_exposeDesktop() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "Expose"') + } + + function action_exposeWindowClass() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "ExposeClass"') + } + + function action_showDesktopGrid() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "ShowDesktopGrid"') + } + + function action_toggleParachute() { + executable.exec('qdbus org.kde.kglobalaccel /component/kwin invokeShortcut "Parachute"') + } + + function activate() { + if (plasmoid.configuration.clickCommand == 'Overview') { + action_showOverview() + } else if (plasmoid.configuration.clickCommand == 'ExposeAll') { + action_exposeAll() + } else if (plasmoid.configuration.clickCommand == 'Expose') { + action_exposeDesktop() + } else if (plasmoid.configuration.clickCommand == 'ExposeClass') { + action_exposeWindowClass() + } else if (plasmoid.configuration.clickCommand == 'ShowDesktopGrid') { + action_showDesktopGrid() + } else if (plasmoid.configuration.clickCommand == 'Parachute') { + action_toggleParachute() + } + } + + Component.onCompleted: { + plasmoid.setAction("showOverview", i18n("Show Overview"), "view-grid"); + plasmoid.setAction("exposeAll", i18n("Present Windows (All desktops)"), "window"); + plasmoid.setAction("exposeDesktop", i18n("Present Windows (Current desktop)"), "window"); + plasmoid.setAction("exposeWindowClass", i18n("Present Windows (Window class)"), "window"); + plasmoid.setAction("showDesktopGrid", i18n("Show Desktop Grid"), "view-grid"); + + // plasmoid.action('configure').trigger() // Uncomment to open the config window on load. + } +} diff --git a/com.github.zren.presentwindows/contents/ui/Unity7Workspaces.qml b/com.github.zren.presentwindows/contents/ui/Unity7Workspaces.qml new file mode 100644 index 0000000..493b105 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/Unity7Workspaces.qml @@ -0,0 +1,127 @@ +import QtQuick 2.0 +import QtQuick.Layouts 1.1 +import org.kde.plasma.plasmoid 2.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +import org.kde.plasma.private.pager 2.0 + +import "lib" + +Item { + id: panelItem + + // Heavy use of the default Pager's code. + // See: /usr/share/plasma/plasmoid/org.kde.plasma.pager/contents/ui/main.qml + PagerModel { + id: pagerModel + + enabled: true + showOnlyCurrentScreen: true //plasmoid.configuration.showOnlyCurrentScreen + screenGeometry: plasmoid.screenGeometry + pagerType: PagerModel.VirtualDesktops + } + + PlasmaCore.FrameSvgItem { + id: taskFrame + anchors.fill: parent + imagePath: "widgets/tasks" + prefix: "normal" + } + + Grid { + id: pagerItemGrid + anchors.fill: parent + + anchors.leftMargin: taskFrame.margins.left + anchors.rightMargin: taskFrame.margins.right + anchors.topMargin: taskFrame.margins.top + anchors.bottomMargin: taskFrame.margins.bottom + + // spacing: PlasmaCore.Units.devicePixelRatio + rows: effectiveRows + columns: effectiveColumns + + readonly property int effectiveRows: { + var rows = 1 + var columns = Math.floor(pagerModel.count / pagerModel.layoutRows) + + if (pagerModel.count % pagerModel.layoutRows > 0) { + columns += 1 + } + + rows = Math.floor(pagerModel.count / columns) + + if (pagerModel.count % columns > 0) { + rows += 1 + } + + return rows + } + + readonly property int effectiveColumns: { + if (!pagerModel.count) { + return 1 + } + + return Math.ceil(pagerModel.count / effectiveRows) + } + + + readonly property real pagerItemSizeRatio: pagerModel.pagerItemSize.width / pagerModel.pagerItemSize.height + // readonly property real widthScaleFactor: columnWidth / pagerModel.pagerItemSize.width + // readonly property real heightScaleFactor: rowHeight / pagerModel.pagerItemSize.height + + property int rowHeight: Math.floor(height / effectiveRows) + property int columnWidth: Math.floor(width / effectiveColumns) + + property color seperatorColor: "#44FFFFFF" + property color activeDesktopFillColor: "#44442027" + + Repeater { + id: repeater + model: pagerModel + + Item { + id: desktop + property int desktopIndex: index + property int desktopColumn: index % pagerItemGrid.columns + property int desktopRow: Math.floor(index / pagerItemGrid.columns) + property bool isActiveDesktop: (index == pagerModel.currentPage) + + width: pagerItemGrid.columnWidth + height: pagerItemGrid.rowHeight + + + Rectangle { + anchors.fill: parent + color: desktop.isActiveDesktop ? pagerItemGrid.activeDesktopFillColor : "transparent" + } + + Rectangle { + id: verticalSeperator + visible: desktop.desktopColumn < pagerItemGrid.columns-1 // Don't show on last column + anchors.top: parent.top + anchors.right: parent.right + anchors.bottom: parent.bottom + width: Math.round(1 * PlasmaCore.Units.devicePixelRatio) + color: pagerItemGrid.seperatorColor + } + Rectangle { + id: horizontalSeperator + visible: desktop.desktopRow < pagerItemGrid.rows-1 // Don't show on last row + anchors.left: parent.left + anchors.bottom: parent.bottom + anchors.right: parent.right + height: Math.round(1 * PlasmaCore.Units.devicePixelRatio) + color: pagerItemGrid.seperatorColor + } + + AppletIcon { + anchors.fill: parent + source: desktop.isActiveDesktop ? "unity7selectedworkspace" : "" + } + } + + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/UnityThemeDetector.qml b/com.github.zren.presentwindows/contents/ui/UnityThemeDetector.qml new file mode 100644 index 0000000..85fc451 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/UnityThemeDetector.qml @@ -0,0 +1,19 @@ +import QtQuick 2.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +QtObject { + readonly property bool shouldUseUnityTheme: { + return PlasmaCore.Theme.themeName == 'UnityAmbiance' + } + readonly property int configState: plasmoid.configuration.useUnityTheme + readonly property bool useUnityTheme: { + // console.log('useUnityTheme', configState, shouldUseUnityTheme) + if (configState == 0) { // Never + return false + } else if (configState == 1) { // DependsOnPlasmaStyle + return shouldUseUnityTheme + } else { // configState == 2 Always + return true + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/config/ConfigGeneral.qml b/com.github.zren.presentwindows/contents/ui/config/ConfigGeneral.qml new file mode 100644 index 0000000..cd034ee --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/config/ConfigGeneral.qml @@ -0,0 +1,149 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Controls 2.5 as QQC2 +import QtQuick.Layouts 1.0 + +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.kirigami 2.5 as Kirigami + +import "../lib" as Lib +import "../libconfig" as LibConfig + + +Kirigami.FormLayout { + + + //------------------------------------------------------- + LibConfig.Heading { + text: i18n("Behavior") + useThickTopMargin: false + } + Lib.DesktopEffectToggle { + id: overviewToggle + effectId: 'overview' + } + Lib.DesktopEffectToggle { + id: presentWindowsToggle + effectId: 'presentwindows' + } + Lib.DesktopEffectToggle { + id: showDesktopGridToggle + effectId: 'desktopgrid' + } + LibConfig.RadioButtonGroup { + id: clickCommandGroup + configKey: 'clickCommand' + Kirigami.FormData.label: i18n("Click") + Kirigami.FormData.buddyFor: null // Note: it attaches to the first CheckBox in the Repeater since it loads first. + model: [] + + //--- + Repeater { + model: [ + { value: 'Overview', text: i18nd("kwin_effects", "Toggle Overview"), effectToggle: overviewToggle }, + { value: 'ExposeAll', text: i18nd("kwin_effects", "Toggle Present Windows (All desktops)"), effectToggle: presentWindowsToggle }, + { value: 'Expose', text: i18nd("kwin_effects", "Toggle Present Windows (Current desktop)"), effectToggle: presentWindowsToggle }, + { value: 'ExposeClass', text: i18nd("kwin_effects", "Toggle Present Windows (Window class)"), effectToggle: presentWindowsToggle }, + { value: 'ShowDesktopGrid', text: i18nd("kwin_effects", "Toggle Desktop Grid"), effectToggle: showDesktopGridToggle }, + ] + RowLayout { + QQC2.RadioButton { + text: modelData.text + enabled: modelData.effectToggle.loaded && modelData.effectToggle.effectEnabled + QQC2.ButtonGroup.group: clickCommandGroup.group + checked: modelData.value === plasmoid.configuration[clickCommandGroup.configKey] + onClicked: { + focus = true + if (clickCommandGroup.configKey) { + plasmoid.configuration[clickCommandGroup.configKey] = modelData.value + } + } + } + QQC2.Button { + visible: modelData.effectToggle.loaded && !modelData.effectToggle.effectEnabled + text: i18n("Enable Desktop Effect") + onClicked: modelData.effectToggle.toggle() + } + } + } + + //--- + Lib.KPackageModel { + id: kwinScriptModel + packageType: 'KWin/Script' + } + Repeater { + visible: kwinScriptModel.loaded + model: [ + { + pluginId: 'Parachute', + url: 'https://store.kde.org/p/1370195/', + }, + ] + RowLayout { + QQC2.RadioButton { + text: modelData.pluginId + enabled: kwinScriptModel.contains(modelData.pluginId) + QQC2.ButtonGroup.group: clickCommandGroup.group + checked: modelData.value === plasmoid.configuration[clickCommandGroup.configKey] + onClicked: { + focus = true + if (clickCommandGroup.configKey) { + plasmoid.configuration[clickCommandGroup.configKey] = modelData.value + } + } + } + LibConfig.Label { + text: '' + modelData.url + '' + } + } + } + } + + //------------------------------------------------------- + LibConfig.Heading { + text: i18n("Appearance") + } + + LibConfig.AppletIconField { + id: iconField + configKey: 'icon' + Kirigami.FormData.label: i18n("Icon") + defaultValue: 'presentwindows-24px' + presetValues: [ + 'presentwindows-24px', + 'presentwindows-22px', + 'presentwindows-16px', + 'edit-group', + 'window', + 'view-app-grid-symbolic', + 'homerun', + ] + } + + + + //------------------------------------------------------- + LibConfig.Heading { + text: i18n("Unity Pager Theme") + } + + LibConfig.RadioButtonGroup { + configKey: 'useUnityTheme' + Kirigami.FormData.label: i18n("Use Unity 7 Theme") + model: [ + { value: 0, text: i18n("Never") }, + { value: 1, text: i18n("When Plasma Style is Unity Ambiance") }, + { value: 2, text: i18n("Always") }, + ] + } + + LibConfig.Label { + text: i18n("Should we use a Virtual Desktop indicator similar to Unity 7? This feature is enabled for the Unity Ambiance Plasma Style by default.") + } + + Label { + text: i18n("Current Plasma Style: %1", PlasmaCore.Theme.themeName) + } + +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/AppletIcon.qml b/com.github.zren.presentwindows/contents/ui/lib/AppletIcon.qml new file mode 100644 index 0000000..844b337 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/AppletIcon.qml @@ -0,0 +1,40 @@ +// Version: 4 + +import QtQuick 2.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +Item { + id: appletIcon + property string source: '' + property bool active: false + readonly property bool usingPackageSvg: filename // plasmoid.file() returns "" if file doesn't exist. + readonly property string filename: source ? plasmoid.file('', 'icons/' + source + '.svg') : '' + readonly property int minSize: Math.min(width, height) + property bool smooth: true + + PlasmaCore.IconItem { + anchors.fill: parent + visible: !appletIcon.usingPackageSvg + source: appletIcon.usingPackageSvg ? '' : appletIcon.source + active: appletIcon.active + smooth: appletIcon.smooth + } + + PlasmaCore.SvgItem { + id: svgItem + anchors.centerIn: parent + readonly property real minSize: Math.min(naturalSize.width, naturalSize.height) + readonly property real widthRatio: naturalSize.width / svgItem.minSize + readonly property real heightRatio: naturalSize.height / svgItem.minSize + width: appletIcon.minSize * widthRatio + height: appletIcon.minSize * heightRatio + + smooth: appletIcon.smooth + + visible: appletIcon.usingPackageSvg + svg: PlasmaCore.Svg { + id: svg + imagePath: appletIcon.filename + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/AppletVersion.qml b/com.github.zren.presentwindows/contents/ui/lib/AppletVersion.qml new file mode 100644 index 0000000..c42f0b1 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/AppletVersion.qml @@ -0,0 +1,49 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.plasma.plasmoid 2.0 + +Item { + implicitWidth: label.implicitWidth + implicitHeight: label.implicitHeight + + property string version: "?" + property string metadataFilepath: plasmoid.file("", "../metadata.desktop") + + PlasmaCore.DataSource { + id: executable + engine: "executable" + connectedSources: [] + onNewData: { + var exitCode = data["exit code"] + var exitStatus = data["exit status"] + var stdout = data["stdout"] + var stderr = data["stderr"] + exited(sourceName, exitCode, exitStatus, stdout, stderr) + disconnectSource(sourceName) // cmd finished + } + function exec(cmd) { + connectSource(cmd) + } + signal exited(string command, int exitCode, int exitStatus, string stdout, string stderr) + } + + Connections { + target: executable + onExited: { + version = stdout.replace('\n', ' ').trim() + } + } + + Label { + id: label + text: i18n("Version: %1", version) + } + + Component.onCompleted: { + var cmd = 'kreadconfig5 --file "' + metadataFilepath + '" --group "Desktop Entry" --key "X-KDE-PluginInfo-Version"' + executable.exec(cmd) + } + +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ConfigIcon.qml b/com.github.zren.presentwindows/contents/ui/lib/ConfigIcon.qml new file mode 100644 index 0000000..7e373a9 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ConfigIcon.qml @@ -0,0 +1,187 @@ +// Version 3 +// Forked to use AppletIcon + +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 +import QtQuick.Dialogs 1.2 + +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.plasma.components 2.0 as PlasmaComponents +import org.kde.kquickcontrolsaddons 2.0 as KQuickAddons + +import ".." + +RowLayout { + id: configIcon + + default property alias _contentChildren: content.data + + property string configKey: '' + property alias value: textField.text + readonly property string configValue: configKey ? plasmoid.configuration[configKey] : "" + onConfigValueChanged: { + if (!textField.focus && value != configValue) { + value = configValue + } + } + property int previewIconSize: units.iconSizes.medium + property string defaultValue: "start-here-kde" + property var presetValues: [] + + onPresetValuesChanged: iconMenu.refresh() + + // Used for binding in presetValue menu loop + function setValue(val) { + configIcon.value = val + } + + // org.kde.plasma.kickoff + Button { + id: iconButton + Layout.minimumWidth: previewFrame.width + units.smallSpacing * 2 + Layout.maximumWidth: Layout.minimumWidth + Layout.minimumHeight: previewFrame.height + units.smallSpacing * 2 + Layout.maximumHeight: Layout.minimumWidth + + + + // just to provide some visual feedback, cannot have checked without checkable enabled + checkable: true + onClicked: { + checked = Qt.binding(function() { // never actually allow it being checked + return iconMenu.status === PlasmaComponents.DialogStatus.Open + }) + + iconMenu.open(0, height) + } + + PlasmaCore.FrameSvgItem { + id: previewFrame + anchors.centerIn: parent + imagePath: plasmoid.location === PlasmaCore.Types.Vertical || plasmoid.location === PlasmaCore.Types.Horizontal + ? "widgets/panel-background" : "widgets/background" + width: previewIconSize + fixedMargins.left + fixedMargins.right + height: previewIconSize + fixedMargins.top + fixedMargins.bottom + + AppletIcon { + anchors.centerIn: parent + width: previewIconSize + height: previewIconSize + source: configIcon.value + } + } + + // QQC Menu can only be opened at cursor position, not a random one + PlasmaComponents.ContextMenu { + id: iconMenu + visualParent: iconButton + + function newMenuItem(parent) { + return Qt.createQmlObject( + "import org.kde.plasma.components 2.0 as PlasmaComponents;" + + "PlasmaComponents.MenuItem {}", + parent); + } + + function newSeparator(parent) { + return Qt.createQmlObject( + "import org.kde.plasma.components 2.0 as PlasmaComponents;" + + "PlasmaComponents.MenuItem { separator: true }", + parent); + } + + function refresh() { + clearMenuItems() + + // Choose... + var menuItem = newMenuItem(iconMenu) + menuItem.text = i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Open icon chooser dialog", "Choose...") + menuItem.icon = "document-open" + menuItem.clicked.connect(function(){ + iconDialog.open() + }) + iconMenu.addMenuItem(menuItem) + + // Clear + var menuItem = newMenuItem(iconMenu) + menuItem.text = i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Reset icon to default", "Clear Icon") + menuItem.icon = "edit-clear" + menuItem.clicked.connect(function(){ + configIcon.value = defaultValue + }) + iconMenu.addMenuItem(menuItem) + + // Preset Values + if (configIcon.presetValues.length > 0) { + menuItem = newSeparator(iconMenu) + iconMenu.addMenuItem(menuItem) + + for (var i = 0; i < configIcon.presetValues.length; i++) { + var presetValue = configIcon.presetValues[i] + menuItem = newMenuItem(iconMenu) + menuItem.text = presetValue + menuItem.icon = presetValue + menuItem.clicked.connect(configIcon.setValue.bind(this, presetValue)) + iconMenu.addMenuItem(menuItem) + } + } + } + + Component.onCompleted: { + refresh() + } + } + } + + ColumnLayout { + id: content + Layout.fillWidth: true + + RowLayout { + TextField { + id: textField + Layout.fillWidth: true + + text: configIcon.configValue + onTextChanged: serializeTimer.restart() + + ToolButton { + iconName: "edit-clear" + onClicked: configIcon.value = defaultValue + + anchors.top: parent.top + anchors.right: parent.right + anchors.bottom: parent.bottom + + width: height + } + } + + Button { + iconName: "document-open" + onClicked: iconDialog.open() + } + } + + // Workaround for crash when using default on a Layout. + // https://bugreports.qt.io/browse/QTBUG-52490 + // Still affecting Qt 5.7.0 + Component.onDestruction: { + while (children.length > 0) { + children[children.length - 1].parent = configIcon; + } + } + } + + KQuickAddons.IconDialog { + id: iconDialog + onIconNameChanged: configIcon.value = iconName + } + + Timer { // throttle + id: serializeTimer + interval: 300 + onTriggered: plasmoid.configuration[configKey] = configIcon.value + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ConfigLabel.qml b/com.github.zren.presentwindows/contents/ui/lib/ConfigLabel.qml new file mode 100644 index 0000000..48b9502 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ConfigLabel.qml @@ -0,0 +1,17 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +Label { + Layout.fillWidth: true + wrapMode: Text.Wrap + linkColor: PlasmaCore.ColorScope.highlightColor + onLinkActivated: Qt.openUrlExternally(link) + + MouseArea { + anchors.fill: parent + acceptedButtons: Qt.NoButton // we don't want to eat clicks on the Text + cursorShape: parent.hoveredLink ? Qt.PointingHandCursor : Qt.ArrowCursor + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ConfigPage.qml b/com.github.zren.presentwindows/contents/ui/lib/ConfigPage.qml new file mode 100644 index 0000000..5bff015 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ConfigPage.qml @@ -0,0 +1,37 @@ +// Version 3 + +import QtQuick 2.0 +import QtQuick.Layouts 1.0 + +Item { + id: page + Layout.fillWidth: true + default property alias _contentChildren: content.data + implicitHeight: content.implicitHeight + + ColumnLayout { + id: content + anchors.left: parent.left + anchors.right: parent.right + anchors.top: parent.top + + // Workaround for crash when using default on a Layout. + // https://bugreports.qt.io/browse/QTBUG-52490 + // Still affecting Qt 5.7.0 + Component.onDestruction: { + while (children.length > 0) { + children[children.length - 1].parent = page; + } + } + } + + property alias showAppletVersion: appletVersionLoader.active + Loader { + id: appletVersionLoader + active: false + visible: active + source: "AppletVersion.qml" + anchors.right: parent.right + anchors.bottom: parent.top + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ConfigSection.qml b/com.github.zren.presentwindows/contents/ui/lib/ConfigSection.qml new file mode 100644 index 0000000..ecdd5bb --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ConfigSection.qml @@ -0,0 +1,52 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 + +// Alternative to GroupBox for when we want the title to always be left aligned. +Rectangle { + id: control + Layout.fillWidth: true + default property alias _contentChildren: content.data + property string label: "" + + color: "#0c000000" + border.width: 2 + border.color: "#10000000" + // radius: 5 + property int padding: 8 + implicitHeight: childrenRect.height + padding + padding + property alias spacing: content.spacing + + Label { + id: title + visible: control.label + text: control.label + font.bold: true + font.pointSize: 13 + anchors.leftMargin: padding + // anchors.topMargin: padding + anchors.left: parent.left + anchors.top: parent.top + anchors.right: parent.right + height: visible ? implicitHeight : padding + } + + ColumnLayout { + id: content + anchors.top: title.bottom + anchors.left: parent.left + anchors.right: parent.right + anchors.margins: padding + // spacing: 0 + // height: childrenRect.height + + // Workaround for crash when using default on a Layout. + // https://bugreports.qt.io/browse/QTBUG-52490 + // Still affecting Qt 5.7.0 + Component.onDestruction: { + while (children.length > 0) { + children[children.length - 1].parent = control; + } + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ConfigTriStateCheckBox.qml b/com.github.zren.presentwindows/contents/ui/lib/ConfigTriStateCheckBox.qml new file mode 100644 index 0000000..ad57d92 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ConfigTriStateCheckBox.qml @@ -0,0 +1,17 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 + +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.plasma.components 2.0 as PlasmaComponents + +import ".." + +CheckBox { + id: configTriStateCheckBox + + property string configKey: '' + partiallyCheckedEnabled: true + checkedState: plasmoid.configuration[configKey] + onClicked: plasmoid.configuration[configKey] = (plasmoid.configuration[configKey] + 1) % 3 +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/DesktopEffectToggle.qml b/com.github.zren.presentwindows/contents/ui/lib/DesktopEffectToggle.qml new file mode 100644 index 0000000..5d138a9 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/DesktopEffectToggle.qml @@ -0,0 +1,67 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import QtQuick.Layouts 1.0 + +QtObject { + id: desktopEffectToggle + property string effectId + property bool loaded: false + property bool effectEnabled: false + + + function toggle() { + executable.toggleState() + } + + property ExecUtil executable: ExecUtil { + id: executable + property string readStateCommand: 'qdbus org.kde.KWin /Effects isEffectLoaded ' + effectId + property string toggleStateCommand: 'qdbus org.kde.KWin /Effects toggleEffect ' + effectId + + // For some reason, the toggleEffect qdbus function does not save to kwinrc. + // So we need to manually write the new state to it, so that the next time + // kwin is launched (next reboot), the desktop effect is still enabled/disabled. + property string saveStateCommand: 'kwriteconfig5 --file ~/.config/kwinrc --group Plugins --key ' + effectId + 'Enabled' // saveStateCommand + ' ' + value + property bool saveOnRead: false + + function readState() { + executable.exec(readStateCommand) + } + function toggleState() { + executable.exec(toggleStateCommand) + } + function saveState() { + var isCurrentlyEnabled = effectEnabled + executable.exec(saveStateCommand + ' ' + (isCurrentlyEnabled ? 'true' : 'false')) + } + Component.onCompleted: { + readState() + } + + onExited: { + if (command == readStateCommand) { + var value = executable.trimOutput(stdout) + value = value === 'true' // cast to boolean + effectEnabled = value + loaded = true + if (saveOnRead) { + saveOnRead = false + saveState() + } + } else if (command == toggleStateCommand) { + saveOnRead = true + readState() + } else if (startsWith(command, saveStateCommand)) { + + } + } + + function startsWith(a, b) { + if (b.length <= a.length) { + return a.substr(0, b.length) == b + } else { + return false + } + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/ExecUtil.qml b/com.github.zren.presentwindows/contents/ui/lib/ExecUtil.qml new file mode 100644 index 0000000..f0f60ff --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/ExecUtil.qml @@ -0,0 +1,24 @@ +import QtQuick 2.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +PlasmaCore.DataSource { + id: executable + engine: "executable" + connectedSources: [] + onNewData: { + var exitCode = data["exit code"] + var exitStatus = data["exit status"] + var stdout = data["stdout"] + var stderr = data["stderr"] + exited(sourceName, exitCode, exitStatus, stdout, stderr) + disconnectSource(sourceName) // cmd finished + } + function exec(cmd) { + connectSource(cmd) + } + signal exited(string command, int exitCode, int exitStatus, string stdout, string stderr) + + function trimOutput(stdout) { + return stdout.replace('\n', ' ').trim() + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/KPackageModel.qml b/com.github.zren.presentwindows/contents/ui/lib/KPackageModel.qml new file mode 100644 index 0000000..2d02ad9 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/KPackageModel.qml @@ -0,0 +1,64 @@ +// Version 1 + +import QtQuick 2.0 + +ListModel { + property string packageType + property bool loaded: false + + readonly property var executable: ExecUtil { + id: executable + property string readStateCommand: '( kpackagetool5 --type="' + packageType + '" --list ; kpackagetool5 --g --type="' + packageType + '" --list ) | cat' + + function readState() { + executable.exec(readStateCommand) + } + Component.onCompleted: { + readState() + } + + function parsePackageList(stdout) { + clear() + var lines = stdout.split('\n') + for (var i = 0; i < lines.length; i++) { + var line = lines[i] + if (line.indexOf(packageType) >= 0) { + // Treat line as: + // Listing service types: KWin/Script in /usr/share/kwin/scripts/ + continue + } + var pluginId = line.trim() + if (pluginId) { + append({ + pluginId: pluginId, + }) + } + } + } + + onExited: { + if (command == readStateCommand) { + parsePackageList(stdout) + loaded = true + } + } + + function startsWith(a, b) { + if (b.length <= a.length) { + return a.substr(0, b.length) == b + } else { + return false + } + } + } + + function contains(pluginId) { + for (var i = 0; i < count; i++) { + var item = get(i) + if (item.pluginId == pluginId) { + return true + } + } + return false + } +} diff --git a/com.github.zren.presentwindows/contents/ui/lib/LinkText.qml b/com.github.zren.presentwindows/contents/ui/lib/LinkText.qml new file mode 100644 index 0000000..31d2b6c --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/lib/LinkText.qml @@ -0,0 +1,13 @@ +import QtQuick 2.0 +import QtQuick.Controls 1.0 +import org.kde.plasma.core 2.0 as PlasmaCore + +Label { + linkColor: PlasmaCore.ColorScope.highlightColor + onLinkActivated: Qt.openUrlExternally(link) + MouseArea { + anchors.fill: parent + acceptedButtons: Qt.NoButton // we don't want to eat clicks on the Text + cursorShape: parent.hoveredLink ? Qt.PointingHandCursor : Qt.ArrowCursor + } +} diff --git a/com.github.zren.presentwindows/contents/ui/libconfig/AppletIconField.qml b/com.github.zren.presentwindows/contents/ui/libconfig/AppletIconField.qml new file mode 100644 index 0000000..d1b283c --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/libconfig/AppletIconField.qml @@ -0,0 +1,157 @@ +// Version 9 +// Forked LibConfig.IconField to use Lib.AppletIcon + +import QtQuick 2.0 +import QtQuick.Controls 2.0 as QQC2 +import QtQuick.Layouts 1.0 + +import org.kde.kirigami 2.0 as Kirigami +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.kquickcontrolsaddons 2.0 as KQuickAddons + +import "../lib" as Lib + +RowLayout { + id: iconField + + default property alias _contentChildren: content.data + + property string configKey: '' + property alias value: textField.text + readonly property string configValue: configKey ? plasmoid.configuration[configKey] : "" + onConfigValueChanged: { + if (!textField.focus && value != configValue) { + value = configValue + } + } + property int previewIconSize: Kirigami.Units.iconSizes.medium + property string defaultValue: "" + property alias placeholderValue: textField.placeholderText + property var presetValues: [] + + // Based on org.kde.plasma.kickoff + QQC2.Button { + id: iconButton + padding: Kirigami.Units.smallSpacing + Layout.alignment: Qt.AlignTop + + // KDE QQC2 sets implicitSize to background.implicitSize ignoring padding/inset properties. + implicitWidth: leftPadding + contentItem.implicitWidth + rightPadding + implicitHeight: topPadding + contentItem.implicitHeight + bottomPadding + + onPressed: iconMenu.opened ? iconMenu.close() : iconMenu.open() + + contentItem: PlasmaCore.FrameSvgItem { + id: previewFrame + imagePath: plasmoid.location === PlasmaCore.Types.Vertical || plasmoid.location === PlasmaCore.Types.Horizontal + ? "widgets/panel-background" : "widgets/background" + implicitWidth: fixedMargins.left + previewIconSize + fixedMargins.right + implicitHeight: fixedMargins.top + previewIconSize + fixedMargins.bottom + + Lib.AppletIcon { + anchors.fill: parent + anchors.leftMargin: previewFrame.fixedMargins.left + anchors.topMargin: previewFrame.fixedMargins.top + anchors.rightMargin: previewFrame.fixedMargins.right + anchors.bottomMargin: previewFrame.fixedMargins.bottom + source: iconField.value || iconField.placeholderValue + active: iconButton.hovered + } + } + + QQC2.Menu { + id: iconMenu + + // Appear below the button + y: +parent.height + + QQC2.MenuItem { + text: i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Open icon chooser dialog", "Choose...") + icon.name: "document-open" + onClicked: dialogLoader.active = true + } + QQC2.MenuItem { + text: i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Reset icon to default", "Clear Icon") + icon.name: "edit-clear" + onClicked: iconField.value = iconField.defaultValue + } + } + } + + ColumnLayout { + id: content + Layout.fillWidth: true + + RowLayout { + QQC2.TextField { + id: textField + Layout.fillWidth: true + + text: iconField.configValue + onTextChanged: serializeTimer.restart() + + rightPadding: clearButton.width + Kirigami.Units.smallSpacing + + QQC2.ToolButton { + id: clearButton + visible: iconField.configValue != iconField.defaultValue + icon.name: iconField.defaultValue === "" ? "edit-clear" : "edit-undo" + onClicked: iconField.value = iconField.defaultValue + + anchors.top: parent.top + anchors.right: parent.right + anchors.bottom: parent.bottom + + width: height + } + } + + QQC2.Button { + id: browseButton + icon.name: "document-open" + onClicked: dialogLoader.active = true + } + } + + Flow { + Layout.fillWidth: true + Layout.maximumWidth: Kirigami.Units.gridUnit * 30 + Repeater { + model: presetValues + QQC2.Button { + icon.name: modelData + text: modelData + onClicked: iconField.value = modelData + } + } + } + } + + Loader { + id: dialogLoader + active: false + sourceComponent: KQuickAddons.IconDialog { + id: dialog + visible: true + modality: Qt.WindowModal + onIconNameChanged: { + iconField.value = iconName + } + onVisibleChanged: { + if (!visible) { + dialogLoader.active = false + } + } + } + } + + Timer { // throttle + id: serializeTimer + interval: 300 + onTriggered: { + if (configKey) { + plasmoid.configuration[configKey] = iconField.value + } + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/libconfig/Heading.qml b/com.github.zren.presentwindows/contents/ui/libconfig/Heading.qml new file mode 100644 index 0000000..e7280bd --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/libconfig/Heading.qml @@ -0,0 +1,83 @@ +// Version 5 + +import QtQuick 2.0 +import QtQuick.Controls 2.5 as QQC2 +import QtQuick.Layouts 1.0 +import org.kde.kirigami 2.5 as Kirigami + +/* +** Example: +** +import './libconfig' as LibConfig +LibConfig.Heading { + text: i18n("SpinBox (Double)") +} +*/ + +// While the following Kirigami is very simple: +// Kirigami.Separator { +// Kirigami.FormData.label: "Heading" +// Kirigami.FormData.isSection: true +// } +// +// I want to be able to adjust the label size and make it bold. +// Kirigami's buddy Heading is level=3, which does not stand out +// very well. I also want to center the heading. +// Since we can't access the Heading in the buddy component, we +// need to make sure the Heading has no text, and draw our own. +ColumnLayout { + id: heading + spacing: 0 + + property string text: "" + property alias separator: separator + property alias label: label + property bool useThickTopMargin: true + + property Item __formLayout: { + if (parent && typeof parent.wideMode === 'boolean') { + return parent + } else if (typeof formLayout !== 'undefined' && typeof formLayout.wideMode === 'boolean') { + return formLayout + } else if (typeof page !== 'undefined' && typeof page.wideMode === 'boolean') { + return page + } else { + return null + } + } + + Layout.fillWidth: true + // Kirigami.FormData.isSection: true + Kirigami.MnemonicData.controlType: Kirigami.MnemonicData.FormLabel + + Kirigami.Separator { + id: separator + visible: false + Layout.fillWidth: true + Layout.topMargin: Kirigami.Units.largeSpacing + Layout.bottomMargin: Kirigami.Units.largeSpacing + } + + Kirigami.Heading { + id: label + Layout.topMargin: useThickTopMargin ? Kirigami.Units.largeSpacing * 3 : Kirigami.Units.largeSpacing + Layout.bottomMargin: Kirigami.Units.smallSpacing + Layout.fillWidth: true + text: heading.text + level: 1 + font.weight: Font.Bold + // horizontalAlignment: (!__formLayout || __formLayout.wideMode) ? Text.AlignHCenter : Text.AlignLeft + verticalAlignment: (!__formLayout || __formLayout.wideMode) ? Text.AlignVCenter : Text.AlignBottom + } +} + +//--- Test Default Kirigami Heading +// Kirigami.Separator { +// property string text: "" +// Kirigami.FormData.label: text +// Kirigami.FormData.isSection: true +// property alias separator: separator +// Item { +// id: separator +// } +// } diff --git a/com.github.zren.presentwindows/contents/ui/libconfig/IconField.qml b/com.github.zren.presentwindows/contents/ui/libconfig/IconField.qml new file mode 100644 index 0000000..250f7f1 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/libconfig/IconField.qml @@ -0,0 +1,154 @@ +// Version 9 + +import QtQuick 2.0 +import QtQuick.Controls 2.0 as QQC2 +import QtQuick.Layouts 1.0 + +import org.kde.kirigami 2.0 as Kirigami +import org.kde.plasma.core 2.0 as PlasmaCore +import org.kde.kquickcontrolsaddons 2.0 as KQuickAddons + +RowLayout { + id: iconField + + default property alias _contentChildren: content.data + + property string configKey: '' + property alias value: textField.text + readonly property string configValue: configKey ? plasmoid.configuration[configKey] : "" + onConfigValueChanged: { + if (!textField.focus && value != configValue) { + value = configValue + } + } + property int previewIconSize: Kirigami.Units.iconSizes.medium + property string defaultValue: "" + property alias placeholderValue: textField.placeholderText + property var presetValues: [] + + // Based on org.kde.plasma.kickoff + QQC2.Button { + id: iconButton + padding: Kirigami.Units.smallSpacing + Layout.alignment: Qt.AlignTop + + // KDE QQC2 sets implicitSize to background.implicitSize ignoring padding/inset properties. + implicitWidth: leftPadding + contentItem.implicitWidth + rightPadding + implicitHeight: topPadding + contentItem.implicitHeight + bottomPadding + + onPressed: iconMenu.opened ? iconMenu.close() : iconMenu.open() + + contentItem: PlasmaCore.FrameSvgItem { + id: previewFrame + imagePath: plasmoid.location === PlasmaCore.Types.Vertical || plasmoid.location === PlasmaCore.Types.Horizontal + ? "widgets/panel-background" : "widgets/background" + implicitWidth: fixedMargins.left + previewIconSize + fixedMargins.right + implicitHeight: fixedMargins.top + previewIconSize + fixedMargins.bottom + + PlasmaCore.IconItem { + anchors.fill: parent + anchors.leftMargin: previewFrame.fixedMargins.left + anchors.topMargin: previewFrame.fixedMargins.top + anchors.rightMargin: previewFrame.fixedMargins.right + anchors.bottomMargin: previewFrame.fixedMargins.bottom + source: iconField.value || iconField.placeholderValue + active: iconButton.hovered + } + } + + QQC2.Menu { + id: iconMenu + + // Appear below the button + y: +parent.height + + QQC2.MenuItem { + text: i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Open icon chooser dialog", "Choose...") + icon.name: "document-open" + onClicked: dialogLoader.active = true + } + QQC2.MenuItem { + text: i18ndc("plasma_applet_org.kde.plasma.kickoff", "@item:inmenu Reset icon to default", "Clear Icon") + icon.name: "edit-clear" + onClicked: iconField.value = iconField.defaultValue + } + } + } + + ColumnLayout { + id: content + Layout.fillWidth: true + + RowLayout { + QQC2.TextField { + id: textField + Layout.fillWidth: true + + text: iconField.configValue + onTextChanged: serializeTimer.restart() + + rightPadding: clearButton.width + Kirigami.Units.smallSpacing + + QQC2.ToolButton { + id: clearButton + visible: iconField.configValue != iconField.defaultValue + icon.name: iconField.defaultValue === "" ? "edit-clear" : "edit-undo" + onClicked: iconField.value = iconField.defaultValue + + anchors.top: parent.top + anchors.right: parent.right + anchors.bottom: parent.bottom + + width: height + } + } + + QQC2.Button { + id: browseButton + icon.name: "document-open" + onClicked: dialogLoader.active = true + } + } + + Flow { + Layout.fillWidth: true + Layout.maximumWidth: Kirigami.Units.gridUnit * 30 + Repeater { + model: presetValues + QQC2.Button { + icon.name: modelData + text: modelData + onClicked: iconField.value = modelData + } + } + } + } + + Loader { + id: dialogLoader + active: false + sourceComponent: KQuickAddons.IconDialog { + id: dialog + visible: true + modality: Qt.WindowModal + onIconNameChanged: { + iconField.value = iconName + } + onVisibleChanged: { + if (!visible) { + dialogLoader.active = false + } + } + } + } + + Timer { // throttle + id: serializeTimer + interval: 300 + onTriggered: { + if (configKey) { + plasmoid.configuration[configKey] = iconField.value + } + } + } +} diff --git a/com.github.zren.presentwindows/contents/ui/libconfig/Label.qml b/com.github.zren.presentwindows/contents/ui/libconfig/Label.qml new file mode 100644 index 0000000..fc8d487 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/libconfig/Label.qml @@ -0,0 +1,18 @@ +// Version 2 + +import QtQuick 2.0 +import QtQuick.Controls 2.0 as QQC2 +import QtQuick.Layouts 1.0 +import org.kde.kirigami 2.0 as Kirigami + +QQC2.Label { + Layout.fillWidth: true + wrapMode: Text.Wrap + linkColor: Kirigami.Theme.highlightColor + onLinkActivated: Qt.openUrlExternally(link) + MouseArea { + anchors.fill: parent + acceptedButtons: Qt.NoButton // we don't want to eat clicks on the Text + cursorShape: parent.hoveredLink ? Qt.PointingHandCursor : Qt.ArrowCursor + } +} diff --git a/com.github.zren.presentwindows/contents/ui/libconfig/RadioButtonGroup.qml b/com.github.zren.presentwindows/contents/ui/libconfig/RadioButtonGroup.qml new file mode 100644 index 0000000..a9427b4 --- /dev/null +++ b/com.github.zren.presentwindows/contents/ui/libconfig/RadioButtonGroup.qml @@ -0,0 +1,65 @@ +// Version 5 + +import QtQuick 2.0 +import QtQuick.Controls 2.5 as QQC2 +import QtQuick.Layouts 1.0 +import org.kde.kirigami 2.5 as Kirigami + +/* +** Example: +** +import './libconfig' as LibConfig +LibConfig.RadioButtonGroup { + configKey: "priority" + model: [ + { value: "a", text: i18n("A") }, + { value: "b", text: i18n("B") }, + { value: "c", text: i18n("C") }, + ] +} +*/ +ColumnLayout { + id: radioButtonGroup + + property string configKey: '' + readonly property var configValue: configKey ? plasmoid.configuration[configKey] : "" + + property alias group: group + QQC2.ButtonGroup { + id: group + } + + property alias model: buttonRepeater.model + + // The main reason we put all the RadioButtons in + // a ColumnLayout is to shrink the spacing between the buttons. + spacing: Kirigami.Units.smallSpacing + + // Assign buddyFor to the first RadioButton so that the Kirigami label aligns with it. + // Repeater is also a visibleChild, so avoid it. + Kirigami.FormData.buddyFor: { + for (var i = 0; i < visibleChildren.length; i++) { + if (!(visibleChildren[i] instanceof Repeater)) { + return visibleChildren[i] + } + } + return null + } + + Repeater { + id: buttonRepeater + QQC2.RadioButton { + visible: typeof modelData.visible !== "undefined" ? modelData.visible : true + enabled: typeof modelData.enabled !== "undefined" ? modelData.enabled : true + text: modelData.text + checked: modelData.value === configValue + QQC2.ButtonGroup.group: radioButtonGroup.group + onClicked: { + focus = true + if (configKey) { + plasmoid.configuration[configKey] = modelData.value + } + } + } + } +} diff --git a/com.github.zren.presentwindows/metadata.json b/com.github.zren.presentwindows/metadata.json new file mode 100644 index 0000000..ceedc72 --- /dev/null +++ b/com.github.zren.presentwindows/metadata.json @@ -0,0 +1,34 @@ +{ + "KPlugin": { + "Authors": [ + { + "Email": "zrenfire@gmail.com", + "Name": "Chris Holland" + } + ], + "Category": "Windows and Tasks", + "Description": "Show desktop windows as mosaics", + "Description[ru]": "Показать окна рабочего стола мозаикой", + "Description[x-test]": "xxShow desktop windows as mosaicsxx", + "EnabledByDefault": true, + "FormFactors": [ + "desktop" + ], + "Icon": "window", + "Id": "com.github.zren.presentwindows", + "License": "GPL-2.0+", + "Name": "Present Windows Button", + "Name[ru]": "Показать рабочий стол", + "Name[x-test]": "xxShow Desktopxx", + "ServiceTypes": [ + "Plasma/Applet" + ], + "Version": "9.0", + "Website": "https://github.com/Zren/plasma-applet-presentwindows" + }, + "X-Plasma-API": "declarativeappletscript", + "X-Plasma-MainScript": "ui/Main.qml", + "X-Plasma-Provides": [ + "org.kde.plasma.windowmanagement" + ] +} diff --git a/com.github.zren.presentwindows/translate/ReadMe.md b/com.github.zren.presentwindows/translate/ReadMe.md new file mode 100644 index 0000000..13c77e7 --- /dev/null +++ b/com.github.zren.presentwindows/translate/ReadMe.md @@ -0,0 +1,40 @@ +> Version 7 of Zren's i18n scripts. + +With KDE Frameworks v5.37 and above, translations are bundled with the `*.plasmoid` file downloaded from the store. + +## Install Translations + +Go to `~/.local/share/plasma/plasmoids/com.github.zren.presentwindows/translate/` and run `sh ./build --restartplasma`. + +## New Translations + +1. Fill out [`template.pot`](template.pot) with your translations then open a [new issue](https://github.com/Zren/plasma-applet-presentwindows/issues/new), name the file `spanish.txt`, attach the txt file to the issue (drag and drop). + +Or if you know how to make a pull request + +1. Copy the `template.pot` file and name it your locale's code (Eg: `en`/`de`/`fr`) with the extension `.po`. Then fill out all the `msgstr ""`. + +## Scripts + +* `sh ./merge` will parse the `i18n()` calls in the `*.qml` files and write it to the `template.pot` file. Then it will merge any changes into the `*.po` language files. +* `sh ./build` will convert the `*.po` files to it's binary `*.mo` version and move it to `contents/locale/...` which will bundle the translations in the `*.plasmoid` without needing the user to manually install them. +* `sh ./plasmoidlocaletest` will run `./build` then `plasmoidviewer` (part of `plasma-sdk`). + +## Links + +* https://zren.github.io/kde/docs/widget/#translations-i18n +* https://techbase.kde.org/Development/Tutorials/Localization/i18n_Build_Systems +* https://api.kde.org/frameworks/ki18n/html/prg_guide.html + +## Examples + +* https://l10n.kde.org/stats/gui/trunk-kf5/team/fr/plasma-desktop/ +* https://github.com/psifidotos/nowdock-plasmoid/tree/master/po +* https://github.com/kotelnik/plasma-applet-redshift-control/tree/master/translations + +## Status +| Locale | Lines | % Done| +|----------|---------|-------| +| Template | 20 | | +| nl | 20/20 | 100% | +| pl | 12/20 | 60% | diff --git a/com.github.zren.presentwindows/translate/build b/com.github.zren.presentwindows/translate/build new file mode 100644 index 0000000..c88cd5d --- /dev/null +++ b/com.github.zren.presentwindows/translate/build @@ -0,0 +1,53 @@ +#!/bin/sh +# Version: 6 + +# This script will convert the *.po files to *.mo files, rebuilding the package/contents/locale folder. +# Feature discussion: https://phabricator.kde.org/D5209 +# Eg: contents/locale/fr_CA/LC_MESSAGES/plasma_applet_org.kde.plasma.eventcalendar.mo + +DIR=`cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd` +plasmoidName=`kreadconfig5 --file="$DIR/../metadata.desktop" --group="Desktop Entry" --key="X-KDE-PluginInfo-Name"` +website=`kreadconfig5 --file="$DIR/../metadata.desktop" --group="Desktop Entry" --key="X-KDE-PluginInfo-Website"` +bugAddress="$website" +packageRoot=".." # Root of translatable sources +projectName="plasma_applet_${plasmoidName}" # project name + +#--- +if [ -z "$plasmoidName" ]; then + echo "[build] Error: Couldn't read plasmoidName." + exit +fi + +if [ -z "$(which msgfmt)" ]; then + echo "[build] Error: msgfmt command not found. Need to install gettext" + echo "[build] Running 'sudo apt install gettext'" + sudo apt install gettext + echo "[build] gettext installation should be finished. Going back to installing translations." +fi + +#--- +echo "[build] Compiling messages" + +catalogs=`find . -name '*.po' | sort` +for cat in $catalogs; do + echo "$cat" + catLocale=`basename ${cat%.*}` + msgfmt -o "${catLocale}.mo" "$cat" + + installPath="$DIR/../contents/locale/${catLocale}/LC_MESSAGES/${projectName}.mo" + + echo "[build] Install to ${installPath}" + mkdir -p "$(dirname "$installPath")" + mv "${catLocale}.mo" "${installPath}" +done + +echo "[build] Done building messages" + +if [ "$1" = "--restartplasma" ]; then + echo "[build] Restarting plasmashell" + killall plasmashell + kstart5 plasmashell + echo "[build] Done restarting plasmashell" +else + echo "[build] (re)install the plasmoid and restart plasmashell to test." +fi diff --git a/com.github.zren.presentwindows/translate/merge b/com.github.zren.presentwindows/translate/merge new file mode 100644 index 0000000..283deb3 --- /dev/null +++ b/com.github.zren.presentwindows/translate/merge @@ -0,0 +1,239 @@ +#!/bin/sh +# Version: 22 + +# https://techbase.kde.org/Development/Tutorials/Localization/i18n_Build_Systems +# https://techbase.kde.org/Development/Tutorials/Localization/i18n_Build_Systems/Outside_KDE_repositories +# https://invent.kde.org/sysadmin/l10n-scripty/-/blob/master/extract-messages.sh + +DIR=`cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd` +plasmoidName=`kreadconfig5 --file="$DIR/../metadata.desktop" --group="Desktop Entry" --key="X-KDE-PluginInfo-Name"` +widgetName="${plasmoidName##*.}" # Strip namespace +website=`kreadconfig5 --file="$DIR/../metadata.desktop" --group="Desktop Entry" --key="X-KDE-PluginInfo-Website"` +bugAddress="$website" +packageRoot=".." # Root of translatable sources +projectName="plasma_applet_${plasmoidName}" # project name + +#--- +if [ -z "$plasmoidName" ]; then + echo "[merge] Error: Couldn't read plasmoidName." + exit +fi + +if [ -z "$(which xgettext)" ]; then + echo "[merge] Error: xgettext command not found. Need to install gettext" + echo "[merge] Running 'sudo apt install gettext'" + sudo apt install gettext + echo "[merge] gettext installation should be finished. Going back to merging translations." +fi + +#--- +echo "[merge] Extracting messages" +potArgs="--from-code=UTF-8 --width=200 --add-location=file" + +# Note: xgettext v0.20.1 (Kubuntu 20.04) and below will attempt to translate Icon, +# so we need to specify Name, GenericName, Comment, and Keywords. +# https://github.com/Zren/plasma-applet-lib/issues/1 +# https://savannah.gnu.org/support/?108887 +find "${packageRoot}" -name '*.desktop' | sort > "${DIR}/infiles.list" +xgettext \ + ${potArgs} \ + --files-from="${DIR}/infiles.list" \ + --language=Desktop \ + -k -kName -kGenericName -kComment -kKeywords \ + -D "${packageRoot}" \ + -D "${DIR}" \ + -o "template.pot.new" \ + || \ + { echo "[merge] error while calling xgettext. aborting."; exit 1; } + +sed -i 's/"Content-Type: text\/plain; charset=CHARSET\\n"/"Content-Type: text\/plain; charset=UTF-8\\n"/' "template.pot.new" + +# See Ki18n's extract-messages.sh for a full example: +# https://invent.kde.org/sysadmin/l10n-scripty/-/blob/master/extract-messages.sh#L25 +# The -kN_ and -kaliasLocale keywords are mentioned in the Outside_KDE_repositories wiki. +# We don't need -kN_ since we don't use intltool-extract but might as well keep it. +# I have no idea what -kaliasLocale is used for. Googling aliasLocale found only listed kde1 code. +# We don't need to parse -ki18nd since that'll extract messages from other domains. +find "${packageRoot}" -name '*.cpp' -o -name '*.h' -o -name '*.c' -o -name '*.qml' -o -name '*.js' | sort > "${DIR}/infiles.list" +xgettext \ + ${potArgs} \ + --files-from="${DIR}/infiles.list" \ + -C -kde \ + -ci18n \ + -ki18n:1 -ki18nc:1c,2 -ki18np:1,2 -ki18ncp:1c,2,3 \ + -kki18n:1 -kki18nc:1c,2 -kki18np:1,2 -kki18ncp:1c,2,3 \ + -kxi18n:1 -kxi18nc:1c,2 -kxi18np:1,2 -kxi18ncp:1c,2,3 \ + -kkxi18n:1 -kkxi18nc:1c,2 -kkxi18np:1,2 -kkxi18ncp:1c,2,3 \ + -kI18N_NOOP:1 -kI18NC_NOOP:1c,2 \ + -kI18N_NOOP2:1c,2 -kI18N_NOOP2_NOSTRIP:1c,2 \ + -ktr2i18n:1 -ktr2xi18n:1 \ + -kN_:1 \ + -kaliasLocale \ + --package-name="${widgetName}" \ + --msgid-bugs-address="${bugAddress}" \ + -D "${packageRoot}" \ + -D "${DIR}" \ + --join-existing \ + -o "template.pot.new" \ + || \ + { echo "[merge] error while calling xgettext. aborting."; exit 1; } + +sed -i 's/# SOME DESCRIPTIVE TITLE./'"# Translation of ${widgetName} in LANGUAGE"'/' "template.pot.new" +sed -i 's/# Copyright (C) YEAR THE PACKAGE'"'"'S COPYRIGHT HOLDER/'"# Copyright (C) $(date +%Y)"'/' "template.pot.new" + +if [ -f "template.pot" ]; then + newPotDate=`grep "POT-Creation-Date:" template.pot.new | sed 's/.\{3\}$//'` + oldPotDate=`grep "POT-Creation-Date:" template.pot | sed 's/.\{3\}$//'` + sed -i 's/'"${newPotDate}"'/'"${oldPotDate}"'/' "template.pot.new" + changes=`diff "template.pot" "template.pot.new"` + if [ ! -z "$changes" ]; then + # There's been changes + sed -i 's/'"${oldPotDate}"'/'"${newPotDate}"'/' "template.pot.new" + mv "template.pot.new" "template.pot" + + addedKeys=`echo "$changes" | grep "> msgid" | cut -c 9- | sort` + removedKeys=`echo "$changes" | grep "< msgid" | cut -c 9- | sort` + echo "" + echo "Added Keys:" + echo "$addedKeys" + echo "" + echo "Removed Keys:" + echo "$removedKeys" + echo "" + + else + # No changes + rm "template.pot.new" + fi +else + # template.pot didn't already exist + mv "template.pot.new" "template.pot" +fi + +potMessageCount=`expr $(grep -Pzo 'msgstr ""\n(\n|$)' "template.pot" | grep -c 'msgstr ""')` +echo "| Locale | Lines | % Done|" > "./Status.md" +echo "|----------|---------|-------|" >> "./Status.md" +entryFormat="| %-8s | %7s | %5s |" +templateLine=`perl -e "printf(\"$entryFormat\", \"Template\", \"${potMessageCount}\", \"\")"` +echo "$templateLine" >> "./Status.md" + +rm "${DIR}/infiles.list" +echo "[merge] Done extracting messages" + +#--- +echo "[merge] Merging messages" +catalogs=`find . -name '*.po' | sort` +for cat in $catalogs; do + echo "[merge] $cat" + catLocale=`basename ${cat%.*}` + + widthArg="" + catUsesGenerator=`grep "X-Generator:" "$cat"` + if [ -z "$catUsesGenerator" ]; then + widthArg="--width=400" + fi + + compendiumArg="" + if [ ! -z "$COMPENDIUM_DIR" ]; then + langCode=`basename "${cat%.*}"` + compendiumPath=`realpath "$COMPENDIUM_DIR/compendium-${langCode}.po"` + if [ -f "$compendiumPath" ]; then + echo "compendiumPath=$compendiumPath" + compendiumArg="--compendium=$compendiumPath" + fi + fi + + cp "$cat" "$cat.new" + sed -i 's/"Content-Type: text\/plain; charset=CHARSET\\n"/"Content-Type: text\/plain; charset=UTF-8\\n"/' "$cat.new" + + msgmerge \ + ${widthArg} \ + --add-location=file \ + --no-fuzzy-matching \ + ${compendiumArg} \ + -o "$cat.new" \ + "$cat.new" "${DIR}/template.pot" + + sed -i 's/# SOME DESCRIPTIVE TITLE./'"# Translation of ${widgetName} in ${catLocale}"'/' "$cat.new" + sed -i 's/# Translation of '"${widgetName}"' in LANGUAGE/'"# Translation of ${widgetName} in ${catLocale}"'/' "$cat.new" + sed -i 's/# Copyright (C) YEAR THE PACKAGE'"'"'S COPYRIGHT HOLDER/'"# Copyright (C) $(date +%Y)"'/' "$cat.new" + + poEmptyMessageCount=`expr $(grep -Pzo 'msgstr ""\n(\n|$)' "$cat.new" | grep -c 'msgstr ""')` + poMessagesDoneCount=`expr $potMessageCount - $poEmptyMessageCount` + poCompletion=`perl -e "printf(\"%d\", $poMessagesDoneCount * 100 / $potMessageCount)"` + poLine=`perl -e "printf(\"$entryFormat\", \"$catLocale\", \"${poMessagesDoneCount}/${potMessageCount}\", \"${poCompletion}%\")"` + echo "$poLine" >> "./Status.md" + + # mv "$cat" "$cat.old" + mv "$cat.new" "$cat" +done +echo "[merge] Done merging messages" + +#--- +echo "[merge] Updating .desktop file" + +# Generate LINGUAS for msgfmt +if [ -f "$DIR/LINGUAS" ]; then + rm "$DIR/LINGUAS" +fi +touch "$DIR/LINGUAS" +for cat in $catalogs; do + catLocale=`basename ${cat%.*}` + echo "${catLocale}" >> "$DIR/LINGUAS" +done + +cp -f "$DIR/../metadata.desktop" "$DIR/template.desktop" +sed -i '/^Name\[/ d; /^GenericName\[/ d; /^Comment\[/ d; /^Keywords\[/ d' "$DIR/template.desktop" + +msgfmt \ + --desktop \ + --template="$DIR/template.desktop" \ + -d "$DIR/" \ + -o "$DIR/new.desktop" + +# Delete empty msgid messages that used the po header +if [ ! -z "$(grep '^Name=$' "$DIR/new.desktop")" ]; then + echo "[merge] Name in metadata.desktop is empty!" + sed -i '/^Name\[/ d' "$DIR/new.desktop" +fi +if [ ! -z "$(grep '^GenericName=$' "$DIR/new.desktop")" ]; then + echo "[merge] GenericName in metadata.desktop is empty!" + sed -i '/^GenericName\[/ d' "$DIR/new.desktop" +fi +if [ ! -z "$(grep '^Comment=$' "$DIR/new.desktop")" ]; then + echo "[merge] Comment in metadata.desktop is empty!" + sed -i '/^Comment\[/ d' "$DIR/new.desktop" +fi +if [ ! -z "$(grep '^Keywords=$' "$DIR/new.desktop")" ]; then + echo "[merge] Keywords in metadata.desktop is empty!" + sed -i '/^Keywords\[/ d' "$DIR/new.desktop" +fi + +# Place translations at the bottom of the desktop file. +translatedLines=`cat "$DIR/new.desktop" | grep "]="` +if [ ! -z "${translatedLines}" ]; then + sed -i '/^Name\[/ d; /^GenericName\[/ d; /^Comment\[/ d; /^Keywords\[/ d' "$DIR/new.desktop" + if [ "$(tail -c 2 "$DIR/new.desktop" | wc -l)" != "2" ]; then + # Does not end with 2 empty lines, so add an empty line. + echo "" >> "$DIR/new.desktop" + fi + echo "${translatedLines}" >> "$DIR/new.desktop" +fi + +# Cleanup +mv "$DIR/new.desktop" "$DIR/../metadata.desktop" +rm "$DIR/template.desktop" +rm "$DIR/LINGUAS" + +#--- +# Populate ReadMe.md +echo "[merge] Updating translate/ReadMe.md" +sed -i -E 's`share\/plasma\/plasmoids\/(.+)\/translate`share/plasma/plasmoids/'"${plasmoidName}"'/translate`' ./ReadMe.md +if [[ "$website" == *"github.com"* ]]; then + sed -i -E 's`\[new issue\]\(https:\/\/github\.com\/(.+)\/(.+)\/issues\/new\)`[new issue]('"${website}"'/issues/new)`' ./ReadMe.md +fi +sed -i '/^|/ d' ./ReadMe.md # Remove status table from ReadMe +cat ./Status.md >> ./ReadMe.md +rm ./Status.md + +echo "[merge] Done" diff --git a/com.github.zren.presentwindows/translate/nl.po b/com.github.zren.presentwindows/translate/nl.po new file mode 100644 index 0000000..525b5f8 --- /dev/null +++ b/com.github.zren.presentwindows/translate/nl.po @@ -0,0 +1,141 @@ +# Translation of presentwindows in nl_NL +# Copyright (C) 2018 +# This file is distributed under the same license as the presentwindows package. +# FIRST AUTHOR , YEAR. +# +msgid "" +msgstr "" +"Project-Id-Version: presentwindows\n" +"Report-Msgid-Bugs-To: https://github.com/Zren/plasma-applet-presentwindows\n" +"POT-Creation-Date: 2023-11-13 13:04+0500\n" +"PO-Revision-Date: 2022-02-25 12:30+0100\n" +"Last-Translator: Heimen Stoffels \n" +"Language-Team: Dutch \n" +"Language: nl_NL\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Generator: Poedit 3.0.1\n" + +#: ../metadata.desktop +msgid "Present Windows Button" +msgstr "Vensters presenteren-knop" + +#: ../metadata.desktop +msgid "Trigger the Present Windows desktop effect's shortcut with a click." +msgstr "Toon het vensteroverzicht met één muisklik." + +#: ../contents/config/config.qml +msgid "General" +msgstr "Algemeen" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Behavior" +msgstr "Gedrag" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Click" +msgstr "Klik" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Enable Desktop Effect" +msgstr "Effect inschakelen" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Appearance" +msgstr "Vormgeving" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Icon" +msgstr "Pictogram" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Unity Pager Theme" +msgstr "Unity-thema" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Use Unity 7 Theme" +msgstr "Unity 7-thema gebruiken" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Never" +msgstr "Nooit" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "When Plasma Style is Unity Ambiance" +msgstr "Als Plasmastijl is ingesteld op Unity Ambience" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Always" +msgstr "Altijd" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "" +"Should we use a Virtual Desktop indicator similar to Unity 7? This feature " +"is enabled for the Unity " +"Ambiance Plasma Style by default." +msgstr "" +"Moet er een virtuelebureaublad-indicator worden getoond, net zoals in Unity " +"7? Deze functie is standaard ingeschakeld als je gebruikmaakt van de Unity Ambiance-plasmastijl." + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Current Plasma Style: %1" +msgstr "Huidige Plasma-stijl: %1" + +#: ../contents/ui/Main.qml +msgid "Show Overview" +msgstr "Overzicht tonen" + +#: ../contents/ui/Main.qml +msgid "Present Windows (All desktops)" +msgstr "Vensters presenteren (alle bureaubladen)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Current desktop)" +msgstr "Vensters presenteren (huidig bureaublad)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Window class)" +msgstr "Vensters presenteren (vensterklasse)" + +#: ../contents/ui/Main.qml +msgid "Show Desktop Grid" +msgstr "Bureaubladrooster tonen" + +#~ msgid "Present Windows Effect" +#~ msgstr "Effect voor Vensters presenteren" + +#~ msgid "" +#~ "Button will not work when the Present Windows desktop effect is disabled." +#~ msgstr "" +#~ "De knop werkt niet als het bureaubladeffect Vensters presenteren is " +#~ "uitgeschakeld." + +#~ msgid "Show Desktop Grid Effect" +#~ msgstr "Bureaubladraster-effect tonen" + +#~ msgid "" +#~ "Button will not work when the Desktop Grid desktop effect is disabled." +#~ msgstr "" +#~ "De knop werkt niet als het bureaubladeffect Bureaubladraster is " +#~ "uitgeschakeld." + +#~ msgid "" +#~ "Should we use a Virtual Desktop indicator similar to Unity 7? This " +#~ "feature is enabled for the Unity Ambiance desktop theme by default." +#~ msgstr "" +#~ "Moet er een virtuele bureaubladindicator in de stijl van Unity 7 worden " +#~ "gebruikt? Deze functie is standaard ingeschakeld voor het bureaubladthema " +#~ "Unity Ambiance." + +#~ msgid "Current Desktop Theme: %1" +#~ msgstr "Huidig bureaubladthema: %1" + +#~ msgid "Version: %1" +#~ msgstr "Versie: %1" + +#~ msgid "Enabled" +#~ msgstr "Ingeschakeld" diff --git a/com.github.zren.presentwindows/translate/nl_NL.po b/com.github.zren.presentwindows/translate/nl_NL.po new file mode 100644 index 0000000..861e8f2 --- /dev/null +++ b/com.github.zren.presentwindows/translate/nl_NL.po @@ -0,0 +1,98 @@ +# Translation of presentwindows in nl_NL +# Copyright (C) 2018 +# This file is distributed under the same license as the presentwindows package. +# FIRST AUTHOR , YEAR. +# +msgid "" +msgstr "" +"Project-Id-Version: presentwindows\n" +"Report-Msgid-Bugs-To: https://github.com/Zren/plasma-applet-presentwindows\n" +"POT-Creation-Date: 2020-04-07 15:50-0400\n" +"PO-Revision-Date: 2018-10-13 21:44+0200\n" +"Last-Translator: Heimen Stoffels \n" +"Language-Team: Dutch \n" +"Language: nl_NL\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.1.1\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +#: ../contents/config/config.qml +msgid "General" +msgstr "Algemeen" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Present Windows Effect" +msgstr "Effect voor Vensters presenteren" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "" +"Button will not work when the Present Windows desktop effect is disabled." +msgstr "" +"De knop werkt niet als het bureaubladeffect Vensters presenteren is " +"uitgeschakeld." + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Show Desktop Grid Effect" +msgstr "Bureaubladraster-effect tonen" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Button will not work when the Desktop Grid desktop effect is disabled." +msgstr "" +"De knop werkt niet als het bureaubladeffect Bureaubladraster is " +"uitgeschakeld." + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Click" +msgstr "Klik" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Icon" +msgstr "Pictogram" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Unity Pager Theme" +msgstr "Unity Pager-thema" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Use Unity 7 Theme" +msgstr "Unity 7-thema gebruiken" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "" +"Should we use a Virtual Desktop indicator similar to Unity 7? This feature " +"is enabled for the Unity " +"Ambiance desktop theme by default." +msgstr "" +"Moet er een virtuele bureaubladindicator in de stijl van Unity 7 worden " +"gebruikt? Deze functie is standaard ingeschakeld voor het bureaubladthema Unity Ambiance." + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Current Desktop Theme: %1" +msgstr "Huidig bureaubladthema: %1" + +#: ../contents/ui/lib/AppletVersion.qml +msgid "Version: %1" +msgstr "Versie: %1" + +#: ../contents/ui/lib/DesktopEffectToggle.qml +msgid "Enabled" +msgstr "Ingeschakeld" + +#: ../contents/ui/Main.qml +msgid "Present Windows (All desktops)" +msgstr "Vensters presenteren (alle bureaubladen)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Current desktop)" +msgstr "Vensters presenteren (huidige bureaublad)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Window class)" +msgstr "Vensters presenteren (vensterklasse)" + +#: ../contents/ui/Main.qml +msgid "Show Desktop Grid" +msgstr "Bureaubladraster tonen" diff --git a/com.github.zren.presentwindows/translate/pl.po b/com.github.zren.presentwindows/translate/pl.po new file mode 100644 index 0000000..0e4880b --- /dev/null +++ b/com.github.zren.presentwindows/translate/pl.po @@ -0,0 +1,126 @@ +# Translation of presentwindows in pl +# Copyright (C) 2020 +# This file is distributed under the same license as the presentwindows package. +# FIRST AUTHOR , YEAR. +# +msgid "" +msgstr "" +"Project-Id-Version: presentwindows\n" +"Report-Msgid-Bugs-To: https://github.com/Zren/plasma-applet-presentwindows\n" +"POT-Creation-Date: 2023-11-13 13:04+0500\n" +"PO-Revision-Date: 2020-05-31 08:11+0200\n" +"Last-Translator: Piotr Komur \n" +"Language-Team: Piotr Komur \n" +"Language: pl\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"X-Generator: Poedit 2.3\n" +"Plural-Forms: nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<12 " +"|| n%100>14) ? 1 : 2);\n" + +#: ../metadata.desktop +msgid "Present Windows Button" +msgstr "" + +#: ../metadata.desktop +msgid "Trigger the Present Windows desktop effect's shortcut with a click." +msgstr "" + +#: ../contents/config/config.qml +msgid "General" +msgstr "Ogólne" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Behavior" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Click" +msgstr "Kliknięcie" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Enable Desktop Effect" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Appearance" +msgstr "Wygląd" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Icon" +msgstr "Ikona" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Unity Pager Theme" +msgstr "Motyw Unity Pager" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Use Unity 7 Theme" +msgstr "Motyw Unity 7" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Never" +msgstr "Nigdy" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "When Plasma Style is Unity Ambiance" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Always" +msgstr "Zawsze" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "" +"Should we use a Virtual Desktop indicator similar to Unity 7? This feature " +"is enabled for the Unity " +"Ambiance Plasma Style by default." +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Current Plasma Style: %1" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Show Overview" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Present Windows (All desktops)" +msgstr "Prezentacja okien (Wszystkie pulpity)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Current desktop)" +msgstr "Prezentacja okien (Bieżący pulpit)" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Window class)" +msgstr "Prezentacja okien (Klasa okna)" + +#: ../contents/ui/Main.qml +msgid "Show Desktop Grid" +msgstr "Siatka pulpitu" + +#~ msgid "Present Windows Effect" +#~ msgstr "Efekt prezentacji okien" + +#~ msgid "" +#~ "Button will not work when the Present Windows desktop effect is disabled." +#~ msgstr "Przycisk nie działa, gdy efekt pulpitu jest nieaktywny." + +#~ msgid "Show Desktop Grid Effect" +#~ msgstr "Efekt siatki pulpitu" + +#~ msgid "" +#~ "Button will not work when the Desktop Grid desktop effect is disabled." +#~ msgstr "Przycisk nie działa, gdy efekt pulpitu jest nieaktywny." + +#~ msgid "Current Desktop Theme: %1" +#~ msgstr "Motyw bieżącego pulpitu: %1" + +#~ msgid "Version: %1" +#~ msgstr "Wersja: %1" + +#~ msgid "Enabled" +#~ msgstr "Włączone" diff --git a/com.github.zren.presentwindows/translate/plasmoidlocaletest b/com.github.zren.presentwindows/translate/plasmoidlocaletest new file mode 100644 index 0000000..dacdedb --- /dev/null +++ b/com.github.zren.presentwindows/translate/plasmoidlocaletest @@ -0,0 +1,181 @@ +#!/bin/bash +# Version 9 +# Requires plasmoidviewer v5.13.0 + +function checkIfLangInstalled { + if [ -x "$(command -v dpkg)" ]; then + dpkg -l ${1} >/dev/null 2>&1 || ( \ + echo -e "${1} not installed.\nInstalling now before continuing.\n" \ + ; sudo apt install ${1} \ + ) || ( \ + echo -e "\nError trying to install ${1}\nPlease run 'sudo apt install ${1}'\n" \ + ; exit 1 \ + ) + elif [ -x "$(command -v pacman)" ]; then + # TODO: run `locale -a` and check if the locale is enabled. + if false; then + # https://wiki.archlinux.org/index.php/Locale + # Uncomment the locale in /etc/locale.gen + # Then run `locale-gen` + echo -e "\nPlease install this locale in System Settings first.\n" + exit 1 + else + echo "" + fi + else + echo -e "\nPackage manager not recognized. If the widget is not translated, please install the package '${1}'\n" + fi +} + +langInput="${1}" +lang="" +languagePack="" + +if [[ "$langInput" =~ ":" ]]; then # String contains a colon so assume it's a locale code. + lang="${langInput}" + IFS=: read -r l1 l2 <<< "${lang}" + languagePack="language-pack-${l2}" +fi + +# https://stackoverflow.com/questions/3191664/list-of-all-locales-and-their-short-codes/28357857#28357857 +declare -a langArr=( + "af_ZA:af:Afrikaans (South Africa)" + "ak_GH:ak:Akan (Ghana)" + "am_ET:am:Amharic (Ethiopia)" + "ar_EG:ar:Arabic (Egypt)" + "as_IN:as:Assamese (India)" + "az_AZ:az:Azerbaijani (Azerbaijan)" + "be_BY:be:Belarusian (Belarus)" + "bem_ZM:bem:Bemba (Zambia)" + "bg_BG:bg:Bulgarian (Bulgaria)" + "bo_IN:bo:Tibetan (India)" + "bs_BA:bs:Bosnian (Bosnia and Herzegovina)" + "ca_ES:ca:Catalan (Spain)" + "chr_US:ch:Cherokee (United States)" + "cs_CZ:cs:Czech (Czech Republic)" + "cy_GB:cy:Welsh (United Kingdom)" + "da_DK:da:Danish (Denmark)" + "de_DE:de:German (Germany)" + "el_GR:el:Greek (Greece)" + "es_MX:es:Spanish (Mexico)" + "et_EE:et:Estonian (Estonia)" + "eu_ES:eu:Basque (Spain)" + "fa_IR:fa:Persian (Iran)" + "ff_SN:ff:Fulah (Senegal)" + "fi_FI:fi:Finnish (Finland)" + "fo_FO:fo:Faroese (Faroe Islands)" + "fr_CA:fr:French (Canada)" + "ga_IE:ga:Irish (Ireland)" + "gl_ES:gl:Galician (Spain)" + "gu_IN:gu:Gujarati (India)" + "gv_GB:gv:Manx (United Kingdom)" + "ha_NG:ha:Hausa (Nigeria)" + "he_IL:he:Hebrew (Israel)" + "hi_IN:hi:Hindi (India)" + "hr_HR:hr:Croatian (Croatia)" + "hu_HU:hu:Hungarian (Hungary)" + "hy_AM:hy:Armenian (Armenia)" + "id_ID:id:Indonesian (Indonesia)" + "ig_NG:ig:Igbo (Nigeria)" + "is_IS:is:Icelandic (Iceland)" + "it_IT:it:Italian (Italy)" + "ja_JP:ja:Japanese (Japan)" + "ka_GE:ka:Georgian (Georgia)" + "kk_KZ:kk:Kazakh (Kazakhstan)" + "kl_GL:kl:Kalaallisut (Greenland)" + "km_KH:km:Khmer (Cambodia)" + "kn_IN:kn:Kannada (India)" + "ko_KR:ko:Korean (South Korea)" + "ko_KR:ko:Korean (South Korea)" + "lg_UG:lg:Ganda (Uganda)" + "lt_LT:lt:Lithuanian (Lithuania)" + "lv_LV:lv:Latvian (Latvia)" + "mg_MG:mg:Malagasy (Madagascar)" + "mk_MK:mk:Macedonian (Macedonia)" + "ml_IN:ml:Malayalam (India)" + "mr_IN:mr:Marathi (India)" + "ms_MY:ms:Malay (Malaysia)" + "mt_MT:mt:Maltese (Malta)" + "my_MM:my:Burmese (Myanmar [Burma])" + "nb_NO:nb:Norwegian Bokmål (Norway)" + "ne_NP:ne:Nepali (Nepal)" + "nl_NL:nl:Dutch (Netherlands)" + "nn_NO:nn:Norwegian Nynorsk (Norway)" + "om_ET:om:Oromo (Ethiopia)" + "or_IN:or:Oriya (India)" + "pa_PK:pa:Punjabi (Pakistan)" + "pl_PL:pl:Polish (Poland)" + "ps_AF:ps:Pashto (Afghanistan)" + "pt_BR:pt:Portuguese (Brazil)" + "ro_RO:ro:Romanian (Romania)" + "ru_RU:ru:Russian (Russia)" + "rw_RW:rw:Kinyarwanda (Rwanda)" + "si_LK:si:Sinhala (Sri Lanka)" + "sk_SK:sk:Slovak (Slovakia)" + "sl_SI:sl:Slovenian (Slovenia)" + "so_SO:so:Somali (Somalia)" + "sq_AL:sq:Albanian (Albania)" + "sr_RS:sr:Serbian (Serbia)" + "sv_SE:sv:Swedish (Sweden)" + "sw_KE:sw:Swahili (Kenya)" + "ta_IN:ta:Tamil (India)" + "te_IN:te:Telugu (India)" + "th_TH:th:Thai (Thailand)" + "ti_ER:ti:Tigrinya (Eritrea)" + "to_TO:to:Tonga (Tonga)" + "tr_TR:tr:Turkish (Turkey)" + "uk_UA:uk:Ukrainian (Ukraine)" + "ur_IN:ur:Urdu (India)" + "uz_UZ:uz:Uzbek (Uzbekistan)" + "vi_VN:vi:Vietnamese (Vietnam)" + "yo_NG:yo:Yoruba (Nigeria)" + "yo_NG:yo:Yoruba (Nigeria)" + "yue_HK:yu:Cantonese (Hong Kong)" + "zh_CN:zh:Chinese (China)" + "zu_ZA:zu:Zulu (South Africa)" +) + +for i in "${langArr[@]}"; do + IFS=: read -r l1 l2 l3 <<< "$i" + if [ "$langInput" == "$l2" ]; then + lang="${l1}:${l2}" + languagePack="language-pack-${l2}" + fi +done + +if [ -z "$lang" ]; then + echo "plasmoidlocaletest doesn't recognize the language '$lang'" + echo "Eg:" + scriptcmd='sh ./plasmoidlocaletest' + for i in "${langArr[@]}"; do + IFS=: read -r l1 l2 l3 <<< "$i" + echo " ${scriptcmd} ${l2} | ${l3}" + done + echo "" + echo "Or use a the full locale code:" + echo " ${scriptcmd} ar_EG:ar" + exit 1 +fi + +IFS=: read -r l1 l2 <<< "${lang}" +l1="${l1}.UTF-8" + +# Check if language is installed +if [ ! -z "$languagePack" ]; then + if [ "$lang" == "zh_CN:zh" ]; then languagePack="language-pack-zh-hans" + fi + + checkIfLangInstalled "$languagePack" || exit 1 +fi + + +echo "LANGUAGE=\"${lang}\"" +echo "LANG=\"${l1}\"" + +scriptDir="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" +packageDir="${scriptDir}/.." + +# Build local translations for plasmoidviewer +sh "${scriptDir}/build" + +LANGUAGE="${lang}" LANG="${l1}" LC_TIME="${l1}" QML_DISABLE_DISK_CACHE=true plasmoidviewer -a "$packageDir" -l topedge -f horizontal -x 0 -y 0 diff --git a/com.github.zren.presentwindows/translate/template.pot b/com.github.zren.presentwindows/translate/template.pot new file mode 100644 index 0000000..2e39a44 --- /dev/null +++ b/com.github.zren.presentwindows/translate/template.pot @@ -0,0 +1,98 @@ +# Translation of presentwindows in LANGUAGE +# Copyright (C) 2023 +# This file is distributed under the same license as the presentwindows package. +# FIRST AUTHOR , YEAR. +# +#, fuzzy +msgid "" +msgstr "" +"Project-Id-Version: presentwindows\n" +"Report-Msgid-Bugs-To: https://github.com/Zren/plasma-applet-presentwindows\n" +"POT-Creation-Date: 2023-11-13 13:04+0500\n" +"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" +"Last-Translator: FULL NAME \n" +"Language-Team: LANGUAGE \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" + +#: ../metadata.desktop +msgid "Present Windows Button" +msgstr "" + +#: ../metadata.desktop +msgid "Trigger the Present Windows desktop effect's shortcut with a click." +msgstr "" + +#: ../contents/config/config.qml +msgid "General" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Behavior" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Click" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Enable Desktop Effect" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Appearance" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Icon" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Unity Pager Theme" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Use Unity 7 Theme" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Never" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "When Plasma Style is Unity Ambiance" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Always" +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Should we use a Virtual Desktop indicator similar to Unity 7? This feature is enabled for the Unity Ambiance Plasma Style by default." +msgstr "" + +#: ../contents/ui/config/ConfigGeneral.qml +msgid "Current Plasma Style: %1" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Show Overview" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Present Windows (All desktops)" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Current desktop)" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Present Windows (Window class)" +msgstr "" + +#: ../contents/ui/Main.qml +msgid "Show Desktop Grid" +msgstr "" diff --git a/push.sh b/push.sh new file mode 100755 index 0000000..345883e --- /dev/null +++ b/push.sh @@ -0,0 +1,5 @@ +#!/bin/bash + +git add . && git commit -m "Update" && git push + +echo "Ready"