diff --git a/h5p/h5plib/v126/classes/local/library/handler.php b/h5p/h5plib/v126/classes/local/library/handler.php deleted file mode 100644 index 1e9c5a0877de7..0000000000000 --- a/h5p/h5plib/v126/classes/local/library/handler.php +++ /dev/null @@ -1,36 +0,0 @@ -. - -namespace h5plib_v126\local\library; - -/** - * Handler for this version of the H5P library. - * - * @package h5plib_v126 - * @copyright 2024 Sara Arjona - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ -class handler extends \core_h5p\local\library\handler { - - /** - * Get the current version of the H5P core library. - * - * @return string - */ - public static function get_h5p_version(): string { - return '126'; - } -} diff --git a/h5p/h5plib/v126/classes/privacy/provider.php b/h5p/h5plib/v126/classes/privacy/provider.php deleted file mode 100644 index 976e1c59f4ea9..0000000000000 --- a/h5p/h5plib/v126/classes/privacy/provider.php +++ /dev/null @@ -1,36 +0,0 @@ -. - -namespace h5plib_v126\privacy; - -/** - * Privacy provider implementation for this version of the H5P library. - * - * @package h5plib_v126 - * @copyright 2024 Sara Arjona - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ -class provider implements \core_privacy\local\metadata\null_provider { - /** - * Get the language string identifier with the component's language - * file to explain why this plugin stores no data. - * - * @return string - */ - public static function get_reason(): string { - return 'privacy:metadata'; - } -} diff --git a/h5p/h5plib/v126/joubel/core/LICENSE.txt b/h5p/h5plib/v126/joubel/core/LICENSE.txt deleted file mode 100644 index 20d40b6bceca3..0000000000000 --- a/h5p/h5plib/v126/joubel/core/LICENSE.txt +++ /dev/null @@ -1,674 +0,0 @@ - GNU GENERAL PUBLIC LICENSE - Version 3, 29 June 2007 - - Copyright (C) 2007 Free Software Foundation, Inc. - Everyone is permitted to copy and distribute verbatim copies - of this license document, but changing it is not allowed. - - Preamble - - The GNU General Public License is a free, copyleft license for -software and other kinds of works. - - The licenses for most software and other practical works are designed -to take away your freedom to share and change the works. By contrast, -the GNU General Public License is intended to guarantee your freedom to -share and change all versions of a program--to make sure it remains free -software for all its users. We, the Free Software Foundation, use the -GNU General Public License for most of our software; it applies also to -any other work released this way by its authors. You can apply it to -your programs, too. - - When we speak of free software, we are referring to freedom, not -price. Our General Public Licenses are designed to make sure that you -have the freedom to distribute copies of free software (and charge for -them if you wish), that you receive source code or can get it if you -want it, that you can change the software or use pieces of it in new -free programs, and that you know you can do these things. - - To protect your rights, we need to prevent others from denying you -these rights or asking you to surrender the rights. Therefore, you have -certain responsibilities if you distribute copies of the software, or if -you modify it: responsibilities to respect the freedom of others. - - For example, if you distribute copies of such a program, whether -gratis or for a fee, you must pass on to the recipients the same -freedoms that you received. You must make sure that they, too, receive -or can get the source code. And you must show them these terms so they -know their rights. - - Developers that use the GNU GPL protect your rights with two steps: -(1) assert copyright on the software, and (2) offer you this License -giving you legal permission to copy, distribute and/or modify it. - - For the developers' and authors' protection, the GPL clearly explains -that there is no warranty for this free software. For both users' and -authors' sake, the GPL requires that modified versions be marked as -changed, so that their problems will not be attributed erroneously to -authors of previous versions. - - Some devices are designed to deny users access to install or run -modified versions of the software inside them, although the manufacturer -can do so. This is fundamentally incompatible with the aim of -protecting users' freedom to change the software. The systematic -pattern of such abuse occurs in the area of products for individuals to -use, which is precisely where it is most unacceptable. Therefore, we -have designed this version of the GPL to prohibit the practice for those -products. If such problems arise substantially in other domains, we -stand ready to extend this provision to those domains in future versions -of the GPL, as needed to protect the freedom of users. - - Finally, every program is threatened constantly by software patents. -States should not allow patents to restrict development and use of -software on general-purpose computers, but in those that do, we wish to -avoid the special danger that patents applied to a free program could -make it effectively proprietary. To prevent this, the GPL assures that -patents cannot be used to render the program non-free. - - The precise terms and conditions for copying, distribution and -modification follow. - - TERMS AND CONDITIONS - - 0. Definitions. - - "This License" refers to version 3 of the GNU General Public License. - - "Copyright" also means copyright-like laws that apply to other kinds of -works, such as semiconductor masks. - - "The Program" refers to any copyrightable work licensed under this -License. Each licensee is addressed as "you". "Licensees" and -"recipients" may be individuals or organizations. - - To "modify" a work means to copy from or adapt all or part of the work -in a fashion requiring copyright permission, other than the making of an -exact copy. The resulting work is called a "modified version" of the -earlier work or a work "based on" the earlier work. - - A "covered work" means either the unmodified Program or a work based -on the Program. - - To "propagate" a work means to do anything with it that, without -permission, would make you directly or secondarily liable for -infringement under applicable copyright law, except executing it on a -computer or modifying a private copy. Propagation includes copying, -distribution (with or without modification), making available to the -public, and in some countries other activities as well. - - To "convey" a work means any kind of propagation that enables other -parties to make or receive copies. Mere interaction with a user through -a computer network, with no transfer of a copy, is not conveying. - - An interactive user interface displays "Appropriate Legal Notices" -to the extent that it includes a convenient and prominently visible -feature that (1) displays an appropriate copyright notice, and (2) -tells the user that there is no warranty for the work (except to the -extent that warranties are provided), that licensees may convey the -work under this License, and how to view a copy of this License. If -the interface presents a list of user commands or options, such as a -menu, a prominent item in the list meets this criterion. - - 1. Source Code. - - The "source code" for a work means the preferred form of the work -for making modifications to it. "Object code" means any non-source -form of a work. - - A "Standard Interface" means an interface that either is an official -standard defined by a recognized standards body, or, in the case of -interfaces specified for a particular programming language, one that -is widely used among developers working in that language. - - The "System Libraries" of an executable work include anything, other -than the work as a whole, that (a) is included in the normal form of -packaging a Major Component, but which is not part of that Major -Component, and (b) serves only to enable use of the work with that -Major Component, or to implement a Standard Interface for which an -implementation is available to the public in source code form. A -"Major Component", in this context, means a major essential component -(kernel, window system, and so on) of the specific operating system -(if any) on which the executable work runs, or a compiler used to -produce the work, or an object code interpreter used to run it. - - The "Corresponding Source" for a work in object code form means all -the source code needed to generate, install, and (for an executable -work) run the object code and to modify the work, including scripts to -control those activities. However, it does not include the work's -System Libraries, or general-purpose tools or generally available free -programs which are used unmodified in performing those activities but -which are not part of the work. For example, Corresponding Source -includes interface definition files associated with source files for -the work, and the source code for shared libraries and dynamically -linked subprograms that the work is specifically designed to require, -such as by intimate data communication or control flow between those -subprograms and other parts of the work. - - The Corresponding Source need not include anything that users -can regenerate automatically from other parts of the Corresponding -Source. - - The Corresponding Source for a work in source code form is that -same work. - - 2. Basic Permissions. - - All rights granted under this License are granted for the term of -copyright on the Program, and are irrevocable provided the stated -conditions are met. This License explicitly affirms your unlimited -permission to run the unmodified Program. The output from running a -covered work is covered by this License only if the output, given its -content, constitutes a covered work. This License acknowledges your -rights of fair use or other equivalent, as provided by copyright law. - - You may make, run and propagate covered works that you do not -convey, without conditions so long as your license otherwise remains -in force. You may convey covered works to others for the sole purpose -of having them make modifications exclusively for you, or provide you -with facilities for running those works, provided that you comply with -the terms of this License in conveying all material for which you do -not control copyright. Those thus making or running the covered works -for you must do so exclusively on your behalf, under your direction -and control, on terms that prohibit them from making any copies of -your copyrighted material outside their relationship with you. - - Conveying under any other circumstances is permitted solely under -the conditions stated below. Sublicensing is not allowed; section 10 -makes it unnecessary. - - 3. Protecting Users' Legal Rights From Anti-Circumvention Law. - - No covered work shall be deemed part of an effective technological -measure under any applicable law fulfilling obligations under article -11 of the WIPO copyright treaty adopted on 20 December 1996, or -similar laws prohibiting or restricting circumvention of such -measures. - - When you convey a covered work, you waive any legal power to forbid -circumvention of technological measures to the extent such circumvention -is effected by exercising rights under this License with respect to -the covered work, and you disclaim any intention to limit operation or -modification of the work as a means of enforcing, against the work's -users, your or third parties' legal rights to forbid circumvention of -technological measures. - - 4. Conveying Verbatim Copies. - - You may convey verbatim copies of the Program's source code as you -receive it, in any medium, provided that you conspicuously and -appropriately publish on each copy an appropriate copyright notice; -keep intact all notices stating that this License and any -non-permissive terms added in accord with section 7 apply to the code; -keep intact all notices of the absence of any warranty; and give all -recipients a copy of this License along with the Program. - - You may charge any price or no price for each copy that you convey, -and you may offer support or warranty protection for a fee. - - 5. Conveying Modified Source Versions. - - You may convey a work based on the Program, or the modifications to -produce it from the Program, in the form of source code under the -terms of section 4, provided that you also meet all of these conditions: - - a) The work must carry prominent notices stating that you modified - it, and giving a relevant date. - - b) The work must carry prominent notices stating that it is - released under this License and any conditions added under section - 7. This requirement modifies the requirement in section 4 to - "keep intact all notices". - - c) You must license the entire work, as a whole, under this - License to anyone who comes into possession of a copy. This - License will therefore apply, along with any applicable section 7 - additional terms, to the whole of the work, and all its parts, - regardless of how they are packaged. This License gives no - permission to license the work in any other way, but it does not - invalidate such permission if you have separately received it. - - d) If the work has interactive user interfaces, each must display - Appropriate Legal Notices; however, if the Program has interactive - interfaces that do not display Appropriate Legal Notices, your - work need not make them do so. - - A compilation of a covered work with other separate and independent -works, which are not by their nature extensions of the covered work, -and which are not combined with it such as to form a larger program, -in or on a volume of a storage or distribution medium, is called an -"aggregate" if the compilation and its resulting copyright are not -used to limit the access or legal rights of the compilation's users -beyond what the individual works permit. Inclusion of a covered work -in an aggregate does not cause this License to apply to the other -parts of the aggregate. - - 6. Conveying Non-Source Forms. - - You may convey a covered work in object code form under the terms -of sections 4 and 5, provided that you also convey the -machine-readable Corresponding Source under the terms of this License, -in one of these ways: - - a) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by the - Corresponding Source fixed on a durable physical medium - customarily used for software interchange. - - b) Convey the object code in, or embodied in, a physical product - (including a physical distribution medium), accompanied by a - written offer, valid for at least three years and valid for as - long as you offer spare parts or customer support for that product - model, to give anyone who possesses the object code either (1) a - copy of the Corresponding Source for all the software in the - product that is covered by this License, on a durable physical - medium customarily used for software interchange, for a price no - more than your reasonable cost of physically performing this - conveying of source, or (2) access to copy the - Corresponding Source from a network server at no charge. - - c) Convey individual copies of the object code with a copy of the - written offer to provide the Corresponding Source. This - alternative is allowed only occasionally and noncommercially, and - only if you received the object code with such an offer, in accord - with subsection 6b. - - d) Convey the object code by offering access from a designated - place (gratis or for a charge), and offer equivalent access to the - Corresponding Source in the same way through the same place at no - further charge. You need not require recipients to copy the - Corresponding Source along with the object code. If the place to - copy the object code is a network server, the Corresponding Source - may be on a different server (operated by you or a third party) - that supports equivalent copying facilities, provided you maintain - clear directions next to the object code saying where to find the - Corresponding Source. Regardless of what server hosts the - Corresponding Source, you remain obligated to ensure that it is - available for as long as needed to satisfy these requirements. - - e) Convey the object code using peer-to-peer transmission, provided - you inform other peers where the object code and Corresponding - Source of the work are being offered to the general public at no - charge under subsection 6d. - - A separable portion of the object code, whose source code is excluded -from the Corresponding Source as a System Library, need not be -included in conveying the object code work. - - A "User Product" is either (1) a "consumer product", which means any -tangible personal property which is normally used for personal, family, -or household purposes, or (2) anything designed or sold for incorporation -into a dwelling. In determining whether a product is a consumer product, -doubtful cases shall be resolved in favor of coverage. For a particular -product received by a particular user, "normally used" refers to a -typical or common use of that class of product, regardless of the status -of the particular user or of the way in which the particular user -actually uses, or expects or is expected to use, the product. A product -is a consumer product regardless of whether the product has substantial -commercial, industrial or non-consumer uses, unless such uses represent -the only significant mode of use of the product. - - "Installation Information" for a User Product means any methods, -procedures, authorization keys, or other information required to install -and execute modified versions of a covered work in that User Product from -a modified version of its Corresponding Source. The information must -suffice to ensure that the continued functioning of the modified object -code is in no case prevented or interfered with solely because -modification has been made. - - If you convey an object code work under this section in, or with, or -specifically for use in, a User Product, and the conveying occurs as -part of a transaction in which the right of possession and use of the -User Product is transferred to the recipient in perpetuity or for a -fixed term (regardless of how the transaction is characterized), the -Corresponding Source conveyed under this section must be accompanied -by the Installation Information. But this requirement does not apply -if neither you nor any third party retains the ability to install -modified object code on the User Product (for example, the work has -been installed in ROM). - - The requirement to provide Installation Information does not include a -requirement to continue to provide support service, warranty, or updates -for a work that has been modified or installed by the recipient, or for -the User Product in which it has been modified or installed. Access to a -network may be denied when the modification itself materially and -adversely affects the operation of the network or violates the rules and -protocols for communication across the network. - - Corresponding Source conveyed, and Installation Information provided, -in accord with this section must be in a format that is publicly -documented (and with an implementation available to the public in -source code form), and must require no special password or key for -unpacking, reading or copying. - - 7. Additional Terms. - - "Additional permissions" are terms that supplement the terms of this -License by making exceptions from one or more of its conditions. -Additional permissions that are applicable to the entire Program shall -be treated as though they were included in this License, to the extent -that they are valid under applicable law. If additional permissions -apply only to part of the Program, that part may be used separately -under those permissions, but the entire Program remains governed by -this License without regard to the additional permissions. - - When you convey a copy of a covered work, you may at your option -remove any additional permissions from that copy, or from any part of -it. (Additional permissions may be written to require their own -removal in certain cases when you modify the work.) You may place -additional permissions on material, added by you to a covered work, -for which you have or can give appropriate copyright permission. - - Notwithstanding any other provision of this License, for material you -add to a covered work, you may (if authorized by the copyright holders of -that material) supplement the terms of this License with terms: - - a) Disclaiming warranty or limiting liability differently from the - terms of sections 15 and 16 of this License; or - - b) Requiring preservation of specified reasonable legal notices or - author attributions in that material or in the Appropriate Legal - Notices displayed by works containing it; or - - c) Prohibiting misrepresentation of the origin of that material, or - requiring that modified versions of such material be marked in - reasonable ways as different from the original version; or - - d) Limiting the use for publicity purposes of names of licensors or - authors of the material; or - - e) Declining to grant rights under trademark law for use of some - trade names, trademarks, or service marks; or - - f) Requiring indemnification of licensors and authors of that - material by anyone who conveys the material (or modified versions of - it) with contractual assumptions of liability to the recipient, for - any liability that these contractual assumptions directly impose on - those licensors and authors. - - All other non-permissive additional terms are considered "further -restrictions" within the meaning of section 10. If the Program as you -received it, or any part of it, contains a notice stating that it is -governed by this License along with a term that is a further -restriction, you may remove that term. If a license document contains -a further restriction but permits relicensing or conveying under this -License, you may add to a covered work material governed by the terms -of that license document, provided that the further restriction does -not survive such relicensing or conveying. - - If you add terms to a covered work in accord with this section, you -must place, in the relevant source files, a statement of the -additional terms that apply to those files, or a notice indicating -where to find the applicable terms. - - Additional terms, permissive or non-permissive, may be stated in the -form of a separately written license, or stated as exceptions; -the above requirements apply either way. - - 8. Termination. - - You may not propagate or modify a covered work except as expressly -provided under this License. Any attempt otherwise to propagate or -modify it is void, and will automatically terminate your rights under -this License (including any patent licenses granted under the third -paragraph of section 11). - - However, if you cease all violation of this License, then your -license from a particular copyright holder is reinstated (a) -provisionally, unless and until the copyright holder explicitly and -finally terminates your license, and (b) permanently, if the copyright -holder fails to notify you of the violation by some reasonable means -prior to 60 days after the cessation. - - Moreover, your license from a particular copyright holder is -reinstated permanently if the copyright holder notifies you of the -violation by some reasonable means, this is the first time you have -received notice of violation of this License (for any work) from that -copyright holder, and you cure the violation prior to 30 days after -your receipt of the notice. - - Termination of your rights under this section does not terminate the -licenses of parties who have received copies or rights from you under -this License. If your rights have been terminated and not permanently -reinstated, you do not qualify to receive new licenses for the same -material under section 10. - - 9. Acceptance Not Required for Having Copies. - - You are not required to accept this License in order to receive or -run a copy of the Program. Ancillary propagation of a covered work -occurring solely as a consequence of using peer-to-peer transmission -to receive a copy likewise does not require acceptance. However, -nothing other than this License grants you permission to propagate or -modify any covered work. These actions infringe copyright if you do -not accept this License. Therefore, by modifying or propagating a -covered work, you indicate your acceptance of this License to do so. - - 10. Automatic Licensing of Downstream Recipients. - - Each time you convey a covered work, the recipient automatically -receives a license from the original licensors, to run, modify and -propagate that work, subject to this License. You are not responsible -for enforcing compliance by third parties with this License. - - An "entity transaction" is a transaction transferring control of an -organization, or substantially all assets of one, or subdividing an -organization, or merging organizations. If propagation of a covered -work results from an entity transaction, each party to that -transaction who receives a copy of the work also receives whatever -licenses to the work the party's predecessor in interest had or could -give under the previous paragraph, plus a right to possession of the -Corresponding Source of the work from the predecessor in interest, if -the predecessor has it or can get it with reasonable efforts. - - You may not impose any further restrictions on the exercise of the -rights granted or affirmed under this License. For example, you may -not impose a license fee, royalty, or other charge for exercise of -rights granted under this License, and you may not initiate litigation -(including a cross-claim or counterclaim in a lawsuit) alleging that -any patent claim is infringed by making, using, selling, offering for -sale, or importing the Program or any portion of it. - - 11. Patents. - - A "contributor" is a copyright holder who authorizes use under this -License of the Program or a work on which the Program is based. The -work thus licensed is called the contributor's "contributor version". - - A contributor's "essential patent claims" are all patent claims -owned or controlled by the contributor, whether already acquired or -hereafter acquired, that would be infringed by some manner, permitted -by this License, of making, using, or selling its contributor version, -but do not include claims that would be infringed only as a -consequence of further modification of the contributor version. For -purposes of this definition, "control" includes the right to grant -patent sublicenses in a manner consistent with the requirements of -this License. - - Each contributor grants you a non-exclusive, worldwide, royalty-free -patent license under the contributor's essential patent claims, to -make, use, sell, offer for sale, import and otherwise run, modify and -propagate the contents of its contributor version. - - In the following three paragraphs, a "patent license" is any express -agreement or commitment, however denominated, not to enforce a patent -(such as an express permission to practice a patent or covenant not to -sue for patent infringement). To "grant" such a patent license to a -party means to make such an agreement or commitment not to enforce a -patent against the party. - - If you convey a covered work, knowingly relying on a patent license, -and the Corresponding Source of the work is not available for anyone -to copy, free of charge and under the terms of this License, through a -publicly available network server or other readily accessible means, -then you must either (1) cause the Corresponding Source to be so -available, or (2) arrange to deprive yourself of the benefit of the -patent license for this particular work, or (3) arrange, in a manner -consistent with the requirements of this License, to extend the patent -license to downstream recipients. "Knowingly relying" means you have -actual knowledge that, but for the patent license, your conveying the -covered work in a country, or your recipient's use of the covered work -in a country, would infringe one or more identifiable patents in that -country that you have reason to believe are valid. - - If, pursuant to or in connection with a single transaction or -arrangement, you convey, or propagate by procuring conveyance of, a -covered work, and grant a patent license to some of the parties -receiving the covered work authorizing them to use, propagate, modify -or convey a specific copy of the covered work, then the patent license -you grant is automatically extended to all recipients of the covered -work and works based on it. - - A patent license is "discriminatory" if it does not include within -the scope of its coverage, prohibits the exercise of, or is -conditioned on the non-exercise of one or more of the rights that are -specifically granted under this License. You may not convey a covered -work if you are a party to an arrangement with a third party that is -in the business of distributing software, under which you make payment -to the third party based on the extent of your activity of conveying -the work, and under which the third party grants, to any of the -parties who would receive the covered work from you, a discriminatory -patent license (a) in connection with copies of the covered work -conveyed by you (or copies made from those copies), or (b) primarily -for and in connection with specific products or compilations that -contain the covered work, unless you entered into that arrangement, -or that patent license was granted, prior to 28 March 2007. - - Nothing in this License shall be construed as excluding or limiting -any implied license or other defenses to infringement that may -otherwise be available to you under applicable patent law. - - 12. No Surrender of Others' Freedom. - - If conditions are imposed on you (whether by court order, agreement or -otherwise) that contradict the conditions of this License, they do not -excuse you from the conditions of this License. If you cannot convey a -covered work so as to satisfy simultaneously your obligations under this -License and any other pertinent obligations, then as a consequence you may -not convey it at all. For example, if you agree to terms that obligate you -to collect a royalty for further conveying from those to whom you convey -the Program, the only way you could satisfy both those terms and this -License would be to refrain entirely from conveying the Program. - - 13. Use with the GNU Affero General Public License. - - Notwithstanding any other provision of this License, you have -permission to link or combine any covered work with a work licensed -under version 3 of the GNU Affero General Public License into a single -combined work, and to convey the resulting work. The terms of this -License will continue to apply to the part which is the covered work, -but the special requirements of the GNU Affero General Public License, -section 13, concerning interaction through a network will apply to the -combination as such. - - 14. Revised Versions of this License. - - The Free Software Foundation may publish revised and/or new versions of -the GNU General Public License from time to time. Such new versions will -be similar in spirit to the present version, but may differ in detail to -address new problems or concerns. - - Each version is given a distinguishing version number. If the -Program specifies that a certain numbered version of the GNU General -Public License "or any later version" applies to it, you have the -option of following the terms and conditions either of that numbered -version or of any later version published by the Free Software -Foundation. If the Program does not specify a version number of the -GNU General Public License, you may choose any version ever published -by the Free Software Foundation. - - If the Program specifies that a proxy can decide which future -versions of the GNU General Public License can be used, that proxy's -public statement of acceptance of a version permanently authorizes you -to choose that version for the Program. - - Later license versions may give you additional or different -permissions. However, no additional obligations are imposed on any -author or copyright holder as a result of your choosing to follow a -later version. - - 15. Disclaimer of Warranty. - - THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY -APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT -HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY -OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, -THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR -PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM -IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF -ALL NECESSARY SERVICING, REPAIR OR CORRECTION. - - 16. Limitation of Liability. - - IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING -WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS -THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY -GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE -USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF -DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD -PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), -EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF -SUCH DAMAGES. - - 17. Interpretation of Sections 15 and 16. - - If the disclaimer of warranty and limitation of liability provided -above cannot be given local legal effect according to their terms, -reviewing courts shall apply local law that most closely approximates -an absolute waiver of all civil liability in connection with the -Program, unless a warranty or assumption of liability accompanies a -copy of the Program in return for a fee. - - END OF TERMS AND CONDITIONS - - How to Apply These Terms to Your New Programs - - If you develop a new program, and you want it to be of the greatest -possible use to the public, the best way to achieve this is to make it -free software which everyone can redistribute and change under these terms. - - To do so, attach the following notices to the program. It is safest -to attach them to the start of each source file to most effectively -state the exclusion of warranty; and each file should have at least -the "copyright" line and a pointer to where the full notice is found. - - - Copyright (C) - - This program is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - This program is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see . - -Also add information on how to contact you by electronic and paper mail. - - If the program does terminal interaction, make it output a short -notice like this when it starts in an interactive mode: - - Copyright (C) - This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'. - This is free software, and you are welcome to redistribute it - under certain conditions; type `show c' for details. - -The hypothetical commands `show w' and `show c' should show the appropriate -parts of the General Public License. Of course, your program's commands -might be different; for a GUI interface, you would use an "about box". - - You should also get your employer (if you work as a programmer) or school, -if any, to sign a "copyright disclaimer" for the program, if necessary. -For more information on this, and how to apply and follow the GNU GPL, see -. - - The GNU General Public License does not permit incorporating your program -into proprietary programs. If your program is a subroutine library, you -may consider it more useful to permit linking proprietary applications with -the library. If this is what you want to do, use the GNU Lesser General -Public License instead of this License. But first, please read -. \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/core/README.txt b/h5p/h5plib/v126/joubel/core/README.txt deleted file mode 100644 index 22cf79866c2e4..0000000000000 --- a/h5p/h5plib/v126/joubel/core/README.txt +++ /dev/null @@ -1,18 +0,0 @@ -This folder contains the general H5P library. The files within this folder are not specific to any framework. - -Any interaction with an LMS, CMS or other frameworks is done through interfaces. Platforms need to implement -the H5PFrameworkInterface(in h5p.classes.php) and also do the following: - - - Provide a form for uploading H5P packages. - - Place the uploaded H5P packages in a temporary directory - +++ - -See existing implementations for details. For instance the Drupal H5P module located at drupal.org/project/h5p - -We will make available documentation and tutorials for creating platform integrations in the future. - -The H5P PHP library is GPL licensed due to GPL code being used for purifying HTML provided by authors. - -## License - -Open Sans font is licensed under Apache license, Version 2.0 \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/core/doc/spec_en.html b/h5p/h5plib/v126/joubel/core/doc/spec_en.html deleted file mode 100644 index de6ddf077ae0f..0000000000000 --- a/h5p/h5plib/v126/joubel/core/doc/spec_en.html +++ /dev/null @@ -1,168 +0,0 @@ -

Overview

-

H5P is a file format for content/applications made using modern, open web technologies (HTML5). The format enables easy installation and transfer of applications/content on different CMSes, LMSes and other platforms. An H5P can be uploaded and published on a platform in mostly the same way one would publish a Flash file today. H5P files may also be updated by simply uploading a new version of the file, the same way as one would using Flash.

-

H5P opens for extensive reuse of code and wide flexibility regarding what may be developed as an H5P.

-

The system uses package files containing all necessary files and libraries for the application to function. These files are based on open formats.

-

Overview of package files

-

Package files are normal zip files, with a naming convention of <filename>.h5p to distinguish from any random zip file. This zip file then requires a specific file structure as described below.

-

There will be a file in JSON format named h5p.json describing the contents of the package and how the system should interpret and use it. This file contains information about title, content type, usage, copyright, licensing, version, language etc. This is described in detail below.

-

There shall be a folder for each included H5P library used by the package. These generic libraries may be reused by other H5P packages. As an example, a multi-choice question task may be used as a standalone block, or be included in a larger H5P package generating a game with quizzes.

-

Package file structure

-

A package contains the following elements:

-
    -
  1. A mandatory file in the root folder named h5p.json
  2. -
  3. An optional image file named h5p.jpg. This is an icon or an image of the application, 512 × 512 pixels. This image may be used by the platform as a preview of the application, and could be included in OG meta tags for use with social media.
  4. -
  5. One content folder, named content. This will contain the preset configuration for the application, as well as any required media files.
  6. -
  7. One or more library directories named the same as the library's internal name.
  8. -
-

h5p.json

-

The h5p.json file is a normal JSON text file containing a JSON object with the following predefined properties.

-

Mandatory properties:

-
    -
  • title - Name of the package. Would typically be used as header for a page displaying the package.
  • -
  • language - Standard language code. Use 'en' for english, 'nb' for norwegian "bokmål". Neutral content use "und".
  • -
  • machineName - Machine readable name of the library. This is the name that will be used for the library folder in the package too.
  • -
  • preloadedDependencies - Libraries that must be loaded on init. Specified as a list of objects with machineName, majorVersion and minorVersion. One would normally list all dependencies here to allow the platform displaying the package to merge JS and CSS files before returning the page.
  • -
  • embedTypes - List of ways to embed the package in the web page. Currently "div" and "iframe" are supported.
  • -

Optional properties:

-
  • contentType - Textual description of the type of content.
  • -
  • description - Textual description of the package.
  • -
  • author - Name of author.
  • -
  • license - Code for the content license. Use the following Creative Commons codes: cc-by, cc-by-sa, cc-by-nd, cc-by-nc, cc-by-nc-sa, cc-by-nc-nd. In addition for public domain: pd, and closed license: cr. More may be added later.
  • -
  • dynamicDependencies - Libraries that may be loaded dynamically during execution.
  • -
  • width - Width of the package content in cases where the package is not dynamically resizable.
  • -
  • height - Height of the package content.
  • -
  • metaKeywords - Suggestion for keywords for the application, as a string. May be used for OG meta tags for social media.
  • -
  • metaDescription - Suggestion for application metaDescription. May be used for OG meta tags for social media.
  • -
-

Eksempel på h5p.json:

-{
- "title": "Biologi-spillet",
- "contentType": "Game",
- "utilization": "Lær om biologi",
- "language": "nb",
- "author": "Amendor AS",
- "license": "cc-by-sa",
- "preloadedDependencies": [
- {
- "machineName": "H5P.Boardgame",
- "majorVersion": 1,
- "minorVersion": 0
- }, {
- "machineName": "H5P.QuestionSet",
- "majorVersion": 1,
- "minorVersion": 0
- }, {
- "machineName": "H5P.MultiChoice",
- "majorVersion": 1, "minorVersion": 0
- }, {
- "machineName": "EmbeddedJS",
- "majorVersion": 1,
- "minorVersion": 0
- } ],
- "embedTypes": ["div", "iframe"],
- "w": 635,
- "h": 500
-}
-

The content folder

-

Contains all the content for the package and its libraries. There shall be no content inside the library folders. The content folder shall contain a file named content.json, containing the JSON object that will be passed to the initializer for the main package library.

- -

Content required by libraries invoked from the main package library will get their contents passed from the main library. The JSON for this will be found within the main content.json for the package, and passed during initialization.

- -

Library folders

- -

A library folder contains all logic, stylesheets and graphics that will be common for all instances of a library. There shall be no content or interface text directly in these folders. All text displayed to the end user shall be passed as part of the library configuration. This make the libraries language independent.

- -

The root of a library folder shall contain a file name library.json formatted similar to the package's hp5.json, but with a few differences. The library shall also have one or more images in the root folder, named library.jpg, library1.jpg etc. Image sizes 512px × 512px, and will be used in the H5P editor tool.

- -

Libraries are not allowed to modify the document tree in ways that will have consequences for the web site or will be noticeable by the user without the library explicitly being initialized from the main package library or another invoked library.

- -

The library shall always include a JavaScript object function named the same as the defined library machineName (defined in library.json and used as the library folder name). This object will be instantiated with the library options as parameter. The resulting object must contain a function attach(target) that will be called after instantiation to attach the library DOM to the main DOM inside target

- -

Example

-

A library called H5P.multichoice would typically be instantiated and attached to the page like this:

-var multichoice = new H5P.multichoice(contentFromJson, contentId);
-multichoice.attach($multichoiceContainer);
- -

library.json

-

Mandatory properties:

-
    -
  • title - Human readable name of the library. May be used in the H5P editor and overviews of installed libraries.
  • -
  • majorVersion - Version major number. (The x in x.y.z). Positive integer.
  • -
  • minorVersion - Version minor number. (The y in x.y.z). Positive integer.
  • -
  • patchVersion - Version patch number. (The z in x.y.z). Positive integer. The system will automatically update to the latest patchVersion installed for all packages that use the library with the same major and minor version number. A new patch version must therefore not change any behaviour of the library, only fix errors.
  • -
  • machineName - Machine readable name for the library. Same as the folder name used.
  • -
  • preloadedJs - List of path to the javascript files required for the library. At least one file need to be present (the one defining the library object). Paths are relative to the library root folder.
  • -
-

Optional properties:

-
    -
  • author - Author name as text.
  • -
  • license - Code describing the library license. Use the following creative commons codes: cc-by, cc-by-sa, cc-by-nd, cc-by-nc, cc-by-nc-sa, cc-by-nc-nd. In addition use pd for public domain, and cr for closed source
  • -
  • description - Textual description of the library.
  • -
  • preloadedDependencies - Libraries that need to be loaded for this library to work. Specified as a list of objects with machineName, majorVersion and minorVersion for the required libraries.
  • -
  • dynamicDependencies - Libraries that may be loaded dynamically during library execution. Specified as a list of objects like preloadedDependencies above.
  • -
  • preloadedCss - List of paths to CSS files to be loaded with the library. Paths are relative to the library root folder.
  • -
  • w - Width in pixels for libraries that use a fixed width. Mandatory if the library shall be embedded in an iframe (see embedTypes below).
  • -
  • h - Height in pixels for libraries that use a fixed height. Mandatory if the library shall be embedded in an iframe (see embedTypes below).
  • -
  • embedTypes - List of possible ways to embed the package in the page. Available values are div and iframe.
  • -
-

Eksempel på library.json:

-{
- "title": "Boardgame",
- "description": "The user is presented with a board with several hotspots. By clicking a hotspot he invokes a mini-game.",
- "majorVersion": 1,
- "minorVersion": 0,
- "patchVersion": 6,
- "runnable": 1,
- "machineName": "H5P.Boardgame",
- "author": "Amendor AS",
- "license": "cc-by-sa",
- "preloadedDependencies": [
- {
- "machineName": "EmbeddedJS",
- "majorVersion": 1,
- "minorVersion": 0
- }, {
- "machineName": "H5P.MultiChoice",
- "majorVersion": 1,
- "minorVersion": 0
- }, {
- "machineName": "H5P.QuestionSet",
- "majorVersion": 1,
- "minorVersion": 0
- } ],
- "preloadedCss": [ {"path": "css/boardgame.css"} ],
- "preloadedJs": [ {"path": "js/boardgame.js"} ],
- "w": 635,
- "h": 500 }
- -

Allowed file types

-

Files that require server side execution or that cannot be regarded an open standard shall not be used. Allowed file types: js, json, png, jpg, gif, svg, css, mp3, wav (audio: PCM), m4a (audio: AAC), mp4 (video: H.264, audio: AAC/MP3), ogg (video: Theora, audio: Vorbis) and webm (video VP8, audio: Vorbis). Administrators of web sites implementing H5P may open for accepting further formats. HTML files shall not be used. HTML for each library shall be inserted from the library scripts to ease code reuse. (By avoiding content being defined in said HTML).

-

API functions

-

The following JavaScript functions are available through h5p:

-
    -
  • H5P.getUserData(namespace, variable)
  • -
  • H5P.setUserData(namespace, variable, data)
  • -
  • H5P.getUserStart(namespace)
  • -
  • H5P.setUserStop(namespace)
  • -
  • H5P.deleteUserData(namespace, variable)
  • -
  • H5P.getGlobalData(namespace, variable)
  • -
  • H5P.setGlobalData(namespace, variable, data)
  • -
  • H5P.deleteGlobalData(namespace, variable)
  • -
-

I tillegg er følgende api funksjoner tilgjengelig via ndla:

-
    -
  • H5P.setUserScore(contentId, score, maxScore)
  • -
-

Best practices

-

H5P is a very open standard. This is positive for flexibility. Most content may be produces as H5P. But this also allows for bad code, security weaknesses, code that may be difficult to reuse. Therefore the following best practices should be followed to get the most from H5P:

-
    -
  • Think reusability when creating a library. H5P support dependencies between libraries, so the same small quiz-library may be used in various larger packages or libraries.
  • -
  • H5P supports library updates. This enables all content using a common library to be updated at once. This must be accounted for when writing new libraries. A library should be as general as possible. The content format should be thought out so there are no changes to the required content data when a library is updated. Note: Multiple versions of a library may exists at the same time, only patch level updates will be automatically installed.
  • -
  • An H5P should not interact directly with the containing web site. It shall only affect elements within its own generated DOM tree. Elements shall also only be injected within the target defined on initialization. This is to avoid dependencies to a specific platform or web page.
  • -
  • Prefix objects, global functions, etc with h5p to minimize the chance of namespace conflicts with the rest of the web page. Remember that there may also be multiple H5P objects inserted on a page, so plan ahead to avoid conflicts.
  • -
  • Content should be responsive.
  • -
  • Content should be WCAG 2 AA compliant
  • -
  • All generated HTML should validate.
  • -
  • All CSS should validate (some browser specific non-standard CSS may at times be required)
  • -
  • Best practices for JavaScript, HTML, etc. should of course also be followed when writing an H5P.
  • -
diff --git a/h5p/h5plib/v126/joubel/core/embed.php b/h5p/h5plib/v126/joubel/core/embed.php deleted file mode 100644 index f8851ece4f3f1..0000000000000 --- a/h5p/h5plib/v126/joubel/core/embed.php +++ /dev/null @@ -1,20 +0,0 @@ - - - - - <?php print $content['title']; ?> - - - - - - - - - -
- - - diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.eot b/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.eot deleted file mode 100644 index 4b2bf455c309f..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.eot and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.svg b/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.svg deleted file mode 100644 index 8150d8ba7f087..0000000000000 --- a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.svg +++ /dev/null @@ -1,114 +0,0 @@ - - - - - - -{ - "fontFamily": "h5p-core-27", - "description": "Font generated by IcoMoon.", - "copyright": "H5P", - "majorVersion": 1, - "minorVersion": 1, - "version": "Version 1.1", - "fontId": "h5p-core-27", - "psName": "h5p-core-27", - "subFamily": "Regular", - "fullName": "h5p-core-27" -} - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.ttf b/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.ttf deleted file mode 100644 index 520db3a9fa50e..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.ttf and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.woff b/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.woff deleted file mode 100644 index 6ba910961f4da..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-core-28.woff and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.eot b/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.eot deleted file mode 100644 index 6afdd28df2a10..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.eot and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.svg b/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.svg deleted file mode 100644 index eec568bca03cd..0000000000000 --- a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.svg +++ /dev/null @@ -1,38 +0,0 @@ - - - - - - -{ - "fontFamily": "h5p-hub", - "description": "Font generated by IcoMoon.", - "majorVersion": 1, - "minorVersion": 3, - "version": "Version 1.3", - "fontId": "h5p-hub", - "psName": "h5p-hub", - "subFamily": "Regular", - "fullName": "h5p-hub" -} - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.ttf b/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.ttf deleted file mode 100644 index d1d40dd32e0b6..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.ttf and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.woff b/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.woff deleted file mode 100644 index c0535345a5132..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/h5p-hub-publish.woff and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/LICENSE-2.0.txt b/h5p/h5plib/v126/joubel/core/fonts/open-sans/LICENSE-2.0.txt deleted file mode 100644 index d645695673349..0000000000000 --- a/h5p/h5plib/v126/joubel/core/fonts/open-sans/LICENSE-2.0.txt +++ /dev/null @@ -1,202 +0,0 @@ - - Apache License - Version 2.0, January 2004 - http://www.apache.org/licenses/ - - TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION - - 1. Definitions. - - "License" shall mean the terms and conditions for use, reproduction, - and distribution as defined by Sections 1 through 9 of this document. - - "Licensor" shall mean the copyright owner or entity authorized by - the copyright owner that is granting the License. - - "Legal Entity" shall mean the union of the acting entity and all - other entities that control, are controlled by, or are under common - control with that entity. For the purposes of this definition, - "control" means (i) the power, direct or indirect, to cause the - direction or management of such entity, whether by contract or - otherwise, or (ii) ownership of fifty percent (50%) or more of the - outstanding shares, or (iii) beneficial ownership of such entity. - - "You" (or "Your") shall mean an individual or Legal Entity - exercising permissions granted by this License. - - "Source" form shall mean the preferred form for making modifications, - including but not limited to software source code, documentation - source, and configuration files. - - "Object" form shall mean any form resulting from mechanical - transformation or translation of a Source form, including but - not limited to compiled object code, generated documentation, - and conversions to other media types. - - "Work" shall mean the work of authorship, whether in Source or - Object form, made available under the License, as indicated by a - copyright notice that is included in or attached to the work - (an example is provided in the Appendix below). - - "Derivative Works" shall mean any work, whether in Source or Object - form, that is based on (or derived from) the Work and for which the - editorial revisions, annotations, elaborations, or other modifications - represent, as a whole, an original work of authorship. For the purposes - of this License, Derivative Works shall not include works that remain - separable from, or merely link (or bind by name) to the interfaces of, - the Work and Derivative Works thereof. - - "Contribution" shall mean any work of authorship, including - the original version of the Work and any modifications or additions - to that Work or Derivative Works thereof, that is intentionally - submitted to Licensor for inclusion in the Work by the copyright owner - or by an individual or Legal Entity authorized to submit on behalf of - the copyright owner. For the purposes of this definition, "submitted" - means any form of electronic, verbal, or written communication sent - to the Licensor or its representatives, including but not limited to - communication on electronic mailing lists, source code control systems, - and issue tracking systems that are managed by, or on behalf of, the - Licensor for the purpose of discussing and improving the Work, but - excluding communication that is conspicuously marked or otherwise - designated in writing by the copyright owner as "Not a Contribution." - - "Contributor" shall mean Licensor and any individual or Legal Entity - on behalf of whom a Contribution has been received by Licensor and - subsequently incorporated within the Work. - - 2. Grant of Copyright License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - copyright license to reproduce, prepare Derivative Works of, - publicly display, publicly perform, sublicense, and distribute the - Work and such Derivative Works in Source or Object form. - - 3. Grant of Patent License. Subject to the terms and conditions of - this License, each Contributor hereby grants to You a perpetual, - worldwide, non-exclusive, no-charge, royalty-free, irrevocable - (except as stated in this section) patent license to make, have made, - use, offer to sell, sell, import, and otherwise transfer the Work, - where such license applies only to those patent claims licensable - by such Contributor that are necessarily infringed by their - Contribution(s) alone or by combination of their Contribution(s) - with the Work to which such Contribution(s) was submitted. If You - institute patent litigation against any entity (including a - cross-claim or counterclaim in a lawsuit) alleging that the Work - or a Contribution incorporated within the Work constitutes direct - or contributory patent infringement, then any patent licenses - granted to You under this License for that Work shall terminate - as of the date such litigation is filed. - - 4. Redistribution. You may reproduce and distribute copies of the - Work or Derivative Works thereof in any medium, with or without - modifications, and in Source or Object form, provided that You - meet the following conditions: - - (a) You must give any other recipients of the Work or - Derivative Works a copy of this License; and - - (b) You must cause any modified files to carry prominent notices - stating that You changed the files; and - - (c) You must retain, in the Source form of any Derivative Works - that You distribute, all copyright, patent, trademark, and - attribution notices from the Source form of the Work, - excluding those notices that do not pertain to any part of - the Derivative Works; and - - (d) If the Work includes a "NOTICE" text file as part of its - distribution, then any Derivative Works that You distribute must - include a readable copy of the attribution notices contained - within such NOTICE file, excluding those notices that do not - pertain to any part of the Derivative Works, in at least one - of the following places: within a NOTICE text file distributed - as part of the Derivative Works; within the Source form or - documentation, if provided along with the Derivative Works; or, - within a display generated by the Derivative Works, if and - wherever such third-party notices normally appear. The contents - of the NOTICE file are for informational purposes only and - do not modify the License. You may add Your own attribution - notices within Derivative Works that You distribute, alongside - or as an addendum to the NOTICE text from the Work, provided - that such additional attribution notices cannot be construed - as modifying the License. - - You may add Your own copyright statement to Your modifications and - may provide additional or different license terms and conditions - for use, reproduction, or distribution of Your modifications, or - for any such Derivative Works as a whole, provided Your use, - reproduction, and distribution of the Work otherwise complies with - the conditions stated in this License. - - 5. Submission of Contributions. Unless You explicitly state otherwise, - any Contribution intentionally submitted for inclusion in the Work - by You to the Licensor shall be under the terms and conditions of - this License, without any additional terms or conditions. - Notwithstanding the above, nothing herein shall supersede or modify - the terms of any separate license agreement you may have executed - with Licensor regarding such Contributions. - - 6. Trademarks. This License does not grant permission to use the trade - names, trademarks, service marks, or product names of the Licensor, - except as required for reasonable and customary use in describing the - origin of the Work and reproducing the content of the NOTICE file. - - 7. Disclaimer of Warranty. Unless required by applicable law or - agreed to in writing, Licensor provides the Work (and each - Contributor provides its Contributions) on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or - implied, including, without limitation, any warranties or conditions - of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A - PARTICULAR PURPOSE. You are solely responsible for determining the - appropriateness of using or redistributing the Work and assume any - risks associated with Your exercise of permissions under this License. - - 8. Limitation of Liability. In no event and under no legal theory, - whether in tort (including negligence), contract, or otherwise, - unless required by applicable law (such as deliberate and grossly - negligent acts) or agreed to in writing, shall any Contributor be - liable to You for damages, including any direct, indirect, special, - incidental, or consequential damages of any character arising as a - result of this License or out of the use or inability to use the - Work (including but not limited to damages for loss of goodwill, - work stoppage, computer failure or malfunction, or any and all - other commercial damages or losses), even if such Contributor - has been advised of the possibility of such damages. - - 9. Accepting Warranty or Additional Liability. While redistributing - the Work or Derivative Works thereof, You may choose to offer, - and charge a fee for, acceptance of support, warranty, indemnity, - or other liability obligations and/or rights consistent with this - License. However, in accepting such obligations, You may act only - on Your own behalf and on Your sole responsibility, not on behalf - of any other Contributor, and only if You agree to indemnify, - defend, and hold each Contributor harmless for any liability - incurred by, or claims asserted against, such Contributor by reason - of your accepting any such warranty or additional liability. - - END OF TERMS AND CONDITIONS - - APPENDIX: How to apply the Apache License to your work. - - To apply the Apache License to your work, attach the following - boilerplate notice, with the fields enclosed by brackets "[]" - replaced with your own identifying information. (Don't include - the brackets!) The text should be enclosed in the appropriate - comment syntax for the file format. We also recommend that a - file or class name and description of purpose be included on the - same "printed page" as the copyright notice for easier - identification within third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2 deleted file mode 100644 index 8f3dd296456e6..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2 deleted file mode 100644 index bae23a0b4916e..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2 deleted file mode 100644 index 81d6c4d7c8a12..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek.woff2 deleted file mode 100644 index 6a34352903299..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-greek.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2 deleted file mode 100644 index 7623759a639d9..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2 deleted file mode 100644 index 31bfbe36764c9..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin.woff2 deleted file mode 100644 index 851d828ebf89f..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-latin.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2 deleted file mode 100644 index d095304c2d6b0..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2 deleted file mode 100644 index 37562053fd592..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2 deleted file mode 100644 index f3ce057e71c38..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2 deleted file mode 100644 index e2364c3a0e9b4..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2 deleted file mode 100644 index 7178a1cca3456..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2 deleted file mode 100644 index bc94bcb9db39b..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2 deleted file mode 100644 index 4f3c8b27c6247..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2 deleted file mode 100644 index 24b78e67d9076..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2 b/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2 deleted file mode 100644 index 9641ccd0e40f7..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2 and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/h5p-default-storage.class.php b/h5p/h5plib/v126/joubel/core/h5p-default-storage.class.php deleted file mode 100644 index 80757d19f5902..0000000000000 --- a/h5p/h5plib/v126/joubel/core/h5p-default-storage.class.php +++ /dev/null @@ -1,593 +0,0 @@ -path = $path; - $this->alteditorpath = $alteditorpath; - } - - /** - * Store the library folder. - * - * @param array $library - * Library properties - */ - public function saveLibrary($library) { - $dest = $this->path . '/libraries/' . H5PCore::libraryToFolderName($library); - - // Make sure destination dir doesn't exist - H5PCore::deleteFileTree($dest); - - // Move library folder - self::copyFileTree($library['uploadDirectory'], $dest); - } - - public function deleteLibrary($library) { - // TODO - } - - /** - * Store the content folder. - * - * @param string $source - * Path on file system to content directory. - * @param array $content - * Content properties - */ - public function saveContent($source, $content) { - $dest = "{$this->path}/content/{$content['id']}"; - - // Remove any old content - H5PCore::deleteFileTree($dest); - - self::copyFileTree($source, $dest); - } - - /** - * Remove content folder. - * - * @param array $content - * Content properties - */ - public function deleteContent($content) { - H5PCore::deleteFileTree("{$this->path}/content/{$content['id']}"); - } - - /** - * Creates a stored copy of the content folder. - * - * @param string $id - * Identifier of content to clone. - * @param int $newId - * The cloned content's identifier - */ - public function cloneContent($id, $newId) { - $path = $this->path . '/content/'; - if (file_exists($path . $id)) { - self::copyFileTree($path . $id, $path . $newId); - } - } - - /** - * Get path to a new unique tmp folder. - * - * @return string - * Path - */ - public function getTmpPath() { - $temp = "{$this->path}/temp"; - self::dirReady($temp); - return "{$temp}/" . uniqid('h5p-'); - } - - /** - * Fetch content folder and save in target directory. - * - * @param int $id - * Content identifier - * @param string $target - * Where the content folder will be saved - */ - public function exportContent($id, $target) { - $source = "{$this->path}/content/{$id}"; - if (file_exists($source)) { - // Copy content folder if it exists - self::copyFileTree($source, $target); - } - else { - // No contnet folder, create emty dir for content.json - self::dirReady($target); - } - } - - /** - * Fetch library folder and save in target directory. - * - * @param array $library - * Library properties - * @param string $target - * Where the library folder will be saved - * @param string $developmentPath - * Folder that library resides in - */ - public function exportLibrary($library, $target, $developmentPath=NULL) { - $folder = H5PCore::libraryToFolderName($library); - - $srcPath = ($developmentPath === NULL ? "/libraries/{$folder}" : $developmentPath); - self::copyFileTree("{$this->path}{$srcPath}", "{$target}/{$folder}"); - } - - /** - * Save export in file system - * - * @param string $source - * Path on file system to temporary export file. - * @param string $filename - * Name of export file. - * @throws Exception Unable to save the file - */ - public function saveExport($source, $filename) { - $this->deleteExport($filename); - - if (!self::dirReady("{$this->path}/exports")) { - throw new Exception("Unable to create directory for H5P export file."); - } - - if (!copy($source, "{$this->path}/exports/{$filename}")) { - throw new Exception("Unable to save H5P export file."); - } - } - - /** - * Removes given export file - * - * @param string $filename - */ - public function deleteExport($filename) { - $target = "{$this->path}/exports/{$filename}"; - if (file_exists($target)) { - unlink($target); - } - } - - /** - * Check if the given export file exists - * - * @param string $filename - * @return boolean - */ - public function hasExport($filename) { - $target = "{$this->path}/exports/{$filename}"; - return file_exists($target); - } - - /** - * Will concatenate all JavaScrips and Stylesheets into two files in order - * to improve page performance. - * - * @param array $files - * A set of all the assets required for content to display - * @param string $key - * Hashed key for cached asset - */ - public function cacheAssets(&$files, $key) { - foreach ($files as $type => $assets) { - if (empty($assets)) { - continue; // Skip no assets - } - - $content = ''; - foreach ($assets as $asset) { - // Get content from asset file - $assetContent = file_get_contents($this->path . $asset->path); - $cssRelPath = preg_replace('/[^\/]+$/', '', $asset->path); - - // Get file content and concatenate - if ($type === 'scripts') { - $content .= $assetContent . ";\n"; - } - else { - // Rewrite relative URLs used inside stylesheets - $content .= preg_replace_callback( - '/url\([\'"]?([^"\')]+)[\'"]?\)/i', - function ($matches) use ($cssRelPath) { - if (preg_match("/^(data:|([a-z0-9]+:)?\/)/i", $matches[1]) === 1) { - return $matches[0]; // Not relative, skip - } - return 'url("../' . $cssRelPath . $matches[1] . '")'; - }, - $assetContent) . "\n"; - } - } - - self::dirReady("{$this->path}/cachedassets"); - $ext = ($type === 'scripts' ? 'js' : 'css'); - $outputfile = "/cachedassets/{$key}.{$ext}"; - file_put_contents($this->path . $outputfile, $content); - $files[$type] = array((object) array( - 'path' => $outputfile, - 'version' => '' - )); - } - } - - /** - * Will check if there are cache assets available for content. - * - * @param string $key - * Hashed key for cached asset - * @return array - */ - public function getCachedAssets($key) { - $files = array(); - - $js = "/cachedassets/{$key}.js"; - if (file_exists($this->path . $js)) { - $files['scripts'] = array((object) array( - 'path' => $js, - 'version' => '' - )); - } - - $css = "/cachedassets/{$key}.css"; - if (file_exists($this->path . $css)) { - $files['styles'] = array((object) array( - 'path' => $css, - 'version' => '' - )); - } - - return empty($files) ? NULL : $files; - } - - /** - * Remove the aggregated cache files. - * - * @param array $keys - * The hash keys of removed files - */ - public function deleteCachedAssets($keys) { - foreach ($keys as $hash) { - foreach (array('js', 'css') as $ext) { - $path = "{$this->path}/cachedassets/{$hash}.{$ext}"; - if (file_exists($path)) { - unlink($path); - } - } - } - } - - /** - * Read file content of given file and then return it. - * - * @param string $file_path - * @return string - */ - public function getContent($file_path) { - return file_get_contents($file_path); - } - - /** - * Save files uploaded through the editor. - * The files must be marked as temporary until the content form is saved. - * - * @param \H5peditorFile $file - * @param int $contentid - */ - public function saveFile($file, $contentId) { - // Prepare directory - if (empty($contentId)) { - // Should be in editor tmp folder - $path = $this->getEditorPath(); - } - else { - // Should be in content folder - $path = $this->path . '/content/' . $contentId; - } - $path .= '/' . $file->getType() . 's'; - self::dirReady($path); - - // Add filename to path - $path .= '/' . $file->getName(); - - copy($_FILES['file']['tmp_name'], $path); - - return $file; - } - - /** - * Copy a file from another content or editor tmp dir. - * Used when copy pasting content in H5P Editor. - * - * @param string $file path + name - * @param string|int $fromid Content ID or 'editor' string - * @param int $toid Target Content ID - */ - public function cloneContentFile($file, $fromId, $toId) { - // Determine source path - if ($fromId === 'editor') { - $sourcepath = $this->getEditorPath(); - } - else { - $sourcepath = "{$this->path}/content/{$fromId}"; - } - $sourcepath .= '/' . $file; - - // Determine target path - $filename = basename($file); - $filedir = str_replace($filename, '', $file); - $targetpath = "{$this->path}/content/{$toId}/{$filedir}"; - - // Make sure it's ready - self::dirReady($targetpath); - - $targetpath .= $filename; - - // Check to see if source exist and if target doesn't - if (!file_exists($sourcepath) || file_exists($targetpath)) { - return; // Nothing to copy from or target already exists - } - - copy($sourcepath, $targetpath); - } - - /** - * Copy a content from one directory to another. Defaults to cloning - * content from the current temporary upload folder to the editor path. - * - * @param string $source path to source directory - * @param string $contentId Id of contentarray - */ - public function moveContentDirectory($source, $contentId = NULL) { - if ($source === NULL) { - return NULL; - } - - // TODO: Remove $contentId and never copy temporary files into content folder. JI-366 - if ($contentId === NULL || $contentId == 0) { - $target = $this->getEditorPath(); - } - else { - // Use content folder - $target = "{$this->path}/content/{$contentId}"; - } - - $contentSource = $source . '/' . 'content'; - $contentFiles = array_diff(scandir($contentSource), array('.','..', 'content.json')); - foreach ($contentFiles as $file) { - if (is_dir("{$contentSource}/{$file}")) { - self::copyFileTree("{$contentSource}/{$file}", "{$target}/{$file}"); - } - else { - copy("{$contentSource}/{$file}", "{$target}/{$file}"); - } - } - - // TODO: Return list of all files so that they can be marked as temporary. JI-366 - } - - /** - * Checks to see if content has the given file. - * Used when saving content. - * - * @param string $file path + name - * @param int $contentId - * @return string File ID or NULL if not found - */ - public function getContentFile($file, $contentId) { - $path = "{$this->path}/content/{$contentId}/{$file}"; - return file_exists($path) ? $path : NULL; - } - - /** - * Checks to see if content has the given file. - * Used when saving content. - * - * @param string $file path + name - * @param int $contentid - * @return string|int File ID or NULL if not found - */ - public function removeContentFile($file, $contentId) { - $path = "{$this->path}/content/{$contentId}/{$file}"; - if (file_exists($path)) { - unlink($path); - - // Clean up any empty parent directories to avoid cluttering the file system - $parts = explode('/', $path); - while (array_pop($parts) !== NULL) { - $dir = implode('/', $parts); - if (is_dir($dir) && count(scandir($dir)) === 2) { // empty contains '.' and '..' - rmdir($dir); // Remove empty parent - } - else { - return; // Not empty - } - } - } - } - - /** - * Check if server setup has write permission to - * the required folders - * - * @return bool True if site can write to the H5P files folder - */ - public function hasWriteAccess() { - return self::dirReady($this->path); - } - - /** - * Check if the file presave.js exists in the root of the library - * - * @param string $libraryFolder - * @param string $developmentPath - * @return bool - */ - public function hasPresave($libraryFolder, $developmentPath = null) { - $path = is_null($developmentPath) ? 'libraries' . '/' . $libraryFolder : $developmentPath; - $filePath = realpath($this->path . '/' . $path . '/' . 'presave.js'); - return file_exists($filePath); - } - - /** - * Check if upgrades script exist for library. - * - * @param string $machineName - * @param int $majorVersion - * @param int $minorVersion - * @return string Relative path - */ - public function getUpgradeScript($machineName, $majorVersion, $minorVersion) { - $upgrades = "/libraries/{$machineName}-{$majorVersion}.{$minorVersion}/upgrades.js"; - if (file_exists($this->path . $upgrades)) { - return $upgrades; - } - else { - return NULL; - } - } - - /** - * Store the given stream into the given file. - * - * @param string $path - * @param string $file - * @param resource $stream - * @return bool - */ - public function saveFileFromZip($path, $file, $stream) { - $filePath = $path . '/' . $file; - - // Make sure the directory exists first - $matches = array(); - preg_match('/(.+)\/[^\/]*$/', $filePath, $matches); - self::dirReady($matches[1]); - - // Store in local storage folder - return file_put_contents($filePath, $stream); - } - - /** - * Recursive function for copying directories. - * - * @param string $source - * From path - * @param string $destination - * To path - * @return boolean - * Indicates if the directory existed. - * - * @throws Exception Unable to copy the file - */ - private static function copyFileTree($source, $destination) { - if (!self::dirReady($destination)) { - throw new \Exception('unabletocopy'); - } - - $ignoredFiles = self::getIgnoredFiles("{$source}/.h5pignore"); - - $dir = opendir($source); - if ($dir === FALSE) { - trigger_error('Unable to open directory ' . $source, E_USER_WARNING); - throw new \Exception('unabletocopy'); - } - - while (false !== ($file = readdir($dir))) { - if (($file != '.') && ($file != '..') && $file != '.git' && $file != '.gitignore' && !in_array($file, $ignoredFiles)) { - if (is_dir("{$source}/{$file}")) { - self::copyFileTree("{$source}/{$file}", "{$destination}/{$file}"); - } - else { - copy("{$source}/{$file}", "{$destination}/{$file}"); - } - } - } - closedir($dir); - } - - /** - * Retrieve array of file names from file. - * - * @param string $file - * @return array Array with files that should be ignored - */ - private static function getIgnoredFiles($file) { - if (file_exists($file) === FALSE) { - return array(); - } - - $contents = file_get_contents($file); - if ($contents === FALSE) { - return array(); - } - - return preg_split('/\s+/', $contents); - } - - /** - * Recursive function that makes sure the specified directory exists and - * is writable. - * - * @param string $path - * @return bool - */ - private static function dirReady($path) { - if (!file_exists($path)) { - $parent = preg_replace("/\/[^\/]+\/?$/", '', $path); - if (!self::dirReady($parent)) { - return FALSE; - } - - mkdir($path, 0777, true); - } - - if (!is_dir($path)) { - trigger_error('Path is not a directory ' . $path, E_USER_WARNING); - return FALSE; - } - - if (!is_writable($path)) { - trigger_error('Unable to write to ' . $path . ' – check directory permissions –', E_USER_WARNING); - return FALSE; - } - - return TRUE; - } - - /** - * Easy helper function for retrieving the editor path - * - * @return string Path to editor files - */ - private function getEditorPath() { - return ($this->alteditorpath !== NULL ? $this->alteditorpath : "{$this->path}/editor"); - } -} diff --git a/h5p/h5plib/v126/joubel/core/h5p-development.class.php b/h5p/h5plib/v126/joubel/core/h5p-development.class.php deleted file mode 100644 index b85df27150dba..0000000000000 --- a/h5p/h5plib/v126/joubel/core/h5p-development.class.php +++ /dev/null @@ -1,193 +0,0 @@ -h5pF = $H5PFramework; - $this->language = $language; - $this->filesPath = $filesPath; - if ($libraries !== NULL) { - $this->libraries = $libraries; - } - else { - $this->findLibraries($filesPath . '/development'); - } - } - - /** - * Get contents of file. - * - * @param string $file File path. - * @return mixed String on success or NULL on failure. - */ - private function getFileContents($file) { - if (file_exists($file) === FALSE) { - return NULL; - } - - $contents = file_get_contents($file); - if ($contents === FALSE) { - return NULL; - } - - return $contents; - } - - /** - * Scans development directory and find all libraries. - * - * @param string $path Libraries development folder - */ - private function findLibraries($path) { - $this->libraries = array(); - - if (is_dir($path) === FALSE) { - return; - } - - $contents = scandir($path); - - for ($i = 0, $s = count($contents); $i < $s; $i++) { - if ($contents[$i][0] === '.') { - continue; // Skip hidden stuff. - } - - $libraryPath = $path . '/' . $contents[$i]; - $libraryJSON = $this->getFileContents($libraryPath . '/library.json'); - if ($libraryJSON === NULL) { - continue; // No JSON file, skip. - } - - $library = json_decode($libraryJSON, TRUE); - if ($library === NULL) { - continue; // Invalid JSON. - } - - // TODO: Validate props? Not really needed, is it? this is a dev site. - - $library['libraryId'] = $this->h5pF->getLibraryId($library['machineName'], $library['majorVersion'], $library['minorVersion']); - - // Convert metadataSettings values to boolean & json_encode it before saving - $library['metadataSettings'] = isset($library['metadataSettings']) ? - H5PMetadata::boolifyAndEncodeSettings($library['metadataSettings']) : - NULL; - - // Save/update library. - $this->h5pF->saveLibraryData($library, $library['libraryId'] === FALSE); - - // Need to decode it again, since it is served from here. - $library['metadataSettings'] = isset($library['metadataSettings']) - ? json_decode($library['metadataSettings']) - : NULL; - - $library['path'] = 'development/' . $contents[$i]; - $this->libraries[H5PDevelopment::libraryToString($library['machineName'], $library['majorVersion'], $library['minorVersion'])] = $library; - } - - // TODO: Should we remove libraries without files? Not really needed, but must be cleaned up some time, right? - - // Go trough libraries and insert dependencies. Missing deps. will just be ignored and not available. (I guess?!) - $this->h5pF->lockDependencyStorage(); - foreach ($this->libraries as $library) { - $this->h5pF->deleteLibraryDependencies($library['libraryId']); - // This isn't optimal, but without it we would get duplicate warnings. - // TODO: You might get PDOExceptions if two or more requests does this at the same time!! - $types = array('preloaded', 'dynamic', 'editor'); - foreach ($types as $type) { - if (isset($library[$type . 'Dependencies'])) { - $this->h5pF->saveLibraryDependencies($library['libraryId'], $library[$type . 'Dependencies'], $type); - } - } - } - $this->h5pF->unlockDependencyStorage(); - // TODO: Deps must be inserted into h5p_nodes_libraries as well... ? But only if they are used?! - } - - /** - * @return array Libraries in development folder. - */ - public function getLibraries() { - return $this->libraries; - } - - /** - * Get library - * - * @param string $name of the library. - * @param int $majorVersion of the library. - * @param int $minorVersion of the library. - * @return array library. - */ - public function getLibrary($name, $majorVersion, $minorVersion) { - $library = H5PDevelopment::libraryToString($name, $majorVersion, $minorVersion); - return isset($this->libraries[$library]) === TRUE ? $this->libraries[$library] : NULL; - } - - /** - * Get semantics for the given library. - * - * @param string $name of the library. - * @param int $majorVersion of the library. - * @param int $minorVersion of the library. - * @return string Semantics - */ - public function getSemantics($name, $majorVersion, $minorVersion) { - $library = H5PDevelopment::libraryToString($name, $majorVersion, $minorVersion); - if (isset($this->libraries[$library]) === FALSE) { - return NULL; - } - return $this->getFileContents($this->filesPath . $this->libraries[$library]['path'] . '/semantics.json'); - } - - /** - * Get translations for the given library. - * - * @param string $name of the library. - * @param int $majorVersion of the library. - * @param int $minorVersion of the library. - * @param $language - * @return string Translation - */ - public function getLanguage($name, $majorVersion, $minorVersion, $language) { - $library = H5PDevelopment::libraryToString($name, $majorVersion, $minorVersion); - - if (isset($this->libraries[$library]) === FALSE) { - return NULL; - } - - return $this->getFileContents($this->filesPath . $this->libraries[$library]['path'] . '/language/' . $language . '.json'); - } - - /** - * Writes library as string on the form "name majorVersion.minorVersion" - * - * @param string $name Machine readable library name - * @param integer $majorVersion - * @param $minorVersion - * @return string Library identifier. - */ - public static function libraryToString($name, $majorVersion, $minorVersion) { - return $name . ' ' . $majorVersion . '.' . $minorVersion; - } -} diff --git a/h5p/h5plib/v126/joubel/core/h5p-event-base.class.php b/h5p/h5plib/v126/joubel/core/h5p-event-base.class.php deleted file mode 100644 index a3569add22b67..0000000000000 --- a/h5p/h5plib/v126/joubel/core/h5p-event-base.class.php +++ /dev/null @@ -1,193 +0,0 @@ - – content view - * embed – viewed through embed code - * shortcode – viewed through internal shortcode - * edit – opened in editor - * delete – deleted - * create – created through editor - * create upload – created through upload - * update – updated through editor - * update upload – updated through upload - * upgrade – upgraded - * - * results, – view own results - * content – view results for content - * set – new results inserted or updated - * - * settings, – settings page loaded - * - * library, – loaded in editor - * create – new library installed - * update – old library updated - * - * @param string $type - * Name of event type - * @param string $sub_type - * Name of event sub type - * @param string $content_id - * Identifier for content affected by the event - * @param string $content_title - * Content title (makes it easier to know which content was deleted etc.) - * @param string $library_name - * Name of the library affected by the event - * @param string $library_version - * Library version - */ - function __construct($type, $sub_type = NULL, $content_id = NULL, $content_title = NULL, $library_name = NULL, $library_version = NULL) { - $this->type = $type; - $this->sub_type = $sub_type; - $this->content_id = $content_id; - $this->content_title = $content_title; - $this->library_name = $library_name; - $this->library_version = $library_version; - $this->time = time(); - - if (self::validLogLevel($type, $sub_type)) { - $this->save(); - } - if (self::validStats($type, $sub_type)) { - $this->saveStats(); - } - } - - /** - * Determines if the event type should be saved/logged. - * - * @param string $type - * Name of event type - * @param string $sub_type - * Name of event sub type - * @return boolean - */ - private static function validLogLevel($type, $sub_type) { - switch (self::$log_level) { - default: - case self::LOG_NONE: - return FALSE; - case self::LOG_ALL: - return TRUE; // Log everything - case self::LOG_ACTIONS: - if (self::isAction($type, $sub_type)) { - return TRUE; // Log actions - } - return FALSE; - } - } - - /** - * Check if the event should be included in the statistics counter. - * - * @param string $type - * Name of event type - * @param string $sub_type - * Name of event sub type - * @return boolean - */ - private static function validStats($type, $sub_type) { - if ( ($type === 'content' && $sub_type === 'shortcode insert') || // Count number of shortcode inserts - ($type === 'library' && $sub_type === NULL) || // Count number of times library is loaded in editor - ($type === 'results' && $sub_type === 'content') ) { // Count number of times results page has been opened - return TRUE; - } - elseif (self::isAction($type, $sub_type)) { // Count all actions - return TRUE; - } - return FALSE; - } - - /** - * Check if event type is an action. - * - * @param string $type - * Name of event type - * @param string $sub_type - * Name of event sub type - * @return boolean - */ - private static function isAction($type, $sub_type) { - if ( ($type === 'content' && in_array($sub_type, array('create', 'create upload', 'update', 'update upload', 'upgrade', 'delete'))) || - ($type === 'library' && in_array($sub_type, array('create', 'update'))) ) { - return TRUE; // Log actions - } - return FALSE; - } - - /** - * A helper which makes it easier for systems to save the data. - * Add all relevant properties to a assoc. array. - * There are no NULL values. Empty string or 0 is used instead. - * Used by both Drupal and WordPress. - * - * @return array with keyed values - */ - protected function getDataArray() { - return array( - 'created_at' => $this->time, - 'type' => $this->type, - 'sub_type' => empty($this->sub_type) ? '' : $this->sub_type, - 'content_id' => empty($this->content_id) ? 0 : $this->content_id, - 'content_title' => empty($this->content_title) ? '' : $this->content_title, - 'library_name' => empty($this->library_name) ? '' : $this->library_name, - 'library_version' => empty($this->library_version) ? '' : $this->library_version - ); - } - - /** - * A helper which makes it easier for systems to save the data. - * Used in WordPress. - * - * @return array with strings - */ - protected function getFormatArray() { - return array( - '%d', - '%s', - '%s', - '%d', - '%s', - '%s', - '%s' - ); - } - - /** - * Stores the event data in the database. - * - * Must be overridden by plugin. - */ - abstract protected function save(); - - /** - * Add current event data to statistics counter. - * - * Must be overridden by plugin. - */ - abstract protected function saveStats(); -} diff --git a/h5p/h5plib/v126/joubel/core/h5p-file-storage.interface.php b/h5p/h5plib/v126/joubel/core/h5p-file-storage.interface.php deleted file mode 100644 index 724a75cab4992..0000000000000 --- a/h5p/h5plib/v126/joubel/core/h5p-file-storage.interface.php +++ /dev/null @@ -1,231 +0,0 @@ - array( - 'type' => 'text', - 'maxLength' => 255 - ), - 'a11yTitle' => array( - 'type' => 'text', - 'maxLength' => 255, - ), - 'authors' => array( - 'type' => 'json' - ), - 'changes' => array( - 'type' => 'json' - ), - 'source' => array( - 'type' => 'text', - 'maxLength' => 255 - ), - 'license' => array( - 'type' => 'text', - 'maxLength' => 32 - ), - 'licenseVersion' => array( - 'type' => 'text', - 'maxLength' => 10 - ), - 'licenseExtras' => array( - 'type' => 'text', - 'maxLength' => 5000 - ), - 'authorComments' => array( - 'type' => 'text', - 'maxLength' => 5000 - ), - 'yearFrom' => array( - 'type' => 'int' - ), - 'yearTo' => array( - 'type' => 'int' - ), - 'defaultLanguage' => array( - 'type' => 'text', - 'maxLength' => 32, - ) - ); - - /** - * JSON encode metadata - * - * @param object $content - * @return string - */ - public static function toJSON($content) { - // Note: deliberatly creating JSON string "manually" to improve performance - return - '{"title":' . (isset($content->title) ? json_encode($content->title) : 'null') . - ',"a11yTitle":' . (isset($content->a11y_title) ? $content->a11y_title : 'null') . - ',"authors":' . (isset($content->authors) ? $content->authors : 'null') . - ',"source":' . (isset($content->source) ? '"' . $content->source . '"' : 'null') . - ',"license":' . (isset($content->license) ? '"' . $content->license . '"' : 'null') . - ',"licenseVersion":' . (isset($content->license_version) ? '"' . $content->license_version . '"' : 'null') . - ',"licenseExtras":' . (isset($content->license_extras) ? json_encode($content->license_extras) : 'null') . - ',"yearFrom":' . (isset($content->year_from) ? $content->year_from : 'null') . - ',"yearTo":' . (isset($content->year_to) ? $content->year_to : 'null') . - ',"changes":' . (isset($content->changes) ? $content->changes : 'null') . - ',"defaultLanguage":' . (isset($content->default_language) ? '"' . $content->default_language . '"' : 'null') . - ',"authorComments":' . (isset($content->author_comments) ? json_encode($content->author_comments) : 'null') . '}'; - } - - /** - * Make the metadata into an associative array keyed by the property names - * @param mixed $metadata Array or object containing metadata - * @param bool $include_title - * @param bool $include_missing For metadata fields not being set, skip 'em. - * Relevant for content upgrade - * @param array $types - * @return array - */ - public static function toDBArray($metadata, $include_title = true, $include_missing = true, &$types = array()) { - $fields = array(); - - if (!is_array($metadata)) { - $metadata = (array) $metadata; - } - - foreach (self::$fields as $key => $config) { - - // Ignore title? - if ($key === 'title' && !$include_title) { - continue; - } - - $exists = array_key_exists($key, $metadata); - - // Don't include missing fields - if (!$include_missing && !$exists) { - continue; - } - - $value = $exists ? $metadata[$key] : null; - - // lowerCamelCase to snake_case - $db_field_name = strtolower(preg_replace('/(? $config['maxLength']) { - $value = mb_substr($value, 0, $config['maxLength']); - } - $types[] = '%s'; - break; - - case 'int': - $value = ($value !== null) ? intval($value) : null; - $types[] = '%d'; - break; - - case 'json': - $value = ($value !== null) ? json_encode($value) : null; - $types[] = '%s'; - break; - } - - $fields[$db_field_name] = $value; - } - - return $fields; - } - - /** - * The metadataSettings field in libraryJson uses 1 for true and 0 for false. - * Here we are converting these to booleans, and also doing JSON encoding. - * This is invoked before the library data is beeing inserted/updated to DB. - * - * @param array $metadataSettings - * @return string - */ - public static function boolifyAndEncodeSettings($metadataSettings) { - // Convert metadataSettings values to boolean - if (isset($metadataSettings['disable'])) { - $metadataSettings['disable'] = $metadataSettings['disable'] === 1; - } - if (isset($metadataSettings['disableExtraTitleField'])) { - $metadataSettings['disableExtraTitleField'] = $metadataSettings['disableExtraTitleField'] === 1; - } - - return json_encode($metadataSettings); - } -} diff --git a/h5p/h5plib/v126/joubel/core/h5p.classes.php b/h5p/h5plib/v126/joubel/core/h5p.classes.php deleted file mode 100644 index 50c698329ae61..0000000000000 --- a/h5p/h5plib/v126/joubel/core/h5p.classes.php +++ /dev/null @@ -1,5554 +0,0 @@ - '/^.{1,255}$/', - 'language' => '/^[-a-zA-Z]{1,10}$/', - 'preloadedDependencies' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - 'mainLibrary' => '/^[$a-z_][0-9a-z_\.$]{1,254}$/i', - 'embedTypes' => array('iframe', 'div'), - ); - - private $h5pOptional = array( - 'contentType' => '/^.{1,255}$/', - 'dynamicDependencies' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - // deprecated - 'author' => '/^.{1,255}$/', - 'authors' => array( - 'name' => '/^.{1,255}$/', - 'role' => '/^\w+$/', - ), - 'source' => '/^(http[s]?:\/\/.+)$/', - 'license' => '/^(CC BY|CC BY-SA|CC BY-ND|CC BY-NC|CC BY-NC-SA|CC BY-NC-ND|CC0 1\.0|GNU GPL|PD|ODC PDDL|CC PDM|U|C)$/', - 'licenseVersion' => '/^(1\.0|2\.0|2\.5|3\.0|4\.0)$/', - 'licenseExtras' => '/^.{1,5000}$/s', - 'yearsFrom' => '/^([0-9]{1,4})$/', - 'yearsTo' => '/^([0-9]{1,4})$/', - 'changes' => array( - 'date' => '/^[0-9]{2}-[0-9]{2}-[0-9]{2} [0-9]{1,2}:[0-9]{2}:[0-9]{2}$/', - 'author' => '/^.{1,255}$/', - 'log' => '/^.{1,5000}$/s' - ), - 'authorComments' => '/^.{1,5000}$/s', - 'w' => '/^[0-9]{1,4}$/', - 'h' => '/^[0-9]{1,4}$/', - // deprecated - 'metaKeywords' => '/^.{1,}$/', - // deprecated - 'metaDescription' => '/^.{1,}$/', - ); - - // Schemas used to validate the library files - private $libraryRequired = array( - 'title' => '/^.{1,255}$/', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - 'patchVersion' => '/^[0-9]{1,5}$/', - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'runnable' => '/^(0|1)$/', - ); - - private $libraryOptional = array( - 'author' => '/^.{1,255}$/', - 'license' => '/^(cc-by|cc-by-sa|cc-by-nd|cc-by-nc|cc-by-nc-sa|cc-by-nc-nd|pd|cr|MIT|GPL1|GPL2|GPL3|MPL|MPL2)$/', - 'description' => '/^.{1,}$/', - 'metadataSettings' => array( - 'disable' => '/^(0|1)$/', - 'disableExtraTitleField' => '/^(0|1)$/' - ), - 'dynamicDependencies' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - 'preloadedDependencies' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - 'editorDependencies' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - 'preloadedJs' => array( - 'path' => '/^((\\\|\/)?[a-z_\-\s0-9\.]+)+\.js$/i', - ), - 'preloadedCss' => array( - 'path' => '/^((\\\|\/)?[a-z_\-\s0-9\.]+)+\.css$/i', - ), - 'dropLibraryCss' => array( - 'machineName' => '/^[\w0-9\-\.]{1,255}$/i', - ), - 'w' => '/^[0-9]{1,4}$/', - 'h' => '/^[0-9]{1,4}$/', - 'embedTypes' => array('iframe', 'div'), - 'fullscreen' => '/^(0|1)$/', - 'coreApi' => array( - 'majorVersion' => '/^[0-9]{1,5}$/', - 'minorVersion' => '/^[0-9]{1,5}$/', - ), - ); - - /** - * Constructor for the H5PValidator - * - * @param H5PFrameworkInterface $H5PFramework - * The frameworks implementation of the H5PFrameworkInterface - * @param H5PCore $H5PCore - */ - public function __construct($H5PFramework, $H5PCore) { - $this->h5pF = $H5PFramework; - $this->h5pC = $H5PCore; - $this->h5pCV = new H5PContentValidator($this->h5pF, $this->h5pC); - } - - /** - * Validates a .h5p file - * - * @param bool $skipContent - * @param bool $upgradeOnly - * @return bool TRUE if the .h5p file is valid - * TRUE if the .h5p file is valid - */ - public function isValidPackage($skipContent = FALSE, $upgradeOnly = FALSE) { - // Create a temporary dir to extract package in. - $tmpDir = $this->h5pF->getUploadedH5pFolderPath(); - $tmpPath = $this->h5pF->getUploadedH5pPath(); - - // Check dependencies, make sure Zip is present - if (!class_exists('ZipArchive')) { - $this->h5pF->setErrorMessage($this->h5pF->t('Your PHP version does not support ZipArchive.'), 'zip-archive-unsupported'); - unlink($tmpPath); - return FALSE; - } - if (!extension_loaded('mbstring')) { - $this->h5pF->setErrorMessage($this->h5pF->t('The mbstring PHP extension is not loaded. H5P need this to function properly'), 'mbstring-unsupported'); - unlink($tmpPath); - return FALSE; - } - - // Only allow files with the .h5p extension: - if (strtolower(substr($tmpPath, -3)) !== 'h5p') { - $this->h5pF->setErrorMessage($this->h5pF->t('The file you uploaded is not a valid HTML5 Package (It does not have the .h5p file extension)'), 'missing-h5p-extension'); - unlink($tmpPath); - return FALSE; - } - - // Extract and then remove the package file. - $zip = new ZipArchive; - - // Open the package - if ($zip->open($tmpPath) !== TRUE) { - $this->h5pF->setErrorMessage($this->h5pF->t('The file you uploaded is not a valid HTML5 Package (We are unable to unzip it)'), 'unable-to-unzip'); - unlink($tmpPath); - return FALSE; - } - - if ($this->h5pC->disableFileCheck !== TRUE) { - list($contentWhitelist, $contentRegExp) = $this->getWhitelistRegExp(FALSE); - list($libraryWhitelist, $libraryRegExp) = $this->getWhitelistRegExp(TRUE); - } - $canInstall = $this->h5pC->mayUpdateLibraries(); - - $valid = TRUE; - $libraries = array(); - - $totalSize = 0; - $mainH5pExists = FALSE; - $contentExists = FALSE; - - // Check for valid file types, JSON files + file sizes before continuing to unpack. - for ($i = 0; $i < $zip->numFiles; $i++) { - $fileStat = $zip->statIndex($i); - - if (!empty($this->h5pC->maxFileSize) && $fileStat['size'] > $this->h5pC->maxFileSize) { - // Error file is too large - $this->h5pF->setErrorMessage($this->h5pF->t('One of the files inside the package exceeds the maximum file size allowed. (%file %used > %max)', array('%file' => $fileStat['name'], '%used' => ($fileStat['size'] / 1048576) . ' MB', '%max' => ($this->h5pC->maxFileSize / 1048576) . ' MB')), 'file-size-too-large'); - $valid = FALSE; - } - $totalSize += $fileStat['size']; - - $fileName = mb_strtolower($fileStat['name']); - if (preg_match('/(^[\._]|\/[\._]|\\\[\._])/', $fileName) !== 0) { - continue; // Skip any file or folder starting with a . or _ - } - elseif ($fileName === 'h5p.json') { - $mainH5pExists = TRUE; - } - elseif ($fileName === 'content/content.json') { - $contentExists = TRUE; - } - elseif (substr($fileName, 0, 8) === 'content/') { - // This is a content file, check that the file type is allowed - if ($skipContent === FALSE && $this->h5pC->disableFileCheck !== TRUE && !preg_match($contentRegExp, $fileName)) { - $this->h5pF->setErrorMessage($this->h5pF->t('File "%filename" not allowed. Only files with the following extensions are allowed: %files-allowed.', array('%filename' => $fileStat['name'], '%files-allowed' => $contentWhitelist)), 'not-in-whitelist'); - $valid = FALSE; - } - } - elseif ($canInstall && strpos($fileName, '/') !== FALSE) { - // This is a library file, check that the file type is allowed - if ($this->h5pC->disableFileCheck !== TRUE && !preg_match($libraryRegExp, $fileName)) { - $this->h5pF->setErrorMessage($this->h5pF->t('File "%filename" not allowed. Only files with the following extensions are allowed: %files-allowed.', array('%filename' => $fileStat['name'], '%files-allowed' => $libraryWhitelist)), 'not-in-whitelist'); - $valid = FALSE; - } - - // Further library validation happens after the files are extracted - } - } - - if (!empty($this->h5pC->maxTotalSize) && $totalSize > $this->h5pC->maxTotalSize) { - // Error total size of the zip is too large - $this->h5pF->setErrorMessage($this->h5pF->t('The total size of the unpacked files exceeds the maximum size allowed. (%used > %max)', array('%used' => ($totalSize / 1048576) . ' MB', '%max' => ($this->h5pC->maxTotalSize / 1048576) . ' MB')), 'total-size-too-large'); - $valid = FALSE; - } - - if ($skipContent === FALSE) { - // Not skipping content, require two valid JSON files from the package - if (!$contentExists) { - $this->h5pF->setErrorMessage($this->h5pF->t('A valid content folder is missing'), 'invalid-content-folder'); - $valid = FALSE; - } - else { - $contentJsonData = $this->getJson($tmpPath, $zip, 'content/content.json'); // TODO: Is this case-senstivie? - if ($contentJsonData === NULL) { - return FALSE; // Breaking error when reading from the archive. - } - elseif ($contentJsonData === FALSE) { - $valid = FALSE; // Validation error when parsing JSON - } - } - - if (!$mainH5pExists) { - $this->h5pF->setErrorMessage($this->h5pF->t('A valid main h5p.json file is missing'), 'invalid-h5p-json-file'); - $valid = FALSE; - } - else { - $mainH5pData = $this->getJson($tmpPath, $zip, 'h5p.json', TRUE); - if ($mainH5pData === NULL) { - return FALSE; // Breaking error when reading from the archive. - } - elseif ($mainH5pData === FALSE) { - $valid = FALSE; // Validation error when parsing JSON - } - elseif (!$this->isValidH5pData($mainH5pData, 'h5p.json', $this->h5pRequired, $this->h5pOptional)) { - $this->h5pF->setErrorMessage($this->h5pF->t('The main h5p.json file is not valid'), 'invalid-h5p-json-file'); // Is this message a bit redundant? - $valid = FALSE; - } - } - } - - if (!$valid) { - // If something has failed during the initial checks of the package - // we will not unpack it or continue validation. - $zip->close(); - unlink($tmpPath); - return FALSE; - } - - // Extract the files from the package - for ($i = 0; $i < $zip->numFiles; $i++) { - $fileName = $zip->statIndex($i)['name']; - - if (preg_match('/(^[\._]|\/[\._]|\\\[\._])/', $fileName) !== 0) { - continue; // Skip any file or folder starting with a . or _ - } - - $isContentFile = (substr($fileName, 0, 8) === 'content/'); - $isFolder = (strpos($fileName, '/') !== FALSE); - - if ($skipContent !== FALSE && $isContentFile) { - continue; // Skipping any content files - } - - if (!($isContentFile || ($canInstall && $isFolder))) { - continue; // Not something we want to unpack - } - - // Get file stream - $fileStream = $zip->getStream($fileName); - if (!$fileStream) { - // This is a breaking error, there's no need to continue. (the rest of the files will fail as well) - $this->h5pF->setErrorMessage($this->h5pF->t('Unable to read file from the package: %fileName', array('%fileName' => $fileName)), 'unable-to-read-package-file'); - $zip->close(); - unlink($path); - H5PCore::deleteFileTree($tmpDir); - return FALSE; - } - - // Use file interface to allow overrides - $this->h5pC->fs->saveFileFromZip($tmpDir, $fileName, $fileStream); - - // Clean up - if (is_resource($fileStream)) { - fclose($fileStream); - } - } - - // We're done with the zip file, clean up the stuff - $zip->close(); - unlink($tmpPath); - - if ($canInstall) { - // Process and validate libraries using the unpacked library folders - $files = scandir($tmpDir); - foreach ($files as $file) { - $filePath = $tmpDir . '/' . $file; - - if ($file === '.' || $file === '..' || $file === 'content' || !is_dir($filePath)) { - continue; // Skip - } - - $libraryH5PData = $this->getLibraryData($file, $filePath, $tmpDir); - if ($libraryH5PData === FALSE) { - $valid = FALSE; - continue; // Failed, but continue validating the rest of the libraries - } - - // Library's directory name must be: - // - - // - or - - // - -. - // where machineName, majorVersion and minorVersion is read from library.json - if ($libraryH5PData['machineName'] !== $file && H5PCore::libraryToFolderName($libraryH5PData) !== $file) { - $this->h5pF->setErrorMessage($this->h5pF->t('Library directory name must match machineName or machineName-majorVersion.minorVersion (from library.json). (Directory: %directoryName , machineName: %machineName, majorVersion: %majorVersion, minorVersion: %minorVersion)', array( - '%directoryName' => $file, - '%machineName' => $libraryH5PData['machineName'], - '%majorVersion' => $libraryH5PData['majorVersion'], - '%minorVersion' => $libraryH5PData['minorVersion'])), 'library-directory-name-mismatch'); - $valid = FALSE; - continue; // Failed, but continue validating the rest of the libraries - } - - $libraryH5PData['uploadDirectory'] = $filePath; - $libraries[H5PCore::libraryToString($libraryH5PData)] = $libraryH5PData; - } - } - - if ($valid) { - if ($upgradeOnly) { - // When upgrading, we only add the already installed libraries, and - // the new dependent libraries - $upgrades = array(); - foreach ($libraries as $libString => &$library) { - // Is this library already installed? - if ($this->h5pF->getLibraryId($library['machineName']) !== FALSE) { - $upgrades[$libString] = $library; - } - } - while ($missingLibraries = $this->getMissingLibraries($upgrades)) { - foreach ($missingLibraries as $libString => $missing) { - $library = $libraries[$libString]; - if ($library) { - $upgrades[$libString] = $library; - } - } - } - - $libraries = $upgrades; - } - - $this->h5pC->librariesJsonData = $libraries; - - if ($skipContent === FALSE) { - $this->h5pC->mainJsonData = $mainH5pData; - $this->h5pC->contentJsonData = $contentJsonData; - $libraries['mainH5pData'] = $mainH5pData; // Check for the dependencies in h5p.json as well as in the libraries - } - - $missingLibraries = $this->getMissingLibraries($libraries); - foreach ($missingLibraries as $libString => $missing) { - if ($this->h5pC->getLibraryId($missing, $libString)) { - unset($missingLibraries[$libString]); - } - } - - if (!empty($missingLibraries)) { - // We still have missing libraries, check if our main library has an upgrade (BUT only if we has content) - $mainDependency = NULL; - if (!$skipContent && !empty($mainH5pData)) { - foreach ($mainH5pData['preloadedDependencies'] as $dep) { - if ($dep['machineName'] === $mainH5pData['mainLibrary']) { - $mainDependency = $dep; - } - } - } - - if ($skipContent || !$mainDependency || !$this->h5pF->libraryHasUpgrade(array( - 'machineName' => $mainDependency['machineName'], - 'majorVersion' => $mainDependency['majorVersion'], - 'minorVersion' => $mainDependency['minorVersion'] - ))) { - foreach ($missingLibraries as $libString => $library) { - if (!empty($mainDependency) && $library['machineName'] === $mainDependency['machineName']) { - $this->h5pF->setErrorMessage($this->h5pF->t('Missing main library @library', array('@library' => $libString )), 'missing-main-library'); - } - else { - $this->h5pF->setErrorMessage($this->h5pF->t('Missing required library @library', array('@library' => $libString)), 'missing-required-library'); - } - $valid = FALSE; - } - if (!$this->h5pC->mayUpdateLibraries()) { - $this->h5pF->setInfoMessage($this->h5pF->t("Note that the libraries may exist in the file you uploaded, but you're not allowed to upload new libraries. Contact the site administrator about this.")); - $valid = FALSE; - } - } - } - } - if (!$valid) { - H5PCore::deleteFileTree($tmpDir); - } - return $valid; - } - - /** - * Help read JSON from the archive - * - * @param string $path - * @param ZipArchive $zip - * @param string $file - * @return mixed JSON content if valid, FALSE for invalid, NULL for breaking error. - */ - private function getJson($path, $zip, $file, $assoc = FALSE) { - // Get stream - $stream = $zip->getStream($file); - if (!$stream) { - // Breaking error, no need to continue validating. - $this->h5pF->setErrorMessage($this->h5pF->t('Unable to read file from the package: %fileName', array('%fileName' => $file)), 'unable-to-read-package-file'); - $zip->close(); - unlink($path); - return NULL; - } - - // Read data - $contents = ''; - while (!feof($stream)) { - $contents .= fread($stream, 2); - } - - // Decode the data - $json = json_decode($contents, $assoc); - if ($json === NULL) { - // JSON cannot be decoded or the recursion limit has been reached. - $this->h5pF->setErrorMessage($this->h5pF->t('Unable to parse JSON from the package: %fileName', array('%fileName' => $file)), 'unable-to-parse-package'); - return FALSE; - } - - // All OK - return $json; - } - - /** - * Help retrieve file type regexp whitelist from plugin. - * - * @param bool $isLibrary Separate list with more allowed file types - * @return string RegExp - */ - private function getWhitelistRegExp($isLibrary) { - $whitelist = $this->h5pF->getWhitelist($isLibrary, H5PCore::$defaultContentWhitelist, H5PCore::$defaultLibraryWhitelistExtras); - return array($whitelist, '/\.(' . preg_replace('/ +/i', '|', preg_quote($whitelist)) . ')$/i'); - } - - /** - * Validates a H5P library - * - * @param string $file - * Name of the library folder - * @param string $filePath - * Path to the library folder - * @param string $tmpDir - * Path to the temporary upload directory - * @return boolean|array - * H5P data from library.json and semantics if the library is valid - * FALSE if the library isn't valid - */ - public function getLibraryData($file, $filePath, $tmpDir) { - if (preg_match('/^[\w0-9\-\.]{1,255}$/i', $file) === 0) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid library name: %name', array('%name' => $file)), 'invalid-library-name'); - return FALSE; - } - $h5pData = $this->getJsonData($filePath . '/' . 'library.json'); - if ($h5pData === FALSE) { - $this->h5pF->setErrorMessage($this->h5pF->t('Could not find library.json file with valid json format for library %name', array('%name' => $file)), 'invalid-library-json-file'); - return FALSE; - } - - // validate json if a semantics file is provided - $semanticsPath = $filePath . '/' . 'semantics.json'; - if (file_exists($semanticsPath)) { - $semantics = $this->getJsonData($semanticsPath, TRUE); - if ($semantics === FALSE) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid semantics.json file has been included in the library %name', array('%name' => $file)), 'invalid-semantics-json-file'); - return FALSE; - } - else { - $h5pData['semantics'] = $semantics; - } - } - - // validate language folder if it exists - $languagePath = $filePath . '/' . 'language'; - if (is_dir($languagePath)) { - $languageFiles = scandir($languagePath); - foreach ($languageFiles as $languageFile) { - if (in_array($languageFile, array('.', '..'))) { - continue; - } - if (preg_match('/^(-?[a-z]+){1,7}\.json$/i', $languageFile) === 0) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid language file %file in library %library', array('%file' => $languageFile, '%library' => $file)), 'invalid-language-file'); - return FALSE; - } - $languageJson = $this->getJsonData($languagePath . '/' . $languageFile, TRUE); - if ($languageJson === FALSE) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid language file %languageFile has been included in the library %name', array('%languageFile' => $languageFile, '%name' => $file)), 'invalid-language-file'); - return FALSE; - } - $parts = explode('.', $languageFile); // $parts[0] is the language code - $h5pData['language'][$parts[0]] = $languageJson; - } - } - - // Check for icon: - $h5pData['hasIcon'] = file_exists($filePath . '/' . 'icon.svg'); - - $validLibrary = $this->isValidH5pData($h5pData, $file, $this->libraryRequired, $this->libraryOptional); - - //$validLibrary = $this->h5pCV->validateContentFiles($filePath, TRUE) && $validLibrary; - - if (isset($h5pData['preloadedJs'])) { - $validLibrary = $this->isExistingFiles($h5pData['preloadedJs'], $tmpDir, $file) && $validLibrary; - } - if (isset($h5pData['preloadedCss'])) { - $validLibrary = $this->isExistingFiles($h5pData['preloadedCss'], $tmpDir, $file) && $validLibrary; - } - if ($validLibrary) { - return $h5pData; - } - else { - return FALSE; - } - } - - /** - * Use the dependency declarations to find any missing libraries - * - * @param array $libraries - * A multidimensional array of libraries keyed with machineName first and majorVersion second - * @return array - * A list of libraries that are missing keyed with machineName and holds objects with - * machineName, majorVersion and minorVersion properties - */ - private function getMissingLibraries($libraries) { - $missing = array(); - foreach ($libraries as $library) { - if (isset($library['preloadedDependencies'])) { - $missing = array_merge($missing, $this->getMissingDependencies($library['preloadedDependencies'], $libraries)); - } - if (isset($library['dynamicDependencies'])) { - $missing = array_merge($missing, $this->getMissingDependencies($library['dynamicDependencies'], $libraries)); - } - if (isset($library['editorDependencies'])) { - $missing = array_merge($missing, $this->getMissingDependencies($library['editorDependencies'], $libraries)); - } - } - return $missing; - } - - /** - * Helper function for getMissingLibraries, searches for dependency required libraries in - * the provided list of libraries - * - * @param array $dependencies - * A list of objects with machineName, majorVersion and minorVersion properties - * @param array $libraries - * An array of libraries keyed with machineName - * @return - * A list of libraries that are missing keyed with machineName and holds objects with - * machineName, majorVersion and minorVersion properties - */ - private function getMissingDependencies($dependencies, $libraries) { - $missing = array(); - foreach ($dependencies as $dependency) { - $libString = H5PCore::libraryToString($dependency); - if (!isset($libraries[$libString])) { - $missing[$libString] = $dependency; - } - } - return $missing; - } - - /** - * Figure out if the provided file paths exists - * - * Triggers error messages if files doesn't exist - * - * @param array $files - * List of file paths relative to $tmpDir - * @param string $tmpDir - * Path to the directory where the $files are stored. - * @param string $library - * Name of the library we are processing - * @return boolean - * TRUE if all the files excists - */ - private function isExistingFiles($files, $tmpDir, $library) { - foreach ($files as $file) { - $path = str_replace(array('/', '\\'), '/', $file['path']); - if (!file_exists($tmpDir . '/' . $library . '/' . $path)) { - $this->h5pF->setErrorMessage($this->h5pF->t('The file "%file" is missing from library: "%name"', array('%file' => $path, '%name' => $library)), 'library-missing-file'); - return FALSE; - } - } - return TRUE; - } - - /** - * Validates h5p.json and library.json data - * - * Error messages are triggered if the data isn't valid - * - * @param array $h5pData - * h5p data - * @param string $library_name - * Name of the library we are processing - * @param array $required - * Validation pattern for required properties - * @param array $optional - * Validation pattern for optional properties - * @return boolean - * TRUE if the $h5pData is valid - */ - private function isValidH5pData($h5pData, $library_name, $required, $optional) { - $valid = $this->isValidRequiredH5pData($h5pData, $required, $library_name); - $valid = $this->isValidOptionalH5pData($h5pData, $optional, $library_name) && $valid; - - // Check the library's required API version of Core. - // If no requirement is set this implicitly means 1.0. - if (isset($h5pData['coreApi']) && !empty($h5pData['coreApi'])) { - if (($h5pData['coreApi']['majorVersion'] > H5PCore::$coreApi['majorVersion']) || - ( ($h5pData['coreApi']['majorVersion'] == H5PCore::$coreApi['majorVersion']) && - ($h5pData['coreApi']['minorVersion'] > H5PCore::$coreApi['minorVersion']) )) { - - $this->h5pF->setErrorMessage( - $this->h5pF->t('The system was unable to install the %component component from the package, it requires a newer version of the H5P plugin. This site is currently running version %current, whereas the required version is %required or higher. You should consider upgrading and then try again.', - array( - '%component' => (isset($h5pData['title']) ? $h5pData['title'] : $library_name), - '%current' => H5PCore::$coreApi['majorVersion'] . '.' . H5PCore::$coreApi['minorVersion'], - '%required' => $h5pData['coreApi']['majorVersion'] . '.' . $h5pData['coreApi']['minorVersion'] - ) - ), - 'api-version-unsupported' - ); - - $valid = false; - } - } - - return $valid; - } - - /** - * Helper function for isValidH5pData - * - * Validates the optional part of the h5pData - * - * Triggers error messages - * - * @param array $h5pData - * h5p data - * @param array $requirements - * Validation pattern - * @param string $library_name - * Name of the library we are processing - * @return boolean - * TRUE if the optional part of the $h5pData is valid - */ - private function isValidOptionalH5pData($h5pData, $requirements, $library_name) { - $valid = TRUE; - - foreach ($h5pData as $key => $value) { - if (isset($requirements[$key])) { - $valid = $this->isValidRequirement($value, $requirements[$key], $library_name, $key) && $valid; - } - // Else: ignore, a package can have parameters that this library doesn't care about, but that library - // specific implementations does care about... - } - - return $valid; - } - - /** - * Validate a requirement given as regexp or an array of requirements - * - * @param mixed $h5pData - * The data to be validated - * @param mixed $requirement - * The requirement the data is to be validated against, regexp or array of requirements - * @param string $library_name - * Name of the library we are validating(used in error messages) - * @param string $property_name - * Name of the property we are validating(used in error messages) - * @return boolean - * TRUE if valid, FALSE if invalid - */ - private function isValidRequirement($h5pData, $requirement, $library_name, $property_name) { - $valid = TRUE; - - if (is_string($requirement)) { - if ($requirement == 'boolean') { - if (!is_bool($h5pData)) { - $this->h5pF->setErrorMessage($this->h5pF->t("Invalid data provided for %property in %library. Boolean expected.", array('%property' => $property_name, '%library' => $library_name))); - $valid = FALSE; - } - } - else { - // The requirement is a regexp, match it against the data - if (is_string($h5pData) || is_int($h5pData)) { - if (preg_match($requirement, $h5pData) === 0) { - $this->h5pF->setErrorMessage($this->h5pF->t("Invalid data provided for %property in %library", array('%property' => $property_name, '%library' => $library_name))); - $valid = FALSE; - } - } - else { - $this->h5pF->setErrorMessage($this->h5pF->t("Invalid data provided for %property in %library", array('%property' => $property_name, '%library' => $library_name))); - $valid = FALSE; - } - } - } - elseif (is_array($requirement)) { - // We have sub requirements - if (is_array($h5pData)) { - if (is_array(current($h5pData))) { - foreach ($h5pData as $sub_h5pData) { - $valid = $this->isValidRequiredH5pData($sub_h5pData, $requirement, $library_name) && $valid; - } - } - else { - $valid = $this->isValidRequiredH5pData($h5pData, $requirement, $library_name) && $valid; - } - } - else { - $this->h5pF->setErrorMessage($this->h5pF->t("Invalid data provided for %property in %library", array('%property' => $property_name, '%library' => $library_name))); - $valid = FALSE; - } - } - else { - $this->h5pF->setErrorMessage($this->h5pF->t("Can't read the property %property in %library", array('%property' => $property_name, '%library' => $library_name))); - $valid = FALSE; - } - return $valid; - } - - /** - * Validates the required h5p data in libraray.json and h5p.json - * - * @param mixed $h5pData - * Data to be validated - * @param array $requirements - * Array with regexp to validate the data against - * @param string $library_name - * Name of the library we are validating (used in error messages) - * @return boolean - * TRUE if all the required data exists and is valid, FALSE otherwise - */ - private function isValidRequiredH5pData($h5pData, $requirements, $library_name) { - $valid = TRUE; - foreach ($requirements as $required => $requirement) { - if (is_int($required)) { - // We have an array of allowed options - return $this->isValidH5pDataOptions($h5pData, $requirements, $library_name); - } - if (isset($h5pData[$required])) { - $valid = $this->isValidRequirement($h5pData[$required], $requirement, $library_name, $required) && $valid; - } - else { - $this->h5pF->setErrorMessage($this->h5pF->t('The required property %property is missing from %library', array('%property' => $required, '%library' => $library_name)), 'missing-required-property'); - $valid = FALSE; - } - } - return $valid; - } - - /** - * Validates h5p data against a set of allowed values(options) - * - * @param array $selected - * The option(s) that has been specified - * @param array $allowed - * The allowed options - * @param string $library_name - * Name of the library we are validating (used in error messages) - * @return boolean - * TRUE if the specified data is valid, FALSE otherwise - */ - private function isValidH5pDataOptions($selected, $allowed, $library_name) { - $valid = TRUE; - foreach ($selected as $value) { - if (!in_array($value, $allowed)) { - $this->h5pF->setErrorMessage($this->h5pF->t('Illegal option %option in %library', array('%option' => $value, '%library' => $library_name)), 'illegal-option-in-library'); - $valid = FALSE; - } - } - return $valid; - } - - /** - * Fetch json data from file - * - * @param string $filePath - * Path to the file holding the json string - * @param boolean $return_as_string - * If true the json data will be decoded in order to validate it, but will be - * returned as string - * @return mixed - * FALSE if the file can't be read or the contents can't be decoded - * string if the $return as string parameter is set - * array otherwise - */ - private function getJsonData($filePath, $return_as_string = FALSE) { - $json = file_get_contents($filePath); - if ($json === FALSE) { - return FALSE; // Cannot read from file. - } - $jsonData = json_decode($json, TRUE); - if ($jsonData === NULL) { - return FALSE; // JSON cannot be decoded or the recursion limit has been reached. - } - return $return_as_string ? $json : $jsonData; - } - - /** - * Helper function that copies an array - * - * @param array $array - * The array to be copied - * @return array - * Copy of $array. All objects are cloned - */ - private function arrayCopy(array $array) { - $result = array(); - foreach ($array as $key => $val) { - if (is_array($val)) { - $result[$key] = self::arrayCopy($val); - } - elseif (is_object($val)) { - $result[$key] = clone $val; - } - else { - $result[$key] = $val; - } - } - return $result; - } -} - -/** - * This class is used for saving H5P files - */ -class H5PStorage { - - public $h5pF; - public $h5pC; - - public $contentId = NULL; // Quick fix so WP can get ID of new content. - - /** - * Constructor for the H5PStorage - * - * @param H5PFrameworkInterface|object $H5PFramework - * The frameworks implementation of the H5PFrameworkInterface - * @param H5PCore $H5PCore - */ - public function __construct(H5PFrameworkInterface $H5PFramework, H5PCore $H5PCore) { - $this->h5pF = $H5PFramework; - $this->h5pC = $H5PCore; - } - - /** - * Saves a H5P file - * - * @param null $content - * @param int $contentMainId - * The main id for the content we are saving. This is used if the framework - * we're integrating with uses content id's and version id's - * @param bool $skipContent - * @param array $options - * @return bool TRUE if one or more libraries were updated - * TRUE if one or more libraries were updated - * FALSE otherwise - */ - public function savePackage($content = NULL, $contentMainId = NULL, $skipContent = FALSE, $options = array()) { - if ($this->h5pC->mayUpdateLibraries()) { - // Save the libraries we processed during validation - $this->saveLibraries(); - } - - if (!$skipContent) { - $basePath = $this->h5pF->getUploadedH5pFolderPath(); - $current_path = $basePath . '/' . 'content'; - - // Save content - if ($content === NULL) { - $content = array(); - } - if (!is_array($content)) { - $content = array('id' => $content); - } - - // Find main library version - foreach ($this->h5pC->mainJsonData['preloadedDependencies'] as $dep) { - if ($dep['machineName'] === $this->h5pC->mainJsonData['mainLibrary']) { - $dep['libraryId'] = $this->h5pC->getLibraryId($dep); - $content['library'] = $dep; - break; - } - } - - $content['params'] = file_get_contents($current_path . '/' . 'content.json'); - - if (isset($options['disable'])) { - $content['disable'] = $options['disable']; - } - $content['id'] = $this->h5pC->saveContent($content, $contentMainId); - $this->contentId = $content['id']; - - try { - // Save content folder contents - $this->h5pC->fs->saveContent($current_path, $content); - } - catch (Exception $e) { - $this->h5pF->setErrorMessage($e->getMessage(), 'save-content-failed'); - } - - // Remove temp content folder - H5PCore::deleteFileTree($basePath); - } - } - - /** - * Helps savePackage. - * - * @return int Number of libraries saved - */ - private function saveLibraries() { - // Keep track of the number of libraries that have been saved - $newOnes = 0; - $oldOnes = 0; - - // Go through libraries that came with this package - foreach ($this->h5pC->librariesJsonData as $libString => &$library) { - // Find local library with same major + minor - $existingLibrary = $this->h5pC->loadLibrary($library['machineName'], $library['majorVersion'], $library['minorVersion']); - - // Assume new library - $new = TRUE; - if (isset($existingLibrary['libraryId'])) { - $new = false; - // We have the library installed already (with the same major + minor) - - $library['libraryId'] = $existingLibrary['libraryId']; - - // Is this a newer patchVersion? - $newerPatchVersion = $existingLibrary['patchVersion'] < $library['patchVersion']; - - if (!$newerPatchVersion) { - $library['saveDependencies'] = FALSE; - // This is an older version, no need to save. - continue; - } - } - - // Indicate that the dependencies of this library should be saved. - $library['saveDependencies'] = TRUE; - - // Convert metadataSettings values to boolean & json_encode it before saving - $library['metadataSettings'] = isset($library['metadataSettings']) ? - H5PMetadata::boolifyAndEncodeSettings($library['metadataSettings']) : - NULL; - - // MOODLE PATCH: The library needs to be saved in database first before creating the files, because the libraryid is used - // as itemid for the files. - // Update our DB - $this->h5pF->saveLibraryData($library, $new); - - // Save library folder - $this->h5pC->fs->saveLibrary($library); - - // Remove cached assets that uses this library - if ($this->h5pC->aggregateAssets && isset($library['libraryId'])) { - $removedKeys = $this->h5pF->deleteCachedAssets($library['libraryId']); - $this->h5pC->fs->deleteCachedAssets($removedKeys); - } - - // Remove tmp folder - H5PCore::deleteFileTree($library['uploadDirectory']); - - if ($existingLibrary) { - $this->h5pC->fs->deleteLibrary($existingLibrary); - } - - if ($new) { - $newOnes++; - } - else { - $oldOnes++; - } - } - - // Go through the libraries again to save dependencies. - $library_ids = array(); - foreach ($this->h5pC->librariesJsonData as &$library) { - if (!$library['saveDependencies']) { - continue; - } - - // TODO: Should the table be locked for this operation? - - // Remove any old dependencies - $this->h5pF->deleteLibraryDependencies($library['libraryId']); - - // Insert the different new ones - if (isset($library['preloadedDependencies'])) { - $this->h5pF->saveLibraryDependencies($library['libraryId'], $library['preloadedDependencies'], 'preloaded'); - } - if (isset($library['dynamicDependencies'])) { - $this->h5pF->saveLibraryDependencies($library['libraryId'], $library['dynamicDependencies'], 'dynamic'); - } - if (isset($library['editorDependencies'])) { - $this->h5pF->saveLibraryDependencies($library['libraryId'], $library['editorDependencies'], 'editor'); - } - - $library_ids[] = $library['libraryId']; - } - - // Make sure libraries dependencies, parameter filtering and export files gets regenerated for all content who uses these libraries. - if (!empty($library_ids)) { - $this->h5pF->clearFilteredParameters($library_ids); - } - - // Tell the user what we've done. - if ($newOnes && $oldOnes) { - if ($newOnes === 1) { - if ($oldOnes === 1) { - // Singular Singular - $message = $this->h5pF->t('Added %new new H5P library and updated %old old one.', array('%new' => $newOnes, '%old' => $oldOnes)); - } - else { - // Singular Plural - $message = $this->h5pF->t('Added %new new H5P library and updated %old old ones.', array('%new' => $newOnes, '%old' => $oldOnes)); - } - } - else { - // Plural - if ($oldOnes === 1) { - // Plural Singular - $message = $this->h5pF->t('Added %new new H5P libraries and updated %old old one.', array('%new' => $newOnes, '%old' => $oldOnes)); - } - else { - // Plural Plural - $message = $this->h5pF->t('Added %new new H5P libraries and updated %old old ones.', array('%new' => $newOnes, '%old' => $oldOnes)); - } - } - } - elseif ($newOnes) { - if ($newOnes === 1) { - // Singular - $message = $this->h5pF->t('Added %new new H5P library.', array('%new' => $newOnes)); - } - else { - // Plural - $message = $this->h5pF->t('Added %new new H5P libraries.', array('%new' => $newOnes)); - } - } - elseif ($oldOnes) { - if ($oldOnes === 1) { - // Singular - $message = $this->h5pF->t('Updated %old H5P library.', array('%old' => $oldOnes)); - } - else { - // Plural - $message = $this->h5pF->t('Updated %old H5P libraries.', array('%old' => $oldOnes)); - } - } - - if (isset($message)) { - $this->h5pF->setInfoMessage($message); - } - } - - /** - * Delete an H5P package - * - * @param $content - */ - public function deletePackage($content) { - $this->h5pC->fs->deleteContent($content); - $this->h5pC->fs->deleteExport(($content['slug'] ? $content['slug'] . '-' : '') . $content['id'] . '.h5p'); - $this->h5pF->deleteContentData($content['id']); - } - - /** - * Copy/clone an H5P package - * - * May for instance be used if the content is being revisioned without - * uploading a new H5P package - * - * @param int $contentId - * The new content id - * @param int $copyFromId - * The content id of the content that should be cloned - * @param int $contentMainId - * The main id of the new content (used in frameworks that support revisioning) - */ - public function copyPackage($contentId, $copyFromId, $contentMainId = NULL) { - $this->h5pC->fs->cloneContent($copyFromId, $contentId); - $this->h5pF->copyLibraryUsage($contentId, $copyFromId, $contentMainId); - } -} - -/** -* This class is used for exporting zips -*/ -Class H5PExport { - public $h5pF; - public $h5pC; - - /** - * Constructor for the H5PExport - * - * @param H5PFrameworkInterface|object $H5PFramework - * The frameworks implementation of the H5PFrameworkInterface - * @param H5PCore $H5PCore - * Reference to an instance of H5PCore - */ - public function __construct(H5PFrameworkInterface $H5PFramework, H5PCore $H5PCore) { - $this->h5pF = $H5PFramework; - $this->h5pC = $H5PCore; - } - - /** - * Reverts the replace pattern used by the text editor - * - * @param string $value - * @return string - */ - private static function revertH5PEditorTextEscape($value) { - return str_replace('<', '<', str_replace('>', '>', str_replace(''', "'", str_replace('"', '"', $value)))); - } - - /** - * Return path to h5p package. - * - * Creates package if not already created - * - * @param array $content - * @return string - */ - public function createExportFile($content) { - - // Get path to temporary folder, where export will be contained - $tmpPath = $this->h5pC->fs->getTmpPath(); - mkdir($tmpPath, 0777, true); - - try { - // Create content folder and populate with files - $this->h5pC->fs->exportContent($content['id'], "{$tmpPath}/content"); - } - catch (Exception $e) { - $this->h5pF->setErrorMessage($this->h5pF->t($e->getMessage()), 'failed-creating-export-file'); - H5PCore::deleteFileTree($tmpPath); - return FALSE; - } - - // Update content.json with content from database - file_put_contents("{$tmpPath}/content/content.json", $content['filtered']); - - // Make embedType into an array - $embedTypes = explode(', ', $content['embedType']); - - // Build h5p.json, the en-/de-coding will ensure proper escaping - $h5pJson = array ( - 'title' => self::revertH5PEditorTextEscape($content['title']), - 'language' => (isset($content['language']) && strlen(trim($content['language'])) !== 0) ? $content['language'] : 'und', - 'mainLibrary' => $content['library']['name'], - 'embedTypes' => $embedTypes - ); - - foreach(array('authors', 'source', 'license', 'licenseVersion', 'licenseExtras' ,'yearFrom', 'yearTo', 'changes', 'authorComments', 'defaultLanguage') as $field) { - if (isset($content['metadata'][$field]) && $content['metadata'][$field] !== '') { - if (($field !== 'authors' && $field !== 'changes') || (!empty($content['metadata'][$field]))) { - $h5pJson[$field] = json_decode(json_encode($content['metadata'][$field], TRUE)); - } - } - } - - // Remove all values that are not set - foreach ($h5pJson as $key => $value) { - if (!isset($value)) { - unset($h5pJson[$key]); - } - } - - // Add dependencies to h5p - foreach ($content['dependencies'] as $dependency) { - $library = $dependency['library']; - - try { - $exportFolder = NULL; - - // Determine path of export library - if (isset($this->h5pC) && isset($this->h5pC->h5pD)) { - - // Tries to find library in development folder - $isDevLibrary = $this->h5pC->h5pD->getLibrary( - $library['machineName'], - $library['majorVersion'], - $library['minorVersion'] - ); - - if ($isDevLibrary !== NULL && isset($library['path'])) { - $exportFolder = "/" . $library['path']; - } - } - - // Export required libraries - $this->h5pC->fs->exportLibrary($library, $tmpPath, $exportFolder); - } - catch (Exception $e) { - $this->h5pF->setErrorMessage($this->h5pF->t($e->getMessage()), 'failed-creating-export-file'); - H5PCore::deleteFileTree($tmpPath); - return FALSE; - } - - // Do not add editor dependencies to h5p json. - if ($dependency['type'] === 'editor') { - continue; - } - - // Add to h5p.json dependencies - $h5pJson[$dependency['type'] . 'Dependencies'][] = array( - 'machineName' => $library['machineName'], - 'majorVersion' => $library['majorVersion'], - 'minorVersion' => $library['minorVersion'] - ); - } - - // Save h5p.json - $results = print_r(json_encode($h5pJson), true); - file_put_contents("{$tmpPath}/h5p.json", $results); - - // Get a complete file list from our tmp dir - $files = array(); - self::populateFileList($tmpPath, $files); - - // Get path to temporary export target file - $tmpFile = $this->h5pC->fs->getTmpPath(); - - // Create new zip instance. - $zip = new ZipArchive(); - $zip->open($tmpFile, ZipArchive::CREATE | ZipArchive::OVERWRITE); - - // Add all the files from the tmp dir. - foreach ($files as $file) { - // Please note that the zip format has no concept of folders, we must - // use forward slashes to separate our directories. - if (file_exists(realpath($file->absolutePath))) { - $zip->addFile(realpath($file->absolutePath), $file->relativePath); - } - } - - // Close zip and remove tmp dir - $zip->close(); - H5PCore::deleteFileTree($tmpPath); - - $filename = $content['slug'] . '-' . $content['id'] . '.h5p'; - try { - // Save export - $this->h5pC->fs->saveExport($tmpFile, $filename); - } - catch (Exception $e) { - $this->h5pF->setErrorMessage($this->h5pF->t($e->getMessage()), 'failed-creating-export-file'); - return false; - } - - unlink($tmpFile); - $this->h5pF->afterExportCreated($content, $filename); - - return true; - } - - /** - * Recursive function the will add the files of the given directory to the - * given files list. All files are objects with an absolute path and - * a relative path. The relative path is forward slashes only! Great for - * use in zip files and URLs. - * - * @param string $dir path - * @param array $files list - * @param string $relative prefix. Optional - */ - private static function populateFileList($dir, &$files, $relative = '') { - $strip = strlen($dir) + 1; - $contents = glob($dir . '/' . '*'); - if (!empty($contents)) { - foreach ($contents as $file) { - $rel = $relative . substr($file, $strip); - if (is_dir($file)) { - self::populateFileList($file, $files, $rel . '/'); - } - else { - $files[] = (object) array( - 'absolutePath' => $file, - 'relativePath' => $rel - ); - } - } - } - } - - /** - * Delete .h5p file - * - * @param array $content object - */ - public function deleteExport($content) { - $this->h5pC->fs->deleteExport(($content['slug'] ? $content['slug'] . '-' : '') . $content['id'] . '.h5p'); - } - - /** - * Add editor libraries to the list of libraries - * - * These are not supposed to go into h5p.json, but must be included with the rest - * of the libraries - * - * TODO This is a private function that is not currently being used - * - * @param array $libraries - * List of libraries keyed by machineName - * @param array $editorLibraries - * List of libraries keyed by machineName - * @return array List of libraries keyed by machineName - */ - private function addEditorLibraries($libraries, $editorLibraries) { - foreach ($editorLibraries as $editorLibrary) { - $libraries[$editorLibrary['machineName']] = $editorLibrary; - } - return $libraries; - } -} - -abstract class H5PPermission { - const DOWNLOAD_H5P = 0; - const EMBED_H5P = 1; - const CREATE_RESTRICTED = 2; - const UPDATE_LIBRARIES = 3; - const INSTALL_RECOMMENDED = 4; - const COPY_H5P = 8; -} - -abstract class H5PDisplayOptionBehaviour { - const NEVER_SHOW = 0; - const CONTROLLED_BY_AUTHOR_DEFAULT_ON = 1; - const CONTROLLED_BY_AUTHOR_DEFAULT_OFF = 2; - const ALWAYS_SHOW = 3; - const CONTROLLED_BY_PERMISSIONS = 4; -} - -abstract class H5PContentHubSyncStatus { - const NOT_SYNCED = 0; - const SYNCED = 1; - const WAITING = 2; - const FAILED = 3; -} - -abstract class H5PContentStatus { - const STATUS_UNPUBLISHED = 0; - const STATUS_DOWNLOADED = 1; - const STATUS_WAITING = 2; - const STATUS_FAILED_DOWNLOAD = 3; - const STATUS_FAILED_VALIDATION = 4; - const STATUS_SUSPENDED = 5; -} - -abstract class H5PHubEndpoints { - const CONTENT_TYPES = 'api.h5p.org/v1/content-types/'; - const SITES = 'api.h5p.org/v1/sites'; - const METADATA = 'hub-api.h5p.org/v1/metadata'; - const CONTENT = 'hub-api.h5p.org/v1/contents'; - const REGISTER = 'hub-api.h5p.org/v1/accounts'; - - public static function createURL($endpoint) { - $protocol = (extension_loaded('openssl') ? 'https' : 'http'); - return "{$protocol}://{$endpoint}"; - } -} - -/** - * Functions and storage shared by the other H5P classes - */ -class H5PCore { - - public static $coreApi = array( - 'majorVersion' => 1, - 'minorVersion' => 26 - ); - public static $styles = array( - 'styles/h5p.css', - 'styles/h5p-confirmation-dialog.css', - 'styles/h5p-core-button.css', - 'styles/h5p-tooltip.css', - ); - public static $scripts = array( - 'js/jquery.js', - 'js/h5p.js', - 'js/h5p-event-dispatcher.js', - 'js/h5p-x-api-event.js', - 'js/h5p-x-api.js', - 'js/h5p-content-type.js', - 'js/h5p-confirmation-dialog.js', - 'js/h5p-action-bar.js', - 'js/request-queue.js', - 'js/h5p-tooltip.js', - ); - public static $adminScripts = array( - 'js/jquery.js', - 'js/h5p-utils.js', - ); - - public static $defaultContentWhitelist = 'json png jpg jpeg gif bmp tif tiff svg eot ttf woff woff2 otf webm mp4 ogg mp3 m4a wav txt pdf rtf doc docx xls xlsx ppt pptx odt ods odp xml csv diff patch swf md textile vtt webvtt gltf glb'; - public static $defaultLibraryWhitelistExtras = 'js css'; - - public $librariesJsonData, $contentJsonData, $mainJsonData, $h5pF, $fs, $h5pD, $disableFileCheck; - const SECONDS_IN_WEEK = 604800; - - private $exportEnabled; - - // Disable flags - const DISABLE_NONE = 0; - const DISABLE_FRAME = 1; - const DISABLE_DOWNLOAD = 2; - const DISABLE_EMBED = 4; - const DISABLE_COPYRIGHT = 8; - const DISABLE_ABOUT = 16; - - const DISPLAY_OPTION_FRAME = 'frame'; - const DISPLAY_OPTION_DOWNLOAD = 'export'; - const DISPLAY_OPTION_EMBED = 'embed'; - const DISPLAY_OPTION_COPYRIGHT = 'copyright'; - const DISPLAY_OPTION_ABOUT = 'icon'; - const DISPLAY_OPTION_COPY = 'copy'; - - // Map flags to string - public static $disable = array( - self::DISABLE_FRAME => self::DISPLAY_OPTION_FRAME, - self::DISABLE_DOWNLOAD => self::DISPLAY_OPTION_DOWNLOAD, - self::DISABLE_EMBED => self::DISPLAY_OPTION_EMBED, - self::DISABLE_COPYRIGHT => self::DISPLAY_OPTION_COPYRIGHT - ); - - /** @var string */ - public $url; - - /** @var int evelopment mode. */ - public $development_mode; - - /** @var bool aggregated files for assets. */ - public $aggregateAssets; - - /** @var string full path of plugin. */ - protected $fullPluginPath; - - /** @var string regex for converting copied files paths. */ - public $relativePathRegExp; - - /** - * Constructor for the H5PCore - * - * @param H5PFrameworkInterface $H5PFramework - * The frameworks implementation of the H5PFrameworkInterface - * @param string|H5PFileStorage $path H5P file storage directory or class. - * @param string $url To file storage directory. - * @param string $language code. Defaults to english. - * @param boolean $export enabled? - */ - public function __construct(H5PFrameworkInterface $H5PFramework, $path, $url, $language = 'en', $export = FALSE) { - $this->h5pF = $H5PFramework; - - $this->fs = ($path instanceof H5PFileStorage ? $path : new H5PDefaultStorage($path)); - - $this->url = $url; - $this->exportEnabled = $export; - $this->development_mode = H5PDevelopment::MODE_NONE; - - $this->aggregateAssets = FALSE; // Off by default.. for now - - $this->detectSiteType(); - $this->fullPluginPath = preg_replace('/\/[^\/]+[\/]?$/', '' , dirname(__FILE__)); - - // Standard regex for converting copied files paths - $this->relativePathRegExp = '/^((\.\.\/){1,2})(.*content\/)?(\d+|editor)\/(.+)$/'; - } - - /** - * Save content and clear cache. - * - * @param array $content - * @param null|int $contentMainId - * @return int Content ID - */ - public function saveContent($content, $contentMainId = NULL) { - if (isset($content['id'])) { - $this->h5pF->updateContent($content, $contentMainId); - } - else { - $content['id'] = $this->h5pF->insertContent($content, $contentMainId); - } - - // Some user data for content has to be reset when the content changes. - $this->h5pF->resetContentUserData($contentMainId ? $contentMainId : $content['id']); - - return $content['id']; - } - - /** - * Load content. - * - * @param int $id for content. - * @return object - */ - public function loadContent($id) { - $content = $this->h5pF->loadContent($id); - - if ($content !== NULL) { - // Validate main content's metadata - $validator = new H5PContentValidator($this->h5pF, $this); - $content['metadata'] = $validator->validateMetadata($content['metadata']); - - $content['library'] = array( - 'id' => $content['libraryId'], - 'name' => $content['libraryName'], - 'majorVersion' => $content['libraryMajorVersion'], - 'minorVersion' => $content['libraryMinorVersion'], - 'embedTypes' => $content['libraryEmbedTypes'], - 'fullscreen' => $content['libraryFullscreen'], - ); - unset($content['libraryId'], $content['libraryName'], $content['libraryEmbedTypes'], $content['libraryFullscreen']); - -// // TODO: Move to filterParameters? -// if (isset($this->h5pD)) { -// // TODO: Remove Drupal specific stuff -// $json_content_path = file_create_path(file_directory_path() . '/' . variable_get('h5p_default_path', 'h5p') . '/content/' . $id . '/content.json'); -// if (file_exists($json_content_path) === TRUE) { -// $json_content = file_get_contents($json_content_path); -// if (json_decode($json_content, TRUE) !== FALSE) { -// drupal_set_message(t('Invalid json in json content'), 'warning'); -// } -// $content['params'] = $json_content; -// } -// } - } - - return $content; - } - - /** - * Filter content run parameters, rebuild content dependency cache and export file. - * - * @param Object|array $content - * @return Object NULL on failure. - */ - public function filterParameters(&$content) { - if (!empty($content['filtered']) && - (!$this->exportEnabled || - ($content['slug'] && - $this->fs->hasExport($content['slug'] . '-' . $content['id'] . '.h5p')))) { - return $content['filtered']; - } - - if (!(isset($content['library']) && isset($content['params']))) { - return NULL; - } - - // Validate and filter against main library semantics. - $validator = new H5PContentValidator($this->h5pF, $this); - $params = (object) array( - 'library' => H5PCore::libraryToString($content['library']), - 'params' => json_decode($content['params']) - ); - if (!$params->params) { - return NULL; - } - $validator->validateLibrary($params, (object) array('options' => array($params->library))); - - // Handle addons: - $addons = $this->h5pF->loadAddons(); - foreach ($addons as $addon) { - $add_to = json_decode($addon['addTo']); - - if (isset($add_to->content->types)) { - foreach($add_to->content->types as $type) { - - if (isset($type->text->regex) && - $this->textAddonMatches($params->params, $type->text->regex)) { - $validator->addon($addon); - - // An addon shall only be added once - break; - } - } - } - } - - $params = json_encode($params->params); - - // Update content dependencies. - $content['dependencies'] = $validator->getDependencies(); - - // Sometimes the parameters are filtered before content has been created - if ($content['id']) { - $this->h5pF->deleteLibraryUsage($content['id']); - $this->h5pF->saveLibraryUsage($content['id'], $content['dependencies']); - - if (!$content['slug']) { - $content['slug'] = $this->generateContentSlug($content); - - // Remove old export file - $this->fs->deleteExport($content['id'] . '.h5p'); - } - - if ($this->exportEnabled) { - // Recreate export file - $exporter = new H5PExport($this->h5pF, $this); - $content['filtered'] = $params; - $exporter->createExportFile($content); - } - - // Cache. - $this->h5pF->updateContentFields($content['id'], array( - 'filtered' => $params, - 'slug' => $content['slug'] - )); - } - return $params; - } - - /** - * Retrieve a value from a nested mixed array structure. - * - * @param Array $params Array to be looked in. - * @param String $path Supposed path to the value. - * @param String [$delimiter='.'] Property delimiter within the path. - * @return Object|NULL The object found or NULL. - */ - private function retrieveValue ($params, $path, $delimiter='.') { - $path = explode($delimiter, $path); - - // Property not found - if (!isset($params[$path[0]])) { - return NULL; - } - - $first = $params[$path[0]]; - - // End of path, done - if (sizeof($path) === 1) { - return $first; - } - - // We cannot go deeper - if (!is_array($first)) { - return NULL; - } - - // Regular Array - if (isset($first[0])) { - foreach($first as $number => $object) { - $found = $this->retrieveValue($object, implode($delimiter, array_slice($path, 1))); - if (isset($found)) { - return $found; - } - } - return NULL; - } - - // Associative Array - return $this->retrieveValue($first, implode('.', array_slice($path, 1))); - } - - /** - * Determine if params contain any match. - * - * @param {object} params - Parameters. - * @param {string} [pattern] - Regular expression to identify pattern. - * @param {boolean} [found] - Used for recursion. - * @return {boolean} True, if params matches pattern. - */ - private function textAddonMatches($params, $pattern, $found = false) { - $type = gettype($params); - if ($type === 'string') { - if (preg_match($pattern, $params) === 1) { - return true; - } - } - elseif ($type === 'array' || $type === 'object') { - foreach ($params as $value) { - $found = $this->textAddonMatches($value, $pattern, $found); - if ($found === true) { - return true; - } - } - } - return false; - } - - /** - * Generate content slug - * - * @param array $content object - * @return string unique content slug - */ - private function generateContentSlug($content) { - $slug = H5PCore::slugify($content['title']); - - $available = NULL; - while (!$available) { - if ($available === FALSE) { - // If not available, add number suffix. - $matches = array(); - if (preg_match('/(.+-)([0-9]+)$/', $slug, $matches)) { - $slug = $matches[1] . (intval($matches[2]) + 1); - } - else { - $slug .= '-2'; - } - } - $available = $this->h5pF->isContentSlugAvailable($slug); - } - - return $slug; - } - - /** - * Find the files required for this content to work. - * - * @param int $id for content. - * @param null $type - * @return array - */ - public function loadContentDependencies($id, $type = NULL) { - $dependencies = $this->h5pF->loadContentDependencies($id, $type); - - if (isset($this->h5pD)) { - $developmentLibraries = $this->h5pD->getLibraries(); - - foreach ($dependencies as $key => $dependency) { - $libraryString = H5PCore::libraryToString($dependency); - if (isset($developmentLibraries[$libraryString])) { - $developmentLibraries[$libraryString]['dependencyType'] = $dependencies[$key]['dependencyType']; - $dependencies[$key] = $developmentLibraries[$libraryString]; - } - } - } - - return $dependencies; - } - - /** - * Get all dependency assets of the given type - * - * @param array $dependency - * @param string $type - * @param array $assets - * @param string $prefix Optional. Make paths relative to another dir. - */ - private function getDependencyAssets($dependency, $type, &$assets, $prefix = '') { - // Check if dependency has any files of this type - if (empty($dependency[$type]) || $dependency[$type][0] === '') { - return; - } - - // Check if we should skip CSS. - if ($type === 'preloadedCss' && (isset($dependency['dropCss']) && $dependency['dropCss'] === '1')) { - return; - } - foreach ($dependency[$type] as $file) { - $assets[] = (object) array( - 'path' => $prefix . '/' . $dependency['path'] . '/' . trim(is_array($file) ? $file['path'] : $file), - 'version' => $dependency['version'] - ); - } - } - - /** - * Combines path with cache buster / version. - * - * @param array $assets - * @return array - */ - public function getAssetsUrls($assets) { - $urls = array(); - - foreach ($assets as $asset) { - $url = $asset->path; - - // Add URL prefix if not external - if (strpos($asset->path, '://') === FALSE) { - $url = $this->url . $url; - } - - // Add version/cache buster if set - if (isset($asset->version)) { - $url .= $asset->version; - } - - $urls[] = $url; - } - - return $urls; - } - - /** - * Return file paths for all dependencies files. - * - * @param array $dependencies - * @param string $prefix Optional. Make paths relative to another dir. - * @return array files. - */ - public function getDependenciesFiles($dependencies, $prefix = '') { - // Build files list for assets - $files = array( - 'scripts' => array(), - 'styles' => array() - ); - - $key = null; - - // Avoid caching empty files - if (empty($dependencies)) { - return $files; - } - - if ($this->aggregateAssets) { - // Get aggregated files for assets - $key = self::getDependenciesHash($dependencies); - - $cachedAssets = $this->fs->getCachedAssets($key); - if ($cachedAssets !== NULL) { - return array_merge($files, $cachedAssets); // Using cached assets - } - } - - // Using content dependencies - foreach ($dependencies as $dependency) { - if (isset($dependency['path']) === FALSE) { - $dependency['path'] = $this->getDependencyPath($dependency); - $dependency['preloadedJs'] = explode(',', $dependency['preloadedJs']); - $dependency['preloadedCss'] = explode(',', $dependency['preloadedCss']); - } - $dependency['version'] = "?ver={$dependency['majorVersion']}.{$dependency['minorVersion']}.{$dependency['patchVersion']}"; - $this->getDependencyAssets($dependency, 'preloadedJs', $files['scripts'], $prefix); - $this->getDependencyAssets($dependency, 'preloadedCss', $files['styles'], $prefix); - } - - if ($this->aggregateAssets) { - // Aggregate and store assets - $this->fs->cacheAssets($files, $key); - - // Keep track of which libraries have been cached in case they are updated - $this->h5pF->saveCachedAssets($key, $dependencies); - } - - return $files; - } - - /** - * Get the path to the dependency. - * - * @param array $dependency - * @return string - */ - protected function getDependencyPath(array $dependency) { - return 'libraries/' . H5PCore::libraryToFolderName($dependency); - } - - private static function getDependenciesHash(&$dependencies) { - // Build hash of dependencies - $toHash = array(); - - // Use unique identifier for each library version - foreach ($dependencies as $dep) { - $toHash[] = "{$dep['machineName']}-{$dep['majorVersion']}.{$dep['minorVersion']}.{$dep['patchVersion']}"; - } - - // Sort in case the same dependencies comes in a different order - sort($toHash); - - // Calculate hash sum - return hash('sha1', implode('', $toHash)); - } - - /** - * Load library semantics. - * - * @param $name - * @param $majorVersion - * @param $minorVersion - * @return string - */ - public function loadLibrarySemantics($name, $majorVersion, $minorVersion) { - $semantics = NULL; - if (isset($this->h5pD)) { - // Try to load from dev lib - $semantics = $this->h5pD->getSemantics($name, $majorVersion, $minorVersion); - } - - if ($semantics === NULL) { - // Try to load from DB. - $semantics = $this->h5pF->loadLibrarySemantics($name, $majorVersion, $minorVersion); - } - - if ($semantics !== NULL) { - $semantics = json_decode($semantics); - $this->h5pF->alterLibrarySemantics($semantics, $name, $majorVersion, $minorVersion); - } - - return $semantics; - } - - /** - * Load library. - * - * @param $name - * @param $majorVersion - * @param $minorVersion - * @return array or null. - */ - public function loadLibrary($name, $majorVersion, $minorVersion) { - $library = NULL; - if (isset($this->h5pD)) { - // Try to load from dev - $library = $this->h5pD->getLibrary($name, $majorVersion, $minorVersion); - if ($library !== NULL) { - $library['semantics'] = $this->h5pD->getSemantics($name, $majorVersion, $minorVersion); - } - } - - if ($library === NULL) { - // Try to load from DB. - $library = $this->h5pF->loadLibrary($name, $majorVersion, $minorVersion); - } - - return $library; - } - - /** - * Deletes a library - * - * @param stdClass $libraryId - */ - public function deleteLibrary($libraryId) { - $this->h5pF->deleteLibrary($libraryId); - } - - /** - * Recursive. Goes through the dependency tree for the given library and - * adds all the dependencies to the given array in a flat format. - * - * @param $dependencies - * @param array $library To find all dependencies for. - * @param int $nextWeight An integer determining the order of the libraries - * when they are loaded - * @param bool $editor Used internally to force all preloaded sub dependencies - * of an editor dependency to be editor dependencies. - * @return int - */ - public function findLibraryDependencies(&$dependencies, $library, $nextWeight = 1, $editor = FALSE) { - foreach (array('dynamic', 'preloaded', 'editor') as $type) { - $property = $type . 'Dependencies'; - if (!isset($library[$property])) { - continue; // Skip, no such dependencies. - } - - if ($type === 'preloaded' && $editor === TRUE) { - // All preloaded dependencies of an editor library is set to editor. - $type = 'editor'; - } - - foreach ($library[$property] as $dependency) { - $dependencyKey = $type . '-' . $dependency['machineName']; - if (isset($dependencies[$dependencyKey]) === TRUE) { - continue; // Skip, already have this. - } - - $dependencyLibrary = $this->loadLibrary($dependency['machineName'], $dependency['majorVersion'], $dependency['minorVersion']); - if ($dependencyLibrary) { - $dependencies[$dependencyKey] = array( - 'library' => $dependencyLibrary, - 'type' => $type - ); - $nextWeight = $this->findLibraryDependencies($dependencies, $dependencyLibrary, $nextWeight, $type === 'editor'); - $dependencies[$dependencyKey]['weight'] = $nextWeight++; - } - else { - // This site is missing a dependency! - $this->h5pF->setErrorMessage($this->h5pF->t('Missing dependency @dep required by @lib.', array('@dep' => H5PCore::libraryToString($dependency), '@lib' => H5PCore::libraryToString($library))), 'missing-library-dependency'); - } - } - } - return $nextWeight; - } - - /** - * Check if a library is of the version we're looking for - * - * Same version means that the majorVersion and minorVersion is the same - * - * @param array $library - * Data from library.json - * @param array $dependency - * Definition of what library we're looking for - * @return boolean - * TRUE if the library is the same version as the dependency - * FALSE otherwise - */ - public function isSameVersion($library, $dependency) { - if ($library['machineName'] != $dependency['machineName']) { - return FALSE; - } - if ($library['majorVersion'] != $dependency['majorVersion']) { - return FALSE; - } - if ($library['minorVersion'] != $dependency['minorVersion']) { - return FALSE; - } - return TRUE; - } - - /** - * Recursive function for removing directories. - * - * @param string $dir - * Path to the directory we'll be deleting - * @return boolean - * Indicates if the directory existed. - */ - public static function deleteFileTree($dir) { - if (!is_dir($dir)) { - return false; - } - if (is_link($dir)) { - // Do not traverse and delete linked content, simply unlink. - unlink($dir); - return; - } - $files = array_diff(scandir($dir), array('.','..')); - foreach ($files as $file) { - $filepath = "$dir/$file"; - // Note that links may resolve as directories - if (!is_dir($filepath) || is_link($filepath)) { - // Unlink files and links - unlink($filepath); - } - else { - // Traverse subdir and delete files - self::deleteFileTree($filepath); - } - } - return rmdir($dir); - } - - /** - * Writes library data as string on the form {machineName} {majorVersion}.{minorVersion} - * - * @param array $library - * With keys (machineName and/or name), majorVersion and minorVersion - * @return string - * On the form {machineName} {majorVersion}.{minorVersion} - */ - public static function libraryToString($library) { - $name = $library['machineName'] ?? $library['name']; - - return "{$name} {$library['majorVersion']}.{$library['minorVersion']}"; - } - - /** - * Get the name of a library's folder name - * - * @return string - */ - public static function libraryToFolderName($library) { - $name = $library['machineName'] ?? $library['name']; - $includePatchVersion = $library['patchVersionInFolderName'] ?? false; - - return "{$name}-{$library['majorVersion']}.{$library['minorVersion']}" . ($includePatchVersion ? ".{$library['patchVersion']}" : ''); - } - - /** - * Parses library data from a string on the form {machineName} {majorVersion}.{minorVersion} - * - * @param string $libraryString - * On the form {machineName} {majorVersion}.{minorVersion} - * @return array|FALSE - * With keys machineName, majorVersion and minorVersion. - * Returns FALSE only if string is not parsable in the normal library - * string formats "Lib.Name-x.y" or "Lib.Name x.y" - */ - public static function libraryFromString($libraryString) { - $re = '/^([\w0-9\-\.]{1,255})[\-\ ]([0-9]{1,5})\.([0-9]{1,5})$/i'; - $matches = array(); - $res = preg_match($re, $libraryString, $matches); - if ($res) { - return array( - 'machineName' => $matches[1], - 'majorVersion' => $matches[2], - 'minorVersion' => $matches[3] - ); - } - return FALSE; - } - - /** - * Determine the correct embed type to use. - * - * @param $contentEmbedType - * @param $libraryEmbedTypes - * @return string 'div' or 'iframe'. - */ - public static function determineEmbedType($contentEmbedType, $libraryEmbedTypes) { - // Detect content embed type - $embedType = strpos(strtolower($contentEmbedType), 'div') !== FALSE ? 'div' : 'iframe'; - - if ($libraryEmbedTypes !== NULL && $libraryEmbedTypes !== '') { - // Check that embed type is available for library - $embedTypes = strtolower($libraryEmbedTypes); - if (strpos($embedTypes, $embedType) === FALSE) { - // Not available, pick default. - $embedType = strpos($embedTypes, 'div') !== FALSE ? 'div' : 'iframe'; - } - } - - return $embedType; - } - - /** - * Get the absolute version for the library as a human readable string. - * - * @param object $library - * @return string - */ - public static function libraryVersion($library) { - return $library->major_version . '.' . $library->minor_version . '.' . $library->patch_version; - } - - /** - * Determine which versions content with the given library can be upgraded to. - * - * @param object $library - * @param array $versions - * @return array - */ - public function getUpgrades($library, $versions) { - $upgrades = array(); - - foreach ($versions as $upgrade) { - if ($upgrade->major_version > $library->major_version || $upgrade->major_version === $library->major_version && $upgrade->minor_version > $library->minor_version) { - $upgrades[$upgrade->id] = H5PCore::libraryVersion($upgrade); - } - } - - return $upgrades; - } - - /** - * Converts all the properties of the given object or array from - * snake_case to camelCase. Useful after fetching data from the database. - * - * Note that some databases does not support camelCase. - * - * @param mixed $arr input - * @param boolean $obj return object - * @return mixed object or array - */ - public static function snakeToCamel($arr, $obj = false) { - $newArr = array(); - - foreach ($arr as $key => $val) { - $next = -1; - while (($next = strpos($key, '_', $next + 1)) !== FALSE) { - $key = substr_replace($key, strtoupper($key[$next + 1]), $next, 2); - } - - $newArr[$key] = $val; - } - - return $obj ? (object) $newArr : $newArr; - } - - /** - * Detects if the site was accessed from localhost, - * through a local network or from the internet. - */ - public function detectSiteType() { - $type = $this->h5pF->getOption('site_type', 'local'); - - // Determine remote/visitor origin - if ($type === 'network' || - ($type === 'local' && - isset($_SERVER['REMOTE_ADDR']) && - !preg_match('/^localhost$|^127(?:\.[0-9]+){0,2}\.[0-9]+$|^(?:0*\:)*?:?0*1$/i', $_SERVER['REMOTE_ADDR']))) { - if (isset($_SERVER['REMOTE_ADDR']) && filter_var($_SERVER['REMOTE_ADDR'], FILTER_VALIDATE_IP, FILTER_FLAG_NO_PRIV_RANGE)) { - // Internet - $this->h5pF->setOption('site_type', 'internet'); - } - elseif ($type === 'local') { - // Local network - $this->h5pF->setOption('site_type', 'network'); - } - } - } - - /** - * Get a list of installed libraries, different minor versions will - * return separate entries. - * - * @return array - * A distinct array of installed libraries - */ - public function getLibrariesInstalled() { - $librariesInstalled = array(); - $libs = $this->h5pF->loadLibraries(); - - foreach($libs as $libName => $library) { - foreach($library as $libVersion) { - $librariesInstalled[$libName.' '.$libVersion->major_version.'.'.$libVersion->minor_version] = $libVersion->patch_version; - } - } - - return $librariesInstalled; - } - - /** - * Easy way to combine similar data sets. - * - * @param array $inputs Multiple arrays with data - * @return array - */ - public function combineArrayValues($inputs) { - $results = array(); - foreach ($inputs as $index => $values) { - foreach ($values as $key => $value) { - $results[$key][$index] = $value; - } - } - return $results; - } - - /** - * Communicate with H5P.org and get content type cache. Each platform - * implementation is responsible for invoking this, eg using cron - * - * @param bool $fetchingDisabled - * @param bool $onlyRegister Only register site with H5P.org - * - * @return bool|object Returns endpoint data if found, otherwise FALSE - */ - public function fetchLibrariesMetadata($fetchingDisabled = FALSE, $onlyRegister = false) { - // Gather data - $uuid = $this->h5pF->getOption('site_uuid', ''); - $platform = $this->h5pF->getPlatformInfo(); - $registrationData = array( - 'uuid' => $uuid, - 'platform_name' => $platform['name'], - 'platform_version' => $platform['version'], - 'h5p_version' => $platform['h5pVersion'], - 'disabled' => $fetchingDisabled ? 1 : 0, - 'local_id' => hash('crc32', $this->fullPluginPath), - 'type' => $this->h5pF->getOption('site_type', 'local'), - 'core_api_version' => H5PCore::$coreApi['majorVersion'] . '.' . - H5PCore::$coreApi['minorVersion'] - ); - - // Register site if it is not registered - if (empty($uuid)) { - $registration = $this->h5pF->fetchExternalData(H5PHubEndpoints::createURL(H5PHubEndpoints::SITES), $registrationData); - - // Failed retrieving uuid - if (!$registration) { - $errorMessage = $this->h5pF->t('Site could not be registered with the hub. Please contact your site administrator.'); - $this->h5pF->setErrorMessage($errorMessage); - $this->h5pF->setErrorMessage( - $this->h5pF->t('The H5P Hub has been disabled until this problem can be resolved. You may still upload libraries through the "H5P Libraries" page.'), - 'registration-failed-hub-disabled' - ); - return FALSE; - } - - // Successfully retrieved new uuid - $json = json_decode($registration); - $registrationData['uuid'] = $json->uuid; - $this->h5pF->setOption('site_uuid', $json->uuid); - $this->h5pF->setInfoMessage( - $this->h5pF->t('Your site was successfully registered with the H5P Hub.') - ); - $uuid = $json->uuid; - // TODO: Uncomment when key is once again available in H5P Settings -// $this->h5pF->setInfoMessage( -// $this->h5pF->t('You have been provided a unique key that identifies you with the Hub when receiving new updates. The key is available for viewing in the "H5P Settings" page.') -// ); - } - - if ($onlyRegister) { - return $uuid; - } - - if ($this->h5pF->getOption('send_usage_statistics', TRUE)) { - $siteData = array_merge( - $registrationData, - array( - 'num_authors' => $this->h5pF->getNumAuthors(), - 'libraries' => json_encode($this->combineArrayValues(array( - 'patch' => $this->getLibrariesInstalled(), - 'content' => $this->h5pF->getLibraryContentCount(), - 'loaded' => $this->h5pF->getLibraryStats('library'), - 'created' => $this->h5pF->getLibraryStats('content create'), - 'createdUpload' => $this->h5pF->getLibraryStats('content create upload'), - 'deleted' => $this->h5pF->getLibraryStats('content delete'), - 'resultViews' => $this->h5pF->getLibraryStats('results content'), - 'shortcodeInserts' => $this->h5pF->getLibraryStats('content shortcode insert') - ))) - ) - ); - } - else { - $siteData = $registrationData; - } - - $result = $this->updateContentTypeCache($siteData); - - // No data received - if (!$result || empty($result)) { - return FALSE; - } - - // Handle libraries metadata - if (isset($result->libraries)) { - foreach ($result->libraries as $library) { - if (isset($library->tutorialUrl) && isset($library->machineName)) { - $this->h5pF->setLibraryTutorialUrl($library->machineNamee, $library->tutorialUrl); - } - } - } - - return $result; - } - - /** - * Create representation of display options as int - * - * @param array $sources - * @param int $current - * @return int - */ - public function getStorableDisplayOptions(&$sources, $current) { - // Download - force setting it if always on or always off - $download = $this->h5pF->getOption(self::DISPLAY_OPTION_DOWNLOAD, H5PDisplayOptionBehaviour::ALWAYS_SHOW); - if ($download == H5PDisplayOptionBehaviour::ALWAYS_SHOW || - $download == H5PDisplayOptionBehaviour::NEVER_SHOW) { - $sources[self::DISPLAY_OPTION_DOWNLOAD] = ($download == H5PDisplayOptionBehaviour::ALWAYS_SHOW); - } - - // Embed - force setting it if always on or always off - $embed = $this->h5pF->getOption(self::DISPLAY_OPTION_EMBED, H5PDisplayOptionBehaviour::ALWAYS_SHOW); - if ($embed == H5PDisplayOptionBehaviour::ALWAYS_SHOW || - $embed == H5PDisplayOptionBehaviour::NEVER_SHOW) { - $sources[self::DISPLAY_OPTION_EMBED] = ($embed == H5PDisplayOptionBehaviour::ALWAYS_SHOW); - } - - foreach (H5PCore::$disable as $bit => $option) { - if (!isset($sources[$option]) || !$sources[$option]) { - $current |= $bit; // Disable - } - else { - $current &= ~$bit; // Enable - } - } - return $current; - } - - /** - * Determine display options visibility and value on edit - * - * @param int $disable - * @return array - */ - public function getDisplayOptionsForEdit($disable = NULL) { - $display_options = array(); - - $current_display_options = $disable === NULL ? array() : $this->getDisplayOptionsAsArray($disable); - - if ($this->h5pF->getOption(self::DISPLAY_OPTION_FRAME, TRUE)) { - $display_options[self::DISPLAY_OPTION_FRAME] = - isset($current_display_options[self::DISPLAY_OPTION_FRAME]) ? - $current_display_options[self::DISPLAY_OPTION_FRAME] : - TRUE; - - // Download - $export = $this->h5pF->getOption(self::DISPLAY_OPTION_DOWNLOAD, H5PDisplayOptionBehaviour::ALWAYS_SHOW); - if ($export == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_ON || - $export == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_OFF) { - $display_options[self::DISPLAY_OPTION_DOWNLOAD] = - isset($current_display_options[self::DISPLAY_OPTION_DOWNLOAD]) ? - $current_display_options[self::DISPLAY_OPTION_DOWNLOAD] : - ($export == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_ON); - } - - // Embed - $embed = $this->h5pF->getOption(self::DISPLAY_OPTION_EMBED, H5PDisplayOptionBehaviour::ALWAYS_SHOW); - if ($embed == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_ON || - $embed == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_OFF) { - $display_options[self::DISPLAY_OPTION_EMBED] = - isset($current_display_options[self::DISPLAY_OPTION_EMBED]) ? - $current_display_options[self::DISPLAY_OPTION_EMBED] : - ($embed == H5PDisplayOptionBehaviour::CONTROLLED_BY_AUTHOR_DEFAULT_ON); - } - - // Copyright - if ($this->h5pF->getOption(self::DISPLAY_OPTION_COPYRIGHT, TRUE)) { - $display_options[self::DISPLAY_OPTION_COPYRIGHT] = - isset($current_display_options[self::DISPLAY_OPTION_COPYRIGHT]) ? - $current_display_options[self::DISPLAY_OPTION_COPYRIGHT] : - TRUE; - } - } - - return $display_options; - } - - /** - * Helper function used to figure out embed & download behaviour - * - * @param string $option_name - * @param H5PPermission $permission - * @param int $id - * @param bool &$value - */ - private function setDisplayOptionOverrides($option_name, $permission, $id, &$value) { - $behaviour = $this->h5pF->getOption($option_name, H5PDisplayOptionBehaviour::ALWAYS_SHOW); - // If never show globally, force hide - if ($behaviour == H5PDisplayOptionBehaviour::NEVER_SHOW) { - $value = false; - } - elseif ($behaviour == H5PDisplayOptionBehaviour::ALWAYS_SHOW) { - // If always show or permissions say so, force show - $value = true; - } - elseif ($behaviour == H5PDisplayOptionBehaviour::CONTROLLED_BY_PERMISSIONS) { - $value = $this->h5pF->hasPermission($permission, $id); - } - } - - /** - * Determine display option visibility when viewing H5P - * - * @param int $display_options - * @param int $id Might be content id or user id. - * Depends on what the platform needs to be able to determine permissions. - * @return array - */ - public function getDisplayOptionsForView($disable, $id) { - $display_options = $this->getDisplayOptionsAsArray($disable); - - if ($this->h5pF->getOption(self::DISPLAY_OPTION_FRAME, TRUE) == FALSE) { - $display_options[self::DISPLAY_OPTION_FRAME] = false; - } - else { - $this->setDisplayOptionOverrides(self::DISPLAY_OPTION_DOWNLOAD, H5PPermission::DOWNLOAD_H5P, $id, $display_options[self::DISPLAY_OPTION_DOWNLOAD]); - $this->setDisplayOptionOverrides(self::DISPLAY_OPTION_EMBED, H5PPermission::EMBED_H5P, $id, $display_options[self::DISPLAY_OPTION_EMBED]); - - if ($this->h5pF->getOption(self::DISPLAY_OPTION_COPYRIGHT, TRUE) == FALSE) { - $display_options[self::DISPLAY_OPTION_COPYRIGHT] = false; - } - } - $display_options[self::DISPLAY_OPTION_COPY] = $this->h5pF->hasPermission(H5PPermission::COPY_H5P, $id); - - return $display_options; - } - - /** - * Convert display options as single byte to array - * - * @param int $disable - * @return array - */ - private function getDisplayOptionsAsArray($disable) { - return array( - self::DISPLAY_OPTION_FRAME => !($disable & H5PCore::DISABLE_FRAME), - self::DISPLAY_OPTION_DOWNLOAD => !($disable & H5PCore::DISABLE_DOWNLOAD), - self::DISPLAY_OPTION_EMBED => !($disable & H5PCore::DISABLE_EMBED), - self::DISPLAY_OPTION_COPYRIGHT => !($disable & H5PCore::DISABLE_COPYRIGHT), - self::DISPLAY_OPTION_ABOUT => !!$this->h5pF->getOption(self::DISPLAY_OPTION_ABOUT, TRUE), - ); - } - - /** - * Small helper for getting the library's ID. - * - * @param array $library - * @param string [$libString] - * @return int Identifier, or FALSE if non-existent - */ - public function getLibraryId($library, $libString = NULL) { - static $libraryIdMap = []; - - if (!$libString) { - $libString = self::libraryToString($library); - } - - if (!isset($libraryIdMap[$libString])) { - $libraryIdMap[$libString] = $this->h5pF->getLibraryId($library['machineName'], $library['majorVersion'], $library['minorVersion']); - } - - return $libraryIdMap[$libString]; - } - - /** - * Convert strings of text into simple kebab case slugs. - * Very useful for readable urls etc. - * - * @param string $input - * @return string - */ - public static function slugify($input) { - // Down low - $input = strtolower($input); - - // Replace common chars - $input = str_replace( - array('æ', 'ø', 'ö', 'ó', 'ô', 'Ò', 'Õ', 'Ý', 'ý', 'ÿ', 'ā', 'ă', 'ą', 'œ', 'å', 'ä', 'á', 'à', 'â', 'ã', 'ç', 'ć', 'ĉ', 'ċ', 'č', 'é', 'è', 'ê', 'ë', 'í', 'ì', 'î', 'ï', 'ú', 'ñ', 'ü', 'ù', 'û', 'ß', 'ď', 'đ', 'ē', 'ĕ', 'ė', 'ę', 'ě', 'ĝ', 'ğ', 'ġ', 'ģ', 'ĥ', 'ħ', 'ĩ', 'ī', 'ĭ', 'į', 'ı', 'ij', 'ĵ', 'ķ', 'ĺ', 'ļ', 'ľ', 'ŀ', 'ł', 'ń', 'ņ', 'ň', 'ʼn', 'ō', 'ŏ', 'ő', 'ŕ', 'ŗ', 'ř', 'ś', 'ŝ', 'ş', 'š', 'ţ', 'ť', 'ŧ', 'ũ', 'ū', 'ŭ', 'ů', 'ű', 'ų', 'ŵ', 'ŷ', 'ź', 'ż', 'ž', 'ſ', 'ƒ', 'ơ', 'ư', 'ǎ', 'ǐ', 'ǒ', 'ǔ', 'ǖ', 'ǘ', 'ǚ', 'ǜ', 'ǻ', 'ǽ', 'ǿ'), - array('ae', 'oe', 'o', 'o', 'o', 'oe', 'o', 'o', 'y', 'y', 'y', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'a', 'c', 'c', 'c', 'c', 'c', 'e', 'e', 'e', 'e', 'i', 'i', 'i', 'i', 'u', 'n', 'u', 'u', 'u', 'es', 'd', 'd', 'e', 'e', 'e', 'e', 'e', 'g', 'g', 'g', 'g', 'h', 'h', 'i', 'i', 'i', 'i', 'i', 'ij', 'j', 'k', 'l', 'l', 'l', 'l', 'l', 'n', 'n', 'n', 'n', 'o', 'o', 'o', 'r', 'r', 'r', 's', 's', 's', 's', 't', 't', 't', 'u', 'u', 'u', 'u', 'u', 'u', 'w', 'y', 'z', 'z', 'z', 's', 'f', 'o', 'u', 'a', 'i', 'o', 'u', 'u', 'u', 'u', 'u', 'a', 'ae', 'oe'), - $input); - - // Replace everything else - $input = preg_replace('/[^a-z0-9]/', '-', $input); - - // Prevent double hyphen - $input = preg_replace('/-{2,}/', '-', $input); - - // Prevent hyphen in beginning or end - $input = trim($input, '-'); - - // Prevent to long slug - if (strlen($input) > 91) { - $input = substr($input, 0, 92); - } - - // Prevent empty slug - if ($input === '') { - $input = 'interactive'; - } - - return $input; - } - - /** - * Makes it easier to print response when AJAX request succeeds. - * - * @param mixed $data - * @since 1.6.0 - */ - public static function ajaxSuccess($data = NULL, $only_data = FALSE) { - $response = array( - 'success' => TRUE - ); - if ($data !== NULL) { - $response['data'] = $data; - - // Pass data flatly to support old methods - if ($only_data) { - $response = $data; - } - } - self::printJson($response); - } - - /** - * Makes it easier to print response when AJAX request fails. - * Will exit after printing error. - * - * @param string $message A human readable error message - * @param string $error_code An machine readable error code that a client - * should be able to interpret - * @param null|int $status_code Http response code - * @param array [$details=null] Better description of the error and possible which action to take - * @since 1.6.0 - */ - public static function ajaxError($message = NULL, $error_code = NULL, $status_code = NULL, $details = NULL) { - $response = array( - 'success' => FALSE - ); - if ($message !== NULL) { - $response['message'] = $message; - } - - if ($error_code !== NULL) { - $response['errorCode'] = $error_code; - } - - if ($details !== NULL) { - $response['details'] = $details; - } - - self::printJson($response, $status_code); - } - - /** - * Print JSON headers with UTF-8 charset and json encode response data. - * Makes it easier to respond using JSON. - * - * @param mixed $data - * @param null|int $status_code Http response code - */ - private static function printJson($data, $status_code = NULL) { - header('Cache-Control: no-cache'); - header('Content-Type: application/json; charset=utf-8'); - print json_encode($data); - } - - /** - * Get a new H5P security token for the given action. - * - * @param string $action - * @return string token - */ - public static function createToken($action) { - // Create and return token - return self::hashToken($action, self::getTimeFactor()); - } - - /** - * Create a time based number which is unique for each 12 hour. - * @return int - */ - private static function getTimeFactor() { - return ceil(time() / (86400 / 2)); - } - - /** - * Generate a unique hash string based on action, time and token - * - * @param string $action - * @param int $time_factor - * @return string - */ - private static function hashToken($action, $time_factor) { - global $SESSION; - if (!isset($SESSION->h5p_token)) { - // Create an unique key which is used to create action tokens for this session. - if (function_exists('random_bytes')) { - $SESSION->h5p_token = base64_encode(random_bytes(15)); - } - else if (function_exists('openssl_random_pseudo_bytes')) { - $SESSION->h5p_token = base64_encode(openssl_random_pseudo_bytes(15)); - } - else { - $SESSION->h5p_token = uniqid('', TRUE); - } - } - - // Create hash and return - return substr(hash('md5', $action . $time_factor . $SESSION->h5p_token), -16, 13); - } - - /** - * Verify if the given token is valid for the given action. - * - * @param string $action - * @param string $token - * @return boolean valid token - */ - public static function validToken($action, $token) { - // Get the timefactor - $time_factor = self::getTimeFactor(); - - // Check token to see if it's valid - return $token === self::hashToken($action, $time_factor) || // Under 12 hours - $token === self::hashToken($action, $time_factor - 1); // Between 12-24 hours - } - - /** - * Update content type cache - * - * @param object $postData Data sent to the hub - * - * @return bool|object Returns endpoint data if found, otherwise FALSE - */ - public function updateContentTypeCache($postData = NULL) { - $interface = $this->h5pF; - - // Make sure data is sent! - if (!isset($postData) || !isset($postData['uuid'])) { - return $this->fetchLibrariesMetadata(); - } - - $postData['current_cache'] = $this->h5pF->getOption('content_type_cache_updated_at', 0); - - $data = $interface->fetchExternalData(H5PHubEndpoints::createURL(H5PHubEndpoints::CONTENT_TYPES), $postData); - - if (! $this->h5pF->getOption('hub_is_enabled', TRUE)) { - return TRUE; - } - - // No data received - if (!$data) { - $interface->setErrorMessage( - $interface->t("Couldn't communicate with the H5P Hub. Please try again later."), - 'failed-communicationg-with-hub' - ); - return FALSE; - } - - $json = json_decode($data); - - // No libraries received - if (!isset($json->contentTypes) || empty($json->contentTypes)) { - $interface->setErrorMessage( - $interface->t('No content types were received from the H5P Hub. Please try again later.'), - 'no-content-types-from-hub' - ); - return FALSE; - } - - // Replace content type cache - $interface->replaceContentTypeCache($json); - - // Inform of the changes and update timestamp - $interface->setInfoMessage($interface->t('Library cache was successfully updated!')); - $interface->setOption('content_type_cache_updated_at', time()); - return $data; - } - - /** - * Update content hub metadata cache - */ - public function updateContentHubMetadataCache($lang = 'en') { - $url = H5PHubEndpoints::createURL(H5PHubEndpoints::METADATA); - $lastModified = $this->h5pF->getContentHubMetadataChecked($lang); - - $headers = array(); - if (!empty($lastModified)) { - $headers['If-Modified-Since'] = $lastModified; - } - $data = $this->h5pF->fetchExternalData("{$url}?lang={$lang}", NULL, TRUE, NULL, TRUE, $headers, NULL, 'GET'); - $lastChecked = new DateTime('now', new DateTimeZone('GMT')); - - if ($data['status'] !== 200 && $data['status'] !== 304) { - // If this was not a success, set the error message and return - $this->h5pF->setErrorMessage( - $this->h5pF->t('No metadata was received from the H5P Hub. Please try again later.') - ); - return null; - } - - // Update timestamp - $this->h5pF->setContentHubMetadataChecked($lastChecked->getTimestamp(), $lang); - - // Not modified - if ($data['status'] === 304) { - return null; - } - $this->h5pF->replaceContentHubMetadataCache($data['data'], $lang); - // TODO: If 200 should we have checked if it decodes? Or 'success'? Not sure if necessary though - return $data['data']; - } - - /** - * Get updated content hub metadata cache - * - * @param string $lang Language as ISO 639-1 code - * - * @return JsonSerializable|string - */ - public function getUpdatedContentHubMetadataCache($lang = 'en') { - $lastUpdate = $this->h5pF->getContentHubMetadataChecked($lang); - if (!$lastUpdate) { - return $this->updateContentHubMetadataCache($lang); - } - - $lastUpdate = new DateTime($lastUpdate); - $expirationTime = $lastUpdate->getTimestamp() + (60 * 60 * 24); // Check once per day - if (time() > $expirationTime) { - $update = $this->updateContentHubMetadataCache($lang); - if (!empty($update)) { - return $update; - } - } - - $storedCache = $this->h5pF->getContentHubMetadataCache($lang); - if (!$storedCache) { - // We don't have the value stored for some reason, reset last update and re-fetch - $this->h5pF->setContentHubMetadataChecked(null, $lang); - return $this->updateContentHubMetadataCache($lang); - } - - return $storedCache; - } - - /** - * Check if the current server setup is valid and set error messages - * - * @return object Setup object with errors and disable hub properties - */ - public function checkSetupErrorMessage() { - $setup = (object) array( - 'errors' => array(), - 'disable_hub' => FALSE - ); - - if (!class_exists('ZipArchive')) { - $setup->errors[] = $this->h5pF->t('Your PHP version does not support ZipArchive.'); - $setup->disable_hub = TRUE; - } - - if (!extension_loaded('mbstring')) { - $setup->errors[] = $this->h5pF->t( - 'The mbstring PHP extension is not loaded. H5P needs this to function properly' - ); - $setup->disable_hub = TRUE; - } - - // Check php version >= 5.2 - $php_version = explode('.', phpversion()); - if ($php_version[0] < 5 || ($php_version[0] === 5 && $php_version[1] < 2)) { - $setup->errors[] = $this->h5pF->t('Your PHP version is outdated. H5P requires version 5.2 to function properly. Version 5.6 or later is recommended.'); - $setup->disable_hub = TRUE; - } - - // Check write access - if (!$this->fs->hasWriteAccess()) { - $setup->errors[] = $this->h5pF->t('A problem with the server write access was detected. Please make sure that your server can write to your data folder.'); - $setup->disable_hub = TRUE; - } - - $max_upload_size = self::returnBytes(ini_get('upload_max_filesize')); - $max_post_size = self::returnBytes(ini_get('post_max_size')); - $byte_threshold = 5000000; // 5MB - if ($max_upload_size < $byte_threshold) { - $setup->errors[] = - $this->h5pF->t('Your PHP max upload size is quite small. With your current setup, you may not upload files larger than %number MB. This might be a problem when trying to upload H5Ps, images and videos. Please consider to increase it to more than 5MB.', array('%number' => number_format($max_upload_size / 1024 / 1024, 2, '.', ' '))); - } - - if ($max_post_size < $byte_threshold) { - $setup->errors[] = - $this->h5pF->t('Your PHP max post size is quite small. With your current setup, you may not upload files larger than %number MB. This might be a problem when trying to upload H5Ps, images and videos. Please consider to increase it to more than 5MB', array('%number' => number_format($max_upload_size / 1024 / 1024, 2, '.', ' '))); - } - - if ($max_upload_size > $max_post_size) { - $setup->errors[] = - $this->h5pF->t('Your PHP max upload size is bigger than your max post size. This is known to cause issues in some installations.'); - } - - // Check SSL - if (!extension_loaded('openssl')) { - $setup->errors[] = - $this->h5pF->t('Your server does not have SSL enabled. SSL should be enabled to ensure a secure connection with the H5P hub.'); - $setup->disable_hub = TRUE; - } - - return $setup; - } - - /** - * Check that all H5P requirements for the server setup is met. - */ - public function checkSetupForRequirements() { - $setup = $this->checkSetupErrorMessage(); - - $this->h5pF->setOption('hub_is_enabled', !$setup->disable_hub); - if (!empty($setup->errors)) { - foreach ($setup->errors as $err) { - $this->h5pF->setErrorMessage($err); - } - } - - if ($setup->disable_hub) { - // Inform how to re-enable hub - $this->h5pF->setErrorMessage( - $this->h5pF->t('H5P hub communication has been disabled because one or more H5P requirements failed.') - ); - $this->h5pF->setErrorMessage( - $this->h5pF->t('When you have revised your server setup you may re-enable H5P hub communication in H5P Settings.') - ); - } - } - - /** - * Return bytes from php_ini string value - * - * @param string $val - * - * @return int|string - */ - public static function returnBytes($val) { - $val = trim($val); - $last = strtolower($val[strlen($val) - 1]); - $bytes = (int) $val; - - switch ($last) { - case 'g': - $bytes *= 1024; - case 'm': - $bytes *= 1024; - case 'k': - $bytes *= 1024; - } - - return $bytes; - } - - /** - * Check if the current user has permission to update and install new - * libraries. - * - * @param bool [$set] Optional, sets the permission - * @return bool - */ - public function mayUpdateLibraries($set = null) { - static $can; - - if ($set !== null) { - // Use value set - $can = $set; - } - - if ($can === null) { - // Ask our framework - $can = $this->h5pF->mayUpdateLibraries(); - } - - return $can; - } - - /** - * Provide localization for the Core JS - * @return array - */ - public function getLocalization() { - return array( - 'fullscreen' => $this->h5pF->t('Fullscreen'), - 'disableFullscreen' => $this->h5pF->t('Disable fullscreen'), - 'download' => $this->h5pF->t('Download'), - 'copyrights' => $this->h5pF->t('Rights of use'), - 'embed' => $this->h5pF->t('Embed'), - 'size' => $this->h5pF->t('Size'), - 'showAdvanced' => $this->h5pF->t('Show advanced'), - 'hideAdvanced' => $this->h5pF->t('Hide advanced'), - 'advancedHelp' => $this->h5pF->t('Include this script on your website if you want dynamic sizing of the embedded content:'), - 'copyrightInformation' => $this->h5pF->t('Rights of use'), - 'close' => $this->h5pF->t('Close'), - 'title' => $this->h5pF->t('Title'), - 'author' => $this->h5pF->t('Author'), - 'year' => $this->h5pF->t('Year'), - 'source' => $this->h5pF->t('Source'), - 'license' => $this->h5pF->t('License'), - 'thumbnail' => $this->h5pF->t('Thumbnail'), - 'noCopyrights' => $this->h5pF->t('No copyright information available for this content.'), - 'reuse' => $this->h5pF->t('Reuse'), - 'reuseContent' => $this->h5pF->t('Reuse Content'), - 'reuseDescription' => $this->h5pF->t('Reuse this content.'), - 'downloadDescription' => $this->h5pF->t('Download this content as a H5P file.'), - 'copyrightsDescription' => $this->h5pF->t('View copyright information for this content.'), - 'embedDescription' => $this->h5pF->t('View the embed code for this content.'), - 'h5pDescription' => $this->h5pF->t('Visit H5P.org to check out more cool content.'), - 'contentChanged' => $this->h5pF->t('This content has changed since you last used it.'), - 'startingOver' => $this->h5pF->t("You'll be starting over."), - 'by' => $this->h5pF->t('by'), - 'showMore' => $this->h5pF->t('Show more'), - 'showLess' => $this->h5pF->t('Show less'), - 'subLevel' => $this->h5pF->t('Sublevel'), - 'confirmDialogHeader' => $this->h5pF->t('Confirm action'), - 'confirmDialogBody' => $this->h5pF->t('Please confirm that you wish to proceed. This action is not reversible.'), - 'cancelLabel' => $this->h5pF->t('Cancel'), - 'confirmLabel' => $this->h5pF->t('Confirm'), - 'licenseU' => $this->h5pF->t('Undisclosed'), - 'licenseCCBY' => $this->h5pF->t('Attribution'), - 'licenseCCBYSA' => $this->h5pF->t('Attribution-ShareAlike'), - 'licenseCCBYND' => $this->h5pF->t('Attribution-NoDerivs'), - 'licenseCCBYNC' => $this->h5pF->t('Attribution-NonCommercial'), - 'licenseCCBYNCSA' => $this->h5pF->t('Attribution-NonCommercial-ShareAlike'), - 'licenseCCBYNCND' => $this->h5pF->t('Attribution-NonCommercial-NoDerivs'), - 'licenseCC40' => $this->h5pF->t('4.0 International'), - 'licenseCC30' => $this->h5pF->t('3.0 Unported'), - 'licenseCC25' => $this->h5pF->t('2.5 Generic'), - 'licenseCC20' => $this->h5pF->t('2.0 Generic'), - 'licenseCC10' => $this->h5pF->t('1.0 Generic'), - 'licenseGPL' => $this->h5pF->t('General Public License'), - 'licenseV3' => $this->h5pF->t('Version 3'), - 'licenseV2' => $this->h5pF->t('Version 2'), - 'licenseV1' => $this->h5pF->t('Version 1'), - 'licensePD' => $this->h5pF->t('Public Domain'), - 'licenseCC010' => $this->h5pF->t('CC0 1.0 Universal (CC0 1.0) Public Domain Dedication'), - 'licensePDM' => $this->h5pF->t('Public Domain Mark'), - 'licenseC' => $this->h5pF->t('Copyright'), - 'contentType' => $this->h5pF->t('Content Type'), - 'licenseExtras' => $this->h5pF->t('License Extras'), - 'changes' => $this->h5pF->t('Changelog'), - 'contentCopied' => $this->h5pF->t('Content is copied to the clipboard'), - 'connectionLost' => $this->h5pF->t('Connection lost. Results will be stored and sent when you regain connection.'), - 'connectionReestablished' => $this->h5pF->t('Connection reestablished.'), - 'resubmitScores' => $this->h5pF->t('Attempting to submit stored results.'), - 'offlineDialogHeader' => $this->h5pF->t('Your connection to the server was lost'), - 'offlineDialogBody' => $this->h5pF->t('We were unable to send information about your completion of this task. Please check your internet connection.'), - 'offlineDialogRetryMessage' => $this->h5pF->t('Retrying in :num....'), - 'offlineDialogRetryButtonLabel' => $this->h5pF->t('Retry now'), - 'offlineSuccessfulSubmit' => $this->h5pF->t('Successfully submitted results.'), - 'mainTitle' => $this->h5pF->t('Sharing :title'), - 'editInfoTitle' => $this->h5pF->t('Edit info for :title'), - 'cancel' => $this->h5pF->t('Cancel'), - 'back' => $this->h5pF->t('Back'), - 'next' => $this->h5pF->t('Next'), - 'reviewInfo' => $this->h5pF->t('Review info'), - 'share' => $this->h5pF->t('Share'), - 'saveChanges' => $this->h5pF->t('Save changes'), - 'registerOnHub' => $this->h5pF->t('Register on the H5P Hub'), - 'updateRegistrationOnHub' => $this->h5pF->t('Save account settings'), - 'requiredInfo' => $this->h5pF->t('Required Info'), - 'optionalInfo' => $this->h5pF->t('Optional Info'), - 'reviewAndShare' => $this->h5pF->t('Review & Share'), - 'reviewAndSave' => $this->h5pF->t('Review & Save'), - 'shared' => $this->h5pF->t('Shared'), - 'currentStep' => $this->h5pF->t('Step :step of :total'), - 'sharingNote' => $this->h5pF->t('All content details can be edited after sharing'), - 'licenseDescription' => $this->h5pF->t('Select a license for your content'), - 'licenseVersion' => $this->h5pF->t('License Version'), - 'licenseVersionDescription' => $this->h5pF->t('Select a license version'), - 'disciplineLabel' => $this->h5pF->t('Disciplines'), - 'disciplineDescription' => $this->h5pF->t('You can select multiple disciplines'), - 'disciplineLimitReachedMessage' => $this->h5pF->t('You can select up to :numDisciplines disciplines'), - 'discipline' => array( - 'searchPlaceholder' => $this->h5pF->t('Type to search for disciplines'), - 'in' => $this->h5pF->t('in'), - 'dropdownButton' => $this->h5pF->t('Dropdown button'), - ), - 'removeChip' => $this->h5pF->t('Remove :chip from the list'), - 'keywordsPlaceholder' => $this->h5pF->t('Add keywords'), - 'keywords' => $this->h5pF->t('Keywords'), - 'keywordsDescription' => $this->h5pF->t('You can add multiple keywords separated by commas. Press "Enter" or "Add" to confirm keywords'), - 'altText' => $this->h5pF->t('Alt text'), - 'reviewMessage' => $this->h5pF->t('Please review the info below before you share'), - 'subContentWarning' => $this->h5pF->t('Sub-content (images, questions etc.) will be shared under :license unless otherwise specified in the authoring tool'), - 'disciplines' => $this->h5pF->t('Disciplines'), - 'shortDescription' => $this->h5pF->t('Short description'), - 'longDescription' => $this->h5pF->t('Long description'), - 'icon' => $this->h5pF->t('Icon'), - 'screenshots' => $this->h5pF->t('Screenshots'), - 'helpChoosingLicense' => $this->h5pF->t('Help me choose a license'), - 'shareFailed' => $this->h5pF->t('Share failed.'), - 'editingFailed' => $this->h5pF->t('Editing failed.'), - 'shareTryAgain' => $this->h5pF->t('Something went wrong, please try to share again.'), - 'pleaseWait' => $this->h5pF->t('Please wait...'), - 'language' => $this->h5pF->t('Language'), - 'level' => $this->h5pF->t('Level'), - 'shortDescriptionPlaceholder' => $this->h5pF->t('Short description of your content'), - 'longDescriptionPlaceholder' => $this->h5pF->t('Long description of your content'), - 'description' => $this->h5pF->t('Description'), - 'iconDescription' => $this->h5pF->t('640x480px. If not selected content will use category icon'), - 'screenshotsDescription' => $this->h5pF->t('Add up to five screenshots of your content'), - 'submitted' => $this->h5pF->t('Submitted!'), - 'isNowSubmitted' => $this->h5pF->t('Is now submitted to H5P Hub'), - 'changeHasBeenSubmitted' => $this->h5pF->t('A change has been submited for'), - 'contentAvailable' => $this->h5pF->t('Your content will normally be available in the Hub within one business day.'), - 'contentUpdateSoon' => $this->h5pF->t('Your content will update soon'), - 'contentLicenseTitle' => $this->h5pF->t('Content License Info'), - 'licenseDialogDescription' => $this->h5pF->t('Click on a specific license to get info about proper usage'), - 'publisherFieldTitle' => $this->h5pF->t('Publisher'), - 'publisherFieldDescription' => $this->h5pF->t('This will display as the "Publisher name" on shared content'), - 'emailAddress' => $this->h5pF->t('Email Address'), - 'publisherDescription' => $this->h5pF->t('Publisher description'), - 'publisherDescriptionText' => $this->h5pF->t('This will be displayed under "Publisher info" on shared content'), - 'contactPerson' => $this->h5pF->t('Contact Person'), - 'phone' => $this->h5pF->t('Phone'), - 'address' => $this->h5pF->t('Address'), - 'city' => $this->h5pF->t('City'), - 'zip' => $this->h5pF->t('Zip'), - 'country' => $this->h5pF->t('Country'), - 'logoUploadText' => $this->h5pF->t('Organization logo or avatar'), - 'acceptTerms' => $this->h5pF->t('I accept the terms of use'), - 'successfullyRegistred' => $this->h5pF->t('You have successfully registered an account on the H5P Hub'), - 'successfullyRegistredDescription' => $this->h5pF->t('You account details can be changed'), - 'successfullyUpdated' => $this->h5pF->t('Your H5P Hub account settings have successfully been changed'), - 'accountDetailsLinkText' => $this->h5pF->t('here'), - 'registrationTitle' => $this->h5pF->t('H5P Hub Registration'), - 'registrationFailed' => $this->h5pF->t('An error occurred'), - 'registrationFailedDescription' => $this->h5pF->t('We were not able to create an account at this point. Something went wrong. Try again later.'), - 'maxLength' => $this->h5pF->t(':length is the maximum number of characters'), - 'keywordExists' => $this->h5pF->t('Keyword already exists!'), - 'licenseDetails' => $this->h5pF->t('License details'), - 'remove' => $this->h5pF->t('Remove'), - 'removeImage' => $this->h5pF->t('Remove image'), - 'cancelPublishConfirmationDialogTitle' => $this->h5pF->t('Cancel sharing'), - 'cancelPublishConfirmationDialogDescription' => $this->h5pF->t('Are you sure you want to cancel the sharing process?'), - 'cancelPublishConfirmationDialogCancelButtonText' => $this->h5pF->t('No'), - 'cancelPublishConfirmationDialogConfirmButtonText' => $this->h5pF->t('Yes'), - 'add' => $this->h5pF->t('Add'), - 'age' => $this->h5pF->t('Typical age'), - 'ageDescription' => $this->h5pF->t('The target audience of this content. Possible input formats separated by commas: "1,34-45,-50,59-".'), - 'invalidAge' => $this->h5pF->t('Invalid input format for Typical age. Possible input formats separated by commas: "1, 34-45, -50, -59-".'), - 'contactPersonDescription' => $this->h5pF->t('H5P will reach out to the contact person in case there are any issues with the content shared by the publisher. The contact person\'s name or other information will not be published or shared with third parties'), - 'emailAddressDescription' => $this->h5pF->t('The email address will be used by H5P to reach out to the publisher in case of any issues with the content or in case the publisher needs to recover their account. It will not be published or shared with any third parties'), - 'copyrightWarning' => $this->h5pF->t('Copyrighted material cannot be shared in the H5P Content Hub. If the content is licensed with a OER friendly license like Creative Commons, please choose the appropriate license. If not this content cannot be shared.'), - 'keywordsExits' => $this->h5pF->t('Keywords already exists!'), - 'someKeywordsExits' => $this->h5pF->t('Some of these keywords already exist'), - 'width' => $this->h5pF->t('width'), - 'height' => $this->h5pF->t('height') - ); - } - - /** - * Publish content on the H5P Hub. - * - * @param bigint $id - * @return stdClass - */ - public function hubRetrieveContent($id) { - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $response = $this->h5pF->fetchExternalData( - H5PHubEndpoints::createURL(H5PHubEndpoints::CONTENT . "/{$id}"), - NULL, TRUE, NULL, TRUE, $headers - ); - - if (empty($response['data'])) { - throw new Exception($this->h5pF->t('Unable to authorize with the H5P Hub. Please check your Hub registration and connection.')); - } - - if (isset($response['status']) && $response['status'] !== 200) { - if ($response['status'] === 404) { - $this->h5pF->setErrorMessage($this->h5pF->t('Content is not shared on the H5P OER Hub.')); - return NULL; - } - throw new Exception($this->h5pF->t("Couldn't communicate with the H5P Hub. Please try again later.")); - } - - $hub_content = json_decode($response['data'])->data; - $hub_content->id = "$hub_content->id"; - return $hub_content; - } - - /** - * Publish content on the H5P Hub. - * - * @param array $data Data from content publishing process - * @param array $files Files to upload with the content publish - * @param bigint $content_hub_id For updating existing content - * @return stdClass - */ - public function hubPublishContent($data, $files, $content_hub_id = NULL) { - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $data['published'] = '1'; - $endpoint = H5PHubEndpoints::CONTENT; - if ($content_hub_id !== NULL) { - $endpoint .= "/{$content_hub_id}"; - $data['_method'] = 'PUT'; - } - - $response = $this->h5pF->fetchExternalData( - H5PHubEndpoints::createURL($endpoint), - $data, TRUE, NULL, TRUE, $headers, $files - ); - - if (empty($response['data']) || $response['status'] === 403) { - throw new Exception($this->h5pF->t('Unable to authorize with the H5P Hub. Please check your Hub registration and connection.')); - } - - if (isset($response['status']) && $response['status'] !== 200) { - throw new Exception($this->h5pF->t('Connecting to the content hub failed, please try again later.')); - } - - $result = json_decode($response['data']); - if (isset($result->success) && $result->success === TRUE) { - return $result; - } - elseif (!empty($result->errors)) { - // Relay any error messages - $e = new Exception($this->h5pF->t('Validation failed.')); - $e->errors = $result->errors; - throw $e; - } - } - - /** - * Creates the authorization header needed to access the private parts of - * the H5P Hub. - * - * @return string - */ - public function hubGetAuthorizationHeader() { - $site_uuid = $this->h5pF->getOption('site_uuid', ''); - $hub_secret = $this->h5pF->getOption('hub_secret', ''); - if (empty($site_uuid)) { - $this->h5pF->setErrorMessage($this->h5pF->t('Missing Site UUID. Please check your Hub registration.')); - } - elseif (empty($hub_secret)) { - $this->h5pF->setErrorMessage($this->h5pF->t('Missing Hub Secret. Please check your Hub registration.')); - } - return 'Basic ' . base64_encode("$site_uuid:$hub_secret"); - } - - /** - * Unpublish content from content hub - * - * @param integer $hubId Content hub id - * - * @return bool True if successful - */ - public function hubUnpublishContent($hubId) { - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $url = H5PHubEndpoints::createURL(H5PHubEndpoints::CONTENT); - $response = $this->h5pF->fetchExternalData("{$url}/{$hubId}", array( - 'published' => '0', - ), true, null, true, $headers, array(), 'PUT'); - - // Remove shared status if successful - if (!empty($response) && $response['status'] === 200) { - $msg = $this->h5pF->t('Content successfully unpublished'); - $this->h5pF->setInfoMessage($msg); - - return true; - } - $msg = $this->h5pF->t('Content unpublish failed'); - $this->h5pF->setErrorMessage($msg); - - return false; - } - - /** - * Sync content with content hub - * - * @param integer $hubId Content hub id - * @param string $exportPath Export path where .h5p for content can be found - * - * @return bool - */ - public function hubSyncContent($hubId, $exportPath) { - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $url = H5PHubEndpoints::createURL(H5PHubEndpoints::CONTENT); - $response = $this->h5pF->fetchExternalData("{$url}/{$hubId}", array( - 'download_url' => $exportPath, - 'resync' => '1', - ), true, null, true, $headers, array(), 'PUT'); - - if (!empty($response) && $response['status'] === 200) { - $msg = $this->h5pF->t('Content sync queued'); - $this->h5pF->setInfoMessage($msg); - return true; - } - - $msg = $this->h5pF->t('Content sync failed'); - $this->h5pF->setErrorMessage($msg); - return false; - } - - /** - * Fetch account info for our site from the content hub - * - * @return array|bool|string False if account is not setup, otherwise data - */ - public function hubAccountInfo() { - $siteUuid = $this->h5pF->getOption('site_uuid', null); - $secret = $this->h5pF->getOption('hub_secret', null); - if (empty($siteUuid) && !empty($secret)) { - $this->h5pF->setErrorMessage($this->h5pF->t('H5P Hub secret is set without a site uuid. This may be fixed by restoring the site uuid or removing the hub secret and registering a new account with the content hub.')); - throw new Exception('Hub secret not set'); - } - - if (empty($siteUuid) || empty($secret)) { - $this->h5pF->setErrorMessage($this->h5pF->t('Missing Site UUID or Hub Secret. Please check your Hub registration.')); - return false; - } - - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $url = H5PHubEndpoints::createURL(H5PHubEndpoints::REGISTER); - $accountInfo = $this->h5pF->fetchExternalData("{$url}/{$siteUuid}", - null, true, null, true, $headers, array(), 'GET'); - - if ($accountInfo['status'] === 401) { - // Unauthenticated, invalid hub secret and site uuid combination - $this->h5pF->setErrorMessage($this->h5pF->t('Hub account authentication info is invalid. This may be fixed by an admin by restoring the hub secret or register a new account with the content hub.')); - return false; - } - - if ($accountInfo['status'] !== 200) { - $this->h5pF->setErrorMessage($this->h5pF->t('Unable to retrieve HUB account information. Please contact support.')); - return false; - } - - return json_decode($accountInfo['data'])->data; - } - - /** - * Register account - * - * @param array $formData Form data. Should include: name, email, description, - * contact_person, phone, address, city, zip, country, remove_logo - * @param object $logo Input image - * - * @return array - */ - public function hubRegisterAccount($formData, $logo) { - - $uuid = $this->h5pF->getOption('site_uuid', ''); - if (empty($uuid)) { - // Attempt to fetch a new site uuid - $uuid = $this->fetchLibrariesMetadata(false, true); - if (!$uuid) { - return [ - 'message' => $this->h5pF->t('Site is missing a unique site uuid and was unable to set a new one. The H5P Content Hub is disabled until this problem can be resolved. Please make sure the H5P Hub is enabled in the H5P settings and try again later.'), - 'status_code' => 403, - 'error_code' => 'MISSING_SITE_UUID', - 'success' => FALSE, - ]; - } - } - - $formData['site_uuid'] = $uuid; - - $headers = []; - $endpoint = H5PHubEndpoints::REGISTER; - // Update if already registered - $hasRegistered = $this->h5pF->getOption('hub_secret'); - if ($hasRegistered) { - $endpoint .= "/{$uuid}"; - $formData['_method'] = 'PUT'; - $headers = [ - 'Authorization' => $this->hubGetAuthorizationHeader(), - ]; - } - - $url = H5PHubEndpoints::createURL($endpoint); - $registration = $this->h5pF->fetchExternalData( - $url, - $formData, - NULL, - NULL, - TRUE, - $headers, - isset($logo) ? ['logo' => $logo] : [] - ); - - try { - $results = json_decode($registration['data']); - } catch (Exception $e) { - return [ - 'message' => 'Could not parse json response.', - 'status_code' => 424, - 'error_code' => 'COULD_NOT_PARSE_RESPONSE', - 'success' => FALSE, - ]; - } - - if (isset($results->errors->site_uuid)) { - return [ - 'message' => 'Site UUID is not unique. This must be fixed by an admin by restoring the hub secret or remove the site uuid and register as a new account with the content hub.', - 'status_code' => 403, - 'error_code' => 'SITE_UUID_NOT_UNIQUE', - 'success' => FALSE, - ]; - } - - if (isset($results->errors->logo)) { - return [ - 'message' => $results->errors->logo[0], - 'status_code' => 400, - 'success' => FALSE, - ]; - } - - if ( - !isset($results->success) - || $results->success === FALSE - || !$hasRegistered && !isset($results->account->secret) - || $registration['status'] !== 200 - ) { - return [ - 'message' => 'Unable to register the account. Please contact support team.', - 'status_code' => 422, - 'error_code' => 'REGISTRATION_FAILED', - 'success' => FALSE, - ]; - } - - if (!$hasRegistered) { - $this->h5pF->setOption('hub_secret', $results->account->secret); - } - - return [ - 'message' => $this->h5pF->t('Account successfully registered.'), - 'status_code' => 200, - 'success' => TRUE, - ]; - } - - /** - * Get status of content from content hub - * - * @param string $hubContentId - * @param int $syncStatus - * - * @return false|int Returns a new H5PContentStatus if successful, else false - */ - public function getHubContentStatus($hubContentId, $syncStatus) { - $headers = array( - 'Authorization' => $this->hubGetAuthorizationHeader(), - 'Accept' => 'application/json', - ); - - $url = H5PHubEndpoints::createURL(H5PHubEndpoints::CONTENT); - $response = $this->h5pF->fetchExternalData("{$url}/{$hubContentId}/status", - null, true, null, true, $headers); - - if (isset($response['status']) && $response['status'] === 403) { - $msg = $this->h5pF->t('The request for content status was unauthorized. This could be because the content belongs to a different account, or your account is not setup properly.'); - $this->h5pF->setErrorMessage($msg); - return false; - } - if (empty($response) || $response['status'] !== 200) { - $msg = $this->h5pF->t('Could not get content hub sync status for content.'); - $this->h5pF->setErrorMessage($msg); - return false; - } - - $data = json_decode($response['data']); - - if (isset($data->messages)) { - // TODO: Is this the right place/way to display them? - - if (!empty($data->messages->info)) { - foreach ($data->messages->info as $info) { - $this->h5pF->setInfoMessage($info); - } - } - if (!empty($data->messages->error)) { - foreach ($data->messages->error as $error) { - $this->h5pF->setErrorMessage($error->message, $error->code); - } - } - } - - $contentStatus = intval($data->status); - // Content status updated - if ($contentStatus !== H5PContentStatus::STATUS_WAITING) { - $newState = H5PContentHubSyncStatus::SYNCED; - if ($contentStatus !== H5PContentStatus::STATUS_DOWNLOADED) { - $newState = H5PContentHubSyncStatus::FAILED; - } - else if (intval($syncStatus) !== $contentStatus) { - // Content status successfully transitioned to synced/downloaded - $successMsg = $this->h5pF->t('Content was successfully shared on the content hub.'); - $this->h5pF->setInfoMessage($successMsg); - } - - return $newState; - } - - return false; - } -} - -/** - * Functions for validating basic types from H5P library semantics. - * @property bool allowedStyles - */ -class H5PContentValidator { - public $h5pF; - public $h5pC; - private $typeMap, $libraries, $dependencies, $nextWeight; - private static $allowed_styleable_tags = array('span', 'p', 'div','h1','h2','h3', 'td'); - - /** @var bool Allowed styles status. */ - protected $allowedStyles; - - /** - * Constructor for the H5PContentValidator - * - * @param object $H5PFramework - * The frameworks implementation of the H5PFrameworkInterface - * @param object $H5PCore - * The main H5PCore instance - */ - public function __construct($H5PFramework, $H5PCore) { - $this->h5pF = $H5PFramework; - $this->h5pC = $H5PCore; - $this->typeMap = array( - 'text' => 'validateText', - 'number' => 'validateNumber', - 'boolean' => 'validateBoolean', - 'list' => 'validateList', - 'group' => 'validateGroup', - 'file' => 'validateFile', - 'image' => 'validateImage', - 'video' => 'validateVideo', - 'audio' => 'validateAudio', - 'select' => 'validateSelect', - 'library' => 'validateLibrary', - ); - $this->nextWeight = 1; - - // Keep track of the libraries we load to avoid loading it multiple times. - $this->libraries = array(); - - // Keep track of all dependencies for the given content. - $this->dependencies = array(); - } - - /** - * Add Addon library. - */ - public function addon($library) { - $depKey = 'preloaded-' . $library['machineName']; - $this->dependencies[$depKey] = array( - 'library' => $library, - 'type' => 'preloaded' - ); - $this->nextWeight = $this->h5pC->findLibraryDependencies($this->dependencies, $library, $this->nextWeight); - $this->dependencies[$depKey]['weight'] = $this->nextWeight++; - } - - /** - * Get the flat dependency tree. - * - * @return array - */ - public function getDependencies() { - return $this->dependencies; - } - - /** - * Validate metadata - * - * @param array $metadata - * @return array Validated & filtered - */ - public function validateMetadata($metadata) { - $semantics = $this->getMetadataSemantics(); - $group = (object)$metadata; - - // Stop complaining about "invalid selected option in select" for - // old content without license chosen. - if (!isset($group->license)) { - $group->license = 'U'; - } - - $this->validateGroup($group, (object) array( - 'type' => 'group', - 'fields' => $semantics, - ), FALSE); - - return (array)$group; - } - - /** - * Validate given text value against text semantics. - * @param $text - * @param $semantics - */ - public function validateText(&$text, $semantics) { - if (!is_string($text)) { - $text = ''; - } - if (isset($semantics->tags)) { - // Not testing for empty array allows us to use the 4 defaults without - // specifying them in semantics. - $tags = array_merge(array('div', 'span', 'p', 'br'), $semantics->tags); - - // Add related tags for table etc. - if (in_array('table', $tags)) { - $tags = array_merge($tags, array('tr', 'td', 'th', 'colgroup', 'thead', 'tbody', 'tfoot')); - } - if (in_array('b', $tags) && ! in_array('strong', $tags)) { - $tags[] = 'strong'; - } - if (in_array('i', $tags) && ! in_array('em', $tags)) { - $tags[] = 'em'; - } - if (in_array('ul', $tags) || in_array('ol', $tags) && ! in_array('li', $tags)) { - $tags[] = 'li'; - } - if (in_array('del', $tags) || in_array('strike', $tags) && ! in_array('s', $tags)) { - $tags[] = 's'; - } - - // Determine allowed style tags - $stylePatterns = array(); - // All styles must be start to end patterns (^...$) - if (isset($semantics->font)) { - if (isset($semantics->font->size) && $semantics->font->size) { - $stylePatterns[] = '/^font-size: *[0-9.]+(em|px|%) *;?$/i'; - } - if (isset($semantics->font->family) && $semantics->font->family) { - $stylePatterns[] = '/^font-family: *[-a-z0-9," ]+;?$/i'; - } - if (isset($semantics->font->color) && $semantics->font->color) { - $stylePatterns[] = '/^color: *(#[a-f0-9]{3}[a-f0-9]{3}?|rgba?\([0-9, ]+\)) *;?$/i'; - } - if (isset($semantics->font->background) && $semantics->font->background) { - $stylePatterns[] = '/^background-color: *(#[a-f0-9]{3}[a-f0-9]{3}?|rgba?\([0-9, ]+\)) *;?$/i'; - } - if (isset($semantics->font->spacing) && $semantics->font->spacing) { - $stylePatterns[] = '/^letter-spacing: *[0-9.]+(em|px|%) *;?$/i'; - } - if (isset($semantics->font->height) && $semantics->font->height) { - $stylePatterns[] = '/^line-height: *[0-9.]+(em|px|%|) *;?$/i'; - } - } - - // Alignment is allowed for all wysiwyg texts - $stylePatterns[] = '/^text-align: *(center|left|right);?$/i'; - - // Strip invalid HTML tags. - $text = $this->filter_xss($text, $tags, $stylePatterns); - } - else { - // Filter text to plain text. - $text = htmlspecialchars($text, ENT_QUOTES, 'UTF-8', FALSE); - } - - // Check if string is within allowed length - if (isset($semantics->maxLength)) { - if (!extension_loaded('mbstring')) { - $this->h5pF->setErrorMessage($this->h5pF->t('The mbstring PHP extension is not loaded. H5P need this to function properly'), 'mbstring-unsupported'); - } - else { - $text = mb_substr($text, 0, $semantics->maxLength); - } - } - - // Check if string is according to optional regexp in semantics - if (!($text === '' && isset($semantics->optional) && $semantics->optional) && isset($semantics->regexp)) { - // Escaping '/' found in patterns, so that it does not break regexp fencing. - $pattern = '/' . str_replace('/', '\\/', $semantics->regexp->pattern) . '/'; - $pattern .= isset($semantics->regexp->modifiers) ? $semantics->regexp->modifiers : ''; - if (preg_match($pattern, $text) === 0) { - // Note: explicitly ignore return value FALSE, to avoid removing text - // if regexp is invalid... - $this->h5pF->setErrorMessage($this->h5pF->t('Provided string is not valid according to regexp in semantics. (value: "%value", regexp: "%regexp")', array('%value' => $text, '%regexp' => $pattern)), 'semantics-invalid-according-regexp'); - $text = ''; - } - } - } - - /** - * Validates content files - * - * @param string $contentPath - * The path containing content files to validate. - * @param bool $isLibrary - * @return bool TRUE if all files are valid - * TRUE if all files are valid - * FALSE if one or more files fail validation. Error message should be set accordingly by validator. - */ - public function validateContentFiles($contentPath, $isLibrary = FALSE) { - if ($this->h5pC->disableFileCheck === TRUE) { - return TRUE; - } - - // Scan content directory for files, recurse into sub directories. - $files = array_diff(scandir($contentPath), array('.','..')); - $valid = TRUE; - $whitelist = $this->h5pF->getWhitelist($isLibrary, H5PCore::$defaultContentWhitelist, H5PCore::$defaultLibraryWhitelistExtras); - - $wl_regex = '/\.(' . preg_replace('/ +/i', '|', preg_quote($whitelist)) . ')$/i'; - - foreach ($files as $file) { - $filePath = $contentPath . '/' . $file; - if (is_dir($filePath)) { - $valid = $this->validateContentFiles($filePath, $isLibrary) && $valid; - } - else { - // Snipped from drupal 6 "file_validate_extensions". Using own code - // to avoid 1. creating a file-like object just to test for the known - // file name, 2. testing against a returned error array that could - // never be more than 1 element long anyway, 3. recreating the regex - // for every file. - if (!extension_loaded('mbstring')) { - $this->h5pF->setErrorMessage($this->h5pF->t('The mbstring PHP extension is not loaded. H5P need this to function properly'), 'mbstring-unsupported'); - $valid = FALSE; - } - else if (!preg_match($wl_regex, mb_strtolower($file))) { - $this->h5pF->setErrorMessage($this->h5pF->t('File "%filename" not allowed. Only files with the following extensions are allowed: %files-allowed.', array('%filename' => $file, '%files-allowed' => $whitelist)), 'not-in-whitelist'); - $valid = FALSE; - } - } - } - return $valid; - } - - /** - * Validate given value against number semantics - * @param $number - * @param $semantics - */ - public function validateNumber(&$number, $semantics) { - // Validate that $number is indeed a number - if (!is_numeric($number)) { - $number = 0; - } - // Check if number is within valid bounds. Move within bounds if not. - if (isset($semantics->min) && $number < $semantics->min) { - $number = $semantics->min; - } - if (isset($semantics->max) && $number > $semantics->max) { - $number = $semantics->max; - } - // Check if number is within allowed bounds even if step value is set. - if (isset($semantics->step)) { - $testNumber = $number - (isset($semantics->min) ? $semantics->min : 0); - $rest = $testNumber % $semantics->step; - if ($rest !== 0) { - $number -= $rest; - } - } - // Check if number has proper number of decimals. - if (isset($semantics->decimals)) { - $number = round($number, $semantics->decimals); - } - } - - /** - * Validate given value against boolean semantics - * @param $bool - * @return bool - */ - public function validateBoolean(&$bool) { - return is_bool($bool); - } - - /** - * Validate select values - * @param $select - * @param $semantics - */ - public function validateSelect(&$select, $semantics) { - $optional = isset($semantics->optional) && $semantics->optional; - $strict = FALSE; - if (isset($semantics->options) && !empty($semantics->options)) { - // We have a strict set of options to choose from. - $strict = TRUE; - $options = array(); - - foreach ($semantics->options as $option) { - // Support optgroup - just flatten options into one - if (isset($option->type) && $option->type === 'optgroup') { - foreach ($option->options as $suboption) { - $options[$suboption->value] = TRUE; - } - } - elseif (isset($option->value)) { - $options[$option->value] = TRUE; - } - } - } - - if (isset($semantics->multiple) && $semantics->multiple) { - // Multi-choice generates array of values. Test each one against valid - // options, if we are strict. First make sure we are working on an - // array. - if (!is_array($select)) { - $select = array($select); - } - - foreach ($select as $key => &$value) { - if ($strict && !$optional && !isset($options[$value])) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid selected option in multi-select.')); - unset($select[$key]); - } - else { - $select[$key] = htmlspecialchars($value, ENT_QUOTES, 'UTF-8', FALSE); - } - } - } - else { - // Single mode. If we get an array in here, we chop off the first - // element and use that instead. - if (is_array($select)) { - $select = $select[0]; - } - - if ($strict && !$optional && !isset($options[$select])) { - $this->h5pF->setErrorMessage($this->h5pF->t('Invalid selected option in select.')); - $select = $semantics->options[0]->value; - } - $select = htmlspecialchars($select, ENT_QUOTES, 'UTF-8', FALSE); - } - } - - /** - * Validate given list value against list semantics. - * Will recurse into validating each item in the list according to the type. - * @param $list - * @param $semantics - */ - public function validateList(&$list, $semantics) { - $field = $semantics->field; - $function = $this->typeMap[$field->type]; - - // Check that list is not longer than allowed length. We do this before - // iterating to avoid unnecessary work. - if (isset($semantics->max)) { - array_splice($list, $semantics->max); - } - - if (!is_array($list)) { - $list = array(); - } - - // Validate each element in list. - foreach ($list as $key => &$value) { - if (!is_int($key)) { - array_splice($list, $key, 1); - continue; - } - $this->$function($value, $field); - if ($value === NULL) { - array_splice($list, $key, 1); - } - } - - if (count($list) === 0) { - $list = NULL; - } - } - - /** - * Validate a file like object, such as video, image, audio and file. - * @param $file - * @param $semantics - * @param array $typeValidKeys - */ - private function _validateFilelike(&$file, $semantics, $typeValidKeys = array()) { - // Do not allow to use files from other content folders. - $matches = array(); - if (preg_match($this->h5pC->relativePathRegExp, $file->path, $matches)) { - $file->path = $matches[5]; - } - - // Remove temporary files suffix - if (substr($file->path, -4, 4) === '#tmp') { - $file->path = substr($file->path, 0, strlen($file->path) - 4); - } - - // Make sure path and mime does not have any special chars - $file->path = htmlspecialchars($file->path, ENT_QUOTES, 'UTF-8', FALSE); - if (isset($file->mime)) { - $file->mime = htmlspecialchars($file->mime, ENT_QUOTES, 'UTF-8', FALSE); - } - - // Remove attributes that should not exist, they may contain JSON escape - // code. - $validKeys = array_merge(array('path', 'mime', 'copyright'), $typeValidKeys); - if (isset($semantics->extraAttributes)) { - $validKeys = array_merge($validKeys, $semantics->extraAttributes); // TODO: Validate extraAttributes - } - $this->filterParams($file, $validKeys); - - if (isset($file->width)) { - $file->width = intval($file->width); - } - - if (isset($file->height)) { - $file->height = intval($file->height); - } - - if (isset($file->codecs)) { - $file->codecs = htmlspecialchars($file->codecs, ENT_QUOTES, 'UTF-8', FALSE); - } - - if (isset($file->bitrate)) { - $file->bitrate = intval($file->bitrate); - } - - if (isset($file->quality)) { - if (!is_object($file->quality) || !isset($file->quality->level) || !isset($file->quality->label)) { - unset($file->quality); - } - else { - $this->filterParams($file->quality, array('level', 'label')); - $file->quality->level = intval($file->quality->level); - $file->quality->label = htmlspecialchars($file->quality->label, ENT_QUOTES, 'UTF-8', FALSE); - } - } - - if (isset($file->copyright)) { - $this->validateGroup($file->copyright, $this->getCopyrightSemantics()); - } - } - - /** - * Validate given file data - * @param $file - * @param $semantics - */ - public function validateFile(&$file, $semantics) { - $this->_validateFilelike($file, $semantics); - } - - /** - * Validate given image data - * @param $image - * @param $semantics - */ - public function validateImage(&$image, $semantics) { - $this->_validateFilelike($image, $semantics, array('width', 'height', 'originalImage')); - } - - /** - * Validate given video data - * @param $video - * @param $semantics - */ - public function validateVideo(&$video, $semantics) { - foreach ($video as &$variant) { - $this->_validateFilelike($variant, $semantics, array('width', 'height', 'codecs', 'quality', 'bitrate')); - } - } - - /** - * Validate given audio data - * @param $audio - * @param $semantics - */ - public function validateAudio(&$audio, $semantics) { - foreach ($audio as &$variant) { - $this->_validateFilelike($variant, $semantics); - } - } - - /** - * Validate given group value against group semantics. - * Will recurse into validating each group member. - * @param $group - * @param $semantics - * @param bool $flatten - */ - public function validateGroup(&$group, $semantics, $flatten = TRUE) { - // Groups with just one field are compressed in the editor to only output - // the child content. (Exemption for fake groups created by - // "validateBySemantics" above) - $function = null; - $field = null; - - $isSubContent = isset($semantics->isSubContent) && $semantics->isSubContent === TRUE; - - if (count($semantics->fields) == 1 && $flatten && !$isSubContent) { - $field = $semantics->fields[0]; - $function = $this->typeMap[$field->type]; - $this->$function($group, $field); - } - else { - foreach ($group as $key => &$value) { - // If subContentId is set, keep value - if($isSubContent && ($key == 'subContentId')){ - continue; - } - - // Find semantics for name=$key - $found = FALSE; - foreach ($semantics->fields as $field) { - if ($field->name == $key) { - if (isset($semantics->optional) && $semantics->optional) { - $field->optional = TRUE; - } - $function = $this->typeMap[$field->type]; - $found = TRUE; - break; - } - } - if ($found) { - if ($function) { - $this->$function($value, $field); - if ($value === NULL) { - unset($group->$key); - } - } - else { - // We have a field type in semantics for which we don't have a - // known validator. - $this->h5pF->setErrorMessage($this->h5pF->t('H5P internal error: unknown content type "@type" in semantics. Removing content!', array('@type' => $field->type)), 'semantics-unknown-type'); - unset($group->$key); - } - } - else { - // If validator is not found, something exists in content that does - // not have a corresponding semantics field. Remove it. - // $this->h5pF->setErrorMessage($this->h5pF->t('H5P internal error: no validator exists for @key', array('@key' => $key))); - unset($group->$key); - } - } - } - } - - /** - * Validate given library value against library semantics. - * Check if provided library is within allowed options. - * - * Will recurse into validating the library's semantics too. - * @param $value - * @param $semantics - */ - public function validateLibrary(&$value, $semantics) { - if (!isset($value->library)) { - $value = NULL; - return; - } - - // Check for array of objects or array of strings - if (is_object($semantics->options[0])) { - $getLibraryNames = function ($item) { - return $item->name; - }; - $libraryNames = array_map($getLibraryNames, $semantics->options); - } - else { - $libraryNames = $semantics->options; - } - - if (!in_array($value->library, $libraryNames)) { - $message = NULL; - // Create an understandable error message: - $machineNameArray = explode(' ', $value->library); - $machineName = $machineNameArray[0]; - foreach ($libraryNames as $semanticsLibrary) { - $semanticsMachineNameArray = explode(' ', $semanticsLibrary); - $semanticsMachineName = $semanticsMachineNameArray[0]; - if ($machineName === $semanticsMachineName) { - // Using the wrong version of the library in the content - $message = $this->h5pF->t('The version of the H5P library %machineName used in this content is not valid. Content contains %contentLibrary, but it should be %semanticsLibrary.', array( - '%machineName' => $machineName, - '%contentLibrary' => $value->library, - '%semanticsLibrary' => $semanticsLibrary - )); - break; - } - } - - // Using a library in content that is not present at all in semantics - if ($message === NULL) { - $message = $this->h5pF->t('The H5P library %library used in the content is not valid', array( - '%library' => $value->library - )); - } - - $this->h5pF->setErrorMessage($message); - $value = NULL; - return; - } - - if (!isset($this->libraries[$value->library])) { - $libSpec = H5PCore::libraryFromString($value->library); - $library = $this->h5pC->loadLibrary($libSpec['machineName'], $libSpec['majorVersion'], $libSpec['minorVersion']); - $library['semantics'] = $this->h5pC->loadLibrarySemantics($libSpec['machineName'], $libSpec['majorVersion'], $libSpec['minorVersion']); - $this->libraries[$value->library] = $library; - } - else { - $library = $this->libraries[$value->library]; - } - - // Validate parameters - $this->validateGroup($value->params, (object) array( - 'type' => 'group', - 'fields' => $library['semantics'], - ), FALSE); - - // Validate subcontent's metadata - if (isset($value->metadata)) { - $value->metadata = $this->validateMetadata($value->metadata); - } - - $validKeys = array('library', 'params', 'subContentId', 'metadata'); - if (isset($semantics->extraAttributes)) { - $validKeys = array_merge($validKeys, $semantics->extraAttributes); - } - - $this->filterParams($value, $validKeys); - if (isset($value->subContentId) && ! preg_match('/^\{?[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\}?$/', $value->subContentId)) { - unset($value->subContentId); - } - - // Find all dependencies for this library - $depKey = 'preloaded-' . $library['machineName']; - if (!isset($this->dependencies[$depKey])) { - $this->dependencies[$depKey] = array( - 'library' => $library, - 'type' => 'preloaded' - ); - - $this->nextWeight = $this->h5pC->findLibraryDependencies($this->dependencies, $library, $this->nextWeight); - $this->dependencies[$depKey]['weight'] = $this->nextWeight++; - } - } - - /** - * Check params for a whitelist of allowed properties - * - * @param array/object $params - * @param array $whitelist - */ - public function filterParams(&$params, $whitelist) { - foreach ($params as $key => $value) { - if (!in_array($key, $whitelist)) { - unset($params->{$key}); - } - } - } - - // XSS filters copied from drupal 7 common.inc. Some modifications done to - // replace Drupal one-liner functions with corresponding flat PHP. - - /** - * Filters HTML to prevent cross-site-scripting (XSS) vulnerabilities. - * - * Based on kses by Ulf Harnhammar, see http://sourceforge.net/projects/kses. - * For examples of various XSS attacks, see: http://ha.ckers.org/xss.html. - * - * This code does four things: - * - Removes characters and constructs that can trick browsers. - * - Makes sure all HTML entities are well-formed. - * - Makes sure all HTML tags and attributes are well-formed. - * - Makes sure no HTML tags contain URLs with a disallowed protocol (e.g. - * javascript:). - * - * @param $string - * The string with raw HTML in it. It will be stripped of everything that can - * cause an XSS attack. - * @param array $allowed_tags - * An array of allowed tags. - * - * @param bool $allowedStyles - * @return mixed|string An XSS safe version of $string, or an empty string if $string is not - * An XSS safe version of $string, or an empty string if $string is not - * valid UTF-8. - * @ingroup sanitation - */ - private function filter_xss($string, $allowed_tags = array('a', 'em', 'strong', 'cite', 'blockquote', 'code', 'ul', 'ol', 'li', 'dl', 'dt', 'dd'), $allowedStyles = FALSE) { - if (strlen($string) == 0) { - return $string; - } - // Only operate on valid UTF-8 strings. This is necessary to prevent cross - // site scripting issues on Internet Explorer 6. (Line copied from - // drupal_validate_utf8) - if (preg_match('/^./us', $string) != 1) { - return ''; - } - - $this->allowedStyles = $allowedStyles; - - // Store the text format. - $this->_filter_xss_split($allowed_tags, TRUE); - // Remove NULL characters (ignored by some browsers). - $string = str_replace(chr(0), '', $string); - // Remove Netscape 4 JS entities. - $string = preg_replace('%&\s*\{[^}]*(\}\s*;?|$)%', '', $string); - - // Defuse all HTML entities. - $string = str_replace('&', '&', $string); - // Change back only well-formed entities in our whitelist: - // Decimal numeric entities. - $string = preg_replace('/&#([0-9]+;)/', '&#\1', $string); - // Hexadecimal numeric entities. - $string = preg_replace('/&#[Xx]0*((?:[0-9A-Fa-f]{2})+;)/', '&#x\1', $string); - // Named entities. - $string = preg_replace('/&([A-Za-z][A-Za-z0-9]*;)/', '&\1', $string); - return preg_replace_callback('% - ( - <(?=[^a-zA-Z!/]) # a lone < - | # or - # a comment - | # or - <[^>]*(>|$) # a string that starts with a <, up until the > or the end of the string - | # or - > # just a > - )%x', array($this, '_filter_xss_split'), $string); - } - - /** - * Processes an HTML tag. - * - * @param $m - * An array with various meaning depending on the value of $store. - * If $store is TRUE then the array contains the allowed tags. - * If $store is FALSE then the array has one element, the HTML tag to process. - * @param bool $store - * Whether to store $m. - * @return string If the element isn't allowed, an empty string. Otherwise, the cleaned up - * If the element isn't allowed, an empty string. Otherwise, the cleaned up - * version of the HTML element. - */ - private function _filter_xss_split($m, $store = FALSE) { - static $allowed_html; - - if ($store) { - $allowed_html = array_flip($m); - return $allowed_html; - } - - $string = $m[1]; - - if (substr($string, 0, 1) != '<') { - // We matched a lone ">" character. - return '>'; - } - elseif (strlen($string) == 1) { - // We matched a lone "<" character. - return '<'; - } - - if (!preg_match('%^<\s*(/\s*)?([a-zA-Z0-9\-]+)\s*([^>]*)>?|()$%', $string, $matches)) { - // Seriously malformed. - return ''; - } - - $slash = trim($matches[1]); - $elem = &$matches[2]; - $attrList = &$matches[3]; - $comment = &$matches[4]; - - if ($comment) { - $elem = '!--'; - } - - if (!isset($allowed_html[strtolower($elem)])) { - // Disallowed HTML element. - return ''; - } - - if ($comment) { - return $comment; - } - - if ($slash != '') { - return ""; - } - - // Is there a closing XHTML slash at the end of the attributes? - $attrList = preg_replace('%(\s?)/\s*$%', '\1', $attrList, -1, $count); - $xhtml_slash = $count ? ' /' : ''; - - // Clean up attributes. - - $attr2 = implode(' ', $this->_filter_xss_attributes($attrList, (in_array($elem, self::$allowed_styleable_tags) ? $this->allowedStyles : FALSE))); - $attr2 = preg_replace('/[<>]/', '', $attr2); - $attr2 = strlen($attr2) ? ' ' . $attr2 : ''; - - return "<$elem$attr2$xhtml_slash>"; - } - - /** - * Processes a string of HTML attributes. - * - * @param $attr - * @param array|bool|object $allowedStyles - * @return array Cleaned up version of the HTML attributes. - * Cleaned up version of the HTML attributes. - */ - private function _filter_xss_attributes($attr, $allowedStyles = FALSE) { - $attrArr = array(); - $mode = 0; - $attrName = ''; - $skip = false; - - while (strlen($attr) != 0) { - // Was the last operation successful? - $working = 0; - switch ($mode) { - case 0: - // Attribute name, href for instance. - if (preg_match('/^([-a-zA-Z]+)/', $attr, $match)) { - $attrName = strtolower($match[1]); - $skip = ( - $attrName == 'style' || - substr($attrName, 0, 2) == 'on' || - substr($attrName, 0, 1) == '-' || - // Ignore long attributes to avoid unnecessary processing overhead. - strlen($attrName) > 96 - ); - $working = $mode = 1; - $attr = preg_replace('/^[-a-zA-Z]+/', '', $attr); - } - break; - - case 1: - // Equals sign or valueless ("selected"). - if (preg_match('/^\s*=\s*/', $attr)) { - $working = 1; $mode = 2; - $attr = preg_replace('/^\s*=\s*/', '', $attr); - break; - } - - if (preg_match('/^\s+/', $attr)) { - $working = 1; $mode = 0; - if (!$skip) { - $attrArr[] = $attrName; - } - $attr = preg_replace('/^\s+/', '', $attr); - } - break; - - case 2: - // Attribute value, a URL after href= for instance. - if (preg_match('/^"([^"]*)"(\s+|$)/', $attr, $match)) { - if ($allowedStyles && $attrName === 'style') { - // Allow certain styles - foreach ($allowedStyles as $pattern) { - if (preg_match($pattern, $match[1])) { - // All patterns are start to end patterns, and CKEditor adds one span per style - $attrArr[] = 'style="' . $match[1] . '"'; - break; - } - } - break; - } - - $thisVal = $this->filter_xss_bad_protocol($match[1]); - - if (!$skip) { - $attrArr[] = "$attrName=\"$thisVal\""; - } - $working = 1; - $mode = 0; - $attr = preg_replace('/^"[^"]*"(\s+|$)/', '', $attr); - break; - } - - if (preg_match("/^'([^']*)'(\s+|$)/", $attr, $match)) { - $thisVal = $this->filter_xss_bad_protocol($match[1]); - - if (!$skip) { - $attrArr[] = "$attrName='$thisVal'"; - } - $working = 1; $mode = 0; - $attr = preg_replace("/^'[^']*'(\s+|$)/", '', $attr); - break; - } - - if (preg_match("%^([^\s\"']+)(\s+|$)%", $attr, $match)) { - $thisVal = $this->filter_xss_bad_protocol($match[1]); - - if (!$skip) { - $attrArr[] = "$attrName=\"$thisVal\""; - } - $working = 1; $mode = 0; - $attr = preg_replace("%^[^\s\"']+(\s+|$)%", '', $attr); - } - break; - } - - if ($working == 0) { - // Not well formed; remove and try again. - $attr = preg_replace('/ - ^ - ( - "[^"]*("|$) # - a string that starts with a double quote, up until the next double quote or the end of the string - | # or - \'[^\']*(\'|$)| # - a string that starts with a quote, up until the next quote or the end of the string - | # or - \S # - a non-whitespace character - )* # any number of the above three - \s* # any number of whitespaces - /x', '', $attr); - $mode = 0; - } - } - - // The attribute list ends with a valueless attribute like "selected". - if ($mode == 1 && !$skip) { - $attrArr[] = $attrName; - } - return $attrArr; - } - -// TODO: Remove Drupal related stuff in docs. - - /** - * Processes an HTML attribute value and strips dangerous protocols from URLs. - * - * @param $string - * The string with the attribute value. - * @param bool $decode - * (deprecated) Whether to decode entities in the $string. Set to FALSE if the - * $string is in plain text, TRUE otherwise. Defaults to TRUE. This parameter - * is deprecated and will be removed in Drupal 8. To process a plain-text URI, - * call _strip_dangerous_protocols() or check_url() instead. - * @return string Cleaned up and HTML-escaped version of $string. - * Cleaned up and HTML-escaped version of $string. - */ - private function filter_xss_bad_protocol($string, $decode = TRUE) { - // Get the plain text representation of the attribute value (i.e. its meaning). - // @todo Remove the $decode parameter in Drupal 8, and always assume an HTML - // string that needs decoding. - if ($decode) { - $string = html_entity_decode($string, ENT_QUOTES, 'UTF-8'); - } - return htmlspecialchars($this->_strip_dangerous_protocols($string), ENT_QUOTES, 'UTF-8', FALSE); - } - - /** - * Strips dangerous protocols (e.g. 'javascript:') from a URI. - * - * This function must be called for all URIs within user-entered input prior - * to being output to an HTML attribute value. It is often called as part of - * check_url() or filter_xss(), but those functions return an HTML-encoded - * string, so this function can be called independently when the output needs to - * be a plain-text string for passing to t(), l(), drupal_attributes(), or - * another function that will call check_plain() separately. - * - * @param $uri - * A plain-text URI that might contain dangerous protocols. - * @return string A plain-text URI stripped of dangerous protocols. As with all plain-text - * A plain-text URI stripped of dangerous protocols. As with all plain-text - * strings, this return value must not be output to an HTML page without - * check_plain() being called on it. However, it can be passed to functions - * expecting plain-text strings. - * @see check_url() - */ - private function _strip_dangerous_protocols($uri) { - static $allowed_protocols; - - if (!isset($allowed_protocols)) { - $allowed_protocols = array_flip(array('ftp', 'http', 'https', 'mailto')); - } - - // Iteratively remove any invalid protocol found. - do { - $before = $uri; - $colonPos = strpos($uri, ':'); - if ($colonPos > 0) { - // We found a colon, possibly a protocol. Verify. - $protocol = substr($uri, 0, $colonPos); - // If a colon is preceded by a slash, question mark or hash, it cannot - // possibly be part of the URL scheme. This must be a relative URL, which - // inherits the (safe) protocol of the base document. - if (preg_match('![/?#]!', $protocol)) { - break; - } - // Check if this is a disallowed protocol. Per RFC2616, section 3.2.3 - // (URI Comparison) scheme comparison must be case-insensitive. - if (!isset($allowed_protocols[strtolower($protocol)])) { - $uri = substr($uri, $colonPos + 1); - } - } - } while ($before != $uri); - - return $uri; - } - - public function getMetadataSemantics() { - static $semantics; - - $cc_versions = array( - (object) array( - 'value' => '4.0', - 'label' => $this->h5pF->t('4.0 International') - ), - (object) array( - 'value' => '3.0', - 'label' => $this->h5pF->t('3.0 Unported') - ), - (object) array( - 'value' => '2.5', - 'label' => $this->h5pF->t('2.5 Generic') - ), - (object) array( - 'value' => '2.0', - 'label' => $this->h5pF->t('2.0 Generic') - ), - (object) array( - 'value' => '1.0', - 'label' => $this->h5pF->t('1.0 Generic') - ) - ); - - $semantics = array( - (object) array( - 'name' => 'title', - 'type' => 'text', - 'label' => $this->h5pF->t('Title'), - 'placeholder' => 'La Gioconda' - ), - (object) array( - 'name' => 'a11yTitle', - 'type' => 'text', - 'label' => $this->h5pF->t('Assistive Technologies label'), - 'optional' => TRUE, - ), - (object) array( - 'name' => 'license', - 'type' => 'select', - 'label' => $this->h5pF->t('License'), - 'default' => 'U', - 'options' => array( - (object) array( - 'value' => 'U', - 'label' => $this->h5pF->t('Undisclosed') - ), - (object) array( - 'type' => 'optgroup', - 'label' => $this->h5pF->t('Creative Commons'), - 'options' => array( - (object) array( - 'value' => 'CC BY', - 'label' => $this->h5pF->t('Attribution (CC BY)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-SA', - 'label' => $this->h5pF->t('Attribution-ShareAlike (CC BY-SA)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-ND', - 'label' => $this->h5pF->t('Attribution-NoDerivs (CC BY-ND)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC', - 'label' => $this->h5pF->t('Attribution-NonCommercial (CC BY-NC)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC-SA', - 'label' => $this->h5pF->t('Attribution-NonCommercial-ShareAlike (CC BY-NC-SA)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC-ND', - 'label' => $this->h5pF->t('Attribution-NonCommercial-NoDerivs (CC BY-NC-ND)'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC0 1.0', - 'label' => $this->h5pF->t('Public Domain Dedication (CC0)') - ), - (object) array( - 'value' => 'CC PDM', - 'label' => $this->h5pF->t('Public Domain Mark (PDM)') - ), - ) - ), - (object) array( - 'value' => 'GNU GPL', - 'label' => $this->h5pF->t('General Public License v3') - ), - (object) array( - 'value' => 'PD', - 'label' => $this->h5pF->t('Public Domain') - ), - (object) array( - 'value' => 'ODC PDDL', - 'label' => $this->h5pF->t('Public Domain Dedication and Licence') - ), - (object) array( - 'value' => 'C', - 'label' => $this->h5pF->t('Copyright') - ) - ) - ), - (object) array( - 'name' => 'licenseVersion', - 'type' => 'select', - 'label' => $this->h5pF->t('License Version'), - 'options' => $cc_versions, - 'optional' => TRUE - ), - (object) array( - 'name' => 'yearFrom', - 'type' => 'number', - 'label' => $this->h5pF->t('Years (from)'), - 'placeholder' => '1991', - 'min' => '-9999', - 'max' => '9999', - 'optional' => TRUE - ), - (object) array( - 'name' => 'yearTo', - 'type' => 'number', - 'label' => $this->h5pF->t('Years (to)'), - 'placeholder' => '1992', - 'min' => '-9999', - 'max' => '9999', - 'optional' => TRUE - ), - (object) array( - 'name' => 'source', - 'type' => 'text', - 'label' => $this->h5pF->t('Source'), - 'placeholder' => 'https://', - 'optional' => TRUE - ), - (object) array( - 'name' => 'authors', - 'type' => 'list', - 'field' => (object) array ( - 'name' => 'author', - 'type' => 'group', - 'fields'=> array( - (object) array( - 'label' => $this->h5pF->t("Author's name"), - 'name' => 'name', - 'optional' => TRUE, - 'type' => 'text' - ), - (object) array( - 'name' => 'role', - 'type' => 'select', - 'label' => $this->h5pF->t("Author's role"), - 'default' => 'Author', - 'options' => array( - (object) array( - 'value' => 'Author', - 'label' => $this->h5pF->t('Author') - ), - (object) array( - 'value' => 'Editor', - 'label' => $this->h5pF->t('Editor') - ), - (object) array( - 'value' => 'Licensee', - 'label' => $this->h5pF->t('Licensee') - ), - (object) array( - 'value' => 'Originator', - 'label' => $this->h5pF->t('Originator') - ) - ) - ) - ) - ) - ), - (object) array( - 'name' => 'licenseExtras', - 'type' => 'text', - 'widget' => 'textarea', - 'label' => $this->h5pF->t('License Extras'), - 'optional' => TRUE, - 'description' => $this->h5pF->t('Any additional information about the license') - ), - (object) array( - 'name' => 'changes', - 'type' => 'list', - 'field' => (object) array( - 'name' => 'change', - 'type' => 'group', - 'label' => $this->h5pF->t('Changelog'), - 'fields' => array( - (object) array( - 'name' => 'date', - 'type' => 'text', - 'label' => $this->h5pF->t('Date'), - 'optional' => TRUE - ), - (object) array( - 'name' => 'author', - 'type' => 'text', - 'label' => $this->h5pF->t('Changed by'), - 'optional' => TRUE - ), - (object) array( - 'name' => 'log', - 'type' => 'text', - 'widget' => 'textarea', - 'label' => $this->h5pF->t('Description of change'), - 'placeholder' => $this->h5pF->t('Photo cropped, text changed, etc.'), - 'optional' => TRUE - ) - ) - ) - ), - (object) array ( - 'name' => 'authorComments', - 'type' => 'text', - 'widget' => 'textarea', - 'label' => $this->h5pF->t('Author comments'), - 'description' => $this->h5pF->t('Comments for the editor of the content (This text will not be published as a part of copyright info)'), - 'optional' => TRUE - ), - (object) array( - 'name' => 'contentType', - 'type' => 'text', - 'widget' => 'none' - ), - (object) array( - 'name' => 'defaultLanguage', - 'type' => 'text', - 'widget' => 'none' - ) - ); - - return $semantics; - } - - public function getCopyrightSemantics() { - static $semantics; - - if ($semantics === NULL) { - $cc_versions = array( - (object) array( - 'value' => '4.0', - 'label' => $this->h5pF->t('4.0 International') - ), - (object) array( - 'value' => '3.0', - 'label' => $this->h5pF->t('3.0 Unported') - ), - (object) array( - 'value' => '2.5', - 'label' => $this->h5pF->t('2.5 Generic') - ), - (object) array( - 'value' => '2.0', - 'label' => $this->h5pF->t('2.0 Generic') - ), - (object) array( - 'value' => '1.0', - 'label' => $this->h5pF->t('1.0 Generic') - ) - ); - - $semantics = (object) array( - 'name' => 'copyright', - 'type' => 'group', - 'label' => $this->h5pF->t('Copyright information'), - 'fields' => array( - (object) array( - 'name' => 'title', - 'type' => 'text', - 'label' => $this->h5pF->t('Title'), - 'placeholder' => 'La Gioconda', - 'optional' => TRUE - ), - (object) array( - 'name' => 'author', - 'type' => 'text', - 'label' => $this->h5pF->t('Author'), - 'placeholder' => 'Leonardo da Vinci', - 'optional' => TRUE - ), - (object) array( - 'name' => 'year', - 'type' => 'text', - 'label' => $this->h5pF->t('Year(s)'), - 'placeholder' => '1503 - 1517', - 'optional' => TRUE - ), - (object) array( - 'name' => 'source', - 'type' => 'text', - 'label' => $this->h5pF->t('Source'), - 'placeholder' => 'http://en.wikipedia.org/wiki/Mona_Lisa', - 'optional' => true, - 'regexp' => (object) array( - 'pattern' => '^http[s]?://.+', - 'modifiers' => 'i' - ) - ), - (object) array( - 'name' => 'license', - 'type' => 'select', - 'label' => $this->h5pF->t('License'), - 'default' => 'U', - 'options' => array( - (object) array( - 'value' => 'U', - 'label' => $this->h5pF->t('Undisclosed') - ), - (object) array( - 'value' => 'CC BY', - 'label' => $this->h5pF->t('Attribution'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-SA', - 'label' => $this->h5pF->t('Attribution-ShareAlike'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-ND', - 'label' => $this->h5pF->t('Attribution-NoDerivs'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC', - 'label' => $this->h5pF->t('Attribution-NonCommercial'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC-SA', - 'label' => $this->h5pF->t('Attribution-NonCommercial-ShareAlike'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'CC BY-NC-ND', - 'label' => $this->h5pF->t('Attribution-NonCommercial-NoDerivs'), - 'versions' => $cc_versions - ), - (object) array( - 'value' => 'GNU GPL', - 'label' => $this->h5pF->t('General Public License'), - 'versions' => array( - (object) array( - 'value' => 'v3', - 'label' => $this->h5pF->t('Version 3') - ), - (object) array( - 'value' => 'v2', - 'label' => $this->h5pF->t('Version 2') - ), - (object) array( - 'value' => 'v1', - 'label' => $this->h5pF->t('Version 1') - ) - ) - ), - (object) array( - 'value' => 'PD', - 'label' => $this->h5pF->t('Public Domain'), - 'versions' => array( - (object) array( - 'value' => '-', - 'label' => '-' - ), - (object) array( - 'value' => 'CC0 1.0', - 'label' => $this->h5pF->t('CC0 1.0 Universal') - ), - (object) array( - 'value' => 'CC PDM', - 'label' => $this->h5pF->t('Public Domain Mark') - ) - ) - ), - (object) array( - 'value' => 'C', - 'label' => $this->h5pF->t('Copyright') - ) - ) - ), - (object) array( - 'name' => 'version', - 'type' => 'select', - 'label' => $this->h5pF->t('License Version'), - 'options' => array() - ) - ) - ); - } - - return $semantics; - } -} diff --git a/h5p/h5plib/v126/joubel/core/images/h5p.svg b/h5p/h5plib/v126/joubel/core/images/h5p.svg deleted file mode 100644 index 9ef61ae8fb95c..0000000000000 --- a/h5p/h5plib/v126/joubel/core/images/h5p.svg +++ /dev/null @@ -1,16 +0,0 @@ - - - - - - - - - diff --git a/h5p/h5plib/v126/joubel/core/images/throbber.gif b/h5p/h5plib/v126/joubel/core/images/throbber.gif deleted file mode 100644 index acddb91dacf53..0000000000000 Binary files a/h5p/h5plib/v126/joubel/core/images/throbber.gif and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/core/js/h5p-action-bar.js b/h5p/h5plib/v126/joubel/core/js/h5p-action-bar.js deleted file mode 100644 index 88cb84825fbc8..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/h5p-action-bar.js +++ /dev/null @@ -1,107 +0,0 @@ -/** - * @class - * @augments H5P.EventDispatcher - * @param {Object} displayOptions - * @param {boolean} displayOptions.export Triggers the display of the 'Download' button - * @param {boolean} displayOptions.copyright Triggers the display of the 'Copyright' button - * @param {boolean} displayOptions.embed Triggers the display of the 'Embed' button - * @param {boolean} displayOptions.icon Triggers the display of the 'H5P icon' link - */ -H5P.ActionBar = (function ($, EventDispatcher) { - "use strict"; - - function ActionBar(displayOptions) { - EventDispatcher.call(this); - - /** @alias H5P.ActionBar# */ - var self = this; - - var hasActions = false; - - // Create action bar - var $actions = H5P.jQuery('
    '); - - /** - * Helper for creating action bar buttons. - * - * @private - * @param {string} type - * @param {string} customClass Instead of type class - */ - var addActionButton = function (type, customClass) { - /** - * Handles selection of action - */ - var handler = function () { - self.trigger(type); - }; - - const $actionList = H5P.jQuery('
  • ', { - 'class': 'h5p-button h5p-noselect h5p-' + (customClass ? customClass : type), - appendTo: $actions - }); - - const $actionButton = H5P.jQuery(''); - H5PLibraryDetails.$next = $(''); - - H5PLibraryDetails.$previous.on('click', function () { - if (H5PLibraryDetails.$previous.hasClass('disabled')) { - return; - } - - H5PLibraryDetails.currentPage--; - H5PLibraryDetails.updatePager(); - H5PLibraryDetails.createContentTable(); - }); - - H5PLibraryDetails.$next.on('click', function () { - if (H5PLibraryDetails.$next.hasClass('disabled')) { - return; - } - - H5PLibraryDetails.currentPage++; - H5PLibraryDetails.updatePager(); - H5PLibraryDetails.createContentTable(); - }); - - // This is the Page x of y widget: - H5PLibraryDetails.$pagerInfo = $(''); - - H5PLibraryDetails.$pager = $('
    ').append(H5PLibraryDetails.$previous, H5PLibraryDetails.$pagerInfo, H5PLibraryDetails.$next); - H5PLibraryDetails.$content.append(H5PLibraryDetails.$pager); - - H5PLibraryDetails.$pagerInfo.on('click', function () { - var width = H5PLibraryDetails.$pagerInfo.innerWidth(); - H5PLibraryDetails.$pagerInfo.hide(); - - // User has updated the pageNumber - var pageNumerUpdated = function () { - var newPageNum = $gotoInput.val()-1; - var intRegex = /^\d+$/; - - $goto.remove(); - H5PLibraryDetails.$pagerInfo.css({display: 'inline-block'}); - - // Check if input value is valid, and that it has actually changed - if (!(intRegex.test(newPageNum) && newPageNum >= 0 && newPageNum < H5PLibraryDetails.getNumPages() && newPageNum != H5PLibraryDetails.currentPage)) { - return; - } - - H5PLibraryDetails.currentPage = newPageNum; - H5PLibraryDetails.updatePager(); - H5PLibraryDetails.createContentTable(); - }; - - // We create an input box where the user may type in the page number - // he wants to be displayed. - // Reson for doing this is when user has ten-thousands of elements in list, - // this is the easiest way of getting to a specified page - var $gotoInput = $('', { - type: 'number', - min : 1, - max: H5PLibraryDetails.getNumPages(), - on: { - // Listen to blur, and the enter-key: - 'blur': pageNumerUpdated, - 'keyup': function (event) { - if (event.keyCode === 13) { - pageNumerUpdated(); - } - } - } - }).css({width: width}); - var $goto = $('', { - 'class': 'h5p-pager-goto' - }).css({width: width}).append($gotoInput).insertAfter(H5PLibraryDetails.$pagerInfo); - - $gotoInput.focus(); - }); - - H5PLibraryDetails.updatePager(); - }; - - /** - * Calculates number of pages - */ - H5PLibraryDetails.getNumPages = function () { - return Math.ceil(H5PLibraryDetails.currentContent.length / H5PLibraryDetails.PAGER_SIZE); - }; - - /** - * Update the pager text, and enables/disables the next and previous buttons as needed - */ - H5PLibraryDetails.updatePager = function () { - H5PLibraryDetails.$pagerInfo.css({display: 'inline-block'}); - - if (H5PLibraryDetails.getNumPages() > 0) { - var message = H5PUtils.translateReplace(H5PLibraryDetails.library.translations.pageXOfY, { - '$x': (H5PLibraryDetails.currentPage+1), - '$y': H5PLibraryDetails.getNumPages() - }); - H5PLibraryDetails.$pagerInfo.html(message); - } - else { - H5PLibraryDetails.$pagerInfo.html(''); - } - - H5PLibraryDetails.$previous.toggleClass('disabled', H5PLibraryDetails.currentPage <= 0); - H5PLibraryDetails.$next.toggleClass('disabled', H5PLibraryDetails.currentContent.length < (H5PLibraryDetails.currentPage+1)*H5PLibraryDetails.PAGER_SIZE); - }; - - /** - * Creates the search element - */ - H5PLibraryDetails.createSearchElement = function () { - - H5PLibraryDetails.$search = $(''); - - var performSeach = function () { - var searchString = $('.h5p-content-search > input').val(); - - // If search string same as previous, just do nothing - if (H5PLibraryDetails.currentFilter === searchString) { - return; - } - - if (searchString.trim().length === 0) { - // If empty search, use the complete list - H5PLibraryDetails.currentContent = H5PLibraryDetails.library.content; - } - else if (H5PLibraryDetails.filterCache[searchString]) { - // If search is cached, no need to filter - H5PLibraryDetails.currentContent = H5PLibraryDetails.filterCache[searchString]; - } - else { - var listToFilter = H5PLibraryDetails.library.content; - - // Check if we can filter the already filtered results (for performance) - if (searchString.length > 1 && H5PLibraryDetails.currentFilter === searchString.substr(0, H5PLibraryDetails.currentFilter.length)) { - listToFilter = H5PLibraryDetails.currentContent; - } - H5PLibraryDetails.currentContent = $.grep(listToFilter, function (content) { - return content.title && content.title.match(new RegExp(searchString, 'i')); - }); - } - - H5PLibraryDetails.currentFilter = searchString; - // Cache the current result - H5PLibraryDetails.filterCache[searchString] = H5PLibraryDetails.currentContent; - H5PLibraryDetails.currentPage = 0; - H5PLibraryDetails.createContentTable(); - - // Display search results: - if (H5PLibraryDetails.$searchResults) { - H5PLibraryDetails.$searchResults.remove(); - } - if (searchString.trim().length > 0) { - H5PLibraryDetails.$searchResults = $('' + H5PLibraryDetails.currentContent.length + ' hits on ' + H5PLibraryDetails.currentFilter + ''); - H5PLibraryDetails.$search.append(H5PLibraryDetails.$searchResults); - } - H5PLibraryDetails.updatePager(); - }; - - var inputTimer; - $('input', H5PLibraryDetails.$search).on('change keypress paste input', function () { - // Here we start the filtering - // We wait at least 500 ms after last input to perform search - if (inputTimer) { - clearTimeout(inputTimer); - } - - inputTimer = setTimeout( function () { - performSeach(); - }, 500); - }); - - H5PLibraryDetails.$content.append(H5PLibraryDetails.$search); - }; - - /** - * Creates the page size selector - */ - H5PLibraryDetails.createPageSizeSelector = function () { - H5PLibraryDetails.$search.append('
    ' + H5PLibraryDetails.library.translations.pageSizeSelectorLabel + ':102050100200
    '); - - // Listen to clicks on the page size selector: - $('.h5p-admin-pager-size-selector > span', H5PLibraryDetails.$search).on('click', function () { - H5PLibraryDetails.PAGER_SIZE = $(this).data('page-size'); - $('.h5p-admin-pager-size-selector > span', H5PLibraryDetails.$search).removeClass('selected'); - $(this).addClass('selected'); - H5PLibraryDetails.currentPage = 0; - H5PLibraryDetails.createContentTable(); - H5PLibraryDetails.updatePager(); - }); - }; - - // Initialize me: - $(document).ready(function () { - if (!H5PLibraryDetails.initialized) { - H5PLibraryDetails.initialized = true; - H5PLibraryDetails.init(); - } - }); - -})(H5P.jQuery); diff --git a/h5p/h5plib/v126/joubel/core/js/h5p-library-list.js b/h5p/h5plib/v126/joubel/core/js/h5p-library-list.js deleted file mode 100644 index 344b7367234e8..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/h5p-library-list.js +++ /dev/null @@ -1,140 +0,0 @@ -/* global H5PAdminIntegration H5PUtils */ -var H5PLibraryList = H5PLibraryList || {}; - -(function ($) { - - /** - * Initializing - */ - H5PLibraryList.init = function () { - var $adminContainer = H5P.jQuery(H5PAdminIntegration.containerSelector).html(''); - - var libraryList = H5PAdminIntegration.libraryList; - if (libraryList.notCached) { - $adminContainer.append(H5PUtils.getRebuildCache(libraryList.notCached)); - } - - // Create library list - $adminContainer.append(H5PLibraryList.createLibraryList(H5PAdminIntegration.libraryList)); - }; - - /** - * Create the library list - * - * @param {object} libraries List of libraries and headers - */ - H5PLibraryList.createLibraryList = function (libraries) { - var t = H5PAdminIntegration.l10n; - if (libraries.listData === undefined || libraries.listData.length === 0) { - return $('
    ' + t.NA + '
    '); - } - - // Create table - var $table = H5PUtils.createTable(libraries.listHeaders); - $table.addClass('libraries'); - - // Add libraries - $.each (libraries.listData, function (index, library) { - var $libraryRow = H5PUtils.createTableRow([ - library.title, - '', - { - text: library.numContent, - class: 'h5p-admin-center' - }, - { - text: library.numContentDependencies, - class: 'h5p-admin-center' - }, - { - text: library.numLibraryDependencies, - class: 'h5p-admin-center' - }, - '
    ' + - '' + - (library.detailsUrl ? '' : '') + - (library.deleteUrl ? '' : '') + - '
    ' - ]); - - H5PLibraryList.addRestricted($('.h5p-admin-restricted', $libraryRow), library.restrictedUrl, library.restricted); - - var hasContent = !(library.numContent === '' || library.numContent === 0); - if (library.upgradeUrl === null) { - $('.h5p-admin-upgrade-library', $libraryRow).remove(); - } - else if (library.upgradeUrl === false || !hasContent) { - $('.h5p-admin-upgrade-library', $libraryRow).attr('disabled', true); - } - else { - $('.h5p-admin-upgrade-library', $libraryRow).attr('title', t.upgradeLibrary).click(function () { - window.location.href = library.upgradeUrl; - }); - } - - // Open details view when clicked - $('.h5p-admin-view-library', $libraryRow).on('click', function () { - window.location.href = library.detailsUrl; - }); - - var $deleteButton = $('.h5p-admin-delete-library', $libraryRow); - if (libraries.notCached !== undefined || - hasContent || - (library.numContentDependencies !== '' && - library.numContentDependencies !== 0) || - (library.numLibraryDependencies !== '' && - library.numLibraryDependencies !== 0)) { - // Disabled delete if content. - $deleteButton.attr('disabled', true); - } - else { - // Go to delete page om click. - $deleteButton.attr('title', t.deleteLibrary).on('click', function () { - window.location.href = library.deleteUrl; - }); - } - - $table.append($libraryRow); - }); - - return $table; - }; - - H5PLibraryList.addRestricted = function ($checkbox, url, selected) { - if (selected === null) { - $checkbox.remove(); - } - else { - $checkbox.change(function () { - $checkbox.attr('disabled', true); - - $.ajax({ - dataType: 'json', - url: url, - cache: false - }).fail(function () { - $checkbox.attr('disabled', false); - - // Reset - $checkbox.attr('checked', !$checkbox.is(':checked')); - }).done(function (result) { - url = result.url; - $checkbox.attr('disabled', false); - }); - }); - - if (selected) { - $checkbox.attr('checked', true); - } - } - }; - - // Initialize me: - $(document).ready(function () { - if (!H5PLibraryList.initialized) { - H5PLibraryList.initialized = true; - H5PLibraryList.init(); - } - }); - -})(H5P.jQuery); diff --git a/h5p/h5plib/v126/joubel/core/js/h5p-resizer.js b/h5p/h5plib/v126/joubel/core/js/h5p-resizer.js deleted file mode 100644 index ed78724ec1a92..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/h5p-resizer.js +++ /dev/null @@ -1,131 +0,0 @@ -// H5P iframe Resizer -(function () { - if (!window.postMessage || !window.addEventListener || window.h5pResizerInitialized) { - return; // Not supported - } - window.h5pResizerInitialized = true; - - // Map actions to handlers - var actionHandlers = {}; - - /** - * Prepare iframe resize. - * - * @private - * @param {Object} iframe Element - * @param {Object} data Payload - * @param {Function} respond Send a response to the iframe - */ - actionHandlers.hello = function (iframe, data, respond) { - // Make iframe responsive - iframe.style.width = '100%'; - - // Bugfix for Chrome: Force update of iframe width. If this is not done the - // document size may not be updated before the content resizes. - iframe.getBoundingClientRect(); - - // Tell iframe that it needs to resize when our window resizes - var resize = function () { - if (iframe.contentWindow) { - // Limit resize calls to avoid flickering - respond('resize'); - } - else { - // Frame is gone, unregister. - window.removeEventListener('resize', resize); - } - }; - window.addEventListener('resize', resize, false); - - // Respond to let the iframe know we can resize it - respond('hello'); - }; - - /** - * Prepare iframe resize. - * - * @private - * @param {Object} iframe Element - * @param {Object} data Payload - * @param {Function} respond Send a response to the iframe - */ - actionHandlers.prepareResize = function (iframe, data, respond) { - // Do not resize unless page and scrolling differs - if (iframe.clientHeight !== data.scrollHeight || - data.scrollHeight !== data.clientHeight) { - - // Reset iframe height, in case content has shrinked. - iframe.style.height = data.clientHeight + 'px'; - respond('resizePrepared'); - } - }; - - /** - * Resize parent and iframe to desired height. - * - * @private - * @param {Object} iframe Element - * @param {Object} data Payload - * @param {Function} respond Send a response to the iframe - */ - actionHandlers.resize = function (iframe, data) { - // Resize iframe so all content is visible. Use scrollHeight to make sure we get everything - iframe.style.height = data.scrollHeight + 'px'; - }; - - /** - * Keyup event handler. Exits full screen on escape. - * - * @param {Event} event - */ - var escape = function (event) { - if (event.keyCode === 27) { - exitFullScreen(); - } - }; - - // Listen for messages from iframes - window.addEventListener('message', function receiveMessage(event) { - if (event.data.context !== 'h5p') { - return; // Only handle h5p requests. - } - - // Find out who sent the message - var iframe, iframes = document.getElementsByTagName('iframe'); - for (var i = 0; i < iframes.length; i++) { - if (iframes[i].contentWindow === event.source) { - iframe = iframes[i]; - break; - } - } - - if (!iframe) { - return; // Cannot find sender - } - - // Find action handler handler - if (actionHandlers[event.data.action]) { - actionHandlers[event.data.action](iframe, event.data, function respond(action, data) { - if (data === undefined) { - data = {}; - } - data.action = action; - data.context = 'h5p'; - event.source.postMessage(data, event.origin); - }); - } - }, false); - - // Let h5p iframes know we're ready! - var iframes = document.getElementsByTagName('iframe'); - var ready = { - context: 'h5p', - action: 'ready' - }; - for (var i = 0; i < iframes.length; i++) { - if (iframes[i].src.indexOf('h5p') !== -1) { - iframes[i].contentWindow.postMessage(ready, '*'); - } - } - -})(); diff --git a/h5p/h5plib/v126/joubel/core/js/h5p-tooltip.js b/h5p/h5plib/v126/joubel/core/js/h5p-tooltip.js deleted file mode 100644 index fad01c58ce61f..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/h5p-tooltip.js +++ /dev/null @@ -1,217 +0,0 @@ -/*global H5P*/ -H5P.Tooltip = (function () { - 'use strict'; - - /** - * Create an accessible tooltip - * - * @param {HTMLElement} triggeringElement The element that should trigger the tooltip - * @param {Object} options Options for tooltip - * @param {String} options.text The text to be displayed in the tooltip - * If not set, will attempt to set text = aria-label of triggeringElement - * @param {String[]} options.classes Extra css classes for the tooltip - * @param {Boolean} options.ariaHidden Whether the hover should be read by screen readers or not (default: true) - * @param {String} options.position Where the tooltip should appear in relation to the - * triggeringElement. Accepted positions are "top" (default), "left", "right" and "bottom" - * - * @constructor - */ - function Tooltip(triggeringElement, options) { - - // Make sure tooltips have unique id - H5P.Tooltip.uniqueId += 1; - const tooltipId = 'h5p-tooltip-' + H5P.Tooltip.uniqueId; - - // Default options - options = options || {}; - options.classes = options.classes || []; - options.ariaHidden = options.ariaHidden || true; - - // Initiate state - let hover = false; - let focus = false; - - // Function used by the escape listener - const escapeFunction = function (e) { - if (e.key === 'Escape') { - tooltip.classList.remove('h5p-tooltip-visible'); - } - } - - // Create element - const tooltip = document.createElement('div'); - - tooltip.classList.add('h5p-tooltip'); - tooltip.id = tooltipId; - tooltip.role = 'tooltip'; - tooltip.innerHTML = options.text || triggeringElement.getAttribute('aria-label') || ''; - tooltip.setAttribute('aria-hidden', options.ariaHidden); - tooltip.classList.add(...options.classes); - - triggeringElement.appendChild(tooltip); - - // Set the initial position based on options.position - switch (options.position) { - case 'left': - tooltip.classList.add('h5p-tooltip-left'); - break; - case 'right': - tooltip.classList.add('h5p-tooltip-right'); - break; - case 'bottom': - tooltip.classList.add('h5p-tooltip-bottom'); - break; - default: - options.position = 'top'; - } - - // Aria-describedby will override aria-hidden - if (!options.ariaHidden) { - triggeringElement.setAttribute('aria-describedby', tooltipId); - } - - // Add event listeners to triggeringElement - triggeringElement.addEventListener('mouseenter', function () { - showTooltip(true); - }); - triggeringElement.addEventListener('mouseleave', function () { - hideTooltip(true); - }); - triggeringElement.addEventListener('focusin', function () { - showTooltip(false); - }); - triggeringElement.addEventListener('focusout', function () { - hideTooltip(false); - }); - - // Prevent clicks on the tooltip from triggering onClick listeners on the triggeringElement - tooltip.addEventListener('click', function (event) { - event.stopPropagation(); - }); - - // Use a mutation observer to listen for aria-label being - // changed for the triggering element. If so, update the tooltip. - // Mutation observer will be used even if the original elements - // doesn't have any aria-label. - new MutationObserver(function (mutations) { - const ariaLabel = mutations[0].target.getAttribute('aria-label'); - if (ariaLabel) { - tooltip.innerHTML = options.text || ariaLabel; - } - }).observe(triggeringElement, { - attributes: true, - attributeFilter: ['aria-label'], - }); - - // Use intersection observer to adjust the tooltip if it is not completely visible - new IntersectionObserver(function (entries) { - entries.forEach((entry) => { - const target = entry.target; - const positionClass = 'h5p-tooltip-' + options.position; - - // Stop adjusting when hidden (to prevent a false positive next time) - if (entry.intersectionRatio === 0) { - ['h5p-tooltip-down', 'h5p-tooltip-left', 'h5p-tooltip-right'] - .forEach(function (adjustmentClass) { - if (adjustmentClass !== positionClass) { - target.classList.remove(adjustmentClass); - } - }); - } - // Adjust if not completely visible when meant to be - else if (entry.intersectionRatio < 1 && (hover || focus)) { - const targetRect = entry.boundingClientRect; - const intersectionRect = entry.intersectionRect; - - // Going out of screen on left side - if (intersectionRect.left > targetRect.left) { - target.classList.add('h5p-tooltip-right'); - target.classList.remove(positionClass); - } - // Going out of screen on right side - else if (intersectionRect.right < targetRect.right) { - target.classList.add('h5p-tooltip-left'); - target.classList.remove(positionClass); - } - - // going out of top of screen - if (intersectionRect.top > targetRect.top) { - target.classList.add('h5p-tooltip-down'); - target.classList.remove(positionClass); - } - // going out of bottom of screen - else if (intersectionRect.bottom < targetRect.bottom) { - target.classList.add('h5p-tooltip-up'); - target.classList.remove(positionClass); - } - } - }); - }).observe(tooltip); - - /** - * Makes the tooltip visible and activates it's functionality - * - * @param {Boolean} triggeredByHover True if triggered by mouse, false if triggered by focus - */ - const showTooltip = function (triggeredByHover) { - if (triggeredByHover) { - hover = true; - } - else { - focus = true; - } - - tooltip.classList.add('h5p-tooltip-visible'); - - // Add listener to iframe body, as esc keypress would not be detected otherwise - document.body.addEventListener('keydown', escapeFunction, true); - } - - /** - * Hides the tooltip and removes listeners - * - * @param {Boolean} triggeredByHover True if triggered by mouse, false if triggered by focus - */ - const hideTooltip = function (triggeredByHover) { - if (triggeredByHover) { - hover = false; - } - else { - focus = false; - } - - // Only hide tooltip if neither hovered nor focused - if (!hover && !focus) { - tooltip.classList.remove('h5p-tooltip-visible'); - - // Remove iframe body listener - document.body.removeEventListener('keydown', escapeFunction, true); - } - } - - /** - * Change the text displayed by the tooltip - * - * @param {String} text The new text to be displayed - * Set to null to use aria-label of triggeringElement instead - */ - this.setText = function (text) { - options.text = text; - tooltip.innerHTML = options.text || triggeringElement.getAttribute('aria-label') || ''; - }; - - /** - * Retrieve tooltip - * - * @return {HTMLElement} - */ - this.getElement = function () { - return tooltip; - }; - } - - return Tooltip; - -})(); - -H5P.Tooltip.uniqueId = -1; diff --git a/h5p/h5plib/v126/joubel/core/js/h5p-utils.js b/h5p/h5plib/v126/joubel/core/js/h5p-utils.js deleted file mode 100644 index b5aa3334e0e96..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/h5p-utils.js +++ /dev/null @@ -1,506 +0,0 @@ -/* global H5PAdminIntegration*/ -var H5PUtils = H5PUtils || {}; - -(function ($) { - /** - * Generic function for creating a table including the headers - * - * @param {array} headers List of headers - */ - H5PUtils.createTable = function (headers) { - var $table = $('
    '); - - if (headers) { - var $thead = $(''); - var $tr = $(''); - - $.each(headers, function (index, value) { - if (!(value instanceof Object)) { - value = { - html: value - }; - } - - $('', value).appendTo($tr); - }); - - $table.append($thead.append($tr)); - } - - return $table; - }; - - /** - * Generic function for creating a table row - * - * @param {array} rows Value list. Object name is used as class name in - */ - H5PUtils.createTableRow = function (rows) { - var $tr = $(''); - - $.each(rows, function (index, value) { - if (!(value instanceof Object)) { - value = { - html: value - }; - } - - $('', value).appendTo($tr); - }); - - return $tr; - }; - - /** - * Generic function for creating a field containing label and value - * - * @param {string} label The label displayed in front of the value - * @param {string} value The value - */ - H5PUtils.createLabeledField = function (label, value) { - var $field = $('
    '); - - $field.append('
    ' + label + '
    '); - $field.append('
    ' + value + '
    '); - - return $field; - }; - - /** - * Replaces placeholder fields in translation strings - * - * @param {string} template The translation template string in the following format: "$name is a $sex" - * @param {array} replacors An js object with key and values. Eg: {'$name': 'Frode', '$sex': 'male'} - */ - H5PUtils.translateReplace = function (template, replacors) { - $.each(replacors, function (key, value) { - template = template.replace(new RegExp('\\'+key, 'g'), value); - }); - return template; - }; - - /** - * Get throbber with given text. - * - * @param {String} text - * @returns {$} - */ - H5PUtils.throbber = function (text) { - return $('
    ', { - class: 'h5p-throbber', - text: text - }); - }; - - /** - * Makes it possbile to rebuild all content caches from admin UI. - * @param {Object} notCached - * @returns {$} - */ - H5PUtils.getRebuildCache = function (notCached) { - var $container = $('

    ' + notCached.message + '

    ' + notCached.progress + '

    '); - var $button = $('').appendTo($container).click(function () { - var $spinner = $('
    ', {class: 'h5p-spinner'}).replaceAll($button); - var parts = ['|', '/', '-', '\\']; - var current = 0; - var spinning = setInterval(function () { - $spinner.text(parts[current]); - current++; - if (current === parts.length) current = 0; - }, 100); - - var $counter = $container.find('.progress'); - var build = function () { - $.post(notCached.url, function (left) { - if (left === '0') { - clearInterval(spinning); - $container.remove(); - location.reload(); - } - else { - var counter = $counter.text().split(' '); - counter[0] = left; - $counter.text(counter.join(' ')); - build(); - } - }); - }; - build(); - }); - - return $container; - }; - - /** - * Generic table class with useful helpers. - * - * @class - * @param {Object} classes - * Custom html classes to use on elements. - * e.g. {tableClass: 'fixed'}. - */ - H5PUtils.Table = function (classes) { - var numCols; - var sortByCol; - var $sortCol; - var sortCol; - var sortDir; - - // Create basic table - var tableOptions = {}; - if (classes.table !== undefined) { - tableOptions['class'] = classes.table; - } - var $table = $('', tableOptions); - var $thead = $('').appendTo($table); - var $tfoot = $('').appendTo($table); - var $tbody = $('').appendTo($table); - - /** - * Add columns to given table row. - * - * @private - * @param {jQuery} $tr Table row - * @param {(String|Object)} col Column properties - * @param {Number} id Used to seperate the columns - */ - var addCol = function ($tr, col, id) { - var options = { - on: {} - }; - - if (!(col instanceof Object)) { - options.text = col; - } - else { - if (col.text !== undefined) { - options.text = col.text; - } - if (col.class !== undefined) { - options.class = col.class; - } - - if (sortByCol !== undefined && col.sortable === true) { - // Make sortable - options.role = 'button'; - options.tabIndex = 0; - - // This is the first sortable column, use as default sort - if (sortCol === undefined) { - sortCol = id; - sortDir = 0; - } - - // This is the sort column - if (sortCol === id) { - options['class'] = 'h5p-sort'; - if (sortDir === 1) { - options['class'] += ' h5p-reverse'; - } - } - - options.on.click = function () { - sort($th, id); - }; - options.on.keypress = function (event) { - if ((event.charCode || event.keyCode) === 32) { // Space - sort($th, id); - } - }; - } - } - - // Append - var $th = $(''); - var $tr = $('').appendTo($newThead); - for (var i = 0; i < cols.length; i++) { - addCol($tr, cols[i], i); - } - - // Update DOM - $thead.replaceWith($newThead); - $thead = $newThead; - }; - - /** - * Set table rows. - * - * @public - * @param {Array} rows Table rows with cols: [[1,'hello',3],[2,'asd',6]] - */ - this.setRows = function (rows) { - var $newTbody = $(''); - - for (var i = 0; i < rows.length; i++) { - var $tr = $('').appendTo($newTbody); - - for (var j = 0; j < rows[i].length; j++) { - $(''); - var $tr = $('').appendTo($newTbody); - $(''); - var $tr = $('').appendTo($newTfoot); - $('
    ', options).appendTo($tr); - if (sortCol === id) { - $sortCol = $th; // Default sort column - } - }; - - /** - * Updates the UI when a column header has been clicked. - * Triggers sorting callback. - * - * @private - * @param {jQuery} $th Table header - * @param {Number} id Used to seperate the columns - */ - var sort = function ($th, id) { - if (id === sortCol) { - // Change sorting direction - if (sortDir === 0) { - sortDir = 1; - $th.addClass('h5p-reverse'); - } - else { - sortDir = 0; - $th.removeClass('h5p-reverse'); - } - } - else { - // Change sorting column - $sortCol.removeClass('h5p-sort').removeClass('h5p-reverse'); - $sortCol = $th.addClass('h5p-sort'); - sortCol = id; - sortDir = 0; - } - - sortByCol({ - by: sortCol, - dir: sortDir - }); - }; - - /** - * Set table headers. - * - * @public - * @param {Array} cols - * Table header data. Can be strings or objects with options like - * "text" and "sortable". E.g. - * [{text: 'Col 1', sortable: true}, 'Col 2', 'Col 3'] - * @param {Function} sort Callback which is runned when sorting changes - * @param {Object} [order] - */ - this.setHeaders = function (cols, sort, order) { - numCols = cols.length; - sortByCol = sort; - - if (order) { - sortCol = order.by; - sortDir = order.dir; - } - - // Create new head - var $newThead = $('
    ', { - html: rows[i][j] - }).appendTo($tr); - } - } - - $tbody.replaceWith($newTbody); - $tbody = $newTbody; - - return $tbody; - }; - - /** - * Set custom table body content. This can be a message or a throbber. - * Will cover all table columns. - * - * @public - * @param {jQuery} $content Custom content - */ - this.setBody = function ($content) { - var $newTbody = $('
    ', { - colspan: numCols - }).append($content).appendTo($tr); - $tbody.replaceWith($newTbody); - $tbody = $newTbody; - }; - - /** - * Set custom table foot content. This can be a pagination widget. - * Will cover all table columns. - * - * @public - * @param {jQuery} $content Custom content - */ - this.setFoot = function ($content) { - var $newTfoot = $('
    ', { - colspan: numCols - }).append($content).appendTo($tr); - $tfoot.replaceWith($newTfoot); - }; - - - /** - * Appends the table to the given container. - * - * @public - * @param {jQuery} $container - */ - this.appendTo = function ($container) { - $table.appendTo($container); - }; - }; - - /** - * Generic pagination class. Creates a useful pagination widget. - * - * @class - * @param {Number} num Total number of items to pagiate. - * @param {Number} limit Number of items to dispaly per page. - * @param {Function} goneTo - * Callback which is fired when the user wants to go to another page. - * @param {Object} l10n - * Localization / translations. e.g. - * { - * currentPage: 'Page $current of $total', - * nextPage: 'Next page', - * previousPage: 'Previous page' - * } - */ - H5PUtils.Pagination = function (num, limit, goneTo, l10n) { - var current = 0; - var pages = Math.ceil(num / limit); - - // Create components - - // Previous button - var $left = $(''; - } - if (contentData.displayOptions.export && contentData.displayOptions.copy) { - html += '
    or
    '; - } - if (contentData.displayOptions.copy) { - html += ''; - } - - const dialog = new H5P.Dialog('reuse', H5P.t('reuseContent'), html, $element); - - // Selecting embed code when dialog is opened - H5P.jQuery(dialog).on('dialog-opened', function (e, $dialog) { - H5P.jQuery('More Info').click(function (e) { - e.stopPropagation(); - }).appendTo($dialog.find('h2')); - $dialog.find('.h5p-download-button').click(function () { - window.location.href = contentData.exportUrl; - instance.triggerXAPI('downloaded'); - dialog.close(); - }); - $dialog.find('.h5p-copy-button').click(function () { - const item = new H5P.ClipboardItem(library); - item.contentId = contentId; - H5P.setClipboard(item); - instance.triggerXAPI('copied'); - dialog.close(); - H5P.attachToastTo( - H5P.jQuery('.h5p-content:first')[0], - H5P.t('contentCopied'), - { - position: { - horizontal: 'centered', - vertical: 'centered', - noOverflowX: true - } - } - ); - }); - H5P.trigger(instance, 'resize'); - }).on('dialog-closed', function () { - H5P.trigger(instance, 'resize'); - }); - - dialog.open(); -}; - -/** - * Display a dialog containing the embed code. - * - * @param {H5P.jQuery} $element - * Element to insert dialog after. - * @param {string} embedCode - * The embed code. - * @param {string} resizeCode - * The advanced resize code - * @param {Object} size - * The content's size. - * @param {number} size.width - * @param {number} size.height - */ -H5P.openEmbedDialog = function ($element, embedCode, resizeCode, size, instance) { - var fullEmbedCode = embedCode + resizeCode; - var dialog = new H5P.Dialog('embed', H5P.t('embed'), '' + H5P.t('size') + ': × px
    ' + H5P.t('showAdvanced') + '

    ' + H5P.t('advancedHelp') + '

    ', $element); - - // Selecting embed code when dialog is opened - H5P.jQuery(dialog).on('dialog-opened', function (event, $dialog) { - var $inner = $dialog.find('.h5p-inner'); - var $scroll = $inner.find('.h5p-scroll-content'); - var diff = $scroll.outerHeight() - $scroll.innerHeight(); - var positionInner = function () { - H5P.trigger(instance, 'resize'); - }; - - // Handle changing of width/height - var $w = $dialog.find('.h5p-embed-size:eq(0)'); - var $h = $dialog.find('.h5p-embed-size:eq(1)'); - var getNum = function ($e, d) { - var num = parseFloat($e.val()); - if (isNaN(num)) { - return d; - } - return Math.ceil(num); - }; - var updateEmbed = function () { - $dialog.find('.h5p-embed-code-container:first').val(fullEmbedCode.replace(':w', getNum($w, size.width)).replace(':h', getNum($h, size.height))); - }; - - $w.change(updateEmbed); - $h.change(updateEmbed); - updateEmbed(); - - // Select text and expand textareas - $dialog.find('.h5p-embed-code-container').each(function () { - H5P.jQuery(this).css('height', this.scrollHeight + 'px').focus(function () { - H5P.jQuery(this).select(); - }); - }); - $dialog.find('.h5p-embed-code-container').eq(0).select(); - positionInner(); - - // Expand advanced embed - var expand = function () { - var $expander = H5P.jQuery(this); - var $content = $expander.next(); - if ($content.is(':visible')) { - $expander.removeClass('h5p-open').text(H5P.t('showAdvanced')).attr('aria-expanded', 'true'); - $content.hide(); - } - else { - $expander.addClass('h5p-open').text(H5P.t('hideAdvanced')).attr('aria-expanded', 'false'); - $content.show(); - } - $dialog.find('.h5p-embed-code-container').each(function () { - H5P.jQuery(this).css('height', this.scrollHeight + 'px'); - }); - positionInner(); - }; - $dialog.find('.h5p-expander').click(expand).keypress(function (event) { - if (event.keyCode === 32) { - expand.apply(this); - return false; - } - }); - }).on('dialog-closed', function () { - H5P.trigger(instance, 'resize'); - }); - - dialog.open(); -}; - -/** - * Show a toast message. - * - * The reference element could be dom elements the toast should be attached to, - * or e.g. the document body for general toast messages. - * - * @param {DOM} element Reference element to show toast message for. - * @param {string} message Message to show. - * @param {object} [config] Configuration. - * @param {string} [config.style=h5p-toast] Style name for the tooltip. - * @param {number} [config.duration=3000] Toast message length in ms. - * @param {object} [config.position] Relative positioning of the toast. - * @param {string} [config.position.horizontal=centered] [before|left|centered|right|after]. - * @param {string} [config.position.vertical=below] [above|top|centered|bottom|below]. - * @param {number} [config.position.offsetHorizontal=0] Extra horizontal offset. - * @param {number} [config.position.offsetVertical=0] Extra vetical offset. - * @param {boolean} [config.position.noOverflowLeft=false] True to prevent overflow left. - * @param {boolean} [config.position.noOverflowRight=false] True to prevent overflow right. - * @param {boolean} [config.position.noOverflowTop=false] True to prevent overflow top. - * @param {boolean} [config.position.noOverflowBottom=false] True to prevent overflow bottom. - * @param {boolean} [config.position.noOverflowX=false] True to prevent overflow left and right. - * @param {boolean} [config.position.noOverflowY=false] True to prevent overflow top and bottom. - * @param {object} [config.position.overflowReference=document.body] DOM reference for overflow. - */ -H5P.attachToastTo = function (element, message, config) { - if (element === undefined || message === undefined) { - return; - } - - const eventPath = function (evt) { - var path = (evt.composedPath && evt.composedPath()) || evt.path; - var target = evt.target; - - if (path != null) { - // Safari doesn't include Window, but it should. - return (path.indexOf(window) < 0) ? path.concat(window) : path; - } - - if (target === window) { - return [window]; - } - - function getParents(node, memo) { - memo = memo || []; - var parentNode = node.parentNode; - - if (!parentNode) { - return memo; - } - else { - return getParents(parentNode, memo.concat(parentNode)); - } - } - - return [target].concat(getParents(target), window); - }; - - /** - * Handle click while toast is showing. - */ - const clickHandler = function (event) { - /* - * A common use case will be to attach toasts to buttons that are clicked. - * The click would remove the toast message instantly without this check. - * Children of the clicked element are also ignored. - */ - var path = eventPath(event); - if (path.indexOf(element) !== -1) { - return; - } - clearTimeout(timer); - removeToast(); - }; - - - - /** - * Remove the toast message. - */ - const removeToast = function () { - document.removeEventListener('click', clickHandler); - if (toast.parentNode) { - toast.parentNode.removeChild(toast); - } - }; - - /** - * Get absolute coordinates for the toast. - * - * @param {DOM} element Reference element to show toast message for. - * @param {DOM} toast Toast element. - * @param {object} [position={}] Relative positioning of the toast message. - * @param {string} [position.horizontal=centered] [before|left|centered|right|after]. - * @param {string} [position.vertical=below] [above|top|centered|bottom|below]. - * @param {number} [position.offsetHorizontal=0] Extra horizontal offset. - * @param {number} [position.offsetVertical=0] Extra vetical offset. - * @param {boolean} [position.noOverflowLeft=false] True to prevent overflow left. - * @param {boolean} [position.noOverflowRight=false] True to prevent overflow right. - * @param {boolean} [position.noOverflowTop=false] True to prevent overflow top. - * @param {boolean} [position.noOverflowBottom=false] True to prevent overflow bottom. - * @param {boolean} [position.noOverflowX=false] True to prevent overflow left and right. - * @param {boolean} [position.noOverflowY=false] True to prevent overflow top and bottom. - * @return {object} - */ - const getToastCoordinates = function (element, toast, position) { - position = position || {}; - position.offsetHorizontal = position.offsetHorizontal || 0; - position.offsetVertical = position.offsetVertical || 0; - - const toastRect = toast.getBoundingClientRect(); - const elementRect = element.getBoundingClientRect(); - - let left = 0; - let top = 0; - - // Compute horizontal position - switch (position.horizontal) { - case 'before': - left = elementRect.left - toastRect.width - position.offsetHorizontal; - break; - case 'after': - left = elementRect.left + elementRect.width + position.offsetHorizontal; - break; - case 'left': - left = elementRect.left + position.offsetHorizontal; - break; - case 'right': - left = elementRect.left + elementRect.width - toastRect.width - position.offsetHorizontal; - break; - case 'centered': - left = elementRect.left + elementRect.width / 2 - toastRect.width / 2 + position.offsetHorizontal; - break; - default: - left = elementRect.left + elementRect.width / 2 - toastRect.width / 2 + position.offsetHorizontal; - } - - // Compute vertical position - switch (position.vertical) { - case 'above': - top = elementRect.top - toastRect.height - position.offsetVertical; - break; - case 'below': - top = elementRect.top + elementRect.height + position.offsetVertical; - break; - case 'top': - top = elementRect.top + position.offsetVertical; - break; - case 'bottom': - top = elementRect.top + elementRect.height - toastRect.height - position.offsetVertical; - break; - case 'centered': - top = elementRect.top + elementRect.height / 2 - toastRect.height / 2 + position.offsetVertical; - break; - default: - top = elementRect.top + elementRect.height + position.offsetVertical; - } - - // Prevent overflow - const overflowElement = document.body; - const bounds = overflowElement.getBoundingClientRect(); - if ((position.noOverflowLeft || position.noOverflowX) && (left < bounds.x)) { - left = bounds.x; - } - if ((position.noOverflowRight || position.noOverflowX) && ((left + toastRect.width) > (bounds.x + bounds.width))) { - left = bounds.x + bounds.width - toastRect.width; - } - if ((position.noOverflowTop || position.noOverflowY) && (top < bounds.y)) { - top = bounds.y; - } - if ((position.noOverflowBottom || position.noOverflowY) && ((top + toastRect.height) > (bounds.y + bounds.height))) { - left = bounds.y + bounds.height - toastRect.height; - } - - return {left: left, top: top}; - }; - - // Sanitization - config = config || {}; - config.style = config.style || 'h5p-toast'; - config.duration = config.duration || 3000; - - // Build toast - const toast = document.createElement('div'); - toast.setAttribute('id', config.style); - toast.classList.add('h5p-toast-disabled'); - toast.classList.add(config.style); - - const msg = document.createElement('span'); - msg.innerHTML = message; - toast.appendChild(msg); - - document.body.appendChild(toast); - - // The message has to be set before getting the coordinates - const coordinates = getToastCoordinates(element, toast, config.position); - toast.style.left = Math.round(coordinates.left) + 'px'; - toast.style.top = Math.round(coordinates.top) + 'px'; - - toast.classList.remove('h5p-toast-disabled'); - const timer = setTimeout(removeToast, config.duration); - - // The toast can also be removed by clicking somewhere - document.addEventListener('click', clickHandler); -}; - -/** - * Copyrights for a H5P Content Library. - * - * @class - */ -H5P.ContentCopyrights = function () { - var label; - var media = []; - var content = []; - - /** - * Set label. - * - * @param {string} newLabel - */ - this.setLabel = function (newLabel) { - label = newLabel; - }; - - /** - * Add sub content. - * - * @param {H5P.MediaCopyright} newMedia - */ - this.addMedia = function (newMedia) { - if (newMedia !== undefined) { - media.push(newMedia); - } - }; - - /** - * Add sub content in front. - * - * @param {H5P.MediaCopyright} newMedia - */ - this.addMediaInFront = function (newMedia) { - if (newMedia !== undefined) { - media.unshift(newMedia); - } - }; - - /** - * Add sub content. - * - * @param {H5P.ContentCopyrights} newContent - */ - this.addContent = function (newContent) { - if (newContent !== undefined) { - content.push(newContent); - } - }; - - /** - * Print content copyright. - * - * @returns {string} HTML. - */ - this.toString = function () { - var html = ''; - - // Add media rights - for (var i = 0; i < media.length; i++) { - html += media[i]; - } - - // Add sub content rights - for (i = 0; i < content.length; i++) { - html += content[i]; - } - - - if (html !== '') { - // Add a label to this info - if (label !== undefined) { - html = '

    ' + label + '

    ' + html; - } - - // Add wrapper - html = '
    ' + html + '
    '; - } - - return html; - }; -}; - -/** - * A ordered list of copyright fields for media. - * - * @class - * @param {Object} copyright - * Copyright information fields. - * @param {Object} [labels] - * Translation of labels. - * @param {Array} [order] - * Order of the fields. - * @param {Object} [extraFields] - * Add extra copyright fields. - */ -H5P.MediaCopyright = function (copyright, labels, order, extraFields) { - var thumbnail; - var list = new H5P.DefinitionList(); - - /** - * Get translated label for field. - * - * @private - * @param {string} fieldName - * @returns {string} - */ - var getLabel = function (fieldName) { - if (labels === undefined || labels[fieldName] === undefined) { - return H5P.t(fieldName); - } - - return labels[fieldName]; - }; - - /** - * Get humanized value for the license field. - * - * @private - * @param {string} license - * @param {string} [version] - * @returns {string} - */ - var humanizeLicense = function (license, version) { - var copyrightLicense = H5P.copyrightLicenses[license]; - - // Build license string - var value = ''; - if (!(license === 'PD' && version)) { - // Add license label - value += (copyrightLicense.hasOwnProperty('label') ? copyrightLicense.label : copyrightLicense); - } - - // Check for version info - var versionInfo; - if (copyrightLicense.versions) { - if (copyrightLicense.versions.default && (!version || !copyrightLicense.versions[version])) { - version = copyrightLicense.versions.default; - } - if (version && copyrightLicense.versions[version]) { - versionInfo = copyrightLicense.versions[version]; - } - } - - if (versionInfo) { - // Add license version - if (value) { - value += ' '; - } - value += (versionInfo.hasOwnProperty('label') ? versionInfo.label : versionInfo); - } - - // Add link if specified - var link; - if (copyrightLicense.hasOwnProperty('link')) { - link = copyrightLicense.link.replace(':version', copyrightLicense.linkVersions ? copyrightLicense.linkVersions[version] : version); - } - else if (versionInfo && copyrightLicense.hasOwnProperty('link')) { - link = versionInfo.link; - } - if (link) { - value = '' + value + ''; - } - - // Generate parenthesis - var parenthesis = ''; - if (license !== 'PD' && license !== 'C') { - parenthesis += license; - } - if (version && version !== 'CC0 1.0') { - if (parenthesis && license !== 'GNU GPL') { - parenthesis += ' '; - } - parenthesis += version; - } - if (parenthesis) { - value += ' (' + parenthesis + ')'; - } - if (license === 'C') { - value += ' ©'; - } - - return value; - }; - - if (copyright !== undefined) { - // Add the extra fields - for (var field in extraFields) { - if (extraFields.hasOwnProperty(field)) { - copyright[field] = extraFields[field]; - } - } - - if (order === undefined) { - // Set default order - order = ['contentType', 'title', 'license', 'author', 'year', 'source', 'licenseExtras', 'changes']; - } - - for (var i = 0; i < order.length; i++) { - var fieldName = order[i]; - if (copyright[fieldName] !== undefined && copyright[fieldName] !== '') { - var humanValue = copyright[fieldName]; - if (fieldName === 'license') { - humanValue = humanizeLicense(copyright.license, copyright.version); - } - if (fieldName === 'source') { - humanValue = (humanValue) ? '' + humanValue + '' : undefined; - } - list.add(new H5P.Field(getLabel(fieldName), humanValue)); - } - } - } - - /** - * Set thumbnail. - * - * @param {H5P.Thumbnail} newThumbnail - */ - this.setThumbnail = function (newThumbnail) { - thumbnail = newThumbnail; - }; - - /** - * Checks if this copyright is undisclosed. - * I.e. only has the license attribute set, and it's undisclosed. - * - * @returns {boolean} - */ - this.undisclosed = function () { - if (list.size() === 1) { - var field = list.get(0); - if (field.getLabel() === getLabel('license') && field.getValue() === humanizeLicense('U')) { - return true; - } - } - return false; - }; - - /** - * Print media copyright. - * - * @returns {string} HTML. - */ - this.toString = function () { - var html = ''; - - if (this.undisclosed()) { - return html; // No need to print a copyright with a single undisclosed license. - } - - if (thumbnail !== undefined) { - html += thumbnail; - } - html += list; - - if (html !== '') { - html = ''; - } - - return html; - }; -}; - -/** - * A simple and elegant class for creating thumbnails of images. - * - * @class - * @param {string} source - * @param {number} width - * @param {number} height - * @param {string} alt - * alternative text for the thumbnail - */ -H5P.Thumbnail = function (source, width, height, alt) { - var thumbWidth, thumbHeight = 100; - if (width !== undefined) { - thumbWidth = Math.round(thumbHeight * (width / height)); - } - - /** - * Print thumbnail. - * - * @returns {string} HTML. - */ - this.toString = function () { - return '' + (alt ? alt : '') + ''; - }; -}; - -/** - * Simple data structure class for storing a single field. - * - * @class - * @param {string} label - * @param {string} value - */ -H5P.Field = function (label, value) { - /** - * Public. Get field label. - * - * @returns {String} - */ - this.getLabel = function () { - return label; - }; - - /** - * Public. Get field value. - * - * @returns {String} - */ - this.getValue = function () { - return value; - }; -}; - -/** - * Simple class for creating a definition list. - * - * @class - */ -H5P.DefinitionList = function () { - var fields = []; - - /** - * Add field to list. - * - * @param {H5P.Field} field - */ - this.add = function (field) { - fields.push(field); - }; - - /** - * Get Number of fields. - * - * @returns {number} - */ - this.size = function () { - return fields.length; - }; - - /** - * Get field at given index. - * - * @param {number} index - * @returns {H5P.Field} - */ - this.get = function (index) { - return fields[index]; - }; - - /** - * Print definition list. - * - * @returns {string} HTML. - */ - this.toString = function () { - var html = ''; - for (var i = 0; i < fields.length; i++) { - var field = fields[i]; - html += '
    ' + field.getLabel() + '
    ' + field.getValue() + '
    '; - } - return (html === '' ? html : '
    ' + html + '
    '); - }; -}; - -/** - * THIS FUNCTION/CLASS IS DEPRECATED AND WILL BE REMOVED. - * - * Helper object for keeping coordinates in the same format all over. - * - * @deprecated - * Will be removed march 2016. - * @class - * @param {number} x - * @param {number} y - * @param {number} w - * @param {number} h - */ -H5P.Coords = function (x, y, w, h) { - if ( !(this instanceof H5P.Coords) ) - return new H5P.Coords(x, y, w, h); - - /** @member {number} */ - this.x = 0; - /** @member {number} */ - this.y = 0; - /** @member {number} */ - this.w = 1; - /** @member {number} */ - this.h = 1; - - if (typeof(x) === 'object') { - this.x = x.x; - this.y = x.y; - this.w = x.w; - this.h = x.h; - } - else { - if (x !== undefined) { - this.x = x; - } - if (y !== undefined) { - this.y = y; - } - if (w !== undefined) { - this.w = w; - } - if (h !== undefined) { - this.h = h; - } - } - return this; -}; - -/** - * Parse library string into values. - * - * @param {string} library - * library in the format "machineName majorVersion.minorVersion" - * @returns {Object} - * library as an object with machineName, majorVersion and minorVersion properties - * return false if the library parameter is invalid - */ -H5P.libraryFromString = function (library) { - var regExp = /(.+)\s(\d+)\.(\d+)$/g; - var res = regExp.exec(library); - if (res !== null) { - return { - 'machineName': res[1], - 'majorVersion': parseInt(res[2]), - 'minorVersion': parseInt(res[3]) - }; - } - else { - return false; - } -}; - -/** - * Get the path to the library - * - * @param {string} library - * The library identifier in the format "machineName-majorVersion.minorVersion". - * @returns {string} - * The full path to the library. - */ -H5P.getLibraryPath = function (library) { - if (H5PIntegration.urlLibraries !== undefined) { - // This is an override for those implementations that has a different libraries URL, e.g. Moodle - return H5PIntegration.urlLibraries + '/' + library; - } - else { - return H5PIntegration.url + '/libraries/' + library; - } -}; - -/** - * Recursivly clone the given object. - * - * @param {Object|Array} object - * Object to clone. - * @param {boolean} [recursive] - * @returns {Object|Array} - * A clone of object. - */ -H5P.cloneObject = function (object, recursive) { - // TODO: Consider if this needs to be in core. Doesn't $.extend do the same? - var clone = object instanceof Array ? [] : {}; - - for (var i in object) { - if (object.hasOwnProperty(i)) { - if (recursive !== undefined && recursive && typeof object[i] === 'object') { - clone[i] = H5P.cloneObject(object[i], recursive); - } - else { - clone[i] = object[i]; - } - } - } - - return clone; -}; - -/** - * Remove all empty spaces before and after the value. - * - * @param {string} value - * @returns {string} - */ -H5P.trim = function (value) { - return value.replace(/^\s+|\s+$/g, ''); - - // TODO: Only include this or String.trim(). What is best? - // I'm leaning towards implementing the missing ones: http://kangax.github.io/compat-table/es5/ - // So should we make this function deprecated? -}; - -/** - * Recursive function that detects deep empty structures. - * - * @param {*} value - * @returns {bool} - */ -H5P.isEmpty = value => { - if (!value && value !== 0 && value !== false) { - return true; // undefined, null, NaN and empty strings. - } - else if (Array.isArray(value)) { - for (let i = 0; i < value.length; i++) { - if (!H5P.isEmpty(value[i])) { - return false; // Array contains a non-empty value - } - } - return true; // Empty array - } - else if (typeof value === 'object') { - for (let prop in value) { - if (value.hasOwnProperty(prop) && !H5P.isEmpty(value[prop])) { - return false; // Object contains a non-empty value - } - } - return true; // Empty object - } - return false; -}; - -/** - * Check if JavaScript path/key is loaded. - * - * @param {string} path - * @returns {boolean} - */ -H5P.jsLoaded = function (path) { - H5PIntegration.loadedJs = H5PIntegration.loadedJs || []; - return H5P.jQuery.inArray(path, H5PIntegration.loadedJs) !== -1; -}; - -/** - * Check if styles path/key is loaded. - * - * @param {string} path - * @returns {boolean} - */ -H5P.cssLoaded = function (path) { - H5PIntegration.loadedCss = H5PIntegration.loadedCss || []; - return H5P.jQuery.inArray(path, H5PIntegration.loadedCss) !== -1; -}; - -/** - * Shuffle an array in place. - * - * @param {Array} array - * Array to shuffle - * @returns {Array} - * The passed array is returned for chaining. - */ -H5P.shuffleArray = function (array) { - // TODO: Consider if this should be a part of core. I'm guessing very few libraries are going to use it. - if (!(array instanceof Array)) { - return; - } - - var i = array.length, j, tempi, tempj; - if ( i === 0 ) return false; - while ( --i ) { - j = Math.floor( Math.random() * ( i + 1 ) ); - tempi = array[i]; - tempj = array[j]; - array[i] = tempj; - array[j] = tempi; - } - return array; -}; - -/** - * Post finished results for user. - * - * @deprecated - * Do not use this function directly, trigger the finish event instead. - * Will be removed march 2016 - * @param {number} contentId - * Identifies the content - * @param {number} score - * Achieved score/points - * @param {number} maxScore - * The maximum score/points that can be achieved - * @param {number} [time] - * Reported time consumption/usage - */ -H5P.setFinished = function (contentId, score, maxScore, time) { - var validScore = typeof score === 'number' || score instanceof Number; - if (validScore && H5PIntegration.postUserStatistics === true) { - /** - * Return unix timestamp for the given JS Date. - * - * @private - * @param {Date} date - * @returns {Number} - */ - var toUnix = function (date) { - return Math.round(date.getTime() / 1000); - }; - - // Post the results - const data = { - contentId: contentId, - score: score, - maxScore: maxScore, - opened: toUnix(H5P.opened[contentId]), - finished: toUnix(new Date()), - time: time - }; - H5P.jQuery.post(H5PIntegration.ajax.setFinished, data) - .fail(function () { - H5P.offlineRequestQueue.add(H5PIntegration.ajax.setFinished, data); - }); - } -}; - -// Add indexOf to browsers that lack them. (IEs) -if (!Array.prototype.indexOf) { - Array.prototype.indexOf = function (needle) { - for (var i = 0; i < this.length; i++) { - if (this[i] === needle) { - return i; - } - } - return -1; - }; -} - -// Need to define trim() since this is not available on older IEs, -// and trim is used in several libs -if (String.prototype.trim === undefined) { - String.prototype.trim = function () { - return H5P.trim(this); - }; -} - -/** - * Trigger an event on an instance - * - * Helper function that triggers an event if the instance supports event handling - * - * @param {Object} instance - * Instance of H5P content - * @param {string} eventType - * Type of event to trigger - * @param {*} data - * @param {Object} extras - */ -H5P.trigger = function (instance, eventType, data, extras) { - // Try new event system first - if (instance.trigger !== undefined) { - instance.trigger(eventType, data, extras); - } - // Try deprecated event system - else if (instance.$ !== undefined && instance.$.trigger !== undefined) { - instance.$.trigger(eventType); - } -}; - -/** - * Register an event handler - * - * Helper function that registers an event handler for an event type if - * the instance supports event handling - * - * @param {Object} instance - * Instance of H5P content - * @param {string} eventType - * Type of event to listen for - * @param {H5P.EventCallback} handler - * Callback that gets triggered for events of the specified type - */ -H5P.on = function (instance, eventType, handler) { - // Try new event system first - if (instance.on !== undefined) { - instance.on(eventType, handler); - } - // Try deprecated event system - else if (instance.$ !== undefined && instance.$.on !== undefined) { - instance.$.on(eventType, handler); - } -}; - -/** - * Generate random UUID - * - * @returns {string} UUID - */ -H5P.createUUID = function () { - return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (char) { - var random = Math.random()*16|0, newChar = char === 'x' ? random : (random&0x3|0x8); - return newChar.toString(16); - }); -}; - -/** - * Create title - * - * @param {string} rawTitle - * @param {number} maxLength - * @returns {string} - */ -H5P.createTitle = function (rawTitle, maxLength) { - if (!rawTitle) { - return ''; - } - if (maxLength === undefined) { - maxLength = 60; - } - var title = H5P.jQuery('
    ') - .text( - // Strip tags - rawTitle.replace(/(<([^>]+)>)/ig,"") - // Escape - ).text(); - if (title.length > maxLength) { - title = title.substr(0, maxLength - 3) + '...'; - } - return title; -}; - -// Wrap in privates -(function ($) { - - /** - * Creates ajax requests for inserting, updateing and deleteing - * content user data. - * - * @private - * @param {number} contentId What content to store the data for. - * @param {string} dataType Identifies the set of data for this content. - * @param {string} subContentId Identifies sub content - * @param {function} [done] Callback when ajax is done. - * @param {object} [data] To be stored for future use. - * @param {boolean} [preload=false] Data is loaded when content is loaded. - * @param {boolean} [invalidate=false] Data is invalidated when content changes. - * @param {boolean} [async=true] - */ - function contentUserDataAjax(contentId, dataType, subContentId, done, data, preload, invalidate, async) { - if (H5PIntegration.user === undefined) { - // Not logged in, no use in saving. - done('Not signed in.'); - return; - } - // Moodle patch to let override this method. - if (H5P.contentUserDataAjax !== undefined) { - return H5P.contentUserDataAjax(contentId, dataType, subContentId, done, data, preload, invalidate, async); - } - // End of Moodle patch. - - var options = { - url: H5PIntegration.ajax.contentUserData.replace(':contentId', contentId).replace(':dataType', dataType).replace(':subContentId', subContentId ? subContentId : 0), - dataType: 'json', - async: async === undefined ? true : async - }; - if (data !== undefined) { - options.type = 'POST'; - options.data = { - data: (data === null ? 0 : data), - preload: (preload ? 1 : 0), - invalidate: (invalidate ? 1 : 0) - }; - } - else { - options.type = 'GET'; - } - if (done !== undefined) { - options.error = function (xhr, error) { - done(error); - }; - options.success = function (response) { - if (!response.success) { - done(response.message); - return; - } - - if (response.data === false || response.data === undefined) { - done(); - return; - } - - done(undefined, response.data); - }; - } - - $.ajax(options); - } - - /** - * Get user data for given content. - * - * @param {number} contentId - * What content to get data for. - * @param {string} dataId - * Identifies the set of data for this content. - * @param {function} done - * Callback with error and data parameters. - * @param {string} [subContentId] - * Identifies which data belongs to sub content. - */ - H5P.getUserData = function (contentId, dataId, done, subContentId) { - if (!subContentId) { - subContentId = 0; // Default - } - - H5PIntegration.contents = H5PIntegration.contents || {}; - var content = H5PIntegration.contents['cid-' + contentId] || {}; - var preloadedData = content.contentUserData; - if (preloadedData && preloadedData[subContentId] && preloadedData[subContentId][dataId] !== undefined) { - if (preloadedData[subContentId][dataId] === 'RESET') { - done(undefined, null); - return; - } - try { - done(undefined, JSON.parse(preloadedData[subContentId][dataId])); - } - catch (err) { - done(err); - } - } - else { - contentUserDataAjax(contentId, dataId, subContentId, function (err, data) { - if (err || data === undefined) { - done(err, data); - return; // Error or no data - } - - // Cache in preloaded - if (content.contentUserData === undefined) { - content.contentUserData = preloadedData = {}; - } - if (preloadedData[subContentId] === undefined) { - preloadedData[subContentId] = {}; - } - preloadedData[subContentId][dataId] = data; - - // Done. Try to decode JSON - try { - done(undefined, JSON.parse(data)); - } - catch (e) { - done(e); - } - }); - } - }; - - /** - * Async error handling. - * - * @callback H5P.ErrorCallback - * @param {*} error - */ - - /** - * Set user data for given content. - * - * @param {number} contentId - * What content to get data for. - * @param {string} dataId - * Identifies the set of data for this content. - * @param {Object} data - * The data that is to be stored. - * @param {Object} [extras] - * Extra properties - * @param {string} [extras.subContentId] - * Identifies which data belongs to sub content. - * @param {boolean} [extras.preloaded=true] - * If the data should be loaded when content is loaded. - * @param {boolean} [extras.deleteOnChange=false] - * If the data should be invalidated when the content changes. - * @param {H5P.ErrorCallback} [extras.errorCallback] - * Callback with error as parameters. - * @param {boolean} [extras.async=true] - */ - H5P.setUserData = function (contentId, dataId, data, extras) { - var options = H5P.jQuery.extend(true, {}, { - subContentId: 0, - preloaded: true, - deleteOnChange: false, - async: true - }, extras); - - try { - data = JSON.stringify(data); - } - catch (err) { - if (options.errorCallback) { - options.errorCallback(err); - } - return; // Failed to serialize. - } - - var content = H5PIntegration.contents['cid-' + contentId]; - if (content === undefined) { - content = H5PIntegration.contents['cid-' + contentId] = {}; - } - if (!content.contentUserData) { - content.contentUserData = {}; - } - var preloadedData = content.contentUserData; - if (preloadedData[options.subContentId] === undefined) { - preloadedData[options.subContentId] = {}; - } - if (data === preloadedData[options.subContentId][dataId]) { - return; // No need to save this twice. - } - - preloadedData[options.subContentId][dataId] = data; - contentUserDataAjax(contentId, dataId, options.subContentId, function (error) { - if (options.errorCallback && error) { - options.errorCallback(error); - } - }, data, options.preloaded, options.deleteOnChange, options.async); - }; - - /** - * Delete user data for given content. - * - * @param {number} contentId - * What content to remove data for. - * @param {string} dataId - * Identifies the set of data for this content. - * @param {string} [subContentId] - * Identifies which data belongs to sub content. - */ - H5P.deleteUserData = function (contentId, dataId, subContentId) { - if (!subContentId) { - subContentId = 0; // Default - } - - // Remove from preloaded/cache - var preloadedData = H5PIntegration.contents['cid-' + contentId].contentUserData; - if (preloadedData && preloadedData[subContentId] && preloadedData[subContentId][dataId]) { - delete preloadedData[subContentId][dataId]; - } - - contentUserDataAjax(contentId, dataId, subContentId, undefined, null); - }; - - /** - * Function for getting content for a certain ID - * - * @param {number} contentId - * @return {Object} - */ - H5P.getContentForInstance = function (contentId) { - var key = 'cid-' + contentId; - var exists = H5PIntegration && H5PIntegration.contents && - H5PIntegration.contents[key]; - - return exists ? H5PIntegration.contents[key] : undefined; - }; - - /** - * Prepares the content parameters for storing in the clipboard. - * - * @class - * @param {Object} parameters The parameters for the content to store - * @param {string} [genericProperty] If only part of the parameters are generic, which part - * @param {string} [specificKey] If the parameters are specific, what content type does it fit - * @returns {Object} Ready for the clipboard - */ - H5P.ClipboardItem = function (parameters, genericProperty, specificKey) { - var self = this; - - /** - * Set relative dimensions when params contains a file with a width and a height. - * Very useful to be compatible with wysiwyg editors. - * - * @private - */ - var setDimensionsFromFile = function () { - if (!self.generic) { - return; - } - var params = self.specific[self.generic]; - if (!params.params.file || !params.params.file.width || !params.params.file.height) { - return; - } - - self.width = 20; // % - self.height = (params.params.file.height / params.params.file.width) * self.width; - }; - - if (!genericProperty) { - genericProperty = 'action'; - parameters = { - action: parameters - }; - } - - self.specific = parameters; - - if (genericProperty && parameters[genericProperty]) { - self.generic = genericProperty; - } - if (specificKey) { - self.from = specificKey; - } - - if (window.H5PEditor && H5PEditor.contentId) { - self.contentId = H5PEditor.contentId; - } - - if (!self.specific.width && !self.specific.height) { - setDimensionsFromFile(); - } - }; - - /** - * Store item in the H5P Clipboard. - * - * @param {H5P.ClipboardItem|*} clipboardItem - */ - H5P.clipboardify = function (clipboardItem) { - if (!(clipboardItem instanceof H5P.ClipboardItem)) { - clipboardItem = new H5P.ClipboardItem(clipboardItem); - } - H5P.setClipboard(clipboardItem); - }; - - /** - * Retrieve parsed clipboard data. - * - * @return {Object} - */ - H5P.getClipboard = function () { - return parseClipboard(); - }; - - /** - * Set item in the H5P Clipboard. - * - * @param {H5P.ClipboardItem|object} clipboardItem - Data to be set. - */ - H5P.setClipboard = function (clipboardItem) { - localStorage.setItem('h5pClipboard', JSON.stringify(clipboardItem)); - - // Trigger an event so all 'Paste' buttons may be enabled. - H5P.externalDispatcher.trigger('datainclipboard', {reset: false}); - }; - - /** - * Get config for a library - * - * @param string machineName - * @return Object - */ - H5P.getLibraryConfig = function (machineName) { - var hasConfig = H5PIntegration.libraryConfig && H5PIntegration.libraryConfig[machineName]; - return hasConfig ? H5PIntegration.libraryConfig[machineName] : {}; - }; - - /** - * Get item from the H5P Clipboard. - * - * @private - * @return {Object} - */ - var parseClipboard = function () { - var clipboardData = localStorage.getItem('h5pClipboard'); - if (!clipboardData) { - return; - } - - // Try to parse clipboard dat - try { - clipboardData = JSON.parse(clipboardData); - } - catch (err) { - console.error('Unable to parse JSON from clipboard.', err); - return; - } - - // Update file URLs and reset content Ids - recursiveUpdate(clipboardData.specific, function (path) { - var isTmpFile = (path.substr(-4, 4) === '#tmp'); - if (!isTmpFile && clipboardData.contentId && !path.match(/^https?:\/\//i)) { - // Comes from existing content - - let prefix; - if (H5PEditor.contentId) { - // .. to existing content - prefix = '../' + clipboardData.contentId + '/'; - } - else { - // .. to new content - prefix = (H5PEditor.contentRelUrl ? H5PEditor.contentRelUrl : '../content/') + clipboardData.contentId + '/'; - } - return path.substr(0, prefix.length) === prefix ? path : prefix + path; - } - - return path; // Will automatically be looked for in tmp folder - }); - - - if (clipboardData.generic) { - // Use reference instead of key - clipboardData.generic = clipboardData.specific[clipboardData.generic]; - } - - return clipboardData; - }; - - /** - * Update file URLs and reset content IDs. - * Useful when copying content. - * - * @private - * @param {object} params Reference - * @param {function} handler Modifies the path to work when pasted - */ - var recursiveUpdate = function (params, handler) { - for (var prop in params) { - if (params.hasOwnProperty(prop) && params[prop] instanceof Object) { - var obj = params[prop]; - if (obj.path !== undefined && obj.mime !== undefined) { - obj.path = handler(obj.path); - } - else { - if (obj.library !== undefined && obj.subContentId !== undefined) { - // Avoid multiple content with same ID - delete obj.subContentId; - } - recursiveUpdate(obj, handler); - } - } - } - }; - - // Init H5P when page is fully loadded - $(document).ready(function () { - - window.addEventListener('storage', function (event) { - // Pick up clipboard changes from other tabs - if (event.key === 'h5pClipboard') { - // Trigger an event so all 'Paste' buttons may be enabled. - H5P.externalDispatcher.trigger('datainclipboard', {reset: event.newValue === null}); - } - }); - - var ccVersions = { - 'default': '4.0', - '4.0': H5P.t('licenseCC40'), - '3.0': H5P.t('licenseCC30'), - '2.5': H5P.t('licenseCC25'), - '2.0': H5P.t('licenseCC20'), - '1.0': H5P.t('licenseCC10'), - }; - - /** - * Maps copyright license codes to their human readable counterpart. - * - * @type {Object} - */ - H5P.copyrightLicenses = { - 'U': H5P.t('licenseU'), - 'CC BY': { - label: H5P.t('licenseCCBY'), - link: 'http://creativecommons.org/licenses/by/:version', - versions: ccVersions - }, - 'CC BY-SA': { - label: H5P.t('licenseCCBYSA'), - link: 'http://creativecommons.org/licenses/by-sa/:version', - versions: ccVersions - }, - 'CC BY-ND': { - label: H5P.t('licenseCCBYND'), - link: 'http://creativecommons.org/licenses/by-nd/:version', - versions: ccVersions - }, - 'CC BY-NC': { - label: H5P.t('licenseCCBYNC'), - link: 'http://creativecommons.org/licenses/by-nc/:version', - versions: ccVersions - }, - 'CC BY-NC-SA': { - label: H5P.t('licenseCCBYNCSA'), - link: 'http://creativecommons.org/licenses/by-nc-sa/:version', - versions: ccVersions - }, - 'CC BY-NC-ND': { - label: H5P.t('licenseCCBYNCND'), - link: 'http://creativecommons.org/licenses/by-nc-nd/:version', - versions: ccVersions - }, - 'CC0 1.0': { - label: H5P.t('licenseCC010'), - link: 'https://creativecommons.org/publicdomain/zero/1.0/' - }, - 'GNU GPL': { - label: H5P.t('licenseGPL'), - link: 'http://www.gnu.org/licenses/gpl-:version-standalone.html', - linkVersions: { - 'v3': '3.0', - 'v2': '2.0', - 'v1': '1.0' - }, - versions: { - 'default': 'v3', - 'v3': H5P.t('licenseV3'), - 'v2': H5P.t('licenseV2'), - 'v1': H5P.t('licenseV1') - } - }, - 'PD': { - label: H5P.t('licensePD'), - versions: { - 'CC0 1.0': { - label: H5P.t('licenseCC010'), - link: 'https://creativecommons.org/publicdomain/zero/1.0/' - }, - 'CC PDM': { - label: H5P.t('licensePDM'), - link: 'https://creativecommons.org/publicdomain/mark/1.0/' - } - } - }, - 'ODC PDDL': 'Public Domain Dedication and Licence', - 'CC PDM': { - label: H5P.t('licensePDM'), - link: 'https://creativecommons.org/publicdomain/mark/1.0/' - }, - 'C': H5P.t('licenseC'), - }; - - /** - * Indicates if H5P is embedded on an external page using iframe. - * @member {boolean} H5P.externalEmbed - */ - - // Relay events to top window. This must be done before H5P.init - // since events may be fired on initialization. - if (H5P.isFramed && H5P.externalEmbed === false) { - H5P.externalDispatcher.on('*', function (event) { - window.parent.H5P.externalDispatcher.trigger.call(this, event); - }); - } - - /** - * Prevent H5P Core from initializing. Must be overriden before document ready. - * @member {boolean} H5P.preventInit - */ - if (!H5P.preventInit) { - // Note that this start script has to be an external resource for it to - // load in correct order in IE9. - H5P.init(document.body); - } - - if (H5PIntegration.saveFreq !== false) { - // When was the last state stored - var lastStoredOn = 0; - // Store the current state of the H5P when leaving the page. - var storeCurrentState = function () { - // Make sure at least 250 ms has passed since last save - var currentTime = new Date().getTime(); - if (currentTime - lastStoredOn > 250) { - lastStoredOn = currentTime; - for (var i = 0; i < H5P.instances.length; i++) { - var instance = H5P.instances[i]; - if (instance.getCurrentState instanceof Function || - typeof instance.getCurrentState === 'function') { - var state = instance.getCurrentState(); - if (state !== undefined) { - // Async is not used to prevent the request from being cancelled. - H5P.setUserData(instance.contentId, 'state', state, {deleteOnChange: true, async: false}); - } - } - } - } - }; - // iPad does not support beforeunload, therefore using unload - H5P.$window.one('beforeunload unload', function () { - // Only want to do this once - H5P.$window.off('pagehide beforeunload unload'); - storeCurrentState(); - }); - // pagehide is used on iPad when tabs are switched - H5P.$window.on('pagehide', storeCurrentState); - } - }); - -})(H5P.jQuery); diff --git a/h5p/h5plib/v126/joubel/core/js/jquery.js b/h5p/h5plib/v126/joubel/core/js/jquery.js deleted file mode 100644 index 4f34d7590ccc8..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/jquery.js +++ /dev/null @@ -1,22 +0,0 @@ -/*! jQuery v3.5.1 | (c) JS Foundation and other contributors | jquery.org/license */ -!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(!e.document)throw new Error("jQuery requires a window with a document");return t(e)}:t(e)}("undefined"!=typeof window?window:this,function(C,e){"use strict";var t=[],r=Object.getPrototypeOf,s=t.slice,g=t.flat?function(e){return t.flat.call(e)}:function(e){return t.concat.apply([],e)},u=t.push,i=t.indexOf,n={},o=n.toString,v=n.hasOwnProperty,a=v.toString,l=a.call(Object),y={},m=function(e){return"function"==typeof e&&"number"!=typeof e.nodeType},x=function(e){return null!=e&&e===e.window},E=C.document,c={type:!0,src:!0,nonce:!0,noModule:!0};function b(e,t,n){var r,i,o=(n=n||E).createElement("script");if(o.text=e,t)for(r in c)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function w(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?n[o.call(e)]||"object":typeof e}var f="3.5.1",S=function(e,t){return new S.fn.init(e,t)};function p(e){var t=!!e&&"length"in e&&e.length,n=w(e);return!m(e)&&!x(e)&&("array"===n||0===t||"number"==typeof t&&0+~]|"+M+")"+M+"*"),U=new RegExp(M+"|>"),X=new RegExp(F),V=new RegExp("^"+I+"$"),G={ID:new RegExp("^#("+I+")"),CLASS:new RegExp("^\\.("+I+")"),TAG:new RegExp("^("+I+"|[*])"),ATTR:new RegExp("^"+W),PSEUDO:new RegExp("^"+F),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+M+"*(even|odd|(([+-]|)(\\d*)n|)"+M+"*(?:([+-]|)"+M+"*(\\d+)|))"+M+"*\\)|)","i"),bool:new RegExp("^(?:"+R+")$","i"),needsContext:new RegExp("^"+M+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+M+"*((?:-\\d)?\\d*)"+M+"*\\)|)(?=[^-]|$)","i")},Y=/HTML$/i,Q=/^(?:input|select|textarea|button)$/i,J=/^h\d$/i,K=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ee=/[+~]/,te=new RegExp("\\\\[\\da-fA-F]{1,6}"+M+"?|\\\\([^\\r\\n\\f])","g"),ne=function(e,t){var n="0x"+e.slice(1)-65536;return t||(n<0?String.fromCharCode(n+65536):String.fromCharCode(n>>10|55296,1023&n|56320))},re=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ie=function(e,t){return t?"\0"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e},oe=function(){T()},ae=be(function(e){return!0===e.disabled&&"fieldset"===e.nodeName.toLowerCase()},{dir:"parentNode",next:"legend"});try{H.apply(t=O.call(p.childNodes),p.childNodes),t[p.childNodes.length].nodeType}catch(e){H={apply:t.length?function(e,t){L.apply(e,O.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function se(t,e,n,r){var i,o,a,s,u,l,c,f=e&&e.ownerDocument,p=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==p&&9!==p&&11!==p)return n;if(!r&&(T(e),e=e||C,E)){if(11!==p&&(u=Z.exec(t)))if(i=u[1]){if(9===p){if(!(a=e.getElementById(i)))return n;if(a.id===i)return n.push(a),n}else if(f&&(a=f.getElementById(i))&&y(e,a)&&a.id===i)return n.push(a),n}else{if(u[2])return H.apply(n,e.getElementsByTagName(t)),n;if((i=u[3])&&d.getElementsByClassName&&e.getElementsByClassName)return H.apply(n,e.getElementsByClassName(i)),n}if(d.qsa&&!N[t+" "]&&(!v||!v.test(t))&&(1!==p||"object"!==e.nodeName.toLowerCase())){if(c=t,f=e,1===p&&(U.test(t)||z.test(t))){(f=ee.test(t)&&ye(e.parentNode)||e)===e&&d.scope||((s=e.getAttribute("id"))?s=s.replace(re,ie):e.setAttribute("id",s=S)),o=(l=h(t)).length;while(o--)l[o]=(s?"#"+s:":scope")+" "+xe(l[o]);c=l.join(",")}try{return H.apply(n,f.querySelectorAll(c)),n}catch(e){N(t,!0)}finally{s===S&&e.removeAttribute("id")}}}return g(t.replace($,"$1"),e,n,r)}function ue(){var r=[];return function e(t,n){return r.push(t+" ")>b.cacheLength&&delete e[r.shift()],e[t+" "]=n}}function le(e){return e[S]=!0,e}function ce(e){var t=C.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function fe(e,t){var n=e.split("|"),r=n.length;while(r--)b.attrHandle[n[r]]=t}function pe(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function de(t){return function(e){return"input"===e.nodeName.toLowerCase()&&e.type===t}}function he(n){return function(e){var t=e.nodeName.toLowerCase();return("input"===t||"button"===t)&&e.type===n}}function ge(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&ae(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function ve(a){return le(function(o){return o=+o,le(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function ye(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}for(e in d=se.support={},i=se.isXML=function(e){var t=e.namespaceURI,n=(e.ownerDocument||e).documentElement;return!Y.test(t||n&&n.nodeName||"HTML")},T=se.setDocument=function(e){var t,n,r=e?e.ownerDocument||e:p;return r!=C&&9===r.nodeType&&r.documentElement&&(a=(C=r).documentElement,E=!i(C),p!=C&&(n=C.defaultView)&&n.top!==n&&(n.addEventListener?n.addEventListener("unload",oe,!1):n.attachEvent&&n.attachEvent("onunload",oe)),d.scope=ce(function(e){return a.appendChild(e).appendChild(C.createElement("div")),"undefined"!=typeof e.querySelectorAll&&!e.querySelectorAll(":scope fieldset div").length}),d.attributes=ce(function(e){return e.className="i",!e.getAttribute("className")}),d.getElementsByTagName=ce(function(e){return e.appendChild(C.createComment("")),!e.getElementsByTagName("*").length}),d.getElementsByClassName=K.test(C.getElementsByClassName),d.getById=ce(function(e){return a.appendChild(e).id=S,!C.getElementsByName||!C.getElementsByName(S).length}),d.getById?(b.filter.ID=function(e){var t=e.replace(te,ne);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n=t.getElementById(e);return n?[n]:[]}}):(b.filter.ID=function(e){var n=e.replace(te,ne);return function(e){var t="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return t&&t.value===n}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=d.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):d.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"===e){while(n=o[i++])1===n.nodeType&&r.push(n);return r}return o},b.find.CLASS=d.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&E)return t.getElementsByClassName(e)},s=[],v=[],(d.qsa=K.test(C.querySelectorAll))&&(ce(function(e){var t;a.appendChild(e).innerHTML="",e.querySelectorAll("[msallowcapture^='']").length&&v.push("[*^$]="+M+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||v.push("\\["+M+"*(?:value|"+R+")"),e.querySelectorAll("[id~="+S+"-]").length||v.push("~="),(t=C.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||v.push("\\["+M+"*name"+M+"*="+M+"*(?:''|\"\")"),e.querySelectorAll(":checked").length||v.push(":checked"),e.querySelectorAll("a#"+S+"+*").length||v.push(".#.+[+~]"),e.querySelectorAll("\\\f"),v.push("[\\r\\n\\f]")}),ce(function(e){e.innerHTML="";var t=C.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&v.push("name"+M+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&v.push(":enabled",":disabled"),a.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&v.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),v.push(",.*:")})),(d.matchesSelector=K.test(c=a.matches||a.webkitMatchesSelector||a.mozMatchesSelector||a.oMatchesSelector||a.msMatchesSelector))&&ce(function(e){d.disconnectedMatch=c.call(e,"*"),c.call(e,"[s!='']:x"),s.push("!=",F)}),v=v.length&&new RegExp(v.join("|")),s=s.length&&new RegExp(s.join("|")),t=K.test(a.compareDocumentPosition),y=t||K.test(a.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},D=t?function(e,t){if(e===t)return l=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!d.sortDetached&&t.compareDocumentPosition(e)===n?e==C||e.ownerDocument==p&&y(p,e)?-1:t==C||t.ownerDocument==p&&y(p,t)?1:u?P(u,e)-P(u,t):0:4&n?-1:1)}:function(e,t){if(e===t)return l=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e==C?-1:t==C?1:i?-1:o?1:u?P(u,e)-P(u,t):0;if(i===o)return pe(e,t);n=e;while(n=n.parentNode)a.unshift(n);n=t;while(n=n.parentNode)s.unshift(n);while(a[r]===s[r])r++;return r?pe(a[r],s[r]):a[r]==p?-1:s[r]==p?1:0}),C},se.matches=function(e,t){return se(e,null,null,t)},se.matchesSelector=function(e,t){if(T(e),d.matchesSelector&&E&&!N[t+" "]&&(!s||!s.test(t))&&(!v||!v.test(t)))try{var n=c.call(e,t);if(n||d.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){N(t,!0)}return 0":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(te,ne),e[3]=(e[3]||e[4]||e[5]||"").replace(te,ne),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||se.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&se.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return G.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&X.test(n)&&(t=h(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(te,ne).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=m[e+" "];return t||(t=new RegExp("(^|"+M+")"+e+"("+M+"|$)"))&&m(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(n,r,i){return function(e){var t=se.attr(e,n);return null==t?"!="===r:!r||(t+="","="===r?t===i:"!="===r?t!==i:"^="===r?i&&0===t.indexOf(i):"*="===r?i&&-1:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function D(e,n,r){return m(n)?S.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?S.grep(e,function(e){return e===n!==r}):"string"!=typeof n?S.grep(e,function(e){return-1)[^>]*|#([\w-]+))$/;(S.fn.init=function(e,t,n){var r,i;if(!e)return this;if(n=n||j,"string"==typeof e){if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:q.exec(e))||!r[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(r[1]){if(t=t instanceof S?t[0]:t,S.merge(this,S.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:E,!0)),N.test(r[1])&&S.isPlainObject(t))for(r in t)m(this[r])?this[r](t[r]):this.attr(r,t[r]);return this}return(i=E.getElementById(r[2]))&&(this[0]=i,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):m(e)?void 0!==n.ready?n.ready(e):e(S):S.makeArray(e,this)}).prototype=S.fn,j=S(E);var L=/^(?:parents|prev(?:Until|All))/,H={children:!0,contents:!0,next:!0,prev:!0};function O(e,t){while((e=e[t])&&1!==e.nodeType);return e}S.fn.extend({has:function(e){var t=S(e,this),n=t.length;return this.filter(function(){for(var e=0;e\x20\t\r\n\f]*)/i,he=/^$|^module$|\/(?:java|ecma)script/i;ce=E.createDocumentFragment().appendChild(E.createElement("div")),(fe=E.createElement("input")).setAttribute("type","radio"),fe.setAttribute("checked","checked"),fe.setAttribute("name","t"),ce.appendChild(fe),y.checkClone=ce.cloneNode(!0).cloneNode(!0).lastChild.checked,ce.innerHTML="",y.noCloneChecked=!!ce.cloneNode(!0).lastChild.defaultValue,ce.innerHTML="",y.option=!!ce.lastChild;var ge={thead:[1,"","
    "],col:[2,"","
    "],tr:[2,"","
    "],td:[3,"","
    "],_default:[0,"",""]};function ve(e,t){var n;return n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[],void 0===t||t&&A(e,t)?S.merge([e],n):n}function ye(e,t){for(var n=0,r=e.length;n",""]);var me=/<|&#?\w+;/;function xe(e,t,n,r,i){for(var o,a,s,u,l,c,f=t.createDocumentFragment(),p=[],d=0,h=e.length;d\s*$/g;function qe(e,t){return A(e,"table")&&A(11!==t.nodeType?t:t.firstChild,"tr")&&S(e).children("tbody")[0]||e}function Le(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function He(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Oe(e,t){var n,r,i,o,a,s;if(1===t.nodeType){if(Y.hasData(e)&&(s=Y.get(e).events))for(i in Y.remove(t,"handle events"),s)for(n=0,r=s[i].length;n").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),E.head.appendChild(r[0])},abort:function(){i&&i()}}});var Ut,Xt=[],Vt=/(=)\?(?=&|$)|\?\?/;S.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=Xt.pop()||S.expando+"_"+Ct.guid++;return this[e]=!0,e}}),S.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Vt.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Vt.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=m(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Vt,"$1"+r):!1!==e.jsonp&&(e.url+=(Et.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||S.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=C[r],C[r]=function(){o=arguments},n.always(function(){void 0===i?S(C).removeProp(r):C[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,Xt.push(r)),o&&m(i)&&i(o[0]),o=i=void 0}),"script"}),y.createHTMLDocument=((Ut=E.implementation.createHTMLDocument("").body).innerHTML="
    ",2===Ut.childNodes.length),S.parseHTML=function(e,t,n){return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(y.createHTMLDocument?((r=(t=E.implementation.createHTMLDocument("")).createElement("base")).href=E.location.href,t.head.appendChild(r)):t=E),o=!n&&[],(i=N.exec(e))?[t.createElement(i[1])]:(i=xe([e],t,o),o&&o.length&&S(o).remove(),S.merge([],i.childNodes)));var r,i,o},S.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1").append(S.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},S.expr.pseudos.animated=function(t){return S.grep(S.timers,function(e){return t===e.elem}).length},S.offset={setOffset:function(e,t,n){var r,i,o,a,s,u,l=S.css(e,"position"),c=S(e),f={};"static"===l&&(e.style.position="relative"),s=c.offset(),o=S.css(e,"top"),u=S.css(e,"left"),("absolute"===l||"fixed"===l)&&-1<(o+u).indexOf("auto")?(a=(r=c.position()).top,i=r.left):(a=parseFloat(o)||0,i=parseFloat(u)||0),m(t)&&(t=t.call(e,n,S.extend({},s))),null!=t.top&&(f.top=t.top-s.top+a),null!=t.left&&(f.left=t.left-s.left+i),"using"in t?t.using.call(e,f):("number"==typeof f.top&&(f.top+="px"),"number"==typeof f.left&&(f.left+="px"),c.css(f))}},S.fn.extend({offset:function(t){if(arguments.length)return void 0===t?this:this.each(function(e){S.offset.setOffset(this,t,e)});var e,n,r=this[0];return r?r.getClientRects().length?(e=r.getBoundingClientRect(),n=r.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===S.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===S.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=S(e).offset()).top+=S.css(e,"borderTopWidth",!0),i.left+=S.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-S.css(r,"marginTop",!0),left:t.left-i.left-S.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===S.css(e,"position"))e=e.offsetParent;return e||re})}}),S.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;S.fn[t]=function(e){return $(this,function(e,t,n){var r;if(x(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),S.each(["top","left"],function(e,n){S.cssHooks[n]=$e(y.pixelPosition,function(e,t){if(t)return t=Be(e,n),Me.test(t)?S(e).position()[n]+"px":t})}),S.each({Height:"height",Width:"width"},function(a,s){S.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){S.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return $(this,function(e,t,n){var r;return x(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?S.css(e,t,i):S.style(e,t,n,i)},s,n?e:void 0,n)}})}),S.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){S.fn[t]=function(e){return this.on(t,e)}}),S.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),S.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){S.fn[n]=function(e,t){return 00'); - - dialog.querySelector('.h5p-confirmation-dialog-text').appendChild(countDownText); - const countDownNum = countDownText.querySelector('.count-down-num'); - - // Create throbber - const throbberWrapper = document.createElement('div'); - throbberWrapper.classList.add('throbber-wrapper'); - const throbber = document.createElement('div'); - throbber.classList.add('sending-requests-throbber'); - throbberWrapper.appendChild(throbber); - - requestQueue.on('requestQueued', function (e) { - // Already processing queue, wait until queue has finished processing before showing dialog - if (e.data && e.data.processingQueue) { - return; - } - - if (!isAttached) { - const rootContent = document.body.querySelector('.h5p-content'); - if (!rootContent) { - return; - } - offlineDialog.appendTo(rootContent); - rootContent.appendChild(throbberWrapper); - isAttached = true; - } - - startCountDown(); - }.bind(this)); - - requestQueue.on('queueEmptied', function (e) { - if (e.data && e.data.length) { - // New requests were added while processing queue or requests failed again. Re-queue requests. - startCountDown(true); - return; - } - - // Successfully emptied queue - clearInterval(currentInterval); - toggleThrobber(false); - intervalIndex = -1; - if (isShowing) { - offlineDialog.hide(); - isShowing = false; - } - - requestQueue.displayToastMessage( - H5P.t('offlineSuccessfulSubmit'), - true, - { - position: { - vertical: 'top', - offsetVertical: '100', - } - } - ); - - }.bind(this)); - - offlineDialog.on('confirmed', function () { - // Show dialog on next render in case it is being hidden by the 'confirm' button - isShowing = false; - setTimeout(function () { - retryRequests(); - }, 100); - }.bind(this)); - - // Initialize listener for when requests are added to queue - window.addEventListener('online', function () { - retryRequests(); - }.bind(this)); - - // Listen for queued requests outside the iframe - window.addEventListener('message', function (event) { - const isValidQueueEvent = window.parent === event.source - && event.data.context === 'h5p' - && event.data.action === 'queueRequest'; - - if (!isValidQueueEvent) { - return; - } - - this.add(event.data.url, event.data.data); - }.bind(this)); - - /** - * Toggle throbber visibility - * - * @param {boolean} [forceShow] Will force throbber visibility if set - */ - const toggleThrobber = function (forceShow) { - isLoading = !isLoading; - if (forceShow !== undefined) { - isLoading = forceShow; - } - - if (isLoading && isShowing) { - offlineDialog.hide(); - isShowing = false; - } - - if (isLoading) { - throbberWrapper.classList.add('show'); - } - else { - throbberWrapper.classList.remove('show'); - } - }; - /** - * Retries the failed requests - */ - const retryRequests = function () { - clearInterval(currentInterval); - toggleThrobber(true); - requestQueue.resumeQueue(); - }; - - /** - * Increments retry interval - */ - const incrementRetryInterval = function () { - intervalIndex += 1; - if (intervalIndex >= retryIntervals.length) { - intervalIndex = retryIntervals.length - 1; - } - }; - - /** - * Starts counting down to retrying queued requests. - * - * @param forceDelayedShow - */ - const startCountDown = function (forceDelayedShow) { - // Already showing, wait for retry - if (isShowing) { - return; - } - - toggleThrobber(false); - if (!isShowing) { - if (forceDelayedShow) { - // Must force delayed show since dialog may be hiding, and confirmation dialog does not - // support this. - setTimeout(function () { - offlineDialog.show(0); - }, 100); - } - else { - offlineDialog.show(0); - } - } - isShowing = true; - startTime = new Date().getTime(); - incrementRetryInterval(); - clearInterval(currentInterval); - currentInterval = setInterval(updateCountDown, 100); - }; - - /** - * Updates the count down timer. Retries requests when time expires. - */ - const updateCountDown = function () { - const time = new Date().getTime(); - const timeElapsed = Math.floor((time - startTime) / 1000); - const timeLeft = retryIntervals[intervalIndex] - timeElapsed; - countDownNum.textContent = timeLeft.toString(); - - // Retry interval reached, retry requests - if (timeLeft <= 0) { - retryRequests(); - } - }; - - /** - * Add request to offline request queue. Only supports posts for now. - * - * @param {string} url The request url - * @param {Object} data The request data - */ - this.add = function (url, data) { - // Only queue request if it failed because we are offline - if (window.navigator.onLine) { - return false; - } - - requestQueue.add(url, data); - }; - }; - - return offlineRequestQueue; -})(H5P.RequestQueue, H5P.ConfirmationDialog); diff --git a/h5p/h5plib/v126/joubel/core/js/settings/h5p-disable-hub.js b/h5p/h5plib/v126/joubel/core/js/settings/h5p-disable-hub.js deleted file mode 100644 index 406e8b2d1e957..0000000000000 --- a/h5p/h5plib/v126/joubel/core/js/settings/h5p-disable-hub.js +++ /dev/null @@ -1,68 +0,0 @@ -/* global H5PDisableHubData */ - -/** - * Global data for disable hub functionality - * - * @typedef {object} H5PDisableHubData Data passed in from the backend - * - * @property {string} selector Selector for the disable hub check-button - * @property {string} overlaySelector Selector for the element that the confirmation dialog will mask - * @property {Array} errors Errors found with the current server setup - * - * @property {string} header Header of the confirmation dialog - * @property {string} confirmationDialogMsg Body of the confirmation dialog - * @property {string} cancelLabel Cancel label of the confirmation dialog - * @property {string} confirmLabel Confirm button label of the confirmation dialog - * - */ -/** - * Utility that makes it possible to force the user to confirm that he really - * wants to use the H5P hub without proper server settings. - */ -(function ($) { - - $(document).on('ready', function () { - - // No data found - if (!H5PDisableHubData) { - return; - } - - // No errors found, no need for confirmation dialog - if (!H5PDisableHubData.errors || !H5PDisableHubData.errors.length) { - return; - } - - H5PDisableHubData.selector = H5PDisableHubData.selector || - '.h5p-settings-disable-hub-checkbox'; - H5PDisableHubData.overlaySelector = H5PDisableHubData.overlaySelector || - '.h5p-settings-container'; - - var dialogHtml = '
    ' + - '

    ' + H5PDisableHubData.errors.join('

    ') + '

    ' + - '

    ' + H5PDisableHubData.confirmationDialogMsg + '

    '; - - // Create confirmation dialog, make sure to include translations - var confirmationDialog = new H5P.ConfirmationDialog({ - headerText: H5PDisableHubData.header, - dialogText: dialogHtml, - cancelText: H5PDisableHubData.cancelLabel, - confirmText: H5PDisableHubData.confirmLabel - }).appendTo($(H5PDisableHubData.overlaySelector).get(0)); - - confirmationDialog.on('confirmed', function () { - enableButton.get(0).checked = true; - }); - - confirmationDialog.on('canceled', function () { - enableButton.get(0).checked = false; - }); - - var enableButton = $(H5PDisableHubData.selector); - enableButton.change(function () { - if ($(this).is(':checked')) { - confirmationDialog.show(enableButton.offset().top); - } - }); - }); -})(H5P.jQuery); diff --git a/h5p/h5plib/v126/joubel/core/readme_moodle.txt b/h5p/h5plib/v126/joubel/core/readme_moodle.txt deleted file mode 100644 index 862029dcc9b04..0000000000000 --- a/h5p/h5plib/v126/joubel/core/readme_moodle.txt +++ /dev/null @@ -1,60 +0,0 @@ -H5P PHP library ---------------- - -Downloaded last release from: https://github.com/h5p/h5p-php-library/tags - -When no new tags are released, a specific commit can also be used. In that case, the version number used in the thirdpartylibs.xml -will be -. For instance, master-f3579c0. - -Import procedure: - * Remove the content in this folder (but the readme_moodle.txt) - * Copy all the files from the folder repository in this directory. - -Removed: - * composer.json - * .gitignore - * .travis.yml - -Changed: - 0. Open the new version of joubel/core/h5p.classes.php and at the beginning of the H5PCore class (around line 2082), check the - value of the coreApi minor and major versions. If they are different from the values in the current Moodle version, instead of - upgrading the library in the current h5plib_vxxx, a new h5plib_vX.Y should be released (as it was done in MDL-80544). The new - h5plib module should be named taking into account that X is the coreApi.majorVersion and Y is the coreApi.minorVersion. - - 1. Replace the $_SESSION references with $SESSION. That implies that the information is saved to backends, so only the Moodle one - should be used by core (core should be free from $_SESSION and always use $SESSION). - More specifically, in h5p.classes.php file, into hashToken() method: - * Declare the global $SESSION. - * Change all the $_SESSION by $SESSION. - * Change all the $_SESSION['xxxx'] by $SESSION->xxxx. - A script for testing this part can be found in MDL-68068 - -2. Add namespace to this library to avoid collision. It means: - - Add the "namespace Moodle;" line at the top of all the h5p*.php files in the root folder. - - Replace \H5Pxxx uses to H5Pxxx (for instance, in h5p-default-storage.class.php there are several references to \H5PCore that - must be replaced with H5PCore). - - Add "use ZipArchive;" in h5p.classes.h5p (check that it's still used before replacing it when upgrading the library). - -3. Check if there are changes in the getLocalization() method in h5p.classes.php and update lang/en/h5p.php accordingly. - If there are changes, check the t() method in h5p/classes/framework.php too (updating or adding new ones). - -4. In saveLibraries() method in core/h5p.classes.php, check $this->h5pF->saveLibraryData is called before $this->h5pC->fs->saveLibrary. -The library needs to be saved in the database first before creating the files, because the libraryid is used as itemid for the files. - -5. Check if new methods have been added to any of the interfaces. If that's the case, implement them in the proper class. For -instance, if a new method is added to h5p-file-storage.interface.php, it should be implemented in h5p/classes/file_storage.php. - -6. Open js/h5p.js and in function contentUserDataAjax() add the following patch: - function contentUserDataAjax(contentId, dataType, subContentId, done, data, preload, invalidate, async) { - if (H5PIntegration.user === undefined) { - // Not logged in, no use in saving. - done('Not signed in.'); - return; - } - // Moodle patch to let override this method. - if (H5P.contentUserDataAjax !== undefined) { - return H5P.contentUserDataAjax(contentId, dataType, subContentId, done, data, preload, invalidate, async); - } - // End of Moodle patch. - - var options = { diff --git a/h5p/h5plib/v126/joubel/core/styles/font-open-sans.css b/h5p/h5plib/v126/joubel/core/styles/font-open-sans.css deleted file mode 100644 index 526559c86795a..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/font-open-sans.css +++ /dev/null @@ -1,494 +0,0 @@ -/* -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. -*/ - -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: italic; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-italic-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 400; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 600; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} -/* cyrillic-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic-ext.woff2') format('woff2'); - unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F; -} -/* cyrillic */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-cyrillic.woff2') format('woff2'); - unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116; -} -/* greek-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek-ext.woff2') format('woff2'); - unicode-range: U+1F00-1FFF; -} -/* greek */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-greek.woff2') format('woff2'); - unicode-range: U+0370-03FF; -} -/* hebrew */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-hebrew.woff2') format('woff2'); - unicode-range: U+0590-05FF, U+200C-2010, U+20AA, U+25CC, U+FB1D-FB4F; -} -/* vietnamese */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-vietnamese.woff2') format('woff2'); - unicode-range: U+0102-0103, U+0110-0111, U+0128-0129, U+0168-0169, U+01A0-01A1, U+01AF-01B0, U+1EA0-1EF9, U+20AB; -} -/* latin-ext */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin-ext.woff2') format('woff2'); - unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF; -} -/* latin */ -@font-face { - font-family: 'Open Sans'; - font-style: normal; - font-weight: 700; - font-stretch: 100%; - font-display: swap; - src: url('../fonts/open-sans/opensans-400-600-700-v28-latin.woff2') format('woff2'); - unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD; -} \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-admin.css b/h5p/h5plib/v126/joubel/core/styles/h5p-admin.css deleted file mode 100644 index 68306126fc01b..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-admin.css +++ /dev/null @@ -1,358 +0,0 @@ -/* Administration interface styling */ - -.h5p-content { - border: 1px solid #DDD; - border-radius: 3px; - padding: 10px; -} - -.h5p-admin-table, -.h5p-admin-table > tbody { - border: none; - width: 100%; -} - -.h5p-admin-table tr:nth-child(odd), -.h5p-data-view tr:nth-child(odd) { - background-color: #F9F9F9; -} -.h5p-admin-table tbody tr:hover { - background-color: #EEE; -} -.h5p-admin-table.empty { - padding: 1em; - background-color: #EEE; - font-size: 1.2em; - font-weight: bold; -} - -.h5p-admin-table.libraries th:last-child, -.h5p-admin-table.libraries td:last-child { - text-align: right; -} - -.h5p-admin-buttons-wrapper { - white-space: nowrap; -} - -.h5p-admin-table.libraries button { - font-size: 2em; - cursor: pointer; - border: 1px solid #AAA; - border-radius: .2em; - background-color: #e0e0e0; - text-shadow: 0 0 0.5em #fff; - padding: 0; - line-height: 1em; - width: 1.125em; - height: 1.05em; - text-indent: -0.125em; - margin: 0.125em 0.125em 0 0.125em; -} -.h5p-admin-upgrade-library:before { - font-family: 'H5P'; - content: "\e888"; -} -.h5p-admin-view-library:before { - font-family: 'H5P'; - content: "\e889"; -} -.h5p-admin-delete-library:before { - font-family: 'H5P'; - content: "\e890"; -} - -.h5p-admin-table.libraries button:hover { - background-color: #d0d0d0; -} -.h5p-admin-table.libraries button:disabled:hover { - background-color: #e0e0e0; - cursor: default; -} - -.h5p-admin-upgrade-library { - color: #339900; -} -.h5p-admin-view-library { - color: #0066cc; -} -.h5p-admin-delete-library { - color: #990000; -} -.h5p-admin-delete-library:disabled, -.h5p-admin-upgrade-library:disabled { - cursor: default; - color: #c0c0c0; -} - -.h5p-library-info { - padding: 1em 1em; - margin: 1em 0; - - width: 350px; - - border: 1px solid #DDD; - border-radius: 3px; -} - -/* Labeled field (label + value) */ -.h5p-labeled-field { - border-bottom: 1px solid #ccc; -} -.h5p-labeled-field:last-child { - border-bottom: none; -} - -.h5p-labeled-field .h5p-label { - display: inline-block; - min-width: 150px; - font-size: 1.2em; - font-weight: bold; - padding: 0.2em; -} - -.h5p-labeled-field .h5p-value { - display: inline-block; - padding: 0.2em; -} - -/* Search element */ -.h5p-content-search { - display: inline-block; - position: relative; - - width: 100%; - padding: 5px 0; - margin-top: 10px; - - border: 1px solid #CCC; - border-radius: 3px; - box-shadow: 2px 2px 5px #888888; -} -.h5p-content-search:before { - font-family: 'H5P'; - vertical-align: bottom; - content: "\e88a"; - font-size: 2em; - line-height: 1.25em; -} -.h5p-content-search input { - font-size: 120%; - line-height: 120%; -} -.h5p-admin-search-results { - margin-left: 10px; - color: #888; -} - -.h5p-admin-pager-size-selector { - position: absolute; - right: 10px; - top: .75em; - display: inline-block; -} -.h5p-admin-pager-size-selector > span { - padding: 5px; - margin-left: 10px; - cursor: pointer; - border: 1px solid #CCC; - border-radius: 3px; -} -.h5p-admin-pager-size-selector > span.selected { - background-color: #edf5fa; -} -.h5p-admin-pager-size-selector > span:hover { - background-color: #555; - color: #FFF; -} - -/* Generic "javascript"-action button */ -button.h5p-admin { - border: 1px solid #AAA; - border-radius: 5px; - padding: 3px 10px; - background-color: #EEE; - cursor: pointer; - display: inline-block; - text-align: center; - color: #222; -} -button.h5p-admin:hover { - background-color: #555; - color: #FFF; -} -button.h5p-admin.disabled, -button.h5p-admin.disabled:hover { - cursor: auto; - color: #CCC; - background-color: #FFF; -} - -/* Pager element */ -.h5p-content-pager { - display: inline-block; - border: 1px solid #CCC; - border-radius: 3px; - box-shadow: 2px 2px 5px #888888; - width: 100%; - text-align: center; - padding: 3px 0; -} -.h5p-content-pager > button { - min-width: 80px; - font-size: 130%; - line-height: 130%; - border: none; - background: none; - font-family: 'H5P'; - font-size: 1.4em; -} -.h5p-content-pager > button:focus { - outline: 0; -} -.h5p-content-pager > button:last-child { - margin-left: 10px; -} -.h5p-content-pager > .pager-info { - cursor: pointer; - padding: 5px; - border-radius: 3px; -} -.h5p-content-pager > .pager-info:hover { - background-color: #555; - color: #FFF; -} -.h5p-content-pager > .pager-info, -.h5p-content-pager > .h5p-pager-goto { - margin: 0 10px; - line-height: 130%; - display: inline-block; -} - -.h5p-admin-header { - margin-top: 1.5em; -} -#h5p-library-upload-form.h5p-admin-upload-libraries-form, -#h5p-content-type-cache-update-form.h5p-admin-upload-libraries-form { - position: relative; - margin: 0; - -} -.h5p-admin-upload-libraries-form .form-submit { - position: absolute; - top: 0; - right: 0; -} -.h5p-spinner { - padding: 0 0.5em; - font-size: 1.5em; - font-weight: bold; -} -#h5p-admin-container .h5p-admin-center { - text-align: center; -} -.h5p-pagination { - text-align: center; -} -.h5p-pagination > span, .h5p-pagination > input { - margin: 0 1em; -} -.h5p-data-view input[type="text"] { - margin-bottom: 0.5em; - margin-right: 0.5em; - float: left; -} -.h5p-data-view input[type="text"]::-ms-clear { - display: none; -} - -.h5p-data-view .h5p-others-contents-toggler-wrapper { - float: right; - line-height: 2; - margin-right: 0.5em; -} - -.h5p-data-view .h5p-others-contents-toggler-label { - font-size: 14px; -} - -.h5p-data-view .h5p-others-contents-toggler { - margin-right: 0.5em; -} - -.h5p-data-view th[role="button"] { - cursor: pointer; -} -.h5p-data-view th[role="button"].h5p-sort:after, -.h5p-data-view th[role="button"]:hover:after, -.h5p-data-view th[role="button"].h5p-sort.h5p-reverse:hover:after { - content: "\25BE"; - position: relative; - left: 0.5em; - top: -1px; -} -.h5p-data-view th[role="button"].h5p-sort.h5p-reverse:after, -.h5p-data-view th[role="button"].h5p-sort:hover:after { - content: "\25B4"; - top: -2px; -} -.h5p-data-view th[role="button"]:hover:after, -.h5p-data-view th[role="button"].h5p-sort.h5p-reverse:hover:after, -.h5p-data-view th[role="button"].h5p-sort:hover:after { - color: #999; -} -.h5p-data-view .h5p-facet { - cursor: pointer; - color: #0073aa; - outline: none; -} -.h5p-data-view .h5p-facet:hover, -.h5p-data-view .h5p-facet:active { - color: #00a0d2; -} -.h5p-data-view .h5p-facet:focus { - color: #124964; - box-shadow: 0 0 0 1px #5b9dd9,0 0 2px 1px rgba(30,140,190,.8); -} -.h5p-data-view .h5p-facet-wrapper { - line-height: 23px; -} -.h5p-data-view .h5p-facet-tag { - margin: 2px 0 0 0.5em; - font-size: 12px; - background: #e8e8e8; - border: 1px solid #cbcbcc; - border-radius: 5px; - color: #5d5d5d; - padding: 0 24px 0 10px; - display: inline-block; - position: relative; -} -.h5p-data-view .h5p-facet-tag > span { - position: absolute; - right: 0; - top: auto; - bottom: auto; - font-size: 18px; - color: #a2a2a2; - outline: none; - width: 21px; - text-indent: 4px; - letter-spacing: 10px; - overflow: hidden; - cursor: pointer; -} -.h5p-data-view .h5p-facet-tag > span:before { - content: "×"; - font-weight: bold; -} -.h5p-data-view .h5p-facet-tag > span:hover, -.h5p-data-view .h5p-facet-tag > span:focus { - color: #a20000; -} -.h5p-data-view .h5p-facet-tag > span:active { - color: #d20000; -} -.content-upgrade-log { - color: red; -} diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-confirmation-dialog.css b/h5p/h5plib/v126/joubel/core/styles/h5p-confirmation-dialog.css deleted file mode 100644 index e849c23e6c6b9..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-confirmation-dialog.css +++ /dev/null @@ -1,183 +0,0 @@ -.h5p-confirmation-dialog-background { - position: fixed; - height: 100%; - width: 100%; - left: 0; - top: 0; - - background: rgba(44, 44, 44, 0.9); - opacity: 1; - visibility: visible; - -webkit-transition: opacity 0.1s, linear 0s, visibility 0s linear 0s; - transition: opacity 0.1s linear 0s, visibility 0s linear 0s; - - z-index: 201; -} - -.h5p-confirmation-dialog-background.hidden { - display: none; -} - -.h5p-confirmation-dialog-background.hiding { - opacity: 0; - visibility: hidden; - -webkit-transition: opacity 0.1s, linear 0s, visibility 0s linear 0.1s; - transition: opacity 0.1s linear 0s, visibility 0s linear 0.1s; -} - -.h5p-confirmation-dialog-popup:focus { - outline: none; -} - -.h5p-confirmation-dialog-popup { - position: absolute; - display: flex; - flex-direction: column; - justify-content: center; - - box-sizing: border-box; - max-width: 35em; - min-width: 25em; - - top: 2em; - left: 50%; - -webkit-transform: translate(-50%, 0%); - -ms-transform: translate(-50%, 0%); - transform: translate(-50%, 0%); - - color: #555; - box-shadow: 0 0 6px 6px rgba(10,10,10,0.3); - - -webkit-transition: transform 0.1s ease-in; - transition: transform 0.1s ease-in; -} - -.h5p-confirmation-dialog-popup.hidden { - -webkit-transform: translate(-50%, 50%); - -ms-transform: translate(-50%, 50%); - transform: translate(-50%, 50%); -} - -.h5p-confirmation-dialog-header { - padding: 1.5em; - background: #fff; - color: #356593; -} - -.h5p-confirmation-dialog-header-text { - font-size: 1.25em; -} - -.h5p-confirmation-dialog-body { - background: #fafbfc; - border-top: solid 1px #dde0e9; - padding: 1.25em 1.5em; -} - -.h5p-confirmation-dialog-text { - margin-bottom: 1.5em; -} - -.h5p-confirmation-dialog-buttons { - float: right; -} - -button.h5p-confirmation-dialog-exit:visited, -button.h5p-confirmation-dialog-exit:link, -button.h5p-confirmation-dialog-exit { - position: absolute; - background: none; - border: none; - font-size: 2.5em; - top: -0.9em; - right: -1.15em; - color: #fff; - cursor: pointer; - text-decoration: none; -} - -button.h5p-confirmation-dialog-exit:focus, -button.h5p-confirmation-dialog-exit:hover { - color: #E4ECF5; -} - -.h5p-confirmation-dialog-exit:before { - font-family: "H5P"; - content: "\e890"; -} - -.h5p-core-button.h5p-confirmation-dialog-confirm-button { - padding-left: 0.75em; - margin-bottom: 0; -} - -.h5p-core-button.h5p-confirmation-dialog-confirm-button:before { - content: "\e601"; - margin-top: -6px; - display: inline-block; -} - -.h5p-confirmation-dialog-popup.offline .h5p-confirmation-dialog-buttons { - float: none; - text-align: center; -} - -.h5p-confirmation-dialog-popup.offline .count-down { - font-family: Arial; - margin-top: 0.15em; - color: #000; -} - -.h5p-confirmation-dialog-popup.offline .h5p-confirmation-dialog-confirm-button:before { - content: "\e90b"; - font-weight: normal; - vertical-align: text-bottom; -} - -.throbber-wrapper { - display: none; - position: absolute; - height: 100%; - width: 100%; - top: 0; - left: 0; - z-index: 1; - background: rgba(44, 44, 44, 0.9); -} - -.throbber-wrapper.show { - display: block; -} - -.throbber-wrapper .throbber-container { - position: absolute; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); -} - -.throbber-wrapper .sending-requests-throbber{ - position: absolute; - top: 7em; - left: 50%; - transform: translateX(-50%); -} - -.throbber-wrapper .sending-requests-throbber:before { - display: block; - font-family: 'H5P'; - content: "\e90b"; - color: white; - font-size: 10em; - animation: request-throbber 1.5s infinite linear; -} - -@keyframes request-throbber { - from { - transform: rotate(0); - } - - to { - transform: rotate(359deg); - } -} diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-core-button.css b/h5p/h5plib/v126/joubel/core/styles/h5p-core-button.css deleted file mode 100644 index e6511dc38e52d..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-core-button.css +++ /dev/null @@ -1,60 +0,0 @@ -button.h5p-core-button:visited, -button.h5p-core-button:link, -button.h5p-core-button { - font-family: "Open Sans", sans-serif; - font-weight: 600; - font-size: 1em; - line-height: 1.2; - padding: 0.5em 1.25em; - border-radius: 2em; - - background: #2579c6; - color: #fff; - - cursor: pointer; - border: none; - box-shadow: none; - outline: none; - - display: inline-block; - text-align: center; - text-shadow: none; - vertical-align: baseline; - text-decoration: none; - - -webkit-transition: initial; - transition: initial; -} -button.h5p-core-button:focus { - background: #1f67a8; -} -button.h5p-core-button:hover { - background: rgba(31, 103, 168, 0.83); -} -button.h5p-core-button:active { - background: #104888; -} -button.h5p-core-button:before { - font-family: 'H5P'; - padding-right: 0.15em; - font-size: 1.5em; - vertical-align: middle; - line-height: 0.7; -} -button.h5p-core-cancel-button:visited, -button.h5p-core-cancel-button:link, -button.h5p-core-cancel-button { - border: none; - background: none; - color: #a00; - margin-right: 1em; - font-size: 1em; - text-decoration: none; - cursor: pointer; -} -button.h5p-core-cancel-button:hover, -button.h5p-core-cancel-button:focus { - background: none; - border: none; - color: #e40000; -} diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-hub-registration.css b/h5p/h5plib/v126/joubel/core/styles/h5p-hub-registration.css deleted file mode 100644 index 2972a103a8e2e..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-hub-registration.css +++ /dev/null @@ -1,368 +0,0 @@ -.h5p-hub-publish button,.h5p-hub-registration button{font-family:"Open Sans", sans-serif;font-size:0.75em;background:transparent;padding:0.75em 2em;color:#757575;border-radius:3em;border:1px solid #d6d6d6;cursor:pointer;font-weight:bold}.h5p-hub-publish button:before,.h5p-hub-registration button:before{font-weight:normal}.h5p-hub-publish button:hover,.h5p-hub-registration button:hover{color:#5f5f5f;border-color:#5f5f5f}.h5p-hub-publish button.h5p-hub-contained.h5p-hub-green,.h5p-hub-registration button.h5p-hub-contained.h5p-hub-green{background-color:#186df7;color:#fff;border-color:#186df7}.h5p-hub-publish button.h5p-hub-contained.h5p-hub-green:hover,.h5p-hub-registration button.h5p-hub-contained.h5p-hub-green:hover{background-color:#2e46a4;border-color:#2e46a4}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green,.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green{border-width:0.125em;background-color:white;border-color:#186df7;color:#186df7}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green:hover,.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green:hover{background-color:#186df7;border-color:#186df7;color:#ffffff}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green[disabled],.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green[disabled]{color:#aac3ec;border-color:#aac3ec;background-color:white;pointer-events:none}.h5p-hub-publish .h5p-hub-navigation button{padding:0.833em;max-width:10.5em} - -.h5p-hub-publish .h5p-hub-form-element,.h5p-hub-registration .h5p-hub-form-element{font-family:Nunito, sans-serif;font-size:16px;padding:1em 0}.h5p-hub-publish .h5p-hub-form-element label,.h5p-hub-registration .h5p-hub-form-element label{display:block;margin-bottom:0.4em;font-weight:bold}.h5p-hub-publish .h5p-hub-form-element.h5p-hub-mandatory label:after,.h5p-hub-registration .h5p-hub-form-element.h5p-hub-mandatory label:after{content:'*';color:#d11e15;position:relative;top:-0.2em}.h5p-hub-publish .h5p-hub-form-element input,.h5p-hub-registration .h5p-hub-form-element input{border-radius:0.208em;width:100%;box-sizing:border-box;line-height:normal}.h5p-hub-publish .h5p-hub-form-element input::placeholder,.h5p-hub-registration .h5p-hub-form-element input::placeholder{font-style:italic}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-description,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-description{font-size:0.833em;color:#707475;margin:0 0 1em 0}.h5p-hub-publish .h5p-hub-form-element textarea,.h5p-hub-registration .h5p-hub-form-element textarea{resize:none;width:100%;border-radius:0.208em;box-sizing:border-box;line-height:1.25em}.h5p-hub-publish .h5p-hub-form-element textarea::-webkit-scrollbar,.h5p-hub-registration .h5p-hub-form-element textarea::-webkit-scrollbar{background-color:rgba(105,117,133,0.25);width:0.333em;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-form-element textarea::-webkit-scrollbar-thumb,.h5p-hub-registration .h5p-hub-form-element textarea::-webkit-scrollbar-thumb{background-color:#697585;background-size:24px 100%;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-form-element textarea::placeholder,.h5p-hub-registration .h5p-hub-form-element textarea::placeholder{font-style:italic}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-link-button,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-link-button{border:none;text-decoration:underline;text-align:right;align-self:flex-start;color:#186df7;font-size:.917em;font-weight:normal;padding:0}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-details-row,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-details-row{display:flex;align-content:center;align-items:center;justify-content:space-between}.h5p-hub-publish .h5p-hub-form-element textarea,.h5p-hub-publish .h5p-hub-form-element input,.h5p-hub-publish .h5p-hub-form-element select,.h5p-hub-registration .h5p-hub-form-element textarea,.h5p-hub-registration .h5p-hub-form-element input,.h5p-hub-registration .h5p-hub-form-element select{padding:.958em;border:2px solid #e2e5ee;box-shadow:none;color:#3c4859;font-family:Nunito, sans-serif}.h5p-hub-publish .h5p-hub-form-element textarea:focus,.h5p-hub-publish .h5p-hub-form-element input:focus,.h5p-hub-publish .h5p-hub-form-element select:focus,.h5p-hub-registration .h5p-hub-form-element textarea:focus,.h5p-hub-registration .h5p-hub-form-element input:focus,.h5p-hub-registration .h5p-hub-form-element select:focus{border-color:#98cbe8;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 8px rgba(152,203,232,0.6);outline:none}.h5p-hub-publish .h5p-hub-step-content>.h5p-hub-form-element:first-child,.h5p-hub-registration .h5p-hub-step-content>.h5p-hub-form-element:first-child{padding-top:0}.h5p-hub-publish .h5p-hub-step-content>.h5p-hub-form-element:last-child,.h5p-hub-registration .h5p-hub-step-content>.h5p-hub-form-element:last-child{padding-bottom:0} - -/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ - -/* Document - ========================================================================== */ - -/** - * 1. Correct the line height in all browsers. - * 2. Prevent adjustments of font size after orientation changes in iOS. - */ - -html { - line-height: 1.15; /* 1 */ - -webkit-text-size-adjust: 100%; /* 2 */ -} - -/* Sections - ========================================================================== */ - -/** - * Remove the margin in all browsers. - */ - -body { - margin: 0; -} - -/** - * Render the `main` element consistently in IE. - */ - -main { - display: block; -} - -/** - * Correct the font size and margin on `h1` elements within `section` and - * `article` contexts in Chrome, Firefox, and Safari. - */ - -h1 { - font-size: 2em; - margin: 0.67em 0; -} - -/* Grouping content - ========================================================================== */ - -/** - * 1. Add the correct box sizing in Firefox. - * 2. Show the overflow in Edge and IE. - */ - -hr { - box-sizing: content-box; /* 1 */ - height: 0; /* 1 */ - overflow: visible; /* 2 */ -} - -/** - * 1. Correct the inheritance and scaling of font size in all browsers. - * 2. Correct the odd `em` font sizing in all browsers. - */ - -pre { - font-family: monospace, monospace; /* 1 */ - font-size: 1em; /* 2 */ -} - -/* Text-level semantics - ========================================================================== */ - -/** - * Remove the gray background on active links in IE 10. - */ - -a { - background-color: transparent; -} - -/** - * 1. Remove the bottom border in Chrome 57- - * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. - */ - -abbr[title] { - border-bottom: none; /* 1 */ - text-decoration: underline; /* 2 */ - text-decoration: underline dotted; /* 2 */ -} - -/** - * Add the correct font weight in Chrome, Edge, and Safari. - */ - -b, -strong { - font-weight: bolder; -} - -/** - * 1. Correct the inheritance and scaling of font size in all browsers. - * 2. Correct the odd `em` font sizing in all browsers. - */ - -code, -kbd, -samp { - font-family: monospace, monospace; /* 1 */ - font-size: 1em; /* 2 */ -} - -/** - * Add the correct font size in all browsers. - */ - -small { - font-size: 80%; -} - -/** - * Prevent `sub` and `sup` elements from affecting the line height in - * all browsers. - */ - -sub, -sup { - font-size: 75%; - line-height: 0; - position: relative; - vertical-align: baseline; -} - -sub { - bottom: -0.25em; -} - -sup { - top: -0.5em; -} - -/* Embedded content - ========================================================================== */ - -/** - * Remove the border on images inside links in IE 10. - */ - -img { - border-style: none; -} - -/* Forms - ========================================================================== */ - -/** - * 1. Change the font styles in all browsers. - * 2. Remove the margin in Firefox and Safari. - */ - -button, -input, -optgroup, -select, -textarea { - font-family: inherit; /* 1 */ - font-size: 100%; /* 1 */ - line-height: 1.15; /* 1 */ - margin: 0; /* 2 */ -} - -/** - * Show the overflow in IE. - * 1. Show the overflow in Edge. - */ - -button, -input { /* 1 */ - overflow: visible; -} - -/** - * Remove the inheritance of text transform in Edge, Firefox, and IE. - * 1. Remove the inheritance of text transform in Firefox. - */ - -button, -select { /* 1 */ - text-transform: none; -} - -/** - * Correct the inability to style clickable types in iOS and Safari. - */ - -button, -[type="button"], -[type="reset"], -[type="submit"] { - -webkit-appearance: button; -} - -/** - * Remove the inner border and padding in Firefox. - */ - -button::-moz-focus-inner, -[type="button"]::-moz-focus-inner, -[type="reset"]::-moz-focus-inner, -[type="submit"]::-moz-focus-inner { - border-style: none; - padding: 0; -} - -/** - * Restore the focus styles unset by the previous rule. - */ - -button:-moz-focusring, -[type="button"]:-moz-focusring, -[type="reset"]:-moz-focusring, -[type="submit"]:-moz-focusring { - outline: 1px dotted ButtonText; -} - -/** - * Correct the padding in Firefox. - */ - -fieldset { - padding: 0.35em 0.75em 0.625em; -} - -/** - * 1. Correct the text wrapping in Edge and IE. - * 2. Correct the color inheritance from `fieldset` elements in IE. - * 3. Remove the padding so developers are not caught out when they zero out - * `fieldset` elements in all browsers. - */ - -legend { - box-sizing: border-box; /* 1 */ - color: inherit; /* 2 */ - display: table; /* 1 */ - max-width: 100%; /* 1 */ - padding: 0; /* 3 */ - white-space: normal; /* 1 */ -} - -/** - * Add the correct vertical alignment in Chrome, Firefox, and Opera. - */ - -progress { - vertical-align: baseline; -} - -/** - * Remove the default vertical scrollbar in IE 10+. - */ - -textarea { - overflow: auto; -} - -/** - * 1. Add the correct box sizing in IE 10. - * 2. Remove the padding in IE 10. - */ - -[type="checkbox"], -[type="radio"] { - box-sizing: border-box; /* 1 */ - padding: 0; /* 2 */ -} - -/** - * Correct the cursor style of increment and decrement buttons in Chrome. - */ - -[type="number"]::-webkit-inner-spin-button, -[type="number"]::-webkit-outer-spin-button { - height: auto; -} - -/** - * 1. Correct the odd appearance in Chrome and Safari. - * 2. Correct the outline style in Safari. - */ - -[type="search"] { - -webkit-appearance: textfield; /* 1 */ - outline-offset: -2px; /* 2 */ -} - -/** - * Remove the inner padding in Chrome and Safari on macOS. - */ - -[type="search"]::-webkit-search-decoration { - -webkit-appearance: none; -} - -/** - * 1. Correct the inability to style clickable types in iOS and Safari. - * 2. Change font properties to `inherit` in Safari. - */ - -::-webkit-file-upload-button { - -webkit-appearance: button; /* 1 */ - font: inherit; /* 2 */ -} - -/* Interactive - ========================================================================== */ - -/* - * Add the correct display in Edge, IE 10+, and Firefox. - */ - -details { - display: block; -} - -/* - * Add the correct display in all browsers. - */ - -summary { - display: list-item; -} - -/* Misc - ========================================================================== */ - -/** - * Add the correct display in IE 10+. - */ - -template { - display: none; -} - -/** - * Add the correct display in IE 10. - */ - -[hidden] { - display: none; -} - -@keyframes spinner{0%{transform:translate3d(-50%, -50%, 0) rotate(0deg)}100%{transform:translate3d(-50%, -50%, 0) rotate(360deg)}}.h5p-hub-publish,.h5p-hub-registration-wrapper{position:relative;margin:2em auto;max-width:1024px;border:1px solid #2c9586;border-radius:0.167em;box-sizing:border-box;box-shadow:0px 0px 20px rgba(0,0,0,0.1);font-family:Nunito, sans-serif;font-size:16px;padding-top:0.25em}.h5p-hub-publish .h5p-hub-header,.h5p-hub-registration-wrapper .h5p-hub-header{padding:1.5em;display:flex;align-items:center;width:100%;box-sizing:border-box;border-bottom:1px solid #d5d5d7;border-bottom-color:#e2e5ee}.h5p-hub-publish .h5p-hub-header .h5p-hub-title,.h5p-hub-registration-wrapper .h5p-hub-header .h5p-hub-title{font-size:1.458em;flex-grow:1;color:black;padding-left:1em}.h5p-hub-publish .h5p-hub-step-panel,.h5p-hub-registration-wrapper .h5p-hub-step-panel{margin:2em;border-radius:0.208em;border:1px solid #e2e5ee}.h5p-hub-publish .h5p-hub-step-title,.h5p-hub-registration-wrapper .h5p-hub-step-title{color:#1a1a1a;font-size:1.250em;font-weight:bold;border-bottom:1px solid #ccc;padding:1.2em 1.8em;line-height:normal;border-bottom-color:#e2e5ee}.h5p-hub-publish .h5p-hub-step-content,.h5p-hub-registration-wrapper .h5p-hub-step-content{padding:2em}.h5p-hub-publish .h5p-hub-footer,.h5p-hub-registration-wrapper .h5p-hub-footer{padding:0 2em 2em;width:100%;box-sizing:border-box;position:relative}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation{display:flex}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation button,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation button{z-index:1}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation button:before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation button:before{font-family:'h5p';font-size:0.75em;content:"\e91a";transform:rotate(180deg);display:inline-block;margin-right:0.75em;line-height:1;position:relative;top:-0.1em}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation .h5p-hub-back:before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation .h5p-hub-back:before{transform:rotate(0)}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation #h5p-hub-share-in-process::before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation #h5p-hub-share-in-process::before{font-family:"h5p";font-size:0.75em;content:"\e928";animation:2s linear infinite spinner;top:0.5em;margin-left:0.55em}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation .h5p-hub-next,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation .h5p-hub-next{margin-left:auto}.h5p-hub-publish .h5p-hub-tip-text-field,.h5p-hub-registration-wrapper .h5p-hub-tip-text-field{top:1.5em}.h5p-hub-publish .h5p-hub-hare-error,.h5p-hub-registration-wrapper .h5p-hub-hare-error{padding:0 2em 2em;text-align:end}.h5p-hub-publish .h5p-hub-bold,.h5p-hub-registration-wrapper .h5p-hub-bold{font-weight:bold}.h5p-hub-publish .h5p-hub-row,.h5p-hub-registration-wrapper .h5p-hub-row{display:flex;flex-wrap:wrap;margin-right:-1em}.h5p-hub-publish .h5p-hub-row .h5p-hub-form-element,.h5p-hub-registration-wrapper .h5p-hub-row .h5p-hub-form-element{flex-basis:100%;flex:1;min-width:17em;align-self:flex-end;border-right:1em solid #ffffff}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-row .h5p-hub-form-element,.h5p-hub-registration-wrapper .h5p-hub-row .h5p-hub-form-element{min-width:calc(100% - 1em)}}.h5p-hub-publish .h5p-hub-columns,.h5p-hub-registration-wrapper .h5p-hub-columns{display:flex;margin-right:-1em}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-columns,.h5p-hub-registration-wrapper .h5p-hub-columns{flex-wrap:wrap}}.h5p-hub-publish .h5p-hub-columns .h5p-hub-column,.h5p-hub-registration-wrapper .h5p-hub-columns .h5p-hub-column{width:50%;border-right:1em solid #ffffff}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-columns .h5p-hub-column,.h5p-hub-registration-wrapper .h5p-hub-columns .h5p-hub-column{width:100%;margin-right:0}}.h5p-hub-publish .h5p-hub-sharing-note,.h5p-hub-registration-wrapper .h5p-hub-sharing-note{text-align:center;color:#707475;width:calc(100% - 4.9em);position:absolute;top:0.65em;box-sizing:border-box;padding:0 9em}.h5p-hub-publish .h5p-hub-sharing-note .h5p-hub-icon-info:before,.h5p-hub-registration-wrapper .h5p-hub-sharing-note .h5p-hub-icon-info:before{font-size:0.5em;display:inline-block;margin-right:1em;background:#d1d7e3;padding:0.4em;border-radius:50%;position:relative;top:-0.25em}.h5p-hub-publish .h5p-hub-sr-only,.h5p-hub-registration-wrapper .h5p-hub-sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0, 0, 0, 0);white-space:nowrap;border:0}.h5p-hub-publish [class^="h5p-hub-icon-"],.h5p-hub-publish [class*=" h5p-hub-icon-"],.h5p-hub-registration-wrapper [class^="h5p-hub-icon-"],.h5p-hub-registration-wrapper [class*=" h5p-hub-icon-"]{font-family:'h5p' !important;speak:none;font-style:normal;font-weight:normal;font-variant:normal;text-transform:none;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.h5p-hub-publish .h5p-hub-icon-dropdown:before,.h5p-hub-registration-wrapper .h5p-hub-icon-dropdown:before{content:"\e925"}.h5p-hub-publish .h5p-hub-icon-info:before,.h5p-hub-registration-wrapper .h5p-hub-icon-info:before{content:"\e93c"}.h5p-hub-publish .h5p-hub-icon-thick-arrow:before,.h5p-hub-registration-wrapper .h5p-hub-icon-thick-arrow:before{content:"\e91a"}.h5p-hub-publish .h5p-hub-icon-check:before,.h5p-hub-registration-wrapper .h5p-hub-icon-check:before{content:"\e933"}.h5p-hub-publish .h5p-hub-icon-close:before,.h5p-hub-registration-wrapper .h5p-hub-icon-close:before{content:"\e93a"}.h5p-hub-publish .h5p-hub-icon-plus:before,.h5p-hub-registration-wrapper .h5p-hub-icon-plus:before{content:"\e939"}.h5p-hub-publish button,.h5p-hub-registration-wrapper button{font-size:.917em;border-width:.125em;border-radius:.208em;padding:.833em 1.65em;background:#fff;color:#3c4859;font-family:Nunito, sans-serif;outline:none;transition:all 0.1s}.h5p-hub-publish button:hover,.h5p-hub-registration-wrapper button:hover{color:#3c4859;background:#e2e5ee;border-color:#e2e5ee}.h5p-hub-registration-wrapper{border:none}.h5p-hub-publish{border:1px solid #e2e5ee}.h5p-hub-share-error{margin:2em} - -.h5p-hub-registration .h5p-hub-message-header{font-weight:bold;font-size:1.042em;font-family:'Open Sans', sans-serif}.h5p-hub-registration .h5p-hub-message-description{margin-top:0.5em;font-size:0.833em;font-family:'Open Sans', sans-serif}.h5p-hub-registration .h5p-hub-message-description a{margin-left:0.3em;color:#000}.h5p-hub-registration .h5p-hub-email-address{display:flex;flex-direction:column;justify-content:flex-end}.h5p-hub-registration .h5p-hub-publisher-description{height:10em}.h5p-hub-registration .h5p-hub-logo-upload-text{font-size:0.917em;color:#707475;margin-bottom:1em;margin-top:0.5em}.h5p-hub-registration .h5p-hub-text-field{max-height:10em;margin-top:1.5em}.h5p-hub-registration .h5p-hub-checkbox{margin-top:1.5em;font-weight:normal;font-size:.917em}.h5p-hub-registration .h5p-hub-register-hub{margin-left:1em;border:2px solid #186df7;background-color:#186df7;color:#fff}.h5p-hub-registration .h5p-hub-register-hub[disabled]{color:#aac3ec;border-color:#aac3ec;background-color:white;pointer-events:none}.h5p-hub-registration .h5p-hub-register-hub:hover{background-color:#2e46a4;border-color:#2e46a4;color:white}.h5p-hub-registration .h5p-hub-footer{display:flex;justify-content:flex-end} - -.h5p-hub-publish .h5p-hub-image-preview,.h5p-hub-registration .h5p-hub-image-preview{width:4em;height:3em;display:inline-block;background-position:center;background-size:contain;background-repeat:no-repeat;background-color:#d0d0d1} - -.h5p-hub-publish .h5p-hub-image-upload-container,.h5p-hub-registration .h5p-hub-image-upload-container{width:4em;height:3em;margin-right:1em;position:relative;display:flex}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close{cursor:pointer;font-size:1em;border:none;outline:none;position:absolute;right:0;padding:0}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close::before{font-size:0.6em;padding:0.5em;line-height:1.1;color:#fff;background:#000;position:absolute;z-index:1;display:inline-block;right:-1.2em;top:-1.2em;border:3px solid #fff;border-radius:50%;cursor:pointer;transition:all 0.3s}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close:hover::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close:hover::before{background:#eee;color:#000}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close:focus::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close:focus::before{outline:#000;outline-style:auto}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload{position:absolute;top:0;left:0;width:100%;height:100%;text-align:center;transition:all 0.3s;border:0.125em dashed #bebdcd;color:#bebdcd}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload input,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload input{position:absolute;left:0;top:0;width:100%;height:100%;cursor:pointer;opacity:0}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload input::-webkit-file-upload-button,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload input::-webkit-file-upload-button{visibility:hidden}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload:hover,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload:hover{color:#777;border-color:#777}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload:before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload:before{content:"+";font-size:2em;display:inline-block;position:relative;top:50%;transform:translateY(-50%)}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected{border-style:solid;border-radius:0.1em}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected:before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected:before{display:none} - -.h5p-hub-registration .h5p-hub-text-field{padding:1.5em;overflow:hidden;overflow-y:scroll;border:1px solid #bebdcd;border-radius:0.208em}.h5p-hub-registration .h5p-hub-text-field::-webkit-scrollbar{background-color:rgba(105,117,133,0.25);width:0.333em;border-radius:5.45256em}.h5p-hub-registration .h5p-hub-text-field::-webkit-scrollbar-thumb{background-color:#697585;background-size:24px 100%;border-radius:5.45256em}.h5p-hub-registration .h5p-hub-text-field .h5p-hub-title{color:#1a1a1a;font-size:1.250em;margin-bottom:0.5em}.h5p-hub-registration .h5p-hub-text-field .h5p-hub-description{color:#707475;font-size:0.833em;margin-bottom:1.5em}.h5p-hub-registration .h5p-hub-text-field .h5p-hub-main-text{color:#707475;font-size:0.833em} - -.h5p-hub-registration .h5p-hub-checkbox{font-size:1.05em;padding:0.5em 0;font-weight:bold;display:flex;cursor:pointer;align-self:start;color:#4d5057;width:100%;position:relative}.h5p-hub-registration .h5p-hub-checkbox>input{visibility:hidden;position:absolute}.h5p-hub-registration .h5p-hub-checkbox:focus{box-shadow:0 0 0.4em 0.01em RGBA(26, 147, 244, 0.7);outline:none}.h5p-hub-registration .h5p-hub-checkbox .h5p-hub-icon:before{font-family:'h5p';content:'\e923';font-style:normal;font-weight:normal;font-size:1em;margin:0.4em 0.4em 0 0.4em;margin-right:1em}.h5p-hub-registration .h5p-hub-checkbox.h5p-hub-checked .h5p-hub-icon:before{content:'\e938';font-family:'h5p';font-style:normal;font-weight:normal;font-size:1em;margin:0.4em 0.4em 0 0.4em;margin-right:1em} - -.h5p-hub-publish .h5p-hub-message,.h5p-hub-registration .h5p-hub-message{font-size:1em;border-left:0.3em solid;padding:1em;margin:0.5em 0}.h5p-hub-publish .h5p-hub-message.h5p-hub-warning,.h5p-hub-registration .h5p-hub-message.h5p-hub-warning{border-color:#fdcc31;background:#fff8e0}.h5p-hub-publish .h5p-hub-message.h5p-hub-error,.h5p-hub-registration .h5p-hub-message.h5p-hub-error{border-color:#c1494b;background-color:#ece0e0;color:#c1494b}.h5p-hub-publish .h5p-hub-message.h5p-hub-success,.h5p-hub-registration .h5p-hub-message.h5p-hub-success{border-color:#03a953;background-color:#d9eae2} - diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-hub-sharing.css b/h5p/h5plib/v126/joubel/core/styles/h5p-hub-sharing.css deleted file mode 100644 index 4ec25b4eeb2d1..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-hub-sharing.css +++ /dev/null @@ -1,400 +0,0 @@ -.h5p-hub-publish button,.h5p-hub-registration button{font-family:"Open Sans", sans-serif;font-size:0.75em;background:transparent;padding:0.75em 2em;color:#757575;border-radius:3em;border:1px solid #d6d6d6;cursor:pointer;font-weight:bold}.h5p-hub-publish button:before,.h5p-hub-registration button:before{font-weight:normal}.h5p-hub-publish button:hover,.h5p-hub-registration button:hover{color:#5f5f5f;border-color:#5f5f5f}.h5p-hub-publish button.h5p-hub-contained.h5p-hub-green,.h5p-hub-registration button.h5p-hub-contained.h5p-hub-green{background-color:#186df7;color:#fff;border-color:#186df7}.h5p-hub-publish button.h5p-hub-contained.h5p-hub-green:hover,.h5p-hub-registration button.h5p-hub-contained.h5p-hub-green:hover{background-color:#2e46a4;border-color:#2e46a4}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green,.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green{border-width:0.125em;background-color:white;border-color:#186df7;color:#186df7}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green:hover,.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green:hover{background-color:#186df7;border-color:#186df7;color:#ffffff}.h5p-hub-publish button.h5p-hub-outlined.h5p-hub-green[disabled],.h5p-hub-registration button.h5p-hub-outlined.h5p-hub-green[disabled]{color:#aac3ec;border-color:#aac3ec;background-color:white;pointer-events:none}.h5p-hub-publish .h5p-hub-navigation button{padding:0.833em;max-width:10.5em} - -.h5p-hub-publish .h5p-hub-stepper{display:flex;width:100%;max-width:40em;align-items:center;flex-direction:row;margin:2em auto;padding:0 2em;box-sizing:border-box}.h5p-hub-publish .h5p-hub-step-connector{flex-grow:1;margin-left:-2em;margin-right:-2em;position:relative;top:-0.6em}.h5p-hub-publish .h5p-hub-step-connector-line{display:block;border-top-style:solid;border-top-width:1px;border-color:#d5d5d7} - -.h5p-hub-publish .h5p-hub-step{text-align:center;font-weight:bold}.h5p-hub-publish .h5p-hub-step .h5p-hub-step-icon{color:#303030;border-radius:50%;border:0.462em solid #e1e1e1;width:3.2em;height:3.2em;line-height:2.276;display:inline-block;z-index:1;position:relative;background:#fff;box-sizing:border-box}.h5p-hub-publish .h5p-hub-step .h5p-hub-step-icon .h5p-hub-icon-check{line-height:2.276}.h5p-hub-publish .h5p-hub-step .h5p-hub-step-label{color:#757575;margin-top:0.5em;min-width:8em}.h5p-hub-publish .h5p-hub-step.h5p-hub-active{color:#20d2b2}.h5p-hub-publish .h5p-hub-step.h5p-hub-active .h5p-hub-step-icon{border-color:#1886f7;color:#186df7}.h5p-hub-publish .h5p-hub-step.h5p-hub-completed .h5p-hub-step-icon{background:#186df7;border-color:#609bff;color:#fff} - -.h5p-hub-publish .h5p-hub-form-element,.h5p-hub-registration .h5p-hub-form-element{font-family:Nunito, sans-serif;font-size:16px;padding:1em 0}.h5p-hub-publish .h5p-hub-form-element label,.h5p-hub-registration .h5p-hub-form-element label{display:block;margin-bottom:0.4em;font-weight:bold}.h5p-hub-publish .h5p-hub-form-element.h5p-hub-mandatory label:after,.h5p-hub-registration .h5p-hub-form-element.h5p-hub-mandatory label:after{content:'*';color:#d11e15;position:relative;top:-0.2em}.h5p-hub-publish .h5p-hub-form-element input,.h5p-hub-registration .h5p-hub-form-element input{border-radius:0.208em;width:100%;box-sizing:border-box;line-height:normal}.h5p-hub-publish .h5p-hub-form-element input::placeholder,.h5p-hub-registration .h5p-hub-form-element input::placeholder{font-style:italic}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-description,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-description{font-size:0.833em;color:#707475;margin:0 0 1em 0}.h5p-hub-publish .h5p-hub-form-element textarea,.h5p-hub-registration .h5p-hub-form-element textarea{resize:none;width:100%;border-radius:0.208em;box-sizing:border-box;line-height:1.25em}.h5p-hub-publish .h5p-hub-form-element textarea::-webkit-scrollbar,.h5p-hub-registration .h5p-hub-form-element textarea::-webkit-scrollbar{background-color:rgba(105,117,133,0.25);width:0.333em;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-form-element textarea::-webkit-scrollbar-thumb,.h5p-hub-registration .h5p-hub-form-element textarea::-webkit-scrollbar-thumb{background-color:#697585;background-size:24px 100%;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-form-element textarea::placeholder,.h5p-hub-registration .h5p-hub-form-element textarea::placeholder{font-style:italic}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-link-button,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-link-button{border:none;text-decoration:underline;text-align:right;align-self:flex-start;color:#186df7;font-size:.917em;font-weight:normal;padding:0}.h5p-hub-publish .h5p-hub-form-element .h5p-hub-details-row,.h5p-hub-registration .h5p-hub-form-element .h5p-hub-details-row{display:flex;align-content:center;align-items:center;justify-content:space-between}.h5p-hub-publish .h5p-hub-form-element textarea,.h5p-hub-publish .h5p-hub-form-element input,.h5p-hub-publish .h5p-hub-form-element select,.h5p-hub-registration .h5p-hub-form-element textarea,.h5p-hub-registration .h5p-hub-form-element input,.h5p-hub-registration .h5p-hub-form-element select{padding:.958em;border:2px solid #e2e5ee;box-shadow:none;color:#3c4859;font-family:Nunito, sans-serif}.h5p-hub-publish .h5p-hub-form-element textarea:focus,.h5p-hub-publish .h5p-hub-form-element input:focus,.h5p-hub-publish .h5p-hub-form-element select:focus,.h5p-hub-registration .h5p-hub-form-element textarea:focus,.h5p-hub-registration .h5p-hub-form-element input:focus,.h5p-hub-registration .h5p-hub-form-element select:focus{border-color:#98cbe8;box-shadow:inset 0 1px 1px rgba(0,0,0,0.075),0 0 8px rgba(152,203,232,0.6);outline:none}.h5p-hub-publish .h5p-hub-step-content>.h5p-hub-form-element:first-child,.h5p-hub-registration .h5p-hub-step-content>.h5p-hub-form-element:first-child{padding-top:0}.h5p-hub-publish .h5p-hub-step-content>.h5p-hub-form-element:last-child,.h5p-hub-registration .h5p-hub-step-content>.h5p-hub-form-element:last-child{padding-bottom:0} - -.h5p-hub-publish select{font-size:0.917em;border-radius:0.208em;padding:0.833em 2.334em 0.833em 0.833em;color:#313131;background-color:#ffffff;font-family:"Open Sans", sans-serif;border:1px solid #bebdcd;width:100%;-webkit-appearance:none;-moz-appearance:none;line-height:1.5}.h5p-hub-publish select:disabled{background-color:#ededed}.h5p-hub-publish select:disabled::before{content:"-"}.h5p-hub-publish select::-ms-expand{display:none}.h5p-hub-publish div.h5p-hub-icon-arrow-down{position:relative;line-height:1.1em;pointer-events:none}.h5p-hub-publish div.h5p-hub-icon-arrow-down:before{content:"\e925";transform:rotate(90deg);position:absolute;right:1.5em;top:1.9em;font-size:0.6em} - -.h5p-hub-publish .h5p-hub-modal-content{z-index:2;background-color:#ffffff;opacity:1;position:relative;margin:2em 3em;border-radius:0.167em;max-height:calc(100% - 4em);overflow:auto}.h5p-hub-publish .h5p-hub-modal-overlay{position:absolute;top:0;left:0px;right:0px;bottom:0px;background-color:rgba(0,0,0,0.7);z-index:2} - -.h5p-hub-hidden-read{width:1px;height:1px;top:-1px;position:absolute;text-indent:1px;overflow:hidden} - -.h5p-hub-publish .h5p-hub-tip{position:relative;display:block}.h5p-hub-publish .h5p-hub-tip.h5p-hub-positioned{position:unset}.h5p-hub-publish .h5p-hub-tip.h5p-hub-bottom .h5p-hub-tip-wrapper{top:115%}.h5p-hub-publish .h5p-hub-tip.h5p-hub-bottom .h5p-hub-tip-wrapper .h5p-hub-arrow:after{top:-0.75em;border-bottom-color:black;border-top-color:transparent}.h5p-hub-publish .h5p-hub-tip.h5p-hub-left-aligned .h5p-hub-tip-wrapper{transform:none;top:-0.85em;right:0}.h5p-hub-publish .h5p-hub-tip.h5p-hub-left-aligned .h5p-hub-tip-wrapper .h5p-hub-tip{white-space:normal}.h5p-hub-publish .h5p-hub-tip.h5p-hub-left-aligned .h5p-hub-tip-wrapper .h5p-hub-arrow:after{left:auto;transform:none;right:1.3em}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper{display:none;position:absolute;top:-3.5em;font-size:14px;font-weight:400;color:white;border-radius:0.4em;z-index:2}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-enter-done,.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-enter,.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-enter-active,.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-exit,.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-exit-active{display:block}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-enter{transform:translate(0, 30%);opacity:0}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-enter-active{transform:translate(0, 0);opacity:1;transition:opacity 0.2s, transform 0.2s}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-exit{transform:translate(0, 0);opacity:1}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper.h5p-hub-tip-wrapper-exit-active{transform:translate(0, 30%);opacity:0;transition:opacity 0.2s, transform 0.2s}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper .h5p-hub-tip{white-space:nowrap;background:black;padding:0.3em 1em;border-radius:0.4em}.h5p-hub-publish .h5p-hub-tip .h5p-hub-tip-wrapper .h5p-hub-arrow:after{content:" ";position:absolute;left:50%;transform:translateX(-50%);pointer-events:none;border:0.4em solid transparent;border-top-color:black} - -.h5p-hub-publish .h5p-hub-accordion-heading{display:flex;align-items:center}.h5p-hub-publish .h5p-hub-accordion-heading .h5p-hub-icon-dropdown{font-size:0.4em;color:#616161;display:inline-block;left:0;display:flex;padding:0.2em}.h5p-hub-publish .h5p-hub-accordion-heading .h5p-hub-icon-dropdown.h5p-hub-open{transform:rotate(90deg)}.h5p-hub-publish .h5p-hub-accordion-button{border:none;position:relative;padding:0.2em;width:100%;justify-self:start;display:flex}.h5p-hub-publish .h5p-hub-accordion-button .h5p-hub-accordion-header{border:none;text-align:left;color:#616161}.h5p-hub-publish .h5p-hub-panel-body{font-size:0.833em;margin:0.8em;margin-bottom:0;color:#707475;overflow:hidden} - -.h5p-hub-publish .h5p-hub-dialog-header{padding:1.708em 2.292em;border-bottom:1px solid #d5d5d7}.h5p-hub-publish .h5p-hub-dialog-header .h5p-hub-dialog-description{font-size:0.833em;color:#707475}.h5p-hub-publish .h5p-hub-dialog-content{padding:1.708em 2.292em}.h5p-hub-publish .h5p-hub-dialog-title{display:flex;justify-content:space-between}.h5p-hub-publish .h5p-hub-dialog-title span{font-size:1.458em;color:#1a1a1a;margin-bottom:0.2em}.h5p-hub-publish .h5p-hub-dialog-title button{max-height:2.77em} - -.h5p-hub-publish .h5p-hub-chips-list{display:flex;list-style:none;padding:0;flex-wrap:wrap;margin:0 0 0.5em}.h5p-hub-publish .h5p-hub-chips-list li{margin:0}.h5p-hub-publish .h5p-hub-chips-list li button{border-radius:0.125em;margin-right:0.5em;margin-top:0.5em;font-style:italic;color:#000000;display:flex;align-items:center;outline:none;border-color:#e2e5ee;padding:0.5em;font-size:0.85em}.h5p-hub-publish .h5p-hub-chips-list li button .h5p-hub-icon-close{font-size:0.8em;margin-left:2em}.h5p-hub-publish .h5p-hub-chips-list li button:focus{background-color:#e8e8e8;border-color:#dddee2}.h5p-hub-publish .h5p-hub-chips-list li button:hover{background-color:#f5f5f5;border-color:#dddee2} - -.h5p-hub-publish .h5p-hub-keywords-input-wrapper{display:flex;align-items:center;border-radius:2em}.h5p-hub-publish .h5p-hub-keywords-input-wrapper .h5p-hub-add-button{position:absolute;right:4.5em;border-radius:1.5em;padding:0.5em 1em;margin-right:0.5em}.h5p-hub-publish .h5p-hub-keywords-input-wrapper .h5p-hub-add-button:focus{border-color:#3ba0f2}.h5p-hub-publish .h5p-hub-keywords-input-wrapper .h5p-hub-add-button.disabled{cursor:default;color:#d6d6d6}.h5p-hub-publish .h5p-hub-keywords-input-wrapper .h5p-hub-add-button.disabled:hover{background-color:transparent}.h5p-hub-publish .h5p-hub-tip-keywords{top:2.5em} - -.h5p-hub-publish .h5p-hub-image-preview,.h5p-hub-registration .h5p-hub-image-preview{width:4em;height:3em;display:inline-block;background-position:center;background-size:contain;background-repeat:no-repeat;background-color:#d0d0d1} - -.h5p-hub-publish .h5p-hub-image-upload-container,.h5p-hub-registration .h5p-hub-image-upload-container{width:4em;height:3em;margin-right:1em;position:relative;display:flex}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close{cursor:pointer;font-size:1em;border:none;outline:none;position:absolute;right:0;padding:0}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close::before{font-size:0.6em;padding:0.5em;line-height:1.1;color:#fff;background:#000;position:absolute;z-index:1;display:inline-block;right:-1.2em;top:-1.2em;border:3px solid #fff;border-radius:50%;cursor:pointer;transition:all 0.3s}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close:hover::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close:hover::before{background:#eee;color:#000}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-icon-close:focus::before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-icon-close:focus::before{outline:#000;outline-style:auto}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload{position:absolute;top:0;left:0;width:100%;height:100%;text-align:center;transition:all 0.3s;border:0.125em dashed #bebdcd;color:#bebdcd}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload input,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload input{position:absolute;left:0;top:0;width:100%;height:100%;cursor:pointer;opacity:0}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload input::-webkit-file-upload-button,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload input::-webkit-file-upload-button{visibility:hidden}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload:hover,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload:hover{color:#777;border-color:#777}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload:before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload:before{content:"+";font-size:2em;display:inline-block;position:relative;top:50%;transform:translateY(-50%)}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected{border-style:solid;border-radius:0.1em}.h5p-hub-publish .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected:before,.h5p-hub-registration .h5p-hub-image-upload-container .h5p-hub-image-upload.h5p-hub-image-selected:before{display:none} - -.h5p-hub-publish .h5p-hub-optional .h5p-hub-keywords .h5p-hub-form-element label{margin-bottom:0.708em}.h5p-hub-publish .h5p-hub-optional .h5p-hub-form-screenshots{display:flex;flex-wrap:wrap;flex-direction:column}.h5p-hub-publish .h5p-hub-optional .h5p-hub-form-screenshots .h5p-hub-image-upload-container{margin-top:0.4em}.h5p-hub-publish .h5p-hub-optional .h5p-hub-form-screenshots .h5p-hub-row{margin-top:0.5em}.h5p-hub-publish .h5p-hub-optional .h5p-hub-form-screenshots .h5p-hub-row .h5p-hub-form-element{padding:0}.h5p-hub-publish .h5p-hub-optional .h5p-hub-form-screenshots .h5p-hub-alt-text{padding:0.3em}.h5p-hub-publish .h5p-hub-optional .h5p-hub-optional-second-part{display:flex}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-optional .h5p-hub-optional-second-part{flex-wrap:wrap}}.h5p-hub-publish .h5p-hub-optional #h5p-hub-form-long-description{height:10.458em}.h5p-hub-publish .h5p-hub-optional #h5p-hub-form-short-description{height:6.333em} - -.h5p-hub-publish .h5p-hub-discipline-element,.h5p-hub-registration .h5p-hub-discipline-element{font-size:1.05em;padding:0.5em 0;font-weight:bold;display:flex;cursor:pointer;align-self:start;color:#4d5057;width:100%;position:relative}.h5p-hub-publish .h5p-hub-discipline-element .h5p-hub-checkbox-error-tip,.h5p-hub-registration .h5p-hub-discipline-element .h5p-hub-checkbox-error-tip{top:5em;position:absolute;margin-left:2.5em;z-index:1}.h5p-hub-publish .h5p-hub-discipline-element:last-child .h5p-hub-checkbox-error-tip,.h5p-hub-registration .h5p-hub-discipline-element:last-child .h5p-hub-checkbox-error-tip{top:2.5em}.h5p-hub-publish .h5p-hub-discipline-element:focus,.h5p-hub-registration .h5p-hub-discipline-element:focus{box-shadow:0 0 0.4em 0.01em RGBA(26, 147, 244, 0.7);outline:none}.h5p-hub-publish .h5p-hub-discipline-element.h5p-hub-parent::after,.h5p-hub-registration .h5p-hub-discipline-element.h5p-hub-parent::after{font-family:'h5p';content:'\e58e';transform:rotate(90deg);font-style:normal;font-size:1.1em;display:inline-block;align-self:center}.h5p-hub-publish .h5p-hub-discipline-element .h5p-hub-content,.h5p-hub-registration .h5p-hub-discipline-element .h5p-hub-content{display:flex;align-items:center;margin-left:0.5em}.h5p-hub-publish .h5p-hub-discipline-element .h5p-hub-bold,.h5p-hub-registration .h5p-hub-discipline-element .h5p-hub-bold{font-weight:bold}.h5p-hub-publish .h5p-hub-discipline-element .h5p-hub-non-bold,.h5p-hub-registration .h5p-hub-discipline-element .h5p-hub-non-bold{font-weight:normal} - -.h5p-hub-publish .h5p-hub-checkbox-list{list-style:none;padding:0;background-color:#ffffff}.h5p-hub-publish .h5p-hub-checkbox-list.h5p-hub-animate-in-right{animation:move-from-right 0.35s}.h5p-hub-publish .h5p-hub-checkbox-list.h5p-hub-animate-in-left{animation:move-from-left 0.35s}@keyframes move-from-right{0%{opacity:0;transform:translateX(30%)}30%{opacity:0}100%{opacity:1;transform:translateX(0)}}@keyframes move-from-left{0%{opacity:0;transform:translateX(-10%)}30%{opacity:0}100%{opacity:1;transform:translateX(0)}} - -.h5p-hub-publish .h5p-hub-bottom-line{border-bottom:2px solid #697585}.h5p-hub-publish .h5p-hub-category-header{background-color:#f2f2f2;color:#313131;font-style:italic;padding:0 0 0 0.9em;font-size:0.81788em} - -.h5p-hub-publish .h5p-hub-search-button{width:100%;position:relative}.h5p-hub-publish .h5p-hub-search-button .h5p-hub-search-field{position:relative;display:flex;align-items:center;justify-content:space-between}.h5p-hub-publish .h5p-hub-search-button .h5p-hub-search-field::before{content:"\e926";font-family:"h5p";font-style:normal;font-size:1em;margin-left:1.25em;color:#b7b8ba;position:absolute}.h5p-hub-publish .h5p-hub-search-button .h5p-hub-search-field .h5p-hub-icon-arrow{margin:1em;font-size:1.30353em;cursor:pointer;color:#000000;transform:rotate(90deg)}.h5p-hub-publish .h5p-hub-search-button .h5p-hub-search-field .h5p-hub-icon-arrow::before{content:"\e925";font-family:"h5p";font-style:normal;font-size:0.6em}.h5p-hub-publish #h5p-hub-filter-search-bar{border:none;padding:0.833em;font-style:italic;width:100%;box-sizing:border-box;font-size:1.15em;padding-left:3.044em}.h5p-hub-publish #h5p-hub-filter-search-bar::placeholder{color:#b7b8ba}.h5p-hub-publish #h5p-hub-filter-search-bar:focus{border-color:#3ba0f2;background-color:#fff;outline:none} - -.h5p-hub-publish .h5p-hub-search-filter{position:relative;border-radius:0.208em;font-size:0.833em;border:2px solid #e2e5ee}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-clear-button{background:none;position:absolute;padding:1.6em;right:3.5em;color:#757575;border:none;top:50%;transform:translateY(-50%)}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-clear-button::before{content:"\e93a";font-family:"h5p";font-style:normal;font-size:1em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-navigate-parent{font-size:1.25em;padding:0.3534em;display:flex;font-weight:bold;font-family:"Open Sans", sans-serif;border-top:1px solid #e6e6e8;align-items:center;border-bottom:solid 1px #e6e6e8}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-navigate-parent button{background-color:#ffffff;position:relative;font-style:normal;font-size:0.8em;align-self:center;border:none;padding:0.75em;margin-right:1em;border-radius:50%;width:2.75em;height:2.75em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-navigate-parent button::before{content:"\e937";font-family:"h5p";transform:rotate(180deg);font-weight:normal;display:inline-block}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-navigate-parent button:hover{background-color:#dadada;cursor:pointer}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list{overflow:hidden;overflow-y:scroll;max-height:25em;margin:0}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li{border-bottom:1px solid #e6e6e8;margin:0}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li:last-child{border-bottom:none}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li:first-child{border-top:1px solid #e6e6e8}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li .h5p-hub-content{padding:0.3em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li .h5p-hub-content .h5p-hub-label-text{margin:0 0.2em 0 1em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li .h5p-hub-content button{border-radius:1.5em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li:focus{border-color:#3ba0f2;background-color:#fff;outline:none;box-shadow:none}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li:hover{background-color:#f2f2f2}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li.h5p-hub-highlighted{outline:1px solid #3ba0f2;outline-offset:-1px;background-color:#fff;box-shadow:none}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list li:hover{background-color:rgba(230,230,232,0.25)}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list::-webkit-scrollbar{background-color:rgba(105,117,133,0.25);width:0.54525em;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-checkbox-list::-webkit-scrollbar-thumb{background-color:#697585;background-size:24px 100%;border-radius:5.45256em}.h5p-hub-publish .h5p-hub-search-filter .h5p-hub-publish .h5p-hub-form-element input{border:none} - -.h5p-hub-publish .h5p-hub-message,.h5p-hub-registration .h5p-hub-message{font-size:1em;border-left:0.3em solid;padding:1em;margin:0.5em 0}.h5p-hub-publish .h5p-hub-message.h5p-hub-warning,.h5p-hub-registration .h5p-hub-message.h5p-hub-warning{border-color:#fdcc31;background:#fff8e0}.h5p-hub-publish .h5p-hub-message.h5p-hub-error,.h5p-hub-registration .h5p-hub-message.h5p-hub-error{border-color:#c1494b;background-color:#ece0e0;color:#c1494b}.h5p-hub-publish .h5p-hub-message.h5p-hub-success,.h5p-hub-registration .h5p-hub-message.h5p-hub-success{border-color:#03a953;background-color:#d9eae2} - -.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-review-header{margin-bottom:1em;color:#363636}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review dl{border-top:1px solid #d5d5d5;margin:1.25em 0 0 0}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition{border-bottom:1px solid #d5d5d5;display:flex;border-bottom-color:#e2e5ee}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition dt{width:25%;min-width:25%;background:#f4f5f9;display:inline-block;padding:0.5em 0 0.5em 1em}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition dd{color:#707475;display:inline-block;margin:0;padding:0.5em 0 0.5em 1em}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition .h5p-hub-img-alt{display:flex;align-items:center}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition .h5p-hub-screenshots{margin-bottom:0.2em}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition .h5p-hub-screenshots:last-child{margin-bottom:0}.h5p-hub-publish .h5p-hub-step-content.h5p-hub-review .h5p-hub-definition .h5p-hub-screenshots .h5p-hub-image-preview{font-size:0.75em;margin-right:0.5em} - -.h5p-hub-publish .h5p-hub-success-page{text-align:center}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-step-icon-success{font-size:0.75em;border-radius:50%;border:0.462em solid #609bff;width:3.2em;height:3.2em;line-height:2.276;display:inline-block;box-sizing:border-box;margin-bottom:0.667em;background:#186df7;color:#fff}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-submitted-text{margin-bottom:2.167em;font-size:0.75em;color:#186df7}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-content-type{letter-spacing:0.146em;color:#303438;margin-bottom:0.7em}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-title{font-size:1.75em;margin-bottom:0.5em;color:#1a1a1a}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-title.h5p-hub-is-edit{margin-bottom:1.75em}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-now-submitted-text{font-size:1.458em;margin-bottom:1.75em;color:#1a1a1a}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-now-submitted-text.h5p-hub-is-edit{margin-bottom:.5em}.h5p-hub-publish .h5p-hub-success-page .h5p-hub-content-available{font-size:0.833em;color:#707475} - -.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal{margin:2em auto;width:30em;max-width:90%}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-header{padding:1.708em 2.292em;border-bottom:1px solid #d5d5d7}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-header .h5p-hub-dialog-title{display:flex;justify-content:space-between}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-header .h5p-hub-dialog-title span{font-size:1.458em;color:#1a1a1a;margin-bottom:0.2em}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-content .h5p-hub-dialog-description{font-size:0.833em;color:#707475}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-content .h5p-hub-button-group{display:flex;justify-content:flex-end;margin-top:2em}.h5p-hub-publish .h5p-hub-cancel-publish-confirmation-modal .h5p-hub-dialog-content .h5p-hub-button-group button{margin-left:1em} - -/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ - -/* Document - ========================================================================== */ - -/** - * 1. Correct the line height in all browsers. - * 2. Prevent adjustments of font size after orientation changes in iOS. - */ - -html { - line-height: 1.15; /* 1 */ - -webkit-text-size-adjust: 100%; /* 2 */ -} - -/* Sections - ========================================================================== */ - -/** - * Remove the margin in all browsers. - */ - -body { - margin: 0; -} - -/** - * Render the `main` element consistently in IE. - */ - -main { - display: block; -} - -/** - * Correct the font size and margin on `h1` elements within `section` and - * `article` contexts in Chrome, Firefox, and Safari. - */ - -h1 { - font-size: 2em; - margin: 0.67em 0; -} - -/* Grouping content - ========================================================================== */ - -/** - * 1. Add the correct box sizing in Firefox. - * 2. Show the overflow in Edge and IE. - */ - -hr { - box-sizing: content-box; /* 1 */ - height: 0; /* 1 */ - overflow: visible; /* 2 */ -} - -/** - * 1. Correct the inheritance and scaling of font size in all browsers. - * 2. Correct the odd `em` font sizing in all browsers. - */ - -pre { - font-family: monospace, monospace; /* 1 */ - font-size: 1em; /* 2 */ -} - -/* Text-level semantics - ========================================================================== */ - -/** - * Remove the gray background on active links in IE 10. - */ - -a { - background-color: transparent; -} - -/** - * 1. Remove the bottom border in Chrome 57- - * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. - */ - -abbr[title] { - border-bottom: none; /* 1 */ - text-decoration: underline; /* 2 */ - text-decoration: underline dotted; /* 2 */ -} - -/** - * Add the correct font weight in Chrome, Edge, and Safari. - */ - -b, -strong { - font-weight: bolder; -} - -/** - * 1. Correct the inheritance and scaling of font size in all browsers. - * 2. Correct the odd `em` font sizing in all browsers. - */ - -code, -kbd, -samp { - font-family: monospace, monospace; /* 1 */ - font-size: 1em; /* 2 */ -} - -/** - * Add the correct font size in all browsers. - */ - -small { - font-size: 80%; -} - -/** - * Prevent `sub` and `sup` elements from affecting the line height in - * all browsers. - */ - -sub, -sup { - font-size: 75%; - line-height: 0; - position: relative; - vertical-align: baseline; -} - -sub { - bottom: -0.25em; -} - -sup { - top: -0.5em; -} - -/* Embedded content - ========================================================================== */ - -/** - * Remove the border on images inside links in IE 10. - */ - -img { - border-style: none; -} - -/* Forms - ========================================================================== */ - -/** - * 1. Change the font styles in all browsers. - * 2. Remove the margin in Firefox and Safari. - */ - -button, -input, -optgroup, -select, -textarea { - font-family: inherit; /* 1 */ - font-size: 100%; /* 1 */ - line-height: 1.15; /* 1 */ - margin: 0; /* 2 */ -} - -/** - * Show the overflow in IE. - * 1. Show the overflow in Edge. - */ - -button, -input { /* 1 */ - overflow: visible; -} - -/** - * Remove the inheritance of text transform in Edge, Firefox, and IE. - * 1. Remove the inheritance of text transform in Firefox. - */ - -button, -select { /* 1 */ - text-transform: none; -} - -/** - * Correct the inability to style clickable types in iOS and Safari. - */ - -button, -[type="button"], -[type="reset"], -[type="submit"] { - -webkit-appearance: button; -} - -/** - * Remove the inner border and padding in Firefox. - */ - -button::-moz-focus-inner, -[type="button"]::-moz-focus-inner, -[type="reset"]::-moz-focus-inner, -[type="submit"]::-moz-focus-inner { - border-style: none; - padding: 0; -} - -/** - * Restore the focus styles unset by the previous rule. - */ - -button:-moz-focusring, -[type="button"]:-moz-focusring, -[type="reset"]:-moz-focusring, -[type="submit"]:-moz-focusring { - outline: 1px dotted ButtonText; -} - -/** - * Correct the padding in Firefox. - */ - -fieldset { - padding: 0.35em 0.75em 0.625em; -} - -/** - * 1. Correct the text wrapping in Edge and IE. - * 2. Correct the color inheritance from `fieldset` elements in IE. - * 3. Remove the padding so developers are not caught out when they zero out - * `fieldset` elements in all browsers. - */ - -legend { - box-sizing: border-box; /* 1 */ - color: inherit; /* 2 */ - display: table; /* 1 */ - max-width: 100%; /* 1 */ - padding: 0; /* 3 */ - white-space: normal; /* 1 */ -} - -/** - * Add the correct vertical alignment in Chrome, Firefox, and Opera. - */ - -progress { - vertical-align: baseline; -} - -/** - * Remove the default vertical scrollbar in IE 10+. - */ - -textarea { - overflow: auto; -} - -/** - * 1. Add the correct box sizing in IE 10. - * 2. Remove the padding in IE 10. - */ - -[type="checkbox"], -[type="radio"] { - box-sizing: border-box; /* 1 */ - padding: 0; /* 2 */ -} - -/** - * Correct the cursor style of increment and decrement buttons in Chrome. - */ - -[type="number"]::-webkit-inner-spin-button, -[type="number"]::-webkit-outer-spin-button { - height: auto; -} - -/** - * 1. Correct the odd appearance in Chrome and Safari. - * 2. Correct the outline style in Safari. - */ - -[type="search"] { - -webkit-appearance: textfield; /* 1 */ - outline-offset: -2px; /* 2 */ -} - -/** - * Remove the inner padding in Chrome and Safari on macOS. - */ - -[type="search"]::-webkit-search-decoration { - -webkit-appearance: none; -} - -/** - * 1. Correct the inability to style clickable types in iOS and Safari. - * 2. Change font properties to `inherit` in Safari. - */ - -::-webkit-file-upload-button { - -webkit-appearance: button; /* 1 */ - font: inherit; /* 2 */ -} - -/* Interactive - ========================================================================== */ - -/* - * Add the correct display in Edge, IE 10+, and Firefox. - */ - -details { - display: block; -} - -/* - * Add the correct display in all browsers. - */ - -summary { - display: list-item; -} - -/* Misc - ========================================================================== */ - -/** - * Add the correct display in IE 10+. - */ - -template { - display: none; -} - -/** - * Add the correct display in IE 10. - */ - -[hidden] { - display: none; -} - -@keyframes spinner{0%{transform:translate3d(-50%, -50%, 0) rotate(0deg)}100%{transform:translate3d(-50%, -50%, 0) rotate(360deg)}}.h5p-hub-publish,.h5p-hub-registration-wrapper{position:relative;margin:2em auto;max-width:1024px;border:1px solid #2c9586;border-radius:0.167em;box-sizing:border-box;box-shadow:0px 0px 20px rgba(0,0,0,0.1);font-family:Nunito, sans-serif;font-size:16px;padding-top:0.25em}.h5p-hub-publish .h5p-hub-header,.h5p-hub-registration-wrapper .h5p-hub-header{padding:1.5em;display:flex;align-items:center;width:100%;box-sizing:border-box;border-bottom:1px solid #d5d5d7;border-bottom-color:#e2e5ee}.h5p-hub-publish .h5p-hub-header .h5p-hub-title,.h5p-hub-registration-wrapper .h5p-hub-header .h5p-hub-title{font-size:1.458em;flex-grow:1;color:black;padding-left:1em}.h5p-hub-publish .h5p-hub-step-panel,.h5p-hub-registration-wrapper .h5p-hub-step-panel{margin:2em;border-radius:0.208em;border:1px solid #e2e5ee}.h5p-hub-publish .h5p-hub-step-title,.h5p-hub-registration-wrapper .h5p-hub-step-title{color:#1a1a1a;font-size:1.250em;font-weight:bold;border-bottom:1px solid #ccc;padding:1.2em 1.8em;line-height:normal;border-bottom-color:#e2e5ee}.h5p-hub-publish .h5p-hub-step-content,.h5p-hub-registration-wrapper .h5p-hub-step-content{padding:2em}.h5p-hub-publish .h5p-hub-footer,.h5p-hub-registration-wrapper .h5p-hub-footer{padding:0 2em 2em;width:100%;box-sizing:border-box;position:relative}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation{display:flex}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation button,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation button{z-index:1}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation button:before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation button:before{font-family:'h5p';font-size:0.75em;content:"\e91a";transform:rotate(180deg);display:inline-block;margin-right:0.75em;line-height:1;position:relative;top:-0.1em}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation .h5p-hub-back:before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation .h5p-hub-back:before{transform:rotate(0)}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation #h5p-hub-share-in-process::before,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation #h5p-hub-share-in-process::before{font-family:"h5p";font-size:0.75em;content:"\e928";animation:2s linear infinite spinner;top:0.5em;margin-left:0.55em}.h5p-hub-publish .h5p-hub-footer .h5p-hub-navigation .h5p-hub-next,.h5p-hub-registration-wrapper .h5p-hub-footer .h5p-hub-navigation .h5p-hub-next{margin-left:auto}.h5p-hub-publish .h5p-hub-tip-text-field,.h5p-hub-registration-wrapper .h5p-hub-tip-text-field{top:1.5em}.h5p-hub-publish .h5p-hub-hare-error,.h5p-hub-registration-wrapper .h5p-hub-hare-error{padding:0 2em 2em;text-align:end}.h5p-hub-publish .h5p-hub-bold,.h5p-hub-registration-wrapper .h5p-hub-bold{font-weight:bold}.h5p-hub-publish .h5p-hub-row,.h5p-hub-registration-wrapper .h5p-hub-row{display:flex;flex-wrap:wrap;margin-right:-1em}.h5p-hub-publish .h5p-hub-row .h5p-hub-form-element,.h5p-hub-registration-wrapper .h5p-hub-row .h5p-hub-form-element{flex-basis:100%;flex:1;min-width:17em;align-self:flex-end;border-right:1em solid #ffffff}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-row .h5p-hub-form-element,.h5p-hub-registration-wrapper .h5p-hub-row .h5p-hub-form-element{min-width:calc(100% - 1em)}}.h5p-hub-publish .h5p-hub-columns,.h5p-hub-registration-wrapper .h5p-hub-columns{display:flex;margin-right:-1em}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-columns,.h5p-hub-registration-wrapper .h5p-hub-columns{flex-wrap:wrap}}.h5p-hub-publish .h5p-hub-columns .h5p-hub-column,.h5p-hub-registration-wrapper .h5p-hub-columns .h5p-hub-column{width:50%;border-right:1em solid #ffffff}@media (max-width: 576px){.h5p-hub-publish .h5p-hub-columns .h5p-hub-column,.h5p-hub-registration-wrapper .h5p-hub-columns .h5p-hub-column{width:100%;margin-right:0}}.h5p-hub-publish .h5p-hub-sharing-note,.h5p-hub-registration-wrapper .h5p-hub-sharing-note{text-align:center;color:#707475;width:calc(100% - 4.9em);position:absolute;top:0.65em;box-sizing:border-box;padding:0 9em}.h5p-hub-publish .h5p-hub-sharing-note .h5p-hub-icon-info:before,.h5p-hub-registration-wrapper .h5p-hub-sharing-note .h5p-hub-icon-info:before{font-size:0.5em;display:inline-block;margin-right:1em;background:#d1d7e3;padding:0.4em;border-radius:50%;position:relative;top:-0.25em}.h5p-hub-publish .h5p-hub-sr-only,.h5p-hub-registration-wrapper .h5p-hub-sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0, 0, 0, 0);white-space:nowrap;border:0}.h5p-hub-publish [class^="h5p-hub-icon-"],.h5p-hub-publish [class*=" h5p-hub-icon-"],.h5p-hub-registration-wrapper [class^="h5p-hub-icon-"],.h5p-hub-registration-wrapper [class*=" h5p-hub-icon-"]{font-family:'h5p' !important;speak:none;font-style:normal;font-weight:normal;font-variant:normal;text-transform:none;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.h5p-hub-publish .h5p-hub-icon-dropdown:before,.h5p-hub-registration-wrapper .h5p-hub-icon-dropdown:before{content:"\e925"}.h5p-hub-publish .h5p-hub-icon-info:before,.h5p-hub-registration-wrapper .h5p-hub-icon-info:before{content:"\e93c"}.h5p-hub-publish .h5p-hub-icon-thick-arrow:before,.h5p-hub-registration-wrapper .h5p-hub-icon-thick-arrow:before{content:"\e91a"}.h5p-hub-publish .h5p-hub-icon-check:before,.h5p-hub-registration-wrapper .h5p-hub-icon-check:before{content:"\e933"}.h5p-hub-publish .h5p-hub-icon-close:before,.h5p-hub-registration-wrapper .h5p-hub-icon-close:before{content:"\e93a"}.h5p-hub-publish .h5p-hub-icon-plus:before,.h5p-hub-registration-wrapper .h5p-hub-icon-plus:before{content:"\e939"}.h5p-hub-publish button,.h5p-hub-registration-wrapper button{font-size:.917em;border-width:.125em;border-radius:.208em;padding:.833em 1.65em;background:#fff;color:#3c4859;font-family:Nunito, sans-serif;outline:none;transition:all 0.1s}.h5p-hub-publish button:hover,.h5p-hub-registration-wrapper button:hover{color:#3c4859;background:#e2e5ee;border-color:#e2e5ee}.h5p-hub-registration-wrapper{border:none}.h5p-hub-publish{border:1px solid #e2e5ee}.h5p-hub-share-error{margin:2em} - diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p-tooltip.css b/h5p/h5plib/v126/joubel/core/styles/h5p-tooltip.css deleted file mode 100644 index 18ee747e7bf97..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p-tooltip.css +++ /dev/null @@ -1,67 +0,0 @@ -.h5p-tooltip { - --translateX: -50%; - --translateY: 0; - - display: none; - position: absolute; - bottom: 100%; - left: 50%; - transform: translateX(var(--translateX)) translateY(var(--translateY)); - - z-index: 4; - - font-size: 0.9rem; - line-height: 1.5rem; - - padding: 0 0.5rem; - white-space: nowrap; - - background: #000; - color: #FFF; - - cursor: default; - - /* To hide the position adjustments and to get a bit more - pleasent popup effect */ - -webkit-animation: 800ms ease 0s normal forwards 1 fadein; - animation: 800ms ease 0s normal forwards 1 fadein; -} - -@keyframes fadein{ - 0% { opacity: 0; } - 80% { opacity: 0; } - 100% { opacity: 1; } -} - -@-webkit-keyframes fadein{ - 0% { opacity: 0; } - 80% { opacity: 0; } - 100% { opacity: 1; } -} - -.h5p-tooltip-bottom { - top: 100%; - bottom: auto; -} - -.h5p-tooltip-left { - --translateY: -50%; - --translateX: 0; - top: 50%; - bottom: auto; - left: auto; - right: 100%; -} - -.h5p-tooltip-right { - --translateY: -50%; - --translateX: 0; - top: 50%; - bottom: auto; - left: 100%; - right: auto; -} - -.h5p-tooltip-visible { - display: block; -} diff --git a/h5p/h5plib/v126/joubel/core/styles/h5p.css b/h5p/h5plib/v126/joubel/core/styles/h5p.css deleted file mode 100644 index c0a18a54ab447..0000000000000 --- a/h5p/h5plib/v126/joubel/core/styles/h5p.css +++ /dev/null @@ -1,640 +0,0 @@ -/* Import common fonts */ -@import 'font-open-sans.css'; -/* General CSS for H5P. Licensed under the MIT License.*/ -/* Custom H5P font to use for icons. */ -@font-face { - font-family: 'h5p'; - src: url('../fonts/h5p-core-28.eot?h1atjl'); - src: url('../fonts/h5p-core-28.eot?h1atjl#iefix') format('embedded-opentype'), - url('../fonts/h5p-core-28.ttf?h1atjl') format('truetype'), - url('../fonts/h5p-core-28.woff?h1atjl') format('woff'), - url('../fonts/h5p-core-28.svg?h1atjl#h5p-core-28') format('svg'); - font-weight: normal; - font-style: normal; -} - -@font-face { - font-family: 'h5p-hub-publish'; - src: url('../fonts/h5p-hub-publish.eot?wy8ylc'); - src: url('../fonts/h5p-hub-publish.eot?wy8ylc#iefix') format('embedded-opentype'), - url('../fonts/h5p-hub-publish.ttf?wy8ylc') format('truetype'), - url('../fonts/h5p-hub-publish.woff?wy8ylc') format('woff'), - url('../fonts/h5p-hub-publish.svg?wy8ylc#h5p-hub') format('svg'); - font-weight: normal; - font-style: normal; - font-display: block; -} - -html.h5p-iframe, html.h5p-iframe > body { - font-family: Sans-Serif; /* Use the browser's default sans-serif font. (Since Heletica doesn't look nice on Windows, and Arial on OS X.) */ - width: 100%; - height: 100%; - margin: 0; - padding: 0; -} -.h5p-semi-fullscreen, .h5p-fullscreen, html.h5p-iframe .h5p-container { - overflow: hidden; -} -.h5p-content { - position: relative; - background: #fefefe; - border: 1px solid #EEE; - border-bottom: none; - box-sizing: border-box; - -moz-box-sizing: border-box; -} -.h5p-noselect -{ - -khtml-user-select: none; - -ms-user-select: none; - -moz-user-select: none; - -webkit-user-select: none; - user-select: none; -} -html.h5p-iframe .h5p-content { - font-size: 16px; - line-height: 1.5em; - width: 100%; - height: auto; - -webkit-text-size-adjust: none; - text-size-adjust: none; -} -html.h5p-iframe .h5p-fullscreen .h5p-content, -html.h5p-iframe .h5p-semi-fullscreen .h5p-content { - height: 100%; -} -.h5p-content.h5p-no-frame, -.h5p-fullscreen .h5p-content, -.h5p-semi-fullscreen .h5p-content { - border: 0; -} -.h5p-container { - position: relative; - z-index: 1; -} -.h5p-iframe-wrapper.h5p-fullscreen { - background-color: #000; -} -body.h5p-semi-fullscreen { - position: fixed; - width: 100%; - height: 100%; -} -.h5p-container.h5p-semi-fullscreen { - position: fixed; - top: 0; - left: 0; - z-index: 101; - width: 100%; - height: 100%; - background-color: #FFF; -} - -.h5p-content-controls { - margin: 0; - position: absolute; - right: 0; - top: 0; - z-index: 3; -} -.h5p-fullscreen .h5p-content-controls { - display: none; -} - -.h5p-content-controls > a:link, .h5p-content-controls > a:visited, a.h5p-disable-fullscreen:link, a.h5p-disable-fullscreen:visited { - color: #e5eef6; -} - -.h5p-enable-fullscreen:before { - font-family: 'H5P'; - content: "\e88c"; -} -.h5p-disable-fullscreen:before { - font-family: 'H5P'; - content: "\e891"; -} -.h5p-enable-fullscreen, .h5p-disable-fullscreen { - cursor: pointer; - color: #EEE; - background: rgb(0,0,0); - background: rgba(0,0,0,0.3); - line-height: 0.975em; - font-size: 2em; - width: 1.125em; - height: 1em; - text-indent: 0.04em; -} -.h5p-disable-fullscreen { - line-height: 0.925em; - width: 1.1em; - height: 0.9em; -} - -.h5p-enable-fullscreen:focus, -.h5p-disable-fullscreen:focus { - outline-style: solid; - outline-width: 1px; - outline-offset: 0.25em; -} - -.h5p-enable-fullscreen:hover, .h5p-disable-fullscreen:hover { - background: rgba(0,0,0,0.5); -} -.h5p-semi-fullscreen .h5p-enable-fullscreen { - display: none; -} - -div.h5p-fullscreen { - width: 100%; - height: 100%; -} -.h5p-iframe-wrapper { - width: auto; - height: auto; -} - -.h5p-fullscreen .h5p-iframe-wrapper, -.h5p-semi-fullscreen .h5p-iframe-wrapper { - width: 100%; - height: 100%; -} - -.h5p-iframe-wrapper.h5p-semi-fullscreen { - width: auto; - height: auto; - background: black; - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - z-index: 100001; -} -.h5p-iframe-wrapper.h5p-semi-fullscreen .buttons { - position: absolute; - top: 0; - right: 0; - z-index: 20; -} -.h5p-iframe-wrapper iframe.h5p-iframe { - /* Hack for IOS landscape / portrait */ - width: 10px; - min-width: 100%; - *width: 100%; - /* End of hack */ - height: 100%; - z-index: 10; - overflow: hidden; - border: 0; - display: block; -} - -.h5p-content ul.h5p-actions { - box-sizing: border-box; - -moz-box-sizing: border-box; - list-style: none; - padding: 0px 10px; - margin: 0; - height: 25px; - font-size: 12px; - background: #FAFAFA; - border-top: 1px solid #EEE; - border-bottom: 1px solid #EEE; - clear: both; - font-family: Sans-Serif; -} -.h5p-fullscreen .h5p-actions, .h5p-semi-fullscreen .h5p-actions { - display: none; -} -.h5p-actions > .h5p-button { - float: left; - cursor: pointer; - margin: 0 0.5em 0 0; - background: none; - padding: 0 0.75em 0 0.25em; - vertical-align: top; - color: #707070; - text-decoration: none; - outline: none; - line-height: 22px; -} -.h5p-actions button:hover { - color: #333; -} -.h5p-actions button:active, -.h5p-actions button:focus, -.h5p-actions .h5p-link:active, -.h5p-actions .h5p-link:focus { - color: #666; -} -.h5p-actions button { - display: inline-flex; - padding: 0; - margin: 0; - color: #6A6A6A; - position: relative; - - /* Disable default button style */ - background: none; - border: none; - font: inherit; - cursor: pointer; - - line-height: 2; -} -.h5p-actions button:focus, -.h5p-actions .h5p-link:focus { - outline-style: solid; - outline-width: thin; - outline-offset: -2px; - outline-color: #5981A1; -} -.h5p-actions button:before { - font-family: 'H5P'; - font-size: 20px; - line-height: 23px; - vertical-align: bottom; - padding-right: 0; -} -.h5p-actions > .h5p-button.h5p-export > button:before { - content: "\e90b"; -} -.h5p-actions > .h5p-button.h5p-copyrights > button:before { - content: "\e88f"; -} -.h5p-actions > .h5p-button.h5p-embed > button:before { - content: "\e892"; -} -.h5p-actions .h5p-link { - float: right; - margin-right: 0; - font-size: 2.0em; - line-height: 23px; - position: relative; - color: #6a6a6a; - text-decoration: none; - outline: none; -} -.h5p-actions .h5p-link:before { - font-family: 'H5P'; - content: "\e88e"; - vertical-align: bottom; -} -.h5p-actions > li { - margin: 0; - list-style: none; -} -.h5p-popup-dialog { - position: absolute; - top: 0; - left: 0; - width: 100%; - min-height: 100%; - z-index: 100; - padding: 2em; - box-sizing: border-box; - -moz-box-sizing: border-box; - opacity: 0; - -webkit-transition: opacity 0.2s; - -moz-transition: opacity 0.2s; - -o-transition: opacity 0.2s; - transition: opacity 0.2s; - background:#000; - background:rgba(0,0,0,0.75); -} -.h5p-popup-dialog.h5p-open { - opacity: 1; -} -.h5p-popup-dialog .h5p-inner { - box-sizing: border-box; - -moz-box-sizing: border-box; - background: #fff; - height: 100%; - max-height: 100%; - position: relative; -} -.h5p-popup-dialog .h5p-inner > h2 { - position: absolute; - box-sizing: border-box; - -moz-box-sizing: border-box; - width: 100%; - margin: 0; - background: #eee; - display: block; - color: #656565; - font-size: 1.25em; - padding: 0.325em 0.5em 0.25em; - line-height: 1.25em; - border-bottom: 1px solid #ccc; - z-index: 2; -} -.h5p-popup-dialog .h5p-inner > h2 > a { - font-size: 12px; - margin-left: 1em; -} -.h5p-embed-dialog .h5p-inner, -.h5p-reuse-dialog .h5p-inner, -.h5p-content-user-data-reset-dialog .h5p-inner { - min-width: 316px; - max-width: 400px; - left: 50%; - top: 50%; - transform: translateX(-50%); -} -.h5p-embed-dialog .h5p-embed-code-container, -.h5p-embed-size { - resize: none; - outline: none; - width: 100%; - padding: 0.375em 0.5em 0.25em; - margin: 0; - overflow: hidden; - border: 1px solid #ccc; - box-shadow: 0 1px 2px 0 #d0d0d0 inset; - font-size: 0.875em; - letter-spacing: 0.065em; - font-family: sans-serif; - white-space: pre; - line-height: 1.5em; - height: 2.0714em; - background: #f5f5f5; - box-sizing: border-box; - -moz-box-sizing: border-box; -} -.h5p-embed-dialog .h5p-embed-code-container:focus { - height: 5em; -} -.h5p-embed-size { - width: 3.5em; - text-align: right; - margin: 0.5em 0; - line-height: 2em; -} -.h5p-popup-dialog .h5p-scroll-content { - border-top: 2.25em solid transparent; - padding: 1em; - box-sizing: border-box; - -moz-box-sizing: border-box; - color: #555555; - z-index: 1; -} -.h5p-popup-dialog.h5p-open .h5p-scroll-content { - overflow: auto; - overflow-x: hidden; - overflow-y: auto; - height: 100%; -} -.h5p-popup-dialog .h5p-scroll-content::-webkit-scrollbar { - width: 8px; -} -.h5p-popup-dialog .h5p-scroll-content::-webkit-scrollbar-track { - background: #e0e0e0; -} -.h5p-popup-dialog .h5p-scroll-content::-webkit-scrollbar-thumb { - box-shadow: 0 0 10px #000 inset; - border-radius: 4px; -} -.h5p-popup-dialog .h5p-close { - cursor: pointer; - font-size: 2em; - position: absolute; - right: 0; - top: 0; - width: 1.125em; - height: 1.125em; - line-height: 1.125em; - color: #656565; - cursor: pointer; - text-indent: -0.065em; - z-index: 3 -} -.h5p-popup-dialog .h5p-close:after { - font-family: 'H5P'; - content: "\e894"; -} -.h5p-popup-dialog .h5p-close:hover:after, -.h5p-popup-dialog .h5p-close:focus:after { - color: #454545; -} -.h5p-popup-dialog .h5p-close:active:after { - color: #252525; -} -.h5p-poopup-dialog h2 { - margin: 0.25em 0 0.5em; -} -.h5p-popup-dialog h3 { - margin: 0.75em 0 0.25em; -} -.h5p-popup-dialog dl { - margin: 0.25em 0 0.75em; -} -.h5p-popup-dialog dt { - float: left; - margin: 0 0.75em 0 0; -} -.h5p-popup-dialog dt:after { - content: ':'; -} -.h5p-popup-dialog dd { - margin: 0; -} -.h5p-expander { - cursor: pointer; - font-size: 1.125em; - margin: 0.5em 0 0; - display: inline-block; -} -.h5p-expander:before { - content: "+"; - width: 1em; - display: inline-block; - font-weight: bold; -} -.h5p-expander.h5p-open:before { - content: "-"; - text-indent: 0.125em; -} -.h5p-expander:hover, -.h5p-expander:focus { - color: #303030; -} -.h5p-expander:active { - color: #202020; -} -.h5p-expander-content { - display: none; -} -.h5p-expander-content p { - margin: 0.5em 0; -} -.h5p-content-copyrights { - border-left: 0.25em solid #d0d0d0; - margin-left: 0.25em; - padding-left: 0.25em; -} -.h5p-throbber { - background: url('../images/throbber.gif?ver=1.2.1') 10px center no-repeat; - padding-left: 38px; - min-height: 30px; - line-height: 30px; -} -.h5p-dialog-ok-button { - cursor: default; - float: right; - outline: none; - border: 2px solid #ccc; - padding: 0.25em 0.75em 0.125em; - background: #eee; -} -.h5p-dialog-ok-button:hover, -.h5p-dialog-ok-button:focus { - background: #fafafa; -} -.h5p-dialog-ok-button:active { - background: #eeffee; -} -.h5p-big-button { - line-height: 1.25; - display: block; - position: relative; - cursor: pointer; - width: 100%; - padding: 1em 1em 1em 3.75em; - text-align: left; - border: 1px solid #dedede; - background: linear-gradient(#ffffff, #f1f1f2); - border-radius: 0.25em; -} -.h5p-big-button:before { - font-family: 'h5p'; - content: "\e893"; - line-height: 1; - font-size: 3em; - color: #2747f7; - position: absolute; - left: 0.125em; - top: 0.125em; -} -.h5p-copy-button:before { - content: "\e905"; -} -.h5p-big-button:hover { - border: 1px solid #2747f7; - background: #eff1fe; -} -.h5p-big-button:active { - border: 1px solid #dedede; - background: #dfe4fe; -} -.h5p-button-title { - color: #2747f7; - font-size: 15px; - font-weight: bold; - margin-bottom: 0.5em; -} -.h5p-button-description { - color: #757575; -} -.h5p-horizontal-line-text { - border-top: 1px solid #dadada; - line-height: 1; - color: #474747; - text-align: center; - position: relative; - margin: 1.25em 0; -} -.h5p-horizontal-line-text > span { - background: white; - padding: 0.5em; - position: absolute; - top: -1em; - left: 50%; - transform: translateX(-50%); -} -.h5p-toast { - font-size: 0.75em; - background-color: rgba(0, 0, 0, 0.9); - color: #fff; - z-index: 110; - position: absolute; - padding: 0 0.5em; - line-height: 2; - border-radius: 4px; - white-space: nowrap; - pointer-events: none; - top: 0; - opacity: 1; - visibility: visible; - transition: opacity 1s; -} -.h5p-toast-disabled { - opacity: 0; - visibility: hidden; -} -.h5p-content code, -.h5peditor code { - color: #3d3d3d; - background: #e0e0e0; - border-radius: 2px; - padding: 0 5px; -} -.h5p-content pre > code, -.h5peditor pre > code { - background-color: #fafafa; - padding: 5px; - display: block; - line-height: normal; - border: 1px solid #c7c7c7; - border-left-width: 4px; - max-width: 100%; - white-space: pre; - overflow: auto; -} - - -/* This is loaded as part of Core and not Editor since this needs to be outside the editor iframe */ -.h5peditor-semi-fullscreen { - width: 100%; - height: 100%; - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - z-index: 101; -} -iframe.h5peditor-semi-fullscreen { - background: #fff; - z-index: 100001; -} - -.h5p-content.using-mouse *:not(textarea):focus { - outline: none !important; -} - -.h5p-content-hub-button:before { - font-family: "h5p"; - margin-right: 0.5em; - font-size: 0.7em; - line-height: 1; -} - -.h5p-content-hub-button.unpublish:before { - content: "\e916"; -} - -.h5p-content-hub-button.waiting:before, -.h5p-content-hub-button.sync:before { - content: "\e917"; -} - -.h5p-content-hub-button.waiting:before { - display: inline-block; - animation: rotate 2s linear infinite; -} - -@keyframes rotate { - to { - transform: rotate(360deg); - } -} diff --git a/h5p/h5plib/v126/joubel/editor/README.md b/h5p/h5plib/v126/joubel/editor/README.md deleted file mode 100644 index 18dc5221925d3..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/README.md +++ /dev/null @@ -1,10 +0,0 @@ -H5P Editor PHP Library -========== - -A general library that is supposed to be used in most PHP implementations of H5P. - -## License - -All code is licensed under MIT License - -Open Sans font is licensed under Apache license, Version 2.0 \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/editor/ckeditor/CHANGES.md b/h5p/h5plib/v126/joubel/editor/ckeditor/CHANGES.md deleted file mode 100644 index 1e7b8e8d89ded..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/ckeditor/CHANGES.md +++ /dev/null @@ -1,2117 +0,0 @@ -CKEditor 4 Changelog -==================== - -## CKEditor 4.17.1 - -Fixed issues: - -* [#4979](https://github.com/ckeditor/ckeditor4/issues/3757): Added cache key in [#4761](https://github.com/ckeditor/ckeditor4/issues/4761) started to breaking relative links for external CSS resources. The fix had been reverted and will be corrected in the upcoming release. - -## CKEditor 4.17 - -**Security Updates:** - -* Fixed XSS vulnerability in the core module reported by [William Bowling](https://github.com/wbowling). - - Issue summary: The vulnerability allowed to inject malformed comments HTML bypassing content sanitization, which could result in executing JavaScript code. See [security advisory](https://github.com/ckeditor/ckeditor4/security/advisories/GHSA-7h26-63m7-qhf2) for more details. - -* Fixed XSS vulnerability in the core module reported by [Maurice Dauer](https://twitter.com/laytonctf). - - Issue summary: The vulnerability allowed to inject malformed HTML bypassing content sanitization, which could result in executing JavaScript code. See [security advisory](https://github.com/ckeditor/ckeditor4/security/advisories/GHSA-pvmx-g8h5-cprj) for more details. - -You can read more details in the relevant security advisory and [contact us](security@cksource.com) if you have more questions. - -**An upgrade is highly recommended!** - -**Highlights:** - -Adobe [ended support of Flash Player](https://www.adobe.com/products/flashplayer/end-of-life.html) on December 31, 2020 and blocked Flash content from running in Flash Player beginning January 12, 2021. -We have decided to deprecate and remove the [Flash](https://ckeditor.com/cke4/addon/flash) plugin from CKEditor 4 to help protect users' systems and discourage using insecure software. - -New Features: - -* [#3433](https://github.com/ckeditor/ckeditor4/issues/3433): Marked required fields in dialogs with asterisk (`*`) symbol. -* [#4374](https://github.com/ckeditor/ckeditor4/issues/4374): Integrated the [Maximize](https://ckeditor.com/cke4/addon/maximize) plugin with browser's History API. -* [#4461](https://github.com/ckeditor/ckeditor4/issues/4461): Introduced the possibility to delay editor initialization while it is in a detached DOM element. -* [#4462](https://github.com/ckeditor/ckeditor4/issues/4462): Introduced support for reattaching editor container element to DOM. -* [#4612](https://github.com/ckeditor/ckeditor4/issues/4612): Allow pasting images as Base64 from [clipboard](https://ckeditor.com/cke4/addon/clipboard) in all browsers except IE. -* [#4681](https://github.com/ckeditor/ckeditor4/issues/4681): Allow drag and drop images as Base64. -* [#4750](https://github.com/ckeditor/ckeditor4/issues/4750): Added notification for pasting and dropping unsupported file types into the editor. -* [#4807](https://github.com/ckeditor/ckeditor4/issues/4807): [Chrome] Improved the performance of pasting large images. Thanks to [FlowIT-JIT](https://github.com/FlowIT-JIT)! -* [#4850](https://github.com/ckeditor/ckeditor4/issues/4850): Added support for loading [content templates](https://ckeditor.com/cke4/addon/templates) from HTML files. Thanks to [Fynn96](https://github.com/Fynn96)! -* [#4874](https://github.com/ckeditor/ckeditor4/issues/4874): Added the [`config.clipboard_handleImages`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-clipboard_handleImages) configuration option for enabling and disabling built-in support for pasting and dropping images in the [Clipboard](https://ckeditor.com/cke4/addon/clipboard) plugin. Thanks to [FlowIT-JIT](https://github.com/FlowIT-JIT)! -* [#4026](https://github.com/ckeditor/ckeditor4/issues/4026): [Preview](https://ckeditor.com/cke4/addon/preview) plugin now uses the [`editor#title`](http://localhost/ckeditor4-docs/build/docs/ckeditor4/latest/api/CKEDITOR_editor.html#property-title) property for the title of the preview window. Thanks to [Ely](https://github.com/Elyasin)! -* [#4467](https://github.com/ckeditor/ckeditor4/issues/4467): Added support for inserting content next to a block [widgets](https://ckeditor.com/cke4/addon/widget) using keyboard navigation. Thanks to [bunglegrind](https://github.com/bunglegrind)! - -Fixed Issues: - -* [#3757](https://github.com/ckeditor/ckeditor4/issues/3757): [Firefox] Fixed: images pasted from [clipboard](https://ckeditor.com/cke4/addon/clipboard) are not inserted as Base64-encoded images. -* [#3876](https://github.com/ckeditor/ckeditor4/issues/3876): Fixed: The [Print](https://ckeditor.com/cke4/addon/print) plugin incorrectly prints links and images. -* [#4444](https://github.com/ckeditor/ckeditor4/issues/4444): [Firefox] Fixed: Print preview is incorrectly loaded from CDN. -* [#4596](https://github.com/ckeditor/ckeditor4/issues/4596): Fixed: Incorrect handling of HSL/HSLA values in [`CKEDITOR.tools.color`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_tools_color.html). -* [#4597](https://github.com/ckeditor/ckeditor4/issues/4597): Fixed: Incorrect color conversion for HSL/HSLA values in [`CKEDITOR.tools.color`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_tools_color.html). -* [#4604](https://github.com/ckeditor/ckeditor4/issues/4604): Fixed: [`CKEDITOR.plugins.clipboard.dataTransfer#getTypes()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_clipboard_dataTransfer.html#method-getTypes) returns no types. -* [#4761](https://github.com/ckeditor/ckeditor4/issues/4761): Fixed: Not all resources loaded by the editor respect the cache key. -* [#4783](https://github.com/ckeditor/ckeditor4/issues/4783): Fixed: The [Accessibility Help](https://ckeditor.com/cke4/addon/a11yhelp) dialog does not contain info about focus being moved back to the editing area upon activating a toolbar button. -* [#4790](https://github.com/ckeditor/ckeditor4/issues/4790): Fixed: Printing page is invoked before the printed page is fully loaded. -* [#4874](https://github.com/ckeditor/ckeditor4/issues/4874): Fixed: Built-in support for pasting and dropping images in the [Clipboard](https://ckeditor.com/cke4/addon/clipboard) plugin restricts third party plugins from handling image pasting. Thanks to [FlowIT-JIT](https://github.com/FlowIT-JIT)! -* [#4888](https://github.com/ckeditor/ckeditor4/issues/4888): Fixed: The [`CKEDITOR.dialog#setState()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_dialog.html#method-setState) method throws error when there is no "OK" button in the dialog. -* [#4858](https://github.com/ckeditor/ckeditor4/issues/4858): Fixed: The [Autolink](https://ckeditor.com/cke4/addon/autolink) plugin incorrectly escapes the `&` characters when pasting links into the editor. -* [#4892](https://github.com/ckeditor/ckeditor4/issues/4892): Fixed: Focus of buttons in dialogs is not visible enough in High Contrast mode. -* [#3858](https://github.com/ckeditor/ckeditor4/issues/3858): Fixed: Pasting content in `ENTER_BR` [enter mode](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-enterMode) crashes the editor. -* [#4891](https://github.com/ckeditor/ckeditor4/issues/4891): Fixed: The [Autogrow](https://ckeditor.com/cke4/addon/autogrow) plugin applies fixed width to the editor. - -API Changes: - -* [#4462](https://github.com/ckeditor/ckeditor4/issues/4462): [`CKEDITOR.editor#getSelection()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_editor.html#method-getSelection) now returns `null` if the editor is in recreating state. -* [#4583](https://github.com/ckeditor/ckeditor4/issues/4583): Added support for new, comma-less color syntax to [`CKEDITOR.tools.color`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_tools_color.html). -* [#4604](https://github.com/ckeditor/ckeditor4/issues/4604): Added the [`CKEDITOR.plugins.clipboard.dataTransfer#isFileTransfer()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_clipboard_dataTransfer.html#method-isFileTransfer) method. -* [#4790](https://github.com/ckeditor/ckeditor4/issues/4790): Added `callback` parameter to [`CKEDITOR.plugins.preview#createPreview()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_preview.html#method-createPreview) method. - -Other Changes: - -* [#4866](https://github.com/ckeditor/ckeditor4/issues/#4866): The [Flash](https://ckeditor.com/cke4/addon/flash) plugin is now deprecated and has been removed from CKEditor 4. -* [#4901](https://github.com/ckeditor/ckeditor4/issues/4901): Redesigned buttons placement in the [Content templates](https://ckeditor.com/cke4/addon/templates) dialog to make it more UX friendly. Thanks to [Fynn96](https://github.com/Fynn96)! - -## CKEditor 4.16.2 - -**Security Updates:** - -* Fixed XSS vulnerability in the [Clipboard](https://ckeditor.com/cke4/addon/clipboard) plugin reported by [Anton Subbotin](https://github.com/skavans). - - Issue summary: The vulnerability allowed to abuse paste functionality using malformed HTML, which could result in injecting arbitrary HTML into the editor. See [security advisory](https://github.com/ckeditor/ckeditor4/security/advisories/GHSA-7889-rm5j-hpgg) for more details. - -* Fixed XSS vulnerability in the [Widget](https://ckeditor.com/cke4/addon/widget) plugin reported by [Anton Subbotin](https://github.com/skavans). - - Issue summary: The vulnerability allowed to abuse undo functionality using malformed [Widget](https://ckeditor.com/cke4/addon/widget) HTML, which could result in executing JavaScript code. See [security advisory](https://github.com/ckeditor/ckeditor4/security/advisories/GHSA-6226-h7ff-ch6c) for more details. - -* Fixed XSS vulnerability in the [Fake Objects](https://ckeditor.com/cke4/addon/fakeobjects) plugin reported by [Mika Kulmala](https://github.com/kulmik). - - Issue summary: The vulnerability allowed to inject malformed [Fake Objects](https://ckeditor.com/cke4/addon/fakeobjects) HTML, which could result in executing JavaScript code. See [security advisory](https://github.com/ckeditor/ckeditor4/security/advisories/GHSA-m94c-37g6-cjhc) for more details. - -You can read more details in the relevant security advisory and [contact us](security@cksource.com) if you have more questions. - -**An upgrade is highly recommended!** - -Fixed Issues: -* [#4777](https://github.com/ckeditor/ckeditor4/issues/4777): Fixed: HTML comments in widgets not processed correctly. -* [#4733](https://github.com/ckeditor/ckeditor4/pull/4733): Fixed: [Link](https://ckeditor.com/cke4/addon/link) prevent duplicate anchors in text with styles. - * [#4728](https://github.com/ckeditor/ckeditor4/issues/4728): Fixed: Multiple anchors in one line and multi-line with text style. - * [#3863](https://github.com/ckeditor/ckeditor4/issues/3863): Fixed: Multiple anchors in single word with text style. -* [#3819](https://github.com/ckeditor/ckeditor4/issues/3819): [Chrome] Fixed: After removing one of the two consecutive spaces, the ` ` character appears in the editor instead of a space. -* [#4666](https://github.com/ckeditor/ckeditor4/pull/4666): [IE] Introduce CSS.escape polyfill. Thanks to [limingli0707](https://github.com/limingli0707)! - * [#681](https://github.com/ckeditor/ckeditor4/issues/681): Fixed: Table elements (td, tr, th, ..) with an id that starts with dot (.) causes javascript runtime err. - * [#641](https://github.com/ckeditor/ckeditor4/issues/641): Fixed: UploadImage Plugin Widgets not working in IE, Opera, Safari, PhantomJS. -* [#3638](https://github.com/ckeditor/ckeditor4/issues/3638): Fixed: Opening the same dialog twice causes it to become hidden under the dialog's page cover. -* [#4247](https://github.com/ckeditor/ckeditor4/issues/4247): Fixed: [Color Button](https://ckeditor.com/cke4/addon/colorbutton)'s incorrect rendering on the first opening. -* [#4555](https://github.com/ckeditor/ckeditor4/issues/4555): Fixed: [Font](https://ckeditor.com/cke4/addon/font) styles with attributes are not applied correctly when used multiple times over the same selection. -* [#4782](https://github.com/ckeditor/ckeditor4/issues/4782): [Firefox] Fixed: `TypeError` is thrown when switching to Source View and back while [Autocomplete](https://ckeditor.com/cke4/addon/autocomplete) plugin is enabled. - -## CKEditor 4.16.1 - -Fixed Issues: -* [#4617](https://github.com/ckeditor/ckeditor4/issues/4617): Fixed: [Autocomplete](https://ckeditor.com/cke4/addon/autocomplete) is not accessible in inline editors. -* [#4493](https://github.com/ckeditor/ckeditor4/issues/4493): Fixed: The [drop-down](https://ckeditor.com/cke4/addon/richcombo) label does not reflect the current value of the drop-down. -* [#1572](https://github.com/ckeditor/ckeditor4/issues/1572): Fixed: A paragraph before or after a [widget](https://ckeditor.com/cke4/addon/widget) cannot be removed. Thanks to [bunglegrind](https://github.com/bunglegrind)! -* [#4301](https://github.com/ckeditor/ckeditor4/issues/4301): Fixed: Pasted content is overwritten when pasted in an initially empty editor with the [`div` Enter mode](https://ckeditor.com/docs/ckeditor4/latest/features/enterkey.html). -* [#4351](https://github.com/ckeditor/ckeditor4/issues/4351): Fixed: Incorrect values for RGBA/HSLA colors in [Color Dialog](https://ckeditor.com/cke4/addon/colordialog). -* [#4509](https://github.com/ckeditor/ckeditor4/issues/4509): Fixed: Incorrect handling of drag & drop inside [widgets](https://ckeditor.com/cke4/addon/widget) and nested editables. -* [#4611](https://github.com/ckeditor/ckeditor4/issues/4611): [Android, iOS] Fixed: Incorrect hover styles for buttons in the toolbar on mobile devices. -* [#4652](https://github.com/ckeditor/ckeditor4/issues/4652): Fixed: [Event data](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_eventInfo.html) set to `false` is treated as an event cancelation. -* [#4659](https://github.com/ckeditor/ckeditor4/issues/4659): Fixed: [`CKEDITOR.htmlParser`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_htmlParser.html) does not treat `--!>` as a comment end tag correctly. - -## CKEditor 4.16 - -**Security Updates:** - -* Fixed ReDoS vulnerability in the [Autolink](https://ckeditor.com/cke4/addon/autolink) plugin. - - Issue summary: It was possible to execute a ReDoS-type attack inside CKEditor 4 by persuading a victim to paste a specially crafted URL-like text into the editor and press Enter or Space. - -* Fixed ReDoS vulnerability in the [Advanced Tab for Dialogs](https://ckeditor.com/cke4/addon/dialogadvtab) plugin. - - Issue summary: It was possible to execute a ReDoS-type attack inside CKEditor 4 by persuading a victim to paste a specially crafted text into the Styles dialog. - -**An upgrade is highly recommended!** - -New Features: - -* [#2800](https://github.com/ckeditor/ckeditor4/issues/2800): Unsupported image formats are now gracefully handled by the [Paste from Word](https://ckeditor.com/cke4/addon/pastefromword) plugin on paste, additionally showing descriptive error messages. -* [#2800](https://github.com/ckeditor/ckeditor4/issues/2800): Unsupported image formats are now gracefully handled by the [Paste from LibreOffice](https://ckeditor.com/cke4/addon/pastefromlibreoffice) plugin on paste, additionally showing descriptive error messages. -* [#3582](https://github.com/ckeditor/ckeditor4/issues/3582): Introduced smart positioning of the [Autocomplete](https://ckeditor.com/cke4/addon/autocomplete) panel used by the [Mentions](https://ckeditor.com/cke4/addon/mentions) and [Emoji](https://ckeditor.com/cke4/addon/emoji) plugins. The panel will now be additionally positioned related to the browser viewport to be always fully visible. -* [#4388](https://github.com/ckeditor/ckeditor4/issues/4388): Added the option to remove an iframe created with the [IFrame Dialog](https://ckeditor.com/cke4/addon/iframe) plugin from the sequential keyboard navigation using the `tabindex` attribute. Thanks to [Timo Kirkkala](https://github.com/kirkkala)! - -Fixed Issues: - -* [#1134](https://github.com/ckeditor/ckeditor4/issues/1134): [Safari] Fixed: [Paste from Word](https://ckeditor.com/cke4/addon/pastefromword) does not embed images. -* [#2800](https://github.com/ckeditor/ckeditor4/issues/2800): Fixed: No images are imported from Microsoft Word when the content is pasted via the [Paste from Word](https://ckeditor.com/cke4/addon/pastefromword) plugin if there is at least one image of unsupported format. -* [#4379](https://github.com/ckeditor/ckeditor4/issues/4379): [Edge] Fixed: Incorrect detection of the [high contrast mode](https://ckeditor.com/docs/ckeditor4/latest/guide/dev_a11y.html#high-contrast-mode). -* [#4422](https://github.com/ckeditor/ckeditor4/issues/4422): Fixed: Missing space between the button name and the keyboard shortcut inside the button label in the [high contrast mode](https://ckeditor.com/docs/ckeditor4/latest/guide/dev_a11y.html#high-contrast-mode). -* [#2208](https://github.com/ckeditor/ckeditor4/issues/2208): [IE] Fixed: The [Autolink](https://ckeditor.com/cke4/addon/autolink) plugin duplicates the native browser implementation. -* [#1824](https://github.com/ckeditor/ckeditor4/issues/1824): Fixed: The [Autolink](https://ckeditor.com/cke4/addon/autolink) plugin should require the [Link](https://ckeditor.com/cke4/addon/link) plugin. -* [#4253](https://github.com/ckeditor/ckeditor4/issues/4253): Fixed: The [Editor Placeholder](https://ckeditor.com/cke4/addon/editorplaceholder) plugin throws an error during the editor initialization with [`config.fullPage`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-fullPage) enabled when there is no `` tag in the editor content. -* [#4372](https://github.com/ckeditor/ckeditor4/issues/4372): Fixed: The [Autogrow](https://ckeditor.com/cke4/addon/autogrow) plugin changes the editor's width when used with an absolute [`config.width`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-width) value. - -API Changes: - -* [#4358](https://github.com/ckeditor/ckeditor4/issues/4358): Introduced the [`CKEDITOR.tools.color`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_tools_color.html) class which adds colors validation and methods for converting colors between various formats: named colors, HEX, RGB, RGBA, HSL and HSLA. -* [#3782](https://github.com/ckeditor/ckeditor4/issues/3782): Moved the [`CKEDITOR.plugins.pastetools.filters.word.images`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_pastetools_filters_word_images.html) filters to the [`CKEDITOR.plugins.pastetools.filters.image`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_pastetools_filters_image.html) namespace. -* [#4297](https://github.com/ckeditor/ckeditor4/issues/4297): All [`CKEDITOR.plugins.pastetools.filters`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_plugins_pastetools_filters.html) are now available under the [`CKEDITOR.pasteTools`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR.html#property-pasteTools) alias. -* [#4394](https://github.com/ckeditor/ckeditor4/issues/4394): Introduced [`CKEDITOR.ajax`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_ajax.html) specialized loading methods for loading binary ([`CKEDITOR.ajax.loadBinary()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_ajax.html#method-loadBinary)) and text ([`CKEDITOR.ajax.loadText()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_ajax.html#method-loadText)) data. - -Other Changes: - -* The [WebSpellChecker](https://ckeditor.com/cke4/addon/wsc) (WSC) plugin is now disabled by default in [Standard and Full presets](https://ckeditor.com/cke4/presets). It can be enabled via [`extraPlugins`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-extraPlugins) configuration option. - -## CKEditor 4.15.1 - -**Security Updates:** - -* Fixed XSS vulnerability in the [Color History feature](https://ckeditor.com/docs/ckeditor4/latest/features/colorbutton.html#color-history) reported by [Mark Wade](https://github.com/mark-wade). - - Issue summary: It was possible to execute an XSS-type attack inside CKEditor 4 by persuading a victim to paste a specially crafted HTML code into the [Color Button](https://ckeditor.com/cke4/addon/colorbutton) dialog. - -**An upgrade is highly recommended!** - -Fixed Issues: - -* [#4293](https://github.com/ckeditor/ckeditor4/issues/4293): Fixed: The [`CKEDITOR.inlineAll()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR.html#method-inlineAll) method tries to initialize inline editor also on elements with an editor already attached to them. -* [#3961](https://github.com/ckeditor/ckeditor4/issues/3961): Fixed: The [Table Resize](https://ckeditor.com/cke4/addon/tableresize) plugin prevents editing of merged cells. -* [#3649](https://github.com/ckeditor/ckeditor4/issues/3649): Fixed: Applying a [block format](https://ckeditor.com/docs/ckeditor4/latest/features/format.html) should remove existing block styles. -* [#4282](https://github.com/ckeditor/ckeditor4/issues/4282): Fixed: The [script loader](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_scriptLoader.html) does not execute callback for scripts already loaded when called for the second time. Thanks to [Alexander Korotkevich](https://github.com/aldoom)! -* [#4273](https://github.com/ckeditor/ckeditor4/issues/4273): Fixed: A memory leak in the [`CKEDITOR.domReady()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR.html#method-domReady) method connected with not removing `load` event listeners. Thanks to [rohit1](https://github.com/rohit1)! -* [#1330](https://github.com/ckeditor/ckeditor4/issues/1330): Fixed: Incomplete CSS margin parsing if an `auto` or `0` value is used. -* [#4286](https://github.com/ckeditor/ckeditor4/issues/4286): Fixed: The [Auto Grow](https://ckeditor.com/cke4/addon/autogrow) plugin causes the editor width to be set to `0` on editor resize. -* [#848](https://github.com/ckeditor/ckeditor4/issues/848): Fixed: Arabic text not being "bound" correctly when pasting. Thanks to [Thomas Hunkapiller](https://github.com/devoidfury) and [J. Ivan Duarte Rodríguez](https://github.com/jidrone-mbm)! - -API Changes: - -* [#3649](https://github.com/ckeditor/ckeditor4/issues/3649): Added a new [`stylesRemove`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_editor.html#event-stylesRemove) editor event. - -Other Changes: - -* [#4262](https://github.com/ckeditor/ckeditor4/issues/4262): Removed the global reference to the `stylesLoaded` variable. Thanks to [Levi Carter](https://github.com/swiftMessenger)! -* Updated the [Export to PDF](https://ckeditor.com/cke4/addon/exportpdf) plugin to `1.0.1` version: - * Improved external CSS support for [classic editor](https://ckeditor.com/docs/ckeditor4/latest/examples/classic.html) by handling exceptions and displaying convenient [error messages](https://ckeditor.com/docs/ckeditor4/latest/guide/dev_errors.html#exportpdf-stylesheets-incaccessible). - -## CKEditor 4.15 - -New features: - -* [#3940](https://github.com/ckeditor/ckeditor4/issues/3940): Introduced the `colorName` property for customizing foreground and background styles in the [Color Button](https://ckeditor.com/cke4/addon/colorbutton) plugin via the [`config.colorButton_foreStyle`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-colorButton_foreStyle) and [`config.colorButton_backStyle`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-colorButton_backStyle) configuration options. -* [#3793](https://github.com/ckeditor/ckeditor4/issues/3793): Introduced the [Editor Placeholder](https://ckeditor.com/cke4/addon/editorplaceholder) plugin. -* [#1795](https://github.com/ckeditor/ckeditor4/issues/1795): The colors picked from the [Color Dialog](https://ckeditor.com/cke4/addon/colordialog) are now stored in the [Color Button](https://ckeditor.com/cke4/addon/colorbutton) palette and can be reused easily. -* [#3783](https://github.com/ckeditor/ckeditor4/issues/3783): The colors used in the document are now displayed as a part of the [Color Button](https://ckeditor.com/cke4/addon/colorbutton) palette. - -Fixed Issues: - -* [#4060](https://github.com/ckeditor/ckeditor4/issues/4060): Fixed: The content inside a [widget](https://ckeditor.com/cke4/addon/widget) nested editable is escaped twice. -* [#4183](https://github.com/ckeditor/ckeditor4/issues/4183): [Safari] Fixed: Incorrect image dimensions when using the [Easy Image](https://ckeditor.com/cke4/addon/easyimage) plugin alongside the [IFrame Editing Area](https://ckeditor.com/cke4/addon/wysiwygarea) plugin. -* [#3693](https://github.com/ckeditor/ckeditor4/issues/3693): Fixed: Incorrect default values for several [Color Button](https://ckeditor.com/cke4/addon/colorbutton) configuration variables in the API documentation. -* [#3795](https://github.com/ckeditor/ckeditor4/issues/3795): Fixed: Setting the [`config.dataIndentationChars`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-dataIndentationChars) configuration option to an empty string is ignored and replaced by a tab (`\t`) character. Thanks to [Thomas Grinderslev](https://github.com/Znegl)! -* [#4107](https://github.com/ckeditor/ckeditor4/issues/4107): Fixed: Multiple [Autocomplete](https://ckeditor.com/cke4/addon/autocomplete) instances cause keyboard navigation issues. -* [#4041](https://github.com/ckeditor/ckeditor4/issues/4041): Fixed: The[`selection.scrollIntoView`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_dom_selection.html#method-scrollIntoView) method throws an error when the editor selection is not set. -* [#3361](https://github.com/ckeditor/ckeditor4/issues/3361): Fixed: Loading multiple [custom editor configurations](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-customConfig) is prone to a race condition between these. -* [#4007](https://github.com/ckeditor/ckeditor4/issues/4007): Fixed: Screen readers do not announce the [Rich Combo](https://ckeditor.com/cke4/addon/richcombo) plugin is collapsed or expanded. -* [#4141](https://github.com/ckeditor/ckeditor4/issues/4141): Fixed: The styles are incorrectly applied when there is a `` DOM element. - -Other Changes: - -* [#13859](https://dev.ckeditor.com/ticket/13859): Test cases created with `bender.tools.createTestsForEditors` will also receive editor bot as a second parameter. - -## CKEditor 4.5.4 - -New Features: - -* [#13632](https://dev.ckeditor.com/ticket/13632): Introduce error logging mechanism. -* [#13730](https://dev.ckeditor.com/ticket/13730): Switch to the new error logging mechanism. - -Fixed Issues: - -* [#9856](https://dev.ckeditor.com/ticket/9856): Fixed: Cannot use the native context menu together with the [Div Editing Area](https://ckeditor.com/cke4/addon/divarea) plugin. Thanks to [Mark Wade](https://github.com/mark-wade)! -* [#12733](https://dev.ckeditor.com/ticket/12733): [IE9+] Fixed: Radio button `onChange` does not work. Thanks to [Iliya Kostadinov](https://github.com/iliyakostadinov)! -* [#13142](https://dev.ckeditor.com/ticket/13142): [Edge] Fixed: *Ctrl+A* and then *Backspace* result in an empty `
    ` element. -* [#13599](https://dev.ckeditor.com/ticket/13599): Fixed: Cross-editor drag and drop of an inline widget results in error/artifacts. -* [#13640](https://dev.ckeditor.com/ticket/13640): [IE] Fixed: Dropping a widget outside the `` element is not handled correctly. -* [#13533](https://dev.ckeditor.com/ticket/13533): Fixed: No progress during upload. -* [#13680](https://dev.ckeditor.com/ticket/13680): Fixed: The parser should allow the `` element to be a child of the `` element. -* [#11724](https://dev.ckeditor.com/ticket/11724): [Touch devices] Fixed: Drop-downs often hide right after opening them. -* [#13690](https://dev.ckeditor.com/ticket/13690): Fixed: Copying content from IE to Chrome adds an extra paragraph. -* [#13284](https://dev.ckeditor.com/ticket/13284): Fixed: Cannot drag and drop a widget if the text caret is placed just after the widget instance. -* [#13516](https://dev.ckeditor.com/ticket/13516): Fixed: CKEditor removes empty HTML5 anchors without the `name` attribute. -* [#13765](https://dev.ckeditor.com/ticket/13765): [Safari 9] Fixed: Problems with rendering samples. - -Other Changes: - -* [#11725](https://dev.ckeditor.com/ticket/11725): Marked [`CKEDITOR.env.mobile`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_env.html#property-mobile) as deprecated. The reason is that it is no longer clear what "mobile" means. -* [#13737](https://dev.ckeditor.com/ticket/13737): Upgraded [Bender.js](https://github.com/benderjs/benderjs) to 0.4.1. - -## CKEditor 4.5.3 - -New Features: - -* [#13501](https://dev.ckeditor.com/ticket/13501): Added the [`config.fileTools_defaultFileName`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_config.html#cfg-fileTools_defaultFileName) option to allow setting a default file name for paste uploads. -* [#13603](https://dev.ckeditor.com/ticket/13603): Added support for uploading dropped BMP images. - -Fixed Issues: - -* [#13590](https://dev.ckeditor.com/ticket/13590): Fixed: Various issues related to the [Paste from Word](https://ckeditor.com/cke4/addon/pastefromword) feature. Fixes also: - * [#11215](https://dev.ckeditor.com/ticket/11215), - * [#8780](https://dev.ckeditor.com/ticket/8780), - * [#12762](https://dev.ckeditor.com/ticket/12762). -* [#13386](https://dev.ckeditor.com/ticket/13386): [Edge] Fixed: Issues with selecting and editing images. -* [#13568](https://dev.ckeditor.com/ticket/13568): Fixed: The [`editor.getSelectedHtml()`](https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_editor.html#method-getSelectedHtml) method returns invalid results for entire content selection. -* [#13453](https://dev.ckeditor.com/ticket/13453): Fixed: Drag&drop of entire editor content throws an error. -* [#13465](https://dev.ckeditor.com/ticket/13465): Fixed: Error is thrown and the widget is lost on drag&drop if it is the only content of the editor. -* [#13414](https://dev.ckeditor.com/ticket/13414): Fixed: Content auto paragraphing in a nested editable despite editor configuration. -* [#13429](https://dev.ckeditor.com/ticket/13429): Fixed: Incorrect selection after content insertion by the [Auto Embed](https://ckeditor.com/cke4/addon/autoembed) plugin. -* [#13388](https://dev.ckeditor.com/ticket/13388): Fixed: [Table Resize](https://ckeditor.com/cke4/addon/tableresize) integration with [Undo](https://ckeditor.com/cke4/addon/undo) is broken. - -Other Changes: - -* [#13637](https://dev.ckeditor.com/ticket/13637): Several icons were refactored. -* Updated [Bender.js](https://github.com/benderjs/benderjs) to 0.3.0 and introduced the ability to run tests via HTTPs ([#13265](https://dev.ckeditor.com/ticket/13265)). - -## CKEditor 4.5.2 - -Fixed Issues: - -* [#13609](https://dev.ckeditor.com/ticket/13609): [Edge] Fixed: The browser crashes when switching to the source mode. Thanks to [Andrew Williams and Mark Smeed](http://webxsolution.com/)! -* [PR#201](https://github.com/ckeditor/ckeditor4/pull/201): Fixed: Buttons in the toolbar configurator cause form submission. Thanks to [colemanw](https://github.com/colemanw)! -* [#13422](https://dev.ckeditor.com/ticket/13422): Fixed: A monospaced font should be used in the `'; - - return ns.createFieldMarkup(this.field, ns.createImportantDescription(this.field.important) + input, id); -}; - -/** - * Validate the current text field. - */ -ns.Textarea.prototype.validate = function () { - var value = H5P.trim(this.$input.val()); - var valid = true; - - // Clear errors before showing new ones - this.$errors.html(''); - - if ((this.field.optional === undefined || !this.field.optional) && !value.length) { - this.$errors.append(ns.createError(ns.t('core', 'requiredProperty', {':property': ns.t('core', 'textField')}))); - valid = false; - } - else if (value.length > this.field.maxLength) { - this.$errors.append(ns.createError(ns.t('core', 'tooLong', {':max': this.field.maxLength}))); - valid = false; - } - else if (this.field.regexp !== undefined && !value.match(new RegExp(this.field.regexp.pattern, this.field.regexp.modifiers))) { - this.$errors.append(ns.createError(ns.t('core', 'invalidFormat'))); - valid = false; - } - - this.$input.toggleClass('error', !valid); - - return ns.checkErrors(this.$errors, this.$input, value); -}; - -/** - * Remove this item. - */ -ns.Textarea.prototype.remove = function () { - this.$item.remove(); -}; - -// Tell the editor what semantic field we are. -ns.widgets.textarea = ns.Textarea; diff --git a/h5p/h5plib/v126/joubel/editor/scripts/h5peditor.js b/h5p/h5plib/v126/joubel/editor/scripts/h5peditor.js deleted file mode 100644 index 98a88cf2ac060..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/scripts/h5peditor.js +++ /dev/null @@ -1,1887 +0,0 @@ -/* global ns */ -/** - * This file contains helper functions for the editor. - */ - -// Grab common resources set in parent window, but avoid sharing back resources set in iframe) -window.ns = window.H5PEditor = H5P.jQuery.extend(false, {}, window.parent.H5PEditor); -ns.$ = H5P.jQuery; - -// Load needed resources from parent. -H5PIntegration = H5P.jQuery.extend(false, {}, window.parent.H5PIntegration); -H5PIntegration.loadedJs = []; -H5PIntegration.loadedCss = []; - -/** - * Constants used within editor - * - * @type {{otherLibraries: string}} - */ -ns.constants = { - otherLibraries: 'Other Libraries', -}; - -/** - * Keep track of our widgets. - */ -ns.widgets = {}; - -/** - * Caches library data (semantics, js and css) - */ -ns.libraryCache = {}; - -/** - * Keeps track of callbacks to run once a library gets loaded. - */ -ns.loadedCallbacks = []; - -/** - * Keep track of which libraries have been loaded in the browser, i.e CSS is - * added and JS have been run - * - * @type {Object} - */ -ns.libraryLoaded = {}; - -/** - * Indiciates if the user is using Internet Explorer. - */ -ns.isIE = navigator.userAgent.match(/; MSIE \d+.\d+;/) !== null; - -/** - * Keep track of renderable common fields. - * - * @type {Object} - */ -ns.renderableCommonFields = {}; - -(() => { - const loading = {}; // Map of callbacks for each src being loaded - - /** - * Help load JavaScripts, prevents double loading. - * - * @param {string} src - * @param {Function} done Callback - */ - ns.loadJs = (src, done) => { - if (H5P.jsLoaded(src)) { - // Already loaded - done(); - return; - } - - if (loading[src] !== undefined) { - // Loading in progress... - loading[src].push(done); - return; - } - - loading[src] = [done]; - - // Load using script tag - var script = document.createElement('script'); - script.type = 'text/javascript'; - script.charset = 'UTF-8'; - script.async = false; - script.onload = function () { - H5PIntegration.loadedJs.push(src); - loading[src].forEach(cb => cb()); - delete loading[src]; - }; - script.onerror = function (err) { - loading[src].forEach(cb => cb(err)); - delete loading[src]; - }; - script.src = src; - document.head.appendChild(script); - }; -})(); - -/** - * Helper function invoked when a library is requested. Will add CSS and eval JS - * if not already done. - * - * @private - * @param {string} libraryName On the form "machineName majorVersion.minorVersion" - * @param {Function} callback - */ -ns.libraryRequested = function (libraryName, callback) { - var libraryData = ns.libraryCache[libraryName]; - - if (!ns.libraryLoaded[libraryName]) { - // Add CSS. - if (libraryData.css !== undefined) { - libraryData.css.forEach(function (path) { - if (!H5P.cssLoaded(path)) { - H5PIntegration.loadedCss.push(path); - if (path) { - ns.$('head').append(''); - } - } - }); - } - - // Add JS - var loadingJs = false; - if (libraryData.javascript !== undefined && libraryData.javascript.length) { - libraryData.javascript.forEach(function (path) { - if (!H5P.jsLoaded(path)) { - loadingJs = true; - ns.loadJs(path, function (err) { - if (err) { - console.error('Error while loading script', err); - return; - } - - var isFinishedLoading = libraryData.javascript.reduce(function (hasLoaded, jsPath) { - return hasLoaded && H5P.jsLoaded(jsPath); - }, true); - - if (isFinishedLoading) { - ns.libraryLoaded[libraryName] = true; - - // Need to set translations after all scripts have been loaded - if (libraryData.translations) { - for (var machineName in libraryData.translations) { - H5PEditor.language[machineName] = libraryData.translations[machineName]; - } - } - - callback(ns.libraryCache[libraryName].semantics); - } - }); - } - }); - } - if (!loadingJs) { - // Don't have to wait for any scripts, run callback - ns.libraryLoaded[libraryName] = true; - callback(ns.libraryCache[libraryName].semantics); - } - } - else { - // Already loaded, run callback - callback(ns.libraryCache[libraryName].semantics); - } -}; - -/** - * Loads the given library, inserts any css and js and - * then runs the callback with the samantics as an argument. - * - * @param {string} libraryName - * On the form machineName majorVersion.minorVersion - * @param {function} callback - * @returns {undefined} - */ -ns.loadLibrary = function (libraryName, callback) { - switch (ns.libraryCache[libraryName]) { - default: - // Get semantics from cache. - ns.libraryRequested(libraryName, callback); - break; - - case 0: - // Add to queue. - if (ns.loadedCallbacks[libraryName] === undefined) { - ns.loadedCallbacks[libraryName] = []; - } - ns.loadedCallbacks[libraryName].push(callback); - break; - - case undefined: - // Load semantics. - ns.libraryCache[libraryName] = 0; // Indicates that others should queue. - ns.loadedCallbacks[libraryName] = []; // Other callbacks to run once loaded. - var library = ns.libraryFromString(libraryName); - - var url = ns.getAjaxUrl('libraries', library); - - // Add content language to URL - if (ns.contentLanguage !== undefined) { - url += (url.indexOf('?') === -1 ? '?' : '&') + 'language=' + ns.contentLanguage; - } - // Add common fields default lanuage to URL - const defaultLanguage = ns.defaultLanguage; // Avoid changes after sending AJAX - if (defaultLanguage !== undefined) { - url += (url.indexOf('?') === -1 ? '?' : '&') + 'default-language=' + defaultLanguage; - } - - // Fire away! - ns.$.ajax({ - url: url, - success: function (libraryData) { - libraryData.translation = { // Used to cache all the translations - en: libraryData.semantics - }; - let languageSemantics = []; - if (libraryData.language !== null) { - languageSemantics = JSON.parse(libraryData.language).semantics; - delete libraryData.language; // Avoid caching a lot of unused data - } - var semantics = ns.$.extend(true, [], libraryData.semantics, languageSemantics); - if (libraryData.defaultLanguage !== null) { - libraryData.translation[defaultLanguage] = JSON.parse(libraryData.defaultLanguage).semantics; - delete libraryData.defaultLanguage; // Avoid caching a lot of unused data - ns.updateCommonFieldsDefault(semantics, libraryData.translation[defaultLanguage]); - } - libraryData.semantics = semantics; - ns.libraryCache[libraryName] = libraryData; - - ns.libraryRequested(libraryName, function (semantics) { - callback(semantics); - - // Run queue. - if (ns.loadedCallbacks[libraryName]) { - for (var i = 0; i < ns.loadedCallbacks[libraryName].length; i++) { - ns.loadedCallbacks[libraryName][i](semantics); - } - } - }); - }, - error: function (jqXHR, textStatus, errorThrown) { - if (window['console'] !== undefined) { - console.warn('Ajax request failed'); - console.warn(jqXHR); - console.warn(textStatus); - console.warn(errorThrown); - } - }, - dataType: 'json' - }); - } -}; - -/** - * Update common fields default values for the given semantics. - * Works by reference. - * - * @param {Array} semantics - * @param {Array} translation - * @param {boolean} [parentIsCommon] Used to indicated that one of the ancestors is a common field - */ -ns.updateCommonFieldsDefault = function (semantics, translation, parentIsCommon) { - for (let i = 0; i < semantics.length; i++) { - const isCommon = (semantics[i].common === true || parentIsCommon); - if (isCommon && semantics[i].default !== undefined && - translation[i] !== undefined && translation[i].default !== undefined) { - // Update value - semantics[i].default = translation[i].default; - } - if (semantics[i].fields !== undefined && semantics[i].fields.length && - translation[i].fields !== undefined && translation[i].fields.length) { - // Look into sub fields - ns.updateCommonFieldsDefault(semantics[i].fields, translation[i].fields, isCommon); - } - if (semantics[i].field !== undefined && translation[i].field !== undefined ) { - // Look into sub field - ns.updateCommonFieldsDefault([semantics[i].field], [translation[i].field], isCommon); - } - } -}; - -/** - * Reset loaded libraries - i.e removes CSS added previously. - * @method - * @return {[type]} - */ -ns.resetLoadedLibraries = function () { - ns.$('head style.h5p-editor-style').remove(); - H5PIntegration.loadedCss = []; - H5PIntegration.loadedJs = []; - ns.loadedCallbacks = []; - ns.libraryLoaded = {}; - ns.libraryCache = {}; -}; - -/** - * Render common fields of content type with given machine name - * - * @param {string} machineName Machine name of content type with common fields - * @param {Array} [libraries] Library data for machine name - */ -ns.renderCommonField = function (machineName, libraries) { - var commonFields = ns.renderableCommonFields[machineName].fields; - var renderableCommonFields = []; - var ancestor; - - commonFields.forEach(function (field) { - if (!field.rendered) { - var commonField = ns.addCommonField( - field.field, - field.parent, - field.params, - field.ancestor, - true - ); - if (commonField.setValues.length === 1) { - renderableCommonFields.push({ - field: field, - instance: commonField.instance - }); - field.instance = commonField.instance; - } - } - field.rendered = true; - }); - - // Render common fields if found - if (renderableCommonFields.length) { - var libraryName = machineName === ns.constants.otherLibraries ? machineName - : (machineName.length ? machineName.split(' ')[0] : ''); - if (libraries.length && libraries[0].title) { - libraryName = libraries[0].title; - } - - // Create a library wrapper - var hasLibraryWrapper = !!ns.renderableCommonFields[machineName].wrapper; - var commonFieldsLibraryWrapper = ns.renderableCommonFields[machineName].wrapper; - if (!hasLibraryWrapper) { - commonFieldsLibraryWrapper = document.createElement('fieldset'); - var libraryWrapperClass = libraryName.replace(/\s+/g, '-').toLowerCase(); - - commonFieldsLibraryWrapper.classList.add('common-fields-library-wrapper'); - commonFieldsLibraryWrapper.classList.add('common-fields-' + libraryWrapperClass); - - var libraryTitle = document.createElement('legend'); - libraryTitle.classList.add('common-field-legend'); - libraryTitle.textContent = libraryName; - libraryTitle.tabIndex = '0'; - libraryTitle.setAttribute('role', 'button'); - libraryTitle.addEventListener('click', function () { - commonFieldsLibraryWrapper.classList.toggle('expanded'); - }); - libraryTitle.addEventListener('keypress', function (e) { - if (e.which === 32) { - commonFieldsLibraryWrapper.classList.toggle('expanded'); - } - }); - commonFieldsLibraryWrapper.appendChild(libraryTitle); - - ns.renderableCommonFields[machineName].wrapper = commonFieldsLibraryWrapper; - } - - renderableCommonFields.forEach(function (commonField) { - commonField.instance.appendTo(ns.$(commonFieldsLibraryWrapper)); - // Gather under a common ancestor - if (commonField.field && commonField.field.ancestor) { - ancestor = commonField.field.ancestor; - - // Ensure that params are updated after common field instance is - // appended since this ensures that defaults are set for common fields - const field = commonField.field; - const library = field.parent.currentLibrary; - const fieldName = field.field.name; - const ancestorField = ancestor.commonFields[library][fieldName]; - ancestorField.params = field.params[fieldName]; - } - }); - - if (!hasLibraryWrapper && ancestor) { - ancestor.$common[0].appendChild(commonFieldsLibraryWrapper); - } - } -}; - -/** - * Recursively traverse parents to find the library our field belongs to - * - * @param parent - * @returns {*} - */ -ns.getParentLibrary = function (parent) { - if (!parent) { - return null; - } - - if (parent.currentLibrary) { - return parent.currentLibrary; - } - - return ns.getParentLibrary(parent.parent); -}; - -/** - * Recursive processing of the semantics chunks. - * - * @param {array} semanticsChunk - * @param {object} params - * @param {jQuery} $wrapper - * @param {mixed} parent - * @param {string} [machineName] Machine name of library that is being processed - * @returns {undefined} - */ -ns.processSemanticsChunk = function (semanticsChunk, params, $wrapper, parent, machineName) { - var ancestor; - parent.children = []; - - if (parent.passReadies === undefined) { - throw 'Widget tried to run processSemanticsChunk without handling ready callbacks. [field:' + parent.field.type + ':' + parent.field.name + ']'; - } - - if (!parent.passReadies) { - // If the parent can't pass ready callbacks we need to take care of them. - parent.readies = []; - } - - for (var i = 0; i < semanticsChunk.length; i++) { - var field = semanticsChunk[i]; - - // Check generic field properties. - if (field.name === undefined) { - throw ns.t('core', 'missingProperty', {':index': i, ':property': 'name'}); - } - if (field.type === undefined) { - throw ns.t('core', 'missingProperty', {':index': i, ':property': 'type'}); - } - - // Set default value. - if (params[field.name] === undefined && field['default'] !== undefined) { - params[field.name] = field['default']; - } - - var widget = ns.getWidgetName(field); - - // TODO: Remove later, this is here for debugging purposes. - if (ns.widgets[widget] === undefined) { - $wrapper.append('
    [field:' + field.type + ':' + widget + ':' + field.name + ']
    '); - continue; - } - - // Add common fields to bottom of form. - if (field.common !== undefined && field.common) { - if (ancestor === undefined) { - ancestor = ns.findAncestor(parent); - } - - var parentLibrary = ns.getParentLibrary(parent); - var library = machineName ? machineName - : (field.library ? field.library - : (parentLibrary ? parentLibrary - : ns.constants.otherLibraries)); - ns.renderableCommonFields[library] = ns.renderableCommonFields[library] || {}; - ns.renderableCommonFields[library].fields = ns.renderableCommonFields[library].fields || []; - - // Add renderable if it doesn't exist - ns.renderableCommonFields[library].fields.push({ - field: field, - parent: parent, - params: params, - ancestor: ancestor, - rendered: false - }); - continue; - } - - var fieldInstance = new ns.widgets[widget](parent, field, params[field.name], function (field, value) { - if (value === undefined) { - delete params[field.name]; - } - else { - params[field.name] = value; - } - }); - fieldInstance.appendTo($wrapper); - parent.children.push(fieldInstance); - } - - // Render all gathered common field - if (ns.renderableCommonFields) { - for (var commonFieldMachineName in ns.renderableCommonFields) { - if (commonFieldMachineName === ns.constants.otherLibraries) { - // No need to grab library info - ns.renderCommonField(commonFieldMachineName); - } - else { - // Get title for common fields group - H5PEditor.LibraryListCache.getLibraries( - [commonFieldMachineName], - ns.renderCommonField.bind(this, commonFieldMachineName) - ); - } - } - } - - if (!parent.passReadies) { - // Run ready callbacks. - for (i = 0; i < parent.readies.length; i++) { - parent.readies[i](); - } - delete parent.readies; - } -}; - -/** - * Attach ancestor of parent's common fields to a new wrapper - * - * @param {Object} parent Parent content type instance that common fields should be attached to - * @param {HTMLElement} wrapper New wrapper of common fields - */ -ns.setCommonFieldsWrapper = function (parent, wrapper) { - var ancestor = ns.findAncestor(parent); - // Hide the ancestor whose children will be reattached elsewhere - wrapper.appendChild(ancestor.$common[0]); -}; - -/** - * Add a field to the common container. - * - * @param {object} field - * @param {object} parent - * @param {object} params - * @param {object} ancestor - * @param {boolean} [skipAppendTo] Skips appending the common field if set - * @returns {undefined} - */ -ns.addCommonField = function (field, parent, params, ancestor, skipAppendTo) { - var commonField; - - // Group all fields based on library name + version - if (ancestor.commonFields[parent.currentLibrary] === undefined) { - ancestor.commonFields[parent.currentLibrary] = {}; - } - - // Field name will have to be unique for library - if (ancestor.commonFields[parent.currentLibrary][field.name] === undefined) { - var widget = ns.getWidgetName(field); - ancestor.commonFields[parent.currentLibrary][field.name] = { - instance: new ns.widgets[widget](parent, field, params[field.name], function (field, value) { - for (var i = 0; i < commonField.setValues.length; i++) { - commonField.setValues[i](field, value); - } - }), - setValues: [], - parents: [] - }; - } - - commonField = ancestor.commonFields[parent.currentLibrary][field.name]; - commonField.parents.push(ns.findLibraryAncestor(parent)); - commonField.setValues.push(function (field, value) { - if (value === undefined) { - delete params[field.name]; - } - else { - params[field.name] = value; - } - }); - - if (commonField.setValues.length === 1) { - ancestor.$common.parent().removeClass('hidden'); - if (!skipAppendTo) { - commonField.instance.appendTo(ancestor.$common); - } - commonField.params = params[field.name]; - } - else { - params[field.name] = commonField.params; - } - - parent.children.push(commonField.instance); - return commonField; -}; - -/** - * Find the nearest library ancestor. Used when adding commonfields. - * - * @param {object} parent - * @returns {ns.findLibraryAncestor.parent|@exp;ns@call;findLibraryAncestor} - */ -ns.findLibraryAncestor = function (parent) { - if (parent.parent === undefined || parent.field.type === 'library') { - return parent; - } - return ns.findLibraryAncestor(parent.parent); -}; - -/** - * getParentZebra - * - * Alternate the background color of fields - * - * @param parent - * @returns {string} to determine background color of callee - */ -ns.getParentZebra = function (parent) { - if (parent.zebra) { - return parent.zebra; - } - else { - return ns.getParentZebra(parent.parent); - } -}; - -/** - * Find the nearest ancestor which handles commonFields. - * - * @param {type} parent - * @returns {@exp;ns@call;findAncestor|ns.findAncestor.parent} - */ -ns.findAncestor = function (parent) { - if (parent.commonFields === undefined) { - return ns.findAncestor(parent.parent); - } - return parent; -}; - -/** - * Call remove on the given children. - * - * @param {Array} children - * @returns {unresolved} - */ -ns.removeChildren = function (children) { - if (children === undefined) { - return; - } - - for (var i = 0; i < children.length; i++) { - // Common fields will be removed by library. - var isCommonField = (children[i].field === undefined || - children[i].field.common === undefined || - !children[i].field.common); - - var hasRemove = (children[i].remove instanceof Function || - typeof children[i].remove === 'function'); - - if (isCommonField && hasRemove) { - children[i].remove(); - } - } -}; - -/** - * Find field from path. - * - * @param {String} path - * @param {Object} parent - * @returns {@exp;ns.Form@call;findField|Boolean} - */ -ns.findField = function (path, parent) { - if (typeof path === 'string') { - path = path.split('/'); - } - - if (path[0] === '..') { - path.splice(0, 1); - return ns.findField(path, parent.parent); - } - if (parent.children) { - for (var i = 0; i < parent.children.length; i++) { - if (parent.children[i].field.name === path[0]) { - path.splice(0, 1); - if (path.length) { - return ns.findField(path, parent.children[i]); - } - else { - return parent.children[i]; - } - } - } - } - - return false; -}; - -/** - * Find a semantics field in the semantics structure by name of the field - * Will return the first found by depth first search if there are identically named fields - * - * @param {string} fieldName Name of the field we wish to find - * @param {Object|Array} semanticsStructure Semantics we wish to find the field within - * @returns {null|Object} Returns the field if found, otherwise null. - */ -ns.findSemanticsField = function (fieldName, semanticsStructure) { - if (Array.isArray(semanticsStructure)) { - for (let i = 0; i < semanticsStructure.length; i++) { - var semanticsField = ns.findSemanticsField(fieldName, semanticsStructure[i]); - if (semanticsField !== null) { - // Return immediately if field is found - return semanticsField; - } - } - return null; - } - else if (semanticsStructure.name === fieldName) { - return semanticsStructure; - } - else if (semanticsStructure.field) { - // Process field - return ns.findSemanticsField(fieldName, semanticsStructure.field); - } - else if (semanticsStructure.fields) { - // Process fields - return ns.findSemanticsField(fieldName, semanticsStructure.fields); - } - else { - // No matching semantics found within known properties and list structures - return null; - } -}; - -/** - * Follow a field and get all changes to its params. - * - * @param {Object} parent The parent object of the field. - * @param {String} path Relative to parent object. - * @param {Function} callback Gets called for params changes. - * @returns {undefined} - */ -ns.followField = function (parent, path, callback) { - if (path === undefined) { - return; - } - - // Find field when tree is ready. - parent.ready(function () { - var def; - - if (path instanceof Object) { - // We have an object with default values - def = H5P.cloneObject(path); - - if (path.field === undefined) { - callback(path, null); - return; // Exit if we have no field to follow. - } - - path = def.field; - delete def.field; - } - - var field = ns.findField(path, parent); - - if (!field) { - throw ns.t('core', 'unknownFieldPath', {':path': path}); - } - if (field.changes === undefined) { - throw ns.t('core', 'noFollow', {':path': path}); - } - - var params = (field.params === undefined ? def : field.params); - callback(params, field.changes.length + 1); - - field.changes.push(function () { - var params = (field.params === undefined ? def : field.params); - callback(params); - }); - }); -}; - -/** - * Create HTML wrapper for error messages. - * - * @param {String} message - * @returns {String} - */ -ns.createError = function (message) { - return '

    ' + message + '

    '; -}; - -/** - * Turn a numbered importance into a string. - * - * @param {string} importance - * @returns {String} - */ -ns.createImportance = function (importance) { - return importance ? 'importance-' + importance : ''; -}; - -/** - * Create HTML wrapper for field items. - * Makes sure the different elements are placed in an consistent order. - * - * @param {string} type - * @param {string} [label] - * @param {string} [description] - * @param {string} [content] - * @deprecated since version 1.12 (Jan. 2017, will be removed Jan. 2018). Use createFieldMarkup instead. - * @see createFieldMarkup - * @returns {string} HTML - */ -ns.createItem = function (type, label, description, content) { - return '
    ' + - (label ? label : '') + - (description ? '
    ' + description + '
    ' : '') + - (content ? content : '') + - '
    ' + - '
    '; -}; - -/** - * An object describing the semantics of a field - * @typedef {Object} SemanticField - * @property {string} name - * @property {string} type - * @property {string} label - * @property {string} [importance] - * @property {string} [description] - * @property {string} [widget] - * @property {boolean} [optional] - */ - -/** - * Create HTML wrapper for a field item. - * Replacement for createItem() - * - * @since 1.12 - * @param {SemanticField} field - * @param {string} content - * @param {string} [inputId] - * @return {string} - */ -ns.createFieldMarkup = function (field, content, inputId) { - content = content || ''; - var markup = this.createLabel(field, '', inputId) + this.createDescription(field.description, inputId) + content; - - return this.wrapFieldMarkup(field, markup); -}; - -/** - * Create HTML wrapper for a boolean field item. - * - * @param {SemanticField} field - * @param {string} content - * @param {string} [inputId] - * - * @return {string} - */ -ns.createBooleanFieldMarkup = function (field, content, inputId) { - var markup = '' + - this.createDescription(field.description, inputId); - - return this.wrapFieldMarkup(field, markup); -}; - -/** - * Wraps a field with some metadata classes, and adds error field - * - * @param {SemanticField} field - * @param {string} markup - * - * @private - * @return {string} - */ -ns.wrapFieldMarkup = function (field, markup) { - // removes undefined and joins - var wrapperClasses = this.joinNonEmptyStrings(['field', 'field-name-' + field.name, field.type, ns.createImportance(field.importance), field.widget]); - - // wrap and return - return '
    ' + - markup + - '
    ' + - '
    '; -}; - -/** - * Joins an array of strings if they are defined and non empty - * - * @param {string[]} arr - * @param {string} [separator] Default is space - * @return {string} - */ -ns.joinNonEmptyStrings = function (arr, separator) { - separator = separator || ' '; - - return arr.filter(function (str) { - return str !== undefined && str.length > 0; - }).join(separator); -}; - -/** - * Create HTML for select options. - * - * @param {String} value - * @param {String} text - * @param {Boolean} selected - * @returns {String} - */ -ns.createOption = function (value, text, selected) { - return ''; -}; - -/** - * Create HTML for text input. - * - * @param {String} value - * @param {number} maxLength - * @param {String} placeholder - * @param {number} [id] - * @param {number} [describedby] - * @returns {String} - */ -ns.createText = function (value, maxLength, placeholder, id, describedby) { - var html = ''; - - return html; -}; - -ns.getNextFieldId = (function (counter) { - /** - * Generates a consistent and unique field ID for the given field. - * - * @param {Object} field - * @return {number} - */ - return function (field) { - return 'field-' + field.name.toLowerCase() + '-' + (counter++); - }; -})(-1); - -/** - * Helps generates a consistent description ID across fields. - * - * @param {string} id - * @return {string} - */ -ns.getDescriptionId = function (id) { - return id + '-description'; -}; - -/** - * Create a label to wrap content in. - * - * @param {SemanticField} field - * @param {String} [content] - * @param {String} [inputId] - * @returns {String} - */ -ns.createLabel = function (field, content, inputId) { - // New items can be added next to the label within the flex-wrapper - var html = ''; -}; - -/** - * Create a description - * @param {String} description - * @param {number} [inputId] Used to reference description from input - * @returns {string} - */ -ns.createDescription = function (description, inputId) { - var html = ''; - if (description !== undefined) { - html += '
    '; - } - return html; -}; - -/** - * Create an important description - * @param {Object} importantDescription - * @returns {String} - */ -ns.createImportantDescription = function (importantDescription) { - var html = ''; - - if (importantDescription !== undefined) { - html += '
    ' + - '
    ' + - '
    ' + - '
    ' + - '' + - ns.t('core', 'hide') + - '' + - '
    ' + - '' + - '' + - '' + - ns.t('core', 'importantInstructions') + - ''; - - if (importantDescription.description !== undefined) { - html += '
    ' + - importantDescription.description + - '
    '; - } - - if (importantDescription.example !== undefined) { - html += '
    ' + - '
    ' + - '' + - ns.t('core', 'example') + - ':' + - '
    ' + - '
    ' + - '' + - importantDescription.example + - '' + - '
    ' + - '
    '; - } - - html += '
    ' + - '' + - ns.t('core', 'showImportantInstructions') + - ''; - } - - return html; -}; - -/** - * Bind events to important description - * @param {Object} widget - * @param {String} fieldName - * @param {Object} parent - */ -ns.bindImportantDescriptionEvents = function (widget, fieldName, parent) { - var context; - - if (!widget.field.important) { - return; - } - - // Generate a context string for using as referance in ex. localStorage. - var librarySelector = ns.findLibraryAncestor(parent); - if (librarySelector.currentLibrary !== undefined) { - var lib = librarySelector.currentLibrary.split(' ')[0]; - context = (lib + '-' + fieldName).replace(/\.|_/g,'-') + '-important-description-open'; - } - - // Set first occurance to visible - ns.storage.get(context, function (value) { - if (value === undefined || value === true) { - widget.$item.addClass('important-description-visible'); - } - }); - - widget.$item.addClass('has-important-description'); - - // Bind events to toggle button and update aria-pressed - widget.$item.find('.important-description-show') - .click(function () { - widget.$item.addClass('important-description-visible'); - ns.storage.set(context, true); - }) - .keydown(function (event) { - if (event.which == 13 || event.which == 32) { - ns.$(this).trigger('click'); - event.preventDefault(); - } - }); - - // Bind events to close button and update aria-pressed of toggle button - widget.$item.find('.important-description-close') - .click(function () { - widget.$item.removeClass('important-description-visible'); - ns.storage.set(context, false); - }) - .keydown(function (event) { - if (event.which == 13 || event.which == 32) { - ns.$(this).trigger('click'); - event.preventDefault(); - } - }); -}; - -/** - * Generate markup for the copy and paste buttons. - * - * @returns {string} HTML - */ -ns.createCopyPasteButtons = function () { - return '
    ' + - '' + - '' + - '
    '; -}; - -/** - * Confirm replace if there is content selected - * - * @param {string} library Current selected library - * @param {number} top Offset - * @param {function} next Next callback - */ -ns.confirmReplace = function (library, top, next) { - if (library) { - // Confirm changing library - var confirmReplace = new H5P.ConfirmationDialog({ - headerText: H5PEditor.t('core', 'pasteContent'), - dialogText: H5PEditor.t('core', 'confirmPasteContent'), - confirmText: H5PEditor.t('core', 'confirmPasteButtonText') - }).appendTo(document.body); - confirmReplace.on('confirmed', next); - confirmReplace.show(top); - } - else { - // No need to confirm - next(); - } -}; - -/** - * Check if any errors has been set. - * - * @param {jQuery} $errors - * @param {jQuery} $input - * @param {String} value - * @returns {mixed} - */ -ns.checkErrors = function ($errors, $input, value) { - if ($errors.children().length) { - $input.keyup(function (event) { - if (event.keyCode === 9) { // TAB - return; - } - $errors.html(''); - $input.removeClass('error'); - $input.unbind('keyup'); - }); - - return false; - } - return value; -}; - -/** - * @param {object} library - * with machineName, majorVersion and minorVersion params - * @returns {string} - * Concatinated version of the library - */ -ns.libraryToString = function (library) { - return library.name + ' ' + library.majorVersion + '.' + library.minorVersion; -}; - -/** - * TODO: Remove from here, and use from H5P instead(move this to the h5p.js...) - * - * @param {string} library - * library in the format machineName majorVersion.minorVersion - * @returns - * library as an object with machineName, majorVersion and minorVersion properties - * return false if the library parameter is invalid - */ -ns.libraryFromString = function (library) { - var regExp = /(.+)\s(\d+)\.(\d+)$/g; - var res = regExp.exec(library); - if (res !== null) { - return { - 'machineName': res[1], - 'majorVersion': res[2], - 'minorVersion': res[3] - }; - } - else { - H5P.error('Invalid überName'); - return false; - } -}; - -/** - * Helper function for detecting field widget. - * - * @param {Object} field - * @returns {String} Widget name - */ -ns.getWidgetName = function (field) { - return (field.widget === undefined ? field.type : field.widget); -}; - -/** - * Mimics how php's htmlspecialchars works (the way we uses it) - */ -ns.htmlspecialchars = function (string) { - return string.toString().replace(//g, '>').replace(/'/g, ''').replace(/"/g, '"'); -}; - -/** - * Makes it easier to add consistent buttons across the editor widget. - * - * @param {string} id Typical CSS class format - * @param {string} title Human readable format - * @param {function} handler Action handler when triggered - * @param {boolean} [displayTitle=false] Show button with text - * @return {H5P.jQuery} - */ -ns.createButton = function (id, title, handler, displayTitle) { - var options = { - class: 'h5peditor-button ' + (displayTitle ? 'h5peditor-button-textual ' : '') + id, - role: 'button', - tabIndex: 0, - 'aria-disabled': 'false', - on: { - click: function () { - handler.call(this); - }, - keydown: function (event) { - switch (event.which) { - case 13: // Enter - case 32: // Space - handler.call(this); - event.preventDefault(); - } - } - } - }; - - // Determine if we're a icon only button or have a textual label - options[displayTitle ? 'html' : 'aria-label'] = title; - - return ns.$('
    ', options); -}; - -/** - * Check if the current library is entitled for the metadata button. True by default. - * - * It will probably be okay to remove this check at some point in time when - * the majority of content types and plugins have been updated to a version - * that supports the metadata system. - * - * @param {string} library - Current library. - * @return {boolean} True, if form should have the metadata button. - */ -ns.enableMetadata = function (library) { - - if (!library || typeof library !== 'string') { - return false; - } - - library = H5P.libraryFromString(library); - if (!library) { - return false; - } - - // This list holds all old libraries (/older versions implicitly) that need an update for metadata - const blackList = [ - // Should never have metadata because it does not make sense - 'H5P.IVHotspot 1.2', - 'H5P.Link 1.3', - 'H5P.TwitterUserFeed 1.0', - 'H5P.GoToQuestion 1.3', - 'H5P.Nil 1.0', - - // Copyright information moved to metadata - 'H5P.Audio 1.2', - 'H5P.Video 1.4', - 'H5P.Image 1.0', - - // Title moved to metadata - 'H5P.DocumentExportPage 1.3', - 'H5P.ExportableTextArea 1.2', - 'H5P.GoalsAssessmentPage 1.3', - 'H5P.GoalsPage 1.4', - 'H5P.StandardPage 1.3', - 'H5P.DragQuestion 1.12', - 'H5P.ImageHotspotQuestion 1.7', - - // Custom editor changed - 'H5P.CoursePresentation 1.19', - 'H5P.InteractiveVideo 1.19' - ]; - - let block = blackList.filter(function (item) { - // + ' ' makes sure to avoid partial matches - return item.indexOf(library.machineName + ' ') !== -1; - }); - if (block.length === 0) { - return true; - } - - block = H5P.libraryFromString(block[0]); - if (library.majorVersion > block.majorVersion || library.majorVersion === block.majorVersion && library.minorVersion > block.minorVersion) { - return true; - } - - return false; -}; - -// Backwards compatibilty -ns.attachToastTo = H5P.attachToastTo; - -/** - * Check if clipboard can be pasted. - * - * @param {Object} clipboard Clipboard data. - * @param {Object} libs Libraries to compare against. - * @return {boolean} True, if content can be pasted. - */ -ns.canPaste = function (clipboard, libs) { - return (this.canPastePlus(clipboard, libs)).canPaste; -}; - -/** - * Check if clipboard can be pasted and give reason if not. - * - * @param {Object} clipboard Clipboard data. - * @param {Object} libs Libraries to compare against. - * @return {Object} Results. {canPaste: boolean, reason: string, description: string}. - */ -ns.canPastePlus = function (clipboard, libs) { - // Clipboard is empty - if (!clipboard || !clipboard.generic) { - return { - canPaste: false, - reason: 'pasteNoContent', - description: ns.t('core', 'pasteNoContent') - }; - } - - // No libraries to compare to - if (libs === undefined) { - return { - canPaste: false, - reason: 'pasteError', - description: ns.t('core', 'pasteError') - }; - } - - // Translate Hub format to common library format - if (libs.libraries !== undefined) { - libs = libs.libraries; - libs.forEach(function (lib) { - lib.name = lib.machineName; - lib.majorVersion = lib.localMajorVersion; - lib.minorVersion = lib.localMinorVersion; - }); - } - - // Check if clipboard library type is available - const machineNameClip = clipboard.generic.library.split(' ')[0]; - let candidates = libs.filter(function (library) { - return library.name === machineNameClip; - }); - if (candidates.length === 0) { - return { - canPaste: false, - reason: 'pasteContentNotSupported', - description: ns.t('core', 'pasteContentNotSupported') - }; - } - - // Check if clipboard library version is available - const versionClip = clipboard.generic.library.split(' ')[1]; - for (let i = 0; i < candidates.length; i++) { - if (candidates[i].majorVersion + '.' + candidates[i].minorVersion === versionClip) { - if (candidates[i].restricted !== true) { - return { - canPaste: true - }; - } - else { - return { - canPaste: false, - reason: 'pasteContentRestricted', - description: ns.t('core', 'pasteContentRestricted') - }; - } - } - } - - // Sort remaining candidates by version number - candidates = candidates - .map(function (candidate) { - return '' + candidate.majorVersion + '.' + candidate.minorVersion; - }) - .map(function (candidate) { - return candidate.replace(/\d+/g, function (d) { - return +d + 1000; - }); - }) - .sort() - .map(function (candidate) { - return candidate.replace(/\d+/g, function (d) { - return +d - 1000; - }); - }); - - // Clipboard library is newer than latest available local library - const candidateMax = candidates.slice(-1)[0]; - if (+candidateMax.split('.')[0] < +versionClip.split('.')[0] || - (+candidateMax.split('.')[0] === +versionClip.split('.')[0] && - +candidateMax.split('.')[1] < +versionClip.split('.')[1])) { - return { - canPaste: false, - reason: 'pasteTooNew', - description: ns.t('core', 'pasteTooNew', { - ':clip': versionClip, - ':local': candidateMax - }) - }; - } - - // Clipboard library is older than latest available local library - const candidateMin = candidates.slice(0, 1)[0]; - if (+candidateMin.split('.')[0] > +versionClip.split('.')[0] || - (+candidateMin.split('.')[0] === +versionClip.split('.')[0] && - +candidateMin.split('.')[1] > +versionClip.split('.')[1])) { - return { - canPaste: false, - reason: 'pasteTooOld', - description: ns.t('core', 'pasteTooOld', { - ':clip': versionClip, - ':local': candidateMin - }) - }; - } - - return { - canPaste: false, - reason: 'pasteError', - description: ns.t('core', 'pasteError') - }; -}; - -// Factory for creating storage instance -ns.storage = (function () { - var instance = { - get: function (key, next) { - var value; - - // Get value from browser storage - if (window.localStorage !== undefined) { - value = !!window.localStorage.getItem(key); - } - - // Try to get a better value from user data storage - try { - H5P.getUserData(0, key, function (err, result) { - if (!err) { - value = result; - } - next(value); - }); - } - catch (err) { - next(value); - } - }, - set: function (key, value) { - - // Store in browser - if (window.localStorage !== undefined) { - window.localStorage.setItem(key, value); - } - - // Try to store in user data storage - try { - H5P.setUserData(0, key, value); - } - catch (err) { /*Intentionally left empty*/ } - } - }; - return instance; -})(); - -/** - * Small helper class for library data. - * - * @class - * @param {string} nameVersionString - */ -ns.ContentType = function ContentType(nameVersionString) { - const libraryNameSplit = nameVersionString.split(' '); - const libraryVersionSplit = libraryNameSplit[1].split('.'); - - this.machineName = libraryNameSplit[0]; - this.majorVersion = libraryVersionSplit[0]; - this.minorVersion = libraryVersionSplit[1]; -}; - -/** - * Look for the best possible upgrade for the given library - * - * @param {ns.ContentType} library - * @param {Array} libraries Where to look - */ -ns.ContentType.getPossibleUpgrade = function (library, libraries) { - let possibleUpgrade; - - for (let i = 0; i < libraries.length; i++) { - const candiate = libraries[i]; - if (candiate.installed !== false && ns.ContentType.hasSameName(candiate, library) && ns.ContentType.isHigherVersion(candiate, library)) { - - // Check if the upgrade is better than the previous upgrade we found - if (!possibleUpgrade || ns.ContentType.isHigherVersion(candiate, possibleUpgrade)) { - possibleUpgrade = candiate; - } - } - } - - return possibleUpgrade; -}; - -/** - * Check if candiate is a higher version than original. - * - * @param {Object} candiate Library object - * @param {Object} original Library object - * @returns {boolean} - */ -ns.ContentType.isHigherVersion = function (candiate, original) { - return (ns.ContentType.getMajorVersion(candiate) > ns.ContentType.getMajorVersion(original) || - (ns.ContentType.getMajorVersion(candiate) == ns.ContentType.getMajorVersion(original) && - ns.ContentType.getMinorVersion(candiate) > ns.ContentType.getMinorVersion(original))); -}; - -/** - * Check if candiate has same name as original. - * - * @param {Object} candiate Library object - * @param {Object} original Library object - * @returns {boolean} - */ -ns.ContentType.hasSameName = function (candiate, original) { - return (ns.ContentType.getName(candiate) === ns.ContentType.getName(original)); -}; - -/** - * Check if candiate has same name as original. - * - * @param {Object} candiate Library object - * @param {Object} original Library object - * @returns {string} - */ -ns.ContentType.getNameVersionString = function (library) { - return ns.ContentType.getName(library) + ' ' + ns.ContentType.getMajorVersion(library) + '.' + ns.ContentType.getMinorVersion(library); -}; - -/** - * Get the major version from a library object. - * - * @param {Object} library - * @returns {number} - */ -ns.ContentType.getMajorVersion = function (library) { - return parseInt((library.localMajorVersion !== undefined ? library.localMajorVersion : library.majorVersion)); -}; - -/** - * Get the minor version from a library object. - * - * @param {Object} library - * @returns {number} - */ -ns.ContentType.getMinorVersion = function (library) { - return parseInt((library.localMinorVersion !== undefined ? library.localMinorVersion : library.minorVersion)); -}; - -/** - * Get the name from a library object. - * - * @param {Object} library - * @returns {string} - */ -ns.ContentType.getName = function (library) { - return (library.machineName !== undefined ? library.machineName : library.name); -}; - - -ns.upgradeContent = (function () { - - /** - * A wrapper for loading library data for the content upgrade scripts. - * - * @param {string} name Library name - * @param {H5P.Version} version - * @param {Function} next Callback - */ - const loadLibrary = function (name, version, next) { - const library = name + ' ' + version.major + '.' + version.minor; - ns.loadLibrary(library, function () { - next(null, ns.libraryCache[library]); - }); - }; - - return function contentUpgrade(fromLibrary, toLibrary, parameters, done) { - ns.loadJs(H5PIntegration.libraryUrl + '/h5p-version.js' + H5PIntegration.pluginCacheBuster, function (err) { - ns.loadJs(H5PIntegration.libraryUrl + '/h5p-content-upgrade-process.js' + H5PIntegration.pluginCacheBuster, function (err) { - // TODO: Avoid stringify the parameters - new H5P.ContentUpgradeProcess(ns.ContentType.getName(fromLibrary), new H5P.Version(fromLibrary), new H5P.Version(toLibrary), JSON.stringify(parameters), 1, function (name, version, next) { - loadLibrary(name, version, function (err, library) { - if (library.upgradesScript) { - ns.loadJs(library.upgradesScript, function (err) { - if (err) { - err = 'Error loading upgrades ' + name + ' ' + version; - } - next(err, library); - }); - } - else { - next(null, library); - } - }); - - }, function (err, result) { - if (err) { - let header = 'Failed'; - let message = 'Could not upgrade content'; - switch (err.type) { - case 'errorTooHighVersion': - message += ': ' + ns.t('core', 'errorTooHighVersion', {'%used': err.used, '%supported': err.supported}); - break; - - case 'errorNotSupported': - message += ': ' + ns.t('core', 'errorNotSupported', {'%used': err.used}); - break; - - case 'errorParamsBroken': - message += ': ' + ns.t('core', 'errorParamsBroken'); - break; - - case 'libraryMissing': - message += ': ' + ns.t('core', 'libraryMissing', {'%lib': err.library}); - break; - - case 'scriptMissing': - message += ': ' + ns.t('core', 'scriptMissing', {'%lib': err.library}); - break; - } - - var confirmErrorDialog = new H5P.ConfirmationDialog({ - headerText: header, - dialogText: message, - confirmText: 'Continue' - }).appendTo(document.body); - confirmErrorDialog.show(); - } - done(err, result); - }); - }); - }); - }; -})(); - -// List of language code mappings used by the editor -ns.supportedLanguages = { - 'aa': 'Afar', - 'ab': 'Abkhazian (аҧсуа бызшәа)', - 'ae': 'Avestan', - 'af': 'Afrikaans', - 'ak': 'Akan', - 'am': 'Amharic (አማርኛ)', - 'ar': 'Arabic (العربية)', - 'as': 'Assamese', - 'ast': 'Asturian', - 'av': 'Avar', - 'ay': 'Aymara', - 'az': 'Azerbaijani (azərbaycan)', - 'ba': 'Bashkir', - 'be': 'Belarusian (Беларуская)', - 'bg': 'Bulgarian (Български)', - 'bh': 'Bihari', - 'bi': 'Bislama', - 'bm': 'Bambara (Bamanankan)', - 'bn': 'Bengali', - 'bo': 'Tibetan', - 'br': 'Breton', - 'bs': 'Bosnian (Bosanski)', - 'ca': 'Catalan (Català)', - 'ce': 'Chechen', - 'ch': 'Chamorro', - 'co': 'Corsican', - 'cr': 'Cree', - 'cs': 'Czech (Čeština)', - 'cu': 'Old Slavonic', - 'cv': 'Chuvash', - 'cy': 'Welsh (Cymraeg)', - 'da': 'Danish (Dansk)', - 'de': 'German (Deutsch)', - 'dv': 'Maldivian', - 'dz': 'Bhutani', - 'ee': 'Ewe (Ɛʋɛ)', - 'el': 'Greek (Ελληνικά)', - 'en': 'English', - 'en-gb': 'English, British', - 'eo': 'Esperanto', - 'es': 'Spanish (Español)', - 'es-mx': 'Spanish, Mexican', - 'et': 'Estonian (Eesti)', - 'eu': 'Basque (Euskera)', - 'fa': 'Persian (فارسی)', - 'ff': 'Fulah (Fulfulde)', - 'fi': 'Finnish (Suomi)', - 'fil': 'Filipino', - 'fj': 'Fiji', - 'fo': 'Faeroese', - 'fr': 'French (Français)', - 'fy': 'Frisian (Frysk)', - 'ga': 'Irish (Gaeilge)', - 'gd': 'Scots Gaelic', - 'gl': 'Galician (Galego)', - 'gn': 'Guarani', - 'gsw-berne': 'Swiss German', - 'gu': 'Gujarati', - 'gv': 'Manx', - 'ha': 'Hausa', - 'he': 'Hebrew (עברית)', - 'hi': 'Hindi (हिन्दी)', - 'ho': 'Hiri Motu', - 'hr': 'Croatian (Hrvatski)', - 'hsb': 'Upper Sorbian (hornjoserbšćina)', - 'ht': 'Haitian Creole', - 'hu': 'Hungarian (Magyar)', - 'hy': 'Armenian (Հայերեն)', - 'hz': 'Herero', - 'ia': 'Interlingua', - 'id': 'Indonesian (Bahasa Indonesia)', - 'ie': 'Interlingue', - 'ig': 'Igbo', - 'ik': 'Inupiak', - 'is': 'Icelandic (Íslenska)', - 'it': 'Italian (Italiano)', - 'iu': 'Inuktitut', - 'ja': 'Japanese (日本語)', - 'jv': 'Javanese', - 'ka': 'Georgian', - 'kg': 'Kongo', - 'ki': 'Kikuyu', - 'kj': 'Kwanyama', - 'kk': 'Kazakh (Қазақ)', - 'kl': 'Greenlandic', - 'km': 'Cambodian', - 'kn': 'Kannada (ಕನ್ನಡ)', - 'ko': 'Korean (한국어)', - 'kr': 'Kanuri', - 'ks': 'Kashmiri', - 'ku': 'Kurdish (Kurdî)', - 'kv': 'Komi', - 'kw': 'Cornish', - 'ky': 'Kyrgyz (Кыргызча)', - 'la': 'Latin (Latina)', - 'lb': 'Luxembourgish', - 'lg': 'Luganda', - 'ln': 'Lingala', - 'lo': 'Laothian', - 'lt': 'Lithuanian (Lietuvių)', - 'lv': 'Latvian (Latviešu)', - 'mg': 'Malagasy', - 'mh': 'Marshallese', - 'mi': 'Māori', - 'mk': 'Macedonian (Македонски)', - 'ml': 'Malayalam (മലയാളം)', - 'mn': 'Mongolian', - 'mo': 'Moldavian', - 'mr': 'Marathi', - 'ms': 'Malay (Bahasa Melayu)', - 'mt': 'Maltese (Malti)', - 'my': 'Burmese', - 'na': 'Nauru', - 'nd': 'North Ndebele', - 'ne': 'Nepali', - 'ng': 'Ndonga', - 'nl': 'Dutch (Nederlands)', - 'nb': 'Norwegian Bokmål (Bokmål)', - 'nn': 'Norwegian Nynorsk (Nynorsk)', - 'nr': 'South Ndebele', - 'nv': 'Navajo', - 'ny': 'Chichewa', - 'oc': 'Occitan', - 'om': 'Oromo', - 'or': 'Oriya', - 'os': 'Ossetian', - 'pa': 'Punjabi', - 'pi': 'Pali', - 'pl': 'Polish (Polski)', - 'ps': 'Pashto (پښتو)', - 'pt': 'Portuguese, International', - 'pt-pt': 'Portuguese, Portugal (Português)', - 'pt-br': 'Portuguese, Brazil (Português)', - 'qu': 'Quechua', - 'rm': 'Rhaeto-Romance', - 'rn': 'Kirundi', - 'ro': 'Romanian (Română)', - 'ru': 'Russian (Русский)', - 'rw': 'Kinyarwanda', - 'sa': 'Sanskrit', - 'sc': 'Sardinian', - 'sco': 'Scots', - 'sd': 'Sindhi', - 'se': 'Northern Sami', - 'sg': 'Sango', - 'sh': 'Serbo-Croatian', - 'si': 'Sinhala (සිංහල)', - 'sk': 'Slovak (Slovenčina)', - 'sl': 'Slovenian (Slovenščina)', - 'sm': 'Samoan', - 'sma': 'Sámi (Southern)', - 'sme': 'Sámi (Northern)', - 'smj': 'Sámi (Lule)', - 'sn': 'Shona', - 'so': 'Somali', - 'sq': 'Albanian (Shqip)', - 'sr': 'Serbian (Српски)', - 'ss': 'Siswati', - 'st': 'Sesotho', - 'su': 'Sudanese', - 'sv': 'Swedish (Svenska)', - 'sw': 'Swahili (Kiswahili)', - 'ta': 'Tamil (தமிழ்)', - 'te': 'Telugu (తెలుగు)', - 'tg': 'Tajik', - 'th': 'Thai (ภาษาไทย)', - 'ti': 'Tigrinya', - 'tk': 'Turkmen', - 'tl': 'Tagalog', - 'tn': 'Setswana', - 'to': 'Tonga', - 'tr': 'Turkish (Türkçe)', - 'ts': 'Tsonga', - 'tt': 'Tatar (Tatarça)', - 'tw': 'Twi', - 'ty': 'Tahitian', - 'ug': 'Uyghur', - 'uk': 'Ukrainian (Українська)', - 'ur': 'Urdu (اردو)', - 'uz': "Uzbek (o'zbek)", - 've': 'Venda', - 'vi': 'Vietnamese (Tiếng Việt)', - 'wo': 'Wolof', - 'xh': 'Xhosa (isiXhosa)', - 'xx-lolspeak': 'Lolspeak)', - 'yi': 'Yiddish', - 'yo': 'Yoruba (Yorùbá)', - 'za': 'Zhuang', - 'zh': 'Chinese', - 'zh-hans': 'Chinese, Simplified (简体中文)', - 'zh-hant': 'Chinese, Traditional (繁體中文)', - 'zh-tw': 'Chinese, Taiwan, Traditional', - 'zu': 'Zulu (isiZulu)' -}; diff --git a/h5p/h5plib/v126/joubel/editor/styles/config.rb b/h5p/h5plib/v126/joubel/editor/styles/config.rb deleted file mode 100644 index 6e00b5a5606cf..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/config.rb +++ /dev/null @@ -1,24 +0,0 @@ -# Require any additional compass plugins here. - -# Set this to the root of your project when deployed: -http_path = "/" -css_dir = "css" -sass_dir = "scss" -images_dir = "../images" -javascripts_dir = "../scripts" - -# You can select your preferred output style here (can be overridden via the command line): -# output_style = :expanded or :nested or :compact or :compressed -output_style = :compressed - -# To enable relative paths to assets via compass helper functions. Uncomment: -# relative_assets = true - -# To disable debugging comments that display the original location of your selectors. Uncomment: -line_comments = false - -# If you prefer the indented syntax, you might want to regenerate this -# project again passing --syntax sass, or you can uncomment this: -# preferred_syntax = :sass -# and then run: -# sass-convert -R --from scss --to sass sass scss && rm -rf sass && mv scss sass diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/application.css b/h5p/h5plib/v126/joubel/editor/styles/css/application.css deleted file mode 100644 index 9ea64891fd2b3..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/application.css +++ /dev/null @@ -1 +0,0 @@ -ul.list-unstyled{list-style:none;padding-left:0;margin:0}.field{margin:20px 0;font-size:16px;padding:0}.field:first-child{margin-top:0}.field:last-child{margin-bottom:0}.tree>.field.group:last-child{margin-bottom:0}.fields>.field.group{margin:10px 0}.fields>.field.group:first-child{margin-top:0}.fields>.field.group:last-child{margin-bottom:0}.field.boolean .h5peditor-label{display:inline}.field .h5p-editor-image-buttons{float:left;clear:both}.field .library{border:0}.field.importance-high>.h5peditor-label-wrapper>.h5peditor-label{font-size:18px;color:#356593}.field .h5p-dialog-anchor{position:relative;margin:-16px}.common-fields-library-wrapper{margin:20px 0;border:none;margin:0;padding:0}.common-fields-library-wrapper:first-child{margin-top:0}.common-fields-library-wrapper:last-child{margin-bottom:0}.common-fields-library-wrapper .common-field-legend{display:none}.group>.title,.h5p-li>.list-item-title-bar,.common>.h5peditor-label{visibility:inherit;cursor:pointer}.group.importance-high>.title,.h5p-li>.list-item-title-bar.importance-high{background:#2579C6;border:1px solid #1f67a8;height:42px}.group.importance-high>.title,.h5p-li>.list-item-title-bar.importance-high>.h5peditor-label,.h5p-li>.list-item-title-bar.importance-high>.title{font-size:16px;font-weight:600;line-height:42px;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.group>.title,.h5p-li>.list-item-title-bar{background:#747275;border:1px solid #636164;height:38px}.group>.title,.h5p-li>.list-item-title-bar>.h5peditor-label,.h5p-li>.list-item-title-bar.importance-medium>.h5peditor-label{font-size:16px;font-weight:600;color:#fff;line-height:38px;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.group.importance-low>.title,.h5p-li>.list-item-title-bar.importance-low,.common>.h5peditor-label{background:#f5f5f5;height:38px;border:1px solid #d0d0d1}.group.importance-low>.title,.h5p-li>.list-item-title-bar.importance-low>.h5peditor-label,.h5p-li>.list-item-title-bar.importance-low>.title,.common>.h5peditor-label{font-size:16px;font-weight:400;color:#212121;line-height:38px}.group{border:none}.group>.title{padding:0 10px;outline:none}.group>.content{position:relative;display:none;margin:0;padding:20px;border:1px solid #d0d0d1;border-top:none;background:#fff}.group.importance-high>.title{color:#fff}.group.importance-low>.title{color:#212121}.group.importance-low>.title:focus:before{outline:1px dashed}.common .content,.content .content{background-color:#fcfcfc}.common .content .content,.content .content .content{background-color:#fff}.common .content .content .content,.content .content .content .content{background-color:#fcfcfc}.common .content .content .content .content,.content .content .content .content .content{background-color:#fff}.common .content .content .content .content .content,.content .content .content .content .content .content{background-color:#fcfcfc}.common .content .content .content .content .content .content,.content .content .content .content .content .content .content{background-color:#fff}.h5p-li>.list-item-title-bar{color:#fff}.h5p-li>.list-item-title-bar>.h5peditor-label{overflow:hidden;margin:0 54px 0 0;padding:0 10px;white-space:nowrap;line-height:38px;outline:none;text-overflow:ellipsis}.h5p-li>.list-item-title-bar>.list-actions{height:100%;float:right;position:relative}.h5p-li>.list-item-title-bar .remove{cursor:pointer;width:1.25em;height:38px;font-size:1.75em;display:inline-block}.h5p-li>.list-item-title-bar .remove:hover{opacity:1;text-decoration:none}.h5p-li>.list-item-title-bar .remove:after{font-family:"H5P";content:"\e890";opacity:0.7;display:inline-block;line-height:38px}.h5p-li>.list-item-title-bar .remove:hover:after{opacity:1}.h5p-li>.list-item-title-bar .order-group{text-align:center;float:right;background:#636164;font-size:16px;cursor:pointer}.h5p-li>.list-item-title-bar .order-up,.h5p-li>.list-item-title-bar .order-down{width:19px;height:19px;line-height:19px}.h5p-li>.list-item-title-bar .order-up:hover,.h5p-li>.list-item-title-bar .order-down:hover{background:#636164}.h5p-li>.list-item-title-bar .order-up:after,.h5p-li>.list-item-title-bar .order-down:after{font-family:"H5P";content:"\e58e";display:inline-block}.h5p-li>.list-item-title-bar .order-down:after{content:"\e58f"}.h5p-li>.list-item-title-bar.importance-high>.title{border:none;margin:0 74px 0 0}.h5p-li>.list-item-title-bar.importance-high .remove{height:42px;line-height:42px;font-size:40px}.h5p-li>.list-item-title-bar.importance-high .remove:after{line-height:42px;opacity:0.7}.h5p-li>.list-item-title-bar.importance-high .remove:hover:after{opacity:1}.h5p-li>.list-item-title-bar.importance-high .order-group{background:#1f67a8;font-size:18px}.h5p-li>.list-item-title-bar.importance-high .order-up,.h5p-li>.list-item-title-bar.importance-high .order-down{width:21px;height:21px;line-height:21px}.h5p-li>.list-item-title-bar.importance-high .order-up:hover,.h5p-li>.list-item-title-bar.importance-high .order-down:hover{background:#1f67a8}.h5p-li>.list-item-title-bar.importance-low>.title{border:none;margin:0 54px 0 0}.h5p-li>.list-item-title-bar.importance-low>.title:before{color:#212121}.h5p-li>.list-item-title-bar.importance-low>.title:focus:before{outline:1px dashed #212121}.h5p-li>.list-item-title-bar.importance-low .remove{height:38px;line-height:38px;font-size:30px}.h5p-li>.list-item-title-bar.importance-low .remove:after{line-height:38px}.h5p-li>.list-item-title-bar.importance-low .remove:after{color:#212121;opacity:0.7}.h5p-li>.list-item-title-bar.importance-low .remove:hover:after{opacity:1}.h5p-li>.list-item-title-bar.importance-low .order-up,.h5p-li>.list-item-title-bar.importance-low .order-down{width:19px;height:19px;background:#d0d0d1;font-size:16px;line-height:19px}.h5p-li>.list-item-title-bar.importance-low .order-up:hover,.h5p-li>.list-item-title-bar.importance-low .order-down:hover{background:#deddde}.h5p-li>.list-item-title-bar.importance-low .order-up:after,.h5p-li>.list-item-title-bar.importance-low .order-down:after{color:#212121}.group.expanded>.content,.listgroup.expanded>.content{display:block}.listgroup>.list-item-title-bar>.h5peditor-label{cursor:pointer}.list-item-title-bar>.title:before,.group>.title:before{content:"\e566";font-family:"H5P";margin-right:5px}.list-item-title-bar>.title:focus:before,.group>.title:focus:before{outline:1px dashed #fff}.listgroup.expanded>.list-item-title-bar>.h5peditor-label:before,.expanded>.title:before{content:"\e565"}.listgroup>.group.field{margin:0;padding:0;min-width:0}.content{display:block;margin:0;padding:20px;border:1px solid #d0d0d1;border-top:none;background:#fff}.common{margin-top:20px}.common>.h5peditor-label{margin:0;padding:0 10px;cursor:pointer;font-size:1em}.common>.h5peditor-label>.icon:before{content:"\e565";font-family:"H5P";margin-right:5px}.common>.h5peditor-label:hover>.icon{opacity:1}.common>.h5peditor-label:focus{outline:none}.common>.h5peditor-label:focus>.icon:before{outline:1px dashed}.common>.fields{min-height:2em;padding:20px;border:1px solid #d0d0d1;border-top:none;background:#fff;position:relative}.common>.fields>.desc{margin:0;font-size:0.875em;color:#666;float:left}.common>.fields p:first-child{margin-bottom:20px}.common.collapsed>.h5peditor-label>.icon:before{content:"\e566"}.common.collapsed>.fields{display:none}.common.hidden{display:none}.h5peditor-button[aria-label]:before{content:attr(aria-label);visibility:hidden;position:absolute;top:115%;right:-10%;z-index:2;padding:0.25em 0.75em;background:#212121;color:#fff;white-space:nowrap;font-size:14px;line-height:1.5;box-shadow:0 0 0.5em #858585}.h5peditor-button[aria-label]:hover:before{visibility:visible}.h5peditor-button[aria-label][aria-disabled="true"]:before{display:none}.h5peditor .h5p-editing-image-button{background:linear-gradient(to bottom, #fff 0, #f2f2f2 100%);font-size:14px;color:#212121;line-height:28px;padding-right:20px;margin-right:0.5em;height:30px;padding-left:0;font-weight:normal}.h5peditor .h5p-editing-image-button:hover:not([disabled]){background:linear-gradient(to bottom, #fff 0, #d0d0d1 100%);border-color:#999}.h5peditor .h5p-editing-image-button:before{font-family:"H5P";content:"\e900";color:#666;padding-right:0.3em;padding-left:0.45em;vertical-align:middle;font-size:1.5em;line-height:0.9}.h5peditor .h5p-editing-image-button.loading:before{content:"\e901"}.h5peditor .h5p-editing-image-button.hidden,.h5peditor .h5p-copyright-button.hidden{display:none !important}.h5peditor .ui-dialog .h5p-editing-image-button,.h5peditor .ui-dialog .h5p-copyright-button{padding-left:0.5em}body.h5p-editor-image-popup{position:relative}.h5p-editing-image-popup-background{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.8);padding:3em 1em;box-sizing:border-box;z-index:102}#darkroom-icons{top:0;left:0}.h5p-editing-image-popup-background.hidden{display:none}.h5p-editing-image-popup{display:inline-block;position:relative;top:0;left:50%;height:auto;width:100%;max-height:100%;-webkit-transform:translateX(-50%);-ms-transform:translateX(-50%);transform:translateX(-50%);background:#fff;-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.h5p-editing-image-header{padding:1em 0.75em}.h5p-editing-image{max-height:100%;max-width:100%}.h5p-editing-image.hidden{display:none}.h5p-editing-image-popup .darkroom-toolbar{position:relative;-webkit-border-radius:0;-moz-border-radius:0;border-radius:0;top:0}.h5p-editing-image-popup .darkroom-toolbar:before{content:initial}.h5p-editing-image-popup .darkroom-image-container{padding:32px 2em}.h5p-editing-image-popup .darkroom-image-container .canvas-container{box-shadow:0 0 8px 3px rgba(0,0,0,0.6);margin:auto}.h5p-editing-image-header-title{display:inline-block;font-size:1.25em}.h5p-editing-image-header-buttons{display:inline-block;float:right}.h5p-editing-image-loading{padding:1em}.h5p-editing-image-loading.hidden{display:none}.h5p-editing-image-editing-container{background:#666}.h5p-editing-image-header-buttons button{padding:0.5em 1.75em;margin:0 0 0 1em;border:1px solid #ccc;border-radius:0.25em;color:#333;cursor:pointer;background:#f2f2f2;background:-webkit-linear-gradient(top, #fff 0, #f2f2f2 100%);background:-ms-linear-gradient(top, #fff 0, #f2f2f2 100%)}.h5p-editing-image-header-buttons button:hover{background:#ededed}.h5p-editing-image-header-buttons .h5p-done{color:#fff;border-color:#20588F;background:#3673B5;background:-webkit-linear-gradient(top, #5A94D3 0, #3673B5 100%);background:-ms-linear-gradient(top, #5A94D3 0, #3673B5 100%)}.h5p-editing-image-header-buttons .h5p-done:hover{background:#3275bc;background:-webkit-linear-gradient(top, #3275bc 0, #285585 100%);background:-ms-linear-gradient(top, #3275bc 0, #285585 100%)}.h5p-editing-image-header-buttons .h5p-remove{background:none;border:none;color:#a00;padding-right:0;padding-left:0;border-radius:0}.h5p-editing-image-header-buttons .h5p-remove:hover{background:none;color:#e40000}.h5p-vtab-wrapper>.h5p-vtab-forms .h5p-remove:after{color:#d0d0d1}html,body{margin:0;padding:0;color:#212121;font-family:"Open Sans",sans-serif;max-width:960px;position:relative}a{text-decoration:none}.h5peditor{font-size:16px}.h5peditor .h5p-more-libraries{font-size:0.875em;margin-top:0.4em}.h5peditor .h5peditor-single>.field.library{border:0;padding:0}.h5peditor .cke_dialog_background_cover{width:100% !important;height:100% !important}.h5peditor .errors p,.h5peditor .h5p-errors{color:#da0001;white-space:normal}.h5peditor textarea{resize:vertical}.h5peditor .h5p-editor-flex-wrapper{display:flex;flex-direction:row}.h5peditor .h5peditor-label-wrapper{display:flex;flex-direction:row}.h5peditor .has-important-description .h5peditor-field-description{display:none}.h5peditor .has-important-description .h5peditor-label-wrapper{float:left}.h5peditor .important-description-visible .h5peditor-field-important-description{display:block}.h5peditor .important-description-visible .h5peditor-label-wrapper{float:none}.h5peditor .important-description-visible .important-description-show{display:none}.h5peditor textarea,.h5peditor .h5peditor-text,.h5peditor .ckeditor{-moz-box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);-webkit-box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);box-sizing:border-box;margin:0;padding:10px;min-height:40px;border:1px solid #d0d0d1;background:#fff;outline:none;font-size:16px;word-wrap:break-word}.h5peditor textarea.error,.h5peditor .h5peditor-text.error,.h5peditor .ckeditor.error{border-color:red}.h5peditor .h5peditor-text,.h5peditor textarea{width:100%;box-sizing:border-box}.h5peditor .h5peditor-text.error,.h5peditor textarea.error{border-color:red}.h5peditor textarea:focus::-webkit-input-placeholder,.h5peditor input:focus::-webkit-input-placeholder{color:#fff}.h5peditor textarea:focus:-moz-placeholder,.h5peditor input:focus:-moz-placeholder{color:#fff}.h5peditor textarea:focus::-moz-placeholder,.h5peditor input:focus::-moz-placeholder{color:#fff}.h5peditor textarea:focus:-ms-input-placeholder,.h5peditor input:focus:-ms-input-placeholder{color:#fff}.h5peditor textarea:focus.h5peditor-ckeditor-placeholder,.h5peditor input:focus.h5peditor-ckeditor-placeholder{color:#fff}.h5peditor ::-webkit-input-placeholder{color:#858585}.h5peditor :-moz-placeholder{color:#858585}.h5peditor ::-moz-placeholder{color:#858585}.h5peditor :-ms-input-placeholder{color:#858585}.h5peditor .h5peditor-ckeditor-placeholder{color:#858585}.h5peditor>select,.h5peditor .h5peditor-language-switcher select,.h5peditor h5peditor-select,.h5peditor .field>select{padding:10px 30px 10px 8px;font-family:"Open Sans",sans-serif;font-size:16px;border:1px solid #d0d0d1;background:#fff url('') no-repeat;background-position:calc(100% - 10px);-moz-box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);-webkit-box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);box-shadow:inset 0px 0px 5px rgba(0,0,0,0.12);-moz-appearance:none;-webkit-appearance:none}.h5peditor>select:disabled,.h5peditor .h5peditor-language-switcher select:disabled,.h5peditor h5peditor-select:disabled,.h5peditor .field>select:disabled{background-color:#efefef}.h5peditor>select{margin-bottom:13px;margin-right:20px}.h5peditor select::-ms-expand{display:none}.h5peditor a:focus{outline:none}.h5peditor textarea:focus,.h5peditor .h5peditor-text:focus{outline:none;background-color:#fff;border-color:#53a0ff}.h5peditor .h5p-ul{padding:0;margin:0;list-style:none}.h5peditor .h5p-ul .h5p-li{margin:10px 0;padding:0;list-style:none}.h5peditor .h5p-ul .h5p-li:first-child{margin-top:0}.h5peditor .h5p-ul .h5p-li:last-child{margin-bottom:0}.h5peditor .h5p-ul .h5p-li.placeholder{box-sizing:border-box;background:#e8f2fa;border:dashed 2px #2782d1}.h5peditor .h5p-ul .h5p-li:hover{text-decoration:none}.h5peditor .h5p-ul .h5p-li:nth-child(2).moving{margin-top:0}.h5peditor .h5p-ul .h5p-li:nth-last-child(2).placeholder{margin-bottom:0}.h5peditor .dimensions input,.h5peditor .coordinates input,.h5peditor .number input{width:75px}.h5peditor .number input[type="range"]{width:300px;float:left;margin:7px 8px 0 0}.h5peditor .h5p-errors{clear:both}.h5peditor .h5p-add-file{float:left;position:relative;background:transparent;border:2px dashed #dddddd;color:#dddddd;margin:0.5em;width:6em;height:4.5em;cursor:pointer;outline:none;box-sizing:border-box;-moz-box-sizing:border-box}.h5peditor .h5p-add-file:focus,.h5peditor .h5p-add-file:hover{color:#999;border-color:#999}.h5peditor .h5p-add-file.hidden{visibility:hidden}.h5peditor .h5p-add-file:after{position:absolute;content:"+";font-size:2em;line-height:2.2em;width:100%;height:100%;text-align:center}.h5peditor .h5p-add-dialog{position:absolute;z-index:1;visibility:hidden;opacity:0;background:#fff;left:1em;right:1em;top:1em;border:1px solid #cdcdcd;box-sizing:border-box;-moz-box-sizing:border-box;-moz-box-shadow:0 0 8px #666;-webkit-box-shadow:0 0 8px #666;box-shadow:0 0 8px #666;-moz-transition:visibility 0s .2s,opacity .2s;-o-transition:visibility 0s .2s,opacity .2s;-webkit-transition:visibility 0s,opacity .2s;-webkit-transition-delay:.2s,0s;transition:visibility 0s .2s,opacity .2s}.h5peditor .h5p-add-dialog.h5p-open{visibility:visible;opacity:1;-moz-transition:visibility 0s 0s,opacity .2s;-o-transition:visibility 0s 0s,opacity .2s;-webkit-transition:visibility 0s,opacity .2s;-webkit-transition-delay:0s,0s;transition:visibility 0s 0s,opacity .2s}.h5peditor .h5p-add-dialog .h5p-add-dialog-table{overflow:hidden}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tablist{overflow:hidden;margin:1em 1em 0 1em}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tab{cursor:pointer;float:left;padding:0.5em 0.75em;border-top:1px solid #ccc;border-right:1px solid #ccc;font-weight:bold;font-size:15px;color:#666;background:#f5f5f5}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tab:first-child{border-left:1px solid #ccc}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tab.selected{color:#333;background:#fff}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tabpanel{border:1px solid #ccc;margin:0 1em 1em 1em;overflow:hidden;padding:1em}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tabpanel h3{margin:0 0 1em 0}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .av-tabpanel .h5p-file-drop-upload{width:90px;margin:0}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-dialog-box{float:left;padding:0;width:60%;margin-right:5%;margin-left:4%;text-align:left;margin-top:30px}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-dialog-box:first-child{width:22%;margin-left:3%;margin-right:1%}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-dialog-box:first-child h3{text-align:center}.h5peditor .h5p-add-dialog .h5p-add-dialog-table h3{color:#666;font-size:1em;margin:1.2em 0;line-height:1.1em}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5peditor-field-description{color:#bbb}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-url-wrapper{background-color:#f5f5f5;border:1px solid #bbb;padding:15px 15px 15px 60px;position:relative}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-url-wrapper:before{font-family:'H5P';position:absolute;left:0;top:0.083em;font-size:3.8em;line-height:1}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-url-wrapper.video:before{content:"\e904";color:#dd0505}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-url-wrapper.audio:before{content:"\e913";color:#747275}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-url-wrapper input{border-color:#bbb;text-align:left;padding-left:20px}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload{position:relative;width:70%;height:90px;margin:0 15%;float:left;background-color:#f5f5f5;text-align:center;cursor:pointer;border:1px solid #bbb}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload .h5p-file-drop-upload-inner{border:2px dashed #bbb;background-color:white;position:absolute;top:3px;bottom:3px;left:3px;right:3px}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload .h5p-file-drop-upload-inner:after{font-family:"H5P";line-height:1.4em;font-size:3.5em;color:#bbb}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload .h5p-file-drop-upload-inner.video:after{content:"\e903"}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload .h5p-file-drop-upload-inner.audio:after{content:"\e914"}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload:hover .h5p-file-drop-upload-inner:after,.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-file-drop-upload.over .h5p-file-drop-upload-inner:after{color:#999}.h5peditor .h5p-add-dialog .h5p-add-dialog-table .h5p-or-vertical{float:left;position:relative;width:5%;height:250px}.h5peditor .h5p-add-dialog .h5p-dialog-box{text-align:center;padding:1em 0.5em}.h5peditor .h5p-add-dialog .h5p-buttons{padding:0.5em;border-top:1px solid #cdcdcd;background:#ddd;text-align:right}.h5peditor .h5p-or{border-bottom:1px solid #cdcdcd;padding:0;margin:0 1em;text-align:center;height:0.5em;line-height:1em}.h5peditor .h5p-or>span{background:#fff;padding:0 0.5em}.h5peditor .h5p-or-vertical{float:left;position:relative;width:50px;height:80px}.h5peditor .h5p-or-vertical .h5p-or-vertical-line{position:absolute;margin:10px 0;left:49%;top:0;bottom:0;width:1px;background:#ccc;z-index:1}.h5peditor .h5p-or-vertical .h5p-or-vertical-word-wrapper{text-align:center;height:18px;position:absolute;left:0;right:0;top:22%;margin-top:-12px;z-index:2}.h5peditor .h5p-or-vertical .h5p-or-vertical-word-wrapper .h5p-or-vertical-word{color:#999;font-weight:bold;font-size:18px;padding:3px;background:#fff}.h5peditor .h5p-file-url{text-align:center}.h5peditor .h5p-thumbnail{margin:0.5em;width:6em;height:4.5em;display:block;float:left;position:relative;-moz-box-shadow:0 0 10px 0 #666;-webkit-box-shadow:0 0 10px 0 #666;box-shadow:0 0 10px 0 #666;border:1px solid #fff;box-sizing:border-box;-moz-box-sizing:border-box}.h5peditor .h5p-thumbnail .h5p-remove{position:absolute;top:0;right:0;cursor:pointer;outline:none;width:1.6em;height:1.6em;line-height:1.6em;overflow:hidden;text-indent:-0.4em;padding:0.065em;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=60);opacity:.6}.h5peditor .h5p-thumbnail .h5p-remove:hover,.h5peditor .h5p-thumbnail .h5p-remove:focus{filter:progid:DXImageTransform.Microsoft.Alpha(enabled=false);opacity:1}.h5peditor .h5p-thumbnail .h5p-remove:after{font-family:"H5P";font-size:2em;color:#fff;content:"\e890";filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=60);opacity:.6}.h5peditor .h5p-type{position:absolute;width:100%;height:100%;font-size:1.1em;line-height:4em;display:block;text-align:center;background:#000;color:#fff;cursor:pointer}.h5peditor .file{position:relative;float:left;margin-bottom:0;max-width:100%}.h5peditor .file.authorList{float:none}.h5peditor .file.field{float:none}.h5peditor .file .thumbnail{display:inline-block;margin:10px 10px 10px 0;-moz-box-shadow:0 0 10px 0 #666;-webkit-box-shadow:0 0 10px 0 #666;box-shadow:0 0 10px 0 #666;border:1px solid #fff;cursor:pointer;max-width:100%;min-width:1em;min-height:1em;background:#fff url('') repeat}.h5peditor .file .thumbnail:focus{-moz-box-shadow:0 0 10px 0 #222;-webkit-box-shadow:0 0 10px 0 #222;box-shadow:0 0 10px 0 #222}.h5peditor .file .add{display:inline-block;cursor:pointer;padding:0.5em 1.5em 0.5em 3em;background:linear-gradient(to bottom, #fbfbfb 0, #f2f2f2 100%);border:1px solid #d0d0d1;border-radius:0.25em;color:#222222;font-weight:bold;line-height:normal}.h5peditor .file .add:hover{background:#ededed}.h5peditor .file .add:focus{box-shadow:0 0 16px 0 rgba(133,188,255,0.84)}.h5peditor .file .add .h5peditor-field-file-upload-text:before{font-family:"H5P";content:"\e902";line-height:1;color:#39c943;font-size:2em;position:absolute;left:0.3em;top:0.1em}.h5peditor .file .remove{display:block;position:absolute;top:7px;right:7px;cursor:pointer}.h5peditor .file .remove:focus:before{filter:progid:DXImageTransform.Microsoft.Alpha(enabled=false);opacity:1}.h5peditor .file .remove:before{font-family:"H5P";font-size:1.4em;color:#fff;content:"\e890";filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=60);opacity:.6;text-shadow:rgba(0,0,0,0.4) 0 0 4px,rgba(0,0,0,0.4) 0 0 4px,rgba(0,0,0,0.4) 0 0 4px}.h5peditor .file .remove:hover{text-decoration:none}.h5peditor .file .remove:hover:before{color:#fff;filter:progid:DXImageTransform.Microsoft.Alpha(enabled=false);opacity:1;text-shadow:rgba(0,0,0,0.4) 0 0 4px,rgba(0,0,0,0.4) 0 0 4px,rgba(0,0,0,0.4) 0 0 4px}.h5peditor .file img{max-width:100%;vertical-align:bottom;max-height:100px}.h5peditor .file .h5p-av-row{overflow:auto;margin-bottom:0.5em}.h5peditor .file .h5p-av-row .h5p-thumbnail{display:table-cell}.h5peditor .file .h5p-av-row .h5p-video-quality{display:table-cell;overflow:hidden;width:10000px;padding-left:1em}.h5peditor .file .h5p-av-row .h5p-video-quality input.h5peditor-text{width:100%}.h5peditor .file .h5p-av-cell{overflow:auto;float:left}.h5peditor .video .file,.h5peditor .audio .file{position:static;overflow:visible}.h5peditor .video .file .thumbnail,.h5peditor .video .file .add,.h5peditor .audio .file .thumbnail,.h5peditor .audio .file .add{float:left}.h5peditor .video .file .add,.h5peditor .audio .file .add{margin-top:8px}.h5peditor .video .file .thumbnail,.h5peditor .audio .file .thumbnail{overflow:visible;position:relative;cursor:auto}.h5peditor .video .file .remove,.h5peditor .audio .file .remove{top:-3px;right:-5px}.h5peditor .video .file .type,.h5peditor .audio .file .type{padding:16px 8px 4px;background:#000;color:#fff;font-size:10px}.h5peditor .video .file .h5peditor-uploading,.h5peditor .audio .file .h5peditor-uploading{float:left;margin:0.5em}.h5peditor .libwrap{margin-top:20px}.h5peditor .libwrap.no-margin,.h5peditor .libwrap:empty{margin-top:0}.h5peditor input[type="checkbox"]{margin:4px 4px 4px 0;vertical-align:bottom}.h5peditor .moving{position:absolute;z-index:1;filter:progid:DXImageTransform.Microsoft.Alpha(Opacity=80);opacity:.8;-webkit-transform:translateZ(0)}.h5peditor .moving .h5peditor-label{cursor:grabbing;cursor:-moz-grabbing;cursor:-webkit-grabbing}.h5peditor .h5peditor-uploading,.h5peditor .h5peditor-loading{padding-top:10px;padding-bottom:6px;font-size:14px}.h5peditor .h5peditor-loading{padding:0.875em 0 1em 3.25em;font-style:italic}.h5peditor .h5p-copyright-button{-moz-border-radius:.25em;-webkit-border-radius:.25em;border-radius:.25em;height:30px;background:linear-gradient(to bottom, #fff 0, #f2f2f2 100%);border:1px solid #d0d0d1;color:#212121;font-size:14px;line-height:28px;padding-right:20px;padding-left:0;clear:both;font-weight:normal}.h5peditor .h5p-copyright-button:before{font-family:"H5P";content:"\e88f";color:#666;padding:0 0.25em 0 0.25em;vertical-align:middle;font-size:1.5em;line-height:0.9}.h5peditor .h5p-copyright-button:hover:not([disabled]){background:linear-gradient(to bottom, #fff 0, #d0d0d1 100%);text-decoration:none;border-color:#999}.h5peditor .h5p-copyright-button:focus{box-shadow:0 0 16px 0 rgba(133,188,255,0.84)}.h5peditor .field.file>.h5p-copyright-button,.h5peditor .field.video>.h5p-copyright-button,.h5peditor .field.audio>.h5p-copyright-button{float:left}.h5peditor .h5p-editor-dialog{position:absolute;z-index:2;margin:5.5em 0 1em;visibility:hidden;opacity:0;height:0;overflow:hidden;background:#fff;-moz-box-shadow:0 0 8px #666;-webkit-box-shadow:0 0 8px #666;box-shadow:0 0 8px #666;-moz-transition:visibility 0s .2s,height 0s .2s,opacity .2s,margin-top .2s;-o-transition:visibility 0s .2s,height 0s .2s,opacity .2s,margin-top .2s;-webkit-transition:visibility 0s,height 0s,opacity .2s,margin-top .2s;-webkit-transition-delay:.2s,.2s,0s,0s;transition:visibility 0s .2s,height 0s .2s,opacity .2s,margin-top .2s}.h5peditor .h5p-editor-dialog.h5p-open{margin-top:3.5em;visibility:visible;opacity:1;height:auto;-moz-transition:visibility 0s 0s,height 0s 0s,opacity .2s,margin-top .2s;-o-transition:visibility 0s 0s,height 0s 0s,opacity .2s,margin-top .2s;-webkit-transition:visibility 0s,height 0s,opacity .2s,margin-top .2s;-webkit-transition-delay:0s,0s,0s,0s;transition:visibility 0s 0s,height 0s 0s,opacity .2s,margin-top .2s}.h5peditor .h5p-editor-dialog>.field{margin:0;border:0;box-shadow:none}.h5peditor .h5p-editor-dialog .content{border:none;background:#fff}.h5peditor .h5p-editor-dialog .content .h5peditor-label{font-size:18px;font-weight:600}.h5peditor .h5p-editor-dialog .h5p-close{color:#494949}.h5peditor .h5p-editor-dialog .h5p-close:before{font-size:2em;right:-0.125em;top:0;position:absolute;z-index:1;font-family:"H5P";content:"\e894";line-height:1em;-moz-transition:scale .2s;-o-transition:scale .2s;-webkit-transition:scale .2s;transition:scale .2s}.h5peditor .h5p-editor-dialog .h5p-close:hover:before{-moz-transform:scale(1.1, 1.1);-ms-transform:scale(1.1, 1.1);-webkit-transform:scale(1.1, 1.1);transform:scale(1.1, 1.1)}.h5peditor .h5p-li>.content>.library{border:0;padding:0}.h5p-editor-dialog.h5p-dialog-wide{width:90%;border-radius:0.208em}.h5peditor-button-textual{-moz-border-radius:.25em;-webkit-border-radius:.25em;border-radius:.25em;background:#747275;background-image:linear-gradient(#7b797c 50%, transparent 50%, transparent);display:inline-block;width:auto;margin:10px 0 0 0;padding:0 20px;box-sizing:border-box;height:38px;border:1px solid #d0d0d1;font-size:16px;font-family:"Open Sans",sans-serif;line-height:38px;color:#fff;cursor:pointer;font-weight:600;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.h5peditor-button-textual:focus{box-shadow:0 0 16px 0 rgba(133,188,255,0.84)}.h5peditor-button-textual:before,.h5peditor-button-textual:after{color:#fff}.h5peditor-button-textual .h5p-editing-image-button:before{font-family:"H5P";content:"\e900";color:#666;padding-right:0.25em;vertical-align:middle;font-size:1.5em;line-height:0.9}.h5peditor-button-textual:hover:not([disabled]){background:#636164;background-image:linear-gradient(#6b696c 50%, transparent 50%, transparent);text-decoration:none}.h5peditor-button-textual.importance-high{background:#2579C6;background-image:linear-gradient(#3080c9 50%, transparent 50%, transparent);border-color:#1f67a8;text-transform:uppercase;height:42px;line-height:42px}.h5peditor-button-textual.importance-high:hover:not([disabled]){background:#1f67a8;background-image:linear-gradient(#2a6fac 50%, transparent 50%, transparent);border-color:#1f67a8}.h5peditor-button-textual.importance-low{background:#f5f5f5;background-image:linear-gradient(#eeeeef 50%, transparent 50%, transparent);border-color:#d0d0d1;color:#212121}.h5peditor-button-textual.importance-low:before,.h5peditor-button-textual.importance-low:after{color:#212121}.h5peditor-button-textual.importance-low:hover:not([disabled]){background:#deddde;background-image:linear-gradient(#deddde 50%, transparent 50%, transparent);border-color:#deddde}.h5peditor-button-textual[disabled],.h5peditor-button-textual[aria-disabled="true"]{background:#f5f5f5;color:#707171;border-color:#eeeeee;cursor:not-allowed}.h5peditor-button-textual[disabled]:before,.h5peditor-button-textual[aria-disabled="true"]:before{font-family:"H5P";content:"\e912";color:#707171;margin-right:0.5em;vertical-align:top}.h5peditor-field-description,.h5p-help-text{font-size:12px;margin-top:0.3em;margin-bottom:1em;font-weight:500;color:#434446;line-height:15px;letter-spacing:0.5px}.h5peditor-field-important-description{position:relative;padding:1.5em 2.2em 2em 2.2em;font-size:0.9em;margin:0 0 20px 0;border:1px solid #ffde78;background-color:#f9f9d8;color:#000;display:none}.h5peditor-field-important-description .important-description-close{position:absolute;font-size:0.9em;line-height:2em;right:1.5em;top:1em;user-select:none;cursor:pointer}.h5peditor-field-important-description .important-description-close:before{font-family:"H5P";content:"\e894";vertical-align:top;font-size:1.6em}.h5peditor-field-important-description .h5p-info-icon{font-size:0.75em}.h5peditor-field-important-description .h5p-info-icon:before{font-family:"H5P";content:"\e909";font-size:2.3em;line-height:0.94em;margin-left:-0.3em;vertical-align:middle}.h5peditor-field-important-description .important-description-title{font-weight:bold;font-size:1.2em;vertical-align:bottom}.h5peditor-field-important-description .important-description-content ul{padding:0;line-height:2em;margin-top:20px}.h5peditor-field-important-description .important-description-content ul li{line-height:1.8em;margin-left:1.5em}.h5peditor-field-important-description .important-description-example{border:1px solid #bbde98;background-color:#d9fac3;display:flex;margin-top:20px}.h5peditor-field-important-description .important-description-example .important-description-example-title{padding:15px;font-weight:bold}.h5peditor-field-important-description .important-description-example .important-description-example-text{padding:15px;width:100%;background-color:#f6fef1}.h5peditor-form{position:relative;background:#fcfcfc;border:1px solid #d0d0d1}.h5peditor-form.h5peditor-form-manager>.tree,.h5peditor-form.h5peditor-form-manager>.common,.h5peditor-form.h5peditor-form-manager>.field{max-width:918px;margin:20px auto;padding:0 20px}.h5peditor-form.h5peditor-semi-fullscreen{margin:0;position:fixed;overflow-y:scroll;margin-top:40px;padding-bottom:40px;box-sizing:border-box;background:#fff;height:calc(100% - 40px)}.h5peditor-label{display:block;margin-bottom:6px;font-weight:600;font-size:16px;color:#454347}#h5peditor-uploader{position:absolute;width:1px;height:1px;top:-1px;border:0;overflow:hidden}.h5p-tutorial-url,.h5p-example-url{font-size:1em;display:inline-flex;align-items:center;line-height:1;margin-right:1em;margin-bottom:0.5em;margin-left:0.25em;height:2rem;border-radius:1rem}.h5p-tutorial-url:focus,.h5p-example-url:focus{box-shadow:0px 0px 5px #1634f7}.h5p-tutorial-url:active,.h5p-example-url:active{border-color:#1C70BA}.h5p-tutorial-url::before,.h5p-example-url::before{font-family:"H5P";font-size:1.25em;border-radius:50%;width:2rem;height:2rem;position:absolute;display:flex;justify-content:center;align-items:center}.h5p-tutorial-url .h5p-tutorial-url-label,.h5p-tutorial-url .h5p-example-url-label,.h5p-example-url .h5p-tutorial-url-label,.h5p-example-url .h5p-example-url-label{font-style:italic;font-weight:bold;text-decoration:underline;margin:0.25em 0.5em;margin-left:2rem;padding-left:0.5em;padding-right:1em}.h5p-tutorial-url:active,.h5p-tutorial-url:hover{background-color:#eaffab}.h5p-tutorial-url::before{content:"\e93e";color:#eaffab;background-color:#749e00}.h5p-tutorial-url .h5p-tutorial-url-label{color:#415702}.h5p-example-url:active,.h5p-example-url:hover{background-color:#f8d4ff}.h5p-example-url::before{content:"\e93d";font-size:1em;color:#f8d4ff;background-color:#8f4f9c}.h5p-example-url .h5p-example-url-label{color:#422447}.h5peditor-widget-select{overflow:hidden;margin:0 0 -1px;padding:0;list-style:none}.h5peditor-widget-option{float:right;border:1px solid #ccc;border-bottom:0;margin-left:0.5em;padding:0.6em 1em;color:#0E1A25;font-size:0.875em;background:#f5f5f5;line-height:1.285714286em;cursor:pointer;outline:none}.h5peditor-widget-option:hover{color:#000}.h5peditor-widget-option:active{color:#8e636a}.h5peditor-widget-active{background:#fff;line-height:1.357142857em}.h5peditor-widgets>.h5peditor-widget-wrapper{border:1px solid #ccc;margin:0 0 0.25em;padding:0.5em}.h5peditor-widgets>.h5peditor-label{float:left;margin-top:5px}.h5p-editor-iframe{margin-bottom:1em}.h5peditor-required:after{content:"*";color:#da0001;margin-left:0.2em;position:relative;top:-0.2em}.h5peditor .cke_bottom,.h5peditor .cke_top{background:#d0d0d1}.h5peditor .cke_chrome{border:1px solid #f5f5f5;background:#d0d0d1}.h5peditor .cke_contents,.h5peditor .cke_toolgroup,.h5peditor .cke_combo_button{border:1px solid #f5f5f5}.important-description-show{background:#f3d55a;padding:0.2em 0.5em 0.2em 1.7em;margin-bottom:0.5em;font-size:0.9em;cursor:pointer;float:right;position:relative;line-height:normal;font-style:italic;font-weight:bold}.important-description-show:before{font-family:"H5P";font-style:normal;font-weight:normal;content:"\e909";font-size:1.8em;position:absolute;left:0em;top:0em;line-height:0.85em;vertical-align:top}.important-description-clear-right{display:block;clear:right}.h5peditor fieldset.common-fields-library-wrapper{width:100%;padding:0 1em 1em;box-sizing:border-box;height:20px;border:solid 1px #e2e5ee;border-radius:4px;margin-bottom:1em}.h5peditor fieldset.common-fields-library-wrapper>*{display:none}.h5peditor fieldset.common-fields-library-wrapper>legend{display:block;cursor:pointer;outline:none;color:#363b42;background-color:#fff;padding:10px;font-weight:bold;font-size:0.875em}.h5peditor fieldset.common-fields-library-wrapper>legend:before{font-family:"H5P";content:"\e566";margin-right:0.5em}.h5peditor fieldset.common-fields-library-wrapper>legend:focus:before{outline:1px dotted #666}.h5peditor fieldset.common-fields-library-wrapper.expanded{height:auto}.h5peditor fieldset.common-fields-library-wrapper.expanded>*{display:block}.h5peditor fieldset.common-fields-library-wrapper.expanded>legend:before{content:"\e565"}.h5p-metadata-button-wrapper{display:flex;flex-direction:row;align-items:center;margin-top:-7px;margin-left:7px;cursor:pointer}.h5p-metadata-button-wrapper.inline-with-selector{display:inline-flex;margin-left:17px}.h5p-metadata-toggler{margin-left:-11px;height:14.5px;max-height:14.5px;padding:0.217rem 0.3rem 0.217rem 0.217rem;font-size:0.625em;font-weight:bold;letter-spacing:0.025em;border:1px solid #b6cada;border-left:none;border-radius:4px;background:linear-gradient(#fff, #deeaf1);color:#356593;z-index:1;justify-content:center;align-content:center;line-height:1.5em}.h5p-metadata-button-tip{width:16.5px;height:16.5px;content:"";background:linear-gradient(-215deg, #fff, #deeaf1);border:1px solid #b6cada;border-right:none;border-top:none;border-radius:4px;transform:rotate(45deg);z-index:0}.h5p-metadata-button-wrapper:hover .h5p-metadata-toggler,.h5p-metadata-button-wrapper:hover .h5p-metadata-button-tip{border-color:#6d9fce}.h5p-metadata-button-wrapper:hover .h5p-metadata-toggler{background:linear-gradient(#f3f8fb, #cee5f3)}.h5p-metadata-button-wrapper:hover .h5p-metadata-button-tip{background:linear-gradient(-215deg, #f3f8fb, #cee5f3)}.h5p-metadata-button-wrapper:active .h5p-metadata-toggler,.h5p-metadata-button-wrapper:active .h5p-metadata-button-tip{color:#294f73;border-color:#6d9fce}.h5p-metadata-button-wrapper:active .h5p-metadata-toggler{background:linear-gradient(#f3f8fb, #bcd4e2)}.h5p-metadata-button-wrapper:active .h5p-metadata-button-tip{background:linear-gradient(-215deg, #f3f8fb, #bcd4e2)}#metadata-title-main-label{margin-top:0}.h5p-metadata-button{background:white;padding:10px;text-align:center;border-radius:5px;border:2px solid #6b6b6b;color:#6b6b6b;font-weight:bold;display:inline;cursor:pointer}.h5p-metadata-button:hover{border-color:#4a4a4a;color:#4a4a4a}.h5p-metadata-button:active{border-color:#000;color:#000}.h5p-metadata-button.inverted{border:2px solid #0a715e;color:#0a715e}.h5p-metadata-button.inverted:hover{border-color:#0d826c;color:#0d826c}.h5p-metadata-button.inverted:active{border-color:#095345;color:#095345}.h5p-metadata-icon-button{border:none;cursor:pointer;color:#6b6b6b;background:transparent;padding:0}.h5p-metadata-icon-button:hover{color:#4a4a4a}.h5p-metadata-icon-button:active{color:#000}.h5p-metadata-wrapper{display:inline-block;background-color:#fff;max-width:700px;width:calc(100% - 4em);text-align:left;border-bottom:20px solid transparent;background-clip:padding-box;max-height:calc(100% - 40px);overflow:auto;margin-bottom:20px}.h5p-metadata-wrapper .h5p-metadata-header{display:flex;border-bottom:1px solid #ced6e3;padding:1.5em}.h5p-metadata-wrapper .h5p-metadata-header .h5p-title-container{flex-grow:1;padding-left:3.5em;white-space:nowrap;overflow:hidden;margin-right:1%;position:relative}.h5p-metadata-wrapper .h5p-metadata-header .h5p-title-container h2{margin:0}.h5p-metadata-wrapper .h5p-metadata-header .h5p-title-container p{margin:0.1em}.h5p-metadata-wrapper .h5p-metadata-header .h5p-title-container:before{position:absolute;font-family:'h5p-metadata-icons';content:'\e903';left:0;top:0;font-size:2em;height:1.4em;line-height:1.3}.h5p-metadata-wrapper .h5p-save{border:solid 2px #0a715e;border-radius:0.3rem;color:#fff;background-color:#0a715e;padding:0.75rem;white-space:nowrap}.h5p-metadata-wrapper .h5p-save:hover{border-color:#0d826c;background-color:#0d826c}.h5p-metadata-wrapper .h5p-save:active{border-color:#095345;background-color:#095345}.h5p-metadata-wrapper .h5peditor-label{font-size:0.8em;color:#333}.h5p-metadata-wrapper .h5peditor-field-description{color:#697484;margin-top:0.35em;letter-spacing:0.5px}.h5p-metadata-wrapper .copyright-form{margin-top:20px}.h5p-metadata-wrapper h2{font-size:1em;overflow:hidden;text-overflow:ellipsis}.h5p-metadata-wrapper p{font-size:0.8333em;color:#697484;overflow:hidden;text-overflow:ellipsis;margin-bottom:0}.h5p-metadata-wrapper .errors p,.h5p-metadata-wrapper .h5p-errors p{color:#da0001;font-size:1em;white-space:normal;text-overflow:unset}.h5p-metadata-wrapper select.h5peditor-select,.h5p-metadata-wrapper input.h5peditor-text,.h5p-metadata-wrapper textarea{border:1px solid #b8c0cd;font-size:0.833em;border-radius:0.208em;box-shadow:none;font-family:"Open Sans",sans-serif}.h5p-metadata-wrapper select:focus,.h5p-metadata-wrapper input:focus,.h5p-metadata-wrapper textarea:focus{box-shadow:inset 0px 0px 10px rgba(0,0,0,0.15)}.h5p-metadata-wrapper select{width:100%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper{display:flex;flex-wrap:wrap;margin:1.5em}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field{margin-bottom:0;width:100%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-license{width:49%;margin-right:1%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-licenseVersion{width:49%;margin-left:1%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-yearFrom{width:24%;margin-right:1%;white-space:nowrap}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-yearFrom .h5peditor-text{width:100%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-yearTo{width:23%;margin-right:1%;margin-left:1%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-yearTo .h5peditor-text{width:100%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper>.field-name-source{width:49%;margin-left:1%}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label{justify-content:space-between}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle{background:none;border:none;cursor:pointer}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle:before{font-family:'h5p-metadata-icons';content:"\e905";margin-right:0.5em;font-size:1.5em;vertical-align:middle}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle:focus,.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle:active,.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle:hover{border:none}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-title label .a11y-title-toggle:hover .h5p-a11y-title-text{text-decoration:underline}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-a11yTitle{transition:max-height 0.2s, margin 0.2s;overflow:hidden}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-a11yTitle.hide{max-height:0;margin:0}.h5p-metadata-wrapper .h5p-metadata-fields-wrapper .field-name-a11yTitle.hidden{display:none}.h5p-metadata-wrapper .field.group>.title{font-weight:400;background:#F6F6F6;color:#323232;border:solid 1px #ced6e3;border-radius:0.208em}.h5p-metadata-wrapper .field.group.expanded>.title{border-radius:0.208em 0.208em 0 0}.h5p-metadata-wrapper .field.group>.title:focus{border-color:#4d90fe}.h5p-metadata-wrapper .field.group>.content{border:solid 1px #ced6e3;border-radius:0 0 0.208em 0.208em;padding:20px;border-top:0}.h5p-metadata-additional-information{width:100%;margin-top:1em}.h5p-metadata-additional-information .content.h5peditor-single{min-height:2em;border:1px solid #d0d0d1;border-top:none;background:#fff}.h5p-metadata-additional-information .title{font-weight:400;background:#F6F6F6;color:#323232;border:solid 1px #ced6e3;border-radius:0.208em}.h5p-metadata-additional-information .title:focus{border-color:#4d90fe}.h5p-metadata-popup-overlay{position:absolute;top:0;left:0;right:0;bottom:0;background-color:rgba(0,0,0,0.85);z-index:101;text-align:center}.h5p-metadata-changelog{width:100%}.h5p-metadata-changelog .field-name-change.expanded .content{display:flex;flex-wrap:wrap}.h5p-metadata-changelog .field-name-change .field-name-date{width:49%;margin-bottom:0.5em;margin-right:1%;margin-top:0}.h5p-metadata-changelog .field-name-change .field-name-date span{width:100%}.h5p-metadata-changelog .field-name-change .field-name-author{width:49%;margin-bottom:0.5em;margin-left:1%;margin-top:0}.h5p-metadata-changelog .field-name-change .field-name-log{width:100%}.h5p-metadata-changelog .field-name-change .h5p-add-author:before{content:"+";margin-right:0.7em}.h5p-metadata-changelog .field-name-change .h5p-cancel{margin-right:10px}.h5p-metadata-changelog .field-name-change .h5p-metadata-changelog-buttons{width:100%}.h5p-metadata-changelog .field-name-change .h5p-metadata-logged-changes{width:100%;margin-top:1em}.h5p-metadata-changelog .field-name-change .h5p-metadata-logged-changes.editing{display:none}.h5p-metadata-changelog .field-name-change .h5p-metadata-logged-changes .h5peditor-field-description{margin-bottom:0}.h5p-metadata-changelog .field-name-change .h5p-metadata-log-wrapper{margin-top:1em;border:solid 1px #ced6e3;border-radius:0.208em;max-height:15.917rem;overflow-y:auto;padding:0 0.5em}.h5p-metadata-changelog .field-name-change .h5p-metadata-log{display:flex;margin-top:0.5em;padding-bottom:0.5em}.h5p-metadata-changelog .field-name-change .h5p-metadata-log .h5p-metadata-delete{margin-left:2em}.h5p-metadata-changelog .field-name-change .h5p-metadata-log .h5p-metadata-delete:after{font-family:'h5p-metadata-icons';content:"\e902";font-size:1.2em}.h5p-metadata-changelog .field-name-change .h5p-metadata-log .h5p-metadata-edit:after{font-family:'h5p-metadata-icons';content:"\e904";font-size:1.2em}.h5p-metadata-changelog .field-name-change .h5p-metadata-log-buttons{min-width:3em;line-height:2.5}.h5p-metadata-changelog .field-name-change .h5p-metadata-log:not(:last-child){border-bottom:solid 1px #ced6e3}.h5p-metadata-changelog .field-name-change .h5p-metadata-log-date{min-width:25%;font-size:0.85em;font-style:italic;line-height:2.5}.h5p-metadata-changelog .field-name-change .h5p-metadata-description-wrapper{flex-grow:1;font-size:0.85em;font-style:italic}.h5p-metadata-changelog .field-name-change .h5p-metadata-description-wrapper{font-weight:bolder}.h5p-metadata-changelog .field-name-change .h5p-metadata-log-author{font-weight:normal}.h5p-metadata-changelog .field-name-change .h5p-metadata-new-log-message{width:100%;background:#C5E7E1;padding:0.5em;margin-bottom:1em;color:#0a715e}.h5p-metadata-author-widget{display:flex;flex-direction:column;border:solid 1px #ced6e3;border-radius:0.208em;width:100%;padding:1em;margin-top:1em}.h5p-metadata-author-widget .h5p-save-author{border:solid 2px #0a715e;color:#fff;background-color:#0a715e}.h5p-metadata-author-widget .h5p-save-author:hover{border-color:#0d826c;color:#fff;background-color:#0d826c}.h5p-metadata-author-widget .h5p-save-author:active{border-color:#095345;color:#fff;background-color:#095345}.h5p-metadata-author-widget .h5p-author-data{display:flex;flex-direction:row;flex-wrap:nowrap}.h5p-metadata-author-widget .h5p-author-data>.field{margin-bottom:0}.h5p-metadata-author-widget .field-name-name{flex-grow:1;margin-right:2%;overflow:hidden;white-space:nowrap;min-width:49%}.h5p-metadata-author-widget .field-name-role{margin-top:0;margin-right:2%;width:25%;flex-grow:1;overflow:hidden;white-space:nowrap}.h5p-metadata-author-widget .authorList{margin-top:1.5em;white-space:nowrap}.h5p-metadata-author-widget .authorList li{display:inline;margin-right:0.5em}.h5p-metadata-author-widget .authorList li span{margin-right:1em;font-style:italic}.h5p-metadata-author-widget .h5p-author-list-wrapper{margin-top:0.5em}.h5p-metadata-author-widget .h5p-author-list-wrapper ul{margin:0;padding:0}.h5p-metadata-author-widget .h5p-author-list-wrapper li{display:inline-block;margin-right:2em;list-style:none;font-weight:bold;font-size:0.8em}.h5p-metadata-author-widget .h5p-author-list-wrapper .h5p-metadata-role{font-style:italic;margin:0 0.6em;font-weight:normal;font-size:0.85em;color:#757575}.h5p-metadata-author-widget .h5p-author-list-wrapper button:after{font-family:'h5p-metadata-icons';content:"\e902";font-size:1em;position:relative;top:-0.1em}.h5peditor-copypaste-wrap{float:right}.h5peditor>.h5peditor-copypaste-wrap{margin-bottom:14px}.h5peditor-copypaste-wrap.hidden{display:none}.h5peditor-clearfix{clear:both}.h5peditor-paste-button,.h5peditor-copy-button{border:1px solid #deeeec;padding:0.5em 0.75em 0.75em 0.5em;margin-left:0.5em;border-radius:0.25em;cursor:pointer;background:#f2faff;line-height:0.5em}.h5peditor-paste-button:hover,.h5peditor-copy-button:hover{border-color:#bbdae8}.h5peditor-paste-button:active,.h5peditor-copy-button:active{background-color:#deeffb}.h5peditor-paste-button.disabled,.h5peditor-copy-button.disabled{color:#707070;background-color:#f5f5f5;border-color:#ededed;cursor:not-allowed}.h5peditor-copy-button:before,.h5peditor-paste-button:before{font-family:"H5P";margin-right:5px;font-size:1.5em;position:relative;top:0.2em;color:#2372b3}.h5peditor-copy-button:before{content:"\e90e"}.h5peditor-paste-button:before{content:"\e910"}.h5peditor-copy-button.disabled:before,.h5peditor-paste-button.disabled:before{color:#707070}.h5peditor-copy-button.disabled:before{content:"\e90f"}.h5peditor-paste-button.disabled:before{content:"\e911"}.h5p-hub .h5peditor-paste-button{font-family:"Open Sans",sans-serif;color:#fff;border:solid 1px #9fc8f4;padding:0 0.75em 0.5em 0.75em;background:transparent;line-height:1;height:auto;margin-top:0.675em;margin-right:0.675em}.h5p-hub .h5peditor-paste-button:hover{background:#4b5460;border-color:#606a78}.h5p-hub .h5peditor-paste-button:active{background:#434b55}.h5p-hub .h5peditor-paste-button.disabled{border-color:#30353d;background:#30353d;color:#c2c2c2}.h5p-hub .h5peditor-paste-button:before{color:#9fc8f4}.h5p-hub .h5peditor-paste-button.disabled:before{color:#c3c3c3}@font-face{font-family:'h5p-fullscreen-bar';src:url("fonts/h5p-fullscreen-bar.eot?p850ul");src:url("fonts/h5p-fullscreen-bar.eot?p850ul#iefix") format("embedded-opentype"),url("fonts/h5p-fullscreen-bar.ttf?p850ul") format("truetype"),url("fonts/h5p-fullscreen-bar.woff?p850ul") format("woff"),url("fonts/h5p-fullscreen-bar.svg?p850ul#h5p") format("svg");font-weight:normal;font-style:normal}.h5peditor-form-manager-head{background:#f5f5f5;color:#414141;border-bottom:1px solid #dfdfdf;height:41px;line-height:40px;box-sizing:border-box;position:relative;display:flex;z-index:3;justify-content:space-between;align-items:flex-start;box-shadow:0px 2px 2px rgba(128,128,128,0.15);padding:0 1em 0 0.5em}.h5peditor-form-manager-button{background:transparent;padding:0;border:0;cursor:pointer;color:#414141;position:relative;white-space:nowrap;line-height:normal}.h5peditor-form-manager-button-inner{display:block;outline:none}.h5peditor-form-manager-breadcrumb{display:flex;min-width:0;flex-grow:1}.h5peditor-form-manager-proceed{display:none;background:#186df7;color:#fff;font-size:14px;font-weight:bold;font-family:Open Sans, sans-serif;border-radius:2px;margin-right:5px;margin-top:5px;padding:5px 10px}.h5peditor-form-manager-proceed:hover{background:#20588f}.h5peditor-form-manager-fullscreen{width:24px;height:24px;font-size:20px;margin-top:9px;margin-left:0.25em}.h5peditor-form-manager-fullscreen .h5peditor-form-manager-button-inner:before{font-family:'h5p-fullscreen-bar';content:'\e929'}.h5peditor-form-manager-fullscreen:after{visibility:hidden;position:absolute;top:120%;right:-10%;z-index:3;padding:0.25em 0.75em;background:#212121;color:#fff;white-space:nowrap;font-size:14px;box-shadow:0 0 0.5em #858585;text-indent:0;font-weight:normal;pointer-events:none;outline:none;line-height:normal;content:attr(aria-label)}.h5peditor-form-manager-fullscreen:hover:after,.h5peditor-form-manager-fullscreen:focus:after{visibility:visible}.h5peditor-semi-fullscreen .h5peditor-form-manager-fullscreen .h5peditor-form-manager-button-inner:before{content:"\e92a"}.h5peditor-semi-fullscreen .h5peditor-form-manager-proceed{display:block}.h5peditor-semi-fullscreen .h5peditor-form-manager-head{position:fixed;top:0;left:0;right:0;z-index:2}.form-manager-exit.form-manager-fullscreen .h5peditor-form-manager-button-inner:before{content:"\f066"}.h5peditor-form-manager-title{position:relative;min-width:20px;font-weight:bold;font-size:14px}.h5peditor-form-manager-title:before{font-family:'h5p-fullscreen-bar';content:"\e928";font-size:1.2em;position:relative;top:0.1em;margin-right:0.25em;speak:none;font-style:normal;font-weight:normal;font-variant:normal;text-transform:none;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.h5peditor-form-manager-title.dragtext:before{content:"\e600"}.h5peditor-form-manager-title.markthewords:before{content:"\e601"}.h5peditor-form-manager-title.multichoice:before{content:"\e603"}.h5peditor-form-manager-title.interactivevideo:before{content:"\e900"}.h5peditor-form-manager-title.audio:before{content:"\e901"}.h5peditor-form-manager-title.truefalse:before{content:"\e902"}.h5peditor-form-manager-title.dialogcards:before{content:"\e903"}.h5peditor-form-manager-title.questionnaire:before{content:"\e904"}.h5peditor-form-manager-title.coursepresentation:before{content:"\e923"}.h5peditor-form-manager-title.dragquestion:before{content:"\e991"}.h5peditor-form-manager-title.summary:before{content:"\e992"}.h5peditor-form-manager-title.singlechoiceset:before{content:"\e993"}.h5peditor-form-manager-title.blanks:before{content:"\e994"}.h5peditor-semi-fullscreen+.h5p-metadata-popup-overlay{position:fixed}.h5peditor-semi-fullscreen+.h5p-metadata-popup-overlay .h5p-metadata-wrapper{margin-top:20px !important}.h5peditor .h5peditor-language-switcher{float:right;white-space:nowrap;margin-bottom:1em}.h5peditor .h5peditor-language-switcher .language-label{padding:0 10px;font-size:15px}.h5peditor .h5peditor-language-switcher select{padding:6px 30px 6px 8px;font-size:15px}.h5peditor-language-notice{display:none;clear:both;font-size:14px;background:#dcf6ff;color:#295b7a;padding:10px 20px 10px 40px;margin:1em 0;line-height:1.5;position:relative}.h5peditor-language-notice:before{font-family:h5p;content:"\e90c";position:absolute;left:15px;font-size:1.125em}.h5peditor-language-notice.show{display:block}.h5peditor-language-notice .first{font-weight:bold}.h5peditor-language-notice a{text-decoration:underline} diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/cke-contents.css b/h5p/h5plib/v126/joubel/editor/styles/css/cke-contents.css deleted file mode 100644 index ed8446a379811..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/cke-contents.css +++ /dev/null @@ -1 +0,0 @@ -@import url(../../ckeditor/contents.css);code{color:#3d3d3d;background:#e0e0e0;border-radius:2px;padding:0 5px}pre>code{background-color:#fafafa;padding:5px;display:block;line-height:normal;border:1px solid #c7c7c7;border-left-width:4px;max-width:100%;white-space:pre;overflow:auto} diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts.css b/h5p/h5plib/v126/joubel/editor/styles/css/fonts.css deleted file mode 100644 index 25579a6654f23..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/fonts.css +++ /dev/null @@ -1,21 +0,0 @@ -@font-face { - font-family: 'h5p-hub'; - src: url('fonts/h5p-hub.eot?626d03'); - src: url('fonts/h5p-hub.eot?626d03#iefix') format('embedded-opentype'), - url('fonts/h5p-hub.ttf?626d03') format('truetype'), - url('fonts/h5p-hub.woff?626d03') format('woff'), - url('fonts/h5p-hub.svg?626d03#h5p-hub') format('svg'); - font-weight: normal; - font-style: normal; -} - -@font-face { - font-family: 'h5p-metadata-icons'; - src: url('fonts/h5p-metadata-icons.eot?qj7uq5'); - src: url('fonts/h5p-metadata-icons.eot?qj7uq5#iefix') format('embedded-opentype'), - url('fonts/h5p-metadata-icons.ttf?qj7uq5') format('truetype'), - url('fonts/h5p-metadata-icons.woff?qj7uq5') format('woff'), - url('fonts/h5p-metadata-icons.svg?qj7uq5#h5p-metadata-icons') format('svg'); - font-weight: normal; - font-style: normal; -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.eot b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.eot deleted file mode 100644 index cdefabfb0e5f5..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.eot and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.svg b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.svg deleted file mode 100644 index f0fc940ccb16d..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.svg +++ /dev/null @@ -1,42 +0,0 @@ - - - - - - -{ - "fontFamily": "h5p", - "description": "Font generated by IcoMoon.", - "majorVersion": 1, - "minorVersion": 1, - "version": "Version 1.1", - "fontId": "h5p", - "psName": "h5p", - "subFamily": "Regular", - "fullName": "h5p" -} - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.ttf b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.ttf deleted file mode 100644 index fc1a55ff72424..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.ttf and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.woff b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.woff deleted file mode 100644 index ae427e3381e5d..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-fullscreen-bar.woff and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.eot b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.eot deleted file mode 100644 index 9e6a10f17481e..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.eot and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.svg b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.svg deleted file mode 100644 index 274e983ef57bb..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.svg +++ /dev/null @@ -1,49 +0,0 @@ - - - - - - -{ - "fontFamily": "h5p-hub", - "description": "Font generated by IcoMoon.", - "majorVersion": 1, - "minorVersion": 2, - "version": "Version 1.2", - "fontId": "h5p-hub", - "psName": "h5p-hub", - "subFamily": "Regular", - "fullName": "h5p-hub" -} - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.ttf b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.ttf deleted file mode 100644 index 1e8baf8a50b58..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.ttf and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.woff b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.woff deleted file mode 100644 index fc14390574eba..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-hub.woff and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.eot b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.eot deleted file mode 100644 index ad0d43a9805fb..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.eot and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.svg b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.svg deleted file mode 100644 index 43e1afcd0e94e..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.svg +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - -{ - "fontFamily": "h5p-metadata-icons", - "description": "Font generated by IcoMoon.", - "majorVersion": 1, - "minorVersion": 1, - "copyright": "H5P", - "fontURL": "", - "version": "Version 1.1", - "fontId": "h5p-metadata-icons", - "psName": "h5p-metadata-icons", - "subFamily": "Regular", - "fullName": "h5p-metadata-icons" -} - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.ttf b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.ttf deleted file mode 100644 index 45b5cb94fe1db..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.ttf and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.woff b/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.woff deleted file mode 100644 index 98142a58fa4a9..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/fonts/h5p-metadata-icons.woff and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/h5p-hub-client.css b/h5p/h5plib/v126/joubel/editor/styles/css/h5p-hub-client.css deleted file mode 100644 index 27909f840d85a..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/h5p-hub-client.css +++ /dev/null @@ -1,84 +0,0 @@ -.h5p-hub article,.h5p-hub aside,.h5p-hub details,.h5p-hub figcaption,.h5p-hub figure,.h5p-hub footer,.h5p-hub header,.h5p-hub hgroup,.h5p-hub menu,.h5p-hub nav,.h5p-hub section{display:block}.h5p-hub ol,.h5p-hub ul{list-style:none}.h5p-hub ul.ul,.h5p-hub .h5p-hub-ul{list-style:disc;margin:.833em 0}.h5p-hub ul.ul li,.h5p-hub .h5p-hub-ul li{margin-left:2.499em}.h5p-hub blockquote,.h5p-hub q{quotes:none}.h5p-hub blockquote:before,.h5p-hub blockquote:after,.h5p-hub q:before,.h5p-hub q:after{content:'';content:none}.h5p-hub table{border-collapse:collapse;border-spacing:0}.h5p-hub dl{margin-top:0;margin-bottom:0}.h5p-hub dd,.h5p-hub dt{padding:0;margin:0}.h5p-hub dt{font-weight:700}.h5p-hub button{border:0;padding:0;margin:0}.h5p-hub{font-family:'Open Sans', sans-serif;font-size:.917em;color:#1f2227}.h5p-hub .h1,.h5p-hub .h2,.h5p-hub .h3,.h5p-hub .h4,.h5p-hub .h5,.h5p-hub .h6,.h5p-hub h1,.h5p-hub h2,.h5p-hub h3,.h5p-hub h4,.h5p-hub h5,.h5p-hub h6{font-family:inherit;color:inherit;font-weight:300;line-height:1.1}.h5p-hub .h1,.h5p-hub .h2,.h5p-hub .h3,.h5p-hub h1,.h5p-hub h2,.h5p-hub h3{margin-top:.833em;margin-bottom:.4165em}.h5p-hub .h4,.h5p-hub .h5,.h5p-hub .h6,.h5p-hub h4,.h5p-hub h5,.h5p-hub h6{margin-top:.4165em;margin-bottom:.4165em;font-weight:600}.h5p-hub h1,.h5p-hub .h1{font-size:1.733em}.h5p-hub h2,.h5p-hub .h2{font-size:1.458em}.h5p-hub h3,.h5p-hub .h3{font-size:1.250em}.h5p-hub h4,.h5p-hub .h4{font-size:1.042em}.h5p-hub h5,.h5p-hub .h5{font-size:0.938em}.h5p-hub h6,.h5p-hub .h6{font-size:0.875em}.h5p-hub ul{padding:0;margin:0}.h5p-hub hr{border:0;height:0;margin:.833em 0;border-bottom:1px solid #ececec}.h5p-hub small,.h5p-hub .small,.h5p-hub .h5p-hub-small{color:#697585;line-height:1.7em;font-size:.95em}.h5p-hub .link,.h5p-hub a{font-family:'Open Sans', sans-serif;color:#1f2227;font-size:1em;font-weight:400;cursor:pointer;text-decoration:underline;background-color:transparent;border:0;padding:0}.h5p-hub .dl-horizontal dt{float:left;width:5em;overflow:hidden;clear:left;text-align:right;text-overflow:ellipsis;white-space:nowrap}.h5p-hub .dl-horizontal dd{margin-left:5.833em}.h5p-hub .additional-information{font-style:italic;color:#697585}.h5p-hub .hidden{display:none}.h5p-hub .h5p-hub-hidden{display:none}.h5p-hub .h5p-hub-button{font-family:'Open Sans', sans-serif;font-size:.95em;font-weight:600;text-align:center;text-decoration:none;line-height:1em;min-width:4em;padding:0.708em 1.5em;border-radius:1.375em;background-color:#f3f3f3;color:#1f2227;display:inline-block;cursor:pointer;border:2px solid transparent;transition:background-color .35s ease}.h5p-hub .h5p-hub-button[disabled],.h5p-hub .h5p-hub-button[aria-disabled]{cursor:not-allowed}.h5p-hub .h5p-hub-button:focus{outline:0;box-shadow:0.06em 0 0.6em 0.1em #7bc1f9}.h5p-hub .h5p-hub-button.h5p-hub-button-primary{background-color:#0a78d1;color:#fff;border-color:#0a78d1}.h5p-hub .h5p-hub-button.h5p-hub-button-primary:hover{background-color:#096ab9;border-color:#096ab9}.h5p-hub .h5p-hub-button.h5p-hub-button-primary:active{background-color:#085ca0;border-color:#085ca0}.h5p-hub .h5p-hub-button.h5p-hub-button-inverse-primary{background-color:#fff;color:#0a78d1;border-color:#0a78d1}.h5p-hub .h5p-hub-button.h5p-hub-button-inverse-primary:hover{color:#085ca0;border-color:#085ca0}.h5p-hub .h5p-hub-button.h5p-hub-button-inverse-primary:active{color:#054070;border-color:#054070}.h5p-hub .h5p-hub-button:before{font-weight:normal}.h5p-hub .h5p-hub-img-responsive{max-width:100%;object-fit:contain}.h5p-hub #h5p-hub-reuse-view .h5p-hub-img-responsive{background-color:#d0d0d1}@-webkit-keyframes spin{100%{transform:rotate(360deg)}}@keyframes spin{100%{transform:rotate(360deg)}}.h5p-hub .h5p-hub-button [class^="icon-"]{margin-right:.4165em;display:inline-block;font-weight:normal;font-size:0.9em}.h5p-hub .h5p-hub-icon-hub-icon:before{font-family:'h5p-hub';content:""}.h5p-hub .h5p-hub-icon-arrow-thick:before{font-family:'h5p-hub';content:""}.h5p-hub .h5p-hub-icon-arrow-thin:before{font-family:'h5p-hub';content:""}.h5p-hub .h5p-hub-icon-accordion-arrow:before{font-family:'h5p-hub';font-weight:normal;content:""}.h5p-hub .h5p-hub-icon-search:before{font-family:'h5p-hub';content:""} - -.h5p-hub .h5p-hub-client-drop-down{background-color:white;color:#697585;font-style:italic;font-weight:600;cursor:pointer;position:relative}.h5p-hub .h5p-hub-client-drop-down .h5p-hub-icon-hub-icon{color:#474f5a;padding:0.833em 1.55em}.h5p-hub .h5p-hub-client-drop-down .h5p-hub-icon-hub-icon:before{font-style:normal;font-weight:normal;margin-right:1em}.h5p-hub .h5p-hub-client-drop-down .h5p-hub-icon-hub-icon:after{font-family:'h5p-hub';content:"";display:inline-block;speak:none;font-style:normal;font-weight:normal;font-variant:normal;text-transform:none;margin-right:.9163em;margin-right:1.8em;display:block;font-size:0.9em;line-height:3.5em;position:absolute;right:0;top:0}.h5p-hub .h5p-hub-client-drop-down .h5p-hub-description{color:transparent;width:1px;height:1px;overflow:hidden;display:inline-block;position:absolute}.h5p-hub .h5p-hub-client-drop-down .h5p-hub-selected{color:#474f5a} - -.h5p-hub .h5p-hub-panel.h5p-hub-section-content-types .h5p-hub-tab-button[aria-selected="true"]{border-bottom:3px solid #0a78d1}.h5p-hub .h5p-hub-panel.h5p-hub-section-reuse .h5p-hub-tab-button[aria-selected="true"]{border-bottom:3px solid #db6f28}.h5p-hub .h5p-hub-panel.h5p-hub-section-upload .h5p-hub-tab-button[aria-selected="true"]{border-bottom:3px solid #ced6e3}.h5p-hub .h5p-hub-tab-panel{border:none}.h5p-hub .h5p-hub-tab-panel [role="tablist"]{background-color:#474f5a;list-style:none;border-bottom:none;padding-left:1em;margin:0}.h5p-hub .h5p-hub-tab-panel [role="tablist"]>span{display:block;height:0.3em;background:#0a78d1;position:relative;transition:all .5s ease}.h5p-hub .h5p-hub-tab-panel [role="tab"]{display:inline-block;text-decoration:none;text-align:center;transition:all .5s ease;color:#f3f3f3;outline-color:#1a93f4;border-bottom:3px solid transparent;transition:border-bottom 0.4s}.h5p-hub .h5p-hub-tab-panel [role="tab"]:nth-child(1) ~ span,.h5p-hub .h5p-hub-tab-panel [role="tab"]:nth-child(1)[aria-selected="true"] ~ span{height:0.27em;width:200px;right:0}.h5p-hub .h5p-hub-tab-panel [role="tab"]:nth-child(2) ~ span,.h5p-hub .h5p-hub-tab-panel [role="tab"]:nth-child(2)[aria-selected="true"] ~ span{height:0.27em;width:4em;right:-230px}.h5p-hub .h5p-hub-tab-panel [role="tab"]>a{display:block;color:#fff;text-decoration:none;padding:.833em}.h5p-hub .h5p-hub-tab-panel [role="tab"]>a:before{font-family:'h5p-hub';content:"";display:inline-block;speak:none;font-style:normal;font-weight:normal;font-variant:normal;text-transform:none;margin-right:.833em}.h5p-hub .h5p-hub-tab-panel [role="tab"]>a:focus:before{border-radius:50%;box-shadow:0 0 0.6em 0.1em #7bc1f9}.h5p-hub .h5p-hub-tab-panel [role="tab"][aria-selected='true']>a:before{content:""} - -.h5p-hub .h5p-hub-no-content{display:flex;box-shadow:0 -0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset}.h5p-hub .h5p-hub-no-content .h5p-hub-no-results-img{width:12.7em;height:4.95em;padding:3em 2em}@media (max-width: 576px){.h5p-hub .h5p-hub-no-content .h5p-hub-no-results-img{width:8em;padding:3em 1em}}.h5p-hub .h5p-hub-no-content .h5p-hub-right-text{padding:3em 2em}.h5p-hub .h5p-hub-no-content .h5p-hub-right-text .h5p-hub-header{font-size:1.2em;font-weight:bold;margin-bottom:0.4em}.h5p-hub .h5p-hub-no-content .h5p-hub-right-text .h5p-hub-suggestion-text{color:#4d4d4d;margin-bottom:0.3em}@media (max-width: 576px){.h5p-hub .h5p-hub-no-content .h5p-hub-right-text{padding:3em 1em}}.h5p-hub .h5p-hub-no-content .h5p-hub-right-text .h5p-hub-url{color:#186df7;font-weight:bold} - -.h5p-hub-list{overflow:auto}.h5p-hub-list ol{margin:1.25em 0;padding:0;list-style:none} - -.h5p-hub ul.h5p-hub-list-of-numbers{margin-top:1.5em;display:flex;flex-direction:row;border-radius:0.15146em;justify-content:center}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element{user-select:none;-ms-user-select:none;padding:0.17039em}@media (max-width: 576px){.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element{padding:0.25em;margin-bottom:1em}}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element a{padding:0.25em 0.75em;font-size:1em;font-weight:bold;text-decoration:none;outline:none;border-radius:0.15146em;border:0.07573em solid #ffffff;display:inline-block}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element a.h5p-hub-highlight{background-color:RGBA(209, 107, 23, 0.1);border:0.07573em solid #d16b17}@media (hover: hover){.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element a:hover{background-color:RGBA(209, 107, 23, 0.05);border:0.07573em solid #ffffff}}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element a:focus{box-shadow:0 0 0.4em 0.01em RGBA(26, 147, 244, 0.7)}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element.disabled{pointer-events:none}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element.disabled a{color:#b3b3b3}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-list-element.disabled a:hover{background-color:#ffffff}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-previous-arrow::after{font-family:'h5p-hub';content:"";transform:rotate(90deg);font-style:normal;font-size:0.7em;display:inline-block}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-next-arrow::before{font-family:'h5p-hub';content:"";transform:rotate(-90deg);font-style:normal;font-size:0.7em;display:inline-block}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-dots{padding:0.17039em;pointer-events:none;user-select:none;-ms-user-select:none}.h5p-hub ul.h5p-hub-list-of-numbers .h5p-hub-dots .h5p-hub-dots-text{padding:0.25em 0.75em} - - - -.h5p-hub .h5p-hub-content-item.h5p-hub-tabular{font-size:16px;display:flex;padding:0.5em 1em;border-bottom:1px solid #ececec;align-items:center;cursor:pointer}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-left{align-self:flex-start}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-middle{flex-grow:10;flex-shrink:10;margin-left:1em;overflow:hidden;min-width:0}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-content-icon{width:7em;height:5.25em;object-fit:contain;background-color:#d0d0d1}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-headline{line-height:1.2}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-title{font-size:1.1em;font-weight:bold;line-height:normal;letter-spacing:-0.5px}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-title.h5p-hub-reviewed:after{font-size:0.8em;font-family:'h5p-hub';font-weight:normal;content:'\e90b';color:#359d94;position:relative;top:-0.2em;margin-left:0.3em}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-by{font-size:1em;color:#666;margin:0 0.3em}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-owner{font-size:1em}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-content-type{font-size:0.9em;color:#303030;letter-spacing:0.65px;margin:0.25em 0}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-summary{color:#72757c;font-size:0.95em;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-right{flex-grow:0;flex-shrink:0;position:relative;font-size:1.5em;width:1.5em;height:1em;text-align:center}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-right:before{color:#72757c;display:inline-block;opacity:0;transition:opacity 0.3s, left 0.3s;font-family:'h5p-hub';content:'\e906';position:relative;left:-0.2em}@media (max-width: 576px){.h5p-hub .h5p-hub-content-item.h5p-hub-tabular .h5p-hub-right{display:none}}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular:hover,.h5p-hub .h5p-hub-content-item.h5p-hub-tabular:focus{background-color:#fafafa}.h5p-hub .h5p-hub-content-item.h5p-hub-tabular:hover .h5p-hub-right:before,.h5p-hub .h5p-hub-content-item.h5p-hub-tabular:focus .h5p-hub-right:before{opacity:1;left:0em} - -.h5p-hub .h5p-hub-content-item.h5p-hub-grid{display:inline-block;width:16.66%;margin-top:0.5em;outline:none}@media (max-width: 768px){.h5p-hub .h5p-hub-content-item.h5p-hub-grid{width:33.33%}.h5p-hub .h5p-hub-content-item.h5p-hub-grid .h5p-hub-content-icon{height:10.5em}}@media (max-width: 576px){.h5p-hub .h5p-hub-content-item.h5p-hub-grid{width:50%}.h5p-hub .h5p-hub-content-item.h5p-hub-grid .h5p-hub-content-icon{height:12em}}.h5p-hub .h5p-hub-content-item.h5p-hub-grid:focus .h5p-hub-grid-item,.h5p-hub .h5p-hub-content-item.h5p-hub-grid:hover .h5p-hub-grid-item{background-color:#fafafa}.h5p-hub .h5p-hub-content-item.h5p-hub-grid:focus .h5p-hub-grid-item{border-color:#5B9DD9}.h5p-hub .h5p-hub-grid-item{cursor:pointer;text-align:center;padding:0.5em;border:1px solid #ddd;margin:0 0.3em}.h5p-hub .h5p-hub-grid-item .h5p-hub-grid-item-title{font-weight:bold;letter-spacing:-0.5px;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden;text-overflow:ellipsis;height:3em}.h5p-hub .h5p-hub-grid-item img{width:100%;height:6em;object-fit:contain;background-color:#d0d0d1}.h5p-hub .h5p-hub-grid-item button{margin:0.5em 0 0.75em} - -.h5p-hub-list.h5p-hub-loading{pointer-events:none}.h5p-hub-list.h5p-hub-loading .h5p-hub-content-item{animation:opacitychange 1s infinite}.h5p-hub-list.h5p-hub-loading .h5p-hub-loading-block{background:#ccc;min-height:1.15em}.h5p-hub-list.h5p-hub-loading .h5p-hub-loading-block.h5p-hub-button{background:transparent;animation:none;border:2px solid #ccc}.h5p-hub-list.h5p-hub-loading .h5p-hub-middle{height:5.25em;display:flex;flex-direction:column;justify-content:space-between}.h5p-hub-list.h5p-hub-loading .h5p-hub-text-long{width:100%}.h5p-hub-list.h5p-hub-loading .h5p-hub-text-medium{width:60%}.h5p-hub-list.h5p-hub-loading .h5p-hub-text-short{width:30%}.h5p-hub-list.h5p-hub-loading .h5p-hub-grid-item{padding:1em}.h5p-hub-list.h5p-hub-loading .h5p-hub-grid-item .h5p-hub-image{height:8vw;min-height:5em}.h5p-hub-list.h5p-hub-loading .h5p-hub-grid-item .h5p-hub-loading-block{margin-bottom:0.5em}.h5p-hub-list.h5p-hub-loading .h5p-hub-grid-item .h5p-hub-loading-block.h5p-hub-button{margin-bottom:0;min-width:0;width:4em;max-width:40%}@keyframes opacitychange{0%{opacity:1}50%{opacity:0.6}100%{opacity:1}} - -.h5p-hub .h5p-hub-content-list{transition:opacity 0.2s, visibility 0s linear 0s}.h5p-hub .h5p-hub-content-list[aria-hidden="true"]{display:inherit;opacity:0;visibility:hidden;transition:opacity 0.2s, visibility 0s linear 0.2s}.h5p-hub .h5p-hub-content-list .h5p-hub-list.h5p-hub-tabular:before{content:" ";width:100%;height:calc(100% - 5.996455834em);position:absolute;pointer-events:none}.h5p-hub .h5p-hub-content-list .h5p-hub-list.h5p-hub-grid ol{margin:0}.h5p-hub .h5p-hub-content-list .h5p-hub-fetching-failed{display:inline-block;font-size:0.925em;border-radius:5px;margin-left:1em;padding:0.5em 0.85em;color:#c64750;background:rgba(198,71,80,0.1)} - -.h5p-hub .h5p-hub-navbar{background-color:transparent;list-style:none;color:#020202;border:0;padding:1.2em 1.2em 0.975422574em 2.292em;margin:0;font-size:1em;opacity:1;visibility:visible;transition:opacity 0.2s, visibility 0s linear 0s;overflow:auto;display:block}.h5p-hub .h5p-hub-navbar.h5p-hub-hidden{opacity:0;visibility:hidden;transition:opacity 0.2s, visibility 0s linear 0.2s}.h5p-hub .h5p-hub-result-header{font-size:1.042em;font-weight:600}.h5p-hub .h5p-hub-result-hits{font-size:0.8em;color:#697585;font-weight:400;padding-left:0.4em}.h5p-hub ul.h5p-hub-sort-by-list{margin-top:0.7844743em}.h5p-hub ul.h5p-hub-sort-by-list>li{float:left;margin-left:0.5em;user-select:none}.h5p-hub ul.h5p-hub-sort-by-list>li>a{display:block;color:#697585;padding:0.25em 0.75em;font-size:0.9em;text-decoration:none}.h5p-hub ul.h5p-hub-sort-by-list>li>a:hover{color:#3c434c}.h5p-hub ul.h5p-hub-sort-by-list>li>a:focus{box-shadow:0 0 1px 1px #1a93f4;outline:none}.h5p-hub ul.h5p-hub-sort-by-list>li>a.h5p-hub-highlight{font-weight:600;color:#000000;text-decoration:underline}.h5p-hub .h5p-hub-sort-by-header{float:left;padding:0.25em 0.75em 0.25em 0;font-size:0.9em;margin-top:0.871637586em} - -.h5p-hub-content-selection-list{padding:2em 1em 0}.h5p-hub-content-selection-list .h5p-hub-header{position:relative;font-weight:bold;padding:0 1em;margin-bottom:0.5em}.h5p-hub-content-selection-list .h5p-hub-title{font-size:1.2em}.h5p-hub-content-selection-list .h5p-hub-action{position:absolute;right:1.5em;top:0}.h5p-hub-content-selection-list:last-child{padding-bottom:1em} - -.h5p-hub .h5p-hub-filter-bar{background-color:#697585;font-size:1em;padding:0 0.833em 0.833em 1.125em;display:flex}.h5p-hub .h5p-hub-filter-bar .h5p-hub-filter-label{color:#ffffff;align-self:center}@media (max-width: 768px){.h5p-hub .h5p-hub-filter-bar{flex-direction:column}.h5p-hub .h5p-hub-filter-bar .h5p-hub-filter-label{align-self:start;margin-left:1em}}.h5p-hub .h5p-hub-filter-buttons{display:flex;flex-direction:row;flex-wrap:wrap}.h5p-hub .h5p-hub-filter-buttons li{outline:none;margin:0.5em 0 0.5em 0}.h5p-hub .h5p-hub-filter-buttons li:focus .h5p-hub-filter-button{box-shadow:0.06em 0 0.6em 0.1em #7bc1f9}.h5p-hub .h5p-hub-filter-buttons li:focus .h5p-hub-filter-button button{background-image:linear-gradient(0deg, #e3e3e3 0%, #fff 100%)}.h5p-hub .h5p-hub-clear-filters{flex-grow:1;align-self:center}@media (max-width: 768px){.h5p-hub .h5p-hub-clear-filters{align-self:start;margin-left:1em}}.h5p-hub .h5p-hub-clear-filters button{font-family:"Open Sans", sans-serif;align-items:center;float:right;display:flex;font-size:1em;background:none;color:#ffffff;font-size:0.88889em;cursor:pointer}.h5p-hub .h5p-hub-clear-filters button:focus{box-shadow:0.06em 0 0.6em 0.1em #7bc1f9;outline:none}.h5p-hub .h5p-hub-clear-filters button::before{font-family:'h5p-hub';content:"";font-style:normal;font-size:0.7em;margin-right:0.7em} - -.h5p-hub .h5p-hub-filter-modal{background:rgba(71,79,90,0.6);top:14.7em;width:100%;height:100%;left:0;position:absolute}.h5p-hub .h5p-hub-filter-dialog-content{padding:1.5em}.h5p-hub .h5p-hub-filter-dialog{position:absolute;width:40em;max-width:100%;box-sizing:border-box;background-color:#ffffff;border-radius:0.25em;box-shadow:2px 5px 5px 1px rgba(0,0,0,0.1)}.h5p-hub .h5p-hub-filter-dialog .h5p-hub-apply-filters-button{background-color:#697585;text-align:center;width:100%;color:#ffffff;font-size:1.15em;font-family:'Open Sans', sans-serif;padding:1em;border-radius:0.25em;margin-top:1em;cursor:pointer}.h5p-hub .h5p-hub-filter-dialog .h5p-hub-apply-filters-button::before{font-family:'h5p-hub';content:"";font-style:normal;font-size:1em;margin:0 0.4em 0 0.4em;margin-right:1em}.h5p-hub .h5p-hub-filter-dialog .h5p-hub-header-text{color:#000000;font-size:1.25em;font-weight:bold;margin-bottom:1em}.h5p-hub .h5p-hub-filter-dialog .h5p-hub-loading:after{font-family:'h5p-hub';content:"";font-size:1em;line-height:1;width:1em;height:1em;padding:.833em;display:block;color:#7c8696;-webkit-animation:spin 2s linear infinite;animation:spin 2s linear infinite;width:1em;margin:0 auto} - -.h5p-hub-message{position:relative;margin:.833em 0;padding:0.5em 1.250em;padding-right:2.250em}.h5p-hub-message .h5p-hub-title{font-size:1.042em;color:#1f2227}.h5p-hub-message .h5p-hub-message-content h2{margin-top:0.4165em;font-size:1.2em}.h5p-hub-message .h5p-hub-description{font-size:.95em;color:#697585}.h5p-hub-message .h5p-hub-message-header{margin:0;font-size:1.19em;line-height:1.9em}.h5p-hub-message .h5p-hub-message-body{font-size:0.925em;margin:0;padding-bottom:0.5em}.h5p-hub-message .h5p-hub-message-body ul.h5p-hub-message-item-list{list-style-type:disc;margin:1em}.h5p-hub-message .h5p-hub-message-content .h5p-hub-message-body{color:#697585}.h5p-hub-message .h5p-hub-get-help{margin-left:0.75em;color:#c62c15;text-decoration:none}.h5p-hub-message .h5p-hub-get-help:before{font-family:'h5p-hub';content:"";font-size:0.9em;margin-right:0.3em;display:inline-block}.h5p-hub-message .h5p-hub-message-close{cursor:pointer;display:block;position:absolute;background-color:transparent;color:#555;right:1.75em;line-height:2.75;font-size:0.75em;width:2.75em;text-align:center}.h5p-hub-message .h5p-hub-message-close:before{font-family:'h5p-hub';content:""}.h5p-hub-message .h5p-hub-message-close:hover{color:#333}.h5p-hub-message.h5p-hub-info{-webkit-box-shadow:0.167em 0 0 0 #00aa54 true;-moz-box-shadow:0.167em 0 0 0 #00aa54 true;box-shadow:0.167em 0 0 0 #00aa54 true;background:rgba(0,170,84,0.15);box-shadow:inset 0.167em 0 #00aa54}.h5p-hub-message.h5p-hub-info .h5p-hub-button{color:#00aa54;border:solid 1px;background-color:transparent;margin-top:1em}.h5p-hub-message.h5p-hub-warning{-webkit-box-shadow:0.167em 0 0 0 #fc3 true;-moz-box-shadow:0.167em 0 0 0 #fc3 true;box-shadow:0.167em 0 0 0 #fc3 true;background:rgba(255,204,51,0.15);box-shadow:inset 0.167em 0 #fc3}.h5p-hub-message.h5p-hub-warning .h5p-hub-button{color:#1f2227;border:solid 1px;background-color:transparent;margin-top:1em}.h5p-hub-message.h5p-hub-error{-webkit-box-shadow:0.167em 0 0 0 #c64750 true;-moz-box-shadow:0.167em 0 0 0 #c64750 true;box-shadow:0.167em 0 0 0 #c64750 true;background:rgba(198,71,80,0.15);box-shadow:inset 0.167em 0 #c64750}.h5p-hub-message.h5p-hub-error .h5p-hub-button{color:#c64750;border:solid 1px;background-color:transparent;margin-top:1em}.h5p-hub-message.h5p-hub-dismissible{overflow:auto;padding-right:3.5em} - -.h5p-hub .h5p-hub-filter-button{margin-left:0.85em}.h5p-hub .h5p-hub-filter-button button{font-family:'Open Sans', sans-serif;font-weight:600;padding:0.25em 0.426em;-ms-user-select:none;user-select:none;display:flex;align-items:center;font-size:1em;box-shadow:2px 2px 3px 1px rgba(0,0,0,0.1);border-radius:2px;background-image:linear-gradient(0deg, #e3e3e3 0%, #fff 100%)}.h5p-hub .h5p-hub-filter-button button.h5p-hub-open{background-image:linear-gradient(0deg, #c2c6cf 0%, #dde2e6 100%)}.h5p-hub .h5p-hub-filter-button button.h5p-hub-filter-checked{background-image:linear-gradient(0deg, #f4ddcd 0%, #fbfffc 100%)}.h5p-hub .h5p-hub-filter-button button.h5p-hub-one-checked{background-image:linear-gradient(0deg, #f4ddcd 0%, #fbfffc 100%)}.h5p-hub .h5p-hub-filter-button button:hover{background-image:linear-gradient(0deg, #c2c6cf 0%, #dde2e6 100%);cursor:pointer}.h5p-hub .h5p-hub-filter-button button .h5p-hub-icon{margin-left:0.3em}@media (min-width: 576px){.h5p-hub .h5p-hub-filter-button button .h5p-hub-icon::after{content:"\e565";font-family:"h5p";font-style:normal;font-size:0.8em;margin-left:0.6em}}.h5p-hub .h5p-hub-filter-button button .h5p-hub-icon.h5p-hub-check::before{font-family:'h5p-hub';content:"";font-style:normal;font-size:0.7em;margin:0 0.4em 0 0.4em} - -.h5p-hub .h5p-hub-checkbox-list{list-style:none}.h5p-hub .h5p-hub-checkbox-list.h5p-hub-animate-in-right{animation:move-from-right 0.35s}.h5p-hub .h5p-hub-checkbox-list.h5p-hub-animate-in-left{animation:move-from-left 0.35s}@keyframes move-from-right{0%{opacity:0;transform:translateX(30%)}30%{opacity:0}100%{opacity:1;transform:translateX(0)}}@keyframes move-from-left{0%{opacity:0;transform:translateX(-10%)}30%{opacity:0}100%{opacity:1;transform:translateX(0)}} - -.h5p-hub .h5p-hub-checkbox{font-size:1.05em;padding:0.5em 0;font-weight:bold;display:flex;cursor:pointer;align-self:start;color:#4d5057;width:100%;position:relative}.h5p-hub .h5p-hub-checkbox:focus{box-shadow:0 0 0.4em 0.01em RGBA(26, 147, 244, 0.7);outline:none}.h5p-hub .h5p-hub-checkbox.h5p-hub-parent::after{font-family:'h5p-hub';content:"";font-family:"h5p";transform:rotate(90deg);font-style:normal;font-size:1.1em;display:block;align-self:center}.h5p-hub .h5p-hub-checkbox .h5p-hub-non-bold{font-weight:normal}.h5p-hub .h5p-hub-checkbox .h5p-hub-content{display:flex}.h5p-hub .h5p-hub-checkbox .h5p-hub-content .h5p-hub-icon::before{font-family:'h5p-hub';content:"";font-style:normal;font-weight:normal;font-size:1em;margin:0.4em 0.4em 0 0.4em;margin-right:1em;color:#697585}.h5p-hub .h5p-hub-checkbox.h5p-hub-checked .h5p-hub-content .h5p-hub-icon::before{font-family:'h5p-hub';content:"";font-style:normal;font-weight:normal;font-size:1em;margin:0.4em 0.4em 0 0.4em;margin-right:1em;color:#697585} - -.h5p-hub .h5p-hub-bottom-line{border-bottom:2px solid #697585}.h5p-hub .h5p-hub-category-header{background-color:#f2f2f2;color:#313131;font-style:italic;padding:0 0 0 0.9em;font-size:0.81788em;text-transform:capitalize} - -.h5p-hub .h5p-hub-search-button{width:100%}.h5p-hub .h5p-hub-search-button .h5p-hub-search-field{position:relative}.h5p-hub .h5p-hub-search-button .h5p-hub-search-field::before{position:absolute;font-family:'h5p-hub';content:"";font-style:normal;font-size:1em;margin-left:1em;color:#b7b8ba;top:50%;transform:translateY(-50%)}.h5p-hub .h5p-hub-search-button .h5p-hub-search-field .h5p-hub-icon-arrow{position:absolute;font-size:1.30353em;top:0.7em;right:1em;cursor:pointer;color:#000000}.h5p-hub .h5p-hub-search-button .h5p-hub-search-field .h5p-hub-icon-arrow::before{content:"\e565";font-family:"h5p";font-style:normal;font-size:1.25em}.h5p-hub .h5p-hub-filter-search-bar{border:0.04545em solid #e6e6e8;padding:0.7em;font-style:italic;font-size:1.25em;width:100%;box-sizing:border-box;padding-left:2.25em}.h5p-hub .h5p-hub-filter-search-bar::placeholder{color:#b7b8ba}.h5p-hub .h5p-hub-filter-search-bar:focus{border-color:#3ba0f2;background-color:#fff;outline:none} - -.h5p-hub .h5p-hub-search-filter{outline:0.06815em solid #bebdcd;position:relative}.h5p-hub .h5p-hub-search-filter .h5p-hub-clear-button{background:none;position:absolute;top:0;padding:1.2em;right:3em;color:#757575}.h5p-hub .h5p-hub-search-filter .h5p-hub-clear-button::before{font-family:'h5p-hub';content:"";font-style:normal;font-size:1em}.h5p-hub .h5p-hub-search-filter .h5p-hub-navigate-parent{font-size:1.25em;padding:0.3534em;display:flex;font-weight:bold;font-family:"Open Sans", sans-serif;border:1px solid #e6e6e8;align-items:center}.h5p-hub .h5p-hub-search-filter .h5p-hub-navigate-parent button{background-color:#ffffff;position:relative;font-style:normal;font-size:0.8em;border-radius:50%;margin-right:0.4em;align-self:center;padding:0.5em;width:2.75em;height:2.75em}.h5p-hub .h5p-hub-search-filter .h5p-hub-navigate-parent button::before{font-family:'h5p-hub';content:"";transform:rotate(180deg);font-weight:normal;display:inline-block}.h5p-hub .h5p-hub-search-filter .h5p-hub-navigate-parent button:hover{background-color:#dadada;cursor:pointer}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list{overflow:hidden;overflow-y:scroll;max-height:25em}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li{border:1px solid #e6e6e8}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li .h5p-hub-content{padding:0.3em}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li .h5p-hub-content .h5p-hub-label-text{margin-right:0.2em}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li:focus{border-color:#3ba0f2;background-color:#fff;outline:none;box-shadow:none}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li:hover{background-color:#f2f2f2}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li.h5p-hub-highlighted{border-color:#3ba0f2;background-color:#fff;box-shadow:none}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list li:hover{background-color:rgba(230,230,232,0.25)}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list::-webkit-scrollbar{background-color:rgba(105,117,133,0.25);width:0.54525em;border-radius:5.45256em}.h5p-hub .h5p-hub-search-filter .h5p-hub-checkbox-list::-webkit-scrollbar-thumb{background-color:#697585;background-size:24px 100%;border-radius:5.45256em} - -.h5p-hub .h5p-hub-downloading-modal-overlay{position:absolute;top:0;left:0;right:0;bottom:0;background-color:rgba(255,255,255,0.8)}.h5p-hub .h5p-hub-downloading-modal{position:absolute;width:30rem;max-width:90%;border-radius:0.25rem;background-color:#ffffff;box-shadow:0px 1px 82px 0px rgba(0,0,0,0.2);left:50%;top:50%;transform:translate(-50%, -50%);transition:background-color 0.3s ease-in;display:flex;flex-wrap:wrap;flex-direction:row;justify-content:center;align-items:center;outline:none;box-sizing:border-box;padding:1em}.h5p-hub .h5p-hub-downloading-modal .h5p-hub-spinner{margin:1em;animation:spin 0.5s linear infinite}.h5p-hub .h5p-hub-downloading-modal span.h5p-hub-downloading-message{font-family:'Open Sans', sans-serif;font-weight:bold;font-size:2rem}.h5p-hub .h5p-hub-downloading-modal span.h5p-hub-downloading-message:focus{outline:none}@keyframes spin{from{transform:rotate(0deg)}to{transform:rotate(360deg)}} - -.h5p-hub-search-wrapper{padding:.833em 1.125em;background-color:#697585;position:relative}.h5p-hub-search-wrapper .h5p-hub-border-wrap{padding:3px;background:#fff;border-radius:1.25em;font-size:1.15em}.h5p-hub-search-wrapper .h5p-hub-icon-search{position:absolute;font-size:1em;top:1.4em;right:2.2em;cursor:pointer;color:#4f5662}.h5p-hub-search-bar{border:1px solid #bfc0c0;padding:0.5em 0.5em 0.5em 1.2em;font-style:italic;font-weight:400;font-size:1em;box-shadow:inset 0 0 10px rgba(0,0,0,0.4);color:#000000;border-radius:1.25em;width:100%;box-sizing:border-box}.h5p-hub-search-bar::placeholder{color:#000000}.h5p-hub-search-bar:focus{border-color:#3ba0f2;background-color:#fff;box-shadow:inset 0 0 10px rgba(0,0,0,0.2);outline:none}.h5p-hub-search-bar:focus::placeholder{color:#697585}.h5p-hub-search-bar:before{background:none;border:4px solid #fff;content:"";display:block;position:absolute;top:4px;left:4px;right:4px;bottom:4px;pointer-events:none}.h5p-hub-search-bar::-ms-clear{display:none;width:0;height:0} - -.h5p-hub .h5p-hub-button.h5p-hub-button-orange{background:transparent;color:#d16b17;border:0.125em solid #d16b17}.h5p-hub .h5p-hub-button.h5p-hub-button-orange:hover{color:#fff;background-color:#d16b17;border-color:#d16b17}.h5p-hub .h5p-hub-button.h5p-hub-button-orange:active{color:#fff;background-color:#c26d13;border-color:#c26d13} - -.h5p-hub-lightbox{position:absolute;top:0;left:0;right:0;bottom:0;background:rgba(0,0,0,0.8)}.h5p-hub-lightbox-inner{position:absolute;left:50%;top:50%;transform:translate(-50%, -50%);width:100%;box-sizing:border-box;padding:2em;text-align:center}.h5p-hub-lightbox-inner:focus{outline:0}.h5p-hub-lightbox-close{position:relative;top:-1.5em;right:0;float:right;width:1.2em;height:1.2em;border-radius:50%;color:#fff;font-size:2em;line-height:1;text-align:center;cursor:pointer;z-index:1;user-select:none;border:2px solid transparent;transition:background 0.2s}.h5p-hub-lightbox-close:focus{border-color:#7bc1f9;box-shadow:0.06em 0 0.6em 0.1em #7bc1f9;outline:0}.h5p-hub-lightbox-close:hover{background:rgba(132,143,158,0.8)}.h5p-hub-license .h5p-hub-lightbox-close:hover{background:rgba(165,175,189,0.43)}.h5p-hub-lightbox-close:before{font-family:'h5p-hub';content:"";font-size:0.6em} - -.h5p-hub-carousel{width:100%;box-sizing:border-box;position:relative}.h5p-hub-carousel .h5p-hub-scroller{overflow:hidden;width:calc(100% - 5em);margin-left:2.5em}.h5p-hub-carousel ul{width:100%;padding:0;margin:0 auto;word-wrap:break-word;transition:margin 0.3s linear;display:flex;flex-direction:row;flex-wrap:nowrap;align-items:center}.h5p-hub-carousel li{padding:0 .3125em;width:20%;margin:0;list-style-type:none;float:left;box-sizing:border-box;overflow:hidden}.h5p-hub-carousel li img{max-width:100%;box-sizing:border-box;border:1px solid #ced6e3}.h5p-hub-carousel li img.h5p-hub-selectable{cursor:pointer}.h5p-hub-carousel li:focus{outline:0}.h5p-hub-carousel li:focus img{border-color:#0a78d1}.h5p-hub-carousel .h5p-hub-progress{color:#fff;position:relative;bottom:-2em;text-align:center}.h5p-hub-carousel .h5p-hub-navigation{width:2.1em;height:2.1em;line-height:2.1em;position:absolute;border-radius:50%;color:white;background-color:rgba(105,117,133,0.8);cursor:pointer;border:2px solid transparent;top:50%;transform:translateY(-50%)}.h5p-hub-carousel .h5p-hub-navigation[disabled]{background-color:rgba(189,195,203,0.8);color:rgba(255,255,255,0.6);pointer-events:none}.h5p-hub-carousel .h5p-hub-navigation:before{display:inline-block;line-height:1em;height:1em}.h5p-hub-carousel .h5p-hub-navigation:hover{background-color:rgba(132,143,158,0.8)}.h5p-hub-carousel .h5p-hub-navigation:focus{outline:0;box-shadow:0.06em 0 0.6em 0.1em #7bc1f9;border-color:#7bc1f9}.h5p-hub-carousel .h5p-hub-prev{left:0}.h5p-hub-carousel .h5p-hub-prev:before{font-family:'h5p-hub';content:""}.h5p-hub-carousel .h5p-hub-next{right:0}.h5p-hub-carousel .h5p-hub-next:before{font-family:'h5p-hub';content:"";transform:rotate(180deg)}.h5p-hub-lightbox-inner .h5p-hub-carousel ul{transition-duration:0.15s}.h5p-hub-lightbox-inner .h5p-hub-carousel li{padding:0}.h5p-hub-lightbox-inner .h5p-hub-carousel li img{border-color:transparent;background:#fff}.h5p-hub-lightbox-inner .h5p-hub-carousel .h5p-hub-navigation{font-size:1.2em}.h5p-hub-lightbox-inner .h5p-hub-carousel .h5p-hub-navigation:focus{background-color:transparent}.h5p-hub-lightbox-inner .h5p-hub-carousel .h5p-hub-next{right:-0.75em}.h5p-hub-lightbox-inner .h5p-hub-carousel .h5p-hub-prev{left:-0.75em} - -.h5p-hub-text-details .h5p-read-more span{outline:none}.h5p-hub-text-details .h5p-read-more .h5p-hub-link{color:#1f2227;outline:none;margin-left:0.5em}.h5p-hub-text-details .h5p-read-more .h5p-hub-link:hover{color:#000}.h5p-hub-text-details .h5p-read-more .h5p-hub-link:focus{box-shadow:0 0 1px 1px #1a93f4} - -.h5p-hub-accordion{border:1px solid #ced6e3;border-bottom:0}.h5p-hub-accordion .h5p-hub-icon-accordion-arrow{margin-top:-4px;font-size:0.7em;margin-right:.833em;margin-left:.833em;display:inline-block;vertical-align:middle;transition:transform 0.1s ease-out}.h5p-hub-accordion [aria-expanded="true"] .h5p-hub-icon-accordion-arrow{transform:rotate(90deg)}.h5p-hub-accordion-toggler{display:block;box-sizing:border-box;text-decoration:none;width:100%;padding:.833em;color:#1f2227}.h5p-hub-accordion-heading{border-bottom:1px solid #ced6e3;background-color:#f3f3f3;cursor:pointer}.h5p-hub-accordion-region{border-bottom:1px solid #ced6e3}.h5p-hub-accordion-region .h5p-hub-panel-body{padding:1em;line-height:1.7em;font-size:0.95em} - -.h5p-hub .publisher-info{overflow:auto}.h5p-hub .publisher-info .h5p-hub-publisher-image{max-width:10em;max-height:10em;height:auto;width:auto;margin-right:1em;float:left}.h5p-hub .h5p-hub-publisher-image{max-width:150px} - -.h5p-hub .h5p-hub-content-detail{background-color:white;padding:1.3em;display:inherit;position:absolute;top:0;left:0;visibility:hidden;outline:none;overflow:auto;width:100%;height:100%;box-sizing:border-box;transform:translateX(100%);transition:transform 0.2s ease-out, visibility 0s linear 0.2s}.h5p-hub .h5p-hub-content-detail::-webkit-scrollbar{background-color:#f3f3f3;width:0.750em}.h5p-hub .h5p-hub-content-detail::-webkit-scrollbar-thumb{background:#5e6978 url("") no-repeat center;background-size:24px 100%}.h5p-hub .h5p-hub-content-detail::-webkit-scrollbar-button:vertical{background-size:20px 20px;background-position-x:-4px;background-position-y:-3px;background-color:#5e6978;background-repeat:no-repeat}.h5p-hub .h5p-hub-content-detail::-webkit-scrollbar-button:vertical:decrement{background-image:url("")}.h5p-hub .h5p-hub-content-detail::-webkit-scrollbar-button:vertical:increment{background-image:url("")}.h5p-hub .h5p-hub-content-detail.h5p-hub-show{transform:translateX(0);visibility:visible;transition:transform 0.2s ease-out, visibility 0s linear 0s}.h5p-hub .h5p-hub-content-detail .h5p-hub-panel[aria-hidden="false"]{display:none}.h5p-hub .h5p-hub-content-detail .h5p-hub-container{max-width:100%;margin-left:auto;margin-right:auto;margin-top:.833em;margin-bottom:1.666em;padding-bottom:1em;display:flex;flex-wrap:wrap;flex-direction:column}.h5p-hub .h5p-hub-content-detail .h5p-hub-container:after{content:" ";display:block;clear:both}@media (min-width: 576px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container{flex-direction:row}}.h5p-hub .h5p-hub-content-detail .h5p-hub-container .h5p-hub-image-wrapper{flex-shrink:0;order:1;display:flex;justify-content:center;align-items:center}@media (min-width: 768px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container .h5p-hub-image-wrapper{justify-content:normal;align-items:normal;margin-top:2em}}.h5p-hub .h5p-hub-content-detail .h5p-hub-container .h5p-hub-image-wrapper img{width:16.5em;height:12.375em}@media (min-width: 768px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container .h5p-hub-image-wrapper img{width:15.5em;height:11.625em}}.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details{order:2;flex-grow:1;margin:2em}.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details .h5p-hub-info-list{order:2}@media (min-width: 576px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details{order:3;flex-basis:100%}.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details .h5p-hub-info-list{display:none}}@media (min-width: 768px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details{order:2;flex-basis:0;margin:2em 2em 0}}.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-text-details h2{margin-top:0;font-size:1.458em;line-height:1.458em}.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-info-list{order:3;flex-shrink:0;min-width:15em;margin:1em;display:none}@media (min-width: 576px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-info-list{display:block;order:2;flex-grow:1}}@media (min-width: 768px){.h5p-hub .h5p-hub-content-detail .h5p-hub-container>.h5p-hub-info-list{order:3;flex-grow:0}}.h5p-hub .h5p-hub-content-detail .h5p-hub-title{outline:0}.h5p-hub .h5p-hub-content-detail .h5p-hub-title.h5p-hub-reviewed:after{font-size:0.8em;font-family:"h5p-hub";font-weight:normal;content:"\e90b";color:#359d94;position:relative;top:-0.2em;margin-left:0.3em}.h5p-hub .h5p-hub-content-detail .h5p-hub-back-button{color:#474f5a;font-size:1.3em;display:inline-block;background:transparent;border:none;text-decoration:none;outline:none;transform:rotate(180deg)}.h5p-hub .h5p-hub-content-detail .h5p-hub-back-button:hover{color:#000}.h5p-hub .h5p-hub-content-detail .h5p-hub-back-button:focus{box-shadow:0 0 1px 1px #1a93f4}.h5p-hub .h5p-hub-content-detail .h5p-hub-back-button.h5p-hub-icon-arrow-thick:before{cursor:pointer}.h5p-hub .h5p-hub-content-detail .h5p-hub-button-bar{text-align:right;margin:1.666em 0}.h5p-hub .h5p-hub-content-detail .h5p-hub-button-download-content{display:inline-flex;align-items:center}.h5p-hub .h5p-hub-content-detail .h5p-hub-button-download-content::before{font-family:"h5p-hub";font-size:0.7em;content:"\e90c";display:inline-block;margin-right:1em;transform:rotate(-90deg)}.h5p-hub .h5p-hub-content-detail .h5p-hub-lightbox-inner.h5p-hub-license::-webkit-scrollbar{background-color:#f3f3f3;width:0.750em}.h5p-hub .h5p-hub-content-detail .h5p-hub-lightbox-inner.h5p-hub-license::-webkit-scrollbar-thumb{background:#5e6978 url("") no-repeat center;background-size:24px 100%}.h5p-hub .h5p-hub-content-detail .h5p-hub-lightbox-inner.h5p-hub-license::-webkit-scrollbar-button:vertical{background-size:20px 20px;background-position-x:-4px;background-position-y:-3px;background-color:#5e6978;background-repeat:no-repeat}.h5p-hub .h5p-hub-content-detail .h5p-hub-lightbox-inner.h5p-hub-license::-webkit-scrollbar-button:vertical:decrement{background-image:url("")}.h5p-hub .h5p-hub-content-detail .h5p-hub-lightbox-inner.h5p-hub-license::-webkit-scrollbar-button:vertical:increment{background-image:url("")} - -.h5p-hub .h5p-hub-info-list{max-width:20em}.h5p-hub .h5p-hub-info-list li{font-size:0.8em;border-bottom:1px solid #dadada;padding:0.85em 0.5em}.h5p-hub .h5p-hub-info-list li .h5p-hub-label{color:#72757c;margin-right:0.3em}.h5p-hub .h5p-hub-info-list li .h5p-hub-content{color:#242424;font-weight:bold}.h5p-hub .h5p-hub-info-list li .h5p-hub-capitalize{text-transform:capitalize}.h5p-hub .h5p-hub-info-list li:last-child{border-bottom:none} - -.h5p-hub .h5p-hub-short-license-info h3{display:inline-block}.h5p-hub .h5p-hub-short-license-info .h5p-hub-short-license-read-more{border:0;background-color:transparent;cursor:pointer}.h5p-hub .h5p-hub-short-license-info a.h5p-hub-short-license-read-more{text-decoration:none}.h5p-hub .h5p-hub-short-license-info .h5p-hub-short-license-read-more:before{font-family:'h5p-hub';content:"";margin-left:0.5em} - -.h5p-hub .h5p-hub-reuse-content-result{position:relative;overflow-y:auto;overflow-x:hidden;max-height:56em;box-shadow:0 0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset}.h5p-hub .h5p-hub-reuse-content-result::-webkit-scrollbar{background-color:#F3F3F3;width:0.750em}.h5p-hub .h5p-hub-reuse-content-result::-webkit-scrollbar-thumb{background:#5e6978 url("") no-repeat center;background-size:24px 100%}.h5p-hub .h5p-hub-reuse-content-result::-webkit-scrollbar-button:vertical{background-size:20px 20px;background-position-x:-4px;background-position-y:-3px;background-color:#5e6978;background-repeat:no-repeat}.h5p-hub .h5p-hub-reuse-content-result::-webkit-scrollbar-button:vertical:decrement{background-image:url("")}.h5p-hub .h5p-hub-reuse-content-result::-webkit-scrollbar-button:vertical:increment{background-image:url("")}.h5p-hub .h5p-hub-reuse-content-result>.h5p-hub-content-list .h5p-hub-fetching-failed{margin-top:2em;margin-left:2em}.h5p-hub .h5p-hub-reuse-content-container{position:relative}.h5p-hub .h5p-hub-content-list-hidden{visibility:hidden} - -.h5p-hub .h5p-hub-content-type-list .h5p-hub-media .h5p-hub-media-body{overflow:visible;zoom:1;width:10000px}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-left{display:table-cell;vertical-align:top}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-body{display:table-cell;vertical-align:middle}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-left{padding-right:.833em}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-object{width:50px}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-heading{margin-top:0;float:left;margin-right:0.5em}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-text-icon{float:left;height:15px}.h5p-hub .h5p-hub-content-type-list .h5p-hub-button{min-width:4em;padding:0.65em 1.5em;float:right}.h5p-hub .h5p-hub-content-type-list .h5p-hub-description{color:#697585;font-size:.95em;float:left;clear:left}.h5p-hub .h5p-hub-content-type-list .h5p-hub-content-type-update-info{margin-right:0;color:#697585;font-size:0.8em;font-style:italic;float:left}.h5p-hub .h5p-hub-content-type-list .h5p-hub-content-type-update-info:before{font-family:'h5p-hub';content:"";font-style:normal;font-size:0.8em;margin-right:0.4em}.h5p-hub .h5p-hub-content-type-list .h5p-hub-button-install:before{font-family:'h5p-hub';content:"";display:inline-block;transform:rotate(-90deg);margin-right:0.7em;font-size:0.9em}@media (min-width: 768px){.h5p-hub .h5p-hub-content-type-list .h5p-hub-button{margin-top:0.25em}.h5p-hub .h5p-hub-content-type-list .h5p-hub-content-type-update-info{margin-right:2em;margin-top:1em;float:right}}@media (min-width: 576px){.h5p-hub .h5p-hub-content-type-list .h5p-hub-media-object{width:100px}} - -.h5p-hub .h5p-hub-content-type-list{overflow:auto;height:37.422137em;opacity:1;transition:opacity 0.2s, visibility 0s linear 0s}.h5p-hub .h5p-hub-content-type-list[aria-hidden="true"]{display:inherit;opacity:0;visibility:hidden;transition:opacity 0.2s, visibility 0s linear 0.2s}.h5p-hub .h5p-hub-content-type-list::-webkit-scrollbar{background-color:#F3F3F3;width:0.750em}.h5p-hub .h5p-hub-content-type-list::-webkit-scrollbar-thumb{background:#5e6978 url("") no-repeat center;background-size:24px 100%}.h5p-hub .h5p-hub-content-type-list::-webkit-scrollbar-button:vertical{background-size:20px 20px;background-position-x:-4px;background-position-y:-3px;background-color:#5e6978;background-repeat:no-repeat}.h5p-hub .h5p-hub-content-type-list::-webkit-scrollbar-button:vertical:decrement{background-image:url("")}.h5p-hub .h5p-hub-content-type-list::-webkit-scrollbar-button:vertical:increment{background-image:url("")}.h5p-hub .h5p-hub-content-type-list:before{content:" ";width:100%;box-shadow:0 0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset,0 -0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset;height:calc(100% - 5.996455834em);position:absolute;pointer-events:none}.h5p-hub .h5p-hub-content-type-list ol{margin:1.25em 0;padding:0;list-style:none}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media{width:100%;padding:0.5em 1.5em 0.5em 1em;box-sizing:border-box;cursor:pointer;overflow:visible;outline:none;margin-top:0}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media:first-child{margin-top:0}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media:hover{background-color:#f5faff}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media:focus{box-shadow:0 0 1px 1px #1a93f4}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media:not(:last-child){border-bottom:1px solid #ececec}.h5p-hub .h5p-hub-content-type-list .h5p-hub-media[tabindex="0"]{background-color:#f5faff}.h5p-hub .h5p-hub-filtering .h5p-hub-content-type-list:before{height:calc(100% - 3.536668484em)}.h5p-hub-no-results{background:no-repeat 4em 0 url();background-size:16% 100%;padding:0 1em 0.5em 17em;margin-top:2em}.h5p-hub-no-results-title{margin-bottom:0.4em;font-size:1.2em;font-weight:bold}.h5p-hub-no-results-desc{color:#4d4d4d} - -.h5p-hub .h5p-hub-panel.h5p-section-content-types{max-height:3.0em;overflow:hidden} - -.h5p-hub-content-type-detail-button-bar{margin:1.666em 0;text-align:right}.h5p-hub-content-type-detail-button-bar .h5p-hub-button-update,.h5p-hub-content-type-detail-button-bar .h5p-hub-button-updating{margin-right:0.6em}.h5p-hub-content-type-detail-button-bar .h5p-hub-button-install:before,.h5p-hub-content-type-detail-button-bar .h5p-hub-button-installing:before,.h5p-hub-content-type-detail-button-bar .h5p-hub-button-update:before,.h5p-hub-content-type-detail-button-bar .h5p-hub-button-updating:before{display:inline-block;margin-right:0.7em;font-weight:normal;font-size:0.9em}.h5p-hub-content-type-detail-button-bar .h5p-hub-button-install:before{font-family:'h5p-hub';content:"";transform:rotate(-90deg)}.h5p-hub-content-type-detail-button-bar .h5p-hub-button-update:before{font-family:'h5p-hub';content:""}.h5p-hub-content-type-detail-button-bar .h5p-hub-button-installing:before,.h5p-hub-content-type-detail-button-bar .h5p-hub-button-updating:before{font-family:'h5p-hub';content:"";-webkit-animation:spin 2s linear infinite;animation:spin 2s linear infinite} - -.h5p-hub .h5p-hub-content-type-detail{background-color:white;padding:1.3em;display:inherit;position:absolute;top:0;left:0;visibility:hidden;outline:none;overflow:auto;width:100%;height:100%;box-sizing:border-box;transform:translateX(100%);transition:transform 0.2s ease-out, visibility 0s linear 0.2s;box-shadow:0 0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset,0 -0.4em 0.4em -0.2em rgba(79,87,99,0.2) inset}.h5p-hub .h5p-hub-content-type-detail::-webkit-scrollbar{background-color:#F3F3F3;width:0.750em}.h5p-hub .h5p-hub-content-type-detail::-webkit-scrollbar-thumb{background:#5e6978 url("") no-repeat center;background-size:24px 100%}.h5p-hub .h5p-hub-content-type-detail::-webkit-scrollbar-button:vertical{background-size:20px 20px;background-position-x:-4px;background-position-y:-3px;background-color:#5e6978;background-repeat:no-repeat}.h5p-hub .h5p-hub-content-type-detail::-webkit-scrollbar-button:vertical:decrement{background-image:url("")}.h5p-hub .h5p-hub-content-type-detail::-webkit-scrollbar-button:vertical:increment{background-image:url("")}.h5p-hub .h5p-hub-content-type-detail.h5p-hub-show{transform:translateX(0);visibility:visible;transition:transform 0.2s ease-out, visibility 0s linear 0s}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-panel[aria-hidden="false"]{display:none}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-container{max-width:100%;margin-left:auto;margin-right:auto;margin-bottom:1.666em;padding-bottom:1em}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-container:after{content:" ";display:block;clear:both}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-image-wrapper{width:100%;float:left;margin-left:0;margin-right:0}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-title{outline:0}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-text-details{width:100%;float:left;margin-left:0;margin-right:0}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-text-details h2{margin-top:0}@media (min-width: 576px){.h5p-hub .h5p-hub-content-type-detail .h5p-hub-image-wrapper{width:32.20339%;float:left;margin-right:1.69492%}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-text-details{width:66.10169%;float:right;margin-right:0}}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-button-bar{margin:1.666em 0;text-align:right}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-button-bar .h5p-hub-button-update,.h5p-hub .h5p-hub-content-type-detail .h5p-hub-button-bar .h5p-hub-button-updating{margin-right:0.6em}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-button-bar .h5p-hub-button-update:before{font-family:'h5p-hub';content:"";margin-right:0.5em;font-weight:normal;font-size:0.9em}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-back-button{color:#474f5a;font-size:1.3em;display:inline-block;background:transparent;border:none;text-decoration:none;outline:none;transform:rotate(180deg)}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-back-button:hover{color:#000}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-back-button:focus{box-shadow:0 0 1px 1px #1a93f4}.h5p-hub .h5p-hub-content-type-detail .h5p-hub-back-button.h5p-hub-icon-arrow-thick:before{cursor:pointer}.h5p-hub-lightbox-inner.h5p-hub-license{background:#fff;text-align:left;max-height:calc(100% - 5em);width:calc(100% - 5em)}.h5p-hub-lightbox-inner.h5p-hub-license :not(.h5p-hub-loading) .h5p-hub-modal-content{max-height:50em;overflow-y:auto}.h5p-hub-lightbox-inner.h5p-hub-license .h5p-hub-loading:after{font-family:'h5p-hub';content:"";font-size:3em;line-height:1;width:1em;height:1em;padding:.833em;display:block;color:#7c8696;-webkit-animation:spin 2s linear infinite;animation:spin 2s linear infinite;width:1em;margin:0 auto}.h5p-hub-lightbox-inner.h5p-hub-license .h5p-hub-lightbox-close{top:-0.8em;right:-0.8em;color:#333}.h5p-hub-lightbox-inner.h5p-hub-license .h5p-hub-modal-header{font-weight:400;font-size:1.6em;margin-bottom:1em}.h5p-hub-lightbox-inner.h5p-hub-license .h5p-hub-modal-title{font-weight:700} - -.h5p-hub .h5p-hub-upload-wrapper{padding:1em 2em 2em 2em}.h5p-hub .h5p-hub-upload-wrapper .h5p-hub-message{margin-top:0}.h5p-hub .h5p-hub-upload-wrapper .h5p-hub-upload-throbber{outline:0;height:10em}.h5p-hub .h5p-hub-upload-wrapper .h5p-hub-upload-throbber.h5p-hub-hidden{display:none}.h5p-hub .h5p-hub-upload-wrapper .h5p-hub-upload-throbber:before{font-family:'h5p-hub';content:"";font-size:4em;line-height:1;width:1em;height:1em;padding:.833em;display:block;color:#7c8696;-webkit-animation:spin 2s linear infinite;animation:spin 2s linear infinite;margin:auto}.h5p-hub .h5p-hub-upload-wrapper a:focus{outline:auto}.h5p-hub h1.h5p-hub-upload-instruction-header{font-size:1.25em;margin-top:.833em;margin-bottom:.833em}.h5p-hub p.h5p-hub-upload-instruction-description{font-size:1.042em} - -.h5p-hub .h5p-hub-upload-form .h5p-hub-input-wrapper{display:inline-block}.h5p-hub .h5p-hub-upload-form .h5p-hub-input-wrapper input[type="file"]{display:none}.h5p-hub .h5p-hub-upload-form [aria-hidden="true"]{display:none}.h5p-hub .h5p-hub-upload-form .h5p-hub-upload-button{background:transparent;color:#697585;border:0.125em solid #697585;margin-left:0.5em}.h5p-hub .h5p-hub-upload-form .h5p-hub-use-button{background:#697585;color:white;margin-left:0.5em}.h5p-hub .h5p-hub-upload-form button[disabled]{opacity:0.5}.h5p-hub .h5p-hub-upload-form .h5p-hub-upload-path{display:inline-block;min-width:14em;padding:10px;padding-left:1.5em;margin-bottom:1em;border:0.125em solid #ced6e3;border-radius:1.375em;outline:0;color:#697585;cursor:pointer}.h5p-hub .h5p-hub-upload-form .h5p-hub-upload-path[disabled]{cursor:default}@media (min-width: 576px){.h5p-hub .h5p-hub-upload-form .h5p-hub-upload-path{width:60%}} - -.h5p-hub{font-family:'Open Sans', sans-serif;padding-top:1px;padding-bottom:.833em;position:relative;overflow:hidden}.h5p-hub .h5p-hub-panel{border:1px solid #697585;background:#fff}.h5p-hub .h5p-hub-panel.h5p-hub-section-content-types{border-color:#0a78d1}.h5p-hub .h5p-hub-panel.h5p-hub-section-reuse{border-color:#db6f28}.h5p-hub .h5p-hub-panel.h5p-hub-section-upload{border-color:#ced6e3}.h5p-hub .h5p-hub-ssection-content-types.open{max-height:58.65em}.h5p-hub .h5p-hub-button-install [class^="icon-"]{margin-right:.4165em;display:inline-block}.h5p-hub .h5p-hub-button-install .h5p-hub-icon-arrow-thick{transform:rotate(-90deg)}.h5p-hub .h5p-hub-content-type-section{position:relative;overflow:hidden}.h5p-hub .h5p-hub-content-type-section.h5p-hub-height-limit{height:34em}.h5p-hub .h5p-hub-content-type-section-view>.h5p-hub-message{margin:0;padding:0.5em 3.5em 0.7em 2.292em}.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button{margin:0.5em 0 0.75em;border-width:2px}.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button:hover,.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button:active{background-color:#c64750;border-color:#c64750;color:#fff}.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button:focus{box-shadow:0.06em 0 0.6em 0.1em #c64750}.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button[disabled]{box-shadow:none;background-color:#c64750;border-color:#c64750;color:#fff;opacity:0.5;padding-left:3em}.h5p-hub .h5p-hub-error .h5p-hub-button.h5p-hub-retry-button[disabled]:before{font-family:'h5p-hub';content:"";font-size:1em;line-height:1;width:1em;height:1em;padding:.833em;display:block;color:#7c8696;-webkit-animation:spin 2s linear infinite;animation:spin 2s linear infinite;position:absolute;margin-left:-2.35em;margin-top:-0.8em;color:#fff}.h5p-hub.h5p-hub-hidden{display:none}.h5p-hub-client-container-using-mouse [role=button]{outline:none} - diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/libs/icons.png b/h5p/h5plib/v126/joubel/editor/styles/css/libs/icons.png deleted file mode 100644 index f5f153e5a0677..0000000000000 Binary files a/h5p/h5plib/v126/joubel/editor/styles/css/libs/icons.png and /dev/null differ diff --git a/h5p/h5plib/v126/joubel/editor/styles/css/libs/zebra_datepicker.min.css b/h5p/h5plib/v126/joubel/editor/styles/css/libs/zebra_datepicker.min.css deleted file mode 100644 index 1ef105e15c2ba..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/css/libs/zebra_datepicker.min.css +++ /dev/null @@ -1 +0,0 @@ -.Zebra_DatePicker{background:#fff;border:1px solid #aaa;border-radius:4px;box-shadow:0 0 10px #ccc;color:#222;font-size:13px;padding:5px;position:absolute;display:table;*width:255px;z-index:1200}.Zebra_DatePicker *,.Zebra_DatePicker :after,.Zebra_DatePicker :before{box-sizing:content-box!important}.Zebra_DatePicker *{padding:0}.Zebra_DatePicker table{border-collapse:collapse;border-radius:4px;border-spacing:0;width:100%}.Zebra_DatePicker td,.Zebra_DatePicker th{border-radius:4px;padding:5px;cursor:pointer;text-align:center;min-width:25px;width:25px}.Zebra_DatePicker td.dp_time_ampm{padding-left:10px}.Zebra_DatePicker .dp_body .dp_not_in_month{color:#666}.Zebra_DatePicker .dp_body .dp_current{color:#3a87ad}.Zebra_DatePicker .dp_body .dp_selected{background:#337ab7;color:#fff}.Zebra_DatePicker .dp_body .dp_disabled{color:#bbb;cursor:text}.Zebra_DatePicker .dp_body .dp_disabled.dp_current{color:#b4d5e6}.Zebra_DatePicker .dp_body .dp_hover{background:#dedede}.Zebra_DatePicker .dp_body .dp_hover.dp_time_control{background-color:#dedede}.Zebra_DatePicker .dp_monthpicker td,.Zebra_DatePicker .dp_timepicker td,.Zebra_DatePicker .dp_yearpicker td{width:33.3333%}.Zebra_DatePicker .dp_timepicker .dp_disabled{border:none;color:#222;font-size:39px;font-weight:700}.Zebra_DatePicker .dp_time_separator div{position:relative}.Zebra_DatePicker .dp_time_separator div:after{content:":";color:#222;font-size:37px;left:100%;position:absolute;z-index:1}.Zebra_DatePicker .dp_header{margin-bottom:5px}@supports (-ms-ime-align:auto){.Zebra_DatePicker .dp_header{font-family:'Segoe UI Symbol',Tahoma,Arial,Helvetica,sans-serif}}.Zebra_DatePicker .dp_footer{margin-top:5px}.Zebra_DatePicker .dp_footer .dp_icon{width:50%}.Zebra_DatePicker .dp_actions td{border-radius:4px}.Zebra_DatePicker .dp_actions .dp_caption{font-weight:700;width:100%}.Zebra_DatePicker .dp_actions .dp_next,.Zebra_DatePicker .dp_actions .dp_previous{*padding:0 10px}.Zebra_DatePicker .dp_actions .dp_hover{background-color:#dedede}.Zebra_DatePicker .dp_daypicker th{cursor:text;font-weight:700}.Zebra_DatePicker.dp_hidden{display:none}.Zebra_DatePicker .dp_icon{height:16px;background-image:url(icons.png);background-repeat:no-repeat;text-indent:-9999px;*text-indent:0}.Zebra_DatePicker .dp_icon.dp_confirm{background-position:center -123px}.Zebra_DatePicker .dp_icon.dp_view_toggler{background-position:center -91px}.Zebra_DatePicker .dp_icon.dp_view_toggler.dp_calendar{background-position:center -59px}button.Zebra_DatePicker_Icon{background:url(icons.png) center top no-repeat;border:none;cursor:pointer;display:block;height:16px;line-height:0;padding:0;position:absolute;text-indent:-9000px;width:16px}button.Zebra_DatePicker_Icon.Zebra_DatePicker_Icon_Disabled{background-position:center -32px;cursor:default} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_copy-paste.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_copy-paste.scss deleted file mode 100644 index a19f3e3db1aa5..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_copy-paste.scss +++ /dev/null @@ -1,93 +0,0 @@ -.h5peditor-copypaste-wrap { - float: right; -} -.h5peditor > .h5peditor-copypaste-wrap { - /* Make sure buttons are visible when Tutorial and Example is missing */ - margin-bottom: 14px; -} -.h5peditor-copypaste-wrap.hidden { - display: none; -} -.h5peditor-clearfix { - clear: both; -} - -.h5peditor-paste-button, -.h5peditor-copy-button { - border: 1px solid #deeeec; - padding: 0.5em 0.75em 0.75em 0.5em; - margin-left: 0.5em; - border-radius: 0.25em; - cursor: pointer; - background: #f2faff; - line-height: 0.5em; - - &:hover { - border-color: #bbdae8; - } - &:active { - background-color: #deeffb - } - &.disabled { - color: #707070; - background-color: #f5f5f5; - border-color: #ededed; - cursor: not-allowed; - } -} - -.h5peditor-copy-button:before, -.h5peditor-paste-button:before { - font-family: $icon-font-family; - margin-right: 5px; - font-size: 1.5em; - position: relative; - top: 0.2em; - color: #2372b3; -} -.h5peditor-copy-button:before { - content: "\e90e"; -} -.h5peditor-paste-button:before { - content: "\e910"; -} -.h5peditor-copy-button.disabled:before, -.h5peditor-paste-button.disabled:before { - color: #707070; -} -.h5peditor-copy-button.disabled:before { - content: "\e90f"; -} -.h5peditor-paste-button.disabled:before { - content: "\e911"; -} -.h5p-hub .h5peditor-paste-button { - font-family: $font-family; - color: #fff; - border: solid 1px #9fc8f4; - padding: 0 0.75em 0.5em 0.75em; - background: transparent; - line-height: 1; - height: auto; - margin-top: 0.675em; - margin-right: 0.675em; - - &:hover { - background: #4b5460; - border-color: #606a78; - } - &:active { - background: #434b55; - } - &.disabled { - border-color: #30353d; - background: #30353d; - color: #c2c2c2; - } -} -.h5p-hub .h5peditor-paste-button:before { - color: #9fc8f4; -} -.h5p-hub .h5peditor-paste-button.disabled:before { - color: #c3c3c3; -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_deprecated.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_deprecated.scss deleted file mode 100644 index 5610d57cb079d..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_deprecated.scss +++ /dev/null @@ -1,13 +0,0 @@ -@import 'variables'; - -// This styling is here for backwards compatibility between releases, and can be removed later - -/* - START HFP-409 - @deprecated (can be removed after May 2017) - Styling that lets old vtabs styling work with the editor upgrade, in Dec 2016 release -*/ -.h5p-vtab-wrapper > .h5p-vtab-forms .h5p-remove:after { - color: $form-border-color; -} -/* END HFP-409 */ diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_form-field.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_form-field.scss deleted file mode 100644 index 4439013a910c7..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_form-field.scss +++ /dev/null @@ -1,52 +0,0 @@ -@import 'variables'; -@import 'mixins'; - -.field { - @include field-margin($padding); - font-size: $font-size-normal; - padding: 0; - - // Assumes last group is `Behavioural settings`, which should - // be grouped with the bottom group `Settings and texts`. - .tree > &.group:last-child { - margin-bottom: 0; - } - - .fields > &.group { - @include field-margin($min-padding); - } - - &.boolean .h5peditor-label { - display: inline; - } - - .h5p-editor-image-buttons { - float: left; - clear: both; - } - - .library { - border: 0; - } - - &.importance-high > .h5peditor-label-wrapper > .h5peditor-label { - font-size: $font-size-large; - color: #356593; - } - - .h5p-dialog-anchor { - position: relative; - margin: -16px; - } -} - -.common-fields-library-wrapper { - @include field-margin($padding); - border: none; - margin: 0; - padding: 0; - - .common-field-legend { - display: none; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_form-groups.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_form-groups.scss deleted file mode 100644 index c5fac197531e0..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_form-groups.scss +++ /dev/null @@ -1,420 +0,0 @@ -@import 'variables'; -@import 'mixins'; - -%title-bar-base { - visibility: inherit; - cursor: pointer; -} -%title-bar-importance-high { - background: $form-item-importance-high-background; - border: 1px solid $form-item-importance-high-border-color; - height: $form-item-height-large; -} -%title-bar-text-importance-high { - font-size: $font-size-normal; - font-weight: 600; - line-height: $form-item-height-large; - // Fix white text on dark background blockiness - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} -%title-bar-importance-medium { - background: $form-item-importance-medium-background; - border: 1px solid $form-item-importance-medium-border-color; - height: $form-item-height-normal; -} -%title-bar-text-importance-medium { - font-size: $font-size-normal; - font-weight: 600; - color: $white; - line-height: $form-item-height-normal; - // Fix white text on dark background blockiness - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; -} - -%title-bar-importance-low { - background: $form-item-importance-low-background; - height: $form-item-height-normal; - border: 1px solid $form-border-color; -} -%title-bar-text-importance-low { - font-size: $font-size-normal; - font-weight: 400; - color: $text-color; - line-height: $form-item-height-normal; -} - -.group { - border: none; - - & > .title { - @extend %title-bar-base; - @extend %title-bar-importance-medium; - @extend %title-bar-text-importance-medium; - padding: 0 $min-padding; - outline: none; - } - - & > .content { - position: relative; - display: none; - margin: 0; - padding: $padding; - border: 1px solid $form-border-color; - border-top: none; - background: $form-content-background; - } - - &.importance-high > .title { - @extend %title-bar-importance-high; - @extend %title-bar-text-importance-high; - color: $white; - } - - &.importance-low > .title { - @extend %title-bar-importance-low; - @extend %title-bar-text-importance-low; - color: $text-color; - - &:focus:before { - outline: 1px dashed; - } - } -} - -// Zebra pattern for lists and groups -.common, .content { - .content { - background-color: $form-background; - - .content { - background-color: $white; - - .content { - background-color: $form-background; - - .content { - background-color: $white; - - .content { - background-color: $form-background; - - .content { - background-color: $white; - } - } - } - } - } - } -} - -.h5p-li > .list-item-title-bar { - @extend %title-bar-base; - @extend %title-bar-importance-medium; - color: $white; - - & > .h5peditor-label { - @extend %title-bar-text-importance-medium; - - overflow: hidden; - margin: 0 $form-item-buttons-width-normal 0 0; - padding: 0 $min-padding; - white-space: nowrap; - line-height: $form-item-height-normal; - outline: none; - text-overflow: ellipsis; - } - - &.importance-high > .h5peditor-label { - @extend %title-bar-text-importance-high; - } - - &.importance-medium > .h5peditor-label { - @extend %title-bar-text-importance-medium; - } - - &.importance-low > .h5peditor-label { - @extend %title-bar-text-importance-low; - } - - & > .list-actions { - height: 100%; - float: right; - position: relative; - } - - .remove { - cursor: pointer; - width: 1.25em; - height: $form-item-height-normal; - font-size: 1.75em; - display: inline-block; - - &:hover { - opacity: 1; - text-decoration: none; - } - } - - .remove:after { - font-family: $icon-font-family; - content: "\e890"; - opacity: 0.7; - display: inline-block; - line-height: $form-item-height-normal; - } - - .remove:hover:after { - opacity: 1; - } - - .order-group { - text-align: center; - float: right; - background: $form-item-importance-medium-border-color; - font-size: $font-size-normal; - cursor: pointer; - } - - .order-up, .order-down { - width: $form-item-height-normal / 2; - height: $form-item-height-normal / 2; - line-height: $form-item-height-normal / 2; - - &:hover { - background: $form-item-importance-medium-background-hover; - } - - &:after { - font-family: $icon-font-family; - content: "\e58e"; - display: inline-block; - } - } - - .order-down:after { - content: "\e58f"; - } - - &.importance-high { - @extend %title-bar-importance-high; - - & > .title { - @extend %title-bar-text-importance-high; - border: none; - margin: 0 $form-item-buttons-width-large 0 0; - } - - .remove { - height: $form-item-height-large; - line-height: $form-item-height-large; - font-size: 40px; - - &:after { - line-height: $form-item-height-large; - opacity: 0.7; - } - &:hover:after { - opacity: 1; - } - } - - .order-group { - background: $form-item-importance-high-border-color; - font-size: $font-size-large; - } - - .order-up, .order-down { - width: $form-item-height-large / 2; - height: $form-item-height-large / 2; - line-height: $form-item-height-large / 2; - - &:hover { - background: $form-item-importance-high-background-hover; - } - } - } - - &.importance-low { - @extend %title-bar-importance-low; - - & > .title { - @extend %title-bar-text-importance-low; - border: none; - margin: 0 $form-item-buttons-width-normal 0 0; - - &:before { - color: $text-color; - } - &:focus:before { - outline: 1px dashed $text-color; - } - } - - .remove { - height: $form-item-height-normal; - line-height: $form-item-height-normal; - font-size: 30px; - - &:after { - line-height: $form-item-height-normal; - } - } - .remove:after { - color: $text-color; - opacity: 0.7; - } - .remove:hover:after { - opacity: 1; - } - - .order-up, .order-down { - width: $form-item-height-normal / 2; - height: $form-item-height-normal / 2; - background: $form-item-importance-low-border-color; - font-size: $font-size-normal; - line-height: $form-item-height-normal / 2; - - &:hover { - background: $form-item-importance-low-background-hover; - } - - &:after { - color: $text-color; - } - } - } -} -.group.expanded, -.listgroup.expanded { - & > .content { - display: block; - } -} - -.listgroup > .list-item-title-bar > .h5peditor-label { - cursor: pointer; -} -.list-item-title-bar > .title, -.group > .title { - &:before { - content: "\e566"; - font-family: $icon-font-family; - margin-right: $min-padding/2; - } - - &:focus:before { - outline: 1px dashed $white; - } -} -.listgroup.expanded > .list-item-title-bar > .h5peditor-label:before, -.expanded > .title:before { - content: "\e565"; -} -.listgroup > .group.field { - margin: 0; - padding: 0; - min-width: 0; -} -.content { - display: block; - margin: 0; - padding: $padding; - border: 1px solid $form-border-color; - border-top: none; - background: $form-content-background; -} - -.common { - margin-top: $padding; - - & > .h5peditor-label { - @extend %title-bar-base; - @extend %title-bar-importance-low; - @extend %title-bar-text-importance-low; - margin: 0; - padding: 0 $min-padding; - cursor: pointer; - font-size: 1em; - - & > .icon:before { - content: "\e565"; - font-family: $icon-font-family; - margin-right: 5px; - } - &:hover > .icon { - opacity: 1; - } - - &:focus { - outline: none; - } - - &:focus > .icon:before { - outline: 1px dashed; - } - } - & > .fields { - min-height: 2em; - padding: $padding; - border: 1px solid $form-border-color; - border-top: none; - background: $form-content-background; - position: relative; - - & > .desc { - margin: 0; - font-size: 0.875em; - color: #666; - float: left; - } - - p:first-child { - margin-bottom: $padding; - } - } - - &.collapsed { - & > .h5peditor-label > .icon:before { - content: "\e566"; - } - & > .fields { - display: none; - } - } - - &.hidden { - display: none; - } -} - -.h5peditor-button[aria-label] { - - &:before { - content: attr(aria-label); - visibility: hidden; - - position: absolute; - top: 115%; - right: -10%; - z-index: 2; - - padding: 0.25em 0.75em; - background: $text-color; - color: $white; - white-space: nowrap; - font-size: $font-size-small; - line-height: 1.5; - box-shadow: 0 0 0.5em $form-input-placeholder-color; - } - - &:hover:before { - visibility: visible; - } - - &[aria-disabled="true"]:before { - display: none; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_fullscreen-bar.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_fullscreen-bar.scss deleted file mode 100644 index 03cc23a4138cb..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_fullscreen-bar.scss +++ /dev/null @@ -1,199 +0,0 @@ -@font-face { - font-family: 'h5p-fullscreen-bar'; - src: url('fonts/h5p-fullscreen-bar.eot?p850ul'); - src: url('fonts/h5p-fullscreen-bar.eot?p850ul#iefix') format('embedded-opentype'), - url('fonts/h5p-fullscreen-bar.ttf?p850ul') format('truetype'), - url('fonts/h5p-fullscreen-bar.woff?p850ul') format('woff'), - url('fonts/h5p-fullscreen-bar.svg?p850ul#h5p') format('svg'); - font-weight: normal; - font-style: normal; -} - -.h5peditor-form-manager-head { - background: #f5f5f5; - color: #414141; - border-bottom: 1px solid #dfdfdf; - height: 41px; - line-height: 40px; - box-sizing: border-box; - position: relative; - display: flex; - z-index: 3; - justify-content: space-between; - align-items: flex-start; - box-shadow: 0px 2px 2px rgba(128,128,128,0.15); - padding: 0 1em 0 0.5em; -} - -.h5peditor-form-manager-button { - background: transparent; - padding: 0; - border: 0; - cursor: pointer; - color: #414141; - position: relative; - white-space: nowrap; - line-height: normal; -} - -.h5peditor-form-manager-button-inner { - display: block; - outline: none; -} - -.h5peditor-form-manager-breadcrumb { - display: flex; - min-width: 0; - flex-grow: 1; -} - -.h5peditor-form-manager-proceed { - display: none; - - background: #186df7; - color: #fff; - - font-size: 14px; - font-weight: bold; - font-family: Open Sans, sans-serif; - border-radius: 2px; - margin-right: 5px; - margin-top: 5px; - padding: 5px 10px; - - &:hover { - background: #20588f; - } -} - -.h5peditor-form-manager-fullscreen { - width: 24px; - height: 24px; - font-size: 20px; - margin-top: 9px; - margin-left: 0.25em; - - .h5peditor-form-manager-button-inner:before { - font-family: 'h5p-fullscreen-bar'; - content: '\e929'; - } - - &:after { - visibility: hidden; - position: absolute; - top: 120%; - right: -10%; - z-index: 3; - padding: 0.25em 0.75em; - background: #212121; - color: #fff; - white-space: nowrap; - font-size: 14px; - box-shadow: 0 0 0.5em #858585; - text-indent: 0; - font-weight: normal; - pointer-events: none; - outline: none; - line-height: normal; - content: attr(aria-label); - } - - &:hover:after, - &:focus:after { - visibility: visible; - } -} - -.h5peditor-semi-fullscreen { - .h5peditor-form-manager-fullscreen .h5peditor-form-manager-button-inner:before { - content: "\e92a"; - } - .h5peditor-form-manager-proceed { - display: block; - } - .h5peditor-form-manager-head { - position: fixed; - top: 0; - left: 0; - right: 0; - z-index: 2; - } -} - -.form-manager-exit.form-manager-fullscreen .h5peditor-form-manager-button-inner:before { - content: "\f066"; -} - -.h5peditor-form-manager-title { - position: relative; - min-width: 20px; - font-weight: bold; - font-size: 14px; - - /* Icons for content types */ - &:before { - font-family: 'h5p-fullscreen-bar'; - content: "\e928"; - font-size: 1.2em; - position: relative; - top: 0.1em; - margin-right: 0.25em; - speak: none; - font-style: normal; - font-weight: normal; - font-variant: normal; - text-transform: none; - line-height: 1; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; - } - - &.dragtext:before { - content: "\e600"; - } - &.markthewords:before { - content: "\e601"; - } - &.multichoice:before { - content: "\e603"; - } - &.interactivevideo:before { - content: "\e900"; - } - &.audio:before { - content: "\e901"; - } - &.truefalse:before { - content: "\e902"; - } - &.dialogcards:before { - content: "\e903"; - } - &.questionnaire:before { - content: "\e904"; - } - &.coursepresentation:before { - content: "\e923"; - } - &.dragquestion:before { - content: "\e991"; - } - &.summary:before { - content: "\e992"; - } - &.singlechoiceset:before { - content: "\e993"; - } - &.blanks:before { - content: "\e994"; - } -} - -/* Makes metadata popup behave good in fullscreen */ -.h5peditor-semi-fullscreen + .h5p-metadata-popup-overlay { - position: fixed; - - .h5p-metadata-wrapper { - margin-top: 20px !important; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit-popup.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit-popup.scss deleted file mode 100644 index d2fdb56d17263..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit-popup.scss +++ /dev/null @@ -1,149 +0,0 @@ -body.h5p-editor-image-popup { - position: relative; -} - -.h5p-editing-image-popup-background { - position: fixed; - top: 0; - left: 0; - - width: 100%; - height: 100%; - - background: rgba(0,0,0,0.8); - padding: 3em 1em; - box-sizing: border-box; - z-index: 102; -} - -#darkroom-icons { - top:0; - left:0; -} - -.h5p-editing-image-popup-background.hidden { - display: none; -} - -.h5p-editing-image-popup { - display: inline-block; - position: relative; - top: 0; - left: 50%; - height: auto; - width: 100%; - max-height: 100%; - -webkit-transform: translateX(-50%); - -ms-transform: translateX(-50%); - transform: translateX(-50%); - background: #fff; - -webkit-box-sizing: border-box; - -moz-box-sizing: border-box; - box-sizing: border-box; -} - -.h5p-editing-image-header { - padding: 1em 0.75em; -} - -.h5p-editing-image { - max-height:100%; - max-width:100%; -} - -.h5p-editing-image.hidden { - display: none; -} - -.h5p-editing-image-popup { - .darkroom-toolbar { - position: relative; - -webkit-border-radius:0; - -moz-border-radius:0; - border-radius:0; - top:0; - } - - .darkroom-toolbar:before { - content: initial; - } - - .darkroom-image-container { - padding: 32px 2em; - } - - .darkroom-image-container .canvas-container { - box-shadow: 0 0 8px 3px rgba(0, 0, 0, 0.6); - margin: auto; - } -} - -.h5p-editing-image-header-title { - display: inline-block; - font-size: 1.25em; -} - -.h5p-editing-image-header-buttons { - display: inline-block; - float: right; -} -.h5p-editing-image-loading { - padding: 1em; -} - -.h5p-editing-image-loading.hidden { - display: none; -} - -.h5p-editing-image-editing-container { - background: #666; -} - -.h5p-editing-image-header-buttons { - button { - padding: 0.5em 1.75em; - margin: 0 0 0 1em; - - border: 1px solid #ccc; - border-radius: 0.25em; - color: #333; - cursor: pointer; - - background: #f2f2f2; - background: -webkit-linear-gradient(top,#fff 0,#f2f2f2 100%); - background: -ms-linear-gradient(top,#fff 0,#f2f2f2 100%); - } - - button:hover { - background: #ededed; - } - - .h5p-done { - color: #fff; - border-color: #20588F; - - background: #3673B5; - background: -webkit-linear-gradient(top,#5A94D3 0,#3673B5 100%); - background: -ms-linear-gradient(top,#5A94D3 0,#3673B5 100%); - } - - .h5p-done:hover { - background: #3275bc; - background: -webkit-linear-gradient(top,#3275bc 0,#285585 100%); - background: -ms-linear-gradient(top,#3275bc 0,#285585 100%); - } - - .h5p-remove { - background: none; - border: none; - color: #a00; - padding-right: 0; - padding-left: 0; - border-radius: 0; - } - - .h5p-remove:hover { - background: none; - color: #e40000; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit.scss deleted file mode 100644 index ebeb318b7d306..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_h5peditor-image-edit.scss +++ /dev/null @@ -1,45 +0,0 @@ -@import 'variables'; - -.h5peditor { - .h5p-editing-image-button { - background: linear-gradient(to bottom, #fff 0, #f2f2f2 100%); - font-size: $font-size-small; - color: $black; - line-height: $form-item-height-small; - padding-right: $padding; - margin-right: 0.5em; - height: 30px; - padding-left: 0; - font-weight: normal; - - &:hover:not([disabled]) { - background: linear-gradient(to bottom, #fff 0, #d0d0d1 100%); - border-color: #999; - } - } - - .h5p-editing-image-button:before { - font-family: $icon-font-family; - content: "\e900"; - color: #666; - padding-right: 0.3em; - padding-left: 0.45em; - vertical-align: middle; - font-size: 1.5em; - line-height: 0.9; - } - - .h5p-editing-image-button.loading:before { - content: "\e901"; - } - - .h5p-editing-image-button.hidden, - .h5p-copyright-button.hidden { - display: none !important; - } - - .ui-dialog .h5p-editing-image-button, - .ui-dialog .h5p-copyright-button { - padding-left: 0.5em; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-author-widget.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-author-widget.scss deleted file mode 100644 index 1770445ddf0b6..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-author-widget.scss +++ /dev/null @@ -1,102 +0,0 @@ -.h5p-metadata-author-widget { - display: flex; - flex-direction: column; - border: solid 1px #ced6e3; - border-radius: 0.208em; - width: 100%; - padding: 1em; - margin-top: 1em; - - .h5p-save-author { - border: solid 2px #0a715e; - color: #fff; - background-color: #0a715e; - } - .h5p-save-author:hover { - border-color: #0d826c; - color: #fff; - background-color: #0d826c; - } - .h5p-save-author:active { - border-color: #095345; - color: #fff; - background-color: #095345; - } - - .h5p-author-data { - display: flex; - flex-direction: row; - flex-wrap: nowrap; - - &>.field { - margin-bottom: 0; - } - } - - .field-name-name { - flex-grow: 1; - margin-right: 2%; - overflow: hidden; - white-space: nowrap; - min-width: 49%; - } - - .field-name-role { - margin-top: 0; - margin-right: 2%; - width: 25%; - flex-grow: 1; - overflow: hidden; - white-space: nowrap; - } - - .authorList { - margin-top: 1.5em; - white-space: nowrap; - - li { - display: inline; - margin-right: 0.5em; - - span { - margin-right: 1em; - font-style: italic; - } - } - } - - .h5p-author-list-wrapper { - margin-top: 0.5em; - - ul { - margin: 0; - padding: 0; - } - - li { - display: inline-block; - margin-right: 2em; - list-style: none; - font-weight: bold; - font-size: 0.8em; - } - - .h5p-metadata-role { - font-style: italic; - margin: 0 0.6em; - font-weight: normal; - font-size: 0.85em; - color: #757575; - } - - button { - &:after { - font-family: 'h5p-metadata-icons'; - content: "\e902"; - font-size: 1em; - position: relative; - top: -0.1em; - } - } - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-changelog-widget.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-changelog-widget.scss deleted file mode 100644 index 656eba1cbaa58..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-changelog-widget.scss +++ /dev/null @@ -1,128 +0,0 @@ -.h5p-metadata-changelog { - width: 100%; -} -.h5p-metadata-changelog .field-name-change { - &.expanded .content { - display: flex; - flex-wrap: wrap; - } - - .field-name-date { - width: 49%; - margin-bottom: 0.5em; - margin-right: 1%; - margin-top: 0; - span { - width: 100%; - } - } - - .field-name-author { - width: 49%; - margin-bottom: 0.5em; - margin-left: 1%; - margin-top: 0; - } - - .field-name-log { - width: 100%; - } - - .h5p-add-author:before { - content: "+"; - margin-right: 0.7em; - } - - .h5p-cancel { - margin-right: 10px; - } - - .h5p-metadata-changelog-buttons { - width: 100%; - } - - .h5p-metadata-logged-changes { - width: 100%; - margin-top: 1em; - - // Hide the change log table while editing - &.editing { - display: none; - } - - .h5peditor-field-description { - margin-bottom: 0; - } - } - - .h5p-metadata-log-wrapper { - margin-top: 1em; - border: solid 1px #ced6e3; - border-radius: 0.208em; - max-height: 15.917rem; - overflow-y: auto; - padding: 0 0.5em; - } - - .h5p-metadata-log { - display: flex; - margin-top: 0.5em; - padding-bottom: 0.5em; - - .h5p-metadata-delete { - margin-left: 2em; - - &:after { - font-family: 'h5p-metadata-icons'; - content: "\e902"; - font-size: 1.2em; - } - } - - .h5p-metadata-edit { - &:after { - font-family: 'h5p-metadata-icons'; - content: "\e904"; - font-size: 1.2em; - } - } - } - - .h5p-metadata-log-buttons { - min-width: 3em; - line-height: 2.5; - } - - .h5p-metadata-log:not(:last-child) { - border-bottom: solid 1px #ced6e3; - } - - .h5p-metadata-log-date { - min-width: 25%; - font-size: 0.85em; - font-style: italic; - line-height: 2.5; - } - - .h5p-metadata-description-wrapper { - flex-grow: 1; - font-size: 0.85em; - font-style: italic; - } - - .h5p-metadata-description-wrapper { - font-weight: bolder; - } - - .h5p-metadata-log-author { - font-weight: normal; - } - - .h5p-metadata-new-log-message { - width: 100%; - background: #C5E7E1; - padding: 0.5em; - margin-bottom: 1em; - color: #0a715e; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-form.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-form.scss deleted file mode 100644 index 1bc1129e553db..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-form.scss +++ /dev/null @@ -1,377 +0,0 @@ -.h5p-metadata-button-wrapper { - display: flex; - flex-direction: row; - align-items: center; - margin-top: -7px; - margin-left: 7px; - cursor: pointer; -} - -.h5p-metadata-button-wrapper.inline-with-selector { - display: inline-flex; - margin-left: 17px; -} - -.h5p-metadata-toggler { - margin-left: -11px; - height: 14.5px; - max-height: 14.5px; - padding: 0.217rem 0.3rem 0.217rem 0.217rem; - font-size: 0.625em; - font-weight: bold; - letter-spacing: 0.025em; - border: 1px solid #b6cada; - border-left: none; - border-radius: 4px; - background: linear-gradient(#fff, #deeaf1); - color: #356593; - z-index: 1; - justify-content:center; - align-content:center; - line-height: 1.5em; -} - -.h5p-metadata-button-tip { - width: 16.5px; - height: 16.5px; - content: ""; - background: linear-gradient(-215deg, #fff, #deeaf1); - border: 1px solid #b6cada; - border-right: none; - border-top: none; - border-radius: 4px; - transform: rotate(45deg); - z-index: 0; -} - -.h5p-metadata-button-wrapper:hover { - .h5p-metadata-toggler, - .h5p-metadata-button-tip { - border-color: #6d9fce; - } - .h5p-metadata-toggler { - background: linear-gradient(#f3f8fb, #cee5f3); - } - .h5p-metadata-button-tip { - background: linear-gradient(-215deg, #f3f8fb, #cee5f3); - } -} - -.h5p-metadata-button-wrapper:active { - .h5p-metadata-toggler, - .h5p-metadata-button-tip { - color: #294f73; - border-color: #6d9fce; - } - .h5p-metadata-toggler { - background: linear-gradient(#f3f8fb, #bcd4e2); - } - .h5p-metadata-button-tip { - background: linear-gradient(-215deg, #f3f8fb, #bcd4e2); - } -} - -#metadata-title-main-label { - margin-top: 0; -} - -.h5p-metadata-button { - background: white; - padding: 10px; - text-align: center; - border-radius: 5px; - border: 2px solid #6b6b6b; - color: #6b6b6b; - font-weight: bold; - display: inline; - cursor: pointer; -} -.h5p-metadata-button:hover { - border-color: #4a4a4a; - color: #4a4a4a; -} -.h5p-metadata-button:active { - border-color: #000; - color: #000; -} - -.h5p-metadata-button.inverted { - border: 2px solid #0a715e; - color: #0a715e; -} -.h5p-metadata-button.inverted:hover { - border-color: #0d826c; - color: #0d826c; -} -.h5p-metadata-button.inverted:active { - border-color: #095345; - color: #095345; -} - -.h5p-metadata-icon-button { - border: none; - cursor: pointer; - color: #6b6b6b; - background: transparent; - padding: 0; -} -.h5p-metadata-icon-button:hover { - color: #4a4a4a; -} -.h5p-metadata-icon-button:active { - color: #000; -} - -.h5p-metadata-wrapper { - display: inline-block; - background-color: #fff; - max-width: 700px; - width: calc(100% - 4em); - text-align: left; - /* Adds a 20px margin on the bottom */ - border-bottom: 20px solid transparent; - background-clip: padding-box; - max-height: calc(100% - 40px); - overflow: auto; - margin-bottom: 20px; - - .h5p-metadata-header { - display: flex; - border-bottom: 1px solid #ced6e3; - padding: 1.5em; - - .h5p-title-container { - flex-grow: 1; - padding-left: 3.5em; - white-space: nowrap; - overflow: hidden; - margin-right: 1%; - position: relative; - - h2 { - margin: 0; - } - - p { - margin: 0.1em; - } - } - - .h5p-title-container:before { - position: absolute; - font-family: 'h5p-metadata-icons'; - content: '\e903'; - left: 0; - top: 0; - font-size: 2em; - height: 1.4em; - line-height: 1.3; - } - } - - .h5p-save { - border: solid 2px #0a715e; - border-radius: 0.3rem; - color: #fff; - background-color: #0a715e; - padding: 0.75rem; - white-space: nowrap; - } - .h5p-save:hover { - border-color: #0d826c; - background-color: #0d826c; - } - .h5p-save:active { - border-color: #095345; - background-color: #095345; - } - - .h5peditor-label { - font-size: 0.8em; - color: #333; - } - - .h5peditor-field-description { - color: #697484; - margin-top: 0.35em; - letter-spacing: 0.5px; - } - - .copyright-form { - margin-top: 20px; - } - - h2 { - font-size: 1em; - overflow: hidden; - text-overflow:ellipsis; - } - - p { - font-size: 0.8333em; - color: #697484; - overflow: hidden; - text-overflow: ellipsis; - margin-bottom: 0; - } - - .errors p, - .h5p-errors p { - color: #da0001; - font-size: 1em; - white-space: normal; - text-overflow: unset; - } - - select.h5peditor-select, - input.h5peditor-text, - textarea { - border: 1px solid #b8c0cd; - font-size: 0.833em; - border-radius: 0.208em; - box-shadow: none; - font-family: $font-family; - } - - select:focus, - input:focus, - textarea:focus { - box-shadow: inset 0px 0px 10px rgba(0,0,0,0.15); - } - - select { - width: 100%; - } - - .h5p-metadata-fields-wrapper { - display: flex; - flex-wrap: wrap; - margin: 1.5em; - - > .field { - margin-bottom: 0; - width: 100%; - } - - > .field-name-license { - width: 49%; - margin-right: 1%; - } - - > .field-name-licenseVersion { - width: 49%; - margin-left: 1%; - } - - > .field-name-yearFrom { - width: 24%; - margin-right: 1%; - white-space: nowrap; - - .h5peditor-text { - width: 100%; - } - } - - > .field-name-yearTo { - width: 23%; - margin-right: 1%; - margin-left: 1%; - - .h5peditor-text { - width: 100%; - } - } - - > .field-name-source { - width: 49%; - margin-left: 1%; - } - - .field-name-title label { - justify-content: space-between; - - .a11y-title-toggle { - background: none; - border: none; - cursor: pointer; - - &:before { - font-family: 'h5p-metadata-icons'; - content: "\e905"; - margin-right: 0.5em; - font-size: 1.5em; - vertical-align: middle; - } - - &:focus, - &:active, - &:hover { - border: none; - } - &:hover .h5p-a11y-title-text { - text-decoration: underline; - } - } - } - - .field-name-a11yTitle { - transition: max-height 0.2s, margin 0.2s; - overflow: hidden; - - &.hide { - max-height: 0; - margin: 0; - } - - &.hidden { - display: none; - } - } - } - - .field.group { - > .title { - font-weight: 400; - background: #F6F6F6; - color: #323232; - border: solid 1px #ced6e3; - border-radius: 0.208em; - } - &.expanded > .title { - border-radius: 0.208em 0.208em 0 0; - } - > .title:focus { - border-color: rgb(77, 144, 254); - } - > .content { - border: solid 1px #ced6e3; - border-radius: 0 0 0.208em 0.208em ; - padding: 20px; - border-top: 0; - } - } -} - -.h5p-metadata-additional-information { - width: 100%; - margin-top: 1em; - - .content.h5peditor-single { - min-height: 2em; - border: 1px solid #d0d0d1; - border-top: none; - background: #fff; - } - - .title { - font-weight: 400; - background: #F6F6F6; - color: #323232; - border: solid 1px #ced6e3; - border-radius: 0.208em; - } - .title:focus { - border-color: rgb(77, 144, 254); - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-popup.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-popup.scss deleted file mode 100644 index 2962a17bc9f3e..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_metadata-popup.scss +++ /dev/null @@ -1,12 +0,0 @@ -.h5p-metadata-popup-overlay { - position: absolute; - top: 0; - left: 0; - right: 0; - bottom: 0; - background-color: rgba(0, 0, 0, 0.85); - // Need to be higer than the different z-indexes in the custom editors - // Like eg popups in CP, DnD and IV - z-index: 101; - text-align: center; -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_mixins.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_mixins.scss deleted file mode 100644 index 96730d36d6c19..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_mixins.scss +++ /dev/null @@ -1,17 +0,0 @@ -@mixin button-background($background, $highlight) { - background: $background; - background-image: - linear-gradient($highlight 50%, transparent 50%, transparent); -} - -@mixin field-margin($margin) { - margin: $margin 0; - - &:first-child { - margin-top: 0; - } - - &:last-child { - margin-bottom: 0; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_tutorial-and-example-links.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_tutorial-and-example-links.scss deleted file mode 100644 index 0dad24aef4a35..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_tutorial-and-example-links.scss +++ /dev/null @@ -1,92 +0,0 @@ -$h5p-tutorial-url-label-color: #415702; -$h5p-tutorial-icon-color: #eaffab; -$h5p-tutorial-icon-bg-color: #749e00; - -$h5p-example-url-label-color: #422447; -$h5p-example-icon-color: #f8d4ff; -$h5p-example-icon-bg-color: #8f4f9c; - -$h5p-tutorial-icon-diameter: 2rem; - -@mixin h5p-tutorial-focus-shadow() { - box-shadow: 0px 0px 5px rgba(22,52,247,1); -} - -.h5p-tutorial-url, -.h5p-example-url { - font-size: 1em; - display: inline-flex; - align-items: center; - line-height: 1; - margin-right: 1em; - margin-bottom: 0.5em; - margin-left: 0.25em; - height: $h5p-tutorial-icon-diameter; - border-radius: $h5p-tutorial-icon-diameter / 2; - - &:focus { - @include h5p-tutorial-focus-shadow; - } - - &:active { - border-color: #1C70BA; - } - - &::before { - font-family: $icon-font-family; - font-size: 1.25em; - border-radius: 50%; - width: $h5p-tutorial-icon-diameter; - height: $h5p-tutorial-icon-diameter; - position: absolute; - display: flex; - justify-content: center; - align-items: center; - } - - .h5p-tutorial-url-label, - .h5p-example-url-label { - font-style: italic; - font-weight: bold; - text-decoration: underline; - margin: 0.25em 0.5em; - margin-left: $h5p-tutorial-icon-diameter; - padding-left: 0.5em; - padding-right: 1em; - } -} - -.h5p-tutorial-url { - &:active, - &:hover { - background-color: $h5p-tutorial-icon-color; - } - - &::before { - content: "\e93e"; - color: $h5p-tutorial-icon-color; - background-color: $h5p-tutorial-icon-bg-color; - } - - .h5p-tutorial-url-label { - color: $h5p-tutorial-url-label-color; - } -} - -.h5p-example-url { - &:active, - &:hover { - background-color: $h5p-example-icon-color; - } - - &::before { - content: "\e93d"; - font-size: 1em; - color: $h5p-example-icon-color; - background-color: $h5p-example-icon-bg-color; - } - - .h5p-example-url-label { - color: $h5p-example-url-label-color; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_utils.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_utils.scss deleted file mode 100644 index e272b7e4cad50..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_utils.scss +++ /dev/null @@ -1,5 +0,0 @@ -ul.list-unstyled { - list-style: none; - padding-left: 0; - margin: 0; -} \ No newline at end of file diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/_variables.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/_variables.scss deleted file mode 100644 index 8b767f6ae078b..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/_variables.scss +++ /dev/null @@ -1,57 +0,0 @@ -// General editor -$black: #212121; -$white: #FFFFFF; -$blue: #285c8b; - -$text-color: $black; - -$font-family: 'Open Sans', sans-serif; - -$icon-font-family: 'H5P'; - -$font-size-large: 18px; -$font-size-normal: 16px; -$font-size-small: 14px; - -$min-padding: 10px; -$padding: 20px; - -// Form -$form-background: #fcfcfc; -$form-border-color: #d0d0d1; -$form-content-background: $white; - -$form-input-color: $white; -$form-input-placeholder-color: #858585; -$form-input-focus-placeholder-color: #fff; -$form-input-box-shadow: inset 0px 0px 5px rgba(0, 0, 0, 0.12); - -$form-item-importance-high-background: #2579C6; -$form-item-importance-high-background-highlight: #3080c9; -$form-item-importance-high-border-color: #1f67a8; -$form-item-importance-high-background-hover: #1f67a8; -$form-item-importance-high-background-hover-highlight: #2a6fac; - -$form-item-importance-medium-background: #747275; -$form-item-importance-medium-background-highlight: #7b797c; -$form-item-importance-medium-border-color: #636164; -$form-item-importance-medium-background-hover: #636164; -$form-item-importance-medium-background-hover-highlight: #6b696c; - -$form-item-importance-low-background: #f5f5f5; -$form-item-importance-low-background-highlight: #eeeeef; -$form-item-importance-low-border-color: #d0d0d1; -$form-item-importance-low-background-hover: #deddde; -$form-item-importance-low-background-hover-highlight: #deddde; - -$form-item-height-large: 42px; // 44px total with border -$form-item-height-normal: 38px; // 40px total with border -$form-item-height-small: 28px; // 30px total with border - -$form-item-buttons-width-large: 74px; -$form-item-buttons-width-normal: 54px; -$form-item-buttons-width-small: 54px; - -// Label -$form-label: #454347; -$form-label-importance-high: #356593; diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/application.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/application.scss deleted file mode 100644 index 3c5d7c878f196..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/application.scss +++ /dev/null @@ -1,1265 +0,0 @@ -@import "compass/css3/border-radius"; -@import "compass/css3/box-shadow"; -@import "compass/css3/opacity"; -@import "compass/css3/text-shadow"; -@import "compass/css3/transition"; -@import "compass/css3/transform"; - -@import "variables"; -@import "mixins"; -@import "utils"; -@import "form-field"; -@import "form-groups"; -@import "h5peditor-image-edit"; -@import "h5peditor-image-edit-popup"; -@import "deprecated"; - -html, body { - margin: 0; - padding: 0; - color: $text-color; - font-family: $font-family; - max-width: 960px; - position: relative; -} -a { - text-decoration: none; -} - -.h5peditor { - font-size: 16px; - - .h5p-more-libraries { - font-size: 0.875em; - margin-top: 0.4em; - } - .h5peditor-single > .field.library { - border: 0; - padding: 0; - } - .cke_dialog_background_cover { - width: 100% !important; - height: 100% !important; - } - - .errors p, - .h5p-errors { - color: #da0001; - white-space: normal; - } - textarea { - resize: vertical; - } - - .h5p-editor-flex-wrapper { - display: flex; - flex-direction: row; - } - - .h5peditor-label-wrapper { - display: flex; - flex-direction: row; - } - - .has-important-description { - .h5peditor-field-description { - display: none; - } - .h5peditor-label-wrapper { - float: left; - } - } - - .important-description-visible { - .h5peditor-field-important-description { - display: block; - } - .h5peditor-label-wrapper { - float: none; - } - .important-description-show { - display: none; - } - } - - textarea, - .h5peditor-text, - .ckeditor { - @include box-shadow($form-input-box-shadow); - box-sizing: border-box; - margin: 0; - padding: $min-padding; - min-height: 40px; - border: 1px solid $form-border-color; - background: $form-input-color; - outline: none; - font-size: 16px; - word-wrap: break-word; - - &.error { - border-color: red; - } - } - .h5peditor-text, textarea { - width: 100%; - box-sizing: border-box; - - &.error { - border-color: red; - } - } - /* Placeholders - need to be on separate lines. If not, - the browsers will invalidate them */ - textarea:focus::-webkit-input-placeholder, - input:focus::-webkit-input-placeholder { - color: $form-input-focus-placeholder-color; - } - textarea:focus:-moz-placeholder, - input:focus:-moz-placeholder { - color: $form-input-focus-placeholder-color; - } - textarea:focus::-moz-placeholder, - input:focus::-moz-placeholder { - color: $form-input-focus-placeholder-color; - } - textarea:focus:-ms-input-placeholder, - input:focus:-ms-input-placeholder { - color: $form-input-focus-placeholder-color; - } - textarea:focus.h5peditor-ckeditor-placeholder, - input:focus.h5peditor-ckeditor-placeholder { - color: $form-input-focus-placeholder-color; - } - - ::-webkit-input-placeholder { - color: $form-input-placeholder-color; - } - :-moz-placeholder { - color: $form-input-placeholder-color; - } - ::-moz-placeholder { - color: $form-input-placeholder-color; - } - :-ms-input-placeholder { - color: $form-input-placeholder-color; - } - .h5peditor-ckeditor-placeholder { - color: $form-input-placeholder-color; - } - & > select, - .h5peditor-language-switcher select, - h5peditor-select, - .field > select { - padding: 10px 30px 10px 8px; - font-family: $font-family; - font-size: 16px; - border: 1px solid $form-border-color; - background: $form-input-color inline-image('down.png') no-repeat; - background-position: calc(100% - 10px); - @include box-shadow($form-input-box-shadow); - -moz-appearance: none; - -webkit-appearance: none; - - &:disabled { - background-color: #efefef; - } - } - & > select { - margin-bottom: 13px; - margin-right: 20px; - } - select::-ms-expand { - display: none; - } - a:focus { - outline: none; - } - textarea:focus, - .h5peditor-text:focus { - outline: none; - background-color: $white; - border-color: #53a0ff; - } - .h5p-ul { - padding: 0; - margin: 0; - list-style: none; - - .h5p-li { - @include field-margin($min-padding); - padding: 0; - list-style: none; - - &.placeholder { - box-sizing: border-box; - background: #e8f2fa; - border: dashed 2px #2782d1; - } - - &:hover { - text-decoration: none; - } - - &:nth-child(2).moving { - margin-top: 0; - } - - &:nth-last-child(2).placeholder { - margin-bottom: 0; - } - } - } - .dimensions input, .coordinates input, .number input { - width: 75px; - } - .number input[type="range"] { - width: 300px; - float: left; - margin: 7px 8px 0 0; - } - .h5p-errors { - clear: both; - } - .h5p-add-file { - float: left; - position: relative; - background: transparent; - border: 2px dashed #dddddd; - color: #dddddd; - margin: 0.5em; - width: 6em; - height: 4.5em; - cursor: pointer; - outline: none; - box-sizing: border-box; - -moz-box-sizing: border-box; - - &:focus, - &:hover { - color: #999; - border-color: #999; - } - - &.hidden { - visibility: hidden; - } - } - - .h5p-add-file:after { - position: absolute; - content: "+"; - font-size: 2em; - line-height: 2.2em; - width: 100%; - height: 100%; - text-align: center; - } - .h5p-add-dialog { - position: absolute; - z-index: 1; - visibility: hidden; - opacity: 0; - background: #fff; - left: 1em; - right: 1em; - top: 1em; - border: 1px solid #cdcdcd; - box-sizing: border-box; - -moz-box-sizing: border-box; - @include box-shadow(0 0 8px #666); - @include transition(visibility 0s 0.2s, opacity 0.2s); - - &.h5p-open { - visibility: visible; - opacity: 1; - @include transition(visibility 0s 0s, opacity 0.2s); - } - - .h5p-add-dialog-table { - overflow: hidden; - - .av-tablist { - overflow: hidden; - margin: 1em 1em 0 1em; - } - - .av-tab { - cursor: pointer; - float: left; - padding: 0.5em 0.75em; - border-top: 1px solid #ccc; - border-right: 1px solid #ccc; - font-weight: bold; - font-size: 15px; - color: #666; - background: #f5f5f5; - - &:first-child { - border-left: 1px solid #ccc; - } - &.selected { - color: #333; - background: #fff; - } - } - - .av-tabpanel { - border: 1px solid #ccc; - margin: 0 1em 1em 1em; - overflow: hidden; - padding: 1em; - - h3 { - margin: 0 0 1em 0; - } - .h5p-file-drop-upload { - width: 90px; - margin: 0; - } - } - - - .h5p-dialog-box { - float: left; - padding: 0; - width: 60%; - margin-right: 5%; - margin-left: 4%; - text-align: left; - margin-top: 30px; - - &:first-child { - width: 22%; - margin-left: 3%; - margin-right: 1%; - - h3 { - text-align: center; - } - } - } - - h3 { - color: #666; - font-size: 1em; - margin: 1.2em 0; - line-height: 1.1em; - } - - .h5peditor-field-description { - color: #bbb; - } - - .h5p-file-url-wrapper { - background-color: #f5f5f5; - border: 1px solid #bbb; - padding: 15px 15px 15px 60px; - position: relative; - - &:before { - font-family: 'H5P'; - position: absolute; - left: 0; - top: 0.083em; - font-size: 3.8em; - line-height: 1; - } - - &.video:before { - content: "\e904"; - color: #dd0505; - } - - &.audio:before { - content: "\e913"; - color: #747275; - } - - input { - border-color: #bbb; - text-align: left; - padding-left: 20px; - } - } - - .h5p-file-drop-upload { - position: relative; - width: 70%; - height: 90px; - margin: 0 15%; - float: left; - background-color: #f5f5f5; - text-align: center; - cursor: pointer; - border: 1px solid #bbb; - - .h5p-file-drop-upload-inner { - border: 2px dashed #bbb; - background-color: white; - position: absolute; - top: 3px; - bottom: 3px; - left: 3px; - right: 3px; - - &:after { - font-family: "H5P"; - line-height: 1.4em; - font-size: 3.5em; - color: #bbb; - } - - &.video:after { - content: "\e903"; - } - - &.audio:after { - content: "\e914"; - } - } - - &:hover, - &.over { - .h5p-file-drop-upload-inner:after { - color: #999; - } - } - } - - .h5p-or-vertical { - float: left; - position: relative; - width: 5%; - height: 250px; - } - } - - .h5p-dialog-box { - text-align: center; - padding: 1em 0.5em; - } - - .h5p-buttons { - padding: 0.5em; - border-top: 1px solid #cdcdcd; - background: #ddd; - text-align: right; - } - } - .h5p-or { - border-bottom: 1px solid #cdcdcd; - padding: 0; - margin: 0 1em; - text-align: center; - height: 0.5em; - line-height: 1em; - & > span { - background: #fff; - padding: 0 0.5em; - } - } - - .h5p-or-vertical { - float: left; - position: relative; - width: 50px; - height: 80px; - - .h5p-or-vertical-line { - position: absolute; - margin: 10px 0; - left: 49%; - top: 0; - bottom: 0; - width: 1px; - background: #ccc; - z-index: 1; - } - .h5p-or-vertical-word-wrapper { - text-align: center; - height: 18px; - position: absolute; - left: 0; - right: 0; - top: 22%; - margin-top: -12px; - z-index: 2; - - .h5p-or-vertical-word { - color: #999; - font-weight: bold; - font-size: 18px; - padding: 3px; - background: #fff; - } - } - } - - .h5p-file-url { - text-align: center; - } - .h5p-thumbnail { - margin: 0.5em; - width: 6em; - height: 4.5em; - display: block; - float: left; - position: relative; - @include box-shadow(0 0 10px 0 #666666); - border: 1px solid #fff; - box-sizing: border-box; - -moz-box-sizing: border-box; - - .h5p-remove { - position: absolute; - top: 0; - right: 0; - cursor: pointer; - outline: none; - width: 1.6em; - height: 1.6em; - line-height: 1.6em; - overflow: hidden; - text-indent: -0.4em; - padding: 0.065em; - @include opacity(0.6); - } - .h5p-remove:hover, - .h5p-remove:focus { - @include opacity(1); - } - .h5p-remove:after { - font-family: $icon-font-family; - font-size: 2em; - color: #fff; - content: "\e890"; - @include opacity(0.6); - } - } - .h5p-type { - position: absolute; - width: 100%; - height: 100%; - font-size: 1.1em; - line-height: 4em; - display: block; - text-align: center; - background: #000; - color: #fff; - cursor: pointer; - } - .file { - position: relative; - float: left; - margin-bottom: 0; - max-width: 100%; - - &.authorList { - float: none; - } - - &.field { - float: none; - } - .thumbnail { - display: inline-block; - margin: 10px 10px 10px 0; - @include box-shadow(0 0 10px 0 #666666); - border: 1px solid #fff; - cursor: pointer; - max-width: 100%; - min-width: 1em; - min-height: 1em; - background: $form-input-color inline-image('transparent-background.png') repeat; - } - .thumbnail:focus { - @include box-shadow(0 0 10px 0 #222); - } - .add { - display: inline-block; - cursor: pointer; - padding: 0.5em 1.5em 0.5em 3em; - background: linear-gradient(to bottom, #fbfbfb 0, #f2f2f2 100%); - border: 1px solid #d0d0d1; - border-radius: 0.25em; - color: #222222; - font-weight: bold; - line-height: normal; - - &:hover { - background: #ededed; - } - - &:focus { - box-shadow: 0 0 16px 0 rgba(133,188,255,0.84); - } - - .h5peditor-field-file-upload-text:before { - font-family: $icon-font-family; - content: "\e902"; - line-height: 1; - color: #39c943; - font-size: 2em; - position: absolute; - left: 0.3em; - top: 0.1em; - } - } - .remove { - display: block; - position: absolute; - top: 7px; - right: 7px; - cursor: pointer; - } - .remove:focus:before { - @include opacity(1); - } - .remove:before { - font-family: $icon-font-family; - font-size: 1.4em; - color: #fff; - content: "\e890"; - @include opacity(0.6); - @include text-shadow(rgba(black, 0.4) 0 0 4px, rgba(black, 0.4) 0 0 4px, rgba(black, 0.4) 0 0 4px); - } - .remove:hover { - text-decoration: none; - } - .remove:hover:before { - color: #fff; - @include opacity(1); - @include text-shadow(rgba(black, 0.4) 0 0 4px, rgba(black, 0.4) 0 0 4px, rgba(black, 0.4) 0 0 4px); - } - img { - max-width: 100%; - vertical-align: bottom; - max-height: 100px; - } - - .h5p-av-row { - overflow: auto; - margin-bottom: 0.5em; - - .h5p-thumbnail { - display: table-cell; - } - - .h5p-video-quality { - display: table-cell; - overflow: hidden; - width: 10000px; - padding-left: 1em; - - input.h5peditor-text { - width: 100%; - } - } - } - - .h5p-av-cell { - overflow: auto; - float: left; - } - } - .video .file, .audio .file { - position: static; - overflow: visible; - - .thumbnail, .add { - float: left; - } - .add { - margin-top: 8px; - } - .thumbnail { - overflow: visible; - position: relative; - cursor: auto; - } - .remove { - top: -3px; - right: -5px; - } - .type { - padding: 16px 8px 4px; - background: #000; - color: #fff; - font-size: 10px; - } - .h5peditor-uploading { - float: left; - margin: 0.5em; - } - } - .libwrap { - margin-top: $padding; - - &.no-margin, - &:empty { - margin-top: 0; - } - } - input[type="checkbox"] { - margin: 4px 4px 4px 0; - vertical-align: bottom; - } - .moving { - position: absolute; - z-index: 1; - @include opacity(0.8); - -webkit-transform: translateZ(0); - - .h5peditor-label { - cursor: grabbing; - cursor: -moz-grabbing; - cursor: -webkit-grabbing; - } - } - - .h5peditor-uploading, .h5peditor-loading { - padding-top: 10px; - padding-bottom: 6px; - font-size: 14px; - } - .h5peditor-loading { - padding: 0.875em 0 1em 3.25em; - font-style: italic; - } - - .h5p-copyright-button { - @include border-radius(0.25em); - height: 30px; - background: linear-gradient(to bottom, #fff 0, #f2f2f2 100%); - border: 1px solid $form-border-color; - color: $black; - font-size: $font-size-small; - line-height: $form-item-height-small; - padding-right: $padding; - padding-left: 0; - clear: both; - font-weight: normal; - - &:before { - font-family: $icon-font-family; - content: "\e88f"; - color: #666; - padding: 0 0.25em 0 0.25em; - vertical-align: middle; - font-size: 1.5em; - line-height: 0.9; - } - - &:hover:not([disabled]) { - background: linear-gradient(to bottom, #fff 0, #d0d0d1 100%); - text-decoration: none; - border-color: #999; - } - } - - .h5p-copyright-button:focus { - box-shadow: 0 0 16px 0 rgba(133, 188, 255, 0.84); - } - - /* Copyright button for video & audio */ - .field.file > .h5p-copyright-button, - .field.video > .h5p-copyright-button, - .field.audio > .h5p-copyright-button { - float: left; - } - - .h5p-editor-dialog { - position: absolute; - z-index: 2; - margin: 5.5em 0 1em; - visibility: hidden; - opacity: 0; - height: 0; - overflow: hidden; - background: #fff; - @include box-shadow(0 0 8px #666); - @include transition(visibility 0s 0.2s, height 0s 0.2s, opacity 0.2s, margin-top 0.2s); - - &.h5p-open { - margin-top: 3.5em; - visibility: visible; - opacity: 1; - height: auto; - @include transition(visibility 0s 0s, height 0s 0s, opacity 0.2s, margin-top 0.2s); - } - - & > .field { - margin: 0; - border: 0; - box-shadow: none; - } - - .content { - border: none; - background: $white; - - .h5peditor-label { - font-size: $font-size-large; - font-weight: 600; - } - } - - .h5p-close { - color: #494949; - - &:before { - font-size: 2em; - right: -0.125em; - top: 0; - position: absolute; - z-index: 1; - font-family: $icon-font-family; - content: "\e894"; - line-height: 1em; - @include transition(scale 0.2s); - } - - &:hover:before { - @include scale(1.1, 1.1); - } - } - } - - .h5p-li > .content > .library { - border: 0; - padding: 0; - } -} - -.h5p-editor-dialog.h5p-dialog-wide { - width: 90%; - border-radius: 0.208em; -} - -/* Placed this outside of .h5peditor context above. Don't want it to be more - specific than neccessary (because of overriding) */ -.h5peditor-button-textual { - @include border-radius(0.25em); - @include button-background( - $form-item-importance-medium-background, - $form-item-importance-medium-background-highlight); - display: inline-block; - width: auto; - margin: $min-padding 0 0 0; - padding: 0 $padding; - box-sizing: border-box; - height: $form-item-height-normal; - border: 1px solid $form-border-color; - font-size: $font-size-normal; - font-family: $font-family; - line-height: $form-item-height-normal; - color: $white; - cursor: pointer; - font-weight: 600; - -webkit-font-smoothing: antialiased; - -moz-osx-font-smoothing: grayscale; - - &:focus { - box-shadow: 0 0 16px 0 rgba(133, 188, 255, 0.84); - } - - &:before, - &:after { - color: $white; - } - - .h5p-editing-image-button:before { - font-family: $icon-font-family; - content: "\e900"; - color: #666; - padding-right: 0.25em; - vertical-align: middle; - font-size: 1.5em; - line-height: 0.9; - } - - &:hover:not([disabled]) { - @include button-background( - $form-item-importance-medium-background-hover, - $form-item-importance-medium-background-hover-highlight); - text-decoration: none; - } - - &.importance-high { - @include button-background( - $form-item-importance-high-background, - $form-item-importance-high-background-highlight); - border-color: $form-item-importance-high-border-color; - text-transform: uppercase; - height: $form-item-height-large; - line-height: $form-item-height-large; - - &:hover:not([disabled]) { - @include button-background( - $form-item-importance-high-background-hover, - $form-item-importance-high-background-hover-highlight); - border-color: $form-item-importance-high-background-hover; - } - } - - &.importance-low { - @include button-background( - $form-item-importance-low-background, - $form-item-importance-low-background-highlight); - border-color: $form-item-importance-low-border-color; - color: $black; - - &:before, - &:after { - color: $black; - } - - &:hover:not([disabled]) { - @include button-background( - $form-item-importance-low-background-hover, - $form-item-importance-low-background-hover-highlight); - border-color: $form-item-importance-low-background-hover; - } - } - - &[disabled], - &[aria-disabled="true"] { - background: #f5f5f5; - color: #707171; - border-color: #eeeeee; - cursor: not-allowed; - - &:before { - font-family: $icon-font-family; - content: "\e912"; - color: #707171; - margin-right: 0.5em; - vertical-align: top; - } - } -} -.h5peditor-field-description, -.h5p-help-text { - font-size: 12px; - margin-top: 0.3em; - margin-bottom: 1em; - font-weight: 500; - color: #434446; - line-height: 15px; - letter-spacing: 0.5px; -} - -.h5peditor-field-important-description { - position: relative; - padding: 1.5em 2.2em 2em 2.2em; - font-size: 0.9em; - margin: 0 0 20px 0; - border: 1px solid #ffde78; - background-color: #f9f9d8; - color: #000; - display: none; - - .important-description-close { - position: absolute; - font-size: 0.9em; - line-height: 2em; - right: 1.5em; - top: 1em; - user-select: none; - cursor: pointer; - - &:before { - font-family: $icon-font-family; - content: "\e894"; - vertical-align: top; - font-size: 1.6em; - } - } - - .h5p-info-icon { - font-size: 0.75em; - - &:before { - font-family: $icon-font-family; - content: "\e909"; - font-size: 2.3em; - line-height: 0.94em; - margin-left: -0.3em; - vertical-align: middle; - } - } - - .important-description-title { - font-weight: bold; - font-size: 1.2em; - vertical-align: bottom; - } - - .important-description-content { - ul { - padding: 0; - line-height: 2em; - margin-top: 20px; - - li { - line-height: 1.8em; - margin-left: 1.5em; - } - } - } - - .important-description-example { - border: 1px solid #bbde98; - background-color: #d9fac3; - display: flex; - margin-top: 20px; - - .important-description-example-title { - padding: 15px; - font-weight: bold; - } - - .important-description-example-text { - padding: 15px; - width: 100%; - background-color: #f6fef1; - } - } -} - -.h5peditor-form { - position: relative; - background: $form-background; - border: 1px solid $form-border-color; - - &.h5peditor-form-manager { - > .tree, - > .common, - > .field { - max-width: 918px; - margin: 20px auto; - padding: 0 20px; - } - } - - &.h5peditor-semi-fullscreen { - margin: 0; - position: fixed; - overflow-y: scroll; - margin-top: 40px; - padding-bottom: 40px; - box-sizing: border-box; - background: #fff; - height: calc(100% - 40px); - } -} - -.h5peditor-label { - display: block; - margin-bottom: 6px; - font-weight: 600; - font-size: $font-size-normal; - color: $form-label; -} - -#h5peditor-uploader { - position: absolute; - width: 1px; - height: 1px; - top: -1px; - border: 0; - overflow: hidden; -} - -@import "tutorial-and-example-links"; - -.h5peditor-widget-select { - overflow: hidden; - margin: 0 0 -1px; - padding: 0; - list-style: none; -} -.h5peditor-widget-option { - float: right; - border: 1px solid #ccc; - border-bottom: 0; - margin-left: 0.5em; - padding: 0.6em 1em; - color: #0E1A25; - font-size: 0.875em; - background: #f5f5f5; - line-height: 1.285714286em; - cursor: pointer; - outline: none; -} -.h5peditor-widget-option:hover { - color: #000; -} -.h5peditor-widget-option:active { - color: #8e636a; /* Pink chocolate */ -} -.h5peditor-widget-active { - background: #fff; - line-height: 1.357142857em; -} -.h5peditor-widgets > .h5peditor-widget-wrapper { - border: 1px solid #ccc; - margin: 0 0 0.25em; - padding: 0.5em; -} -.h5peditor-widgets > .h5peditor-label { - float: left; - margin-top: 5px; -} -.h5p-editor-iframe { - margin-bottom: 1em; -} -.h5peditor-required:after { - content: "*"; - color: #da0001; - margin-left: 0.2em; - position: relative; - top: -0.2em; -} - -/* Help CKEditor blend into the H5PEditor */ -.h5peditor .cke_bottom, -.h5peditor .cke_top { - background: #d0d0d1; -} -.h5peditor .cke_chrome { - border: 1px solid #f5f5f5; - background: #d0d0d1; -} -.h5peditor .cke_contents, -.h5peditor .cke_toolgroup, -.h5peditor .cke_combo_button { - border: 1px solid #f5f5f5; -} - -/* The show important description button */ -.important-description-show { - background: #f3d55a; - padding: 0.2em 0.5em 0.2em 1.7em; - margin-bottom: 0.5em; - font-size: 0.9em; - cursor: pointer; - float: right; - position: relative; - line-height: normal; - - font-style: italic; - font-weight: bold; - - &:before { - font-family: $icon-font-family; - font-style: normal; - font-weight: normal; - content: "\e909"; - font-size: 1.8em; - position: absolute; - left: 0em; - top: 0em; - line-height: 0.85em; - vertical-align: top; - } -} - -.important-description-clear-right { - display: block; - clear: right; -} - -.h5peditor fieldset.common-fields-library-wrapper { - width: 100%; - padding: 0 1em 1em; - box-sizing: border-box; - height: 20px; - border: solid 1px #e2e5ee; - border-radius: 4px; - margin-bottom: 1em; - - & > * { - display: none; - } - & > legend { - display: block; - cursor: pointer; - outline: none; - color: #363b42; - background-color: #fff; - padding: 10px; - font-weight: bold; - font-size: 0.875em; - - &:before { - font-family: "H5P"; - content: "\e566"; - margin-right: 0.5em; - } - &:focus:before { - outline: 1px dotted #666; - } - } - - &.expanded { - height: auto; - - & > * { - display: block; - } - & > legend:before { - content: "\e565"; - } - } -} - -@import "metadata-form"; -@import "metadata-popup"; -@import "metadata-changelog-widget"; -@import "metadata-author-widget"; -@import "copy-paste"; -@import "fullscreen-bar"; - -.h5peditor .h5peditor-language-switcher { - float: right; - white-space: nowrap; - margin-bottom: 1em; - - .language-label { - padding: 0 10px; - font-size: 15px; - } - select { - padding: 6px 30px 6px 8px; - font-size: 15px; - } -} -.h5peditor-language-notice { - display: none; - clear: both; - font-size: 14px; - background: #dcf6ff; - color: #295b7a; - padding: 10px 20px 10px 40px; - margin: 1em 0; - line-height: 1.5; - position: relative; - - &:before { - font-family: h5p; - content: "\e90c"; - position: absolute; - left: 15px; - font-size: 1.125em; - } - &.show { - display: block; - } - .first { - font-weight: bold; - } - a { - text-decoration: underline; - } -} diff --git a/h5p/h5plib/v126/joubel/editor/styles/scss/cke-contents.scss b/h5p/h5plib/v126/joubel/editor/styles/scss/cke-contents.scss deleted file mode 100644 index 7d4e9a3f3330d..0000000000000 --- a/h5p/h5plib/v126/joubel/editor/styles/scss/cke-contents.scss +++ /dev/null @@ -1,19 +0,0 @@ -@import "../../ckeditor/contents.css"; - -code { - color: #3d3d3d; - background: #e0e0e0; - border-radius: 2px; - padding: 0 5px; -} -pre > code { - background-color: #fafafa; - padding: 5px; - display: block; - line-height: normal; - border: 1px solid #c7c7c7; - border-left-width: 4px; - max-width: 100%; - white-space: pre; - overflow: auto; -} diff --git a/h5p/h5plib/v126/lang/en/h5plib_v126.php b/h5p/h5plib/v126/lang/en/h5plib_v126.php deleted file mode 100644 index 05550f31397de..0000000000000 --- a/h5p/h5plib/v126/lang/en/h5plib_v126.php +++ /dev/null @@ -1,246 +0,0 @@ -. - -/** - * Strings for component 'h5p_v126' - * - * @package h5plib_v126 - * @copyright 2024 Sara Arjona - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ - -defined('MOODLE_INTERNAL') || die(); - -$string['editor:a11ytitleshowlabel'] = 'Show label for AT'; -$string['editor:a11ytitlehidelabel'] = 'Hide label for AT'; -$string['editor:atoz'] = 'A to Z'; -$string['editor:add'] = 'Add'; -$string['editor:addauthor'] = 'Save author'; -$string['editor:addentity'] = 'Add :entity'; -$string['editor:addfile'] = 'Add file'; -$string['editor:addnewchange'] = 'Add new change'; -$string['editor:addtitle'] = 'Add title'; -$string['editor:addvideodescription'] = 'H5P supports all external video sources formatted as mp4, webm or ogv, like Vimeo Pro, and has support for YouTube and Panopto links.'; -$string['editor:avtablistlabel'] = 'Insert using'; -$string['editor:belowmin'] = 'The :property value is below the minimum of :min.'; -$string['editor:cancel'] = 'Cancel'; -$string['editor:cancellabel'] = 'Cancel'; -$string['editor:changefile'] = 'Change file'; -$string['editor:changelanguage'] = 'Change language to :language?'; -$string['editor:changelibrary'] = 'Change content type?'; -$string['editor:changelogdescription'] = 'Some licences require that changes made to the original work, or derivatives are logged and displayed. You may log your changes here for licensing reasons or just to allow yourself and others to keep track of the changes made to this content.'; -$string['editor:close'] = 'Close'; -$string['editor:commonfields'] = 'Text overrides and translations'; -$string['editor:commonfieldsdescription'] = 'Here you can edit settings or translate texts used in this content.'; -$string['editor:confirmchangelibrary'] = 'By doing this you will lose all work done with the current content type. Are you sure you wish to change content type?'; -$string['editor:confirmdeletechangelog'] = 'Are you sure you would like to delete this change log entry?'; -$string['editor:confirmimageremoval'] = 'This will remove your image. Are you sure you wish to proceed?'; -$string['editor:confirmpastebuttontext'] = 'Replace content'; -$string['editor:confirmpastecontent'] = 'By doing this you will replace the current content with the content from your clipboard. The current content will be lost. Are you sure you wish to continue?'; -$string['editor:confirmremoval'] = 'Are you sure you wish to remove this :type?'; -$string['editor:confirmremoveauthor'] = 'Are you sure you would like to remove this author?'; -$string['editor:contenttypebackbuttonlabel'] = 'Back'; -$string['editor:contenttypecacheoutdated'] = 'Content type list outdated'; -$string['editor:contenttypecacheoutdateddesc'] = 'Your site is having difficulties connecting to H5P.org to check for content type updates. You may not be able to update or install new content types.'; -$string['editor:contenttypedemobuttonlabel'] = 'Content demo'; -$string['editor:contenttypedetailbuttonlabel'] = 'Details'; -$string['editor:contenttypegetbuttonlabel'] = 'Get'; -$string['editor:contenttypeiconalttext'] = 'Icon'; -$string['editor:contenttypeinstallbuttonlabel'] = 'Install'; -$string['editor:contenttypeinstallerror'] = ':contentType could not be installed. Contact your administrator.'; -$string['editor:contenttypeinstallsuccess'] = ':contentType successfully installed!'; -$string['editor:contenttypeinstallingbuttonlabel'] = 'Installing'; -$string['editor:contenttypelicensepaneltitle'] = 'Licence'; -$string['editor:contenttypenotinstalled'] = 'Content type not installed'; -$string['editor:contenttypenotinstalleddesc'] = 'You do not have permission to install content types.'; -$string['editor:contenttypeowner'] = 'By :owner'; -$string['editor:contenttyperestricted'] = 'Restricted content type'; -$string['editor:contenttyperestricteddesc'] = 'The use of this content type has been restricted by an administrator.'; -$string['editor:contenttypesearchfieldplaceholder'] = 'Search for content types'; -$string['editor:contenttypesectionall'] = 'All content types'; -$string['editor:contenttypeunsupportedapiversioncontent'] = 'Contact your system administrator to provide you with the necessary updates'; -$string['editor:contenttypeunsupportedapiversiontitle'] = 'This content type requires a newer core version'; -$string['editor:contenttypeupdateavailable'] = 'Update available'; -$string['editor:contenttypeupdatebuttonlabel'] = 'Update'; -$string['editor:contenttypeupdatesuccess'] = ':contentType successfully updated!'; -$string['editor:contenttypeupdatingbuttonlabel'] = 'Updating'; -$string['editor:contenttypeusebuttonlabel'] = 'Use'; -$string['editor:contributetranslations'] = 'If you want to complete the translation for :language, see contributing translations to H5P.'; -$string['editor:copiedbutton'] = 'Copied'; -$string['editor:copiedtoclipboard'] = 'Content is copied to the clipboard'; -$string['editor:copybutton'] = 'Copy'; -$string['editor:copytoclipboard'] = 'Copy H5P content to the clipboard'; -$string['editor:createcontenttablabel'] = 'Create content'; -$string['editor:currentmenuselected'] = 'current selection'; -$string['editor:editcopyright'] = 'Edit copyright'; -$string['editor:editimage'] = 'Edit image'; -$string['editor:editmode'] = 'Editing mode'; -$string['editor:enteraudiotitle'] = 'Paste link or other audio source URL'; -$string['editor:enteraudiourl'] = 'Enter audio source URL'; -$string['editor:enterfullscreenbuttonlabel'] = 'Enter fullscreen'; -$string['editor:entervideotitle'] = 'Paste YouTube link or other video source URL'; -$string['editor:entervideourl'] = 'Enter video URL'; -$string['editor:errorcalculatingmaxscore'] = 'Could not calculate the max score for this content. The max score is assumed to be 0. Contact your administrator if this isn’t correct.'; -$string['editor:errorcommunicatinghubcontent'] = 'An error occured. Please try again.'; -$string['editor:errorcommunicatinghubtitle'] = 'Not able to communicate with hub.'; -$string['editor:errorheader'] = 'An error occured'; -$string['editor:errornotsupported'] = 'Parameters contain %used which is not supported.'; -$string['editor:errorparamsbroken'] = 'Parameters are broken.'; -$string['editor:errortoohighversion'] = 'Parameters contain %used while only %supported or earlier are supported.'; -$string['editor:example'] = 'Example'; -$string['editor:exceedsmax'] = 'The :property value exceeds the maximum of :max.'; -$string['editor:exitfullscreenbuttonlabel'] = 'Exit fullscreen'; -$string['editor:expandcollapse'] = 'Expand/Collapse'; -$string['editor:filetolarge'] = 'The file you are trying to upload might be too large.'; -$string['editor:fillinthefieldsbelow'] = 'Fill in the fields below'; -$string['editor:gethelp'] = 'Get help'; -$string['editor:h5pfileuploadservererrorcontent'] = 'An unexpected error occurred. Check your server error log for more details.'; -$string['editor:h5pfileuploadservererrortitle'] = 'The H5P file could not be uploaded'; -$string['editor:h5pfilevalidationfailedcontent'] = 'Make sure the uploaded H5P contains valid H5P content. H5P files containing only libraries should be uploaded through the H5P libraries page.'; -$string['editor:h5pfilevalidationfailedtitle'] = 'Could not validate H5P file.'; -$string['editor:h5pfilewrongextensioncontent'] = 'Only files with the .h5p extension are allowed.'; -$string['editor:h5pfilewrongextensiontitle'] = 'The selected file could not be uploaded'; -$string['editor:height'] = 'height'; -$string['editor:hide'] = 'Hide'; -$string['editor:hideimportantinstructions'] = 'Hide important instructions'; -$string['editor:hubpanellabel'] = 'Select content type'; -$string['editor:illegaldecimalnumber'] = ':property can only contain numbers with max :decimals decimals.'; -$string['editor:imagelightboxprogress'] = ':num of :total'; -$string['editor:imagelightboxtitle'] = 'Images'; -$string['editor:importantinstructions'] = 'Important instructions'; -$string['editor:insert'] = 'Insert'; -$string['editor:invalidformat'] = 'Field value contains an invalid format or characters that are forbidden.'; -$string['editor:language'] = 'Language'; -$string['editor:librarymissing'] = 'Missing required library %lib.'; -$string['editor:licensecandistribute'] = 'Can distribute'; -$string['editor:licensecanholdliable'] = 'Can hold liable'; -$string['editor:licensecanmodify'] = 'Can modify'; -$string['editor:licensecansublicense'] = 'Can sub-licence'; -$string['editor:licensecanusecommercially'] = 'Can use commercially'; -$string['editor:licensecannotholdliable'] = 'Cannot hold liable'; -$string['editor:licensedescription'] = 'Some of the features of this licence are indicated below. Click the info icon above to read the original licence text.'; -$string['editor:licensefetchdetailsfailed'] = 'Failed fetching licence details'; -$string['editor:licensemodalsubtitle'] = 'Select a licence to view information about proper usage'; -$string['editor:licensemodaltitle'] = 'Licence details'; -$string['editor:licensemustincludecopyright'] = 'Must include copyright'; -$string['editor:licensemustincludelicense'] = 'Must include licence'; -$string['editor:licenseunspecified'] = 'Unspecified'; -$string['editor:listbelowmin'] = 'The list needs at least :min items for the content to function properly.'; -$string['editor:listexceedsmax'] = 'The list exceeds the maximum of :max items.'; -$string['editor:listlabel'] = 'List'; -$string['editor:loading'] = 'Loading, please wait...'; -$string['editor:loadingimageeditor'] = 'Loading image editor, please wait...'; -$string['editor:logthischange'] = 'Log this change'; -$string['editor:loggedchanges'] = 'Logged changes'; -$string['editor:maxscoresemanticsmissing'] = 'Could not find the expected semantics in the content.'; -$string['editor:metadata'] = 'Metadata'; -$string['editor:metadatasharingandlicensinginfo'] = 'Metadata (sharing and licencing info)'; -$string['editor:missingproperty'] = 'Field :index is missing its :property property.'; -$string['editor:missingtranslation'] = '[Missing translation :key]'; -$string['editor:newchangehasbeenlogged'] = 'New change has been logged'; -$string['editor:newestfirst'] = 'Newest first'; -$string['editor:nextimage'] = 'Next image'; -$string['editor:nochangeshavebeenlogged'] = 'No changes have been logged'; -$string['editor:nocontenttypesavailable'] = 'No content types are available'; -$string['editor:nocontenttypesavailabledesc'] = 'Your site is having difficulties connecting to H5P.org and listing the available content types.'; -$string['editor:nofollow'] = 'Cannot follow field ":path".'; -$string['editor:nolanguagessupported'] = 'No languages supported'; -$string['editor:noresultsfound'] = 'No results found'; -$string['editor:noresultsfounddesc'] = 'There is no content type that matches your search criteria.'; -$string['editor:nosemantics'] = 'Error, could not load the content type form.'; -$string['editor:notalltextschanged'] = 'Not all texts were changed, as there is only partial coverage for :language.'; -$string['editor:notimagefield'] = '":path" is not an image.'; -$string['editor:notimageordimensionsfield'] = '":path" is not an image or dimensions field.'; -$string['editor:numresults'] = ':num results'; -$string['editor:numberfield'] = 'number field'; -$string['editor:ok'] = 'OK'; -$string['editor:onlynumbers'] = 'The :property value can only contain numbers.'; -$string['editor:or'] = 'or'; -$string['editor:orderitemdown'] = 'Order item down'; -$string['editor:orderitemup'] = 'Order item up'; -$string['editor:outofstep'] = 'The :property value can only be changed in steps of :step.'; -$string['editor:pasteandreplacebutton'] = 'Paste and replace'; -$string['editor:pasteandreplacefromclipboard'] = 'Replace existing content with H5P content from the clipboard'; -$string['editor:pastebutton'] = 'Paste'; -$string['editor:pastecontent'] = 'Replace content'; -$string['editor:pastecontentnotsupported'] = 'The content in the H5P clipboard is not supported in this context'; -$string['editor:pastecontentrestricted'] = 'The content in the clipboard has been restricted on this site'; -$string['editor:pasteerror'] = 'Cannot paste from clipboard'; -$string['editor:pastefromclipboard'] = 'Paste H5P content from the clipboard'; -$string['editor:pastenocontent'] = 'No H5P content on the clipboard'; -$string['editor:pastetoonew'] = 'The content in the H5P clipboard is of a higher version (:clip) than what is supported in this context (:local). If possible, try to have this content upgraded first, then try pasting the content here again.'; -$string['editor:pastetooold'] = 'The content in the H5P clipboard is of a lower version (:clip) than what is supported in this context (:local). If possible, try to have the content you want to paste upgraded, then copy it again and try pasting it here.'; -$string['editor:popularfirst'] = 'Popular first'; -$string['editor:previousimage'] = 'Previous image'; -$string['editor:proceedbuttonlabel'] = 'Proceed to save'; -$string['editor:readless'] = 'Read less'; -$string['editor:readmore'] = 'Read more'; -$string['editor:recentlyusedfirst'] = 'Recently used first'; -$string['editor:reloadbuttonlabel'] = 'Reload'; -$string['editor:removefile'] = 'Remove file'; -$string['editor:removeimage'] = 'Remove image'; -$string['editor:removeitem'] = 'Remove item'; -$string['editor:requiredproperty'] = 'The :property is required and must have a value.'; -$string['editor:resettooriginallabel'] = 'Reset to original'; -$string['editor:savelabel'] = 'Save'; -$string['editor:savemetadata'] = 'Save metadata'; -$string['editor:screenshots'] = 'Screenshots'; -$string['editor:scriptmissing'] = 'Could not load upgrades script for %lib.'; -$string['editor:searchresults'] = 'Search results'; -$string['editor:selectfiletoupload'] = 'Select file to upload'; -$string['editor:selectlibrary'] = 'Select the library you wish to use for your content.'; -$string['editor:semanticserror'] = 'Semantics error: :error'; -$string['editor:show'] = 'Show'; -$string['editor:showimportantinstructions'] = 'Show instructions'; -$string['editor:tabtitlebasicfileupload'] = 'File upload'; -$string['editor:tabtitleinputlinkurl'] = 'Link/URL'; -$string['editor:textfield'] = 'text field'; -$string['editor:thecontenttype'] = 'the content type'; -$string['editor:thiswillpotentially'] = 'This will potentially reset all the text and translations. You can\'t undo this. The content itself will not be changed. Do you want to proceed?'; -$string['editor:title'] = 'Title'; -$string['editor:toolong'] = 'Field value is too long; it should contain :max letters or less.'; -$string['editor:tryagain'] = 'Try again'; -$string['editor:tutorial'] = 'Tutorial'; -$string['editor:unabletointerpreterror'] = 'Unable to interpret response.'; -$string['editor:unabletointerpretsolution'] = 'Please check your error log.'; -$string['editor:unknownfieldpath'] = 'Unable to find ":path".'; -$string['editor:unknownfileuploaderror'] = 'Unknown file upload error'; -$string['editor:unknownlibrary'] = 'Unfortunately, the selected content type \'%lib\' isn\'t installed on this system.'; -$string['editor:untitled'] = 'Untitled :libraryTitle'; -$string['editor:uploadaudiotitle'] = 'Upload audio file'; -$string['editor:uploaderror'] = 'File upload error'; -$string['editor:uploadfilebuttonchangelabel'] = 'Change file'; -$string['editor:uploadfilebuttonlabel'] = 'Upload a file'; -$string['editor:uploadinstructionscontent'] = 'You may start with examples from H5P.org.'; -$string['editor:uploadinstructionstitle'] = 'Upload an H5P file.'; -$string['editor:uploadplaceholder'] = 'No file chosen'; -$string['editor:uploadsuccess'] = ':title was successfully uploaded!'; -$string['editor:uploadtablabel'] = 'Upload'; -$string['editor:uploadvideotitle'] = 'Upload video file'; -$string['editor:uploading'] = 'Uploading, please wait...'; -$string['editor:uploadingthrobber'] = 'Now uploading...'; -$string['editor:usedforsearchingreportsandcopyrightinformation'] = 'Used for searching, reports and copyright information'; -$string['editor:videoquality'] = 'Video quality label'; -$string['editor:videoqualitydefaultlabel'] = 'Quality :index'; -$string['editor:videoqualitydescription'] = 'This label helps the user identify the current quality of the video. E.g. 1080p, 720p, HD or Mobile'; -$string['editor:warningchangebrowsingtoseeresults'] = 'Click All to get the list of all the content types you can install.'; -$string['editor:warningnocontenttypesinstalled'] = 'You don\'t have any content types installed.'; -$string['editor:warningupdateavailablebody'] = 'Update to the latest version for an improved experience.'; -$string['editor:warningupdateavailabletitle'] = 'A new version of :contentType is available.'; -$string['editor:width'] = 'width'; -$string['pluginname'] = 'H5P framework v1.26'; -$string['pluginname_help'] = 'H5P framework version 1.26.'; -$string['privacy:metadata'] = 'The H5P framework v1.26 does not store any personal data.'; diff --git a/h5p/h5plib/v126/thirdpartylibs.xml b/h5p/h5plib/v126/thirdpartylibs.xml deleted file mode 100644 index 0fa7a2f7df1f7..0000000000000 --- a/h5p/h5plib/v126/thirdpartylibs.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - joubel/core - h5p-php-library - The general H5P library. - 1.26 - GPL - 3.0+ - https://github.com/h5p/h5p-php-library/ - - Joubel - - - - joubel/editor - h5p-editor-php-library - A general library that is supposed to be used in most PHP implementations of H5P. - moodle-1.23 - GPL - 3.0+ - https://github.com/h5p/h5p-editor-php-library/ - - Joubel - - - diff --git a/h5p/h5plib/v126/version.php b/h5p/h5plib/v126/version.php deleted file mode 100644 index 132d5a432fb1d..0000000000000 --- a/h5p/h5plib/v126/version.php +++ /dev/null @@ -1,29 +0,0 @@ -. - -/** - * Version information. - * - * @package h5plib_v126 - * @copyright 2024 Sara Arjona - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ - -defined('MOODLE_INTERNAL') || die(); - -$plugin->version = 2024042200; // The current module version (Date: YYYYMMDDXX). -$plugin->requires = 2024041600; // Requires this Moodle version. -$plugin->component = 'h5plib_v126'; // Full name of the plugin (used for diagnostics). diff --git a/lib/db/upgrade.php b/lib/db/upgrade.php index 57ed56d25ac5b..f90f3073cd227 100644 --- a/lib/db/upgrade.php +++ b/lib/db/upgrade.php @@ -1421,5 +1421,20 @@ function xmldb_main_upgrade($oldversion) { upgrade_main_savepoint(true, 2024092000.01); } + if ($oldversion < 2024092400.01) { + // If h5plib_v126 is no longer present, remove it. + if (!file_exists($CFG->dirroot . '/h5p/h5plib/v126/version.php')) { + // Clean config. + uninstall_plugin('h5plib', 'v126'); + } + + // If h5plib_v127 is present, set it as the default one. + if (file_exists($CFG->dirroot . '/h5p/h5plib/v127/version.php')) { + set_config('h5plibraryhandler', 'h5plib_v127'); + } + + upgrade_main_savepoint(true, 2024092400.01); + } + return true; } diff --git a/lib/plugins.json b/lib/plugins.json index 1233bce742ca1..38a4d2f01059f 100644 --- a/lib/plugins.json +++ b/lib/plugins.json @@ -274,7 +274,7 @@ "rubric" ], "h5plib": [ - "v126" + "v127" ], "local": [], "logstore": [ @@ -605,7 +605,8 @@ "tidy" ], "h5plib": [ - "v124" + "v124", + "v126" ], "logstore": [ "legacy" diff --git a/version.php b/version.php index 88f8ced370f1c..c524a3a329172 100644 --- a/version.php +++ b/version.php @@ -29,7 +29,7 @@ defined('MOODLE_INTERNAL') || die(); -$version = 2024092400.00; // YYYYMMDD = weekly release date of this DEV branch. +$version = 2024092400.01; // YYYYMMDD = weekly release date of this DEV branch. // RR = release increments - 00 in DEV branches. // .XX = incremental changes. $release = '4.5dev+ (Build: 20240924)'; // Human-friendly version name