NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
/** * GNU GENERAL PUBLIC LICENSE * Version 3, 29 June 2007 * * Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/> * 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. * * <one line to give the program's name and a brief idea of what it does.> * Copyright (C) <year> <name of author> * * 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 <https://www.gnu.org/licenses/>. * * 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: * * <program> Copyright (C) <year> <name of author> * 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 * <https://www.gnu.org/licenses/>. * * 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 * <https://www.gnu.org/licenses/why-not-lgpl.html>. */ var UserScriptStorage, updateBodyCssClassByView; var CSS_PREFIX = 'nyamka-from-romano-userscript'; var timeout = function(ms) { return new Promise(function(resolve) { return setTimeout(resolve, ms); }); }; var createElement = function(htmlText) { var element; if (htmlText.startsWith("<!DOCTYPE HTML PUBLIC")) { element = document.createElement('html'); } else { element = document.createElement('div'); } element.innerHTML = htmlText; if (element.children.length === 1) { return element.children[0]; } return element; }; var createStyle = function(styleText) { var element; element = document.createElement('style'); element.type = 'text/css'; element.updateStyleText = function(styleText) { styleText = styleText || ''; if (element.styleSheet) { element.styleSheet.cssText = styleText; } else { while (element.lastElementChild) { element.removeChild(element.lastElementChild); } element.appendChild(document.createTextNode(styleText)); } }; element.updateStyleText(styleText); return element; }; var extname = function(filename) { var data, splitPathRe; splitPathRe = /^(\/?|)([\s\S]*?)((?:\.{1,2}|[^\/]+?|)(\.[^.\/]*|))(?:[\/]*)$/; data = splitPathRe.exec(filename).slice(1); return data[3].toLowerCase(); }; var isObject = function(value) { return typeof value === 'object' && (value !== null && value !== (void 0)); }; var isString = function(value) { return typeof value === 'string'; }; var isEmpty = function(value) { if (value === null || value === void 0) { return true; } if (value.length > 0) { return false; } if (value.length === 0) { return true; } if (typeof value !== 'object') { return false; } return Object.keys(value).length === 0; }; var generateUniqueId = (function() { var id; id = 0; return function() { id += 1; return id; }; })(); var getInputCaretPosition = function(input) { if (input.selectionStart || input.selectionStart === '0') { return { startPos: input.selectionStart, endPos: input.selectionEnd }; } return { startPos: input.value.length, endPos: input.value.length }; }; var copyToClipboard = function(data, dataType) { var clipboardItem; dataType = dataType || 'text/plain'; clipboardItem = {}; clipboardItem[dataType] = new Blob([data], { type: dataType }); if (dataType.startsWith('text/') && dataType !== 'text/plain') { clipboardItem['text/plain'] = new Blob([data], { type: 'text/plain' }); } return navigator.clipboard.write([new ClipboardItem(clipboardItem)]); }; var downloadDataAsFile = function(value, fileName, fileType) { var a, blob, event; if (isObject(value)) { fileName = fileName || "download.json"; fileType = fileType || "application/json"; value = JSON.stringify(value, null, 4); } fileName = fileName || "download"; fileType = fileType || "text/plain"; blob = new Blob([value], { type: fileType }); event = document.createEvent('MouseEvents'); a = document.createElement('a'); a.download = fileName; a.href = window.URL.createObjectURL(blob); a.dataset.downloadurl = [fileType, a.download, a.href].join(':'); event.initEvent('click', true, false, unsafeWindow, 0, 0, 0, 0, 0, false, false, false, false, 0, null); a.dispatchEvent(event); }; updateBodyCssClassByView = function() { var cssClass, i, len, pathname, ref; ref = document.body.classList || []; for (i = 0, len = ref.length; i < len; i++) { cssClass = ref[i]; if (cssClass && cssClass.startsWith(`${CSS_PREFIX}-app-view`)) { document.body.classList.remove(cssClass); } } pathname = location.pathname.split('/')[1]; if (pathname === 'user_properties') { pathname = 'user-properties'; } else if (pathname.startsWith('user')) { pathname = 'user'; } document.body.classList.add(`${CSS_PREFIX}-app-view-${pathname}`); }; var onLocationChange = (function() { var allCallbacks; allCallbacks = null; updateBodyCssClassByView(); return function(cb) { var cbId, onChange, origPushState, origReplaceState; if (allCallbacks === null) { // декорируем функции смены url в 1 и единственный раз onChange = function() { var error, k, ref, v; updateBodyCssClassByView(); ref = allCallbacks || {}; for (k in ref) { v = ref[k]; if (v) { try { v(); } catch (error1) { error = error1; console.warn(error); } } } }; unsafeWindow.addEventListener('popstate', onChange); unsafeWindow.addEventListener('pushstate', onChange); unsafeWindow.addEventListener('hashchange', onChange); origPushState = unsafeWindow.history.pushState; unsafeWindow.history.pushState = function() { var result; result = origPushState.apply(unsafeWindow.history, arguments); onChange(); return result; }; origReplaceState = unsafeWindow.history.replaceState; unsafeWindow.history.replaceState = function() { var result; result = origReplaceState.apply(unsafeWindow.history, arguments); onChange(); return result; }; updateBodyCssClassByView(); } // добавляем коллбак к очереди cbId = generateUniqueId(); allCallbacks = allCallbacks || {}; allCallbacks[cbId] = cb; // и сразу запускаем функцию cb(); // возвращаем функцию дерегистратор return function() { delete onLocationChangeCallbacks[cbId]; }; }; })(); var onDomMutations = (function() { var allCallbacks, observer; observer = null; allCallbacks = null; return function(cb) { var MutationObserver, cbId; if (!observer) { MutationObserver = unsafeWindow.MutationObserver || unsafeWindow.WebkitMutationObserver || unsafeWindow.MozMutationObserver; observer = new MutationObserver(function(mutationsList) { var error, k, ref, v; if (isEmpty(allCallbacks)) { observer.disconnect(); return; } ref = allCallbacks || {}; for (k in ref) { v = ref[k]; if (v) { try { v(mutationsList); } catch (error1) { error = error1; console.warn(error); } } } }); observer.observe(document.body, { childList: true, subtree: true }); } // добавляем коллбак к очереди cbId = generateUniqueId(); allCallbacks = allCallbacks || {}; allCallbacks[cbId] = cb; // возвращаем функцию дерегистратор return function() { delete allCallbacks[cbId]; if (isEmpty(allCallbacks)) { observer.disconnect(); observer = null; } }; }; })(); var onClipboardPaste = function(cb) { var onPaste; onPaste = function(event) { var clipboardData; clipboardData = event.clipboardData || {}; clipboardData.items = clipboardData.items || []; cb(clipboardData); }; document.addEventListener('paste', onPaste); // возвращаем функцию дерегистратор return function() { document.removeEventListener('paste', onPaste); }; }; var loadOnce = (function() { var callbacks, loadOnceDone, loadStarted, loaded, preLoadOnce; loadStarted = {}; loaded = {}; callbacks = {}; loadOnceDone = function(_type, url, data) { var cb, i, len, ref; switch (_type) { case 'js': case 'css': loaded[url] = data || true; break; case 'data': if (extname(url) === '.json') { try { data = JSON.parse(data); } catch (error1) { } } loaded[url] = data; } loadStarted[url] = false; ref = callbacks[url] || []; for (i = 0, len = ref.length; i < len; i++) { cb = ref[i]; cb(data); } delete callbacks[url]; }; preLoadOnce = function(url, cb, notLoadedFn) { if (!loadStarted[url] && !loaded[url]) { loadStarted[url] = true; callbacks[url] = callbacks[url] || []; if (cb) { callbacks[url].push(cb); } return notLoadedFn(); } if (!cb) { return; } if (loaded[url]) { cb(loaded[url]); } else { callbacks[url] = callbacks[url] || []; callbacks[url].push(cb); } }; return { js: function(url, options) { options = options || {}; return new Promise(function(resolve, reject) { preLoadOnce(url, resolve, function() { var script; if (options.module) { import(url).then(function(result){ loadOnceDone('js', url, result); }, reject); return; } script = document.createElement('script'); script.onload = function() { loadOnceDone('js', url, void 0); }; script.onerror = reject; script.src = url; script.type = "text/javascript"; document.getElementsByTagName('head')[0].appendChild(script); }); }); }, css: function(url) { return new Promise(function(resolve, reject) { preLoadOnce(url, resolve, function() { var link; link = document.createElement("link"); link.rel = "stylesheet"; link.href = url; if (link.addEventListener) { link.addEventListener("load", function() { return loadOnceDone('css', url, void 0, true); }); } else if (link.attachEvent) { link.attachEvent("onload", function() { return loadOnceDone('css', url, void 0, true); }); } else { // поддержки коллбэка нет. просто запускаем loadOnceDone loadOnceDone('css', url, void 0); } document.getElementsByTagName('head')[0].appendChild(link); }); }); } }; })(); UserScriptStorage = (function() { class UserScriptStorage { constructor() { // следим сразу за изменениями localStorage window.addEventListener('storage', (event) => { var key; if (event.storageArea !== localStorage || event.key.startsWith(this.prefix)) { return; } key = event.key.split(this.prefix)[1]; if (key) { this[key] = this.loadData(key); } }); } registerVariable(key, defaultValue) { this.variables[key] = defaultValue; this[key] = this.loadData(key); } loadData(key) { var data; try { data = JSON.parse(localStorage.getItem(this.prefix + key)); } catch (error1) { data = null; } if (data === null || data === (void 0)) { return this.variables[key] || null; } return data; } saveData() { var key; for (key in this.variables) { localStorage.setItem(this.prefix + key, JSON.stringify(this[key])); } } } UserScriptStorage.prototype.prefix = 'nyamka-from-romano-'; UserScriptStorage.prototype.variables = {}; return UserScriptStorage; }).call(undefined); var storage = new UserScriptStorage(); var faviconImg$3 = ""; var favicon$3 = faviconImg$3; var getUserId = function(url) { var userId; url = url || ''; userId = ''; if (url.indexOf('/user') > -1) { userId = url.split('/user')[1]; } userId = userId.split('?')[0]; return userId; }; var getCurrentUserId = function() { var profileLink; profileLink = document.querySelector('#site-content-right li.first a'); if (profileLink) { return getUserId(profileLink.href); } return null; }; var isCurrentSite$3 = function() { return location.host === 'fanfics.me'; }; var getChapterEditor$4 = function() { return { title: document.querySelector("input#chapterName"), body: document.querySelector('textarea#chapterText') }; }; var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n<t.length;n++)e.setAttribute(t[n],r.attributes[t[n]]);var a="prepend"===s?"afterbegin":"beforeend";return i.insertAdjacentElement(a,e),e}} var css$c = ".nyamka-from-romano-userscript-blur .MessageLeft, .nyamka-from-romano-userscript-blur .MessageText, .nyamka-from-romano-userscript-blur a.user, .nyamka-from-romano-userscript-blur .MessageCommentLeft,\n.nyamka-from-romano-userscript-blur .MessageRight > .small.light, .nyamka-from-romano-userscript-blur.user, .nyamka-from-romano-userscript-blur > a > img {\n -webkit-filter: grayscale(1) blur(7px);\n -moz-filter: grayscale(1) blur(7px);\n filter: grayscale(1) blur(7px);\n position: relative;\n pointer-events: none;\n}\n.nyamka-from-romano-userscript-blur .MessageLeft::after, .nyamka-from-romano-userscript-blur .MessageText::after, .nyamka-from-romano-userscript-blur a.user::after, .nyamka-from-romano-userscript-blur .MessageCommentLeft::after,\n.nyamka-from-romano-userscript-blur .MessageRight > .small.light::after, .nyamka-from-romano-userscript-blur.user::after, .nyamka-from-romano-userscript-blur > a > img::after {\n content: \"\";\n position: absolute;\n top: 0;\n left: 0;\n height: 100%;\n width: 100%;\n}\n@media (max-width: 700px) {\n .nyamka-from-romano-userscript-blur .MessageLeft, .nyamka-from-romano-userscript-blur .MessageText, .nyamka-from-romano-userscript-blur a.user, .nyamka-from-romano-userscript-blur .MessageCommentLeft,\n.nyamka-from-romano-userscript-blur .MessageRight > .small.light, .nyamka-from-romano-userscript-blur.user, .nyamka-from-romano-userscript-blur > a > img {\n -webkit-filter: grayscale(1) blur(5px);\n -moz-filter: grayscale(1) blur(5px);\n filter: grayscale(1) blur(5px);\n }\n}\n.nyamka-from-romano-userscript-blur .LikeButton, .nyamka-from-romano-userscript-blur .LikeCounter, .nyamka-from-romano-userscript-blur a.normal_link, .nyamka-from-romano-userscript-blur .MessageButtons, .nyamka-from-romano-userscript-blur .DropDownManagement {\n display: none;\n}\n.nyamka-from-romano-userscript-blur .MessageRightUser > .small.light {\n display: none;\n}\n.nyamka-from-romano-userscript-blur .MessageRightUser > span[data-show-member] {\n pointer-events: none;\n}\n\nbody.nyamka-from-romano-userscript-app-view-notifications .ContentTable.Messages:not(.visible) {\n position: relative;\n pointer-events: none;\n}\nbody.nyamka-from-romano-userscript-app-view-notifications .ContentTable.Messages:not(.visible)::after {\n content: \"\";\n position: absolute;\n background: #fcfaf2;\n top: 0;\n left: 0;\n height: 100%;\n width: 100%;\n}\n\n@media (max-width: 1023px) {\n body[data-nyamka-script=fanfics-me] .MessageButtons .LikeButton {\n visibility: visible;\n }\n body[data-nyamka-script=fanfics-me] .MessageButtons .LikeCounter {\n margin-left: 10px;\n }\n}"; n(css$c,{}); var hideElement, initSettings, processCommentBody, processLikeBody, processNotificationBody, indexOf$5 = [].indexOf; hideElement = function(node, forceBlur) { if (!node) { return; } if (storage.usersHateListUseCssClass || forceBlur) { if (node.classList) { node.classList.add(`${CSS_PREFIX}-blur`); } else if (node.className) { node.className = (node.className || '') + ` ${CSS_PREFIX}-blur`; } return; } if (node.style) { node.style.display = 'none'; } else { node.textContent = ''; } }; processCommentBody = function(node) { var author, authorId, j, len, ref, user, userId; // пытаемся найти id комментатора author = node.querySelector('.MessageRightUser a') || {}; authorId = getUserId(author.href); // добавляем id пользователя, если его нет. так потом можно будет нормально скрывать кнопки // для собственных мимимишек if (authorId && !node.getAttribute('data-poster')) { node.setAttribute('data-poster', authorId); } if (authorId && storage.usersHateList[authorId]) { // чао бамбино, сеньорита! hideElement(node); return; } ref = node.querySelectorAll('.MessageText a.user'); // ищем теперь упоминание любого пользователя из hateList в комменте for (j = 0, len = ref.length; j < len; j++) { user = ref[j]; userId = getUserId(user.href); if (storage.usersHateList[userId]) { // ага! зачем читать упоминания о вас, мой дорогой? hideElement(node); return; } } }; processNotificationBody = function(node) { var authorId, i, j, len, textContent, user, userId, users, usersContainer; usersContainer = node.querySelector('.MessageRightUser') || {}; textContent = (usersContainer.textContent || '').toLowerCase(); if (textContent.indexOf('упомянул вас') > -1 || textContent.indexOf('упомянула вас') > -1) { authorId = getUserId(usersContainer.querySelector('a').href); if (authorId && storage.usersHateList[authorId]) { // чао бамбино, сеньорита! hideElement(node); return; } return; } // обычный список мимимишек users = usersContainer.querySelectorAll('a.user'); for (i = j = 0, len = users.length; j < len; i = ++j) { user = users[i]; userId = getUserId(user.href); if (!userId || !storage.usersHateList[userId]) { continue; } if (users.length === 1) { hideElement(node); return; } // скрываем только имя и запятую hideElement(user); hideElement(user.nextSibling); if (i === 0) { hideElement(node.querySelector('.MessageLeft'), true); } } }; processLikeBody = function(node) { var author, authorId; author = node.querySelector('a') || {}; authorId = getUserId(author.href); if (authorId && storage.usersHateList[authorId]) { // чао бамбино, сеньорита! hideElement(node); return; } }; initSettings = function() { var checkbox, infos, j, knownUserIds, len, oblivionSettings, ref, ref1, tdStat, tr, userId, v, valueToChecked; if (!document.getElementById('bl_list')) { return setTimeout(initSettings, 300); } valueToChecked = function(value) { if (value) { return 'checked'; } return ''; }; knownUserIds = []; ref = document.getElementById('bl_list').querySelectorAll('tr'); for (j = 0, len = ref.length; j < len; j++) { tr = ref[j]; tdStat = tr.querySelector('td.stat'); userId = tr.id.split('bl_member_')[1]; if (!tdStat || !userId) { continue; } checkbox = createElement(`<div> <label style="cursor: pointer"> <input type='checkbox' data-user-id="${userId}" ${valueToChecked(storage.usersHateList[userId])}> <small style='font-size: 90%'>Скрывать связанные комментарии, уведомления, мимимишки</small> </label> </div>`); checkbox.querySelector('input').addEventListener('click', function() { userId = this.getAttribute('data-user-id'); storage.usersHateList[userId] = !!this.checked; storage.saveData(); }); tr.children[1].appendChild(checkbox); knownUserIds.push(userId); } ref1 = storage.usersHateList; for (userId in ref1) { v = ref1[userId]; if (!v || !knownUserIds.includes(userId)) { delete storage.usersHateList[userId]; } } storage.saveData(); infos = document.getElementsByClassName('InfoContainer'); if (infos[0]) { oblivionSettings = createElement(`<div class='InfoContainer'> <b>Дополнительные настройки "Забвения пользователей"</b> <br> <label> <input type='checkbox' class='ChekboxImg' ${valueToChecked(storage.usersHateListUseCssClass)}> Вместо агрессивного скрытия использовать везде блюр </label> </div>`); oblivionSettings.querySelector('input').addEventListener('click', function() { storage.usersHateListUseCssClass = !!this.checked; storage.saveData(); }); infos[0].parentNode.appendChild(oblivionSettings); } }; var init$4 = function() { var addHideMimimiStyle, currentUserId, styleHideOwnMimimi; if (!isCurrentSite$3()) { return; } // переменные в сторадже и миграции // TODO: удалить нижние строки с localStorage, когда пройдет минимум 2-3 месяца if (localStorage.getItem('oblivion-userscript-usersHateList')) { localStorage.setItem(storage.prefix + 'usersHateList', localStorage.getItem('oblivion-userscript-usersHateList')); localStorage.removeItem('oblivion-userscript-usersHateList'); } if (localStorage.getItem('oblivion-userscript-useCssClass')) { localStorage.setItem(storage.prefix + 'usersHateListUseCssClass', localStorage.getItem('oblivion-userscript-useCssClass')); localStorage.removeItem('oblivion-userscript-useCssClass'); } storage.registerVariable('usersHateList', {}); storage.registerVariable('usersHateListUseCssClass', false); currentUserId = null; styleHideOwnMimimi = null; addHideMimimiStyle = function() { if (!currentUserId) { currentUserId = getCurrentUserId(); } if (currentUserId && !styleHideOwnMimimi) { styleHideOwnMimimi = createStyle(`table[data-poster="${currentUserId}"] > tbody > tr > td > .MessageBody > .MessageButtons .LikeButton, table[data-poster="${currentUserId}"] > tbody > tr > td > .MessageButtons .LikeButton, .MessageAlone[data-poster="${currentUserId}"] > .Message .LikeButton{ display: none; }`); document.head.appendChild(styleHideOwnMimimi); } }; // наблюдение за новым контентом на странице onLocationChange(function() { if (location.pathname === '/user_properties' && location.search.indexOf('action=black_list') > -1) { initSettings(); return; } if (location.pathname.startsWith('/message')) { document.addEventListener('DOMContentLoaded', function() { var container, posterId; container = document.querySelector('.MessageAlone'); if (!container) { return; } addHideMimimiStyle(); if (!container.getAttribute('data-poster')) { posterId = getUserId(document.querySelector('.MessageAloneHead > a').href); container.setAttribute('data-poster', posterId); } }); return; } if (location.pathname === '/notifications') { document.addEventListener('DOMContentLoaded', function() { var container, j, len, node, ref; container = document.querySelector('#site-content .ContentTable.Messages'); if (isEmpty(storage.usersHateList)) { if (container) { container.classList.add('visible'); } return; } ref = document.querySelectorAll('table.Message'); for (j = 0, len = ref.length; j < len; j++) { node = ref[j]; processNotificationBody(node); } if (container) { container.classList.add('visible'); } }); return; } }); onDomMutations(function(mutationsList) { var comment, j, k, l, len, len1, len2, len3, like, m, n, node, ref, ref1, ref2; addHideMimimiStyle(); //if isEmpty(storage.usersHateList) // return // нас интересуют ТОЛЬКО новые ноды for (j = 0, len = mutationsList.length; j < len; j++) { m = mutationsList[j]; if (m.addedNodes.length) { ref = m.addedNodes; for (k = 0, len1 = ref.length; k < len1; k++) { node = ref[k]; if (!(node)) { continue; } // нашли блок комментов if (indexOf$5.call(node.classList || [], 'MessageComments') >= 0 || indexOf$5.call(node.classList || [], 'MessageCommentsContainer') >= 0) { ref1 = node.querySelectorAll('table.MessageBody'); for (l = 0, len2 = ref1.length; l < len2; l++) { comment = ref1[l]; processCommentBody(comment); } } if (location.pathname === '/notifications' && indexOf$5.call(node.classList || [], 'Message') >= 0) { processNotificationBody(node); } // список мимимикнувших в модальных окнах if (indexOf$5.call(node.classList || [], 'LikesList') >= 0) { ref2 = node.children; for (n = 0, len3 = ref2.length; n < len3; n++) { like = ref2[n]; processLikeBody(like); } } } } } }); }; var css$b = ".notyf__message {\n text-align: left;\n}"; n(css$b,{}); var showNotify; showNotify = async function(options) { var notify; try { Notyf; } catch (error1) { await Promise.all([loadOnce.js('https://cdn.jsdelivr.net/npm/notyf@3.10.0/notyf.min.js'), loadOnce.css('https://cdn.jsdelivr.net/npm/notyf@3.10.0/notyf.min.css')]); } notify = new Notyf({ duration: 3000, position: { y: 'top' } }); notify[options.type](options.text); }; var notification = { success: function(text) { return showNotify({ type: 'success', text }); }, error: function(text) { return showNotify({ type: 'error', text }); } }; var css$a = ".nyamka-from-romano-userscript-dropdown {\n position: relative;\n display: inline-block;\n}\n.nyamka-from-romano-userscript-dropdown-button .caret {\n display: inline-block;\n width: 0;\n height: 0;\n margin-left: 2px;\n vertical-align: middle;\n border-top: 4px dashed;\n border-top: 4px solid\\9 ;\n border-right: 4px solid transparent;\n border-left: 4px solid transparent;\n}\n.nyamka-from-romano-userscript-dropdown-content {\n display: none;\n position: absolute;\n background-color: #D8D8C4;\n min-width: 220px;\n border: 1px solid #CCCCCC;\n box-shadow: 0px 8px 16px 0px rgba(0, 0, 0, 0.2);\n z-index: 1;\n padding: 0;\n}\n.nyamka-from-romano-userscript-dropdown-content li {\n color: #333;\n padding: 6px 15px;\n text-decoration: none;\n display: block;\n line-height: 1.2;\n cursor: pointer;\n}\n.nyamka-from-romano-userscript-dropdown-content li:hover, .nyamka-from-romano-userscript-dropdown-content li:focus {\n background: #c3a63f;\n color: black;\n}\n.nyamka-from-romano-userscript-dropdown-content li img {\n width: 18px;\n height: auto;\n margin-right: 6px;\n}\n.nyamka-from-romano-userscript-dropdown-content.show {\n display: block !important;\n}\n\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-dropdown-button {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-dropdown-button:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-dropdown-button:focus {\n background: #d8d8c4;\n}\n\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-button {\n color: #4f2d01;\n background: white;\n font-size: 12px;\n margin-bottom: 5px;\n margin-right: 5px;\n padding: 4px 8px;\n border: 1px solid #4f2d014d;\n border-radius: 5px;\n}\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-button:hover, body[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-button:focus {\n color: #4f2d01;\n border-color: #4f2d014d;\n}\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-content {\n background-color: #f6ecda;\n border-radius: 3px;\n padding: 4px 0;\n}\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-content li:hover, body[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-dropdown-content li:focus {\n background: #c69e6b;\n}\n\nbody[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-dropdown-content {\n margin-left: 0 !important;\n border-radius: 1px;\n}\nbody[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-dropdown-content li:hover, body[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-dropdown-content li:focus {\n background: #55a4f8;\n color: white;\n}"; n(css$a,{}); var watchForOutsideClick; watchForOutsideClick = function(dropdownContent) { var callback; callback = function(event) { if (!event.target.matches(`.${CSS_PREFIX}-dropdown-button`)) { if (dropdownContent.classList.contains('show')) { dropdownContent.classList.remove('show'); unsafeWindow.removeEventListener('click', callback); } } }; unsafeWindow.addEventListener('click', callback); }; var Dropdown = class Dropdown { constructor(title, items) { var i, item, len, liItem, ref; this.element = createElement(`<div class='${CSS_PREFIX}-dropdown'> <button class='${CSS_PREFIX}-dropdown-button'>${title} <span class='caret'></span></button> <ul class='${CSS_PREFIX}-dropdown-content' role='menu'> </ul> </div>`); this.element.children[0].addEventListener('click', (event) => { event.preventDefault(); this.element.children[1].classList.toggle('show'); watchForOutsideClick(this.element.children[1]); }); ref = items || []; for (i = 0, len = ref.length; i < len; i++) { item = ref[i]; liItem = createElement(`<li role='menuitem'>${item.text}</li>`); if (item.onClick) { liItem.addEventListener('click', (function(item) { return function(event) { event.preventDefault(); item.onClick(event, item); }; })(item)); } this.element.children[1].appendChild(liItem); } return; } open() { this.element.children[1].classList.add('show'); } close() { this.element.children[1].classList.remove('show'); } }; var css$9 = ".nyamka-from-romano-userscript-modal-content {\n text-align: left;\n}\n\nbody[data-nyamka-script=fanfics-me] #nyamka-from-romano-userscript-modals-container {\n display: none;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content {\n padding: 10px;\n text-align: left;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content .form-inline {\n margin-bottom: 0.7em;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content .form-inline label {\n margin-bottom: 0.5em;\n display: inline-block;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content .form-inline select {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content .form-inline select:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content .form-inline select:focus {\n background: #d8d8c4;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content h4, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content p {\n margin-bottom: 0.5em;\n text-align: left;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content ol {\n text-align: left;\n list-style: decimal;\n padding-left: 2em;\n margin-bottom: 0.5em;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-modal-content ol li {\n margin-bottom: 0.5em;\n}\n\nbody[data-nyamka-script=gikami-ru] #nyamka-from-romano-userscript-modals-container .modal-header .modal-title {\n order: 0;\n}\nbody[data-nyamka-script=gikami-ru] #nyamka-from-romano-userscript-modals-container .modal-header button.close {\n order: 1;\n}\nbody[data-nyamka-script=gikami-ru] #nyamka-from-romano-userscript-modals-container .modal-header button.close span {\n margin: 0;\n padding: 0px 10px;\n}\nbody[data-nyamka-script=gikami-ru] #nyamka-from-romano-userscript-modals-container hr {\n margin: 15px 0;\n}"; n(css$9,{}); var getJQuery, getModalType, indexOf$4 = [].indexOf; getJQuery = function() { try { if ($) { return $; } } catch (error1) { } try { return jQuery; } catch (error1) { } return void 0; }; getModalType = (function() { var modalType; modalType = null; return function() { var jq, test; if (!modalType) { jq = getJQuery(); test = jq(createElement('<div></div>')); if (test.arcticmodal) { modalType = 'arctic'; } else { modalType = 'bootstrap'; } } return modalType; }; })(); var Modal = class Modal { constructor(options) { var closeButton, modalType, modalsContainer; this.options = options || {}; modalsContainer = document.getElementById(`${CSS_PREFIX}-modals-container`); if (!modalsContainer) { modalsContainer = createElement(`<div id='${CSS_PREFIX}-modals-container'></div>`); document.body.appendChild(modalsContainer); } modalType = getModalType(); if (modalType === 'arctic') { closeButton = "<div class='box-modal_close arcticmodal-close'>Закрыть</div>"; if (this.options.disableClose) { closeButton = ''; } this.element = createElement(`<div class='box-modal'> ${closeButton} <div> <div class='modal-title'>${this.options.title || ''}</div> <div class='${CSS_PREFIX}-modal-content'> ${this.options.content} </div> <div class='clear'></div> </div> </div>`); } else if (modalType === 'bootstrap') { closeButton = `<button type="button" class="close" data-dismiss="modal" aria-label="Закрыть"> <span aria-hidden="true">×</span> </button>`; if (this.options.disableClose) { closeButton = ''; } this.element = createElement(`<div class='modal fade' role='dialog' aria-hidden="true"> <div class="modal-dialog" role="document"> <div class="modal-content"> <div class="modal-header"> ${closeButton} <h5 class="modal-title">${this.options.title || ''}</h5> </div> <div class="modal-body"> ${this.options.content} </div> </div> </div> </div>`); } modalsContainer.appendChild(this.element); } isVisible() { var modalType, modalsContainer; modalType = getModalType(); if (modalType === 'arctic') { modalsContainer = document.getElementById(`${CSS_PREFIX}-modals-container`); if (!modalsContainer) { return false; } return this.element.parentNode !== modalsContainer; } else if (modalType === 'bootstrap') { return indexOf$4.call(this.element.classList, 'in') >= 0 || indexOf$4.call(this.element.classList, 'show') >= 0; // bootstrap4 } return false; } show() { var jq, modalType; modalType = getModalType(); jq = getJQuery(); if (modalType === 'arctic') { jq(this.element).arcticmodal(); } else if (modalType === 'bootstrap') { jq(this.element).modal({ keyboard: !this.options.disableClose }); } } close() { var jq, modalType; modalType = getModalType(); jq = getJQuery(); if (modalType === 'arctic') { try { jq(this.element).arcticmodal('close'); } catch (error1) { } } else if (modalType === 'bootstrap') { jq(this.element).modal('hide'); } } destroy() { var modalsContainer; this.close(); modalsContainer = document.getElementById(`${CSS_PREFIX}-modals-container`); if (!modalsContainer) { return; } try { modalsContainer.removeChild(this.element); } catch (error1) { } } }; var faviconImg$2 = ""; var favicon$2 = faviconImg$2; var isCurrentSite$2 = function() { return location.host === 'ficbook.net'; }; var getChapterEditor$3 = function() { var baseEditorTab, container; //container = document.querySelector('div.text-editor-container') container = document.querySelector('form#savePartForm'); if (!container) { return; } baseEditorTab = container.querySelector("a#jsToFicbookEditor"); // обязательно переключаемся на обычный редактор if (baseEditorTab) { baseEditorTab.click(); } return { title: container.querySelector("input#titleInput"), body: container.querySelector('textarea[name="content"]') }; }; var faviconImg$1 = ""; var favicon$1 = faviconImg$1; var isCurrentSite$1 = function() { return location.host === 'archiveofourown.org'; }; var getChapterEditor$2 = function() { return void 0; }; var isCurrentSite = function() { return location.host === 'gikami.ru'; }; var getChapterEditor$1 = function() { var body, bodyEditor, container, editor, i, len, ref; container = document.querySelector('form[name="form_book"]'); if (!container) { return; } ref = unsafeWindow._mediumEditors || []; for (i = 0, len = ref.length; i < len; i++) { editor = ref[i]; if (editor && editor.elements[0]) { if (editor.elements[0].id.startsWith('medium-editor-book')) { bodyEditor = editor; break; } } } if (!bodyEditor) { throw "Can't find editor for body!"; } // создаем "прокси" для того, чтоб не ломать код, который ожидает обычный textarea для тела редактора body = new Proxy({}, { get: function(target, property) { if (property === 'addEventListener' || property === 'removeEventListener') { return function() { return bodyEditor.elements[0][property].apply(bodyEditor.elements[0], arguments); }; } if (property === 'value') { return bodyEditor.getContent(); } throw "Not implemented"; }, set: function(target, property, value) { if (property !== 'value') { throw "Not implemented"; } bodyEditor.elements[0].click(); bodyEditor.elements[0].focus(); bodyEditor.selectAllContents(); bodyEditor.pasteHTML(value + '', { cleanAttrs: [], cleanTags: [], unwrapTags: [] }); } }); return { title: container.querySelector('input[name="title"]'), body: body }; }; var faviconImg = ""; var favicon = faviconImg; var getChapterEditor = function() { if (isCurrentSite$3()) { return getChapterEditor$4(); } if (isCurrentSite$2()) { return getChapterEditor$3(); } if (isCurrentSite$1()) { return getChapterEditor$2(); } if (isCurrentSite()) { return getChapterEditor$1(); } return void 0; }; var fanficsMePreprocess, ficbookNetPreprocess, lineToParagraph; fanficsMePreprocess = function(text, options) { var converted, end, getNoteMark, j, len, line, note, noteTag, notes, ref, start; options = options || {}; noteTag = options.noteTag || 'note'; converted = []; notes = []; getNoteMark = function(index, inText) { if (options.noteType === '*') { return "*".repeat(index); } // number if (inText) { return `(${index})`; } return `${index})`; }; ref = text.split('\n').filter(function(line) { return !!line.trim(); }); for (j = 0, len = ref.length; j < len; j++) { line = ref[j]; start = line.indexOf(`<${noteTag}>`); end = line.indexOf(`</${noteTag}>`); if (start > -1) { note = line.substring(start + noteTag.length + 2, end); line = line.substring(0, start) + "<b>" + getNoteMark(notes.length + 1, true) + "</b>" + line.substr(end + noteTag.length + 3); notes.push(note); } converted.push(line); } converted = converted.join('\n'); if (notes.length) { converted = converted + "\n<center>---</center>\n<i>ПРИМЕЧАНИЯ:</i>\n" + notes.map(function(note, index) { return `<i><b>${getNoteMark(index + 1)}</b> — ${note}</i>`; }).join('\n'); } return converted; }; ficbookNetPreprocess = function(text, options) { var converted, j, len, line, ref; options = options || {}; if (options.convertNoteTag) { text = text.replaceAll("<footnote>", `<${options.convertNoteTag}>`).replaceAll("</footnote>", `</${options.convertNoteTag}>`); } if (options.extractNoteTag) { options.noteTag = options.noteTag || 'footnote'; text = fanficsMePreprocess(text, options); } converted = []; ref = text.split('\n'); for (j = 0, len = ref.length; j < len; j++) { line = ref[j]; if (line.startsWith('<tab>')) { line = line.substr(5); } converted.push(line); } return converted.join('\n'); }; lineToParagraph = function(line, nextLine, paragraphType) { nextLine = nextLine || ''; if (!line.trim()) { return line; } if (paragraphType === 'tab' && !line.startsWith('<center>') && !line.endsWith('</center>') && line) { return `<tab>${line}`; } if (paragraphType === '4-spaces' && !line.startsWith('<center>') && !line.endsWith('</center>')) { return ` ${line}`; } if (paragraphType === '2em' && !line.startsWith('<center>') && !line.endsWith('</center>')) { return `<2em>${line}`; } if (paragraphType === 'new-line') { if (!line.startsWith('<center>') && !line.endsWith('</center>') && nextLine.trim()) { return `${line}\n `; } } return line; }; var fanficsMe = { toFicbookNet: function(text) { var converted, i, j, len, line, lines; converted = []; text = fanficsMePreprocess(text, { noteType: storage.exportNotesStrategyForFicbookNet }); text = text.replaceAll('<u>', '').replaceAll('</u>', '').replaceAll('<hr>', '<tab>\n').replaceAll('<empty-line>', '\n<tab>'); lines = text.split('\n'); for (i = j = 0, len = lines.length; j < len; i = ++j) { line = lines[i]; if (!line.trim()) { continue; } if (line === '<tab>') { converted.push(line); continue; } if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1 && line.indexOf('<center>') === -1) { line = `<center>${line}</center>`; } else if (line.indexOf('<center>') === -1 && line.indexOf('<right>') === -1) { line = lineToParagraph(line, lines[i + 1], storage.exportParagraphStrategyForFicbookNet); } converted.push(line); } return converted.join('\n'); }, toAuthorToday: function(text) { var converted, j, len, line, lines; converted = []; text = fanficsMePreprocess(text, { noteType: storage.exportNotesStrategyForAuthorToday }); text = text.replaceAll('<hr>', ' ').replaceAll('<empty-line>', ' '); lines = text.split('\n'); for (j = 0, len = lines.length; j < len; j++) { line = lines[j]; if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1) { line = line.replaceAll('<center>', '').replaceAll('</center>', ''); line = `<p style="text-align: center;">${line}</p>`; } else if (line.indexOf('<center>') > -1) { line = line.replace("<center>", "<p style=\"text-align: center;\">").replace('</center>', '</p>'); } else if (line.indexOf('<right>') > -1) { line = line.replace("<right>", "<p style=\"text-align: right;\">").replace('</right>', '</p>'); } else { line = `<p>${line}</p>`; } converted.push(line); } return converted.join('\n'); }, toAO3: function(text) { var converted, i, j, len, line, lines; converted = []; text = fanficsMePreprocess(text, { noteType: storage.exportNotesStrategyForAO3 }); text = text.replaceAll('<i>', '<em>').replaceAll('</i>', '</em>').replaceAll('<s>', '<strike>').replaceAll('</s>', '</strike>').replaceAll('<hr>', '<hr/>').replaceAll('<empty-line>', ' '); lines = text.split('\n'); for (i = j = 0, len = lines.length; j < len; i = ++j) { line = lines[i]; if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1) { line = line.replaceAll('<center>', '').replaceAll('</center>', ''); line = `<p align='center'>${line}</p>`; } else if (line === '<center>---</center>') { line = "<hr/>"; } else if (line.indexOf('<center>') > -1) { line = line.replace("<center>", "<p align='center'>").replace('</center>', '</p>'); } else if (line.indexOf('<right>') > -1) { line = line.replace("<right>", "<p align='right'>").replace('</right>', '</p>'); } else if (line.indexOf('<hr/>') === -1) { line = lineToParagraph(line, lines[i + 1], storage.exportParagraphStrategyForAO3); line = line.split('\n').join("</p><p>"); line = `<p>${line}</p>`; } converted.push(line); } return converted.join('\n'); } }; var ficbookNet = { toFanficsMe: function(text) { return ficbookNetPreprocess(text, { convertNoteTag: 'note' }); }, toAuthorToday: function(text) { var converted, j, len, line, ref; converted = []; text = ficbookNetPreprocess(text, { extractNoteTag: true, noteType: storage.exportNotesStrategyForAuthorToday }); ref = text.split('\n'); for (j = 0, len = ref.length; j < len; j++) { line = ref[j]; if (!line.trim()) { line = ' '; } if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1) { line = line.replaceAll('<center>', '').replaceAll('</center>', ''); line = `<p style='text-align: center;'>${line}</p>`; } else if (line.indexOf('<center>') > -1) { line = line.replace("<center>", "<p style='text-align: center;'>").replace('</center>', '</p>'); } else if (line.indexOf('<right>') > -1) { line = line.replace("<right>", "<p style='text-align: right;'>").replace('</right>', '</p>'); } else { line = `<p>${line}</p>`; } converted.push(line); } return converted.join('\n'); }, toAO3: function(text) { var converted, i, j, len, line, lines; converted = []; text = ficbookNetPreprocess(text, { extractNoteTag: true, noteType: storage.exportNotesStrategyForAO3 }); text = text.replaceAll('<i>', '<em>').replaceAll('</i>', '</em>').replaceAll('<s>', '<strike>').replaceAll('</s>', '</strike>'); lines = text.split('\n'); for (i = j = 0, len = lines.length; j < len; i = ++j) { line = lines[i]; if (!line.trim()) { line = ' '; } if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1) { line = line.replaceAll('<center>', '').replaceAll('</center>', ''); line = `<p align='center'>${line}</p>`; } else if (line.indexOf('<center>') > -1) { line = line.replace("<center>", "<p align='center'>").replace('</center>', '</p>'); } else if (line.indexOf('<right>') > -1) { line = line.replace("<right>", "<p align='right'>").replace('</right>', '</p>'); } else { line = lineToParagraph(line, lines[i + 1], storage.exportParagraphStrategyForAO3); line = line.split('\n').join("</p><p>"); line = `<p>${line}</p>`; } converted.push(line); } return converted.join('\n'); }, toGikami: function(text) { var converted, i, j, len, line, lines; converted = []; text = ficbookNetPreprocess(text, { extractNoteTag: true }); lines = text.split('\n'); for (i = j = 0, len = lines.length; j < len; i = ++j) { line = lines[i]; if (!line.trim()) { line = ' '; } if (line.indexOf('***') > -1 || line.indexOf('* * *') > -1) { line = line.replaceAll('<center>', '').replaceAll('</center>', ''); line = `<p align='center'>${line}</p>`; } else if (line.indexOf('<center>') > -1) { line = line.replace("<center>", "<p style='text-align: center;'>").replace('</center>', '</p>'); } else if (line.indexOf('<right>') > -1) { line = line.replace("<right>", "<p style='text-align: right;'>").replace('</right>', '</p>'); } else { line = lineToParagraph(line, lines[i + 1], storage.exportParagraphStrategyForGikami); line = line.split('\n').join("</p><p>"); line = `<p>${line}</p>`.replace("<p><2em>", "<p style='text-indent: 2em'>"); } converted.push(line); } return converted.join('\n'); } }; var execute = async function(text) { var tp; try { /* настройки такие же, как и на ficbook.net */ Typograf; } catch (error1) { await loadOnce.js('https://cdn.jsdelivr.net/npm/typograf@6.12.1/dist/typograf.min.js'); Typograf.addRule({ name: "common/other/typographicSpaces", handler: (t) => { return t.replace("! ≫", "!≫"); } }); } tp = new Typograf({ locale: ['ru'] }); tp.setSetting("common/punctuation/quote", "ru", { left: "«", right: "»", removeDuplicateQuotes: true }); tp.disableRule("common/nbsp/*"); tp.disableRule("ru/nbsp/*"); tp.disableRule("ru/dash/koe"); tp.disableRule("ru/dash/to"); tp.disableRule("ru/punctuation/ano"); return tp.execute(text || ''); }; var SVG_ICON_FLOWER_EMPTY = "<svg focusable=\"false\" data-icon=\"flower\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 512 512\"><path fill=\"currentColor\" d=\"M461.5 256c30.7-26.8 50.5-65.8 50.5-109.7C512 65.5 446.5 0 365.7 0 321.8 0 282.8 19.8 256 50.5 229.2 19.8 190.2 0 146.3 0 65.5 0 0 65.5 0 146.3c0 43.9 19.8 82.9 50.5 109.7C19.8 282.8 0 321.8 0 365.7 0 446.5 65.5 512 146.3 512c43.9 0 82.9-19.8 109.7-50.5 26.8 30.7 65.8 50.5 109.7 50.5 80.8 0 146.3-65.5 146.3-146.3 0-43.9-19.8-82.9-50.5-109.7zm-95.8 224c-32.9 0-63.3-14-85.6-39.5L256 412.9l-24.1 27.6C209.6 466 179.2 480 146.3 480 83.3 480 32 428.7 32 365.7c0-32.9 14-63.3 39.5-85.6L99.1 256l-27.6-24.1C46 209.6 32 179.2 32 146.3 32 83.3 83.3 32 146.3 32c32.9 0 63.3 14 85.6 39.5L256 99.1l24.1-27.6C302.4 46 332.8 32 365.7 32c63 0 114.3 51.3 114.3 114.3 0 32.9-14 63.3-39.5 85.6L412.9 256l27.6 24.1c25.5 22.3 39.5 52.7 39.5 85.6 0 63-51.3 114.3-114.3 114.3zM256 160c-52.9 0-96 43.1-96 96s43.1 96 96 96 96-43.1 96-96-43.1-96-96-96zm0 160c-35.3 0-64-28.7-64-64s28.7-64 64-64 64 28.7 64 64-28.7 64-64 64z\" class=\"\"></path></svg>\n"; var SVG_ICON_FLOWER_FULL = "<svg focusable=\"false\" data-icon=\"flower\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 512 512\"><path fill=\"currentColor\" d=\"M480 160A128 128 0 0 0 352 32c-38.45 0-72.54 17.3-96 44.14C232.54 49.3 198.45 32 160 32A128 128 0 0 0 32 160c0 38.45 17.3 72.54 44.14 96C49.3 279.46 32 313.55 32 352a128 128 0 0 0 128 128c38.45 0 72.54-17.3 96-44.14C279.46 462.7 313.55 480 352 480a128 128 0 0 0 128-128c0-38.45-17.3-72.54-44.14-96C462.7 232.54 480 198.45 480 160zM256 336a80 80 0 1 1 80-80 80 80 0 0 1-80 80z\" class=\"\"></path></svg>\n"; var SVG_ICON_INDENT_EMPTY = "<svg focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 448 512\"><path fill=\"currentColor\" d=\"M27.31 148.7a15.63 15.63 0 0 0-11.17-4.7A16 16 0 0 0 0 160v192a16 16 0 0 0 16.13 16 15.58 15.58 0 0 0 11.18-4.7l96-96a16 16 0 0 0 0-22.62zM32 313.36V198.64L89.37 256zM440 48H8a8 8 0 0 0-8 8v16a8 8 0 0 0 8 8h432a8 8 0 0 0 8-8V56a8 8 0 0 0-8-8zm0 384H8a8 8 0 0 0-8 8v16a8 8 0 0 0 8 8h432a8 8 0 0 0 8-8v-16a8 8 0 0 0-8-8zm0-128H200a8 8 0 0 0-8 8v16a8 8 0 0 0 8 8h240a8 8 0 0 0 8-8v-16a8 8 0 0 0-8-8zm0-128H200a8 8 0 0 0-8 8v16a8 8 0 0 0 8 8h240a8 8 0 0 0 8-8v-16a8 8 0 0 0-8-8z\" class=\"\"></path></svg>\n"; var SVG_ICON_INDENT_FULL = "<svg focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 448 512\"><path fill=\"currentColor\" d=\"M27.31 363.3l96-96a16 16 0 0 0 0-22.62l-96-96C17.27 138.66 0 145.78 0 160v192c0 14.31 17.33 21.3 27.31 11.3zM432 416H16a16 16 0 0 0-16 16v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16v-32a16 16 0 0 0-16-16zm3.17-128H204.83A12.82 12.82 0 0 0 192 300.83v38.34A12.82 12.82 0 0 0 204.83 352h230.34A12.82 12.82 0 0 0 448 339.17v-38.34A12.82 12.82 0 0 0 435.17 288zm0-128H204.83A12.82 12.82 0 0 0 192 172.83v38.34A12.82 12.82 0 0 0 204.83 224h230.34A12.82 12.82 0 0 0 448 211.17v-38.34A12.82 12.82 0 0 0 435.17 160zM432 32H16A16 16 0 0 0 0 48v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16V48a16 16 0 0 0-16-16z\" class=\"\"></path></svg>\n"; var SVG_ICON_PASTE = "<svg focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 448 512\"><path fill=\"currentColor\" d=\"M128 184c0-30.879 25.122-56 56-56h136V56c0-13.255-10.745-24-24-24h-80.61C204.306 12.89 183.637 0 160 0s-44.306 12.89-55.39 32H24C10.745 32 0 42.745 0 56v336c0 13.255 10.745 24 24 24h104V184zm32-144c13.255 0 24 10.745 24 24s-10.745 24-24 24-24-10.745-24-24 10.745-24 24-24zm184 248h104v200c0 13.255-10.745 24-24 24H184c-13.255 0-24-10.745-24-24V184c0-13.255 10.745-24 24-24h136v104c0 13.2 10.8 24 24 24zm104-38.059V256h-96v-96h6.059a24 24 0 0 1 16.97 7.029l65.941 65.941a24.002 24.002 0 0 1 7.03 16.971z\"></path></svg>\n"; var IMG_HOW_TO_EXPORT = ""; var css$8 = ".nyamka-from-romano-userscript-markup-tools-autoformat .autoformat-active svg, .nyamka-from-romano-userscript-markup-tools-autoformat .autoformat svg, .nyamka-from-romano-userscript-markup-tools-autoformat .paste svg, .nyamka-from-romano-userscript-markup-tools-autotab .autoformat-active svg, .nyamka-from-romano-userscript-markup-tools-autotab .autoformat svg, .nyamka-from-romano-userscript-markup-tools-autotab .paste svg, .nyamka-from-romano-userscript-markup-tools-paste .autoformat-active svg, .nyamka-from-romano-userscript-markup-tools-paste .autoformat svg, .nyamka-from-romano-userscript-markup-tools-paste .paste svg {\n width: 11px;\n}\n.nyamka-from-romano-userscript-markup-tools-autoformat .autoformat-active, .nyamka-from-romano-userscript-markup-tools-autotab .autoformat-active, .nyamka-from-romano-userscript-markup-tools-paste .autoformat-active {\n display: none;\n}\n.nyamka-from-romano-userscript-markup-tools-autoformat.active .autoformat-active, .nyamka-from-romano-userscript-markup-tools-autotab.active .autoformat-active, .nyamka-from-romano-userscript-markup-tools-paste.active .autoformat-active {\n display: inline-block;\n}\n.nyamka-from-romano-userscript-markup-tools-autoformat.active .autoformat, .nyamka-from-romano-userscript-markup-tools-autotab.active .autoformat, .nyamka-from-romano-userscript-markup-tools-paste.active .autoformat {\n display: none;\n}\n\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autoformat, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autotab {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autoformat:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autoformat:focus, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autotab:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autotab:focus {\n background: #d8d8c4;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autoformat.active, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autotab.active {\n background: #c3a63f;\n color: black;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autoformat.active:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools-autotab.active:hover {\n background: #cfb866;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-markup-tools button {\n line-height: 15px;\n padding: 4px 6px;\n font-size: 14px;\n margin-right: 5px;\n margin: 5px 2px;\n margin-left: 0;\n}\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-markup-tools {\n margin-top: 6px;\n}\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-markup-tools button {\n margin-bottom: 0;\n}\n\nbody[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-markup-tools {\n margin-top: 6px;\n}\nbody[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-markup-tools button {\n padding: 3px 7px;\n margin-right: 6px;\n font-weight: normal;\n margin-bottom: 6px;\n font-size: 14px;\n}\nbody[data-nyamka-script=gikami-ru] form.form_book input.editor-title, body[data-nyamka-script=gikami-ru] form.form_book .medium-editor-book-element {\n border: 1px solid #dedfe2;\n border-radius: 8px;\n padding-left: 5px;\n padding-right: 5px;\n}\nbody[data-nyamka-script=gikami-ru] form.form_book .card-footer {\n position: fixed;\n bottom: 0;\n left: 0;\n right: 0;\n border-radius: 0;\n background: #23282d;\n}"; n(css$8,{}); var getItemAsString = function(item) { return new Promise(function(resolve, reject) { item.getAsString(resolve); }); }; var pasteModal = (function() { var modal, onClipboardPasteRmWatcher; modal = null; onClipboardPasteRmWatcher = null; return function() { if (modal) { modal.destroy(); } modal = new Modal({ title: "Вставка html-текста с сохранением форматирования", content: `<div class='${CSS_PREFIX}-chapters-importer-modal'> <div class="${CSS_PREFIX}-chapters-importer-modal-paste-area visible" style="margin: 0"> <div class='${CSS_PREFIX}-chapters-importer-modal-svg-paste'> ${SVG_ICON_PASTE} </div> <p><b>Вставьте html-текст из буфера обмена по Ctrl+V</b></p> <div><textarea class='form-control' placeholder="Вставьте текст из буфера обмена в это текстовое поле"></textarea></div> </div> </div>` }); return new Promise(function(resolve, reject) { onClipboardPasteRmWatcher = onClipboardPaste(async function(clipboardData) { var data, itemHtml, itemText, items; if (!modal || !modal.isVisible()) { onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; return; } data = null; items = Array.prototype.slice.call(clipboardData.items); itemText = items.find(function(item) { return item.type === 'text/plain'; }); itemHtml = items.find(function(item) { return item.type === 'text/html'; }); if (itemHtml) { data = (await getItemAsString(itemHtml)); } else if (itemText) { data = (await getItemAsString(itemText)); } if (data) { onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; modal.destroy(); resolve(data); } }); modal.show(); }); }; })(); var addTabsToText, exportChapterText, getOnFormatButtonClickFunction, initMarkupTools, onShowSettingsClick, indexOf$3 = [].indexOf; exportChapterText = async function(exportProvider) { var editor, modifier, text; modifier = exportProvider.split(':')[1]; exportProvider = exportProvider.split(':')[0]; editor = getChapterEditor(); if (!editor || !editor.body) { notification.error('Не получилось найти редактор текста на странице'); return; } text = editor.body.value || ''; if (isCurrentSite$3()) { text = fanficsMe[`to${exportProvider}`](text); } else if (isCurrentSite$2()) { text = ficbookNet[`to${exportProvider}`](text); if (exportProvider === 'FanficsMe' && modifier === 'blog') { text = text.replaceAll("<i>", "[i]").replaceAll("</i>", "[/i]").replaceAll("<b>", "[b]").replaceAll("</b>", "[/b]").replaceAll("<s>", "[s]").replaceAll("</s>", "[/s]").replaceAll("<center>", "").replaceAll("</center>", "").replaceAll("<right>", "").replaceAll("</right>", "").replaceAll("<hr>", "***"); } } (await copyToClipboard(text, 'text/html')); notification.success('Текст скопирован в буфер обмена!'); }; if (isCurrentSite()) { addTabsToText = function(text) { var formatted, i, len, line, ref; formatted = []; text = text.split("</p>").join("</p>\n"); ref = text.split('\n'); for (i = 0, len = ref.length; i < len; i++) { line = ref[i]; if (line.indexOf("<center>") === -1 && line.indexOf("<right>") === -1 && line.indexOf('***') === -1 && line.indexOf('* * *') === -1 && line.indexOf('text-indent: 2em') === -1 && line.trim()) { line = line.replace("<p>", "<p style='text-indent: 2em'>"); } formatted.push(line); } return formatted.join('\n'); }; } else { addTabsToText = function(text) { var formatted, i, len, line, ref; formatted = []; ref = text.split('\n'); for (i = 0, len = ref.length; i < len; i++) { line = ref[i]; if (line.indexOf("<center>") === -1 && line.indexOf("<right>") === -1 && line.indexOf('***') === -1 && line.indexOf('* * *') === -1 && !line.startsWith('<tab>') && line.trim()) { line = `<tab>${line}`; } formatted.push(line); } return formatted.join('\n'); }; } getOnFormatButtonClickFunction = function(transformFunction) { var textBeforeFormat; textBeforeFormat = null; return async function(event) { var cleanTextBeforeFormat, editor; event.preventDefault(); if (textBeforeFormat !== null && indexOf$3.call(this.classList, 'active') >= 0) { // нужно восстановление текста editor = getChapterEditor(); if (!editor || !editor.body) { return; } editor.body.value = textBeforeFormat; this.classList.remove('active'); return; } editor = getChapterEditor() || {}; if (!editor.body || !editor.body.value || !editor.body.value.trim()) { return; } cleanTextBeforeFormat = () => { this.classList.remove('active'); textBeforeFormat = null; editor = getChapterEditor() || {}; if (!editor.body) { return; } ['input', 'cut', 'paste'].forEach(function(eventName) { editor.body.removeEventListener(eventName, cleanTextBeforeFormat); }); }; // следим за тем, что если текст изменился, то и отменить форматирование уже нельзя textBeforeFormat = editor.body.value; editor.body.value = (await transformFunction(editor.body.value)); this.classList.add('active'); ['input', 'cut', 'paste'].forEach(function(eventName) { editor.body.addEventListener(eventName, cleanTextBeforeFormat); }); }; }; onShowSettingsClick = (function() { var getNotesSettings, modal; modal = null; getNotesSettings = function(siteName) { if (!isCurrentSite$3() && !isCurrentSite$2()) { return ''; } return `<div class='form-group form-inline'> <label>Экспортировать сноски как...</label> <select class='form-control input-sm' data-property='exportNotesStrategyFor${siteName}'> <option value='*'>звездочки, пример: *</option> <option value='number'>цифры, пример: (1)</option> </select> </div>`; }; return function() { var i, len, property, ref, selectElement, settings; if (modal) { modal.destroy(); } settings = []; if (isCurrentSite$3()) { settings.push(`<h4>Настройки экспорта в Ficbook-разметку</h4> <div class='form-group form-inline'> <label>Разделение абзацев c помощью...</label> <select class='form-control input-sm' data-property='exportParagraphStrategyForFicbookNet'> <option value='tab'><tab> в начале предложений</option> <option value=''>никак не отделять</option> </select> </div> ${getNotesSettings('FicbookNet')}`); } settings.push(`<h4>Настройки экспорта в AO3-разметку</h4> <div class='form-group form-inline'> <label>Красная строка с помощью...</label> <select class='form-control input-sm' data-property='exportParagraphStrategyForAO3'> <option value='4-spaces'>4-х пробелов в начале предложений</option> <option value=''>никак не отделять</option> </select> </div> ${getNotesSettings('AO3')}`); if (isCurrentSite$3() || isCurrentSite$2()) { settings.push(`<h4>Настройки экспорта в AuthorToday-разметку</h4> ${getNotesSettings('AuthorToday')}`); } modal = new Modal({ title: "Настройки и помощь", content: `${settings.join('<hr/>')} <hr/> <h4>Помощь</h4> <p>При нажатии на "экспорт" текст главы из этой страницы будет конвертирован в верстку выбранного типа и скопирован в буфер обмена.</p> <p>После этого вы можете перейти на другой сайт и просто вставить текст в редактор.</p> <p>Если верстка, на ваш взгляд, вставилась неправильно, то перейдите в режим редактирования html и вставьте заново текст (на скриншоте ниже показано на какие кнопки нужно нажимать).</p> <img style="width: 100%; max-width: 450px;" src="${IMG_HOW_TO_EXPORT}"> <hr/> <p><b>ВНИМАНИЕ! Не вся верстка может быть конвертирована! Например:</b></p> <ol> <li>тег <b><hr></b> в верстке для AuthorToday и Ficbook будет заменен пустой строкой</li> <li>перечеркнутые слова с помощью тега <b><s></b> в верстке для AuthorToday скорее всего пропадут (какой-то внутренний баг их редактора текста)</li> <li>на AO3 нет такого понятия как "красная строка", поэтому вы сами должны выбрать, как должны разделяться абзацы при экспорте</li> <li>теги сносок <b><note></b> и <b><footnote></b> при невозможности конвертирования будут указаны как "ПРИМЕЧАНИЯ" в конце текста</li> </ol>` }); ref = modal.element.querySelectorAll('select'); for (i = 0, len = ref.length; i < len; i++) { selectElement = ref[i]; property = selectElement.getAttribute('data-property'); if (!property) { continue; } selectElement.value = storage[property]; selectElement.addEventListener('change', function() { property = this.getAttribute('data-property'); storage[property] = this.value; storage.saveData(); }); } return modal.show(); }; })(); initMarkupTools = function() { var autoFormatButton, autoTabButton, container, dropdown, exporterItems, mEditor, pasteButton, toolbar; if (isCurrentSite$3()) { container = document.getElementsByClassName('TextEditorContainer')[0]; } else if (isCurrentSite$2()) { container = document.querySelector('.text-editor-container .editor-section'); } else if (isCurrentSite()) { mEditor = document.querySelector('textarea[name="content"]'); if (mEditor) { mEditor.parentNode.insertBefore(createElement("<div></div>"), mEditor.parentNode.children[0]); container = mEditor.parentNode.children[0]; } } if (!container) { setTimeout(initMarkupTools, 500); return; } if (container.querySelector(`.${CSS_PREFIX}-markup-tools`)) { return; } exporterItems = []; if (!isCurrentSite$2()) { exporterItems.push({ text: `<img src='${favicon$2}'>Ficbook-разметку`, onClick: function() { exportChapterText('FicbookNet'); } }); } if (!isCurrentSite$3()) { exporterItems.push({ //text: 'Fanfics-разметку (фанфики)' text: `<img src='${favicon$3}'>Fanfics-разметку`, onClick: function() { exportChapterText('FanficsMe'); } }); } /* exporterItems.push({ text: 'Fanfics-разметку (блогозаписи)' onClick: ()-> exportChapterText('FanficsMe:blog') return }) */ // остальные exporterItems.push({ text: `<img src='${favicon}'>Author.today-разметку`, onClick: function() { exportChapterText('AuthorToday'); } }); exporterItems.push({ text: `<img src='${favicon$1}'>AO3-разметку`, onClick: function() { exportChapterText('AO3'); } }); /* if !gikami.isCurrentSite() exporterItems.push({ text: 'Gikami-разметку' onClick: ()-> exportChapterText('Gikami') return }) */ exporterItems.push({ text: 'ℹ️ Настройка/помощь', onClick: function(event) { onShowSettingsClick(); } }); // помещаем в один контейнер все кнопки toolbar = createElement(`<div class='${CSS_PREFIX}-markup-tools'> </div>`); dropdown = new Dropdown('Экспорт в...', exporterItems); toolbar.appendChild(dropdown.element); if (isCurrentSite$3()) { autoFormatButton = createElement(`<button class='${CSS_PREFIX}-markup-tools-autoformat' title="Убрать лишние пробелы, заменить символы и т.п."> <span class='autoformat-active'>${SVG_ICON_FLOWER_FULL}</span> <span class='autoformat'>${SVG_ICON_FLOWER_EMPTY}</span> <span class='autoformat-active'>Отменить форматирование</span> <span class='autoformat'>Отформатировать текст</span> </button>`); autoFormatButton.addEventListener('click', getOnFormatButtonClickFunction(execute)); toolbar.appendChild(autoFormatButton); } if (isCurrentSite$2()) { autoTabButton = createElement(`<button class='btn btn-for-formatting-text ${CSS_PREFIX}-markup-tools-autotab' title="Автоматическая простановка <tab>"> <span class='autoformat-active'>${SVG_ICON_INDENT_FULL}</span> <span class='autoformat'>${SVG_ICON_INDENT_EMPTY}</span> <span class='autoformat-active'>Отменить простановку <tab></span> <span class='autoformat'>Проставить <tab></span> </button>`); } if (isCurrentSite()) { autoTabButton = createElement(`<button class='btn btn-for-formatting-text ${CSS_PREFIX}-markup-tools-autotab' title="Автоматическая простановка абзацев"> <span class='autoformat-active'>${SVG_ICON_INDENT_FULL}</span> <span class='autoformat'>${SVG_ICON_INDENT_EMPTY}</span> <span class='autoformat-active'>Отменить простановку абзацев</span> <span class='autoformat'>Проставить абзацы</span> </button>`); } if (autoTabButton) { autoTabButton.addEventListener('click', getOnFormatButtonClickFunction(addTabsToText)); toolbar.appendChild(autoTabButton); } if (isCurrentSite()) { pasteButton = createElement(`<button class='btn btn-for-formatting-text ${CSS_PREFIX}-markup-tools-paste' title="Вставка из буфера обмена с сохранением форматирования"> <span class='paste'>${SVG_ICON_PASTE}</span> Вставка из буфера обмена </button>`); pasteButton.addEventListener('click', async function(event) { var editor; event.preventDefault(); editor = getChapterEditor$1(); editor.body.value = (await pasteModal()); }); toolbar.appendChild(pasteButton); } container.appendChild(toolbar); }; var init$3 = function() { ['FanficsMe', 'FicbookNet', 'AuthorToday', 'AO3'].forEach(function(siteName) { storage.registerVariable(`exportNotesStrategyFor${siteName}`, 'number'); }); if (isCurrentSite$3()) { storage.registerVariable("exportParagraphStrategyForFicbookNet", 'tab'); } storage.registerVariable("exportParagraphStrategyForAO3", '4-spaces'); storage.registerVariable("exportParagraphStrategyForGikami", '2em'); onLocationChange(function() { if (isCurrentSite$3() && location.pathname === '/fic_write' && (location.search.indexOf('soaction=chapter_edit') > -1 || location.search.indexOf('soaction=chapter_new') > -1)) { initMarkupTools(); } if (isCurrentSite$2() && (/\/home\/myfics\/\d+.\/parts\/\d+.?$/.test(location.pathname) || /\/home\/myfics\/\d+.\/addpart/.test(location.pathname))) { initMarkupTools(); } if (isCurrentSite() && (/\/book\/\d+.\/add$/.test(location.pathname) || /\/book\/\d+.\/redact\/\d+.?$/.test(location.pathname))) { initMarkupTools(); } }); }; var css$7 = ".nyamka-from-romano-userscript-image-cropper-modal-loading {\n padding: 2em 0;\n padding-bottom: 3em;\n text-align: center;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-loading p {\n text-align: center;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-loading p img {\n position: relative;\n top: 4px;\n left: 4px;\n max-width: 16px;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-cropper {\n display: none;\n max-width: 100%;\n min-width: 100%;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-preview {\n overflow: hidden;\n max-height: 80vh;\n}\n@media (max-width: 700px) {\n .nyamka-from-romano-userscript-image-cropper-modal-preview {\n max-height: 65vh;\n }\n}\n.nyamka-from-romano-userscript-image-cropper-modal-btns-block {\n display: none;\n text-align: center;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-btns-block button {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n margin: 10px;\n margin-top: 0;\n}\n.nyamka-from-romano-userscript-image-cropper-modal-btns-block button:hover, .nyamka-from-romano-userscript-image-cropper-modal-btns-block button:focus {\n background: #d8d8c4;\n}"; n(css$7,{}); var imageCropperModal = (function() { var modal; modal = null; return function(imageFile) { if (modal) { modal.destroy(); } modal = new Modal({ title: "Обрезка картинки", content: `<div class='${CSS_PREFIX}-image-cropper-modal'> <div class='${CSS_PREFIX}-image-cropper-modal-loading'> <p> Подождите, пока зарузится редактор <img src="https://fanfics.me/images/load_2.gif"> </p> </div> <div class='${CSS_PREFIX}-image-cropper-modal-btns-block'> <button class='upload-cropped'>Загрузить обрезанный файл</button> <button class='upload-original'>Загрузить файл</button> </div> <div class='${CSS_PREFIX}-image-cropper-modal-preview'> <img class='${CSS_PREFIX}-image-cropper-modal-cropper' src="${URL.createObjectURL(imageFile)}"> </div> <div class='clear'></div> </div>` }); return new Promise(async function(resolve, reject) { var cropper; modal.show(); try { Cropper; } catch (error1) { await Promise.all([loadOnce.js('https://cdnjs.cloudflare.com/ajax/libs/cropperjs/1.5.12/cropper.min.js'), loadOnce.css('https://cdnjs.cloudflare.com/ajax/libs/cropperjs/1.5.12/cropper.min.css')]); } modal.element.querySelector(`.${CSS_PREFIX}-image-cropper-modal-loading`).style.display = 'none'; modal.element.querySelector(`.${CSS_PREFIX}-image-cropper-modal-cropper`).style.display = 'block'; modal.element.querySelector(`.${CSS_PREFIX}-image-cropper-modal-btns-block`).style.display = 'block'; cropper = new Cropper(modal.element.querySelector(`.${CSS_PREFIX}-image-cropper-modal-cropper`)); modal.element.querySelector("button.upload-cropped").addEventListener('click', function() { cropper.getCroppedCanvas().toBlob(function(blob) { modal.destroy(); modal = null; blob.name = imageFile.name; resolve(blob); }); }); modal.element.querySelector("button.upload-original").addEventListener('click', function() { modal.destroy(); modal = null; resolve(imageFile); }); }); }; })(); var IMG_IMGBB_API_KEY = ""; var css$6 = ".nyamka-from-romano-userscript-imgbb-api-key-modal img {\n width: 100%;\n max-width: 350px;\n margin-top: 0.5em;\n}\n.nyamka-from-romano-userscript-imgbb-api-key-modal input {\n border: 1px solid #D8D8C4;\n padding: 5px;\n background: #fff;\n color: #888;\n}\n.nyamka-from-romano-userscript-imgbb-api-key-modal input:focus {\n color: black;\n}\n.nyamka-from-romano-userscript-imgbb-api-key-modal button {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n}\n.nyamka-from-romano-userscript-imgbb-api-key-modal button:hover, .nyamka-from-romano-userscript-imgbb-api-key-modal button:focus {\n background: #d8d8c4;\n}\n.nyamka-from-romano-userscript-imgbb-api-key-modal input, .nyamka-from-romano-userscript-imgbb-api-key-modal button {\n margin: 0.5em 0;\n margin-right: 0.5em;\n}"; n(css$6,{}); var imgBbApiKeyModal = (function() { var modal; modal = null; return function() { var imgbbApiKey, inputElement, saveButtonElement; if (modal) { modal.destroy(); } modal = new Modal({ title: "Настройка загрузки картинок", content: `<div class='${CSS_PREFIX}-imgbb-api-key-modal'> <input type='text' required='' placeholder='Вставьте сюда API key'> <button>Сохранить API key</button> <hr> <p>Для работы с загрузкой файлов вам нужно зарегистрировать API key на сайте <b>https://ru.imgbb.com</b></p> <ol> <li>Перейдите на <a href='https://imgbb.com/login' target='_blank'>https://imgbb.com/login</a> и войдите на сайт (или зарегистрируйтесь, если вас там нет)</li> <li>Перейдите на <a href='https://api.imgbb.com/' target='_blank'>https://api.imgbb.com</a> и найдите на странице большую голубую кнопку "Get API key". Нажмите на нее</li> <li> Скопируйте полученый ключ из текстового инпута (на скриншоте ниже это "bla-blabla-blabla-bla") в текстовый инпут в этом модальном окне <div></div> <img src="${IMG_IMGBB_API_KEY}"> </li> </ol> <p><b>ВНИМАНИЕ:</b> если вы будете удалять фото в личном кабинете https://ru.imgbb.com, но тут они тоже удалятся!</p> </div>` }); imgbbApiKey = storage.imgbbApiKey; saveButtonElement = modal.element.querySelector('button'); inputElement = modal.element.querySelector('input'); inputElement.value = imgbbApiKey; saveButtonElement.disabled = !imgbbApiKey; return new Promise(function(resolve, reject) { modal.show(); inputElement.addEventListener('input', function() { imgbbApiKey = this.value; saveButtonElement.disabled = !this.value; }); saveButtonElement.addEventListener('click', function() { if (!imgbbApiKey) { return; } storage.imgbbApiKey = imgbbApiKey; storage.saveData(); modal.destroy(); modal = null; resolve(imgbbApiKey); }); }); }; })(); var SVG_ICON_UPLOAD = "<svg focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 640 512\"><path fill=\"currentColor\" d=\"M537.6 226.6c4.1-10.7 6.4-22.4 6.4-34.6 0-53-43-96-96-96-19.7 0-38.1 6-53.3 16.2C367 64.2 315.3 32 256 32c-88.4 0-160 71.6-160 160 0 2.7.1 5.4.2 8.1C40.2 219.8 0 273.2 0 336c0 79.5 64.5 144 144 144h368c70.7 0 128-57.3 128-128 0-61.9-44-113.6-102.4-125.4zM393.4 288H328v112c0 8.8-7.2 16-16 16h-48c-8.8 0-16-7.2-16-16V288h-65.4c-14.3 0-21.4-17.2-11.3-27.3l105.4-105.4c6.2-6.2 16.4-6.2 22.6 0l105.4 105.4c10.1 10.1 2.9 27.3-11.3 27.3z\" class=\"\"></path></svg>\n"; var css$5 = ".nyamka-from-romano-userscript-upload-image-modal-dnd-area {\n display: none;\n text-align: center;\n padding: 4em 2em;\n margin-bottom: 1em;\n background-color: #c3a63f;\n outline: 2px dashed #fcfaf2;\n outline-offset: -10px;\n -webkit-transition: outline-offset 0.15s ease-in-out, background-color 0.15s linear;\n transition: outline-offset 0.15s ease-in-out, background-color 0.15s linear;\n text-align: center;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area.is-dragover {\n outline-offset: -20px;\n outline-color: #c3a63f;\n background-color: #d8d8c4;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area.is-dragover p, .nyamka-from-romano-userscript-upload-image-modal-dnd-area.is-dragover label, .nyamka-from-romano-userscript-upload-image-modal-dnd-area.is-dragover svg {\n color: #c3a63f;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area p, .nyamka-from-romano-userscript-upload-image-modal-dnd-area label {\n text-align: center !important;\n color: #fcfaf2;\n margin: 0.5em 0;\n font-size: 120%;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area label {\n text-decoration: underline;\n cursor: pointer;\n}\n.nyamka-from-romano-userscript-upload-image-modal-dnd-area input[type=file] {\n display: none;\n}\n.nyamka-from-romano-userscript-upload-image-modal-svg-upload {\n max-width: 5em;\n text-align: center;\n margin: 0 auto;\n color: #fcfaf2;\n}\n.nyamka-from-romano-userscript-upload-image-modal-errors {\n display: none;\n padding: 5em 0;\n text-align: center;\n text-align: center;\n}\n.nyamka-from-romano-userscript-upload-image-modal-errors.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-upload-image-modal-errors ul {\n margin-bottom: 1em;\n text-align: center;\n}\n.nyamka-from-romano-userscript-upload-image-modal-errors ul li {\n color: #ca0000;\n margin-bottom: 0.5em;\n}\n.nyamka-from-romano-userscript-upload-image-modal-description p {\n margin: 0.5em 0;\n text-align: left;\n}\n.nyamka-from-romano-userscript-upload-image-modal-description .change-api-key {\n cursor: pointer;\n text-decoration: underline;\n}\n.nyamka-from-romano-userscript-upload-image-modal-upload-progress {\n display: none;\n padding: 2em 0;\n text-align: center;\n display: none;\n}\n.nyamka-from-romano-userscript-upload-image-modal-upload-progress.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-upload-image-modal-upload-progress li {\n margin: 0.5em 0;\n text-align: center;\n}\n.nyamka-from-romano-userscript-upload-image-modal-upload-progress button {\n display: none;\n margin-top: 3em;\n}\n.nyamka-from-romano-userscript-upload-image-modal-upload-progress button.visible {\n display: inline-block;\n}\n.nyamka-from-romano-userscript-upload-image-modal button {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n margin: 0.5em 0;\n margin-right: 0.5em;\n}\n.nyamka-from-romano-userscript-upload-image-modal button:hover, .nyamka-from-romano-userscript-upload-image-modal button:focus {\n background: #d8d8c4;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status {\n font-weight: bold;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status.fail {\n color: #ca0000;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status.fail img {\n display: none;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status.success {\n color: green;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status.success img {\n display: none;\n}\n.nyamka-from-romano-userscript-upload-image-modal .status img {\n position: relative;\n top: 3px;\n left: 3px;\n max-width: 16px;\n}\n.nyamka-from-romano-userscript-upload-image-modal .visible-lg {\n display: none;\n}\n@media (min-width: 1023px) {\n .nyamka-from-romano-userscript-upload-image-modal .visible-lg {\n display: block;\n }\n}"; n(css$5,{}); var IMAGE_ALLOWED_EXT, IMAGE_ALLOWED_MAX_SIZE, isFileAllowedForUpload, uploadImage, indexOf$2 = [].indexOf; IMAGE_ALLOWED_EXT = ['jpg', 'jpeg', 'png', 'gif', 'tiff', 'bmp']; IMAGE_ALLOWED_MAX_SIZE = 30 * 1024 * 1024; // 30 mb isFileAllowedForUpload = function(file) { var ref; if (ref = file.type.split('/')[1], indexOf$2.call(IMAGE_ALLOWED_EXT, ref) < 0) { return `Файл <b>${file.name}</b> нельзя загружать, т.к. это неразрешенный формат ${file.type || 'unknown'}. Выберите другой файл`; } if (file.size > IMAGE_ALLOWED_MAX_SIZE) { return `Файл <b>${file.name}</b> нельзя загружать, т.к. его размер >= 30 мегабайтам. Уменьшите картинку в фоторедакторе`; } return true; }; uploadImage = async function(image) { var form; form = new FormData(); form.append('image', image, image.name); return (await fetch(`https://api.imgbb.com/1/upload?key=${storage.imgbbApiKey}`, { method: 'POST', body: form })); }; var uploadImageModal = (function() { var modal, onClipboardPasteRmWatcher; modal = null; onClipboardPasteRmWatcher = null; return async function() { if (!storage.imgbbApiKey) { (await imgBbApiKeyModal()); } if (modal) { modal.destroy(); } if (onClipboardPasteRmWatcher) { onClipboardPasteRmWatcher(); } modal = new Modal({ title: "Загрузка изображения", content: `<div class='${CSS_PREFIX}-upload-image-modal'> <div class="${CSS_PREFIX}-upload-image-modal-dnd-area visible"> <div class='${CSS_PREFIX}-upload-image-modal-svg-upload'> ${SVG_ICON_UPLOAD} </div> <p><b>Перетащите изображение в эту область для начала загрузки</b></p> <p><small>или</small></p> <label for='${CSS_PREFIX}-label-image-upload' role='button'><b>Выберите изображение</b></label> <input type='file' id='${CSS_PREFIX}-label-image-upload' accept='${IMAGE_ALLOWED_EXT.map(function(ext) { return "image/" + ext; })}' multiple> <div class='visible-lg'> <p><small>или</small></p> <p><b>Вставьте изображение из буфера обмена по Ctrl+V</b> <small>(может работать не во всех браузерах и не со всеми картинками)</small></p> </div> </div> <div class='${CSS_PREFIX}-upload-image-modal-upload-progress'> <div></div> <button>Вставить ссылки в текст</button> </div> <div class='${CSS_PREFIX}-upload-image-modal-errors'> <div class='${CSS_PREFIX}-upload-image-modal-errors-content'></div> <button>Выбрать файлы заново</button> </div> <div class='${CSS_PREFIX}-upload-image-modal-description'> <p>Файлы загружаются через <a href='https://ru.imgbb.com/' target='_blank'>https://ru.imgbb.com</a></p> <p>Чтоб настроить API key <a role='button' class='change-api-key'>нажмите сюда</a></p> </div> </div>` }); return new Promise(function(resolve, reject) { var areaElement, errorsElement, highlightArea, onFilesSelect, unhighlightArea; modal.show(); highlightArea = function(event) { areaElement.classList.add('is-dragover'); }; unhighlightArea = function() { areaElement.classList.remove('is-dragover'); }; onFilesSelect = async function(files) { var cropped, errors, file, isAllUploaded, isAllowed, j, len, progressElement, progressLis, promises, results; console.log('SELECTED', files); // проверяем, разрешена ли загрузка для выбранных файлов errors = []; for (j = 0, len = files.length; j < len; j++) { file = files[j]; isAllowed = isFileAllowedForUpload(file); if (isAllowed !== true) { errors.push(isAllowed); } } if (errors.length) { areaElement.classList.remove('visible'); errorsElement.classList.add('visible'); errorsElement.querySelector(`.${CSS_PREFIX}-upload-image-modal-errors-content`).replaceChildren(createElement(`<ul> ${errors.map(function(text) { return `<li>${text}</li>`; }).join('')} </ul>`)); return; } if (files.length === 1) { // открываем мод окно с обрезкой cropped = (await imageCropperModal(files[0])); files = [cropped]; } if (files.length) { // начинаем загрузку progressLis = []; promises = Array.prototype.slice.call(files).map(async function(file, i) { var body, result; progressLis.push(`${file.name} - <span class='status'>загружается<img src='https://fanfics.me/images/load_2.gif'></span>`); result = (await uploadImage(file)); try { body = (await result.json()); } catch (error1) { body = { error: { message: 'Bad Request' } }; } if (result.status !== 200) { progressElement.querySelectorAll('li')[i].children[0].textContent = `Не получилось загрузить. Ошибка сервера ru.imgbb.com: ${body.error.message}`; progressElement.querySelectorAll('li')[i].children[0].classList.add('fail'); return false; } progressElement.querySelectorAll('li')[i].children[0].textContent = "Загружено"; progressElement.querySelectorAll('li')[i].children[0].classList.add('success'); if (body.data && body.data.url) { if (extname(body.data.url) === '.gif') { return body.data.url + '?.jpg'; } return body.data.url; } progressElement.querySelectorAll('li')[i].children[0].textContent = "Не получилось загрузить"; progressElement.querySelectorAll('li')[i].children[0].classList.add('fail'); return false; }); areaElement.classList.remove('visible'); errorsElement.classList.remove('visible'); progressElement = modal.element.querySelector(`.${CSS_PREFIX}-upload-image-modal-upload-progress`); progressElement.classList.add('visible'); progressElement.children[0].replaceChildren(createElement(`<ul> ${progressLis.map(function(html) { return `<li>${html}</li>`; }).join('')} </ul>`)); progressElement.querySelector('button').addEventListener('click', function() { modal.destroy(); onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; // готовые ссылки для вставки в документ resolve(results.filter(function(url) { return !!url; }).join('\n')); }); results = (await Promise.all(promises)); isAllUploaded = results.every(function(r) { return !!r; }); if (isAllUploaded) { // если все загружено, то сразу и закрываем мод окно modal.destroy(); onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; // готовые ссылки для вставки в документ resolve(results.filter(function(url) { return !!url; }).join('\n')); return; } progressElement.querySelector('button').classList.add('visible'); } }; modal.element.querySelector('.change-api-key').addEventListener('click', imgBbApiKeyModal); errorsElement = modal.element.querySelector(`.${CSS_PREFIX}-upload-image-modal-errors`); errorsElement.querySelector('button').addEventListener('click', function() { areaElement.classList.add('visible'); errorsElement.classList.remove('visible'); areaElement.querySelector("input[type='file']").value = ''; }); areaElement = modal.element.querySelector(`.${CSS_PREFIX}-upload-image-modal-dnd-area`); areaElement.querySelector("input[type='file']").addEventListener('change', function(event) { return onFilesSelect(this.files || []); }); ['drag', 'dragstart', 'dragend', 'dragover', 'dragenter', 'dragleave', 'drop'].forEach(function(eventName) { areaElement.addEventListener(eventName, function(event) { event.preventDefault(); event.stopPropagation(); }); }); ['dragover', 'dragenter'].forEach(function(eventName) { areaElement.addEventListener(eventName, highlightArea); }); ['dragleave', 'dragend', 'drop'].forEach(function(eventName) { areaElement.addEventListener(eventName, unhighlightArea); }); areaElement.addEventListener('drop', function(event) { var droppedFiles; if (event.originalEvent) { droppedFiles = event.originalEvent.dataTransfer.files; } else { droppedFiles = event.dataTransfer.files; } onFilesSelect(droppedFiles || []); }); onClipboardPasteRmWatcher = onClipboardPaste(function(clipboardData) { var files, item, j, len, ref; if (!modal || !modal.isVisible()) { onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; return; } files = []; ref = clipboardData.items; for (j = 0, len = ref.length; j < len; j++) { item = ref[j]; if (item.kind === 'file') { files.push(item.getAsFile()); } } if (files.length) { onFilesSelect(files); } }); }); }; })(); var SVG_ICON_SMILEY_FACE = "<svg focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 496 512\"><path fill=\"currentColor\" d=\"M248 8C111 8 0 119 0 256s111 248 248 248 248-111 248-248S385 8 248 8zm141.4 389.4c-37.8 37.8-88 58.6-141.4 58.6s-103.6-20.8-141.4-58.6S48 309.4 48 256s20.8-103.6 58.6-141.4S194.6 56 248 56s103.6 20.8 141.4 58.6S448 202.6 448 256s-20.8 103.6-58.6 141.4zM328 224c17.7 0 32-14.3 32-32s-14.3-32-32-32-32 14.3-32 32 14.3 32 32 32zm-160 0c17.7 0 32-14.3 32-32s-14.3-32-32-32-32 14.3-32 32 14.3 32 32 32zm194.4 64H133.6c-8.2 0-14.5 7-13.5 15 7.5 59.2 58.9 105 121.1 105h13.6c62.2 0 113.6-45.8 121.1-105 1-8-5.3-15-13.5-15z\" class=\"\"></path></svg>\n"; var css$4 = ".emoji-picker__wrapper {\n z-index: 100;\n}\n.emoji-picker__wrapper .emoji-picker.light, .emoji-picker__wrapper .emoji-picker.dark {\n --emoji-size: 1.2em !important;\n}"; n(css$4,{}); var TRANSLATIONS; TRANSLATIONS = { search: 'Искать эмоджи...', categories: { recents: 'Недавние', smileys: 'Смайлы и эмоции', people: 'Люди и тело', animals: 'Животные и природа', food: 'Еда и напитки', activities: 'Мероприятия', travel: 'Путешествия и места', objects: 'Объекты', symbols: 'Символы', flags: 'Флаги', custom: 'Пользовательские' }, notFound: 'Эмоджи не найдены' }; var generateEmojiPickerButton = function(options) { var button, picker; options = options || {}; options.cssClass = options.cssClass || `${CSS_PREFIX}-emoji-button`; picker = null; button = createElement(`<button class='${options.cssClass}' title='Эмоджи'>${SVG_ICON_SMILEY_FACE}</button>`); button.addEventListener('click', async function(event) { var emojiModule, theme; event.preventDefault(); if (!picker) { emojiModule = (await loadOnce.js("https://cdn.jsdelivr.net/npm/@joeattardi/emoji-button@4.6.0/dist/index.min.js", { module: true })); theme = 'light'; if (storage.isDarkModeEnabled) { theme = 'dark'; } picker = new emojiModule.EmojiButton({ i18n: TRANSLATIONS, showVariants: false, initialCategory: 'recents', theme }); picker.on('emoji', function(selection) { event = new CustomEvent('emojiSelected', { detail: selection }); button.dispatchEvent(event); }); } picker.togglePicker(button); }); return button; }; var css$3 = "button.nyamka-from-romano-userscript-tools-upload-button, button.nyamka-from-romano-userscript-tools-emoji-button {\n display: inline-block;\n padding: 3px 10px 4px 10px;\n font-size: 12px;\n font-family: Tahoma, Roboto, Verdana, Sans-Serif, Lucida Sans, ubuntu;\n border-radius: 2px;\n background: #efefdb;\n border: #92927e 1px solid;\n cursor: pointer;\n margin-left: 1em;\n}\nbutton.nyamka-from-romano-userscript-tools-upload-button:hover, button.nyamka-from-romano-userscript-tools-upload-button:focus, button.nyamka-from-romano-userscript-tools-emoji-button:hover, button.nyamka-from-romano-userscript-tools-emoji-button:focus {\n background: #d8d8c4;\n}\nbutton.nyamka-from-romano-userscript-tools-upload-button svg, button.nyamka-from-romano-userscript-tools-emoji-button svg {\n display: none;\n}\n@media (max-width: 700px) {\n button.nyamka-from-romano-userscript-tools-upload-button, button.nyamka-from-romano-userscript-tools-emoji-button {\n padding-top: 0;\n padding-bottom: 0;\n position: relative;\n line-height: 1;\n top: 4px;\n }\n button.nyamka-from-romano-userscript-tools-upload-button span.text, button.nyamka-from-romano-userscript-tools-emoji-button span.text {\n display: none;\n }\n button.nyamka-from-romano-userscript-tools-upload-button svg, button.nyamka-from-romano-userscript-tools-emoji-button svg {\n width: 1.9em;\n display: inline-block;\n }\n}\n\nbutton.nyamka-from-romano-userscript-tools-emoji-button {\n padding-left: 3px;\n padding-right: 3px;\n padding-top: 0;\n}\nbutton.nyamka-from-romano-userscript-tools-emoji-button svg {\n display: inline-block;\n width: 1.2em;\n position: relative;\n top: 3.5px;\n}\n@media (max-width: 1023px) {\n button.nyamka-from-romano-userscript-tools-emoji-button {\n display: none;\n }\n}\n\n.MessageCommentNewInstr button.nyamka-from-romano-userscript-tools-upload-button, .MessageCommentNewInstr button.nyamka-from-romano-userscript-tools-emoji-button, .DialogNewMsgButton button.nyamka-from-romano-userscript-tools-upload-button, .DialogNewMsgButton button.nyamka-from-romano-userscript-tools-emoji-button {\n float: left;\n top: 0;\n}\n.MessageCommentNewInstr button.nyamka-from-romano-userscript-tools-emoji-button, .DialogNewMsgButton button.nyamka-from-romano-userscript-tools-emoji-button {\n padding-bottom: 3.3px;\n}\n\nbody[data-ffme-extender-script=true] button.nyamka-from-romano-userscript-tools-upload-button, body[data-ffme-extender-script=true] button.nyamka-from-romano-userscript-tools-emoji-button {\n display: none;\n}\nbody[data-ffme-extender-script=true] button.ffe-emoji svg {\n width: 15px;\n padding-top: 2px;\n}\nbody[data-ffme-extender-script=true] .MessageNewInfoHead {\n margin-top: 5px;\n padding-left: 0;\n}\nbody[data-ffme-extender-script=true] .MessageNewInfoHead button:first-of-type {\n margin-left: 0;\n}"; n(css$3,{}); var getEditorForm, getTextareaPlaceholderDiv, injectButtons, injectButtonsIntoFfmeToolbar, insertTextInCommentForm, onEmojiButtonClick, onUploadButtonClick, indexOf$1 = [].indexOf; getEditorForm = function(child) { if (!child || !child.classList || !child.nodeName) { return; } if (child.classList && ((indexOf$1.call(child.classList, 'MessageCommentNewForm') >= 0) || (indexOf$1.call(child.classList, 'MessageNewForm') >= 0) || (indexOf$1.call(child.classList, 'DialogNewMsgForm') >= 0))) { return child; } if (child.nodeName === 'FORM' && child.id && (child.id.startsWith('messageEditForm') || child.id.startsWith('message_commentEditForm'))) { return child; } return getEditorForm(child.parentNode); }; getTextareaPlaceholderDiv = function(child) { var mn; if (!child) { return; } if (child.nodeName === 'FORM' && child.id && child.id.startsWith('message_commentEditForm')) { return; } // форма редактора уже созданного коммента в треде mn = child.querySelector('div.MessageNew'); if (mn) { return mn; } return getTextareaPlaceholderDiv(child.parentNode); }; insertTextInCommentForm = async function(commentForm, delimeter, insertFunction) { var caretPosition, i, isEmptyTextarea, len, newText, ref, tarea, textarea, valueBegin, valueEnd; textarea = null; ref = commentForm.querySelectorAll('textarea'); for (i = 0, len = ref.length; i < len; i++) { tarea = ref[i]; if (tarea.tabIndex !== '-1') { textarea = tarea; } } if (!textarea) { return; } caretPosition = getInputCaretPosition(textarea); newText = (await insertFunction()); if (!newText) { return; } valueBegin = textarea.value.substring(0, caretPosition.startPos); valueEnd = textarea.value.substring(caretPosition.endPos, textarea.value.length); isEmptyTextarea = !textarea.value.trim(); textarea.value = ''; delimeter = delimeter || ''; textarea.value = valueBegin; if (delimeter && valueBegin[valueBegin.length - 1] !== delimeter) { textarea.value += delimeter; } textarea.value += newText; caretPosition = textarea.value.length; if (valueEnd && valueEnd[0] !== delimeter) { textarea.value += delimeter; } textarea.value += valueEnd; setTimeout(function() { var placeholder; textarea.focus(); textarea.click(); textarea.setSelectionRange(caretPosition, caretPosition); if (!isEmptyTextarea) { return; } // ищем элемент, который мимикрирует под textarea и нажимаем на него, чтоб сфокусировался textarea placeholder = getTextareaPlaceholderDiv(textarea); if (placeholder) { placeholder.click(); } }); }; onUploadButtonClick = function(commentForm) { insertTextInCommentForm(commentForm, '\n', uploadImageModal); }; onEmojiButtonClick = function(commentForm, emojiSymbol) { insertTextInCommentForm(commentForm, '', function() { return emojiSymbol.emoji; }); }; injectButtons = function(commentForm) { var emojiButton, inputSubmit, tools, uploadButton; if (!commentForm || commentForm.querySelector(`button.${CSS_PREFIX}-tools-upload-button`)) { return; } tools = commentForm.querySelector('.MessageCommentNewInstr') || commentForm.querySelector('.MessageNewInstr') || commentForm.querySelector('.CommentEditManagement') || commentForm.querySelector('.DialogNewMsgButton > div'); // панель инструментов новых сообщений в личке if (!tools) { return; } uploadButton = createElement(`<button class='${CSS_PREFIX}-tools-upload-button'>${SVG_ICON_UPLOAD}<span class='text'>Загрузить картинку</span></button>`); emojiButton = generateEmojiPickerButton({ cssClass: `${CSS_PREFIX}-tools-emoji-button` }); inputSubmit = tools.querySelector("input[type='submit']"); if (inputSubmit.nextSibling) { tools.insertBefore(uploadButton, inputSubmit.nextSibling); tools.insertBefore(emojiButton, uploadButton.nextSibling); } else { tools.appendChild(uploadButton); tools.appendChild(emojiButton); } uploadButton.addEventListener('click', function(event) { event.preventDefault(); onUploadButtonClick(commentForm); }); emojiButton.addEventListener('emojiSelected', function(event) { event.preventDefault(); onEmojiButtonClick(commentForm, event.detail); }); }; injectButtonsIntoFfmeToolbar = function(commentForm) { var emojiButton, eraserButton, helloButton, uploadButton, uploadButtonPatched; if (!commentForm || commentForm.querySelector("button[data-nyamka-script-patched='true']")) { return; } // перехватить события от нажатия не получится // полностью заменяем кнопку uploadButton = commentForm.querySelector('button.ffe-image'); uploadButtonPatched = uploadButton.cloneNode(true); // помечаем кнопку, чтоб опять не патчить форму. НЕ ПЕРЕНОСИТЬ ЭТО СВОЙСТВО В form! uploadButtonPatched.setAttribute('data-nyamka-script-patched', true); uploadButton.parentNode.replaceChild(uploadButtonPatched, uploadButton); uploadButtonPatched.addEventListener('click', function(event) { event.preventDefault(); onUploadButtonClick(commentForm); }); // заменяем кнопку привет-ведьмы emojiButton = generateEmojiPickerButton({ cssClass: "ffe-blog-text-attr ffe-emoji" }); helloButton = commentForm.querySelector('button.ffe-hello'); if (helloButton) { helloButton.parentNode.replaceChild(emojiButton, helloButton); } else { // это редактор нового поста в профиле eraserButton = commentForm.querySelector('button.ffe-eraser'); if (eraserButton && eraserButton.nextSibling) { eraserButton.parentNode.insertBefore(emojiButton, eraserButton.nextSibling); } else { emojiButton = null; } } if (emojiButton) { emojiButton.addEventListener('emojiSelected', function(event) { event.preventDefault(); onEmojiButtonClick(commentForm, event.detail); }); } // помечаем страницу как такую, которая содержит сторонний скрипт document.body.setAttribute('data-ffme-extender-script', true); }; var init$2 = function() { if (!isCurrentSite$3()) { return; } storage.registerVariable('imgbbApiKey', null); // наблюдение за новым контентом на странице onLocationChange(function() { if (location.pathname.startsWith('/user')) { document.addEventListener('DOMContentLoaded', function() { return injectButtons(document.querySelector('#MessageNewPost')); }); return; } if (location.pathname.startsWith('/message')) { document.addEventListener('DOMContentLoaded', function() { return injectButtons(document.querySelector('.MessageCommentNewForm > form')); }); return; } }); onDomMutations(function(mutationsList) { var form, i, j, len, len1, m, node, ref; for (i = 0, len = mutationsList.length; i < len; i++) { m = mutationsList[i]; if (m.addedNodes.length) { ref = m.addedNodes; for (j = 0, len1 = ref.length; j < len1; j++) { node = ref[j]; if (!(node)) { continue; } if (node.nodeName === 'BUTTON' && indexOf$1.call(node.classList, 'ffe-image') >= 0) { // панель от Fanfics Extender. кнопка загрузки картинок injectButtonsIntoFfmeToolbar(getEditorForm(node), node); return; } if (node.nodeName === 'DIV' && (indexOf$1.call(node.classList, 'MessageCommentNewForm_Container') >= 0 || indexOf$1.call(node.classList, 'MessageNewForm') >= 0)) { // нашли панель для работы с сообщениями injectButtons(node); return; } if (node.nodeName === 'FORM' && indexOf$1.call(node.classList, 'DialogNewMsgForm') >= 0) { // редактор сообщений в личке injectButtons(node); return; } if (node.nodeName === 'TR' && (indexOf$1.call(node.classList, 'MessageComments') >= 0)) { form = node.querySelector('.MessageCommentNewForm_Container'); if (form) { injectButtons(form); return; } } if (node.nodeName === 'TEXTAREA' && (indexOf$1.call(node.classList, 'MessageCommentNew') >= 0 || indexOf$1.call(node.classList, 'CommentEditText') >= 0 || indexOf$1.call(node.classList, 'MessageNew') >= 0)) { // редактор нового поста на странице профиля form = getEditorForm(node); if (form) { injectButtons(form); return; } } if (node.nodeName === 'INPUT' && node.type === 'submit' && indexOf$1.call(node.classList, 'modern_button') >= 0) { // когда в треде создали новый коммент (не на отдельной странице, а в списке "блого-записей") // или когда в личке создали новое сообщение form = getEditorForm(node); if (form) { injectButtons(form); return; } } } } } }); }; var OfficeHtmlToSimpleHtmlConverter, getNodeName, indexOf = [].indexOf; getNodeName = function(node) { if (node && node.nodeName) { return node.nodeName.toLowerCase(); } return ''; }; OfficeHtmlToSimpleHtmlConverter = class OfficeHtmlToSimpleHtmlConverter { constructor() { this.allowedSimpleTags = ['b', 'i', 's', 'u']; this.allowedTags = ['h1', 'h2', 'center', 'right']; this.ignoreBackgrounds = [ "#cccccc", "rgb(204, 204, 204)", //серый "#b7b7b7", "rgb(183, 183, 183)", "#999999", "rgb(153, 153, 153)", "#666666", "rgb(102, 102, 102)", "#434343", "rgb(67, 67, 67)", "#ff0000", "rgb(255, 0, 0)", //красный "#cc0000", "rgb(204, 0, 0)" ]; } isNodeNameSimpleTag(nodeName) { if (isString(nodeName)) { return indexOf.call(this.allowedSimpleTags, nodeName) >= 0; } return true; } generateOpenTag(nodeName) { if (isString(nodeName)) { nodeName = [nodeName]; } return nodeName.filter((n) => { return indexOf.call(this.allowedTags, n) >= 0 || indexOf.call(this.allowedSimpleTags, n) >= 0; }).map(function(n) { return `<${n}>`; }).join(''); } generateCloseTag(nodeName) { var closeTag; if (isString(nodeName)) { nodeName = [nodeName]; } closeTag = nodeName.filter((n) => { return indexOf.call(this.allowedTags, n) >= 0 || indexOf.call(this.allowedSimpleTags, n) >= 0; }).reverse().map(function(n) { return `</${n}>`; }).join(''); if (indexOf.call(nodeName, 'center') >= 0 || indexOf.call(nodeName, 'right') >= 0) { closeTag += '\n'; } return closeTag; } getNodeName(elementDom) { var _nodeName, nodeName, ref, ref1; nodeName = getNodeName(elementDom); if ((nodeName === 'span' || nodeName === 'p') && elementDom && elementDom.style) { _nodeName = []; /* порядок тегов важен! сначала должны находиться center/right */ if ((ref = elementDom.style['text-align']) === 'center' || ref === 'right') { _nodeName.push(elementDom.style['text-align']); } if ((ref1 = elementDom.style['font-weight']) === '700' || ref1 === '900') { _nodeName.push('b'); } if ((elementDom.style['font-style']) === 'italic') { _nodeName.push('i'); } if ((elementDom.style['text-decoration']) === 'underline') { _nodeName.push('u'); } if (_nodeName.length) { return _nodeName; } } return nodeName; } isElementIgnored(elementDom) { var ref, ref1; if (elementDom && elementDom.style) { return (ref = elementDom.style['background-color'], indexOf.call(this.ignoreBackgrounds, ref) >= 0) || (ref1 = elementDom.style['background'], indexOf.call(this.ignoreBackgrounds, ref1) >= 0); } return false; } domToSimpleHtml(elementDom) { var child, j, len, nodeName, ref, simpleHtml; if (this.isElementIgnored(elementDom)) { return ''; } nodeName = this.getNodeName(elementDom); if (nodeName === 'br') { return '<empty-line>\n'; } if (nodeName === 'hr') { return '<hr>\n'; } if ((!elementDom.children || !elementDom.children.length) && !elementDom.textContent) { if (nodeName === 'p' && !nodeName.length) { return '<empty-line>\n'; } return ''; } if (nodeName === 'h1' || nodeName === 'h2') { return `<${nodeName}>${elementDom.textContent}</${nodeName}>\n`; } simpleHtml = this.generateOpenTag(nodeName); if (!elementDom.children || !elementDom.children.length) { simpleHtml += elementDom.textContent; } ref = elementDom.children || []; for (j = 0, len = ref.length; j < len; j++) { child = ref[j]; simpleHtml = `${simpleHtml}${this.domToSimpleHtml(child)}`; } simpleHtml = simpleHtml + this.generateCloseTag(nodeName); if (nodeName === 'p') { simpleHtml += '\n'; } return simpleHtml; } cleanGoogleDocsHtmlText(htmlText) { var div, doc, j, k, len, len1, mainNode, node, nodes, ref; doc = createElement(htmlText); nodes = []; try { mainNode = doc.querySelector("b[id^=docs-internal-guid-]"); if (mainNode) { nodes = mainNode.children; } } catch (error1) { } if (!nodes.length) { ref = createElement(htmlText).children; // быстро найти не получилось for (j = 0, len = ref.length; j < len; j++) { node = ref[j]; if (getNodeName(node) !== 'meta') { nodes.push(node); } } if ((nodes.length === 2 && getNodeName(nodes[1]) === 'br') || nodes.length === 1) { nodes = nodes[0].children; } } div = document.createElement('div'); for (k = 0, len1 = nodes.length; k < len1; k++) { node = nodes[k]; // баг с игнором дублирующихся нодов div.appendChild(node.cloneNode(true)); } return div; } cleanLibreOfficeHtmlText(htmlText) { var doc; // TODO: доделать парсе для libreOffice doc = createElement(htmlText); return doc.querySelector('body'); } cleanDefaultHtmlText(htmlText) { return this.cleanGoogleDocsHtmlText(htmlText); } detectOfficeType(htmlText) { if (htmlText.includes('id="docs-internal-guid-')) { return 'googleDocs'; } if (htmlText.startsWith("<!DOCTYPE HTML PUBLIC") && (htmlText.indexOf("LibreOffice") > -1 || htmlText.indexOf("OpenOffice") > -1)) { return 'libreOffice'; } return 'html'; } convert(htmlText) { var cleaned, docType, domElement, text; docType = this.detectOfficeType(htmlText); if (docType === 'googleDocs') { domElement = this.cleanGoogleDocsHtmlText(htmlText); } else if (docType === 'libreOffice') { domElement = this.cleanLibreOfficeHtmlText(htmlText); } else { domElement = this.cleanDefaultHtmlText(htmlText); } text = this.domToSimpleHtml(domElement); // вырезаем теги ignore cleaned = text; Array.from(text.matchAll(/<ignore>(.*?)<\/ignore>/g)).map(function(value) { cleaned = cleaned.replaceAll(value, ''); }); return { text: cleaned, type: docType }; } }; var htmlToSimpleHtml = function(htmlText) { var c; c = new OfficeHtmlToSimpleHtmlConverter(); return c.convert(htmlText); }; var simpleHtmlToChapters = function(textHtml) { var allMatchedTitles, body, chapters, endIndex, i, j, len, match, title; // парсим на заголовки и блоки chapters = []; textHtml = textHtml.replaceAll("<h2>", "<h1>").replaceAll("</h2>", "</h1>"); allMatchedTitles = Array.from(textHtml.matchAll(/<h1>(.*?)<\/h1>/g)); for (i = j = 0, len = allMatchedTitles.length; j < len; i = ++j) { match = allMatchedTitles[i]; title = match[1].trim() || ""; if (allMatchedTitles[i + 1]) { endIndex = allMatchedTitles[i + 1].index; } else { endIndex = textHtml.length - 1; } body = textHtml.substring(match.index + match[0].length, endIndex).trim(); chapters.push({title, body}); } if (!chapters.length) { chapters = [ { title: "", body: textHtml } ]; } return chapters; }; var IMG_GDOCS_PALETTE = ""; var css$2 = ".nyamka-from-romano-userscript-chapters-importer-modal-paste-area {\n display: none;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-paste-area.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-paste-area textarea {\n width: 100%;\n min-height: 5em;\n resize: none;\n margin-bottom: 15px;\n box-sizing: border-box;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-paste-area p {\n text-align: center !important;\n color: #fcfaf2;\n margin: 0.5em 0;\n font-size: 120%;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-paste-area svg, .nyamka-from-romano-userscript-chapters-importer-modal-paste-area p {\n display: none;\n}\n@media (min-width: 1023px) {\n .nyamka-from-romano-userscript-chapters-importer-modal-paste-area {\n text-align: center;\n padding: 2em 2em;\n margin-bottom: 1em;\n background-color: #c3a63f;\n outline: 2px dashed #fcfaf2;\n outline-offset: -10px;\n -webkit-transition: outline-offset 0.15s ease-in-out, background-color 0.15s linear;\n transition: outline-offset 0.15s ease-in-out, background-color 0.15s linear;\n text-align: center;\n }\n .nyamka-from-romano-userscript-chapters-importer-modal-paste-area svg, .nyamka-from-romano-userscript-chapters-importer-modal-paste-area p {\n display: inline-block;\n }\n .nyamka-from-romano-userscript-chapters-importer-modal-paste-area textarea {\n display: none;\n }\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-svg-paste {\n max-width: 5em;\n text-align: center;\n margin: 0 auto;\n color: #fcfaf2;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-errors {\n display: none;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-errors.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-errors p {\n color: #ca0000;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-description {\n display: none;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-description.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-debug {\n display: none;\n text-align: center;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-debug.visible {\n display: block;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal-debug button {\n text-align: center;\n margin: 0 auto;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal .bg-grey, .nyamka-from-romano-userscript-chapters-importer-modal .bg-red {\n padding: 2px 4px;\n padding-top: 1px;\n border-radius: 2px;\n white-space: nowrap;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal .bg-grey {\n background: gray;\n color: black;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal .bg-red {\n background: red;\n color: black;\n}\n.nyamka-from-romano-userscript-chapters-importer-modal img {\n max-width: 350px;\n width: 100%;\n}\n\n@media (min-width: 1023px) {\n body[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-chapters-importer-modal-paste-area {\n background-color: #4e2b18;\n outline: 2px dashed #faf5ea;\n }\n}\n\n@media (min-width: 1023px) {\n body[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-chapters-importer-modal-paste-area {\n background-color: #54a6ff;\n outline: 2px dashed #f0f2f5;\n }\n}"; n(css$2,{}); var chaptersImporterModal = (function() { var debugData, modal, onClipboardPasteRmWatcher; modal = null; onClipboardPasteRmWatcher = null; debugData = null; return function(multipleChapters) { var exportSettings, i, len, property, ref, selectElement, title; if (modal) { modal.destroy(); } debugData = { status: 'OK', rawOfficeType: 'unknown', rawOfficeHtml: '', convertedHtml: '', error: null }; title = "Импортирование главы из Google Docs"; exportSettings = []; if (multipleChapters) { title = "Импортирование глав из Google Docs"; } if (exportSettings.length) { //exportSettings.unshift("<h3>Настроийки импорта</h3>") exportSettings.push("<hr>"); } modal = new Modal({ title: title, content: `<div class='${CSS_PREFIX}-chapters-importer-modal'> <div class="${CSS_PREFIX}-chapters-importer-modal-paste-area visible"> <div class='${CSS_PREFIX}-chapters-importer-modal-svg-paste'> ${SVG_ICON_PASTE} </div> <p><b>Вставьте текст из Google Docs из буфера обмена по Ctrl+V</b></p> <div><textarea class='form-control' placeholder="Вставьте текст из Google Docs из буфера обмена в это текстовое поле"></textarea></div> </div> ${exportSettings.join('')} <div class='${CSS_PREFIX}-chapters-importer-modal-description visible'> <p><i>Импортированы будут заголовки 1 и 2 уровня, жирный, курсив и перечеркнутый текст. Также будет импортированы выравнивания (по центру, по правому краю) и горизонтальные линии. Картинки, таблицы, списки нормально импортироваться не смогут.</i></p> <p>Заголовки глав должны быть оформлены стилями "Заголовок 1" или "Заголовок 2" (оба варианта будут считаться названиями глав).</p> <p>Можно импортировать сноски с тегами <b><note></note></b> и <b><footnote></footnote></b></p> <p> Если вы хотите, чтоб какой-то текст был проигнорирован при импорте, то выделите его красным цветом <span class='bg-red'>вот так</span> или выделите серым цветом <span class='bg-grey'>вот так</span> или заключите текст в тег <b><ignore></ignore></b> </p> <p>Кнопка выделения текста в Google docs:</p> <img src="${IMG_GDOCS_PALETTE}"> </div> <div class='${CSS_PREFIX}-chapters-importer-modal-errors'> <p>Возникли ошибки при импорте документа:</p> <ul> </ul> </div> <div class="${CSS_PREFIX}-chapters-importer-modal-debug"> <hr> <button class='btn btn-default'>Скачать debug-данные</button> </div> <div class='clear'></div> </div>` }); ref = modal.element.querySelectorAll('select'); for (i = 0, len = ref.length; i < len; i++) { selectElement = ref[i]; property = selectElement.getAttribute('data-property'); if (!property) { continue; } selectElement.value = storage[property]; selectElement.addEventListener('change', function() { property = this.getAttribute('data-property'); storage[property] = this.value; storage.saveData(); }); } return new Promise(function(resolve, reject) { var displayErrorAndSaveToDebug, processOfficeHtml; displayErrorAndSaveToDebug = function(error) { var ul; debugData.status = 'FAIL'; if (!(error instanceof Array || Array.isArray(error))) { debugData.error = { text: error.text || error.toString(), stack: error.stack }; ul = modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-errors > ul`); ul.style.padding = 0; ul.appendChild(createElement(`${error.text || error.toString()}`)); if (error.stack) { ul.appendChild(createElement(`<code>${error.stack}</code>`)); } } modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-errors`).classList.add('visible'); modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-debug`).classList.add('visible'); modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-paste-area`).classList.remove('visible'); modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-description`).classList.remove('visible'); console.warn("Ошибка в chaptersImporterModal: ", error); console.log(debugData); reject(error); }; processOfficeHtml = function(officeHtml) { var chapters, converted, error, titles; debugData.rawOfficeHtml = officeHtml; console.log(officeHtml); try { converted = htmlToSimpleHtml(officeHtml); debugData.convertedHtml = converted.text; debugData.rawOfficeType = converted.type; } catch (error1) { error = error1; displayErrorAndSaveToDebug(error); return; } try { chapters = simpleHtmlToChapters(converted.text); debugData.chapters = chapters; } catch (error1) { error = error1; displayErrorAndSaveToDebug(error); return; } if (!multipleChapters && chapters.length > 1) { // такой импорт производить нельзя titles = chapters.map(function(ch) { return `'${ch.title}'`; }); displayErrorAndSaveToDebug({ text: `<b>Обнаружено ${chapters.length} заголовкa(-ов) в тексте</b>: ${titles.join(', ')}.<br><b>Такое не разрешено.</b> Вставьте в модальное окно импорта текст без заголовка h1/h2 или с ОДНИМ заголовком h1/h2` }); return; } return chapters; }; //console.log processOfficeHtml(TEST_TEXT) //console.log debugData onClipboardPasteRmWatcher = onClipboardPaste(async function(clipboardData) { var chapters, data, itemHtml, itemText, items; if (!modal || !modal.isVisible()) { onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; return; } data = null; items = Array.prototype.slice.call(clipboardData.items); itemText = items.find(function(item) { return item.type === 'text/plain'; }); itemHtml = items.find(function(item) { return item.type === 'text/html'; }); if (itemHtml) { data = (await getItemAsString(itemHtml)); } else if (itemText) { data = (await getItemAsString(itemText)); } if (onClipboardPasteRmWatcher) { onClipboardPasteRmWatcher(); onClipboardPasteRmWatcher = null; } if (!data) { displayErrorAndSaveToDebug({ text: "Текст в буфере обмена не обнаружен" }); return; } chapters = processOfficeHtml(data); if (chapters) { modal.destroy(); resolve(chapters); } }); modal.element.querySelector(`.${CSS_PREFIX}-chapters-importer-modal-debug > button`).addEventListener('click', function() { downloadDataAsFile(debugData, 'debugData.json'); }); modal.show(); }); }; })(); var SVG_ICON_GOOGLE_DRIVE = "<svg aria-hidden=\"true\" focusable=\"false\" role=\"img\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 512 512\"><path fill=\"currentColor\" d=\"M339 314.9L175.4 32h161.2l163.6 282.9H339zm-137.5 23.6L120.9 480h310.5L512 338.5H201.5zM154.1 67.4L0 338.5 80.6 480 237 208.8 154.1 67.4z\" class=\"\"></path></svg>\n"; var css$1 = "body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-chapters-importer-button {\n border: 1px solid #CCCCCC;\n padding: 5px 6px;\n background: #F0F0EE;\n text-align: center;\n white-space: nowrap;\n cursor: pointer;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-chapters-importer-button:hover, body[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-chapters-importer-button:focus {\n background: #d8d8c4;\n}\nbody[data-nyamka-script=fanfics-me] .nyamka-from-romano-userscript-chapters-importer-button svg {\n width: 11px;\n}\n\nbody[data-nyamka-script=ficbook-net] .nyamka-from-romano-userscript-chapters-importer-button svg, body[data-nyamka-script=gikami-ru] .nyamka-from-romano-userscript-chapters-importer-button svg {\n width: 15px;\n max-width: 15px;\n margin-bottom: -2px;\n}"; n(css$1,{}); var importChapters, injectImportButton; importChapters = function(chapters, allowMultipleChapters) { var editor; if (isCurrentSite$2()) { chapters = chapters.map(function(chapter) { chapter.body = chapter.body.replaceAll("<note>", "<footnote>").replaceAll("</note>", "</footnote>"); return chapter; }); } if (!allowMultipleChapters) { editor = getChapterEditor(); if (!editor || !editor.title || !editor.body) { notification.error("Не получилось найти на сайте форму для редактирования главы"); return; } if (chapters[0].title) { editor.title.value = chapters[0].title; } editor.body.value = chapters[0].body; notification.success("Текст импортирован. Теперь вы можете сохранить главу"); } }; injectImportButton = function(allowMultipleChapters) { var btnClass, buttonImport, container; container = document.querySelector('.nyamka-from-romano-userscript-markup-tools') || document.querySelector('.operations > .add-part') || document.querySelector('.medium-editor-book-stat'); //gikami if (!container) { return setTimeout(function() { return injectImportButton(allowMultipleChapters); }, 500); } if (container.querySelector(`button.${CSS_PREFIX}-chapters-importer-button`)) { return; } if (allowMultipleChapters) { btnClass = 'btn-default btn-sm mb-5'; } else { btnClass = 'btn-for-formatting-text'; } buttonImport = createElement(`<button class='btn ${btnClass} ${CSS_PREFIX}-chapters-importer-button''> ${SVG_ICON_GOOGLE_DRIVE} Импортировать </button>`); container.appendChild(buttonImport); return buttonImport.addEventListener('click', async function(event) { var chapters, error; event.preventDefault(); try { chapters = (await chaptersImporterModal()); } catch (error1) { error = error1; console.log(error); return; } importChapters(chapters, allowMultipleChapters); }); }; //chaptersImporterModal() //console.log ficbookNet.getChaptersListForFanfic() var init$1 = function() { onLocationChange(function() { if (isCurrentSite$2()) { if (/\/home\/myfics\/\d+.\/parts\/\d+.?$/.test(location.pathname) || /\/home\/myfics\/\d+.\/addpart/.test(location.pathname)) { injectImportButton(); } return; } if (isCurrentSite$3()) { if (location.pathname === '/fic_write' && (location.search.indexOf('soaction=chapter_edit') > -1 || location.search.indexOf('soaction=chapter_new') > -1)) { injectImportButton(); } return; } if (isCurrentSite()) { if (/\/book\/\d+.\/add$/.test(location.pathname) || /\/book\/\d+.\/redact\/\d+.?$/.test(location.pathname)) { injectImportButton(); } return; } }); }; var css = "body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] {\n background: #666666;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar3, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .AuthorIntroduction .title, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar {\n background: #555555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar3 {\n border-bottom: none;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-pers, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-menu li a {\n color: #bbb;\n text-shadow: 1px 1px 1px #000000;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #back, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTextContainer h2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .tr .title {\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #take_bookmark {\n color: #3caa3c;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #back:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #take_bookmark:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .read_properties_button:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .read_properties_button.hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HeaderSlideMenu {\n background-color: #555;\n border-color: #333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HeaderSlideMenu .light {\n color: #bbbbbb;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HeaderSlideMenu li {\n border-color: #555555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HeaderSlideMenu li:hover {\n background-color: #777777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTextContainer, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTextContainerIn, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTable, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .summary_text_fic3, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .horizontal-bar, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #CommentForm .title {\n background: #333;\n border-color: #333;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .horizontal-bar, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTextContainer, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ReadTable {\n border-color: #777777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] a {\n color: #1e1e93;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] a:visited {\n color: #1e1e93;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] h1, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsCount {\n color: #bbbbbb;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #CommentForm textarea, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #CommentForm .comment_area {\n background: #777;\n border-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .comments .title, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .SerieContainerTitle, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .CommentItemTop {\n background: #777;\n border-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .dotted_button:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .chContents_1:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .chContents_3:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .chNav_large:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .chNav:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .blockquote {\n background-color: #333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-menu li:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-menu li.hover {\n background-color: #333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-pers:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .topbar-pers.hover {\n background-color: #333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_Management, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksList, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsCount {\n border-color: #555;\n background-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_Subscription, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksListHead {\n color: #bbb;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_Subscription1:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_Subscription2:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuSecondLine {\n border-color: #777;\n background-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuSecondLine .activ2 a {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuSecondLine .activ2 a:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuSecondLine a:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksListHead, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .chNav_large {\n border-color: #777;\n background-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksList1 .marks_list li:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksList1 .marks_list_add li:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .totop:hover {\n background-color: #777;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_List, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MarksList1 .marks_list {\n filter: brightness(70%);\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .t-l, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .t-rl, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .t-lb, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .t-rlb, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsNext {\n border-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FM_note_container {\n background-color: #555;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content {\n background: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-right {\n background: #4d4d4d;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-right .main_menu li {\n border-color: #404040;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-right .main_menu li a {\n color: #8f8fff;\n text-shadow: 0 0 1px black;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-right .main_menu li a:visited {\n color: #8f8fff;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-right .main_menu li.checked {\n border-color: #262626;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .main_menu2.insert_main_menu a {\n color: #8f8fff;\n text-shadow: 0 0 1px black;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .main_menu2.insert_main_menu a:visited {\n color: #8f8fff;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-center, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuFirstLine {\n background: #666666;\n color: #eeeeee;\n border-color: #4d4d4d;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-center li > a:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuFirstLine li > a:hover {\n background: #595959;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #site-content-center li.activ2 > a, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .HorMenuFirstLine li.activ2 > a {\n background: #4d4d4d;\n border-color: #404040;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft h2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileRight h2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FanartRight h2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Sidebar_NativeAd, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageAloneHead {\n background: #4d4d4d;\n border-color: #333333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft h2 a, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileRight h2 a, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FanartRight h2 a, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Sidebar_NativeAd a, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageAloneHead a {\n color: #8f8fff;\n text-shadow: 0 0 1px black;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft h2 a:visited, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileRight h2 a:visited, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FanartRight h2 a:visited, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Sidebar_NativeAd a:visited, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageAloneHead a:visited {\n color: #8f8fff;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeftFic, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft_PersonalCard {\n border-color: #333333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeftFic:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft_PersonalCard:hover {\n background: transparent;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileLeft_h2_descr, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FanartRight_h2_descr, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Sidebar_NativeAd_title {\n background: gray;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Profile_Head {\n position: relative;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Profile_Head h1 > a {\n color: white;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Profile_Head::before {\n content: \"\";\n position: absolute;\n top: 0;\n left: 0;\n right: 0;\n bottom: 0;\n background: #00000082;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsShow, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .dropdown-click .body, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageManagement, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .profile_info_more,\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FanartTable2 .TdImg {\n background: #737373;\n color: #eeeeee !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .dropdown-click .header {\n background: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsHide {\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentsHide:hover {\n background: #636363;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageNewManagementPrivate img {\n filter: contrast(4.5);\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageNewManagementPrivate .small > div {\n background: #737373 !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageNewManagementPrivate li:hover {\n background: #636363;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .gen {\n color: #6e006e;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .red {\n color: #cb0000;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Comments, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .green, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .date {\n color: #4dd34d !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .blue {\n color: #00006e;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .light, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .small_light_link, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .small_light, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DateUpdate, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .FicDopInfo_Link,\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .ProfileInfo_main tr td {\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .header-search, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] div.MessageNew, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] textarea.MessageNew, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] input.MessageCommentNew, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] textarea.MessageCommentNew,\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .input_3, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] textarea.DialogNewMsgTextarea, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] textarea.CommentEditText, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .fdmain-search {\n background-color: #e6e6e6;\n color: #1a1a1a;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .header-search-answ {\n background: #737373;\n border-color: #595959;\n box-shadow: 0 0 6px #202020;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .header-search-answ li:hover {\n background: #8c8c8c;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] img.avatar, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] img.open-picture, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MyCommentsItemText img, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogItem_Avatar img, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogNewMsgHeader img,\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Message .MessageLeft a > img, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageCommentLeft a > img {\n filter: brightness(0.7);\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteFormContainer {\n border: 1px solid #404040;\n background: #595959;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteFormContainer h2 {\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteForm_Answer {\n background: #EFEFDB;\n color: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteForm_Answer .light {\n color: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteFormNoVoted .PollVoteForm_Answer:hover {\n background: #D8D8C4;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollVoteForm_AnswerPersentBg {\n background: #D8D8C4;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PollMultiple .PollVoteForm_Answer.checked {\n background: #D8D8C4;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MessageAloneTextFormat_Container .checked {\n color: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .spoiler {\n background: #595959;\n color: #595959;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .spoiler.selected {\n background: #595959;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .spoiler::before {\n background: #404040;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .fic_info_title.nav {\n background: #757575;\n color: #eeeeee !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .fic_info_content {\n background: #6e6e6e;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .RecommentCloud {\n background: #545454;\n border-color: #3b3b3b;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .RecommentCloud::before {\n border-color: transparent transparent #3b3b3b #3b3b3b;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .RecommentCloud::after {\n border-color: transparent transparent #545454 #545454;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #FicWriteMenu td.activ #publ_count, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #FicWriteMenu td.activ #moderation_count, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #FicWriteMenu td.activ #moderation_count, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #FicWriteMenu td.activ #deleted_count, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #FicWriteMenu td.activ #series_count {\n color: #eeeeee !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .TextEditorContainer .TextFormat, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .TextEditorContainer .TextFormat2 {\n color: #2e2e2e;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .TextFormat3 {\n color: #2e2e2e;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogItem:hover, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogItem.notread, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogMsg.NotRead, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogMsg.DialogMsgSelected {\n background: #545454;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogNewMsgForm {\n background: #666666;\n border-color: #545454;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogItem_LastMessage.notread, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogItem_LastMessage.notread .small.light, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogNewMsgAtchButton, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogNewMsgAtchButton .small.light {\n color: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .DialogNewMsgButton img {\n filter: contrast(4.5);\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] #data-container table.FullWidth {\n background: #666666 !important;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .box-modal {\n background: #666666;\n color: #eeeeee;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .box-modal .box-modal_close.arcticmodal-close {\n color: #262626;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Footer {\n background: #333333;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Footer a {\n color: #8f8fff;\n text-shadow: 0 0 1px black;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .Footer a:visited {\n color: #8f8fff;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .PropertiesMenu .sp .activ {\n background: #595959;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MyCommentsItemShow {\n color: #eeeeee !important;\n background: #595959;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .MyCommentsItemShow.MyCommentsItemShowEmpty {\n background: transparent;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .InfoContainer, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .InfoContainer h2, body.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me] .InfoContainer .light {\n color: #666666;\n}\nbody.nyamka-from-romano-userscript-darkmode[data-nyamka-script=fanfics-me].nyamka-from-romano-userscript-app-view-notifications .ContentTable.Messages:not(.visible)::after {\n background: #666666 !important;\n}\n\n.nyamka-from-romano-userscript-darkmode-settings label {\n display: inline-block;\n padding: 10px;\n cursor: pointer;\n}"; n(css,{}); var getMenuContainer, getSettings; getMenuContainer = async function() { var j, len, menu, ref; ref = [200, 500, 700, 1000, 1500]; for (j = 0, len = ref.length; j < len; j++) { menu = document.querySelector('.main_menu2.insert_main_menu'); if (!menu) { await timeout(); } else { break; } } return menu; }; getSettings = function() { var checked, settings; checked = ''; if (storage.isDarkModeEnabled) { checked = 'checked'; document.body.classList.add(`${CSS_PREFIX}-darkmode`); } settings = createElement(`<li data-nyamka-script='true' class='${CSS_PREFIX}-darkmode-settings'> <label> <input type='checkbox' ${checked}> Ночной режим включен </label> </li>`); settings.querySelector('input').addEventListener('click', function() { storage.isDarkModeEnabled = !!this.checked; storage.saveData(); if (!storage.isDarkModeEnabled) { document.body.classList.remove(`${CSS_PREFIX}-darkmode`); } else { document.body.classList.add(`${CSS_PREFIX}-darkmode`); } }); return settings; }; var init = function() { var settings; if (!isCurrentSite$3()) { return; } storage.registerVariable('isDarkModeEnabled', false); settings = getSettings(); onLocationChange(async function() { var menu; menu = (await getMenuContainer()); if (!menu || menu.parentNode.querySelector("li[data-nyamka-script]")) { return; } menu.parentNode.insertBefore(settings, menu.nextSibling); }); }; /* // ==UserScript== // @name Нямка от Рóмана - dev // @namespace http://tampermonkey.net/ // @version 0.38 // @description пакет улучшений фикрайтерских сайтов // @license GPL-3.0-or-later // @author Roman O // @match https://fanfics.me/* // @match https://ficbook.net/* // @grant unsafeWindow // @run-at document-body // ==/UserScript== */ document.body.setAttribute('data-nyamka-script', location.host.replaceAll('.', '-')); init(); init$4(); init$2(); init$3(); //siteNotifications.init() init$1();