pax_global_header00006660000000000000000000000064141741417610014520gustar00rootroot0000000000000052 comment=1cf93d5c5a96db23f8e1845465938f1647c25884 simple-image-filter-1.1.2/000077500000000000000000000000001417414176100153555ustar00rootroot00000000000000simple-image-filter-1.1.2/.clog.toml000066400000000000000000000002211417414176100172470ustar00rootroot00000000000000[clog] repository = "https://github.com/dependon/openQtCamera" from-latest-tag = true changelog = "CHANGELOG.md" [sections] Others = ["chore"] simple-image-filter-1.1.2/.github/000077500000000000000000000000001417414176100167155ustar00rootroot00000000000000simple-image-filter-1.1.2/.github/workflows/000077500000000000000000000000001417414176100207525ustar00rootroot00000000000000simple-image-filter-1.1.2/.github/workflows/c-cpp.yml000066400000000000000000000005521417414176100225010ustar00rootroot00000000000000name: C/C++ CI on: push: branches: [ master ] pull_request: branches: [ master ] jobs: build: runs-on: ubuntu-latest steps: - uses: actions/checkout@v2 - name: installCppcheck run: sudo apt install cppcheck --fix-missing - name: cppcheck run: cppcheck . # - name: make distcheck # run: make distcheck simple-image-filter-1.1.2/.gitignore000066400000000000000000000011121417414176100173400ustar00rootroot00000000000000# Compiled Object files *.slo *.lo *.o # Compiled Dynamic libraries *.so *.dylib # Compiled Static libraries *.lai *.la *.a build*/ include*/ *.pro.user* *.DS_Store *.qm *.autosave vendor/include/* vendor/lib/* .vscode/* *.log # debian debian/* !debian/changelog !debian/rules !debian/control !debian/compat !debian/source/* *.obj *.tlog *.exe *.ilk *.props *.vcxproj *.pdb *.filters /debug/moc_statusbarwidget.cpp /debug/moc_pushbutton.cpp /debug/moc_mainwidget.cpp /debug/moc_imageview.cpp /debug/moc_imagethread.cpp /debug/moc_application.cpp /ui_mainwidget.h simple-image-filter-1.1.2/LICENSE000066400000000000000000001045151417414176100163700ustar00rootroot00000000000000 GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007 Copyright (C) 2007 Free Software Foundation, Inc. Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. Preamble The GNU General Public License is a free, copyleft license for software and other kinds of works. The licenses for most software and other practical works are designed to take away your freedom to share and change the works. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change all versions of a program--to make sure it remains free software for all its users. We, the Free Software Foundation, use the GNU General Public License for most of our software; it applies also to any other work released this way by its authors. You can apply it to your programs, too. When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for them if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs, and that you know you can do these things. To protect your rights, we need to prevent others from denying you these rights or asking you to surrender the rights. Therefore, you have certain responsibilities if you distribute copies of the software, or if you modify it: responsibilities to respect the freedom of others. For example, if you distribute copies of such a program, whether gratis or for a fee, you must pass on to the recipients the same freedoms that you received. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights. Developers that use the GNU GPL protect your rights with two steps: (1) assert copyright on the software, and (2) offer you this License giving you legal permission to copy, distribute and/or modify it. For the developers' and authors' protection, the GPL clearly explains that there is no warranty for this free software. For both users' and authors' sake, the GPL requires that modified versions be marked as changed, so that their problems will not be attributed erroneously to authors of previous versions. Some devices are designed to deny users access to install or run modified versions of the software inside them, although the manufacturer can do so. This is fundamentally incompatible with the aim of protecting users' freedom to change the software. The systematic pattern of such abuse occurs in the area of products for individuals to use, which is precisely where it is most unacceptable. Therefore, we have designed this version of the GPL to prohibit the practice for those products. If such problems arise substantially in other domains, we stand ready to extend this provision to those domains in future versions of the GPL, as needed to protect the freedom of users. Finally, every program is threatened constantly by software patents. States should not allow patents to restrict development and use of software on general-purpose computers, but in those that do, we wish to avoid the special danger that patents applied to a free program could make it effectively proprietary. To prevent this, the GPL assures that patents cannot be used to render the program non-free. The precise terms and conditions for copying, distribution and modification follow. TERMS AND CONDITIONS 0. Definitions. "This License" refers to version 3 of the GNU General Public License. "Copyright" also means copyright-like laws that apply to other kinds of works, such as semiconductor masks. "The Program" refers to any copyrightable work licensed under this License. Each licensee is addressed as "you". "Licensees" and "recipients" may be individuals or organizations. To "modify" a work means to copy from or adapt all or part of the work in a fashion requiring copyright permission, other than the making of an exact copy. The resulting work is called a "modified version" of the earlier work or a work "based on" the earlier work. A "covered work" means either the unmodified Program or a work based on the Program. To "propagate" a work means to do anything with it that, without permission, would make you directly or secondarily liable for infringement under applicable copyright law, except executing it on a computer or modifying a private copy. Propagation includes copying, distribution (with or without modification), making available to the public, and in some countries other activities as well. To "convey" a work means any kind of propagation that enables other parties to make or receive copies. Mere interaction with a user through a computer network, with no transfer of a copy, is not conveying. An interactive user interface displays "Appropriate Legal Notices" to the extent that it includes a convenient and prominently visible feature that (1) displays an appropriate copyright notice, and (2) tells the user that there is no warranty for the work (except to the extent that warranties are provided), that licensees may convey the work under this License, and how to view a copy of this License. If the interface presents a list of user commands or options, such as a menu, a prominent item in the list meets this criterion. 1. Source Code. The "source code" for a work means the preferred form of the work for making modifications to it. "Object code" means any non-source form of a work. A "Standard Interface" means an interface that either is an official standard defined by a recognized standards body, or, in the case of interfaces specified for a particular programming language, one that is widely used among developers working in that language. The "System Libraries" of an executable work include anything, other than the work as a whole, that (a) is included in the normal form of packaging a Major Component, but which is not part of that Major Component, and (b) serves only to enable use of the work with that Major Component, or to implement a Standard Interface for which an implementation is available to the public in source code form. A "Major Component", in this context, means a major essential component (kernel, window system, and so on) of the specific operating system (if any) on which the executable work runs, or a compiler used to produce the work, or an object code interpreter used to run it. The "Corresponding Source" for a work in object code form means all the source code needed to generate, install, and (for an executable work) run the object code and to modify the work, including scripts to control those activities. However, it does not include the work's System Libraries, or general-purpose tools or generally available free programs which are used unmodified in performing those activities but which are not part of the work. For example, Corresponding Source includes interface definition files associated with source files for the work, and the source code for shared libraries and dynamically linked subprograms that the work is specifically designed to require, such as by intimate data communication or control flow between those subprograms and other parts of the work. The Corresponding Source need not include anything that users can regenerate automatically from other parts of the Corresponding Source. The Corresponding Source for a work in source code form is that same work. 2. Basic Permissions. All rights granted under this License are granted for the term of copyright on the Program, and are irrevocable provided the stated conditions are met. This License explicitly affirms your unlimited permission to run the unmodified Program. The output from running a covered work is covered by this License only if the output, given its content, constitutes a covered work. This License acknowledges your rights of fair use or other equivalent, as provided by copyright law. You may make, run and propagate covered works that you do not convey, without conditions so long as your license otherwise remains in force. You may convey covered works to others for the sole purpose of having them make modifications exclusively for you, or provide you with facilities for running those works, provided that you comply with the terms of this License in conveying all material for which you do not control copyright. Those thus making or running the covered works for you must do so exclusively on your behalf, under your direction and control, on terms that prohibit them from making any copies of your copyrighted material outside their relationship with you. Conveying under any other circumstances is permitted solely under the conditions stated below. Sublicensing is not allowed; section 10 makes it unnecessary. 3. Protecting Users' Legal Rights From Anti-Circumvention Law. No covered work shall be deemed part of an effective technological measure under any applicable law fulfilling obligations under article 11 of the WIPO copyright treaty adopted on 20 December 1996, or similar laws prohibiting or restricting circumvention of such measures. When you convey a covered work, you waive any legal power to forbid circumvention of technological measures to the extent such circumvention is effected by exercising rights under this License with respect to the covered work, and you disclaim any intention to limit operation or modification of the work as a means of enforcing, against the work's users, your or third parties' legal rights to forbid circumvention of technological measures. 4. Conveying Verbatim Copies. You may convey verbatim copies of the Program's source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice; keep intact all notices stating that this License and any non-permissive terms added in accord with section 7 apply to the code; keep intact all notices of the absence of any warranty; and give all recipients a copy of this License along with the Program. You may charge any price or no price for each copy that you convey, and you may offer support or warranty protection for a fee. 5. Conveying Modified Source Versions. You may convey a work based on the Program, or the modifications to produce it from the Program, in the form of source code under the terms of section 4, provided that you also meet all of these conditions: a) The work must carry prominent notices stating that you modified it, and giving a relevant date. b) The work must carry prominent notices stating that it is released under this License and any conditions added under section 7. This requirement modifies the requirement in section 4 to "keep intact all notices". c) You must license the entire work, as a whole, under this License to anyone who comes into possession of a copy. This License will therefore apply, along with any applicable section 7 additional terms, to the whole of the work, and all its parts, regardless of how they are packaged. This License gives no permission to license the work in any other way, but it does not invalidate such permission if you have separately received it. d) If the work has interactive user interfaces, each must display Appropriate Legal Notices; however, if the Program has interactive interfaces that do not display Appropriate Legal Notices, your work need not make them do so. A compilation of a covered work with other separate and independent works, which are not by their nature extensions of the covered work, and which are not combined with it such as to form a larger program, in or on a volume of a storage or distribution medium, is called an "aggregate" if the compilation and its resulting copyright are not used to limit the access or legal rights of the compilation's users beyond what the individual works permit. Inclusion of a covered work in an aggregate does not cause this License to apply to the other parts of the aggregate. 6. Conveying Non-Source Forms. You may convey a covered work in object code form under the terms of sections 4 and 5, provided that you also convey the machine-readable Corresponding Source under the terms of this License, in one of these ways: a) Convey the object code in, or embodied in, a physical product (including a physical distribution medium), accompanied by the Corresponding Source fixed on a durable physical medium customarily used for software interchange. b) Convey the object code in, or embodied in, a physical product (including a physical distribution medium), accompanied by a written offer, valid for at least three years and valid for as long as you offer spare parts or customer support for that product model, to give anyone who possesses the object code either (1) a copy of the Corresponding Source for all the software in the product that is covered by this License, on a durable physical medium customarily used for software interchange, for a price no more than your reasonable cost of physically performing this conveying of source, or (2) access to copy the Corresponding Source from a network server at no charge. c) Convey individual copies of the object code with a copy of the written offer to provide the Corresponding Source. This alternative is allowed only occasionally and noncommercially, and only if you received the object code with such an offer, in accord with subsection 6b. d) Convey the object code by offering access from a designated place (gratis or for a charge), and offer equivalent access to the Corresponding Source in the same way through the same place at no further charge. You need not require recipients to copy the Corresponding Source along with the object code. If the place to copy the object code is a network server, the Corresponding Source may be on a different server (operated by you or a third party) that supports equivalent copying facilities, provided you maintain clear directions next to the object code saying where to find the Corresponding Source. Regardless of what server hosts the Corresponding Source, you remain obligated to ensure that it is available for as long as needed to satisfy these requirements. e) Convey the object code using peer-to-peer transmission, provided you inform other peers where the object code and Corresponding Source of the work are being offered to the general public at no charge under subsection 6d. A separable portion of the object code, whose source code is excluded from the Corresponding Source as a System Library, need not be included in conveying the object code work. A "User Product" is either (1) a "consumer product", which means any tangible personal property which is normally used for personal, family, or household purposes, or (2) anything designed or sold for incorporation into a dwelling. In determining whether a product is a consumer product, doubtful cases shall be resolved in favor of coverage. For a particular product received by a particular user, "normally used" refers to a typical or common use of that class of product, regardless of the status of the particular user or of the way in which the particular user actually uses, or expects or is expected to use, the product. A product is a consumer product regardless of whether the product has substantial commercial, industrial or non-consumer uses, unless such uses represent the only significant mode of use of the product. "Installation Information" for a User Product means any methods, procedures, authorization keys, or other information required to install and execute modified versions of a covered work in that User Product from a modified version of its Corresponding Source. The information must suffice to ensure that the continued functioning of the modified object code is in no case prevented or interfered with solely because modification has been made. If you convey an object code work under this section in, or with, or specifically for use in, a User Product, and the conveying occurs as part of a transaction in which the right of possession and use of the User Product is transferred to the recipient in perpetuity or for a fixed term (regardless of how the transaction is characterized), the Corresponding Source conveyed under this section must be accompanied by the Installation Information. But this requirement does not apply if neither you nor any third party retains the ability to install modified object code on the User Product (for example, the work has been installed in ROM). The requirement to provide Installation Information does not include a requirement to continue to provide support service, warranty, or updates for a work that has been modified or installed by the recipient, or for the User Product in which it has been modified or installed. Access to a network may be denied when the modification itself materially and adversely affects the operation of the network or violates the rules and protocols for communication across the network. Corresponding Source conveyed, and Installation Information provided, in accord with this section must be in a format that is publicly documented (and with an implementation available to the public in source code form), and must require no special password or key for unpacking, reading or copying. 7. Additional Terms. "Additional permissions" are terms that supplement the terms of this License by making exceptions from one or more of its conditions. Additional permissions that are applicable to the entire Program shall be treated as though they were included in this License, to the extent that they are valid under applicable law. If additional permissions apply only to part of the Program, that part may be used separately under those permissions, but the entire Program remains governed by this License without regard to the additional permissions. When you convey a copy of a covered work, you may at your option remove any additional permissions from that copy, or from any part of it. (Additional permissions may be written to require their own removal in certain cases when you modify the work.) You may place additional permissions on material, added by you to a covered work, for which you have or can give appropriate copyright permission. Notwithstanding any other provision of this License, for material you add to a covered work, you may (if authorized by the copyright holders of that material) supplement the terms of this License with terms: a) Disclaiming warranty or limiting liability differently from the terms of sections 15 and 16 of this License; or b) Requiring preservation of specified reasonable legal notices or author attributions in that material or in the Appropriate Legal Notices displayed by works containing it; or c) Prohibiting misrepresentation of the origin of that material, or requiring that modified versions of such material be marked in reasonable ways as different from the original version; or d) Limiting the use for publicity purposes of names of licensors or authors of the material; or e) Declining to grant rights under trademark law for use of some trade names, trademarks, or service marks; or f) Requiring indemnification of licensors and authors of that material by anyone who conveys the material (or modified versions of it) with contractual assumptions of liability to the recipient, for any liability that these contractual assumptions directly impose on those licensors and authors. All other non-permissive additional terms are considered "further restrictions" within the meaning of section 10. If the Program as you received it, or any part of it, contains a notice stating that it is governed by this License along with a term that is a further restriction, you may remove that term. If a license document contains a further restriction but permits relicensing or conveying under this License, you may add to a covered work material governed by the terms of that license document, provided that the further restriction does not survive such relicensing or conveying. If you add terms to a covered work in accord with this section, you must place, in the relevant source files, a statement of the additional terms that apply to those files, or a notice indicating where to find the applicable terms. Additional terms, permissive or non-permissive, may be stated in the form of a separately written license, or stated as exceptions; the above requirements apply either way. 8. Termination. You may not propagate or modify a covered work except as expressly provided under this License. Any attempt otherwise to propagate or modify it is void, and will automatically terminate your rights under this License (including any patent licenses granted under the third paragraph of section 11). However, if you cease all violation of this License, then your license from a particular copyright holder is reinstated (a) provisionally, unless and until the copyright holder explicitly and finally terminates your license, and (b) permanently, if the copyright holder fails to notify you of the violation by some reasonable means prior to 60 days after the cessation. Moreover, your license from a particular copyright holder is reinstated permanently if the copyright holder notifies you of the violation by some reasonable means, this is the first time you have received notice of violation of this License (for any work) from that copyright holder, and you cure the violation prior to 30 days after your receipt of the notice. Termination of your rights under this section does not terminate the licenses of parties who have received copies or rights from you under this License. If your rights have been terminated and not permanently reinstated, you do not qualify to receive new licenses for the same material under section 10. 9. Acceptance Not Required for Having Copies. You are not required to accept this License in order to receive or run a copy of the Program. Ancillary propagation of a covered work occurring solely as a consequence of using peer-to-peer transmission to receive a copy likewise does not require acceptance. However, nothing other than this License grants you permission to propagate or modify any covered work. These actions infringe copyright if you do not accept this License. Therefore, by modifying or propagating a covered work, you indicate your acceptance of this License to do so. 10. Automatic Licensing of Downstream Recipients. Each time you convey a covered work, the recipient automatically receives a license from the original licensors, to run, modify and propagate that work, subject to this License. You are not responsible for enforcing compliance by third parties with this License. An "entity transaction" is a transaction transferring control of an organization, or substantially all assets of one, or subdividing an organization, or merging organizations. If propagation of a covered work results from an entity transaction, each party to that transaction who receives a copy of the work also receives whatever licenses to the work the party's predecessor in interest had or could give under the previous paragraph, plus a right to possession of the Corresponding Source of the work from the predecessor in interest, if the predecessor has it or can get it with reasonable efforts. You may not impose any further restrictions on the exercise of the rights granted or affirmed under this License. For example, you may not impose a license fee, royalty, or other charge for exercise of rights granted under this License, and you may not initiate litigation (including a cross-claim or counterclaim in a lawsuit) alleging that any patent claim is infringed by making, using, selling, offering for sale, or importing the Program or any portion of it. 11. Patents. A "contributor" is a copyright holder who authorizes use under this License of the Program or a work on which the Program is based. The work thus licensed is called the contributor's "contributor version". A contributor's "essential patent claims" are all patent claims owned or controlled by the contributor, whether already acquired or hereafter acquired, that would be infringed by some manner, permitted by this License, of making, using, or selling its contributor version, but do not include claims that would be infringed only as a consequence of further modification of the contributor version. For purposes of this definition, "control" includes the right to grant patent sublicenses in a manner consistent with the requirements of this License. Each contributor grants you a non-exclusive, worldwide, royalty-free patent license under the contributor's essential patent claims, to make, use, sell, offer for sale, import and otherwise run, modify and propagate the contents of its contributor version. In the following three paragraphs, a "patent license" is any express agreement or commitment, however denominated, not to enforce a patent (such as an express permission to practice a patent or covenant not to sue for patent infringement). To "grant" such a patent license to a party means to make such an agreement or commitment not to enforce a patent against the party. If you convey a covered work, knowingly relying on a patent license, and the Corresponding Source of the work is not available for anyone to copy, free of charge and under the terms of this License, through a publicly available network server or other readily accessible means, then you must either (1) cause the Corresponding Source to be so available, or (2) arrange to deprive yourself of the benefit of the patent license for this particular work, or (3) arrange, in a manner consistent with the requirements of this License, to extend the patent license to downstream recipients. "Knowingly relying" means you have actual knowledge that, but for the patent license, your conveying the covered work in a country, or your recipient's use of the covered work in a country, would infringe one or more identifiable patents in that country that you have reason to believe are valid. If, pursuant to or in connection with a single transaction or arrangement, you convey, or propagate by procuring conveyance of, a covered work, and grant a patent license to some of the parties receiving the covered work authorizing them to use, propagate, modify or convey a specific copy of the covered work, then the patent license you grant is automatically extended to all recipients of the covered work and works based on it. A patent license is "discriminatory" if it does not include within the scope of its coverage, prohibits the exercise of, or is conditioned on the non-exercise of one or more of the rights that are specifically granted under this License. You may not convey a covered work if you are a party to an arrangement with a third party that is in the business of distributing software, under which you make payment to the third party based on the extent of your activity of conveying the work, and under which the third party grants, to any of the parties who would receive the covered work from you, a discriminatory patent license (a) in connection with copies of the covered work conveyed by you (or copies made from those copies), or (b) primarily for and in connection with specific products or compilations that contain the covered work, unless you entered into that arrangement, or that patent license was granted, prior to 28 March 2007. Nothing in this License shall be construed as excluding or limiting any implied license or other defenses to infringement that may otherwise be available to you under applicable patent law. 12. No Surrender of Others' Freedom. If conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot convey a covered work so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not convey it at all. For example, if you agree to terms that obligate you to collect a royalty for further conveying from those to whom you convey the Program, the only way you could satisfy both those terms and this License would be to refrain entirely from conveying the Program. 13. Use with the GNU Affero General Public License. Notwithstanding any other provision of this License, you have permission to link or combine any covered work with a work licensed under version 3 of the GNU Affero General Public License into a single combined work, and to convey the resulting work. The terms of this License will continue to apply to the part which is the covered work, but the special requirements of the GNU Affero General Public License, section 13, concerning interaction through a network will apply to the combination as such. 14. Revised Versions of this License. The Free Software Foundation may publish revised and/or new versions of the GNU General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. Each version is given a distinguishing version number. If the Program specifies that a certain numbered version of the GNU General Public License "or any later version" applies to it, you have the option of following the terms and conditions either of that numbered version or of any later version published by the Free Software Foundation. If the Program does not specify a version number of the GNU General Public License, you may choose any version ever published by the Free Software Foundation. If the Program specifies that a proxy can decide which future versions of the GNU General Public License can be used, that proxy's public statement of acceptance of a version permanently authorizes you to choose that version for the Program. Later license versions may give you additional or different permissions. However, no additional obligations are imposed on any author or copyright holder as a result of your choosing to follow a later version. 15. Disclaimer of Warranty. THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 16. Limitation of Liability. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. 17. Interpretation of Sections 15 and 16. If the disclaimer of warranty and limitation of liability provided above cannot be given local legal effect according to their terms, reviewing courts shall apply local law that most closely approximates an absolute waiver of all civil liability in connection with the Program, unless a warranty or assumption of liability accompanies a copy of the Program in return for a fee. END OF TERMS AND CONDITIONS How to Apply These Terms to Your New Programs If you develop a new program, and you want it to be of the greatest possible use to the public, the best way to achieve this is to make it free software which everyone can redistribute and change under these terms. To do so, attach the following notices to the program. It is safest to attach them to the start of each source file to most effectively state the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found. Copyright (C) This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . Also add information on how to contact you by electronic and paper mail. If the program does terminal interaction, make it output a short notice like this when it starts in an interactive mode: Copyright (C) This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. This is free software, and you are welcome to redistribute it under certain conditions; type `show c' for details. The hypothetical commands `show w' and `show c' should show the appropriate parts of the General Public License. Of course, your program's commands might be different; for a GUI interface, you would use an "about box". You should also get your employer (if you work as a programmer) or school, if any, to sign a "copyright disclaimer" for the program, if necessary. For more information on this, and how to apply and follow the GNU GPL, see . The GNU General Public License does not permit incorporating your program into proprietary programs. If your program is a subroutine library, you may consider it more useful to permit linking proprietary applications with the library. If this is what you want to do, use the GNU Lesser General Public License instead of this License. But first, please read . simple-image-filter-1.1.2/README.md000066400000000000000000000016331417414176100166370ustar00rootroot00000000000000# simple-image-filter What is simple-image-filter? At present, simple-image-filter is a QT software that can handle image rotation, clipping, basic beauty, filter and other functions ImageProcess,using Qt,filter # dowload windows release https://github.com/dependon/simple-image-filter/releases/download/1.1.0/simple_image.exe chineseweb: https://depend.lanzouo.com/i6tCpxdfg6d password:8s3b ## arch linux install yay -s simple-image-filter-git ## deepin/debian build sudo apt install qtcreator git clone https://github.com/dependon/simple-image-filter.git mkdir build cd build qmake .. make ## Dermabrasion filter [![gfQ4ld.png](https://z3.ax1x.com/2021/05/18/gfQ4ld.png)](https://imgtu.com/i/gfQ4ld) ## more filter [![gfQ70P.png](https://z3.ax1x.com/2021/05/18/gfQ70P.png)](https://imgtu.com/i/gfQ70P) ## License simple-image-filter is released under GPLv3+. Copyright (C) 2020 ~ 2021 LiuMingHang. simple-image-filter-1.1.2/debian/000077500000000000000000000000001417414176100165775ustar00rootroot00000000000000simple-image-filter-1.1.2/debian/changelog000066400000000000000000000002421417414176100204470ustar00rootroot00000000000000simple-image-filter (1.1.0) unstable; urgency=low * Initial release. -- Deepin Packages Builder Mon, 13 Feb 2017 16:32:34 +0800 simple-image-filter-1.1.2/debian/compat000066400000000000000000000000021417414176100177750ustar00rootroot000000000000009 simple-image-filter-1.1.2/debian/control000066400000000000000000000006711417414176100202060ustar00rootroot00000000000000Source: simple-image-filter Section: utils Priority: optional Maintainer: liuminghang Build-Depends: debhelper (>= 9),pkg-config, dpkg-dev, qt5-qmake, qtbase5-dev,qttools5-dev-tools Standards-Version: 0.0-3 Homepage: simple-image-filter Package: simple-image-filter Architecture: any Depends: ${shlibs:Depends}, ${misc:Depends} Recommends: libqt5libqgtk2, kimageformat-plugins Description: simple image filter. simple-image-filter-1.1.2/debian/rules000077500000000000000000000003021417414176100176520ustar00rootroot00000000000000#!/usr/bin/make -f include /usr/share/dpkg/default.mk export QT_SELECT = qt5 %: dh $@ --parallel override_dh_auto_configure: dh_auto_configure -- DEFINES+="VERSION=$(DEB_VERSION_UPSTREAM)" simple-image-filter-1.1.2/debian/simple-image-filter.1000066400000000000000000000010251417414176100225130ustar00rootroot00000000000000.TH simple-image-filter 1 "September 15, 2021" "version 1" "General Commands Manual" .SH NAME simple\-image\-filter \- Image Filter .SH SYNOPSIS .B simple\-image\-filter .SH DESCRIPTION .B Image Filter is an image processing software, can handle image rotation, clip, basic beautification, filter and other graph functions. .SH OPTIONS .PP This program does not accept any command line options. .PP This manual page was written by .MT dluyaning@\:gmail.com Lu YaNing .ME for the Debian project (and may be used by others). simple-image-filter-1.1.2/debian/simple-image-filter.manpages000066400000000000000000000000351417414176100241460ustar00rootroot00000000000000debian/simple-image-filter.1 simple-image-filter-1.1.2/simple-image-filter-process.pro000077500000000000000000000003021417414176100234050ustar00rootroot00000000000000#------------------------------------------------- # # Project created by QtCreator 2017-03-20T14:23:41 # #------------------------------------------------- TEMPLATE = subdirs SUBDIRS += src simple-image-filter-1.1.2/src/000077500000000000000000000000001417414176100161445ustar00rootroot00000000000000simple-image-filter-1.1.2/src/api/000077500000000000000000000000001417414176100167155ustar00rootroot00000000000000simple-image-filter-1.1.2/src/api/imageapi.cpp000066400000000000000000000677311417414176100212130ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "imageapi.h" #include #include #include typedef struct { uint8_t R; uint8_t G; uint8_t B; uint8_t L; } RGBL; typedef struct { float H; float S; float V; } HSV; static void RGB_TO_HSV(const RGBL *input, HSV *output) // convert RGB value to HSV value { float r, g, b, minRGB, maxRGB, deltaRGB; r = input->R / 255.0f; g = input->G / 255.0f; b = input->B / 255.0f; minRGB = QImageAPI::RgbMin(r, g, b); maxRGB = QImageAPI::RgbMax(r, g, b); deltaRGB = maxRGB - minRGB; output->V = maxRGB; if (maxRGB != 0.0f) output->S = deltaRGB / maxRGB; else output->S = 0.0f; if (output->S <= 0.0f) { output->H = 0.0f; } else { if (r == maxRGB) { output->H = (g - b) / deltaRGB; } else { if (g == maxRGB) { output->H = 2.0f + (b - r) / deltaRGB; } else { if (b == maxRGB) { output->H = 4.0f + (r - g) / deltaRGB; } } } output->H = output->H * 60.0f; if (output->H < 0.0f) { output->H += 360; } output->H /= 360; } } static void HSV_TO_RGB(HSV *input, RGBL *output) //convert HSV value to RGB value { float R, G, B; int k; float aa, bb, cc, f; if (input->S <= 0.0f) R = G = B = input->V; else { if (input->H == 1.0f) input->H = 0.0f; input->H *= 6.0f; k = (int)floor(input->H); f = input->H - k; aa = input->V * (1.0f - input->S); bb = input->V * (1.0f - input->S * f); cc = input->V * (1.0f - (input->S * (1.0f - f))); switch (k) { case 0: R = input->V; G = cc; B = aa; break; case 1: R = bb; G = input->V; B = aa; break; case 2: R = aa; G = input->V; B = cc; break; case 3: R = aa; G = bb; B = input->V; break; case 4: R = cc; G = aa; B = input->V; break; case 5: R = input->V; G = aa; B = bb; break; } } output->R = (unsigned char)(R * 255); output->G = (unsigned char)(G * 255); output->B = (unsigned char)(B * 255); } void adjustBrightness(RGBL &rgb_v, int step) { HSV hsv_v; RGB_TO_HSV(&rgb_v, &hsv_v); rgb_v.L = hsv_v.V; rgb_v.L += step; if (rgb_v.L <= 0) { rgb_v.L = 1; } else if (rgb_v.L >= 100) { rgb_v.L = 100; } hsv_v.V = rgb_v.L / 100.0; HSV_TO_RGB(&hsv_v, &rgb_v); } QImageAPI::QImageAPI() { } int QImageAPI::RgbMax(int red, int green, int blue) { int ret = 0; if (red >= green) { ret = red; } else { ret = green; } if (ret < blue) { ret = blue; } return ret; } int QImageAPI::RgbMin(int red, int green, int blue) { int ret = 0; if (red >= green) { ret = green; } else { ret = red; } if (ret > blue) { ret = blue; } return ret; } int QImageAPI::Bound(int range_left, int data, int range_right) { int index = data; if (data > range_right) { index = range_right; } else if (data < range_left) { index = range_left; } return index; } QImage QImageAPI::QImageD_RunBEEPSHorizontalVertical(const QImage &img, double spatialDecay, double photometricStandardDeviation) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); double c = -0.5 / (photometricStandardDeviation * photometricStandardDeviation); double mu = spatialDecay / (2 - spatialDecay); double *exptable = new double[256]; double *g_table = new double[256]; for (int i = 0; i <= 255; i++) { exptable[i] = (1 - spatialDecay) * exp(c * i * i); g_table[i] = mu * i; } int width = img.width(); int height = img.height(); int length = width * height; double *data2Red = new double[length]; double *data2Green = new double[length]; double *data2Blue = new double[length]; int size = imgCopy.width() * imgCopy.height(); uint8_t *rgb = imgCopy.bits(); for (int i = 0; i < size; i++) { data2Red[i] = rgb[i * 3]; data2Green[i] = rgb[i * 3 + 1]; data2Blue[i] = rgb[i * 3 + 2]; } double *gRed = new double[length]; double *pRed = new double[length]; double *rRed = new double[length]; double *gGreen = new double[length]; double *pGreen = new double[length]; double *rGreen = new double[length]; double *gBlue = new double[length]; double *pBlue = new double[length]; double *rBlue = new double[length]; memcpy(pRed, data2Red, sizeof(double) * length); memcpy(rRed, data2Red, sizeof(double) * length); memcpy(pGreen, data2Green, sizeof(double) * length); memcpy(rGreen, data2Green, sizeof(double) * length); memcpy(pBlue, data2Blue, sizeof(double) * length); memcpy(rBlue, data2Blue, sizeof(double) * length); double rho0 = 1.0 / (2 - spatialDecay); for (int k2 = 0; k2 < height; ++k2) { int startIndex = k2 * width; double mu = 0.0; for (int k = startIndex + 1, K = startIndex + width; k < K; ++k) { int div0Red = fabs(pRed[k] - pRed[k - 1]); mu = exptable[div0Red]; pRed[k] = pRed[k - 1] * mu + pRed[k] * (1.0 - mu); int div0Green = fabs(pGreen[k] - pGreen[k - 1]); mu = exptable[div0Green]; pGreen[k] = pGreen[k - 1] * mu + pGreen[k] * (1.0 - mu); int div0Blue = fabs(pBlue[k] - pBlue[k - 1]); mu = exptable[div0Blue]; pBlue[k] = pBlue[k - 1] * mu + pBlue[k] * (1.0 - mu); } for (int k = startIndex + width - 2; startIndex <= k; --k) { int div0Red = fabs(rRed[k] - rRed[k + 1]); double mu = exptable[div0Red]; rRed[k] = rRed[k + 1] * mu + rRed[k] * (1.0 - mu); int div0Green = fabs(rGreen[k] - rGreen[k + 1]); mu = exptable[div0Green]; rGreen[k] = rGreen[k + 1] * mu + rGreen[k] * (1.0 - mu); int div0Blue = fabs(rBlue[k] - rBlue[k + 1]); mu = exptable[div0Blue]; rBlue[k] = rBlue[k + 1] * mu + rBlue[k] * (1.0 - mu); } for (int k = startIndex, K = startIndex + width; k < K; k++) { rRed[k] = (rRed[k] + pRed[k]) * rho0 - g_table[(int)data2Red[k]]; rGreen[k] = (rGreen[k] + pGreen[k]) * rho0 - g_table[(int)data2Green[k]]; rBlue[k] = (rBlue[k] + pBlue[k]) * rho0 - g_table[(int)data2Blue[k]]; } } int m = 0; for (int k2 = 0; k2 < height; k2++) { int n = k2; for (int k1 = 0; k1 < width; k1++) { gRed[n] = rRed[m]; gGreen[n] = rGreen[m]; gBlue[n] = rBlue[m]; m++; n += height; } } memcpy(pRed, gRed, sizeof(double) * height * width); memcpy(rRed, gRed, sizeof(double) * height * width); memcpy(pGreen, gGreen, sizeof(double) * height * width); memcpy(rGreen, gGreen, sizeof(double) * height * width); memcpy(pBlue, gBlue, sizeof(double) * height * width); memcpy(rBlue, gBlue, sizeof(double) * height * width); for (int k1 = 0; k1 < width; ++k1) { int startIndex = k1 * height; double mu = 0.0; for (int k = startIndex + 1, K = startIndex + height; k < K; ++k) { int div0Red = fabs(pRed[k] - pRed[k - 1]); mu = exptable[div0Red]; pRed[k] = pRed[k - 1] * mu + pRed[k] * (1.0 - mu); int div0Green = fabs(pGreen[k] - pGreen[k - 1]); mu = exptable[div0Green]; pGreen[k] = pGreen[k - 1] * mu + pGreen[k] * (1.0 - mu); int div0Blue = fabs(pBlue[k] - pBlue[k - 1]); mu = exptable[div0Blue]; pBlue[k] = pBlue[k - 1] * mu + pBlue[k] * (1.0 - mu); } for (int k = startIndex + height - 2; startIndex <= k; --k) { int div0Red = fabs(rRed[k] - rRed[k + 1]); mu = exptable[div0Red]; rRed[k] = rRed[k + 1] * mu + rRed[k] * (1.0 - mu); int div0Green = fabs(rGreen[k] - rGreen[k + 1]); mu = exptable[div0Green]; rGreen[k] = rGreen[k + 1] * mu + rGreen[k] * (1.0 - mu); int div0Blue = fabs(rBlue[k] - rBlue[k + 1]); mu = exptable[div0Blue]; rBlue[k] = rBlue[k + 1] * mu + rBlue[k] * (1.0 - mu); } } double init_gain_mu = spatialDecay / (2 - spatialDecay); for (int k = 0; k < length; ++k) { rRed[k] = (rRed[k] + pRed[k]) * rho0 - gRed[k] * init_gain_mu; rGreen[k] = (rGreen[k] + pGreen[k]) * rho0 - gGreen[k] * init_gain_mu; rBlue[k] = (rBlue[k] + pBlue[k]) * rho0 - gBlue[k] * init_gain_mu; } m = 0; int nRowBytes = (width * 24 + 31) / 32 * 4; int lineNum_24 = 0; for (int k1 = 0; k1 < width; ++k1) { int n = k1; for (int k2 = 0; k2 < height; ++k2) { data2Red[n] = rRed[m]; data2Green[n] = rGreen[m]; data2Blue[n] = rBlue[m]; lineNum_24 = k2 * nRowBytes; rgb[lineNum_24 + k1 * 3] = data2Red[n]; rgb[lineNum_24 + k1 * 3 + 1] = data2Green[n]; rgb[lineNum_24 + k1 * 3 + 2] = data2Blue[n]; m++; n += width; } } delete []data2Red; data2Red = nullptr; delete []data2Green ; data2Green = nullptr; delete []data2Blue; data2Blue = nullptr; delete []pRed; pRed = nullptr; delete []rRed; rRed = nullptr; delete []gRed; gRed = nullptr; delete []pGreen; pGreen = nullptr; delete []rGreen; rGreen = nullptr; delete []gGreen; gGreen = nullptr; delete []pBlue; pBlue = nullptr; delete []rBlue; rBlue = nullptr; delete []gBlue; gBlue = nullptr; delete []exptable; exptable = nullptr; delete []g_table; g_table = nullptr; qDebug() << "磨皮结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::warnImage(const QImage &img, int index) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } QColor frontColor; int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { int r = rgb[i * 3] + index; int g = rgb[i * 3 + 1] + index; int b = rgb[i * 3 + 2] ; rgb[i * 3] = r > 255 ? 255 : r; rgb[i * 3 + 1] = g > 255 ? 255 : g; rgb[i * 3 + 2] = b > 255 ? 255 : b; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::coolImage(const QImage &img, int index) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } QColor frontColor; int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { int r = rgb[i * 3] ; int g = rgb[i * 3 + 1] ; int b = rgb[i * 3 + 2] + index; rgb[i * 3] = r > 255 ? 255 : r; rgb[i * 3 + 1] = g > 255 ? 255 : g; rgb[i * 3 + 2] = b > 255 ? 255 : b; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::GrayScaleImage(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } QColor frontColor; int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { int average = (rgb[i * 3] + rgb[i * 3 + 1] + rgb[i * 3 + 2]) / 3; rgb[i * 3] = average > 255 ? 255 : average; rgb[i * 3 + 1] = average > 255 ? 255 : average; rgb[i * 3 + 2] = average > 255 ? 255 : average; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::lightContrastImage(const QImage &img, int light, int Contrast) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } int r; int g; int b; int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { r = light * 0.01 * rgb[i * 3] - 150 + Contrast; g = light * 0.01 * rgb[i * 3 + 1] - 150 + Contrast; b = light * 0.01 * rgb[i * 3 + 2] - 150 + Contrast; r = Bound(0, r, 255); g = Bound(0, g, 255); b = Bound(0, b, 255); rgb[i * 3] = r; rgb[i * 3 + 1] = g; rgb[i * 3 + 2] = b; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::InverseColorImage(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { rgb[i * 3] = 255 - rgb[i * 3] ; rgb[i * 3 + 1] = 255 - rgb[i * 3 + 1] ; rgb[i * 3 + 2] = 255 - rgb[i * 3 + 2] ; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::oldImage(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); if (nullptr == rgb) { return QImage(); } int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { float r = 0.393 * rgb[i * 3] + 0.769 * rgb[i * 3 + 1] + 0.189 * rgb[i * 3 + 2]; float g = 0.349 * rgb[i * 3] + 0.686 * rgb[i * 3 + 1] + 0.168 * rgb[i * 3 + 2]; float b = 0.272 * rgb[i * 3] + 0.534 * rgb[i * 3 + 1] + 0.131 * rgb[i * 3 + 2]; r = Bound(0, r, 255); g = Bound(0, g, 255); b = Bound(0, b, 255); rgb[i * 3] = r; rgb[i * 3 + 1] = g ; rgb[i * 3 + 2] = b ; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::LaplaceSharpen(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; int width = img.width(); int height = img.height(); int window[3][3] = {0, -1, 0, -1, 4, -1, 0, -1, 0}; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(width, height, QImage::Format_RGB888); } else { imgCopy = QImage(img); } QImage imgCopyrgbImg = QImage(img).convertToFormat(QImage::Format_RGB888); uint8_t *rgbImg = imgCopyrgbImg.bits(); uint8_t *rgb = imgCopy.bits(); int nRowBytes = (width * 24 + 31) / 32 * 4; int lineNum_24 = 0; for (int x = 1; x < img.width(); x++) { for (int y = 1; y < img.height(); y++) { int sumR = 0; int sumG = 0; int sumB = 0; for (int m = x - 1; m <= x + 1; m++) for (int n = y - 1; n <= y + 1; n++) { if (m >= 0 && m < width && n < height) { lineNum_24 = n * nRowBytes; sumR += rgbImg[lineNum_24 + m * 3] * window[n - y + 1][m - x + 1]; sumG += rgbImg[lineNum_24 + m * 3 + 1] * window[n - y + 1][m - x + 1]; sumB += rgbImg[lineNum_24 + m * 3 + 2] * window[n - y + 1][m - x + 1]; } } int old_r = rgbImg[lineNum_24 + x * 3]; sumR += old_r; sumR = qBound(0, sumR, 255); int old_g = rgbImg[lineNum_24 + x * 3 + 1]; sumG += old_g; sumG = qBound(0, sumG, 255); int old_b = rgbImg[lineNum_24 + x * 3 + 2]; sumB += old_b; sumB = qBound(0, sumB, 255); lineNum_24 = y * nRowBytes; rgb[lineNum_24 + x * 3] = sumR; rgb[lineNum_24 + x * 3 + 1] = sumG; rgb[lineNum_24 + x * 3 + 2] = sumB; } } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::SobelEdge(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); double *Gx = new double[9]; double *Gy = new double[9]; /* Sobel */ Gx[0] = 1.0; Gx[1] = 0.0; Gx[2] = -1.0; Gx[3] = 2.0; Gx[4] = 0.0; Gx[5] = -2.0; Gx[6] = 1.0; Gx[7] = 0.0; Gx[8] = -1.0; Gy[0] = -1.0; Gy[1] = -2.0; Gy[2] = - 1.0; Gy[3] = 0.0; Gy[4] = 0.0; Gy[5] = 0.0; Gy[6] = 1.0; Gy[7] = 2.0; Gy[8] = 1.0; QRgb pixel; QImage grayImage = GreyScale(img); int height = grayImage.height(); int width = grayImage.width(); QImage imgCopy = QImage(width, height, QImage::Format_RGB888); uint8_t *rgbImg = grayImage.bits(); uint8_t *rgb = imgCopy.bits(); int nRowBytes = (width * 24 + 31) / 32 * 4; int lineNum_24 = 0; float *sobel_norm = new float[width * height]; float max = 0.0; QColor my_color; for (int x = 0; x < width; x++) { for (int y = 0; y < height; y++) { double value_gx = 0.0; double value_gy = 0.0; for (int k = 0; k < 3; k++) { for (int p = 0; p < 3; p++) { if ((x + 1 + 1 - k < width) && (y + 1 + 1 - p < height)) { pixel = grayImage.pixel(x + 1 + 1 - k, y + 1 + 1 - p); lineNum_24 = (y + 1 + 1 - p) * nRowBytes; value_gx += Gx[p * 3 + k] * rgbImg[lineNum_24 + (x + 1 + 1 - k) * 3]; value_gy += Gy[p * 3 + k] * rgbImg[lineNum_24 + (x + 1 + 1 - k) * 3]; } } sobel_norm[x + y * width] = abs(value_gx) + abs(value_gy); max = sobel_norm[x + y * width] > max ? sobel_norm[x + y * width] : max; } } } for (int i = 0; i < width; i++) { for (int j = 0; j < height; j++) { my_color.setHsv(0, 0, 255 - int(255.0 * sobel_norm[i + j * width] / max)); lineNum_24 = j * nRowBytes; rgb[lineNum_24 + i * 3] = my_color.red(); rgb[lineNum_24 + i * 3 + 1] = my_color.green(); rgb[lineNum_24 + i * 3 + 2] = my_color.blue(); } } delete[] sobel_norm; qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::GreyScale(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { int average = (rgb[i * 3] * 299 + rgb[i * 3 + 1] * 587 + rgb[i * 3 + 1] * 114 + 500) / 1000; rgb[i * 3] = average; rgb[i * 3 + 1] = average; rgb[i * 3 + 2] = average; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::Binaryzation(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); int newGray = 0; int gray = 0; int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { gray = (rgb[i * 3] + rgb[i * 3 + 1] + rgb[i * 3 + 2]) / 3; if (gray > 128) newGray = 255; else newGray = 0; rgb[i * 3] = newGray; rgb[i * 3 + 1] = newGray; rgb[i * 3 + 2] = newGray; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::ContourExtraction(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); int width = img.width(); int height = img.height(); int pixel[8]; QImage binImg = Binaryzation(img); QImage newImg = QImage(width, height, QImage::Format_RGB888); newImg.fill(Qt::white); uint8_t *rgb = newImg.bits(); uint8_t *binrgb = binImg.bits(); int nRowBytes = (width * 24 + 31) / 32 * 4; int lineNum_24 = 0; for (int y = 1; y < height - 1; y++) { for (int x = 1; x < width - 1; x++) { memset(pixel, 0, 8); lineNum_24 = y * nRowBytes; if (binrgb[lineNum_24 + x * 3] == 0) { rgb[lineNum_24 + x * 3] = 0; rgb[lineNum_24 + x * 3 + 1] = 0; rgb[lineNum_24 + x * 3 + 2] = 0; pixel[0] = binrgb[(y - 1) * nRowBytes + (x - 1) * 3]; pixel[1] = binrgb[(y) * nRowBytes + (x - 1) * 3]; pixel[2] = binrgb[(y + 1) * nRowBytes + (x - 1) * 3]; pixel[3] = binrgb[(y - 1) * nRowBytes + (x) * 3]; pixel[4] = binrgb[(y + 1) * nRowBytes + (x) * 3]; pixel[5] = binrgb[(y - 1) * nRowBytes + (x + 1) * 3]; pixel[6] = binrgb[(y) * nRowBytes + (x + 1) * 3]; pixel[7] = binrgb[(y + 1) * nRowBytes + (x + 1) * 3]; if (pixel[0] + pixel[1] + pixel[2] + pixel[3] + pixel[4] + pixel[5] + pixel[6] + pixel[7] == 0) { rgb[lineNum_24 + x * 3] = 255; rgb[lineNum_24 + x * 3 + 1] = 255; rgb[lineNum_24 + x * 3 + 2] = 255; } } } } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return newImg; } /***************************************************************************** * Flip * **************************************************************************/ QImage QImageAPI::Horizontal(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage copyImage(QSize(img.width(), img.height()), QImage::Format_ARGB32); copyImage = img.mirrored(true, false); qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return copyImage; } QImage QImageAPI::Metal(const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage *baseImage = new QImage(img); QImage darkImage = QImageAPI::Brightness(-100, img); QImage greyImage = QImageAPI::GreyScale(darkImage); QPainter painter; QImage newImage = baseImage->scaled(QSize(img.width(), img.height())); painter.begin(&newImage); painter.setOpacity(0.5); painter.drawImage(0, 0, greyImage); painter.end(); qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return newImage; } QImage QImageAPI::Brightness(int delta, const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); int r, g, b; QImage imgCopy; if (img.format() != QImage::Format_RGB888) { imgCopy = QImage(img).convertToFormat(QImage::Format_RGB888); } else { imgCopy = QImage(img); } uint8_t *rgb = imgCopy.bits(); int size = img.width() * img.height(); for (int i = 0; i < size ; i++) { r = rgb[i * 3] + delta; g = rgb[i * 3 + 1] + delta; b = rgb[i * 3 + 2] + delta; r = qBound(0, r, 255); g = qBound(0, g, 255); b = qBound(0, b, 255); rgb[i * 3] = r ; rgb[i * 3 + 1] = g ; rgb[i * 3 + 2] = b ; } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return imgCopy; } QImage QImageAPI::transparencyImg(int delta, const QImage &img) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage newImage(img.width(), img.height(), QImage::Format_ARGB32); QColor oldColor; int r, g, b; for (int x = 0; x < newImage.width(); x++) { for (int y = 0; y < newImage.height(); y++) { oldColor = QColor(img.pixel(x, y)); r = oldColor.red() ; g = oldColor.green() ; b = oldColor.blue() ; newImage.setPixel(x, y, qRgba(r, g, b, delta)); } } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return newImage; } QImage QImageAPI::StaurationImg(const QImage &origin, int saturation) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); int r, g, b, rgbMin, rgbMax; float k = saturation / 100.0f * 128; int alpha = 0; QImage newImage(origin); QColor tmpColor; for (int x = 0; x < newImage.width(); x++) { for (int y = 0; y < newImage.height(); y++) { tmpColor = QColor(origin.pixel(x, y)); r = tmpColor.red(); g = tmpColor.green(); b = tmpColor.blue(); rgbMin = RgbMin(r, g, b); rgbMax = RgbMax(r, g, b); int delta = (rgbMax - rgbMin); int value = (rgbMax + rgbMin); if (delta == 0) { continue; } int L = value >> 1; int S = L < 128 ? (delta << 7) / value : (delta << 7) / (510 - value); if (k >= 0) { alpha = k + S >= 128 ? S : 128 - k; alpha = 128 * 128 / alpha - 128; } else alpha = k; r = r + ((r - L) * alpha >> 7); g = g + ((g - L) * alpha >> 7); b = b + ((b - L) * alpha >> 7); r = Bound(0, r, 255); g = Bound(0, g, 255); b = Bound(0, b, 255); newImage.setPixel(x, y, qRgb(r, g, b)); } } qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return newImage; } QImage QImageAPI::Vertical(const QImage &origin) { qint64 startTime = QDateTime::currentMSecsSinceEpoch(); QImage newImage(QSize(origin.width(), origin.height()), QImage::Format_ARGB32); newImage = origin.mirrored(false, true); qDebug() << "结束:" << QDateTime::currentMSecsSinceEpoch() - startTime; return newImage; } simple-image-filter-1.1.2/src/api/imageapi.h000066400000000000000000000056711417414176100206530ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef IIMAGEAPI_H #define IIMAGEAPI_H #include #include #include #include #include #include #include class QImageAPI { public: QImageAPI(); //Take RGB maximum static int RgbMax(int red, int green, int blue); //Take RGB minimum static int RgbMin(int red, int green, int blue); //Boundary judgment static int Bound(int range_left, int data, int range_right); //qimage Skin grinding // static void QImageD_RunBEEPSHorizontalVertical(QImage *img, QImage *imgCopy, double spatialDecay = 0.02, double photometricStandardDeviation = 10); static QImage QImageD_RunBEEPSHorizontalVertical(const QImage &img, double spatialDecay, double photometricStandardDeviation); //Warm color filter static QImage warnImage(const QImage &img, int index = 30); //Cool color filter static QImage coolImage(const QImage &img, int index = 30); //Grayscale filter static QImage GrayScaleImage(const QImage &img); //Brightness and saturation static QImage lightContrastImage(const QImage &img, int light = 100, int Contrast = 150); //Anti color filter static QImage InverseColorImage(const QImage &img); //Old photo filter static QImage oldImage(const QImage &img); //laplacian sharpening static QImage LaplaceSharpen(const QImage &img); //Sobel Edge Detector static QImage SobelEdge(const QImage &img); //Greyscale static QImage GreyScale(const QImage &img); //Contour acquisition static QImage ContourExtraction(const QImage &img); //Flip horizontally static QImage Horizontal(const QImage &img); //Flip vertical static QImage Vertical(const QImage &origin); //Binarization static QImage Binaryzation(const QImage &img); //Metal wire drawing effect static QImage Metal(const QImage &img); //Adjust image brightness static QImage Brightness(int delta, const QImage &img); //Transparency static QImage transparencyImg(int delta, const QImage &img); //Saturation (- 100 - 100) static QImage StaurationImg(const QImage &origin, int saturation); }; #endif // IIMAGEAPI_H simple-image-filter-1.1.2/src/application.cpp000066400000000000000000000020161417414176100211520ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "application.h" #include Application::Application(int &argc, char **argv) : AppS(argc, argv) { this->setApplicationName(tr("simple-image-filter")); this->setApplicationDisplayName(tr("simple-image-filter")); } simple-image-filter-1.1.2/src/application.h000066400000000000000000000040421417414176100206200ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef APPLICATION_H #define APPLICATION_H #include "mainwidget.h" #include enum MenuItemId { IdNormal, IdBEEP, IdWarn, IdCool, IdGrayScale, IdInverseColor, Idold, IdlightContrast, IdLaplaceSharpen, IdSobel, IdHorizontal, IdVertical, IdContourExtraction, IdBinaryzation, IdMetal, IdTransparency }; enum isChange { Change, UnChange, }; #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DApplication AppS; #else #include typedef QApplication AppS; #endif #define App (static_cast(QCoreApplication::instance())) class Application : public AppS { Q_OBJECT public: Application(int &argc, char **argv); signals: //设置栈窗口(0为打开的初始界面,1为图像显示界面) void setStackWidget(const int &index); //图形处理信号,用于线程处理完毕发送给UI显示 // void sigFilterImage(QImage *); void sigFilterImage(QImage, isChange is = Change); //鼠标移动信号,由于在view里面鼠标移动后,事件没有渗透到主窗口,特此写了一个信号 void sigMouseMove(); //重置亮度和对比度 void sigResetLightContrast(); private: }; #endif // APPLICATION_H simple-image-filter-1.1.2/src/control/000077500000000000000000000000001417414176100176245ustar00rootroot00000000000000simple-image-filter-1.1.2/src/control/combox.cpp000066400000000000000000000015601417414176100216210ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "combox.h" Combox::Combox(QWidget *parent) : MyComboBox(parent) { } simple-image-filter-1.1.2/src/control/combox.h000066400000000000000000000021271417414176100212660ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef COMBOX_H #define COMBOX_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DComboBox MyComboBox; #else #include typedef QComboBox MyComboBox; #endif class Combox: public MyComboBox { Q_OBJECT public: explicit Combox(QWidget *parent = nullptr); }; #endif // COMBOX_H simple-image-filter-1.1.2/src/control/dialog.cpp000066400000000000000000000015561417414176100215760ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "dialog.h" Dialog::Dialog(QWidget *parent) : MyDialog(parent) { } simple-image-filter-1.1.2/src/control/dialog.h000066400000000000000000000021021417414176100212270ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef DIALOG_H #define DIALOG_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DWidget MyDialog; #else #include typedef QWidget MyDialog; #endif class Dialog : public MyDialog { Q_OBJECT public: Dialog(QWidget *parent = nullptr); }; #endif // DIALOG_H simple-image-filter-1.1.2/src/control/label.cpp000066400000000000000000000015521417414176100214120ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "label.h" Label::Label(QWidget *parent) : MyLabel(parent) { } simple-image-filter-1.1.2/src/control/label.h000066400000000000000000000020651417414176100210570ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef LABEL_H #define LABEL_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DLabel MyLabel; #else #include typedef QLabel MyLabel; #endif class Label : public MyLabel { Q_OBJECT public: Label(QWidget *parent = nullptr); }; #endif // LABEL_H simple-image-filter-1.1.2/src/control/lineedit.cpp000066400000000000000000000015661417414176100221350ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "lineedit.h" LineEdit::LineEdit(QWidget *parent) : MyLineEdit(parent) { } simple-image-filter-1.1.2/src/control/lineedit.h000066400000000000000000000021321417414176100215700ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef LINEEDIT_H #define LINEEDIT_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DLineEdit MyLineEdit; #else #include typedef QLineEdit MyLineEdit; #endif class LineEdit: public MyLineEdit { Q_OBJECT public: LineEdit(QWidget *parent = nullptr); }; #endif // LINEEDIT_H simple-image-filter-1.1.2/src/control/menu.cpp000066400000000000000000000015461417414176100213020ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "menu.h" Menu::Menu(QWidget *parent) : MyMunu(parent) { } simple-image-filter-1.1.2/src/control/menu.h000066400000000000000000000020621417414176100207410ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef MENU_H #define MENU_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DMenu MyMunu; #else #include typedef QMenu MyMunu; #endif class Menu : public MyMunu { Q_OBJECT public: explicit Menu(QWidget *parent = nullptr); }; #endif // MENU_H simple-image-filter-1.1.2/src/control/openimagebutton.cpp000066400000000000000000000017751417414176100235420ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "openimagebutton.h" openImageButton::openImageButton(QWidget *parent): SuggestBtn(parent) { } openImageButton::openImageButton(const QString &text, QWidget *parent) : SuggestBtn(parent) { setText(text); } simple-image-filter-1.1.2/src/control/openimagebutton.h000066400000000000000000000023411417414176100231750ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef OPENIMAGEBUTTON_H #define OPENIMAGEBUTTON_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DSuggestButton SuggestBtn; #else #include typedef QPushButton SuggestBtn; #endif class openImageButton: public SuggestBtn { Q_OBJECT public: explicit openImageButton(QWidget *parent = nullptr); explicit openImageButton(const QString &text, QWidget *parent = nullptr); }; #endif // OPENIMAGEBUTTON_H simple-image-filter-1.1.2/src/control/pushbutton.cpp000066400000000000000000000017371417414176100225530ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "pushbutton.h" PushButton::PushButton(QWidget *parent): PushBtn(parent) { } PushButton::PushButton(const QString &text, QWidget *parent) : PushBtn(parent) { setText(text); } simple-image-filter-1.1.2/src/control/pushbutton.h000066400000000000000000000022651417414176100222150ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef ICONBUTTON_H #define ICONBUTTON_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DPushButton PushBtn; #else #include typedef QPushButton PushBtn; #endif class PushButton: public PushBtn { Q_OBJECT public: explicit PushButton(QWidget *parent = nullptr); explicit PushButton(const QString &text, QWidget *parent = nullptr); }; #endif // ICONBUTTON_H simple-image-filter-1.1.2/src/control/spinbox.cpp000066400000000000000000000015621417414176100220160ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "spinbox.h" SpinBox::SpinBox(QWidget *parent) : MySpinBox(parent) { } simple-image-filter-1.1.2/src/control/spinbox.h000066400000000000000000000021141417414176100214550ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef SPINBOX_H #define SPINBOX_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DSpinBox MySpinBox; #else #include typedef QSpinBox MySpinBox; #endif class SpinBox: public MySpinBox { Q_OBJECT public: SpinBox(QWidget *parent = nullptr); }; #endif // SPINBOX_H simple-image-filter-1.1.2/src/control/statusbarwidget.cpp000066400000000000000000000020121417414176100235370ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "statusbarwidget.h" #include StatusBarWidget::StatusBarWidget(QWidget *parent) : FloatWidget(parent) { this->setAcceptDrops(true);//启用拖动事件 #ifdef USE_DTK setBlurBackgroundEnabled(true); #endif } simple-image-filter-1.1.2/src/control/statusbarwidget.h000066400000000000000000000023051417414176100232110ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef STATUSBARWIDGET_H #define STATUSBARWIDGET_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DFloatingWidget FloatWidget; #else #include typedef QWidget FloatWidget; #endif #include class StatusBarWidget : public FloatWidget { Q_OBJECT public: explicit StatusBarWidget(QWidget *parent = nullptr); signals: public slots: }; #endif // STATUSBARWIDGET_H simple-image-filter-1.1.2/src/control/toolbutton.cpp000066400000000000000000000017371417414176100225510ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "toolbutton.h" ToolButton::ToolButton(QWidget *parent): ToolBtn(parent) { } ToolButton::ToolButton(const QString &text, QWidget *parent) : ToolBtn(parent) { setText(text); } simple-image-filter-1.1.2/src/control/toolbutton.h000066400000000000000000000022661417414176100222140ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef TOOLBUTTON_H #define TOOLBUTTON_H #ifdef USE_DTK #include DWIDGET_USE_NAMESPACE typedef DToolButton ToolBtn; #else #include typedef QToolButton ToolBtn; #endif class ToolButton: public ToolBtn { Q_OBJECT public: explicit ToolButton(QWidget *parent = nullptr); explicit ToolButton(const QString &text, QWidget *parent = nullptr); }; #endif // TOOLBUTTON_H simple-image-filter-1.1.2/src/help/000077500000000000000000000000001417414176100170745ustar00rootroot00000000000000simple-image-filter-1.1.2/src/help/helpwidget.cpp000066400000000000000000000017641417414176100217440ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "helpwidget.h" #include "ui_helpwidget.h" helpWidget::helpWidget(QWidget *parent) : QWidget(parent), ui(new Ui::helpWidget) { ui->setupUi(this); } helpWidget::~helpWidget() { delete ui; } simple-image-filter-1.1.2/src/help/helpwidget.h000066400000000000000000000020761417414176100214060ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef HELPWIDGET_H #define HELPWIDGET_H #include namespace Ui { class helpWidget; } class helpWidget : public QWidget { Q_OBJECT public: explicit helpWidget(QWidget *parent = nullptr); ~helpWidget(); private: Ui::helpWidget *ui; }; #endif // HELPWIDGET_H simple-image-filter-1.1.2/src/help/helpwidget.ui000066400000000000000000000216631417414176100215770ustar00rootroot00000000000000 helpWidget 0 0 431 497 Form Help Qt::AlignCenter Simple image filter tool is a tool with the functions of using filter, clipping, transparency adjustment, brightness adjustment and contrast adjustment. It can be used to do some simple image processing. Qt::AutoText true Help Qt::AlignCenter F2 Qt::AlignCenter Open Qt::AlignCenter Ctrl+O Qt::AlignCenter Withdrawal step Qt::AlignCenter Ctrl+Z Qt::AlignCenter Reset Qt::AlignCenter Ctrl+E Qt::AlignCenter Fit Image Qt::AlignCenter Ctrl+R Qt::AlignCenter Fit Window Qt::AlignCenter Ctrl+T Qt::AlignCenter Rotate-90 Qt::AlignCenter Ctrl+Left Qt::AlignCenter Rotate+90 Qt::AlignCenter Ctrl+Right Qt::AlignCenter Save Qt::AlignCenter Ctrl+S Qt::AlignCenter ScaleImage Qt::AlignCenter Ctrl+Y Qt::AlignCenter Zoom in Qt::AlignCenter Ctrl+ '+' Qt::AlignCenter Zoom out Qt::AlignCenter Ctrl+ '-' Qt::AlignCenter Display shortcuts Qt::AlignCenter Ctrl + Shift + ? Qt::AlignCenter simple-image-filter-1.1.2/src/icon/000077500000000000000000000000001417414176100170745ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/close.svg000066400000000000000000000020061417414176100207200ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/dcc_11_36px.svg000066400000000000000000000030401417414176100215240ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/dcc_fit_36px.svg000066400000000000000000000043251417414176100220740ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/dcc_left_36px.svg000066400000000000000000000012021417414176100222330ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/dcc_right_36px.svg000066400000000000000000000011701417414176100224220ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/open.svg000066400000000000000000000020061417414176100205540ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/reset.svg000066400000000000000000000037501417414176100207440ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/save.svg000066400000000000000000000035561417414176100205640ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/scale.svg000066400000000000000000000032621417414176100207070ustar00rootroot00000000000000simple-image-filter-1.1.2/src/icon/simple-image-filter.png000066400000000000000000006467541417414176100234640ustar00rootroot00000000000000PNG  IHDRߊgAMA a cHRMz&u0`:pQ<bKGDIDATxw%Gq6?3s YZ  #"p"hM6_^ců M0N6&@Ht93?gݻ:ܓLzꩧ(ƈ mhCІvrYІ6 mh+CІ6 CІ6 CІ6 CІ6 CІ6 CІ6 CІ6 CІ6 CІ6 CІ6 0w,6JDk}\CІ6}NІ6 mh І6 mh}hCІvRR'|?CІ6Ss%:vCGСmhCe+qִYޣy>;5CІ6VU>vw罡 І6{;QB}9-.8۩CǾ6tCІ2[Mg}4{f;w;C~mЇ6 mn[[mg}4Q\kYy@}>>t}hCN|z{ogG;sz-`Y' qCІv٩/A/Rԑ> 1ޡ _X]XN4]?ŶWsSi+W\3=_^o̶c[Cޟ-SWn{\﹞1y N%;Zۉb?z ?Z'> ة{h}g`{yC:dlrС/iJdJu8!VYo??Q![Ƿ\q[=3׭[(VU5uwjoy[ƑS޼N}%|8҆}JaT}rmc5z '~4e 7n8m۶mιΖ-[;犑cy/<)Oz_·o,v~}\:>g߿?9~:z|#{nw9EQl)bkg[kYY_۱cOwQm> /7oZyr'"~Wkql )澜(Xg|R>Gp q1RСІvR*;꥾bRuvE]tQQ 35!wB8p7?~@jEC>tCښqpC|uq5lxc{ge[׭[wneGGGϷ֎t:K(<9du]O}ӟo;v}C܇6chC|M կ~[nK.زev"`,!5fW݈hȿ>O|?a\~aІv6h_[K>___{G+rIOz sscEQ$|hbhϭϩk|;|Yo|J+bܩWx,|+^s9\pEnymbttt1(:>ܹeNya7{S}N{޻G2 -%O>tС픰8_g|^{(۷o|d||,wݳ,z4R&TGk:sˉO>ǵ=z<SΡsC;vC|c+ǟ򔧜wim:;lv ~1֎fwlhlcbKp;cKGGG;۷olذ,ƺy c|N|}9tu }s=ﵣ}xWZy ! C|kC`O?sFGGtGFFvέ/cVɎ#_LdD_["`61ˡv#S%R{"Z8lЇ6`C- oڴbcLg|||1u:8Nv||bJdNxvygٟϱ/O{8{ O@:xBgf]v.rkmqm ;6fYv&N;|zBN|9ҡm[fW܎Nzc+~pDcq[h_lO;!^+\߶m[/x6l?묳S힙e<ϳ֎ёė*R Σ1G8 :وܗe>ޗSņ!ǥw!9+z3::~Æ :F/5?Vv8BNhEI6D>7!c[mǽ|N>ϽbCFv@೟%>ZB >oxݺuZk-[lw΍\`]8Ds E9RRg;cic؇B'Sz:ޡ>ͫ3O2_W^Ȧ3,˶y~v=v<RFߛωͥǹrl];?|(l"w*ޡ/VYo?BB{6lpsn|||R"ʺeڜV|B|)N\ϯs:C" Ύ̛9|]#tN2":<0t+Co8Ղߟ p opƍ#۶m^\l< mv":"A<P^bǯ qꨱ$g.L۰FG4-cAկoކAq~&l>m.~jއS}?5׭[g7o9s/vGGG)bKQ[ɑ/%oNxہ_$bojνg;q51lvW|m8|hc`8t ꚕTe,=>8.8<<ʇ~*lܴ LR,<mvH.WW_W?GyYgaÆ-ׯ?'˲[kNYohNtG>n~h<ugEs=.dKӈv3fN>o]w zMwbjjݷ8ppj A0d@O]/|B1 {C—>yJllҵuT~=):VgwV R!OX5y~@ٖ-[.7dccc!N%Gφ#_F})6_N|s7Dw/q v܋Ç&119CPUb SdMLr""/, ,Qr̀C1" Lj:ԑU#ǛI[?zZ{Þԧi:tGo'g{4y)[ }J@### .(kAgZk7bh'ꎼasK97ӾQ+uN{LLLcrr&Kdrď),!(FYh,XK؍1!"c1eU0sǿ7i:>ҵմYr>{@Ky5\q{gqƖM6ނ/֎t:K(Nz;|YaG! ٹt}7~n߉9&'qxbU]"&wN؁g9;UD811HM#;c AK020py5BACd?Qz#Qz2zGMObϮ?U{ҵco'Cq.'b]m|}9Ϭd!x9sK.<ϋ-[l|1f(  dS͑k@a "u߻n}'&go!L > k<-&gM`  G+ "  b$ڷ0$6enLd Ĵ]Ι;!i yt>T!1c,2o0k|}c>61ۆ|v":\iq︬(/ޖȺu.,bcQgyZX]9x^tL~d'C.`q~RAW^yS_nݙyo9976;USS=u۽޽paLOpa)'MV2YNd[/{f\4%H]BkEAcA!3ZޛxpA2LCIQ6B@`<24h@C&b#zGpr1.~9d Q؉rf ;3OE]taټy6km!xQPhah+ّ󫷢7ދއazj=ص&jnsc;7i5J62JNڐ\m8Fy3=wV1eyՈ<&G f|]37{eC6;u2,22kJ! K'L}z?~+lJ׵|umv8"r#h:3kv{Q۷+en{ve[s[DԵݑ+af;ngg~?pS8pM]G9 LfYAMsfxܰd`IQID4$ƤHH<3e%7 0wN x1E1*]6 LD%fDzώ>~1&+R\ܡ m9v7t+fpm;q9crz{C=}q-uckYIhFIMwSeɁ#$rh\ȰK;}Y>FR?Z=q@#Ηok b!yccޕ w$Dc s m¡CD@SHt.{,!BN=:`ežyԖZ6C[C _֣N9y֫ꁧ~y7n8Ns^e[,rCG>Vӑ+aj;nMcރػg?&{&%2aQvTcaLfBjEq e32 c#eM%l$в:dVH[|1~ o*}icw+D "q=8HhGCbCl,/V BRsAo@]թAb. ",dq-@KP{cARXƠ2|3_ów1ҵձrAy_ٹu<ݖȽwcz؃C&gA޻SS3ؽBhD]cak3 6`!9':T5!1Ƕ+G)q77sN$q>(?X2Be/MWt?t5Q4E9p%qĀm& B r"t"^E·k?adbdQ0>:TbGs2T> "o8tkZ88 DgyfOK~f;cl15Ý݇waw3{ЉYMY 6 48k2 Z2~dGX5Z7F[AӈXB%^",F*X$o/Hsbχ6a+`j;nMI޷3=ݿu-b"Rl9 "`HZʴ4Zd@*9j L0Y8ß˝CPϖR&KkDB@%G{&h^ (gǚ܂HgܰD#iȶ|5cY8~cl6,6p sowk]0Ԇc${ ' XЂ3QGp:1ExQħդ߇avq&)e. ,0=܂uGg~$f~6)x;p#sƁ@t?nݺ+u?ss9>VjYOp-;!w{ v۟zb[2fXjٌ֡(:U 0F0 g9ҹK9kkTɁYcR1@2Te9LNiP*E^5V)\bkYԾp /|2.|% ^|7=ߵOKVreS-N7d8>R,Jtyο+) ZFD?=ܞ0=jj&,Lg@?֘+L%~#s賻aЗmk`⊟ʝymo~VL]¾}_ vݛFJX#L/vʙqP&ѱ!#ky[P9F E8jg-5ŐXeBu5eMdرIr!p}Jn3|{陵2=2FNN]r>xp psMZ?>\x^>j|9]-zVR\k]Y:F5q\G9ʪlMQ.d~a񶂏ȳ1LZ,g3TY< Wc܎CN7;B7~Yg=wO?i}nwƮ=WصwHYįN{d0B@#ia:lI$m;c3y hfsg Ȭr%,FV%$7dRyңRE@FA)9hEʊ! FJ\kDVTчLORH5[O(2m_i,cKEQiUUT1@ʂ VI=L5$B?= K۾{':&ٰtmv#糝\P?-[Z:V5n.>4{-(k~8v{n7%Pb5EbxhN "'/C%G:7DȬ%8h.Ra[E͐I"5c6>x#MLhMΘ#Qp YPK>Vc471oGdyj斧!DpWûv8BO|pK19Jˋ?mOATJ9̀ADe=| >_"xϨpto-BŐmrڑRSւOdaCDbWUx 7BK'KM0 \cajZCs՝0tC[ :=ػ Į=صwo"8QY&q%p[wfj|6#N%ZA+Ѻ#{ ad,K dY. j0s$m; c9v9&b*v{dE ’);sx 0*UYx]X,S cPKc jOr&sl!z$60Ǿ;~I/y .|%Kr qbS]5B_@D:gI-JF&ta>ཇ5Fw&uGGЛItA*D97w}*7K`G1Boy !Nvuax9tq??3~|ONJ~LL=سYQU3Đ;x\&H q9ǰ >pPpfAY)R6&Nl^V}Ty- H LfP˦ZKHp#o{w^ݩgy]'䩸!M|snR0 }1{ |#\pDUi#UY0==ݒl%.\G()Dp+5!-W~q`adXU}a2b-E>ڻs1kckC_,Bx핯|%~nOЎUep};a :8]}4Z!%YSMSE ;bLDÛq !i+D,% ^+YLp$ .%\waE]8 q=c^{! a2W7,y{޵:~o_w;Žb8t` D,qR.e$gMo+CYrV%Fcd믛.qC u +^6 6ObtZ,:T9G" <~-9]0"+0SR*'FBYۿwkc_-;ސsQ;=a\44>ͻph$ fL`5۳:YD֚j ,Q&C&rff~ ˜bN`dyPU^-q2 )G7We)yFo|D=GC>4| ҟf1j$eBZc]Vn(NLN4G*f, >".%~O!mf1&O.WH@V[yJO 9#/53獑f(FUɹv~.'ᒇmKcm9Ŷ1g?b7'8NhXiT-!.sУz ~ k,|Z[$s>!8@(rdnX԰Wek ID5;޹SYͮC:e&ŵsG@}~EYBGAuֻ ܷs7ʲ}{E,#:pŀ1)mQ-96@SCmd.Cfr됹 Cfe8\3%2断1(uY򣯅"-Ѷ͐tPW#FGGPZu}hQ<6E1ĥ`y5JZ*VaTcQ6 k]s }% pR7HqܫаcD=B4}w݃{׊vb@! z 'N=Y>s>goc)s}f_\/5aѨI&r&(RZ6*(gzx>_/qp95/,r&1ݍ$oIM)Έ)R]ot>V{̎Cyώ^{CqLo܁']صs?]{QUYŭ(io9r#UZF;l9&qs;wblQwf*,g͸~)n`yk->qz}P#u8 |(^r>0⸫LlbcLFW+x@Dc W]+s'*]Tk W4kΥ/FϥJ:풪# b.QN!z33RZk'"/+&WЈ5G@YǿIa6~]I=J!K 7c})9uy(6qW(G?,ׁ[qM;V9u gufB+7BB1pHLc5Dؿa:,][HSMo/\637܉{{b}8tPa ;f:gZIMp4G1i[$&@JLrN9P jgr8ggQBȌ0Q.&!g`6d Iw 2*A k^C"o{\{G{nٞWpks wZ\≝?!U@Qq5Zei}# /ҽMIr p1:33YT;x9_$\ss}.}*rѱQބwxU{V ʙ(DEj_2 NM5E7w*%TKUHq{򢃙iD 5J箫D0IāC<Ğ9Pוt# WU$T'z$'"xO@ւ Đu]UIM>9.m=y|)ٱ"-Utط=xO(>IN!)o{ūz[9߹&{]0/jc5Ło~t5KVfk/h\sҖKZ:3{o!ܱwSݨj{N[ 8uY_4Is%G dY=qFXͤk!g r%.]YjM8NMςe F> g f&ER&9cf{'S"ɸŤhP xLURէGa4\geO+#,Nk<~jnu0&Q9O33 Ur_1Dt;#HU]"цrJB䶛{8Z*;_rm#ν\-(;.fRԺh.9Wh-hX,SzٯAK4 5ՙ5q'eku$M5RtxyXx_ (#|ͯ) ,\:X3XCLB cNJyJ׶aXvTvB:n]}X&Mkؽgۭl&_m2 ]&P7߈FPR޺ZU7<ל:`!9k?X,"OJ" kTe]rEjh4nJHQ#_vǬ-^-e:݆Bz9F@F@IIL9gۼ$p7/َ&dbbs^UYqhkBr1aV*m/>bګ'oohVͧokgnp5$Dh!&( g_P"HҊ4I"ByKYW "nY:b2taGjPҵҡvre3タ훿ya") O6Μ r֘"+JLZ2B>cX1u2D_ v #.Ҧ#V|ÀD4r>Y^GLz=ɥmY;QU'9c;ʦy:=xʙR"~HvLR>/Q0XSLqZ(߈5+zZa%Ek "c`6a}M3A6}UUDЅkk.IEKȚ$ JzܩWyϹl\ܫq#ཤwXGH8߿ן5'$6-"#B-ܗ&*n̳1|NR#GJeЅ`GyN9"Kzig+$s0sj 0,];*;=-hҗN_~& ɐ;v$$쌅Ae06[50;r%H]t,1ZaA{sNJ^$- <$ #2,O8w @@ "PR0Hf续͚A\˖cS޷[(CXF<6\!>lno:V#es%?"TDS F>p;[7)> Z:@<+>A]9#JYpgźJo<Ҷp'`#wjjʏw_W2{ygZWeJhJk/~J\m#K~`߃ " LF4 D-9Xpxeq+W ϵ8ž4*r##x;߂[cCy1Lz uJ(=}⊨@,eJ$D΀רB Gz/foED%E`@nǃ4r\k#{Ӝ='obE6cַ{]]{\w޽9glp96lλؒ𿬃-|z梋MEb.]/ߨͰnt[m@eq9 rdƢp13=Zݹ N;tQYˤ&ː0aOz$N;c+u'!T3 tkKRQd#a9J1MжL D摽7iɬ*W;(>.:%MGV%|9{5#!xPLމF -ҽ,r+!Aqa2F=msdдKP92*ac關K~풜罅>\[b2]KdR֫~ʙ+Pq͠S$FWskxf,[q vI~1:+yVp֯ڒxAEcFq~/wCf#.Cdq wȌC2׉tkYٴ"qGrfА1Z\E;v7/2Wo5UVJPpAsW 0#qxLF!S9Rd&crڀh !'KeZ&KC#'`oEU}c>NML9MJP{rnڔjԟȓa)2'ZhgIի]ݤ)Rќ:EPU 'Rn\jijYk{X[xvl^l{N|!jڐBl?7|z\AoQ;"H?!BC\ZcҐ@}R"M1G)@APZLZ\Ee9ֻWJ׆ [ Rܼ\mbb{86/{ }WO~mFl1ks+F;VIȠ ŠZ2븙)%%b[D̲i4gK4^.sJΙfY-Y?m#hκtF:3߇!ιe?l\2qčJEWxR8Dj617=BS.b3,B(:];iLIv0HΒ \`V&QaȉBS\%{CY%nZIjA􍃗ߴ֠BJVH $(]G>o1ӽ`>՞~Ȍ~#qx `Xq[s_i=;p$osƊ#\~4c~g𶗼 {=:$ $嚶[!B#f(C)+UA"(BUqW:`m3Gjf3ų_|Dad)st+AFkgKv諥i"]ݎ1ce9F]0b GPf4DYr1E<7DYl`&rM_V1D8ˑp׃sNv4ꪆjiiҤ11<+ 41SX&(9EY8ǽuzLvKzS3)U-d!%J0߶uL>KQuwj MU YY5%"o7Ѻ 97d\&g\hdSK(2v!eq/"y(y6 6"v2I a3[I-f3W>CC~K9}83qmG{oz/_1{>?y_"PMᱏxՓsha7]oyfت'p< 10ѫz@|ա|/xh+1W|<#z<^fnapKt?yzvui`󬼪ڻnkRևqĨ0 gơ2-;}U$75ֈ;W7{3˳Đ 'm;o}|rye#Wq$z~blJ`EdfƯ*~g]:|f=ᨼn>~ 0"sUqh!ۥ ƴHdu+A/}NђFApiBU#ϴ9zk }!y5>n6@! 1=U=Lse՗D@D.x;g\t&'Gny{|]/3ry1ׁ>bAB^|>˷}>ܮ|m9*rD!z )Wuզ.9BX(ȓS=x4- b8@Z\bT/3t}ԭCGD[:8c=>>~l5S6Ǩ-PX-NH:3}%ERh`6yB*d"zQa] /6%V%BaDd|%sJXy ?\s6У\2~[[F02ik#>lq-m(,G0TOZHАڄ:#nq,۝Rj m$[MHs\nOQ۹v0a6vs c1uJ5oz% FDg<θL\k[<"_):{[ ;3oZ>VYʢJTT j z7q n ŗ^s>V_w^ڟ7zIqIX /K*%oF␥8XӺR*WB(pFDTȜvXssS;Aш2%pv`,:>;vkGgkCgffʏ-q }> "e 6G9d7QZ%*Cw<0ڷZ'Fo~$ϲ !nIu Qra`z!!Ed6OΏ"GUVXs?XܐCo~T|&%Q`#`` L ˓WYoIBpT,5D1rClpXO'f3V}mAh! 9W'㍲CTSCk""T( nlPF*"qG[/8W= xj|ooPUrfQ8脱fZPE+"C%1P@"?~_ۯB'Ge8ih`iݑ.?o~O' ] ݁ZP8+!UEaV blz$M)ĻEy8soϸ[` b- LZ$2a߾#9t=+'2!}v"Dmk'vܵ_/bfs]uTL!3a[ybx@ KY$`[ym@d,#ϋBl!9]jGnڈˤ#B Ǯ_GݓqsG- Z?^?| JN* Dk@αCw,)|Kjo?TEeX@U="{wcap_r^W~ 6r$|h-R& D AR*[# J.Lh=oc={gǮzag,BKM$HKvqt];4S m1[Ky6<{ pgP0r&2cc@qsJ+aÄ$ @p6Af)t"؆:p,L,Uʸ,%@ %cWQ#}h)7xiy%+yerͱ;;n ԓ/yPH\!D _{ "#U/,^%20т i$w{sB!^-<[Q}ݴ:rp1)^55kA Q[Uc1bEg>ϖp,_>06rPB5U2=vIDp9+MG_{ ֍rRu9׿u|?]q齢n,؄5$p%Z"VBWd[qǍHuԹMDATgg:WK3KזoۡE߿qo5P\ >HORNA!tܔ@ g6K .AdѪ9F,?0ahnk#^05׍GooʺSJ9HS55,]ΔasN=Iݬ`e9_$%>!QW5.x㺏} ud#ŃB3TҔ ѝλ(hݛ 'CcMcA^woI*/hG5b:F\ u6Ht"ĺRSoyRtiOd R_Ϳ/|S@ց.L\XڲxG|?vX} P!V%(:ֈցC$3x!_c] n)/{ko~ZV"Rx@ 6Cx{mEɸ \cp AF-n_Khtm5z==;eۡ@d2!)VnMek2GUU5C;ŀz[2mbX'=iTEr֤{@FO7ck!@ 㝤s㖧 JMZģd?0:>ӷ Ye^h!=k>gqd8J*` %zbSWN bըœǦs6\=öG=[Õ5GTW Q@w0 \r8wv-J^p˒fU'tE#ڻ[-s}~x;&IxD5Zs{P(R&@ 5FzOaB*ekI}j#mh\k+:د%G+onPXgKܹ9PsIce$|L"Aöpέ WkOmД%0-6SYu!iY#^hdd ĚƴT9ZqNe .W$ 5|Y62HK"_nwaGFT&E)rsr9Ru䤧9aZ+ vHQkH*R A1yN jq`Thɓ7qL gpV*uq$*i瞎'ɸXј~ П +@;YΓ@C ֻO-'6_7^juRϑ=e4 PpWxcdfj?QWG_l|OriNtel, WukFȂd\k*pR@ERYhSn܍K/ jӇk˷]ޞڏwz}m27r>npVژt#%' 1Mj"wZ 9ËƀEY\}gu6 g[ܕ-%Y$%6UKѴ81q aEӒRcf3}X|;HqsE1g,{Ë uUD #+tP"(Gש+KII%A7! 4kD 뼉GS(\D.7(-6u*VzuB@~EhDεō,hS V kiuMyrɭ{(A{ T0h6g2e0f9bް?Q`)ğx*嫒#~YG}$'Ỳ,.尝.20,aC g0:bL`z3xOΫu}597O?՘Hع רE5b+ȥ0ùUU"Lk" qZTZF0SFwݾ#RR#-֢I\ȽanܘY΃H_}ѵO{C%E7C}Ak$2fyPɡ;tQ!B A#3YCgڗ^I^^XúpI͙k]ʴxRǨ@ 7ic.59~MߣPrؽq6{+dAx`Om8SF'UY#)4bs* nfzOz5$S[يK3?zqgrru'߃a| t;03 ȋ(@E!kd}#ϣh`"֍,"etC;wչZ瞅OˢPna͸K=ҵiS'2EV\*t5$#2Rtgw=_fR)C+1DJcUIvB}`M:J> `w}(meXtC!6 "wo U* ID0xPPJVnLfc jbߚ̤dM9&ydHW~y?4\Ñ?H@Ct,_@6C& ,¢7=p1HS+-~(;s*6DTǃlqoW>Pn4:q`d A)=8:TdŃƵ^QVޮ>C\T4cuA@K~/<棅֗Z?:6Ƶ06>! o a̷(Vg['6! D#5 }.QQ+0*sYsrҵZΛ@SSSsF諩7~urM#Fm<-j89Ϣ} U2.r,# of ڇ9o^sIʨj>p~zUXD>;E9OYշ1 jNP{i8qJ*[V#_=OE.ۊm e}Un:~&:)程-9C`95NjAkC̼FDga9 N&gUFTgAeYQ }CJzktO jʲ ʠleLrV&9e<g[Ajh4yRBڊeoDLi肶|BhA .HtÑP>x|m~7}_児:09Pn``#質[q6eE =ڰ|DcEgO4U ak7@6~`. , ,L#xꣷa,hZKs^||ㆯ^&}&gR$*tn4 SII$(vY{N>$g\X'唲 Kזgkm-.`ΝToTB8!|PWA'*1R]U Lo1zy3Y/{hQ G.h~64ֈ@s15v0HU9Nzp֥[_vSWY(QMg$0"*BObKu@kF&"e B)Kx65ڍ^V;#Br X@|@]Jނ1GkYWބؑEHMbEBo7 q)cj?G8U'E4'ǂԠk"w+4b*+h` ޸1·>ߺYc_{ex|;^cjrYPކF-' uyf%ROyMhz%FQrM .ٌqm)۰tmvrqQ<rxo_Z\q96a8LoUIoL鑻%J;jվڜOBz}#Eb{9ʺpT>m,ϸgf UBiY 6Л,o}0%у&7ع:E\a]5Ƙ %)mI|&'}혓`tXr1%)W"3<D(VI-mH\a"A" DmQÙA6=1,P4pp=Z-B``6HEci"`!-K\" Iф\9A9,!}YC][PUwX@"ܪwEP9Խ#[f9)ϝD^bXIz;:H#(^Gɯ?c#~UAp9A> ~/>!P#CtT0Z7y;u ^-2%V(.-D>$nB{쐁 iZ@*?:v>]1pQuB<;,NY s@o=%rs>qxo9ru C|,^F>T]_Dy eUŽv`ffZq}4оǥi;=ƕ/K1,][ez1m !LcƖC #ClKHZRRfCݪd|bf՚{a>FWF"cLe0 U#ʦWv\Kt}ᶩ?6qN4gJVJsyk!FE!j zॾ]AKsB@\z}P3iދ@ uf02܁.0Ayٽ&O%lIOxbtw> qZq-6l^ǪnBbr'9>zyaV[I uD@@bOȎJ1`R.C@T֟!O$~7KWԓ&f"k'_c5ż#4*{=^ɷ1G`դCdEv(!zYxI-W S2Y H{/*[@JOD$5ߟ`qCj_zï?s#/sxcXԙ/> 7aFF`Go]ن @ ;:퀊 &0ɹVBng.9h3w "LLP$;hVDL'ՌIwHFϝlsQ\:H5ꊼߋIØ Toe_>7`ύu-Z s$V4֔:8J'Uid74\c/PWqM)$K{J1Q朻|X6k\ǖG֍ r&UVCe Kaj笰MVwN nqLh2}tzi+H^䈒/&n#\ƹĕА")Jj&u#&0Hc)Q}"G!"@sYюi ?*}@"VQWuH>*#3w#5[sڥ gns~x~B9O6<IS0.ܺl؈l&t6mh3[,lrAH+h#eq(6I$5Y8;2+#=2̬9q]TK j‹cdA!E#c٦\a|ߏ^υ>wo~us:uQğIU*3c]2T1uBʺN?Oz0==2 8d$c`p-wCo}Ёҵ-`'B=7^Y; pz% mphB2aj#JL9fwjq`ؾ}2DE~DozF:ٮW n`)Xv)e$gL6R]w.J%g&w0 0qu0Zj{GRoD'L` #\Pʄt]Mf^thd# 肋 7鱙EeAYW ~-xKzCN(h~'E!N(a:nr+,G3a ̝J"sUOn +c!:EJ7tJ@0rku/39MeS#[[+~Q}9Np |iW>/8c-_J g>Gn7dP'i7@sYA_ø[`3-\TMG{ieylD4,][k\eY 1nizQ"ۨ!fA$6CYǺ疒 oS5(Xg26=S8 _߫ff4dMڍFΨu&{5)]Ϭל̑4Iq P](iLsړpl`\P `aL@ ]rU>-(4Ɖ^UURC([h!xY *o>g W>X`R밖_aGe\Cp#@:Lz+FGPrF.DB#3>%36<+^ >klڎ5Z Ք;Þ1.,jiz8jSzbALnL,_|>2pn|N}1x˯ Ϋqxiq$}|^Ӕ,IOMadN*I}LۓcT b}5@hDfBN}s6,][u]mǜݸiF ʺL9c&m6\Sf:'5Vu"%tJ竽ot֭]ƍqaGA.Jh amo#NymhyAS#/HVJLr> I\$5r… ]$/rY. \x~C,w<q>Ԡ׮wKwʃ_jt)ߛ-_}kjx7hBJ=iF(Xn H){FI*u9탕 RR^Z3IsF}K hBſ#YyΪpы`0:>7,#| P=JdTVeCK0;x q΢Ig|fBJ}덧R  -rhfa:":8LZr4oD^}/G5Pms:ȣLjugle˻eo݉_}ߧ1g/Llut:0c-yrtk$BLac}(2 S#g ֬ KJd(5Xr֐s'6)A3Ad8T=(tVg~ >Og+2k6} #25=h}ϵ?󨫾zwpƶJҚkMadB'צIԼqmA)- []VqN}sfG)e-N6;BZܱY7z$E2J$_ :Z4i%ߟA F)aSn'Q Ó ׸MoK(b!DFJ'+t#d;J7!r-4 QdC0΢St13=@zE& - ;m=g.ّ/sߺObz^0#ہ"[Ls ZmUivcd4nQ_1ZH .$5%;t݌%JxTV`yY603ITJ @5σSVArRts/v\XScA8Zс9Ⱥ9bxP9XBM3x>R#` e/='$@7?m=e9/S%aiN;0.lg42 "頎E!(YTnAcVgT>Eda&"p"ye&m!!T%dd("e>sa YQkUo]!y`4=NU,\dktAó_[^_jSmu oyOWȊ-蝯I=:2 Iڼe|EC(Ga3ʁ 00Vxynǝ;WDڽ!V" ~4t#{N7߼Kȿ0$u;'ZW~(%E&E}6 s?.k ל;}Pc%J/^-9o$TYi?lqmڰּ].{%6!E&1ع1{p`~]|I6Ār(9ԩqIR UOQژIx?My:rqkˌ2Uu_ƑRcXf7^NUY&H(^(C DIDATK٢PFBN4ǓN)e'JHv|'j6_rToB=?K֫_?7=u8G2knjL~qvML_2sI֘Gg wzRaD([|UUQ@mdH8ܨa1;\((zl(>eIrH qZn@7 u-Rh ~Y#:QMyO 2gx+uJ˖b߸u7 &u;(02ƁqPwAgl ) F:O߾V2LZ܈DO{ӒeܥQXM0QM,2ݽ#d"nh[bQ<,T(]*Jr(xĹvCg.E/<Iļ8^HYB>w<,嚷K>uxӯMᄡJuJF5JqihuT$& Xi&&F-E"% `Qa}\֎+e[4tmi֤sH@=S-nzE c3anEj dED0ҹ(HYKzRR1 e|V :c!#eܪvtFҶ\rp.C>\wr6>MMDF&`Եg>O *ϥZrc4478iH8N} PV}ؖq<%Oe9rՄok~?C&h oye,Zt-WTD㦻v'\801ӔJJDJ! ߁$bcCDx#yo?t 85h|1w )pŗF6&e0֢*Ծ߮VYךH{PVrmÌOZ*>kWJyR,d~1mCMh@CXK>ֺx623Q{STU-3:} =襮|t(`cthBVnk;^BX~Yd6a}Hkd؂vF7,#,K*z$锤($Btڧr)DXdA Gʦ Ys KsvȉQ j2 h{6Ȑ1`Xs4&⼤-VA$ȸ9n^K>NJZ2侈"*Ggx?goݸdp)jK_/]۾}KzMmS T"^^TқUϝ*i=]߅7s֬VHJeå-hk̛^033d?@N}i$:ԲV P"n "C<BӅk&87giLc<|0ЕulByKYhc @rHyJ`r DdVAV<<MQDbDͷ.pv4$Ҵ2aDA|ͿWvP5l#nz<ܡ3x*.}G񱰛ڇo:va#w4$ 3܍o6jKXe5c xӲ/:߁$QCJU$À!AC y\,)d! NX7@$=IlPZLA vp_~ w!E&庀Y%^ӿ^riN:x^d֐G(Uy$#(s[´|glq.ow ]V(KϥЩlZ#a:oyA Bw.ce1Y|Fu7Mi)cքgQ]Sk)~E@c |]ct:{A 9#jn6cG>#!(̠d 1pUPqVt @֢tL`A,yR 2bG@]֘DDDw}W}LRrؑd]0EtsV|:_e%B nMhpש uU煞:NDro&zYt#]ې ϰ:gmy|.}>nM}šI;x`"-CXgdưf%rP b) {^g-3ȲL,&0@} u]IM/,i 8A}059HIgV8DTLn-F>F ɦ*˄ EJ]v{%z'ĝ L4L M6XbWꀑcx̳¶GoOLX8ۼCxo~{vc ;N!] )A.C16-cȝAkc)ErŕR6s$Ss HE *Ʋbr~1 E)3z硩F %L><2&HDXdJd6ZQh6:r04U> "60Bg (oxc.K|%\9_~+OV4Sۨ>yI).7YXAr3·t4a Th6 mA;bfr{0)\I 3} 1 FJ%m: J;Cë8_3Dfqw=֠gm?'bwy"WkHUdRoarTD厅p|/egCثNg'|HfT'k]Y6Zd!znRxsK#_h;{oxϧ{ttAy;2 N8Ɉӑ1#+ j#1ԙzz>#H$&j]/3}`tԾqFdlaz!BWM`5: EQ@hZvhJm\/ed rēS{N&]5ц ^ph7KYyì(aeYMۃ]>Ğv @v߽(`Faq׳@`Y͎bd(2ȹ Cl: ap85m"R۾ubhD+P{F ߦKlt0%n^X&+1%l2ɲxeUI~cTQ p$^^;@CL}Q&\& ]s^o[cwWsD&54lhbQ8n ylBA.֨}l^$7^~Kh]#_n-{&̷?;w0mtflnuu`FGa:]ΝgeVMg5ؠY0"9XJdݼfms0 iJN@ u1:.7",pY.yLDDHzɣG08NR*3 /Hgeyd8J{w?Ԝcg9cle{ P:F.V=47UAk^M A2tȑ=~]ߙ8oo۶m/`kY*x_&C}cLo^Vf:G%IHehHnׯ=7P >5z33`d@kI @Qh9(_uWSRH&H.$CKţ"L^z=>ҰQv [D%!}qÄ,dY'4 B,UiNQHRށ|=>(.Zv"^zw &/kBCG\V+Uj9&jCf.iV`sOq#'[CD 0u-a_tcX;'؈i #<"H?5֞ΔF#D`S=-fZK҈`=m;L2Ү4̤6~pPz:]\cȏg$w}v!] ؑLgk&`pfML`c8lv#`9_]W@iyTYffB6} QR鄉>|͐tYz"G#qr ,A":eY!&x #:><φnj\" jVpȋUrX"V\hd(rl0y"GxgI _ox;:VN:7u~&&&d(N[-%B eV)@<{ $I׵0 Clz\{K\#~|8zd`K!jV<\1Q(8êc}%y#uƭe.f ^nр$h;Eאԃki$V3e gT1J>c% C&QV4BHmkcr"Lf=+=pKYR#~H$׎ր 0:yغ1L@;8}+.W 57TWX$nfzIMǨ/B#Zhhc ԩ2rDʆQD1,ZkV,eGRAyFo%zkSǏ:}0PՕ4IXEj7 g*_~0=1}{biYJȹ,I7W}zUYٰej"r 4Oj৆eȴ|c#"&M,G7# UߞxV2d.sK,FGR=1E#Dk:# m]$NeIܤæ<1W kH}*ꒇ?{\V) NJn^;c]B!7'&gjGAZ҂m2cT\#ViK!tquֳ|1 X;,3䃒d ˙( iy0}a񧁔G81!DAz/Oz?~g_s.2#A::x0OK֔ԡFgxuVݬE0wl}29fܽ- Vu+T9|>)3  ??)m=LP:TG L<G敇MNpcpZuKVr0M<7@k=&7I@"4Fm̃*AOYRup ˸"݅_rNT<| EP POꑻbɑ\\1j aL/#^Lo2{U1 ns{^,Qp@N_\ 97 W"uhhK{e;". o߫?ObӪ1>Pw,{q9$*<+z=ӂ^F=(aYd(W|O\d2#E hc[cNSsPhr D<0י9 ICs_Mз kO#3U'o?In]3wOvA2.5,1x9n|:ALTǃĘ$7e{mt;6: nӱCCkX^i_Qw~?yB@^8yqDgZ$lf+IB">΢CXRp xF&)R{t˕8ՖQMC "!G<bHu8e %/>c\2J<IBhǒ7>eUn.=-~%A L&)ܤA&pnR>U BDdRJGІΩlJZC`nCt2YSi8b09˚ N:<b)Q;LR!j{b'Q커AFasLj{<)7Bw`)A*G5~3^CJͻ>"á]- }i5POj􋾌j!$0)af̐Vsy^EvWsT(qZTl5Av?<:|Z:'Ϳ34>qpōgaxrߠioFJ  '凮;p(۸J{bH*Lx| UU{~~Ї>tjqlL.|U+ \{d: @.߬ժ#B)&( r9gGw*s GA`GJd+$ޕ}*1c#rFh\1x7}6z$TvSO6&xȳg*TMuAPzEͶEҟ᛾Ex?~79'WEs0MuT)CU)bK;Pf}c]q02$~޻RUI P 9T!9clphFS{Ĕao20il?Dz(Bܘbe3D|'. @cX^,0CQ8;ZLԪ71/r^ܒܛ9GM'0>@|h:ğ%r]sjs,dZ D<_<^G?Ї2,۠C;I~s76Ev眇2=?w?]]>}&{|wU}k/'١X:Czv+nTM-ՃXG,ݺRU` 3r F{1s[ۨg 꺒~9ÐecpGb0Y}' ȩ mJ 1%ChUUz q@626@Ε@KlL-T om=<_wasC?z+(Clj 4Bc6σwp^2Wg |";ZPqE4Z 7z]̂P fOsg1w8rN:QWU!jn儀Y O0@W@Dc. !+d@5-xx\%TRUEDA>h.$Ęg;2h^$]G~eVw^|꯿?1*앀ɓGLtnjg&35.~>yNid9zfgrYoK)7?O}ƴlooZG6 v!w~dR!k|`BwzprkeO<:Hz,Kxu4B]NO\,V'߲ޑ*uQt]v)q֞C2睱yøC*<8z;MJ5I{fʧS|//B=~W_pB]7~6XG6C5`@1TA`"o!fO:w }*PXe4hocdGf.})+uL I1_ c .TFb#v@^\!ɾTw%DLIqlfmc5ݧB³U@ uSkR%خ$m9}/4ՆF2If~f J2B4 J(oǣe^_{<߿'+{~9G9x tL$e5$Gvo@|no 0O=dΔ7/*Ъ@Q{H]DO:5cNiY|RBX"e+؃H},dm&)ZQ>Рp잳XL]Ýp|T5"'D5HCX`H2L:Od}<(['9K[Ox ^{>BX_G< 0Cl?i>Lx3F!*!x(?Bs~^%t)caӾ1FLl`ʒg0PUӺw"Ŋ0$f`$‡v&305w<*QP?+bߡķ`8x?7Xηrk7جr38ID I F, f"/! w;z.XlKᤨnD9GFtt"}fuz|.sd2cT ^}o[REk97}R뿃L\1ugBh?kX]wvCWB{Ĝp,%2I<B3glu1;V*vZGn[ jB+ y6<@Jl᲌ e?[4SOV|^*40+'9zV`|7|r8Wq)=/~PfU5T.4Jŵ݁q~ڐ\ѥ2&rnZ lmrHgI"( y2ۨҪ)&gwLArkRIL2Z:gO\a'䨋tq ^{ [x`S \<MafG]=uHFAcV(ꉙE#^ʹڞh]B 18: |W_?ƅbڻ}~'(%wQa6lbp]ĸ^J =~W!]Bp;>}vK*._?r!#;جg\d#:^$A;3_\%-pK < Dʨ]py9UͤAja0W; =(P sF3kзJh:̷v^L#"V Df}<~@|:9<`עNȵN& 0ik/E^'܉蜣l{stkP@|nK@?W^ֶ%=h߼bCHb`Q-S? @r&' Μ:06y۲p2`%yDv{i+tW mR4zͪ/@5Z.Dfd0%Y3dfQMe朳 w(#aWלUҹKȟ^{E~iG>M&pF,L%dƛDQ@G/@Ӡb Bs!x}*i(\Pc6vzmj[U58G9tҭU{7J„$p^t->$f^Njz4!U$ķI:@2&"jppPVz*^Û;hމ*L CD.{0~es |5r\H[ՀzC "23EA=c F&_rܭ=? 듳m m‰cad5(  ѸCC-!6v\+Romk ٱxrr.>vX,>LWn'3'9"$u"+ ]@vT aˎNXXV8s V`,nJLQ!*I樖mjdN+9aQJ)RFFil]aN+~'T/2_s-o~TU߬M'k3 a:7"J*N,js?9LbRUA5 T*+FhE !Vi-DqG*d]wc, e缠(<*j݅`s膖沏3hz&Ҷ͈`b"g#kE1 (#ŰAvGuѦ]P3'5mA 9T]NҪ`2Vr,0{Ƃ\N5e._um_9c!\~A?5!M̌ˠQʘKF>Ή0",bF.Wf9_i2'$$ #( Gߣmu9_zStD9+ɖiP8#PyϳJOO|Bk2&%/a ~K#{зrЁ+t'OK)7踉C<#=r#r  5Idn3>$Y ] ?9W|UWt΋.5+x˪]JPGS &.'T\W^-E9z3S/R p%(:#c2~7~i㟎/7|G6p͕WZfg3PUֈYZ8',V}ѣ6@=N씼,~! Uf=(; \EˍAP׸AZ ƳJ/XzFlx" .P8^s~ 3Bc:AVBtM 85< 4,β#{ "K Ȩ\\ЭV{kt2E۶ȩZR{{JЎJS5{ L?#pwKJH>_so{]%U^UŐKzH>7+S?{1naIN.cWrZܦP>ECL(#")N4p.`|Ine'cqM<' {/l9$.dn0Ɉ4WEn;,WKQK-!m"1e Ҥ4h$!(8TZ[93(֭i4e<3ӳ}+OGUJ|w 9C=#63T ou ).^wXg )4ȉd(a7"`%)Qxryi3&^O*ƒ&{ydrޢ[yY 2"f$; RRײ 0B B>)NҚ#qC$#&]+ $qPxپeʾ$K Pai=[u@R]M$rm8"!㑇gȿ;?}q9{/I&KVCum0CҒ:yY#j;ʶF>cύw]A;'3$b,KD),A|>ANŽ̪+6 AD]y;>v҅\qfְE& .WS"bvcI/=^dX h"7$0 xw?W>́SUzʯ׽]5 nsSfɏAsh:*!!P1nxl"˺Ĕ#%Oě:oyOamsMWfy¼$#A3YN+g14Jߘ95:\@,ˮ75ÃuUP!+ץe0lrrtMU.U/0RW5>}~2 8B "T -uΥ>F:VBZ `Da3#gA0lG$9=/NyE+$"\s;~'3&3i4'8N>ras>{}ԯ{4wmƿU:̙J{v+6G!u1cD2eTuU9s,07ċX,4h5,x8DՍ1Y >xTu^lVIr hC} dc{7+qM}ւq>ۥbx_կ{\]í t k#11b = iT\&HuVrYN佐Fڃா0v!1MIY=+H4IJv^ ؔӌЇ:e!ffmU*Џ{}.U>RU+j\'bΨu?ٗ=v]ʣc{}ZI+ e# ġ(Jb>92Ezk;%GYo9+LOs]cW77HYE}&i3)\ze_|}/+qW&u7 MҮA%1Єh[( j@1ay#1K[ȠAzpnt@o)S!+.>.CH)=Sc ;c!YUJ rQ/$ 3O CyBNI+DHViϊd oqGYoYG4`92d?&|Q౬qQj3|6<ߖ<x[+5ؗ֍ .T`oANQ rUG - .!z'7xIAC7/~F"$Z9n1ss0iKra8@q΁pשq,R2f= .3#AlF20rFIjɆwӺk>acJ "/_o>$gae`sO$PZl^%٤`Υ _S TrEH'p!n~^HIx(ӣ3T: ɍYB`lr opgF|b?ƛ4'h¦׆~zbmp 0s;{M coׁܶ)Ksq>>u@H e@pO@Hxj eH C2NxF+7on~e:33""( jҠS^dڠk .gLBOn#fA%-* jh IFM| Y y˭m IEet K%*@#l3'Ճf0y'5#iE :, d@VҽK+c\y5 .xwT( zs`@ʲ@Ib9>V*A_H6$2QXâ-#Lrաi*dE(\p0 eSl~>uӠk;L Uɐy1'x,REx7 29cs gg"AZeͧF2բLSK`spŪcL:XG}]uH3xVH@ʡmHz򙚤KaqGSy?]+el|A_|/<“G*m~K(?Һ'vUN$h%eӍ,.>|?WՉ_|~ܶ#w8aYLgPK@12ZhPmWAݪ+c=Ds­lФl̨ٝ }阐-6҇CS 8ѓc'ՔbY\"T˝%\1PWx {b2;;QK /y^\] \# o47pUC@$'JvBȀweToGd$Dh6)rb7K(ùQҏ%x#F{QF]} ja#u4"Bv(l2`81[pf$/ri59k/~x!!`9D v=W>BfE i*tmWzm(jʂSn ]h;;K AT Ɉ_~9t{{\8$O߅7M7ѝ#4¢[“pPuq]>޿޻z6 >]^,dܫx%2g)!XƆ澍Y>U^B]G RbUmYl#)aGkOfʺF"}oLf4۵I2WuJt+aJxX߽:t.u%fWVť~H^Ԡ[=!N66X` qHʪj?yʠDʩ{fױFyu9 *'6*Lc`"hj#~꧞oaҜlߖ~:/3G?;;\a]dECn=W%~_9VegҠ;;wySQ|C߻!xo,S剣Q󍪍LD;r.E9!"Jo=GDd)K=/dme,b#w O%TD\22!PFV]0sD#cZ"RH}7Яڽo>R+xɟ506ov- L&@SB|v!qWr\fmnu+S;SBTZ)!ǬbnF"2 ]ܜ >H_S=YW]=__WUf~*mNf0@P6b7# zl΋~~IlxFP H_p ]7킽Xk-(w?~f@`Jd&BW$ʈC#-vBgYjtFTG"|n+Tӧo%q B*3`$ Lea9Jg%9s"2b4"̤b0ٔ"dm: T-ⓝS<h AsmBo@ Q]ףkkxgd K`pd~[;^0HD e7Vǧj$3fIDe@µ @J '{MMcfF͌6BF ]o?}: :Uޚ@T3K< tw.}@DzOEpŭe/Ư9/)ɔHL22F$V9J_[$9a@vhV$])ʵ/p>N; _!HaD¦\|nD}'a&b7x3xĖ>ů-OS;k#[O`Z LrϬڥ&!FĵCvwf_L6ݓ*mfщM\/qnϪڎ?~A2&/赍u,@X+TI`GcFvk8C).F @rȔqT^#5S<ߌy7F>gFRToJ ԫ=)bXbG icb#ʊf#Aq9LgZbVR!Q;" UbeF%VVYf N%a KsEi ʁ;eF00AfR-y>K9;FNﰜZ l#NpZdK{Bu]admSDB@yZ @r/:bӞ3ѳ Z{BUup#,zrq%GA¥db21Gӗ ]Jۿor|WRlO0ކ~*eĐjxҖk>')n?{Wh%]mрLgV~>vѫmnaBBG=("'L< 1#32<2exBrAnv@˄TzPffLm Pw}(Y{;ެW)&G`L"{΅t|*|]>B ZMsox*8HI;)Ep{?"+)9 v9m[3~߼~=׊o?IA1[lŕ9CDz*hSŭL\zҵ ~gfYsAâ]EVFX-刞#(bg {?C򈳂__OOj\[GX߄3!MM5P/\I q,(T4,EK"J vFGC.*VfʏJ2rzޫ|#4KI ;|]#To" >MJ$Jbףu 1H'"G ox[-( DXM0P"*ߋS%QqJq I1 RY,#"8v/9#. dTU AUwŅJ {%ahKjYm B+ ^*BI{-ȗD\+z`e$mO?+>m/xsd@eMLc'N>K=%B+ s[.}jy~|":ހzG-. yg`68Io;!OOR);]!! &ai32B2PFLJA]r:R 4mW|~C0\uTڳۧb!?}b a}MY h2WAt0jkuI>P/BJ*8u+["jzi3oziH{KFd Z(iE}-əbdvPRDNTl $(,/AXO/Sү&"I K4+NC~pϻ_pex3B1PSw4 ~14 2gL3BdfҢX)*n oRXIxVfN+nN"BQfϑzӣ|JHHJr NeHiyPnZ6^ڹ*5rAIq ˖p5e6%WAp#$)jw׽/~m_SO>;z)㩬Y0iK98HrX·9/Q6!7k1& ngoZ` :_ vA136o"刞ԈB+*@'^]p.!9ř5rf4K,ս6@aN3v\CA,Yu6EbYB]O ڡ}Ņs=~z;[7;+;uQpSuLAA]S I r=RxyD~TjFT(%P&[kit ,,<5"ryUDH #x`"x%v9c'cYj'd't>kߗ yd.#yE+9+,MB\C4hz[.nIg!u;"#Βu1U8ut»4a'֘Fi?p߼OG>eE }9_?y␨5bzoL$ZٔD&UGt9&ԾOAGU{Dk1 9A3QRmki} ؇b`qڀ}헼i|@Tl*P5|'1F2vQ]39Q$B{8B/{ ]?b@kD&>3Z82/A^ ]A(%*6]h_eZ;tQRPZ*+;,^,UѥH<4twp䇔lIe^Npykպ1"kc,p?U{":TG%PF,)5Ca2JNӵ ;KP%Fʇ1vxN@ܪ[N/7PD6uY @*h]/Eco_H_7$GѸ_ RB9t.0̢2aTU %Iz O{.^zQۮO?zQ<4~"yGQ=\)Gw=W\kz;MĐC_#un{u݉Ͻg_{m}c} x-w@@$X2;aG\">p39(( 1KEȧ=yð~رRz;;&y *uI-/\Wb=@":< nh,HUpѕsAۅ>\Wވ4_U5 a:EM' skU] 8Q~Е̒;%2{*HKLWĕKNޥ@Rw o,DAS%T -hD!&!`Ѹ_q;/!sBH˕u%BCUUx<G`ms7w|_vwxqg0i$Ϧ7n6N ,v$Go\X* R)(gvR WLjd =HeM ݰMo"1y=  ֤>I+|p*鬵PW3,HYHX$i= Σ;-crv /xW|>L8}BDJx@l@A\;9Udҡ uKe5>ت=xŭztSDm_f|XWNUaWkͪu|P ۷ dU&ެMKqHKA]\gt)!26w 8vȨs(NhߊGbumb >%" Ѣ&BBbCCLB3glE=S%pO"kG,'BEid{y꺨 09KzrDҬnͨ0ZT_y5p?|-ÒQ쀺 h۬de(' wE3Xfѱ ;Fvnl~*ߣ쀈@W QzԀ9 M'!R*}]-\O\ { Y9zݣ$C[J5،UUxb7$#T][zp{F׀jt%?o‹y~n/ܿ- VZ$ۧO,#O=uW|npt 'OLz[ %T#IC=aO}$ڡ% _`^-3sP5ݤ+S?u%UΔ{e h; +'!8yq RWW2hm}YD%' %\F-QCףT"CgK²M@t"IFԵC>}1pH̒DV IQS ڵ,5p ysmf m%kw*ϣNh]C/p端8T@̌Ov$VxqA-N8v8xǫz^8e6g>w/}9_I}:+g}l&xv0u/HQJŧXi D,xz&8xL'3<|ðqD*ث6nMob~<}5۠rܦS'"SM'zwCTO!FZnQgK_HPÇ̧> \Fפn]k~7KWBv[@8tT#:!b1)S8:USVQJ`x!u~yC[U8F _ ;1L HX 2jWמR1#!L$w%) eP{¢rx"w>Lh;`\{U[ys'c@lO0 NEF%%Xur OA9@EQP\}2]N2cՉ-ME ڏBޒqh06o )`2Tr  yRe8|ő$T|+ZZ d*鏻BLJ- ̗qJUQ R$;ҥrY!j€V],d uQw%o5z!A0p]P75b8 }o#.xY:ӜC UZlvϱm6?*+[|ԭQ]ׄV$ AU U/{||g'8'䔰SRZagjC-X%P U7AE3Vǽo"2ǛH 2Cu#CuMO181q95@@Nf O t H-EnTë<諿RC}po{pr)?D"jvWoͩ,-;{t[r AT}ߟp./؇q) M$d$&9#QTmN g< ~w݁| \|}k?{KjNoU4JL`EՉ#Qət[l>E:X}zeg"$11N̻&`x3Wyt1$S' ˖ 'paqZegZY󲮊! &TAH: 8A [rrb g lv\1Kڒ3`!a;]&8JP7'>I Oٴr՗`]Me:'z{Ep>7d-F$.v$іlsBլ#ho-7G1kɏc4H m b`m|.Kay: K^rK~ pG @_5R]o}bA{?Ǡ] )+ca@Q1|l xM2]SCn:V]ם  gWì_sơ ` [s,BF*k 1bg=ᣇwq0HR_~XNt*~rWWp7hF^=, J"DY3bear~>uN.l#+o1⦽rIycedgw̬LW!l@P6z q5]<ŽI Rj]I:ʖ#>  *aWwv0[_b=L*H%h_=[ጰygԳdžMQ#YEvͷgnxmQ;]3=gGIL g*U:}(Mm0KP6ݿ.?ؕj? =H5QNz҆y3R/Ng]΢C N WdIcmwA%(\AV p D-8Iw**#""&-uS+jf^~>rLRBLxY*PHX,``nkH(B*709!uK9<0U6 U-w#vR9?6~|(;'u- wH[53P2^FuCUMM 5{D4۹jzz ֶ^4N\G~{j E{9GMH#2cNQΒ1/>^ZGϻx@5&0\$1rmJm۳q{ qc!|Fv~Ӟ{gUt}'>~'\qtXok} '@>ͦd0]G΀kjPnsբى}YY!b ȃO2ym!J)]g9)BL3ThOs.dYךHcݮ4e[5!@S.S“ѶȪCUx瘷Ϋ"N8g;k\OY|ca;o iYt13Voڮ>IsN. vK3c[2W'3Ro<7on": y6MAZ|r o/k̀c[ WEJ6s.>Y0JnDFM[n|K%V[6LjebӰ>&93*(!Q5W(!*[O:bֶE g^# 3RILX(!%ˈڛW5a- ,>Z N3pxHXCN p(oʁn7]"-(ᒼr-ܤ㟮%(Xxľ|K\7~pΗI'NZyds6ºc{f?Egv9V9dsU"p9ݕ9rTJTQz|pIx '+r'"ϼ/0?~T~ҀP›P]! {  LQ B ӈI@9ZeBPfh2 H5233rvwD!*Ljb1~l(XاjU&Ƞ젳]jK+bМ}V$a1_S0 vʨf,V-|ƫy#&U\$"%j  <|ʯ%;WSK/0X^] ;Ȉyrإj4vN|R>7'XPV0I ʆ7#Yp;'`2p(jBXꚖThTe-*k"-9܁kƭϵ6Kg_~>ӧw1eN5s?r9uEC[ur I# o awo~}9g4)h45}*Oѫ?HE9vRnө2'7ЩZF AK#a(zɤjF3;-Nq^P-O6^enAR .I$K*+mfVY㱳Fkt#e]^Yu`m2rI:8Qp)Ѯ*XY2!rymùDF{8pFԕ׆ â3eR#Y>nAUr &TX9t}D mX8O[JnD0%b_x@.6V[7gU&Զ[I sDNCrI u -R4(=gόӧ k$(!k]z]H>`~1lIDAT^9=4ox}IQҰlVB"a]n,nÍ7x|L/y ?gq70qO}<vRTo~Mxs̏?U ; S[l@< [uЮ2EęU ;{JZQMgKdrFuVųyFe3j}z䱝Eo j7$QmTXҷcQeI@HFҼ 5oC9akE$0XӨɤ3Tt!!6u즐oE$ʡYGGh\m]p3 B!qW?b{ z'm38eqǁ\*|]#v-w ,K8hX0cWsHd8&p5&PFW q! &0bJ p#z g7$ol N9C&-gHM:q?tŕ:q\,l̂yֶ<4uplu-6ob{[ 4& mss} 6֬{Tr{B~vvZџ] עtw?`.~k3 {na4Ml(}\+=X58̧qO}"Do6~~/i:4N !B=yDf1pbVb".D)9nћoدSHl6D76: |n_KY՝B] Tu%wJ*y:R?t+~wUr2B"] 0 L;BbI _gʫZ&c9hvR9uoD-]PI6h'gu מx23P9/ؿ*3\I`K1:vvW`SDƖ3Vcr \ \eW'jpP&`e _5#(&*.h@y"'uD:S;'Azl9Jxrfn-Н8#Lfkgfq%--dQQ,8?Wϋ^sn\sk_g8u 9JQ6m#kkkRpٟ(7 ~qQv9< ykNcJEv5%QJܪس B|!M<_[[o­: 7YN |UE F*4MqZoԕFU MY"Ԡ%ihy!:-s'DU%|c- pQ0LX{s %? I/ݘ$ش1* v-ʜLj`o:Ϛ88Oh 9!ȩy )j"iY%f%+1ZC Y63Vm'LlEZ9h_^:$.'5yPa B5!UT)rl+!573m) $̓R8j~@SW ^9(>,gpʥ=B;#-9o;[C.z&A)ka@@֝[&Ĕ1&FD!*y/,H#HH$d'}D"^ D0e+QL5&rNHxs\ ,@Iv>7y1qa9GpOK{EĪTM% Fs:fֹ*CF9y'4&Zg nߩ0 _>+zrbҷ.8>x 'T{0x2DkP"xx$$I^R J2Ґefu|" GDD2O~I8řr[! W(3Rdd9!/E[(kXv(['EzG5΁T!x0bzY}DߵOنא㪴,Dp.cwS42zj# )G9sBAqƜ/"_BPn%y\xvSq%zS5tݺ~G-k?zy~[^}^ s=Ķ_|]ם~*>~w;y{b6)#x ^7,iH'p)\#vU T5PpA#;,e]s&lf N, \}i &f2ܸ:*@h*/22; ԵCl& V!Ĺ)!''a530~,l"mw YB'1*LfĔ}(]I.rfT@$'0Kp0Xx : $GxsBpy_f$ S@8(kAfs)xYpIcz Tke@拭-IMIH3a;KQv^Y*t6@ 5ﴗG(Ș9WKe#dtZIRhlzЀ D;V R)Yb*ܖ<piIJ]U䘵/=W~7gnX+FRf'>Gۧ.>xϳTUU`~n|:::G~w}~W'>y3<d*꾙+{a*4b+sʯvXKI*?\2D:*q#?gsOȡi*tf BgL䑝j|?V(*F1Q!xQ˜ iƌW`NWac:;>q)lb ;I@g2BקhsTΎJ<GNSJąx=3 =)Q3r8@Ӝ+@^Lrl?/> !19ˤX.;sƄz:cҙ猍` k_^r1-G1?uL@LзBhll"Jt hqOwK;'qC4(Dv%AH0'W br!ZS5ؚ=$+J0IYV{jN)99#eSϰڽDr'Ok2I@?ïKn]L"p}{|ZM&8+_wu=q`E~?3'>~%5P5pDS* H9b'D98 CtXyF'V9OJ xw֐&egˮ[׉9JQ4ԥ$o9ov,=G&GEM'-}maVh".G&xGnTA+\q `6ނ͐so o(q?H̔Qky1!s 1B0PnPSI(&&Yɣr_Qߙ%sN:CNc#ܗ*x =? rG!wFN=[5h9A\[p,D8#t% 8Ge΋- XnIZ؈T>vm׶0yvqqJ݉?;(Vg#AFO:%״wl!N]0`g֭g%Bd瑻  o~r+U鶎o#"<яƋ^"VarwzF٫R=~|lk@mo|\.ollW}ѾY oj'%8OHM#z]YU*=g*arRF&c6V)OZτV:pÌIvQF"g ,n.s+EG3 *:>PY'ӗCЊiR,{cT=i]༘0҃sg0?^TA]Q5.)g.Օ*b_ΐ }jN/xI9ԷF4^gz֦hNbֱ4Mˉv$)UzUTZ͛p%9 km(ɜ?I *.[Kv>y?/M7S05ST>H5\2D*`EcE#hu+9ˢNL&aW;&64ct۾$1" Z~qo.s{KS"TcMjy9hKR-kfenKJc̨*طIWc" ̡]E%țV*J$c= %6( yiw` 99 } uR' 9p%Auߙ_-smp DFq{ 41I=F䘑&yKGeMf"ַ p6R7Tyߺе-ڝ3w8y 5uN v-di#"拹|zDÇ6p~gPX휒hߝ(7!N_z}x;DZ3G\S rFE.WDCZdp)`?rㅿ#gW^:\7M]W vwIÎzuwzPp |]J|;7_wu=]L gGOO *Jʴ@U'MI~\F7ӊ+{ĒTEn@3bb-dT!UUt(ptɫXH>iౙZ"x/ĩl<JB!qL&YSuVZxHF@\`yy\~k5gƪ%ɰ9bQ9ij>%xM ~mv}:H[whs*羮uR9BVpDh)Ve ܆.0l=%8=w2F9W5{dƽgqj[]1~]ICTݎK5 ~2? K]L`Ѓ}C8~Fm@O?Gycށ?0eP۷2|78Kmu{ammmg@>n+B__sO7PZ?8&'OF}xȾtKIdr AG= v2iOЈǼM*g3mä4ZnK9i+(pU6~W_O|2~VI횉7M  ?NS%Jؙ2V2hO918r+Zȱh t22{ܰ ,AJԖ}TQXl0z6`dQcV[+1eTsb ,a&5Vm>*.9Z&r.,B:ce0ijM|TuXs].&3*!Jl~@Zg-Y$q>|>DAȬBq_ )!$> ;<&,j̹jZ*K9[Ir15c/ױΗ;Nk1LQ^4+1%I0m7ͻ` ʾfù PSd B2U8XdndJgΨVU9z9a V#sv|H*![ %"!X1Ю跏ᡏ~%O˺8ޮJ<K_ҲO awRu3]n}Hć>eeau{a}}7>=Bo?cM!Ly]z } >]NmM:.w;_P~}{?S=Ə_#j ׳1MPS/$$YtI<qZj-xdVޫ)4Oc:TT$!ՕU| @%3KH֧2Ԙ.#a!a6'n=&T16hjkeIPz^,IbLCMg:)]$8R;3B#IϜ ˑ)>Ҹ$o 3KLi5uT-Y*kd)HHm 4 [T}Td , 'K9yZ3| w@JV|X~vof?<#g_'eI@.iuo]B. [zY$G,nAzSͩ#w~ Vg"mfJAVr V `6l5K.MI (G+<Pi*"s]΃ݨ?Zv BHzxmN, x@ L}YxO$h) bY(Ho$=UHSE {>P 9:6-&A;, ` !xC_4qtCK$#[mě>| !H4-q͓gZWj#rfL*!",IT`vX'c+.p?wd" v"jf i kⓚy3gP sFZ-Al\c%xubq 3`I9RPoOɉ9HJj^zFMӠZ̀U,;`qaviuZ,!T2[[[Xa4@zQl=q{\/Ͻp;W^y(w/? t%>xrwqfFu躮ʅ3(Lf0"U5X0~E;@K FHǕ1_nGdyXJ[UX-!TZ{EfXwݤ~LNu5Vì IuC\:\pfZό}&? L$=':gK"*ztuU#Ƅ\%]ApA8@d:j*ڄ@|.w< J^l5Y`l7/lq"VB8_[՟Quja~$r٪դdTlPiq\R2 In)Ԃ8:}ح;VcrӵM,z6{ؗe2U kW۾EUS#u-j d:.N^]k:% rssrApx޸w]r}׼.} lvm@q5{YSF۶X[[+[ܶ[1@?' 9Røkl&orH=nF/ns@/[]qkfQ_n ܾ5o^g>G&SP8Z]#+s 7г2+Nf1 τYx[hӛs:!wsyʘNHY6sk̄4l 3;Ԃ !!Sy]@yg&P.fUdDUE(t)\}W%;eVkIen{E`EItbỤ2?060袐2ڮ QIzĘ۫^KB]y,Vv-T*9b_CեUNq?r. fqfPbnd#SDRQ,>tlSIU܄t1d<L؄)0\υE(_#Vlk;޵[W|&)%(A'e GTUkA,$LWMRB@{r==K䳯o&<_;WSݕUnS3N1Ljrj*Ou\.ZJpfeM7~7vq(7$D#v/X__1BǁtA[??~8"ef"hn:W "R^+,?ET݆pʡrWB0y!1S2MRK)6Ty`>E6j"Ai2l OtD*6q! b4^E؉kuNWkU5ơ+!`V$#EI>u;ʣK_<.y}1Etxd[U!]פ^°CbuCbd.{+8'=jKpSƩS}.w%A!mnw|<4M=,[ !/TkA0L NrDqW5>:Oi=axK^r:␠}yPovYw;L~{XnER޶~W_{% 0%fJoA^eZ*7}sfEy6xyid块>Y 2[Mư9T$ba륌D=ϺލOpF?'?pR .l6Ų_/B+"20C,*ges;ޡ#vg< ;QYpcѹ)x\No]dDi͟\$hca:0# ǐy+3#[^ϑ1eHN ~e.%a]R o&j;Yj^vPP 5`P%+iIn&c? =$R2|UՠoWAg0 78愶>^*BVb ^"}/;0C ljEjb{kcT=;˾{p{\wVw4 iZژEwOmZv8¬ ?5ǙeIk6g;8Ǎwp5&n6Lp5@ $Ƕ13%%u][]~}ߟv{.u]s 2ǁ=5oCx?.T*ZShZ֩s旌]Lձ`JLdTKψuЧ O1eG$B'< BЅk6u2XfRU]du\b=Ln!8]\iT&HࠤR񬺨iV*Q#`x !eieQcn;QU9x ٢,=ZߗrSؐa}~|.?7wmиU!)'fgɡn@ϓ dYa+JmFz%p`3P IEAʡ:MP[<=gNg.$8u FL!'_-6`>߁!]Cd'S }6F)4T%K[HnE=q|m< /f6#P|}qswKollrAkbn^Ƕq8jj} ,uU Z9QF33^=AYjCxt SޡnBZE;&cX}m >ˬ|)k PUL7fRڤƪ6TT 2C3fy/KpaĜPTz6!rȬG*Jk!C]xr ὐlG7;Qs :u=EϜ9/VhDE ^FvYieU ?otWRgu,c˖qcZs! \p3 #L|2q 1K6eI,Y}S}i^kͼ?Zԩd${ks>!ƌ:.z9\ ,b09`bmh%5cAHo3@b |pEf \nq_YYQc(xy$0njmv z, .~@X=OcƢÞ'NRck/xxϾ Mdgn{ =ilp|}qT$S ;6mLHBD,5l~x7.p{8:-ٶ-N8!܅scO?0>?5ٗ﹮밺?8Bu>M$Wn~U^Ȁ~wh3ι+/a ɝzFOD<7]{!\lU 'q4 PXxi |QH}@ƥDŘ`Pf  '_nT*n ^Yj\uJ@) [rIr^ ӓ8oEUnmPq zhgI´wjPM|@N]BBc@ ﳜе ĈQAh[Ľsjp0-( OI Ȉ8!=uK 0/BXn1!d<$p?϶ T\ sR`!I%O&K ޅܜ)7쎲63_YrgVZ=/Ǐ[p§ى>Lcx| |ѯK}kC뮻o4x 2m3a2y+#i˨rh;x6qfmag6 J5!K!-嗌!AXLx5׷? oyCi;U$AIX=)n\W)5 ư BֲT%c`S^+8@'=SC2Z#=ffGBX']AspȄ.oybpcsJޔj)zj[ - S"+=YFyo.ڃs1ba!lzB5ȦZ7b%9smrƏJq5K {lm QhaHd9v\N";b?68mm;^?C4{ã[:/_ckS :q7רF5eyN-Q1!eH{ɘOS$5쮱{F]Tɩ{bCpݡݸJy/ZzPK.CpIvmښ0Tv_j>&wW|l'+0NWqU^[Db5:6=WT$!;=FRNA*W S0u=GtO3*l]Hxӈi tē5{Xoj# F[Uu` j{*tm[\[dD0(EmBQtUUa)c8bTYݯ X=^p\g /0W0oQ!̙3{{5Nv{ыO?}d2=n֟v~_ؿ{>`r(p7YYJtUg0Nz{nvc 1k F#˨۴jVJ/XUhIǴT3񨆳YK`4a?iNIF60"N잕MQ X _FGӅS}OOu R}]Ǟ"ғx+ x3e8E c-Q=rV7I +祮BB,5+Bʲbq~bq^qJ-gA_2@|0 BHEsTF~,SF8Af[3f7Oc{ ySo!jnQfѡp|N}BHrQF)&:ލgNsmgQ5b"Д cct5mmCb+KɐS>6=RM_hf[ػgoW}KLpbcs_?s?9Y{[8ђE9.,;`~5P,rS"?grۃS CO7/..窪5 *=?зӟ,Vg+XsyW̬Uh+X0]1p8eHG b̠_+!,-նt|p2 *[xahC, mIU,uސdW6,c2rb3Ey&Yӓѹg郹 QzҙyoR`*<8:HXE 1.ɇVܓl4̄'99$sPV%I+Fu3\z1Mz~w<ar} T#Ĥ-i)vmu))h1NdRHkS`]8 u cNz`\lS3} TBd}PTB\1dXVr8=9')!K{a)N8 ߃I5[K}HrGX8 QIY~5O/¿āa0ת\Vh~SF|t G#x&^+^c*/#bsV2qQjFuiaYهrf5ζ mSc(TLrJRTK ]_Q("aXYr8,cn8/x?!?~c||:~}wk> =,n gewX~pn֏/ܝ~/ß~&uՈ3;"J"zy_Z ccL@A56b3`CU*bzrĆWz ;fatÕf8<ˈ7M96`2Dc\OHfV>fjuΡ5Å'~W&9 H[Tx7| $vD?Kq/ũSgK$F'%& l7xIvl2rYI@Lb*KO =p%PyW6~ dOʢ%TlJL٭JĿtN hgd(O #vԉy#_'2\!VLXyqV\c3ܞBhc-~yWhY|ْ r*'[ڦq*s]0(ոB9732?C=?X'7o鞔q'?kӹWM;<~b VZyblQn R5k/Ni ٿ2Ƒat h=3ťG84§YŴl?/B8<6Ce'MxWu2cz7#Im[8 IXW^zkn0.ٽ8mq9|Mg>_0պ"dVsm[t]wb>GS@_GD}㖏|6uV<+T#TV}.D60 gb!60lB%PW8+Ʃ W\sJBAIreމ%WZ8J;>$ssq\j5ؓB SlVXAWyXg|tW夒/._ׅ+@@¾1.Ps'nA昰~zEWYD id?S%u*Xӯ!B@$*#($iZ#`0'm .țf>92bU1a4uĈZ&ا(i U ۏ IXd'9`D׌F;IM :DqzctL@ F#q{BwG# q@ L[|H"e$JoY9y=E/14ݟ*|V?C藎͍om8~jnpUŤVX=U ^,]P9qվ8{=\ W[Br3F~>YY Wmv:xkpf+o; g6[|⾓x&Y`]_2:nn 2(qAZ+}{vylk}|~Y#?sW۶L?y{7~c ϩ/!@>vh׮]ꈾF٬BCwطC;}G> a{yX_^®eL׷XtϩȱMTDϸCQ3 M IY`X7euIoϢCr<=Fc_Zju~ڗ$9UI[iW*Lj3\HQc飫&q{ǂ;":n,)vSѧ` Xb䠨\NBY#kcP9E-q6)oDbO=t^E>Xq7c£$c #YP*ss٩GYtz8ɉZTD9z,s #Lx: -;dD E0@C QW|Um 0|M@0[؍8](|0 u~4RǪ#\$HZa3˴/T;&|w x|?;Qbƣȩ$Zo sЕ.N"`iq+ň"n8/݃\-oqG=J@ 蚅1M8UUF;RDxEl; gFf PxĔER KY;|"xNy~쒈u5kljpk>_{pಽK{]ĉ;ӧ|˷5o'ɢCtc??O.t*VAm*}{'B7c . ?E8 3ј揎p#d/Q{`Al%CMP4 [''&x$zI6,WZZ%sVyATj1`GU05P?V V \T̑TQ{!k>g*(R eVLIAL#S Vgލ\'#C yU%%kk@!3i*')N_.pݛi@2OA599,Ό7* DqKH9m)%3U5~/$d4>g&Mx<ڦJ44@l^"ŎM,Jg=& UT;kXXB8OJN}0V$z!Y: g~GN,p- cTޣX_]G7!d\~_oط4_{yROӧwk[?ӝwީ0Z^|0>4c̻oы^{3melOf(>?6VY٭}-UjOF ],b_~)=&eALJ9gQ08/^{FoR YN o: Iފ(Q-wdz9XóbU1QsÊޖ@] w1<;B5_r.'r@#pceʙpbź[F.趴91.ϻr?)/ )Fd (fRC,CC@|/G 0m>SݢF8g1a-0j0[BLyTR G.СE++k϶:u gΜmRN?}8TuO~_,C___qK3 777W;UO%s~p8~&:Aj6zjLˢP%!sS`dV0]Usα5&Ή4KpNZWG6vHcHX/нaV8|Psybv2[`G"+sYwH[9Mϩқ4x5Pϲqr!у"ֺ2cJ%:u%,Gh)fH~z8SȜTl& OHqS2l9q2˂ZX xƔHOXҀ\b<9GQ+iV]*S W ѝ u|Mcr[Fp@0 ÁEl(d BZ? N+i8F c406q4I60/F8rΞxBa )^Hqң˅>hu]x^^J{l{0x'7JB݅۽1fMC߷2Đ'%ik}l^pBj)s;\|#V J@ʋ.S$AJSI=2^Y.fr)|x u]`1lʂHḇImqUg<x̄:t]??wj_Bպ|o||ɚ_ #9a? E_?7w= #"PY W2*0RD_ҫH eajKŖT֏*Qͅ[0c%b汵ʩ\3 ڒd\!cJ,Q> Pn4b0|P%nL9I=T)Hc #fUM$3 l-D""C7D8_B(O$c9dI*thm<@ ]rhSBVLLsB'l{C:brD)tw3" >$ A~D-/Gҗp>y]` SuyO?t []?`5P//1W9Ƃ\@(9{CxuRY`!)7ky_* `vrIRm`Y~*˛TB⡐χhALI{0xUpE˸u|u<~v&`)?EnAmawGG\vR93xeߏb2u׿{B̙ cWۖy< 7t.U~ܜp;x׏7;YL&%.V퇕 T݌E0EY`e*x W躮TBE[U$2Y``<߅g7EaϬ#>4Wpgqg*u~Xw:ԢZFZ5j LFG]BjBc7DY=Yn55zWwq1MYe{ 22 ]} N\ED9E=,W^߽P^tnt_wώ ^ZnRr4- w]ܟɟ"1X[[cӂۃN2?-{gxf4fA0/-֜!NEL7ISfA y1B !p-=f=ˬʹ5SYf "maLq5-ՙ%A~4eg>~'K xx3A[ow v޾WԿf5\oYb/B ä`?~تF3"$+_rv)ݵѨF*k 5paRRlanZ;@s\Fڎ_a52 [S8[qO$vyZl%;"yPN L,9c։kHېԤYzl!T1`m49S)pK(6aϷrJF hԓ{p+ |^rh҃HIkEңI(T'dnou8ffbW}p;ŧ>5G@HBZcc>$Uh2C̈́&-Xz@X r!52F.D b=J'Hw/k{uׂ*!rf Uj XV!B'o/951A|ej;k\]}+32Zxٕ{kacKpVk8˹Z5U;-Mlp 724T\!ޮk$Ip.)Vd*Ky.BL8{o}e1c}}=@N3άNq'57^a Oߏ&na4ᢋ.YdO&__N"8<]:o>|Էg2SzU;}Rjr9>+C@m﹐6PC~1R3D<[m0Eh3,< , 1G$fy:OHWW;T4ED pE&QT")A* J g:&֤ #N\+UuQ@ 8PfaBKfWK9'f !Oڣk.ȉwB:E~5Hd\i"DEoHRVa1|8ylCI6u4G\~})'LG+\)gf$eX_ۚEc;E p(_g̉< n+5i)Îƈ&t :Lf[2a—s95&0ȰfIDwavB̎Ӡ^~d6Jj6a Ǐկzqk׮3bo}K=^ h!{q]/|7W9Kً2Jᆓ(VׅJ71'd8o%۠8 Zp֟ F>a ҳGxMGphWA\Frr`3⳧NY/ +ŝvQKߡC_C?C|!x࿓d*t7w73O?{AU%*wN5Au][zxH![g=b"ʣIZ4gyU؂ q&9[nG蟖`+++XXX(콟sQf9sdFBK&,guШ0IOsm!%(*_'G@(NM9FzE!wZ* N(V" F[ZRŒB"q>%´ n\&."Eg0M>?rLoΕXyu⡇={׾w_r4t CGPOI־Fysqm=X]Y{+PY^l1u+ ]QЀ<4%HMm5iK6LAݳ #bR޿Z!I%8ja,u<4r@j($o6gstڍQ. i ^o2B]{#wP` ޱT z!7Qf# iy e6瘐B)sK"elm cX `%(r;]䬐^.7:)[$gVs7*@;AmH)2)-̐/Q*]rly.O8BY Lɒ(ȼ*rt#rhY&'PU-Nhj`sfRQOi#-$-گ{.Z"8KOmrd|PewQo^ k)˒X.r>2+;j26o:WA+96DLF/bYbaT *eJhpG>Xls~ /a*h{7߽-o@ ON J4 Oqbr.ޝ~dN{~<7pg8ɡsyI-5DϘL}pM}1{,TBt3?$xi5Nd0w=0;! |SAL9b?o3RCxs`b"Xm0m\  yFދ˛JHSVr P0qJ i)GG5X.3N )WXX9nx6 qKB]=2Dn{HT^mIg'Ml,LEj axKi[ dUWE弛IH#R!oz˷f] 뿹~™6Yˈ`C$#qiNT[yk1 .ӊ R\H`j"FUzh(:C{tqt M?uAO%Q2ڀg=^+p`τ;G0C3w Nx2#í(uXYY9'f /yK OT!W{b<1fSo iKCzcg T$Ոoz1EC\ĚIP E$2A!8ZH מu,\x\`T׈]"krkLbHReD(.*Px ]5W$:V`ؽxo@Z&1`Pua]'Kre}o8%l  d,D\ L&StGF 9АR7 ɹ޽ſ Ae|hH@ȉeH壵طT//b/b&'ekm !a^eKy sUPuiĶ$Nds仹$#@(SEq1gxA|nĥy1u, <u67pGn%.{Ζg-žU2[樓i[ 22eT08^K'.u%4@5e<#Y]Lc!rT\5^✐rxq7Kk - vU_Af88 sVη k(o @}l1/0㘭@L* Y7D3ɹfd#* Ю]+x QqWx]v?/}& QZvZˆTwXTx/@F$$ZQ|¼jV%0zL% #H$aЦ9Gr07}bBeckj (_ EhߠϺWZ !9f/|l]+qʄ=x,Ykԝs#GϿu2{oݯT&Ƹo w*p tp{`дm޹U=݃~VUfm e!mHHe\dAmveȦUF`1T*A[#A: \m2bA=T硉̟]H!h0ԛ*v&9=l[RY/'uTʊrŧ9WܼЕx\ =!A.)_$LCޙƕ/&l"iK$sH8YcQ7;\׊r,tVؐ-BW0jYvY\约`xC,P(WC-r5 +{YplsExIѬGj7Ap J7r&k#lz{D&юe$FH-FE&`4PAaɛr89_gq4ؐeZvU]v7c?/>ŽƘRcpuYVq8r@ .3w|O۷6Q_y o=LxA?84X*4?<΅I󘩴oo)IwI}}MƳ\bu]ם)Mp`}C&9`ye`zx27fOއ'UU{+V] K^7}7]y};^עJ o^9/?~|9w .@4td>ŠNF1- w JJ*=>2Cmb,:Ox[.]zrPUV#lVq5c@X\# Wk5Bb@ !4]dgn!Rɘn zX4$m=)AI܋1X^^.cSo[Mg:!򰕇V}dqE4{_O)\LT'AMuh .5Zk'BÊ^D)̉H]IDAT<}%w^SW8eMA r}{%YR*'=x3/M]*8p`|[_|+^q}@SB JI@\? gϑ˦GKV=Z*k{AQm+Ƚ4%`2ąbI,ǘSDSNL9fĮCuL3᭤-YP_<ѨYT{,)˦ >MVK{J.W%i"NtIyXc:F'jU,$TRPV)ʲJJi< J]*L t;Koa̪m3xMB6Y- "2HmkVg,Klj$qjv/<:I.c+vlD Q1{z\4;#!8;"R]A4)%9 _\/9G[c{|yyc1`I&IՇd߼c>?ԟB' >$&FW{N۫Ow=߿)()#-LJȩpD~8ÿK_%>%Ra`-u5P+[݀8!e@$FL $IIg9+#&-lƐ{CBٌm*@g Sh|fx{-׬;x w69˨Y*peՇ7N\Wzz݃bd'՟羍'%׿qkr'oec*X-2\}TYGMaH?(7 k,q@![N&b#sQzԧ@2jq4iĶDb8jNx+ji&pV$/(L(~#cao7Wv}|Sxoe(߶γix/®q.ЄT(:vJu]lKeOdfJP|<ǂ|@e$n$r!i>{k }'ThYMd1(2X:JWBp//7% >r1ݽ+++/_סؿ&Эֻ{F7* ?{3.A澯6(gTcRYF\9/me{HH'BTRp=Ǯ.L5^%~[JL(CN^kzYT`ya<,m9 $s"O\甐sD׵ȑYE gW1Ev!)'¸Pʤ,&8G2:!D@^ F H^՟lDŌG 002ژ (\F9Z%ՏFrroB:.> 5ʘ,,HE\更X[f*|T! 8zyc! FSto N) A:B( =9-MX*0YZ B+,토 2nV%%iQpۄoe?A޽vr}VݒZB&U!t\^s!\wD3vWUu%PI>QIui*yXK?K"I6i^\_KB~N}ɒwDB2ʾ(:s*ZD@۽xv/-ȵѽ~)[~t9mL;ѵ767x7\ ۡI Љz}C,,,S?| }ߋfW bHcBjhYCw$Kђ`Hb@Mz/ҌjT4ic;Kr 1.fZu@zTM:JUc ѷY˒r\9;_9;NH˨Eiz\!t<ׅ|Jؚr5ϬspWp+#p W<ݥ,TW=7Bئ0AUA0PF r)U\es啝@`YngplС Z '%Q&v.e[B'b13mf*467,2z(NZ:A9RdR`]y;}yMlBY `+_;?+3:~mPl]I[/Q*s(+ǁȄN "+~q}h}go}^lI 9Lmk ZT/}.U^*P3cʽ&KJ>#eP[ nZj(sX{JXzL>Ҋ\6/tcGF,wI+]M ,YuH&Cس<,LFB (٠6p_}d.=6e]{d29'ؙ5ҟo>`y#۫͐{UU>+J?v⯓bHIp7*٤Ψ2{d0b zĆ1l?@hy*%0 ( COF=[s{oIr.!bۙRږTs_[^c8{j呷wϹwf9 wÏg@S&35ޮU@$p1b4fEDuO8d뽞\{f B $:/%!bu> R1 C0IW)F#d>'1aT96jH SeT"6qEp9ĀfKO5u@)gL*&XYX!$m;4x1lcbH=4s+3/s0\Tt[ u"](Z',Vbx<% Ȓ$$"eX3u@nr:Pl4)-#dP{'3V-qNZ$l]KJ]ٓ=7[@xV%EŪ{pȯUT 2)DRb%IBU|dVC5BUƕ5u9@ oRaسgO':?ehqjK,{D(W/%>|5YBb *im~74)5/ZdJ`-07ƴ訷DDE'{ߣ7 {(Rb*(b@r>rN2؋k;C¿a /^s?͖J]8p{J }sߋY;|5OHkf.vm;UԹǻYB2_ce_Ǥ&# k9QF cJ UTB2TCs Aw]Yx{siTWzE>8gDTTe]3&t%9 [bۡm:ڀ͍ۚ׷ZΙr|OUWtժI`V'>ʥ5M2_ʜ9㍰duemA锏Sxk BQ{/_K0q %qb #.P2IH]iAG%xw[@.QLVRWW(6/f)9sSD64Q `-dJ˄Rq m]r0E踆Yc)Gcdz<&=pe]wc'ӿUп/=s zOw)5ף^Y2r NK.z7U@a\c}fc"WÍjFUΉBN\e\d!JfK$l̀ 4(d<a,l̢Z, R١+gȔ"b׉\@X2v}}kAQpY}998_WLiZu)SG6X3"& B35VcΙgM( h%bPjg)5xn<̀Ԃ?ghi!gvraks1'd9GN5iOqxc-+&B|ϓ3S@j7$~ ݞ-^MNMm{U5*Xd2؇>k6Qנ8p8y 9og>}?ݽ{wއ>;?[' Q:{NNekcq{xʂqϦ̥2VC5 qTJy>à}$MEh2T1*9z& cfK顋M/7ʈHfB c@1cIjSlBS* Dcd+cS k¨$FQ"'>%q_TeV~F5=[I=+uT{$SH=ۏ˒RlE2#Nyۖf0@Yi7nJf`@an-M#WLV΀*婧@_j _OZ Vݙn2⤦xA?&8Q_Ƕr F \%Gť{a 52|?~3_?w;nչDfXYmH85>x Yg^Ż'AT* DGA{|a ]F{ }ya.}9|/+Q:1V@I.&ZAHfU(sF#)6iNFJQ2!ΣRCPYZ3&u#-A |14!*8x S֌z^,OՔП}/^`<{xlW[˝>xs/=vï ]KAW@ 9*1t]ǁQ{БSZ^D,gl[aѳKyXda|hqThDNG90$nm׫Řdl:~-T"^ 4Q@L˘n 56]%Y/J%lVip%ᘌǬW3:#Y\ B'@o2(hH2V3c']7dWAG9`2:5$eYHTЬ CWsBs;h2F`1Ss)dѱ:jtik7|cWvY;U@rr8Q 3z/ _׈de/209$7Éq0PJ(E@5&ddlG Mא#RR0+ym2QYٳ7\w{iW##Ѩ{1tOo%W>ihd8؎6r!V G=eyP& ,zxPk^kxr%Y!5 y! ${dJCeZ`0y템.J"m^р> )!$nEmHdQ9kG, :vid!7q0VWĉ¯,aģx;pSsK;:~~8|K~'~??anɔ>l@?677xӛ{À#\{ n9 s۾H>h#!JhAƢe\LA(3egghPV. 7;#!u;IJ!he˃I\i*A;茴1 Cp0Pk;G˦C ZfkQv!"&w )uq5mxgv'a)nv,W`aZ`#⟢,Fr$H@V^PyFף\S,)ﯪhWjVFJ @-tp Riho82%.qdrrľb@Hy D@+K*EmjC3G[قA(!שK1JPra_,> ushq$REDk[4kp/ԲG؀Ԇ3O7<:\s8p`_? +++%=ʇ<Ų?ߘ={A6cc[x;,'+g\ co2~l㵪^ ~VRo-OS:OˋF7hh2`UB>Ohb$Y$)FD jCgTR`fQw,$e9d2ɥ:&ꊐ5 }ګ1ٵ"{_}sؽjKKKX[[;'>UU[뮓燁4d}n0P//˱g8簰G7N uم^W^sdU6F S3{Y^wJ}t%ã^"JU}JDXX`F0E HduG 8M[,kQEȜ&;NTR0Vבe-e@*Ę6 C;WHS^phQw##hε=fcVZ2U6 ( tb3`,뙑ц(Z + =-g912ukcD T6k"S 7.L2ш-4;q_h M߂N?z{Gd8X#G6fkěYd#"FlO2 '5yYd!'rYN P9?\Pfb V8rbxuqUO +%HZ2X^Zxb7u9hn+`!zIX`;2ZTk%{Pr]})eRnt%#UVS)k"s~YтL:s x9 e8=v)3Gs&'qq͋^`7g«sP*СC/~^.Ǔ >۶3??t%8_5~*BP<(_ClI K="֚|0R@5e HR-Ӹ,/ʾ`}uGLfu%iyLc 8kxc&mڞG6f¨֬[\I/L&ؚMa`5eY6)l+fQ#&c2I #7C X$c(պbSD+9 &qB`Ir.rs 1 &bby ]'m^K!23K8JM:? Y Q@stm/*ֺ,)}uP?E]$vD+ƶ`?+.`$A5&Y B^L6AȘRY1XzsǞ$)BJ W5ChiO\pW2W1 U +0l7}diịjqղ݌ a,޿>u Ca}ަ$V]8(%⪫^r_=Hɸ=d -yV>Mm77TG,#m+Ã,UFLjJg"TݑYnkG(3YT%M>akcS&Vg׽EVo><Ȣdcy_敕9$O*7M7VW]v`׮]8{ly*|~pxe|hCt :lN[< %hX&zSxk`%> [9lMgcaTpȞ9#̜G4H"r6XUX\dPp<)[-p=2ZD 3XrGաs|E=ϑgB[~[_x:s؃׌f%lsE̩5T`k$;#nE2*>~1sO gfWdDQs~A⤪k\pݵW㚫ӞvyߞQ-F<9]W6Vܜkw$',FDG6xG7;t?/N^A!e8?rT<2Ir[;2-m"H;I*=t^)Z,Q?:lb,e/˺$ILsV}^zd:cn*S$+qN jTPEB&#QIn8.tgpﭷ|Tbudx\ПaPb޽oS?+ҟT@Ͻ򕯜,OOZP9E so o: 5ahD8빊nyvz؟M1)8H$4@6P8{; ; [#=2'b[`!c-pd&S߂L^"Rh$cJ]ovRede˛euŢ4XW#pL6¹ 2 kF1Xr I>l gHjR/l\{  1,094 q9G3v-n8\ #"85S CюK!߰,CUAUm\ #8Sxy7fgBC!D3eLurA3?/5ZnaA 0Rs XXeG&6psuȜS_ dΈYN}Ou6)̠#h撟XK `wn(ׅ3R}˞B\֊.R\FQ T*s e|MzI{. gN#lk,zf!ǝޥp`n:u[e]`,#}#/U]K@FfEqr]5΅$>//xN?k)#'Mp\q袋lIcx:N?g <%O"JSEAߥw 'WIS$d"Voaf/C{6_M4COt_[bU ~-+c¨ɐHu~aP[^!ŭ)+r@X_dxTlDrh4Hi<|JUS 谻X|ojZ W Gu,|Akܸ3j9p4z)ko ָ+\|7~=f53o_|xUx1 M=}chbJȗ(z4VHn$_}`Fp[[| x-CBtQ`j#9_1e!TzyD[E▇}tb6g+T8*QMWԫd1[$z큃q1oihP<훋 l\81 SϏr]bsscoz8# 6e\ymGQێ Kr|%> ?v7Uoc pk۶/9g=<,--ե!~΋JyÓoq ~KqG $]ڮeX5JjViHO.FCSЍPF>`n)&&6׍d׫Ah6|hF"Sƀt!BƸmk= Vo䦜'4zj5HoC@ B`Gg3 HzҿФ̲v,7b¿״<笕z@~& C_w=~oF ‚Nض3T륟iCx䱱-ʘ5P;?) ϪRdmD15IA@xL E-'Vj!Qȡ߁ݽd=%'[[.:fs,7HLH4lwwR#&ҍ1O05zoȺ t.n}|"{YRR0r P畜P'~\ʶ‘eY _w_R90T(u1Q;-"t{qmU -1\|9!mkid,;HH _b%T,c8Y ̐h@Аl"% Qw˱T 1!ADl6FnC(aH;P 6%b#l-Ll@bb &YK]"fPcٙa< =^ciqUUa<askQQBf=b:L&XXX1=vqV6bҟU Ż&8$*"_[l[#(D6HIz]b3U;͉(dTdgT '1ծ"W;V!$je8έwb?*B)vnOuH9g|ĉ'B_XX!穾]|ϻݸ}lr_.GF+ihex(+ ]3Q px\zCqJu-jFF_N`&W[cwZtzv&zDfSpN|ؒ5hkqΡ2m'+NQM 9tHALby?]@H3r"g ;ϭ8c1lHAYd9bF@ k!ύr  0QX`h18Ew RϦ#\x[Ln6w0SF8fBËH"'AZ)Hr)pEmM ޤ'ȩA;iw.Da>|G\}b<?}~ *Tc{ATJ$ rخ*M2EtgrD*%]_`~lNuUyצ~>yvZ؞rD0g_/GG륥bE7N7n}B.-7uśӞ2K췼E51 !0Ha:#֞28Z;|Y0$gx mCB1Pel%o u]U\AE}Ay;3J4ت䄑ȶr,ЩuX[٨( Cl :(3RȁH"WUHq1"HbsBjR׉Y +吐;!HbǦE,f43>R;Pꀜ`Mp$r AnGH'X~UKɦ+aFzrܮZ4[kd'c?q5`jAy"&bg [*SZN&{ &Y9 qn& mgM"M +++ػw/c A' u]Ŧi g1;v!`6X > l;kaӴR4dbJb\XDyaQyrB׊1RYǘ:JbB+Ai!H$ L9^DRbе]#Y L<ğKѠz 1EV|@) )nwAQ NBc{7 #?v )Ƌ9<@$G"&}LϮR_/..5㪫z^#؟rB{衇^R:|/If|Qxػw/sض D8< ܸ )^#u뒛9YkH 2Sl%APJRj4VXWLOA5!igבB@8+Q7" Ҧ9q G&EW!qog-3ur1B%qTQHL-+BZ_Թn 9tƍ`B<=F0Dy [M jlTc ` R=z- "O,J4Ɓiï3fn{Ů]0lVpp2S|>ގtݏb&Iyg7IPȕzEMBEw$֪$51 ĮMY]ts0g~6؝ť3(_oo3o\wukU`/'s/\HXfX.ϕѣG\}圯>$#Y[[å^Çjn|N_?QE~>b/m"uUq~3ȢQ%B1Alfl! .Oػ׸&={ Zf Qn+Pa"1mz,Sjqԯ]y;ڪ= &'ft"(5[/(p;g-%ugάu_w#qVIsh΋B~;=4F#LS!̇8zMDO=Ydx vاnӘзϤO{G=|s 0tvҷyc%Ew(WyTѲ\RɃ=̘AMdɿ!(u}Px\]`hAȡQ0pOִ [{$' U2#HAM#ϵt74O\( ku"cKCnlpB?O!v@AHsIg\mge%1 ! Yo٬2!v\%PJ"؂RC @`sDm"L7 <vHS8-S\ZZ'uHu (mq^܈gƻi1 !B-":/o5Xn9y5}z* Oftǀ4Ƽ'*o|ӟHInm#)NchÏ"\v"PЋf6׾ι<WP(Sk2 6zn/r?2q(PF i/[ Iϩ^y|,+\O`pL@Uq.gr!APj,[f:-+ L!BS}U+C bk7\uv4P9 A<1,іrZ ϡ[5ߍw9WկB=Bw1ʽxn@ݟ3oPfd3?|Y\/l3}#[o~uTa dUJCkB 됉1Kv$o&ڒԀj) pP5tG@S;g4MZ1MS!2pZ/J`ƮS4eBТQeS*mgֳWPGFsBX DAQ5SV1M}aHs52^odI#v =8IKMXdgK^9)Nlޟ y&dIp]1"-0 {f'WIRe3Іtw/mh1z=bhbkBwG{K:>+:=jS*΁FpI˙˩O)Ȧޖԁۓ̵X$3v~,FrTb%/н񱏮ͭVnK.=--_)18r_[78|xu]E?\Ne$sH %L'}5R!+w:w?%(uH9R(rzAZ۠FR=99k$bkq5M;1k嵓e))! ur3hE8'm3ű40xFI L ɮG׵mcѶʺhP%2GC%nV2lT,CCz{oѻCѴõG9[ KN֝GRc߂(ӥd` YN)Z==91uy3]lS*IC߮v.ɆȻQ&[kf yc'{?y`ͨ_+8^fLgy}џ9WTsx;ށQYX6csJk__}G}"d̨jD /#bJ&S-CxӳVy,LEV΢on u}Oݢ|`{:ECzT)ЮVzX@ "SuQUS#NUqNZ,FQ!DA ?9(YNB @X/)|:ɒ?"n2:hBIpjܯtbHMF j@i¡YD/9; )' mnNn~9JZjO5sr"}I*y@K&P =a]9i|/4vlY@ wb lZJ(=ʙ#!IvqſWxk_W ,+3=6}߯EqZXnԾt~&ebL >IӒ;Ҫ ;%)idq\Ln+BUe8൝Ցe'L"3zҳ.X1~*} ֓s (KG)WERTIfM+=ut/% Mk*,BB1)E&5[Z>!fev櫶,C*Nb∤_U6+8o!:'n%m/D33^ցq 2 Nj/ҟz꩏;~㢅< w淚Nn;/}o֡Tj'v(1r Q>uO:4aӠ @( mZ]d *lM"*(O|]@A`nv@{XEE"Wp%'tB9:1mo4өl|mr9KpN59U : d0] >r)ȋrǤԫr7w_LjIUa%I0) /tJ*m/+xsj/oIO}"’V v$ Ÿ*$A[[tZkY2ƪe LQ"uq$;C0a54g}}E"[MN8:1Vb/m% Ρ0#֞r4t4_RQ4Ve&!+)R7%G`&B'E*5iV .}ZsJ]pըz)'d!PH7og̢d& HY-Fdcej+bi&=^uF ]rѷ/̘XIY2(J͛7RJ?5O@fwŶOv+ԎԧM9)X] @ϫqS>'TH$gsS<(\Eb읓6NQ55|׍҈NZvlM m1A[~LR -n(~^-:NpT5rku,ڛ/ xX[* XH('mKj(Lk}[qMV-qL!ŸGjEQkU.^@Q mnk.@w3qe0<&+ّ QZ =.'憎MiGToyFl)X7 ׳3Z0dNw EO^Fpz:8Y{ٹ=mw麻Mڒ5Z![-ՀqfZ}(}@ QCT8CA>(hNk*b.Z0Z>҇.A۞\޲ML 5֋>@TO>HEky\?[P$NBKsh*Iw.ECELL7&U"Suike,c},!UWհYPUIwȘ*-CqնTWcX?+0m/mtȱCR1i GPε]XtQ-bжRDlg\R׾Ԯ^DYjAa2X%Qa$8O?[&#xpZ `l%vkm c.4JUnV</Vt"ݐFd24K@,-2@cA,+W9.5b (Ce:bs)ޖt5*VLJB+iCԠuDPq;#X[/a'R@W`2@6A1ڲ5)ۇ8<8/E l |ֱ )֑ٝ3R CtR4v }Kwߡ{ENI\X% gV>#.ABߙlȥmd.}_Qj^U}fO,H.s KY)ٌE 9v%rϪ;.CQWZh)E(ĄO?wO9tAJ^Đ* sX?C,W$P j6qeυ펜r11eڰfǦQgff3PKnݺw"ɟ,Ʒi4Ms((^67+(bi5oqOC=)X_guh]hVK"*J,]K9_9ڝ!EjFz'nNW<^_iMRHIxDA. JV Pp + uh[ԆFV=q@u[QE#W.z8` b TҊ$ԍֳv (9Ep;+/e*ȁRWG cdN)g ~W޹]3I%T%~fq JsAb>Hۢ!1}ƍX.W_V (T~~?B/=mԪ P2dbi9CDW H wX/@N[NI QH`.+{ԫ56#MJx Bb(1[YS~6".O$}B@Et_30pgoS9.0DŽ!(ѕ>o(i=ϕq:_!iZ6ZoZ4wJQ"/ƒ^r[)"%(T&Ts;ݠˢD$)Ar~DÝ"daZsbJ}뫯\%HM^ntb_#/O_;N9Gι_4T;}2RrJoDR~0Zpf܅P?S?uF3nt|WDIx.-z8 HEК9v(TP:SKKj,d6YfR>!!;8SN59iw' ql~U:69IYI9O#yū$VXώXPk:>y-rXa{),U[f`hM>l9+~>ts>v6őF [䔚1MVIQ·t'VpJ@S~%j rYbޣ׋)]Sg8_uK9kRh e5By6f"8/HGI뭷r1nAu@o:)]Iǫ6L2@bMKnDO]UU@e&>dUlLu,} S`m%;1𱗨<m }/ۮ(>mW-r軕Ե+n.ۙaB!! %M@Τ*sBhdXR"RA)_T:Đ2;MvNIl{}"%j4%EU Fq(PvOjSSjP d UL`)l#kdKPM'CJO8kXT]NpȽ2 ũǚN-8,g¦69u@H\δ%8WURgK炁bS@RԾwGW+'=HݜU .S@Ra?%U pGeA};0zԷzaѹ(gv:͛TxjA͞)d?0Jy1}i!X(Cbqol1Lw|^Ōr1jCf8qrJ8)[z"c4Nf v BZ{}KDwk;h`IT Ѽ1)nF[k9@4[`vp6bʘw/ {)>3ݨ_/: ,dԫ-oyץΣi5J7M-,/\aꛞ=<>YIϒ)%ۗ}r@V#pE!|h`5\#UBrgr]5B4qDq+$%Yt3L&m+BAQ@A5;fq|[Ae Ȳ Nw-HYIH!ͥLxV1;rhKQ뒦I=|,TF@0͇%ʸd4+i4~ZXwi/g'ƾ%eCv쳰 2a5ky$h$3Y. kyCKtmU5# J1Rr 0̅!'>WQrd\~_K?KuxB}宇U `uQ2j+cwxmL(\:Wv{OH*JYhgD$9LU/3}8HrG8yȩp #!d爴:QP%rbkc$u I)3Xt(M,ǧy6R5\J-,LYVͪ!3N}HG JJ$5qF.mޖ E"E* 'te`\ tjs3Jr1ͨGYJ'kobw[opem{Ƴc}8t7_ۯw=8YX?8Xm]2t9Ha'vZW^s#YyWr5׈C'8*JCA;Ub\I`%J**glm7Xz~G"z)DL&] b􋹤ÛIhW-jrL=L%-Lw.auz[zf"L< {%I<ȥI1 nfDc35G$lT ' tcp -zR: <.zvikA#NJBeGNX K(|ٜX(6.# o/omoJv}.i+4{ت=\J:OƼW FJ U82\%s% Ŝ8c_ؽD^[-xoGC{XIgn N te]F0;+_9\ Tb]nDBJ!).QNMИwB 'cl]>,?%e'?dIbl| n<2ܺPiB^hԨ83CxsP2"U3%_E +zZ,8'~(~lqXh9 ߼0/8u+bYGԦ pngArE&җ*-t.dU3D>z|]% p-R/7ʕH)a>_AAgk>{hjC?B^wXŽA&¢u缶J;;Y2fFBPCzQeR͍V=nܼO{,m>v vo ck[">N) qtruٞqז{Ư#D.5trTTQi2u]U5|3'H +A} p()++*<*C\W¼f4M WHUbGH)`cTDFH]]g>n-š򗬋AeH?هdo9'-[ G;MuWCnf/]rZ6#{Lmlc{κ?ywO?ty.΋LƯ=;W> CeaSwpKAT$4B c2k]'? Gp+ [xu y5GVGzi9wsp@=\\\\ >qsh%>OhNp@h-epLJD"~ihkR(h51R%;Jm"<<\pD[&EUrXӐ&bPKJX8@fyt-:W2[w ?<61qLNiaқú1 aGG8u )JMU JKX^ϴBCt 4֯~W~t$r^E#w$17""Ȋ.lItpU-j BHB fԦ$*lj)*Ԩ 0 VA;xJ&#GDZW@ ;+9U}N庥rB\ P-HR^<&JїH]"r;G\r}nD">5z 8fs޶y d3ׯ_A"09o{YM/:'|}ί@ЮVy9d{$CbX<4D9?ĺoQ9+b??ׯ_LH8={ԟV1=m/%)x*p5HJW%4TUT!;!oJE \ B-a1 vqo aGmLHfrK8iy+*t"(ĠF?H$Q-ܯ=.ĤLnUe[3+CN޶s<ǕcT#.m0sX{"n85mUۈ릆 z]1}ߋ>Å i2ָq.рËaƄB/p#v v-ŘN9Z]4Վ*#3=+NJʙf -;U3E4eq˩L]VTℐ{mHaCܸuqpt[FhfE'ϊrUf\?/"4MSfbbRQ; o׾> I%^#a<ΔAIbLt‘zѳ5rXEǗdž lRHUDt:7E\.$ݩ R6A𶪮ʘr.Y፲p7~ F7[:sSiœ<`yf5E4M rQܨM/YzJ E͌j6E34z}>3laqG^Dk; FxPGQ2,Ci=R2R9qfgcP0' ĀW07S `N岌{5)$.yy["k7lxIlAr0h'4*SYh-XoZ/>r,F6Rx>73DkSA'8(T]$۬R0ӧ]-[A:&?oӟ*: {1/M*OvUu d9{5Vnv~>G;#,Wm~١ufvȨ6BH%{@RVr/ZУs?c;G#U4P~Nb Y˘*^_SP9U!ѵcD4UۼGH*HsB< x8<:Qk+gE،R3Vr%&i-!*RbuE5(bA[OzkW͵If${+W`0 綆gyZzO䢋91υpL^\%oSgy5J)iGoξɁX[+%:wb$waU0&8_i PiI J@q.F5K9z'`[~jtF);%ʤ( &XϬk i\$%JgVaN}jC;ž>LIZT% EFTa9>@}sD_jIqV+f~mEs[@0ϲQ'"n0gjk8?Qı8U6g͝ K5Ls>7{|yBj$Iq<pm<&~ԍ"g[W~0Oo'Cb~@#gKV3*|1jՁᑲ"!4]Tx5jK1}+ayzvq:၏QBhjfQՔ}R*g*:Q+lnAK9+׻tˤiznڍk,., -aL)kGj Kq~ Qn^yEd+`c쐳|=6V'1ľxW&Ѝiq7?zƍg82#3ŕs;7JﺮϻHоKKrJ6s^=/ݟFP(0B#1™KFtRGRC}&W'BMs}n<))MXVH'^(L5漗El"Z?4@X!3)]Բ5rcNtRC\)5!ЫΤe$53Esb鵤ɍg}%_BRd)sh*>I䫄Q8b]dScRcx#^MU1shdfߒ·f-m Z/ﳷLsfPx6)k 4̥3QEk {: F[> 菾w6[v8S:\֮D辮qJg8灣۷q|H-j nO1nA*<jY$" ,YqBRĘEіnIn'oY "E~K@豳k7ZKeսR?n$1S:%%ԸEQ_C41n+7" l˽SY"kO+Y2<{yBZkiXͲ%~`K6RyO &QzN+{_ÏO~t^{MW0+@MD[T9e ~/۹fO_AK8*MeJ& iQ[.u\XM:FRj$HiHۋQ+=(J^a(>u2^u&9Ia^~ Գ}x>l8{1%F?{D9.2XKk}*X;hddsJh 8WFQڹt#:kƃ>A.uS:Ͻm;x&^g4grh?opW1rU#a.1+ xipkx>Ef8@,*kyvJ˶-- i9׵Fr4%xlQ* H2r(i*2)պ(06႕:A!A@Rе+Km:XfA89ɖ-Ce b hr2c#RX,岯PbȇyKȓ i]TOPh^-sz +⏔n>۷ Nnzrumyo{lCfe*JfJ#ntbOO ߇R\bBkw;rAZwuq iBe(=Gڲ0Za.lm9A\rvc=rAS8ZBӺt"ךTˑYtHS61"]H}9.mz)ҪG:R8쵛8)؆]a~Y̭7{a<Ɨk.+ۣ%I+젪u/qCd;ɜ+$ @ȲLlE t&Š7ݳYV3._=1C=+} ]f{OgQ{^|#) 66{u%ˀȧZ*݄lKȄRXEMi,,ڌ;%N|UnGӸZEֈ;BTËg^*TZĒ$tWK=lEs|ɡa9#l5HRM}Q&T|pwFu7g8r%Őb!-6 jV}՗k-D#OdNTS@ n+O}Vx;eqE#^ $+y 5̒n,mHԂԜSߪy0bs4݊F9IJQjw[5ia8҈`,)͙# i$R1ʆ;u`p1^~ nĵk7p|t,N8 G? NnOM(Nlc^s \\%=3[Clv<z%ڂjo7:}F[u؛Gwpp`5t3HT*B/ w@}윻:猟ɟ, i(kOp#J?nwW X8gEGGfecKt b"1*/hHPΊf{*#(18ETISH&w|R9%M+!e!,7HƲE1bC  &iʵYlrf0?9T_f"\wL?;OOgq 80QQ52ptY3fLh[ȕ <{%'mp2r(MUU] \'Sݹfk sö+EP FK)"^}uWsΚWNvVU$Q=RN&,])ڜBj()"+IWtx YrR0 gIӧ\rA|Nܲ Wmã<-88l>GF%Fh jpFw|dsX\c VYEd<:"p1r*knv&B5Ύu3URu-wumMQzCCζݱ~|#{y“O>O.4^d2|.|N4v ë/,w^J %fR [_cF[h$ >DfG~rTÕט/PJryHf"X ,xHgZ68AΫi\IqIY}_p4sY*9"Uoδ>WjghZ更u?0*7S Asv 'iyʌjv)aj+bjqmԶiN5Fd~Z=\3 ]zZ}iK'.su\9-8@;ZOFEύyϭ[Mu[&x[`/_zg FeH3QB9,i5зIub3Q9ŒF7# h7CRZRI[-eKKB F)4ks0R${lqr:Ǎ[xx-;ա=RJlc9hydeKMo u}XuoQJZ9+I/R{1#K`v:%=<\P7ͷ;<>>>edsr/C~7͛9w\tpmگ\RޭmLs7~Ϳ#M@5ԪI>L=>4v,<ˤ(}zkWת LFKzTLISPZΑhh6+^b:unl; ؾzxZJ=2q2+$}YMa/@SC;Qf3`M? }$l8d  mISMKp| w ֛w5wjHiM'/t4T2R8]hH ^+4zN{R 9Lt hOK5^D;2£R TRyz?(nc[K'~ق}F0~7/B+W}nwCͫ=; T^ P~ubhEewi O'[?;."2iSYG#| b 1S+rxGP "DI(8rۥ,O50ù_ Dm2ڑTn0n:@rSP&FvIFjm7H aHCq/~6H!(՝Sq pΙ!b|s97=wkIjXQ<$)^7<%Z0Zԃm)a@%,j]'ݕ{g Q2933[86SO} kyk{\|٥FqLPOjY7Y&PC'"xkT0愔{/QD5Oȡou5i-^ M"-=''3g d>|1GzMRV`j;4@pUc튵z9a}2VD`GN?T4b.{o!NKToar*='hFZ$eMeOr)pUo &qy܁gn޼~ZOX@;e cuvMD&E1D.<WKe?Y(1C1>c8Blٸ/.ڮbtbX!6A~Y-iye,݆ yv_5p*zKsqz@.Ξݽ@6ڲ~x4+FȖϜ; ̴P 6 & (Q|`58!kv}kTϳ<===]}ǹ?~zw[O>$>6f^lbRvUyARW(;A psEd45diXnDkzSOBo'?V2P/g(3$T6/a5+=Qҿej&&2dk%af" 5 DDNOMF.Syxyؠr/} iUv^qČܜPm5}EHIYO$%K!R;ŚK[kFD+½8eʝO(]_76<qF:Y"*Hf[SΚ,~# L- y[61Aԃ~^;+3coP+> _Uh*W̓Z=k\Rж&A9ɖJ66BBH E"uO3ڶG۶XhڮE2b0sJ0PK=B4R1Z:(RΆ)!B7?FȜ{B/v. BmxM, #)3bvK#R?{lH) ͵W關yp>&t,7=7Zۛ}'56g~/3~~ Z]ںp %ZD3׾Z.KD8Bkɜ< d=Z Rev\Ӂ9 t<,,K .2!u`L4)_wcl_yD7HdOEkE|\t?b@}d(FL,Q^ơ](RCդbTҬ@{tT^gb m 쉮:.7 3BG{z[U޽~5SA:W%Zkmz / M-6;gMY[rZBinKnjJ6@A,N @kݰIliXv=n޸| ѱ[tŒ\( 2 qBUO qv ̝Uu.-_'SHer?bIc!$1\ٍ`~ [s0L%[fYyK/Dh&[\\aj-2$MeŮ]1#/`K5ޫ1RRUH}'s:*OTHαӞkX9KFʎ\WIm~q c,f_c୏?PSwAߜpN-F]rkF}?Ƌs|?X.pO9N;^MoWUT27U0eUk 7%=^1?lQj]O}}uM`+J,aHY;62&$1um (eEY v%9^hJ|V.1 0%N3K[@Yb顖nFu|[&w#Q95% hiJɎn ),R`@Z7rG_HFX- fŐc[qxpS0ꈐlo.46 Rq@ pUhrE(JYŁw0Г_UMF K")uZոo*yCzuE%idz5Dz\$>J=4Xè Q3{Ahrsd\ԉyX@4A&Nڂiay8W晁Ԍ sJu!HMdsi@\9, WTT\+5/51l]K 5C fK,]=R Ѝ%gˮĸNA17o.1?vJ)w|>~Gx>|g˗/E7- (,ozF`0(ڄDYSC+h/NsGT(5)5`l.JRbC/шSt8< _hT{X4l3i& -ktR.BQ5LGoI8G8ʥeZ؈Jﵤ9BmLk(3z y(Wav!mT$IKP{Yf/"fn6!O B֖йi@X.?t9lfv<~1>o .UD% dCh }RBDVdlLGnѩѷNa)9!0i jdS4ZR[9j81)g[.gfl+ dR%X.ITx͸[϶b|])5gR Tښ14d줦h_9GxW+ʕ1:<"}0F+gym3DojQ%?)0 R+00{A-<K\'u~}:&BlHgulׁ3= [ m۞7 nݺU{xuY ƒx%fnvbS e~꘥^x ,sWgkakgW^NX͏W(CB\֫QVIVΚP K\]2@=If,΋2f`)typIlEʜDR)1F.'O$ݷ%cjԙGеi IDLlLs,a]ި@Ba?ƒyյtE0SKmR}/ĠߡmL6y{Y.Z9g]34zkI-y" J64}Nm$/i[ ̧=n!9&<]((PuFPfV 1ZC$[ JkOֺ R5/,Lq[rS?)x8j70-YɛFYr[i)_*ϭ;GD"&+G, [:FpR YGqғŸ7ޡϓy5zPEN+jBO|ⓨU0>oQȕ#D.qLk8.VA=ͰKTR6Gb1ࢶf4Y!/q5ޢ;p#Cpp)Or!deN>KlR쏶NChy2YHɬ0LH%n& ?G-XO9in(#uIBid*,ʎa$HxKX^VpRDlOwR Ga6k$YZW^<=#Cn?_GUUٹ(miTそ8" ƒy }LL &͡7g+ơW)Pq)YˢH`YJ AR"!%uј3PaRs4ps ofK5+,L$NgHU] B8%f?d#0fl8CNe3U:+.9;zgگ#_{ݣeN|h "V /qG/4Gn' G>FDEsƏȏ@¦fǯ{6_3>ڇ/zr;ۤ>qH|XaLF`)iԺg-)@զVC=p;d 2it޵x#:!CI'+[`=I)"(yhk/Q,59hH*X`Ӓ ! &RA -c\p% uČ"pp󖠫׀~ Q9F@&:3o~}Dfjծ۰VE')Q-uV"q7bf3fl+y YA5F5$O_#:3>-|#OEЖ3٩Fiw`#ۙ3mrZRSTNF,,a80q8 Qs^rqԕXƢ4=pZQ$ݏQ/CSjH1*b`nj*ڹ0ij j@L5xi|c,R*썺V^nZFOF~&ΥN @}Ob1_2/-\~ ZVksl!ocm6 _AG!y~(z׻@.Zu?{8I99auzTPe#+ B)ZRj[i81knVBوYIiӝ˗V Ws_WSiLأ>!=cqSAzSZ{RvHeelJc)Q-8>ISִQBzTMR4=FgJ:Q#̅S:8rк`}uڶQXJAǺFL˂ dmN0H fHh ஜvϜm梵x{AOi+ :GnFs<Junu۷oǿmԾy 7|&-lMk=ٞ|`r娜Q"OV#UOȖ$͍7(N9ֶb vuARS gMo#t2ua IoU2+ډA6-ՉpVR;@%&Jd}Ft?p3uUW[[xW}Z޿K_Q^V8Dli8*@vӘoϟw!YҟxޟؠmF6c=vww7o}=ݿ7WտR; <0.)Il慠k_Ǜa^7e0%V`@ )OWOPvqŒ~Z3E!BK@qEǚ<vq d5i we |5*c1پ3&ӽ#h샑svL ȟiTUufS}3W_ڲz_)254v@o*QY#Tu\Ck%xDol Fxmy#f乪q>ZVҿ9Q@ ^u: ?9ip)拊i8+ N':CT.TSDPS'$C͂-M;G uؽzDw o-cS*|<!kld>NF87Jﺮϼq4H]~sW\Y#LpN_j@+F bk{WB/g) ZJs1LkF?i6gχֵoiPGƶ u)Ғ3|_@wr}h9vgym@h1mK}ٕдتxfn.C[͜,X0;`Csk2F,Yڹd )WuɡmEdPsN7MO5/ P+'0-K n!Jf'җImȡ5Ȋ}aE0E6Pcmߥ*Pp;e}fu#k)Ξk묡!s5qF0r([gfT2 #q.W KUk Ttso^ozL^TOr7w? ̹'yQ[߄GGR2 Mԁ9 ;[⑳PMuutliR!Df%tX +8GB+4M׍IrB<\VHwOQ7ML}[R4r0؄dnmy} 婜W-,<kJ˾횾E}q^ç?rb݀y(3Ơ#L=QU՟lx~O=Tyʕ+߷_~5 J;5@Idz7K[VֲAV"E++>?2;;%2xCѢV5 f/@Qz!70-Ca)dhIIVBYU KV8V D~퉐6XH߫YȔKMZ3*A:>V2BM;Qތ֚K.鞟ɘ33sO>x8r} ;7q!@c\rx^v4gfw}}˛_G_Pȫ0 f^?1] FP CM:giӺԻ+A֛i-7bj\(@(|z];P5Skx1"Gͷ?CRJBEn i @4ʶ:bJP6 Mݨ^4g5$Jb.L:fG~6Nn')>?, ^`Yi!@~qѨ& sB"rw*Y?ohAa&H 6wMMXnNGz=w:f䄺Q6P拟y"r\ds ab=VD'`'%5*;ٌEfpBx׻ލԙx-oՔZW젩$ oz+G^wz*/)xt.t'reڵPZ)aݺHp-F6iQ'P8+P|633&2 8f^Ld 9 8 ؗT/Hf\Us|>A&x/)G-8)/e_z"uG^76Ɨʶ8e$nΫlsI_C]8J֍ZSO}wx{[ی/8s(6M.G֏mm]b#Pa) "! x?pkd& 7qe֨Q$w&Z5rz%P_K|f[p!aW1 qT^S?C&[Ł! #]:xlOO`$tafL#͈DeCҺZ, =3ڳ?gjqtt$j ɾ(o$+v,T{/ѠGYY <.J)A4ȟv0gKD7ik2r ۥD7ZzaŔ2_S2ACdžD 7V"p53Κ6<:-gKDU_70|wt;ͰliR9P7q3\HR7&ߒqeȴttqGj^x(+3uHlε;Eu]hی?_ n\b]kjl}'#yƷYRnIAC"+$BxVPyc;i.Xdg3ݔd",e`kCRF&Q:$? jo#vh x]8,:#TWBjcٸ;L.=,㕺:?)z-' T(T1`]F`xGǭ[bL^ #uǠ2f2d4v=>J=ꏣ^i{X7r'&,㬟Xޏ~\.M7o]PK+]>J&\͢3)\]Lu9aWzI#Fo ι~q L_(L14eF +c7bow;[ U"]sAK0L1@LGrVkqv&Sn7+tܧ2 .)"QkʻԋUAQLt20+&ґ^St,-q"*48B:pWsWԼl;-h|sGWx7xp qu{%/Ǵs1žHw// c>y^~Acc~no͛7r8 |gygfEQ_ 7iĈg ֺy*0qP)[h+LP,5 >ԜbʨFyTb 9JگBbܼfH 1o#ktwB) kh^HD~ڄoh_ՓRG_H!+SA͈!oBd`9hkFs*_gnin ӲOQ"jܘzd4⩈0\g`o'q6L{hmǒcUu4I |us %0P{RwqĄ5ʹ8\P č7h77X[}ppP~׼dW]+mW^LC܊MJ<5\4SE3qq,D:H=88ICsEN} }[7aHTH2PD)?t+(Ϫd䜼M09]k9X1TUpOD39S\U\VƤnK5&_~?^zu̗F|<\.~ff}r fZzxoiy?%=rwQߙ6oF݉4 _Z@ƈܯ$i<˝zAIjra%ج|F5aFxRkjŹֱpBLic~b%=w]oXbh[Af364ŔvC`ј735IR ^u]zGRP7U/%V<3 NjQJRڼ<ؖaY n{ Q UI^ɕ[3p52־&y8-F]Yr1pz|(5W $'OSj1󆎫fr!}p]Qw!dc3Fv".7~p5*;ͽ¢t_vbCt+۸|iTբcoZ yK4&,D̿T~i#5EF(Y|{V&>S pQ6uuHJ^mg/l͉}-C%ã-p>{3b -jѡ2W. ZfNVZy1۝2ofYA_~V~w=yχ_>jb~tIyi7Nxo-t]\VG8ƈwe|r](o7cߏq{HuC $=Z^aEͫkI$Ҡ٬!X-rtg_jMjtnI/9(rr=$ EZ#TK <91qn<M,XU,n(r)nܺ[)b;CeSЯfXm -ujQȨ* 4=R" ePm=wʊ9z sF=ZNwGzϮ3Gp@UL#lA!bwXW-K#V 2 yaؘS TQkmYH97;Iʕ+kQE?ϋoݺ#rθ:q$4{_K_V\Tj"c˂TJ?A-Xi0X0r2sV83^Iua,)֭eBS% g 4[\W)6DzIƞiK]4;:6q̤f4ભQ \qNCa1R{ kp=Ǖ̥h&5b{o-g_ո3-s뺵=MCO/ |l/LͬEO0J;{YwcvzXgR|'^&]~yՐ~F><'~zUE>] p'4O?3=~sRJx/^E_z\_{BgV"&i}=6bgؖzgHPʯ0D@c`2Lș4^ Kgi7om+쏥eD5Z*MB5ṕEZguE!ʆ|(NfH Ś ՝'4\[ Gr_ը|^_&4!M \)NuSFiyeWU&p7֫1J*HB0AVDAAmAпb6B (-t1B!!IUj|Uo^?{W@́zs=Z[ߗСCGN8Wzڲ1 ? K=B[KP,..Ӊ=ml'z2{ 8 D+ytJBlaŬYEl&]Ma[KVFEEw/'F]5[|APPeRˡ `YѲvɃ+P0( Bk])x=!5SZ׏ΧỶ?GRS/+M?2.\Sn`)ubE(B'h>%QsTRg#,;EBx`O|7w}_3曂4ox?Կ|)sm/wuWO}3~ư*/d,SJׇhәv =bYRSH; ^*id"((=/!Z#sr(3`7Le&O?dcSe&Os@F@p +d-*sbH*9=u E)lGidBcb=υg5+ʁn-][b@=BԘR3!1b;!QC1@܀ѡT:/ h?^wŁZʽ׵<(x^CR@u7.B $;$쵼?`I<˲ĝwCb 6v{lg)(N]8Ί;w ),0W5e *nLpɥ|kz'it$sy.}.c NRZ5i%jV xg1*ÈsEdDz eB9TonS8Bh }8kqf[LbcccZxނS]GFk|,K2 DwuׯkIp~P,j$I_r{F+p.2lTuEW+A`c}'B fC%Fr )4 KVSO^ٹ1iէIF?c ϲṳ̃&Zih@ T%w9Yi-JR G MPK{2^{WVVSudZ?x)mrNPl~IݯxQ]XtW!/5o`5v [p g0??/ gIwX]]";?mmg>b휙 |&, lښށFga8;NAk(Zzɧ֡q)ymbqd!?sfX&~ *8Sb(qFj-qD<ꨁ'C9* 1hhIXV7R6d}'%Ovφk\9^{h஻:xaE >W'.>&J8ѓY 7ޛhv@]ƻq:431ԻW5?Ӯj%}jD@lf$ *P7Gz+Zaq%VX m+/pH߰jFβ .I LUZD$9uIkO"͡^ǎciyMPzna`0BJoH](mD ξJ}KJ2zx k km"i (j K$DȻ)`4fdϳXQ{/V\SO"S ٲ!cD{ Glbȵ#|%t*409០jaZ|g|4z@DX[[}ҨnOݭڭ34/z":SLM$*E H&FuQkBZ PN)kKp=ٵtxjĻ]̤մ w99uxUdCd}uWb-})MV8(ry1I8ԾaB|:LDgf!|wF-ΑhфI8WcXMd.=^`(|nO~fR^'ƍmj˷<CSƘu)rp𯍢hS;?ݲ0/z+}!Ojv շh!k ]>MB*?!F7V6]خ1C9KƎ4t-K 0@JbWGT:@p`K[leB0g]LwjZj/$bV>LThDЃêUsrKEn*k+@p^6[gFR1p<h ŀRp3J@.;ݗؠd+0Us,*O?x +6!EH&F/}ν 恘 m? шR%J79Z-u]_1Yq Q0l^[҇4766֭|k:>aԿ:&<(+FL,.>r?5&ְv#."D&-5' t$ꇗSN:Т=b3h?6-|ZV[͙x#\?pe ;qypj LpvԽqr|02@`ݧ?p>G>,A*F4F3(ZM* <"d`Nm U^IN7 OR :D R[EḍS ̙,9N.lj%Qo Qz}OD@S,Ri&)*%MyWM&6J& F/ J)t~GS!zwH4"QH#|-EkP n#5Q"֛(1`A 3lz=m7芕?BZXsdֵoS0EI*nԶ䧯gg}:h$'O&" Z)̥:36.x}!vZ bsR&)坰NfX[BlǓq! 炤Ջ|&DE!SlLD$$dWm87Cm j8mbfiK{kjz-yҭt! kGnܚ" c#&IbϪEU $r>4fQy՘:.jhEou 2!?@N6gB*޷:W+bͻO-$&c4>O2e$e7Ȼ0J!KW~E\%(0H?4||[OYWu@sXlٗbRyCyuacc#yN/^YY>&+.fDN\W"=rl߹ss$%C!jk=< gzPҖt# 'Nܘbo6d;t5((TX}͖Yhc[c-:3H.઒ 1V]a׃Yt;e6ѓ_WsmO}  !Oɾ<(?tdmp]R"Uq/J.z<{B*GDZW$Ji8j`ܜebTpt6BRZꋬ rDq-cd2yWAɃs$=\M ,,YljGq"?| ?X2-~?QJrnYh1Q@Z@ٜR{ 5BG iA ott:7+Y.{?8٧%n?u:SUTTmժo̜o X`9}8:rY-BVTmWySH˺{[${=Ki+"2inHgy'25>w=3TVYaiii"i&kkk+B$q ^2$y v< #2n.́k+N+_&Ţ-\[Ui=;Ik>H)kPqhIq:eX@&J3 Y]R 4hī #!2W Exw!0w/lFFM7tg?٣$ʘ#h-6Gg c> h4&:5MlJ+.KV=#Y")B )5|٤~뎀"DqeE5[H]{2QMAIğʬpOy"z]][Cq=~0ĉ%X\B_,a+N*nK[b$b$m* *טu];`!oρiLs0sk[kS9W\T|&<@m 5U2+:7ư]SIÖkE)([kMDHH_X:E?v%euR|cPY;VM^8F% Qr ;ޔh4`6_ioŸyvu3kűcǂ8Y׎=È>gvڢ~!|ls3MQ8߾mb-*:|O?ktS4sZ[a˾\C) Q9Lh9 E*^L(\fDLL7ThT5u8&ǥ 5&sT$$M3 &4gњEOyt K/ۇgԌ!x^I~ ܨ)jvFvIQ-UçŪm#}mom&63a¿@g*Jjz&19*ސ IB: ehg}PH'RlM:QTU*aY)/*!Wil׳V`۠щOO!5tYX ޣӸkoxç3d.8@YMk_{R*s =IZ'mst>뗋ĥϾuU.ԕP J2؂-KjemaW>mZUdkc!J4BY$}!WB) 4AbBd [_ sBd fu=PKJt8qb ڤP"okђRמǟ)WT>Emщ (jH۔rErP.gtrlYGLȎz&=! HG#)70tHX9wP5(_!#{Vt85&4QhgOad1D*JD RnQEЦA%it-iQ5 t05;N\pGUBLЋ?e0jmm=3!ߍ1{;l 7o>1Q,6sD)c.">τ|-}Rˤ߷މL^jN B<"ff%GLhZr=l"A)(*uđ[Ƀ I t F^Q΋ptМF{=?v'Pl,+[1)A %Fys})־Wc $wBuN}9u|I^M+B:uR&OA ZxSq r< @'"(3qQ<勾UA\!% HNS9qOQ rT96av@Kwg$I;v9Ǐ}>ΙYYY׿|&svH ? }3pzعcs۶m'KzK։@]]k-vr;?$H ;dY$ "p8#drp)&2 TWRNGq:C4M$ig3&nD~0M:Ĉ4;)/~ve!>|ip=}VGK0q); G#m%Bt`ԁku]c̪|m9}. oc@OfwV/> ܥ G(es%hd%Ji'A"*e@Pq֮ZͭpJXb\HDP9 z@7(J{GD`#Gb\IfTQeܺXf"k~}08<ؑ1&ndi~06.(_ q _MXA6)v*|(-6D,|U_ׅd6IBZV} ̛T-(2)t2PU٨a}CCEm6\O|Z 5<z>'=WWWq=9_.営H9WCXl۳!JSPJUqxX',)eLjTYLƖbα7=gd-SʡIK&f) -X)P4 $- @&qC8%Y`X>:a(b N`&ڭއ˯Ikq! gkR%VVVe)$qEZছnr6@é> _8`XEե yJ$Ӻ+_msm+^ U :teYV[iWݯ~Ϋ߂7,t|G(TϜu '  w-Toਊ 0&n'EHFqeٗz1"J/+X`0`},$TI2]Ln)+H/ueAP1zQ9P.$i>$rsKAѠx7=4)iFWe1ʠ?ډ\rɹer"6Hk]]Y2(& JZ24P {P4L2W=/L9ma_Kʙ\_[4ɉ =?#y/}Ki?ZiS>?PJ; |l4m'NYq z}X6B`[F!bY=xм,J} I@u(UT*Bx-I>rfM 1 Ӏj4X>H8"JXkAWlJGɉkNq ^E#fV|rxy;EUE1` TzGDf,V7߾a @@@W~z@k.8m^Wɯڿ|p/FD9?ICLhMh;!iPQu`]kcLnU=" :2[EԚutz2 E? ,^BSp8~l'N@Y:iVMR29It"|­Rp3s8M R E֗A"LB=*T:ZW#O.~9nl$M$)Ք0E%4_"DJ@ 0ý1r2H ݓwA-nSBL,64E=8[蟬7riX ưZL꧒y!ǮJցҵZk}fffw܍fFSgtIp뭷cb3Xhhԣ=p _}/x9g9F# 1lу'J'JB[!7v8P:Zrպ•0*.+#I90vq ՞jQi@k6iTD2it2%6ɻ,|RXQ1s(1RH>o=< 0 H f5/˫fPFܫ&w_ϯbtMH{4&(Ah܍ZN5Ta| R)L`S)?!iL>3Rĕ7ke7ڦd*17f*Ngd풒)td,D^eɪA@DPqFT I={.ySgM>W] ҽmllI:}uRh۸x{%L(o_--{8Sn@*?MEQWFHo.",:qȑSrSp죩Ѵ;p4`lTF qNʵёd$=GNZpـK}6&ls!<մEMlL]*ݐs^kb*%﵃ l 9O{tR !S)hqII,Ծ[T" q#s9bΧw%*r[y_#E J'BۯҮWNd8ԮkUB`Gsrt: ;XF0ie˜n'4-XGaxI{' I*,"Rs44t!Z(PF>R Ɵ|h>7j1Z>?>޽{C^k KзU $p[@zFU#K hKj܂ux$MQu~&,ܟbmHzr6ӀϷR8)Dl%;" z\zq(RQ`{cr c0BB&؉ /{ZK?9eR]k]Nڀs8$l6Q3.><٨4%њnbVxܘr%ҙN$ $I0 IH^sӡySIEKrQ)P 4BRJ3tuG >t<×B=,Aу8Οgyt^81J!sϸĉrlQ1D˘Z$'& jYKbw*W }-@ d+@{RSu}G)cb"'-jrj VDH[3 T +*̲VаeFR>xոj6ʤ L)pEq(BϓE(j n̂ůwjc׮]A kDx_Q%&Kgqt&|~x<{Gڵ;Ʈ];h5 _2kZ%U<iDUyތb*rmYO-Y(SPY\ >}t(13=%W}h5aZ=3M`gʲO]dZ{7xNjln$`66`ǝwyy?ޫMIQn;}aMmd+G9X iT=E6eH8 )jTz-5tCYe (e%;zkpdn0p m>is/ 9P 2pynU6v:S)YhhVp6jJD :Atg$B3M 5DɁ [IoE $5VUO\)6ކeRL`M3(X>b&2T)BsnU2/@?b]&u~(ZH,g(P, ^xoz]x;'ErǏnzH_4Sv6Avvc^F=hx{6vډنHd\A&ZCׇ8k@B{w-Rys=*n^B)#lD S bs@ (Hi !<ԣ(:4ќنn޾?xs@g~&vg ǵ60==믿8|'٥kgg|[T& =3S0㽹cǎ]OxZۖχw܉w݈c67X\\DjĂߌ~ԻR ?@`6l989 Ӝ=lf9-:t,(ߨs:f'1 c8k S+Yʫt*ݮ]^FOIi_'ĵ aBΗDmZ{@mYY+%_8"vq(eG MrP@ԮbOAsۗd{!|jTq/7ow?R0-P:٪Zdr5jZ7tG!7 zbC" rPQRAzȼ@ ;}ڿ~R5ܙȭ}JŨvG锴ͯM2u>9mb"('?acڈH< j`&lo/g=r ~gt4^nG={o߾uB נwmWXY@6, f;H7v]qB[4\W -F:Ši4(2&t@ $iQ''**,AEhB`o &V;`n.ػMwjl:lj{/=Widdj6oQ <=zMoz?pN5 qivl?jP@Hv=74g4N>WFx(_%/y  3q&կxʥ+$!XDI y.nNŚtK$*VX(YU=D1{Z"m"ﯣX#Ƴ dU+msD:E*N(>&R1wAHWaElWZ-LԒ^WSnZ y?n|/U5WHjmM{J.rU B"&#C(dDLmRQTBt>1eD9e [ TB& y KBԜG9Xcڄ@W} ~&W+[;5 ["n֡ʣ)/,,sE̮1KX+FZVw$܇݊|$h{Ƥ01$ "he%S]23DLO!ѻ$g*+&8u4 XԹ@@MV,L[FhxB Z?.k=Q + đFǯ={s0s;LNx{]? ;CSMU&eXND9dY׽}wq:NmStߦV'ƍe:pJA=FEkhi:uW~,9{>t:VWWiQ:_#w~O! ( Uβ; w k lYP& }a !n`|FE!55mi}bfjk4Zѡyu9U<0+L=K)K,=ԁ`)| ^է #H,M9UO7S3==lhpx޻Nx5Op]^`1O!A?amMJ4v rZ8Q&iN/*mAEMP)wi+u"n nڕ@E5Eh'Eٳg_x!ZhwťghYQXYYjI@}\d)mw/߂S<toF{4b3!J38E>WFx_ oxCxСCa믍Kފ@v@[A]y O@j+ 夘Y*@Roͼi`*dn ҲLO{vжz%p.$\ a2Lp8y~p]C[G,- hUkS;&".VPg1&刳Vq6ITd R0iĝ9y8M.B06NDҊAh cfzADX\˄p<$>6PIm<"ʝ uEɒ2dnɷ(OTi$9>ӋXr>jذ-^ EgKɆR?B6 ?: +knne2QKie@&ڤ"*ӃLA_'AEM X(K2J*jN:^(|Ũ\^%t( 99.K@GhgL KnM!NZ|O,a҄IS)Zmav~J3[H,½uB}^dj46cPuv19{vO{R^~} o߾}:q;DX;v ^obރ?+ќ2)KIC|ʹq28'mJoiD6qWz66zcgu {jlQƀ]  ٔ|̥Ya,)ojtoeP̜vRZT.,XƵ:(+iy20k=)5TuCOd7o'WI6lUl&%TcTG hWQ/#\/#dХ +9SW6 &8uu9?iYר5&r3(^XR.hL`H{>m?= caa7x9-A}t,9!`Zc-{%9nW/շٟ3L|c- MFek1$Q&JuMDQ9O*((H0}Oq{z%L,"\)D * F+ y7LDZ\^+<o@` ^6JC6UޔIky-Ϣ6/w:|<סf%B~:)pon;9k96JVAt ?,{b*K8R *` +B kQF,am(m R&kmՒA0ی1Lp<4pꄴNF>7);pߏ]}o~:V;Zz@?iO;k|>>33}h6Xlf}}!E雭3 kk|1bgbRJh +gM%TjYΠg}l v*5L*Xv qJT2 L[JzY{, :L-ŃD&W)DeB&^Hkr|+79*bM6:7}V,LI"JADvK|-Q ?A:>+?dF'Cd\LP,WekPQd"@*w2 p܁-{ۗL"~$+BVD)_!N;( v܆jhSI X2>Pt~?^|%,C <ݻw#MS|_F;}0>h˛_֘)(q[>^[ ;T=O^1V̝UPRT@hœ$0ǠX&vO7{Nuk@ٷeY^>,7i}5v܉Wz8UR܃(}Od{Gt89JP47x|5_rNc9ZMҕRx+߈(dp=nxמZ ʖQ ҁ#(Bȡ՚ARѡ5JuZ,aXda} q,<0U;ItQ=Y8pǗJMD!*%h;hX 4 z^KQ3Ơ4ŃMDdEM#vwh1=4 \b%U͕3!mϙt"t'f4AT qH{iʮԎ'0 ֦&Tt*+,$\б|:GS(Sd)l192P PRJh t³oy ZFȲ ~^g}6v܉n ǎz>{gtz};[t}و@rwd`i୎㸷B-jPF/.1spBQ8C(Bt>o{u{E|c;SZFFIpnY!>z?=ЁQXp=6B$xߏ92ч|zf >zo3 LReYp}|u J1H(QK>M;> zrO-y!J^w|KBԋf!⬋cm0@@ y80XPK9Űe1z.@vx!W3I?D:nPPUgݳOŖJTK krE(WFp.SF䶎UFcWit!Qp(=ɛ d{Qp_Ldo6s!)ʢ rV ] t~:8AIN9qQKtWCNCȐrt{}>8ࠌ*ҔV=GŁCܜPnS^>"8[@e^DR'*\\`CtHR{u &i5 t.;4wGDFlOa vIۚw#~)9'd#t9."رwy'8cЂs6g,؀0[>-"#T]H]G+Ty*i (N-bIZ&rEkkK)Z+NPRԡJox;IZ[y,Bh MI YK*>wE0 &/ⴉ2+49E56v[׭SI\2=Ap:t'SH&'׷0⌃W'6N^{Et6{+JS{rHd@C(Dj:nCNQ\+G7h:dHTl9r=VUIoT0Q<-.PdzDK/ʡٚ^[n#,!9 %f>gx53˲D'˲={ /ıcp-H}}(k[6v؁v=vlz0<>Z֢(SF??>)nIL9뮻}_۝G}30~< `U[@y{oe<\}a_U\r%}Qz`q{e}(aECG^2WmQ\GSx\h+V%o"DFCl-Q tSuy SD3 Q7NJ3 ܥ*xstOy2@gACc5$&Պs1eB ޵3|k(őQ4W1JREb&J)E('|u~W1{CV>mYCO2pڀGD2UC:"vKv X6"g[{@E%Y%!%a M(d;Z 5wkF+5T _u񏋿>g̚W;}JlWJAA)ldC,J2@)k /qUï' ]{3o{3~{ßFk||4WE(=QԻ9. ۷oߎ{w)3>i*OZtL6ȼ:Ioח&wX _;>0i#q. ?68qYgɓ'? piiu5z|R$]3]1ҳlhڋ"{mwMih=9YkQOO9yR(?oLQ:\W5hzrB#VkbC5  V\R7U 9m4://JM?Fl +QY[ jA=\K5 ͥ/TPUDmBgMnPdO=EN9DQ 2-*2Al* "D IMFHPL() JOʗ8o iZhw(%iܧ;{ %v=pt-#g.WQLQ>tRlO8Pʂ2&nlLGpR񳏽i6xb1]F_VGA݃y˲x۷]w݅oęf6ƽ?Iw-=-I{q$ѿeOF_o%^nc۶m/}閷xa4(}'EuӕӂVf~H&.[~P:̖oM/cVɩ|Al""LOO3zcN}fuGo h1Z$:QNiVpEͼQSb&r|08M-zإ歔3,;?UM΢vq4Tn:YQ:'֕=㛘ֈ9(n~h8[ QDI'fњNHK$9ĺ4`ɍ\[UҝOc(>IZ2qWS<@fޚPoӱ9t,ϵ֞L!= N)dG'cQT[I+&I%];|2&AڙG{jqkg?06X_>©tA0/ Ͻ~,Nw5T8O$h4HF#nw܁믿SSSxSHr}vVN0SgO?S<~_}2Zze)ncǎ~G6# 7cyE{dOxWEh=x[gzүWc`rW ~y,ҁG[L n~ =>L 481bwzF1 RLC׌[<IaK'M8EY.mb> Ib膻بIrͶ2y-j̡,PD(%(2 4@VJ&%y' PPK9 Dns!*=-5l`3>rd!&v7Y>&N;p*4Ťl J8@s&E?ӓ2aKTW2xMv 6Oh4cf P eL( ITF+!tH3Ӛ]8_z_}9]Ͷ|=JGt_kO]tvڅ{Dec3GǀII$epM׽Yn?W4G]AVS> juoTt~ J-J7HI6AC9(QEP *6g i`G 4ΰ@eƉ;D>׃ :8vq,ooJ)_Dz%T2k| xvE آ@`\fUM$Ф༂\{&*:tds!WPN` ;WZ7ʍʦqj'KWP' 4,9@iA[ C^&SDpjlZ!%)~Op7`J+m7C/}2y:DGlGrup/FY馛aL&ƃqoƓqfmucḺ$2Rvh4n򗿼a0+qQ,@WNx/60S;1Q$擟䫦Mx_uUx[{Ɗ r/x/OcmPi0k-ү}z٣(6ʥpIznϐWM] HA.p~%8y׬M (-.w3 ~& IJ kTH{pn6peY&y6OWk7X$&%O0L:{Po>*o6RQrȠ s DK9S>Fem[|2kA! "(W蝼[JS0a/p*_3CT@^6 jd+>FAϷ-?bjF@{ȴܗl% &Ih|^߿>33K/v<Ήc}8O}+?}?2n=i O7#1XXX,:t#gތ4ߺk=A`s1<ȃf =xW_(Lw /b3[mc&.I~E< h=B|Ҫ b d/! +tb\#B. kڧI:}emb|`E& 0^ R7U\6R@}xTR7f $I 2[>WR@2 NHðm*ڪQ~l,m9ŕ}MPp c(_= z6n!ވjOP:Q&DU39|/B8E/Q*={ࢋ.`0׿u<g~qMbO"m6?hN!M"ōͮwݿ3eY?wzQ} [OSׁnw#rP Uzxx[ǩzHoZ6Ç>pSiV5Ƭ;n[^یy{%yeY/`y*An3^xLTd A)=kjpe%C Gp'Gfe_6_'B9,N}9#Z"BGhwfI}O`yg` "c@u!RÑ)xQ_95A=9S;PlQ'PAݴazo“o蘉rql}M+A~-m$UArEPg=#h@Z6exr\Γl)WxynPԮA:  ¿;em0E^<d3eY"2osq|+_:sطo^aaa~ѣGOK-Gߛ4ƌ[΄7I!q]v!Iu6 d&'Eul6x`O`8P)N)wzȿ!`/'n$)lރ4!o{ xرcֶ{n= Lǎ =ߌԻR ?BҞTوĪf+qRb½ }[u E P/_c鲰 OHY ">' 0 aԹVTAй5 ,~ L"L>nO=FIQԂAn挂+!ދ|TUkVJ#IPR+e &8CY }D +O*͓(S8Fc`P@Q}Ͷ6Vc~,,<, դ@+()/x~{kr&ߓQާp׀?KKK馛pȑ){k-كqa,.. 7PcםNgYqͮs.ޚͦ=p?c\io5q?)qKdMoեu6 6g??MD&Go|񒗼; 8p A_8(:_wHEW"1;)BxDQn ѕLl(0`-{iH S+Y4yȈь)~ < Lhc;E6Mө+eYGV ɓ'<(u[ȍ: T+H\͂S*9HkLy*+_G1X gVUV=_7 G+f J?V!J'h.W4oP<$=s$d=Z@B*j M $>F2jB4 EQ]LP ݀r%+<vK|ӟUD- DmNKփ2H䠓)r erַ)_z8@'G`'O* @J dJlYƐ4yr \ȼmo¿yw؜@5.*?I{Eh6Cu(p=<z)ك?袋xq V>Flθh[Cs!Ÿ]Zk`nni'n>p;ARQ /jN2zX 6g>/WQK_R o&q}[9ߵU?6Pݬe7f eB=; -qn"&{Is}`٘EeNPJ(+'оAl՞BVek{mBΡ=ݕ|n9v%'*Yfr@)8sy_vSi;?jaqi_#bft=?~p2c .b\q8쳑$ pmힲ@Lعsg(=i/F۷oGDY;躏G9G{G5y] T| 9(-g 6Ӽ?RŅdP>N6ۙdp&4v/^^l)l@ '{UyyP{q~0#"5D)[`HlH᜔Y 6&g]}%{{6P(yR"PkJ%[$ hZGh W8E?}|(Oz*$DV3?RzF2-q *"rJ'])̞u1Vݜ/5w6z'&)Dl縕ʹ@ V7I}Ĭ1L{!)+/Ikŀ}뭟,3JѨL 68q@)Y+ȱG1|4K.5= ÅGQ]jNf5EܜF_c&:T LkdsSqm(.*[Y/LpIܺFD*%LsC' LځwO{X>/G}/^v/`ekGRuV/^ce]=u۷og}vGw+I}[NRJ ӭNN(bp>|_-ã b6Q(o^| 9(Flf4JoQW9xmgE~|ebf[ڼg[)sq\or$iVT֟$}Q TO$,oBI+[i׺M$@lCdAڰ˙@D(QDY+|Iٔ<u%B߿Ǽ B4Q`MZ8kWhRq!m- /7q;5nPQ[Q( h)d%DqL&6x[{Iz[d 沈^rL`DIE S[_ >9 y6O~5Mq.JlZ?>Ϡmc!I#v^BV.9r{/۷??q}~sm,--ayyYvl7NRŀ'kkwdm<^c>@>^V+eyypȑXVƁrJ/;n em,?سgO㻿w_z饳^zۓ~4>^MkƁf۷yѝv% |4" G|}K-eM>b}09Jy\qZ/9dty@!Zr+ @@:kֺEDun[@'-_~PHiʁ_JD m`-A:b&99>bRyvv%7"\"0Hݑ,ݺp!'D R4 n&H ~'^ Y0L@(j2#hu%Q"$EA,389 TdF nЇ*'>i6r9 .w3PY@y~Y5= ^D } 揈|z< 1Zz@?iO;k0dgffHJ);v kkk:Jǔ?]_wǠV2#Eu9YEjarLЊPndTHΧd}ݳ}Y k+e0 8(Ñe[@)`QWZxS_zLv%Nߖ ~bY$3쟀- Xڻ.G"9&K@lcP>0RoSp!R-kwwٻϟwOQ M`j;,Z@@l99tWkBd](-)| V]8y&͑DEi}YgzO{'Ls})wy'vTF|{V_'~k~ _ðjPd0DQDfSܹ3K+>mۖdYFFEnk5= 5oߎ'O0`jj*d&6666m۶aHVWWmѠ(1[n??꯬ÙDuW+q?g0&QzJuUW [Qȏ/9~hL}ح_flebQs@2Y2sM Z.e.auc($Eg&%Q/T} R''ɺdh5;{Q +R{S9ܭik(ZRsv2OQ[+_~ ?c_bZIc$NnD0Ag҈)?V&2Vi#Z/7!S1- 2|ދb'-Q(::Tmf]61 z%`KhC7/`AY@oZM׼W>$uq?=}|}a|S Iҥ駏#s/ =PnbY ;`vܙc(˲u{ݓ:Nn:kRJ'I:Vi)}+{DP`:sPչib4CR8k٩ &9 )m{P h7V1g mi ws! @ [Erdo%\$itݏ>7`i+u^'hhU|s nh[ǽk,K,..~VWW3 )v N[KfiAœWFTk58mD :m .@HP$ OJ.e WBʬ ":P x/m@e:4EZ7s|K_7߼eow_;Q`0@bcc}݇j7f@8j̿TceǸׂ6 qq >)N[,ԣ3EQ\ Lػw/&j^_g+ _rhD( VlbV)Dn fqF"^#ioۭarC"ŵp/, tWwħu2 3[WRo:~lv@hHjDY2y1?e!r0(-,ŢT&ʛxC*@,"$HYYy+< R5AE E@zZ3![H[PD $:yu2]Ptmn'}Tr93˯|Dt:C[kqwCCu:7^_?q 7|enT4y`2S#Nz̿7'}@>z`00=~]Ո 0G "2L9}ïگA)Fv=V_קUH3=줼V/~ w57XNz60QSP1s\)EOԴaMxNsuQ59uv n r"t#P8s@M우(!j%}2&)pHJd-Q_t* Pݢ/ÓΕQ@dbX/JpagexVͽD%JA6C"*@q՘'(lJ7 l0 E6=fŲ*^+z$p1{^=/{rxtߟ@mBR;S0J)ND8~8>OceeehOzzʥǼx>5|d4RzVڑƁZ RF[u0p F~蓎u8ch}\}3Gvs6r<oa#6SuI*y(q%`ҁq_یAr\/F*)nT.`CkLJbJ0XBގȭ}DGM@E Ōo(aΛPiQ@8F^i2&&r63j-v^U׈zԏ6:ٽKަhDh"7H0leRWkkLQs8B`P2WISPQ(NP?Y$&Fl5τ TAUfÓ)o0XO}Wg-li+={(u(tN1^^8Wy[ZZkߋee+(q{}SŃG|OQ8 ~yзlQl9'x@ݽ{7~wwHlfcccˠY[@߿GZ#N(kBʲPqCWHqIS lTkdEɭ:~EW# נӴ,F J+8 8KTmb!Y57!ɾ'8 x6:HTD>qœ(衱1/6@_ [BlbpR )%'D5ff &JP949L-?ԫ]$p13V5NYW/~N>=ߊVO[kxm݇v:ЪGXaq*nߣoRgR+,վIˣeb3 <9?{ 09J[ߊ>a@{Q_t-hgp^zvi(K)VxpeEe3RߖtvP"cP92J5|)\ Og{[Ї,21P`mpѳ^C_ktWLv%-TVut?9׹@S { :*ikx9>%hΝEu X5"us: IkEzd#கɈVш鶽ڳX;zh9t=*S`xBD*r9JUL/) v?C//)6Oo >mo\pk=Eɝs<sϧ1>JO=uQ0̐c~&m蛁|R|e& LE_3X6El[kgQz‡?Ft1nOV[Ɓ}:o{+C$ˇmI}K y< Ng[}#dY" K‚,Nv9 ̻}t9B+YМb7)Wr[($AB"/0w|[ېW&蜭HjI Dlq7:(>:ְ*l*\6X7y{QDe:)˞ݸ?%&vyRS&e@:3 e'c'_,tg^$/z&Ҫck_Ov;d'O5\šj97$*7^s5g@l\w3}qo0͢>Q)g觋i~ E_3X+Ƙ^W",7?ml_AZ`M*ih-lnD%j섴9lЅRؒk1 1ci"ebt)uqp4THU ^ٮ"ZÉIpAs}zԅNRSz79WIB]6|Lrv_בXIA $Q晳`!Ytە*FVlģ T`3PZ{/<GKWIDATyv}63QhZzN(v]p ׾}?|sx`f(TF'RAl\{XOIf@}OJcOˣ~x.b0?c۶m%,..UnQVخ^_hR Q1QrXycp Br`0[m ˍBLO6(VrZ9>O7eQh$ ($9 kNg%ۆ=o"'Q#9ʐ4v,qy`1tT$=(zM Bv:}G|yAUO͟kdk DI 5XG%P ;"%{9(I/]'oi3S ؆'><\ꭰYSP@s~ ߅}^';l6)rZz?,`~CyY'W0ǁC63Du '펂h)&.hڃ[7wkjjTWjYo}[X__`08E2|@kmiVuO 7@i`b'yu3g剖>p(t;܆* - (U@s8 Íd`Jn1ĂSBnDmض<,=p+(bFw )Om>Vq?JPqQ4y;)"@ -d lb㐜W,D&EYܦ S`EӯX0+udTY8BI,K''hOObPʆ9+*ns*Ń+{P nt^ 6w-':y^'"8p!uroR5w}^s5nS]'pzsqmYAf}ZzI < Hx_h x{w('yf~˜]{}p|yW30PN\8*W-ϸ0ǍtSDτv+kkhXa9TԄR1 籡֫?&h_+ /ǑAh6"c2dAF &h,\B,}l^E=׉d1J1+4r?a*@JC!JZ(|ڰ:)Pc8两ť()Fg1-mF#N;(tYI}R^?5 IPpɓ/bp33vf:\2Z:thgkur1UqJG?ʳlbmx]g3k#%orxN+z[sv^d#WMq6kA5HѬ/k7֥6;Q٫T`jr< Lg#-.‘ %QVXk}˨MAfekfN?Niz~|!skz Naj qzEI S{DsGI<#wK_w ts_w>sD>xfK K,"nw}u$' T/[x[XəzfAt-:o^ʌyaEЏ}gs=(z ;rJ*|_K۷ -.5[=Mn6s6ϟ_?vlfF7yu 0@*Q*B s*/Q8 *"Agz~[/)d{XB^)p\0&( uZxxĻ> jNWˠnތ bMQssL 9/++5x,z:+~͔ۣ "_ɕP2B\r| ;y !y.ZGH6>Cۻw/vre|ȵA4v}Ml43Q:y[K0ERǤ=,&!ѽFJbjl6!@ɐ^7qZVxHǣQ2.0i~%^)Cmszq| u^Ezi9'.pɡUX6#s !<%q /m>g7?7]}(ntugTJO~Z#<{?h%dk4kzxqwz s)nܸAp6ok*]j9Ɩܮ]mCށlh;g}dLdLԨ&5nB/j(9G5/pu4&V~|D5&8f*>ރ#Yim뚍mw񅷭w97K| X,לߵk6mZc[";,w[su5D:y2oK議>"6c<ӿEQdk(l}olYQ\o{x2d\TmJwM~Q7ABFDKKԳN). 5a 325ฺMH:$H㢂XBom[k5r<E {[vL4MLL׾Xr#Of6d1Ycli6K_.a/D)G7p67 &XM1mW׃F4:LySJԿ=. l<]:9/GCF&ܜUq]@w+4y]Ck))3o"TwأGԼ^x9\A7 }>+yK/g9lUm. |Ҷ~{˔Fxlk: [a#C:JO͸WTlw 馛x{ރR˗399"&H=I,@F%O h=Cmbߔ'u7:JG :y:ݞ끰6]FFM{K4'#N WzpLGWڵjj#\MȆc+uH#p{QCqF<.7=SKÌc5a18d @t%oU}˖316 oTpSg1vlW';k9|ȵ.1[d3[,*l@ 6 hoApoꪫXVKxV*vն.syYMrG&O h)rۇyfF}ґ:^|AT T:v@PR2s4M@%#Gucڵhnm/PQ qsDq~lUcJ+ $B7lTLm]SRoM=:U1z1 VGHQG5'W1uT7QCJ?4+_Oe}ߩkݮK~~ڡT*߲,5뮻i|,[`}!6w'%a @iÆ p/2;<@GB>(;Ecuwg WqEkF\ xzpniiUxVյfcM}SE]Q*GG<׃ t=iN a*%b}JWO+=&d^8z}E`t1^9n1))}!̘@/|3Zɀ]Msy/ ȵ*ػw/z+xC`ӦM޽e- J(='I= ݻ[_RoM#IdͼE/`ʕLLL4Kzߎ?|󟚑,Bm"TpKCTo>B:v<-s5%}Duʳq=ՔEF9tgp"J/NǐvW>}bTm,2n|}G+z#iw#6x=K.#T|T3*tȪ7PRZ덼ϝq?э$\c:1z<kಕ=Wwv+ 贴xçqE۲Ns;L!\^N/7 P('S1QwgDž(QuǻA֡C d2.|ؘ! Pu50F aZ=oy=.}sOW{mBr>0 "֮|;|S3?#Ҏ0?].cpppF\.yfkkj|IF議E1d\cll3}}}_T<ӯ:&''U]Sϊ;-˪֭Z^ԲUW9?,V7@էqpPzvCmfb̩7ޢ!AyyCBϦ7*D2/5P8ؒjrrQ&Q!AјBAx?z 4-hj"ZA/~ kkj$O fL3mZڸq5A\/~[jugzNvfǬ ԴJ+k~ B9uD$T(q2%ZPCEO"@+v dTp Ƞb"u4y9\zL[*G/jW8DRkK?h;s,7gɕyO}=Y X4}7ut];i|,/Y6B_oXq-=6oiٳWIJ7RrUWO)6F,Y]Y}~=oxHUEڷD8vѤ<=ZT&daf(]F9ESA!BF.YyM*jp 1_S1 "Dط\ϸ|AvQ2}@ C.{7 nF}ArVt֭.jmM-,Nl}jsimkjjL?x|}cpߜw./U'O˵޽nښœ6B_$t뙞.a8 fzzzzؿ?jӞ\iymc/o|ut(?Ik |=(AuDjyZ qv(S\tak aN5RĬ  ABH:q ~H6@uC:+9>'w%r(6q<=666ƦM%'l /.7Lt^WW|Xˎ)o}[]t-}rrZU:^it]pwݭ=ȣ*B6תEQ>O!KX|B#?8+.iߠ&$];-\ץoZV{ﵶ-İ"bb37Ap6ckǮTرcGfPy]o DR>ZGQįkNkkj`F苏,3tqСO ]Il檫ooZز {۠UXFJ9&"/nLh}(|WTOj##q"r0"W\ѨP8^T5D. -h4Zg؟ )JC2Rzw u8R0i|&_?U3E5XG\^*1cry.#jmM-,lȘEl&7nA<G_8fݾ};gʳ""a?CaP!;32 s"e"BW)ݧ=҅M'kxxmԪƌ]!8Xq]e'@ \\-A"BVPn\gɏ9̿YݦgCZ{P`hhhwq5d[a#\zK-m֭W\W\3=3#<6^7{Оx"s=ڱL C Zحأ PM{Q0r DVfJ)bA9QZ^Q DU=3@5PnFG* R8~_Z[ |M/n^sAx׍;Z 룿ey?5Dnq"Kv޽vb3;w?)\r BV^=3=nӤ>^r޳[~ʁRj{z =FM Pȩ5QHׁh(QZA Qc!\S$b?5\PBMMNL=<SGRCLFEzF]FҢ3Q=ϣ* bb.K-o*|HγO\ץACqSݺNnmM-N&6B_zHGZ xJ|>v Ðo}Zfá;vcluq~H/o܌_$ CjJ@U*{yK.R{x( yHJ Px#iu7oU =COkV~w/wroPݻeEsMϧ z{{1sNnvkkj`¢6B_b 6 hoApcWڵk+[r$qkKˊ;O?`#SQ7(-G "E!<nPC>qu]~©1 ШlP)=k@ @pTEVf'tSO6 ϘZ\'O:tnbkm<ҷuL6B_Ȫ'U\ ؿFFFr"+.kz1999c;`Fɵǹ/祟>+? 9$"ULN%ߨ< PҘ5>GNWHd$x;?{׮oܱDmnO"˱|:wmmM-[Xt/Ad]IIPڰag0O}SwyTUm~eYnT#t{Mgx%hT;g5YV-HD0kahVqu}#3HΧ%jBnY!կ%rY`#VXB_Hz3}F駟~gmT{k_kZhڵc6n۲Ku?|9B05]VacP"j~"A'hјkT/ 9\O9gH󽻑_Mo8΂ښurm1555㜎NnmM-l r_Rl&ݻw[nݿ*ޚKN`卫8Q>zt#e0q4D VQa">B/Ϫ3W{9nBY|ylٲښZ"7}cоAn`dDח޽{œh76[c]ɟpۯ>x'?>Jhq:œ͜%&ng8yy&̗w0uvjRchGÒF'fLϛ8J& }~Wn:@:oݺuvEDY?w:l%n\ˎ@%}yzCCC-7 >kkjba#VBZz3-={|b7uLҗg?Y4=a( rY k٨fcBJww2Vo}.1P`Ŋ3֭[%r K'Rcl9cKuk׮MJ,u![lK.tܖ-[foRA.}e,_v4Y3oEfE-K1Q'߿?ѣG3x" '}})bOB?TZlfgs=(z~-b3W_}uvO<~۹uuZ݌ʛRbȊ+1&''9qʑZ,l~!-6yZOF[SK6BXD2 %=ӽz^V_}c-;fnFwQzRAiƳ"td'+ߌ_MT_+V1vAnVkkjbIF ۉf1_}SljjGyϪUr1뮻%Tŀ% "t&{ WVJvg=vjc%MsgDB-[h0 /O~f5D {OX XB_{\fQzqƍs~c4I+sAAR񞝈}.uNݏzK^zΝ;[N~2ښZ" {OX XB_`B^x[f)=mݺu\yM"۾}{s:>w]^pgpѶivetm.itX7F4&'r-޽ښZ"8ia#v+6ׯ_(V@A.odppb3{g,6b~>ߣ/s mlloښZ27{bFOt-Jre>d4 z>ZlfllJrLJmENjNـNǨ{œ6B_,=L2J/fyf1cl_Xv-rōXjsػ=f7fڵ3ykkj|`K E$t,6SJ{E###_WJ9؞gpe5mݺ{;N\g͚5-ur{Z[SK {OX XB_,=2Cl&&$l(mذ3an0%ь?OsZWU{v:ηct/J]RԲ|bb;G}:5HS-K;0b3gy淕Ryo81Zxnؾ} {w:λ1RYbE˺ x{& {OX ئ' `֭[Jf!;`ƍ(83jM>pl osQsin7m۸۩j-EYK6B_,}LK;g=Q Lj#b7MlipǵM7 f͚ch?1ꊰ|S-}hK[j^z\ٯ81_|#w`xڳPKgq===-MNNyfm5g}O/sZ,,/=t.6lvh~=\rIsۛ~~-ylX|9k֬i& C~sw7mM{"oښZ+=b1` }(z3񂾾/=a}_2gqFsݎ;8x`=:c!a8fcnjsY{2kkjаTŀ%R♞gbqk rqB˵^@sݞ={xuCvi3[oNnmM-jX*6Qzތҟԧ>m7=q_+V/gdd\.裏2555cn&b(c)T*y͙Nf,s%rkkj1_Tŀ%Rx6=3J߰a?a_8->9򔧴lcqƌ}EP`ttիW,Ðz͛7#n>Vy {OX XB_,lA.x>::sϽ9_3>p>|y=Ǿ>YjՌu;w?!^v?c<58n{b`oی@/|_*nK/3@;MMMQ. jRD.#Ӄ349-2tk=[[SZ,,/=[|>w饗~' Q_4.ׯmo{˖-;scӦM<#- ݦحʼnZ,,/⽷ f>u>N8Mr<q8y_\@P:;vࡇbǎ-ytЬʼnZ,,/=x.b37n)~DIr<uq38?s9~addÇ399;wd||cu"N:sNh4{bA`KL/ .9We$IV뺙߲u}̓'Gٺy%mkzmaaa,z$6ҋ7n $_iŽ=&I2Oo' oɨ<" * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "mainwidget.h" #include "application.h" #ifdef USE_DTK #include #include #include #include #include #include #include "help/helpwidget.h" DWIDGET_USE_NAMESPACE using namespace Dtk::Core; using namespace Dtk::Widget; int main(int argc, char *argv[]) { qRegisterMetaType("isChange"); Application a(argc, argv); a.loadTranslator(); DMainWindow w; MainWidget widget(&w); for (int i = 1; i < argc; ++i) { widget.openImage(argv[i]); break; } a.setApplicationDescription( QObject::tr( "github/simple-image-filter
" "mail: liuminghang0821@gmail.com

" "Published under GPL V3" ) ); a.setApplicationVersion(DApplication::buildVersion("1.0.0")); w.setWindowIcon(QIcon(":/icon/simple-image-filter.png")); w.titlebar()->setIcon(QIcon(":/icon/simple-image-filter.png")); w.setCentralWidget(&widget); w.resize(800, 600); w.show(); DMenu *menu = new DMenu(&w); QAction *action = new QAction(menu); action->setText(QObject::tr("Help")); QObject::connect(action, &QAction::triggered, [ = ] { helpWidget *widget = new helpWidget(); widget->show(); DDialog ss; ss.setIcon(QIcon(":/icon/simple-image-filter.png")); ss.setFixedSize(430, 520); widget->setParent(&ss); ss.exec(); }); menu->addAction(action); w.titlebar()->setMenu(menu); //看图居中打开 Dtk::Widget::moveToCenter(&w); return a.exec(); } #else #include #include #include #define TRANSALTIONPATH "/usr/share/simple-image-filter/translations" int main(int argc, char *argv[]) { Application a(argc, argv); #ifdef Q_OS_LINUX QDir dir(TRANSALTIONPATH); if (dir.exists()) { QDirIterator qmIt(TRANSALTIONPATH, QStringList() << QString("*%1.qm").arg(QLocale::system().name()), QDir::Files); while (qmIt.hasNext()) { qmIt.next(); QFileInfo finfo = qmIt.fileInfo(); QTranslator *translator = new QTranslator; if (translator->load(finfo.baseName(), finfo.absolutePath())) { qApp->installTranslator(translator); } } } #endif qRegisterMetaType("isChange"); MainWidget w; w.resize(800, 600); w.show(); return a.exec(); } #endif simple-image-filter-1.1.2/src/mainwidget.cpp000066400000000000000000000553071417414176100210120ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "mainwidget.h" #include "ui_mainwidget.h" #include "statusbarwidget.h" #include "toolbutton.h" #include "imageview.h" #include "application.h" #include "menu.h" #include "shortcut/shortcut.h" #include "help/helpwidget.h" #include #include #include #include #include #include #include #include #include #include #include #include #ifdef USE_DTK #include #endif MainWidget::MainWidget(QWidget *parent) : QWidget(parent), ui(new Ui::MainWidget) { ui->setupUi(this); Init(); qApp->installEventFilter(this); } bool MainWidget::Init() { bool iRet = false; this->setAcceptDrops(true);//启用拖动事件 if (!ui->mainImageView->scene()) { QGraphicsScene *scene = new QGraphicsScene(ui->mainImageView); ui->mainImageView->setScene(scene); QGraphicsScene *basicScene = new QGraphicsScene(ui->basicImageView); ui->basicImageView->setScene(basicScene); ui->basicImageView->setViewId(ViewId::Basic); } if (!m_statusbarWidget) { m_statusbarWidget = new StatusBarWidget(this); m_Vlayout = new QVBoxLayout(m_statusbarWidget); m_layoutframe = new QFrame(m_statusbarWidget); m_Hlayout = new QHBoxLayout(m_layoutframe); m_layoutframe->setLayout(m_Hlayout); m_Vlayout->addWidget(m_layoutframe); m_statusbarWidget->setLayout(m_Vlayout); m_statusbarWidget->setFixedSize(6 * 90, 180); m_statusbarWidget->move((this->width() - m_statusbarWidget->width()) / 2, this->height() - m_statusbarWidget->height()); initBtn(); initConnect(); initMenu(); initShortcut(); iRet = true; } setCurrentWidget(0); return iRet; } void MainWidget::initBtn() { if (m_statusbarWidget) { m_lightSliderWidget = new QFrame(this); m_contrastSliderWidget = new QFrame(this); m_lightSliderLabel = new Label(m_lightSliderWidget); m_lightSliderLabel->setText(QObject::tr("Brightness")); m_lightSliderLabel->setFixedWidth(70); m_lightSliderLabel->setAlignment(Qt::AlignCenter); m_contrastSliderLabel = new Label(m_contrastSliderWidget); m_contrastSliderLabel->setText(QObject::tr("Contrast")); m_contrastSliderLabel->setFixedWidth(70); m_contrastSliderLabel->setAlignment(Qt::AlignCenter); m_lightSliderWidgetLayout = new QHBoxLayout(m_lightSliderWidget); m_lightSliderWidget->setLayout(m_lightSliderWidgetLayout); m_contrastSliderWidgetLayout = new QHBoxLayout(m_contrastSliderWidget); m_contrastSliderWidget->setLayout(m_contrastSliderWidgetLayout); m_lightSlider = new QSlider(Qt::Horizontal, m_lightSliderWidget); //这个初始值,是让这个控件水平布局 m_lightSlider->setRange(0, 300); m_lightSlider->setValue(100); m_contrastSlider = new QSlider(Qt::Horizontal, m_contrastSliderWidget); //这个初始值,是让这个控件水平布局 m_contrastSlider->setRange(0, 300); m_contrastSlider->setValue(150); m_lightSliderWidgetLayout->addWidget(m_lightSliderLabel); m_lightSliderWidgetLayout->addWidget(m_lightSlider); m_lightSliderWidgetLayout->setContentsMargins(0, 0, 0, 0); m_contrastSliderWidgetLayout->addWidget(m_contrastSliderLabel); m_contrastSliderWidgetLayout->addWidget(m_contrastSlider); m_contrastSliderWidgetLayout->setContentsMargins(0, 0, 0, 0); connect(m_lightSlider, &QSlider::sliderReleased, this, [ = ] { ui->mainImageView->lightContrastImage(m_lightSlider->value(), m_contrastSlider->value()); }); connect(m_contrastSlider, &QSlider::sliderReleased, this, [ = ] { ui->mainImageView->lightContrastImage(m_lightSlider->value(), m_contrastSlider->value()); }); m_Vlayout->addWidget(m_lightSliderWidget); m_Vlayout->addWidget(m_contrastSliderWidget); m_Vlayout->setContentsMargins(5, 5, 5, 5); m_openBtn = new ToolButton(m_statusbarWidget); m_openBtn->setShortcut(QKeySequence("Ctrl+O")); m_openBtn->setFixedSize(60, 60); m_openBtn->setToolTip(QObject::tr("Open")); m_openBtn->setIcon(QIcon(":/icon/open.svg")); m_openBtn->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_openBtn); connect(m_openBtn, &QToolButton::clicked, this, &MainWidget::openImageFile); m_resetBtn = new ToolButton(m_statusbarWidget); m_resetBtn->setShortcut(QKeySequence("Ctrl+E")); m_resetBtn->setFixedSize(60, 60); m_resetBtn->setToolTip(QObject::tr("Reset")); m_resetBtn->setIcon(QIcon(":/icon/reset.svg")); m_resetBtn->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_resetBtn); connect(m_resetBtn, &QToolButton::clicked, ui->mainImageView, &ImageView::resetImage); m_fitImageBtn = new ToolButton(m_statusbarWidget); m_fitImageBtn->setShortcut(QKeySequence("Ctrl+R")); m_fitImageBtn->setFixedSize(60, 60); m_fitImageBtn->setToolTip(QObject::tr("FitImage")); m_fitImageBtn->setIcon(QIcon(":/icon/dcc_11_36px.svg")); m_fitImageBtn->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_fitImageBtn); connect(m_fitImageBtn, &QToolButton::clicked, ui->mainImageView, &ImageView::fitImage); connect(m_fitImageBtn, &QToolButton::clicked, ui->basicImageView, &ImageView::fitImage); m_fitWindowBtn = new ToolButton(m_statusbarWidget); m_fitWindowBtn->setShortcut(QKeySequence("Ctrl+T")); m_fitWindowBtn->setFixedSize(60, 60); m_fitWindowBtn->setToolTip(QObject::tr("FitWindow")); m_fitWindowBtn->setIcon(QIcon(":/icon/dcc_fit_36px.svg")); m_fitWindowBtn->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_fitWindowBtn); connect(m_fitWindowBtn, &QToolButton::clicked, ui->mainImageView, &ImageView::fitWindow); connect(m_fitWindowBtn, &QToolButton::clicked, ui->basicImageView, &ImageView::fitWindow); m_rotateLeft = new ToolButton(); m_rotateLeft->setShortcut(QKeySequence("Ctrl+Left")); m_rotateLeft->setFixedSize(60, 60); m_rotateLeft->setToolTip(QObject::tr("Rotate-90")); m_rotateLeft->setIcon(QIcon(":/icon/dcc_left_36px.svg")); m_rotateLeft->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_rotateLeft); connect(m_rotateLeft, &QToolButton::clicked, ui->mainImageView, [ = ] { ui->mainImageView->RotateImage(-90); }); m_rotateRight = new ToolButton(); m_rotateRight->setShortcut(QKeySequence("Ctrl+Right")); m_rotateRight->setFixedSize(60, 60); m_rotateRight->setToolTip(QObject::tr("Rotate+90")); m_rotateRight->setIcon(QIcon(":/icon/dcc_right_36px.svg")); m_rotateRight->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_rotateRight); connect(m_rotateRight, &QToolButton::clicked, ui->mainImageView, [ = ] { ui->mainImageView->RotateImage(90); }); m_saveBtn = new ToolButton(); m_saveBtn->setShortcut(QKeySequence("Ctrl+S")); m_saveBtn->setFixedSize(60, 60); m_saveBtn->setToolTip(QObject::tr("Save")); m_saveBtn->setIcon(QIcon(":/icon/save.svg")); m_saveBtn->setIconSize(QSize(36, 36)); m_Hlayout->addWidget(m_saveBtn); connect(m_saveBtn, &QToolButton::clicked, ui->mainImageView, &ImageView::savecurrentPic); //delete scale // m_scaleImageBtn = new ToolButton(); // m_scaleImageBtn->setShortcut(QKeySequence("Ctrl+Y")); // m_scaleImageBtn->setFixedSize(60, 60); // m_scaleImageBtn->setToolTip(QObject::tr("ScaleImage")); // m_scaleImageBtn->setIcon(QIcon(":/icon/scale.svg")); // m_scaleImageBtn->setIconSize(QSize(36, 36)); // m_Hlayout->addWidget(m_scaleImageBtn); // connect(m_scaleImageBtn, &QToolButton::clicked, ui->mainImageView, &ImageView::scaleImage); } } void MainWidget::initConnect() { connect(App, &Application::setStackWidget, this, &MainWidget::setCurrentWidget); connect(App, &Application::sigMouseMove, this, [ = ] { if (/*window()->isFullScreen() &&*/ m_statusbarWidget) { QPoint pos = mapFromGlobal(QCursor::pos()); if (height() - 20 < pos.y() && height() > pos.y() && height() == m_statusbarWidget->y()) { QPropertyAnimation *animation = new QPropertyAnimation(m_statusbarWidget, "pos"); animation->setDuration(200); animation->setEasingCurve(QEasingCurve::NCurveTypes); animation->setStartValue( QPoint((width() - m_statusbarWidget->width()) / 2, m_statusbarWidget->y())); animation->setEndValue(QPoint((width() - m_statusbarWidget->width()) / 2, height() - m_statusbarWidget->height() - 10)); animation->start(QAbstractAnimation::DeleteWhenStopped); } else if (height() - m_statusbarWidget->height() - 10 > pos.y() && height() - m_statusbarWidget->height() - 10 == m_statusbarWidget->y()) { QPropertyAnimation *animation = new QPropertyAnimation(m_statusbarWidget, "pos"); animation->setDuration(200); animation->setEasingCurve(QEasingCurve::NCurveTypes); animation->setStartValue( QPoint((width() - m_statusbarWidget->width()) / 2, m_statusbarWidget->y())); animation->setEndValue(QPoint((width() - m_statusbarWidget->width()) / 2, height())); animation->start(QAbstractAnimation::DeleteWhenStopped); } } }); connect(App, &Application::sigResetLightContrast, this, [ = ] { if (m_lightSlider) { m_lightSlider->setValue(100); } if (m_contrastSlider) { m_contrastSlider->setValue(150); } }); } void MainWidget::initMenu() { if (!m_leftMenu) { m_leftMenu = new Menu(this); m_openAction = new QAction(m_leftMenu); m_openAction->setText(tr("Open images")); m_leftMenu->addAction(m_openAction); connect(m_openAction, &QAction::triggered, this, &MainWidget::openImageFile); m_copyAction = new QAction(m_leftMenu); m_copyAction->setText(tr("Copy")); m_leftMenu->addAction(m_copyAction); connect(m_copyAction, &QAction::triggered, this, [ = ] { QClipboard *cb = qApp->clipboard(); QMimeData *newMimeData = new QMimeData(); newMimeData->setImageData(ui->mainImageView->image()); cb->setMimeData(newMimeData, QClipboard::Clipboard); }); m_saveBAction = new QAction(m_leftMenu); m_saveBAction->setText(tr("Save")); m_leftMenu->addAction(m_saveBAction); connect(m_saveBAction, &QAction::triggered, ui->mainImageView, &ImageView::savecurrentPic); m_CXAction = new QAction(m_leftMenu); m_CXAction->setText(QObject::tr("Withdrawal step")); m_leftMenu->addAction(m_CXAction); connect(m_CXAction, &QAction::triggered, ui->mainImageView, &ImageView::setLastImage); m_resetAction = new QAction(m_leftMenu); m_resetAction->setText(QObject::tr("Reset")); m_leftMenu->addAction(m_resetAction); connect(m_resetAction, &QAction::triggered, ui->mainImageView, &ImageView::resetImage); m_filterMenu = new Menu(this); m_filteraction = new QAction(m_leftMenu); m_filteraction->setText(tr("Filter")); m_filteraction->setMenu(m_filterMenu); m_leftMenu->addAction(m_filteraction); m_oldFilter = new QAction(m_filterMenu); m_oldFilter->setText(tr("Old photo filter")); m_filterMenu->addAction(m_oldFilter); connect(m_oldFilter, &QAction::triggered, ui->mainImageView, &ImageView::oldIMage); m_beepFilter = new QAction(m_filterMenu); m_beepFilter->setText(tr("Bilateral filtering")); m_filterMenu->addAction(m_beepFilter); connect(m_beepFilter, &QAction::triggered, this, [ = ] { ui->mainImageView->BEEPImage(); }); m_warnFilter = new QAction(m_filterMenu); m_warnFilter->setText(tr("Warm color filter")); m_filterMenu->addAction(m_warnFilter); connect(m_warnFilter, &QAction::triggered, this, [ = ] { ui->mainImageView->warnImage(); }); m_coolFilter = new QAction(m_filterMenu); m_coolFilter->setText(tr("Cool color filter")); m_filterMenu->addAction(m_coolFilter); connect(m_coolFilter, &QAction::triggered, this, [ = ] { ui->mainImageView->coolImage(); }); m_grayScaleFilter = new QAction(m_filterMenu); m_grayScaleFilter->setText(tr("Gray filter")); m_filterMenu->addAction(m_grayScaleFilter); connect(m_grayScaleFilter, &QAction::triggered, ui->mainImageView, &ImageView::GrayScaleImage); // m_lightContrastImage = new QAction(m_leftMenu); // m_lightContrastImage->setText(tr("lightContrastImage")); // m_leftMenu->addAction(m_lightContrastImage); // connect(m_lightContrastImage, &QAction::triggered, this, [ = ] { // ui->mainImageView->lightContrastImage(); // }); m_inverseColorFilter = new QAction(m_filterMenu); m_inverseColorFilter->setText(tr("Anti color filter")); m_filterMenu->addAction(m_inverseColorFilter); connect(m_inverseColorFilter, &QAction::triggered, ui->mainImageView, &ImageView::InverseColorImage); m_Metal = new QAction(m_filterMenu); m_Metal->setText(tr("Metal filter")); m_filterMenu->addAction(m_Metal); connect(m_Metal, &QAction::triggered, ui->mainImageView, &ImageView::Metal); m_ContourExtractionFilter = new QAction(m_filterMenu); //轮廓获取(右键菜单) m_ContourExtractionFilter->setText(QObject::tr("Contour Extraction Filter")); m_filterMenu->addAction(m_ContourExtractionFilter); connect(m_ContourExtractionFilter, &QAction::triggered, ui->mainImageView, &ImageView::ContourExtraction); m_BinaryzationFilter = new QAction(m_filterMenu); //二值化滤镜(右键菜单) m_BinaryzationFilter->setText(QObject::tr("Binaryzation Filter")); m_filterMenu->addAction(m_BinaryzationFilter); connect(m_BinaryzationFilter, &QAction::triggered, ui->mainImageView, &ImageView::Binaryzation); m_sharpeningMenu = new Menu(this); m_sharpeningAction = new QAction(m_leftMenu); m_sharpeningAction->setText(tr("Sharpening")); m_sharpeningAction->setMenu(m_sharpeningMenu); m_leftMenu->addAction(m_sharpeningAction); m_laplaceSharpen = new QAction(m_sharpeningMenu); m_laplaceSharpen->setText(tr("Laplacian sharpening")); m_sharpeningMenu->addAction(m_laplaceSharpen); connect(m_laplaceSharpen, &QAction::triggered, ui->mainImageView, &ImageView::LaplaceSharpenImage); m_soder = new QAction(m_sharpeningMenu); m_soder->setText(tr("Soder sharpening")); m_sharpeningMenu->addAction(m_soder); connect(m_soder, &QAction::triggered, ui->mainImageView, &ImageView::soderImage); m_flip = new QAction(m_leftMenu); m_flip->setText(tr("Flip")); m_leftMenu->addAction(m_flip); m_leftFlip = new Menu(this); m_flip->setMenu(m_leftFlip); m_flipVertical = new QAction(m_leftMenu); m_flipVertical->setText(tr("Flip Vertical")); m_leftFlip->addAction(m_flipVertical); connect(m_flipVertical, &QAction::triggered, ui->mainImageView, &ImageView::flipVertical); m_flipHorizontal = new QAction(m_leftMenu); m_flipHorizontal->setText(tr("Flip Horizontal")); m_leftFlip->addAction(m_flipHorizontal); connect(m_flipHorizontal, &QAction::triggered, ui->mainImageView, &ImageView::flipHorizontal); m_scale = new QAction(m_leftMenu); m_scale->setText(tr("Crop a picture")); m_leftMenu->addAction(m_scale); connect(m_scale, &QAction::triggered, ui->mainImageView, &ImageView::scaled); m_transparency = new QAction(m_leftMenu); m_transparency->setText(tr("Set Transparency")); m_leftMenu->addAction(m_transparency); connect(m_transparency, &QAction::triggered, ui->mainImageView, &ImageView::SetTransparency); } } bool MainWidget::Uninit() { bool iRet = false; if (m_statusbarWidget) { m_statusbarWidget->deleteLater(); m_statusbarWidget = nullptr; iRet = true; } return iRet; } void MainWidget::setCurrentWidget(const int &index) { switch (index) { case 0: ui->stackedWidget->setCurrentIndex(0); m_statusbarWidget->hide(); break; case 1: ui->stackedWidget->setCurrentIndex(1); if (m_statusbarWidget) { m_statusbarWidget->show(); m_statusbarWidget->move((this->width() - m_statusbarWidget->width()) / 2, this->height() - m_statusbarWidget->height()); } break; default: break; } } void MainWidget::openImage(const QString &path) { if (!path.isEmpty()) { ui->mainImageView->openImage(path); ui->basicImageView->openImage(path); setWindowTitle(QFileInfo(path).fileName()); } } void MainWidget::initShortcut() { QShortcut *sc = new QShortcut(QKeySequence("Ctrl+="), this); sc->setContext(Qt::WindowShortcut); connect(sc, &QShortcut::activated, this, [ = ] { ui->mainImageView->setScaleValue(1.1); }); sc = new QShortcut(QKeySequence("Ctrl+-"), this); sc->setContext(Qt::WindowShortcut); connect(sc, &QShortcut::activated, this, [ = ] { ui->mainImageView->setScaleValue(0.9); }); sc = new QShortcut(QKeySequence("Ctrl+C"), this); connect(sc, &QShortcut::activated, this, [ = ] { QClipboard *cb = qApp->clipboard(); QMimeData *newMimeData = new QMimeData(); newMimeData->setImageData(ui->mainImageView->image()); cb->setMimeData(newMimeData, QClipboard::Clipboard); }); sc = new QShortcut(QKeySequence("Ctrl+Z"), this); connect(sc, &QShortcut::activated, this, [ = ] { ui->mainImageView->setLastImage(); }); #ifdef USE_DTK QShortcut *scViewShortcut = new QShortcut(QKeySequence("Ctrl+Shift+/"), this); // connect(scE, SIGNAL(activated()), dApp, SLOT(quit())); connect(scViewShortcut, &QShortcut::activated, this, [ = ] { qDebug() << "receive Ctrl+Shift+/"; QRect rect = window()->geometry(); QPoint pos(rect.x() + rect.width() / 2, rect.y() + rect.height() / 2); Shortcut sc; QStringList shortcutString; QString param1 = "-j=" + sc.toStr(); QString param2 = "-p=" + QString::number(pos.x()) + "," + QString::number(pos.y()); shortcutString << "-b" << param1 << param2; qDebug() << shortcutString; QProcess::startDetached("deepin-shortcut-viewer", shortcutString); }); #endif QShortcut *F2Shortcut = new QShortcut(QKeySequence("F2"), this); connect(F2Shortcut, &QShortcut::activated, this, [ = ] { helpWidget *widget = new helpWidget(); widget->show(); #ifdef USE_DTK DDialog ss; ss.setIcon(QIcon(":/icon/simple-image-filter.png")); #else QDialog ss; #endif ss.setFixedSize(430, 520); widget->setParent(&ss); ss.exec(); }); QShortcut *F1Shortcut = new QShortcut(QKeySequence("F1"), this); connect(F1Shortcut, &QShortcut::activated, this, [ = ] { helpWidget *widget = new helpWidget(); widget->show(); #ifdef USE_DTK DDialog ss; ss.setIcon(QIcon(":/icon/simple-image-filter.png")); #else QDialog ss; #endif ss.setFixedSize(430, 520); widget->setParent(&ss); ss.exec(); }); } MainWidget::~MainWidget() { Uninit(); delete ui; } void MainWidget::showEvent(QShowEvent *event) { if (m_statusbarWidget) { if (m_statusbarWidget->isVisible()) { m_statusbarWidget->move((this->width() - m_statusbarWidget->width()) / 2, this->height() - m_statusbarWidget->height()); } } return QWidget::showEvent(event); } void MainWidget::resizeEvent(QResizeEvent *event) { if (m_statusbarWidget) { if (m_statusbarWidget->isVisible()) { m_statusbarWidget->move((this->width() - m_statusbarWidget->width()) / 2, this->height() - m_statusbarWidget->height()); } } return QWidget::resizeEvent(event); } void MainWidget::openImageFile() { QString picPath = QStandardPaths::writableLocation(QStandardPaths::PicturesLocation); QString strPath = QFileDialog::getOpenFileName(this, "", picPath); openImage(strPath); } bool MainWidget::eventFilter(QObject *obj, QEvent *event) { Q_UNUSED(obj) if (event->type() == QEvent::Drop) { } if (event->type() == QEvent::DragMove) { event->accept(); } if (event->type() == QEvent::DragEnter) { QList urls = static_cast(event)->mimeData()->urls(); if (urls.isEmpty()) { return false; } QStringList paths; for (QUrl url : urls) { const QString path = url.toLocalFile(); paths << path; } if (!paths.isEmpty()) { ui->mainImageView->openImage(paths.at(0)); ui->basicImageView->openImage(paths.at(0)); } event->accept(); } if (event->type() == QEvent::Resize && this->isVisible()) { } return false; } void MainWidget::contextMenuEvent(QContextMenuEvent *event) { if (1 == ui->stackedWidget->currentIndex() && m_leftMenu) { m_leftMenu->exec(QCursor::pos()); event->accept(); } } void MainWidget::on_openPicture_clicked() { openImageFile(); } simple-image-filter-1.1.2/src/mainwidget.h000066400000000000000000000130131417414176100204430ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef MAINWIDGET_H #define MAINWIDGET_H #include #include "label.h" class Menu; class QFrame; class QSlider; class ToolButton; class QVBoxLayout; class QHBoxLayout; class StatusBarWidget; class QGraphicsPixmapItem; class ToolButton; class QPropertyAnimation; namespace Ui { class MainWidget; } class MainWidget : public QWidget { Q_OBJECT public: explicit MainWidget(QWidget *parent = nullptr); //初始化函数 bool Init(); //初始化状态栏btn函数 void initBtn(); //初始化信号槽 void initConnect(); //初始化右键菜单 void initMenu(); //反初始化 bool Uninit(); //通过路径打开图片 void openImage(const QString &path); //初始化快捷键 void initShortcut(); ~MainWidget() override; protected: //窗口显示事件 void showEvent(QShowEvent *event) override; //窗口大小改变事件 void resizeEvent(QResizeEvent *event) override; //事件钩子 bool eventFilter(QObject *obj, QEvent *event) override; //右键菜单事件 void contextMenuEvent(QContextMenuEvent *event) override; private slots: //打开文管选择图片文件 void openImageFile(); //设置栈窗口(0为打开的初始界面,1为图像显示界面) void setCurrentWidget(const int &index); //在窗口1打开文管选择文件 void on_openPicture_clicked(); private: Ui::MainWidget *ui; StatusBarWidget *m_statusbarWidget{nullptr}; //下方状态栏 ToolButton *m_openBtn{nullptr}; //打开按钮(下方状态栏) ToolButton *m_fitImageBtn{nullptr}; //适应图片大小按钮(下方状态栏) ToolButton *m_fitWindowBtn{nullptr}; //适应窗口大小按钮(下方状态栏) ToolButton *m_rotateRight{nullptr}; //向右旋转按钮按钮(下方状态栏) ToolButton *m_rotateLeft{nullptr}; //向左旋转按钮按钮(下方状态栏) ToolButton *m_saveBtn{nullptr}; //保存图片按钮(下方状态栏) ToolButton *m_resetBtn{nullptr}; //还原原来的图片状态按钮(下方状态栏) ToolButton *m_scaleImageBtn{nullptr}; //还原原来的图片状态按钮(下方状态栏) Menu *m_leftMenu{nullptr}; //右键菜单 QAction *m_openAction{nullptr}; //打开按钮(右键菜单) QAction *m_copyAction{nullptr}; //复制功能(右键菜单) QAction *m_fitImageAction{nullptr};//适应图片大小按钮(右键菜单) QAction *m_fitWindowAction{nullptr}; //适应窗口大小按钮(右键菜单) QAction *m_rotateRightAction{nullptr}; //向右旋转按钮按钮(右键菜单) QAction *m_rotateLeftAction{nullptr}; //向左旋转按钮按钮(右键菜单) QAction *m_saveBAction{nullptr}; //保存图片按钮(右键菜单) QAction *m_CXAction{nullptr}; //撤销回上一次按钮 QAction *m_resetAction{nullptr}; //还原原来的图片状态按钮(右键菜单) Menu *m_filterMenu{nullptr}; QAction *m_filteraction{nullptr}; QAction *m_oldFilter{nullptr}; //老照片滤镜(右键菜单) QAction *m_beepFilter{nullptr}; //磨皮滤镜(右键菜单) QAction *m_warnFilter{nullptr}; //暖色滤镜(右键菜单) QAction *m_coolFilter{nullptr}; //冷色滤镜(右键菜单) QAction *m_grayScaleFilter{nullptr}; //黑白滤镜(右键菜单) QAction *m_lightContrastImage{nullptr}; //对比度和亮度(右键菜单) QAction *m_inverseColorFilter{nullptr}; //反色滤镜(右键菜单) QAction *m_ContourExtractionFilter{nullptr}; //轮廓获取(右键菜单) QAction *m_BinaryzationFilter{nullptr}; //二值化滤镜(右键菜单) QAction *m_transparency{nullptr}; //设置透明度 QAction *m_Metal{nullptr}; //金属拉丝 QAction *m_flip{nullptr}; //翻转按钮 Menu *m_leftFlip{nullptr}; //翻转菜单 QAction *m_flipVertical{nullptr}; //垂直翻转 QAction *m_flipHorizontal{nullptr}; //水平翻转 QAction *m_ContourExtraction{nullptr}; //水平翻转 Menu *m_sharpeningMenu{nullptr};//锐化菜单 QAction *m_sharpeningAction{nullptr}; QAction *m_laplaceSharpen{nullptr}; //拉普拉斯锐化 QAction *m_soder{nullptr}; //soder锐化 QAction *m_scale{nullptr}; QSlider *m_lightSlider {nullptr}; //亮度slider QSlider *m_contrastSlider {nullptr};//对比度slider QVBoxLayout *m_Vlayout {nullptr}; QHBoxLayout *m_Hlayout {nullptr}; QFrame *m_layoutframe {nullptr}; //新增slider窗口 QFrame *m_lightSliderWidget{nullptr}; QFrame *m_contrastSliderWidget{nullptr}; Label *m_lightSliderLabel{nullptr}; Label *m_contrastSliderLabel{nullptr}; QHBoxLayout *m_lightSliderWidgetLayout{nullptr}; QHBoxLayout *m_contrastSliderWidgetLayout{nullptr}; }; #endif // MAINWIDGET_H simple-image-filter-1.1.2/src/mainwidget.ui000066400000000000000000000077401417414176100206430ustar00rootroot00000000000000 MainWidget 0 0 876 444 MainWidget 0 0 0 0 0 0 1 Qt::Horizontal 40 20 302 36 openPicture Qt::Horizontal 40 20 0 0 0 0 0 0 0 0 0 0 0 openImageButton QPushButton
openimagebutton.h
ImageView QGraphicsView
imageview.h
simple-image-filter-1.1.2/src/qrc.qrc000066400000000000000000000007131417414176100174410ustar00rootroot00000000000000 icon/dcc_fit_36px.svg icon/dcc_left_36px.svg icon/dcc_right_36px.svg icon/dcc_11_36px.svg icon/scale.svg icon/save.svg icon/open.svg icon/reset.svg icon/close.svg icon/simple-image-filter.png simple-image-filter-1.1.2/src/scaledialog.cpp000066400000000000000000000031101417414176100211120ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "scaledialog.h" #include "ui_scaledialog.h" #include "application.h" scaleDialog::scaleDialog(const QImage &img, QWidget *parent) : QWidget(parent), m_img(img), ui(new Ui::scaleDialog) { ui->setupUi(this); ui->widthSpinBox->setValue(m_img.width()); ui->heightSpinBox->setValue(m_img.height()); setWindowTitle(tr("Scale window")); } scaleDialog::~scaleDialog() { delete ui; } void scaleDialog::on_okBtn_clicked() { App->sigFilterImage(m_img.scaled(ui->widthSpinBox->value(), ui->heightSpinBox->value())); QWidget *widget = static_cast(parent()); if (widget) { widget->close(); } } void scaleDialog::on_cancelBtn_clicked() { QWidget *widget = static_cast(parent()); if (widget) { widget->close(); } } simple-image-filter-1.1.2/src/scaledialog.h000066400000000000000000000022721417414176100205670ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef SCALEDIALOG_H #define SCALEDIALOG_H #include namespace Ui { class scaleDialog; } class scaleDialog : public QWidget { Q_OBJECT public: explicit scaleDialog(const QImage &img, QWidget *parent = nullptr); ~scaleDialog(); private slots: void on_okBtn_clicked(); void on_cancelBtn_clicked(); private: Ui::scaleDialog *ui; QImage m_img; }; #endif // SCALEDIALOG_H simple-image-filter-1.1.2/src/scaledialog.ui000066400000000000000000000060431417414176100207550ustar00rootroot00000000000000 scaleDialog 0 0 374 212 Dialog 176 36 1111111 1111111 ok cancel width 1 10000 height 1 10000 Scaled image Qt::AlignCenter openImageButton QPushButton
openimagebutton.h
Label QLabel
label.h
PushButton QPushButton
pushbutton.h
SpinBox QSpinBox
spinbox.h
simple-image-filter-1.1.2/src/shortcut/000077500000000000000000000000001417414176100200175ustar00rootroot00000000000000simple-image-filter-1.1.2/src/shortcut/shortcut.cpp000077500000000000000000000062221417414176100224030ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "shortcut.h" namespace { const QString VIEW_GROUP = "SHORTCUTVIEW"; const QString ALBUM_GROUP = "SHORTCUTALBUM"; } // namespace Shortcut::Shortcut(QObject *parent) : QObject(parent) { ShortcutGroup group1; ShortcutGroup group2; ShortcutGroup group3; group1.groupName = QObject::tr("Simple Image Filter"); group3.groupName = QObject::tr("Settings"); //整理代码结构,解决显示界面出现和快捷键不一致问题,是由于配置文件导致,但配置文件无效,改为不使用配置文件 group1.groupItems << ShortcutItem(QObject::tr("Open"), "Ctrl+O") << ShortcutItem(QObject::tr("Reset"), "Ctrl+E") << ShortcutItem(QObject::tr("FitImage"), "Ctrl+R") << ShortcutItem(QObject::tr("FitWindow"), "Ctrl+T") << ShortcutItem(QObject::tr("Rotate-90"), "Ctrl+Left") << ShortcutItem(QObject::tr("Rotate+90"), "Ctrl+Right") << ShortcutItem(QObject::tr("Save"), "Ctrl+S") << ShortcutItem(QObject::tr("ScaleImage"), "Ctrl+Y") << ShortcutItem(QObject::tr("Zoom in"), "Ctrl+ '+'") << ShortcutItem(QObject::tr("Zoom out"), "Ctrl+ '-'") << ShortcutItem(QObject::tr("Copy"), "Ctrl+C") << ShortcutItem(QObject::tr("Withdrawal step"), "Ctrl+Z"); group3.groupItems << ShortcutItem(QObject::tr("Help"), "F2") << ShortcutItem(QObject::tr("Display shortcuts"), "Ctrl + Shift + ?"); #ifndef LITE_DIV m_shortcutGroups << group1 << group2 << group3; #else m_shortcutGroups << group1 << group3; #endif // convert to json object QJsonArray jsonGroups; for (auto scg : m_shortcutGroups) { QJsonObject jsonGroup; jsonGroup.insert("groupName", scg.groupName); QJsonArray jsonGroupItems; for (auto sci : scg.groupItems) { QJsonObject jsonItem; jsonItem.insert("name", sci.name); jsonItem.insert("value", sci.value); jsonGroupItems.append(jsonItem); } jsonGroup.insert("groupItems", jsonGroupItems); jsonGroups.append(jsonGroup); } m_shortcutObj.insert("shortcut", jsonGroups); } QString Shortcut::toStr() { QJsonDocument doc(m_shortcutObj); return doc.toJson().data(); } simple-image-filter-1.1.2/src/shortcut/shortcut.h000077500000000000000000000025321417414176100220500ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef SHORTCUT_H #define SHORTCUT_H #include #include #include #include struct ShortcutItem { QString name; QString value; ShortcutItem(QString n, QString v): name(n), value(v) {} }; struct ShortcutGroup { QString groupName; QList groupItems; }; class Shortcut : public QObject { Q_OBJECT public: explicit Shortcut(QObject *parent = nullptr); QString toStr(); private: QJsonObject m_shortcutObj; QList m_shortcutGroups; }; #endif // SHORTCUT_H simple-image-filter-1.1.2/src/src.pro000066400000000000000000000063711417414176100174640ustar00rootroot00000000000000#------------------------------------------------- # # Project created by QtCreator 2020-12-14T15:11:40 # #------------------------------------------------- QT += core gui concurrent greaterThan(QT_MAJOR_VERSION, 4): QT += widgets TARGET = simple-image-filter TEMPLATE = app CONFIG -= app_bundle CONFIG += c++11 link_pkgconfig qtHaveModule(dtkwidget){ PKGCONFIG += dtkwidget DEFINES += USE_DTK } !isEmpty(target.path) # The following define makes your compiler emit warnings if you use # any feature of Qt which has been marked as deprecated (the exact warnings # depend on your compiler). Please consult the documentation of the # deprecated API in order to know how to port your code away from it. DEFINES += QT_DEPRECATED_WARNINGS # You can also make your code fail to compile if you use deprecated APIs. # In order to do so, uncomment the following line. # You can also select to disable deprecated APIs only up to a certain version of Qt. #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0 CONFIG += c++11 INCLUDEPATH += control \ thread \ view \ api SOURCES += \ main.cpp \ mainwidget.cpp \ application.cpp \ control/statusbarwidget.cpp \ control/pushbutton.cpp \ thread/imagethread.cpp \ view/imageview.cpp \ api/imageapi.cpp \ control/menu.cpp \ control/openimagebutton.cpp \ scaledialog.cpp \ control/dialog.cpp \ control/combox.cpp \ control/lineedit.cpp \ control/label.cpp \ control/spinbox.cpp \ transparencywidget.cpp \ control/toolbutton.cpp \ shortcut/shortcut.cpp \ help/helpwidget.cpp HEADERS += \ mainwidget.h \ application.h \ control/statusbarwidget.h \ control/pushbutton.h \ thread/imagethread.h \ view/imageview.h \ api/imageapi.h \ control/menu.h \ control/openimagebutton.h \ scaledialog.h \ control/dialog.h \ control/combox.h \ control/lineedit.h \ control/label.h \ control/spinbox.h \ transparencywidget.h \ control/toolbutton.h \ shortcut/shortcut.h \ help/helpwidget.h FORMS += \ mainwidget.ui \ scaledialog.ui \ transparencywidget.ui \ help/helpwidget.ui #./translation/ #TRANSLATIONS += simple-image-filter_zh_CN.ts CONFIG(release, debug|release) { TRANSLATIONS = $$files($$PWD/translations/*.ts) #遍历目录中的ts文件,调用lrelease将其生成为qm文件 for(tsfile, TRANSLATIONS) { qmfile = $$replace(tsfile, .ts$, .qm) system(lrelease $$tsfile -qm $$qmfile) | error("Failed to lrelease") } } APPSHAREDIR = /usr/share/simple-image-filter # Default rules for deployment. qnx: target.path = /tmp/$${TARGET}/bin else: unix:!android: target.path = /usr/bin translations.path = $$APPSHAREDIR/translations translations.files = $$PWD/translations/*.qm desktop.path = /usr/share/applications/ desktop.files = $$PWD/install/simple-image-filter.desktop icon.path =/usr/share/icons icon.files=$$PWD/icon/simple-image-filter.png !isEmpty(target.path): INSTALLS += target translations icon desktop RESOURCES += \ qrc.qrc TRANSLATIONS += \ translations/simple-image-filter_zh_CN.ts simple-image-filter-1.1.2/src/thread/000077500000000000000000000000001417414176100174135ustar00rootroot00000000000000simple-image-filter-1.1.2/src/thread/imagethread.cpp000066400000000000000000000065271417414176100224030ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "imagethread.h" #include ImageRunnable::ImageRunnable() { setAutoDelete(true); } ImageRunnable::~ImageRunnable() { } void ImageRunnable::setData(QImage &img, ImageFilterInfo info) { m_info = info; m_iimg = img; } void ImageRunnable::run() { switch (m_info.id) { case MenuItemId::IdNormal: break; case MenuItemId::Idold: m_iimg = QImageAPI::oldImage(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdBEEP: m_iimg = QImageAPI::QImageD_RunBEEPSHorizontalVertical(m_iimg, m_info.spatialDecay, m_info.photometricStandardDeviation); App->sigFilterImage(m_iimg); break; case MenuItemId::IdCool: m_iimg = QImageAPI::coolImage(m_iimg, m_info.coolImageDecay); App->sigFilterImage(m_iimg); break; case MenuItemId::IdWarn: m_iimg = QImageAPI::warnImage(m_iimg, m_info.warnImageDecay); App->sigFilterImage(m_iimg); break; case MenuItemId::IdGrayScale: m_iimg = QImageAPI::GrayScaleImage(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdInverseColor: m_iimg = QImageAPI::InverseColorImage(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdlightContrast: m_iimg = QImageAPI::lightContrastImage(m_iimg, m_info.lightDecay, m_info.ContrastDecay); App->sigFilterImage(m_iimg, UnChange); break; case MenuItemId::IdLaplaceSharpen: m_iimg = QImageAPI::LaplaceSharpen(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdSobel: m_iimg = QImageAPI::SobelEdge(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdHorizontal: m_iimg = QImageAPI::Horizontal(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdVertical: m_iimg = QImageAPI::Vertical(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdMetal: m_iimg = QImageAPI::Metal(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdContourExtraction: m_iimg = QImageAPI::ContourExtraction(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdBinaryzation: m_iimg = QImageAPI::Binaryzation(m_iimg); App->sigFilterImage(m_iimg); break; case MenuItemId::IdTransparency: m_iimg = QImageAPI::transparencyImg(m_info.transparency, m_iimg); App->sigFilterImage(m_iimg); break; default: break; } } simple-image-filter-1.1.2/src/thread/imagethread.h000066400000000000000000000030601417414176100220350ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef IMAGETHREAD_H #define IMAGETHREAD_H #include "imageapi.h" #include "application.h" #include #include #include #include struct ImageFilterInfo { MenuItemId id; double spatialDecay{0.02}; double photometricStandardDeviation{10}; int warnImageDecay{30}; int coolImageDecay{30}; int lightDecay{100}; int ContrastDecay{100}; int transparency{255}; }; class ImageRunnable : public QObject, public QRunnable { Q_OBJECT public : ImageRunnable(); ~ImageRunnable(); //设置数据 void setData(QImage &img, ImageFilterInfo info); protected: virtual void run(); private: ImageFilterInfo m_info; QImage *m_img {nullptr}; QImage m_iimg{nullptr}; }; #endif // IMAGETHREAD_H simple-image-filter-1.1.2/src/translations/000077500000000000000000000000001417414176100206655ustar00rootroot00000000000000simple-image-filter-1.1.2/src/translations/simple-image-filter_zh_CN.ts000066400000000000000000000545161417414176100261650ustar00rootroot00000000000000 Application simple-image-filter 简单图像处理工具 ClippingWidget scale image 裁剪图片 Preview: 预览: OriginalImage: 另外选择图片: OutputShape: 输出形状: CropperShape: 裁剪形状: FixedSize: 固定尺寸: MinimumSize: 最小的尺寸: EnableOpacity: 是否支持透明: Opacity: 透明度: ShowDragSquare: 显示拖动方块: DragSquareEdge: 拖动方块大小: DragSquareColor: 拖动方块颜色: ShowRectBorder: 显示矩形边框: RectBorderColor: 矩形边框颜色: Save 保存 Quit 退出 Rect/Square 矩形/方形 Ellipse/Circle 椭圆/圆 Rect 矩形 Square 正方形 FixedRect 固定矩形 Ellipse 椭圆 Circle FixedEllipse 固定椭圆 Select a picture 选择一张图片 Error 错误 Load image failed 加载图片失败 ImageView Save Image 保存图片 .png .png Open File 打开图片 png error 错误 open file error 打开文件失败 Reset tips 重置提示 Reset picture? 是否重置图片? ok 确定 cancel 取消 MainWidget MainWidget 主窗口 openPicture 打开图片 Open images 打开图片 Copy 复制 Save 保存 Filter 滤镜 Old photo filter 老照片滤镜 Bilateral filtering 双边滤波 Warm color filter 暖色滤镜 Cool color filter 冷色滤镜 Gray filter 灰度滤镜 Anti color filter 反色滤镜 Metal filter 金属滤镜 Sharpening 锐化 Laplacian sharpening 拉普拉斯锐化 Soder sharpening Soder 锐化 Flip 翻转 Flip Vertical 垂直翻转 Flip Horizontal 水平翻转 Crop a picture 裁剪图片大小 Set Transparency 设置透明度 QObject <a href='https://github.com/dependon/simple-image-filter/'>github/simple-image-filter</a><br/><span style='font-size:12pt;'>mail: liuminghang0821@gmail.com</span><br/><br/>Published under GPL V3 <a href='https://github.com/dependon/simple-image-filter/'>github/simple-image-filter</a><br/><span style='font-size:12pt;'>mail: liuminghang0821@qq.com</span><br/><br/>Published under GPL V3 Help 帮助 Brightness 亮度 Contrast 对比度 Open 打开图片 Reset 重做 FitImage 适应图片 FitWindow 适应窗口 Rotate-90 旋转-90 Rotate+90 旋转+90 Save 图片保存 ScaleImage 裁剪图片 Withdrawal step 撤销到上一步 Contour Extraction Filter 轮廓提取 Binaryzation Filter 图像二值化 Simple Image Filter 简单图像处理工具 Settings 设置 Zoom in 放大 Zoom out 放小 Copy 复制 Display shortcuts 显示快捷键窗口 TransparencyWidget Form 窗口 Set Transparency 设置透明度 ok 确定 确定 helpWidget Form 窗口 Help 帮助 Simple image filter tool is a tool with the functions of using filter, clipping, transparency adjustment, brightness adjustment and contrast adjustment. It can be used to do some simple image processing. 简单图像处理工具是一个具有使用滤镜,裁剪,透明度调节,亮度调节,对比度调节等功能的工具,可以用来做一些简单的图片处理. F2 F2 Open 打开图片 Ctrl+O Ctrl+O Withdrawal step 撤销到上一步 Ctrl+Z Reset 重做 Ctrl+E Ctrl+E Fit Image 适应图片 Ctrl+R Ctrl+R Fit Window 适应窗口 Ctrl+T Ctrl+T Rotate-90 旋转-90 Ctrl+Left Rotate+90 旋转+90 Ctrl+Right Save 保存 Ctrl+S ScaleImage 裁剪图片 Ctrl+Y Zoom in 放大 Ctrl+ '+' Zoom out 放小 Ctrl+ '-' Display shortcuts 显示快捷键窗口 Ctrl + Shift + ? scaleDialog Dialog 裁剪窗口 width height Scaled image 裁剪图片 ok 确定 cancel 取消 Scale window 裁剪窗口 simple-image-filter-1.1.2/src/transparencywidget.cpp000066400000000000000000000023711417414176100225700ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "transparencywidget.h" #include "ui_transparencywidget.h" TransparencyWidget::TransparencyWidget(QWidget *parent) : QWidget(parent), ui(new Ui::TransparencyWidget) { ui->setupUi(this); } TransparencyWidget::~TransparencyWidget() { delete ui; } void TransparencyWidget::on_okBtn_clicked() { emit transparencyChange(ui->spinBox->value()); QWidget *widget = static_cast(parent()); if (widget) { widget->close(); } } simple-image-filter-1.1.2/src/transparencywidget.h000066400000000000000000000023421417414176100222330ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef TRANSPARENCYWIDGET_H #define TRANSPARENCYWIDGET_H #include namespace Ui { class TransparencyWidget; } class TransparencyWidget : public QWidget { Q_OBJECT public: explicit TransparencyWidget(QWidget *parent = nullptr); ~TransparencyWidget(); signals: void transparencyChange(const int &index); private slots: void on_okBtn_clicked(); private: Ui::TransparencyWidget *ui; }; #endif // TRANSPARENCYWIDGET_H simple-image-filter-1.1.2/src/transparencywidget.ui000066400000000000000000000033011417414176100224150ustar00rootroot00000000000000 TransparencyWidget 0 0 357 107 Form Set Transparency Qt::AlignCenter 0 255 255 ok openImageButton QPushButton
openimagebutton.h
SpinBox QSpinBox
spinbox.h
Label QLabel
label.h
simple-image-filter-1.1.2/src/view/000077500000000000000000000000001417414176100171165ustar00rootroot00000000000000simple-image-filter-1.1.2/src/view/imageview.cpp000066400000000000000000000400271417414176100216020ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "imageview.h" #include "mainwidget.h" #include "imagethread.h" #include "scaledialog.h" #include "transparencywidget.h" #include #include #include #include #include #include #include #include #include #include #include #ifdef USE_DTK #include #include #endif #ifdef Q_OS_LINUX #include #endif const qreal MAX_SCALE_FACTOR = 20.0; const qreal MIN_SCALE_FACTOR = 0.029; #define devicePixelRatioF devicePixelRatio ImageView::ImageView(QWidget *parent, ViewId id): QGraphicsView(parent), m_cureentId(id) { setMouseTracking(true); setDragMode(ScrollHandDrag); QThreadPool::globalInstance()->setMaxThreadCount(1); if (Basic != id) { connect(App, &Application::sigFilterImage, this, &ImageView::openFilterImage); connect(App, &Application::sigFilterImage, this, &ImageView::addhisImage); } } void ImageView::openImage(const QString &path) { if (scene()) { if (m_currentImage) { delete m_currentImage; m_currentImage = nullptr; } m_currentImage = new QImage(path); if (!m_currentImage->isNull()) { //删除历史,重置亮度对比度 m_hisImage.clear(); emit App->sigResetLightContrast(); QPixmap pic = QPixmap::fromImage(*m_currentImage); if (Basic != m_cureentId) { App->setStackWidget(1); } scene()->clear(); m_pixmapItem = new QGraphicsPixmapItem(pic); m_pixmapItem->setTransformationMode(Qt::SmoothTransformation); QRectF rect = m_pixmapItem->boundingRect(); setSceneRect(rect); scene()->addItem(m_pixmapItem); fitWindow(); m_currentPath = path; } else { // App->setStackWidget(0); } m_FilterImage = image(); } } void ImageView::openFilterImage(QImage img, isChange is) { #ifdef Q_OS_LINUX malloc_trim(0); #endif if (!img.isNull() && scene()) { if (Change == is) { m_FilterImage = img; } QPixmap pic = QPixmap::fromImage(img); if (!pic.isNull()) { if (Basic != m_cureentId) { App->setStackWidget(1); } scene()->clear(); m_pixmapItem = new QGraphicsPixmapItem(pic); m_pixmapItem->setTransformationMode(Qt::SmoothTransformation); QRectF rect = m_pixmapItem->boundingRect(); setSceneRect(rect); scene()->addItem(m_pixmapItem); fitWindow(); } } } qreal ImageView::windowRelativeScale() const { QRectF bf = sceneRect(); if (1.0 * width() / height() > 1.0 * bf.width() / bf.height()) { return 1.0 * height() / bf.height(); } else { return 1.0 * width() / bf.width(); } } void ImageView::fitWindow() { qreal wrs = windowRelativeScale(); m_scal = wrs; resetTransform(); scale(wrs, wrs); // if (wrs - 1 > -0.01 && wrs - 1 < 0.01) { // emit checkAdaptImageBtn(); // } else { // emit disCheckAdaptImageBtn(); // } m_isFitImage = false; m_isFitWindow = true; } void ImageView::fitImage() { resetTransform(); m_scal = 1.0; scale(1, 1); m_isFitImage = true; m_isFitWindow = false; } void ImageView::RotateImage(const int &index) { if (!m_pixmapItem && scene()) return; QPixmap pixmap = m_pixmapItem->pixmap(); QMatrix rotate; rotate.rotate(index); pixmap = pixmap.transformed(rotate, Qt::FastTransformation); pixmap.setDevicePixelRatio(devicePixelRatioF()); scene()->clear(); resetTransform(); m_pixmapItem = new QGraphicsPixmapItem(pixmap); m_pixmapItem->setTransformationMode(Qt::SmoothTransformation); // Make sure item show in center of view after reload QRectF rect = m_pixmapItem->boundingRect(); setSceneRect(rect); scene()->addItem(m_pixmapItem); autoFit(); m_rotateAngel += index; m_FilterImage = image(); } void ImageView::savecurrentPic() { QString desktop = QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + "/Desktop"; QString filename = QFileDialog::getSaveFileName(this, tr("Save Image"), desktop, tr(".png")); //选择路径 if (!filename.contains(".png")) { filename = filename + ".png"; } image().save(filename); } void ImageView::scaleImage() { // ClippingWidget *dialog = new ClippingWidget(); // if (m_pixmapItem) { // dialog->setChooseCurrentImage(m_pixmapItem->pixmap()); // } // dialog->show(); //#ifdef USE_DTK // DDialog ss; // ss.setIcon(QIcon(":/icon/simple-image-filter.png")); //#else // QDialog ss; //#endif // ss.setFixedSize(1050, 610); // dialog->setParent(&ss); // ss.exec(); } void ImageView::savecurrentPicAs() { QFileDialog fileDialog; QString desktop = QStandardPaths::writableLocation(QStandardPaths::HomeLocation) + "/Desktop"; QString fileName = fileDialog.getSaveFileName(this, tr("Open File"), desktop, tr("png")); if (fileName == "") { return; } QFile file(fileName); if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) { QMessageBox::warning(this, tr("error"), tr("open file error")); return; } else { image().save(fileName, "png"); } } void ImageView::openImage(QImage *img) { if (!img->isNull() && scene()) { QPixmap pic = QPixmap::fromImage(*img); if (!pic.isNull()) { if (Basic != m_cureentId) { App->setStackWidget(1); } scene()->clear(); m_pixmapItem = new QGraphicsPixmapItem(pic); m_pixmapItem->setTransformationMode(Qt::SmoothTransformation); QRectF rect = m_pixmapItem->boundingRect(); setSceneRect(rect); scene()->addItem(m_pixmapItem); fitWindow(); } } } qreal ImageView::imageRelativeScale() const { return transform().m11() / devicePixelRatioF(); } void ImageView::autoFit() { if (image().isNull()) return; QSize image_size = image().size(); // change some code in graphicsitem.cpp line100. if ((image_size.width() >= width() || image_size.height() >= height() - 150) && width() > 0 && height() > 0) { fitWindow(); } else { fitImage(); } } void ImageView::mouseMoveEvent(QMouseEvent *event) { if (Basic != m_cureentId) { App->sigMouseMove(); } return QGraphicsView::mouseMoveEvent(event); } void ImageView::oldIMage() { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::Idold; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::resetImage() { #ifdef USE_DTK DDialog *pDDialog = new DDialog(QString(tr("Reset tips")), QString(tr("Reset picture?")), nullptr); pDDialog->setIcon(QIcon(":/icon/simple-image-filter.png")); pDDialog->setWindowFlags(pDDialog->windowFlags() | Qt::WindowStaysOnTopHint); pDDialog->addButton(QString(tr("ok")), false, DDialog::ButtonRecommend); pDDialog->addButton(QString(tr("cancel")), true, DDialog::ButtonNormal); connect(pDDialog, &DDialog::buttonClicked, [ = ](int index, const QString & text) { if (index == 0) { openImage(m_currentImage); m_FilterImage = *m_currentImage; emit App->sigResetLightContrast(); } else { // close(); } }); pDDialog->exec(); #else openImage(m_currentImage); m_FilterImage = *m_currentImage; #endif } void ImageView::BEEPImage(double spatialDecay, double photometricStandardDeviation) { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdBEEP; info.spatialDecay = spatialDecay; info.photometricStandardDeviation = photometricStandardDeviation; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::warnImage(int index) { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdWarn; info.warnImageDecay = index; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::coolImage(int index) { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdCool; info.coolImageDecay = index; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::GrayScaleImage() { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdGrayScale; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::lightContrastImage(int light, int Contrast) { if (m_currentImage) { QImage lightContrastImage(m_FilterImage); ImageFilterInfo info; info.id = MenuItemId::IdlightContrast; info.lightDecay = light; info.ContrastDecay = Contrast; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(lightContrastImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::InverseColorImage() { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdInverseColor; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::LaplaceSharpenImage() { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdLaplaceSharpen; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::soderImage() { if (m_currentImage) { ImageFilterInfo info; info.id = MenuItemId::IdSobel; ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::flipVertical() { ImageFilterInfo info; info.id = MenuItemId::IdVertical; playThread(info); } void ImageView::flipHorizontal() { ImageFilterInfo info; info.id = MenuItemId::IdHorizontal; playThread(info); } void ImageView::Binaryzation() { ImageFilterInfo info; info.id = MenuItemId::IdBinaryzation; playThread(info); } void ImageView::ContourExtraction() { ImageFilterInfo info; info.id = MenuItemId::IdContourExtraction; playThread(info); } void ImageView::Metal() { ImageFilterInfo info; info.id = MenuItemId::IdMetal; playThread(info); } void ImageView::scaled() { scaleDialog *scaleWidget = new scaleDialog(m_FilterImage); scaleWidget->show(); #ifdef USE_DTK DDialog ss; ss.setIcon(QIcon(":/icon/simple-image-filter.png")); #else QDialog ss; #endif // ss.setTitle(tr("Scaled image")); ss.setFixedSize(374, 214); scaleWidget->setParent(&ss); ss.exec(); } void ImageView::SetTransparency() { TransparencyWidget *widget = new TransparencyWidget(); widget->show(); connect(widget, &TransparencyWidget::transparencyChange, this, [ = ](const int &index) { ImageFilterInfo info; info.id = MenuItemId::IdTransparency; info.transparency = index; playThread(info); }); #ifdef USE_DTK DDialog ss; ss.setIcon(QIcon(":/icon/simple-image-filter.png")); #else QDialog ss; #endif // ss.setTitle(tr("Set picture transparency")); ss.setFixedSize(374, 120); widget->setParent(&ss); ss.exec(); } void ImageView::setLastImage() { qDebug() << m_hisImage.count(); if (m_hisImage.count() > 1) { m_hisImage.pop_back(); openFilterImage(m_hisImage.last(), Change); } else { if (m_hisImage.count() > 0) { m_hisImage.pop_back(); } emit App->sigResetLightContrast(); openFilterImage(*m_currentImage, Change); } } void ImageView::addhisImage(QImage img, isChange is) { Q_UNUSED(is); if (m_hisImage.count() > 10) { m_hisImage.pop_front(); } m_hisImage.push_back(img); } const QImage ImageView::image() { if (m_pixmapItem) { return m_pixmapItem->pixmap().toImage(); } else { return QImage(); } } void ImageView::setViewId(ViewId id) { m_cureentId = id; if (Basic == id) { disconnect(App, &Application::sigFilterImage, this, &ImageView::openFilterImage); } } void ImageView::playThread(const ImageFilterInfo &info) { if (m_currentImage) { ImageRunnable *imgThread = new ImageRunnable(); imgThread->setData(m_FilterImage, info); QThreadPool::globalInstance()->start(imgThread); } } void ImageView::resizeEvent(QResizeEvent *event) { return QGraphicsView::resizeEvent(event); } void ImageView::wheelEvent(QWheelEvent *event) { qreal factor = qPow(1.2, event->delta() / 240.0); scaleAtPoint(event->pos(), factor); event->accept(); } void ImageView::scaleAtPoint(QPoint pos, qreal factor) { // Remember zoom anchor point. const QPointF targetPos = pos; const QPointF targetScenePos = mapToScene(targetPos.toPoint()); // Do the scaling. setScaleValue(factor); // Restore the zoom anchor point. // // The Basic idea here is we don't care how the scene is scaled or transformed, // we just want to restore the anchor point to the target position we've // remembered, in the coordinate of the view/viewport. const QPointF curPos = mapFromScene(targetScenePos); const QPointF centerPos = QPointF(width() / 2.0, height() / 2.0) + (curPos - targetPos); const QPointF centerScenePos = mapToScene(centerPos.toPoint()); centerOn(static_cast(centerScenePos.x()), static_cast(centerScenePos.y())); } void ImageView::setScaleValue(qreal v) { //由于矩阵被旋转,通过矩阵获取缩放因子,计算缩放比例错误,因此记录过程中的缩放因子来判断缩放比例 m_scal *= v; qDebug() << m_scal; scale(v, v); //const qreal irs = imageRelativeScale() * devicePixelRatioF(); // Rollback if (v < 1 && /*irs <= MIN_SCALE_FACTOR)*/m_scal < 0.03) { const qreal minv = MIN_SCALE_FACTOR / m_scal; // if (minv < 1.09) return; scale(minv, minv); m_scal *= minv; } else if (v > 1 && /*irs >= MAX_SCALE_FACTOR*/m_scal > 20) { const qreal maxv = MAX_SCALE_FACTOR / m_scal; scale(maxv, maxv); m_scal *= maxv; } else { m_isFitImage = false; m_isFitWindow = false; } // qreal rescale = imageRelativeScale() * devicePixelRatioF(); // if (rescale - 1 > -0.01 && rescale - 1 < 0.01) { // emit checkAdaptImageBtn(); // } else { // emit disCheckAdaptImageBtn(); // } } simple-image-filter-1.1.2/src/view/imageview.h000066400000000000000000000077211417414176100212530ustar00rootroot00000000000000/* * Copyright (C) 2020 ~ 2021 LiuMingHang. * * Author: LiuMingHang * * Maintainer: LiuMingHang * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #ifndef IMAGEVIEW_H #define IMAGEVIEW_H #include "application.h" #include #include class ImageFilterInfo; class QGraphicsPixmapItem; class CropperWidget; class scaleDialog; //Normal indicates normal mode, and Basic mode does not accept global signals enum ViewId { Normal, Basic, }; class ImageView : public QGraphicsView { Q_OBJECT public: ImageView(QWidget *parent = nullptr, ViewId id = Normal); //Open picture via path void openImage(const QString &path); //For mouse wheel sliding qreal windowRelativeScale() const; qreal imageRelativeScale() const; void scaleAtPoint(QPoint pos, qreal factor); void setScaleValue(qreal v); //Adaptive window void autoFit(); void mouseMoveEvent(QMouseEvent *event) override; //Returns the current picture img const QImage image(); //Set view mode void setViewId(ViewId id); void playThread(const ImageFilterInfo &info); public slots: //Fit window size void fitWindow(); //Fit to picture size void fitImage(); //Rotate the picture and feel the index angle, - is left and + is righ void RotateImage(const int &index); //Save picture void savecurrentPic(); //Crop picture void scaleImage(); //Save as void savecurrentPicAs(); //Open this picture void openImage(QImage *img); void openFilterImage(QImage img, isChange is); //Old photo filter void oldIMage(); //Reset diagram void resetImage(); //qimage Skin grinding void BEEPImage(double spatialDecay = 0.02, double photometricStandardDeviation = 10); //Warm color filter void warnImage(int index = 30); //Cool color filter void coolImage(int index = 30); //Grayscale filter void GrayScaleImage(); //Brightness and saturation void lightContrastImage(int light = 100, int Contrast = 150); //Anti color filter void InverseColorImage(); //lpls void LaplaceSharpenImage(); //soder void soderImage(); //Flip vertically void flipVertical(); //Flip horizontally void flipHorizontal(); //Contour acquisition void ContourExtraction(); //metal wire-drawing void Metal(); //Crop picture resolution void scaled(); //Set picture transparency void SetTransparency(); //Set as last processed picture void setLastImage(); //Add new cache void addhisImage(QImage img, isChange is = Change); //Binarization void Binaryzation(); protected: void resizeEvent(QResizeEvent *event) override; void wheelEvent(QWheelEvent *event) override; private: QString m_currentPath;//Current picture path QGraphicsPixmapItem *m_pixmapItem{nullptr};//Item of the current image bool m_isFitImage = false;//Is it suitable for pictures bool m_isFitWindow = false;//Fit window qreal m_scal = 1.0; int m_rotateAngel = 0; //Rotation angle QImage *m_currentImage{nullptr};//Current original image QImage m_FilterImage{nullptr};//Currently processed image QImage m_lightContrastImage{nullptr};//Brightness exposure image ViewId m_cureentId{Normal};//Current mode QList m_hisImage;//Historical pictures }; #endif // IMAGEVIEW_H